mirror of
https://github.com/dense-analysis/ale.git
synced 2025-12-06 12:44:23 +08:00
Merge pull request #2035 from jparise/lsp_config_callback
Add a `lsp_config_callback` linter option
This commit is contained in:
@@ -85,6 +85,14 @@ function! ale#assert#LSPOptions(expected_options) abort
|
|||||||
AssertEqual a:expected_options, l:initialization_options
|
AssertEqual a:expected_options, l:initialization_options
|
||||||
endfunction
|
endfunction
|
||||||
|
|
||||||
|
function! ale#assert#LSPConfig(expected_config) abort
|
||||||
|
let l:buffer = bufnr('')
|
||||||
|
let l:linter = s:GetLinter()
|
||||||
|
let l:config = ale#lsp_linter#GetConfig(l:buffer, l:linter)
|
||||||
|
|
||||||
|
AssertEqual a:expected_config, l:config
|
||||||
|
endfunction
|
||||||
|
|
||||||
function! ale#assert#LSPLanguage(expected_language) abort
|
function! ale#assert#LSPLanguage(expected_language) abort
|
||||||
let l:buffer = bufnr('')
|
let l:buffer = bufnr('')
|
||||||
let l:linter = s:GetLinter()
|
let l:linter = s:GetLinter()
|
||||||
@@ -147,6 +155,7 @@ function! ale#assert#SetUpLinterTest(filetype, name) abort
|
|||||||
command! -nargs=+ AssertLinter :call ale#assert#Linter(<args>)
|
command! -nargs=+ AssertLinter :call ale#assert#Linter(<args>)
|
||||||
command! -nargs=0 AssertLinterNotExecuted :call ale#assert#LinterNotExecuted()
|
command! -nargs=0 AssertLinterNotExecuted :call ale#assert#LinterNotExecuted()
|
||||||
command! -nargs=+ AssertLSPOptions :call ale#assert#LSPOptions(<args>)
|
command! -nargs=+ AssertLSPOptions :call ale#assert#LSPOptions(<args>)
|
||||||
|
command! -nargs=+ AssertLSPConfig :call ale#assert#LSPConfig(<args>)
|
||||||
command! -nargs=+ AssertLSPLanguage :call ale#assert#LSPLanguage(<args>)
|
command! -nargs=+ AssertLSPLanguage :call ale#assert#LSPLanguage(<args>)
|
||||||
command! -nargs=+ AssertLSPProject :call ale#assert#LSPProject(<args>)
|
command! -nargs=+ AssertLSPProject :call ale#assert#LSPProject(<args>)
|
||||||
command! -nargs=+ AssertLSPAddress :call ale#assert#LSPAddress(<args>)
|
command! -nargs=+ AssertLSPAddress :call ale#assert#LSPAddress(<args>)
|
||||||
@@ -172,6 +181,10 @@ function! ale#assert#TearDownLinterTest() abort
|
|||||||
delcommand AssertLSPOptions
|
delcommand AssertLSPOptions
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
if exists(':AssertLSPConfig')
|
||||||
|
delcommand AssertLSPConfig
|
||||||
|
endif
|
||||||
|
|
||||||
if exists(':AssertLSPLanguage')
|
if exists(':AssertLSPLanguage')
|
||||||
delcommand AssertLSPLanguage
|
delcommand AssertLSPLanguage
|
||||||
endif
|
endif
|
||||||
|
|||||||
@@ -257,7 +257,17 @@ function! ale#linter#PreProcess(filetype, linter) abort
|
|||||||
let l:obj.initialization_options = a:linter.initialization_options
|
let l:obj.initialization_options = a:linter.initialization_options
|
||||||
endif
|
endif
|
||||||
|
|
||||||
if has_key(a:linter, 'lsp_config')
|
if has_key(a:linter, 'lsp_config_callback')
|
||||||
|
if has_key(a:linter, 'lsp_config')
|
||||||
|
throw 'Only one of `lsp_config` or `lsp_config_callback` should be set'
|
||||||
|
endif
|
||||||
|
|
||||||
|
let l:obj.lsp_config_callback = a:linter.lsp_config_callback
|
||||||
|
|
||||||
|
if !s:IsCallback(l:obj.lsp_config_callback)
|
||||||
|
throw '`lsp_config_callback` must be a callback if defined'
|
||||||
|
endif
|
||||||
|
elseif has_key(a:linter, 'lsp_config')
|
||||||
if type(a:linter.lsp_config) isnot v:t_dict
|
if type(a:linter.lsp_config) isnot v:t_dict
|
||||||
throw '`lsp_config` must be a Dictionary'
|
throw '`lsp_config` must be a Dictionary'
|
||||||
endif
|
endif
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ function! ale#lsp#Register(executable_or_address, project, init_options) abort
|
|||||||
" initialized: 0 if the connection is ready, 1 otherwise.
|
" initialized: 0 if the connection is ready, 1 otherwise.
|
||||||
" init_request_id: The ID for the init request.
|
" init_request_id: The ID for the init request.
|
||||||
" init_options: Options to send to the server.
|
" init_options: Options to send to the server.
|
||||||
|
" config: Configuration settings to send to the server.
|
||||||
" callback_list: A list of callbacks for handling LSP responses.
|
" callback_list: A list of callbacks for handling LSP responses.
|
||||||
" message_queue: Messages queued for sending to callbacks.
|
" message_queue: Messages queued for sending to callbacks.
|
||||||
" capabilities_queue: The list of callbacks to call with capabilities.
|
" capabilities_queue: The list of callbacks to call with capabilities.
|
||||||
@@ -32,6 +33,7 @@ function! ale#lsp#Register(executable_or_address, project, init_options) abort
|
|||||||
\ 'initialized': 0,
|
\ 'initialized': 0,
|
||||||
\ 'init_request_id': 0,
|
\ 'init_request_id': 0,
|
||||||
\ 'init_options': a:init_options,
|
\ 'init_options': a:init_options,
|
||||||
|
\ 'config': {},
|
||||||
\ 'callback_list': [],
|
\ 'callback_list': [],
|
||||||
\ 'message_queue': [],
|
\ 'message_queue': [],
|
||||||
\ 'capabilities_queue': [],
|
\ 'capabilities_queue': [],
|
||||||
@@ -210,6 +212,25 @@ function! s:UpdateCapabilities(conn, capabilities) abort
|
|||||||
endif
|
endif
|
||||||
endfunction
|
endfunction
|
||||||
|
|
||||||
|
" Update a connection's configuration dictionary and notify LSP servers
|
||||||
|
" of any changes since the last update. Returns 1 if a configuration
|
||||||
|
" update was sent; otherwise 0 will be returned.
|
||||||
|
function! ale#lsp#UpdateConfig(conn_id, buffer, config) abort
|
||||||
|
let l:conn = get(s:connections, a:conn_id, {})
|
||||||
|
|
||||||
|
if empty(l:conn) || a:config ==# l:conn.config " no-custom-checks
|
||||||
|
return 0
|
||||||
|
endif
|
||||||
|
|
||||||
|
let l:conn.config = a:config
|
||||||
|
let l:message = ale#lsp#message#DidChangeConfiguration(a:buffer, a:config)
|
||||||
|
|
||||||
|
call ale#lsp#Send(a:conn_id, l:message)
|
||||||
|
|
||||||
|
return 1
|
||||||
|
endfunction
|
||||||
|
|
||||||
|
|
||||||
function! ale#lsp#HandleInitResponse(conn, response) abort
|
function! ale#lsp#HandleInitResponse(conn, response) abort
|
||||||
if get(a:response, 'method', '') is# 'initialize'
|
if get(a:response, 'method', '') is# 'initialize'
|
||||||
let a:conn.initialized = 1
|
let a:conn.initialized = 1
|
||||||
|
|||||||
@@ -140,6 +140,18 @@ function! ale#lsp_linter#GetOptions(buffer, linter) abort
|
|||||||
return l:initialization_options
|
return l:initialization_options
|
||||||
endfunction
|
endfunction
|
||||||
|
|
||||||
|
function! ale#lsp_linter#GetConfig(buffer, linter) abort
|
||||||
|
let l:config = {}
|
||||||
|
|
||||||
|
if has_key(a:linter, 'lsp_config_callback')
|
||||||
|
let l:config = ale#util#GetFunction(a:linter.lsp_config_callback)(a:buffer)
|
||||||
|
elseif has_key(a:linter, 'lsp_config')
|
||||||
|
let l:config = a:linter.lsp_config
|
||||||
|
endif
|
||||||
|
|
||||||
|
return l:config
|
||||||
|
endfunction
|
||||||
|
|
||||||
" Given a buffer, an LSP linter, start up an LSP linter and get ready to
|
" Given a buffer, an LSP linter, start up an LSP linter and get ready to
|
||||||
" receive messages for the document.
|
" receive messages for the document.
|
||||||
function! ale#lsp_linter#StartLSP(buffer, linter) abort
|
function! ale#lsp_linter#StartLSP(buffer, linter) abort
|
||||||
@@ -188,14 +200,9 @@ function! ale#lsp_linter#StartLSP(buffer, linter) abort
|
|||||||
call ale#lsp#MarkConnectionAsTsserver(l:conn_id)
|
call ale#lsp#MarkConnectionAsTsserver(l:conn_id)
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
let l:config = ale#lsp_linter#GetConfig(a:buffer, a:linter)
|
||||||
let l:language_id = ale#util#GetFunction(a:linter.language_callback)(a:buffer)
|
let l:language_id = ale#util#GetFunction(a:linter.language_callback)(a:buffer)
|
||||||
|
|
||||||
if !empty(get(a:linter, 'lsp_config'))
|
|
||||||
" set LSP configuration options (workspace/didChangeConfiguration)
|
|
||||||
let l:config_message = ale#lsp#message#DidChangeConfiguration(a:buffer, a:linter.lsp_config)
|
|
||||||
call ale#lsp#Send(l:conn_id, l:config_message)
|
|
||||||
endif
|
|
||||||
|
|
||||||
let l:details = {
|
let l:details = {
|
||||||
\ 'buffer': a:buffer,
|
\ 'buffer': a:buffer,
|
||||||
\ 'connection_id': l:conn_id,
|
\ 'connection_id': l:conn_id,
|
||||||
@@ -204,6 +211,8 @@ function! ale#lsp_linter#StartLSP(buffer, linter) abort
|
|||||||
\ 'language_id': l:language_id,
|
\ 'language_id': l:language_id,
|
||||||
\}
|
\}
|
||||||
|
|
||||||
|
call ale#lsp#UpdateConfig(l:conn_id, a:buffer, l:config)
|
||||||
|
|
||||||
if ale#lsp#OpenDocument(l:conn_id, a:buffer, l:language_id)
|
if ale#lsp#OpenDocument(l:conn_id, a:buffer, l:language_id)
|
||||||
if g:ale_history_enabled && !empty(l:command)
|
if g:ale_history_enabled && !empty(l:command)
|
||||||
call ale#history#Add(a:buffer, 'started', l:conn_id, l:command)
|
call ale#history#Add(a:buffer, 'started', l:conn_id, l:command)
|
||||||
|
|||||||
@@ -306,6 +306,7 @@ given the above setup are as follows.
|
|||||||
`AssertLinterNotExecuted` - Check that linters will not be executed.
|
`AssertLinterNotExecuted` - Check that linters will not be executed.
|
||||||
`AssertLSPLanguage language` - Check the language given to an LSP server.
|
`AssertLSPLanguage language` - Check the language given to an LSP server.
|
||||||
`AssertLSPOptions options_dict` - Check the options given to an LSP server.
|
`AssertLSPOptions options_dict` - Check the options given to an LSP server.
|
||||||
|
`AssertLSPConfig config_dict` - Check the config given to an LSP server.
|
||||||
`AssertLSPProject project_root` - Check the root given to an LSP server.
|
`AssertLSPProject project_root` - Check the root given to an LSP server.
|
||||||
`AssertLSPAddress address` - Check the address to an LSP server.
|
`AssertLSPAddress address` - Check the address to an LSP server.
|
||||||
|
|
||||||
|
|||||||
13
doc/ale.txt
13
doc/ale.txt
@@ -2685,6 +2685,9 @@ ale#linter#Define(filetype, linter) *ale#linter#Define()*
|
|||||||
`initialization_options_callback` may be defined to
|
`initialization_options_callback` may be defined to
|
||||||
pass initialization options to the LSP.
|
pass initialization options to the LSP.
|
||||||
|
|
||||||
|
An optional `lsp_config` or `lsp_config_callback` may
|
||||||
|
be defined to pass configuration settings to the LSP.
|
||||||
|
|
||||||
`address_callback` A |String| or |Funcref| for a callback function
|
`address_callback` A |String| or |Funcref| for a callback function
|
||||||
accepting a buffer number. A |String| should be
|
accepting a buffer number. A |String| should be
|
||||||
returned with an address to connect to.
|
returned with an address to connect to.
|
||||||
@@ -2745,6 +2748,16 @@ ale#linter#Define(filetype, linter) *ale#linter#Define()*
|
|||||||
This can be used in place of `initialization_options`
|
This can be used in place of `initialization_options`
|
||||||
when more complicated processing is needed.
|
when more complicated processing is needed.
|
||||||
|
|
||||||
|
`lsp_config` A |Dictionary| of configuration settings for LSPs.
|
||||||
|
This will be fed (as JSON) to the LSP in the
|
||||||
|
workspace/didChangeConfiguration command.
|
||||||
|
|
||||||
|
`lsp_config_callback` A |String| or |Funcref| for a callback function
|
||||||
|
accepting a buffer number. A |Dictionary| should be
|
||||||
|
returned for configuration settings to pass the LSP.
|
||||||
|
This can be used in place of `lsp_config` when more
|
||||||
|
complicated processing is needed.
|
||||||
|
|
||||||
Only one of `command`, `command_callback`, or `command_chain` should be
|
Only one of `command`, `command_callback`, or `command_chain` should be
|
||||||
specified. `command_callback` is generally recommended when a command string
|
specified. `command_callback` is generally recommended when a command string
|
||||||
needs to be generated dynamically, or any global options are used.
|
needs to be generated dynamically, or any global options are used.
|
||||||
|
|||||||
@@ -26,4 +26,5 @@ Execute(should set correct LSP values):
|
|||||||
|
|
||||||
AssertLSPLanguage 'elixir'
|
AssertLSPLanguage 'elixir'
|
||||||
AssertLSPOptions {}
|
AssertLSPOptions {}
|
||||||
|
AssertLSPConfig {}
|
||||||
AssertLSPProject ale#path#Simplify(g:dir . '/mix_paths/wrapped_project')
|
AssertLSPProject ale#path#Simplify(g:dir . '/mix_paths/wrapped_project')
|
||||||
|
|||||||
@@ -56,7 +56,7 @@ Execute(should set go-langserver for go app1):
|
|||||||
call ale#test#SetFilename('go_paths/go1/prj1/file.go')
|
call ale#test#SetFilename('go_paths/go1/prj1/file.go')
|
||||||
|
|
||||||
AssertLSPLanguage 'go'
|
AssertLSPLanguage 'go'
|
||||||
AssertLSPOptions {}
|
AssertLSPConfig {}
|
||||||
AssertLSPProject ale#path#Simplify(g:dir . '/go_paths/go1')
|
AssertLSPProject ale#path#Simplify(g:dir . '/go_paths/go1')
|
||||||
|
|
||||||
Execute(should set go-langserver for go app2):
|
Execute(should set go-langserver for go app2):
|
||||||
@@ -64,4 +64,5 @@ Execute(should set go-langserver for go app2):
|
|||||||
|
|
||||||
AssertLSPLanguage 'go'
|
AssertLSPLanguage 'go'
|
||||||
AssertLSPOptions {}
|
AssertLSPOptions {}
|
||||||
|
AssertLSPConfig {}
|
||||||
AssertLSPProject ale#path#Simplify(g:dir . '/go_paths/go2')
|
AssertLSPProject ale#path#Simplify(g:dir . '/go_paths/go2')
|
||||||
|
|||||||
@@ -9,11 +9,13 @@ Execute(should set sbtserver for sbt project with build.sbt):
|
|||||||
call ale#test#SetFilename('../scala_fixtures/valid_sbt_project/Main.scala')
|
call ale#test#SetFilename('../scala_fixtures/valid_sbt_project/Main.scala')
|
||||||
AssertLSPLanguage 'scala'
|
AssertLSPLanguage 'scala'
|
||||||
AssertLSPOptions {}
|
AssertLSPOptions {}
|
||||||
|
AssertLSPConfig {}
|
||||||
AssertLSPProject ale#path#Simplify(g:dir . 'command_callback/../scala_fixtures/valid_sbt_project')
|
AssertLSPProject ale#path#Simplify(g:dir . 'command_callback/../scala_fixtures/valid_sbt_project')
|
||||||
AssertLSPAddress '127.0.0.1:4273'
|
AssertLSPAddress '127.0.0.1:4273'
|
||||||
Execute(should not set sbtserver for sbt project without build.sbt):
|
Execute(should not set sbtserver for sbt project without build.sbt):
|
||||||
call ale#test#SetFilename('../scala_fixtures/invalid_sbt_project/Main.scala')
|
call ale#test#SetFilename('../scala_fixtures/invalid_sbt_project/Main.scala')
|
||||||
AssertLSPLanguage 'scala'
|
AssertLSPLanguage 'scala'
|
||||||
AssertLSPOptions {}
|
AssertLSPOptions {}
|
||||||
|
AssertLSPConfig {}
|
||||||
AssertLSPProject ''
|
AssertLSPProject ''
|
||||||
AssertLSPAddress '127.0.0.1:4273'
|
AssertLSPAddress '127.0.0.1:4273'
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ Before:
|
|||||||
\ 'initialized': 0,
|
\ 'initialized': 0,
|
||||||
\ 'init_request_id': 0,
|
\ 'init_request_id': 0,
|
||||||
\ 'init_options': {},
|
\ 'init_options': {},
|
||||||
|
\ 'config': {},
|
||||||
\ 'callback_list': [],
|
\ 'callback_list': [],
|
||||||
\ 'message_queue': [],
|
\ 'message_queue': [],
|
||||||
\ 'capabilities_queue': [],
|
\ 'capabilities_queue': [],
|
||||||
|
|||||||
17
test/lsp/test_update_config.vader
Normal file
17
test/lsp/test_update_config.vader
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
Before:
|
||||||
|
runtime autoload/ale/lsp.vim
|
||||||
|
|
||||||
|
let g:conn_id = ale#lsp#Register('executable', '/foo/bar', {})
|
||||||
|
|
||||||
|
After:
|
||||||
|
Restore
|
||||||
|
|
||||||
|
unlet! g:conn_id
|
||||||
|
|
||||||
|
runtime autoload/ale/lsp.vim
|
||||||
|
|
||||||
|
Execute(Only send updates when the configuration dictionary changes):
|
||||||
|
AssertEqual 0, ale#lsp#UpdateConfig(g:conn_id, bufnr(''), {})
|
||||||
|
AssertEqual 1, ale#lsp#UpdateConfig(g:conn_id, bufnr(''), {'a': 1})
|
||||||
|
AssertEqual 0, ale#lsp#UpdateConfig(g:conn_id, bufnr(''), {'a': 1})
|
||||||
|
AssertEqual 1, ale#lsp#UpdateConfig(g:conn_id, bufnr(''), {})
|
||||||
@@ -501,6 +501,31 @@ Execute(PreProcess should throw when initialization_options_callback is not a ca
|
|||||||
\})
|
\})
|
||||||
AssertEqual '`initialization_options_callback` must be a callback if defined', g:vader_exception
|
AssertEqual '`initialization_options_callback` must be a callback if defined', g:vader_exception
|
||||||
|
|
||||||
|
Execute(PreProcess should complain about using lsp_config and lsp_config_callback together):
|
||||||
|
let g:linter = {
|
||||||
|
\ 'name': 'x',
|
||||||
|
\ 'lsp': 'socket',
|
||||||
|
\ 'address_callback': 'X',
|
||||||
|
\ 'language': 'x',
|
||||||
|
\ 'project_root_callback': 'x',
|
||||||
|
\ 'lsp_config': 'x',
|
||||||
|
\ 'lsp_config_callback': 'x',
|
||||||
|
\}
|
||||||
|
|
||||||
|
AssertThrows call ale#linter#PreProcess('testft', g:linter)
|
||||||
|
AssertEqual 'Only one of `lsp_config` or `lsp_config_callback` should be set', g:vader_exception
|
||||||
|
|
||||||
|
Execute(PreProcess should throw when lsp_config_callback is not a callback):
|
||||||
|
AssertThrows call ale#linter#PreProcess('testft', {
|
||||||
|
\ 'name': 'foo',
|
||||||
|
\ 'lsp': 'socket',
|
||||||
|
\ 'address_callback': 'X',
|
||||||
|
\ 'language': 'x',
|
||||||
|
\ 'project_root_callback': 'x',
|
||||||
|
\ 'lsp_config_callback': {},
|
||||||
|
\})
|
||||||
|
AssertEqual '`lsp_config_callback` must be a callback if defined', g:vader_exception
|
||||||
|
|
||||||
Execute(PreProcess should accept LSP configuration options via lsp_config):
|
Execute(PreProcess should accept LSP configuration options via lsp_config):
|
||||||
let g:ale_lsp_configuration = {
|
let g:ale_lsp_configuration = {
|
||||||
\ 'foo': 'bar'
|
\ 'foo': 'bar'
|
||||||
@@ -517,7 +542,6 @@ Execute(PreProcess should accept LSP configuration options via lsp_config):
|
|||||||
|
|
||||||
AssertEqual {'foo': 'bar'}, ale#linter#PreProcess('testft', g:linter).lsp_config
|
AssertEqual {'foo': 'bar'}, ale#linter#PreProcess('testft', g:linter).lsp_config
|
||||||
|
|
||||||
|
|
||||||
Execute(PreProcess should throw when lsp_config is not a Dictionary):
|
Execute(PreProcess should throw when lsp_config is not a Dictionary):
|
||||||
AssertThrows call ale#linter#PreProcess('testft', {
|
AssertThrows call ale#linter#PreProcess('testft', {
|
||||||
\ 'name': 'foo',
|
\ 'name': 'foo',
|
||||||
@@ -528,4 +552,3 @@ Execute(PreProcess should throw when lsp_config is not a Dictionary):
|
|||||||
\ 'lsp_config': 'x',
|
\ 'lsp_config': 'x',
|
||||||
\})
|
\})
|
||||||
AssertEqual '`lsp_config` must be a Dictionary', g:vader_exception
|
AssertEqual '`lsp_config` must be a Dictionary', g:vader_exception
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user