-
Notifications
You must be signed in to change notification settings - Fork 768
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
6 changed files
with
368 additions
and
66 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,138 @@ | ||
# Copyright (C) 2017 Davit Samvelyan [email protected] | ||
# Synopsys. | ||
# | ||
# This file is part of ycmd. | ||
# | ||
# ycmd 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. | ||
# | ||
# ycmd 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 ycmd. If not, see <http://www.gnu.org/licenses/>. | ||
|
||
from __future__ import unicode_literals | ||
from __future__ import print_function | ||
from __future__ import division | ||
from __future__ import absolute_import | ||
# Not installing aliases from python-future; it's unreliable and slow. | ||
from builtins import * # noqa | ||
|
||
from future.utils import iteritems | ||
|
||
import os | ||
import threading | ||
|
||
from collections import defaultdict | ||
from ycmd import responses | ||
from ycmd.completers.general.filename_completer import ( GetPathType, | ||
GetPathTypeName ) | ||
|
||
|
||
class IncludeEntry( object ): | ||
""" Represents single include completion candidate. | ||
name is the name/string of the completion candidate, | ||
entry_type is an integer indicating whether the candidate is a | ||
'File', 'Dir' or both (See EXTRA_INFO_MAP in filename_completer). """ | ||
|
||
def __init__( self, name, entry_type ): | ||
self.name = name | ||
self.entry_type = entry_type | ||
|
||
|
||
class IncludeList( object ): | ||
""" | ||
Helper class for combining include completion candidates from | ||
several include paths. | ||
self._includes is a dictionary whose keys are | ||
IncludeEntry `name`s and values are IncludeEntry `entry_type`s. | ||
""" | ||
|
||
def __init__( self ): | ||
self._includes = defaultdict( int ) | ||
|
||
|
||
def AddIncludes( self, includes ): | ||
for include in includes: | ||
self._includes[ include.name ] |= include.entry_type | ||
|
||
|
||
def GetIncludes( self ): | ||
includes = [] | ||
for name, include_type in iteritems( self._includes ): | ||
includes.append( responses.BuildCompletionData( | ||
name, GetPathTypeName( include_type ) ) ) | ||
return includes | ||
|
||
|
||
class IncludeCache( object ): | ||
""" | ||
Holds a dictionary representing the include path cache. | ||
Dictionary keys are the include path directories. | ||
Dictionary values are tuples whose first object | ||
represents `mtime` of the dictionary key and the other | ||
object is an IncludeList. | ||
""" | ||
|
||
def __init__( self ): | ||
self._cache = {} | ||
self._cache_lock = threading.Lock() | ||
|
||
|
||
def GetIncludes( self, path ): | ||
includes = self._GetCached( path ) | ||
|
||
if includes is None: | ||
includes = self._ListIncludes( path ) | ||
self._AddToCache( path, includes ) | ||
|
||
return includes | ||
|
||
|
||
def _AddToCache( self, path, includes, mtime = None ): | ||
if not mtime: | ||
mtime = _GetModificationTime( path ) | ||
if mtime: | ||
with self._cache_lock: | ||
self._cache[ path ] = { 'mtime': mtime, 'includes': includes } | ||
|
||
|
||
def _GetCached( self, path ): | ||
includes = None | ||
with self._cache_lock: | ||
cache_entry = self._cache.get( path ) | ||
if cache_entry: | ||
mtime = _GetModificationTime( path ) | ||
if mtime > cache_entry[ 'mtime' ]: | ||
self._AddToCache( path, self._ListIncludes( path ), mtime ) | ||
else: | ||
includes = cache_entry[ 'includes' ] | ||
|
||
return includes | ||
|
||
|
||
def _ListIncludes( self, path ): | ||
try: | ||
names = os.listdir( path ) | ||
except OSError: | ||
return [] | ||
|
||
includes = [] | ||
for name in names: | ||
inc_path = os.path.join( path, name ) | ||
entry_type = GetPathType( inc_path ) | ||
includes.append( IncludeEntry( name, entry_type ) ) | ||
|
||
return includes | ||
|
||
|
||
def _GetModificationTime( path ): | ||
try: | ||
return os.path.getmtime( path ) | ||
except OSError: | ||
return 0 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.