vscode/cli/build.rs
Connor Peet 52840e3ca5
cli: cleanup build (#190213)
- Remove the `prepare` script entirely
- Variables are now populated from the product.json during build. Most
  variables are mapped automatically, with some special handling in a
	few cases. `build.rs` is now much more self-contained.
- Look for the `product.overrides.json` for vscode developers instead of
  looking for a peer `vscode-distro` folder

Fixes #178691
2023-08-10 20:14:30 -07:00

194 lines
5 KiB
Rust

/*---------------------------------------------------------------------------------------------
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the MIT License. See License.txt in the project root for license information.
*--------------------------------------------------------------------------------------------*/
const FILE_HEADER: &str = "/*---------------------------------------------------------------------------------------------\n * Copyright (c) Microsoft Corporation. All rights reserved.\n * Licensed under the MIT License. See License.txt in the project root for license information.\n *--------------------------------------------------------------------------------------------*/";
use std::{
collections::HashMap,
env, fs, io,
path::{Path, PathBuf},
process::{self},
str::FromStr,
};
use serde::{de::DeserializeOwned, Deserialize};
use serde_json::Value;
fn main() {
let files = enumerate_source_files().expect("expected to enumerate files");
ensure_file_headers(&files).expect("expected to ensure file headers");
apply_build_environment_variables();
}
fn camel_case_to_constant_case(key: &str) -> String {
let mut output = String::new();
let mut prev_upper = false;
for c in key.chars() {
if c.is_uppercase() {
if prev_upper {
output.push(c.to_ascii_lowercase());
} else {
output.push('_');
output.push(c.to_ascii_uppercase());
}
prev_upper = true;
} else if c.is_lowercase() {
output.push(c.to_ascii_uppercase());
prev_upper = false;
} else {
output.push(c);
prev_upper = false;
}
}
output
}
fn set_env_vars_from_map_keys(prefix: &str, map: impl IntoIterator<Item = (String, Value)>) {
let mut win32_app_ids = vec![];
for (key, value) in map {
//#region special handling
let value = match key.as_str() {
"tunnelServerQualities" | "serverLicense" => {
Value::String(serde_json::to_string(&value).unwrap())
}
"nameLong" => {
if let Value::String(s) = &value {
let idx = s.find(" - ");
println!(
"cargo:rustc-env=VSCODE_CLI_QUALITYLESS_PRODUCT_NAME={}",
idx.map(|i| &s[..i]).unwrap_or(s)
);
}
value
}
"tunnelApplicationConfig" => {
if let Value::Object(v) = value {
set_env_vars_from_map_keys(&format!("{}_{}", prefix, "TUNNEL"), v);
}
continue;
}
_ => value,
};
if key.contains("win32") && key.contains("AppId") {
if let Value::String(s) = value {
win32_app_ids.push(s);
continue;
}
}
//#endregion
if let Value::String(s) = value {
println!(
"cargo:rustc-env={}_{}={}",
prefix,
camel_case_to_constant_case(&key),
s
);
}
}
if !win32_app_ids.is_empty() {
println!(
"cargo:rustc-env=VSCODE_CLI_WIN32_APP_IDS={}",
win32_app_ids.join(",")
);
}
}
fn read_json_from_path<T>(path: &Path) -> T
where
T: DeserializeOwned,
{
let mut file = fs::File::open(path).expect("failed to open file");
serde_json::from_reader(&mut file).expect("failed to deserialize JSON")
}
fn apply_build_from_product_json(path: &Path) {
let json: HashMap<String, Value> = read_json_from_path(path);
set_env_vars_from_map_keys("VSCODE_CLI", json);
}
#[derive(Deserialize)]
struct PackageJson {
pub version: String,
}
fn apply_build_environment_variables() {
let repo_dir = env::current_dir().unwrap().join("..");
let package_json = read_json_from_path::<PackageJson>(&repo_dir.join("package.json"));
println!(
"cargo:rustc-env=VSCODE_CLI_VERSION={}",
package_json.version
);
match env::var("VSCODE_CLI_PRODUCT_JSON") {
Ok(v) => {
let path = if cfg!(windows) {
PathBuf::from_str(&v.replace('/', "\\")).unwrap()
} else {
PathBuf::from_str(&v).unwrap()
};
println!("cargo:warning=loading product.json from <{:?}>", path);
apply_build_from_product_json(&path);
}
Err(_) => {
apply_build_from_product_json(&repo_dir.join("product.json"));
let overrides = repo_dir.join("product.overrides.json");
if overrides.exists() {
apply_build_from_product_json(&overrides);
}
}
};
}
fn ensure_file_headers(files: &[PathBuf]) -> Result<(), io::Error> {
let mut ok = true;
let crlf_header_str = str::replace(FILE_HEADER, "\n", "\r\n");
let crlf_header = crlf_header_str.as_bytes();
let lf_header = FILE_HEADER.as_bytes();
for file in files {
let contents = fs::read(file)?;
if !(contents.starts_with(lf_header) || contents.starts_with(crlf_header)) {
eprintln!("File missing copyright header: {}", file.display());
ok = false;
}
}
if !ok {
process::exit(1);
}
Ok(())
}
/// Gets all "rs" files in the source directory
fn enumerate_source_files() -> Result<Vec<PathBuf>, io::Error> {
let mut files = vec![];
let mut queue = vec![];
let current_dir = env::current_dir()?.join("src");
queue.push(current_dir);
while !queue.is_empty() {
for entry in fs::read_dir(queue.pop().unwrap())? {
let entry = entry?;
let ftype = entry.file_type()?;
if ftype.is_dir() {
queue.push(entry.path());
} else if ftype.is_file() && entry.file_name().to_string_lossy().ends_with(".rs") {
files.push(entry.path());
}
}
}
Ok(files)
}