1
1
import * as vscode from "vscode" ;
2
2
import type { Config } from "./config" ;
3
3
import { log } from "./util" ;
4
- import { expectNotUndefined , unwrapUndefinable } from "./undefinable" ;
4
+ import { unwrapUndefinable } from "./undefinable" ;
5
+ import * as toolchain from "./toolchain" ;
5
6
6
7
// This ends up as the `type` key in tasks.json. RLS also uses `cargo` and
7
8
// our configuration should be compatible with it so use the same key.
8
- export const TASK_TYPE = "cargo" ;
9
+ export const CARGO_TASK_TYPE = "cargo" ;
10
+ export const SHELL_TASK_TYPE = "shell" ;
9
11
10
- export const TASK_SOURCE = "rust" ;
12
+ export const RUST_TASK_SOURCE = "rust" ;
11
13
12
- export interface RustTargetDefinition extends vscode . TaskDefinition {
13
- // The cargo command, such as "run" or "check".
14
+ export type RustTargetDefinition = {
15
+ readonly type : typeof CARGO_TASK_TYPE | typeof SHELL_TASK_TYPE ;
16
+ } & vscode . TaskDefinition &
17
+ RustTarget ;
18
+ export type RustTarget = {
19
+ // The command to run, usually `cargo`.
14
20
command : string ;
15
- // Additional arguments passed to the cargo command.
21
+ // Additional arguments passed to the command.
16
22
args ?: string [ ] ;
17
- // The working directory to run the cargo command in.
23
+ // The working directory to run the command in.
18
24
cwd ?: string ;
19
25
// The shell environment.
20
26
env ?: { [ key : string ] : string } ;
21
- // Override the cargo executable name, such as
22
- // "my_custom_cargo_bin".
23
- overrideCargo ?: string ;
24
- }
27
+ } ;
25
28
26
29
class RustTaskProvider implements vscode . TaskProvider {
27
30
private readonly config : Config ;
@@ -31,6 +34,10 @@ class RustTaskProvider implements vscode.TaskProvider {
31
34
}
32
35
33
36
async provideTasks ( ) : Promise < vscode . Task [ ] > {
37
+ if ( ! vscode . workspace . workspaceFolders ) {
38
+ return [ ] ;
39
+ }
40
+
34
41
// Detect Rust tasks. Currently we do not do any actual detection
35
42
// of tasks (e.g. aliases in .cargo/config) and just return a fixed
36
43
// set of tasks that always exist. These tasks cannot be removed in
@@ -45,15 +52,23 @@ class RustTaskProvider implements vscode.TaskProvider {
45
52
{ command : "run" , group : undefined } ,
46
53
] ;
47
54
55
+ // FIXME: The server should provide this
56
+ const cargo = await toolchain . cargoPath ( ) ;
57
+
48
58
const tasks : vscode . Task [ ] = [ ] ;
49
- for ( const workspaceTarget of vscode . workspace . workspaceFolders || [ ] ) {
59
+ for ( const workspaceTarget of vscode . workspace . workspaceFolders ) {
50
60
for ( const def of defs ) {
61
+ const definition = {
62
+ command : cargo ,
63
+ args : [ def . command ] ,
64
+ } ;
65
+ const exec = await targetToExecution ( definition , this . config . cargoRunner ) ;
51
66
const vscodeTask = await buildRustTask (
52
67
workspaceTarget ,
53
- { type : TASK_TYPE , command : def . command } ,
68
+ { ... definition , type : CARGO_TASK_TYPE } ,
54
69
`cargo ${ def . command } ` ,
55
70
this . config . problemMatcher ,
56
- this . config . cargoRunner ,
71
+ exec ,
57
72
) ;
58
73
vscodeTask . group = def . group ;
59
74
tasks . push ( vscodeTask ) ;
@@ -67,16 +82,24 @@ class RustTaskProvider implements vscode.TaskProvider {
67
82
// VSCode calls this for every cargo task in the user's tasks.json,
68
83
// we need to inform VSCode how to execute that command by creating
69
84
// a ShellExecution for it.
70
-
71
- const definition = task . definition as RustTargetDefinition ;
72
-
73
- if ( definition . type === TASK_TYPE ) {
85
+ if ( task . definition . type === CARGO_TASK_TYPE ) {
86
+ const taskDefinition = task . definition as RustTargetDefinition ;
87
+ const cargo = await toolchain . cargoPath ( ) ;
88
+ const exec = await targetToExecution (
89
+ {
90
+ command : cargo ,
91
+ args : [ taskDefinition . command ] . concat ( taskDefinition . args || [ ] ) ,
92
+ cwd : taskDefinition . cwd ,
93
+ env : taskDefinition . env ,
94
+ } ,
95
+ this . config . cargoRunner ,
96
+ ) ;
74
97
return await buildRustTask (
75
98
task . scope ,
76
- definition ,
99
+ taskDefinition ,
77
100
task . name ,
78
101
this . config . problemMatcher ,
79
- this . config . cargoRunner ,
102
+ exec ,
80
103
) ;
81
104
}
82
105
@@ -89,34 +112,31 @@ export async function buildRustTask(
89
112
definition : RustTargetDefinition ,
90
113
name : string ,
91
114
problemMatcher : string [ ] ,
92
- customRunner ?: string ,
93
- throwOnError : boolean = false ,
115
+ exec : vscode . ProcessExecution | vscode . ShellExecution ,
94
116
) : Promise < vscode . Task > {
95
- const exec = await cargoToExecution ( definition , customRunner , throwOnError ) ;
96
-
97
117
return new vscode . Task (
98
118
definition ,
99
119
// scope can sometimes be undefined. in these situations we default to the workspace taskscope as
100
120
// recommended by the official docs: https://code.visualstudio.com/api/extension-guides/task-provider#task-provider)
101
121
scope ?? vscode . TaskScope . Workspace ,
102
122
name ,
103
- TASK_SOURCE ,
123
+ RUST_TASK_SOURCE ,
104
124
exec ,
105
125
problemMatcher ,
106
126
) ;
107
127
}
108
128
109
- async function cargoToExecution (
110
- definition : RustTargetDefinition ,
111
- customRunner : string | undefined ,
112
- throwOnError : boolean ,
129
+ export async function targetToExecution (
130
+ definition : RustTarget ,
131
+ customRunner ? : string ,
132
+ throwOnError : boolean = false ,
113
133
) : Promise < vscode . ProcessExecution | vscode . ShellExecution > {
114
134
if ( customRunner ) {
115
135
const runnerCommand = `${ customRunner } .buildShellExecution` ;
116
136
117
137
try {
118
138
const runnerArgs = {
119
- kind : TASK_TYPE ,
139
+ kind : CARGO_TASK_TYPE ,
120
140
args : definition . args ,
121
141
cwd : definition . cwd ,
122
142
env : definition . env ,
@@ -136,37 +156,14 @@ async function cargoToExecution(
136
156
// fallback to default processing
137
157
}
138
158
}
139
-
140
- // this is a cargo task; do Cargo-esque processing
141
- if ( definition . type === TASK_TYPE ) {
142
- // Check whether we must use a user-defined substitute for cargo.
143
- // Split on spaces to allow overrides like "wrapper cargo".
144
- const cargoCommand = definition . overrideCargo ?. split ( " " ) ?? [ definition . command ] ;
145
-
146
- const definitionArgs = expectNotUndefined (
147
- definition . args ,
148
- "args were not provided via runnables; this is a bug." ,
149
- ) ;
150
- const args = [ ...cargoCommand . slice ( 1 ) , ...definitionArgs ] ;
151
- const processName = unwrapUndefinable ( cargoCommand [ 0 ] ) ;
152
-
153
- return new vscode . ProcessExecution ( processName , args , {
154
- cwd : definition . cwd ,
155
- env : definition . env ,
156
- } ) ;
157
- } else {
158
- // we've been handed a process definition by rust-analyzer, trust all its inputs
159
- // and make a shell execution.
160
- const args = unwrapUndefinable ( definition . args ) ;
161
-
162
- return new vscode . ProcessExecution ( definition . command , args , {
163
- cwd : definition . cwd ,
164
- env : definition . env ,
165
- } ) ;
166
- }
159
+ const args = unwrapUndefinable ( definition . args ) ;
160
+ return new vscode . ProcessExecution ( definition . command , args , {
161
+ cwd : definition . cwd ,
162
+ env : definition . env ,
163
+ } ) ;
167
164
}
168
165
169
166
export function activateTaskProvider ( config : Config ) : vscode . Disposable {
170
167
const provider = new RustTaskProvider ( config ) ;
171
- return vscode . tasks . registerTaskProvider ( TASK_TYPE , provider ) ;
168
+ return vscode . tasks . registerTaskProvider ( CARGO_TASK_TYPE , provider ) ;
172
169
}
0 commit comments