2020-04-28 14:30:49 +00:00
|
|
|
import * as cp from 'child_process';
|
2020-05-05 22:42:04 +00:00
|
|
|
import * as os from 'os';
|
|
|
|
import * as path from 'path';
|
2020-04-28 14:30:49 +00:00
|
|
|
import * as readline from 'readline';
|
2020-04-30 12:25:04 +00:00
|
|
|
import { OutputChannel } from 'vscode';
|
2020-05-05 22:42:04 +00:00
|
|
|
import { isValidExecutable } from './util';
|
2020-04-28 14:30:49 +00:00
|
|
|
|
|
|
|
interface CompilationArtifact {
|
|
|
|
fileName: string;
|
|
|
|
name: string;
|
|
|
|
kind: string;
|
|
|
|
isTest: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
export class Cargo {
|
2020-05-05 22:22:02 +00:00
|
|
|
constructor(readonly rootFolder: string, readonly output: OutputChannel) { }
|
2020-04-28 14:30:49 +00:00
|
|
|
|
2020-05-05 22:22:02 +00:00
|
|
|
private async artifactsFromArgs(cargoArgs: string[]): Promise<CompilationArtifact[]> {
|
2020-04-30 15:41:48 +00:00
|
|
|
const artifacts: CompilationArtifact[] = [];
|
2020-04-28 14:30:49 +00:00
|
|
|
|
|
|
|
try {
|
|
|
|
await this.runCargo(cargoArgs,
|
|
|
|
message => {
|
2020-04-30 15:41:48 +00:00
|
|
|
if (message.reason === 'compiler-artifact' && message.executable) {
|
|
|
|
const isBinary = message.target.crate_types.includes('bin');
|
|
|
|
const isBuildScript = message.target.kind.includes('custom-build');
|
2020-04-28 14:30:49 +00:00
|
|
|
if ((isBinary && !isBuildScript) || message.profile.test) {
|
|
|
|
artifacts.push({
|
|
|
|
fileName: message.executable,
|
|
|
|
name: message.target.name,
|
|
|
|
kind: message.target.kind[0],
|
|
|
|
isTest: message.profile.test
|
2020-04-30 15:41:48 +00:00
|
|
|
});
|
2020-04-28 14:30:49 +00:00
|
|
|
}
|
2020-05-05 22:22:02 +00:00
|
|
|
} else if (message.reason === 'compiler-message') {
|
2020-04-30 12:25:04 +00:00
|
|
|
this.output.append(message.message.rendered);
|
|
|
|
}
|
2020-04-28 14:30:49 +00:00
|
|
|
},
|
2020-05-05 22:22:02 +00:00
|
|
|
stderr => this.output.append(stderr),
|
2020-04-28 14:30:49 +00:00
|
|
|
);
|
2020-05-05 22:22:02 +00:00
|
|
|
} catch (err) {
|
2020-04-30 12:25:04 +00:00
|
|
|
this.output.show(true);
|
2020-04-28 14:30:49 +00:00
|
|
|
throw new Error(`Cargo invocation has failed: ${err}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
return artifacts;
|
|
|
|
}
|
|
|
|
|
2020-05-05 22:22:02 +00:00
|
|
|
async executableFromArgs(args: readonly string[]): Promise<string> {
|
|
|
|
const cargoArgs = [...args, "--message-format=json"];
|
2020-05-14 10:30:05 +00:00
|
|
|
if( cargoArgs[0] == "run" ) {
|
|
|
|
// a runnable from the quick pick.
|
|
|
|
cargoArgs[0] = "build";
|
|
|
|
}
|
2020-04-28 14:30:49 +00:00
|
|
|
|
2020-04-30 15:41:48 +00:00
|
|
|
const artifacts = await this.artifactsFromArgs(cargoArgs);
|
2020-04-28 14:30:49 +00:00
|
|
|
|
2020-04-30 15:41:48 +00:00
|
|
|
if (artifacts.length === 0) {
|
2020-04-28 14:30:49 +00:00
|
|
|
throw new Error('No compilation artifacts');
|
|
|
|
} else if (artifacts.length > 1) {
|
|
|
|
throw new Error('Multiple compilation artifacts are not supported.');
|
|
|
|
}
|
|
|
|
|
|
|
|
return artifacts[0].fileName;
|
|
|
|
}
|
|
|
|
|
2020-05-05 22:22:02 +00:00
|
|
|
private runCargo(
|
2020-04-28 14:30:49 +00:00
|
|
|
cargoArgs: string[],
|
|
|
|
onStdoutJson: (obj: any) => void,
|
|
|
|
onStderrString: (data: string) => void
|
|
|
|
): Promise<number> {
|
2020-05-05 22:22:02 +00:00
|
|
|
return new Promise((resolve, reject) => {
|
2020-05-05 22:42:04 +00:00
|
|
|
let cargoPath;
|
|
|
|
try {
|
|
|
|
cargoPath = getCargoPathOrFail();
|
|
|
|
} catch (err) {
|
|
|
|
return reject(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
const cargo = cp.spawn(cargoPath, cargoArgs, {
|
2020-04-28 14:30:49 +00:00
|
|
|
stdio: ['ignore', 'pipe', 'pipe'],
|
2020-05-05 22:22:02 +00:00
|
|
|
cwd: this.rootFolder
|
2020-04-28 14:30:49 +00:00
|
|
|
});
|
|
|
|
|
2020-05-05 22:22:02 +00:00
|
|
|
cargo.on('error', err => reject(new Error(`could not launch cargo: ${err}`)));
|
|
|
|
|
|
|
|
cargo.stderr.on('data', chunk => onStderrString(chunk.toString()));
|
2020-04-28 14:30:49 +00:00
|
|
|
|
2020-04-30 15:41:48 +00:00
|
|
|
const rl = readline.createInterface({ input: cargo.stdout });
|
2020-04-28 14:30:49 +00:00
|
|
|
rl.on('line', line => {
|
2020-04-30 15:41:48 +00:00
|
|
|
const message = JSON.parse(line);
|
2020-04-28 14:30:49 +00:00
|
|
|
onStdoutJson(message);
|
|
|
|
});
|
|
|
|
|
|
|
|
cargo.on('exit', (exitCode, _) => {
|
2020-04-30 15:41:48 +00:00
|
|
|
if (exitCode === 0)
|
2020-04-28 14:30:49 +00:00
|
|
|
resolve(exitCode);
|
|
|
|
else
|
|
|
|
reject(new Error(`exit code: ${exitCode}.`));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
2020-05-05 22:22:02 +00:00
|
|
|
}
|
2020-05-05 22:42:04 +00:00
|
|
|
|
|
|
|
// Mirrors `ra_env::get_path_for_executable` implementation
|
|
|
|
function getCargoPathOrFail(): string {
|
|
|
|
const envVar = process.env.CARGO;
|
|
|
|
const executableName = "cargo";
|
|
|
|
|
|
|
|
if (envVar) {
|
|
|
|
if (isValidExecutable(envVar)) return envVar;
|
|
|
|
|
|
|
|
throw new Error(`\`${envVar}\` environment variable points to something that's not a valid executable`);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isValidExecutable(executableName)) return executableName;
|
|
|
|
|
|
|
|
const standardLocation = path.join(os.homedir(), '.cargo', 'bin', executableName);
|
|
|
|
|
|
|
|
if (isValidExecutable(standardLocation)) return standardLocation;
|
|
|
|
|
|
|
|
throw new Error(
|
|
|
|
`Failed to find \`${executableName}\` executable. ` +
|
|
|
|
`Make sure \`${executableName}\` is in \`$PATH\`, ` +
|
|
|
|
`or set \`${envVar}\` to point to a valid executable.`
|
|
|
|
);
|
|
|
|
}
|