This guide includes background and tips for working on the Radius Go codebase.
Go is a great language for newcomers! Due to its simple style and uncomplicated design, we find that new contributors can get going without a long learning process.
For learning Go, we recommend the following resources:
We're happy to accept pull-requests and give code review feedback aimed at newbies. If you have programmed in other languages before, we are confident you can pick up Go and start contributing easily.
Get stuck while working on a change? Want to get advice on coding style or existing code? Creating a forum post on our Discord server is a great way to get help.
You'll want to run the following command often:
make build test lint
This will build, run unit tests, and run linters to point out any problems. It's a good idea to run this if you're about to make a git commit
.
If you're looking for something, use the code organization guide to find what you're looking for.
We enforce coding style through using gofmt.
We stick to the usual philosophy of Go projects regarding styling, meaning that we prefer to avoid bikeshedding and debates about styling:
gofmt isn't anybody's preferred style, but it's adequate for everybody.
If you're using a modern editor with Go support, chances are it is already integrated with gofmt
and this will mostly be automatic. If there's any question about how to style a piece of code, following the style of the surrounding code is a safe bet.
We also mostly agree with Google's Go Style Guide, but don't follow it strictly or enforce everything written there. If you're new to working on a Go project, this is a great read that will get you thinking critically about the small decisions you will make when writing Go code. You can find an appendix further down with some cases where we disagree with and don't follow their guidance.
One thing we do require is godoc comments on exported packages, types, variables, constants, and functions. We like this because it has two good effects:
- Encourages you to minimize the exported surface-area, thus simplifying the design.
- Requires you to document clearly the purpose code you expect other parts of the codebase to call.
Right now we don't have automated enforcement of this rule, so expect it to come up in code review if you forget.
Most of this is standard practice for error handling in Go. We find that beginners struggle to understand and apply the right patterns so we're providing some advice.
The Google Go Style Guide has some excellent guidance for errors.
Radius code SHOULD NOT suppress errors without a good reason.
// Bad: Don't do this
result, _ := someErrorReturningFunc()
useResult(result)
If you have a good reason to ignore an error then you should:
- Document the rationale with a comment.
- Handle a specific error type rather than all errors.
- Propagate the error for all other cases.
// Good: do this
result, err := findWidget(id)
if errors.Is(err, &WidgetDoesNotExistError{}) {
// If the widget does not exist we want to create it before continuing.
err := createWidget(id)
if err != nil {
return err
}
// Widget exists, so let's continue
} else if err != nil {
return err
}
We run golint-ci as part of the pull-request process for static analysis. We don't have many customizations and mostly rely on the defaults.
We run CodeQL as part of the pull-request process for security analysis. If the CodeQL analysis finds a security issue it will be reported as part of the PR checks. CodeQL is not currently required to pass for a PR to be merged, as it may be triggered by other alerts within the repo.
If CodeQL fails due to your changes, please work with the maintainers to resolve the issue.
We mostly ignore the Google style guide's guidance for testing. In particular, we really like assertion libraries and use them in our tests.