haskell source code formatter
 
 
Go to file
Lennart Spitzner 44af042436 Merge branch 'release' 2018-03-04 19:03:11 +01:00
doc Add missing code blocks to layout docs 2017-12-09 15:09:22 +00:00
doc-svg-gen Update doc-svg-gen.cabal to prevent new-configure annoyance 2017-12-15 20:55:31 +01:00
src/Language/Haskell Improve layouting of RecordUpd, Fix minor issue for HsLet 2018-02-19 21:33:43 +01:00
src-brittany Correct some commandline help output 2018-02-14 17:18:15 +01:00
src-idemtests Refactor all modules: +Internal; Add public Brittany module 2017-05-22 21:25:08 +02:00
src-libinterfacetests Refactor exposed function (now parsePrintModule) 2017-06-03 18:23:03 +02:00
src-literatetests Improve layouting of RecordUpd, Fix minor issue for HsLet 2018-02-19 21:33:43 +01:00
src-unittests Support hanging type signature config option 2017-12-28 21:38:31 +01:00
srcinc Remove dependency on either package 2017-11-27 18:11:49 +01:00
.gitignore Add dist-newstyle .gitignore 2017-07-28 18:03:21 +02:00
.travis.yml Rename the ghc-8.0.2 stack yaml 2017-12-28 17:30:58 +01:00
ChangeLog.md Bump to 0.9.0.1, Add changelog 2018-02-14 17:18:22 +01:00
LICENSE Add license (AGPLv3) 2017-06-03 18:23:03 +02:00
README.md Fix spelling of my name 2018-02-14 06:54:39 -08:00
Setup.hs Squash previous history 2016-07-30 15:22:07 +02:00
brittany-sample.gif Add sample gif to README.md 2016-08-12 15:01:15 +02:00
brittany.cabal Bump to 0.9.0.1, Add changelog 2018-02-14 17:18:22 +01:00
stack-8.0.2.yaml Add ghc-exactprint-0.5.6.0 to extra-deps in stack.yaml 2018-02-14 15:20:22 +01:00
stack.yaml stack.yaml: update to lts-10.0 2017-12-28 17:30:58 +01:00

README.md

brittany Hackage version Stackage version Build Status

haskell source code formatter

Output sample

(see more examples and comparisons)

This project's goals roughly are to:

  • Always retain the semantics of the source being transformed;
  • Be idempotent (this also directly ensures that only valid haskell is produced);
  • Support the full GHC-haskell syntax including syntactic extensions (but excluding -XCPP which is too hard);
  • Retain newlines and comments unmodified;
  • Be clever about using the available horizontal space while not overflowing it if it cannot be avoided;
  • Be clever about aligning things horizontally (this can be turned off completely however);
  • Have linear complexity in the size of the input.

In theory, the core algorithm inside brittany reaches these goals. It is rather clever about making use of horizontal space while still being linear in the size of the input (although the constant factor is not small). See these examples of clever layouting.

But brittany is not finished yet, and there are some open issues that yet require fixing:

  • only type-signatures and function/value bindings are processed; other module elements (data-decls, classes, instances, imports/exports etc.) are not transformed in any way; this extends to e.g. bindings inside class instance definitions - they won't be touched (yet).
  • By using ghc-exactprint as the parser, brittany supports full GHC including extensions, but some of the less common syntactic elements (even of 2010 haskell) are not handled.
  • There are some known issues regarding handling of in-source comments. There are cases where comments are not copied to the output (this will be detected and the user will get an error); there are other cases where comments are moved slightly; there are also cases where comments result in wonky newline insertion (although this should be a purely aesthetic issue.)
  • There is an open performance issue on large inputs (due to an accidentally quadratic sub-algorithm); noticable for inputs with >1k loc. (fixed in 0.8.0.3)

Try without Installing

You can paste haskell code over here to test how it gets formatted by brittany. (Rg. privacy: the server does log the size of the input, but not the full requests.)

Other usage notes

  • Supports GHC versions 8.0.* and 8.2.*.
  • included in stackage with lts>=10.0 (or nightlies dating to >=2017-11-15)
  • config (file) documentation is lacking.
  • some config values can not be configured via commandline yet.
  • uses/creates user config file in ~/.config/brittany/config.yaml; also reads (the first) brittany.yaml found in current or parent directories.

Installation

  • via cabal "old-build"

    # optionally:
    # mkdir brittany
    # cd brittany
    # cabal sandbox init
    cabal install brittany --bindir=$HOME/.cabal/bin # -w $PATH_TO_GHC_8_0
    
  • via cabal new-build

    cabal unpack brittany
    cd brittany-0.8.0.2
    # cabal new-configure -w $PATH_TO_GHC_8_0
    cabal new-build exe:brittany
    # and it should be safe to just copy the executable, e.g.
    cp `./find dist-newstyle/build/ -type f -name brittany` $HOME/.cabal/bin/
    
  • via stack using a sufficiently recent stackage snapshot (dated to >= 2017-11-15)

    stack install brittany # --resolver lts-10.0
    

    (earlier ltss did not include brittany yet, but the repo should contain a stack.yaml that works with ghc-8.0.)

  • on ArchLinux via the britanny AUR package using aura:

    aura -A brittany
    

Editor Integration

Sublime text

In this gist I have described a haskell setup that includes a shortcut to run brittany formatting.

VSCode

This extension connects commandline brittany to VSCode formatting API. Thanks to @MaxGabriel.

Via HIE

haskell-ide-engine includes a brittany plugin that directly uses the brittany library. Relevant for any editors that properly support the language-server-protocol.

Neovim / Vim 8

The Neoformat plugin comes with support for brittany built in.

Usage

  • Default mode of operation: Transform a single module, from stdin to stdout. Can pass one or multiple files as input, and there is a flag to override them in place instead of using stdout (since 0.9.0.0). So:

    brittany                           # stdin -> stdout
    brittany mysource.hs               # ./mysource.hs -> stdout
    brittany --write-mode=inplace *.hs # apply formatting to all ./*.hs inplace
    
  • For stdin/stdout usage it makes sense to enable certain syntactic extensions by default, i.e. to add something like this to your ~/.config/brittany/config.yaml (execute brittany once to create default):

    conf_forward:
      options_ghc:
      - -XLambdaCase
      - -XMultiWayIf
      - -XGADTs
      - -XPatternGuards
      - -XViewPatterns
      - -XRecursiveDo
      - -XTupleSections
      - -XExplicitForAll
      - -XImplicitParams
      - -XQuasiQuotes
      - -XTemplateHaskell
      - -XBangPatterns
    

Feature Requests, Contribution, Documentation

This currently is a one-person project in the sense that 90% of the code is written by one person. And (unfortunately) it is not my job to keep improving this project. Please forgive that as a consequence my time to invest on new features is rather limited.

Nonetheless I consider it "in active development" :)

One way of speeding things up is to make your own contributions. There is a good amount of high-level documentation at

the documentation index

Note that most development happens on the dev branch of this repository!

License

Copyright (C) 2016-2017 Lennart Spitzner

This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License, version 3, as published by the Free Software Foundation.

This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more details.

You should have received a copy of the GNU Affero General Public License along with this program. If not, see http://www.gnu.org/licenses/.