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/>.
|
|
|
|
|
2013-10-14 15:29:00 -07:00
|
|
|
from server_utils import SetUpPythonPath
|
|
|
|
SetUpPythonPath()
|
2013-10-01 16:21:17 -07:00
|
|
|
|
2013-10-14 15:29:00 -07:00
|
|
|
import sys
|
2013-09-20 17:24:34 -07:00
|
|
|
import logging
|
|
|
|
import json
|
2013-10-08 16:21:43 -07:00
|
|
|
import argparse
|
2013-10-11 20:12:04 -07:00
|
|
|
import waitress
|
2013-10-17 12:21:59 -07:00
|
|
|
import signal
|
2013-09-20 17:24:34 -07:00
|
|
|
from ycm import user_options_store
|
2013-10-08 16:21:43 -07:00
|
|
|
from ycm import extra_conf_store
|
2013-10-23 11:02:25 -07:00
|
|
|
from ycm.server.watchdog_plugin import WatchdogPlugin
|
2013-09-20 17:24:34 -07:00
|
|
|
|
2013-09-27 16:20:35 -07:00
|
|
|
|
2013-10-14 15:29:00 -07:00
|
|
|
def YcmCoreSanityCheck():
|
|
|
|
if 'ycm_core' in sys.modules:
|
|
|
|
raise RuntimeError( 'ycm_core already imported, ycmd has a bug!' )
|
2013-09-24 14:00:27 -07:00
|
|
|
|
|
|
|
|
2013-10-17 22:12:28 -07:00
|
|
|
# We manually call sys.exit() on SIGTERM and SIGINT so that atexit handlers are
|
|
|
|
# properly executed.
|
2013-10-17 12:21:59 -07:00
|
|
|
def SetUpSignalHandler():
|
2013-10-17 20:14:56 -07:00
|
|
|
def SignalHandler( signum, frame ):
|
2013-10-17 22:12:28 -07:00
|
|
|
sys.exit()
|
2013-10-17 12:21:59 -07:00
|
|
|
|
|
|
|
for sig in [ signal.SIGTERM,
|
2013-10-17 20:16:44 -07:00
|
|
|
signal.SIGINT ]:
|
2013-10-17 12:21:59 -07:00
|
|
|
signal.signal( sig, SignalHandler )
|
|
|
|
|
|
|
|
|
2013-09-20 17:24:34 -07:00
|
|
|
def Main():
|
|
|
|
parser = argparse.ArgumentParser()
|
|
|
|
parser.add_argument( '--host', type = str, default = 'localhost',
|
2013-09-25 11:12:34 -07:00
|
|
|
help = 'server hostname')
|
2013-09-20 17:24:34 -07:00
|
|
|
parser.add_argument( '--port', type = int, default = 6666,
|
2013-09-25 11:12:34 -07:00
|
|
|
help = 'server port')
|
2013-09-20 17:24:34 -07:00
|
|
|
parser.add_argument( '--log', type = str, default = 'info',
|
2013-09-25 11:12:34 -07:00
|
|
|
help = 'log level, one of '
|
|
|
|
'[debug|info|warning|error|critical]' )
|
2013-10-23 12:33:27 -07:00
|
|
|
parser.add_argument( '--idle_suicide_seconds', type = int, default = 0,
|
2013-10-23 11:02:25 -07:00
|
|
|
help = 'num idle seconds before server shuts down')
|
2013-09-24 14:00:27 -07:00
|
|
|
parser.add_argument( '--options_file', type = str, default = '',
|
2013-09-25 11:12:34 -07:00
|
|
|
help = 'file with user options, in JSON format' )
|
2013-09-20 17:24:34 -07:00
|
|
|
args = parser.parse_args()
|
|
|
|
|
|
|
|
numeric_level = getattr( logging, args.log.upper(), None )
|
|
|
|
if not isinstance( numeric_level, int ):
|
|
|
|
raise ValueError( 'Invalid log level: %s' % args.log )
|
|
|
|
|
2013-10-14 15:29:00 -07:00
|
|
|
# Has to be called before any call to logging.getLogger()
|
2013-09-20 17:24:34 -07:00
|
|
|
logging.basicConfig( format = '%(asctime)s - %(levelname)s - %(message)s',
|
|
|
|
level = numeric_level )
|
|
|
|
|
2013-10-14 15:29:00 -07:00
|
|
|
options = None
|
|
|
|
if args.options_file:
|
|
|
|
options = json.load( open( args.options_file, 'r' ) )
|
|
|
|
user_options_store.SetAll( options )
|
2013-10-15 11:19:56 -07:00
|
|
|
# This ensures that ycm_core is not loaded before extra conf
|
|
|
|
# preload was run.
|
2013-10-14 15:29:00 -07:00
|
|
|
YcmCoreSanityCheck()
|
|
|
|
extra_conf_store.CallGlobalExtraConfYcmCorePreloadIfExists()
|
|
|
|
|
2013-10-15 11:19:56 -07:00
|
|
|
# This can't be a top-level import because it transitively imports
|
|
|
|
# ycm_core which we want to be imported ONLY after extra conf
|
|
|
|
# preload has executed.
|
2013-10-23 11:02:25 -07:00
|
|
|
from ycm.server import handlers
|
2013-10-14 15:29:00 -07:00
|
|
|
handlers.UpdateUserOptions( options )
|
2013-10-17 12:21:59 -07:00
|
|
|
SetUpSignalHandler()
|
2013-10-23 12:33:27 -07:00
|
|
|
handlers.app.install( WatchdogPlugin( args.idle_suicide_seconds ) )
|
2013-10-14 15:29:00 -07:00
|
|
|
waitress.serve( handlers.app,
|
|
|
|
host = args.host,
|
|
|
|
port = args.port,
|
|
|
|
threads = 10 )
|
2013-09-20 17:24:34 -07:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
Main()
|
|
|
|
|