Skip to content

Commit 1811210

Browse files
committed
Auto merge of #15476 - Wilfred:implement-saved-file3, r=Veykril
Substitute $saved_file in custom check commands If the custom command has a $saved_file placeholder, and we know the file being saved, replace the placeholder and run a check command. If there's a placeholder and we don't know the saved file, do nothing. This is a simplified version of #15381, which I hope is easier to review.
2 parents 818c30c + cdbf54f commit 1811210

File tree

6 files changed

+66
-18
lines changed

6 files changed

+66
-18
lines changed

crates/flycheck/src/lib.rs

+48-11
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ use std::{
1414

1515
use command_group::{CommandGroup, GroupChild};
1616
use crossbeam_channel::{never, select, unbounded, Receiver, Sender};
17-
use paths::AbsPathBuf;
17+
use paths::{AbsPath, AbsPathBuf};
1818
use rustc_hash::FxHashMap;
1919
use serde::Deserialize;
2020
use stdx::process::streaming_output;
@@ -102,13 +102,15 @@ impl FlycheckHandle {
102102
}
103103

104104
/// Schedule a re-start of the cargo check worker to do a workspace wide check.
105-
pub fn restart_workspace(&self) {
106-
self.sender.send(StateChange::Restart(None)).unwrap();
105+
pub fn restart_workspace(&self, saved_file: Option<AbsPathBuf>) {
106+
self.sender.send(StateChange::Restart { package: None, saved_file }).unwrap();
107107
}
108108

109109
/// Schedule a re-start of the cargo check worker to do a package wide check.
110110
pub fn restart_for_package(&self, package: String) {
111-
self.sender.send(StateChange::Restart(Some(package))).unwrap();
111+
self.sender
112+
.send(StateChange::Restart { package: Some(package), saved_file: None })
113+
.unwrap();
112114
}
113115

114116
/// Stop this cargo check worker.
@@ -159,7 +161,7 @@ pub enum Progress {
159161
}
160162

161163
enum StateChange {
162-
Restart(Option<String>),
164+
Restart { package: Option<String>, saved_file: Option<AbsPathBuf> },
163165
Cancel,
164166
}
165167

@@ -186,6 +188,8 @@ enum Event {
186188
CheckEvent(Option<CargoMessage>),
187189
}
188190

191+
const SAVED_FILE_PLACEHOLDER: &str = "$saved_file";
192+
189193
impl FlycheckActor {
190194
fn new(
191195
id: usize,
@@ -221,7 +225,7 @@ impl FlycheckActor {
221225
tracing::debug!(flycheck_id = self.id, "flycheck cancelled");
222226
self.cancel_check_process();
223227
}
224-
Event::RequestStateChange(StateChange::Restart(package)) => {
228+
Event::RequestStateChange(StateChange::Restart { package, saved_file }) => {
225229
// Cancel the previously spawned process
226230
self.cancel_check_process();
227231
while let Ok(restart) = inbox.recv_timeout(Duration::from_millis(50)) {
@@ -231,7 +235,11 @@ impl FlycheckActor {
231235
}
232236
}
233237

234-
let command = self.check_command(package.as_deref());
238+
let command =
239+
match self.check_command(package.as_deref(), saved_file.as_deref()) {
240+
Some(c) => c,
241+
None => continue,
242+
};
235243
let formatted_command = format!("{:?}", command);
236244

237245
tracing::debug!(?command, "will restart flycheck");
@@ -305,7 +313,14 @@ impl FlycheckActor {
305313
}
306314
}
307315

308-
fn check_command(&self, package: Option<&str>) -> Command {
316+
/// Construct a `Command` object for checking the user's code. If the user
317+
/// has specified a custom command with placeholders that we cannot fill,
318+
/// return None.
319+
fn check_command(
320+
&self,
321+
package: Option<&str>,
322+
saved_file: Option<&AbsPath>,
323+
) -> Option<Command> {
309324
let (mut cmd, args) = match &self.config {
310325
FlycheckConfig::CargoCommand {
311326
command,
@@ -358,7 +373,7 @@ impl FlycheckActor {
358373
cmd.arg("--target-dir").arg(target_dir);
359374
}
360375
cmd.envs(extra_env);
361-
(cmd, extra_args)
376+
(cmd, extra_args.clone())
362377
}
363378
FlycheckConfig::CustomCommand {
364379
command,
@@ -387,12 +402,34 @@ impl FlycheckActor {
387402
}
388403
}
389404

390-
(cmd, args)
405+
if args.contains(&SAVED_FILE_PLACEHOLDER.to_owned()) {
406+
// If the custom command has a $saved_file placeholder, and
407+
// we're saving a file, replace the placeholder in the arguments.
408+
if let Some(saved_file) = saved_file {
409+
let args = args
410+
.iter()
411+
.map(|arg| {
412+
if arg == SAVED_FILE_PLACEHOLDER {
413+
saved_file.to_string()
414+
} else {
415+
arg.clone()
416+
}
417+
})
418+
.collect();
419+
(cmd, args)
420+
} else {
421+
// The custom command has a $saved_file placeholder,
422+
// but we had an IDE event that wasn't a file save. Do nothing.
423+
return None;
424+
}
425+
} else {
426+
(cmd, args.clone())
427+
}
391428
}
392429
};
393430

394431
cmd.args(args);
395-
cmd
432+
Some(cmd)
396433
}
397434

398435
fn send(&self, check_task: Message) {

crates/rust-analyzer/src/config.rs

+5
Original file line numberDiff line numberDiff line change
@@ -209,6 +209,11 @@ config_data! {
209209
/// by changing `#rust-analyzer.check.invocationStrategy#` and
210210
/// `#rust-analyzer.check.invocationLocation#`.
211211
///
212+
/// If `$saved_file` is part of the command, rust-analyzer will pass
213+
/// the absolute path of the saved file to the provided command. This is
214+
/// intended to be used with non-Cargo build systems.
215+
/// Note that `$saved_file` is experimental and may be removed in the futureg.
216+
///
212217
/// An example command would be:
213218
///
214219
/// ```bash

crates/rust-analyzer/src/handlers/notification.rs

+6-4
Original file line numberDiff line numberDiff line change
@@ -168,7 +168,7 @@ pub(crate) fn handle_did_save_text_document(
168168
} else if state.config.check_on_save() {
169169
// No specific flycheck was triggered, so let's trigger all of them.
170170
for flycheck in state.flycheck.iter() {
171-
flycheck.restart_workspace();
171+
flycheck.restart_workspace(None);
172172
}
173173
}
174174
Ok(())
@@ -314,14 +314,16 @@ fn run_flycheck(state: &mut GlobalState, vfs_path: VfsPath) -> bool {
314314
Some((idx, package))
315315
});
316316

317+
let saved_file = vfs_path.as_path().map(|p| p.to_owned());
318+
317319
// Find and trigger corresponding flychecks
318320
for flycheck in world.flycheck.iter() {
319321
for (id, package) in workspace_ids.clone() {
320322
if id == flycheck.id() {
321323
updated = true;
322324
match package.filter(|_| !world.config.flycheck_workspace()) {
323325
Some(package) => flycheck.restart_for_package(package),
324-
None => flycheck.restart_workspace(),
326+
None => flycheck.restart_workspace(saved_file.clone()),
325327
}
326328
continue;
327329
}
@@ -330,7 +332,7 @@ fn run_flycheck(state: &mut GlobalState, vfs_path: VfsPath) -> bool {
330332
// No specific flycheck was triggered, so let's trigger all of them.
331333
if !updated {
332334
for flycheck in world.flycheck.iter() {
333-
flycheck.restart_workspace();
335+
flycheck.restart_workspace(saved_file.clone());
334336
}
335337
}
336338
Ok(())
@@ -372,7 +374,7 @@ pub(crate) fn handle_run_flycheck(
372374
}
373375
// No specific flycheck was triggered, so let's trigger all of them.
374376
for flycheck in state.flycheck.iter() {
375-
flycheck.restart_workspace();
377+
flycheck.restart_workspace(None);
376378
}
377379
Ok(())
378380
}

crates/rust-analyzer/src/main_loop.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,6 @@ use std::{
88

99
use always_assert::always;
1010
use crossbeam_channel::{select, Receiver};
11-
use flycheck::FlycheckHandle;
1211
use ide_db::base_db::{SourceDatabaseExt, VfsPath};
1312
use lsp_server::{Connection, Notification, Request};
1413
use lsp_types::notification::Notification as _;
@@ -337,7 +336,7 @@ impl GlobalState {
337336
if became_quiescent {
338337
if self.config.check_on_save() {
339338
// Project has loaded properly, kick off initial flycheck
340-
self.flycheck.iter().for_each(FlycheckHandle::restart_workspace);
339+
self.flycheck.iter().for_each(|flycheck| flycheck.restart_workspace(None));
341340
}
342341
if self.config.prefill_caches() {
343342
self.prime_caches_queue.request_op("became quiescent".to_owned(), ());

docs/user/generated_config.adoc

+5
Original file line numberDiff line numberDiff line change
@@ -234,6 +234,11 @@ each of them, with the working directory being the workspace root
234234
by changing `#rust-analyzer.check.invocationStrategy#` and
235235
`#rust-analyzer.check.invocationLocation#`.
236236

237+
If `$saved_file` is part of the command, rust-analyzer will pass
238+
the absolute path of the saved file to the provided command. This is
239+
intended to be used with non-Cargo build systems.
240+
Note that `$saved_file` is experimental and may be removed in the futureg.
241+
237242
An example command would be:
238243

239244
```bash

editors/code/package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -775,7 +775,7 @@
775775
]
776776
},
777777
"rust-analyzer.check.overrideCommand": {
778-
"markdownDescription": "Override the command rust-analyzer uses instead of `cargo check` for\ndiagnostics on save. The command is required to output json and\nshould therefore include `--message-format=json` or a similar option\n(if your client supports the `colorDiagnosticOutput` experimental\ncapability, you can use `--message-format=json-diagnostic-rendered-ansi`).\n\nIf you're changing this because you're using some tool wrapping\nCargo, you might also want to change\n`#rust-analyzer.cargo.buildScripts.overrideCommand#`.\n\nIf there are multiple linked projects/workspaces, this command is invoked for\neach of them, with the working directory being the workspace root\n(i.e., the folder containing the `Cargo.toml`). This can be overwritten\nby changing `#rust-analyzer.check.invocationStrategy#` and\n`#rust-analyzer.check.invocationLocation#`.\n\nAn example command would be:\n\n```bash\ncargo check --workspace --message-format=json --all-targets\n```\n.",
778+
"markdownDescription": "Override the command rust-analyzer uses instead of `cargo check` for\ndiagnostics on save. The command is required to output json and\nshould therefore include `--message-format=json` or a similar option\n(if your client supports the `colorDiagnosticOutput` experimental\ncapability, you can use `--message-format=json-diagnostic-rendered-ansi`).\n\nIf you're changing this because you're using some tool wrapping\nCargo, you might also want to change\n`#rust-analyzer.cargo.buildScripts.overrideCommand#`.\n\nIf there are multiple linked projects/workspaces, this command is invoked for\neach of them, with the working directory being the workspace root\n(i.e., the folder containing the `Cargo.toml`). This can be overwritten\nby changing `#rust-analyzer.check.invocationStrategy#` and\n`#rust-analyzer.check.invocationLocation#`.\n\nIf `$saved_file` is part of the command, rust-analyzer will pass\nthe absolute path of the saved file to the provided command. This is\nintended to be used with non-Cargo build systems.\nNote that `$saved_file` is experimental and may be removed in the futureg.\n\nAn example command would be:\n\n```bash\ncargo check --workspace --message-format=json --all-targets\n```\n.",
779779
"default": null,
780780
"type": [
781781
"null",

0 commit comments

Comments
 (0)