Skip to content

Commit 123b4de

Browse files
dbarnettsnu5mumr1k
authored andcommitted
Add ruff as formatter for Python
1 parent 6fa1616 commit 123b4de

File tree

6 files changed

+154
-3
lines changed

6 files changed

+154
-3
lines changed

README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,7 @@ codefmt` if codefmt is installed (and helptags have been generated).
3636
* Nix (nixpkgs-fmt)
3737
* OCaml ([ocamlformat](https://github.com/ocaml-ppx/ocamlformat))
3838
* Protocol Buffers (clang-format)
39-
* Python (Autopep8, Black, isort, or YAPF)
39+
* Python (Autopep8, Black, isort, Ruff, or YAPF)
4040
* Ruby ([rubocop](https://rubocop.org))
4141
* Rust ([rustfmt](https://github.com/rust-lang/rustfmt))
4242
* Shell (shfmt)

autoload/codefmt/ruff.vim

+71
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
" Copyright 2017 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+
16+
let s:plugin = maktaba#plugin#Get('codefmt')
17+
18+
19+
function! s:FormatWithArgs(args) abort
20+
let l:executable = s:plugin.Flag('ruff_executable')
21+
let l:lines = getline(1, line('$'))
22+
let l:cmd = [l:executable, 'format'] + a:args
23+
if !empty(@%)
24+
let l:cmd += ['--stdin-filename=' . @%]
25+
endif
26+
let l:input = join(l:lines, "\n")
27+
let l:result = maktaba#syscall#Create(l:cmd).WithStdin(l:input).Call(0)
28+
if v:shell_error
29+
call maktaba#error#Shout('Error formatting file: %s', l:result.stderr)
30+
return
31+
endif
32+
let l:formatted = split(l:result.stdout, "\n")
33+
34+
call maktaba#buffer#Overwrite(1, line('$'), l:formatted)
35+
endfunction
36+
37+
38+
""
39+
" @private
40+
" Formatter: ruff
41+
function! codefmt#ruff#GetFormatter() abort
42+
let l:formatter = {
43+
\ 'name': 'ruff',
44+
\ 'setup_instructions': 'Install ruff ' .
45+
\ '(https://docs.astral.sh/ruff/).'}
46+
47+
function l:formatter.IsAvailable() abort
48+
return executable(s:plugin.Flag('ruff_executable'))
49+
endfunction
50+
51+
function l:formatter.AppliesToBuffer() abort
52+
return codefmt#formatterhelpers#FiletypeMatches(&filetype, 'python')
53+
endfunction
54+
55+
function l:formatter.Format() abort
56+
call s:FormatWithArgs([])
57+
endfunction
58+
59+
""
60+
" Reformat the current buffer with ruff or the binary named in
61+
" @flag(ruff_executable), only targeting the range between {startline} and
62+
" {endline}.
63+
" @throws ShellError
64+
function l:formatter.FormatRange(startline, endline) abort
65+
call maktaba#ensure#IsNumber(a:startline)
66+
call maktaba#ensure#IsNumber(a:endline)
67+
call s:FormatWithArgs(['--range=' . a:startline . ':' . a:endline])
68+
endfunction
69+
70+
return l:formatter
71+
endfunction

doc/codefmt.txt

+5-1
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,7 @@ The current list of defaults by filetype is:
5050
* lua: luaformatterfiveone
5151
* nix: nixpkgs-fmt
5252
* ocaml: ocamlformat
53-
* python: autopep8, black, yapf
53+
* python: autopep8, black, ruff, yapf
5454
* ruby: rubocop
5555
* rust: rustfmt
5656
* sh: shfmt
@@ -98,6 +98,10 @@ The path to the mix executable for Elixir. String, list, or callable that
9898
takes no args and returns a string or a list with command line arguments.
9999
Default: 'mix' `
100100

101+
*codefmt:ruff_executable*
102+
The path to the ruff executable.
103+
Default: 'ruff' `
104+
101105
*codefmt:yapf_executable*
102106
The path to the yapf executable.
103107
Default: 'yapf' `

