use std::fs;
use std::path::Path;
use tonic_build::Builder;
fn main() -> Result<(), Box<dyn std::error::Error>> {
let out_dir = std::env::var("OUT_DIR").unwrap();
println!("cargo:warning=Proto output dir: {}", out_dir);
let builder = tonic_build::configure()
.protoc_arg("--experimental_allow_proto3_optional")
.build_server(false)
.compile_well_known_types(true)
.extern_path(
".google.protobuf.BytesValue",
"::prost::alloc::vec::Vec<u8>",
)
.extern_path(
".google.protobuf.StringValue",
"::prost::alloc::string::String",
)
.extern_path(".google.protobuf", "::prost_wkt_types")
.type_attribute(
".",
"#[derive(::serde_derive::Serialize, ::serde_derive::Deserialize)]",
)
.type_attribute(".", "#[serde(rename_all = \"camelCase\")]");
let builder = add_field_attributes(builder);
builder
.compile_protos(&["proto/api.proto", "proto/common.proto"], &["proto"])
.unwrap();
let code_snippet = r#"// This code snippet is custom inserted by the build script.
// Since the generated code does not support deserializing a string to a u64,
// we need to add a custom deserializer function and add in serde annotatotions to individual
// fields below that need this.
// See build.rs for more details.
use serde::Deserialize;
use base64::{Engine as _, engine::general_purpose};
pub fn string_to_u64<'de, D>(deserializer: D) -> Result<u64, D::Error>
where
D: serde::Deserializer<'de>,
{
let s = String::deserialize(deserializer)?;
s.parse::<u64>().map_err(serde::de::Error::custom)
}
pub fn string_to_i64<'de, D>(deserializer: D) -> Result<i64, D::Error>
where
D: serde::Deserializer<'de>,
{
let s = String::deserialize(deserializer)?;
s.parse::<i64>().map_err(serde::de::Error::custom)
}
pub fn string_to_u8s<'de, D>(deserializer: D) -> Result<Vec<u8>, D::Error>
where
D: serde::Deserializer<'de>,
{
let s = <String>::deserialize(deserializer).map_err(serde::de::Error::custom)?;
general_purpose::STANDARD.decode(s).map_err(serde::de::Error::custom)
}
pub fn string_to_f64<'de, D>(deserializer: D) -> Result<f64, D::Error>
where
D: serde::Deserializer<'de>,
{
let s = String::deserialize(deserializer)?;
s.parse::<f64>().map_err(serde::de::Error::custom)
}
pub fn string_to_u64s<'de, D>(deserializer: D) -> Result<Vec<u64>, D::Error>
where
D: serde::Deserializer<'de>,
{
let s = Vec::<String>::deserialize(deserializer)?;
s.into_iter()
.map(|item| item.parse::<u64>().map_err(serde::de::Error::custom))
.collect()
}
// End of custom code snippet
"#;
let generated_file_path = Path::new(&out_dir).join("api.rs");
let mut generated_code = fs::read_to_string(&generated_file_path)?;
generated_code = format!("{}{}", code_snippet, generated_code);
generated_code = modify_64bit_fields(generated_code);
fs::write(generated_file_path, generated_code)?;
Ok(())
}
fn modify_64bit_fields(content: String) -> String {
let re = regex::Regex::new(r"( *)(pub\s+)?(\w+\s*:\s*(?:::prost::alloc::vec::Vec<)?([ui](64|8)>?).*)").unwrap();
re.replace_all(&content, |caps: ®ex::Captures| {
let padding = &caps[1];
let access_modifier = &caps[2];
let field = &caps[3];
let mut field_type = String::from(&caps[4]);
field_type = field_type.replace(">", "s");
format!(
"{}#[serde(deserialize_with = \"string_to_{}\")]\n{}{}{}",
padding,
field_type,
padding,
access_modifier,
field
)
}).to_string()
}
fn add_field_attributes(builder: Builder) -> Builder {
builder
.field_attribute("programID", "#[serde(rename = \"programID\")]")
.field_attribute("accountID", "#[serde(rename = \"accountID\")]")
}