rust-analyzer/editors/code/src/toolchain.ts

221 lines
7.5 KiB
TypeScript
Raw Normal View History

2022-05-17 17:15:06 +00:00
import * as cp from "child_process";
import * as os from "os";
import * as path from "path";
import * as readline from "readline";
import * as vscode from "vscode";
import { execute, log, memoizeAsync } from "./util";
import { unwrapNullable } from "./nullable";
import { unwrapUndefinable } from "./undefinable";
interface CompilationArtifact {
fileName: string;
name: string;
kind: string;
isTest: boolean;
}
2020-05-20 18:03:49 +00:00
export interface ArtifactSpec {
cargoArgs: string[];
filter?: (artifacts: CompilationArtifact[]) => CompilationArtifact[];
}
export class Cargo {
constructor(
readonly rootFolder: string,
readonly output: vscode.OutputChannel,
2023-07-11 13:35:10 +00:00
readonly env: Record<string, string>,
) {}
2020-05-20 18:03:49 +00:00
// Made public for testing purposes
static artifactSpec(args: readonly string[]): ArtifactSpec {
const cargoArgs = [...args, "--message-format=json"];
// arguments for a runnable from the quick pick should be updated.
// see crates\rust-analyzer\src\main_loop\handlers.rs, handle_code_lens
switch (cargoArgs[0]) {
2022-05-17 17:15:06 +00:00
case "run":
cargoArgs[0] = "build";
break;
case "test": {
if (!cargoArgs.includes("--no-run")) {
cargoArgs.push("--no-run");
}
break;
2020-05-20 18:03:49 +00:00
}
}
const result: ArtifactSpec = { cargoArgs: cargoArgs };
if (cargoArgs[0] === "test" || cargoArgs[0] === "bench") {
// for instance, `crates\rust-analyzer\tests\heavy_tests\main.rs` tests
// produce 2 artifacts: {"kind": "bin"} and {"kind": "test"}
2022-05-17 17:15:06 +00:00
result.filter = (artifacts) => artifacts.filter((it) => it.isTest);
}
2020-05-20 18:03:49 +00:00
return result;
}
2020-05-20 18:03:49 +00:00
private async getArtifacts(spec: ArtifactSpec): Promise<CompilationArtifact[]> {
const artifacts: CompilationArtifact[] = [];
try {
2022-05-17 17:15:06 +00:00
await this.runCargo(
spec.cargoArgs,
(message) => {
if (message.reason === "compiler-artifact" && message.executable) {
const isBinary = message.target.crate_types.includes("bin");
const isBuildScript = message.target.kind.includes("custom-build");
if ((isBinary && !isBuildScript) || message.profile.test) {
artifacts.push({
fileName: message.executable,
name: message.target.name,
kind: message.target.kind[0],
2022-05-17 17:15:06 +00:00
isTest: message.profile.test,
2020-04-30 15:41:48 +00:00
});
}
2022-05-17 17:15:06 +00:00
} else if (message.reason === "compiler-message") {
this.output.append(message.message.rendered);
}
},
2023-07-11 13:35:10 +00:00
(stderr) => this.output.append(stderr),
);
2020-05-05 22:22:02 +00:00
} catch (err) {
this.output.show(true);
throw new Error(`Cargo invocation has failed: ${err}`);
}
return spec.filter?.(artifacts) ?? artifacts;
}
async executableFromArgs(args: readonly string[]): Promise<string> {
const artifacts = await this.getArtifacts(Cargo.artifactSpec(args));
2020-04-30 15:41:48 +00:00
if (artifacts.length === 0) {
2022-05-17 17:15:06 +00:00
throw new Error("No compilation artifacts");
} else if (artifacts.length > 1) {
2022-05-17 17:15:06 +00:00
throw new Error("Multiple compilation artifacts are not supported.");
}
const artifact = unwrapUndefinable(artifacts[0]);
return artifact.fileName;
}
2021-08-15 16:19:45 +00:00
private async runCargo(
cargoArgs: string[],
onStdoutJson: (obj: any) => void,
2023-07-11 13:35:10 +00:00
onStderrString: (data: string) => void,
): Promise<number> {
2021-08-15 16:19:45 +00:00
const path = await cargoPath();
return await new Promise((resolve, reject) => {
const cargo = cp.spawn(path, cargoArgs, {
2022-05-17 17:15:06 +00:00
stdio: ["ignore", "pipe", "pipe"],
cwd: this.rootFolder,
env: this.env,
});
2022-05-17 17:15:06 +00:00
cargo.on("error", (err) => reject(new Error(`could not launch cargo: ${err}`)));
2020-05-05 22:22:02 +00:00
2022-05-17 17:15:06 +00:00
cargo.stderr.on("data", (chunk) => onStderrString(chunk.toString()));
2020-04-30 15:41:48 +00:00
const rl = readline.createInterface({ input: cargo.stdout });
2022-05-17 17:15:06 +00:00
rl.on("line", (line) => {
2020-04-30 15:41:48 +00:00
const message = JSON.parse(line);
onStdoutJson(message);
});
2022-05-17 17:15:06 +00:00
cargo.on("exit", (exitCode, _) => {
if (exitCode === 0) resolve(exitCode);
else reject(new Error(`exit code: ${exitCode}.`));
});
});
}
2020-05-05 22:22:02 +00:00
}
/** Mirrors `project_model::sysroot::discover_sysroot_dir()` implementation*/
2021-08-15 16:19:45 +00:00
export async function getSysroot(dir: string): Promise<string> {
const rustcPath = await getPathForExecutable("rustc");
// do not memoize the result because the toolchain may change between runs
2021-08-15 16:19:45 +00:00
return await execute(`${rustcPath} --print sysroot`, { cwd: dir });
}
export async function getRustcId(dir: string): Promise<string> {
2021-08-15 16:19:45 +00:00
const rustcPath = await getPathForExecutable("rustc");
// do not memoize the result because the toolchain may change between runs
const data = await execute(`${rustcPath} -V -v`, { cwd: dir });
const rx = /commit-hash:\s(.*)$/m;
const result = unwrapNullable(rx.exec(data));
const first = unwrapUndefinable(result[1]);
return first;
}
2020-08-12 14:52:28 +00:00
/** Mirrors `toolchain::cargo()` implementation */
2024-06-17 12:06:01 +00:00
// FIXME: The server should provide this
2021-08-15 16:19:45 +00:00
export function cargoPath(): Promise<string> {
return getPathForExecutable("cargo");
}
2020-08-12 14:52:28 +00:00
/** Mirrors `toolchain::get_path_for_executable()` implementation */
2021-08-15 16:19:45 +00:00
export const getPathForExecutable = memoizeAsync(
// We apply caching to decrease file-system interactions
2021-08-15 16:19:45 +00:00
async (executableName: "cargo" | "rustc" | "rustup"): Promise<string> => {
{
const envVar = process.env[executableName.toUpperCase()];
if (envVar) return envVar;
}
2021-08-15 16:19:45 +00:00
if (await lookupInPath(executableName)) return executableName;
const cargoHome = getCargoHome();
if (cargoHome) {
const standardPath = vscode.Uri.joinPath(cargoHome, "bin", executableName);
2021-08-17 07:24:13 +00:00
if (await isFileAtUri(standardPath)) return standardPath.fsPath;
}
return executableName;
2023-07-11 13:35:10 +00:00
},
);
2021-08-15 16:19:45 +00:00
async function lookupInPath(exec: string): Promise<boolean> {
const paths = process.env["PATH"] ?? "";
2022-05-17 17:15:06 +00:00
const candidates = paths.split(path.delimiter).flatMap((dirInPath) => {
const candidate = path.join(dirInPath, exec);
2022-05-17 17:15:06 +00:00
return os.type() === "Windows_NT" ? [candidate, `${candidate}.exe`] : [candidate];
});
2021-08-15 16:19:45 +00:00
for await (const isFile of candidates.map(isFileAtPath)) {
if (isFile) {
return true;
}
}
return false;
}
function getCargoHome(): vscode.Uri | null {
const envVar = process.env["CARGO_HOME"];
if (envVar) return vscode.Uri.file(envVar);
try {
// hmm, `os.homedir()` seems to be infallible
// it is not mentioned in docs and cannot be inferred by the type signature...
return vscode.Uri.joinPath(vscode.Uri.file(os.homedir()), ".cargo");
} catch (err) {
log.error("Failed to read the fs info", err);
}
return null;
}
2021-06-15 17:29:02 +00:00
async function isFileAtPath(path: string): Promise<boolean> {
return isFileAtUri(vscode.Uri.file(path));
}
async function isFileAtUri(uri: vscode.Uri): Promise<boolean> {
try {
return ((await vscode.workspace.fs.stat(uri)).type & vscode.FileType.File) !== 0;
} catch {
return false;
}
}