2015-10-08 11:36:34 +02:00
|
|
|
# Copyright (C) 2015 YouCompleteMe contributors.
|
|
|
|
#
|
|
|
|
# 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/>.
|
|
|
|
|
2016-02-27 16:12:24 -08:00
|
|
|
from __future__ import unicode_literals
|
|
|
|
from __future__ import print_function
|
|
|
|
from __future__ import division
|
|
|
|
from __future__ import absolute_import
|
|
|
|
from future import standard_library
|
|
|
|
standard_library.install_aliases()
|
|
|
|
from builtins import * # noqa
|
|
|
|
|
2015-10-08 11:36:34 +02:00
|
|
|
import os
|
2016-01-11 13:33:43 +01:00
|
|
|
import sys
|
2015-10-08 11:36:34 +02:00
|
|
|
import vim
|
|
|
|
import functools
|
2016-01-12 21:24:23 +01:00
|
|
|
import re
|
2015-10-08 11:36:34 +02:00
|
|
|
|
2016-02-28 14:39:55 -08:00
|
|
|
# Can't import these from setup.py because it makes nosetests go crazy.
|
2015-10-08 11:36:34 +02:00
|
|
|
DIR_OF_CURRENT_SCRIPT = os.path.dirname( os.path.abspath( __file__ ) )
|
2015-10-12 21:19:37 +02:00
|
|
|
DIR_OF_YCMD = os.path.join( DIR_OF_CURRENT_SCRIPT, '..', '..', 'third_party',
|
|
|
|
'ycmd' )
|
2016-01-11 13:33:43 +01:00
|
|
|
WIN_PYTHON_PATH = os.path.join( sys.exec_prefix, 'python.exe' )
|
2016-02-28 17:44:48 -08:00
|
|
|
PYTHON_BINARY_REGEX = re.compile(
|
2016-02-29 10:26:50 -08:00
|
|
|
r'python((2(\.[67])?)|(3(\.[3-9])?))?(.exe)?$' )
|
2015-10-08 11:36:34 +02:00
|
|
|
|
|
|
|
|
|
|
|
def Memoize( obj ):
|
|
|
|
cache = obj.cache = {}
|
|
|
|
|
|
|
|
@functools.wraps( obj )
|
|
|
|
def memoizer( *args, **kwargs ):
|
|
|
|
key = str( args ) + str( kwargs )
|
|
|
|
if key not in cache:
|
|
|
|
cache[ key ] = obj( *args, **kwargs )
|
|
|
|
return cache[ key ]
|
|
|
|
return memoizer
|
|
|
|
|
|
|
|
|
|
|
|
@Memoize
|
|
|
|
def PathToPythonInterpreter():
|
2015-10-12 21:19:37 +02:00
|
|
|
from ycmd import utils
|
|
|
|
|
2016-03-12 22:04:57 +01:00
|
|
|
python_interpreter = vim.eval( 'g:ycm_server_python_interpreter' )
|
2015-10-08 11:36:34 +02:00
|
|
|
|
2016-01-11 13:33:43 +01:00
|
|
|
if python_interpreter:
|
2016-01-23 16:58:24 +01:00
|
|
|
if IsPythonVersionCorrect( python_interpreter ):
|
|
|
|
return python_interpreter
|
2015-10-08 11:36:34 +02:00
|
|
|
|
2016-03-12 22:04:57 +01:00
|
|
|
raise RuntimeError( "Path in 'g:ycm_server_python_interpreter' option "
|
2016-02-28 17:44:48 -08:00
|
|
|
"does not point to a valid Python 2.6+ or 3.3+." )
|
2015-10-08 11:36:34 +02:00
|
|
|
|
2016-04-25 20:24:41 -07:00
|
|
|
python_interpreter = _PathToPythonUsedDuringBuild()
|
|
|
|
if IsPythonVersionCorrect( python_interpreter ):
|
|
|
|
return python_interpreter
|
|
|
|
|
2016-01-11 13:33:43 +01:00
|
|
|
# On UNIX platforms, we use sys.executable as the Python interpreter path.
|
|
|
|
# We cannot use sys.executable on Windows because for unknown reasons, it
|
|
|
|
# returns the Vim executable. Instead, we use sys.exec_prefix to deduce the
|
|
|
|
# interpreter path.
|
|
|
|
python_interpreter = ( WIN_PYTHON_PATH if utils.OnWindows() else
|
|
|
|
sys.executable )
|
2015-10-08 11:36:34 +02:00
|
|
|
|
2016-01-23 16:58:24 +01:00
|
|
|
if IsPythonVersionCorrect( python_interpreter ):
|
|
|
|
return python_interpreter
|
|
|
|
|
2016-02-29 10:26:50 -08:00
|
|
|
# As a last resort, we search python in the PATH. We prefer Python 2 over 3
|
|
|
|
# for the sake of backwards compatibility with ycm_extra_conf.py files out
|
|
|
|
# there; few people wrote theirs to work on py3.
|
|
|
|
# So we check 'python2' before 'python' because on some distributions (Arch
|
|
|
|
# Linux for example), python refers to python3.
|
2016-01-23 16:58:24 +01:00
|
|
|
python_interpreter = utils.PathToFirstExistingExecutable( [ 'python2',
|
2016-02-28 17:44:48 -08:00
|
|
|
'python',
|
|
|
|
'python3' ] )
|
2016-01-23 16:58:24 +01:00
|
|
|
|
|
|
|
if IsPythonVersionCorrect( python_interpreter ):
|
|
|
|
return python_interpreter
|
2015-10-08 11:36:34 +02:00
|
|
|
|
2016-02-28 17:44:48 -08:00
|
|
|
raise RuntimeError( "Cannot find Python 2.6+ or 3.3+. You can set its path "
|
2016-03-12 22:04:57 +01:00
|
|
|
"using the 'g:ycm_server_python_interpreter' "
|
2016-01-23 16:58:24 +01:00
|
|
|
"option." )
|
2016-01-11 13:33:43 +01:00
|
|
|
|
|
|
|
|
2016-04-25 20:24:41 -07:00
|
|
|
def _PathToPythonUsedDuringBuild():
|
|
|
|
from ycmd import utils
|
|
|
|
|
|
|
|
try:
|
|
|
|
filepath = os.path.join( DIR_OF_YCMD, 'PYTHON_USED_DURING_BUILDING' )
|
|
|
|
return utils.ReadFile( filepath ).strip()
|
|
|
|
# We need to check for IOError for Python2 and OSError for Python3
|
|
|
|
except ( IOError, OSError ):
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
2016-01-12 21:24:23 +01:00
|
|
|
def EndsWithPython( path ):
|
2016-02-28 17:44:48 -08:00
|
|
|
"""Check if given path ends with a python 2.6+ or 3.3+ name."""
|
2016-04-25 20:24:41 -07:00
|
|
|
return path and PYTHON_BINARY_REGEX.search( path ) is not None
|
2016-01-12 21:24:23 +01:00
|
|
|
|
|
|
|
|
|
|
|
def IsPythonVersionCorrect( path ):
|
2016-02-28 17:44:48 -08:00
|
|
|
"""Check if given path is the Python interpreter version 2.6+ or 3.3+."""
|
2015-10-12 21:19:37 +02:00
|
|
|
from ycmd import utils
|
|
|
|
|
2016-01-12 21:24:23 +01:00
|
|
|
if not EndsWithPython( path ):
|
|
|
|
return False
|
|
|
|
|
|
|
|
command = [ path,
|
2016-01-11 13:33:43 +01:00
|
|
|
'-c',
|
|
|
|
"import sys;"
|
|
|
|
"major, minor = sys.version_info[ :2 ];"
|
2016-02-28 17:44:48 -08:00
|
|
|
"good_python = ( major == 2 and minor >= 6 ) "
|
|
|
|
"or ( major == 3 and minor >= 3 ) or major > 3;"
|
|
|
|
# If this looks weird, remember that:
|
|
|
|
# int( True ) == 1
|
|
|
|
# int( False ) == 0
|
|
|
|
"sys.exit( not good_python )" ]
|
2016-01-11 13:33:43 +01:00
|
|
|
|
|
|
|
return utils.SafePopen( command ).wait() == 0
|
2015-10-08 11:36:34 +02:00
|
|
|
|
|
|
|
|
|
|
|
def PathToServerScript():
|
2015-10-12 21:19:37 +02:00
|
|
|
return os.path.join( DIR_OF_YCMD, 'ycmd' )
|