#![cfg_attr(feature = "clippy", feature(plugin))]
#![cfg_attr(feature = "clippy", plugin(clippy))]
use fs_extra::dir::CopyOptions;
use glob::glob;
use std::env;
use std::path::{Path, PathBuf};
use std::process::Command;
fn main() -> Result<(), Box<dyn std::error::Error>> {
let out_dir = PathBuf::from(env::var("OUT_DIR")?);
let build_path = Path::new(".").join("libpg_query");
let out_header_path = out_dir.join("pg_query").with_extension("h");
let out_protobuf_path = out_dir.join("protobuf");
let target = env::var("TARGET").unwrap();
println!("cargo:rerun-if-changed=libpg_query");
println!("cargo:rustc-link-search=native={}", out_dir.display());
println!("cargo:rustc-link-lib=static=pg_query");
let source_paths = vec![
build_path.join("pg_query").with_extension("h"),
build_path.join("Makefile"),
build_path.join("src"),
build_path.join("protobuf"),
build_path.join("vendor"),
];
let copy_options = CopyOptions { overwrite: true, ..CopyOptions::default() };
fs_extra::copy_items(&source_paths, &out_dir, ©_options)?;
let mut build = cc::Build::new();
build
.files(glob(out_dir.join("src/*.c").to_str().unwrap()).unwrap().map(|p| p.unwrap()))
.files(glob(out_dir.join("src/postgres/*.c").to_str().unwrap()).unwrap().map(|p| p.unwrap()))
.file(out_dir.join("vendor/protobuf-c/protobuf-c.c"))
.file(out_dir.join("vendor/xxhash/xxhash.c"))
.file(out_dir.join("protobuf/pg_query.pb-c.c"))
.include(out_dir.join("."))
.include(out_dir.join("./vendor"))
.include(out_dir.join("./src/postgres/include"))
.include(out_dir.join("./src/include"))
.warnings(false); if env::var("PROFILE").unwrap() == "debug" || env::var("DEBUG").unwrap() == "1" {
build.define("USE_ASSERT_CHECKING", None);
}
if target.contains("windows") {
build.include(out_dir.join("./src/postgres/include/port/win32"));
if target.contains("msvc") {
build.include(out_dir.join("./src/postgres/include/port/win32_msvc"));
}
}
build.compile("pg_query");
bindgen::Builder::default()
.header(out_header_path.to_str().ok_or("Invalid header path")?)
.generate()
.map_err(|_| "Unable to generate bindings")?
.write_to_file(out_dir.join("bindings.rs"))?;
let protoc_exists = Command::new("protoc").arg("--version").status().is_ok();
let is_built_by_docs_rs = env::var("DOCS_RS").is_ok();
if !is_built_by_docs_rs && (env::var("REGENERATE_PROTOBUF").is_ok() || protoc_exists) {
println!("generating protobuf bindings");
let src_dir = PathBuf::from(env::var("CARGO_MANIFEST_DIR")?).join("src");
env::set_var("OUT_DIR", &src_dir);
let mut prost_build = prost_build::Config::new();
prost_build.type_attribute(".", "#[derive(serde::Serialize)]");
prost_build.compile_protos(&[&out_protobuf_path.join("pg_query").with_extension("proto")], &[&out_protobuf_path])?;
std::fs::rename(src_dir.join("pg_query.rs"), src_dir.join("protobuf.rs"))?;
env::set_var("OUT_DIR", &out_dir);
} else {
println!("skipping protobuf generation");
}
Ok(())
}