diff options
Diffstat (limited to 'crates')
17 files changed, 1503 insertions, 1504 deletions
diff --git a/crates/ra_flycheck/Cargo.toml b/crates/ra_flycheck/Cargo.toml index eac502da5..1aa39bade 100644 --- a/crates/ra_flycheck/Cargo.toml +++ b/crates/ra_flycheck/Cargo.toml | |||
@@ -9,12 +9,8 @@ doctest = false | |||
9 | 9 | ||
10 | [dependencies] | 10 | [dependencies] |
11 | crossbeam-channel = "0.4.0" | 11 | crossbeam-channel = "0.4.0" |
12 | lsp-types = { version = "0.74.0", features = ["proposed"] } | ||
13 | log = "0.4.8" | 12 | log = "0.4.8" |
14 | cargo_metadata = "0.10.0" | 13 | cargo_metadata = "0.10.0" |
15 | serde_json = "1.0.48" | 14 | serde_json = "1.0.48" |
16 | jod-thread = "0.1.1" | 15 | jod-thread = "0.1.1" |
17 | ra_toolchain = { path = "../ra_toolchain" } | 16 | ra_toolchain = { path = "../ra_toolchain" } |
18 | |||
19 | [dev-dependencies] | ||
20 | insta = "0.16.0" | ||
diff --git a/crates/ra_flycheck/src/conv.rs b/crates/ra_flycheck/src/conv.rs deleted file mode 100644 index 817543deb..000000000 --- a/crates/ra_flycheck/src/conv.rs +++ /dev/null | |||
@@ -1,341 +0,0 @@ | |||
1 | //! This module provides the functionality needed to convert diagnostics from | ||
2 | //! `cargo check` json format to the LSP diagnostic format. | ||
3 | use cargo_metadata::diagnostic::{ | ||
4 | Applicability, Diagnostic as RustDiagnostic, DiagnosticLevel, DiagnosticSpan, | ||
5 | DiagnosticSpanMacroExpansion, | ||
6 | }; | ||
7 | use lsp_types::{ | ||
8 | CodeAction, Diagnostic, DiagnosticRelatedInformation, DiagnosticSeverity, DiagnosticTag, | ||
9 | Location, NumberOrString, Position, Range, TextEdit, Url, WorkspaceEdit, | ||
10 | }; | ||
11 | use std::{ | ||
12 | collections::HashMap, | ||
13 | fmt::Write, | ||
14 | path::{Component, Path, PathBuf, Prefix}, | ||
15 | str::FromStr, | ||
16 | }; | ||
17 | |||
18 | #[cfg(test)] | ||
19 | mod test; | ||
20 | |||
21 | /// Converts a Rust level string to a LSP severity | ||
22 | fn map_level_to_severity(val: DiagnosticLevel) -> Option<DiagnosticSeverity> { | ||
23 | match val { | ||
24 | DiagnosticLevel::Ice => Some(DiagnosticSeverity::Error), | ||
25 | DiagnosticLevel::Error => Some(DiagnosticSeverity::Error), | ||
26 | DiagnosticLevel::Warning => Some(DiagnosticSeverity::Warning), | ||
27 | DiagnosticLevel::Note => Some(DiagnosticSeverity::Information), | ||
28 | DiagnosticLevel::Help => Some(DiagnosticSeverity::Hint), | ||
29 | DiagnosticLevel::Unknown => None, | ||
30 | } | ||
31 | } | ||
32 | |||
33 | /// Check whether a file name is from macro invocation | ||
34 | fn is_from_macro(file_name: &str) -> bool { | ||
35 | file_name.starts_with('<') && file_name.ends_with('>') | ||
36 | } | ||
37 | |||
38 | /// Converts a Rust macro span to a LSP location recursively | ||
39 | fn map_macro_span_to_location( | ||
40 | span_macro: &DiagnosticSpanMacroExpansion, | ||
41 | workspace_root: &PathBuf, | ||
42 | ) -> Option<Location> { | ||
43 | if !is_from_macro(&span_macro.span.file_name) { | ||
44 | return Some(map_span_to_location(&span_macro.span, workspace_root)); | ||
45 | } | ||
46 | |||
47 | if let Some(expansion) = &span_macro.span.expansion { | ||
48 | return map_macro_span_to_location(&expansion, workspace_root); | ||
49 | } | ||
50 | |||
51 | None | ||
52 | } | ||
53 | |||
54 | /// Converts a Rust span to a LSP location, resolving macro expansion site if neccesary | ||
55 | fn map_span_to_location(span: &DiagnosticSpan, workspace_root: &PathBuf) -> Location { | ||
56 | if span.expansion.is_some() { | ||
57 | let expansion = span.expansion.as_ref().unwrap(); | ||
58 | if let Some(macro_range) = map_macro_span_to_location(&expansion, workspace_root) { | ||
59 | return macro_range; | ||
60 | } | ||
61 | } | ||
62 | |||
63 | map_span_to_location_naive(span, workspace_root) | ||
64 | } | ||
65 | |||
66 | /// Converts a Rust span to a LSP location | ||
67 | fn map_span_to_location_naive(span: &DiagnosticSpan, workspace_root: &PathBuf) -> Location { | ||
68 | let mut file_name = workspace_root.clone(); | ||
69 | file_name.push(&span.file_name); | ||
70 | let uri = url_from_path_with_drive_lowercasing(file_name).unwrap(); | ||
71 | |||
72 | let range = Range::new( | ||
73 | Position::new(span.line_start as u64 - 1, span.column_start as u64 - 1), | ||
74 | Position::new(span.line_end as u64 - 1, span.column_end as u64 - 1), | ||
75 | ); | ||
76 | |||
77 | Location { uri, range } | ||
78 | } | ||
79 | |||
80 | /// Converts a secondary Rust span to a LSP related information | ||
81 | /// | ||
82 | /// If the span is unlabelled this will return `None`. | ||
83 | fn map_secondary_span_to_related( | ||
84 | span: &DiagnosticSpan, | ||
85 | workspace_root: &PathBuf, | ||
86 | ) -> Option<DiagnosticRelatedInformation> { | ||
87 | if let Some(label) = &span.label { | ||
88 | let location = map_span_to_location(span, workspace_root); | ||
89 | Some(DiagnosticRelatedInformation { location, message: label.clone() }) | ||
90 | } else { | ||
91 | // Nothing to label this with | ||
92 | None | ||
93 | } | ||
94 | } | ||
95 | |||
96 | /// Determines if diagnostic is related to unused code | ||
97 | fn is_unused_or_unnecessary(rd: &RustDiagnostic) -> bool { | ||
98 | if let Some(code) = &rd.code { | ||
99 | match code.code.as_str() { | ||
100 | "dead_code" | "unknown_lints" | "unreachable_code" | "unused_attributes" | ||
101 | | "unused_imports" | "unused_macros" | "unused_variables" => true, | ||
102 | _ => false, | ||
103 | } | ||
104 | } else { | ||
105 | false | ||
106 | } | ||
107 | } | ||
108 | |||
109 | /// Determines if diagnostic is related to deprecated code | ||
110 | fn is_deprecated(rd: &RustDiagnostic) -> bool { | ||
111 | if let Some(code) = &rd.code { | ||
112 | match code.code.as_str() { | ||
113 | "deprecated" => true, | ||
114 | _ => false, | ||
115 | } | ||
116 | } else { | ||
117 | false | ||
118 | } | ||
119 | } | ||
120 | |||
121 | enum MappedRustChildDiagnostic { | ||
122 | Related(DiagnosticRelatedInformation), | ||
123 | SuggestedFix(CodeAction), | ||
124 | MessageLine(String), | ||
125 | } | ||
126 | |||
127 | fn map_rust_child_diagnostic( | ||
128 | rd: &RustDiagnostic, | ||
129 | workspace_root: &PathBuf, | ||
130 | ) -> MappedRustChildDiagnostic { | ||
131 | let spans: Vec<&DiagnosticSpan> = rd.spans.iter().filter(|s| s.is_primary).collect(); | ||
132 | if spans.is_empty() { | ||
133 | // `rustc` uses these spanless children as a way to print multi-line | ||
134 | // messages | ||
135 | return MappedRustChildDiagnostic::MessageLine(rd.message.clone()); | ||
136 | } | ||
137 | |||
138 | let mut edit_map: HashMap<Url, Vec<TextEdit>> = HashMap::new(); | ||
139 | for &span in &spans { | ||
140 | match (&span.suggestion_applicability, &span.suggested_replacement) { | ||
141 | (Some(Applicability::MachineApplicable), Some(suggested_replacement)) => { | ||
142 | let location = map_span_to_location(span, workspace_root); | ||
143 | let edit = TextEdit::new(location.range, suggested_replacement.clone()); | ||
144 | edit_map.entry(location.uri).or_default().push(edit); | ||
145 | } | ||
146 | _ => {} | ||
147 | } | ||
148 | } | ||
149 | |||
150 | if !edit_map.is_empty() { | ||
151 | MappedRustChildDiagnostic::SuggestedFix(CodeAction { | ||
152 | title: rd.message.clone(), | ||
153 | kind: Some("quickfix".to_string()), | ||
154 | diagnostics: None, | ||
155 | edit: Some(WorkspaceEdit::new(edit_map)), | ||
156 | command: None, | ||
157 | is_preferred: None, | ||
158 | }) | ||
159 | } else { | ||
160 | MappedRustChildDiagnostic::Related(DiagnosticRelatedInformation { | ||
161 | location: map_span_to_location(spans[0], workspace_root), | ||
162 | message: rd.message.clone(), | ||
163 | }) | ||
164 | } | ||
165 | } | ||
166 | |||
167 | #[derive(Debug)] | ||
168 | pub(crate) struct MappedRustDiagnostic { | ||
169 | pub location: Location, | ||
170 | pub diagnostic: Diagnostic, | ||
171 | pub fixes: Vec<CodeAction>, | ||
172 | } | ||
173 | |||
174 | /// Converts a Rust root diagnostic to LSP form | ||
175 | /// | ||
176 | /// This flattens the Rust diagnostic by: | ||
177 | /// | ||
178 | /// 1. Creating a LSP diagnostic with the root message and primary span. | ||
179 | /// 2. Adding any labelled secondary spans to `relatedInformation` | ||
180 | /// 3. Categorising child diagnostics as either `SuggestedFix`es, | ||
181 | /// `relatedInformation` or additional message lines. | ||
182 | /// | ||
183 | /// If the diagnostic has no primary span this will return `None` | ||
184 | pub(crate) fn map_rust_diagnostic_to_lsp( | ||
185 | rd: &RustDiagnostic, | ||
186 | workspace_root: &PathBuf, | ||
187 | ) -> Vec<MappedRustDiagnostic> { | ||
188 | let primary_spans: Vec<&DiagnosticSpan> = rd.spans.iter().filter(|s| s.is_primary).collect(); | ||
189 | if primary_spans.is_empty() { | ||
190 | return vec![]; | ||
191 | } | ||
192 | |||
193 | let severity = map_level_to_severity(rd.level); | ||
194 | |||
195 | let mut source = String::from("rustc"); | ||
196 | let mut code = rd.code.as_ref().map(|c| c.code.clone()); | ||
197 | if let Some(code_val) = &code { | ||
198 | // See if this is an RFC #2103 scoped lint (e.g. from Clippy) | ||
199 | let scoped_code: Vec<&str> = code_val.split("::").collect(); | ||
200 | if scoped_code.len() == 2 { | ||
201 | source = String::from(scoped_code[0]); | ||
202 | code = Some(String::from(scoped_code[1])); | ||
203 | } | ||
204 | } | ||
205 | |||
206 | let mut needs_primary_span_label = true; | ||
207 | let mut related_information = vec![]; | ||
208 | let mut tags = vec![]; | ||
209 | |||
210 | for secondary_span in rd.spans.iter().filter(|s| !s.is_primary) { | ||
211 | let related = map_secondary_span_to_related(secondary_span, workspace_root); | ||
212 | if let Some(related) = related { | ||
213 | related_information.push(related); | ||
214 | } | ||
215 | } | ||
216 | |||
217 | let mut fixes = vec![]; | ||
218 | let mut message = rd.message.clone(); | ||
219 | for child in &rd.children { | ||
220 | let child = map_rust_child_diagnostic(&child, workspace_root); | ||
221 | match child { | ||
222 | MappedRustChildDiagnostic::Related(related) => related_information.push(related), | ||
223 | MappedRustChildDiagnostic::SuggestedFix(code_action) => fixes.push(code_action), | ||
224 | MappedRustChildDiagnostic::MessageLine(message_line) => { | ||
225 | write!(&mut message, "\n{}", message_line).unwrap(); | ||
226 | |||
227 | // These secondary messages usually duplicate the content of the | ||
228 | // primary span label. | ||
229 | needs_primary_span_label = false; | ||
230 | } | ||
231 | } | ||
232 | } | ||
233 | |||
234 | if is_unused_or_unnecessary(rd) { | ||
235 | tags.push(DiagnosticTag::Unnecessary); | ||
236 | } | ||
237 | |||
238 | if is_deprecated(rd) { | ||
239 | tags.push(DiagnosticTag::Deprecated); | ||
240 | } | ||
241 | |||
242 | primary_spans | ||
243 | .iter() | ||
244 | .map(|primary_span| { | ||
245 | let location = map_span_to_location(&primary_span, workspace_root); | ||
246 | |||
247 | let mut message = message.clone(); | ||
248 | if needs_primary_span_label { | ||
249 | if let Some(primary_span_label) = &primary_span.label { | ||
250 | write!(&mut message, "\n{}", primary_span_label).unwrap(); | ||
251 | } | ||
252 | } | ||
253 | |||
254 | // If error occurs from macro expansion, add related info pointing to | ||
255 | // where the error originated | ||
256 | if !is_from_macro(&primary_span.file_name) && primary_span.expansion.is_some() { | ||
257 | let def_loc = map_span_to_location_naive(&primary_span, workspace_root); | ||
258 | related_information.push(DiagnosticRelatedInformation { | ||
259 | location: def_loc, | ||
260 | message: "Error originated from macro here".to_string(), | ||
261 | }); | ||
262 | } | ||
263 | |||
264 | let diagnostic = Diagnostic { | ||
265 | range: location.range, | ||
266 | severity, | ||
267 | code: code.clone().map(NumberOrString::String), | ||
268 | source: Some(source.clone()), | ||
269 | message, | ||
270 | related_information: if !related_information.is_empty() { | ||
271 | Some(related_information.clone()) | ||
272 | } else { | ||
273 | None | ||
274 | }, | ||
275 | tags: if !tags.is_empty() { Some(tags.clone()) } else { None }, | ||
276 | }; | ||
277 | |||
278 | MappedRustDiagnostic { location, diagnostic, fixes: fixes.clone() } | ||
279 | }) | ||
280 | .collect() | ||
281 | } | ||
282 | |||
283 | /// Returns a `Url` object from a given path, will lowercase drive letters if present. | ||
284 | /// This will only happen when processing windows paths. | ||
285 | /// | ||
286 | /// When processing non-windows path, this is essentially the same as `Url::from_file_path`. | ||
287 | pub fn url_from_path_with_drive_lowercasing( | ||
288 | path: impl AsRef<Path>, | ||
289 | ) -> Result<Url, Box<dyn std::error::Error + Send + Sync>> { | ||
290 | let component_has_windows_drive = path.as_ref().components().any(|comp| { | ||
291 | if let Component::Prefix(c) = comp { | ||
292 | match c.kind() { | ||
293 | Prefix::Disk(_) | Prefix::VerbatimDisk(_) => return true, | ||
294 | _ => return false, | ||
295 | } | ||
296 | } | ||
297 | false | ||
298 | }); | ||
299 | |||
300 | // VSCode expects drive letters to be lowercased, where rust will uppercase the drive letters. | ||
301 | if component_has_windows_drive { | ||
302 | let url_original = Url::from_file_path(&path) | ||
303 | .map_err(|_| format!("can't convert path to url: {}", path.as_ref().display()))?; | ||
304 | |||
305 | let drive_partition: Vec<&str> = url_original.as_str().rsplitn(2, ':').collect(); | ||
306 | |||
307 | // There is a drive partition, but we never found a colon. | ||
308 | // This should not happen, but in this case we just pass it through. | ||
309 | if drive_partition.len() == 1 { | ||
310 | return Ok(url_original); | ||
311 | } | ||
312 | |||
313 | let joined = drive_partition[1].to_ascii_lowercase() + ":" + drive_partition[0]; | ||
314 | let url = Url::from_str(&joined).expect("This came from a valid `Url`"); | ||
315 | |||
316 | Ok(url) | ||
317 | } else { | ||
318 | Ok(Url::from_file_path(&path) | ||
319 | .map_err(|_| format!("can't convert path to url: {}", path.as_ref().display()))?) | ||
320 | } | ||
321 | } | ||
322 | |||
323 | // `Url` is not able to parse windows paths on unix machines. | ||
324 | #[cfg(target_os = "windows")] | ||
325 | #[cfg(test)] | ||
326 | mod path_conversion_windows_tests { | ||
327 | use super::url_from_path_with_drive_lowercasing; | ||
328 | #[test] | ||
329 | fn test_lowercase_drive_letter_with_drive() { | ||
330 | let url = url_from_path_with_drive_lowercasing("C:\\Test").unwrap(); | ||
331 | |||
332 | assert_eq!(url.to_string(), "file:///c:/Test"); | ||
333 | } | ||
334 | |||
335 | #[test] | ||
336 | fn test_drive_without_colon_passthrough() { | ||
337 | let url = url_from_path_with_drive_lowercasing(r#"\\localhost\C$\my_dir"#).unwrap(); | ||
338 | |||
339 | assert_eq!(url.to_string(), "file://localhost/C$/my_dir"); | ||
340 | } | ||
341 | } | ||
diff --git a/crates/ra_flycheck/src/conv/test.rs b/crates/ra_flycheck/src/conv/test.rs deleted file mode 100644 index 4e81455ca..000000000 --- a/crates/ra_flycheck/src/conv/test.rs +++ /dev/null | |||
@@ -1,1072 +0,0 @@ | |||
1 | //! This module contains the large and verbose snapshot tests for the | ||
2 | //! conversions between `cargo check` json and LSP diagnostics. | ||
3 | #[cfg(not(windows))] | ||
4 | use crate::*; | ||
5 | |||
6 | #[cfg(not(windows))] | ||
7 | fn parse_diagnostic(val: &str) -> cargo_metadata::diagnostic::Diagnostic { | ||
8 | serde_json::from_str::<cargo_metadata::diagnostic::Diagnostic>(val).unwrap() | ||
9 | } | ||
10 | |||
11 | #[test] | ||
12 | #[cfg(not(windows))] | ||
13 | fn snap_rustc_incompatible_type_for_trait() { | ||
14 | let diag = parse_diagnostic( | ||
15 | r##"{ | ||
16 | "message": "method `next` has an incompatible type for trait", | ||
17 | "code": { | ||
18 | "code": "E0053", | ||
19 | "explanation": "\nThe parameters of any trait method must match between a trait implementation\nand the trait definition.\n\nHere are a couple examples of this error:\n\n```compile_fail,E0053\ntrait Foo {\n fn foo(x: u16);\n fn bar(&self);\n}\n\nstruct Bar;\n\nimpl Foo for Bar {\n // error, expected u16, found i16\n fn foo(x: i16) { }\n\n // error, types differ in mutability\n fn bar(&mut self) { }\n}\n```\n" | ||
20 | }, | ||
21 | "level": "error", | ||
22 | "spans": [ | ||
23 | { | ||
24 | "file_name": "compiler/ty/list_iter.rs", | ||
25 | "byte_start": 1307, | ||
26 | "byte_end": 1350, | ||
27 | "line_start": 52, | ||
28 | "line_end": 52, | ||
29 | "column_start": 5, | ||
30 | "column_end": 48, | ||
31 | "is_primary": true, | ||
32 | "text": [ | ||
33 | { | ||
34 | "text": " fn next(&self) -> Option<&'list ty::Ref<M>> {", | ||
35 | "highlight_start": 5, | ||
36 | "highlight_end": 48 | ||
37 | } | ||
38 | ], | ||
39 | "label": "types differ in mutability", | ||
40 | "suggested_replacement": null, | ||
41 | "suggestion_applicability": null, | ||
42 | "expansion": null | ||
43 | } | ||
44 | ], | ||
45 | "children": [ | ||
46 | { | ||
47 | "message": "expected type `fn(&mut ty::list_iter::ListIterator<'list, M>) -> std::option::Option<&ty::Ref<M>>`\n found type `fn(&ty::list_iter::ListIterator<'list, M>) -> std::option::Option<&'list ty::Ref<M>>`", | ||
48 | "code": null, | ||
49 | "level": "note", | ||
50 | "spans": [], | ||
51 | "children": [], | ||
52 | "rendered": null | ||
53 | } | ||
54 | ], | ||
55 | "rendered": "error[E0053]: method `next` has an incompatible type for trait\n --> compiler/ty/list_iter.rs:52:5\n |\n52 | fn next(&self) -> Option<&'list ty::Ref<M>> {\n | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ types differ in mutability\n |\n = note: expected type `fn(&mut ty::list_iter::ListIterator<'list, M>) -> std::option::Option<&ty::Ref<M>>`\n found type `fn(&ty::list_iter::ListIterator<'list, M>) -> std::option::Option<&'list ty::Ref<M>>`\n\n" | ||
56 | } | ||
57 | "##, | ||
58 | ); | ||
59 | |||
60 | let workspace_root = PathBuf::from("/test/"); | ||
61 | let diag = map_rust_diagnostic_to_lsp(&diag, &workspace_root); | ||
62 | insta::assert_debug_snapshot!(diag); | ||
63 | } | ||
64 | |||
65 | #[test] | ||
66 | #[cfg(not(windows))] | ||
67 | fn snap_rustc_unused_variable() { | ||
68 | let diag = parse_diagnostic( | ||
69 | r##"{ | ||
70 | "message": "unused variable: `foo`", | ||
71 | "code": { | ||
72 | "code": "unused_variables", | ||
73 | "explanation": null | ||
74 | }, | ||
75 | "level": "warning", | ||
76 | "spans": [ | ||
77 | { | ||
78 | "file_name": "driver/subcommand/repl.rs", | ||
79 | "byte_start": 9228, | ||
80 | "byte_end": 9231, | ||
81 | "line_start": 291, | ||
82 | "line_end": 291, | ||
83 | "column_start": 9, | ||
84 | "column_end": 12, | ||
85 | "is_primary": true, | ||
86 | "text": [ | ||
87 | { | ||
88 | "text": " let foo = 42;", | ||
89 | "highlight_start": 9, | ||
90 | "highlight_end": 12 | ||
91 | } | ||
92 | ], | ||
93 | "label": null, | ||
94 | "suggested_replacement": null, | ||
95 | "suggestion_applicability": null, | ||
96 | "expansion": null | ||
97 | } | ||
98 | ], | ||
99 | "children": [ | ||
100 | { | ||
101 | "message": "#[warn(unused_variables)] on by default", | ||
102 | "code": null, | ||
103 | "level": "note", | ||
104 | "spans": [], | ||
105 | "children": [], | ||
106 | "rendered": null | ||
107 | }, | ||
108 | { | ||
109 | "message": "consider prefixing with an underscore", | ||
110 | "code": null, | ||
111 | "level": "help", | ||
112 | "spans": [ | ||
113 | { | ||
114 | "file_name": "driver/subcommand/repl.rs", | ||
115 | "byte_start": 9228, | ||
116 | "byte_end": 9231, | ||
117 | "line_start": 291, | ||
118 | "line_end": 291, | ||
119 | "column_start": 9, | ||
120 | "column_end": 12, | ||
121 | "is_primary": true, | ||
122 | "text": [ | ||
123 | { | ||
124 | "text": " let foo = 42;", | ||
125 | "highlight_start": 9, | ||
126 | "highlight_end": 12 | ||
127 | } | ||
128 | ], | ||
129 | "label": null, | ||
130 | "suggested_replacement": "_foo", | ||
131 | "suggestion_applicability": "MachineApplicable", | ||
132 | "expansion": null | ||
133 | } | ||
134 | ], | ||
135 | "children": [], | ||
136 | "rendered": null | ||
137 | } | ||
138 | ], | ||
139 | "rendered": "warning: unused variable: `foo`\n --> driver/subcommand/repl.rs:291:9\n |\n291 | let foo = 42;\n | ^^^ help: consider prefixing with an underscore: `_foo`\n |\n = note: #[warn(unused_variables)] on by default\n\n" | ||
140 | }"##, | ||
141 | ); | ||
142 | |||
143 | let workspace_root = PathBuf::from("/test/"); | ||
144 | let diag = map_rust_diagnostic_to_lsp(&diag, &workspace_root); | ||
145 | insta::assert_debug_snapshot!(diag); | ||
146 | } | ||
147 | |||
148 | #[test] | ||
149 | #[cfg(not(windows))] | ||
150 | fn snap_rustc_wrong_number_of_parameters() { | ||
151 | let diag = parse_diagnostic( | ||
152 | r##"{ | ||
153 | "message": "this function takes 2 parameters but 3 parameters were supplied", | ||
154 | "code": { | ||
155 | "code": "E0061", | ||
156 | "explanation": "\nThe number of arguments passed to a function must match the number of arguments\nspecified in the function signature.\n\nFor example, a function like:\n\n```\nfn f(a: u16, b: &str) {}\n```\n\nMust always be called with exactly two arguments, e.g., `f(2, \"test\")`.\n\nNote that Rust does not have a notion of optional function arguments or\nvariadic functions (except for its C-FFI).\n" | ||
157 | }, | ||
158 | "level": "error", | ||
159 | "spans": [ | ||
160 | { | ||
161 | "file_name": "compiler/ty/select.rs", | ||
162 | "byte_start": 8787, | ||
163 | "byte_end": 9241, | ||
164 | "line_start": 219, | ||
165 | "line_end": 231, | ||
166 | "column_start": 5, | ||
167 | "column_end": 6, | ||
168 | "is_primary": false, | ||
169 | "text": [ | ||
170 | { | ||
171 | "text": " pub fn add_evidence(", | ||
172 | "highlight_start": 5, | ||
173 | "highlight_end": 25 | ||
174 | }, | ||
175 | { | ||
176 | "text": " &mut self,", | ||
177 | "highlight_start": 1, | ||
178 | "highlight_end": 19 | ||
179 | }, | ||
180 | { | ||
181 | "text": " target_poly: &ty::Ref<ty::Poly>,", | ||
182 | "highlight_start": 1, | ||
183 | "highlight_end": 41 | ||
184 | }, | ||
185 | { | ||
186 | "text": " evidence_poly: &ty::Ref<ty::Poly>,", | ||
187 | "highlight_start": 1, | ||
188 | "highlight_end": 43 | ||
189 | }, | ||
190 | { | ||
191 | "text": " ) {", | ||
192 | "highlight_start": 1, | ||
193 | "highlight_end": 8 | ||
194 | }, | ||
195 | { | ||
196 | "text": " match target_poly {", | ||
197 | "highlight_start": 1, | ||
198 | "highlight_end": 28 | ||
199 | }, | ||
200 | { | ||
201 | "text": " ty::Ref::Var(tvar, _) => self.add_var_evidence(tvar, evidence_poly),", | ||
202 | "highlight_start": 1, | ||
203 | "highlight_end": 81 | ||
204 | }, | ||
205 | { | ||
206 | "text": " ty::Ref::Fixed(target_ty) => {", | ||
207 | "highlight_start": 1, | ||
208 | "highlight_end": 43 | ||
209 | }, | ||
210 | { | ||
211 | "text": " let evidence_ty = evidence_poly.resolve_to_ty();", | ||
212 | "highlight_start": 1, | ||
213 | "highlight_end": 65 | ||
214 | }, | ||
215 | { | ||
216 | "text": " self.add_evidence_ty(target_ty, evidence_poly, evidence_ty)", | ||
217 | "highlight_start": 1, | ||
218 | "highlight_end": 76 | ||
219 | }, | ||
220 | { | ||
221 | "text": " }", | ||
222 | "highlight_start": 1, | ||
223 | "highlight_end": 14 | ||
224 | }, | ||
225 | { | ||
226 | "text": " }", | ||
227 | "highlight_start": 1, | ||
228 | "highlight_end": 10 | ||
229 | }, | ||
230 | { | ||
231 | "text": " }", | ||
232 | "highlight_start": 1, | ||
233 | "highlight_end": 6 | ||
234 | } | ||
235 | ], | ||
236 | "label": "defined here", | ||
237 | "suggested_replacement": null, | ||
238 | "suggestion_applicability": null, | ||
239 | "expansion": null | ||
240 | }, | ||
241 | { | ||
242 | "file_name": "compiler/ty/select.rs", | ||
243 | "byte_start": 4045, | ||
244 | "byte_end": 4057, | ||
245 | "line_start": 104, | ||
246 | "line_end": 104, | ||
247 | "column_start": 18, | ||
248 | "column_end": 30, | ||
249 | "is_primary": true, | ||
250 | "text": [ | ||
251 | { | ||
252 | "text": " self.add_evidence(target_fixed, evidence_fixed, false);", | ||
253 | "highlight_start": 18, | ||
254 | "highlight_end": 30 | ||
255 | } | ||
256 | ], | ||
257 | "label": "expected 2 parameters", | ||
258 | "suggested_replacement": null, | ||
259 | "suggestion_applicability": null, | ||
260 | "expansion": null | ||
261 | } | ||
262 | ], | ||
263 | "children": [], | ||
264 | "rendered": "error[E0061]: this function takes 2 parameters but 3 parameters were supplied\n --> compiler/ty/select.rs:104:18\n |\n104 | self.add_evidence(target_fixed, evidence_fixed, false);\n | ^^^^^^^^^^^^ expected 2 parameters\n...\n219 | / pub fn add_evidence(\n220 | | &mut self,\n221 | | target_poly: &ty::Ref<ty::Poly>,\n222 | | evidence_poly: &ty::Ref<ty::Poly>,\n... |\n230 | | }\n231 | | }\n | |_____- defined here\n\n" | ||
265 | }"##, | ||
266 | ); | ||
267 | |||
268 | let workspace_root = PathBuf::from("/test/"); | ||
269 | let diag = map_rust_diagnostic_to_lsp(&diag, &workspace_root); | ||
270 | insta::assert_debug_snapshot!(diag); | ||
271 | } | ||
272 | |||
273 | #[test] | ||
274 | #[cfg(not(windows))] | ||
275 | fn snap_clippy_pass_by_ref() { | ||
276 | let diag = parse_diagnostic( | ||
277 | r##"{ | ||
278 | "message": "this argument is passed by reference, but would be more efficient if passed by value", | ||
279 | "code": { | ||
280 | "code": "clippy::trivially_copy_pass_by_ref", | ||
281 | "explanation": null | ||
282 | }, | ||
283 | "level": "warning", | ||
284 | "spans": [ | ||
285 | { | ||
286 | "file_name": "compiler/mir/tagset.rs", | ||
287 | "byte_start": 941, | ||
288 | "byte_end": 946, | ||
289 | "line_start": 42, | ||
290 | "line_end": 42, | ||
291 | "column_start": 24, | ||
292 | "column_end": 29, | ||
293 | "is_primary": true, | ||
294 | "text": [ | ||
295 | { | ||
296 | "text": " pub fn is_disjoint(&self, other: Self) -> bool {", | ||
297 | "highlight_start": 24, | ||
298 | "highlight_end": 29 | ||
299 | } | ||
300 | ], | ||
301 | "label": null, | ||
302 | "suggested_replacement": null, | ||
303 | "suggestion_applicability": null, | ||
304 | "expansion": null | ||
305 | } | ||
306 | ], | ||
307 | "children": [ | ||
308 | { | ||
309 | "message": "lint level defined here", | ||
310 | "code": null, | ||
311 | "level": "note", | ||
312 | "spans": [ | ||
313 | { | ||
314 | "file_name": "compiler/lib.rs", | ||
315 | "byte_start": 8, | ||
316 | "byte_end": 19, | ||
317 | "line_start": 1, | ||
318 | "line_end": 1, | ||
319 | "column_start": 9, | ||
320 | "column_end": 20, | ||
321 | "is_primary": true, | ||
322 | "text": [ | ||
323 | { | ||
324 | "text": "#![warn(clippy::all)]", | ||
325 | "highlight_start": 9, | ||
326 | "highlight_end": 20 | ||
327 | } | ||
328 | ], | ||
329 | "label": null, | ||
330 | "suggested_replacement": null, | ||
331 | "suggestion_applicability": null, | ||
332 | "expansion": null | ||
333 | } | ||
334 | ], | ||
335 | "children": [], | ||
336 | "rendered": null | ||
337 | }, | ||
338 | { | ||
339 | "message": "#[warn(clippy::trivially_copy_pass_by_ref)] implied by #[warn(clippy::all)]", | ||
340 | "code": null, | ||
341 | "level": "note", | ||
342 | "spans": [], | ||
343 | "children": [], | ||
344 | "rendered": null | ||
345 | }, | ||
346 | { | ||
347 | "message": "for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#trivially_copy_pass_by_ref", | ||
348 | "code": null, | ||
349 | "level": "help", | ||
350 | "spans": [], | ||
351 | "children": [], | ||
352 | "rendered": null | ||
353 | }, | ||
354 | { | ||
355 | "message": "consider passing by value instead", | ||
356 | "code": null, | ||
357 | "level": "help", | ||
358 | "spans": [ | ||
359 | { | ||
360 | "file_name": "compiler/mir/tagset.rs", | ||
361 | "byte_start": 941, | ||
362 | "byte_end": 946, | ||
363 | "line_start": 42, | ||
364 | "line_end": 42, | ||
365 | "column_start": 24, | ||
366 | "column_end": 29, | ||
367 | "is_primary": true, | ||
368 | "text": [ | ||
369 | { | ||
370 | "text": " pub fn is_disjoint(&self, other: Self) -> bool {", | ||
371 | "highlight_start": 24, | ||
372 | "highlight_end": 29 | ||
373 | } | ||
374 | ], | ||
375 | "label": null, | ||
376 | "suggested_replacement": "self", | ||
377 | "suggestion_applicability": "Unspecified", | ||
378 | "expansion": null | ||
379 | } | ||
380 | ], | ||
381 | "children": [], | ||
382 | "rendered": null | ||
383 | } | ||
384 | ], | ||
385 | "rendered": "warning: this argument is passed by reference, but would be more efficient if passed by value\n --> compiler/mir/tagset.rs:42:24\n |\n42 | pub fn is_disjoint(&self, other: Self) -> bool {\n | ^^^^^ help: consider passing by value instead: `self`\n |\nnote: lint level defined here\n --> compiler/lib.rs:1:9\n |\n1 | #![warn(clippy::all)]\n | ^^^^^^^^^^^\n = note: #[warn(clippy::trivially_copy_pass_by_ref)] implied by #[warn(clippy::all)]\n = help: for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#trivially_copy_pass_by_ref\n\n" | ||
386 | }"##, | ||
387 | ); | ||
388 | |||
389 | let workspace_root = PathBuf::from("/test/"); | ||
390 | let diag = map_rust_diagnostic_to_lsp(&diag, &workspace_root); | ||
391 | insta::assert_debug_snapshot!(diag); | ||
392 | } | ||
393 | |||
394 | #[test] | ||
395 | #[cfg(not(windows))] | ||
396 | fn snap_rustc_mismatched_type() { | ||
397 | let diag = parse_diagnostic( | ||
398 | r##"{ | ||
399 | "message": "mismatched types", | ||
400 | "code": { | ||
401 | "code": "E0308", | ||
402 | "explanation": "\nThis error occurs when the compiler was unable to infer the concrete type of a\nvariable. It can occur for several cases, the most common of which is a\nmismatch in the expected type that the compiler inferred for a variable's\ninitializing expression, and the actual type explicitly assigned to the\nvariable.\n\nFor example:\n\n```compile_fail,E0308\nlet x: i32 = \"I am not a number!\";\n// ~~~ ~~~~~~~~~~~~~~~~~~~~\n// | |\n// | initializing expression;\n// | compiler infers type `&str`\n// |\n// type `i32` assigned to variable `x`\n```\n" | ||
403 | }, | ||
404 | "level": "error", | ||
405 | "spans": [ | ||
406 | { | ||
407 | "file_name": "runtime/compiler_support.rs", | ||
408 | "byte_start": 1589, | ||
409 | "byte_end": 1594, | ||
410 | "line_start": 48, | ||
411 | "line_end": 48, | ||
412 | "column_start": 65, | ||
413 | "column_end": 70, | ||
414 | "is_primary": true, | ||
415 | "text": [ | ||
416 | { | ||
417 | "text": " let layout = alloc::Layout::from_size_align_unchecked(size, align);", | ||
418 | "highlight_start": 65, | ||
419 | "highlight_end": 70 | ||
420 | } | ||
421 | ], | ||
422 | "label": "expected usize, found u32", | ||
423 | "suggested_replacement": null, | ||
424 | "suggestion_applicability": null, | ||
425 | "expansion": null | ||
426 | } | ||
427 | ], | ||
428 | "children": [], | ||
429 | "rendered": "error[E0308]: mismatched types\n --> runtime/compiler_support.rs:48:65\n |\n48 | let layout = alloc::Layout::from_size_align_unchecked(size, align);\n | ^^^^^ expected usize, found u32\n\n" | ||
430 | }"##, | ||
431 | ); | ||
432 | |||
433 | let workspace_root = PathBuf::from("/test/"); | ||
434 | let diag = map_rust_diagnostic_to_lsp(&diag, &workspace_root); | ||
435 | insta::assert_debug_snapshot!(diag); | ||
436 | } | ||
437 | |||
438 | #[test] | ||
439 | #[cfg(not(windows))] | ||
440 | fn snap_handles_macro_location() { | ||
441 | let diag = parse_diagnostic( | ||
442 | r##"{ | ||
443 | "rendered": "error[E0277]: can't compare `{integer}` with `&str`\n --> src/main.rs:2:5\n |\n2 | assert_eq!(1, \"love\");\n | ^^^^^^^^^^^^^^^^^^^^^^ no implementation for `{integer} == &str`\n |\n = help: the trait `std::cmp::PartialEq<&str>` is not implemented for `{integer}`\n = note: this error originates in a macro outside of the current crate (in Nightly builds, run with -Z external-macro-backtrace for more info)\n\n", | ||
444 | "children": [ | ||
445 | { | ||
446 | "children": [], | ||
447 | "code": null, | ||
448 | "level": "help", | ||
449 | "message": "the trait `std::cmp::PartialEq<&str>` is not implemented for `{integer}`", | ||
450 | "rendered": null, | ||
451 | "spans": [] | ||
452 | } | ||
453 | ], | ||
454 | "code": { | ||
455 | "code": "E0277", | ||
456 | "explanation": "\nYou tried to use a type which doesn't implement some trait in a place which\nexpected that trait. Erroneous code example:\n\n```compile_fail,E0277\n// here we declare the Foo trait with a bar method\ntrait Foo {\n fn bar(&self);\n}\n\n// we now declare a function which takes an object implementing the Foo trait\nfn some_func<T: Foo>(foo: T) {\n foo.bar();\n}\n\nfn main() {\n // we now call the method with the i32 type, which doesn't implement\n // the Foo trait\n some_func(5i32); // error: the trait bound `i32 : Foo` is not satisfied\n}\n```\n\nIn order to fix this error, verify that the type you're using does implement\nthe trait. Example:\n\n```\ntrait Foo {\n fn bar(&self);\n}\n\nfn some_func<T: Foo>(foo: T) {\n foo.bar(); // we can now use this method since i32 implements the\n // Foo trait\n}\n\n// we implement the trait on the i32 type\nimpl Foo for i32 {\n fn bar(&self) {}\n}\n\nfn main() {\n some_func(5i32); // ok!\n}\n```\n\nOr in a generic context, an erroneous code example would look like:\n\n```compile_fail,E0277\nfn some_func<T>(foo: T) {\n println!(\"{:?}\", foo); // error: the trait `core::fmt::Debug` is not\n // implemented for the type `T`\n}\n\nfn main() {\n // We now call the method with the i32 type,\n // which *does* implement the Debug trait.\n some_func(5i32);\n}\n```\n\nNote that the error here is in the definition of the generic function: Although\nwe only call it with a parameter that does implement `Debug`, the compiler\nstill rejects the function: It must work with all possible input types. In\norder to make this example compile, we need to restrict the generic type we're\naccepting:\n\n```\nuse std::fmt;\n\n// Restrict the input type to types that implement Debug.\nfn some_func<T: fmt::Debug>(foo: T) {\n println!(\"{:?}\", foo);\n}\n\nfn main() {\n // Calling the method is still fine, as i32 implements Debug.\n some_func(5i32);\n\n // This would fail to compile now:\n // struct WithoutDebug;\n // some_func(WithoutDebug);\n}\n```\n\nRust only looks at the signature of the called function, as such it must\nalready specify all requirements that will be used for every type parameter.\n" | ||
457 | }, | ||
458 | "level": "error", | ||
459 | "message": "can't compare `{integer}` with `&str`", | ||
460 | "spans": [ | ||
461 | { | ||
462 | "byte_end": 155, | ||
463 | "byte_start": 153, | ||
464 | "column_end": 33, | ||
465 | "column_start": 31, | ||
466 | "expansion": { | ||
467 | "def_site_span": { | ||
468 | "byte_end": 940, | ||
469 | "byte_start": 0, | ||
470 | "column_end": 6, | ||
471 | "column_start": 1, | ||
472 | "expansion": null, | ||
473 | "file_name": "<::core::macros::assert_eq macros>", | ||
474 | "is_primary": false, | ||
475 | "label": null, | ||
476 | "line_end": 36, | ||
477 | "line_start": 1, | ||
478 | "suggested_replacement": null, | ||
479 | "suggestion_applicability": null, | ||
480 | "text": [ | ||
481 | { | ||
482 | "highlight_end": 35, | ||
483 | "highlight_start": 1, | ||
484 | "text": "($ left : expr, $ right : expr) =>" | ||
485 | }, | ||
486 | { | ||
487 | "highlight_end": 3, | ||
488 | "highlight_start": 1, | ||
489 | "text": "({" | ||
490 | }, | ||
491 | { | ||
492 | "highlight_end": 33, | ||
493 | "highlight_start": 1, | ||
494 | "text": " match (& $ left, & $ right)" | ||
495 | }, | ||
496 | { | ||
497 | "highlight_end": 7, | ||
498 | "highlight_start": 1, | ||
499 | "text": " {" | ||
500 | }, | ||
501 | { | ||
502 | "highlight_end": 34, | ||
503 | "highlight_start": 1, | ||
504 | "text": " (left_val, right_val) =>" | ||
505 | }, | ||
506 | { | ||
507 | "highlight_end": 11, | ||
508 | "highlight_start": 1, | ||
509 | "text": " {" | ||
510 | }, | ||
511 | { | ||
512 | "highlight_end": 46, | ||
513 | "highlight_start": 1, | ||
514 | "text": " if ! (* left_val == * right_val)" | ||
515 | }, | ||
516 | { | ||
517 | "highlight_end": 15, | ||
518 | "highlight_start": 1, | ||
519 | "text": " {" | ||
520 | }, | ||
521 | { | ||
522 | "highlight_end": 25, | ||
523 | "highlight_start": 1, | ||
524 | "text": " panic !" | ||
525 | }, | ||
526 | { | ||
527 | "highlight_end": 57, | ||
528 | "highlight_start": 1, | ||
529 | "text": " (r#\"assertion failed: `(left == right)`" | ||
530 | }, | ||
531 | { | ||
532 | "highlight_end": 16, | ||
533 | "highlight_start": 1, | ||
534 | "text": " left: `{:?}`," | ||
535 | }, | ||
536 | { | ||
537 | "highlight_end": 18, | ||
538 | "highlight_start": 1, | ||
539 | "text": " right: `{:?}`\"#," | ||
540 | }, | ||
541 | { | ||
542 | "highlight_end": 47, | ||
543 | "highlight_start": 1, | ||
544 | "text": " & * left_val, & * right_val)" | ||
545 | }, | ||
546 | { | ||
547 | "highlight_end": 15, | ||
548 | "highlight_start": 1, | ||
549 | "text": " }" | ||
550 | }, | ||
551 | { | ||
552 | "highlight_end": 11, | ||
553 | "highlight_start": 1, | ||
554 | "text": " }" | ||
555 | }, | ||
556 | { | ||
557 | "highlight_end": 7, | ||
558 | "highlight_start": 1, | ||
559 | "text": " }" | ||
560 | }, | ||
561 | { | ||
562 | "highlight_end": 42, | ||
563 | "highlight_start": 1, | ||
564 | "text": " }) ; ($ left : expr, $ right : expr,) =>" | ||
565 | }, | ||
566 | { | ||
567 | "highlight_end": 49, | ||
568 | "highlight_start": 1, | ||
569 | "text": "({ $ crate :: assert_eq ! ($ left, $ right) }) ;" | ||
570 | }, | ||
571 | { | ||
572 | "highlight_end": 53, | ||
573 | "highlight_start": 1, | ||
574 | "text": "($ left : expr, $ right : expr, $ ($ arg : tt) +) =>" | ||
575 | }, | ||
576 | { | ||
577 | "highlight_end": 3, | ||
578 | "highlight_start": 1, | ||
579 | "text": "({" | ||
580 | }, | ||
581 | { | ||
582 | "highlight_end": 37, | ||
583 | "highlight_start": 1, | ||
584 | "text": " match (& ($ left), & ($ right))" | ||
585 | }, | ||
586 | { | ||
587 | "highlight_end": 7, | ||
588 | "highlight_start": 1, | ||
589 | "text": " {" | ||
590 | }, | ||
591 | { | ||
592 | "highlight_end": 34, | ||
593 | "highlight_start": 1, | ||
594 | "text": " (left_val, right_val) =>" | ||
595 | }, | ||
596 | { | ||
597 | "highlight_end": 11, | ||
598 | "highlight_start": 1, | ||
599 | "text": " {" | ||
600 | }, | ||
601 | { | ||
602 | "highlight_end": 46, | ||
603 | "highlight_start": 1, | ||
604 | "text": " if ! (* left_val == * right_val)" | ||
605 | }, | ||
606 | { | ||
607 | "highlight_end": 15, | ||
608 | "highlight_start": 1, | ||
609 | "text": " {" | ||
610 | }, | ||
611 | { | ||
612 | "highlight_end": 25, | ||
613 | "highlight_start": 1, | ||
614 | "text": " panic !" | ||
615 | }, | ||
616 | { | ||
617 | "highlight_end": 57, | ||
618 | "highlight_start": 1, | ||
619 | "text": " (r#\"assertion failed: `(left == right)`" | ||
620 | }, | ||
621 | { | ||
622 | "highlight_end": 16, | ||
623 | "highlight_start": 1, | ||
624 | "text": " left: `{:?}`," | ||
625 | }, | ||
626 | { | ||
627 | "highlight_end": 22, | ||
628 | "highlight_start": 1, | ||
629 | "text": " right: `{:?}`: {}\"#," | ||
630 | }, | ||
631 | { | ||
632 | "highlight_end": 72, | ||
633 | "highlight_start": 1, | ||
634 | "text": " & * left_val, & * right_val, $ crate :: format_args !" | ||
635 | }, | ||
636 | { | ||
637 | "highlight_end": 33, | ||
638 | "highlight_start": 1, | ||
639 | "text": " ($ ($ arg) +))" | ||
640 | }, | ||
641 | { | ||
642 | "highlight_end": 15, | ||
643 | "highlight_start": 1, | ||
644 | "text": " }" | ||
645 | }, | ||
646 | { | ||
647 | "highlight_end": 11, | ||
648 | "highlight_start": 1, | ||
649 | "text": " }" | ||
650 | }, | ||
651 | { | ||
652 | "highlight_end": 7, | ||
653 | "highlight_start": 1, | ||
654 | "text": " }" | ||
655 | }, | ||
656 | { | ||
657 | "highlight_end": 6, | ||
658 | "highlight_start": 1, | ||
659 | "text": " }) ;" | ||
660 | } | ||
661 | ] | ||
662 | }, | ||
663 | "macro_decl_name": "assert_eq!", | ||
664 | "span": { | ||
665 | "byte_end": 38, | ||
666 | "byte_start": 16, | ||
667 | "column_end": 27, | ||
668 | "column_start": 5, | ||
669 | "expansion": null, | ||
670 | "file_name": "src/main.rs", | ||
671 | "is_primary": false, | ||
672 | "label": null, | ||
673 | "line_end": 2, | ||
674 | "line_start": 2, | ||
675 | "suggested_replacement": null, | ||
676 | "suggestion_applicability": null, | ||
677 | "text": [ | ||
678 | { | ||
679 | "highlight_end": 27, | ||
680 | "highlight_start": 5, | ||
681 | "text": " assert_eq!(1, \"love\");" | ||
682 | } | ||
683 | ] | ||
684 | } | ||
685 | }, | ||
686 | "file_name": "<::core::macros::assert_eq macros>", | ||
687 | "is_primary": true, | ||
688 | "label": "no implementation for `{integer} == &str`", | ||
689 | "line_end": 7, | ||
690 | "line_start": 7, | ||
691 | "suggested_replacement": null, | ||
692 | "suggestion_applicability": null, | ||
693 | "text": [ | ||
694 | { | ||
695 | "highlight_end": 33, | ||
696 | "highlight_start": 31, | ||
697 | "text": " if ! (* left_val == * right_val)" | ||
698 | } | ||
699 | ] | ||
700 | } | ||
701 | ] | ||
702 | }"##, | ||
703 | ); | ||
704 | |||
705 | let workspace_root = PathBuf::from("/test/"); | ||
706 | let diag = map_rust_diagnostic_to_lsp(&diag, &workspace_root); | ||
707 | insta::assert_debug_snapshot!(diag); | ||
708 | } | ||
709 | |||
710 | #[test] | ||
711 | #[cfg(not(windows))] | ||
712 | fn snap_macro_compiler_error() { | ||
713 | let diag = parse_diagnostic( | ||
714 | r##"{ | ||
715 | "rendered": "error: Please register your known path in the path module\n --> crates/ra_hir_def/src/path.rs:265:9\n |\n265 | compile_error!(\"Please register your known path in the path module\")\n | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^\n | \n ::: crates/ra_hir_def/src/data.rs:80:16\n |\n80 | let path = path![std::future::Future];\n | -------------------------- in this macro invocation\n\n", | ||
716 | "children": [], | ||
717 | "code": null, | ||
718 | "level": "error", | ||
719 | "message": "Please register your known path in the path module", | ||
720 | "spans": [ | ||
721 | { | ||
722 | "byte_end": 8285, | ||
723 | "byte_start": 8217, | ||
724 | "column_end": 77, | ||
725 | "column_start": 9, | ||
726 | "expansion": { | ||
727 | "def_site_span": { | ||
728 | "byte_end": 8294, | ||
729 | "byte_start": 7858, | ||
730 | "column_end": 2, | ||
731 | "column_start": 1, | ||
732 | "expansion": null, | ||
733 | "file_name": "crates/ra_hir_def/src/path.rs", | ||
734 | "is_primary": false, | ||
735 | "label": null, | ||
736 | "line_end": 267, | ||
737 | "line_start": 254, | ||
738 | "suggested_replacement": null, | ||
739 | "suggestion_applicability": null, | ||
740 | "text": [ | ||
741 | { | ||
742 | "highlight_end": 28, | ||
743 | "highlight_start": 1, | ||
744 | "text": "macro_rules! __known_path {" | ||
745 | }, | ||
746 | { | ||
747 | "highlight_end": 37, | ||
748 | "highlight_start": 1, | ||
749 | "text": " (std::iter::IntoIterator) => {};" | ||
750 | }, | ||
751 | { | ||
752 | "highlight_end": 33, | ||
753 | "highlight_start": 1, | ||
754 | "text": " (std::result::Result) => {};" | ||
755 | }, | ||
756 | { | ||
757 | "highlight_end": 29, | ||
758 | "highlight_start": 1, | ||
759 | "text": " (std::ops::Range) => {};" | ||
760 | }, | ||
761 | { | ||
762 | "highlight_end": 33, | ||
763 | "highlight_start": 1, | ||
764 | "text": " (std::ops::RangeFrom) => {};" | ||
765 | }, | ||
766 | { | ||
767 | "highlight_end": 33, | ||
768 | "highlight_start": 1, | ||
769 | "text": " (std::ops::RangeFull) => {};" | ||
770 | }, | ||
771 | { | ||
772 | "highlight_end": 31, | ||
773 | "highlight_start": 1, | ||
774 | "text": " (std::ops::RangeTo) => {};" | ||
775 | }, | ||
776 | { | ||
777 | "highlight_end": 40, | ||
778 | "highlight_start": 1, | ||
779 | "text": " (std::ops::RangeToInclusive) => {};" | ||
780 | }, | ||
781 | { | ||
782 | "highlight_end": 38, | ||
783 | "highlight_start": 1, | ||
784 | "text": " (std::ops::RangeInclusive) => {};" | ||
785 | }, | ||
786 | { | ||
787 | "highlight_end": 27, | ||
788 | "highlight_start": 1, | ||
789 | "text": " (std::ops::Try) => {};" | ||
790 | }, | ||
791 | { | ||
792 | "highlight_end": 22, | ||
793 | "highlight_start": 1, | ||
794 | "text": " ($path:path) => {" | ||
795 | }, | ||
796 | { | ||
797 | "highlight_end": 77, | ||
798 | "highlight_start": 1, | ||
799 | "text": " compile_error!(\"Please register your known path in the path module\")" | ||
800 | }, | ||
801 | { | ||
802 | "highlight_end": 7, | ||
803 | "highlight_start": 1, | ||
804 | "text": " };" | ||
805 | }, | ||
806 | { | ||
807 | "highlight_end": 2, | ||
808 | "highlight_start": 1, | ||
809 | "text": "}" | ||
810 | } | ||
811 | ] | ||
812 | }, | ||
813 | "macro_decl_name": "$crate::__known_path!", | ||
814 | "span": { | ||
815 | "byte_end": 8427, | ||
816 | "byte_start": 8385, | ||
817 | "column_end": 51, | ||
818 | "column_start": 9, | ||
819 | "expansion": { | ||
820 | "def_site_span": { | ||
821 | "byte_end": 8611, | ||
822 | "byte_start": 8312, | ||
823 | "column_end": 2, | ||
824 | "column_start": 1, | ||
825 | "expansion": null, | ||
826 | "file_name": "crates/ra_hir_def/src/path.rs", | ||
827 | "is_primary": false, | ||
828 | "label": null, | ||
829 | "line_end": 277, | ||
830 | "line_start": 270, | ||
831 | "suggested_replacement": null, | ||
832 | "suggestion_applicability": null, | ||
833 | "text": [ | ||
834 | { | ||
835 | "highlight_end": 22, | ||
836 | "highlight_start": 1, | ||
837 | "text": "macro_rules! __path {" | ||
838 | }, | ||
839 | { | ||
840 | "highlight_end": 43, | ||
841 | "highlight_start": 1, | ||
842 | "text": " ($start:ident $(:: $seg:ident)*) => ({" | ||
843 | }, | ||
844 | { | ||
845 | "highlight_end": 51, | ||
846 | "highlight_start": 1, | ||
847 | "text": " $crate::__known_path!($start $(:: $seg)*);" | ||
848 | }, | ||
849 | { | ||
850 | "highlight_end": 87, | ||
851 | "highlight_start": 1, | ||
852 | "text": " $crate::path::ModPath::from_simple_segments($crate::path::PathKind::Abs, vec![" | ||
853 | }, | ||
854 | { | ||
855 | "highlight_end": 76, | ||
856 | "highlight_start": 1, | ||
857 | "text": " $crate::path::__name![$start], $($crate::path::__name![$seg],)*" | ||
858 | }, | ||
859 | { | ||
860 | "highlight_end": 11, | ||
861 | "highlight_start": 1, | ||
862 | "text": " ])" | ||
863 | }, | ||
864 | { | ||
865 | "highlight_end": 8, | ||
866 | "highlight_start": 1, | ||
867 | "text": " });" | ||
868 | }, | ||
869 | { | ||
870 | "highlight_end": 2, | ||
871 | "highlight_start": 1, | ||
872 | "text": "}" | ||
873 | } | ||
874 | ] | ||
875 | }, | ||
876 | "macro_decl_name": "path!", | ||
877 | "span": { | ||
878 | "byte_end": 2966, | ||
879 | "byte_start": 2940, | ||
880 | "column_end": 42, | ||
881 | "column_start": 16, | ||
882 | "expansion": null, | ||
883 | "file_name": "crates/ra_hir_def/src/data.rs", | ||
884 | "is_primary": false, | ||
885 | "label": null, | ||
886 | "line_end": 80, | ||
887 | "line_start": 80, | ||
888 | "suggested_replacement": null, | ||
889 | "suggestion_applicability": null, | ||
890 | "text": [ | ||
891 | { | ||
892 | "highlight_end": 42, | ||
893 | "highlight_start": 16, | ||
894 | "text": " let path = path![std::future::Future];" | ||
895 | } | ||
896 | ] | ||
897 | } | ||
898 | }, | ||
899 | "file_name": "crates/ra_hir_def/src/path.rs", | ||
900 | "is_primary": false, | ||
901 | "label": null, | ||
902 | "line_end": 272, | ||
903 | "line_start": 272, | ||
904 | "suggested_replacement": null, | ||
905 | "suggestion_applicability": null, | ||
906 | "text": [ | ||
907 | { | ||
908 | "highlight_end": 51, | ||
909 | "highlight_start": 9, | ||
910 | "text": " $crate::__known_path!($start $(:: $seg)*);" | ||
911 | } | ||
912 | ] | ||
913 | } | ||
914 | }, | ||
915 | "file_name": "crates/ra_hir_def/src/path.rs", | ||
916 | "is_primary": true, | ||
917 | "label": null, | ||
918 | "line_end": 265, | ||
919 | "line_start": 265, | ||
920 | "suggested_replacement": null, | ||
921 | "suggestion_applicability": null, | ||
922 | "text": [ | ||
923 | { | ||
924 | "highlight_end": 77, | ||
925 | "highlight_start": 9, | ||
926 | "text": " compile_error!(\"Please register your known path in the path module\")" | ||
927 | } | ||
928 | ] | ||
929 | } | ||
930 | ] | ||
931 | } | ||
932 | "##, | ||
933 | ); | ||
934 | |||
935 | let workspace_root = PathBuf::from("/test/"); | ||
936 | let diag = map_rust_diagnostic_to_lsp(&diag, &workspace_root); | ||
937 | insta::assert_debug_snapshot!(diag); | ||
938 | } | ||
939 | |||
940 | #[test] | ||
941 | #[cfg(not(windows))] | ||
942 | fn snap_multi_line_fix() { | ||
943 | let diag = parse_diagnostic( | ||
944 | r##"{ | ||
945 | "rendered": "warning: returning the result of a let binding from a block\n --> src/main.rs:4:5\n |\n3 | let a = (0..10).collect();\n | -------------------------- unnecessary let binding\n4 | a\n | ^\n |\n = note: `#[warn(clippy::let_and_return)]` on by default\n = help: for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#let_and_return\nhelp: return the expression directly\n |\n3 | \n4 | (0..10).collect()\n |\n\n", | ||
946 | "children": [ | ||
947 | { | ||
948 | "children": [], | ||
949 | "code": null, | ||
950 | "level": "note", | ||
951 | "message": "`#[warn(clippy::let_and_return)]` on by default", | ||
952 | "rendered": null, | ||
953 | "spans": [] | ||
954 | }, | ||
955 | { | ||
956 | "children": [], | ||
957 | "code": null, | ||
958 | "level": "help", | ||
959 | "message": "for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#let_and_return", | ||
960 | "rendered": null, | ||
961 | "spans": [] | ||
962 | }, | ||
963 | { | ||
964 | "children": [], | ||
965 | "code": null, | ||
966 | "level": "help", | ||
967 | "message": "return the expression directly", | ||
968 | "rendered": null, | ||
969 | "spans": [ | ||
970 | { | ||
971 | "byte_end": 55, | ||
972 | "byte_start": 29, | ||
973 | "column_end": 31, | ||
974 | "column_start": 5, | ||
975 | "expansion": null, | ||
976 | "file_name": "src/main.rs", | ||
977 | "is_primary": true, | ||
978 | "label": null, | ||
979 | "line_end": 3, | ||
980 | "line_start": 3, | ||
981 | "suggested_replacement": "", | ||
982 | "suggestion_applicability": "MachineApplicable", | ||
983 | "text": [ | ||
984 | { | ||
985 | "highlight_end": 31, | ||
986 | "highlight_start": 5, | ||
987 | "text": " let a = (0..10).collect();" | ||
988 | } | ||
989 | ] | ||
990 | }, | ||
991 | { | ||
992 | "byte_end": 61, | ||
993 | "byte_start": 60, | ||
994 | "column_end": 6, | ||
995 | "column_start": 5, | ||
996 | "expansion": null, | ||
997 | "file_name": "src/main.rs", | ||
998 | "is_primary": true, | ||
999 | "label": null, | ||
1000 | "line_end": 4, | ||
1001 | "line_start": 4, | ||
1002 | "suggested_replacement": "(0..10).collect()", | ||
1003 | "suggestion_applicability": "MachineApplicable", | ||
1004 | "text": [ | ||
1005 | { | ||
1006 | "highlight_end": 6, | ||
1007 | "highlight_start": 5, | ||
1008 | "text": " a" | ||
1009 | } | ||
1010 | ] | ||
1011 | } | ||
1012 | ] | ||
1013 | } | ||
1014 | ], | ||
1015 | "code": { | ||
1016 | "code": "clippy::let_and_return", | ||
1017 | "explanation": null | ||
1018 | }, | ||
1019 | "level": "warning", | ||
1020 | "message": "returning the result of a let binding from a block", | ||
1021 | "spans": [ | ||
1022 | { | ||
1023 | "byte_end": 55, | ||
1024 | "byte_start": 29, | ||
1025 | "column_end": 31, | ||
1026 | "column_start": 5, | ||
1027 | "expansion": null, | ||
1028 | "file_name": "src/main.rs", | ||
1029 | "is_primary": false, | ||
1030 | "label": "unnecessary let binding", | ||
1031 | "line_end": 3, | ||
1032 | "line_start": 3, | ||
1033 | "suggested_replacement": null, | ||
1034 | "suggestion_applicability": null, | ||
1035 | "text": [ | ||
1036 | { | ||
1037 | "highlight_end": 31, | ||
1038 | "highlight_start": 5, | ||
1039 | "text": " let a = (0..10).collect();" | ||
1040 | } | ||
1041 | ] | ||
1042 | }, | ||
1043 | { | ||
1044 | "byte_end": 61, | ||
1045 | "byte_start": 60, | ||
1046 | "column_end": 6, | ||
1047 | "column_start": 5, | ||
1048 | "expansion": null, | ||
1049 | "file_name": "src/main.rs", | ||
1050 | "is_primary": true, | ||
1051 | "label": null, | ||
1052 | "line_end": 4, | ||
1053 | "line_start": 4, | ||
1054 | "suggested_replacement": null, | ||
1055 | "suggestion_applicability": null, | ||
1056 | "text": [ | ||
1057 | { | ||
1058 | "highlight_end": 6, | ||
1059 | "highlight_start": 5, | ||
1060 | "text": " a" | ||
1061 | } | ||
1062 | ] | ||
1063 | } | ||
1064 | ] | ||
1065 | } | ||
1066 | "##, | ||
1067 | ); | ||
1068 | |||
1069 | let workspace_root = PathBuf::from("/test/"); | ||
1070 | let diag = map_rust_diagnostic_to_lsp(&diag, &workspace_root); | ||
1071 | insta::assert_debug_snapshot!(diag); | ||
1072 | } | ||
diff --git a/crates/ra_flycheck/src/lib.rs b/crates/ra_flycheck/src/lib.rs index d5efb6ab3..178485c9e 100644 --- a/crates/ra_flycheck/src/lib.rs +++ b/crates/ra_flycheck/src/lib.rs | |||
@@ -1,10 +1,9 @@ | |||
1 | //! cargo_check provides the functionality needed to run `cargo check` or | 1 | //! cargo_check provides the functionality needed to run `cargo check` or |
2 | //! another compatible command (f.x. clippy) in a background thread and provide | 2 | //! another compatible command (f.x. clippy) in a background thread and provide |
3 | //! LSP diagnostics based on the output of the command. | 3 | //! LSP diagnostics based on the output of the command. |
4 | mod conv; | ||
5 | 4 | ||
6 | use std::{ | 5 | use std::{ |
7 | io::{self, BufRead, BufReader}, | 6 | io::{self, BufReader}, |
8 | path::PathBuf, | 7 | path::PathBuf, |
9 | process::{Command, Stdio}, | 8 | process::{Command, Stdio}, |
10 | time::Instant, | 9 | time::Instant, |
@@ -12,14 +11,6 @@ use std::{ | |||
12 | 11 | ||
13 | use cargo_metadata::Message; | 12 | use cargo_metadata::Message; |
14 | use crossbeam_channel::{never, select, unbounded, Receiver, RecvError, Sender}; | 13 | use crossbeam_channel::{never, select, unbounded, Receiver, RecvError, Sender}; |
15 | use lsp_types::{ | ||
16 | CodeAction, CodeActionOrCommand, Diagnostic, Url, WorkDoneProgress, WorkDoneProgressBegin, | ||
17 | WorkDoneProgressEnd, WorkDoneProgressReport, | ||
18 | }; | ||
19 | |||
20 | use crate::conv::{map_rust_diagnostic_to_lsp, MappedRustDiagnostic}; | ||
21 | |||
22 | pub use crate::conv::url_from_path_with_drive_lowercasing; | ||
23 | 14 | ||
24 | #[derive(Clone, Debug, PartialEq, Eq)] | 15 | #[derive(Clone, Debug, PartialEq, Eq)] |
25 | pub enum FlycheckConfig { | 16 | pub enum FlycheckConfig { |
@@ -61,10 +52,17 @@ pub enum CheckTask { | |||
61 | ClearDiagnostics, | 52 | ClearDiagnostics, |
62 | 53 | ||
63 | /// Request adding a diagnostic with fixes included to a file | 54 | /// Request adding a diagnostic with fixes included to a file |
64 | AddDiagnostic { url: Url, diagnostic: Diagnostic, fixes: Vec<CodeActionOrCommand> }, | 55 | AddDiagnostic { workspace_root: PathBuf, diagnostic: cargo_metadata::diagnostic::Diagnostic }, |
65 | 56 | ||
66 | /// Request check progress notification to client | 57 | /// Request check progress notification to client |
67 | Status(WorkDoneProgress), | 58 | Status(Status), |
59 | } | ||
60 | |||
61 | #[derive(Debug)] | ||
62 | pub enum Status { | ||
63 | Being, | ||
64 | Progress(String), | ||
65 | End, | ||
68 | } | 66 | } |
69 | 67 | ||
70 | pub enum CheckCommand { | 68 | pub enum CheckCommand { |
@@ -131,9 +129,7 @@ impl FlycheckThread { | |||
131 | 129 | ||
132 | fn clean_previous_results(&self, task_send: &Sender<CheckTask>) { | 130 | fn clean_previous_results(&self, task_send: &Sender<CheckTask>) { |
133 | task_send.send(CheckTask::ClearDiagnostics).unwrap(); | 131 | task_send.send(CheckTask::ClearDiagnostics).unwrap(); |
134 | task_send | 132 | task_send.send(CheckTask::Status(Status::End)).unwrap(); |
135 | .send(CheckTask::Status(WorkDoneProgress::End(WorkDoneProgressEnd { message: None }))) | ||
136 | .unwrap(); | ||
137 | } | 133 | } |
138 | 134 | ||
139 | fn should_recheck(&mut self) -> bool { | 135 | fn should_recheck(&mut self) -> bool { |
@@ -155,52 +151,24 @@ impl FlycheckThread { | |||
155 | fn handle_message(&self, msg: CheckEvent, task_send: &Sender<CheckTask>) { | 151 | fn handle_message(&self, msg: CheckEvent, task_send: &Sender<CheckTask>) { |
156 | match msg { | 152 | match msg { |
157 | CheckEvent::Begin => { | 153 | CheckEvent::Begin => { |
158 | task_send | 154 | task_send.send(CheckTask::Status(Status::Being)).unwrap(); |
159 | .send(CheckTask::Status(WorkDoneProgress::Begin(WorkDoneProgressBegin { | ||
160 | title: "Running `cargo check`".to_string(), | ||
161 | cancellable: Some(false), | ||
162 | message: None, | ||
163 | percentage: None, | ||
164 | }))) | ||
165 | .unwrap(); | ||
166 | } | 155 | } |
167 | 156 | ||
168 | CheckEvent::End => { | 157 | CheckEvent::End => { |
169 | task_send | 158 | task_send.send(CheckTask::Status(Status::End)).unwrap(); |
170 | .send(CheckTask::Status(WorkDoneProgress::End(WorkDoneProgressEnd { | ||
171 | message: None, | ||
172 | }))) | ||
173 | .unwrap(); | ||
174 | } | 159 | } |
175 | 160 | ||
176 | CheckEvent::Msg(Message::CompilerArtifact(msg)) => { | 161 | CheckEvent::Msg(Message::CompilerArtifact(msg)) => { |
177 | task_send | 162 | task_send.send(CheckTask::Status(Status::Progress(msg.target.name))).unwrap(); |
178 | .send(CheckTask::Status(WorkDoneProgress::Report(WorkDoneProgressReport { | ||
179 | cancellable: Some(false), | ||
180 | message: Some(msg.target.name), | ||
181 | percentage: None, | ||
182 | }))) | ||
183 | .unwrap(); | ||
184 | } | 163 | } |
185 | 164 | ||
186 | CheckEvent::Msg(Message::CompilerMessage(msg)) => { | 165 | CheckEvent::Msg(Message::CompilerMessage(msg)) => { |
187 | let map_result = map_rust_diagnostic_to_lsp(&msg.message, &self.workspace_root); | 166 | task_send |
188 | if map_result.is_empty() { | 167 | .send(CheckTask::AddDiagnostic { |
189 | return; | 168 | workspace_root: self.workspace_root.clone(), |
190 | } | 169 | diagnostic: msg.message, |
191 | 170 | }) | |
192 | for MappedRustDiagnostic { location, diagnostic, fixes } in map_result { | 171 | .unwrap(); |
193 | let fixes = fixes | ||
194 | .into_iter() | ||
195 | .map(|fix| { | ||
196 | CodeAction { diagnostics: Some(vec![diagnostic.clone()]), ..fix }.into() | ||
197 | }) | ||
198 | .collect(); | ||
199 | |||
200 | task_send | ||
201 | .send(CheckTask::AddDiagnostic { url: location.uri, diagnostic, fixes }) | ||
202 | .unwrap(); | ||
203 | } | ||
204 | } | 172 | } |
205 | 173 | ||
206 | CheckEvent::Msg(Message::BuildScriptExecuted(_msg)) => {} | 174 | CheckEvent::Msg(Message::BuildScriptExecuted(_msg)) => {} |
@@ -271,11 +239,11 @@ impl FlycheckThread { | |||
271 | } | 239 | } |
272 | } | 240 | } |
273 | 241 | ||
274 | #[derive(Debug)] | 242 | // #[derive(Debug)] |
275 | pub struct DiagnosticWithFixes { | 243 | // pub struct DiagnosticWithFixes { |
276 | diagnostic: Diagnostic, | 244 | // diagnostic: Diagnostic, |
277 | fixes: Vec<CodeAction>, | 245 | // fixes: Vec<CodeAction>, |
278 | } | 246 | // } |
279 | 247 | ||
280 | enum CheckEvent { | 248 | enum CheckEvent { |
281 | Begin, | 249 | Begin, |
@@ -300,15 +268,11 @@ fn run_cargo( | |||
300 | // erroneus output. | 268 | // erroneus output. |
301 | let stdout = BufReader::new(child.stdout.take().unwrap()); | 269 | let stdout = BufReader::new(child.stdout.take().unwrap()); |
302 | let mut read_at_least_one_message = false; | 270 | let mut read_at_least_one_message = false; |
303 | 271 | for message in cargo_metadata::Message::parse_stream(stdout) { | |
304 | for line in stdout.lines() { | ||
305 | let line = line?; | ||
306 | |||
307 | let message = serde_json::from_str::<cargo_metadata::Message>(&line); | ||
308 | let message = match message { | 272 | let message = match message { |
309 | Ok(message) => message, | 273 | Ok(message) => message, |
310 | Err(err) => { | 274 | Err(err) => { |
311 | log::error!("Invalid json from cargo check, ignoring ({}): {:?} ", err, line); | 275 | log::error!("Invalid json from cargo check, ignoring ({})", err); |
312 | continue; | 276 | continue; |
313 | } | 277 | } |
314 | }; | 278 | }; |
diff --git a/crates/rust-analyzer/Cargo.toml b/crates/rust-analyzer/Cargo.toml index 8c94f430a..21e705597 100644 --- a/crates/rust-analyzer/Cargo.toml +++ b/crates/rust-analyzer/Cargo.toml | |||
@@ -29,6 +29,7 @@ rustc-hash = "1.1.0" | |||
29 | serde = { version = "1.0.106", features = ["derive"] } | 29 | serde = { version = "1.0.106", features = ["derive"] } |
30 | serde_json = "1.0.48" | 30 | serde_json = "1.0.48" |
31 | threadpool = "1.7.1" | 31 | threadpool = "1.7.1" |
32 | cargo_metadata = "0.10.0" | ||
32 | 33 | ||
33 | stdx = { path = "../stdx" } | 34 | stdx = { path = "../stdx" } |
34 | 35 | ||
@@ -53,6 +54,7 @@ winapi = "0.3.8" | |||
53 | 54 | ||
54 | [dev-dependencies] | 55 | [dev-dependencies] |
55 | tempfile = "3.1.0" | 56 | tempfile = "3.1.0" |
57 | insta = "0.16.0" | ||
56 | test_utils = { path = "../test_utils" } | 58 | test_utils = { path = "../test_utils" } |
57 | 59 | ||
58 | [features] | 60 | [features] |
diff --git a/crates/rust-analyzer/src/diagnostics.rs b/crates/rust-analyzer/src/diagnostics.rs index e7924f0a3..4bdd45a7d 100644 --- a/crates/rust-analyzer/src/diagnostics.rs +++ b/crates/rust-analyzer/src/diagnostics.rs | |||
@@ -1,4 +1,5 @@ | |||
1 | //! Book keeping for keeping diagnostics easily in sync with the client. | 1 | //! Book keeping for keeping diagnostics easily in sync with the client. |
2 | pub(crate) mod to_proto; | ||
2 | 3 | ||
3 | use std::{collections::HashMap, sync::Arc}; | 4 | use std::{collections::HashMap, sync::Arc}; |
4 | 5 | ||
diff --git a/crates/ra_flycheck/src/conv/snapshots/ra_flycheck__conv__test__snap_clippy_pass_by_ref.snap b/crates/rust-analyzer/src/diagnostics/snapshots/rust_analyzer__diagnostics__to_proto__tests__snap_clippy_pass_by_ref.snap index 4c9db0385..d7f9ec049 100644 --- a/crates/ra_flycheck/src/conv/snapshots/ra_flycheck__conv__test__snap_clippy_pass_by_ref.snap +++ b/crates/rust-analyzer/src/diagnostics/snapshots/rust_analyzer__diagnostics__to_proto__tests__snap_clippy_pass_by_ref.snap | |||
@@ -1,5 +1,5 @@ | |||
1 | --- | 1 | --- |
2 | source: crates/ra_flycheck/src/conv/test.rs | 2 | source: crates/rust-analyzer/src/diagnostics/to_proto.rs |
3 | expression: diag | 3 | expression: diag |
4 | --- | 4 | --- |
5 | [ | 5 | [ |
diff --git a/crates/ra_flycheck/src/conv/snapshots/ra_flycheck__conv__test__snap_handles_macro_location.snap b/crates/rust-analyzer/src/diagnostics/snapshots/rust_analyzer__diagnostics__to_proto__tests__snap_handles_macro_location.snap index 7cde4d867..a59faf254 100644 --- a/crates/ra_flycheck/src/conv/snapshots/ra_flycheck__conv__test__snap_handles_macro_location.snap +++ b/crates/rust-analyzer/src/diagnostics/snapshots/rust_analyzer__diagnostics__to_proto__tests__snap_handles_macro_location.snap | |||
@@ -1,5 +1,5 @@ | |||
1 | --- | 1 | --- |
2 | source: crates/ra_flycheck/src/conv/test.rs | 2 | source: crates/rust-analyzer/src/diagnostics/to_proto.rs |
3 | expression: diag | 3 | expression: diag |
4 | --- | 4 | --- |
5 | [ | 5 | [ |
diff --git a/crates/ra_flycheck/src/conv/snapshots/ra_flycheck__conv__test__snap_macro_compiler_error.snap b/crates/rust-analyzer/src/diagnostics/snapshots/rust_analyzer__diagnostics__to_proto__tests__snap_macro_compiler_error.snap index 1cc37e087..3c78e7f36 100644 --- a/crates/ra_flycheck/src/conv/snapshots/ra_flycheck__conv__test__snap_macro_compiler_error.snap +++ b/crates/rust-analyzer/src/diagnostics/snapshots/rust_analyzer__diagnostics__to_proto__tests__snap_macro_compiler_error.snap | |||
@@ -1,5 +1,5 @@ | |||
1 | --- | 1 | --- |
2 | source: crates/ra_flycheck/src/conv/test.rs | 2 | source: crates/rust-analyzer/src/diagnostics/to_proto.rs |
3 | expression: diag | 3 | expression: diag |
4 | --- | 4 | --- |
5 | [ | 5 | [ |
diff --git a/crates/ra_flycheck/src/conv/snapshots/ra_flycheck__conv__test__snap_multi_line_fix.snap b/crates/rust-analyzer/src/diagnostics/snapshots/rust_analyzer__diagnostics__to_proto__tests__snap_multi_line_fix.snap index 615ed8378..076b3cf27 100644 --- a/crates/ra_flycheck/src/conv/snapshots/ra_flycheck__conv__test__snap_multi_line_fix.snap +++ b/crates/rust-analyzer/src/diagnostics/snapshots/rust_analyzer__diagnostics__to_proto__tests__snap_multi_line_fix.snap | |||
@@ -1,5 +1,5 @@ | |||
1 | --- | 1 | --- |
2 | source: crates/ra_flycheck/src/conv/test.rs | 2 | source: crates/rust-analyzer/src/diagnostics/to_proto.rs |
3 | expression: diag | 3 | expression: diag |
4 | --- | 4 | --- |
5 | [ | 5 | [ |
diff --git a/crates/ra_flycheck/src/conv/snapshots/ra_flycheck__conv__test__snap_rustc_incompatible_type_for_trait.snap b/crates/rust-analyzer/src/diagnostics/snapshots/rust_analyzer__diagnostics__to_proto__tests__snap_rustc_incompatible_type_for_trait.snap index 0df0fce18..46d0c56d2 100644 --- a/crates/ra_flycheck/src/conv/snapshots/ra_flycheck__conv__test__snap_rustc_incompatible_type_for_trait.snap +++ b/crates/rust-analyzer/src/diagnostics/snapshots/rust_analyzer__diagnostics__to_proto__tests__snap_rustc_incompatible_type_for_trait.snap | |||
@@ -1,5 +1,5 @@ | |||
1 | --- | 1 | --- |
2 | source: crates/ra_flycheck/src/conv/test.rs | 2 | source: crates/rust-analyzer/src/diagnostics/to_proto.rs |
3 | expression: diag | 3 | expression: diag |
4 | --- | 4 | --- |
5 | [ | 5 | [ |
diff --git a/crates/ra_flycheck/src/conv/snapshots/ra_flycheck__conv__test__snap_rustc_mismatched_type.snap b/crates/rust-analyzer/src/diagnostics/snapshots/rust_analyzer__diagnostics__to_proto__tests__snap_rustc_mismatched_type.snap index 28ebcb3b3..4182929ba 100644 --- a/crates/ra_flycheck/src/conv/snapshots/ra_flycheck__conv__test__snap_rustc_mismatched_type.snap +++ b/crates/rust-analyzer/src/diagnostics/snapshots/rust_analyzer__diagnostics__to_proto__tests__snap_rustc_mismatched_type.snap | |||
@@ -1,5 +1,5 @@ | |||
1 | --- | 1 | --- |
2 | source: crates/ra_flycheck/src/conv/test.rs | 2 | source: crates/rust-analyzer/src/diagnostics/to_proto.rs |
3 | expression: diag | 3 | expression: diag |
4 | --- | 4 | --- |
5 | [ | 5 | [ |
diff --git a/crates/ra_flycheck/src/conv/snapshots/ra_flycheck__conv__test__snap_rustc_unused_variable.snap b/crates/rust-analyzer/src/diagnostics/snapshots/rust_analyzer__diagnostics__to_proto__tests__snap_rustc_unused_variable.snap index 5e0873281..69138c15b 100644 --- a/crates/ra_flycheck/src/conv/snapshots/ra_flycheck__conv__test__snap_rustc_unused_variable.snap +++ b/crates/rust-analyzer/src/diagnostics/snapshots/rust_analyzer__diagnostics__to_proto__tests__snap_rustc_unused_variable.snap | |||
@@ -1,5 +1,5 @@ | |||
1 | --- | 1 | --- |
2 | source: crates/ra_flycheck/src/conv/test.rs | 2 | source: crates/rust-analyzer/src/diagnostics/to_proto.rs |
3 | expression: diag | 3 | expression: diag |
4 | --- | 4 | --- |
5 | [ | 5 | [ |
diff --git a/crates/ra_flycheck/src/conv/snapshots/ra_flycheck__conv__test__snap_rustc_wrong_number_of_parameters.snap b/crates/rust-analyzer/src/diagnostics/snapshots/rust_analyzer__diagnostics__to_proto__tests__snap_rustc_wrong_number_of_parameters.snap index e500d3cd6..f6ab05004 100644 --- a/crates/ra_flycheck/src/conv/snapshots/ra_flycheck__conv__test__snap_rustc_wrong_number_of_parameters.snap +++ b/crates/rust-analyzer/src/diagnostics/snapshots/rust_analyzer__diagnostics__to_proto__tests__snap_rustc_wrong_number_of_parameters.snap | |||
@@ -1,5 +1,5 @@ | |||
1 | --- | 1 | --- |
2 | source: crates/ra_flycheck/src/conv/test.rs | 2 | source: crates/rust-analyzer/src/diagnostics/to_proto.rs |
3 | expression: diag | 3 | expression: diag |
4 | --- | 4 | --- |
5 | [ | 5 | [ |
diff --git a/crates/rust-analyzer/src/diagnostics/to_proto.rs b/crates/rust-analyzer/src/diagnostics/to_proto.rs new file mode 100644 index 000000000..0c8108591 --- /dev/null +++ b/crates/rust-analyzer/src/diagnostics/to_proto.rs | |||
@@ -0,0 +1,1408 @@ | |||
1 | //! This module provides the functionality needed to convert diagnostics from | ||
2 | //! `cargo check` json format to the LSP diagnostic format. | ||
3 | use cargo_metadata::diagnostic::{ | ||
4 | Applicability, Diagnostic as RustDiagnostic, DiagnosticLevel, DiagnosticSpan, | ||
5 | DiagnosticSpanMacroExpansion, | ||
6 | }; | ||
7 | use lsp_types::{ | ||
8 | CodeAction, Diagnostic, DiagnosticRelatedInformation, DiagnosticSeverity, DiagnosticTag, | ||
9 | Location, NumberOrString, Position, Range, TextEdit, Url, WorkspaceEdit, | ||
10 | }; | ||
11 | use std::{ | ||
12 | collections::HashMap, | ||
13 | fmt::Write, | ||
14 | path::{Component, Path, PathBuf, Prefix}, | ||
15 | str::FromStr, | ||
16 | }; | ||
17 | |||
18 | /// Converts a Rust level string to a LSP severity | ||
19 | fn map_level_to_severity(val: DiagnosticLevel) -> Option<DiagnosticSeverity> { | ||
20 | match val { | ||
21 | DiagnosticLevel::Ice => Some(DiagnosticSeverity::Error), | ||
22 | DiagnosticLevel::Error => Some(DiagnosticSeverity::Error), | ||
23 | DiagnosticLevel::Warning => Some(DiagnosticSeverity::Warning), | ||
24 | DiagnosticLevel::Note => Some(DiagnosticSeverity::Information), | ||
25 | DiagnosticLevel::Help => Some(DiagnosticSeverity::Hint), | ||
26 | DiagnosticLevel::Unknown => None, | ||
27 | } | ||
28 | } | ||
29 | |||
30 | /// Check whether a file name is from macro invocation | ||
31 | fn is_from_macro(file_name: &str) -> bool { | ||
32 | file_name.starts_with('<') && file_name.ends_with('>') | ||
33 | } | ||
34 | |||
35 | /// Converts a Rust macro span to a LSP location recursively | ||
36 | fn map_macro_span_to_location( | ||
37 | span_macro: &DiagnosticSpanMacroExpansion, | ||
38 | workspace_root: &PathBuf, | ||
39 | ) -> Option<Location> { | ||
40 | if !is_from_macro(&span_macro.span.file_name) { | ||
41 | return Some(map_span_to_location(&span_macro.span, workspace_root)); | ||
42 | } | ||
43 | |||
44 | if let Some(expansion) = &span_macro.span.expansion { | ||
45 | return map_macro_span_to_location(&expansion, workspace_root); | ||
46 | } | ||
47 | |||
48 | None | ||
49 | } | ||
50 | |||
51 | /// Converts a Rust span to a LSP location, resolving macro expansion site if neccesary | ||
52 | fn map_span_to_location(span: &DiagnosticSpan, workspace_root: &PathBuf) -> Location { | ||
53 | if span.expansion.is_some() { | ||
54 | let expansion = span.expansion.as_ref().unwrap(); | ||
55 | if let Some(macro_range) = map_macro_span_to_location(&expansion, workspace_root) { | ||
56 | return macro_range; | ||
57 | } | ||
58 | } | ||
59 | |||
60 | map_span_to_location_naive(span, workspace_root) | ||
61 | } | ||
62 | |||
63 | /// Converts a Rust span to a LSP location | ||
64 | fn map_span_to_location_naive(span: &DiagnosticSpan, workspace_root: &PathBuf) -> Location { | ||
65 | let mut file_name = workspace_root.clone(); | ||
66 | file_name.push(&span.file_name); | ||
67 | let uri = url_from_path_with_drive_lowercasing(file_name).unwrap(); | ||
68 | |||
69 | let range = Range::new( | ||
70 | Position::new(span.line_start as u64 - 1, span.column_start as u64 - 1), | ||
71 | Position::new(span.line_end as u64 - 1, span.column_end as u64 - 1), | ||
72 | ); | ||
73 | |||
74 | Location { uri, range } | ||
75 | } | ||
76 | |||
77 | /// Converts a secondary Rust span to a LSP related information | ||
78 | /// | ||
79 | /// If the span is unlabelled this will return `None`. | ||
80 | fn map_secondary_span_to_related( | ||
81 | span: &DiagnosticSpan, | ||
82 | workspace_root: &PathBuf, | ||
83 | ) -> Option<DiagnosticRelatedInformation> { | ||
84 | if let Some(label) = &span.label { | ||
85 | let location = map_span_to_location(span, workspace_root); | ||
86 | Some(DiagnosticRelatedInformation { location, message: label.clone() }) | ||
87 | } else { | ||
88 | // Nothing to label this with | ||
89 | None | ||
90 | } | ||
91 | } | ||
92 | |||
93 | /// Determines if diagnostic is related to unused code | ||
94 | fn is_unused_or_unnecessary(rd: &RustDiagnostic) -> bool { | ||
95 | if let Some(code) = &rd.code { | ||
96 | match code.code.as_str() { | ||
97 | "dead_code" | "unknown_lints" | "unreachable_code" | "unused_attributes" | ||
98 | | "unused_imports" | "unused_macros" | "unused_variables" => true, | ||
99 | _ => false, | ||
100 | } | ||
101 | } else { | ||
102 | false | ||
103 | } | ||
104 | } | ||
105 | |||
106 | /// Determines if diagnostic is related to deprecated code | ||
107 | fn is_deprecated(rd: &RustDiagnostic) -> bool { | ||
108 | if let Some(code) = &rd.code { | ||
109 | match code.code.as_str() { | ||
110 | "deprecated" => true, | ||
111 | _ => false, | ||
112 | } | ||
113 | } else { | ||
114 | false | ||
115 | } | ||
116 | } | ||
117 | |||
118 | enum MappedRustChildDiagnostic { | ||
119 | Related(DiagnosticRelatedInformation), | ||
120 | SuggestedFix(CodeAction), | ||
121 | MessageLine(String), | ||
122 | } | ||
123 | |||
124 | fn map_rust_child_diagnostic( | ||
125 | rd: &RustDiagnostic, | ||
126 | workspace_root: &PathBuf, | ||
127 | ) -> MappedRustChildDiagnostic { | ||
128 | let spans: Vec<&DiagnosticSpan> = rd.spans.iter().filter(|s| s.is_primary).collect(); | ||
129 | if spans.is_empty() { | ||
130 | // `rustc` uses these spanless children as a way to print multi-line | ||
131 | // messages | ||
132 | return MappedRustChildDiagnostic::MessageLine(rd.message.clone()); | ||
133 | } | ||
134 | |||
135 | let mut edit_map: HashMap<Url, Vec<TextEdit>> = HashMap::new(); | ||
136 | for &span in &spans { | ||
137 | match (&span.suggestion_applicability, &span.suggested_replacement) { | ||
138 | (Some(Applicability::MachineApplicable), Some(suggested_replacement)) => { | ||
139 | let location = map_span_to_location(span, workspace_root); | ||
140 | let edit = TextEdit::new(location.range, suggested_replacement.clone()); | ||
141 | edit_map.entry(location.uri).or_default().push(edit); | ||
142 | } | ||
143 | _ => {} | ||
144 | } | ||
145 | } | ||
146 | |||
147 | if !edit_map.is_empty() { | ||
148 | MappedRustChildDiagnostic::SuggestedFix(CodeAction { | ||
149 | title: rd.message.clone(), | ||
150 | kind: Some("quickfix".to_string()), | ||
151 | diagnostics: None, | ||
152 | edit: Some(WorkspaceEdit::new(edit_map)), | ||
153 | command: None, | ||
154 | is_preferred: None, | ||
155 | }) | ||
156 | } else { | ||
157 | MappedRustChildDiagnostic::Related(DiagnosticRelatedInformation { | ||
158 | location: map_span_to_location(spans[0], workspace_root), | ||
159 | message: rd.message.clone(), | ||
160 | }) | ||
161 | } | ||
162 | } | ||
163 | |||
164 | #[derive(Debug)] | ||
165 | pub(crate) struct MappedRustDiagnostic { | ||
166 | pub location: Location, | ||
167 | pub diagnostic: Diagnostic, | ||
168 | pub fixes: Vec<CodeAction>, | ||
169 | } | ||
170 | |||
171 | /// Converts a Rust root diagnostic to LSP form | ||
172 | /// | ||
173 | /// This flattens the Rust diagnostic by: | ||
174 | /// | ||
175 | /// 1. Creating a LSP diagnostic with the root message and primary span. | ||
176 | /// 2. Adding any labelled secondary spans to `relatedInformation` | ||
177 | /// 3. Categorising child diagnostics as either `SuggestedFix`es, | ||
178 | /// `relatedInformation` or additional message lines. | ||
179 | /// | ||
180 | /// If the diagnostic has no primary span this will return `None` | ||
181 | pub(crate) fn map_rust_diagnostic_to_lsp( | ||
182 | rd: &RustDiagnostic, | ||
183 | workspace_root: &PathBuf, | ||
184 | ) -> Vec<MappedRustDiagnostic> { | ||
185 | let primary_spans: Vec<&DiagnosticSpan> = rd.spans.iter().filter(|s| s.is_primary).collect(); | ||
186 | if primary_spans.is_empty() { | ||
187 | return vec![]; | ||
188 | } | ||
189 | |||
190 | let severity = map_level_to_severity(rd.level); | ||
191 | |||
192 | let mut source = String::from("rustc"); | ||
193 | let mut code = rd.code.as_ref().map(|c| c.code.clone()); | ||
194 | if let Some(code_val) = &code { | ||
195 | // See if this is an RFC #2103 scoped lint (e.g. from Clippy) | ||
196 | let scoped_code: Vec<&str> = code_val.split("::").collect(); | ||
197 | if scoped_code.len() == 2 { | ||
198 | source = String::from(scoped_code[0]); | ||
199 | code = Some(String::from(scoped_code[1])); | ||
200 | } | ||
201 | } | ||
202 | |||
203 | let mut needs_primary_span_label = true; | ||
204 | let mut related_information = vec![]; | ||
205 | let mut tags = vec![]; | ||
206 | |||
207 | for secondary_span in rd.spans.iter().filter(|s| !s.is_primary) { | ||
208 | let related = map_secondary_span_to_related(secondary_span, workspace_root); | ||
209 | if let Some(related) = related { | ||
210 | related_information.push(related); | ||
211 | } | ||
212 | } | ||
213 | |||
214 | let mut fixes = vec![]; | ||
215 | let mut message = rd.message.clone(); | ||
216 | for child in &rd.children { | ||
217 | let child = map_rust_child_diagnostic(&child, workspace_root); | ||
218 | match child { | ||
219 | MappedRustChildDiagnostic::Related(related) => related_information.push(related), | ||
220 | MappedRustChildDiagnostic::SuggestedFix(code_action) => fixes.push(code_action), | ||
221 | MappedRustChildDiagnostic::MessageLine(message_line) => { | ||
222 | write!(&mut message, "\n{}", message_line).unwrap(); | ||
223 | |||
224 | // These secondary messages usually duplicate the content of the | ||
225 | // primary span label. | ||
226 | needs_primary_span_label = false; | ||
227 | } | ||
228 | } | ||
229 | } | ||
230 | |||
231 | if is_unused_or_unnecessary(rd) { | ||
232 | tags.push(DiagnosticTag::Unnecessary); | ||
233 | } | ||
234 | |||
235 | if is_deprecated(rd) { | ||
236 | tags.push(DiagnosticTag::Deprecated); | ||
237 | } | ||
238 | |||
239 | primary_spans | ||
240 | .iter() | ||
241 | .map(|primary_span| { | ||
242 | let location = map_span_to_location(&primary_span, workspace_root); | ||
243 | |||
244 | let mut message = message.clone(); | ||
245 | if needs_primary_span_label { | ||
246 | if let Some(primary_span_label) = &primary_span.label { | ||
247 | write!(&mut message, "\n{}", primary_span_label).unwrap(); | ||
248 | } | ||
249 | } | ||
250 | |||
251 | // If error occurs from macro expansion, add related info pointing to | ||
252 | // where the error originated | ||
253 | if !is_from_macro(&primary_span.file_name) && primary_span.expansion.is_some() { | ||
254 | let def_loc = map_span_to_location_naive(&primary_span, workspace_root); | ||
255 | related_information.push(DiagnosticRelatedInformation { | ||
256 | location: def_loc, | ||
257 | message: "Error originated from macro here".to_string(), | ||
258 | }); | ||
259 | } | ||
260 | |||
261 | let diagnostic = Diagnostic { | ||
262 | range: location.range, | ||
263 | severity, | ||
264 | code: code.clone().map(NumberOrString::String), | ||
265 | source: Some(source.clone()), | ||
266 | message, | ||
267 | related_information: if !related_information.is_empty() { | ||
268 | Some(related_information.clone()) | ||
269 | } else { | ||
270 | None | ||
271 | }, | ||
272 | tags: if !tags.is_empty() { Some(tags.clone()) } else { None }, | ||
273 | }; | ||
274 | |||
275 | MappedRustDiagnostic { location, diagnostic, fixes: fixes.clone() } | ||
276 | }) | ||
277 | .collect() | ||
278 | } | ||
279 | |||
280 | /// Returns a `Url` object from a given path, will lowercase drive letters if present. | ||
281 | /// This will only happen when processing windows paths. | ||
282 | /// | ||
283 | /// When processing non-windows path, this is essentially the same as `Url::from_file_path`. | ||
284 | pub fn url_from_path_with_drive_lowercasing( | ||
285 | path: impl AsRef<Path>, | ||
286 | ) -> Result<Url, Box<dyn std::error::Error + Send + Sync>> { | ||
287 | let component_has_windows_drive = path.as_ref().components().any(|comp| { | ||
288 | if let Component::Prefix(c) = comp { | ||
289 | match c.kind() { | ||
290 | Prefix::Disk(_) | Prefix::VerbatimDisk(_) => return true, | ||
291 | _ => return false, | ||
292 | } | ||
293 | } | ||
294 | false | ||
295 | }); | ||
296 | |||
297 | // VSCode expects drive letters to be lowercased, where rust will uppercase the drive letters. | ||
298 | if component_has_windows_drive { | ||
299 | let url_original = Url::from_file_path(&path) | ||
300 | .map_err(|_| format!("can't convert path to url: {}", path.as_ref().display()))?; | ||
301 | |||
302 | let drive_partition: Vec<&str> = url_original.as_str().rsplitn(2, ':').collect(); | ||
303 | |||
304 | // There is a drive partition, but we never found a colon. | ||
305 | // This should not happen, but in this case we just pass it through. | ||
306 | if drive_partition.len() == 1 { | ||
307 | return Ok(url_original); | ||
308 | } | ||
309 | |||
310 | let joined = drive_partition[1].to_ascii_lowercase() + ":" + drive_partition[0]; | ||
311 | let url = Url::from_str(&joined).expect("This came from a valid `Url`"); | ||
312 | |||
313 | Ok(url) | ||
314 | } else { | ||
315 | Ok(Url::from_file_path(&path) | ||
316 | .map_err(|_| format!("can't convert path to url: {}", path.as_ref().display()))?) | ||
317 | } | ||
318 | } | ||
319 | |||
320 | #[cfg(test)] | ||
321 | mod tests { | ||
322 | use super::*; | ||
323 | |||
324 | // `Url` is not able to parse windows paths on unix machines. | ||
325 | #[test] | ||
326 | #[cfg(target_os = "windows")] | ||
327 | fn test_lowercase_drive_letter_with_drive() { | ||
328 | let url = url_from_path_with_drive_lowercasing("C:\\Test").unwrap(); | ||
329 | |||
330 | assert_eq!(url.to_string(), "file:///c:/Test"); | ||
331 | } | ||
332 | |||
333 | #[test] | ||
334 | #[cfg(target_os = "windows")] | ||
335 | fn test_drive_without_colon_passthrough() { | ||
336 | let url = url_from_path_with_drive_lowercasing(r#"\\localhost\C$\my_dir"#).unwrap(); | ||
337 | |||
338 | assert_eq!(url.to_string(), "file://localhost/C$/my_dir"); | ||
339 | } | ||
340 | |||
341 | #[cfg(not(windows))] | ||
342 | fn parse_diagnostic(val: &str) -> cargo_metadata::diagnostic::Diagnostic { | ||
343 | serde_json::from_str::<cargo_metadata::diagnostic::Diagnostic>(val).unwrap() | ||
344 | } | ||
345 | |||
346 | #[test] | ||
347 | #[cfg(not(windows))] | ||
348 | fn snap_rustc_incompatible_type_for_trait() { | ||
349 | let diag = parse_diagnostic( | ||
350 | r##"{ | ||
351 | "message": "method `next` has an incompatible type for trait", | ||
352 | "code": { | ||
353 | "code": "E0053", | ||
354 | "explanation": "\nThe parameters of any trait method must match between a trait implementation\nand the trait definition.\n\nHere are a couple examples of this error:\n\n```compile_fail,E0053\ntrait Foo {\n fn foo(x: u16);\n fn bar(&self);\n}\n\nstruct Bar;\n\nimpl Foo for Bar {\n // error, expected u16, found i16\n fn foo(x: i16) { }\n\n // error, types differ in mutability\n fn bar(&mut self) { }\n}\n```\n" | ||
355 | }, | ||
356 | "level": "error", | ||
357 | "spans": [ | ||
358 | { | ||
359 | "file_name": "compiler/ty/list_iter.rs", | ||
360 | "byte_start": 1307, | ||
361 | "byte_end": 1350, | ||
362 | "line_start": 52, | ||
363 | "line_end": 52, | ||
364 | "column_start": 5, | ||
365 | "column_end": 48, | ||
366 | "is_primary": true, | ||
367 | "text": [ | ||
368 | { | ||
369 | "text": " fn next(&self) -> Option<&'list ty::Ref<M>> {", | ||
370 | "highlight_start": 5, | ||
371 | "highlight_end": 48 | ||
372 | } | ||
373 | ], | ||
374 | "label": "types differ in mutability", | ||
375 | "suggested_replacement": null, | ||
376 | "suggestion_applicability": null, | ||
377 | "expansion": null | ||
378 | } | ||
379 | ], | ||
380 | "children": [ | ||
381 | { | ||
382 | "message": "expected type `fn(&mut ty::list_iter::ListIterator<'list, M>) -> std::option::Option<&ty::Ref<M>>`\n found type `fn(&ty::list_iter::ListIterator<'list, M>) -> std::option::Option<&'list ty::Ref<M>>`", | ||
383 | "code": null, | ||
384 | "level": "note", | ||
385 | "spans": [], | ||
386 | "children": [], | ||
387 | "rendered": null | ||
388 | } | ||
389 | ], | ||
390 | "rendered": "error[E0053]: method `next` has an incompatible type for trait\n --> compiler/ty/list_iter.rs:52:5\n |\n52 | fn next(&self) -> Option<&'list ty::Ref<M>> {\n | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ types differ in mutability\n |\n = note: expected type `fn(&mut ty::list_iter::ListIterator<'list, M>) -> std::option::Option<&ty::Ref<M>>`\n found type `fn(&ty::list_iter::ListIterator<'list, M>) -> std::option::Option<&'list ty::Ref<M>>`\n\n" | ||
391 | } | ||
392 | "##, | ||
393 | ); | ||
394 | |||
395 | let workspace_root = PathBuf::from("/test/"); | ||
396 | let diag = map_rust_diagnostic_to_lsp(&diag, &workspace_root); | ||
397 | insta::assert_debug_snapshot!(diag); | ||
398 | } | ||
399 | |||
400 | #[test] | ||
401 | #[cfg(not(windows))] | ||
402 | fn snap_rustc_unused_variable() { | ||
403 | let diag = parse_diagnostic( | ||
404 | r##"{ | ||
405 | "message": "unused variable: `foo`", | ||
406 | "code": { | ||
407 | "code": "unused_variables", | ||
408 | "explanation": null | ||
409 | }, | ||
410 | "level": "warning", | ||
411 | "spans": [ | ||
412 | { | ||
413 | "file_name": "driver/subcommand/repl.rs", | ||
414 | "byte_start": 9228, | ||
415 | "byte_end": 9231, | ||
416 | "line_start": 291, | ||
417 | "line_end": 291, | ||
418 | "column_start": 9, | ||
419 | "column_end": 12, | ||
420 | "is_primary": true, | ||
421 | "text": [ | ||
422 | { | ||
423 | "text": " let foo = 42;", | ||
424 | "highlight_start": 9, | ||
425 | "highlight_end": 12 | ||
426 | } | ||
427 | ], | ||
428 | "label": null, | ||
429 | "suggested_replacement": null, | ||
430 | "suggestion_applicability": null, | ||
431 | "expansion": null | ||
432 | } | ||
433 | ], | ||
434 | "children": [ | ||
435 | { | ||
436 | "message": "#[warn(unused_variables)] on by default", | ||
437 | "code": null, | ||
438 | "level": "note", | ||
439 | "spans": [], | ||
440 | "children": [], | ||
441 | "rendered": null | ||
442 | }, | ||
443 | { | ||
444 | "message": "consider prefixing with an underscore", | ||
445 | "code": null, | ||
446 | "level": "help", | ||
447 | "spans": [ | ||
448 | { | ||
449 | "file_name": "driver/subcommand/repl.rs", | ||
450 | "byte_start": 9228, | ||
451 | "byte_end": 9231, | ||
452 | "line_start": 291, | ||
453 | "line_end": 291, | ||
454 | "column_start": 9, | ||
455 | "column_end": 12, | ||
456 | "is_primary": true, | ||
457 | "text": [ | ||
458 | { | ||
459 | "text": " let foo = 42;", | ||
460 | "highlight_start": 9, | ||
461 | "highlight_end": 12 | ||
462 | } | ||
463 | ], | ||
464 | "label": null, | ||
465 | "suggested_replacement": "_foo", | ||
466 | "suggestion_applicability": "MachineApplicable", | ||
467 | "expansion": null | ||
468 | } | ||
469 | ], | ||
470 | "children": [], | ||
471 | "rendered": null | ||
472 | } | ||
473 | ], | ||
474 | "rendered": "warning: unused variable: `foo`\n --> driver/subcommand/repl.rs:291:9\n |\n291 | let foo = 42;\n | ^^^ help: consider prefixing with an underscore: `_foo`\n |\n = note: #[warn(unused_variables)] on by default\n\n" | ||
475 | }"##, | ||
476 | ); | ||
477 | |||
478 | let workspace_root = PathBuf::from("/test/"); | ||
479 | let diag = map_rust_diagnostic_to_lsp(&diag, &workspace_root); | ||
480 | insta::assert_debug_snapshot!(diag); | ||
481 | } | ||
482 | |||
483 | #[test] | ||
484 | #[cfg(not(windows))] | ||
485 | fn snap_rustc_wrong_number_of_parameters() { | ||
486 | let diag = parse_diagnostic( | ||
487 | r##"{ | ||
488 | "message": "this function takes 2 parameters but 3 parameters were supplied", | ||
489 | "code": { | ||
490 | "code": "E0061", | ||
491 | "explanation": "\nThe number of arguments passed to a function must match the number of arguments\nspecified in the function signature.\n\nFor example, a function like:\n\n```\nfn f(a: u16, b: &str) {}\n```\n\nMust always be called with exactly two arguments, e.g., `f(2, \"test\")`.\n\nNote that Rust does not have a notion of optional function arguments or\nvariadic functions (except for its C-FFI).\n" | ||
492 | }, | ||
493 | "level": "error", | ||
494 | "spans": [ | ||
495 | { | ||
496 | "file_name": "compiler/ty/select.rs", | ||
497 | "byte_start": 8787, | ||
498 | "byte_end": 9241, | ||
499 | "line_start": 219, | ||
500 | "line_end": 231, | ||
501 | "column_start": 5, | ||
502 | "column_end": 6, | ||
503 | "is_primary": false, | ||
504 | "text": [ | ||
505 | { | ||
506 | "text": " pub fn add_evidence(", | ||
507 | "highlight_start": 5, | ||
508 | "highlight_end": 25 | ||
509 | }, | ||
510 | { | ||
511 | "text": " &mut self,", | ||
512 | "highlight_start": 1, | ||
513 | "highlight_end": 19 | ||
514 | }, | ||
515 | { | ||
516 | "text": " target_poly: &ty::Ref<ty::Poly>,", | ||
517 | "highlight_start": 1, | ||
518 | "highlight_end": 41 | ||
519 | }, | ||
520 | { | ||
521 | "text": " evidence_poly: &ty::Ref<ty::Poly>,", | ||
522 | "highlight_start": 1, | ||
523 | "highlight_end": 43 | ||
524 | }, | ||
525 | { | ||
526 | "text": " ) {", | ||
527 | "highlight_start": 1, | ||
528 | "highlight_end": 8 | ||
529 | }, | ||
530 | { | ||
531 | "text": " match target_poly {", | ||
532 | "highlight_start": 1, | ||
533 | "highlight_end": 28 | ||
534 | }, | ||
535 | { | ||
536 | "text": " ty::Ref::Var(tvar, _) => self.add_var_evidence(tvar, evidence_poly),", | ||
537 | "highlight_start": 1, | ||
538 | "highlight_end": 81 | ||
539 | }, | ||
540 | { | ||
541 | "text": " ty::Ref::Fixed(target_ty) => {", | ||
542 | "highlight_start": 1, | ||
543 | "highlight_end": 43 | ||
544 | }, | ||
545 | { | ||
546 | "text": " let evidence_ty = evidence_poly.resolve_to_ty();", | ||
547 | "highlight_start": 1, | ||
548 | "highlight_end": 65 | ||
549 | }, | ||
550 | { | ||
551 | "text": " self.add_evidence_ty(target_ty, evidence_poly, evidence_ty)", | ||
552 | "highlight_start": 1, | ||
553 | "highlight_end": 76 | ||
554 | }, | ||
555 | { | ||
556 | "text": " }", | ||
557 | "highlight_start": 1, | ||
558 | "highlight_end": 14 | ||
559 | }, | ||
560 | { | ||
561 | "text": " }", | ||
562 | "highlight_start": 1, | ||
563 | "highlight_end": 10 | ||
564 | }, | ||
565 | { | ||
566 | "text": " }", | ||
567 | "highlight_start": 1, | ||
568 | "highlight_end": 6 | ||
569 | } | ||
570 | ], | ||
571 | "label": "defined here", | ||
572 | "suggested_replacement": null, | ||
573 | "suggestion_applicability": null, | ||
574 | "expansion": null | ||
575 | }, | ||
576 | { | ||
577 | "file_name": "compiler/ty/select.rs", | ||
578 | "byte_start": 4045, | ||
579 | "byte_end": 4057, | ||
580 | "line_start": 104, | ||
581 | "line_end": 104, | ||
582 | "column_start": 18, | ||
583 | "column_end": 30, | ||
584 | "is_primary": true, | ||
585 | "text": [ | ||
586 | { | ||
587 | "text": " self.add_evidence(target_fixed, evidence_fixed, false);", | ||
588 | "highlight_start": 18, | ||
589 | "highlight_end": 30 | ||
590 | } | ||
591 | ], | ||
592 | "label": "expected 2 parameters", | ||
593 | "suggested_replacement": null, | ||
594 | "suggestion_applicability": null, | ||
595 | "expansion": null | ||
596 | } | ||
597 | ], | ||
598 | "children": [], | ||
599 | "rendered": "error[E0061]: this function takes 2 parameters but 3 parameters were supplied\n --> compiler/ty/select.rs:104:18\n |\n104 | self.add_evidence(target_fixed, evidence_fixed, false);\n | ^^^^^^^^^^^^ expected 2 parameters\n...\n219 | / pub fn add_evidence(\n220 | | &mut self,\n221 | | target_poly: &ty::Ref<ty::Poly>,\n222 | | evidence_poly: &ty::Ref<ty::Poly>,\n... |\n230 | | }\n231 | | }\n | |_____- defined here\n\n" | ||
600 | }"##, | ||
601 | ); | ||
602 | |||
603 | let workspace_root = PathBuf::from("/test/"); | ||
604 | let diag = map_rust_diagnostic_to_lsp(&diag, &workspace_root); | ||
605 | insta::assert_debug_snapshot!(diag); | ||
606 | } | ||
607 | |||
608 | #[test] | ||
609 | #[cfg(not(windows))] | ||
610 | fn snap_clippy_pass_by_ref() { | ||
611 | let diag = parse_diagnostic( | ||
612 | r##"{ | ||
613 | "message": "this argument is passed by reference, but would be more efficient if passed by value", | ||
614 | "code": { | ||
615 | "code": "clippy::trivially_copy_pass_by_ref", | ||
616 | "explanation": null | ||
617 | }, | ||
618 | "level": "warning", | ||
619 | "spans": [ | ||
620 | { | ||
621 | "file_name": "compiler/mir/tagset.rs", | ||
622 | "byte_start": 941, | ||
623 | "byte_end": 946, | ||
624 | "line_start": 42, | ||
625 | "line_end": 42, | ||
626 | "column_start": 24, | ||
627 | "column_end": 29, | ||
628 | "is_primary": true, | ||
629 | "text": [ | ||
630 | { | ||
631 | "text": " pub fn is_disjoint(&self, other: Self) -> bool {", | ||
632 | "highlight_start": 24, | ||
633 | "highlight_end": 29 | ||
634 | } | ||
635 | ], | ||
636 | "label": null, | ||
637 | "suggested_replacement": null, | ||
638 | "suggestion_applicability": null, | ||
639 | "expansion": null | ||
640 | } | ||
641 | ], | ||
642 | "children": [ | ||
643 | { | ||
644 | "message": "lint level defined here", | ||
645 | "code": null, | ||
646 | "level": "note", | ||
647 | "spans": [ | ||
648 | { | ||
649 | "file_name": "compiler/lib.rs", | ||
650 | "byte_start": 8, | ||
651 | "byte_end": 19, | ||
652 | "line_start": 1, | ||
653 | "line_end": 1, | ||
654 | "column_start": 9, | ||
655 | "column_end": 20, | ||
656 | "is_primary": true, | ||
657 | "text": [ | ||
658 | { | ||
659 | "text": "#![warn(clippy::all)]", | ||
660 | "highlight_start": 9, | ||
661 | "highlight_end": 20 | ||
662 | } | ||
663 | ], | ||
664 | "label": null, | ||
665 | "suggested_replacement": null, | ||
666 | "suggestion_applicability": null, | ||
667 | "expansion": null | ||
668 | } | ||
669 | ], | ||
670 | "children": [], | ||
671 | "rendered": null | ||
672 | }, | ||
673 | { | ||
674 | "message": "#[warn(clippy::trivially_copy_pass_by_ref)] implied by #[warn(clippy::all)]", | ||
675 | "code": null, | ||
676 | "level": "note", | ||
677 | "spans": [], | ||
678 | "children": [], | ||
679 | "rendered": null | ||
680 | }, | ||
681 | { | ||
682 | "message": "for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#trivially_copy_pass_by_ref", | ||
683 | "code": null, | ||
684 | "level": "help", | ||
685 | "spans": [], | ||
686 | "children": [], | ||
687 | "rendered": null | ||
688 | }, | ||
689 | { | ||
690 | "message": "consider passing by value instead", | ||
691 | "code": null, | ||
692 | "level": "help", | ||
693 | "spans": [ | ||
694 | { | ||
695 | "file_name": "compiler/mir/tagset.rs", | ||
696 | "byte_start": 941, | ||
697 | "byte_end": 946, | ||
698 | "line_start": 42, | ||
699 | "line_end": 42, | ||
700 | "column_start": 24, | ||
701 | "column_end": 29, | ||
702 | "is_primary": true, | ||
703 | "text": [ | ||
704 | { | ||
705 | "text": " pub fn is_disjoint(&self, other: Self) -> bool {", | ||
706 | "highlight_start": 24, | ||
707 | "highlight_end": 29 | ||
708 | } | ||
709 | ], | ||
710 | "label": null, | ||
711 | "suggested_replacement": "self", | ||
712 | "suggestion_applicability": "Unspecified", | ||
713 | "expansion": null | ||
714 | } | ||
715 | ], | ||
716 | "children": [], | ||
717 | "rendered": null | ||
718 | } | ||
719 | ], | ||
720 | "rendered": "warning: this argument is passed by reference, but would be more efficient if passed by value\n --> compiler/mir/tagset.rs:42:24\n |\n42 | pub fn is_disjoint(&self, other: Self) -> bool {\n | ^^^^^ help: consider passing by value instead: `self`\n |\nnote: lint level defined here\n --> compiler/lib.rs:1:9\n |\n1 | #![warn(clippy::all)]\n | ^^^^^^^^^^^\n = note: #[warn(clippy::trivially_copy_pass_by_ref)] implied by #[warn(clippy::all)]\n = help: for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#trivially_copy_pass_by_ref\n\n" | ||
721 | }"##, | ||
722 | ); | ||
723 | |||
724 | let workspace_root = PathBuf::from("/test/"); | ||
725 | let diag = map_rust_diagnostic_to_lsp(&diag, &workspace_root); | ||
726 | insta::assert_debug_snapshot!(diag); | ||
727 | } | ||
728 | |||
729 | #[test] | ||
730 | #[cfg(not(windows))] | ||
731 | fn snap_rustc_mismatched_type() { | ||
732 | let diag = parse_diagnostic( | ||
733 | r##"{ | ||
734 | "message": "mismatched types", | ||
735 | "code": { | ||
736 | "code": "E0308", | ||
737 | "explanation": "\nThis error occurs when the compiler was unable to infer the concrete type of a\nvariable. It can occur for several cases, the most common of which is a\nmismatch in the expected type that the compiler inferred for a variable's\ninitializing expression, and the actual type explicitly assigned to the\nvariable.\n\nFor example:\n\n```compile_fail,E0308\nlet x: i32 = \"I am not a number!\";\n// ~~~ ~~~~~~~~~~~~~~~~~~~~\n// | |\n// | initializing expression;\n// | compiler infers type `&str`\n// |\n// type `i32` assigned to variable `x`\n```\n" | ||
738 | }, | ||
739 | "level": "error", | ||
740 | "spans": [ | ||
741 | { | ||
742 | "file_name": "runtime/compiler_support.rs", | ||
743 | "byte_start": 1589, | ||
744 | "byte_end": 1594, | ||
745 | "line_start": 48, | ||
746 | "line_end": 48, | ||
747 | "column_start": 65, | ||
748 | "column_end": 70, | ||
749 | "is_primary": true, | ||
750 | "text": [ | ||
751 | { | ||
752 | "text": " let layout = alloc::Layout::from_size_align_unchecked(size, align);", | ||
753 | "highlight_start": 65, | ||
754 | "highlight_end": 70 | ||
755 | } | ||
756 | ], | ||
757 | "label": "expected usize, found u32", | ||
758 | "suggested_replacement": null, | ||
759 | "suggestion_applicability": null, | ||
760 | "expansion": null | ||
761 | } | ||
762 | ], | ||
763 | "children": [], | ||
764 | "rendered": "error[E0308]: mismatched types\n --> runtime/compiler_support.rs:48:65\n |\n48 | let layout = alloc::Layout::from_size_align_unchecked(size, align);\n | ^^^^^ expected usize, found u32\n\n" | ||
765 | }"##, | ||
766 | ); | ||
767 | |||
768 | let workspace_root = PathBuf::from("/test/"); | ||
769 | let diag = map_rust_diagnostic_to_lsp(&diag, &workspace_root); | ||
770 | insta::assert_debug_snapshot!(diag); | ||
771 | } | ||
772 | |||
773 | #[test] | ||
774 | #[cfg(not(windows))] | ||
775 | fn snap_handles_macro_location() { | ||
776 | let diag = parse_diagnostic( | ||
777 | r##"{ | ||
778 | "rendered": "error[E0277]: can't compare `{integer}` with `&str`\n --> src/main.rs:2:5\n |\n2 | assert_eq!(1, \"love\");\n | ^^^^^^^^^^^^^^^^^^^^^^ no implementation for `{integer} == &str`\n |\n = help: the trait `std::cmp::PartialEq<&str>` is not implemented for `{integer}`\n = note: this error originates in a macro outside of the current crate (in Nightly builds, run with -Z external-macro-backtrace for more info)\n\n", | ||
779 | "children": [ | ||
780 | { | ||
781 | "children": [], | ||
782 | "code": null, | ||
783 | "level": "help", | ||
784 | "message": "the trait `std::cmp::PartialEq<&str>` is not implemented for `{integer}`", | ||
785 | "rendered": null, | ||
786 | "spans": [] | ||
787 | } | ||
788 | ], | ||
789 | "code": { | ||
790 | "code": "E0277", | ||
791 | "explanation": "\nYou tried to use a type which doesn't implement some trait in a place which\nexpected that trait. Erroneous code example:\n\n```compile_fail,E0277\n// here we declare the Foo trait with a bar method\ntrait Foo {\n fn bar(&self);\n}\n\n// we now declare a function which takes an object implementing the Foo trait\nfn some_func<T: Foo>(foo: T) {\n foo.bar();\n}\n\nfn main() {\n // we now call the method with the i32 type, which doesn't implement\n // the Foo trait\n some_func(5i32); // error: the trait bound `i32 : Foo` is not satisfied\n}\n```\n\nIn order to fix this error, verify that the type you're using does implement\nthe trait. Example:\n\n```\ntrait Foo {\n fn bar(&self);\n}\n\nfn some_func<T: Foo>(foo: T) {\n foo.bar(); // we can now use this method since i32 implements the\n // Foo trait\n}\n\n// we implement the trait on the i32 type\nimpl Foo for i32 {\n fn bar(&self) {}\n}\n\nfn main() {\n some_func(5i32); // ok!\n}\n```\n\nOr in a generic context, an erroneous code example would look like:\n\n```compile_fail,E0277\nfn some_func<T>(foo: T) {\n println!(\"{:?}\", foo); // error: the trait `core::fmt::Debug` is not\n // implemented for the type `T`\n}\n\nfn main() {\n // We now call the method with the i32 type,\n // which *does* implement the Debug trait.\n some_func(5i32);\n}\n```\n\nNote that the error here is in the definition of the generic function: Although\nwe only call it with a parameter that does implement `Debug`, the compiler\nstill rejects the function: It must work with all possible input types. In\norder to make this example compile, we need to restrict the generic type we're\naccepting:\n\n```\nuse std::fmt;\n\n// Restrict the input type to types that implement Debug.\nfn some_func<T: fmt::Debug>(foo: T) {\n println!(\"{:?}\", foo);\n}\n\nfn main() {\n // Calling the method is still fine, as i32 implements Debug.\n some_func(5i32);\n\n // This would fail to compile now:\n // struct WithoutDebug;\n // some_func(WithoutDebug);\n}\n```\n\nRust only looks at the signature of the called function, as such it must\nalready specify all requirements that will be used for every type parameter.\n" | ||
792 | }, | ||
793 | "level": "error", | ||
794 | "message": "can't compare `{integer}` with `&str`", | ||
795 | "spans": [ | ||
796 | { | ||
797 | "byte_end": 155, | ||
798 | "byte_start": 153, | ||
799 | "column_end": 33, | ||
800 | "column_start": 31, | ||
801 | "expansion": { | ||
802 | "def_site_span": { | ||
803 | "byte_end": 940, | ||
804 | "byte_start": 0, | ||
805 | "column_end": 6, | ||
806 | "column_start": 1, | ||
807 | "expansion": null, | ||
808 | "file_name": "<::core::macros::assert_eq macros>", | ||
809 | "is_primary": false, | ||
810 | "label": null, | ||
811 | "line_end": 36, | ||
812 | "line_start": 1, | ||
813 | "suggested_replacement": null, | ||
814 | "suggestion_applicability": null, | ||
815 | "text": [ | ||
816 | { | ||
817 | "highlight_end": 35, | ||
818 | "highlight_start": 1, | ||
819 | "text": "($ left : expr, $ right : expr) =>" | ||
820 | }, | ||
821 | { | ||
822 | "highlight_end": 3, | ||
823 | "highlight_start": 1, | ||
824 | "text": "({" | ||
825 | }, | ||
826 | { | ||
827 | "highlight_end": 33, | ||
828 | "highlight_start": 1, | ||
829 | "text": " match (& $ left, & $ right)" | ||
830 | }, | ||
831 | { | ||
832 | "highlight_end": 7, | ||
833 | "highlight_start": 1, | ||
834 | "text": " {" | ||
835 | }, | ||
836 | { | ||
837 | "highlight_end": 34, | ||
838 | "highlight_start": 1, | ||
839 | "text": " (left_val, right_val) =>" | ||
840 | }, | ||
841 | { | ||
842 | "highlight_end": 11, | ||
843 | "highlight_start": 1, | ||
844 | "text": " {" | ||
845 | }, | ||
846 | { | ||
847 | "highlight_end": 46, | ||
848 | "highlight_start": 1, | ||
849 | "text": " if ! (* left_val == * right_val)" | ||
850 | }, | ||
851 | { | ||
852 | "highlight_end": 15, | ||
853 | "highlight_start": 1, | ||
854 | "text": " {" | ||
855 | }, | ||
856 | { | ||
857 | "highlight_end": 25, | ||
858 | "highlight_start": 1, | ||
859 | "text": " panic !" | ||
860 | }, | ||
861 | { | ||
862 | "highlight_end": 57, | ||
863 | "highlight_start": 1, | ||
864 | "text": " (r#\"assertion failed: `(left == right)`" | ||
865 | }, | ||
866 | { | ||
867 | "highlight_end": 16, | ||
868 | "highlight_start": 1, | ||
869 | "text": " left: `{:?}`," | ||
870 | }, | ||
871 | { | ||
872 | "highlight_end": 18, | ||
873 | "highlight_start": 1, | ||
874 | "text": " right: `{:?}`\"#," | ||
875 | }, | ||
876 | { | ||
877 | "highlight_end": 47, | ||
878 | "highlight_start": 1, | ||
879 | "text": " & * left_val, & * right_val)" | ||
880 | }, | ||
881 | { | ||
882 | "highlight_end": 15, | ||
883 | "highlight_start": 1, | ||
884 | "text": " }" | ||
885 | }, | ||
886 | { | ||
887 | "highlight_end": 11, | ||
888 | "highlight_start": 1, | ||
889 | "text": " }" | ||
890 | }, | ||
891 | { | ||
892 | "highlight_end": 7, | ||
893 | "highlight_start": 1, | ||
894 | "text": " }" | ||
895 | }, | ||
896 | { | ||
897 | "highlight_end": 42, | ||
898 | "highlight_start": 1, | ||
899 | "text": " }) ; ($ left : expr, $ right : expr,) =>" | ||
900 | }, | ||
901 | { | ||
902 | "highlight_end": 49, | ||
903 | "highlight_start": 1, | ||
904 | "text": "({ $ crate :: assert_eq ! ($ left, $ right) }) ;" | ||
905 | }, | ||
906 | { | ||
907 | "highlight_end": 53, | ||
908 | "highlight_start": 1, | ||
909 | "text": "($ left : expr, $ right : expr, $ ($ arg : tt) +) =>" | ||
910 | }, | ||
911 | { | ||
912 | "highlight_end": 3, | ||
913 | "highlight_start": 1, | ||
914 | "text": "({" | ||
915 | }, | ||
916 | { | ||
917 | "highlight_end": 37, | ||
918 | "highlight_start": 1, | ||
919 | "text": " match (& ($ left), & ($ right))" | ||
920 | }, | ||
921 | { | ||
922 | "highlight_end": 7, | ||
923 | "highlight_start": 1, | ||
924 | "text": " {" | ||
925 | }, | ||
926 | { | ||
927 | "highlight_end": 34, | ||
928 | "highlight_start": 1, | ||
929 | "text": " (left_val, right_val) =>" | ||
930 | }, | ||
931 | { | ||
932 | "highlight_end": 11, | ||
933 | "highlight_start": 1, | ||
934 | "text": " {" | ||
935 | }, | ||
936 | { | ||
937 | "highlight_end": 46, | ||
938 | "highlight_start": 1, | ||
939 | "text": " if ! (* left_val == * right_val)" | ||
940 | }, | ||
941 | { | ||
942 | "highlight_end": 15, | ||
943 | "highlight_start": 1, | ||
944 | "text": " {" | ||
945 | }, | ||
946 | { | ||
947 | "highlight_end": 25, | ||
948 | "highlight_start": 1, | ||
949 | "text": " panic !" | ||
950 | }, | ||
951 | { | ||
952 | "highlight_end": 57, | ||
953 | "highlight_start": 1, | ||
954 | "text": " (r#\"assertion failed: `(left == right)`" | ||
955 | }, | ||
956 | { | ||
957 | "highlight_end": 16, | ||
958 | "highlight_start": 1, | ||
959 | "text": " left: `{:?}`," | ||
960 | }, | ||
961 | { | ||
962 | "highlight_end": 22, | ||
963 | "highlight_start": 1, | ||
964 | "text": " right: `{:?}`: {}\"#," | ||
965 | }, | ||
966 | { | ||
967 | "highlight_end": 72, | ||
968 | "highlight_start": 1, | ||
969 | "text": " & * left_val, & * right_val, $ crate :: format_args !" | ||
970 | }, | ||
971 | { | ||
972 | "highlight_end": 33, | ||
973 | "highlight_start": 1, | ||
974 | "text": " ($ ($ arg) +))" | ||
975 | }, | ||
976 | { | ||
977 | "highlight_end": 15, | ||
978 | "highlight_start": 1, | ||
979 | "text": " }" | ||
980 | }, | ||
981 | { | ||
982 | "highlight_end": 11, | ||
983 | "highlight_start": 1, | ||
984 | "text": " }" | ||
985 | }, | ||
986 | { | ||
987 | "highlight_end": 7, | ||
988 | "highlight_start": 1, | ||
989 | "text": " }" | ||
990 | }, | ||
991 | { | ||
992 | "highlight_end": 6, | ||
993 | "highlight_start": 1, | ||
994 | "text": " }) ;" | ||
995 | } | ||
996 | ] | ||
997 | }, | ||
998 | "macro_decl_name": "assert_eq!", | ||
999 | "span": { | ||
1000 | "byte_end": 38, | ||
1001 | "byte_start": 16, | ||
1002 | "column_end": 27, | ||
1003 | "column_start": 5, | ||
1004 | "expansion": null, | ||
1005 | "file_name": "src/main.rs", | ||
1006 | "is_primary": false, | ||
1007 | "label": null, | ||
1008 | "line_end": 2, | ||
1009 | "line_start": 2, | ||
1010 | "suggested_replacement": null, | ||
1011 | "suggestion_applicability": null, | ||
1012 | "text": [ | ||
1013 | { | ||
1014 | "highlight_end": 27, | ||
1015 | "highlight_start": 5, | ||
1016 | "text": " assert_eq!(1, \"love\");" | ||
1017 | } | ||
1018 | ] | ||
1019 | } | ||
1020 | }, | ||
1021 | "file_name": "<::core::macros::assert_eq macros>", | ||
1022 | "is_primary": true, | ||
1023 | "label": "no implementation for `{integer} == &str`", | ||
1024 | "line_end": 7, | ||
1025 | "line_start": 7, | ||
1026 | "suggested_replacement": null, | ||
1027 | "suggestion_applicability": null, | ||
1028 | "text": [ | ||
1029 | { | ||
1030 | "highlight_end": 33, | ||
1031 | "highlight_start": 31, | ||
1032 | "text": " if ! (* left_val == * right_val)" | ||
1033 | } | ||
1034 | ] | ||
1035 | } | ||
1036 | ] | ||
1037 | }"##, | ||
1038 | ); | ||
1039 | |||
1040 | let workspace_root = PathBuf::from("/test/"); | ||
1041 | let diag = map_rust_diagnostic_to_lsp(&diag, &workspace_root); | ||
1042 | insta::assert_debug_snapshot!(diag); | ||
1043 | } | ||
1044 | |||
1045 | #[test] | ||
1046 | #[cfg(not(windows))] | ||
1047 | fn snap_macro_compiler_error() { | ||
1048 | let diag = parse_diagnostic( | ||
1049 | r##"{ | ||
1050 | "rendered": "error: Please register your known path in the path module\n --> crates/ra_hir_def/src/path.rs:265:9\n |\n265 | compile_error!(\"Please register your known path in the path module\")\n | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^\n | \n ::: crates/ra_hir_def/src/data.rs:80:16\n |\n80 | let path = path![std::future::Future];\n | -------------------------- in this macro invocation\n\n", | ||
1051 | "children": [], | ||
1052 | "code": null, | ||
1053 | "level": "error", | ||
1054 | "message": "Please register your known path in the path module", | ||
1055 | "spans": [ | ||
1056 | { | ||
1057 | "byte_end": 8285, | ||
1058 | "byte_start": 8217, | ||
1059 | "column_end": 77, | ||
1060 | "column_start": 9, | ||
1061 | "expansion": { | ||
1062 | "def_site_span": { | ||
1063 | "byte_end": 8294, | ||
1064 | "byte_start": 7858, | ||
1065 | "column_end": 2, | ||
1066 | "column_start": 1, | ||
1067 | "expansion": null, | ||
1068 | "file_name": "crates/ra_hir_def/src/path.rs", | ||
1069 | "is_primary": false, | ||
1070 | "label": null, | ||
1071 | "line_end": 267, | ||
1072 | "line_start": 254, | ||
1073 | "suggested_replacement": null, | ||
1074 | "suggestion_applicability": null, | ||
1075 | "text": [ | ||
1076 | { | ||
1077 | "highlight_end": 28, | ||
1078 | "highlight_start": 1, | ||
1079 | "text": "macro_rules! __known_path {" | ||
1080 | }, | ||
1081 | { | ||
1082 | "highlight_end": 37, | ||
1083 | "highlight_start": 1, | ||
1084 | "text": " (std::iter::IntoIterator) => {};" | ||
1085 | }, | ||
1086 | { | ||
1087 | "highlight_end": 33, | ||
1088 | "highlight_start": 1, | ||
1089 | "text": " (std::result::Result) => {};" | ||
1090 | }, | ||
1091 | { | ||
1092 | "highlight_end": 29, | ||
1093 | "highlight_start": 1, | ||
1094 | "text": " (std::ops::Range) => {};" | ||
1095 | }, | ||
1096 | { | ||
1097 | "highlight_end": 33, | ||
1098 | "highlight_start": 1, | ||
1099 | "text": " (std::ops::RangeFrom) => {};" | ||
1100 | }, | ||
1101 | { | ||
1102 | "highlight_end": 33, | ||
1103 | "highlight_start": 1, | ||
1104 | "text": " (std::ops::RangeFull) => {};" | ||
1105 | }, | ||
1106 | { | ||
1107 | "highlight_end": 31, | ||
1108 | "highlight_start": 1, | ||
1109 | "text": " (std::ops::RangeTo) => {};" | ||
1110 | }, | ||
1111 | { | ||
1112 | "highlight_end": 40, | ||
1113 | "highlight_start": 1, | ||
1114 | "text": " (std::ops::RangeToInclusive) => {};" | ||
1115 | }, | ||
1116 | { | ||
1117 | "highlight_end": 38, | ||
1118 | "highlight_start": 1, | ||
1119 | "text": " (std::ops::RangeInclusive) => {};" | ||
1120 | }, | ||
1121 | { | ||
1122 | "highlight_end": 27, | ||
1123 | "highlight_start": 1, | ||
1124 | "text": " (std::ops::Try) => {};" | ||
1125 | }, | ||
1126 | { | ||
1127 | "highlight_end": 22, | ||
1128 | "highlight_start": 1, | ||
1129 | "text": " ($path:path) => {" | ||
1130 | }, | ||
1131 | { | ||
1132 | "highlight_end": 77, | ||
1133 | "highlight_start": 1, | ||
1134 | "text": " compile_error!(\"Please register your known path in the path module\")" | ||
1135 | }, | ||
1136 | { | ||
1137 | "highlight_end": 7, | ||
1138 | "highlight_start": 1, | ||
1139 | "text": " };" | ||
1140 | }, | ||
1141 | { | ||
1142 | "highlight_end": 2, | ||
1143 | "highlight_start": 1, | ||
1144 | "text": "}" | ||
1145 | } | ||
1146 | ] | ||
1147 | }, | ||
1148 | "macro_decl_name": "$crate::__known_path!", | ||
1149 | "span": { | ||
1150 | "byte_end": 8427, | ||
1151 | "byte_start": 8385, | ||
1152 | "column_end": 51, | ||
1153 | "column_start": 9, | ||
1154 | "expansion": { | ||
1155 | "def_site_span": { | ||
1156 | "byte_end": 8611, | ||
1157 | "byte_start": 8312, | ||
1158 | "column_end": 2, | ||
1159 | "column_start": 1, | ||
1160 | "expansion": null, | ||
1161 | "file_name": "crates/ra_hir_def/src/path.rs", | ||
1162 | "is_primary": false, | ||
1163 | "label": null, | ||
1164 | "line_end": 277, | ||
1165 | "line_start": 270, | ||
1166 | "suggested_replacement": null, | ||
1167 | "suggestion_applicability": null, | ||
1168 | "text": [ | ||
1169 | { | ||
1170 | "highlight_end": 22, | ||
1171 | "highlight_start": 1, | ||
1172 | "text": "macro_rules! __path {" | ||
1173 | }, | ||
1174 | { | ||
1175 | "highlight_end": 43, | ||
1176 | "highlight_start": 1, | ||
1177 | "text": " ($start:ident $(:: $seg:ident)*) => ({" | ||
1178 | }, | ||
1179 | { | ||
1180 | "highlight_end": 51, | ||
1181 | "highlight_start": 1, | ||
1182 | "text": " $crate::__known_path!($start $(:: $seg)*);" | ||
1183 | }, | ||
1184 | { | ||
1185 | "highlight_end": 87, | ||
1186 | "highlight_start": 1, | ||
1187 | "text": " $crate::path::ModPath::from_simple_segments($crate::path::PathKind::Abs, vec![" | ||
1188 | }, | ||
1189 | { | ||
1190 | "highlight_end": 76, | ||
1191 | "highlight_start": 1, | ||
1192 | "text": " $crate::path::__name![$start], $($crate::path::__name![$seg],)*" | ||
1193 | }, | ||
1194 | { | ||
1195 | "highlight_end": 11, | ||
1196 | "highlight_start": 1, | ||
1197 | "text": " ])" | ||
1198 | }, | ||
1199 | { | ||
1200 | "highlight_end": 8, | ||
1201 | "highlight_start": 1, | ||
1202 | "text": " });" | ||
1203 | }, | ||
1204 | { | ||
1205 | "highlight_end": 2, | ||
1206 | "highlight_start": 1, | ||
1207 | "text": "}" | ||
1208 | } | ||
1209 | ] | ||
1210 | }, | ||
1211 | "macro_decl_name": "path!", | ||
1212 | "span": { | ||
1213 | "byte_end": 2966, | ||
1214 | "byte_start": 2940, | ||
1215 | "column_end": 42, | ||
1216 | "column_start": 16, | ||
1217 | "expansion": null, | ||
1218 | "file_name": "crates/ra_hir_def/src/data.rs", | ||
1219 | "is_primary": false, | ||
1220 | "label": null, | ||
1221 | "line_end": 80, | ||
1222 | "line_start": 80, | ||
1223 | "suggested_replacement": null, | ||
1224 | "suggestion_applicability": null, | ||
1225 | "text": [ | ||
1226 | { | ||
1227 | "highlight_end": 42, | ||
1228 | "highlight_start": 16, | ||
1229 | "text": " let path = path![std::future::Future];" | ||
1230 | } | ||
1231 | ] | ||
1232 | } | ||
1233 | }, | ||
1234 | "file_name": "crates/ra_hir_def/src/path.rs", | ||
1235 | "is_primary": false, | ||
1236 | "label": null, | ||
1237 | "line_end": 272, | ||
1238 | "line_start": 272, | ||
1239 | "suggested_replacement": null, | ||
1240 | "suggestion_applicability": null, | ||
1241 | "text": [ | ||
1242 | { | ||
1243 | "highlight_end": 51, | ||
1244 | "highlight_start": 9, | ||
1245 | "text": " $crate::__known_path!($start $(:: $seg)*);" | ||
1246 | } | ||
1247 | ] | ||
1248 | } | ||
1249 | }, | ||
1250 | "file_name": "crates/ra_hir_def/src/path.rs", | ||
1251 | "is_primary": true, | ||
1252 | "label": null, | ||
1253 | "line_end": 265, | ||
1254 | "line_start": 265, | ||
1255 | "suggested_replacement": null, | ||
1256 | "suggestion_applicability": null, | ||
1257 | "text": [ | ||
1258 | { | ||
1259 | "highlight_end": 77, | ||
1260 | "highlight_start": 9, | ||
1261 | "text": " compile_error!(\"Please register your known path in the path module\")" | ||
1262 | } | ||
1263 | ] | ||
1264 | } | ||
1265 | ] | ||
1266 | } | ||
1267 | "##, | ||
1268 | ); | ||
1269 | |||
1270 | let workspace_root = PathBuf::from("/test/"); | ||
1271 | let diag = map_rust_diagnostic_to_lsp(&diag, &workspace_root); | ||
1272 | insta::assert_debug_snapshot!(diag); | ||
1273 | } | ||
1274 | |||
1275 | #[test] | ||
1276 | #[cfg(not(windows))] | ||
1277 | fn snap_multi_line_fix() { | ||
1278 | let diag = parse_diagnostic( | ||
1279 | r##"{ | ||
1280 | "rendered": "warning: returning the result of a let binding from a block\n --> src/main.rs:4:5\n |\n3 | let a = (0..10).collect();\n | -------------------------- unnecessary let binding\n4 | a\n | ^\n |\n = note: `#[warn(clippy::let_and_return)]` on by default\n = help: for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#let_and_return\nhelp: return the expression directly\n |\n3 | \n4 | (0..10).collect()\n |\n\n", | ||
1281 | "children": [ | ||
1282 | { | ||
1283 | "children": [], | ||
1284 | "code": null, | ||
1285 | "level": "note", | ||
1286 | "message": "`#[warn(clippy::let_and_return)]` on by default", | ||
1287 | "rendered": null, | ||
1288 | "spans": [] | ||
1289 | }, | ||
1290 | { | ||
1291 | "children": [], | ||
1292 | "code": null, | ||
1293 | "level": "help", | ||
1294 | "message": "for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#let_and_return", | ||
1295 | "rendered": null, | ||
1296 | "spans": [] | ||
1297 | }, | ||
1298 | { | ||
1299 | "children": [], | ||
1300 | "code": null, | ||
1301 | "level": "help", | ||
1302 | "message": "return the expression directly", | ||
1303 | "rendered": null, | ||
1304 | "spans": [ | ||
1305 | { | ||
1306 | "byte_end": 55, | ||
1307 | "byte_start": 29, | ||
1308 | "column_end": 31, | ||
1309 | "column_start": 5, | ||
1310 | "expansion": null, | ||
1311 | "file_name": "src/main.rs", | ||
1312 | "is_primary": true, | ||
1313 | "label": null, | ||
1314 | "line_end": 3, | ||
1315 | "line_start": 3, | ||
1316 | "suggested_replacement": "", | ||
1317 | "suggestion_applicability": "MachineApplicable", | ||
1318 | "text": [ | ||
1319 | { | ||
1320 | "highlight_end": 31, | ||
1321 | "highlight_start": 5, | ||
1322 | "text": " let a = (0..10).collect();" | ||
1323 | } | ||
1324 | ] | ||
1325 | }, | ||
1326 | { | ||
1327 | "byte_end": 61, | ||
1328 | "byte_start": 60, | ||
1329 | "column_end": 6, | ||
1330 | "column_start": 5, | ||
1331 | "expansion": null, | ||
1332 | "file_name": "src/main.rs", | ||
1333 | "is_primary": true, | ||
1334 | "label": null, | ||
1335 | "line_end": 4, | ||
1336 | "line_start": 4, | ||
1337 | "suggested_replacement": "(0..10).collect()", | ||
1338 | "suggestion_applicability": "MachineApplicable", | ||
1339 | "text": [ | ||
1340 | { | ||
1341 | "highlight_end": 6, | ||
1342 | "highlight_start": 5, | ||
1343 | "text": " a" | ||
1344 | } | ||
1345 | ] | ||
1346 | } | ||
1347 | ] | ||
1348 | } | ||
1349 | ], | ||
1350 | "code": { | ||
1351 | "code": "clippy::let_and_return", | ||
1352 | "explanation": null | ||
1353 | }, | ||
1354 | "level": "warning", | ||
1355 | "message": "returning the result of a let binding from a block", | ||
1356 | "spans": [ | ||
1357 | { | ||
1358 | "byte_end": 55, | ||
1359 | "byte_start": 29, | ||
1360 | "column_end": 31, | ||
1361 | "column_start": 5, | ||
1362 | "expansion": null, | ||
1363 | "file_name": "src/main.rs", | ||
1364 | "is_primary": false, | ||
1365 | "label": "unnecessary let binding", | ||
1366 | "line_end": 3, | ||
1367 | "line_start": 3, | ||
1368 | "suggested_replacement": null, | ||
1369 | "suggestion_applicability": null, | ||
1370 | "text": [ | ||
1371 | { | ||
1372 | "highlight_end": 31, | ||
1373 | "highlight_start": 5, | ||
1374 | "text": " let a = (0..10).collect();" | ||
1375 | } | ||
1376 | ] | ||
1377 | }, | ||
1378 | { | ||
1379 | "byte_end": 61, | ||
1380 | "byte_start": 60, | ||
1381 | "column_end": 6, | ||
1382 | "column_start": 5, | ||
1383 | "expansion": null, | ||
1384 | "file_name": "src/main.rs", | ||
1385 | "is_primary": true, | ||
1386 | "label": null, | ||
1387 | "line_end": 4, | ||
1388 | "line_start": 4, | ||
1389 | "suggested_replacement": null, | ||
1390 | "suggestion_applicability": null, | ||
1391 | "text": [ | ||
1392 | { | ||
1393 | "highlight_end": 6, | ||
1394 | "highlight_start": 5, | ||
1395 | "text": " a" | ||
1396 | } | ||
1397 | ] | ||
1398 | } | ||
1399 | ] | ||
1400 | } | ||
1401 | "##, | ||
1402 | ); | ||
1403 | |||
1404 | let workspace_root = PathBuf::from("/test/"); | ||
1405 | let diag = map_rust_diagnostic_to_lsp(&diag, &workspace_root); | ||
1406 | insta::assert_debug_snapshot!(diag); | ||
1407 | } | ||
1408 | } | ||
diff --git a/crates/rust-analyzer/src/main_loop.rs b/crates/rust-analyzer/src/main_loop.rs index 13d305b97..15e5bb354 100644 --- a/crates/rust-analyzer/src/main_loop.rs +++ b/crates/rust-analyzer/src/main_loop.rs | |||
@@ -25,7 +25,7 @@ use lsp_types::{ | |||
25 | WorkDoneProgressBegin, WorkDoneProgressCreateParams, WorkDoneProgressEnd, | 25 | WorkDoneProgressBegin, WorkDoneProgressCreateParams, WorkDoneProgressEnd, |
26 | WorkDoneProgressReport, | 26 | WorkDoneProgressReport, |
27 | }; | 27 | }; |
28 | use ra_flycheck::{url_from_path_with_drive_lowercasing, CheckTask}; | 28 | use ra_flycheck::{CheckTask, Status}; |
29 | use ra_ide::{Canceled, FileId, LibraryData, LineIndex, SourceRootId}; | 29 | use ra_ide::{Canceled, FileId, LibraryData, LineIndex, SourceRootId}; |
30 | use ra_prof::profile; | 30 | use ra_prof::profile; |
31 | use ra_project_model::{PackageRoot, ProjectWorkspace}; | 31 | use ra_project_model::{PackageRoot, ProjectWorkspace}; |
@@ -37,7 +37,7 @@ use threadpool::ThreadPool; | |||
37 | 37 | ||
38 | use crate::{ | 38 | use crate::{ |
39 | config::{Config, FilesWatcher}, | 39 | config::{Config, FilesWatcher}, |
40 | diagnostics::DiagnosticTask, | 40 | diagnostics::{to_proto::url_from_path_with_drive_lowercasing, DiagnosticTask}, |
41 | from_proto, lsp_ext, | 41 | from_proto, lsp_ext, |
42 | main_loop::{ | 42 | main_loop::{ |
43 | pending_requests::{PendingRequest, PendingRequests}, | 43 | pending_requests::{PendingRequest, PendingRequests}, |
@@ -736,22 +736,61 @@ fn on_check_task( | |||
736 | task_sender.send(Task::Diagnostic(DiagnosticTask::ClearCheck))?; | 736 | task_sender.send(Task::Diagnostic(DiagnosticTask::ClearCheck))?; |
737 | } | 737 | } |
738 | 738 | ||
739 | CheckTask::AddDiagnostic { url, diagnostic, fixes } => { | 739 | CheckTask::AddDiagnostic { workspace_root, diagnostic } => { |
740 | let path = url.to_file_path().map_err(|()| format!("invalid uri: {}", url))?; | 740 | let diagnostics = crate::diagnostics::to_proto::map_rust_diagnostic_to_lsp( |
741 | let file_id = match world_state.vfs.read().path2file(&path) { | 741 | &diagnostic, |
742 | Some(file) => FileId(file.0), | 742 | &workspace_root, |
743 | None => { | 743 | ); |
744 | log::error!("File with cargo diagnostic not found in VFS: {}", path.display()); | 744 | for diag in diagnostics { |
745 | return Ok(()); | 745 | let path = diag |
746 | } | 746 | .location |
747 | }; | 747 | .uri |
748 | .to_file_path() | ||
749 | .map_err(|()| format!("invalid uri: {}", diag.location.uri))?; | ||
750 | let file_id = match world_state.vfs.read().path2file(&path) { | ||
751 | Some(file) => FileId(file.0), | ||
752 | None => { | ||
753 | log::error!( | ||
754 | "File with cargo diagnostic not found in VFS: {}", | ||
755 | path.display() | ||
756 | ); | ||
757 | return Ok(()); | ||
758 | } | ||
759 | }; | ||
748 | 760 | ||
749 | task_sender | 761 | task_sender.send(Task::Diagnostic(DiagnosticTask::AddCheck( |
750 | .send(Task::Diagnostic(DiagnosticTask::AddCheck(file_id, diagnostic, fixes)))?; | 762 | file_id, |
763 | diag.diagnostic, | ||
764 | diag.fixes.into_iter().map(|it| it.into()).collect(), | ||
765 | )))?; | ||
766 | } | ||
751 | } | 767 | } |
752 | 768 | ||
753 | CheckTask::Status(progress) => { | 769 | CheckTask::Status(status) => { |
754 | if world_state.config.client_caps.work_done_progress { | 770 | if world_state.config.client_caps.work_done_progress { |
771 | let progress = match status { | ||
772 | Status::Being => { | ||
773 | lsp_types::WorkDoneProgress::Begin(lsp_types::WorkDoneProgressBegin { | ||
774 | title: "Running `cargo check`".to_string(), | ||
775 | cancellable: Some(false), | ||
776 | message: None, | ||
777 | percentage: None, | ||
778 | }) | ||
779 | } | ||
780 | Status::Progress(target) => { | ||
781 | lsp_types::WorkDoneProgress::Report(lsp_types::WorkDoneProgressReport { | ||
782 | cancellable: Some(false), | ||
783 | message: Some(target), | ||
784 | percentage: None, | ||
785 | }) | ||
786 | } | ||
787 | Status::End => { | ||
788 | lsp_types::WorkDoneProgress::End(lsp_types::WorkDoneProgressEnd { | ||
789 | message: None, | ||
790 | }) | ||
791 | } | ||
792 | }; | ||
793 | |||
755 | let params = lsp_types::ProgressParams { | 794 | let params = lsp_types::ProgressParams { |
756 | token: lsp_types::ProgressToken::String( | 795 | token: lsp_types::ProgressToken::String( |
757 | "rustAnalyzer/cargoWatcher".to_string(), | 796 | "rustAnalyzer/cargoWatcher".to_string(), |
diff --git a/crates/rust-analyzer/src/world.rs b/crates/rust-analyzer/src/world.rs index 6333c15b2..367272925 100644 --- a/crates/rust-analyzer/src/world.rs +++ b/crates/rust-analyzer/src/world.rs | |||
@@ -11,7 +11,7 @@ use std::{ | |||
11 | use crossbeam_channel::{unbounded, Receiver}; | 11 | use crossbeam_channel::{unbounded, Receiver}; |
12 | use lsp_types::Url; | 12 | use lsp_types::Url; |
13 | use parking_lot::RwLock; | 13 | use parking_lot::RwLock; |
14 | use ra_flycheck::{url_from_path_with_drive_lowercasing, Flycheck, FlycheckConfig}; | 14 | use ra_flycheck::{Flycheck, FlycheckConfig}; |
15 | use ra_ide::{ | 15 | use ra_ide::{ |
16 | Analysis, AnalysisChange, AnalysisHost, CrateGraph, FileId, LibraryData, SourceRootId, | 16 | Analysis, AnalysisChange, AnalysisHost, CrateGraph, FileId, LibraryData, SourceRootId, |
17 | }; | 17 | }; |
@@ -22,7 +22,9 @@ use stdx::format_to; | |||
22 | 22 | ||
23 | use crate::{ | 23 | use crate::{ |
24 | config::Config, | 24 | config::Config, |
25 | diagnostics::{CheckFixes, DiagnosticCollection}, | 25 | diagnostics::{ |
26 | to_proto::url_from_path_with_drive_lowercasing, CheckFixes, DiagnosticCollection, | ||
27 | }, | ||
26 | main_loop::pending_requests::{CompletedRequest, LatestRequests}, | 28 | main_loop::pending_requests::{CompletedRequest, LatestRequests}, |
27 | vfs_glob::{Glob, RustPackageFilterBuilder}, | 29 | vfs_glob::{Glob, RustPackageFilterBuilder}, |
28 | LspError, Result, | 30 | LspError, Result, |