Auto merge of #2592 - micbou:prioritize-errors-over-warnings, r=Valloric

[READY] Prioritize errors over warnings on the same line

When multiple diagnostics are on the same line, warnings may have higher priority than errors. See #2141 for an example. This happens for several reasons:
 - we sort diagnostics by column and then by kind. We should do the opposite;
 - we place all signs; each sign overriding the previous one. Since only one sign is visible by line, we should just insert the first one (which has the highest priority);
 - we draw highlighting matches (squiggles) in the same order as diagnostics; each match being drawn over the previous ones. We should draw them in reverse order (from lowest to highest priority).

Fixes #2141.

<!-- Reviewable:start -->
---
This change is [<img src="https://reviewable.io/review_button.svg" height="34" align="absmiddle" alt="Reviewable"/>](https://reviewable.io/reviews/valloric/youcompleteme/2592)
<!-- Reviewable:end -->
This commit is contained in:
Homu 2017-03-30 05:12:35 +09:00
commit f4e6defa33
5 changed files with 201 additions and 32 deletions

View File

@ -108,8 +108,9 @@ class DiagnosticInterface( object ):
self._diag_message_needs_clearing = False self._diag_message_needs_clearing = False
return return
text = diags[ 0 ][ 'text' ] first_diag = diags[ 0 ]
if diags[ 0 ].get( 'fixit_available', False ): text = first_diag[ 'text' ]
if first_diag.get( 'fixit_available', False ):
text += ' (FixIt)' text += ' (FixIt)'
vimsupport.PostVimMessage( text, warning = False, truncate = True ) vimsupport.PostVimMessage( text, warning = False, truncate = True )
@ -132,7 +133,8 @@ def _UpdateSquiggles( buffer_number_to_line_to_diags ):
line_to_diags = buffer_number_to_line_to_diags[ vim.current.buffer.number ] line_to_diags = buffer_number_to_line_to_diags[ vim.current.buffer.number ]
for diags in itervalues( line_to_diags ): for diags in itervalues( line_to_diags ):
for diag in diags: # Insert squiggles in reverse order so that errors overlap warnings.
for diag in reversed( diags ):
location_extent = diag[ 'location_extent' ] location_extent = diag[ 'location_extent' ]
is_error = _DiagnosticIsError( diag ) is_error = _DiagnosticIsError( diag )
@ -201,23 +203,23 @@ def _GetKeptAndNewSigns( placed_signs, buffer_number_to_line_to_diags,
continue continue
for line, diags in iteritems( line_to_diags ): for line, diags in iteritems( line_to_diags ):
for diag in diags: # Only one sign is visible by line.
sign = _DiagSignPlacement( next_sign_id, first_diag = diags[ 0 ]
line, sign = _DiagSignPlacement( next_sign_id,
buffer_number, line,
_DiagnosticIsError( diag ) ) buffer_number,
if sign not in placed_signs: _DiagnosticIsError( first_diag ) )
new_signs += [ sign ] if sign not in placed_signs:
next_sign_id += 1 new_signs.append( sign )
else: next_sign_id += 1
# We use .index here because `sign` contains a new id, but else:
# we need the sign with the old id to unplace it later on. # We use .index here because `sign` contains a new id, but
# We won't be placing the new sign. # we need the sign with the old id to unplace it later on.
kept_signs += [ placed_signs[ placed_signs.index( sign ) ] ] # We won't be placing the new sign.
kept_signs.append( placed_signs[ placed_signs.index( sign ) ] )
return new_signs, kept_signs, next_sign_id return new_signs, kept_signs, next_sign_id
def _PlaceNewSigns( kept_signs, new_signs ): def _PlaceNewSigns( kept_signs, new_signs ):
placed_signs = kept_signs[:] placed_signs = kept_signs[:]
for sign in new_signs: for sign in new_signs:
@ -226,7 +228,7 @@ def _PlaceNewSigns( kept_signs, new_signs ):
if sign in placed_signs: if sign in placed_signs:
continue continue
vimsupport.PlaceSign( sign.id, sign.line, sign.buffer, sign.is_error ) vimsupport.PlaceSign( sign.id, sign.line, sign.buffer, sign.is_error )
placed_signs.append(sign) placed_signs.append( sign )
return placed_signs return placed_signs
@ -247,10 +249,10 @@ def _ConvertDiagListToDict( diag_list ):
for line_to_diags in itervalues( buffer_to_line_to_diags ): for line_to_diags in itervalues( buffer_to_line_to_diags ):
for diags in itervalues( line_to_diags ): for diags in itervalues( line_to_diags ):
# We also want errors to be listed before warnings so that errors aren't # We want errors to be listed before warnings so that errors aren't hidden
# hidden by the warnings; Vim won't place a sign oven an existing one. # by the warnings.
diags.sort( key = lambda diag: ( diag[ 'location' ][ 'column_num' ], diags.sort( key = lambda diag: ( diag[ 'kind' ],
diag[ 'kind' ] ) ) diag[ 'location' ][ 'column_num' ] ) )
return buffer_to_line_to_diags return buffer_to_line_to_diags

View File

@ -48,8 +48,8 @@ def PresentDialog_Confirm_Call( message ):
def PlaceSign_Call( sign_id, line_num, buffer_num, is_error ): def PlaceSign_Call( sign_id, line_num, buffer_num, is_error ):
sign_name = 'YcmError' if is_error else 'YcmWarning' sign_name = 'YcmError' if is_error else 'YcmWarning'
return call( 'sign place {0} line={1} name={2} buffer={3}' return call( 'sign place {0} name={1} line={2} buffer={3}'
.format( sign_id, line_num, sign_name, buffer_num ) ) .format( sign_id, sign_name, line_num, buffer_num ) )
def UnplaceSign_Call( sign_id, buffer_num ): def UnplaceSign_Call( sign_id, buffer_num ):

View File

@ -41,6 +41,9 @@ BWIPEOUT_REGEX = re.compile(
'^(?:silent! )bwipeout!? (?P<buffer_number>[0-9]+)$' ) '^(?:silent! )bwipeout!? (?P<buffer_number>[0-9]+)$' )
GETBUFVAR_REGEX = re.compile( GETBUFVAR_REGEX = re.compile(
'^getbufvar\((?P<buffer_number>[0-9]+), "&(?P<option>.+)"\)$' ) '^getbufvar\((?P<buffer_number>[0-9]+), "&(?P<option>.+)"\)$' )
MATCHADD_REGEX = re.compile(
'^matchadd\(\'(?P<group>.+)\', \'(?P<pattern>.+)\'\)$' )
MATCHDELETE_REGEX = re.compile( '^matchdelete\((?P<id>)\)$' )
# One-and only instance of mocked Vim object. The first 'import vim' that is # One-and only instance of mocked Vim object. The first 'import vim' that is
# executed binds the vim module to the instance of MagicMock that is created, # executed binds the vim module to the instance of MagicMock that is created,
@ -52,6 +55,8 @@ GETBUFVAR_REGEX = re.compile(
# https://github.com/Valloric/YouCompleteMe/pull/1694 # https://github.com/Valloric/YouCompleteMe/pull/1694
VIM_MOCK = MagicMock() VIM_MOCK = MagicMock()
VIM_MATCHES = []
@contextlib.contextmanager @contextlib.contextmanager
def CurrentWorkingDirectory( path ): def CurrentWorkingDirectory( path ):
@ -130,6 +135,30 @@ def _MockVimOptionsEval( value ):
return None return None
def _MockVimMatchEval( value ):
if value == 'getmatches()':
return VIM_MATCHES
match = MATCHADD_REGEX.search( value )
if match:
group = match.group( 'group' )
option = match.group( 'pattern' )
vim_match = VimMatch( group, option )
VIM_MATCHES.append( vim_match )
return vim_match.id
match = MATCHDELETE_REGEX.search( value )
if match:
identity = match.group( 'id' )
for index, vim_match in enumerate( VIM_MATCHES ):
if vim_match.id == identity:
VIM_MATCHES.pop( index )
return -1
return 0
return None
def _MockVimEval( value ): def _MockVimEval( value ):
if value == 'g:ycm_min_num_of_chars_for_completion': if value == 'g:ycm_min_num_of_chars_for_completion':
return 0 return 0
@ -154,6 +183,10 @@ def _MockVimEval( value ):
if result is not None: if result is not None:
return result return result
result = _MockVimMatchEval( value )
if result is not None:
return result
raise ValueError( 'Unexpected evaluation: {0}'.format( value ) ) raise ValueError( 'Unexpected evaluation: {0}'.format( value ) )
@ -217,6 +250,23 @@ class VimBuffer( object ):
return [ ToUnicode( x ) for x in self.contents ] return [ ToUnicode( x ) for x in self.contents ]
class VimMatch( object ):
def __init__( self, group, pattern ):
self.id = len( VIM_MATCHES )
self.group = group
self.pattern = pattern
def __eq__( self, other ):
return self.group == other.group and self.pattern == other.pattern
def __repr__( self ):
return "VimMatch( group = '{0}', pattern = '{1}' )".format( self.group,
self.pattern )
@contextlib.contextmanager @contextlib.contextmanager
def MockVimBuffers( buffers, current_buffer, cursor_position = ( 1, 1 ) ): def MockVimBuffers( buffers, current_buffer, cursor_position = ( 1, 1 ) ):
"""Simulates the Vim buffers list |buffers| where |current_buffer| is the """Simulates the Vim buffers list |buffers| where |current_buffer| is the

View File

@ -23,7 +23,7 @@ from __future__ import absolute_import
from builtins import * # noqa from builtins import * # noqa
from ycm.tests.test_utils import ( ExtendedMock, MockVimBuffers, MockVimModule, from ycm.tests.test_utils import ( ExtendedMock, MockVimBuffers, MockVimModule,
VimBuffer ) VimBuffer, VimMatch )
MockVimModule() MockVimModule()
import os import os
@ -32,7 +32,7 @@ from hamcrest import ( assert_that, contains, empty, is_in, is_not, has_length,
matches_regexp ) matches_regexp )
from mock import call, MagicMock, patch from mock import call, MagicMock, patch
from ycm.tests import StopServer, YouCompleteMeInstance from ycm.tests import StopServer, test_utils, YouCompleteMeInstance
from ycmd.responses import ServerError from ycmd.responses import ServerError
@ -348,8 +348,8 @@ def YouCompleteMe_ShowDiagnostics_DiagnosticsFound_DoNotOpenLocationList_test(
'text': 'error text', 'text': 'error text',
'location': { 'location': {
'filepath': 'buffer', 'filepath': 'buffer',
'column_num': 2, 'line_num': 19,
'line_num': 19 'column_num': 2
} }
} }
@ -388,8 +388,8 @@ def YouCompleteMe_ShowDiagnostics_DiagnosticsFound_OpenLocationList_test(
'text': 'error text', 'text': 'error text',
'location': { 'location': {
'filepath': 'buffer', 'filepath': 'buffer',
'column_num': 2, 'line_num': 19,
'line_num': 19 'column_num': 2
} }
} }
@ -413,3 +413,120 @@ def YouCompleteMe_ShowDiagnostics_DiagnosticsFound_OpenLocationList_test(
'valid': 1 'valid': 1
} ] ) } ] )
open_location_list.assert_called_once_with( focus = True ) open_location_list.assert_called_once_with( focus = True )
@YouCompleteMeInstance( { 'echo_current_diagnostic': 1,
'enable_diagnostic_signs': 1,
'enable_diagnostic_highlighting': 1 } )
@patch( 'ycm.youcompleteme.YouCompleteMe.FiletypeCompleterExistsForFiletype',
return_value = True )
@patch( 'ycm.vimsupport.PostVimMessage', new_callable = ExtendedMock )
@patch( 'vim.command', new_callable = ExtendedMock )
def YouCompleteMe_UpdateDiagnosticInterface_PrioritizeErrorsOverWarnings_test(
ycm, vim_command, post_vim_message, *args ):
contents = """int main() {
int x, y;
x == y
}"""
# List of diagnostics returned by ycmd for the above code.
diagnostics = [ {
'kind': 'ERROR',
'text': "expected ';' after expression",
'location': {
'filepath': 'buffer',
'line_num': 3,
'column_num': 9
},
# Looks strange but this is really what ycmd is returning.
'location_extent': {
'start': {
'filepath': '',
'line_num': 0,
'column_num': 0,
},
'end': {
'filepath': '',
'line_num': 0,
'column_num': 0,
}
},
'ranges': [],
'fixit_available': True
}, {
'kind': 'WARNING',
'text': 'equality comparison result unused',
'location': {
'filepath': 'buffer',
'line_num': 3,
'column_num': 7,
},
'location_extent': {
'start': {
'filepath': 'buffer',
'line_num': 3,
'column_num': 5,
},
'end': {
'filepath': 'buffer',
'line_num': 3,
'column_num': 7,
}
},
'ranges': [ {
'start': {
'filepath': 'buffer',
'line_num': 3,
'column_num': 3,
},
'end': {
'filepath': 'buffer',
'line_num': 3,
'column_num': 9,
}
} ],
'fixit_available': True
} ]
current_buffer = VimBuffer( 'buffer',
filetype = 'c',
contents = contents.splitlines(),
number = 5,
window = 2 )
test_utils.VIM_MATCHES = []
with MockVimBuffers( [ current_buffer ], current_buffer, ( 3, 1 ) ):
with patch( 'ycm.client.event_notification.EventNotification.Response',
return_value = diagnostics ):
ycm.OnFileReadyToParse()
ycm.HandleFileParseRequest( block = True )
# Error match is added after warning matches.
assert_that(
test_utils.VIM_MATCHES,
contains(
VimMatch( 'YcmWarningSection', '\%3l\%5c\_.\{-}\%3l\%7c' ),
VimMatch( 'YcmWarningSection', '\%3l\%3c\_.\{-}\%3l\%9c' ),
# FIXME: match should be inserted at the end of line 3 (missing ";").
VimMatch( 'YcmErrorSection', '\%0l\%0c' )
)
)
# Only the error sign is placed.
vim_command.assert_has_exact_calls( [
call( 'sign define ycm_dummy_sign' ),
call( 'sign place 3 name=ycm_dummy_sign line=3 buffer=5' ),
call( 'sign place 1 name=YcmError line=3 buffer=5' ),
call( 'sign undefine ycm_dummy_sign' ),
call( 'sign unplace 3 buffer=5' )
] )
# When moving the cursor on the diagnostics, the error is displayed to the
# user, not the warning.
ycm.OnCursorMoved()
post_vim_message.assert_has_exact_calls( [
call( "expected ';' after expression (FixIt)",
truncate = True, warning = False )
] )

View File

@ -175,8 +175,8 @@ def PlaceSign( sign_id, line_num, buffer_num, is_error = True ):
line_num = 1 line_num = 1
sign_name = 'YcmError' if is_error else 'YcmWarning' sign_name = 'YcmError' if is_error else 'YcmWarning'
vim.command( 'sign place {0} line={1} name={2} buffer={3}'.format( vim.command( 'sign place {0} name={1} line={2} buffer={3}'.format(
sign_id, line_num, sign_name, buffer_num ) ) sign_id, sign_name, line_num, buffer_num ) )
def PlaceDummySign( sign_id, buffer_num, line_num ): def PlaceDummySign( sign_id, buffer_num, line_num ):