1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
#![warn(missing_docs)]
use std::{convert, env, fmt, io, path::PathBuf, process, string};
pub fn locate_manifest() -> Result<PathBuf, LocateManifestError> {
let cargo = env::var("CARGO").unwrap_or("cargo".to_owned());
let output = process::Command::new(cargo)
.arg("locate-project")
.output()?;
if !output.status.success() {
return Err(LocateManifestError::CargoExecution {
stderr: output.stderr,
});
}
let output = String::from_utf8(output.stdout)?;
let parsed = json::parse(&output)?;
let root = parsed["root"].as_str().ok_or(LocateManifestError::NoRoot)?;
Ok(PathBuf::from(root))
}
#[derive(Debug)]
pub enum LocateManifestError {
Io(io::Error),
CargoExecution {
stderr: Vec<u8>,
},
StringConversion(string::FromUtf8Error),
ParseJson(json::Error),
NoRoot,
}
impl fmt::Display for LocateManifestError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
LocateManifestError::Io(err) => {
write!(f, "An I/O error occurred while trying to execute `cargo locate-project`: {}", err)
}
LocateManifestError::CargoExecution { stderr } => {
write!(f, "The command `cargo locate-project` did not exit successfully.\n\
Stderr: {}", String::from_utf8_lossy(stderr))
}
LocateManifestError::StringConversion(err) => {
write!(f, "The output of `cargo locate-project` was not valid UTF-8: {}", err)
}
LocateManifestError::ParseJson(err) => {
write!(f, "The output of `cargo locate-project` was not valid JSON: {}", err)
}
LocateManifestError::NoRoot => {
write!(f, "The JSON output of `cargo locate-project` did not contain the expected \"root\" string.")
}
}
}
}
impl std::error::Error for LocateManifestError {
fn source(&self) -> Option<&(dyn std::error::Error + 'static)> {
match self {
LocateManifestError::Io(err) => Some(err),
LocateManifestError::CargoExecution { stderr: _ } => None,
LocateManifestError::StringConversion(err) => Some(err),
LocateManifestError::ParseJson(err) => Some(err),
LocateManifestError::NoRoot => None,
}
}
}
impl convert::From<io::Error> for LocateManifestError {
fn from(source: io::Error) -> Self {
LocateManifestError::Io(source)
}
}
impl convert::From<string::FromUtf8Error> for LocateManifestError {
fn from(source: string::FromUtf8Error) -> Self {
LocateManifestError::StringConversion(source)
}
}
impl convert::From<json::Error> for LocateManifestError {
fn from(source: json::Error) -> Self {
LocateManifestError::ParseJson(source)
}
}
#[test]
fn test_manifest_path() {
use std::path::Path;
let manifest_path = locate_manifest().expect("failed to retrieve cargo manifest path");
let manual_path = Path::new(file!())
.parent()
.unwrap()
.join("../Cargo.toml")
.canonicalize()
.unwrap();
assert_eq!(manifest_path, manual_path);
}