mix format in vim from anywhere (or just in umbrella apps)

Spoiler alert: This post is about setting up vim so that you run mix format automatically when you save a file and have it detect the nearest .formatter.exs

I have this weird issue with mix format and umbrella apps.

The issue is that mix format assumes you have a .formatter.exs file in the current directory. If you don’t, it doesn’t look upwards in the file tree. It simply assumes you want to run it with the default config. You can change this behaviour by using the --dot-formatter flag to explicitly point to the formatter file you want to use.

Now, in vim you can also use ale to run mix format on save. If you don’t know ale, take the time to do so. I set up ale to do this precisely by adding the following line to my (n)vim config:

let g:ale_fixers['elixir'] = ['mix_format']

Most of the time you will be good to go with this and you won’t find any more issues.

But when I’m working with Elixir umbrella apps, I sometimes cd to the apps/<app> directly so that my Ctrl+P doesn’t get all cluttered by similarly named files from different applications. If you have a .formatter.exs file with custom rules and ale set up to run mix format on file save, things get tricky. mix format won’t detect your config unless you explicitly set the --dot-formatter flag.

As of a few weeks ago, my PR to allow custom mix format options in ale has been accepted.

With this, we can attempt to find the nearest .formatter.exs and dynamically pass the location to ale. Here’s a bit of vimscript to do so (ignore my n00bness writing vimscript):

function! LoadNearestFormatter()
  let l:formatters = []
  let l:directory = fnameescape(expand("%:p:h"))

  for l:fmt in findfile(".formatter.exs", l:directory . ";", -1)
     call insert(l:formatters, l:fmt)
  endfor

  call reverse(l:formatters)

  let g:ale_fixers['elixir'] = ['mix_format']

  if len(l:formatters) > 0
    let g:ale_elixir_mix_format_options = "--dot-formatter " . l:formatters[0]
  endif
endfunction

call LoadNearestFormatter()

If you have a better version of this, please let me know.

That bit of code will look for .formatter.exs files along the file tree and if any is found, it passes that along with the correct option.

You can also define a .formatter.exs in your home and use it as fallback for when no other .formatter.exs is present, but I’d advise against this.

@naps62 has the following vim code instead:

let l:git_root = system("git rev-parse --show-toplevel")[:-2]
let l:fmt = findfile(".formatter.exs", l:git_root)
let g:ale_elixir_mix_format_options = "--dot-formatter " . l:fmt

It’s a lot less verbose and it won’t search for .formatter.exs files in your home directory.

And that’s it! Now you can use ale to automatically run mix format inside umbrella apps.


I tend to write here so you can subscribe. I also tweet and do open source sometimes. If you are into that type of things, hit that follow button.

 
8
Kudos
 
8
Kudos

Now read this

you must return here with a shrubbery: the pixels camp quizshow qualifier treasure hunt

This is the story of how I locked myself inside my room for 29 hours and only left after finishing one of the craziest tech wargames/treasure hunt I have ever taken part in. Prelude # For those of you who don’t know, Pixels Camp has a... Continue →