4580: Fix invoking cargo without consulting CARGO env var or standard installation paths r=matklad a=Veetaha

Followup for #4329

The pr essentially fixes [this bug](https://youtu.be/EzQ7YIIo1rY?t=2189)

cc @lefticus

Co-authored-by: veetaha <veetaha2@gmail.com>
This commit is contained in:
bors[bot] 2020-06-02 11:58:28 +00:00 committed by GitHub
commit 131ccd9540
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
14 changed files with 140 additions and 81 deletions

1
Cargo.lock generated
View file

@ -1119,6 +1119,7 @@ dependencies = [
"memmap", "memmap",
"ra_mbe", "ra_mbe",
"ra_proc_macro", "ra_proc_macro",
"ra_toolchain",
"ra_tt", "ra_tt",
"serde_derive", "serde_derive",
"test_utils", "test_utils",

View file

@ -22,3 +22,4 @@ cargo_metadata = "0.10.0"
difference = "2.0.0" difference = "2.0.0"
# used as proc macro test target # used as proc macro test target
serde_derive = "1.0.106" serde_derive = "1.0.106"
ra_toolchain = { path = "../ra_toolchain" }

View file

@ -2,7 +2,6 @@
use crate::dylib; use crate::dylib;
use crate::ProcMacroSrv; use crate::ProcMacroSrv;
pub use difference::Changeset as __Changeset;
use ra_proc_macro::ListMacrosTask; use ra_proc_macro::ListMacrosTask;
use std::str::FromStr; use std::str::FromStr;
use test_utils::assert_eq_text; use test_utils::assert_eq_text;
@ -13,7 +12,7 @@ mod fixtures {
// Use current project metadata to get the proc-macro dylib path // Use current project metadata to get the proc-macro dylib path
pub fn dylib_path(crate_name: &str, version: &str) -> std::path::PathBuf { pub fn dylib_path(crate_name: &str, version: &str) -> std::path::PathBuf {
let command = Command::new("cargo") let command = Command::new(ra_toolchain::cargo())
.args(&["check", "--message-format", "json"]) .args(&["check", "--message-format", "json"])
.output() .output()
.unwrap() .unwrap()

View file

@ -121,12 +121,21 @@ pub struct RunnablesParams {
pub position: Option<Position>, pub position: Option<Position>,
} }
// Must strictly correspond to the executable name
#[derive(Serialize, Deserialize, Debug)]
#[serde(rename_all = "lowercase")]
pub enum RunnableKind {
Cargo,
Rustc,
Rustup,
}
#[derive(Deserialize, Serialize, Debug)] #[derive(Deserialize, Serialize, Debug)]
#[serde(rename_all = "camelCase")] #[serde(rename_all = "camelCase")]
pub struct Runnable { pub struct Runnable {
pub range: Range, pub range: Range,
pub label: String, pub label: String,
pub bin: String, pub kind: RunnableKind,
pub args: Vec<String>, pub args: Vec<String>,
pub extra_args: Vec<String>, pub extra_args: Vec<String>,
pub env: FxHashMap<String, String>, pub env: FxHashMap<String, String>,

View file

@ -426,7 +426,7 @@ pub fn handle_runnables(
res.push(lsp_ext::Runnable { res.push(lsp_ext::Runnable {
range: Default::default(), range: Default::default(),
label: format!("cargo {} -p {}", cmd, spec.package), label: format!("cargo {} -p {}", cmd, spec.package),
bin: "cargo".to_string(), kind: lsp_ext::RunnableKind::Cargo,
args: vec![cmd.to_string(), "--package".to_string(), spec.package.clone()], args: vec![cmd.to_string(), "--package".to_string(), spec.package.clone()],
extra_args: Vec::new(), extra_args: Vec::new(),
env: FxHashMap::default(), env: FxHashMap::default(),
@ -438,7 +438,7 @@ pub fn handle_runnables(
res.push(lsp_ext::Runnable { res.push(lsp_ext::Runnable {
range: Default::default(), range: Default::default(),
label: "cargo check --workspace".to_string(), label: "cargo check --workspace".to_string(),
bin: "cargo".to_string(), kind: lsp_ext::RunnableKind::Cargo,
args: vec!["check".to_string(), "--workspace".to_string()], args: vec!["check".to_string(), "--workspace".to_string()],
extra_args: Vec::new(), extra_args: Vec::new(),
env: FxHashMap::default(), env: FxHashMap::default(),
@ -982,10 +982,11 @@ fn to_lsp_runnable(
target.map_or_else(|| "run binary".to_string(), |t| format!("run {}", t)) target.map_or_else(|| "run binary".to_string(), |t| format!("run {}", t))
} }
}; };
Ok(lsp_ext::Runnable { Ok(lsp_ext::Runnable {
range: to_proto::range(&line_index, runnable.range), range: to_proto::range(&line_index, runnable.range),
label, label,
bin: "cargo".to_string(), kind: lsp_ext::RunnableKind::Cargo,
args, args,
extra_args, extra_args,
env: { env: {

View file

@ -79,7 +79,7 @@ fn foo() {
{ {
"args": [ "test" ], "args": [ "test" ],
"extraArgs": [ "foo", "--nocapture" ], "extraArgs": [ "foo", "--nocapture" ],
"bin": "cargo", "kind": "cargo",
"env": { "RUST_BACKTRACE": "short" }, "env": { "RUST_BACKTRACE": "short" },
"cwd": null, "cwd": null,
"label": "test foo", "label": "test foo",
@ -91,7 +91,7 @@ fn foo() {
{ {
"args": ["check", "--workspace"], "args": ["check", "--workspace"],
"extraArgs": [], "extraArgs": [],
"bin": "cargo", "kind": "cargo",
"env": {}, "env": {},
"cwd": null, "cwd": null,
"label": "cargo check --workspace", "label": "cargo check --workspace",
@ -141,7 +141,7 @@ fn main() {}
{ {
"args": [ "test", "--package", "foo", "--test", "spam" ], "args": [ "test", "--package", "foo", "--test", "spam" ],
"extraArgs": [ "test_eggs", "--exact", "--nocapture" ], "extraArgs": [ "test_eggs", "--exact", "--nocapture" ],
"bin": "cargo", "kind": "cargo",
"env": { "RUST_BACKTRACE": "short" }, "env": { "RUST_BACKTRACE": "short" },
"label": "test test_eggs", "label": "test test_eggs",
"range": { "range": {
@ -153,7 +153,7 @@ fn main() {}
{ {
"args": [ "check", "--package", "foo" ], "args": [ "check", "--package", "foo" ],
"extraArgs": [], "extraArgs": [],
"bin": "cargo", "kind": "cargo",
"env": {}, "env": {},
"label": "cargo check -p foo", "label": "cargo check -p foo",
"range": { "range": {
@ -165,7 +165,7 @@ fn main() {}
{ {
"args": [ "test", "--package", "foo" ], "args": [ "test", "--package", "foo" ],
"extraArgs": [], "extraArgs": [],
"bin": "cargo", "kind": "cargo",
"env": {}, "env": {},
"label": "cargo test -p foo", "label": "cargo test -p foo",
"range": { "range": {

View file

@ -427,7 +427,7 @@ interface Runnable {
/// The label to show in the UI. /// The label to show in the UI.
label: string; label: string;
/// The following fields describe a process to spawn. /// The following fields describe a process to spawn.
bin: string; kind: "cargo" | "rustc" | "rustup";
args: string[]; args: string[];
/// Args for cargo after `--`. /// Args for cargo after `--`.
extraArgs: string[]; extraArgs: string[];

View file

@ -3,7 +3,7 @@ import * as vscode from 'vscode';
import * as path from 'path'; import * as path from 'path';
import * as ra from './lsp_ext'; import * as ra from './lsp_ext';
import { Cargo } from './cargo'; import { Cargo } from './toolchain';
import { Ctx } from "./ctx"; import { Ctx } from "./ctx";
const debugOutput = vscode.window.createOutputChannel("Debug"); const debugOutput = vscode.window.createOutputChannel("Debug");

View file

@ -45,10 +45,13 @@ export interface RunnablesParams {
textDocument: lc.TextDocumentIdentifier; textDocument: lc.TextDocumentIdentifier;
position: lc.Position | null; position: lc.Position | null;
} }
export type RunnableKind = "cargo" | "rustc" | "rustup";
export interface Runnable { export interface Runnable {
range: lc.Range; range: lc.Range;
label: string; label: string;
bin: string; kind: RunnableKind;
args: string[]; args: string[];
extraArgs: string[]; extraArgs: string[];
env: { [key: string]: string }; env: { [key: string]: string };

View file

@ -1,6 +1,7 @@
import * as vscode from 'vscode'; import * as vscode from 'vscode';
import * as lc from 'vscode-languageclient'; import * as lc from 'vscode-languageclient';
import * as ra from './lsp_ext'; import * as ra from './lsp_ext';
import * as toolchain from "./toolchain";
import { Ctx, Cmd } from './ctx'; import { Ctx, Cmd } from './ctx';
import { startDebugSession, getDebugConfiguration } from './debug'; import { startDebugSession, getDebugConfiguration } from './debug';
@ -175,7 +176,7 @@ export function createTask(spec: ra.Runnable): vscode.Task {
const definition: CargoTaskDefinition = { const definition: CargoTaskDefinition = {
type: 'cargo', type: 'cargo',
label: spec.label, label: spec.label,
command: spec.bin, command: toolchain.getPathForExecutable(spec.kind),
args: spec.extraArgs ? [...spec.args, '--', ...spec.extraArgs] : spec.args, args: spec.extraArgs ? [...spec.args, '--', ...spec.extraArgs] : spec.args,
env: Object.assign({}, process.env, spec.env), env: Object.assign({}, process.env, spec.env),
}; };

View file

@ -1,4 +1,5 @@
import * as vscode from 'vscode'; import * as vscode from 'vscode';
import * as toolchain from "./toolchain";
// This ends up as the `type` key in tasks.json. RLS also uses `cargo` and // This ends up as the `type` key in tasks.json. RLS also uses `cargo` and
// our configuration should be compatible with it so use the same key. // our configuration should be compatible with it so use the same key.
@ -24,6 +25,8 @@ class CargoTaskProvider implements vscode.TaskProvider {
// set of tasks that always exist. These tasks cannot be removed in // set of tasks that always exist. These tasks cannot be removed in
// tasks.json - only tweaked. // tasks.json - only tweaked.
const cargoPath = toolchain.cargoPath();
return [ return [
{ command: 'build', group: vscode.TaskGroup.Build }, { command: 'build', group: vscode.TaskGroup.Build },
{ command: 'check', group: vscode.TaskGroup.Build }, { command: 'check', group: vscode.TaskGroup.Build },
@ -46,7 +49,7 @@ class CargoTaskProvider implements vscode.TaskProvider {
`cargo ${command}`, `cargo ${command}`,
'rust', 'rust',
// What to do when this command is executed. // What to do when this command is executed.
new vscode.ShellExecution('cargo', [command]), new vscode.ShellExecution(cargoPath, [command]),
// Problem matchers. // Problem matchers.
['$rustc'], ['$rustc'],
); );

View file

@ -1,9 +1,10 @@
import * as cp from 'child_process'; import * as cp from 'child_process';
import * as os from 'os'; import * as os from 'os';
import * as path from 'path'; import * as path from 'path';
import * as fs from 'fs';
import * as readline from 'readline'; import * as readline from 'readline';
import { OutputChannel } from 'vscode'; import { OutputChannel } from 'vscode';
import { isValidExecutable } from './util'; import { log, memoize } from './util';
interface CompilationArtifact { interface CompilationArtifact {
fileName: string; fileName: string;
@ -17,7 +18,11 @@ export interface ArtifactSpec {
filter?: (artifacts: CompilationArtifact[]) => CompilationArtifact[]; filter?: (artifacts: CompilationArtifact[]) => CompilationArtifact[];
} }
export function artifactSpec(args: readonly string[]): ArtifactSpec { export class Cargo {
constructor(readonly rootFolder: string, readonly output: OutputChannel) { }
// Made public for testing purposes
static artifactSpec(args: readonly string[]): ArtifactSpec {
const cargoArgs = [...args, "--message-format=json"]; const cargoArgs = [...args, "--message-format=json"];
// arguments for a runnable from the quick pick should be updated. // arguments for a runnable from the quick pick should be updated.
@ -42,9 +47,6 @@ export function artifactSpec(args: readonly string[]): ArtifactSpec {
return result; return result;
} }
export class Cargo {
constructor(readonly rootFolder: string, readonly output: OutputChannel) { }
private async getArtifacts(spec: ArtifactSpec): Promise<CompilationArtifact[]> { private async getArtifacts(spec: ArtifactSpec): Promise<CompilationArtifact[]> {
const artifacts: CompilationArtifact[] = []; const artifacts: CompilationArtifact[] = [];
@ -77,7 +79,7 @@ export class Cargo {
} }
async executableFromArgs(args: readonly string[]): Promise<string> { async executableFromArgs(args: readonly string[]): Promise<string> {
const artifacts = await this.getArtifacts(artifactSpec(args)); const artifacts = await this.getArtifacts(Cargo.artifactSpec(args));
if (artifacts.length === 0) { if (artifacts.length === 0) {
throw new Error('No compilation artifacts'); throw new Error('No compilation artifacts');
@ -94,14 +96,7 @@ export class Cargo {
onStderrString: (data: string) => void onStderrString: (data: string) => void
): Promise<number> { ): Promise<number> {
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
let cargoPath; const cargo = cp.spawn(cargoPath(), cargoArgs, {
try {
cargoPath = getCargoPathOrFail();
} catch (err) {
return reject(err);
}
const cargo = cp.spawn(cargoPath, cargoArgs, {
stdio: ['ignore', 'pipe', 'pipe'], stdio: ['ignore', 'pipe', 'pipe'],
cwd: this.rootFolder cwd: this.rootFolder
}); });
@ -126,26 +121,54 @@ export class Cargo {
} }
} }
// Mirrors `ra_env::get_path_for_executable` implementation /** Mirrors `ra_toolchain::cargo()` implementation */
function getCargoPathOrFail(): string { export function cargoPath(): string {
const envVar = process.env.CARGO; return getPathForExecutable("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; /** Mirrors `ra_toolchain::get_path_for_executable()` implementation */
export const getPathForExecutable = memoize(
// We apply caching to decrease file-system interactions
(executableName: "cargo" | "rustc" | "rustup"): string => {
{
const envVar = process.env[executableName.toUpperCase()];
if (envVar) return envVar;
}
const standardLocation = path.join(os.homedir(), '.cargo', 'bin', executableName); if (lookupInPath(executableName)) return executableName;
if (isValidExecutable(standardLocation)) return standardLocation; try {
// hmm, `os.homedir()` seems to be infallible
// it is not mentioned in docs and cannot be infered by the type signature...
const standardPath = path.join(os.homedir(), ".cargo", "bin", executableName);
throw new Error( if (isFile(standardPath)) return standardPath;
`Failed to find \`${executableName}\` executable. ` + } catch (err) {
`Make sure \`${executableName}\` is in \`$PATH\`, ` + log.error("Failed to read the fs info", err);
`or set \`${envVar}\` to point to a valid executable.` }
return executableName;
}
); );
function lookupInPath(exec: string): boolean {
const paths = process.env.PATH ?? "";;
const candidates = paths.split(path.delimiter).flatMap(dirInPath => {
const candidate = path.join(dirInPath, exec);
return os.type() === "Windows_NT"
? [candidate, `${candidate}.exe`]
: [candidate];
});
return candidates.some(isFile);
}
function isFile(suspectPath: string): boolean {
// It is not mentionned in docs, but `statSync()` throws an error when
// the path doesn't exist
try {
return fs.statSync(suspectPath).isFile();
} catch {
return false;
}
} }

View file

@ -99,3 +99,21 @@ export function isValidExecutable(path: string): boolean {
export function setContextValue(key: string, value: any): Thenable<void> { export function setContextValue(key: string, value: any): Thenable<void> {
return vscode.commands.executeCommand('setContext', key, value); return vscode.commands.executeCommand('setContext', key, value);
} }
/**
* Returns a higher-order function that caches the results of invoking the
* underlying function.
*/
export function memoize<Ret, TThis, Param extends string>(func: (this: TThis, arg: Param) => Ret) {
const cache = new Map<string, Ret>();
return function(this: TThis, arg: Param) {
const cached = cache.get(arg);
if (cached) return cached;
const result = func.call(this, arg);
cache.set(arg, result);
return result;
};
}

View file

@ -1,25 +1,25 @@
import * as assert from 'assert'; import * as assert from 'assert';
import * as cargo from '../../src/cargo'; import { Cargo } from '../../src/toolchain';
suite('Launch configuration', () => { suite('Launch configuration', () => {
suite('Lens', () => { suite('Lens', () => {
test('A binary', async () => { test('A binary', async () => {
const args = cargo.artifactSpec(["build", "--package", "pkg_name", "--bin", "pkg_name"]); const args = Cargo.artifactSpec(["build", "--package", "pkg_name", "--bin", "pkg_name"]);
assert.deepEqual(args.cargoArgs, ["build", "--package", "pkg_name", "--bin", "pkg_name", "--message-format=json"]); assert.deepEqual(args.cargoArgs, ["build", "--package", "pkg_name", "--bin", "pkg_name", "--message-format=json"]);
assert.deepEqual(args.filter, undefined); assert.deepEqual(args.filter, undefined);
}); });
test('One of Multiple Binaries', async () => { test('One of Multiple Binaries', async () => {
const args = cargo.artifactSpec(["build", "--package", "pkg_name", "--bin", "bin1"]); const args = Cargo.artifactSpec(["build", "--package", "pkg_name", "--bin", "bin1"]);
assert.deepEqual(args.cargoArgs, ["build", "--package", "pkg_name", "--bin", "bin1", "--message-format=json"]); assert.deepEqual(args.cargoArgs, ["build", "--package", "pkg_name", "--bin", "bin1", "--message-format=json"]);
assert.deepEqual(args.filter, undefined); assert.deepEqual(args.filter, undefined);
}); });
test('A test', async () => { test('A test', async () => {
const args = cargo.artifactSpec(["test", "--package", "pkg_name", "--lib", "--no-run"]); const args = Cargo.artifactSpec(["test", "--package", "pkg_name", "--lib", "--no-run"]);
assert.deepEqual(args.cargoArgs, ["test", "--package", "pkg_name", "--lib", "--no-run", "--message-format=json"]); assert.deepEqual(args.cargoArgs, ["test", "--package", "pkg_name", "--lib", "--no-run", "--message-format=json"]);
assert.notDeepEqual(args.filter, undefined); assert.notDeepEqual(args.filter, undefined);
@ -28,7 +28,7 @@ suite('Launch configuration', () => {
suite('QuickPick', () => { suite('QuickPick', () => {
test('A binary', async () => { test('A binary', async () => {
const args = cargo.artifactSpec(["run", "--package", "pkg_name", "--bin", "pkg_name"]); const args = Cargo.artifactSpec(["run", "--package", "pkg_name", "--bin", "pkg_name"]);
assert.deepEqual(args.cargoArgs, ["build", "--package", "pkg_name", "--bin", "pkg_name", "--message-format=json"]); assert.deepEqual(args.cargoArgs, ["build", "--package", "pkg_name", "--bin", "pkg_name", "--message-format=json"]);
assert.deepEqual(args.filter, undefined); assert.deepEqual(args.filter, undefined);
@ -36,14 +36,14 @@ suite('Launch configuration', () => {
test('One of Multiple Binaries', async () => { test('One of Multiple Binaries', async () => {
const args = cargo.artifactSpec(["run", "--package", "pkg_name", "--bin", "bin2"]); const args = Cargo.artifactSpec(["run", "--package", "pkg_name", "--bin", "bin2"]);
assert.deepEqual(args.cargoArgs, ["build", "--package", "pkg_name", "--bin", "bin2", "--message-format=json"]); assert.deepEqual(args.cargoArgs, ["build", "--package", "pkg_name", "--bin", "bin2", "--message-format=json"]);
assert.deepEqual(args.filter, undefined); assert.deepEqual(args.filter, undefined);
}); });
test('A test', async () => { test('A test', async () => {
const args = cargo.artifactSpec(["test", "--package", "pkg_name", "--lib"]); const args = Cargo.artifactSpec(["test", "--package", "pkg_name", "--lib"]);
assert.deepEqual(args.cargoArgs, ["test", "--package", "pkg_name", "--lib", "--message-format=json", "--no-run"]); assert.deepEqual(args.cargoArgs, ["test", "--package", "pkg_name", "--lib", "--message-format=json", "--no-run"]);
assert.notDeepEqual(args.filter, undefined); assert.notDeepEqual(args.filter, undefined);