From c4468b355d94ac46bf1a47d55eabc4a8b55b9b02 Mon Sep 17 00:00:00 2001 From: Adam Cozzette Date: Wed, 11 Dec 2024 16:59:04 -0800 Subject: [PATCH] Rust: update protobuf_codegen crate to rely on separately installed protoc PiperOrigin-RevId: 705288486 --- rust/BUILD | 15 --- rust/release_crates/BUILD | 2 + rust/release_crates/cargo_test.sh | 6 +- rust/release_crates/protobuf_codegen/BUILD | 1 - .../protobuf_codegen/src/lib.rs | 101 +++++++----------- upb_generator/minitable/BUILD | 2 +- 6 files changed, 45 insertions(+), 82 deletions(-) diff --git a/rust/BUILD b/rust/BUILD index 9ec6bc197f5b..99c1c95b5dc5 100644 --- a/rust/BUILD +++ b/rust/BUILD @@ -291,18 +291,3 @@ pkg_filegroup( prefix = "libupb", visibility = ["//rust/release_crates:__subpackages__"], ) - -# Bundle only the linux-x86_64 protoc for testing. -pkg_cross_compiled_binaries( - name = "vendored_protocs_test", - cpus = [ - "linux-x86_64", - ], - prefix = "bin", - tags = ["manual"], - targets = [ - "//upb_generator/minitable:protoc-gen-upb_minitable", - "//:protoc", - ], - visibility = ["//rust/release_crates:__subpackages__"], -) diff --git a/rust/release_crates/BUILD b/rust/release_crates/BUILD index 960eedf33f48..ae90441d9c8d 100644 --- a/rust/release_crates/BUILD +++ b/rust/release_crates/BUILD @@ -8,6 +8,8 @@ sh_binary( "//rust/release_crates/protobuf:protobuf_crate", "//rust/release_crates/protobuf_codegen:protobuf_codegen_crate", "//rust/release_crates/protobuf_example:protobuf_example_crate", + "//src/google/protobuf/compiler:protoc", + "//upb_generator/minitable:protoc-gen-upb_minitable", ], tags = ["manual"], deps = ["@bazel_tools//tools/bash/runfiles"], diff --git a/rust/release_crates/cargo_test.sh b/rust/release_crates/cargo_test.sh index 598fe019900f..54062a9a6dc5 100755 --- a/rust/release_crates/cargo_test.sh +++ b/rust/release_crates/cargo_test.sh @@ -56,7 +56,11 @@ mkdir $EXAMPLE_ROOT EXAMPLE_TAR=$(rlocation com_google_protobuf/rust/release_crates/protobuf_example/protobuf_example_crate.tar) echo "Expanding protobuf_example crate tar" -tar -xvf $EXAMPLE_TAR -C $EXAMPLE_ROOT +tar -xvf $EXAMPLE_TAR -C $EXAMPLE_ROOT + +# Put the Bazel-built protoc and plugin at the beginning of $PATH +PATH=$(dirname $(rlocation com_google_protobuf/protoc)):$PATH +PATH=$(dirname $(rlocation com_google_protobuf/upb_generator/minitable/protoc-gen-upb_minitable)):$PATH cd $CRATE_ROOT CARGO_HOME=$CARGO_HOME cargo test diff --git a/rust/release_crates/protobuf_codegen/BUILD b/rust/release_crates/protobuf_codegen/BUILD index e2ddf85deec0..2c9f18bce878 100644 --- a/rust/release_crates/protobuf_codegen/BUILD +++ b/rust/release_crates/protobuf_codegen/BUILD @@ -7,7 +7,6 @@ pkg_tar( srcs = [ ":protobuf_codegen_files", "//:LICENSE", - "//rust:vendored_protocs_test", ], tags = ["manual"], visibility = ["//rust:__subpackages__"], diff --git a/rust/release_crates/protobuf_codegen/src/lib.rs b/rust/release_crates/protobuf_codegen/src/lib.rs index 68f394f07a1e..2ee6088f354f 100644 --- a/rust/release_crates/protobuf_codegen/src/lib.rs +++ b/rust/release_crates/protobuf_codegen/src/lib.rs @@ -4,8 +4,6 @@ use std::path::{Path, PathBuf}; pub struct CodeGen { inputs: Vec, output_dir: PathBuf, - protoc_path: Option, - protoc_gen_upb_minitable_path: Option, includes: Vec, } @@ -16,8 +14,6 @@ impl CodeGen { Self { inputs: Vec::new(), output_dir: PathBuf::from(std::env::var("OUT_DIR").unwrap()).join("protobuf_generated"), - protoc_path: None, - protoc_gen_upb_minitable_path: None, includes: Vec::new(), } } @@ -37,20 +33,6 @@ impl CodeGen { self } - pub fn protoc_path(&mut self, protoc_path: impl AsRef) -> &mut Self { - self.protoc_path = Some(protoc_path.as_ref().to_owned()); - self - } - - pub fn protoc_gen_upb_minitable_path( - &mut self, - protoc_gen_upb_minitable_path: impl AsRef, - ) -> &mut Self { - self.protoc_gen_upb_minitable_path = - Some(protoc_gen_upb_minitable_path.as_ref().to_owned()); - self - } - pub fn include(&mut self, include: impl AsRef) -> &mut Self { self.includes.push(include.as_ref().to_owned()); self @@ -83,6 +65,33 @@ impl CodeGen { .collect() } + fn protoc_version() -> String { + let output = std::process::Command::new("protoc").arg("--version").output().unwrap().stdout; + + // The output of protoc --version looks something like "libprotoc XX.Y", with a + // possible suffix starting with a dash. We want to return just the + // "XX.Y" part. + let mut s = + String::from_utf8(output).unwrap().strip_prefix("libprotoc ").unwrap().to_string(); + let first_dash = s.find('-'); + if let Some(i) = first_dash { + s.truncate(i); + } + s + } + + fn expected_protoc_version() -> String { + let mut s = VERSION.to_string(); + let first_dash = s.find('-'); + if let Some(i) = first_dash { + s.truncate(i); + } + let mut v: Vec<&str> = s.split('.').collect(); + assert_eq!(v.len(), 3); + v.remove(0); + v.join(".") + } + pub fn generate_and_compile(&self) -> Result<(), String> { let upb_version = std::env::var("DEP_UPB_VERSION").expect("DEP_UPB_VERSION should have been set, make sure that the Protobuf crate is a dependency"); if VERSION != upb_version { @@ -92,12 +101,16 @@ impl CodeGen { ); } - let protoc_path = if let Some(path) = &self.protoc_path { - path.clone() - } else { - protoc_path().expect("To be a supported platform") - }; - let mut cmd = std::process::Command::new(protoc_path); + let protoc_version = Self::protoc_version(); + let expected_protoc_version = Self::expected_protoc_version(); + if protoc_version != expected_protoc_version { + panic!( + "Expected protoc version {} but found {}", + expected_protoc_version, protoc_version + ); + } + + let mut cmd = std::process::Command::new("protoc"); for input in &self.inputs { cmd.arg(input); } @@ -105,12 +118,6 @@ impl CodeGen { // Attempt to make the directory if it doesn't exist let _ = std::fs::create_dir(&self.output_dir); } - let protoc_gen_upb_minitable_path = if let Some(path) = &self.protoc_gen_upb_minitable_path - { - path.clone() - } else { - protoc_gen_upb_minitable_path().expect("To be a supported platform") - }; for include in &self.includes { println!("cargo:rerun-if-changed={}", include.display()); @@ -118,10 +125,6 @@ impl CodeGen { cmd.arg(format!("--rust_out={}", self.output_dir.display())) .arg("--rust_opt=experimental-codegen=enabled,kernel=upb") - .arg(format!( - "--plugin=protoc-gen-upb_minitable={}", - protoc_gen_upb_minitable_path.display() - )) .arg(format!("--upb_minitable_out={}", self.output_dir.display())); for include in &self.includes { cmd.arg(format!("--proto_path={}", include.display())); @@ -161,33 +164,3 @@ impl CodeGen { Ok(()) } } - -fn get_path_for_arch() -> Option { - let mut path = PathBuf::from(env!("CARGO_MANIFEST_DIR")); - path.push("bin"); - match (std::env::consts::OS, std::env::consts::ARCH) { - ("macos", "x86_64") => path.push("osx-x86_64"), - ("macos", "aarch64") => path.push("osx-aarch_64"), - ("linux", "aarch64") => path.push("linux-aarch_64"), - ("linux", "powerpc64") => path.push("linux-ppcle_64"), - ("linux", "s390x") => path.push("linux-s390_64"), - ("linux", "x86") => path.push("linux-x86_32"), - ("linux", "x86_64") => path.push("linux-x86_64"), - ("windows", "x86") => path.push("win32"), - ("windows", "x86_64") => path.push("win64"), - _ => return None, - }; - Some(path) -} - -pub fn protoc_path() -> Option { - let mut path = get_path_for_arch()?; - path.push("protoc"); - Some(path) -} - -pub fn protoc_gen_upb_minitable_path() -> Option { - let mut path = get_path_for_arch()?; - path.push("protoc-gen-upb_minitable"); - Some(path) -} diff --git a/upb_generator/minitable/BUILD b/upb_generator/minitable/BUILD index a5eadfed9344..83e75b0d7d7c 100644 --- a/upb_generator/minitable/BUILD +++ b/upb_generator/minitable/BUILD @@ -93,7 +93,7 @@ bootstrap_cc_binary( visibility = [ "//editions/codegen_tests:__pkg__", "//net/proto2/contrib/protoc_explorer:__pkg__", - "//rust:__pkg__", + "//rust:__subpackages__", "//third_party/prototiller/transformer:__pkg__", ], )