2017-11-26 12:24:18 +00:00
|
|
|
Before:
|
|
|
|
Save g:ale_completion_delay
|
|
|
|
Save g:ale_completion_max_suggestions
|
|
|
|
Save g:ale_completion_info
|
|
|
|
Save &l:omnifunc
|
|
|
|
Save &l:completeopt
|
|
|
|
|
|
|
|
let g:ale_completion_enabled = 1
|
|
|
|
|
|
|
|
call ale#test#SetDirectory('/testplugin/test/completion')
|
|
|
|
call ale#test#SetFilename('dummy.txt')
|
|
|
|
|
|
|
|
runtime autoload/ale/lsp.vim
|
|
|
|
|
2017-11-26 13:01:01 +00:00
|
|
|
let g:message_list = []
|
2018-07-22 19:04:45 +01:00
|
|
|
let g:capability_checked = ''
|
2018-08-24 13:16:58 +01:00
|
|
|
let g:conn_id = v:null
|
2017-11-26 12:24:18 +00:00
|
|
|
let g:Callback = ''
|
2018-07-22 21:14:48 +01:00
|
|
|
let g:wait_callback_list = []
|
2017-11-26 12:24:18 +00:00
|
|
|
|
2018-07-22 19:04:45 +01:00
|
|
|
function! ale#lsp_linter#StartLSP(buffer, linter) abort
|
2018-08-24 13:16:58 +01:00
|
|
|
let g:conn_id = ale#lsp#Register('executable', '/foo/bar', {})
|
|
|
|
call ale#lsp#MarkDocumentAsOpen(g:conn_id, a:buffer)
|
2018-06-15 09:53:13 +01:00
|
|
|
|
2017-11-26 12:24:18 +00:00
|
|
|
return {
|
2018-06-15 09:53:13 +01:00
|
|
|
\ 'buffer': a:buffer,
|
2018-08-24 13:16:58 +01:00
|
|
|
\ 'connection_id': g:conn_id,
|
2017-11-26 12:24:18 +00:00
|
|
|
\ 'project_root': '/foo/bar',
|
2018-06-15 09:53:13 +01:00
|
|
|
\ 'language_id': 'python',
|
2017-11-26 12:24:18 +00:00
|
|
|
\}
|
|
|
|
endfunction
|
|
|
|
|
2018-07-16 17:57:07 +01:00
|
|
|
" Pretend we're in insert mode for most tests.
|
|
|
|
function! ale#util#Mode(...) abort
|
|
|
|
return 'i'
|
|
|
|
endfunction
|
|
|
|
|
2018-08-24 13:16:58 +01:00
|
|
|
function! ale#lsp#WaitForCapability(conn_id, capability, callback) abort
|
2018-07-22 19:04:45 +01:00
|
|
|
let g:capability_checked = a:capability
|
2018-07-22 21:14:48 +01:00
|
|
|
call add(g:wait_callback_list, a:callback)
|
2018-07-22 19:04:45 +01:00
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! ale#lsp#RegisterCallback(conn_id, callback) abort
|
|
|
|
let g:Callback = a:callback
|
|
|
|
endfunction
|
|
|
|
|
2017-11-26 12:24:18 +00:00
|
|
|
" Replace the Send function for LSP, so we can monitor calls to it.
|
2018-08-24 13:16:58 +01:00
|
|
|
function! ale#lsp#Send(conn_id, message) abort
|
2017-11-26 13:01:01 +00:00
|
|
|
call add(g:message_list, a:message)
|
2018-07-22 21:14:48 +01:00
|
|
|
|
|
|
|
return 1
|
2017-11-26 12:24:18 +00:00
|
|
|
endfunction
|
|
|
|
|
|
|
|
After:
|
|
|
|
Restore
|
|
|
|
|
2018-08-24 13:16:58 +01:00
|
|
|
if g:conn_id isnot v:null
|
|
|
|
call ale#lsp#RemoveConnectionWithID(g:conn_id)
|
|
|
|
endif
|
|
|
|
|
2017-11-26 13:01:01 +00:00
|
|
|
unlet! g:message_list
|
2018-07-22 19:04:45 +01:00
|
|
|
unlet! g:capability_checked
|
2018-07-22 21:14:48 +01:00
|
|
|
unlet! g:wait_callback_list
|
2018-08-24 13:16:58 +01:00
|
|
|
unlet! g:conn_id
|
2017-11-26 12:24:18 +00:00
|
|
|
unlet! g:Callback
|
|
|
|
unlet! b:ale_old_omnifunc
|
|
|
|
unlet! b:ale_old_completopt
|
|
|
|
unlet! b:ale_completion_info
|
|
|
|
unlet! b:ale_completion_response
|
|
|
|
unlet! b:ale_completion_parser
|
|
|
|
unlet! b:ale_complete_done_time
|
|
|
|
unlet! b:ale_linters
|
2018-05-28 15:16:15 +01:00
|
|
|
unlet! b:ale_tsserver_completion_names
|
2017-11-26 12:24:18 +00:00
|
|
|
|
2018-07-16 17:57:07 +01:00
|
|
|
" Reset the function.
|
|
|
|
function! ale#util#Mode(...) abort
|
|
|
|
return call('mode', a:000)
|
|
|
|
endfunction
|
|
|
|
|
2017-11-26 12:24:18 +00:00
|
|
|
call ale#test#RestoreDirectory()
|
|
|
|
call ale#linter#Reset()
|
|
|
|
|
|
|
|
" Stop any timers we left behind.
|
|
|
|
" This stops the tests from failing randomly.
|
|
|
|
call ale#completion#StopTimer()
|
|
|
|
|
|
|
|
runtime autoload/ale/completion.vim
|
|
|
|
runtime autoload/ale/lsp.vim
|
2018-07-06 01:00:30 +01:00
|
|
|
runtime autoload/ale/lsp_linter.vim
|
2017-11-26 12:24:18 +00:00
|
|
|
|
|
|
|
Given typescript(Some typescript file):
|
|
|
|
foo
|
|
|
|
somelongerline
|
|
|
|
bazxyzxyzxyz
|
|
|
|
|
|
|
|
Execute(The right message should be sent for the initial tsserver request):
|
|
|
|
runtime ale_linters/typescript/tsserver.vim
|
|
|
|
let b:ale_linters = ['tsserver']
|
|
|
|
" The cursor position needs to match what was saved before.
|
|
|
|
call setpos('.', [bufnr(''), 1, 3, 0])
|
|
|
|
|
|
|
|
call ale#completion#GetCompletions()
|
|
|
|
|
2018-07-22 19:04:45 +01:00
|
|
|
" We shouldn't register the callback yet.
|
|
|
|
AssertEqual '''''', string(g:Callback)
|
|
|
|
|
2018-07-22 21:14:48 +01:00
|
|
|
AssertEqual 1, len(g:wait_callback_list)
|
2018-07-22 19:04:45 +01:00
|
|
|
AssertEqual 'completion', g:capability_checked
|
2018-08-24 13:16:58 +01:00
|
|
|
call map(g:wait_callback_list, 'v:val([g:conn_id, ''/foo/bar''])')
|
2018-07-22 19:04:45 +01:00
|
|
|
|
2017-11-26 12:24:18 +00:00
|
|
|
" We should send the right callback.
|
|
|
|
AssertEqual
|
|
|
|
\ 'function(''ale#completion#HandleTSServerResponse'')',
|
|
|
|
\ string(g:Callback)
|
|
|
|
" We should send the right message.
|
|
|
|
AssertEqual
|
2017-11-26 13:01:01 +00:00
|
|
|
\ [[0, 'ts@completions', {'file': expand('%:p'), 'line': 1, 'offset': 3, 'prefix': 'fo'}]],
|
|
|
|
\ g:message_list
|
2017-11-26 12:24:18 +00:00
|
|
|
" We should set up the completion info correctly.
|
|
|
|
AssertEqual
|
|
|
|
\ {
|
|
|
|
\ 'line_length': 3,
|
2018-08-24 13:16:58 +01:00
|
|
|
\ 'conn_id': g:conn_id,
|
2017-11-26 12:24:18 +00:00
|
|
|
\ 'column': 3,
|
2018-07-22 21:14:48 +01:00
|
|
|
\ 'request_id': 1,
|
2017-11-26 12:24:18 +00:00
|
|
|
\ 'line': 1,
|
|
|
|
\ 'prefix': 'fo',
|
|
|
|
\ },
|
|
|
|
\ get(b:, 'ale_completion_info', {})
|
|
|
|
|
|
|
|
Execute(The right message sent to the tsserver LSP when the first completion message is received):
|
|
|
|
" The cursor position needs to match what was saved before.
|
|
|
|
call setpos('.', [bufnr(''), 1, 1, 0])
|
|
|
|
let b:ale_completion_info = {
|
|
|
|
\ 'conn_id': 123,
|
|
|
|
\ 'prefix': 'f',
|
|
|
|
\ 'request_id': 4,
|
|
|
|
\ 'line': 1,
|
|
|
|
\ 'column': 1,
|
|
|
|
\}
|
|
|
|
" We should only show up to this many suggestions.
|
|
|
|
let g:ale_completion_max_suggestions = 3
|
|
|
|
|
|
|
|
" Handle the response for completions.
|
|
|
|
call ale#completion#HandleTSServerResponse(123, {
|
|
|
|
\ 'request_seq': 4,
|
|
|
|
\ 'command': 'completions',
|
|
|
|
\ 'body': [
|
|
|
|
\ {'name': 'Baz'},
|
|
|
|
\ {'name': 'dingDong'},
|
|
|
|
\ {'name': 'Foo'},
|
|
|
|
\ {'name': 'FooBar'},
|
|
|
|
\ {'name': 'frazzle'},
|
|
|
|
\ {'name': 'FFS'},
|
|
|
|
\ ],
|
|
|
|
\})
|
|
|
|
|
2018-05-28 15:16:15 +01:00
|
|
|
" We should save the names we got in the buffer, as TSServer doesn't return
|
|
|
|
" details for every name.
|
|
|
|
AssertEqual
|
|
|
|
\ ['Foo', 'FooBar', 'frazzle'],
|
|
|
|
\ get(b:, 'ale_tsserver_completion_names', [])
|
|
|
|
|
2017-11-26 12:24:18 +00:00
|
|
|
" The entry details messages should have been sent.
|
|
|
|
AssertEqual
|
2017-11-26 13:01:01 +00:00
|
|
|
\ [[
|
2017-11-26 12:24:18 +00:00
|
|
|
\ 0,
|
|
|
|
\ 'ts@completionEntryDetails',
|
|
|
|
\ {
|
|
|
|
\ 'file': expand('%:p'),
|
|
|
|
\ 'entryNames': ['Foo', 'FooBar', 'frazzle'],
|
|
|
|
\ 'offset': 1,
|
|
|
|
\ 'line': 1,
|
|
|
|
\ },
|
2017-11-26 13:01:01 +00:00
|
|
|
\ ]],
|
|
|
|
\ g:message_list
|
2017-11-26 12:24:18 +00:00
|
|
|
|
|
|
|
Given python(Some Python file):
|
|
|
|
foo
|
|
|
|
somelongerline
|
|
|
|
bazxyzxyzxyz
|
|
|
|
|
|
|
|
Execute(The right message should be sent for the initial LSP request):
|
|
|
|
runtime ale_linters/python/pyls.vim
|
|
|
|
let b:ale_linters = ['pyls']
|
|
|
|
" The cursor position needs to match what was saved before.
|
|
|
|
call setpos('.', [bufnr(''), 1, 5, 0])
|
|
|
|
|
|
|
|
call ale#completion#GetCompletions()
|
|
|
|
|
2018-07-22 19:04:45 +01:00
|
|
|
" We shouldn't register the callback yet.
|
|
|
|
AssertEqual '''''', string(g:Callback)
|
|
|
|
|
2018-07-22 21:14:48 +01:00
|
|
|
AssertEqual 1, len(g:wait_callback_list)
|
2018-07-22 19:04:45 +01:00
|
|
|
AssertEqual 'completion', g:capability_checked
|
2018-08-24 13:16:58 +01:00
|
|
|
call map(g:wait_callback_list, 'v:val([g:conn_id, ''/foo/bar''])')
|
2018-07-22 19:04:45 +01:00
|
|
|
|
2017-11-26 12:24:18 +00:00
|
|
|
" We should send the right callback.
|
|
|
|
AssertEqual
|
|
|
|
\ 'function(''ale#completion#HandleLSPResponse'')',
|
|
|
|
\ string(g:Callback)
|
|
|
|
" We should send the right message.
|
|
|
|
" The character index needs to be at most the index of the last character on
|
|
|
|
" the line, or integration with pyls will be broken.
|
2017-11-26 13:01:01 +00:00
|
|
|
"
|
|
|
|
" We need to send the message for changing the document first.
|
2017-11-26 12:24:18 +00:00
|
|
|
AssertEqual
|
2017-11-26 13:01:01 +00:00
|
|
|
\ [
|
|
|
|
\ [1, 'textDocument/didChange', {
|
|
|
|
\ 'textDocument': {
|
|
|
|
\ 'uri': ale#path#ToURI(expand('%:p')),
|
|
|
|
\ 'version': g:ale_lsp_next_version_id - 1,
|
|
|
|
\ },
|
|
|
|
\ 'contentChanges': [{'text': join(getline(1, '$'), "\n") . "\n"}]
|
|
|
|
\ }],
|
|
|
|
\ [0, 'textDocument/completion', {
|
2017-11-26 12:24:18 +00:00
|
|
|
\ 'textDocument': {'uri': ale#path#ToURI(expand('%:p'))},
|
2019-02-08 19:31:38 +00:00
|
|
|
\ 'position': {'line': 0, 'character': 3},
|
2017-11-26 13:01:01 +00:00
|
|
|
\ }],
|
|
|
|
\ ],
|
|
|
|
\ g:message_list
|
2017-11-26 12:24:18 +00:00
|
|
|
" We should set up the completion info correctly.
|
|
|
|
AssertEqual
|
|
|
|
\ {
|
|
|
|
\ 'line_length': 3,
|
2018-08-24 13:16:58 +01:00
|
|
|
\ 'conn_id': g:conn_id,
|
2017-11-26 12:24:18 +00:00
|
|
|
\ 'column': 3,
|
2018-07-22 21:14:48 +01:00
|
|
|
\ 'request_id': 1,
|
2017-11-26 12:24:18 +00:00
|
|
|
\ 'line': 1,
|
|
|
|
\ 'prefix': 'fo',
|
2018-07-22 21:14:48 +01:00
|
|
|
\ 'completion_filter': 'ale#completion#python#CompletionItemFilter',
|
2017-11-26 12:24:18 +00:00
|
|
|
\ },
|
|
|
|
\ get(b:, 'ale_completion_info', {})
|
2018-07-22 21:14:48 +01:00
|
|
|
|
|
|
|
Execute(Two completion requests shouldn't be sent in a row):
|
|
|
|
call ale#linter#PreventLoading('python')
|
|
|
|
call ale#linter#Define('python', {
|
|
|
|
\ 'name': 'foo',
|
|
|
|
\ 'lsp': 'stdio',
|
|
|
|
\ 'executable': 'foo',
|
|
|
|
\ 'command': 'foo',
|
|
|
|
\ 'project_root_callback': {-> '/foo/bar'},
|
|
|
|
\})
|
|
|
|
call ale#linter#Define('python', {
|
|
|
|
\ 'name': 'bar',
|
|
|
|
\ 'lsp': 'stdio',
|
|
|
|
\ 'executable': 'foo',
|
|
|
|
\ 'command': 'foo',
|
|
|
|
\ 'project_root_callback': {-> '/foo/bar'},
|
|
|
|
\})
|
|
|
|
let b:ale_linters = ['foo', 'bar']
|
|
|
|
|
|
|
|
" The cursor position needs to match what was saved before.
|
|
|
|
call setpos('.', [bufnr(''), 1, 5, 0])
|
|
|
|
|
|
|
|
call ale#completion#GetCompletions()
|
|
|
|
|
|
|
|
" We shouldn't register the callback yet.
|
|
|
|
AssertEqual '''''', string(g:Callback)
|
|
|
|
|
|
|
|
AssertEqual 2, len(g:wait_callback_list)
|
|
|
|
AssertEqual 'completion', g:capability_checked
|
|
|
|
call map(g:wait_callback_list, 'v:val([347, ''/foo/bar''])')
|
|
|
|
|
|
|
|
" We should only send one completion message for two LSP servers.
|
|
|
|
AssertEqual
|
|
|
|
\ [
|
|
|
|
\ [1, 'textDocument/didChange', {
|
|
|
|
\ 'textDocument': {
|
|
|
|
\ 'uri': ale#path#ToURI(expand('%:p')),
|
|
|
|
\ 'version': g:ale_lsp_next_version_id - 1,
|
|
|
|
\ },
|
|
|
|
\ 'contentChanges': [{'text': join(getline(1, '$'), "\n") . "\n"}]
|
|
|
|
\ }],
|
|
|
|
\ [0, 'textDocument/completion', {
|
|
|
|
\ 'textDocument': {'uri': ale#path#ToURI(expand('%:p'))},
|
2019-02-08 19:31:38 +00:00
|
|
|
\ 'position': {'line': 0, 'character': 3},
|
2018-07-22 21:14:48 +01:00
|
|
|
\ }],
|
|
|
|
\ ],
|
|
|
|
\ g:message_list
|