instant/flags.vim

+4
Original file line numberDiff line numberDiff line change
@@ -87,6 +87,10 @@ call s:plugin.Flag('js_beautify_executable', 'js-beautify')
8787
" takes no args and returns a string or a list with command line arguments.
8888
call s:plugin.Flag('mix_executable', 'mix')
8989

90+
""
91+
" The path to the ruff executable.
92+
call s:plugin.Flag('ruff_executable', 'ruff')
93+
9094
""
9195
" The path to the yapf executable.
9296
call s:plugin.Flag('yapf_executable', 'yapf')

plugin/register.vim

+2-1
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,7 @@
4444
" * lua: luaformatterfiveone
4545
" * nix: nixpkgs-fmt
4646
" * ocaml: ocamlformat
47-
" * python: autopep8, black, yapf
47+
" * python: autopep8, black, ruff, yapf
4848
" * ruby: rubocop
4949
" * rust: rustfmt
5050
" * sh: shfmt
@@ -82,6 +82,7 @@ call s:registry.AddExtension(codefmt#nixpkgs_fmt#GetFormatter())
8282
call s:registry.AddExtension(codefmt#autopep8#GetFormatter())
8383
call s:registry.AddExtension(codefmt#isort#GetFormatter())
8484
call s:registry.AddExtension(codefmt#black#GetFormatter())
85+
call s:registry.AddExtension(codefmt#ruff#GetFormatter())
8586
call s:registry.AddExtension(codefmt#yapf#GetFormatter())
8687
call s:registry.AddExtension(codefmt#rubocop#GetFormatter())
8788
call s:registry.AddExtension(codefmt#rustfmt#GetFormatter())

vroom/ruff.vroom

+71
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
The built-in ruff formatter knows how to format python code.
2+
If you aren't familiar with basic codefmt usage yet, see main.vroom first.
3+
4+
We'll set up codefmt and configure the vroom environment, then jump into some
5+
examples.
6+
7+
:source $VROOMDIR/setupvroom.vim
8+
9+
:let g:repeat_calls = []
10+
:function FakeRepeat(...)<CR>
11+
| call add(g:repeat_calls, a:000)<CR>
12+
:endfunction
13+
:call maktaba#test#Override('repeat#set', 'FakeRepeat')
14+
15+
:call codefmt#SetWhetherToPerformIsAvailableChecksForTesting(0)
16+
17+
18+
The ruff formatter expects the ruff executable to be installed on your
19+
system.
20+
21+
:silent file somefile.py
22+
% f()
23+
:FormatCode ruff
24+
! ruff format --stdin-filename=somefile.py.*
25+
$ f()
26+
27+
The name or path of the ruff executable can be configured via the
28+
ruff_executable flag if the default of "ruff" doesn't work.
29+
30+
:Glaive codefmt ruff_executable='/somepath/ruff'
31+
:FormatCode ruff
32+
! /somepath/ruff format.*
33+
$ f()
34+
:Glaive codefmt ruff_executable='ruff'
35+
36+
37+
You can format any buffer with ruff specifying the formatter explicitly.
38+
39+
@clear
40+
% if True: pass
41+
42+
:FormatCode ruff
43+
! ruff format.*
44+
$ if True:
45+
$ pass
46+
if True:
47+
pass
48+
@end
49+
50+
It can format specific line ranges of code using :FormatLines.
51+
52+
@clear
53+
% some_tuple=( 1,2, 3,'a' );<CR>
54+
|if bar : bar+=1; bar=bar* bar<CR>
55+
|else: bar-=1;
56+
57+
:2,3FormatLines ruff
58+
! ruff format .*--range=2:3.*
59+
$ some_tuple=( 1,2, 3,'a' );
60+
$ if bar:
61+
$ bar += 1
62+
$ bar = bar * bar
63+
$ else:
64+
$ bar -= 1
65+
some_tuple=( 1,2, 3,'a' );
66+
if bar:
67+
bar += 1
68+
bar = bar * bar
69+
else:
70+
bar -= 1
71+
@end

0 commit comments

Comments
 (0)