Styleguide¶
This document is a guide to aid Juju code reviewers.
Pull requests¶
A Pull Request (PR) needs a justification. This could be:
An issue, ie “Fixes LP 12349000”
A description justifying the change
A link to a design document or mailing list discussion
PRs should be kept as small as possible, addressing one discrete issue outlined in the PR justification. The smaller the diff, the better.
If your PR addresses several distinct issues, please split the proposal into one PR per issue.
Once your PR is up and reviewed, don’t comment on the review saying “i’ve done this”, unless you’ve pushed the code.
Functions¶
What the function does should be evident by its signature.
How a function does what it does should also be clear. If a function is large and verbose, breaking apart it’s business logic into helper functions can make it easier to read. Conversely, if the function’s logic is too opaque, in-lining a helper function or variable may help.
If a function has named return values in it’s signature, it should use a bare return:
func AwesomeFunc(i int) (s string, err error) {
// ...
return // Don't use the statement: "return s, err"
}
Errors¶
The text of error messages should be lower case without a trailing period, because they are often included in other strings or output:
// INCORRECT
return fmt.Errorf("Cannot read config %q.", configFilePath)
// CORRECT
return fmt.Errorf("cannot read agent config %q", configFilePath)
If a function call only returns an error, assign and handle the error within one if statement (unless doing so makes the code harder to read, for example if the line of the function call is already quite long):
// PREFERRED
if err := someFunc(); err != nil {
return err
}
If an error is thrown away, why it is not handled is explained in a comment.
The juju/errors package should be used to handle errors:
// Trace always returns an annotated error. Trace records the
// location of the Trace call, and adds it to the annotation stack.
// If the error argument is nil, the result of Trace is nil.
if err := SomeFunc(); err != nil {
return errors.Trace(err)
}
// Errorf creates a new annotated error and records the location that the
// error is created. This should be a drop in replacement for fmt.Errorf.
return errors.Errorf("validation failed: %s", message)
// Annotate is used to add extra context to an existing error. The location of
// the Annotate call is recorded with the annotations. The file, line and
// function are also recorded.
// If the error argument is nil, the result of Annotate is nil.
if err := SomeFunc(); err != nil {
return errors.Annotate(err, "failed to frombulate")
}
See github.com/juju/errors/annotation.go for more error handling functions.
Tests¶
Please read ../doc/how-to-write-tests.txt
Each test should test a discrete behaviour and is idempotent
The behaviour being tested is clear from the function name, as well as the expected result (i.e. success or failure)
Repeated boilerplate in test functions is extracted into it’s own helper function or
SetUpTest
External functionality, that is not being directly tested, should be mocked out
Variables in other modules are not patched directly. Instead, create a local variable and patch that:
// ----------
// in main.go
// ----------
import somepackage
var someVar = somepackage.SomeVar
// ---------------
// in main_test.go
// ---------------
func (s *SomeSuite) TestSomethingc(c *gc.C) {
s.PatchValue(someVar, newValue)
// ....
}
If your test functions are under package_test
and they need to test something
from package that is not exported, create an exported alias to it in export_test.go
:
// -----------
// in tools.go
// -----------
package tools
var someVar = value
// -----------------
// in export_test.go
// -----------------
package tools
var SomeVar = someVar
// ---------------
// in main_test.go
// ---------------
package tools_test
import tools
func (s *SomeSuite) TestSomethingc(c *gc.C) {
s.PatchValue(tools.SomeVar, newValue)
// ...
}
Layout¶
Imports are grouped into 3 sections: standard library, 3rd party libraries, juju/juju library:
import (
"fmt"
"io"
"github.com/juju/errors"
"github.com/juju/loggo"
gc "gopkg.in/check.v1"
"github.com/juju/juju/environs"
"github.com/juju/juju/environs/config"
)
API¶
Please read ../doc/api.txt
Client calls to the API are, by default, batch calls
CLI commands¶
The base Command
interface is found in cmd/cmd.go
.
Commands need to provide an Info
method that returns an Info struct.
The info struct contains: name, args, purpose and a detailed description. This information is used to provide the default help for the command.
In the same package, there is CommandBase
whose purpose is to be composed
into new commands, and provides a default no-op SetFlags implementation, a
default Init method that checks for no extra args, and a default Help method.
Supercommands¶
Supercommand
s are commands that do many things, and have “sub-commands” that
provide this functionality. Git and Bazaar are common examples of
“supercommands”. Subcommands must also provide the Command
interface, and
are registered using the Register
method. The name and aliases are
registered with the supercommand. If there is a duplicate name registered,
the whole thing panics.
Supercommands need to be created with the NewSuperCommand
function in order
to provide a fully constructed object.
The ‘help’ subcommand¶
All supercommand instances get a help command. This provides the basic help functionality to get all the registered commands, with the addition of also being able to provide non-command help topics which can be added.
Help topics have a name
which is what is matched from the command line, a
short
one line description that is shown when <cmd> help
is called,
and a long
text that is output when the topic is requested.
Execution¶
The Main
method in the cmd package handles the execution of a command.
A new gnuflag.FlagSet
is created and passed to the command in SetFlags
.
This is for the command to register the flags that it knows how to handle.
The args are then parsed, and passed through to the Init
method for the
command to decide what to do with the positional arguments.
The command is then Run
and passed in an execution Context
that defines
the standard input and output streams, and has the current working directory.
Interactive commands¶
Some commands in Juju are interactive. To keep the UX consistent across the product, please follow these guidelines. Note, many of these guidelines are supported by the interact package at github.com/juju/juju/cmd/juju/interact .
All interactive commands should begin with a short blurb telling the user that they’ve started an interactive wizard.
Starting interactive bootstrap process.
Prompts should be a short imperative sentence with the first letter capitalized, ending with a colon and a space before waiting for user input.
Enter your mother's maiden name:
Prompts should tell the user what to do, not ask them questions.
Enter a name:
rather thanWhat name do you want?
The only time a prompt should end with a question mark is for yes/no questions.
Use foobar as network? (Y/n):
Yes/no questions should always end with (y/n), with the default answer capitalized.
Try to always format the question so you can make yes the default.
Prompts that request the user choose from a list of options should start
Select a ...
Prompts that request the user enter text not from a list should start
Enter ....
Most prompts should have a reasonable default, shown in brackets just before the colon, which can be accepted by just hitting enter.
Select a cloud [localhost]:
Questions that have a list of answers should print out those options before the prompt.
Options should be consistently sorted in a logical manner (usually alphabetical).
Options should be a single short word, if possible.
Selecting from a list of options should always be case insensitive.
If an incorrect selection is entered, print a short error and reprint the prompt, but do not reprint the list of options. No blank line should be printed between the error and the corresponding prompt.
Always print a blank line between prompts.
The core/state package¶
Return values: ok vs err¶
By convention, anything that could reasonably fail must use a separate channel to communicate the failure. Broadly speaking, methods that can fail in more than one way must return an error; those that can only fail in one way (eg Machine.InstanceId: the value is either valid or missing) are expected to return a bool for consistency’s sake, even if the type of the return value is such that failure can be signalled in- band.
Changes to entities¶
Entity objects reflect remote state that may change at any time, but we don’t want to make that too obvious. By convention, the only methods that should change an entity’s in-memory document are as follows:
Refresh(), which should update every field.
Methods that set fields remotely should update only those fields locally.
The upshot of this is that it’s not appropriate to call Refresh on any argument to a state method, including receivers; if you’re in a context in which that would be helpful, you should clone the entity first. This is simple enough that there’s no Clone() method, but it would be kinda nice to implement them in our Copious Free Time; I think there are places outside state that would also find it useful.
Care and feeding of mgo/txn¶
Just about all our writes to mongodb are mediated by the mgo/txn package, and using this correctly demands some care. Not all the code has been written in a fully aware state, and cases in which existing practice is divergent from the advice given below should be regarded with some suspicion.
The txn package lets you make watchable changes to mongodb via lists of operations with the following critical properties:
transactions can apply to more than one document (this is rather the point of having them).
transactions will complete if every assert in the transaction passes; they will not run at all if any assert fails.
multi-document transactions are not atomic; the operations are applied in the order specified by the list.
operations, and hence assertions, can only be applied to documents with ids that are known at the time the operation list is built; this means that it takes extra work to specify a condition like “no unit document newer than X exists”.
The second point above deserves further discussion. Whenever you are implementing a state change, you should consider the impact of the following possible mongodb states on your code:
if mongodb is already in a state consistent with the transaction having already been run – which is always possible – you should return immediately without error.
if mongodb is in a state that indicates the transaction is not valid – eg trying to add a unit to a dying service – you should return immediately with a descriptive error.
Each of the above situations should generally be checked as part of preparing a new []txn.Op, but in some cases it’s convenient to trust (to begin with) that the entity’s in-memory state reflects reality. Regardless, your job is to build a list of operations which assert that:
the transaction still needs to be applied.
the transaction is actively valid.
facts on which the transaction list’s form depends remain true.
If you’re really lucky you’ll get to write a transaction in which the third requirement collapses to nothing, but that’s not really the norm. In practice, you need to be prepared for the run to return txn.ErrAborted; if this happens, you need to check for previous success (and return nil) or for known cause of invalidity (and return an error describing it).
If neither of these cases apply, you should assume it’s an assertion failure of the third kind; in that case, you should build a new transaction based on more recent state and try again. If ErrAborteds just keep coming, give up; there’s an ErrExcessiveContention that helps to describe the situation.
Watching entities, and selecting groups thereof¶
The mgo/txn log enables very convenient notifications of changes to particular documents and groups thereof. The state/watcher package converts the txn event log into events and send them down client- supplied channels for further processing; the state code itself implements watchers in terms of these events.
All the internally-relevant watching code is implemented in the file state/watcher.go. These constructs can be broadly divided into groups as follows:
single-document watchers: dead simple, they notify of every change to a given doc. SettingsWatcher bucks the convention of NotifyWatcher in that it reads whole *Settings~s to send down the channel rather than just sending notifications (we probably shouldn’t do that, but I don’t think there’s time to fix it right now).
entity group watchers: pretty simple, very common; they notify of every change to the Life field among a group of entities in the same collection, with group membership determined in a wide variety of ways.
relation watchers: of a similar nature to entity group watchers, but generating carefully-ordered events from observed changes to several different collections, none of which have Life fields.
Implementation of new watchers is not necessarily a simple task, unless it’s a genuinely trivial refactoring (or, given lack of generics, copy- and-paste job). Unless you already know exactly what you’re doing, please start a discussion about what you’re trying to do before you embark on further work in this area.
Transactions and reference counts¶
As described above, it’s difficult to assert things like “no units of service X exist”. It can be done, but not without additional work, and we’re generally using reference counts to enable this sort of thing.
In general, reference counts should not be stored within entities: such changes are part of the operation of juju and are not important enough to be reflected as a constant stream of events. We didn’t figure this out until recently, though, so relationDoc has a UnitCount, and serviceDoc has both a UnitCount and a RelationCount. This doesn’t matter for the relation – nothing watches relation docs directly – but I fear it matters quite hard for application, because every added or removed unit or relation will be an unnecessary event sent to every unit agent.
We’ll deal with that when it’s a problem rather than just a concern; but new reference counts should not be thoughtlessly added to entity documents, and opportunities to separate them from existing docs should be considered seriously.
Comments¶
Comments should be full sentences with one space after a period, proper capitalization and punctuation.
Avoid adding non-useful information, such as flagging the layout of a file with block comments or explaining behaviour that is immediately evident from reading the code.
All exported symbols (names, functions, methods, types, constants and variables) must have a comment.
Anything that is unintuitive at first sight must have a comment, such as:
non-trivial unexported type or function declarations (e.g. if a type implements an interface)
the breaking of a convention (e.g. not handling an error)
a particular behaviour the reader would have to think ‘more deeply’ about to understand
Top-level name comments start with the name of the thing. For example, a top-level, exported function:
A TODO comment needs to be linked to a bug in Launchpad. Include the lp bug number in the TODO with the following format:
// TODO(username) yyyy-mm-dd bug #1234567
e.g.
// TODO(axw) 2013-09-24 bug #1229507
Each file must have a copyright notice. The copyright notice must include the year the file was created and the year the file was last updated, if applicable.
Note that the blank line following the notice separates the comment from the package, ensuring it does not appear in godoc.