Skip to content

Commit f30608c

Browse files
authored
Add ruby formatting support using Rubocop. (#220)
Note: This change is based on an abandoned change by zinovyev #132
1 parent 09ee74c commit f30608c

File tree

7 files changed

+151
-5
lines changed

7 files changed

+151
-5
lines changed

README.md

+1
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@ helpfiles in the `doc/` directory. The helpfiles are also available via
2828
* OCaml ([ocamlformat](https://github.com/ocaml-ppx/ocamlformat))
2929
* Proto (clang-format)
3030
* Python (Autopep8, Black, or YAPF)
31+
* Ruby ([rubocop](https://rubocop.org))
3132
* Rust ([rustfmt](https://github.com/rust-lang/rustfmt))
3233
* TypeScript (clang-format)
3334
* Shell (shfmt)

autoload/codefmt/formatterhelpers.vim

+19-4
Original file line numberDiff line numberDiff line change
@@ -75,6 +75,7 @@ endfunction
7575

7676
""
7777
" @public
78+
" @usage startline endline cmd \[ignoreerrors] \[skipfirstnlines]
7879
" Attempt to format a range of lines from {startline} to {endline} in the
7980
" current buffer via a formatter that doesn't natively support range
8081
" formatting, which is invoked via {cmd} (a system call represented by either
@@ -87,21 +88,35 @@ endfunction
8788
" the tool for range formatting and post a URL for that feature request above
8889
" code that calls it.
8990
"
90-
" @throws ShellError if the {cmd} system call fails
91+
" If [ignoreerrors] is nonzero, the syscall ignores errors. This can be helpful
92+
" for formatters that return nonzero results for reasons unrelated to
93+
" formatting. If [skipfirstnlines] is set to a nonzero number N, the first
94+
" N lines of the formatter output are trimmed. This can be used to trim
95+
" always-present headers.
96+
"
97+
" @throws ShellError if the {cmd} system call fails (and [ignoreerrors] is 0)
9198
" @throws WrongType
9299
function! codefmt#formatterhelpers#AttemptFakeRangeFormatting(
93-
\ startline, endline, cmd) abort
100+
\ startline, endline, cmd, ...) abort
94101
call maktaba#ensure#IsNumber(a:startline)
95102
call maktaba#ensure#IsNumber(a:endline)
96103

104+
let l:ignoreerrors = a:0 >= 1 ? a:1 : 0
105+
let l:skipfirstnlines = a:0 >= 2 ? a:2 : 0
106+
107+
call maktaba#ensure#IsNumber(l:ignoreerrors)
108+
call maktaba#ensure#IsNumber(l:skipfirstnlines)
109+
97110
let l:lines = getline(1, line('$'))
98111
let l:input = join(l:lines[a:startline - 1 : a:endline - 1], "\n")
99112

100-
let l:result = maktaba#syscall#Create(a:cmd).WithStdin(l:input).Call()
113+
let l:result =
114+
\ maktaba#syscall#Create(a:cmd).WithStdin(l:input).Call(!l:ignoreerrors)
101115
let l:formatted = split(l:result.stdout, "\n")
102116
" Special case empty slice: neither l:lines[:0] nor l:lines[:-1] is right.
103117
let l:before = a:startline > 1 ? l:lines[ : a:startline - 2] : []
104-
let l:full_formatted = l:before + l:formatted + l:lines[a:endline :]
118+
let l:full_formatted = l:before + l:formatted[l:skipfirstnlines :]
119+
\ + l:lines[a:endline :]
105120

106121
call maktaba#buffer#Overwrite(1, line('$'), l:full_formatted)
107122
endfunction

autoload/codefmt/rubocop.vim

+64
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
" Copyright 2023 Google Inc. All rights reserved.
2+
"
3+
" Licensed under the Apache License, Version 2.0 (the "License");
4+
" you may not use this file except in compliance with the License.
5+
" You may obtain a copy of the License at
6+
"
7+
" http://www.apache.org/licenses/LICENSE-2.0
8+
"
9+
" Unless required by applicable law or agreed to in writing, software
10+
" distributed under the License is distributed on an "AS IS" BASIS,
11+
" WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
" See the License for the specific language governing permissions and
13+
" limitations under the License.
14+
15+
let s:plugin = maktaba#plugin#Get('codefmt')
16+
17+
18+
""
19+
" @private
20+
" Formatter: rubocop
21+
function! codefmt#rubocop#GetFormatter() abort
22+
let l:formatter = {
23+
\ 'name': 'rubocop',
24+
\ 'setup_instructions': 'Install rubocop ' .
25+
\ '(https://rubygems.org/gems/rubocop).'}
26+
27+
function l:formatter.IsAvailable() abort
28+
return executable(s:plugin.Flag('rubocop_executable'))
29+
endfunction
30+
31+
function l:formatter.AppliesToBuffer() abort
32+
return &filetype is# 'eruby' || &filetype is# 'ruby'
33+
endfunction
34+
35+
""
36+
" Reformat the current buffer with rubocop or the binary named in
37+
" @flag(rubocop_executable), only targeting the range between {startline} and
38+
" {endline}.
39+
"
40+
" @throws ShellError
41+
function l:formatter.FormatRange(startline, endline) abort
42+
" See flag explanations at:
43+
" https://docs.rubocop.org/rubocop/1.51/usage/basic_usage.html
44+
let l:cmd = [s:plugin.Flag('rubocop_executable'), '--stdin', @%, '-a', '--no-color', '-fq', '-o', '/dev/null']
45+
46+
" Rubocop exits with an error condition if there are lint errors, even
47+
" after successfully formatting. This is annoying for our purpuoses,
48+
" because we have no way to distinguish lint errors from a 'real' falure.
49+
" Use Call(0) to suppress maktaba's error handling.
50+
let l:ignoreerrors = 1
51+
" Rubocop is primarily a linter, and by default it outputs lint errors
52+
" first, followed by a dividing line, and then the formatted result.
53+
" '-o /dev/null' in the command line suppresses any lint errors, but the
54+
" divider is always printed.
55+
let l:skipfirstnlines = 1
56+
57+
" Rubocop does not support range formatting; see bug:
58+
" https://github.com/Shopify/ruby-lsp/issues/203
59+
call codefmt#formatterhelpers#AttemptFakeRangeFormatting(
60+
\ a:startline, a:endline, l:cmd, l:ignoreerrors, l:skipfirstnlines)
61+
endfunction
62+
63+
return l:formatter
64+
endfunction

doc/codefmt.txt

+1-1
Original file line numberDiff line numberDiff line change
@@ -383,7 +383,7 @@ codefmt#formatterhelpers#Format({cmd}) *codefmt#formatterhelpers#Format()*
383383
Throws ERROR(ShellError) if the {cmd} system call fails
384384

385385
codefmt#formatterhelpers#AttemptFakeRangeFormatting({startline}, {endline},
386-
{cmd}) *codefmt#formatterhelpers#AttemptFakeRangeFormatting()*
386+
{cmd}, {ignoreerrors}, {skipfirstnlines}) *codefmt#formatterhelpers#AttemptFakeRangeFormatting()*
387387
Attempt to format a range of lines from {startline} to {endline} in the
388388
current buffer via a formatter that doesn't natively support range
389389
formatting, which is invoked via {cmd} (a system call represented by either

instant/flags.vim

+4
Original file line numberDiff line numberDiff line change
@@ -174,6 +174,10 @@ call s:plugin.Flag('prettier_options', [])
174174
" The path to the swift-format executable.
175175
call s:plugin.Flag('swift_format_executable', 'swift-format')
176176

177+
""
178+
" The path to the Rubocop executable.
179+
call s:plugin.Flag('rubocop_executable', 'rubocop')
180+
177181
""
178182
" @private
179183
function s:LookupPrettierExecutable() abort

plugin/register.vim

+2
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,7 @@
4444
" * nix: nixpkgs-fmt
4545
" * ocaml: ocamlformat
4646
" * python: autopep8, black, yapf
47+
" * ruby: rubocop
4748
" * rust: rustfmt
4849
" * sh: shfmt
4950
" * swift: swift-format
@@ -80,6 +81,7 @@ call s:registry.AddExtension(codefmt#autopep8#GetFormatter())
8081
call s:registry.AddExtension(codefmt#isort#GetFormatter())
8182
call s:registry.AddExtension(codefmt#black#GetFormatter())
8283
call s:registry.AddExtension(codefmt#yapf#GetFormatter())
84+
call s:registry.AddExtension(codefmt#rubocop#GetFormatter())
8385
call s:registry.AddExtension(codefmt#rustfmt#GetFormatter())
8486
call s:registry.AddExtension(codefmt#shfmt#GetFormatter())
8587
call s:registry.AddExtension(codefmt#swiftformat#GetFormatter())

vroom/rubocop.vroom

+60
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,60 @@
1+
Rubocop is a linter and formatter for ruby.
2+
If you aren't familiar with basic codefmt usage yet, see main.vroom first.
3+
4+
First, set up the vroom environment.
5+
6+
:source $VROOMDIR/setupvroom.vim
7+
8+
:let g:repeat_calls = []
9+
:function FakeRepeat(...)<CR>
10+
| call add(g:repeat_calls, a:000)<CR>
11+
:endfunction
12+
:call maktaba#test#Override('repeat#set', 'FakeRepeat')
13+
14+
:call codefmt#SetWhetherToPerformIsAvailableChecksForTesting(0)
15+
16+
By default, the rubocop executable is called. To use this plugin, rubocop
17+
must be installed on your system. (But not for testing; vroom intercepts
18+
system calls.)
19+
:FormatCode rubocop
20+
! rubocop .*
21+
22+
23+
24+
The name and path of the Rubocop executable can be configured with a flag:
25+
:Glaive codefmt rubocop_executable=some_other_program
26+
:FormatCode rubocop
27+
! some_other_program .*-fq.*
28+
:Glaive codefmt rubocop_executable=rubocop
29+
30+
31+
32+
Rubocop does basic whitespace management. Though because it's primarily a
33+
linter, it outputs lint errors first, then a separator. Even with the lint
34+
errors disabled (-fq), you still get the separator.
35+
36+
% def SomeClass <CR> <CR> end
37+
:FormatCode rubocop
38+
! rubocop .*-fq.*
39+
$ =========
40+
$ def SomeClass
41+
$ end
42+
def SomeClass
43+
end
44+
45+
46+
47+
Being a linter, Rubocop cares about style as well as formatting.
48+
When a buffer is stylistically fine, it returns 0, and everything is OK.
49+
But sometimes it will return 1 even though things have gone well;
50+
we should still use the output when that happens.
51+
52+
% def SomeClass <CR> <CR> end
53+
:FormatCode rubocop
54+
! rubocop .*
55+
$ =========
56+
$ def SomeClass
57+
$ end
58+
$ 1 (status)
59+
def SomeClass
60+
end

0 commit comments

Comments
 (0)