1
- #![ feature( plugin, custom_derive) ]
2
- #![ plugin( serde_macros) ]
3
-
4
1
extern crate serde;
5
2
extern crate serde_json;
3
+ extern crate racer;
4
+ extern crate rustfmt;
6
5
7
6
use analysis:: { AnalysisHost , Span } ;
8
7
use vfs:: { Vfs , Change } ;
9
8
use build:: * ;
10
9
use std:: sync:: Arc ;
11
10
use std:: path:: Path ;
12
11
12
+ use self :: racer:: core:: complete_from_file;
13
+ use self :: racer:: core:: find_definition;
14
+ use self :: racer:: core;
15
+ use self :: rustfmt:: { Input as FmtInput , format_input} ;
16
+ use self :: rustfmt:: config:: { self , WriteMode } ;
17
+
13
18
use std:: fs:: { File , OpenOptions } ;
14
19
use std:: fmt:: Debug ;
15
- use serde:: Serialize ;
20
+ use std:: panic;
21
+ use serde:: { Serialize , Deserialize } ;
16
22
use ide:: VscodeKind ;
17
23
18
24
use std:: io:: { self , Read , Write , Error , ErrorKind } ;
@@ -25,13 +31,13 @@ const RUSTW_TIMEOUT: u64 = 500;
25
31
// For now this is a catch-all for any error back to the consumer of the RLS
26
32
const MethodNotFound : i64 = -32601 ;
27
33
28
- #[ derive( Debug , Deserialize , Serialize ) ]
34
+ #[ derive( Clone , Debug , Deserialize , Serialize ) ]
29
35
struct Position {
30
36
line : usize ,
31
37
character : usize
32
38
}
33
39
34
- #[ derive( Debug , Deserialize , Serialize ) ]
40
+ #[ derive( Clone , Debug , Deserialize , Serialize ) ]
35
41
struct Range {
36
42
start : Position ,
37
43
end : Position ,
@@ -84,6 +90,49 @@ struct SymbolInformation {
84
90
location : Location ,
85
91
}
86
92
93
+ #[ derive( Debug , Deserialize ) ]
94
+ struct CompilerMessageCode {
95
+ code : String
96
+ }
97
+
98
+ #[ derive( Debug , Deserialize ) ]
99
+ struct CompilerSpan {
100
+ file_name : String ,
101
+ line_start : usize ,
102
+ column_start : usize ,
103
+ line_end : usize ,
104
+ column_end : usize ,
105
+ }
106
+
107
+ #[ derive( Debug , Deserialize ) ]
108
+ struct CompilerMessage {
109
+ message : String ,
110
+ code : Option < CompilerMessageCode > ,
111
+ level : String ,
112
+ spans : Vec < CompilerSpan > ,
113
+ }
114
+
115
+ #[ derive( Debug , Clone , Serialize ) ]
116
+ struct Diagnostic {
117
+ range : Range ,
118
+ severity : u32 ,
119
+ code : String ,
120
+ message : String ,
121
+ }
122
+
123
+ #[ derive( Debug , Serialize ) ]
124
+ struct PublishDiagnosticsParams {
125
+ uri : String ,
126
+ diagnostics : Vec < Diagnostic > ,
127
+ }
128
+
129
+ #[ derive( Debug , Serialize ) ]
130
+ struct NotificationMessage < T > where T : Debug +Serialize {
131
+ jsonrpc : String ,
132
+ method : String ,
133
+ params : T ,
134
+ }
135
+
87
136
#[ allow( non_snake_case) ]
88
137
#[ derive( Debug , Deserialize ) ]
89
138
struct ReferenceParams {
@@ -133,6 +182,7 @@ enum Method {
133
182
GotoDef ( TextDocumentPositionParams ) ,
134
183
FindAllRef ( ReferenceParams ) ,
135
184
Symbols ( DocumentSymbolParams ) ,
185
+ Complete ( TextDocumentPositionParams ) ,
136
186
}
137
187
138
188
#[ derive( Debug , Serialize ) ]
@@ -164,6 +214,12 @@ struct InitializeCapabilities {
164
214
capabilities : ServerCapabilities
165
215
}
166
216
217
+ #[ derive( Debug , Serialize ) ]
218
+ struct CompletionItem {
219
+ label : String ,
220
+ detail : String ,
221
+ }
222
+
167
223
#[ derive( Debug , Serialize ) ]
168
224
struct ResponseSuccess < T > where T : Debug +Serialize {
169
225
jsonrpc : String ,
@@ -277,6 +333,12 @@ fn parse_message(input: &str) -> io::Result<ServerMessage> {
277
333
serde_json:: from_value ( params. unwrap ( ) . to_owned ( ) ) . unwrap ( ) ;
278
334
Ok ( ServerMessage :: Request ( Request { id : id, method : Method :: FindAllRef ( method) } ) )
279
335
}
336
+ "textDocument/completion" => {
337
+ let id = ls_command. lookup ( "id" ) . unwrap ( ) . as_u64 ( ) . unwrap ( ) as usize ;
338
+ let method: TextDocumentPositionParams =
339
+ serde_json:: from_value ( params. unwrap ( ) . to_owned ( ) ) . unwrap ( ) ;
340
+ Ok ( ServerMessage :: Request ( Request { id : id, method : Method :: Complete ( method) } ) )
341
+ }
280
342
"textDocument/documentSymbol" => {
281
343
let id = ls_command. lookup ( "id" ) . unwrap ( ) . as_u64 ( ) . unwrap ( ) as usize ;
282
344
let method: DocumentSymbolParams =
@@ -334,10 +396,51 @@ impl LSService {
334
396
335
397
let result = self . build_queue . request_build ( project_path, priority) ;
336
398
match result {
337
- BuildResult :: Success ( _) | BuildResult :: Failure ( _) => {
338
- let reply = serde_json:: to_string ( & result) . unwrap ( ) ;
399
+ BuildResult :: Success ( ref x) | BuildResult :: Failure ( ref x) => {
400
+ let result: Vec < Diagnostic > = x. iter ( ) . map ( |msg| {
401
+ match serde_json:: from_str :: < CompilerMessage > ( & msg) {
402
+ Ok ( method) => {
403
+ let diag = Diagnostic {
404
+ range : Range {
405
+ start : Position {
406
+ line : method. spans [ 0 ] . line_start ,
407
+ character : method. spans [ 0 ] . column_start ,
408
+ } ,
409
+ end : Position {
410
+ line : method. spans [ 0 ] . line_end ,
411
+ character : method. spans [ 0 ] . column_end ,
412
+ }
413
+ } ,
414
+ severity : if method. level == "error" { 1 } else { 2 } ,
415
+ code : match method. code {
416
+ Some ( c) => c. code . clone ( ) ,
417
+ None => String :: new ( ) ,
418
+ } ,
419
+ message : method. message . clone ( ) ,
420
+ } ;
421
+ let out = NotificationMessage {
422
+ jsonrpc : "2.0" . into ( ) ,
423
+ method : "textDocument/publishDiagnostics" . to_string ( ) ,
424
+ params : PublishDiagnosticsParams {
425
+ uri : "file://" . to_string ( ) + & method. spans [ 0 ] . file_name ,
426
+ diagnostics : vec ! [ diag. clone( ) ]
427
+ }
428
+ } ;
429
+ let output = serde_json:: to_string ( & out) . unwrap ( ) ;
430
+ output_response ( output) ;
431
+ diag
432
+ }
433
+ Err ( e) => {
434
+ io:: stderr ( ) . write ( & format ! ( "<<ERROR>> {:?}" , e) . into_bytes ( ) ) ;
435
+ io:: stderr ( ) . write ( & format ! ( "<<FROM>> {}" , msg) . into_bytes ( ) ) ;
436
+ panic ! ( ) ;
437
+ }
438
+ }
439
+ } ) . collect ( ) ;
440
+
441
+ //let reply = serde_json::to_string(&result).unwrap();
339
442
// println!("build result: {:?}", result);
340
- log ( format ! ( "build result: {:?}" , result) ) ;
443
+ // log(format!("build result: {:?}", result));
341
444
342
445
let file_name = Path :: new ( & project_path) . file_name ( )
343
446
. unwrap ( )
@@ -433,6 +536,54 @@ impl LSService {
433
536
output_response ( output) ;
434
537
}
435
538
539
+ fn complete ( & self , id : usize , params : TextDocumentPositionParams ) {
540
+ fn adjust_vscode_pos_for_racer ( mut source : Position ) -> Position {
541
+ source. line += 1 ;
542
+ source
543
+ }
544
+
545
+ fn adjust_racer_pos_for_vscode ( mut source : Position ) -> Position {
546
+ if source. line > 0 {
547
+ source. line -= 1 ;
548
+ }
549
+ source
550
+ }
551
+
552
+ let vfs: & Vfs = & self . vfs ;
553
+
554
+ let pos = adjust_vscode_pos_for_racer ( params. position ) ;
555
+ let fname: String = params. textDocument . uri . chars ( ) . skip ( "file://" . len ( ) ) . collect ( ) ;
556
+ let file_path = & Path :: new ( & fname) ;
557
+
558
+ let result: Vec < CompletionItem > = panic:: catch_unwind ( move || {
559
+
560
+ let cache = core:: FileCache :: new ( ) ;
561
+ let session = core:: Session :: from_path ( & cache, file_path, file_path) ;
562
+ for ( path, txt) in vfs. get_changed_files ( ) {
563
+ session. cache_file_contents ( & path, txt) ;
564
+ }
565
+
566
+ let src = session. load_file ( file_path) ;
567
+
568
+ let pos = session. load_file ( file_path) . coords_to_point ( pos. line , pos. character ) . unwrap ( ) ;
569
+ let results = complete_from_file ( & src. code , file_path, pos, & session) ;
570
+
571
+ results. map ( |comp| CompletionItem {
572
+ label : comp. matchstr . clone ( ) ,
573
+ detail : comp. contextstr . clone ( ) ,
574
+ } ) . collect ( )
575
+ } ) . unwrap_or ( vec ! [ ] ) ;
576
+
577
+ let out = ResponseSuccess {
578
+ jsonrpc : "2.0" . into ( ) ,
579
+ id : id,
580
+ result : result
581
+ } ;
582
+
583
+ let output = serde_json:: to_string ( & out) . unwrap ( ) ;
584
+ output_response ( output) ;
585
+ }
586
+
436
587
fn find_all_refs ( & self , id : usize , params : ReferenceParams ) {
437
588
let t = thread:: current ( ) ;
438
589
let uri = params. textDocument . uri . clone ( ) ;
@@ -684,6 +835,10 @@ pub fn run_server(analysis: Arc<AnalysisHost>, vfs: Arc<Vfs>, build_queue: Arc<B
684
835
try!( log. write_all ( & format ! ( "command(goto): {:?}\n " , params) . into_bytes ( ) ) ) ;
685
836
service. goto_def ( id, params) ;
686
837
}
838
+ Method :: Complete ( params) => {
839
+ try!( log. write_all ( & format ! ( "command(complete): {:?}\n " , params) . into_bytes ( ) ) ) ;
840
+ service. complete ( id, params) ;
841
+ }
687
842
Method :: Symbols ( params) => {
688
843
try!( log. write_all ( & format ! ( "command(goto): {:?}\n " , params) . into_bytes ( ) ) ) ;
689
844
service. symbols ( id, params) ;
@@ -722,10 +877,11 @@ pub fn run_server(analysis: Arc<AnalysisHost>, vfs: Arc<Vfs>, build_queue: Arc<B
722
877
}
723
878
} ;
724
879
725
- let output = serde_json:: to_string ( & result) . unwrap ( ) ;
726
- output_response ( output) ;
727
880
service. current_project = Some ( init. rootPath . clone ( ) ) ;
728
881
service. build ( & init. rootPath , BuildPriority :: Immediate ) ;
882
+
883
+ let output = serde_json:: to_string ( & result) . unwrap ( ) ;
884
+ output_response ( output) ;
729
885
}
730
886
}
731
887
}
0 commit comments