Write a unit test

See also: Unit testing

This document demonstrates how to write a unit test for Juju.

Prepare for the test

Create package_test.go

[note type=caution] This step is necessary only if this file doesn’t already exist. [/note]

Each package requires a package_test.go file if we wish any of our tests to run.

Below is a standard package_test.go file for an example package called magic. We import the “testing” package from the standard library and then the gocheck package as gc. We also create a function Test that will be the entry-point into our test suites.

// Copyright 20XX Canonical Ltd.
// Licensed under the AGPLv3, see LICENCE file for details.

package magic_test

import (
  "testing"

  gc "gopkg.in/check.v1"
)

func Test(t *testing.T) {
  gc.TestingT(t)
}

[note type=caution] You will sometimes see package_test.go files which use testing.MgoTestPackage as their entrypoint. This is required to run old-style JujuConnSuite tests, which test against a running instance of MongoDB.

These tests are deprecated and are actively being removed. No more should be added. [/note]

Create

`

In the code directory, for each file that you want to test (say, a source code file called magic1.go), create a <code-filename>_test.go (e.g. magic1_test.go).

Import gocheck

See also: gocheck

In magic1_test.go, import the gocheck package as gc:

import (
gc "gopkg.in/check.v1"
)

[note type=information] gc is the usual alias for gocheck across all the Juju repositories. [/note]

Add a unit test suite

See also: Unit test suite

Also in magic1_test.go, add a unit test suite.

Once the test suite structure has been created, it needs to be registered with gc or the tests will not run. You can do by passing a pointer to an instance of our suite to the gc.Suite function.

type magicSuite struct{}

var _ = gc.Suite(&magicSuite{})

Write the test

See also: Checker

In magic1_test.go, below the test suite, start adding your unit test functions.

The process is as follows: You target some behavior (usually a function) in the code file (in our case, magic1). You then write a test for it, where the test usually follows the same Given, When, Then logic.

For example, suppose your magic1.go file defines a simple function called Sum:

func Sum(a, b int) int {
return a + b
}

Then, in your magic1_test.go file you can write a test for it as follows (where gc.Equals is a Checker):

// GIVEN a equals 5 AND b equals 3
// WHEN a and b are summed
// THEN we get 8
func (s *magicSuite) TestSum(c *gc.C) {
a := 5
b := 3

res := magic.Sum(a, b)

c.Assert(res, gc.Equals, 8)
}

Run the test

Finally, to run the test, do:

go test github.com/juju/juju/x/y/magic/

This will run all the tests registered in the magic package, including the one we just wrote.

You can also chose to run specific tests or suites, using the -check.f flag for gocheck

go test github.com/juju/juju/x/y/magic/ -check.f magicSuite         # run the magicSuite only
go test github.com/juju/juju/x/y/magic/ -check.f magicSuite.TestSum # run the test TestSum in magicSuite only

[note type=information] See more here gocheck > Selecting which tests to run . [/note]

Debug a test

If you need to reproduce a failing test but can’t reproduce it easily, use this script: juju/scripts/unit-test/stress-race.bash.

Tip

Where to run? Running on a small or medium instance on AWS will likely help trigger races more quickly than your local hardware. Particularly useful are instances that shares CPU time – t._n_ instances currently. If you locally build the test to stress you may still need to rsync over the build environment as some tests look for files in the build tree. You’ll also need to install mongo.

Tip

How many times to run? It has been noticed that, if the test runs 100 times without failure, things are probably all right.