Auto merge of #2489 - puremourning:use-server-start-column, r=micbou

[READY] Reset the start column in omnifunc completer

# PR Prelude

Thank you for working on YCM! :)

**Please complete these steps and check these boxes (by putting an `x` inside
the brackets) _before_ filing your PR:**

- [X] I have read and understood YCM's [CONTRIBUTING][cont] document.
- [X] I have read and understood YCM's [CODE_OF_CONDUCT][code] document.
- [X] I have included tests for the changes in my PR. If not, I have included a
  rationale for why I haven't.
- [X] **I understand my PR may be closed if it becomes obvious I didn't
  actually perform all of these steps.**

# Why this change is necessary and useful

A number of issues have been raised over the years where YCM doesn't interact great with the user's omnifunc. This is commonly because we ignore the omnifunc's `findstart` response, and use our own implementation (`base.CompletionStartColumn`).

In combination with valloric/ycmd#681 this change uses the omnifunc's start column for completions and fixes valloric/ycmd#671 and others, such as:

- https://github.com/Valloric/YouCompleteMe/issues/1322 probably (not yet tested)
- https://github.com/Valloric/YouCompleteMe/issues/1957
- https://github.com/Valloric/YouCompleteMe/issues/1816

Note: This is just an initial test for sharing the code to gauge reaction. Not fully tested yet.

[cont]: https://github.com/Valloric/YouCompleteMe/blob/master/CONTRIBUTING.md
[code]: https://github.com/Valloric/YouCompleteMe/blob/master/CODE_OF_CONDUCT.md

<!-- 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/2489)
<!-- Reviewable:end -->
This commit is contained in:
zzbot 2017-07-02 18:44:06 -07:00 committed by GitHub
commit 25a2e3120c
3 changed files with 58 additions and 25 deletions

View File

