-
Notifications
You must be signed in to change notification settings - Fork 12
Coding Standards
This file contains agreed upon coding standards and best practices as well as, proposals for changes or new standards. Proposals are prefixed with [PROPOSAL]
and are voted by the wallet backend team through poll on Slack. To be accepted, a practice should be voted with majority + 1, neutral votes counting as positive.
Each proposal should start with a section justifying for the standard with rational arguments. When it makes sense, we may also add examples of good and bad practices to make the point clearer.
Why
This is the kind of details we don't want to be fighting over constantly. The
.editorconfig
are widely used, easily written and supported by any decent editor out there. Agreeing on such rules prevent our version control system to go crazy because people use different encoding or indentation style. It makes the overall code more consistent.
A .editorconfig
(see https://editorconfig.org/) at the root of the project specifies for various filetype:
- Line length
- Indentation style (spaces vs tabs)
- Encoding
This file should be parsed and enforced by any contributor's editor.
Why
- To maximize readability. Human eyes find it much harder to scan long lines. For people with imperfect vision, it can be even harder. Narrower code can be read quickly without having to scan from side to side. Although monitors have grown in width and resolution in recent years, human eyes haven't changed.
- To easily view multiple sources side-by-side. This is particularly important when working on a laptop. With a readable font size of around 11 pt, 80 characters is about half the horizontal distance across a laptop monitor. Trying to fit 90 or 100 characters into the same width requires a smaller font, raising the level of discomfort for people with poorer vision.
- To avoid horizontal scrolling when viewing source code. When reading most source code, we already have to scroll vertically. Horizontal scrolling means that readers have to move their viewpoint in two dimensions rather than one. This requires more effort and can cause strain for people reading your code.
Source code, including comments, should not exceed 80 characters in length.
See Details and Examples
If you find yourself exceeding 80 characters, there are several strategies you can use.
By inserting carriage returns in the right place, we can often reveal the underlying structure of an expression. Haskell allows you to break up long expressions so that they occur over multiple lines. For example:
-- BAD
instance Bi Block where
encode block = encodeListLen 3 <> encode (blockHeader block) <> encode (blockBody block) <> encode (blockExtraData block)
-- GOOD
instance Bi Block where
encode block =
encodeListLen 3
<> encode (blockHeader block)
<> encode (blockBody block)
<> encode (blockExtraData block)
-- BAD
mkMagicalBlock
:: MagicProtocolId -- A unique key specifying a magic protocol.
-> MagicType -- The type of magic used in this block signing.
-> MagicalKey -- The magical key used in this block signing.
-> Maybe Delegation.MagicalCertificate -- A magical certificate of delegation, in case the specified 'MagicalKey' does not have the right to sign this block.
-> Block
-- GOOD
mkMagicalBlock
:: MagicProtocolId
-- ^ A unique key specifying a magic protocol.
-> MagicType
-- ^ The type of magic used in this block signing.
-> MagicalKey
-- ^ The magical key used in this block signing.
-> Maybe Delegation.MagicalCertificate
-- ^ A magical certificate of delegation, in case the specified
-- 'MagicalKey' does not have the right to sign this block.
-> Block
Haskell provides convenient support for multi-line string literals:
-- BAD
errorAccountFundsCompletelyExhausted = "The funds in this account have been completely spent, and its balance is now zero. Either add more funds to this account or use a different account for this transaction."
-- GOOD
errorAccountFundsCompletelyExhausted =
"The funds in this account have been completely spent, and its balance \
\is now zero. Either add more funds to this account or use a different \
\account for this transaction."
If your function contains so many levels of nesting that it's hard to keep things within 80 characters (even with careful use of wrapping), consider breaking your function up into smaller parts.
Why
Haskellers have a tendency to over-align everything vertically for the sake of readability. In practice, this is much more of an habit than a real gain in readability. Aligning content based on a function name, variable name or record field tends to create unnecessary long diffs and needless conflicts in version control systems when making a change to add an argument, variable or parameters. Favoring new-line and fixed-length alignment plays nicer with version control.
Variables, arguments, fields and tokens in general shouldn't be aligned based on the length of a previous token. Rather, tokens should go over a new line and be indented one-level extra when it makes sense, or not be aligned at all.
See Examples
-- GOOD
data AddressPool address = AddressPool
{ _addresses :: !(Map address Word32)
, _gap :: !AddressPoolGap
}
-- GOOD
data AddressPool address = AddressPool
{ _addresses
:: !(Map address Word32)
, _gap
:: !AddressPoolGap
}
-- GOOD
deriveAccountPrivateKey
:: PassPhrase -- Passphrase used to encrypt Master Private Key
-> EncryptedSecretKey -- Master Private Key
-> Word32 -- Hardened Account Key Index
-> Maybe EncryptedSecretKey -- Account Private Key
deriveAccountPrivateKey passPhrase masterEncPrvKey accountIx =
-- BAD
myFunction :: Word64 -> Maybe String
myFunction w = let res = Wrap w in
case someOp res of
Left _err -> Nothing
Right () -> Just coin
-- BAD
myFunction :: Int
-> Maybe ByteString
-> Set Word32
-> Update DB (Either [Word32]
(Map Word32 ([String], Set ByteString)))
-- BAD
data MyRecord = MyRecord
{ _myRecordLongNameField :: !String
, _myRecordShort :: ![Int]
}
Why
Type-aliases convey a false sense of type-safety. While they usually make things a bit better for the reader, they have a tendency to spread through the code-base transforming those sweet help spot into traps. We can't define proper instances on type aliases, and we treat them as different type whereas they are behind the scene, just another one.
Instead of writing type aliases, one should favor wrapping up values in newtype when it makes sense, or, have them wrapped into a tagged type with a phantom type to convey some extra meaning while still preserving type safeness. By using newtypes, we actually extend our program vocabulary and increase its robustness.
See examples
-- GOOD
newtype HardenedIndex = HardenedIndex { getHardenedIndex :: Word32 }
deriveAccount :: HardenedIndex -> XPrv -> XPrv
-- GOOD
data Scheme = Seq | Rnd
newtype Key (* :: Scheme) = Key { getKey :: XPrv }
deriveAccount :: Word32 -> Key 'Seq -> Key 'Seq
-- GOOD
newtype Tagged (* :: Symbol) = Tagged { getTagged :: String }
startNode :: Tagged "nodeId" -> IO ()
-- BAD
type HardenedIndex = Word32
deriveAccount :: HardenedIndex -> XPrv -> XPrv