-
Notifications
You must be signed in to change notification settings - Fork 106
/
build.rs
113 lines (99 loc) · 3.66 KB
/
build.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
use regex::Regex;
use std::env;
use std::fs::{self, read_to_string, File};
use std::io::Write;
use std::path::Path;
use std::process::Command;
use std::result::Result;
const COMMAND_REGEX: &str = r"pub fn (.*)\(app: &mut Application\) -> Result";
fn main() {
generate_commands();
set_build_revision();
}
/// This build task generates a Rust snippet which, when included later on in
/// build process, adds logic to construct a HashMap<String, Command> for all
/// public commands declared in the commands module. This facilitates runtime
/// command referencing via string, which is required for command mode, as well
/// as user-defined keymaps.
fn generate_commands() {
let mut output = create_output_file().unwrap();
write_commands(&mut output).unwrap();
finalize_output_file(&mut output).unwrap();
}
fn create_output_file() -> Result<File, String> {
let out_dir = env::var("OUT_DIR").expect("The compiler did not provide $OUT_DIR");
let out_file: std::path::PathBuf = [&out_dir, "hash_map"].iter().collect();
let mut file = File::create(&out_file).map_err(|_| {
format!(
"Couldn't create output file: {}",
out_file.to_string_lossy()
)
})?;
file.write(
"{\n let mut commands: HashMap<&'static str, Command> = HashMap::new();\n".as_bytes(),
)
.map_err(|_| "Failed to write command hash init")?;
Ok(file)
}
fn write_commands(output: &mut File) -> Result<(), &str> {
let expression = Regex::new(COMMAND_REGEX).expect("Failed to compile command matching regex");
let entries =
fs::read_dir("./src/commands/").map_err(|_| "Failed to read command module directory")?;
for entry in entries {
let path = entry
.map_err(|_| "Failed to read command module directory entry")?
.path();
let module_name = module_name(&path).unwrap();
let content = read_to_string(&path).map_err(|_| "Failed to read command module data")?;
for captures in expression.captures_iter(&content) {
let function_name = captures.get(1).unwrap().as_str();
write_command(output, &module_name, function_name)?;
}
}
Ok(())
}
fn write_command(
output: &mut File,
module_name: &str,
function_name: &str,
) -> Result<usize, &'static str> {
output
.write(
format!(
" commands.insert(\"{module_name}::{function_name}\", {module_name}::{function_name});\n"
)
.as_bytes(),
)
.map_err(|_| "Failed to write command")
}
fn finalize_output_file(output: &mut File) -> Result<usize, &str> {
output
.write(" commands\n}\n".as_bytes())
.map_err(|_| "Failed to write command hash return")
}
fn module_name(path: &Path) -> Result<String, &str> {
path.file_name()
.and_then(|name| {
name.to_string_lossy()
.split('.')
.next()
.map(|n| n.to_string())
})
.ok_or("Unable to parse command module from file name")
}
fn set_build_revision() {
// Skip if the environment variable is already set
let revision = env::var("BUILD_REVISION");
if revision.map(|r| !r.is_empty()) == Ok(true) {
return;
}
// Run the Git command to get the current commit hash
let output = Command::new("git")
.args(&["rev-parse", "--short", "HEAD"])
.output()
.expect("Failed to execute git command");
// Parse the hash
let build_revision = String::from_utf8(output.stdout).expect("Invalid UTF-8 sequence");
// Write the hash as an environment variable
println!("cargo:rustc-env=BUILD_REVISION={}", build_revision.trim());
}