Better handling of crashed ycmd; restart command
Previously the YCM Vim client would go bonkers when ycmd crashed. Now the user can continue using Vim just without YCM functionality. Also added a :YcmRestartServer command to let the user restart ycmd if it crashed. With a little luck, this will be rarely necessary.
This commit is contained in:
parent
436017bd4d
commit
18002e17df
@ -566,7 +566,11 @@ function! youcompleteme#Complete( findstart, base )
|
|||||||
return -2
|
return -2
|
||||||
endif
|
endif
|
||||||
|
|
||||||
return pyeval( 'ycm_state.CreateCompletionRequest().CompletionStartColumn()' )
|
py request = ycm_state.CreateCompletionRequest()
|
||||||
|
if !pyeval( 'bool(request)' )
|
||||||
|
return -2
|
||||||
|
endif
|
||||||
|
return pyeval( 'request.CompletionStartColumn()' )
|
||||||
else
|
else
|
||||||
return s:CompletionsForQuery( a:base )
|
return s:CompletionsForQuery( a:base )
|
||||||
endif
|
endif
|
||||||
@ -584,6 +588,13 @@ function! youcompleteme#OmniComplete( findstart, base )
|
|||||||
endfunction
|
endfunction
|
||||||
|
|
||||||
|
|
||||||
|
function! s:RestartServer()
|
||||||
|
py ycm_state.RestartServer()
|
||||||
|
endfunction
|
||||||
|
|
||||||
|
command! YcmRestartServer call s:RestartServer()
|
||||||
|
|
||||||
|
|
||||||
function! s:ShowDetailedDiagnostic()
|
function! s:ShowDetailedDiagnostic()
|
||||||
py ycm_state.ShowDetailedDiagnostic()
|
py ycm_state.ShowDetailedDiagnostic()
|
||||||
endfunction
|
endfunction
|
||||||
|
@ -198,6 +198,7 @@ def _GetCompleterForRequestData( request_data ):
|
|||||||
|
|
||||||
@atexit.register
|
@atexit.register
|
||||||
def _ServerShutdown():
|
def _ServerShutdown():
|
||||||
|
LOGGER.info( 'Server shutting down' )
|
||||||
if SERVER_STATE:
|
if SERVER_STATE:
|
||||||
SERVER_STATE.Shutdown()
|
SERVER_STATE.Shutdown()
|
||||||
extra_conf_store.Shutdown()
|
extra_conf_store.Shutdown()
|
||||||
|
@ -88,12 +88,18 @@ class YouCompleteMe( object ):
|
|||||||
self._server_popen = subprocess.Popen( args,
|
self._server_popen = subprocess.Popen( args,
|
||||||
stdout = fstdout,
|
stdout = fstdout,
|
||||||
stderr = fstderr )
|
stderr = fstderr )
|
||||||
self._CheckIfServerCrashed()
|
self._NotifyUserIfServerCrashed()
|
||||||
|
|
||||||
|
|
||||||
def _CheckIfServerCrashed( self ):
|
def _IsServerAlive( self ):
|
||||||
server_crashed = self._server_popen.poll()
|
returncode = self._server_popen.poll()
|
||||||
if server_crashed:
|
# When the process hasn't finished yet, poll() returns None.
|
||||||
|
return returncode is None
|
||||||
|
|
||||||
|
|
||||||
|
def _NotifyUserIfServerCrashed( self ):
|
||||||
|
if self._IsServerAlive():
|
||||||
|
return
|
||||||
if self._server_stderr:
|
if self._server_stderr:
|
||||||
with open( self._server_stderr, 'r' ) as server_stderr_file:
|
with open( self._server_stderr, 'r' ) as server_stderr_file:
|
||||||
vimsupport.PostMultiLineNotice( SERVER_CRASH_MESSAGE_STDERR_FILE +
|
vimsupport.PostMultiLineNotice( SERVER_CRASH_MESSAGE_STDERR_FILE +
|
||||||
@ -102,6 +108,12 @@ class YouCompleteMe( object ):
|
|||||||
vimsupport.PostVimMessage( SERVER_CRASH_MESSAGE_SAME_STDERR )
|
vimsupport.PostVimMessage( SERVER_CRASH_MESSAGE_SAME_STDERR )
|
||||||
|
|
||||||
|
|
||||||
|
def RestartServer( self ):
|
||||||
|
vimsupport.PostVimMessage( 'Restarting ycmd server...' )
|
||||||
|
self.OnVimLeave()
|
||||||
|
self._SetupServer()
|
||||||
|
|
||||||
|
|
||||||
def CreateCompletionRequest( self, force_semantic = False ):
|
def CreateCompletionRequest( self, force_semantic = False ):
|
||||||
# We have to store a reference to the newly created CompletionRequest
|
# We have to store a reference to the newly created CompletionRequest
|
||||||
# because VimScript can't store a reference to a Python object across
|
# because VimScript can't store a reference to a Python object across
|
||||||
@ -111,17 +123,23 @@ class YouCompleteMe( object ):
|
|||||||
self._omnicomp.ShouldUseNow() ):
|
self._omnicomp.ShouldUseNow() ):
|
||||||
self._latest_completion_request = OmniCompletionRequest( self._omnicomp )
|
self._latest_completion_request = OmniCompletionRequest( self._omnicomp )
|
||||||
else:
|
else:
|
||||||
self._latest_completion_request = CompletionRequest( force_semantic )
|
self._latest_completion_request = ( CompletionRequest( force_semantic )
|
||||||
|
if self._IsServerAlive() else
|
||||||
|
None )
|
||||||
return self._latest_completion_request
|
return self._latest_completion_request
|
||||||
|
|
||||||
|
|
||||||
def SendCommandRequest( self, arguments, completer ):
|
def SendCommandRequest( self, arguments, completer ):
|
||||||
|
if self._IsServerAlive():
|
||||||
return SendCommandRequest( arguments, completer )
|
return SendCommandRequest( arguments, completer )
|
||||||
|
|
||||||
|
|
||||||
def GetDefinedSubcommands( self ):
|
def GetDefinedSubcommands( self ):
|
||||||
|
if self._IsServerAlive():
|
||||||
return BaseRequest.PostDataToHandler( BuildRequestData(),
|
return BaseRequest.PostDataToHandler( BuildRequestData(),
|
||||||
'defined_subcommands' )
|
'defined_subcommands' )
|
||||||
|
else:
|
||||||
|
return []
|
||||||
|
|
||||||
|
|
||||||
def GetCurrentCompletionRequest( self ):
|
def GetCurrentCompletionRequest( self ):
|
||||||
@ -143,9 +161,11 @@ class YouCompleteMe( object ):
|
|||||||
|
|
||||||
|
|
||||||
def OnFileReadyToParse( self ):
|
def OnFileReadyToParse( self ):
|
||||||
self._CheckIfServerCrashed()
|
|
||||||
self._omnicomp.OnFileReadyToParse( None )
|
self._omnicomp.OnFileReadyToParse( None )
|
||||||
|
|
||||||
|
if not self._IsServerAlive():
|
||||||
|
self._NotifyUserIfServerCrashed()
|
||||||
|
|
||||||
extra_data = {}
|
extra_data = {}
|
||||||
if self._user_options[ 'collect_identifiers_from_tags_files' ]:
|
if self._user_options[ 'collect_identifiers_from_tags_files' ]:
|
||||||
extra_data[ 'tag_files' ] = _GetTagFiles()
|
extra_data[ 'tag_files' ] = _GetTagFiles()
|
||||||
@ -159,26 +179,35 @@ class YouCompleteMe( object ):
|
|||||||
|
|
||||||
|
|
||||||
def OnBufferUnload( self, deleted_buffer_file ):
|
def OnBufferUnload( self, deleted_buffer_file ):
|
||||||
|
if not self._IsServerAlive():
|
||||||
|
return
|
||||||
SendEventNotificationAsync( 'BufferUnload',
|
SendEventNotificationAsync( 'BufferUnload',
|
||||||
{ 'unloaded_buffer': deleted_buffer_file } )
|
{ 'unloaded_buffer': deleted_buffer_file } )
|
||||||
|
|
||||||
|
|
||||||
def OnBufferVisit( self ):
|
def OnBufferVisit( self ):
|
||||||
|
if not self._IsServerAlive():
|
||||||
|
return
|
||||||
extra_data = {}
|
extra_data = {}
|
||||||
_AddUltiSnipsDataIfNeeded( extra_data )
|
_AddUltiSnipsDataIfNeeded( extra_data )
|
||||||
SendEventNotificationAsync( 'BufferVisit', extra_data )
|
SendEventNotificationAsync( 'BufferVisit', extra_data )
|
||||||
|
|
||||||
|
|
||||||
def OnInsertLeave( self ):
|
def OnInsertLeave( self ):
|
||||||
|
if not self._IsServerAlive():
|
||||||
|
return
|
||||||
SendEventNotificationAsync( 'InsertLeave' )
|
SendEventNotificationAsync( 'InsertLeave' )
|
||||||
|
|
||||||
|
|
||||||
def OnVimLeave( self ):
|
def OnVimLeave( self ):
|
||||||
|
if self._IsServerAlive():
|
||||||
self._server_popen.terminate()
|
self._server_popen.terminate()
|
||||||
os.remove( self._temp_options_filename )
|
os.remove( self._temp_options_filename )
|
||||||
|
|
||||||
|
|
||||||
def OnCurrentIdentifierFinished( self ):
|
def OnCurrentIdentifierFinished( self ):
|
||||||
|
if not self._IsServerAlive():
|
||||||
|
return
|
||||||
SendEventNotificationAsync( 'CurrentIdentifierFinished' )
|
SendEventNotificationAsync( 'CurrentIdentifierFinished' )
|
||||||
|
|
||||||
|
|
||||||
@ -200,6 +229,8 @@ class YouCompleteMe( object ):
|
|||||||
|
|
||||||
|
|
||||||
def ShowDetailedDiagnostic( self ):
|
def ShowDetailedDiagnostic( self ):
|
||||||
|
if not self._IsServerAlive():
|
||||||
|
return
|
||||||
debug_info = BaseRequest.PostDataToHandler( BuildRequestData(),
|
debug_info = BaseRequest.PostDataToHandler( BuildRequestData(),
|
||||||
'detailed_diagnostic' )
|
'detailed_diagnostic' )
|
||||||
if 'message' in debug_info:
|
if 'message' in debug_info:
|
||||||
@ -207,8 +238,11 @@ class YouCompleteMe( object ):
|
|||||||
|
|
||||||
|
|
||||||
def DebugInfo( self ):
|
def DebugInfo( self ):
|
||||||
|
if self._IsServerAlive():
|
||||||
debug_info = BaseRequest.PostDataToHandler( BuildRequestData(),
|
debug_info = BaseRequest.PostDataToHandler( BuildRequestData(),
|
||||||
'debug_info' )
|
'debug_info' )
|
||||||
|
else:
|
||||||
|
debug_info = 'Server crashed, no debug info from server'
|
||||||
debug_info += '\nServer running at: {0}'.format(
|
debug_info += '\nServer running at: {0}'.format(
|
||||||
BaseRequest.server_location )
|
BaseRequest.server_location )
|
||||||
if self._server_stderr or self._server_stdout:
|
if self._server_stderr or self._server_stdout:
|
||||||
|
Loading…
x
Reference in New Issue
Block a user