@ -83,6 +83,13 @@ class OmniCompleter( Completer ):
# FIXME: Technically, if the return is -1 we should raise an error # FIXME: Technically, if the return is -1 we should raise an error
return [] return []
# Use the start column calculated by the omnifunc, rather than our own
# interpretation. This is important for certain languages where our
# identifier detection is either incorrect or not compatible with the
# behaviour of the omnifunc. Note: do this before calling the omnifunc
# because it affects the value returned by 'query'
request_data[ 'start_column' ] = return_value + 1
omnifunc_call = [ self._omnifunc, omnifunc_call = [ self._omnifunc,
"(0,'", "(0,'",
vimsupport.EscapeForVim( request_data[ 'query' ] ), vimsupport.EscapeForVim( request_data[ 'query' ] ),

View File

@ -27,7 +27,6 @@ from builtins import * # noqa
from future.utils import PY2 from future.utils import PY2
from mock import patch, call from mock import patch, call
from nose.tools import eq_ from nose.tools import eq_
from hamcrest import contains_string
from ycm.tests.test_utils import ExpectedFailure, ExtendedMock, MockVimModule from ycm.tests.test_utils import ExpectedFailure, ExtendedMock, MockVimModule
MockVimModule() MockVimModule()
@ -120,7 +119,7 @@ def OmniCompleter_GetCompletions_Cache_ListFilter_test( ycm ):
# And get the completions # And get the completions
with patch( 'vim.eval', with patch( 'vim.eval',
new_callable = ExtendedMock, new_callable = ExtendedMock,
side_effect = [ 6, omnifunc_result ] ) as vim_eval: side_effect = [ 5, omnifunc_result ] ) as vim_eval:
results = ycm._omnicomp.ComputeCandidates( request_data ) results = ycm._omnicomp.ComputeCandidates( request_data )
vim_eval.assert_has_exact_calls( [ vim_eval.assert_has_exact_calls( [
@ -150,7 +149,7 @@ def OmniCompleter_GetCompletions_NoCache_List_test( ycm ):
# And get the completions # And get the completions
with patch( 'vim.eval', with patch( 'vim.eval',
new_callable = ExtendedMock, new_callable = ExtendedMock,
side_effect = [ 6, omnifunc_result ] ) as vim_eval: side_effect = [ 5, omnifunc_result ] ) as vim_eval:
results = ycm._omnicomp.ComputeCandidates( request_data ) results = ycm._omnicomp.ComputeCandidates( request_data )
vim_eval.assert_has_exact_calls( [ vim_eval.assert_has_exact_calls( [
@ -181,7 +180,7 @@ def OmniCompleter_GetCompletions_NoCache_ListFilter_test( ycm ):
# And get the completions # And get the completions
with patch( 'vim.eval', with patch( 'vim.eval',
new_callable = ExtendedMock, new_callable = ExtendedMock,
side_effect = [ 6, omnifunc_result ] ) as vim_eval: side_effect = [ 5, omnifunc_result ] ) as vim_eval:
results = ycm._omnicomp.ComputeCandidates( request_data ) results = ycm._omnicomp.ComputeCandidates( request_data )
vim_eval.assert_has_exact_calls( [ vim_eval.assert_has_exact_calls( [
@ -194,11 +193,8 @@ def OmniCompleter_GetCompletions_NoCache_ListFilter_test( ycm ):
eq_( results, omnifunc_result ) eq_( results, omnifunc_result )
@ExpectedFailure( 'We ignore the result of the call to findstart and use our ' @YouCompleteMeInstance( { 'cache_omnifunc': 0 } )
'own interpretation of where the identifier should be', def OmniCompleter_GetCompletsions_NoCache_UseFindStart_test( ycm ):
contains_string( "test_omnifunc(0,'t')" ) )
@YouCompleteMeInstance( { 'cache_omnifunc': 1 } )
def OmniCompleter_GetCompletsions_UseFindStart_test( ycm ):
contents = 'test.t' contents = 'test.t'
request_data = BuildRequestWrap( line_num = 1, request_data = BuildRequestWrap( line_num = 1,
column_num = 7, column_num = 7,
@ -217,20 +213,50 @@ def OmniCompleter_GetCompletsions_UseFindStart_test( ycm ):
# And get the completions # And get the completions
with patch( 'vim.eval', with patch( 'vim.eval',
new_callable = ExtendedMock, new_callable = ExtendedMock,
side_effect = [ 1, omnifunc_result ] ) as vim_eval: side_effect = [ 0, omnifunc_result ] ) as vim_eval:
results = ycm._omnicomp.ComputeCandidates( request_data ) results = ycm._omnicomp.ComputeCandidates( request_data )
vim_eval.assert_has_exact_calls( [ vim_eval.assert_has_exact_calls( [
call( 'test_omnifunc(1,"")' ), call( 'test_omnifunc(1,"")' ),
# Fails here: actual result is that the findstart result (1) is ignored
# and we use the 't' query as we normally would on the server side
call( "test_omnifunc(0,'test.t')" ), call( "test_omnifunc(0,'test.t')" ),
] ) ] )
eq_( results, omnifunc_result ) eq_( results, omnifunc_result )
@YouCompleteMeInstance( { 'cache_omnifunc': 1 } )
def OmniCompleter_GetCompletsions_Cache_UseFindStart_test( ycm ):
contents = 'test.t'
request_data = BuildRequestWrap( line_num = 1,
column_num = 7,
contents = contents )
eq_( request_data[ 'query' ], 't' )
# Make sure there is an omnifunc set up.
with patch( 'vim.eval', return_value = ToBytesOnPY2( 'test_omnifunc' ) ):
ycm._omnicomp.OnFileReadyToParse( request_data )
omnifunc_result = [ ToBytesOnPY2( 'a' ),
ToBytesOnPY2( 'b' ),
ToBytesOnPY2( 'cdef' ) ]
# And get the completions
with patch( 'vim.eval',
new_callable = ExtendedMock,
side_effect = [ 0, omnifunc_result ] ) as vim_eval:
results = ycm._omnicomp.ComputeCandidates( request_data )
vim_eval.assert_has_exact_calls( [
call( 'test_omnifunc(1,"")' ),
call( "test_omnifunc(0,'test.t')" ),
] )
# There are no results because the query 'test.t' doesn't match any
# candidate (and cache_omnifunc=1, so we FilterAndSortCandidates)
eq_( results, [] )
@YouCompleteMeInstance( { 'cache_omnifunc': 1 } ) @YouCompleteMeInstance( { 'cache_omnifunc': 1 } )
def OmniCompleter_GetCompletions_Cache_Object_test( ycm ): def OmniCompleter_GetCompletions_Cache_Object_test( ycm ):
contents = 'test.t' contents = 'test.t'
@ -255,7 +281,7 @@ def OmniCompleter_GetCompletions_Cache_Object_test( ycm ):
# And get the completions # And get the completions
with patch( 'vim.eval', with patch( 'vim.eval',
new_callable = ExtendedMock, new_callable = ExtendedMock,
side_effect = [ 6, omnifunc_result ] ) as vim_eval: side_effect = [ 5, omnifunc_result ] ) as vim_eval:
results = ycm._omnicomp.ComputeCandidates( request_data ) results = ycm._omnicomp.ComputeCandidates( request_data )
@ -300,7 +326,7 @@ def OmniCompleter_GetCompletions_Cache_ObjectList_test( ycm ):
# And get the completions # And get the completions
with patch( 'vim.eval', with patch( 'vim.eval',
new_callable = ExtendedMock, new_callable = ExtendedMock,
side_effect = [ 6, omnifunc_result ] ) as vim_eval: side_effect = [ 5, omnifunc_result ] ) as vim_eval:
results = ycm._omnicomp.ComputeCandidates( request_data ) results = ycm._omnicomp.ComputeCandidates( request_data )
@ -345,7 +371,7 @@ def OmniCompleter_GetCompletions_NoCache_ObjectList_test( ycm ):
# And get the completions # And get the completions
with patch( 'vim.eval', with patch( 'vim.eval',
new_callable = ExtendedMock, new_callable = ExtendedMock,
side_effect = [ 6, omnifunc_result ] ) as vim_eval: side_effect = [ 5, omnifunc_result ] ) as vim_eval:
results = ycm._omnicomp.ComputeCandidates( request_data ) results = ycm._omnicomp.ComputeCandidates( request_data )
@ -394,7 +420,7 @@ def OmniCompleter_GetCompletions_Cache_ObjectListObject_test( ycm ):
# And get the completions # And get the completions
with patch( 'vim.eval', with patch( 'vim.eval',
new_callable = ExtendedMock, new_callable = ExtendedMock,
side_effect = [ 6, omnifunc_result ] ) as vim_eval: side_effect = [ 5, omnifunc_result ] ) as vim_eval:
results = ycm._omnicomp.ComputeCandidates( request_data ) results = ycm._omnicomp.ComputeCandidates( request_data )
@ -441,7 +467,7 @@ def OmniCompleter_GetCompletions_NoCache_ObjectListObject_test( ycm ):
# And get the completions # And get the completions
with patch( 'vim.eval', with patch( 'vim.eval',
new_callable = ExtendedMock, new_callable = ExtendedMock,
side_effect = [ 6, omnifunc_result ] ) as vim_eval: side_effect = [ 5, omnifunc_result ] ) as vim_eval:
results = ycm._omnicomp.ComputeCandidates( request_data ) results = ycm._omnicomp.ComputeCandidates( request_data )
@ -474,7 +500,7 @@ def OmniCompleter_GetCompletions_Cache_List_Unicode_test( ycm ):
# And get the completions # And get the completions
with patch( 'vim.eval', with patch( 'vim.eval',
new_callable = ExtendedMock, new_callable = ExtendedMock,
side_effect = [ 6, omnifunc_result ] ) as vim_eval: side_effect = [ 12, omnifunc_result ] ) as vim_eval:
results = ycm._omnicomp.ComputeCandidates( request_data ) results = ycm._omnicomp.ComputeCandidates( request_data )
vim_eval.assert_has_exact_calls( [ vim_eval.assert_has_exact_calls( [
@ -504,7 +530,7 @@ def OmniCompleter_GetCompletions_NoCache_List_Unicode_test( ycm ):
# And get the completions # And get the completions
with patch( 'vim.eval', with patch( 'vim.eval',
new_callable = ExtendedMock, new_callable = ExtendedMock,
side_effect = [ 6, omnifunc_result ] ) as vim_eval: side_effect = [ 12, omnifunc_result ] ) as vim_eval:
results = ycm._omnicomp.ComputeCandidates( request_data ) results = ycm._omnicomp.ComputeCandidates( request_data )
vim_eval.assert_has_exact_calls( [ vim_eval.assert_has_exact_calls( [
@ -535,7 +561,7 @@ def OmniCompleter_GetCompletions_Cache_List_Filter_Unicode_test( ycm ):
# And get the completions # And get the completions
with patch( 'vim.eval', with patch( 'vim.eval',
new_callable = ExtendedMock, new_callable = ExtendedMock,
side_effect = [ 6, omnifunc_result ] ) as vim_eval: side_effect = [ 12, omnifunc_result ] ) as vim_eval:
results = ycm._omnicomp.ComputeCandidates( request_data ) results = ycm._omnicomp.ComputeCandidates( request_data )
vim_eval.assert_has_exact_calls( [ vim_eval.assert_has_exact_calls( [
@ -564,7 +590,7 @@ def OmniCompleter_GetCompletions_NoCache_List_Filter_Unicode_test( ycm ):
# And get the completions # And get the completions
with patch( 'vim.eval', with patch( 'vim.eval',
new_callable = ExtendedMock, new_callable = ExtendedMock,
side_effect = [ 6, omnifunc_result ] ) as vim_eval: side_effect = [ 12, omnifunc_result ] ) as vim_eval:
results = ycm._omnicomp.ComputeCandidates( request_data ) results = ycm._omnicomp.ComputeCandidates( request_data )
vim_eval.assert_has_exact_calls( [ vim_eval.assert_has_exact_calls( [
@ -610,7 +636,7 @@ def OmniCompleter_GetCompletions_Cache_ObjectList_Unicode_test( ycm ):
# And get the completions # And get the completions
with patch( 'vim.eval', with patch( 'vim.eval',
new_callable = ExtendedMock, new_callable = ExtendedMock,
side_effect = [ 6, omnifunc_result ] ) as vim_eval: side_effect = [ 12, omnifunc_result ] ) as vim_eval:
results = ycm._omnicomp.ComputeCandidates( request_data ) results = ycm._omnicomp.ComputeCandidates( request_data )
@ -666,7 +692,7 @@ def OmniCompleter_GetCompletions_Cache_ObjectListObject_Unicode_test( ycm ):
# And get the completions # And get the completions
with patch( 'vim.eval', with patch( 'vim.eval',
new_callable = ExtendedMock, new_callable = ExtendedMock,
side_effect = [ 6, omnifunc_result ] ) as vim_eval: side_effect = [ 12, omnifunc_result ] ) as vim_eval:
results = ycm._omnicomp.ComputeCandidates( request_data ) results = ycm._omnicomp.ComputeCandidates( request_data )

2
third_party/ycmd vendored

@ -1 +1 @@
Subproject commit 7618a8d7e25ebddd5fce64df24047086be7906cb Subproject commit 490b3ee26d7930704743585ab1ac6f925dbc7202