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

151 lines
4.6 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;
use std::path::PathBuf;
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");
let protoc_path = protoc_bin_vendored::protoc_bin_path().unwrap();
2022-02-09 05:08:49 +00:00
#[cfg(feature = "dart")]
gen_pb_for_dart(crate_name, root, &paths, &file_names, &protoc_path);
2022-02-09 05:08:49 +00:00
protoc_rust::Codegen::new()
.out_dir("./src/protobuf/model")
.protoc_path(protoc_path)
2022-02-09 05:08:49 +00:00
.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>, proto_path: &PathBuf) {
2022-02-09 10:35:50 +00:00
if std::env::var("CARGO_MAKE_WORKING_DIRECTORY").is_err() {
log::warn!("CARGO_MAKE_WORKING_DIRECTORY was not set, skip generate dart pb");
return;
}
if std::env::var("FLUTTER_FLOWY_SDK_PATH").is_err() {
log::warn!("FLUTTER_FLOWY_SDK_PATH was not set, skip generate dart pb");
return;
}
let workspace_dir = std::env::var("CARGO_MAKE_WORKING_DIRECTORY").unwrap();
let flutter_sdk_path = std::env::var("FLUTTER_FLOWY_SDK_PATH").unwrap();
let output = format!("{}/{}/{}", workspace_dir, flutter_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_dart_plugin();
let proto_path = proto_path.to_str().unwrap().to_owned();
paths.iter().for_each(|path| {
if cmd_lib::run_cmd! {
${proto_path} --dart_out=${output} --proto_path=${root} ${path}
}
.is_err()
{
panic!("Generate dart 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
}