2012-08-04 17:46:54 -07:00
|
|
|
#!/usr/bin/env python
|
|
|
|
#
|
2014-01-13 11:08:43 -08:00
|
|
|
# Copyright (C) 2011, 2012 Google Inc.
|
2012-08-04 17:46:54 -07:00
|
|
|
#
|
|
|
|
# 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-09-20 17:24:34 -07:00
|
|
|
import tempfile
|
|
|
|
import os
|
2013-09-23 13:27:32 -07:00
|
|
|
import sys
|
|
|
|
import signal
|
2013-10-03 10:14:31 -07:00
|
|
|
import functools
|
2013-10-14 20:38:45 -07:00
|
|
|
import socket
|
2013-10-24 10:26:55 -07:00
|
|
|
import stat
|
2014-03-19 13:16:49 -07:00
|
|
|
import json
|
2013-10-24 20:21:03 -07:00
|
|
|
from distutils.spawn import find_executable
|
2013-12-11 13:41:04 +01:00
|
|
|
import subprocess
|
2014-03-19 15:13:35 -07:00
|
|
|
import collections
|
2013-09-20 17:24:34 -07:00
|
|
|
|
2013-10-11 11:11:02 -07:00
|
|
|
WIN_PYTHON27_PATH = 'C:\python27\pythonw.exe'
|
|
|
|
WIN_PYTHON26_PATH = 'C:\python26\pythonw.exe'
|
|
|
|
|
|
|
|
|
2012-08-04 17:46:54 -07:00
|
|
|
def IsIdentifierChar( char ):
|
|
|
|
return char.isalnum() or char == '_'
|
|
|
|
|
|
|
|
|
|
|
|
def SanitizeQuery( query ):
|
|
|
|
return query.strip()
|
2013-04-22 22:19:26 -07:00
|
|
|
|
2013-09-07 11:58:42 -07:00
|
|
|
|
2014-03-19 15:13:35 -07:00
|
|
|
# Given an object, returns a str object that's utf-8 encoded.
|
2014-01-13 10:00:05 -08:00
|
|
|
def ToUtf8IfNeeded( value ):
|
|
|
|
if isinstance( value, unicode ):
|
|
|
|
return value.encode( 'utf8' )
|
|
|
|
if isinstance( value, str ):
|
|
|
|
return value
|
|
|
|
return str( value )
|
2013-09-20 17:24:34 -07:00
|
|
|
|
|
|
|
|
2014-03-19 15:13:35 -07:00
|
|
|
# Recurses through the object if it's a dict/iterable and converts all the
|
|
|
|
# unicode objects to utf-8 strings.
|
|
|
|
def RecursiveEncodeUnicodeToUtf8( value ):
|
|
|
|
if isinstance( value, unicode ):
|
|
|
|
return value.encode( 'utf8' )
|
|
|
|
if isinstance( value, str ):
|
|
|
|
return value
|
|
|
|
elif isinstance( value, collections.Mapping ):
|
|
|
|
return dict( map( RecursiveEncodeUnicodeToUtf8, value.iteritems() ) )
|
|
|
|
elif isinstance( value, collections.Iterable ):
|
|
|
|
return type( value )( map( RecursiveEncodeUnicodeToUtf8, value ) )
|
|
|
|
else:
|
|
|
|
return value
|
|
|
|
|
|
|
|
|
2014-03-19 13:16:49 -07:00
|
|
|
def ToUtf8Json( data ):
|
2014-03-19 15:13:35 -07:00
|
|
|
return json.dumps( RecursiveEncodeUnicodeToUtf8( data ),
|
|
|
|
ensure_ascii = False,
|
|
|
|
# This is the encoding of INPUT str data
|
|
|
|
encoding = 'utf-8' )
|
2014-03-19 13:16:49 -07:00
|
|
|
|
|
|
|
|
2013-09-20 17:24:34 -07:00
|
|
|
def PathToTempDir():
|
2013-10-06 18:27:47 -07:00
|
|
|
tempdir = os.path.join( tempfile.gettempdir(), 'ycm_temp' )
|
|
|
|
if not os.path.exists( tempdir ):
|
|
|
|
os.makedirs( tempdir )
|
2013-10-24 10:26:55 -07:00
|
|
|
# Needed to support multiple users working on the same machine;
|
|
|
|
# see issue 606.
|
|
|
|
MakeFolderAccessibleToAll( tempdir )
|
2013-10-06 18:27:47 -07:00
|
|
|
return tempdir
|
2013-09-23 13:27:32 -07:00
|
|
|
|
|
|
|
|
2013-10-24 10:26:55 -07:00
|
|
|
def MakeFolderAccessibleToAll( path_to_folder ):
|
|
|
|
current_stat = os.stat( path_to_folder )
|
|
|
|
# readable, writable and executable by everyone
|
|
|
|
flags = ( current_stat.st_mode | stat.S_IROTH | stat.S_IWOTH | stat.S_IXOTH
|
|
|
|
| stat.S_IRGRP | stat.S_IWGRP | stat.S_IXGRP )
|
|
|
|
os.chmod( path_to_folder, flags )
|
|
|
|
|
|
|
|
|
2013-10-28 12:17:18 -07:00
|
|
|
def RunningInsideVim():
|
|
|
|
try:
|
|
|
|
import vim # NOQA
|
|
|
|
return True
|
|
|
|
except ImportError:
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
2013-10-14 20:38:45 -07:00
|
|
|
def GetUnusedLocalhostPort():
|
|
|
|
sock = socket.socket()
|
|
|
|
# This tells the OS to give us any free port in the range [1024 - 65535]
|
|
|
|
sock.bind( ( '', 0 ) )
|
|
|
|
port = sock.getsockname()[ 1 ]
|
|
|
|
sock.close()
|
|
|
|
return port
|
|
|
|
|
|
|
|
|
2013-12-20 13:01:48 -08:00
|
|
|
def RemoveIfExists( filename ):
|
|
|
|
try:
|
|
|
|
os.remove( filename )
|
|
|
|
except OSError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2013-11-14 10:50:05 -08: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
|
2013-10-11 11:11:02 -07:00
|
|
|
def PathToPythonInterpreter():
|
2013-11-14 10:50:05 -08:00
|
|
|
if not RunningInsideVim():
|
|
|
|
return sys.executable
|
|
|
|
|
|
|
|
import vim # NOQA
|
|
|
|
user_path_to_python = vim.eval( 'g:ycm_path_to_python_interpreter' )
|
|
|
|
if user_path_to_python:
|
|
|
|
return user_path_to_python
|
|
|
|
|
2013-10-24 20:21:03 -07:00
|
|
|
# We check for 'python2' before 'python' because some OS's (I'm looking at you
|
|
|
|
# Arch Linux) have made the... interesting decision to point /usr/bin/python
|
|
|
|
# to python3.
|
2013-11-05 15:30:34 -08:00
|
|
|
python_names = [ 'python2', 'python' ]
|
|
|
|
if OnWindows():
|
|
|
|
# On Windows, 'pythonw' doesn't pop-up a console window like running
|
|
|
|
# 'python' does.
|
|
|
|
python_names.insert( 0, 'pythonw' )
|
|
|
|
|
|
|
|
path_to_python = PathToFirstExistingExecutable( python_names )
|
2013-11-01 10:53:16 -07:00
|
|
|
if path_to_python:
|
|
|
|
return path_to_python
|
|
|
|
|
|
|
|
# On Windows, Python may not be on the PATH at all, so we check some common
|
|
|
|
# install locations.
|
|
|
|
if OnWindows():
|
|
|
|
if os.path.exists( WIN_PYTHON27_PATH ):
|
|
|
|
return WIN_PYTHON27_PATH
|
|
|
|
elif os.path.exists( WIN_PYTHON26_PATH ):
|
|
|
|
return WIN_PYTHON26_PATH
|
|
|
|
raise RuntimeError( 'Python 2.7/2.6 not installed!' )
|
2013-10-24 20:21:03 -07:00
|
|
|
|
|
|
|
|
2013-10-24 20:24:08 -07:00
|
|
|
def PathToFirstExistingExecutable( executable_name_list ):
|
2013-10-24 20:21:03 -07:00
|
|
|
for executable_name in executable_name_list:
|
|
|
|
path = find_executable( executable_name )
|
|
|
|
if path:
|
|
|
|
return path
|
|
|
|
return None
|
2013-10-11 11:11:02 -07:00
|
|
|
|
|
|
|
|
|
|
|
def OnWindows():
|
|
|
|
return sys.platform == 'win32'
|
|
|
|
|
|
|
|
|
2014-02-26 12:54:14 -07:00
|
|
|
def OnCygwin():
|
|
|
|
return sys.platform == 'cygwin'
|
|
|
|
|
|
|
|
|
2013-09-23 13:27:32 -07:00
|
|
|
# From here: http://stackoverflow.com/a/8536476/1672783
|
|
|
|
def TerminateProcess( pid ):
|
2013-10-11 11:11:02 -07:00
|
|
|
if OnWindows():
|
2013-09-23 13:27:32 -07:00
|
|
|
import ctypes
|
|
|
|
PROCESS_TERMINATE = 1
|
|
|
|
handle = ctypes.windll.kernel32.OpenProcess( PROCESS_TERMINATE,
|
|
|
|
False,
|
|
|
|
pid )
|
|
|
|
ctypes.windll.kernel32.TerminateProcess( handle, -1 )
|
|
|
|
ctypes.windll.kernel32.CloseHandle( handle )
|
|
|
|
else:
|
|
|
|
os.kill( pid, signal.SIGTERM )
|
2013-10-01 16:21:17 -07:00
|
|
|
|
|
|
|
|
|
|
|
def AddThirdPartyFoldersToSysPath():
|
|
|
|
path_to_third_party = os.path.join(
|
|
|
|
os.path.dirname( os.path.abspath( __file__ ) ),
|
|
|
|
'../../third_party' )
|
|
|
|
|
|
|
|
for folder in os.listdir( path_to_third_party ):
|
|
|
|
sys.path.insert( 0, os.path.realpath( os.path.join( path_to_third_party,
|
|
|
|
folder ) ) )
|
|
|
|
|
2013-10-07 13:09:34 -07:00
|
|
|
def ForceSemanticCompletion( request_data ):
|
|
|
|
return ( 'force_semantic' in request_data and
|
|
|
|
bool( request_data[ 'force_semantic' ] ) )
|
2013-12-11 13:41:04 +01:00
|
|
|
|
2013-12-24 18:53:23 -08:00
|
|
|
|
|
|
|
# A wrapper for subprocess.Popen that works around a Popen bug on Windows.
|
|
|
|
def SafePopen( *args, **kwargs ):
|
2014-01-14 23:22:48 +08:00
|
|
|
if kwargs.get( 'stdin' ) is None:
|
2013-12-11 13:41:04 +01:00
|
|
|
# We need this on Windows otherwise bad things happen. See issue #637.
|
2013-12-24 18:53:23 -08:00
|
|
|
kwargs[ 'stdin' ] = subprocess.PIPE if OnWindows() else None
|
2013-12-11 13:41:04 +01:00
|
|
|
|
2013-12-24 18:53:23 -08:00
|
|
|
return subprocess.Popen( *args, **kwargs )
|
2013-12-11 13:41:04 +01:00
|
|
|
|
|
|
|
|