我的工作区中有以下项目:
Workspacefolder
|
+-- Project A
| |
| +-- build.rs
|
+-- Dep
| |
| +-- test.json
|
+-Cargo.toml
在
Project A
中,有build.rs
想要以不依赖平台且与CI配合良好的方式打开test.json
。
我正在寻找一个
CARGO_WORKSPACE
环境变量,因为这样我就可以说 Path::new(&workspace_dir).join("/Dep/test.json")
。
现在有一个更简单的方法:
fn workspace_dir() -> PathBuf {
let output = std::process::Command::new(env!("CARGO"))
.arg("locate-project")
.arg("--workspace")
.arg("--message-format=plain")
.output()
.unwrap()
.stdout;
let cargo_path = Path::new(std::str::from_utf8(&output).unwrap().trim());
cargo_path.parent().unwrap().to_path_buf()
}
不,不适用于与 Rust 1.16.0 捆绑的 Cargo 版本。您可以通过打印构建脚本中的所有环境变量来自行验证这一点:
use std::fs::File;
use std::io::Write;
fn main() {
let mut dump = File::create("/tmp/dump").expect("unable to open");
for (k, v) in std::env::vars() {
writeln!(&mut dump, "{} -> {}", k, v).expect("unable to write")
}
}
在我的机器上,这会产生:
$ sort /tmp/dump | grep CARGO
CARGO_CFG_DEBUG_ASSERTIONS ->
CARGO_CFG_TARGET_ARCH -> x86_64
CARGO_CFG_TARGET_ENDIAN -> little
CARGO_CFG_TARGET_ENV ->
CARGO_CFG_TARGET_FAMILY -> unix
CARGO_CFG_TARGET_OS -> macos
CARGO_CFG_TARGET_POINTER_WIDTH -> 64
CARGO_CFG_UNIX ->
CARGO_HOME -> /Users/shep/.cargo
CARGO_MANIFEST_DIR -> /private/tmp/the-workspace/project-a
CARGO_PKG_AUTHORS -> An Devloper <[email protected]>
CARGO_PKG_DESCRIPTION ->
CARGO_PKG_HOMEPAGE ->
CARGO_PKG_NAME -> project-a
CARGO_PKG_VERSION -> 0.1.0
CARGO_PKG_VERSION_MAJOR -> 0
CARGO_PKG_VERSION_MINOR -> 1
CARGO_PKG_VERSION_PATCH -> 0
CARGO_PKG_VERSION_PRE ->
我不知道为什么你不能这样做
Path::new(&manifest_dir).join("..").join("Dep").join("test.json")
我已将每个目录拆分为一个单独的调用 - 完全无需指定目录分隔符来与平台无关。
对于货物版本 1.63.0,我管理的是:
use std::{env, path::PathBuf, process::Command};
pub fn get_workspace_root() -> anyhow::Result<PathBuf> {
let current_dir = env::current_dir()?;
let cmd_output = Command::new("cargo")
.args(["metadata", "--format-version=1"])
.output()?;
if !cmd_output.status.success() {
return Ok(current_dir);
}
let json =
serde_json::from_str::<serde_json::Value>(String::from_utf8(cmd_output.stdout)?.as_str())?;
let path = match json.get("workspace_root") {
Some(val) => match val.as_str() {
Some(val) => val,
None => return Ok(current_dir),
},
None => return Ok(current_dir),
};
Ok(PathBuf::from(path))
}