Fix #363 - Detect virtualenv executables and fix import paths for mypy. Use lint_file for mypy
This commit is contained in:
parent
702b203c51
commit
a9c5e14fc9
@ -1,18 +1,41 @@
|
|||||||
" Author: Keith Smiley <k@keith.so>, w0rp <devw0rp@gmail.com>
|
" Author: Keith Smiley <k@keith.so>, w0rp <devw0rp@gmail.com>
|
||||||
" Description: mypy support for optional python typechecking
|
" Description: mypy support for optional python typechecking
|
||||||
|
|
||||||
|
let g:ale_python_mypy_executable =
|
||||||
|
\ get(g:, 'ale_python_mypy_executable', 'mypy')
|
||||||
let g:ale_python_mypy_options = get(g:, 'ale_python_mypy_options', '')
|
let g:ale_python_mypy_options = get(g:, 'ale_python_mypy_options', '')
|
||||||
|
let g:ale_python_mypy_use_global = get(g:, 'ale_python_mypy_use_global', 0)
|
||||||
|
|
||||||
|
" (cd /home/w0rp/git/wazoku/wazoku-spotlight && /home/w0rp/git/wazoku/wazoku-spotlight/ve-py3/bin/mypy --show-column-numbers /home/w0rp/git/wazoku/wazoku-spotlight/central/models/__init__.py) | grep ^central/models/__init__.p
|
||||||
|
|
||||||
|
function! ale_linters#python#mypy#GetExecutable(buffer) abort
|
||||||
|
if !ale#Var(a:buffer, 'python_mypy_use_global')
|
||||||
|
let l:virtualenv = ale#python#FindVirtualenv(a:buffer)
|
||||||
|
|
||||||
|
if !empty(l:virtualenv)
|
||||||
|
let l:ve_mypy = l:virtualenv . '/bin/mypy'
|
||||||
|
|
||||||
|
if executable(l:ve_mypy)
|
||||||
|
return l:ve_mypy
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
|
||||||
|
return ale#Var(a:buffer, 'python_mypy_executable')
|
||||||
|
endfunction
|
||||||
|
|
||||||
function! ale_linters#python#mypy#GetCommand(buffer) abort
|
function! ale_linters#python#mypy#GetCommand(buffer) abort
|
||||||
let l:automatic_stubs_dir = ale#path#FindNearestDirectory(a:buffer, 'stubs')
|
let l:project_root = ale#python#FindProjectRoot(a:buffer)
|
||||||
" TODO: Add Windows support
|
let l:cd_command = !empty(l:project_root)
|
||||||
let l:automatic_stubs_command = (has('unix') && !empty(l:automatic_stubs_dir))
|
\ ? ale#path#CdString(l:project_root)
|
||||||
\ ? 'MYPYPATH=' . l:automatic_stubs_dir . ' '
|
|
||||||
\ : ''
|
\ : ''
|
||||||
|
let l:executable = ale_linters#python#mypy#GetExecutable(a:buffer)
|
||||||
|
|
||||||
return 'mypy --show-column-numbers '
|
return l:cd_command
|
||||||
|
\ . fnameescape(l:executable)
|
||||||
|
\ . ' --show-column-numbers '
|
||||||
\ . ale#Var(a:buffer, 'python_mypy_options')
|
\ . ale#Var(a:buffer, 'python_mypy_options')
|
||||||
\ . ' %t'
|
\ . ' %s'
|
||||||
endfunction
|
endfunction
|
||||||
|
|
||||||
function! ale_linters#python#mypy#Handle(buffer, lines) abort
|
function! ale_linters#python#mypy#Handle(buffer, lines) abort
|
||||||
@ -23,21 +46,20 @@ function! ale_linters#python#mypy#Handle(buffer, lines) abort
|
|||||||
" Lines like these should be ignored below:
|
" Lines like these should be ignored below:
|
||||||
"
|
"
|
||||||
" file.py:4: note: (Stub files are from https://github.com/python/typeshed)
|
" file.py:4: note: (Stub files are from https://github.com/python/typeshed)
|
||||||
let l:pattern = '\v^[a-zA-Z]?:?[^:]+:(\d+):?(\d+)?: ([^:]+): (.+)$'
|
let l:pattern = '\v^([a-zA-Z]?:?[^:]+):(\d+):?(\d+)?: (error|warning): (.+)$'
|
||||||
let l:output = []
|
let l:output = []
|
||||||
|
let l:buffer_filename = expand('#' . a:buffer . ':p')
|
||||||
|
|
||||||
for l:match in ale#util#GetMatches(a:lines, l:pattern)
|
for l:match in ale#util#GetMatches(a:lines, l:pattern)
|
||||||
if l:match[4] =~# 'Stub files are from'
|
if l:buffer_filename[-len(l:match[1]):] !=# l:match[1]
|
||||||
" The lines telling us where to get stub files from make it so
|
|
||||||
" we can't read the actual errors, so exclude them.
|
|
||||||
continue
|
continue
|
||||||
endif
|
endif
|
||||||
|
|
||||||
call add(l:output, {
|
call add(l:output, {
|
||||||
\ 'lnum': l:match[1] + 0,
|
\ 'lnum': l:match[2] + 0,
|
||||||
\ 'col': l:match[2] + 0,
|
\ 'col': l:match[3] + 0,
|
||||||
\ 'text': l:match[4],
|
\ 'type': l:match[4] =~# 'error' ? 'E' : 'W',
|
||||||
\ 'type': l:match[3] =~# 'error' ? 'E' : 'W',
|
\ 'text': l:match[5],
|
||||||
\})
|
\})
|
||||||
endfor
|
endfor
|
||||||
|
|
||||||
@ -46,7 +68,8 @@ endfunction
|
|||||||
|
|
||||||
call ale#linter#Define('python', {
|
call ale#linter#Define('python', {
|
||||||
\ 'name': 'mypy',
|
\ 'name': 'mypy',
|
||||||
\ 'executable': 'mypy',
|
\ 'executable_callback': 'ale_linters#python#mypy#GetExecutable',
|
||||||
\ 'command_callback': 'ale_linters#python#mypy#GetCommand',
|
\ 'command_callback': 'ale_linters#python#mypy#GetCommand',
|
||||||
\ 'callback': 'ale_linters#python#mypy#Handle',
|
\ 'callback': 'ale_linters#python#mypy#Handle',
|
||||||
|
\ 'lint_file': 1,
|
||||||
\})
|
\})
|
||||||
|
@ -34,6 +34,14 @@ g:ale_python_flake8_options *g:ale_python_flake8_options*
|
|||||||
-------------------------------------------------------------------------------
|
-------------------------------------------------------------------------------
|
||||||
mypy *ale-python-mypy*
|
mypy *ale-python-mypy*
|
||||||
|
|
||||||
|
g:ale_python_mypy_executable *g:ale_python_mypy_executable*
|
||||||
|
*b:ale_python_mypy_executable*
|
||||||
|
Type: |String|
|
||||||
|
Default: `'mypy'`
|
||||||
|
|
||||||
|
This variable can be changed to modify the executable used for mypy.
|
||||||
|
|
||||||
|
|
||||||
g:ale_python_mypy_options *g:ale_python_mypy_options*
|
g:ale_python_mypy_options *g:ale_python_mypy_options*
|
||||||
*b:ale_python_mypy_options*
|
*b:ale_python_mypy_options*
|
||||||
Type: |String|
|
Type: |String|
|
||||||
@ -43,6 +51,18 @@ g:ale_python_mypy_options *g:ale_python_mypy_options*
|
|||||||
invocation.
|
invocation.
|
||||||
|
|
||||||
|
|
||||||
|
g:ale_python_mypy_use_global *g:ale_python_mypy_use_global*
|
||||||
|
*b:ale_python_mypy_use_global*
|
||||||
|
Type: |Number|
|
||||||
|
Default: `0`
|
||||||
|
|
||||||
|
This variable controls whether or not ALE will search for mypy in a
|
||||||
|
virtualenv directory first. If this variable is set to `1`, then ALE will
|
||||||
|
always use |g:ale_python_mypy_executable| for the executable path.
|
||||||
|
|
||||||
|
Both variables can be set with `b:` buffer variables instead.
|
||||||
|
|
||||||
|
|
||||||
-------------------------------------------------------------------------------
|
-------------------------------------------------------------------------------
|
||||||
pylint *ale-python-pylint*
|
pylint *ale-python-pylint*
|
||||||
|
|
||||||
@ -67,6 +87,8 @@ g:ale_python_pylint_options *g:ale_python_pylint_options*
|
|||||||
|
|
||||||
let g:ale_python_pylint_executable = 'python3' " or 'python' for Python 2
|
let g:ale_python_pylint_executable = 'python3' " or 'python' for Python 2
|
||||||
let g:ale_python_pylint_options = '-rcfile /path/to/pylint.rc'
|
let g:ale_python_pylint_options = '-rcfile /path/to/pylint.rc'
|
||||||
|
" The virtualenv detection needs to be disabled.
|
||||||
|
let g:ale_python_pylint_use_global = 0
|
||||||
|
|
||||||
after making sure it's installed for the appropriate Python versions (e.g.
|
after making sure it's installed for the appropriate Python versions (e.g.
|
||||||
`python3 -m pip install --user pylint`).
|
`python3 -m pip install --user pylint`).
|
||||||
@ -77,9 +99,9 @@ g:ale_python_pylint_use_global *g:ale_python_pylint_use_global*
|
|||||||
Type: |Number|
|
Type: |Number|
|
||||||
Default: `0`
|
Default: `0`
|
||||||
|
|
||||||
This variable controls whether or not ALE will search pylint in a virtualenv
|
This variable controls whether or not ALE will search for pylint in a
|
||||||
directory first. If this variable is set to `1`, then ALE will always use
|
virtualenv directory first. If this variable is set to `1`, then ALE will
|
||||||
|g:ale_python_pylint_executable| for the executable path.
|
always use |g:ale_python_pylint_executable| for the executable path.
|
||||||
|
|
||||||
Both variables can be set with `b:` buffer variables instead.
|
Both variables can be set with `b:` buffer variables instead.
|
||||||
|
|
||||||
|
0
test/command_callback/python_paths/with_virtualenv/env/bin/mypy
vendored
Executable file
0
test/command_callback/python_paths/with_virtualenv/env/bin/mypy
vendored
Executable file
73
test/command_callback/test_mypy_command_callback.vader
Normal file
73
test/command_callback/test_mypy_command_callback.vader
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
Before:
|
||||||
|
runtime ale_linters/python/mypy.vim
|
||||||
|
silent! execute 'cd /testplugin/test/command_callback'
|
||||||
|
let g:dir = getcwd()
|
||||||
|
|
||||||
|
After:
|
||||||
|
silent execute 'cd ' . fnameescape(g:dir)
|
||||||
|
" Set the file to something else,
|
||||||
|
" or we'll cause issues when running other tests
|
||||||
|
silent file 'dummy.py'
|
||||||
|
unlet! g:dir
|
||||||
|
|
||||||
|
call ale#linter#Reset()
|
||||||
|
let g:ale_python_mypy_executable = 'mypy'
|
||||||
|
let g:ale_python_mypy_options = ''
|
||||||
|
let g:ale_python_mypy_use_global = 0
|
||||||
|
|
||||||
|
Execute(The mypy callbacks should return the correct default values):
|
||||||
|
AssertEqual
|
||||||
|
\ 'mypy',
|
||||||
|
\ ale_linters#python#mypy#GetExecutable(bufnr(''))
|
||||||
|
AssertEqual
|
||||||
|
\ 'cd ' . g:dir . ' && mypy --show-column-numbers %s',
|
||||||
|
\ ale_linters#python#mypy#GetCommand(bufnr(''))
|
||||||
|
|
||||||
|
Execute(The mypy executable should be configurable, and escaped properly):
|
||||||
|
let g:ale_python_mypy_executable = 'executable with spaces'
|
||||||
|
|
||||||
|
AssertEqual
|
||||||
|
\ 'executable with spaces',
|
||||||
|
\ ale_linters#python#mypy#GetExecutable(bufnr(''))
|
||||||
|
AssertEqual
|
||||||
|
\ 'cd ' . g:dir . ' && executable\ with\ spaces --show-column-numbers %s',
|
||||||
|
\ ale_linters#python#mypy#GetCommand(bufnr(''))
|
||||||
|
|
||||||
|
Execute(The mypy command callback should let you set options):
|
||||||
|
let g:ale_python_mypy_options = '--some-option'
|
||||||
|
|
||||||
|
AssertEqual
|
||||||
|
\ 'cd ' . g:dir . ' && mypy --show-column-numbers --some-option %s',
|
||||||
|
\ ale_linters#python#mypy#GetCommand(bufnr(''))
|
||||||
|
|
||||||
|
Execute(The mypy command should switch directories to the detected project root):
|
||||||
|
silent execute 'file ' . fnameescape(g:dir . '/python_paths/no_virtualenv/subdir/foo/bar.py')
|
||||||
|
|
||||||
|
AssertEqual
|
||||||
|
\ 'mypy',
|
||||||
|
\ ale_linters#python#mypy#GetExecutable(bufnr(''))
|
||||||
|
AssertEqual
|
||||||
|
\ 'cd ' . g:dir . '/python_paths/no_virtualenv/subdir && mypy --show-column-numbers %s',
|
||||||
|
\ ale_linters#python#mypy#GetCommand(bufnr(''))
|
||||||
|
|
||||||
|
Execute(The mypy callbacks should detect virtualenv directories and switch to the project root):
|
||||||
|
silent execute 'file ' . fnameescape(g:dir . '/python_paths/with_virtualenv/subdir/foo/bar.py')
|
||||||
|
|
||||||
|
AssertEqual
|
||||||
|
\ g:dir . '/python_paths/with_virtualenv/env/bin/mypy',
|
||||||
|
\ ale_linters#python#mypy#GetExecutable(bufnr(''))
|
||||||
|
AssertEqual
|
||||||
|
\ 'cd ' . g:dir . '/python_paths/with_virtualenv/subdir && '
|
||||||
|
\ . g:dir . '/python_paths/with_virtualenv/env/bin/mypy --show-column-numbers %s',
|
||||||
|
\ ale_linters#python#mypy#GetCommand(bufnr(''))
|
||||||
|
|
||||||
|
Execute(You should able able to use the global mypy instead):
|
||||||
|
silent execute 'file ' . fnameescape(g:dir . '/python_paths/with_virtualenv/subdir/foo/bar.py')
|
||||||
|
let g:ale_python_mypy_use_global = 1
|
||||||
|
|
||||||
|
AssertEqual
|
||||||
|
\ 'mypy',
|
||||||
|
\ ale_linters#python#mypy#GetExecutable(bufnr(''))
|
||||||
|
AssertEqual
|
||||||
|
\ 'cd ' . g:dir . '/python_paths/with_virtualenv/subdir && mypy --show-column-numbers %s',
|
||||||
|
\ ale_linters#python#mypy#GetCommand(bufnr(''))
|
@ -3,30 +3,45 @@ Before:
|
|||||||
|
|
||||||
After:
|
After:
|
||||||
call ale#linter#Reset()
|
call ale#linter#Reset()
|
||||||
|
silent file something_else.py
|
||||||
|
|
||||||
Execute(The mypy handler should parse lines correctly):
|
Execute(The mypy handler should parse lines correctly):
|
||||||
|
silent file foo/bar/__init__.py
|
||||||
|
|
||||||
AssertEqual
|
AssertEqual
|
||||||
\ [
|
\ [
|
||||||
\ {
|
\ {
|
||||||
\ 'lnum': 4,
|
\ 'lnum': 15,
|
||||||
\ 'col': 0,
|
\ 'col': 3,
|
||||||
\ 'text': "No library stub file for module 'django.db'",
|
\ 'text': 'Argument 1 to "somefunc" has incompatible type "int"; expected "str"',
|
||||||
\ 'type': 'E',
|
\ 'type': 'E',
|
||||||
\ },
|
\ },
|
||||||
\ {
|
\ {
|
||||||
\ 'lnum': 40,
|
\ 'lnum': 72,
|
||||||
\ 'col': 5,
|
\ 'col': 1,
|
||||||
\ 'text': "Some other problem",
|
\ 'text': 'Some warning',
|
||||||
\ 'type': 'E',
|
\ 'type': 'W',
|
||||||
\ },
|
\ },
|
||||||
\ ],
|
\ ],
|
||||||
\ ale_linters#python#mypy#Handle(347, [
|
\ ale_linters#python#mypy#Handle(bufnr(''), [
|
||||||
\ "file.py:4: error: No library stub file for module 'django.db'",
|
\ 'foo/bar/baz.py: note: In class "SomeClass":',
|
||||||
\ 'file.py:4: note: (Stub files are from https://github.com/python/typeshed)',
|
\ 'foo/bar/baz.py:768:38: error: Cannot determine type of ''SOME_SYMBOL''',
|
||||||
\ "file.py:40:5: error: Some other problem",
|
\ 'foo/bar/baz.py: note: In class "AnotherClass":',
|
||||||
|
\ 'foo/bar/baz.py:821:38: error: Cannot determine type of ''SOME_SYMBOL''',
|
||||||
|
\ 'foo/bar/__init__.py:92: note: In module imported here:',
|
||||||
|
\ 'foo/bar/other.py: note: In class "YetAnotherClass":',
|
||||||
|
\ 'foo/bar/other.py:38:44: error: Cannot determine type of ''ANOTHER_SYMBOL''',
|
||||||
|
\ 'foo/bar/__init__.py: note: At top level:',
|
||||||
|
\ 'foo/bar/__init__.py:15:3: error: Argument 1 to "somefunc" has incompatible type "int"; expected "str"',
|
||||||
|
\ 'another_module/bar.py:14: note: In module imported here,',
|
||||||
|
\ 'another_module/__init__.py:16: note: ... from here,',
|
||||||
|
\ 'foo/bar/__init__.py:72:1: warning: Some warning',
|
||||||
\ ])
|
\ ])
|
||||||
|
|
||||||
Execute(The mypy handler should handle Windows names with spaces):
|
Execute(The mypy handler should handle Windows names with spaces):
|
||||||
|
" This test works on Unix, where this is seen as a single filename
|
||||||
|
silent file C:\\something\\with\ spaces.py
|
||||||
|
|
||||||
AssertEqual
|
AssertEqual
|
||||||
\ [
|
\ [
|
||||||
\ {
|
\ {
|
||||||
@ -36,6 +51,6 @@ Execute(The mypy handler should handle Windows names with spaces):
|
|||||||
\ 'type': 'E',
|
\ 'type': 'E',
|
||||||
\ },
|
\ },
|
||||||
\ ],
|
\ ],
|
||||||
\ ale_linters#python#mypy#Handle(347, [
|
\ ale_linters#python#mypy#Handle(bufnr(''), [
|
||||||
\ "C:\something\with spaces.py:4: error: No library stub file for module 'django.db'",
|
\ 'C:\something\with spaces.py:4: error: No library stub file for module ''django.db''',
|
||||||
\ ])
|
\ ])
|
||||||
|
Loading…
Reference in New Issue
Block a user