#![allow(unused_imports)] #![allow(unused_attributes)] #![allow(dead_code)] #[cfg(feature = "proto_gen")] use crate::proto_gen::*; use log::info; use std::fs::File; use std::io::Write; use std::process::Command; use walkdir::WalkDir; pub fn gen_files(crate_name: &str, root: &str) { #[cfg(feature = "proto_gen")] let _ = gen_protos(crate_name); let mut paths = vec![]; let mut file_names = vec![]; for (path, file_name) in WalkDir::new(root).into_iter().filter_map(|e| e.ok()).map(|e| { let path = e.path().to_str().unwrap().to_string(); let file_name = e.path().file_stem().unwrap().to_str().unwrap().to_string(); (path, file_name) }) { if path.ends_with(".proto") { // https://stackoverflow.com/questions/49077147/how-can-i-force-build-rs-to-run-again-without-cleaning-my-whole-project println!("cargo:rerun-if-changed={}", path); paths.push(path); file_names.push(file_name); } } println!("cargo:rerun-if-changed=build.rs"); #[cfg(feature = "dart")] gen_pb_for_dart(crate_name, root, &paths, &file_names); protoc_rust::Codegen::new() .out_dir("./src/protobuf/model") .inputs(&paths) .include(root) .run() .expect("Running protoc failed."); } #[cfg(feature = "dart")] fn gen_pb_for_dart(name: &str, root: &str, paths: &Vec, file_names: &Vec) { let output = format!( "{}/{}/{}", env!("CARGO_MAKE_WORKING_DIRECTORY"), env!("FLUTTER_FLOWY_SDK_PATH"), name ); if !std::path::Path::new(&output).exists() { std::fs::create_dir_all(&output).unwrap(); } check_pb_compiler(); check_pb_dart_plugin(); paths.iter().for_each(|path| { if cmd_lib::run_cmd! { protoc --dart_out=${output} --proto_path=${root} ${path} } .is_err() { panic!("Generate pb file failed with: {}", path) }; }); let protobuf_dart = format!("{}/protobuf.dart", output); match std::fs::OpenOptions::new() .create(true) .write(true) .append(false) .truncate(true) .open(&protobuf_dart) { Ok(ref mut file) => { let mut export = String::new(); export.push_str("// Auto-generated, do not edit \n"); for file_name in file_names { let c = format!("export './{}.pb.dart';\n", file_name); export.push_str(c.as_ref()); } file.write_all(export.as_bytes()).unwrap(); File::flush(file).unwrap(); } Err(err) => { panic!("Failed to open file: {}", err); } } } fn check_pb_compiler() { assert!(run_command("command -v protoc"), "protoc was not installed correctly"); } fn check_pb_dart_plugin() { assert!( run_command("command -v protoc-gen-dart"), "protoc-gen-dart was not installed correctly" ); } fn run_command(cmd: &str) -> bool { let output = if cfg!(target_os = "windows") { Command::new("cmd") .arg("/C") .arg(cmd) .status() .expect("failed to execute process") } else { Command::new("sh") .arg("-c") .arg(cmd) .status() .expect("failed to execute process") }; output.success() } #[cfg(feature = "proto_gen")] fn gen_protos(crate_name: &str) -> Vec { let cache_path = env!("CARGO_MANIFEST_DIR"); let root = std::fs::canonicalize(".").unwrap().as_path().display().to_string(); let crate_context = ProtoGenerator::gen(crate_name, &root, cache_path); let proto_crates = crate_context .iter() .map(|info| info.protobuf_crate.clone()) .collect::>(); crate_context .into_iter() .map(|info| info.files) .flatten() .for_each(|file| { println!("cargo:rerun-if-changed={}", file.file_path); }); proto_crates }