Skip to content

Commit

Permalink
Add a terraform linter
Browse files Browse the repository at this point in the history
This linter uses the check functionality built into terraform. ALE
already has a fixer using `terraform fmt` but this doesn't provide error
messages. ALE already has a linter using `tflint` but this requires an
extra application to be installed.

For example this linter will give a warning that ! is an illegal
character in the line below:

    variable "example" !{}

This linter runs the buffer through the command below and parses the
output:

    terraform fmt -no-color -check=true -

This commit includes a basic implementation, documentation and tests.
The only option is to control which executable is run.

Tested with:

    $ terraform -version
    Terraform v0.11.13
  • Loading branch information
maxwell-k committed May 23, 2019
1 parent 1a9b8a5 commit 88fa0b9
Show file tree
Hide file tree
Showing 6 changed files with 108 additions and 3 deletions.
49 changes: 49 additions & 0 deletions ale_linters/terraform/terraform.vim
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
" Author: Keith Maxwell <keith.maxwell@gmail.com>
" Description: terraform fmt to check for errors

call ale#Set('terraform_terraform_executable', 'terraform')

function! ale_linters#terraform#terraform#GetExecutable(buffer) abort
return ale#Var(a:buffer, 'terraform_terraform_executable')
endfunction

function! ale_linters#terraform#terraform#GetCommand(buffer) abort
return ale#Escape(ale_linters#terraform#terraform#GetExecutable(a:buffer))
\ . ' fmt -no-color --check=true -'
endfunction

function! ale_linters#terraform#terraform#Handle(buffer, lines) abort
let l:head = '^Error running fmt: In <standard input>: '
let l:output = []
let l:patterns = [
\ l:head.'At \(\d\+\):\(\d\+\): \(.*\)$',
\ l:head.'\(.*\)$'
\]

for l:match in ale#util#GetMatches(a:lines, l:patterns)
if len(l:match[2]) > 0
call add(l:output, {
\ 'lnum': str2nr(l:match[1]),
\ 'col': str2nr(l:match[2]),
\ 'text': l:match[3],
\ 'type': 'E',
\})
else
call add(l:output, {
\ 'lnum': line('$'),
\ 'text': l:match[1],
\ 'type': 'E',
\})
endif
endfor

return l:output
endfunction

call ale#linter#Define('terraform', {
\ 'name': 'terraform',
\ 'output_stream': 'stderr',
\ 'executable': function('ale_linters#terraform#terraform#GetExecutable'),
\ 'command': function('ale_linters#terraform#terraform#GetCommand'),
\ 'callback': 'ale_linters#terraform#terraform#Handle',
\})
2 changes: 1 addition & 1 deletion doc/ale-hcl.txt
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ ALE HCL Integration *ale-hcl-options*
===============================================================================
terraform-fmt *ale-hcl-terraform-fmt*

See |ale-terraform-fmt| for information about the available options.
See |ale-terraform-fmt-fixer| for information about the available options.

===============================================================================
vim:tw=78:ts=2:sts=2:sw=2:ft=help:norl:
14 changes: 13 additions & 1 deletion doc/ale-terraform.txt
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ ALE Terraform Integration *ale-terraform-options*


===============================================================================
fmt *ale-terraform-fmt*
terraform-fmt-fixer *ale-terraform-fmt-fixer*

g:ale_terraform_fmt_executable *g:ale_terraform_fmt_executable*
*b:ale_terraform_fmt_executable*
Expand All @@ -20,6 +20,18 @@ g:ale_terraform_fmt_options *g:ale_terraform_fmt_options*
Default: `''`


===============================================================================
terraform *ale-terraform-terraform*

g:ale_terraform_terraform_executable *g:ale_terraform_terraform_executable*
*b:ale_terraform_terraform_executable*

Type: |String|
Default: `'terraform'`

This variable can be changed to use a different executable for terraform.


===============================================================================
tflint *ale-terraform-tflint*

Expand Down
3 changes: 2 additions & 1 deletion doc/ale.txt
Original file line number Diff line number Diff line change
Expand Up @@ -2246,7 +2246,8 @@ documented in additional help files.
tcl.....................................|ale-tcl-options|
nagelfar..............................|ale-tcl-nagelfar|
terraform...............................|ale-terraform-options|
fmt...................................|ale-terraform-fmt|
terraform-fmt-fixer...................|ale-terraform-fmt-fixer|
terraform.............................|ale-terraform-terraform|
tflint................................|ale-terraform-tflint|
tex.....................................|ale-tex-options|
chktex................................|ale-tex-chktex|
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
" Based upon :help ale-development
Before:
call ale#assert#SetUpLinterTest('terraform', 'terraform')

After:
call ale#assert#TearDownLinterTest()

Execute(The default command should be correct):
AssertLinter 'terraform', ale#Escape('terraform') . ' fmt -no-color --check=true -'
34 changes: 34 additions & 0 deletions test/handler/test_terraform_handler.vader
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
Before:
" Load the file which defines the linter.
runtime ale_linters/terraform/terraform.vim

After:
" Unload all linters again.
call ale#linter#Reset()

Execute(The output should be correct):
AssertEqual
\ [
\ {
\ 'lnum': 1,
\ 'col': 20,
\ 'type': 'E',
\ 'text': 'illegal char',
\ },
\ {
\ 'lnum': 2,
\ 'col': 14,
\ 'type': 'E',
\ 'text': 'literal not terminated',
\ },
\ {
\ 'lnum': 1,
\ 'type': 'E',
\ 'text': 'object expected closing RBRACE got: EOF',
\ },
\ ],
\ ale_linters#terraform#terraform#Handle(bufnr(''), [
\ 'Error running fmt: In <standard input>: At 1:20: illegal char',
\ 'Error running fmt: In <standard input>: At 2:14: literal not terminated',
\ 'Error running fmt: In <standard input>: object expected closing RBRACE got: EOF',
\ ])

0 comments on commit 88fa0b9

Please sign in to comment.