- 
                Notifications
    You must be signed in to change notification settings 
- Fork 44
Style guide
We use Fourmolu and EditorConfig to enforce style. If you do not format your code before you open a pull request, it will be formatted for you automatically.
Lines of code usually should not exceed 80 columns. In exceptional circumstances, lines may be as long as 100 columns. A limit is needed so that developers can reasonably work on two units side-by-side. The 80 column limit is a common figure that affords most developers ample space. If a block of code is difficult to fit within that limit, it should probably be refactored anyway.
Whitespace is not allowed at the end of lines.
Put top-level pragmas immediately following the relevant declaration:
id :: a -> a
id x = x
{-# INLINE id #-}Import declarations are sorted alphabetically. Imports are not grouped because this interferes with tools that manage them automatically.
Prefer qualified imports for terms, except between closely related modules. Prefer unqualified imports for types, except to disambiguate between similar names.
Use CamelCase names, except in the test suite where prefixes (e.g. test_) have a special meaning.
Don't capitalize all letters when using an abbreviation.
For example, write HttpServer instead of HTTPServer,
except two letter abbreviations which are the entire name (e.g. IO).
Prefer using full names instead of abbreviations when the abbreviation is not obvious.
Do not use prefixes on exported names to replace namespaces unless that helps readability. Even then, consider using a separate import with an appropriate alias. As an example, ‘delay’ below is a generic name, which may be harder to understand locally, so you should provide some local disambiguation.
-- BAD:
import SomeModule
f nc = NetworkConfig.delay nc
-- BAD:
import SomeModule
f NetworkConfig {..} = delay
-- BAD:
import SomeModule as Something
f nc = Something.delay nc
-- GOOD (separate import for the struct):
import SomeModule as NetworkConfig (NetworkConfig(..))
f nc = NetworkConfig.delay nc
-- GOOD:
import SomeModule
f NetworkConfig { delay } = delayHowever, be mindful of the fact that a generic name like delay is harder to understand without context.
Don't use short names like n, sk, f unless their meaning is clear from context (function name, types, other variables, etc.),
or unless they are so abstract that it is difficult to be more precise.
Use singular when naming modules e.g. use Data.Map and Data.ByteString.Internal instead of Data.Maps and Data.ByteString.Internals.
Put test cases and helpers for a module in a module with the corresponding name under the Test module hierarchy, i.e. put tests for a module Data.Foo in Test.Data.Foo.
If data type has only one constructor then this data type name should be same as constructor name:
data User = User Int StringAs above, the constructor of a newtype should be named the same as the type.
Field name for newtype should start with either get or un followed by type name.
For wrappers with monadic semantics it should start with run.
newtype Coin = Coin { getCoin :: Int }Use the DuplicateRecordFields extension to allow for duplicate field names and do not use unqualified field names.
data NetworkConfig = NetworkConfig
    { delay :: Microsecond
    , port  :: Word
    }
f nc = delay nc -- not allowed
f NetworkConfig { delay } = delay  -- NamedFieldPuns: better
-- verbose, but also good
f NetworkConfig { delay = specificName } = specificName 
-- In case of ambiguity, to replace the prefixes below:
import {- not necessarily qualified -} SomeModule as NetworkConfig (NetworkConfig(..))
f nc = NetworkConfig.delay nc  -- allowed: named importPractice test-driven development: every change to the code should be preceded by a failing test.
Write a test case for every user-facing message.
Write complete sentences with correct capitalization and punctuation.
Write documentation for every top-level declaration (function, class, or type). Describe the fields and constructors of every data type. Write an explicit type signature for every top-level function and describe the arguments.
-- | Send a message on a socket.  The socket must be in a connected
-- state.  Returns the number of bytes sent.  Applications are
-- responsible for ensuring that all data has been sent.
send :: Socket      -- ^ Connected socket
     -> ByteString  -- ^ Data to send
     -> IO Int      -- ^ Bytes sentFor functions the documentation should give enough information to apply the function without looking at the function's definition.
-- | Bla bla bla.
data Person = Person
    { age  :: !Int     -- ^ Age
    , name :: !String  -- ^ First name
    }For fields that require longer comments format them like so:
data Record = Record
    { -- | This is a very very very long comment that is split over
      -- multiple lines.
      field1 :: !Text
      -- | This is a second very very very long comment that is split
      -- over multiple lines.
    , field2 :: !Int
    }Separate end-of-line comments from the code using 2 spaces.
data Parser = Parser
    !Int         -- Current position
    !ByteString  -- Remaining input
foo :: Int -> Int
foo n = salt * 32 + 9
  where
    salt = 453645243  -- Magic hash salt.Use in-line links economically. You are encouraged to add links for API names. It is not necessary to add links for all API names in a Haddock comment. We therefore recommend adding a link to an API name if:
- The user might actually want to click on it for more information (in your judgment), and
- Only for the first occurrence of each API name in the comment (don't bother repeating a link)
By default, use the Strict language feature.
Don't use lazy fields or bindings unless the benefit can be demonstrated with a profiling report.
Avoid over-using point-free style. For example, this is hard to read:
-- Bad:
f = (g .) . hDo not mix sum and record types. Prefer using record types instead of raw product types. Do not make product types where there could be any reasonable ambiguity about what the arguments mean. Do not make product types where it’s not clear what the arguments mean, regardless of ambiguity.
-- Bad:
data Exists = Exists Sort Sort Variable Pattern
-- non-commutative operator, non-obvious operand order.
-- sum expression from lower_limit to upper_limit
data Sum = Sum Exp Exp Exp
-- non-commutative operator, non-obvious operand order,
-- non-obvious operand meaning (i.e. many people wouldn't think
-- about the d(exp) part as being part of the integral).
-- integrate expression from lower_limit to upper_limit d(expression)
data Integral = Integral Exp Exp Exp Exp
-- Acceptable:
data Exists = Exists PatternSort VariableSort Variable Pattern
-- commutative operator, ambiguity does not matter
data Add = Add Exp Exp
-- non-commutative operator, but really obvious operator order
-- that everyone knows.
data Div = Div Exp ExpThe package description lists many extensions that are enabled throughout the project.
The following language extensions may be enabled on a per-module basis:
- AllowAmbiguousTypes
- PolyKinds
- TemplateHaskell
Use the class From to define (total) conversions between types.
For example,
instance From String Text where
    from = Text.packInstances should be homomorphisms preserving some structure of the input. If the preserved structure is not obvious, please document it.
Always use at least one type application or annotation with from.
Rationale: The signature of from is very generic and it is difficult to tell how it is being used when its type is inferred.
The following examples are all acceptable:
-- acceptable: both types are explicit and adjacent to 'from'
toText :: String -> Text
toText = from
-- acceptable: uses type application
someFunction =
    let toText = from @String
    in _The history of a pull request should tell a reasonable story, rather than recording an exact history.
At the tip of the pull request, the build must succeed with the --pedantic Stack option,
i.e. with the -Wall -Werror GHC options.
All tests must pass at the tip of the pull request.
If a pull request introduces a bug that is later fixed in the same pull request,
the bug should be removed from history by squashing (using git rebase) its resolution into its introduction.
The --fixup option to git commit is useful for automatically squashing bugs.
A bug may be preserved in the pull request history if it is especially tricky or demonstrates a problem in other code or tools;
in this case the resolution must be rebased to immediately follow the commit that introduced the bug.
Pull requests will be squashed and merged after approval.
Every pull request will be reviewed by another member before merging. The reviewer will use the review checklist in the pull request template to evaluate the pull request.