2013-09-20 17:24:34 -07:00
|
|
|
#!/usr/bin/env python
|
|
|
|
#
|
|
|
|
# Copyright (C) 2013 Strahinja Val Markovic <val@markovic.io>
|
|
|
|
#
|
|
|
|
# This file is part of YouCompleteMe.
|
|
|
|
#
|
|
|
|
# YouCompleteMe is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# YouCompleteMe is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with YouCompleteMe. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
import sys
|
|
|
|
import os
|
2013-09-23 14:33:14 -07:00
|
|
|
import atexit
|
2013-09-20 17:24:34 -07:00
|
|
|
|
|
|
|
# We want to have the YouCompleteMe/python directory on the Python PATH because
|
|
|
|
# all the code already assumes that it's there. This is a relic from before the
|
|
|
|
# client/server architecture.
|
|
|
|
# TODO: Fix things so that this is not needed anymore when we split ycmd into a
|
|
|
|
# separate repository.
|
|
|
|
sys.path.insert( 0, os.path.join(
|
|
|
|
os.path.dirname( os.path.abspath( __file__ ) ),
|
|
|
|
'../..' ) )
|
|
|
|
|
|
|
|
import logging
|
|
|
|
import time
|
|
|
|
import json
|
|
|
|
import bottle
|
|
|
|
from bottle import run, request, response
|
|
|
|
import server_state
|
|
|
|
from ycm import user_options_store
|
|
|
|
import argparse
|
|
|
|
|
|
|
|
# num bytes for the request body buffer; request.json only works if the request
|
|
|
|
# size is less than this
|
|
|
|
bottle.Request.MEMFILE_MAX = 300 * 1024
|
|
|
|
|
|
|
|
user_options_store.LoadDefaults()
|
|
|
|
SERVER_STATE = server_state.ServerState( user_options_store.GetAll() )
|
|
|
|
|
|
|
|
LOGGER = logging.getLogger( __name__ )
|
|
|
|
app = bottle.Bottle()
|
|
|
|
|
|
|
|
|
|
|
|
@app.post( '/event_notification' )
|
|
|
|
def EventNotification():
|
|
|
|
LOGGER.info( 'Received event notification')
|
|
|
|
request_data = request.json
|
|
|
|
event_name = request_data[ 'event_name' ]
|
|
|
|
LOGGER.debug( 'Event name: %s', event_name )
|
|
|
|
|
|
|
|
event_handler = 'On' + event_name
|
|
|
|
getattr( SERVER_STATE.GetGeneralCompleter(), event_handler )( request_data )
|
|
|
|
|
|
|
|
filetypes = request_data[ 'filetypes' ]
|
|
|
|
if SERVER_STATE.FiletypeCompletionUsable( filetypes ):
|
|
|
|
getattr( SERVER_STATE.GetFiletypeCompleter( filetypes ),
|
|
|
|
event_handler )( request_data )
|
|
|
|
|
|
|
|
|
|
|
|
@app.post( '/run_completer_command' )
|
|
|
|
def RunCompleterCommand():
|
|
|
|
LOGGER.info( 'Received command request')
|
|
|
|
request_data = request.json
|
|
|
|
completer_target = request_data[ 'completer_target' ]
|
|
|
|
|
|
|
|
if completer_target == 'identifier':
|
|
|
|
completer = SERVER_STATE.GetGeneralCompleter()
|
|
|
|
else:
|
|
|
|
completer = SERVER_STATE.GetFiletypeCompleter()
|
|
|
|
|
|
|
|
return _JsonResponse(
|
|
|
|
completer.OnUserCommand( request_data[ 'command_arguments' ],
|
|
|
|
request_data ) )
|
|
|
|
|
|
|
|
|
|
|
|
@app.post( '/get_completions' )
|
|
|
|
def GetCompletions():
|
|
|
|
LOGGER.info( 'Received completion request')
|
|
|
|
request_data = request.json
|
|
|
|
do_filetype_completion = SERVER_STATE.ShouldUseFiletypeCompleter(
|
|
|
|
request_data )
|
|
|
|
LOGGER.debug( 'Using filetype completion: %s', do_filetype_completion )
|
|
|
|
filetypes = request_data[ 'filetypes' ]
|
|
|
|
completer = ( SERVER_STATE.GetFiletypeCompleter( filetypes ) if
|
|
|
|
do_filetype_completion else
|
|
|
|
SERVER_STATE.GetGeneralCompleter() )
|
|
|
|
|
|
|
|
# This is necessary so that general_completer_store fills up
|
|
|
|
# _current_query_completers.
|
|
|
|
# TODO: Fix this.
|
|
|
|
completer.ShouldUseNow( request_data )
|
|
|
|
|
|
|
|
# TODO: This should not be async anymore, server is multi-threaded
|
|
|
|
completer.CandidatesForQueryAsync( request_data )
|
|
|
|
while not completer.AsyncCandidateRequestReady():
|
|
|
|
time.sleep( 0.03 )
|
|
|
|
return _JsonResponse( completer.CandidatesFromStoredRequest() )
|
|
|
|
|
|
|
|
|
2013-09-24 12:53:44 -07:00
|
|
|
@app.get( '/user_options' )
|
|
|
|
def GetUserOptions():
|
|
|
|
LOGGER.info( 'Received user options GET request')
|
|
|
|
return _JsonResponse( dict( SERVER_STATE.user_options ) )
|
|
|
|
|
|
|
|
|
|
|
|
@app.post( '/user_options' )
|
|
|
|
def SetUserOptions():
|
|
|
|
LOGGER.info( 'Received user options POST request')
|
2013-09-24 14:00:27 -07:00
|
|
|
_SetUserOptions( request.json )
|
2013-09-20 17:24:34 -07:00
|
|
|
|
|
|
|
|
|
|
|
@app.post( '/filetype_completion_available')
|
|
|
|
def FiletypeCompletionAvailable():
|
|
|
|
LOGGER.info( 'Received filetype completion available request')
|
|
|
|
return _JsonResponse( SERVER_STATE.FiletypeCompletionAvailable(
|
|
|
|
request.json[ 'filetypes' ] ) )
|
|
|
|
|
|
|
|
|
|
|
|
def _JsonResponse( data ):
|
|
|
|
response.set_header( 'Content-Type', 'application/json' )
|
|
|
|
return json.dumps( data )
|
|
|
|
|
|
|
|
|
2013-09-23 14:33:14 -07:00
|
|
|
@atexit.register
|
|
|
|
def _ServerShutdown():
|
|
|
|
SERVER_STATE.Shutdown()
|
2013-09-23 13:27:32 -07:00
|
|
|
|
|
|
|
|
2013-09-24 14:00:27 -07:00
|
|
|
def _SetUserOptions( options ):
|
|
|
|
global SERVER_STATE
|
|
|
|
|
|
|
|
SERVER_STATE = server_state.ServerState( options )
|
|
|
|
user_options_store.SetAll( options )
|
|
|
|
|
|
|
|
|
2013-09-20 17:24:34 -07:00
|
|
|
def Main():
|
|
|
|
global LOGGER
|
|
|
|
parser = argparse.ArgumentParser()
|
|
|
|
parser.add_argument( '--host', type = str, default = 'localhost',
|
|
|
|
help='server hostname')
|
|
|
|
parser.add_argument( '--port', type = int, default = 6666,
|
|
|
|
help='server port')
|
|
|
|
parser.add_argument( '--log', type = str, default = 'info',
|
|
|
|
help='log level, one of '
|
|
|
|
'[debug|info|warning|error|critical]')
|
2013-09-24 14:00:27 -07:00
|
|
|
parser.add_argument( '--options_file', type = str, default = '',
|
|
|
|
help='file with user options, in JSON format')
|
2013-09-20 17:24:34 -07:00
|
|
|
args = parser.parse_args()
|
|
|
|
|
2013-09-24 14:00:27 -07:00
|
|
|
if args.options_file:
|
|
|
|
_SetUserOptions( json.load( open( args.options_file, 'r' ) ) )
|
|
|
|
|
2013-09-20 17:24:34 -07:00
|
|
|
numeric_level = getattr( logging, args.log.upper(), None )
|
|
|
|
if not isinstance( numeric_level, int ):
|
|
|
|
raise ValueError( 'Invalid log level: %s' % args.log )
|
|
|
|
|
|
|
|
logging.basicConfig( format = '%(asctime)s - %(levelname)s - %(message)s',
|
|
|
|
level = numeric_level )
|
|
|
|
|
|
|
|
LOGGER = logging.getLogger( __name__ )
|
|
|
|
run( app = app, host = args.host, port = args.port, server='cherrypy' )
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
Main()
|
|
|
|
|