2014-01-13 11:08:43 -08:00
|
|
|
# Copyright (C) 2013 Google Inc.
|
2014-01-04 14:28:27 -08: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/>.
|
|
|
|
|
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
|
|
|
|
|
|
|
|
from future.utils import itervalues, iteritems
|
2014-09-14 01:15:40 +07:00
|
|
|
from collections import defaultdict, namedtuple
|
2014-01-04 14:28:27 -08:00
|
|
|
from ycm import vimsupport
|
2016-10-10 14:08:51 -04:00
|
|
|
from ycm.diagnostic_filter import DiagnosticFilter
|
2014-01-04 16:45:34 -08:00
|
|
|
import vim
|
2014-01-04 14:28:27 -08:00
|
|
|
|
|
|
|
|
|
|
|
class DiagnosticInterface( object ):
|
2014-01-08 18:43:17 -08:00
|
|
|
def __init__( self, user_options ):
|
|
|
|
self._user_options = user_options
|
2014-01-04 16:45:34 -08:00
|
|
|
# Line and column numbers are 1-based
|
|
|
|
self._buffer_number_to_line_to_diags = defaultdict(
|
|
|
|
lambda: defaultdict( list ) )
|
2014-01-04 14:28:27 -08:00
|
|
|
self._next_sign_id = 1
|
2014-01-04 16:45:34 -08:00
|
|
|
self._previous_line_number = -1
|
2014-01-10 12:18:24 -08:00
|
|
|
self._diag_message_needs_clearing = False
|
2014-09-14 01:15:40 +07:00
|
|
|
self._placed_signs = []
|
2014-01-04 16:45:34 -08:00
|
|
|
|
|
|
|
|
|
|
|
def OnCursorMoved( self ):
|
|
|
|
line, _ = vimsupport.CurrentLineAndColumn()
|
|
|
|
line += 1 # Convert to 1-based
|
|
|
|
if line != self._previous_line_number:
|
|
|
|
self._previous_line_number = line
|
2014-01-08 18:43:17 -08:00
|
|
|
|
|
|
|
if self._user_options[ 'echo_current_diagnostic' ]:
|
|
|
|
self._EchoDiagnosticForLine( line )
|
2014-01-04 14:28:27 -08:00
|
|
|
|
2015-12-04 14:45:11 -08:00
|
|
|
|
|
|
|
def GetErrorCount( self ):
|
2015-12-04 15:59:18 -08:00
|
|
|
return len( self._FilterDiagnostics( _DiagnosticIsError ) )
|
2015-12-04 14:45:11 -08:00
|
|
|
|
|
|
|
|
|
|
|
def GetWarningCount( self ):
|
2015-12-04 15:59:18 -08:00
|
|
|
return len( self._FilterDiagnostics( _DiagnosticIsWarning ) )
|
2015-12-04 14:45:11 -08:00
|
|
|
|
|
|
|
|
2016-01-11 23:16:49 +04:00
|
|
|
def PopulateLocationList( self, diags ):
|
|
|
|
vimsupport.SetLocationList(
|
|
|
|
vimsupport.ConvertDiagnosticsToQfList( diags ) )
|
|
|
|
|
|
|
|
|
2014-01-04 14:28:27 -08:00
|
|
|
def UpdateWithNewDiagnostics( self, diags ):
|
2016-10-10 14:08:51 -04:00
|
|
|
diag_filter = DiagnosticFilter.from_filetype( self._user_options, vimsupport.CurrentFiletypes() )
|
|
|
|
normalized_diags = [ _NormalizeDiagnostic( x ) for x in diags if diag_filter.Accept(x) ]
|
2014-10-29 10:16:38 -07:00
|
|
|
self._buffer_number_to_line_to_diags = _ConvertDiagListToDict(
|
|
|
|
normalized_diags )
|
2014-01-08 18:43:17 -08:00
|
|
|
|
2014-01-08 19:43:21 -08:00
|
|
|
if self._user_options[ 'enable_diagnostic_signs' ]:
|
2014-09-14 01:15:40 +07:00
|
|
|
self._placed_signs, self._next_sign_id = _UpdateSigns(
|
|
|
|
self._placed_signs,
|
|
|
|
self._buffer_number_to_line_to_diags,
|
|
|
|
self._next_sign_id )
|
2014-01-08 18:43:17 -08:00
|
|
|
|
2014-01-08 19:43:21 -08:00
|
|
|
if self._user_options[ 'enable_diagnostic_highlighting' ]:
|
2014-01-08 18:43:17 -08:00
|
|
|
_UpdateSquiggles( self._buffer_number_to_line_to_diags )
|
2014-01-04 16:45:34 -08:00
|
|
|
|
2014-01-08 19:43:21 -08:00
|
|
|
if self._user_options[ 'always_populate_location_list' ]:
|
2016-01-11 23:16:49 +04:00
|
|
|
self.PopulateLocationList( normalized_diags )
|
2014-01-08 19:09:40 -08:00
|
|
|
|
2014-01-04 16:45:34 -08:00
|
|
|
def _EchoDiagnosticForLine( self, line_num ):
|
|
|
|
buffer_num = vim.current.buffer.number
|
|
|
|
diags = self._buffer_number_to_line_to_diags[ buffer_num ][ line_num ]
|
|
|
|
if not diags:
|
2014-01-10 12:18:24 -08:00
|
|
|
if self._diag_message_needs_clearing:
|
|
|
|
# Clear any previous diag echo
|
2016-08-28 08:34:09 +02:00
|
|
|
vimsupport.PostVimMessage( '', warning = False )
|
2014-01-10 12:18:24 -08:00
|
|
|
self._diag_message_needs_clearing = False
|
2014-01-04 16:45:34 -08:00
|
|
|
return
|
2015-08-05 22:09:07 +01:00
|
|
|
|
|
|
|
text = diags[ 0 ][ 'text' ]
|
|
|
|
if diags[ 0 ].get( 'fixit_available', False ):
|
|
|
|
text += ' (FixIt)'
|
|
|
|
|
2016-08-28 08:34:09 +02:00
|
|
|
vimsupport.PostVimMessage( text, warning = False, truncate = True )
|
2014-01-10 12:18:24 -08:00
|
|
|
self._diag_message_needs_clearing = True
|
2014-01-04 16:45:34 -08:00
|
|
|
|
|
|
|
|
2015-12-04 15:59:18 -08:00
|
|
|
def _FilterDiagnostics( self, predicate ):
|
|
|
|
matched_diags = []
|
|
|
|
line_to_diags = self._buffer_number_to_line_to_diags[
|
|
|
|
vim.current.buffer.number ]
|
|
|
|
|
2016-02-27 16:12:24 -08:00
|
|
|
for diags in itervalues( line_to_diags ):
|
|
|
|
matched_diags.extend( list( filter( predicate, diags ) ) )
|
2015-12-04 15:59:18 -08:00
|
|
|
return matched_diags
|
|
|
|
|
|
|
|
|
2014-01-04 18:32:42 -08:00
|
|
|
def _UpdateSquiggles( buffer_number_to_line_to_diags ):
|
|
|
|
vimsupport.ClearYcmSyntaxMatches()
|
|
|
|
line_to_diags = buffer_number_to_line_to_diags[ vim.current.buffer.number ]
|
|
|
|
|
2016-02-27 16:12:24 -08:00
|
|
|
for diags in itervalues( line_to_diags ):
|
2014-01-04 18:32:42 -08:00
|
|
|
for diag in diags:
|
2014-01-05 16:09:40 -08:00
|
|
|
location_extent = diag[ 'location_extent' ]
|
2014-01-05 13:30:22 -08:00
|
|
|
is_error = _DiagnosticIsError( diag )
|
|
|
|
|
2014-01-09 15:48:48 -08:00
|
|
|
if location_extent[ 'start' ][ 'line_num' ] < 0:
|
|
|
|
location = diag[ 'location' ]
|
|
|
|
vimsupport.AddDiagnosticSyntaxMatch(
|
2014-05-09 15:57:04 -07:00
|
|
|
location[ 'line_num' ],
|
|
|
|
location[ 'column_num' ] )
|
2014-01-09 15:48:48 -08:00
|
|
|
else:
|
|
|
|
vimsupport.AddDiagnosticSyntaxMatch(
|
2014-05-09 15:57:04 -07:00
|
|
|
location_extent[ 'start' ][ 'line_num' ],
|
|
|
|
location_extent[ 'start' ][ 'column_num' ],
|
|
|
|
location_extent[ 'end' ][ 'line_num' ],
|
|
|
|
location_extent[ 'end' ][ 'column_num' ],
|
2014-01-09 15:48:48 -08:00
|
|
|
is_error = is_error )
|
2014-01-05 13:30:22 -08:00
|
|
|
|
|
|
|
for diag_range in diag[ 'ranges' ]:
|
|
|
|
vimsupport.AddDiagnosticSyntaxMatch(
|
2014-05-09 15:57:04 -07:00
|
|
|
diag_range[ 'start' ][ 'line_num' ],
|
|
|
|
diag_range[ 'start' ][ 'column_num' ],
|
|
|
|
diag_range[ 'end' ][ 'line_num' ],
|
|
|
|
diag_range[ 'end' ][ 'column_num' ],
|
2014-01-05 13:30:22 -08:00
|
|
|
is_error = is_error )
|
2014-01-04 18:32:42 -08:00
|
|
|
|
|
|
|
|
2014-09-14 01:15:40 +07:00
|
|
|
def _UpdateSigns( placed_signs, buffer_number_to_line_to_diags, next_sign_id ):
|
|
|
|
new_signs, kept_signs, next_sign_id = _GetKeptAndNewSigns(
|
|
|
|
placed_signs, buffer_number_to_line_to_diags, next_sign_id
|
|
|
|
)
|
2014-09-19 12:30:15 -07:00
|
|
|
# Dummy sign used to prevent "flickering" in Vim when last mark gets
|
|
|
|
# deleted from buffer. Dummy sign prevents Vim to collapsing the sign column
|
|
|
|
# in that case.
|
|
|
|
# There's also a vim bug which causes the whole window to redraw in some
|
|
|
|
# conditions (vim redraw logic is very complex). But, somehow, if we place a
|
|
|
|
# dummy sign before placing other "real" signs, it will not redraw the
|
2014-09-14 01:15:40 +07:00
|
|
|
# buffer (patch to vim pending).
|
2014-09-19 12:30:15 -07:00
|
|
|
dummy_sign_needed = not kept_signs and new_signs
|
2014-09-14 01:15:40 +07:00
|
|
|
|
|
|
|
if dummy_sign_needed:
|
2014-09-19 12:30:15 -07:00
|
|
|
vimsupport.PlaceDummySign( next_sign_id + 1,
|
|
|
|
vim.current.buffer.number,
|
|
|
|
new_signs[ 0 ].line )
|
2014-09-14 01:15:40 +07:00
|
|
|
|
2014-09-19 12:30:15 -07:00
|
|
|
# We place only those signs that haven't been placed yet.
|
2014-09-14 01:15:40 +07:00
|
|
|
new_placed_signs = _PlaceNewSigns( kept_signs, new_signs )
|
|
|
|
|
2014-09-19 12:30:15 -07:00
|
|
|
# We use incremental placement, so signs that already placed on the correct
|
|
|
|
# lines will not be deleted and placed again, which should improve performance
|
|
|
|
# in case of many diags. Signs which don't exist in the current diag should be
|
|
|
|
# deleted.
|
|
|
|
_UnplaceObsoleteSigns( kept_signs, placed_signs )
|
2014-09-14 01:15:40 +07:00
|
|
|
|
|
|
|
if dummy_sign_needed:
|
|
|
|
vimsupport.UnPlaceDummySign( next_sign_id + 1, vim.current.buffer.number )
|
|
|
|
|
|
|
|
return new_placed_signs, next_sign_id
|
|
|
|
|
2014-09-19 12:30:15 -07:00
|
|
|
|
2014-09-14 01:15:40 +07:00
|
|
|
def _GetKeptAndNewSigns( placed_signs, buffer_number_to_line_to_diags,
|
|
|
|
next_sign_id ):
|
|
|
|
new_signs = []
|
|
|
|
kept_signs = []
|
2016-03-06 17:13:43 +00:00
|
|
|
for buffer_number, line_to_diags in iteritems(
|
|
|
|
buffer_number_to_line_to_diags ):
|
2014-01-10 14:39:52 -08:00
|
|
|
if not vimsupport.BufferIsVisible( buffer_number ):
|
|
|
|
continue
|
|
|
|
|
2016-02-27 16:12:24 -08:00
|
|
|
for line, diags in iteritems( line_to_diags ):
|
2014-01-10 14:39:52 -08:00
|
|
|
for diag in diags:
|
2014-09-14 01:15:40 +07:00
|
|
|
sign = _DiagSignPlacement( next_sign_id,
|
|
|
|
line,
|
|
|
|
buffer_number,
|
|
|
|
_DiagnosticIsError( diag ) )
|
|
|
|
if sign not in placed_signs:
|
|
|
|
new_signs += [ sign ]
|
|
|
|
next_sign_id += 1
|
|
|
|
else:
|
2014-09-19 12:30:15 -07:00
|
|
|
# We use .index here because `sign` contains a new id, but
|
|
|
|
# we need the sign with the old id to unplace it later on.
|
|
|
|
# We won't be placing the new sign.
|
2014-09-14 01:15:40 +07:00
|
|
|
kept_signs += [ placed_signs[ placed_signs.index( sign ) ] ]
|
|
|
|
return new_signs, kept_signs, next_sign_id
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _PlaceNewSigns( kept_signs, new_signs ):
|
2014-10-29 10:02:41 -07:00
|
|
|
placed_signs = kept_signs[:]
|
2014-09-14 01:15:40 +07:00
|
|
|
for sign in new_signs:
|
2014-09-19 12:30:15 -07:00
|
|
|
# Do not set two signs on the same line, it will screw up storing sign
|
|
|
|
# locations.
|
2014-09-14 01:15:40 +07:00
|
|
|
if sign in placed_signs:
|
|
|
|
continue
|
2014-09-19 12:30:15 -07:00
|
|
|
vimsupport.PlaceSign( sign.id, sign.line, sign.buffer, sign.is_error )
|
2014-10-29 10:02:41 -07:00
|
|
|
placed_signs.append(sign)
|
2014-09-14 01:15:40 +07:00
|
|
|
return placed_signs
|
|
|
|
|
|
|
|
|
2014-09-19 12:30:15 -07:00
|
|
|
def _UnplaceObsoleteSigns( kept_signs, placed_signs ):
|
2014-09-14 01:15:40 +07:00
|
|
|
for sign in placed_signs:
|
|
|
|
if sign not in kept_signs:
|
|
|
|
vimsupport.UnplaceSignInBuffer( sign.buffer, sign.id )
|
2014-01-10 14:39:52 -08:00
|
|
|
|
|
|
|
|
2014-01-04 16:45:34 -08:00
|
|
|
def _ConvertDiagListToDict( diag_list ):
|
|
|
|
buffer_to_line_to_diags = defaultdict( lambda: defaultdict( list ) )
|
|
|
|
for diag in diag_list:
|
2014-01-05 12:58:13 -08:00
|
|
|
location = diag[ 'location' ]
|
|
|
|
buffer_number = vimsupport.GetBufferNumberForFilename(
|
|
|
|
location[ 'filepath' ] )
|
2014-05-09 15:57:04 -07:00
|
|
|
line_number = location[ 'line_num' ]
|
2014-01-05 12:58:13 -08:00
|
|
|
buffer_to_line_to_diags[ buffer_number ][ line_number ].append( diag )
|
|
|
|
|
2016-02-27 16:12:24 -08:00
|
|
|
for line_to_diags in itervalues( buffer_to_line_to_diags ):
|
|
|
|
for diags in itervalues( line_to_diags ):
|
2014-01-04 16:45:34 -08:00
|
|
|
# We also want errors to be listed before warnings so that errors aren't
|
|
|
|
# hidden by the warnings; Vim won't place a sign oven an existing one.
|
2014-01-05 12:58:13 -08:00
|
|
|
diags.sort( key = lambda diag: ( diag[ 'location' ][ 'column_num' ],
|
|
|
|
diag[ 'kind' ] ) )
|
2014-01-04 16:45:34 -08:00
|
|
|
return buffer_to_line_to_diags
|
2014-01-04 14:28:27 -08:00
|
|
|
|
2014-01-04 18:32:42 -08:00
|
|
|
|
|
|
|
def _DiagnosticIsError( diag ):
|
2014-06-03 14:49:53 -07:00
|
|
|
return diag[ 'kind' ] == 'ERROR'
|
2014-01-04 18:32:42 -08:00
|
|
|
|
2014-09-14 01:15:40 +07:00
|
|
|
|
2015-12-04 14:45:11 -08:00
|
|
|
def _DiagnosticIsWarning( diag ):
|
|
|
|
return diag[ 'kind' ] == 'WARNING'
|
|
|
|
|
|
|
|
|
2014-10-29 10:16:38 -07:00
|
|
|
def _NormalizeDiagnostic( diag ):
|
|
|
|
def ClampToOne( value ):
|
|
|
|
return value if value > 0 else 1
|
|
|
|
|
|
|
|
location = diag[ 'location' ]
|
|
|
|
location[ 'column_num' ] = ClampToOne( location[ 'column_num' ] )
|
|
|
|
location[ 'line_num' ] = ClampToOne( location[ 'line_num' ] )
|
|
|
|
return diag
|
|
|
|
|
|
|
|
|
2016-03-06 17:13:43 +00:00
|
|
|
class _DiagSignPlacement(
|
|
|
|
namedtuple( "_DiagSignPlacement",
|
|
|
|
[ 'id', 'line', 'buffer', 'is_error' ] ) ):
|
2014-09-19 12:30:15 -07:00
|
|
|
# We want two signs that have different ids but the same location to compare
|
|
|
|
# equal. ID doesn't matter.
|
|
|
|
def __eq__( self, other ):
|
|
|
|
return ( self.line == other.line and
|
|
|
|
self.buffer == other.buffer and
|
|
|
|
self.is_error == other.is_error )
|