Skip to content

Commit 39e7d9a

Browse files
committed
Add support for file operations in CodeActions
1 parent 47371eb commit 39e7d9a

File tree

3 files changed

+72
-14
lines changed

3 files changed

+72
-14
lines changed

ycmd/completers/language_server/language_server_completer.py

+22-6
Original file line numberDiff line numberDiff line change
@@ -3590,21 +3590,37 @@ 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+
if 'edits' in text_document_edit:
3606+
uri = text_document_edit[ 'textDocument' ][ 'uri' ]
3607+
edits = text_document_edit[ 'edits' ]
3608+
chunks.extend( TextEditToChunks( request_data, uri, edits ) )
3609+
elif kind == 'rename':
3610+
chunks.append(
3611+
responses.RenameChunk(
3612+
old_filepath = lsp.UriToFilePath(
3613+
text_document_edit[ 'oldUri' ] ),
3614+
new_filepath = lsp.UriToFilePath(
3615+
text_document_edit[ 'newUri' ] ) ) )
3616+
elif kind == 'delete':
3617+
chunks.append(
3618+
responses.DeleteChunk(
3619+
lsp.UriToFilePath( text_document_edit[ 'uri' ] ) ) )
3620+
elif kind == 'create':
3621+
chunks.append(
3622+
responses.CreateChunk(
3623+
lsp.UriToFilePath( text_document_edit[ 'uri' ] ) ) )
36083624
return responses.FixIt(
36093625
responses.Location( request_data[ 'line_num' ],
36103626
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

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

270270

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

@@ -276,6 +314,12 @@ def __init__( self, replacement_text: str, range: Range ):
276314
self.replacement_text = replacement_text
277315
self.range = range
278316

317+
def ToYcmdProtocol( self ):
318+
return {
319+
'replacement_text': self.replacement_text,
320+
'range': BuildRangeData( self.range ),
321+
}
322+
279323

280324
def BuildDiagnosticData( diagnostic ):
281325
kind = ( diagnostic.kind_.name if hasattr( diagnostic.kind_, 'name' )
@@ -314,12 +358,6 @@ def BuildFixItResponse( fixits ):
314358
can be used to apply arbitrary changes to arbitrary files and is suitable for
315359
both quick fix and refactor operations"""
316360

317-
def BuildFixitChunkData( chunk ):
318-
return {
319-
'replacement_text': chunk.replacement_text,
320-
'range': BuildRangeData( chunk.range ),
321-
}
322-
323361
def BuildFixItData( fixit ):
324362
if hasattr( fixit, 'resolve' ):
325363
result = {
@@ -331,7 +369,7 @@ def BuildFixItData( fixit ):
331369
else:
332370
result = {
333371
'location': BuildLocationData( fixit.location ),
334-
'chunks' : [ BuildFixitChunkData( x ) for x in fixit.chunks ],
372+
'chunks' : [ x.ToYcmdProtocol() for x in fixit.chunks ],
335373
'text': fixit.text,
336374
'kind': fixit.kind,
337375
'resolve': False

0 commit comments

Comments
 (0)