mirror of
https://github.com/lightningnetwork/lnd.git
synced 2025-08-25 05:02:32 +02:00
Merge pull request #9696 from yyforyongyu/add-aider
Add `development_guidelines.md` for both human and machine
This commit is contained in:
4
.github/pull_request_template.md
vendored
4
.github/pull_request_template.md
vendored
@@ -12,8 +12,8 @@ Steps for reviewers to follow to test the change.
|
||||
|
||||
### Code Style and Documentation
|
||||
- [ ] The change is not [insubstantial](https://github.com/lightningnetwork/lnd/blob/master/docs/code_contribution_guidelines.md#substantial-contributions-only). Typo fixes are not accepted to fight bot spam.
|
||||
- [ ] The change obeys the [Code Documentation and Commenting](https://github.com/lightningnetwork/lnd/blob/master/docs/code_contribution_guidelines.md#code-documentation-and-commenting) guidelines, and lines wrap at 80.
|
||||
- [ ] Commits follow the [Ideal Git Commit Structure](https://github.com/lightningnetwork/lnd/blob/master/docs/code_contribution_guidelines.md#ideal-git-commit-structure).
|
||||
- [ ] The change obeys the [Code Documentation and Commenting](https://github.com/lightningnetwork/lnd/blob/master/docs/development_guidelines.md#code-documentation-and-commenting) guidelines, and lines wrap at 80.
|
||||
- [ ] Commits follow the [Ideal Git Commit Structure](https://github.com/lightningnetwork/lnd/blob/master/docs/development_guidelines.md#ideal-git-commit-structure).
|
||||
- [ ] Any new logging statements use an appropriate subsystem and logging level.
|
||||
- [ ] Any new lncli commands have appropriate tags in the comments for the rpc in the proto file.
|
||||
- [ ] [There is a change description in the release notes](https://github.com/lightningnetwork/lnd/tree/master/docs/release-notes), or `[skip ci]` in the commit message for small changes.
|
||||
|
@@ -5,16 +5,8 @@
|
||||
4. [Substantial contributions only](#substantial-contributions-only)
|
||||
5. [Development Practices](#development-practices)
|
||||
1. [Share Early, Share Often](#share-early-share-often)
|
||||
1. [Testing](#testing)
|
||||
1. [Code Documentation and Commenting](#code-documentation-and-commenting)
|
||||
1. [Model Git Commit Messages](#model-git-commit-messages)
|
||||
1. [Ideal Git Commit Structure](#ideal-git-commit-structure)
|
||||
1. [Sign Your Git Commits](#sign-your-git-commits)
|
||||
1. [Code Spacing and formatting](#code-spacing-and-formatting)
|
||||
1. [Pointing to Remote Dependent Branches in Go Modules](#pointing-to-remote-dependent-branches-in-go-modules)
|
||||
1. [Use of Log Levels](#use-of-log-levels)
|
||||
1. [Use of Golang submodules](#use-of-golang-submodules)
|
||||
6. [Code Approval Process](#code-approval-process)
|
||||
1. [Development Guidelines](#development-guidelines)
|
||||
5. [Code Approval Process](#code-approval-process)
|
||||
1. [Code Review](#code-review)
|
||||
1. [Rework Code (if needed)](#rework-code-if-needed)
|
||||
1. [Acceptance](#acceptance)
|
||||
@@ -146,341 +138,23 @@ This approach has several benefits:
|
||||
- The quicker your changes are merged to master, the less time you will need to
|
||||
spend rebasing and otherwise trying to keep up with the main code base
|
||||
|
||||
## Testing
|
||||
## Development Guidelines
|
||||
|
||||
One of the major design goals of all of `lnd`'s packages and the daemon itself is
|
||||
to aim for a high degree of test coverage. This is financial software so bugs
|
||||
and regressions in the core logic can cost people real money. For this reason
|
||||
every effort must be taken to ensure the code is as accurate and bug-free as
|
||||
possible. Thorough testing is a good way to help achieve that goal.
|
||||
The `lnd` project emphasizes code readability and maintainability through
|
||||
specific development guidelines. Key aspects include: thorough code
|
||||
documentation with clear function comments and meaningful inline comments;
|
||||
consistent code spacing to separate logical blocks; adherence to an 80-character
|
||||
line limit with specific rules for wrapping function calls, definitions, and log
|
||||
messages (including structured logging); comprehensive unit and integration
|
||||
testing for all changes; well-structured Git commit messages with package
|
||||
prefixes and atomic commits; signing Git commits; proper handling of Go module
|
||||
dependencies and submodules; and appropriate use of log levels. Developers are
|
||||
encouraged to configure their editors to align with these standards.
|
||||
|
||||
Unless a new feature you submit is completely trivial, it will probably be
|
||||
rejected unless it is also accompanied by adequate test coverage for both
|
||||
positive and negative conditions. That is to say, the tests must ensure your
|
||||
code works correctly when it is fed correct data as well as incorrect data
|
||||
(error paths).
|
||||
For the complete set of rules and examples, please refer to the detailed
|
||||
[Development Guidelines](development_guidelines.md).
|
||||
|
||||
|
||||
Go provides an excellent test framework that makes writing test code and
|
||||
checking coverage statistics straightforward. For more information about the
|
||||
test coverage tools, see the [golang cover blog post](https://blog.golang.org/cover).
|
||||
|
||||
A quick summary of test practices follows:
|
||||
- All new code should be accompanied by tests that ensure the code behaves
|
||||
correctly when given expected values, and, perhaps even more importantly, that
|
||||
it handles errors gracefully
|
||||
- When you fix a bug, it should be accompanied by tests which exercise the bug
|
||||
to both prove it has been resolved and to prevent future regressions
|
||||
- Changes to publicly exported packages such as
|
||||
[brontide](https://github.com/lightningnetwork/lnd/tree/master/brontide) should
|
||||
be accompanied by unit tests exercising the new or changed behavior.
|
||||
- Changes to behavior within the daemon's interaction with the P2P protocol,
|
||||
or RPC's will need to be accompanied by integration tests which use the
|
||||
[`networkHarness`framework](https://github.com/lightningnetwork/lnd/blob/master/lntest/harness.go)
|
||||
contained within `lnd`. For example integration tests, see
|
||||
[`lnd_test.go`](https://github.com/lightningnetwork/lnd/blob/master/itest/lnd_test.go).
|
||||
- The itest log files are automatically scanned for `[ERR]` lines. There
|
||||
shouldn't be any of those in the logs, see [Use of Log Levels](#use-of-log-levels).
|
||||
|
||||
Throughout the process of contributing to `lnd`, you'll likely also be
|
||||
extensively using the commands within our `Makefile`. As a result, we recommend
|
||||
[perusing the make file documentation](https://github.com/lightningnetwork/lnd/blob/master/docs/MAKEFILE.md).
|
||||
|
||||
## Code Documentation and Commenting
|
||||
|
||||
- At a minimum every function must be commented with its intended purpose and
|
||||
any assumptions that it makes
|
||||
- Function comments must always begin with the name of the function per
|
||||
[Effective Go](https://golang.org/doc/effective_go.html)
|
||||
- Function comments should be complete sentences since they allow a wide
|
||||
variety of automated presentations such as [godoc.org](https://godoc.org)
|
||||
- The general rule of thumb is to look at it as if you were completely
|
||||
unfamiliar with the code and ask yourself, would this give me enough
|
||||
information to understand what this function does and how I'd probably want
|
||||
to use it?
|
||||
- Exported functions should also include detailed information the caller of the
|
||||
function will likely need to know and/or understand:<br /><br />
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
// generates a revocation key
|
||||
func DeriveRevocationPubkey(commitPubKey *btcec.PublicKey,
|
||||
revokePreimage []byte) *btcec.PublicKey {
|
||||
```
|
||||
**RIGHT**
|
||||
```go
|
||||
// DeriveRevocationPubkey derives the revocation public key given the
|
||||
// counterparty's commitment key, and revocation preimage derived via a
|
||||
// pseudo-random-function. In the event that we (for some reason) broadcast a
|
||||
// revoked commitment transaction, then if the other party knows the revocation
|
||||
// preimage, then they'll be able to derive the corresponding private key to
|
||||
// this private key by exploiting the homomorphism in the elliptic curve group:
|
||||
// * https://en.wikipedia.org/wiki/Group_homomorphism#Homomorphisms_of_abelian_groups
|
||||
//
|
||||
// The derivation is performed as follows:
|
||||
//
|
||||
// revokeKey := commitKey + revokePoint
|
||||
// := G*k + G*h
|
||||
// := G * (k+h)
|
||||
//
|
||||
// Therefore, once we divulge the revocation preimage, the remote peer is able to
|
||||
// compute the proper private key for the revokeKey by computing:
|
||||
// revokePriv := commitPriv + revokePreimge mod N
|
||||
//
|
||||
// Where N is the order of the sub-group.
|
||||
func DeriveRevocationPubkey(commitPubKey *btcec.PublicKey,
|
||||
revokePreimage []byte) *btcec.PublicKey {
|
||||
```
|
||||
- Comments in the body of the code are highly encouraged, but they should
|
||||
explain the intention of the code as opposed to just calling out the
|
||||
obvious<br /><br />
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
// return err if amt is less than 546
|
||||
if amt < 546 {
|
||||
return err
|
||||
}
|
||||
```
|
||||
**RIGHT**
|
||||
```go
|
||||
// Treat transactions with amounts less than the amount which is considered dust
|
||||
// as non-standard.
|
||||
if amt < 546 {
|
||||
return err
|
||||
}
|
||||
```
|
||||
**NOTE:** The above should really use a constant as opposed to a magic number,
|
||||
but it was left as a magic number to show how much of a difference a good
|
||||
comment can make.
|
||||
|
||||
## Code Spacing and formatting
|
||||
|
||||
Code in general (and Open Source code specifically) is _read_ by developers many
|
||||
more times during its lifecycle than it is modified. With this fact in mind, the
|
||||
Golang language was designed for readability (among other goals).
|
||||
While the enforced formatting of `go fmt` and some best practices already
|
||||
eliminate many discussions, the resulting code can still look and feel very
|
||||
differently among different developers.
|
||||
|
||||
We aim to enforce a few additional rules to unify the look and feel of all code
|
||||
in `lnd` to help improve the overall readability.
|
||||
|
||||
**Please refer to the [code formatting rules
|
||||
document](./code_formatting_rules.md)** to see the list of additional style
|
||||
rules we enforce.
|
||||
|
||||
## Model Git Commit Messages
|
||||
|
||||
This project prefers to keep a clean commit history with well-formed commit
|
||||
messages. This section illustrates a model commit message and provides a bit
|
||||
of background for it. This content was originally created by Tim Pope and made
|
||||
available on his website, however that website is no longer active, so it is
|
||||
being provided here.
|
||||
|
||||
Here’s a model Git commit message:
|
||||
|
||||
```text
|
||||
Short (50 chars or less) summary of changes
|
||||
|
||||
More detailed explanatory text, if necessary. Wrap it to about 72
|
||||
characters or so. In some contexts, the first line is treated as the
|
||||
subject of an email and the rest of the text as the body. The blank
|
||||
line separating the summary from the body is critical (unless you omit
|
||||
the body entirely); tools like rebase can get confused if you run the
|
||||
two together.
|
||||
|
||||
Write your commit message in the present tense: "Fix bug" and not "Fixed
|
||||
bug." This convention matches up with commit messages generated by
|
||||
commands like git merge and git revert.
|
||||
|
||||
Further paragraphs come after blank lines.
|
||||
|
||||
- Bullet points are okay, too
|
||||
- Typically a hyphen or asterisk is used for the bullet, preceded by a
|
||||
single space, with blank lines in between, but conventions vary here
|
||||
- Use a hanging indent
|
||||
```
|
||||
|
||||
Here are some of the reasons why wrapping your commit messages to 72 columns is
|
||||
a good thing.
|
||||
|
||||
- git log doesn't do any special wrapping of the commit messages. With
|
||||
the default pager of less -S, this means your paragraphs flow far off the edge
|
||||
of the screen, making them difficult to read. On an 80 column terminal, if we
|
||||
subtract 4 columns for the indent on the left and 4 more for symmetry on the
|
||||
right, we’re left with 72 columns.
|
||||
- git format-patch --stdout converts a series of commits to a series of emails,
|
||||
using the messages for the message body. Good email netiquette dictates we
|
||||
wrap our plain text emails such that there’s room for a few levels of nested
|
||||
reply indicators without overflow in an 80 column terminal.
|
||||
|
||||
In addition to the Git commit message structure adhered to within the daemon
|
||||
all short-[commit messages are to be prefixed according to the convention
|
||||
outlined in the Go project](https://golang.org/doc/contribute.html#change). All
|
||||
commits should begin with the subsystem or package primarily affected by the
|
||||
change. In the case of a widespread change, the packages are to be delimited by
|
||||
either a '+' or a ','. This prefix seems minor but can be extremely helpful in
|
||||
determining the scope of a commit at a glance, or when bug hunting to find a
|
||||
commit which introduced a bug or regression.
|
||||
|
||||
## Ideal Git Commit Structure
|
||||
|
||||
Within the project we prefer small, contained commits for a pull request over a
|
||||
single giant commit that touches several files/packages. Ideal commits build on
|
||||
their own, in order to facilitate easy usage of tools like `git bisect` to `git
|
||||
cherry-pick`. It's preferred that commits contain an isolated change in a
|
||||
single package. In this case, the commit header message should begin with the
|
||||
prefix of the modified package. For example, if a commit was made to modify the
|
||||
`lnwallet` package, it should start with `lnwallet: `.
|
||||
|
||||
In the case of changes that only build in tandem with changes made in other
|
||||
packages, it is permitted for a single commit to be made which contains several
|
||||
prefixes such as: `lnwallet+htlcswitch`. This prefix structure along with the
|
||||
requirement for atomic contained commits (when possible) make things like
|
||||
scanning the set of commits and debugging easier. In the case of changes that
|
||||
touch several packages, and can only compile with the change across several
|
||||
packages, a `multi: ` prefix should be used.
|
||||
|
||||
Examples of common patterns w.r.t commit structures within the project:
|
||||
|
||||
* It is common that during the work on a PR, existing bugs are found and
|
||||
fixed. If they can be fixed in isolation, they should have their own
|
||||
commit.
|
||||
* File restructuring like moving a function to another file or changing order
|
||||
of functions: with a separate commit because it is much easier to review
|
||||
the real changes that go on top of the restructuring.
|
||||
* Preparatory refactorings that are functionally equivalent: own commit.
|
||||
* Project or package wide file renamings should be in their own commit.
|
||||
* Ideally if a new package/struct/sub-system is added in a PR, there should
|
||||
be a single commit which adds the new functionality, with follow up
|
||||
individual commits that begin to integrate the functionality within the
|
||||
codebase.
|
||||
* If a PR only fixes a trivial issue, such as updating documentation on a
|
||||
small scale, fix typos, or any changes that do not modify the code, the
|
||||
commit message of the HEAD commit of the PR should end with `[skip ci]` to
|
||||
skip the CI checks. When pushing to such an existing PR, the latest commit
|
||||
being pushed should end with `[skip ci]` as to not inadvertently trigger the
|
||||
CI checks.
|
||||
|
||||
## Sign your git commits
|
||||
|
||||
When contributing to `lnd` it is recommended to sign your git commits. This is
|
||||
easy to do and will help in assuring the integrity of the tree. See [mailing
|
||||
list entry](https://lists.linuxfoundation.org/pipermail/bitcoin-dev/2014-May/005877.html)
|
||||
for more information.
|
||||
|
||||
### How to sign your commits?
|
||||
|
||||
Provide the `-S` flag (or `--gpg-sign`) to git commit when you commit
|
||||
your changes, for example
|
||||
|
||||
```shell
|
||||
$ git commit -m "Commit message" -S
|
||||
```
|
||||
|
||||
Optionally you can provide a key id after the `-S` option to sign with a
|
||||
specific key.
|
||||
|
||||
To instruct `git` to auto-sign every commit, add the following lines to your
|
||||
`~/.gitconfig` file:
|
||||
|
||||
```text
|
||||
[commit]
|
||||
gpgsign = true
|
||||
```
|
||||
|
||||
### What if I forgot?
|
||||
|
||||
You can retroactively sign your previous commit using `--amend`, for example
|
||||
|
||||
```shell
|
||||
$ git commit -S --amend
|
||||
```
|
||||
|
||||
If you need to go further back, you can use the interactive rebase
|
||||
command with 'edit'. Replace `HEAD~3` with the base commit from which
|
||||
you want to start.
|
||||
|
||||
```shell
|
||||
$ git rebase -i HEAD~3
|
||||
```
|
||||
|
||||
Replace 'pick' by 'edit' for the commit that you want to sign and the
|
||||
rebasing will stop after that commit. Then you can amend the commit as
|
||||
above. Afterwards, do
|
||||
|
||||
```shell
|
||||
$ git rebase --continue
|
||||
```
|
||||
|
||||
As this will rewrite history, you cannot do this when your commit is
|
||||
already merged. In that case, too bad, better luck next time.
|
||||
|
||||
If you rewrite history for another reason - for example when squashing
|
||||
commits - make sure that you re-sign as the signatures will be lost.
|
||||
|
||||
Multiple commits can also be re-signed with `git rebase`. For example, signing
|
||||
the last three commits can be done with:
|
||||
|
||||
```shell
|
||||
$ git rebase --exec 'git commit --amend --no-edit -n -S' -i HEAD~3
|
||||
```
|
||||
|
||||
### How to check if commits are signed?
|
||||
|
||||
Use `git log` with `--show-signature`,
|
||||
|
||||
```shell
|
||||
$ git log --show-signature
|
||||
```
|
||||
|
||||
You can also pass the `--show-signature` option to `git show` to check a single
|
||||
commit.
|
||||
|
||||
## Pointing to Remote Dependent Branches in Go Modules
|
||||
|
||||
It's common that a developer may need to make a change in a dependent project
|
||||
of `lnd` such as `btcd`, `neutrino`, `btcwallet`, etc. In order to test changes
|
||||
without testing infrastructure, or simply make a PR into `lnd` that will build
|
||||
without any further work, the `go.mod` and `go.sum` files will need to be
|
||||
updated. Luckily, the `go mod` command has a handy tool to do this
|
||||
automatically so developers don't need to manually edit the `go.mod` file:
|
||||
```shell
|
||||
$ go mod edit -replace=IMPORT-PATH-IN-LND@LND-VERSION=DEV-FORK-IMPORT-PATH@DEV-FORK-VERSION
|
||||
```
|
||||
|
||||
Here's an example replacing the `lightning-onion` version checked into `lnd` with a version in roasbeef's fork:
|
||||
```shell
|
||||
$ go mod edit -replace=github.com/lightningnetwork/lightning-onion@v0.0.0-20180605012408-ac4d9da8f1d6=github.com/roasbeef/lightning-onion@2e5ae87696046298365ab43bcd1cf3a7a1d69695
|
||||
```
|
||||
|
||||
## Use of Log Levels
|
||||
|
||||
There are six log levels available: `trace`, `debug`, `info`, `warn`, `error` and `critical`.
|
||||
|
||||
Only use `error` for internal errors that are never expected to happen during
|
||||
normal operation. No event triggered by external sources (rpc, chain backend,
|
||||
etc) should lead to an `error` log.
|
||||
|
||||
## Use of Golang submodules
|
||||
|
||||
Changes to packages that are their own submodules (e.g. they contain a `go.mod`
|
||||
and `go.sum` file, for example `tor/go.mod`) require a specific process.
|
||||
We want to avoid the use of local replace directives in the root `go.mod`,
|
||||
therefore changes to a submodule are a bit involved.
|
||||
|
||||
The main process for updating and then using code in a submodule is as follows:
|
||||
- Create a PR for the changes to the submodule itself (e.g. edit something in
|
||||
the `tor` package)
|
||||
- Wait for the PR to be merged and a new tag (for example `tor/v1.0.x`) to be
|
||||
pushed.
|
||||
- Create a second PR that bumps the updated submodule in the root `go.mod` and
|
||||
uses the new functionality in the main module.
|
||||
|
||||
Of course the two PRs can be opened at the same time and be built on top of each
|
||||
other. But the merge and tag push order should always be maintained.
|
||||
|
||||
# Code Approval Process
|
||||
|
||||
This section describes the code approval process that is used for code
|
||||
|
@@ -1,336 +0,0 @@
|
||||
# Code formatting rules
|
||||
|
||||
## Why this emphasis on formatting?
|
||||
|
||||
Code in general (and Open Source code specifically) is _read_ by developers many
|
||||
more times during its lifecycle than it is modified. With this fact in mind, the
|
||||
Golang language was designed for readability (among other goals).
|
||||
While the enforced formatting of `go fmt` and some best practices already
|
||||
eliminate many discussions, the resulting code can still look and feel very
|
||||
differently among different developers.
|
||||
|
||||
We aim to enforce a few additional rules to unify the look and feel of all code
|
||||
in `lnd` to help improve the overall readability.
|
||||
|
||||
## Spacing
|
||||
|
||||
Blocks of code within `lnd` should be segmented into logical stanzas of
|
||||
operation. Such spacing makes the code easier to follow at a skim, and reduces
|
||||
unnecessary line noise. Coupled with the commenting scheme specified in the
|
||||
[contribution guide](./code_contribution_guidelines.md#code-documentation-and-commenting),
|
||||
proper spacing allows readers to quickly scan code, extracting semantics quickly.
|
||||
Functions should _not_ just be laid out as a bare contiguous block of code.
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
witness := make([][]byte, 4)
|
||||
witness[0] = nil
|
||||
if bytes.Compare(pubA, pubB) == -1 {
|
||||
witness[1] = sigB
|
||||
witness[2] = sigA
|
||||
} else {
|
||||
witness[1] = sigA
|
||||
witness[2] = sigB
|
||||
}
|
||||
witness[3] = witnessScript
|
||||
return witness
|
||||
```
|
||||
**RIGHT**
|
||||
```go
|
||||
witness := make([][]byte, 4)
|
||||
|
||||
// When spending a p2wsh multi-sig script, rather than an OP_0, we add
|
||||
// a nil stack element to eat the extra pop.
|
||||
witness[0] = nil
|
||||
|
||||
// When initially generating the witnessScript, we sorted the serialized
|
||||
// public keys in descending order. So we do a quick comparison in order
|
||||
// to ensure the signatures appear on the Script Virtual Machine stack in
|
||||
// the correct order.
|
||||
if bytes.Compare(pubA, pubB) == -1 {
|
||||
witness[1] = sigB
|
||||
witness[2] = sigA
|
||||
} else {
|
||||
witness[1] = sigA
|
||||
witness[2] = sigB
|
||||
}
|
||||
|
||||
// Finally, add the preimage as the last witness element.
|
||||
witness[3] = witnessScript
|
||||
|
||||
return witness
|
||||
```
|
||||
|
||||
Additionally, we favor spacing between stanzas within syntax like: switch case
|
||||
statements and select statements.
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
switch {
|
||||
case a:
|
||||
<code block>
|
||||
case b:
|
||||
<code block>
|
||||
case c:
|
||||
<code block>
|
||||
case d:
|
||||
<code block>
|
||||
default:
|
||||
<code block>
|
||||
}
|
||||
```
|
||||
**RIGHT**
|
||||
```go
|
||||
switch {
|
||||
// Brief comment detailing instances of this case (repeat below).
|
||||
case a:
|
||||
<code block>
|
||||
|
||||
case b:
|
||||
<code block>
|
||||
|
||||
case c:
|
||||
<code block>
|
||||
|
||||
case d:
|
||||
<code block>
|
||||
|
||||
default:
|
||||
<code block>
|
||||
}
|
||||
```
|
||||
|
||||
## Additional Style Constraints
|
||||
|
||||
Before a PR is submitted, the proposer should ensure that the file passes the
|
||||
set of linting scripts run by `make lint`. These include `gofmt`. In addition
|
||||
to `gofmt` we've opted to enforce the following style guidelines.
|
||||
|
||||
### 80 character line length
|
||||
|
||||
ALL columns (on a best effort basis) should be wrapped to 80 line columns.
|
||||
Editors should be set to treat a **tab as 8 spaces**.
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
myKey := "0214cd678a565041d00e6cf8d62ef8add33b4af4786fb2beb87b366a2e151fcee7"
|
||||
```
|
||||
|
||||
**RIGHT**
|
||||
```go
|
||||
myKey := "0214cd678a565041d00e6cf8d62ef8add33b4af4786fb2beb87b366a2e1" +
|
||||
"51fcee7"
|
||||
```
|
||||
|
||||
### Wrapping long function calls
|
||||
|
||||
When wrapping a line that contains a function call as the unwrapped line exceeds
|
||||
the column limit, the close parenthesis should be placed on its own line.
|
||||
Additionally, all arguments should begin in a new line after the open parenthesis.
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
value, err := bar(a,
|
||||
a, b, c)
|
||||
```
|
||||
|
||||
**RIGHT**
|
||||
```go
|
||||
value, err := bar(
|
||||
a, a, b, c,
|
||||
)
|
||||
```
|
||||
|
||||
#### Exception for log and error message formatting
|
||||
|
||||
**Note that the above guidelines don't apply to log or error messages.** For
|
||||
log and error messages, committers should attempt to minimize the number of
|
||||
lines utilized, while still adhering to the 80-character column limit. For
|
||||
example:
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
return fmt.Errorf(
|
||||
"this is a long error message with a couple (%d) place holders",
|
||||
len(things),
|
||||
)
|
||||
|
||||
log.Debugf(
|
||||
"Something happened here that we need to log: %v",
|
||||
longVariableNameHere,
|
||||
)
|
||||
```
|
||||
|
||||
**RIGHT**
|
||||
```go
|
||||
return fmt.Errorf("this is a long error message with a couple (%d) place "+
|
||||
"holders", len(things))
|
||||
|
||||
log.Debugf("Something happened here that we need to log: %v",
|
||||
longVariableNameHere)
|
||||
```
|
||||
|
||||
This helps to visually distinguish those formatting statements (where nothing
|
||||
of consequence happens except for formatting an error message or writing
|
||||
to a log) from actual method or function calls. This compact formatting should
|
||||
be used for calls to formatting functions like `fmt.Errorf`,
|
||||
`log.(Trace|Debug|Info|Warn|Error)f` and `fmt.Printf`.
|
||||
But not for statements that are important for the flow or logic of the code,
|
||||
like `require.NoErrorf()`.
|
||||
|
||||
#### Exceptions and additional styling for structured logging
|
||||
|
||||
When making use of structured logging calls (there are any `btclog.Logger`
|
||||
methods ending in `S`), a few different rules and exceptions apply.
|
||||
|
||||
1) **Static messages:** Structured log calls take a `context.Context` as a first
|
||||
parameter and a _static_ string as the second parameter (the `msg` parameter).
|
||||
Formatted strings should ideally not be used for the construction of the `msg`
|
||||
parameter. Instead, key-value pairs (or `slog` attributes) should be used to
|
||||
provide additional variables to the log line.
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
log.DebugS(ctx, fmt.Sprintf("User %d just spent %.8f to open a channel", userID, 0.0154))
|
||||
```
|
||||
|
||||
**RIGHT**
|
||||
```go
|
||||
log.InfoS(ctx, "Channel open performed",
|
||||
slog.Int("user_id", userID),
|
||||
btclog.Fmt("amount", "%.8f", 0.00154))
|
||||
```
|
||||
|
||||
2) **Key-value attributes**: The third parameter in any structured log method is
|
||||
a variadic list of the `any` type but it is required that these are provided in
|
||||
key-value pairs such that an associated `slog.Attr` variable can be created for
|
||||
each key-value pair. The simplest way to specify this is to directly pass in the
|
||||
key-value pairs as raw literals as follows:
|
||||
|
||||
```go
|
||||
log.InfoS(ctx, "Channel open performed", "user_id", userID, "amount", 0.00154)
|
||||
```
|
||||
This does work, but it becomes easy to make a mistake and accidentally leave out
|
||||
a value for each key provided leading to a nonsensical log line. To avoid this,
|
||||
it is suggested to make use of the various `slog.Attr` helper functions as
|
||||
follows:
|
||||
|
||||
```go
|
||||
log.InfoS(ctx, "Channel open performed",
|
||||
slog.Int("user_id", userID),
|
||||
btclog.Fmt("amount", "%.8f", 0.00154))
|
||||
```
|
||||
|
||||
3) **Line wrapping**: Structured log lines are an exception to the 80-character
|
||||
line wrapping rule. This is so that the key-value pairs can be easily read and
|
||||
reasoned about. If it is the case that there is only a single key-value pair
|
||||
and the entire log line is still less than 80 characters, it is acceptable to
|
||||
have the key-value pair on the same line as the log message. However, if there
|
||||
are multiple key-value pairs, it is suggested to use the one line per key-value
|
||||
pair format. Due to this suggestion, it is acceptable for any single key-value
|
||||
pair line to exceed 80 characters for the sake of readability.
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
// Example 1.
|
||||
log.InfoS(ctx, "User connected",
|
||||
"user_id", userID)
|
||||
|
||||
// Example 2.
|
||||
log.InfoS(ctx, "Channel open performed", "user_id", userID,
|
||||
btclog.Fmt("amount", "%.8f", 0.00154), "channel_id", channelID)
|
||||
|
||||
// Example 3.
|
||||
log.InfoS(ctx, "Bytes received",
|
||||
"user_id", userID,
|
||||
btclog.Hex("peer_id", peerID.SerializeCompressed()),
|
||||
btclog.Hex("message", []bytes{
|
||||
0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08,
|
||||
})))
|
||||
```
|
||||
|
||||
**RIGHT**
|
||||
```go
|
||||
// Example 1.
|
||||
log.InfoS(ctx, "User connected", "user_id", userID)
|
||||
|
||||
// Example 2.
|
||||
log.InfoS(ctx, "Channel open performed",
|
||||
slog.Int("user_id", userID),
|
||||
btclog.Fmt("amount", "%.8f", 0.00154),
|
||||
slog.String("channel_id", channelID))
|
||||
|
||||
// Example 3.
|
||||
log.InfoS(ctx, "Bytes received",
|
||||
"user_id", userID,
|
||||
btclog.Hex("peer_id", peerID.SerializeCompressed()),
|
||||
btclog.Hex("message", []bytes{0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08})))
|
||||
```
|
||||
|
||||
### Wrapping long function definitions
|
||||
|
||||
If one is forced to wrap lines of function arguments that exceed the
|
||||
80-character limit, then indentation must be kept on the following lines. Also,
|
||||
lines should not end with an open parenthesis if the function definition isn't
|
||||
finished yet.
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
func foo(a, b, c,
|
||||
) (d, error) {
|
||||
|
||||
func bar(a, b, c) (
|
||||
d, error,
|
||||
) {
|
||||
|
||||
func baz(a, b, c) (
|
||||
d, error) {
|
||||
```
|
||||
**RIGHT**
|
||||
```go
|
||||
func foo(a, b,
|
||||
c) (d, error) {
|
||||
|
||||
func baz(a, b, c) (d,
|
||||
error) {
|
||||
|
||||
func longFunctionName(
|
||||
a, b, c) (d, error) {
|
||||
```
|
||||
|
||||
If a function declaration spans multiple lines the body should start with an
|
||||
empty line to help visually distinguishing the two elements.
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
func foo(a, b, c,
|
||||
d, e) error {
|
||||
var a int
|
||||
}
|
||||
```
|
||||
**RIGHT**
|
||||
```go
|
||||
func foo(a, b, c,
|
||||
d, e) error {
|
||||
|
||||
var a int
|
||||
}
|
||||
```
|
||||
|
||||
## Recommended settings for your editor
|
||||
|
||||
To make it easier to follow the rules outlined above, we recommend setting up
|
||||
your editor with at least the following two settings:
|
||||
|
||||
1. Set your tabulator width (also called "tab size") to **8 spaces**.
|
||||
2. Set a ruler or visual guide at 80 character.
|
||||
|
||||
Note that the two above settings are automatically applied in editors that
|
||||
support the `EditorConfig` scheme (for example GoLand, GitHub, GitLab,
|
||||
VisualStudio). In addition, specific settings for Visual Studio Code are checked
|
||||
into the code base as well.
|
||||
|
||||
Other editors (for example Atom, Notepad++, Vim, Emacs and so on) might install
|
||||
a plugin to understand the rules in the `.editorconfig` file.
|
||||
|
||||
In Vim, you might want to use `set colorcolumn=80`.
|
702
docs/development_guidelines.md
Normal file
702
docs/development_guidelines.md
Normal file
@@ -0,0 +1,702 @@
|
||||
# Development Guidelines
|
||||
1. [Code Documentation and Commenting](#code-documentation-and-commenting)
|
||||
1. [Code Spacing and Formatting](#code-spacing-and-formatting)
|
||||
1. [Additional Style Constraints](#additional-style-constraints)
|
||||
1. [Recommended settings for your editor](#recommended-settings-for-your-editor)
|
||||
1. [Testing](#testing)
|
||||
1. [Model Git Commit Messages](#model-git-commit-messages)
|
||||
1. [Ideal Git Commit Structure](#ideal-git-commit-structure)
|
||||
1. [Sign Your Git Commits](#sign-your-git-commits)
|
||||
1. [Pointing to Remote Dependent Branches in Go Modules](#pointing-to-remote-dependent-branches-in-go-modules)
|
||||
1. [Use of Log Levels](#use-of-log-levels)
|
||||
1. [Use of Golang submodules](#use-of-golang-submodules)
|
||||
|
||||
## Why this emphasis on formatting?
|
||||
|
||||
Code in general (and Open Source code specifically) is _read_ by developers many
|
||||
more times during its lifecycle than it is modified. With this fact in mind, the
|
||||
Golang language was designed for readability (among other goals).
|
||||
While the enforced formatting of `go fmt` and some best practices already
|
||||
eliminate many discussions, the resulting code can still look and feel very
|
||||
differently among different developers.
|
||||
|
||||
We aim to enforce a few additional rules to unify the look and feel of all code
|
||||
in `lnd` to help improve the overall readability.
|
||||
|
||||
## Code Documentation and Commenting
|
||||
|
||||
- At a minimum every function must be commented with its intended purpose and
|
||||
any assumptions that it makes
|
||||
- Function comments must always begin with the name of the function per
|
||||
[Effective Go](https://golang.org/doc/effective_go.html)
|
||||
- Function comments should be complete sentences since they allow a wide
|
||||
variety of automated presentations such as [godoc.org](https://godoc.org)
|
||||
- The general rule of thumb is to look at it as if you were completely
|
||||
unfamiliar with the code and ask yourself, would this give me enough
|
||||
information to understand what this function does and how I'd probably want to
|
||||
use it?
|
||||
- Exported functions should also include detailed information the caller of the
|
||||
function will likely need to know and/or understand:<br /><br />
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
// generates a revocation key
|
||||
func DeriveRevocationPubkey(commitPubKey *btcec.PublicKey,
|
||||
revokePreimage []byte) *btcec.PublicKey {
|
||||
```
|
||||
**RIGHT**
|
||||
```go
|
||||
// DeriveRevocationPubkey derives the revocation public key given the
|
||||
// counterparty's commitment key, and revocation preimage derived via a
|
||||
// pseudo-random-function. In the event that we (for some reason) broadcast a
|
||||
// revoked commitment transaction, then if the other party knows the revocation
|
||||
// preimage, then they'll be able to derive the corresponding private key to
|
||||
// this private key by exploiting the homomorphism in the elliptic curve group:
|
||||
// * https://en.wikipedia.org/wiki/Group_homomorphism#Homomorphisms_of_abelian_groups
|
||||
//
|
||||
// The derivation is performed as follows:
|
||||
//
|
||||
// revokeKey := commitKey + revokePoint
|
||||
// := G*k + G*h
|
||||
// := G * (k+h)
|
||||
//
|
||||
// Therefore, once we divulge the revocation preimage, the remote peer is able to
|
||||
// compute the proper private key for the revokeKey by computing:
|
||||
// revokePriv := commitPriv + revokePreimge mod N
|
||||
//
|
||||
// Where N is the order of the sub-group.
|
||||
func DeriveRevocationPubkey(commitPubKey *btcec.PublicKey,
|
||||
revokePreimage []byte) *btcec.PublicKey {
|
||||
```
|
||||
- Comments in the body of the code are highly encouraged, but they should
|
||||
explain the intention of the code as opposed to just calling out the
|
||||
obvious<br /><br />
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
// return err if amt is less than 546
|
||||
if amt < 546 {
|
||||
return err
|
||||
}
|
||||
```
|
||||
**RIGHT**
|
||||
```go
|
||||
// Treat transactions with amounts less than the amount which is considered dust
|
||||
// as non-standard.
|
||||
if amt < 546 {
|
||||
return err
|
||||
}
|
||||
```
|
||||
**NOTE:** The above should really use a constant as opposed to a magic number,
|
||||
but it was left as a magic number to show how much of a difference a good
|
||||
comment can make.
|
||||
|
||||
## Code Spacing and formatting
|
||||
|
||||
Code in general (and Open Source code specifically) is _read_ by developers many
|
||||
more times during its lifecycle than it is modified. With this fact in mind, the
|
||||
Golang language was designed for readability (among other goals).
|
||||
While the enforced formatting of `go fmt` and some best practices already
|
||||
eliminate many discussions, the resulting code can still look and feel very
|
||||
differently among different developers.
|
||||
|
||||
We aim to enforce a few additional rules to unify the look and feel of all code
|
||||
in `lnd` to help improve the overall readability.
|
||||
|
||||
Blocks of code within `lnd` should be segmented into logical stanzas of
|
||||
operation. Such spacing makes the code easier to follow at a skim, and reduces
|
||||
unnecessary line noise. Coupled with the commenting scheme specified in the
|
||||
[contribution guide](#code-documentation-and-commenting),
|
||||
proper spacing allows readers to quickly scan code, extracting semantics quickly.
|
||||
Functions should _not_ just be laid out as a bare contiguous block of code.
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
witness := make([][]byte, 4)
|
||||
witness[0] = nil
|
||||
if bytes.Compare(pubA, pubB) == -1 {
|
||||
witness[1] = sigB
|
||||
witness[2] = sigA
|
||||
} else {
|
||||
witness[1] = sigA
|
||||
witness[2] = sigB
|
||||
}
|
||||
witness[3] = witnessScript
|
||||
return witness
|
||||
```
|
||||
**RIGHT**
|
||||
```go
|
||||
witness := make([][]byte, 4)
|
||||
|
||||
// When spending a p2wsh multi-sig script, rather than an OP_0, we add
|
||||
// a nil stack element to eat the extra pop.
|
||||
witness[0] = nil
|
||||
|
||||
// When initially generating the witnessScript, we sorted the serialized
|
||||
// public keys in descending order. So we do a quick comparison in order
|
||||
// to ensure the signatures appear on the Script Virtual Machine stack in
|
||||
// the correct order.
|
||||
if bytes.Compare(pubA, pubB) == -1 {
|
||||
witness[1] = sigB
|
||||
witness[2] = sigA
|
||||
} else {
|
||||
witness[1] = sigA
|
||||
witness[2] = sigB
|
||||
}
|
||||
|
||||
// Finally, add the preimage as the last witness element.
|
||||
witness[3] = witnessScript
|
||||
|
||||
return witness
|
||||
```
|
||||
|
||||
Additionally, we favor spacing between stanzas within syntax like: switch case
|
||||
statements and select statements.
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
switch {
|
||||
case a:
|
||||
<code block>
|
||||
case b:
|
||||
<code block>
|
||||
case c:
|
||||
<code block>
|
||||
case d:
|
||||
<code block>
|
||||
default:
|
||||
<code block>
|
||||
}
|
||||
```
|
||||
**RIGHT**
|
||||
```go
|
||||
switch {
|
||||
// Brief comment detailing instances of this case (repeat below).
|
||||
case a:
|
||||
<code block>
|
||||
|
||||
case b:
|
||||
<code block>
|
||||
|
||||
case c:
|
||||
<code block>
|
||||
|
||||
case d:
|
||||
<code block>
|
||||
|
||||
default:
|
||||
<code block>
|
||||
}
|
||||
```
|
||||
|
||||
## Additional Style Constraints
|
||||
|
||||
Before a PR is submitted, the proposer should ensure that the file passes the
|
||||
set of linting scripts run by `make lint`. These include `gofmt`. In addition
|
||||
to `gofmt` we've opted to enforce the following style guidelines.
|
||||
|
||||
### 80 character line length
|
||||
|
||||
ALL columns (on a best effort basis) should be wrapped to 80 line columns.
|
||||
Editors should be set to treat a **tab as 8 spaces**.
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
myKey := "0214cd678a565041d00e6cf8d62ef8add33b4af4786fb2beb87b366a2e151fcee7"
|
||||
```
|
||||
|
||||
**RIGHT**
|
||||
```go
|
||||
myKey := "0214cd678a565041d00e6cf8d62ef8add33b4af4786fb2beb87b366a2e1" +
|
||||
"51fcee7"
|
||||
```
|
||||
|
||||
### Wrapping long function calls
|
||||
|
||||
When wrapping a line that contains a function call as the unwrapped line exceeds
|
||||
the column limit, the close parenthesis should be placed on its own line.
|
||||
Additionally, all arguments should begin in a new line after the open parenthesis.
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
value, err := bar(a,
|
||||
a, b, c)
|
||||
```
|
||||
|
||||
**RIGHT**
|
||||
```go
|
||||
value, err := bar(
|
||||
a, a, b, c,
|
||||
)
|
||||
```
|
||||
|
||||
#### Exception for log and error message formatting
|
||||
|
||||
**Note that the above guidelines don't apply to log or error messages.** For
|
||||
log and error messages, committers should attempt to minimize the number of
|
||||
lines utilized, while still adhering to the 80-character column limit. For
|
||||
example:
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
return fmt.Errorf(
|
||||
"this is a long error message with a couple (%d) place holders",
|
||||
len(things),
|
||||
)
|
||||
|
||||
log.Debugf(
|
||||
"Something happened here that we need to log: %v",
|
||||
longVariableNameHere,
|
||||
)
|
||||
```
|
||||
|
||||
**RIGHT**
|
||||
```go
|
||||
return fmt.Errorf("this is a long error message with a couple (%d) place "+
|
||||
"holders", len(things))
|
||||
|
||||
log.Debugf("Something happened here that we need to log: %v",
|
||||
longVariableNameHere)
|
||||
```
|
||||
|
||||
This helps to visually distinguish those formatting statements (where nothing
|
||||
of consequence happens except for formatting an error message or writing
|
||||
to a log) from actual method or function calls. This compact formatting should
|
||||
be used for calls to formatting functions like `fmt.Errorf`,
|
||||
`log.(Trace|Debug|Info|Warn|Error)f` and `fmt.Printf`.
|
||||
But not for statements that are important for the flow or logic of the code,
|
||||
like `require.NoErrorf()`.
|
||||
|
||||
#### Exceptions and additional styling for structured logging
|
||||
|
||||
When making use of structured logging calls (there are any `btclog.Logger`
|
||||
methods ending in `S`), a few different rules and exceptions apply.
|
||||
|
||||
1) **Static messages:** Structured log calls take a `context.Context` as a first
|
||||
parameter and a _static_ string as the second parameter (the `msg` parameter).
|
||||
Formatted strings should ideally not be used for the construction of the `msg`
|
||||
parameter. Instead, key-value pairs (or `slog` attributes) should be used to
|
||||
provide additional variables to the log line.
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
log.DebugS(ctx, fmt.Sprintf("User %d just spent %.8f to open a channel", userID, 0.0154))
|
||||
```
|
||||
|
||||
**RIGHT**
|
||||
```go
|
||||
log.InfoS(ctx, "Channel open performed",
|
||||
slog.Int("user_id", userID),
|
||||
btclog.Fmt("amount", "%.8f", 0.00154))
|
||||
```
|
||||
|
||||
2) **Key-value attributes**: The third parameter in any structured log method is
|
||||
a variadic list of the `any` type but it is required that these are provided in
|
||||
key-value pairs such that an associated `slog.Attr` variable can be created for
|
||||
each key-value pair. The simplest way to specify this is to directly pass in the
|
||||
key-value pairs as raw literals as follows:
|
||||
|
||||
```go
|
||||
log.InfoS(ctx, "Channel open performed", "user_id", userID, "amount", 0.00154)
|
||||
```
|
||||
This does work, but it becomes easy to make a mistake and accidentally leave out
|
||||
a value for each key provided leading to a nonsensical log line. To avoid this,
|
||||
it is suggested to make use of the various `slog.Attr` helper functions as
|
||||
follows:
|
||||
|
||||
```go
|
||||
log.InfoS(ctx, "Channel open performed",
|
||||
slog.Int("user_id", userID),
|
||||
btclog.Fmt("amount", "%.8f", 0.00154))
|
||||
```
|
||||
|
||||
3) **Line wrapping**: Structured log lines are an exception to the 80-character
|
||||
line wrapping rule. This is so that the key-value pairs can be easily read and
|
||||
reasoned about. If it is the case that there is only a single key-value pair
|
||||
and the entire log line is still less than 80 characters, it is acceptable to
|
||||
have the key-value pair on the same line as the log message. However, if there
|
||||
are multiple key-value pairs, it is suggested to use the one line per key-value
|
||||
pair format. Due to this suggestion, it is acceptable for any single key-value
|
||||
pair line to exceed 80 characters for the sake of readability.
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
// Example 1.
|
||||
log.InfoS(ctx, "User connected",
|
||||
"user_id", userID)
|
||||
|
||||
// Example 2.
|
||||
log.InfoS(ctx, "Channel open performed", "user_id", userID,
|
||||
btclog.Fmt("amount", "%.8f", 0.00154), "channel_id", channelID)
|
||||
|
||||
// Example 3.
|
||||
log.InfoS(ctx, "Bytes received",
|
||||
"user_id", userID,
|
||||
btclog.Hex("peer_id", peerID.SerializeCompressed()),
|
||||
btclog.Hex("message", []bytes{
|
||||
0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08,
|
||||
})))
|
||||
```
|
||||
|
||||
**RIGHT**
|
||||
```go
|
||||
// Example 1.
|
||||
log.InfoS(ctx, "User connected", "user_id", userID)
|
||||
|
||||
// Example 2.
|
||||
log.InfoS(ctx, "Channel open performed",
|
||||
slog.Int("user_id", userID),
|
||||
btclog.Fmt("amount", "%.8f", 0.00154),
|
||||
slog.String("channel_id", channelID))
|
||||
|
||||
// Example 3.
|
||||
log.InfoS(ctx, "Bytes received",
|
||||
"user_id", userID,
|
||||
btclog.Hex("peer_id", peerID.SerializeCompressed()),
|
||||
btclog.Hex("message", []bytes{0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08})))
|
||||
```
|
||||
|
||||
### Wrapping long function definitions
|
||||
|
||||
If one is forced to wrap lines of function arguments that exceed the
|
||||
80-character limit, then indentation must be kept on the following lines. Also,
|
||||
lines should not end with an open parenthesis if the function definition isn't
|
||||
finished yet.
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
func foo(a, b, c,
|
||||
) (d, error) {
|
||||
|
||||
func bar(a, b, c) (
|
||||
d, error,
|
||||
) {
|
||||
|
||||
func baz(a, b, c) (
|
||||
d, error) {
|
||||
```
|
||||
**RIGHT**
|
||||
```go
|
||||
func foo(a, b,
|
||||
c) (d, error) {
|
||||
|
||||
func baz(a, b, c) (d,
|
||||
error) {
|
||||
|
||||
func longFunctionName(
|
||||
a, b, c) (d, error) {
|
||||
```
|
||||
|
||||
If a function declaration spans multiple lines the body should start with an
|
||||
empty line to help visually distinguishing the two elements.
|
||||
|
||||
**WRONG**
|
||||
```go
|
||||
func foo(a, b, c,
|
||||
d, e) error {
|
||||
var a int
|
||||
}
|
||||
```
|
||||
**RIGHT**
|
||||
```go
|
||||
func foo(a, b, c,
|
||||
d, e) error {
|
||||
|
||||
var a int
|
||||
}
|
||||
```
|
||||
|
||||
## Recommended settings for your editor
|
||||
|
||||
To make it easier to follow the rules outlined above, we recommend setting up
|
||||
your editor with at least the following two settings:
|
||||
|
||||
1. Set your tabulator width (also called "tab size") to **8 spaces**.
|
||||
2. Set a ruler or visual guide at 80 character.
|
||||
|
||||
Note that the two above settings are automatically applied in editors that
|
||||
support the `EditorConfig` scheme (for example GoLand, GitHub, GitLab,
|
||||
VisualStudio). In addition, specific settings for Visual Studio Code are checked
|
||||
into the code base as well.
|
||||
|
||||
Other editors (for example Atom, Notepad++, Vim, Emacs and so on) might install
|
||||
a plugin to understand the rules in the `.editorconfig` file.
|
||||
|
||||
In Vim, you might want to use `set colorcolumn=80`.
|
||||
|
||||
|
||||
## Testing
|
||||
|
||||
One of the major design goals of all of `lnd`'s packages and the daemon itself is
|
||||
to aim for a high degree of test coverage. This is financial software so bugs
|
||||
and regressions in the core logic can cost people real money. For this reason
|
||||
every effort must be taken to ensure the code is as accurate and bug-free as
|
||||
possible. Thorough testing is a good way to help achieve that goal.
|
||||
|
||||
Unless a new feature you submit is completely trivial, it will probably be
|
||||
rejected unless it is also accompanied by adequate test coverage for both
|
||||
positive and negative conditions. That is to say, the tests must ensure your
|
||||
code works correctly when it is fed correct data as well as incorrect data
|
||||
(error paths).
|
||||
|
||||
|
||||
Go provides an excellent test framework that makes writing test code and
|
||||
checking coverage statistics straightforward. For more information about the
|
||||
test coverage tools, see the [golang cover blog post](https://blog.golang.org/cover).
|
||||
|
||||
A quick summary of test practices follows:
|
||||
|
||||
- All new code should be accompanied by tests that ensure the code behaves
|
||||
correctly when given expected values, and, perhaps even more importantly, that
|
||||
it handles errors gracefully.
|
||||
- When you fix a bug, it should be accompanied by tests which exercise the bug
|
||||
to both prove it has been resolved and to prevent future regressions.
|
||||
- Changes to publicly exported packages such as
|
||||
[brontide](https://github.com/lightningnetwork/lnd/tree/master/brontide) should
|
||||
be accompanied by unit tests exercising the new or changed behavior.
|
||||
- Changes to behavior within the daemon's interaction with the P2P protocol,
|
||||
or RPC's will need to be accompanied by integration tests which use the
|
||||
[`networkHarness`framework](https://github.com/lightningnetwork/lnd/blob/master/lntest/harness.go)
|
||||
contained within `lnd`. For example integration tests, see
|
||||
[`lnd_test.go`](https://github.com/lightningnetwork/lnd/blob/master/itest/lnd_test.go).
|
||||
- The itest log files are automatically scanned for `[ERR]` lines. There
|
||||
shouldn't be any of those in the logs, see [Use of Log Levels](#use-of-log-levels).
|
||||
|
||||
Throughout the process of contributing to `lnd`, you'll likely also be
|
||||
extensively using the commands within our `Makefile`. As a result, we recommend
|
||||
[perusing the make file documentation](https://github.com/lightningnetwork/lnd/blob/master/docs/MAKEFILE.md).
|
||||
|
||||
|
||||
Before committing the changes made, you should run unit tests to validate the
|
||||
changes, and provide a new integration test when necessary. The unit tests
|
||||
should pass at two levels,
|
||||
|
||||
- Run `make unit-debug log="stdlog trace" pkg=$pkg case=$case timeout=10s`,
|
||||
where `pkg` is the package that's updated, and `case` is the newly added or
|
||||
affected unit test. This command should run against all the newly added and
|
||||
affected test cases. In addition, you should pay attention to the logs added
|
||||
here, and make sure they are correctly formatted and no spammy. Also notice
|
||||
the timeout - 10 seconds should be more than enough to run a single unit test
|
||||
case, if it takes longer than that, consider break the test to make it more
|
||||
"unit".
|
||||
|
||||
- Run `make unit pkg=$pkg timeout=5m` to make sure all existing unit tests still
|
||||
pass.
|
||||
|
||||
- If there are newly added integration tests, or the changes may alter the
|
||||
workflow of specific areas, run `make itest icase=$icase` to validate the
|
||||
behavior, where `icase` is the affected test case.
|
||||
|
||||
|
||||
## Model Git Commit Messages
|
||||
|
||||
This project prefers to keep a clean commit history with well-formed commit
|
||||
messages. This section illustrates a model commit message and provides a bit
|
||||
of background for it. This content was originally created by Tim Pope and made
|
||||
available on his website, however that website is no longer active, so it is
|
||||
being provided here.
|
||||
|
||||
Here’s a model Git commit message:
|
||||
|
||||
```text
|
||||
Short (50 chars or less) summary of changes
|
||||
|
||||
More detailed explanatory text, if necessary. Wrap it to about 72
|
||||
characters or so. In some contexts, the first line is treated as the
|
||||
subject of an email and the rest of the text as the body. The blank
|
||||
line separating the summary from the body is critical (unless you omit
|
||||
the body entirely); tools like rebase can get confused if you run the
|
||||
two together.
|
||||
|
||||
Write your commit message in the present tense: "Fix bug" and not "Fixed
|
||||
bug." This convention matches up with commit messages generated by
|
||||
commands like git merge and git revert.
|
||||
|
||||
Further paragraphs come after blank lines.
|
||||
|
||||
- Bullet points are okay, too
|
||||
- Typically a hyphen or asterisk is used for the bullet, preceded by a
|
||||
single space, with blank lines in between, but conventions vary here
|
||||
- Use a hanging indent
|
||||
```
|
||||
|
||||
Here are some of the reasons why wrapping your commit messages to 72 columns is
|
||||
a good thing.
|
||||
|
||||
- git log doesn't do any special wrapping of the commit messages. With
|
||||
the default pager of less -S, this means your paragraphs flow far off the edge
|
||||
of the screen, making them difficult to read. On an 80 column terminal, if we
|
||||
subtract 4 columns for the indent on the left and 4 more for symmetry on the
|
||||
right, we’re left with 72 columns.
|
||||
- git format-patch --stdout converts a series of commits to a series of emails,
|
||||
using the messages for the message body. Good email netiquette dictates we
|
||||
wrap our plain text emails such that there’s room for a few levels of nested
|
||||
reply indicators without overflow in an 80 column terminal.
|
||||
|
||||
|
||||
In addition to the Git commit message structure adhered to within the daemon
|
||||
all short-[commit messages are to be prefixed according to the convention
|
||||
outlined in the Go project](https://golang.org/doc/contribute.html#change). All
|
||||
commits should begin with the subsystem or package primarily affected by the
|
||||
change. In the case of a widespread change, the packages are to be delimited by
|
||||
either a '+' or a ','. This prefix seems minor but can be extremely helpful in
|
||||
determining the scope of a commit at a glance, or when bug hunting to find a
|
||||
commit which introduced a bug or regression.
|
||||
|
||||
## Ideal Git Commit Structure
|
||||
|
||||
Within the project we prefer small, contained commits for a pull request over a
|
||||
single giant commit that touches several files/packages. Ideal commits build on
|
||||
their own, in order to facilitate easy usage of tools like `git bisect` to `git
|
||||
cherry-pick`. It's preferred that commits contain an isolated change in a
|
||||
single package. In this case, the commit header message should begin with the
|
||||
prefix of the modified package. For example, if a commit was made to modify the
|
||||
`lnwallet` package, it should start with `lnwallet: `.
|
||||
|
||||
In the case of changes that only build in tandem with changes made in other
|
||||
packages, it is permitted for a single commit to be made which contains several
|
||||
prefixes such as: `lnwallet+htlcswitch`. This prefix structure along with the
|
||||
requirement for atomic contained commits (when possible) make things like
|
||||
scanning the set of commits and debugging easier. In the case of changes that
|
||||
touch several packages, and can only compile with the change across several
|
||||
packages, a `multi: ` prefix should be used.
|
||||
|
||||
Examples of common patterns w.r.t commit structures within the project:
|
||||
|
||||
* It is common that during the work on a PR, existing bugs are found and
|
||||
fixed. If they can be fixed in isolation, they should have their own
|
||||
commit.
|
||||
* File restructuring like moving a function to another file or changing order
|
||||
of functions: with a separate commit because it is much easier to review
|
||||
the real changes that go on top of the restructuring.
|
||||
* Preparatory refactorings that are functionally equivalent: own commit.
|
||||
* Project or package wide file renamings should be in their own commit.
|
||||
* Ideally if a new package/struct/sub-system is added in a PR, there should
|
||||
be a single commit which adds the new functionality, with follow up
|
||||
individual commits that begin to integrate the functionality within the
|
||||
codebase.
|
||||
* If a PR only fixes a trivial issue, such as updating documentation on a
|
||||
small scale, fix typos, or any changes that do not modify the code, the
|
||||
commit message of the HEAD commit of the PR should end with `[skip ci]` to
|
||||
skip the CI checks. When pushing to such an existing PR, the latest commit
|
||||
being pushed should end with `[skip ci]` as to not inadvertently trigger the
|
||||
CI checks.
|
||||
|
||||
## Sign your git commits
|
||||
|
||||
When contributing to `lnd` it is recommended to sign your git commits. This is
|
||||
easy to do and will help in assuring the integrity of the tree. See [mailing
|
||||
list entry](https://lists.linuxfoundation.org/pipermail/bitcoin-dev/2014-May/005877.html)
|
||||
for more information.
|
||||
|
||||
### How to sign your commits?
|
||||
|
||||
Provide the `-S` flag (or `--gpg-sign`) to git commit when you commit
|
||||
your changes, for example
|
||||
|
||||
```shell
|
||||
$ git commit -m "Commit message" -S
|
||||
```
|
||||
|
||||
Optionally you can provide a key id after the `-S` option to sign with a
|
||||
specific key.
|
||||
|
||||
To instruct `git` to auto-sign every commit, add the following lines to your
|
||||
`~/.gitconfig` file:
|
||||
|
||||
```text
|
||||
[commit]
|
||||
gpgsign = true
|
||||
```
|
||||
|
||||
### What if I forgot?
|
||||
|
||||
You can retroactively sign your previous commit using `--amend`, for example
|
||||
|
||||
```shell
|
||||
$ git commit -S --amend
|
||||
```
|
||||
|
||||
If you need to go further back, you can use the interactive rebase
|
||||
command with 'edit'. Replace `HEAD~3` with the base commit from which
|
||||
you want to start.
|
||||
|
||||
```shell
|
||||
$ git rebase -i HEAD~3
|
||||
```
|
||||
|
||||
Replace 'pick' by 'edit' for the commit that you want to sign and the
|
||||
rebasing will stop after that commit. Then you can amend the commit as
|
||||
above. Afterwards, do
|
||||
|
||||
```shell
|
||||
$ git rebase --continue
|
||||
```
|
||||
|
||||
As this will rewrite history, you cannot do this when your commit is
|
||||
already merged. In that case, too bad, better luck next time.
|
||||
|
||||
If you rewrite history for another reason - for example when squashing
|
||||
commits - make sure that you re-sign as the signatures will be lost.
|
||||
|
||||
Multiple commits can also be re-signed with `git rebase`. For example, signing
|
||||
the last three commits can be done with:
|
||||
|
||||
```shell
|
||||
$ git rebase --exec 'git commit --amend --no-edit -n -S' -i HEAD~3
|
||||
```
|
||||
|
||||
### How to check if commits are signed?
|
||||
|
||||
Use `git log` with `--show-signature`,
|
||||
|
||||
```shell
|
||||
$ git log --show-signature
|
||||
```
|
||||
|
||||
You can also pass the `--show-signature` option to `git show` to check a single
|
||||
commit.
|
||||
|
||||
|
||||
## Pointing to Remote Dependent Branches in Go Modules
|
||||
|
||||
It's common that a developer may need to make a change in a dependent project
|
||||
of `lnd` such as `btcd`, `neutrino`, `btcwallet`, etc. In order to test changes
|
||||
without testing infrastructure, or simply make a PR into `lnd` that will build
|
||||
without any further work, the `go.mod` and `go.sum` files will need to be
|
||||
updated. Luckily, the `go mod` command has a handy tool to do this
|
||||
automatically so developers don't need to manually edit the `go.mod` file:
|
||||
```shell
|
||||
$ go mod edit -replace=IMPORT-PATH-IN-LND@LND-VERSION=DEV-FORK-IMPORT-PATH@DEV-FORK-VERSION
|
||||
```
|
||||
|
||||
Here's an example replacing the `lightning-onion` version checked into `lnd` with a version in roasbeef's fork:
|
||||
```shell
|
||||
$ go mod edit -replace=github.com/lightningnetwork/lightning-onion@v0.0.0-20180605012408-ac4d9da8f1d6=github.com/roasbeef/lightning-onion@2e5ae87696046298365ab43bcd1cf3a7a1d69695
|
||||
```
|
||||
|
||||
## Use of Log Levels
|
||||
|
||||
There are six log levels available: `trace`, `debug`, `info`, `warn`, `error` and `critical`.
|
||||
|
||||
Only use `error` for internal errors that are never expected to happen during
|
||||
normal operation. No event triggered by external sources (rpc, chain backend,
|
||||
etc) should lead to an `error` log.
|
||||
|
||||
## Use of Golang submodules
|
||||
|
||||
Changes to packages that are their own submodules (e.g. they contain a `go.mod`
|
||||
and `go.sum` file, for example `tor/go.mod`) require a specific process.
|
||||
We want to avoid the use of local replace directives in the root `go.mod`,
|
||||
therefore changes to a submodule are a bit involved.
|
||||
|
||||
The main process for updating and then using code in a submodule is as follows:
|
||||
- Create a PR for the changes to the submodule itself (e.g. edit something in
|
||||
the `tor` package)
|
||||
- Wait for the PR to be merged and a new tag (for example `tor/v1.0.x`) to be
|
||||
pushed.
|
||||
- Create a second PR that bumps the updated submodule in the root `go.mod` and
|
||||
uses the new functionality in the main module.
|
||||
|
||||
Of course the two PRs can be opened at the same time and be built on top of each
|
||||
other. But the merge and tag push order should always be maintained.
|
@@ -85,7 +85,7 @@ entry point for anyone not yet familiar with the process:
|
||||
- Bonus: Write a unit test (or test case) that you can propose to add to the PR.
|
||||
- Go through the PR commit by commit and line by line
|
||||
- Is the flow of the commits easy to understand? Does the order make sense?
|
||||
See the [code contribution guidelines](code_contribution_guidelines.md)
|
||||
See the [code contribution guidelines](development_guidelines.md#ideal-git-commit-structure)
|
||||
for more information on this.
|
||||
- Is the granularity of the commit structure easy to understand? Should
|
||||
commits be split up or be combined?
|
||||
|
Reference in New Issue
Block a user