Skip to content

Commit 16ea84f

Browse files
committed
Add support for file operations in CodeActions
1 parent d8c3b69 commit 16ea84f

File tree

3 files changed

+80
-14
lines changed

3 files changed

+80
-14
lines changed

ycmd/completers/language_server/language_server_completer.py

+26-6
Original file line numberDiff line numberDiff line change
@@ -3590,21 +3590,41 @@ def WorkspaceEditToFixIt( request_data,
35903590
if not workspace_edit:
35913591
return None
35923592

3593+
chunks = []
35933594
if 'changes' in workspace_edit:
3594-
chunks = []
35953595
# We sort the filenames to make the response stable. Edits are applied in
35963596
# strict sequence within a file, but apply to files in arbitrary order.
35973597
# However, it's important for the response to be stable for the tests.
35983598
for uri in sorted( workspace_edit[ 'changes' ].keys() ):
35993599
chunks.extend( TextEditToChunks( request_data,
36003600
uri,
36013601
workspace_edit[ 'changes' ][ uri ] ) )
3602-
else:
3603-
chunks = []
3602+
if 'documentChanges' in workspace_edit:
36043603
for text_document_edit in workspace_edit[ 'documentChanges' ]:
3605-
uri = text_document_edit[ 'textDocument' ][ 'uri' ]
3606-
edits = text_document_edit[ 'edits' ]
3607-
chunks.extend( TextEditToChunks( request_data, uri, edits ) )
3604+
kind = text_document_edit.get( 'kind', '' )
3605+
options = text_document_edit.get( 'options', {} )
3606+
if 'edits' in text_document_edit:
3607+
uri = text_document_edit[ 'textDocument' ][ 'uri' ]
3608+
edits = text_document_edit[ 'edits' ]
3609+
chunks.extend( TextEditToChunks( request_data, uri, edits ) )
3610+
elif kind == 'rename':
3611+
chunks.append(
3612+
responses.RenameChunk(
3613+
old_filepath = lsp.UriToFilePath(
3614+
text_document_edit[ 'oldUri' ] ),
3615+
new_filepath = lsp.UriToFilePath(
3616+
text_document_edit[ 'newUri' ] ),
3617+
options = options ) )
3618+
elif kind == 'delete':
3619+
chunks.append(
3620+
responses.DeleteChunk(
3621+
lsp.UriToFilePath( text_document_edit[ 'uri' ] ),
3622+
options = options ) )
3623+
elif kind == 'create':
3624+
chunks.append(
3625+
responses.CreateChunk(
3626+
lsp.UriToFilePath( text_document_edit[ 'uri' ] ),
3627+
options = options ) )
36083628
return responses.FixIt(
36093629
responses.Location( request_data[ 'line_num' ],
36103630
request_data[ 'column_num' ],

ycmd/completers/language_server/language_server_protocol.py

+5-1
Original file line numberDiff line numberDiff line change
@@ -307,7 +307,11 @@ def Initialize( request_id,
307307
'didChangeWatchedFiles': {
308308
'dynamicRegistration': True
309309
},
310-
'workspaceEdit': { 'documentChanges': True, },
310+
'workspaceEdit': {
311+
'resourceOperations': [ 'create', 'rename', 'delete' ],
312+
'documentChanges': True,
313+
'failureHandling': 'abort'
314+
},
311315
'symbol': {
312316
'symbolKind': {
313317
'valueSet': list( range( 1, len( SYMBOL_KIND ) ) ),

ycmd/responses.py

+49-7
Original file line numberDiff line numberDiff line change
@@ -268,6 +268,47 @@ def __init__( self, location: Location, chunks, text = '', kind = None ):
268268
self.kind = kind
269269

270270

271+
class DeleteChunk:
272+
def __init__( self, filepath, options ):
273+
self.filepath = filepath
274+
self.options = options
275+
276+
def ToYcmdProtocol( self ):
277+
return {
278+
'filepath': self.filepath,
279+
'kind': 'delete',
280+
'options': self.options
281+
}
282+
283+
284+
class CreateChunk:
285+
def __init__( self, filepath, options ):
286+
self.filepath = filepath
287+
self.options = options
288+
289+
def ToYcmdProtocol( self ):
290+
return {
291+
'filepath': self.filepath,
292+
'kind': 'create',
293+
'options': self.options
294+
}
295+
296+
297+
class RenameChunk:
298+
def __init__( self, old_filepath, new_filepath, options ):
299+
self.old_filepath = old_filepath
300+
self.new_filepath = new_filepath
301+
self.options = options
302+
303+
def ToYcmdProtocol( self ):
304+
return {
305+
'new_filepath': self.new_filepath,
306+
'old_filepath': self.old_filepath,
307+
'kind': 'rename',
308+
'options': self.options
309+
}
310+
311+
271312
class FixItChunk:
272313
"""An individual replacement within a FixIt (aka Refactor)"""
273314

@@ -276,6 +317,13 @@ def __init__( self, replacement_text: str, range: Range ):
276317
self.replacement_text = replacement_text
277318
self.range = range
278319

320+
def ToYcmdProtocol( self ):
321+
return {
322+
'replacement_text': self.replacement_text,
323+
'range': BuildRangeData( self.range ),
324+
'kind': 'change'
325+
}
326+
279327

280328
def BuildDiagnosticData( diagnostic ):
281329
kind = ( diagnostic.kind_.name if hasattr( diagnostic.kind_, 'name' )
@@ -314,12 +362,6 @@ def BuildFixItResponse( fixits ):
314362
can be used to apply arbitrary changes to arbitrary files and is suitable for
315363
both quick fix and refactor operations"""
316364

317-
def BuildFixitChunkData( chunk ):
318-
return {
319-
'replacement_text': chunk.replacement_text,
320-
'range': BuildRangeData( chunk.range ),
321-
}
322-
323365
def BuildFixItData( fixit ):
324366
if hasattr( fixit, 'resolve' ):
325367
result = {
@@ -331,7 +373,7 @@ def BuildFixItData( fixit ):
331373
else:
332374
result = {
333375
'location': BuildLocationData( fixit.location ),
334-
'chunks' : [ BuildFixitChunkData( x ) for x in fixit.chunks ],
376+
'chunks' : [ x.ToYcmdProtocol() for x in fixit.chunks ],
335377
'text': fixit.text,
336378
'kind': fixit.kind,
337379
'resolve': False

0 commit comments

Comments
 (0)