AppFlowy/shared-lib/lib-infra/src/pb.rs

142 lines
4.0 KiB
Rust
Raw Normal View History

#![allow(unused_imports)]
2022-02-08 15:52:00 +00:00
#![allow(unused_attributes)]
2022-02-08 23:23:57 +00:00
#![allow(dead_code)]
2022-02-09 05:08:49 +00:00
#[cfg(feature = "proto_gen")]
use crate::proto_gen::*;
use log::info;
use std::fs::File;
use std::io::Write;
2022-02-08 23:23:57 +00:00
use std::process::Command;
use walkdir::WalkDir;
2022-02-09 10:17:06 +00:00
pub fn gen_files(crate_name: &str, root: &str) {
2022-02-09 05:08:49 +00:00
#[cfg(feature = "proto_gen")]
2022-02-09 10:17:06 +00:00
let _ = gen_protos(crate_name);
2022-02-09 05:08:49 +00:00
let mut paths = vec![];
let mut file_names = vec![];
2022-02-09 05:08:49 +00:00
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);
}
}
2022-02-09 05:08:49 +00:00
println!("cargo:rerun-if-changed=build.rs");
#[cfg(feature = "dart")]
2022-02-09 10:17:06 +00:00
gen_pb_for_dart(crate_name, root, &paths, &file_names);
2022-02-09 05:08:49 +00:00
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<String>, file_names: &Vec<String>) {
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();
}
2022-02-08 23:23:57 +00:00
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()
{
2022-02-08 17:14:54 +00:00
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);
}
}
}
2022-02-08 23:23:57 +00:00
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()
}
2022-02-09 05:08:49 +00:00
#[cfg(feature = "proto_gen")]
2022-02-09 10:17:06 +00:00
fn gen_protos(crate_name: &str) -> Vec<ProtobufCrate> {
let cache_path = env!("CARGO_MANIFEST_DIR");
2022-02-09 05:08:49 +00:00
let root = std::fs::canonicalize(".").unwrap().as_path().display().to_string();
2022-02-09 10:17:06 +00:00
let crate_context = ProtoGenerator::gen(crate_name, &root, cache_path);
2022-02-09 05:08:49 +00:00
let proto_crates = crate_context
.iter()
.map(|info| info.protobuf_crate.clone())
.collect::<Vec<_>>();
crate_context
.into_iter()
.map(|info| info.files)
.flatten()
.for_each(|file| {
println!("cargo:rerun-if-changed={}", file.file_path);
});
proto_crates
}