use std::env; use std::num::NonZeroUsize; use std::path::PathBuf; use std::process::Command; fn main() { let out_path = PathBuf::from(env::var("OUT_DIR").unwrap()); let regocpp_path = out_path.join("rego-cpp"); let num_procs = std::thread::available_parallelism().unwrap_or(NonZeroUsize::new(1).unwrap()); let rego_git_repo = std::env::var("REGOCPP_REPO") .unwrap_or("https://github.com/microsoft/rego-cpp.git".to_string()); let rego_git_tag = std::env::var("REGOCPP_TAG").unwrap_or("main".to_string()); if !regocpp_path.exists() { Command::new("git") .args(&["clone", ®o_git_repo]) .current_dir(&out_path) .status() .expect("failed to execute process"); } else { Command::new("git") .args(&["pull", "--rebase"]) .current_dir(®ocpp_path) .status() .expect("failed to execute process"); } Command::new("git") .args(&["checkout", ®o_git_tag]) .current_dir(®ocpp_path) .status() .expect("failed to execute process"); let regocpp_config = match std::env::var("PROFILE").as_ref() { Ok(build_type) => { if build_type == "debug" { "RelWithDebInfo" } else { "Release" } } Err(_) => "Release", }; Command::new("cmake") .args(&[ "-S", ".", "-B", "build", "-DCMAKE_BUILD_TYPE={}" .replace("{}", regocpp_config) .as_str(), "-DCMAKE_INSTALL_PREFIX=rust", "-DREGOCPP_COPY_EXAMPLES=ON", "-DREGOCPP_USE_MANUAL_TZDATA=ON" ]) .current_dir(®ocpp_path) .status() .expect("failed to execute process"); let regocpp_build_path = regocpp_path.join("build"); Command::new("cmake") .args(&[ "--build", ".", "--config", regocpp_config, "--parallel", &num_procs.to_string(), "--target", "install", ]) .current_dir(®ocpp_build_path) .status() .expect("failed to execute process"); let rego_path = regocpp_path.join("rust"); let libdir_path = rego_path.join("lib"); let header_path = rego_path.join("include").join("rego").join("rego_c.h"); let header_path_str = header_path.to_str().unwrap(); println!("cargo:rustc-link-search={}", libdir_path.to_str().unwrap()); println!("cargo:rustc-link-lib=static=re2"); println!("cargo:rustc-link-lib=static=snmalloc-new-override"); println!("cargo:rustc-link-lib=static=rego"); println!("cargo:rustc-link-lib=static=json"); println!("cargo:rustc-link-lib=static=yaml"); println!("cargo:rustc-link-lib=static=date-tz"); if cfg!(windows) { println!("cargo:rustc-link-arg=mincore.lib"); println!("cargo:rustc-link-arg=shell32.lib"); } else if cfg!(target_os = "macos") { println!("cargo:rustc-link-lib=c++"); } else { println!("cargo:rustc-link-lib=stdc++"); } println!("cargo:rerun-if-changed={}", header_path_str); // The bindgen::Builder is the main entry point // to bindgen, and lets you build up options for // the resulting bindings. let bindings = bindgen::Builder::default() // The input header we would like to generate // bindings for. .header(header_path_str) // Tell cargo to invalidate the built crate whenever any of the // included header files changed. .parse_callbacks(Box::new(bindgen::CargoCallbacks)) // Finish the builder and generate the bindings. .generate() // Unwrap the Result and panic on failure. .expect("Unable to generate bindings"); // Write the bindings to the $OUT_DIR/bindings.rs file. let out_path = PathBuf::from(env::var("OUT_DIR").unwrap()); bindings .write_to_file(out_path.join("bindings.rs")) .expect("Couldn't write bindings!"); }