git_version_macro/
utils.rs

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
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
use std::ffi::OsStr;
use std::path::{PathBuf, Path};
use std::process::Command;

/// Run `git describe` for the current working directory with custom flags to get version information from git.
pub fn describe<I, S>(dir: impl AsRef<Path>, args: I) -> Result<String, String>
where
	I: IntoIterator<Item = S>,
	S: AsRef<OsStr>,
{
	let dir = dir.as_ref();
	run_git("git describe", Command::new("git")
		.arg("-C")
		.arg(dir)
		.arg("describe").args(args))
}

/// Get the git directory for the given directory.
pub fn git_dir(dir: impl AsRef<Path>) -> Result<PathBuf, String> {
	let dir = dir.as_ref();
	let path = run_git("git rev-parse", Command::new("git")
		.arg("-C")
		.arg(dir)
		.args(["rev-parse", "--git-dir"]))?;
	Ok(dir.join(path))
}

/// Run `git submodule foreach` command to discover submodules in the project.
pub fn get_submodules(dir: impl AsRef<Path>) -> Result<Vec<String>, String> {
	let dir = dir.as_ref();
	let result = run_git("git submodule",
		Command::new("git")
			.arg("-C")
			.arg(dir)
			.arg("submodule")
			.arg("foreach")
			.arg("--quiet")
			.arg("--recursive")
			.arg("echo $displaypath"),
	)?;

	Ok(result.lines()
		.filter(|x| !x.is_empty())
		.map(|x| x.to_owned())
		.collect()
	)
}

pub fn canonicalize_path(path: &Path) -> syn::Result<String> {
	path.canonicalize()
		.map_err(|e| error!("failed to canonicalize {}: {}", path.display(), e))?
		.into_os_string()
		.into_string()
		.map_err(|file| error!("invalid UTF-8 in path to {}", PathBuf::from(file).display()))
}

/// Create a token stream representing dependencies on the git state.
pub fn git_dependencies() -> syn::Result<proc_macro2::TokenStream> {
	let manifest_dir = std::env::var_os("CARGO_MANIFEST_DIR")
		.ok_or_else(|| error!("CARGO_MANIFEST_DIR is not set"))?;
	let git_dir = git_dir(manifest_dir).map_err(|e| error!("failed to determine .git directory: {}", e))?;

	let deps: Vec<_> = ["logs/HEAD", "index"]
		.iter()
		.flat_map(|&file| {
			canonicalize_path(&git_dir.join(file))
			.map_err(|e| eprintln!("Failed to add dependency on the git state: {}. Git state changes might not trigger a rebuild.", e))
			.ok()
		})
		.collect();

	Ok(quote::quote! {
		#( include_bytes!(#deps); )*
	})
}

fn run_git(program: &str, command: &mut std::process::Command) -> Result<String, String> {
	let output = command
		.stdout(std::process::Stdio::piped())
		.stderr(std::process::Stdio::piped())
		.spawn()
		.map_err(|e| {
			if e.kind() == std::io::ErrorKind::NotFound {
				format!("Command `{}` not found: is git installed?", command.get_program().to_string_lossy())
			} else {
				format!("Failed to run `{}`: {}", command.get_program().to_string_lossy(), e)
			}
		})?
		.wait_with_output()
		.map_err(|e| format!("Failed to wait for `{}`: {}", program, e))?;

	let output = collect_output(program, output)?;
	let output = strip_trailing_newline(output);
	let output =
		String::from_utf8(output).map_err(|_| format!("Failed to parse output of `{}`: output contains invalid UTF-8", program))?;
	Ok(output)
}

/// Check if a command ran successfully, and if not, return a verbose error.
fn collect_output(program: &str, output: std::process::Output) -> Result<Vec<u8>, String> {
	// If the command succeeded, just return the output as is.
	if output.status.success() {
		return Ok(output.stdout);

	// If the command terminated with non-zero exit code, return an error.
	} else if let Some(status) = output.status.code() {
		// Include the first line of stderr in the error message, if it's valid UTF-8 and not empty.
		let message = output
			.stderr
			.split(|c| *c == b'\n')
			.next()
			.and_then(|x| std::str::from_utf8(x).ok())
			.filter(|x| !x.is_empty());
		if let Some(message) = message {
			return Err(format!("{} exited with status {}: {}", program, status, message));
		} else {
			return Err(format!("{} exited with status {}", program, status));
		}
	}

	// The command was killed by a signal.
	#[cfg(unix)]
	{
		use std::os::unix::process::ExitStatusExt;
		if let Some(signal) = output.status.signal() {
			// Include the signal number on Unix.
			return Err(format!("{} killed by signal {}", program, signal));
		}
	}

	Err(format!("{} exitted with error", program))
}

/// Remove a trailing newline from a byte string.
fn strip_trailing_newline(mut input: Vec<u8>) -> Vec<u8> {
	if input.last().copied() == Some(b'\n') {
		input.pop();
	}
	input
}

#[test]
fn test_git_dir() {
	use assert2::{assert, let_assert};
	use std::path::Path;

	let_assert!(Ok(git_dir) = git_dir("."));
	let_assert!(Ok(git_dir) = git_dir.canonicalize());
	let_assert!(Ok(expected) = Path::new(env!("CARGO_MANIFEST_DIR")).join("../.git").canonicalize());
	assert!(git_dir == expected);
}