Using GoConvey for Testing in Go

GoConvey is a testing framework for Go that provides a readable DSL for writing test cases with nested structures and web UI support for test execution. It builds on Go’s standard testing package while making test outputs more structured and human-readable.


Installation

You can install GoConvey using:

go get github.com/smartystreets/goconvey

Hands-on Example: Testing a Function with GoConvey

1. Create a Function to Test

We’ll write a simple function that adds two integers:

package mathutils
 
func Add(a, b int) int {
    return a + b
}

2. Write a GoConvey Test

Create a test file (mathutils_test.go) in the same package:

package mathutils
 
import (
    "testing"
 
    . "github.com/smartystreets/goconvey/convey"
)
 
func TestAdd(t *testing.T) {
    Convey("Given two numbers", t, func() {
        a, b := 3, 5
 
        Convey("When they are added", func() {
            result := Add(a, b)
 
            Convey("Then the result should be their sum", func() {
                So(result, ShouldEqual, 8)
            })
        })
    })
}

3. Run the Tests

To execute tests, use:

go test ./...

Or to use the GoConvey web UI, navigate to your project directory and run:

$GOPATH/bin/goconvey

Then open http://localhost:8080 in a browser.

Key Features of GoConvey in the Example

  1. Convey(description, t, func()) – Defines a test case with a clear description.
  2. Nested Convey blocks – Allows structuring tests hierarchically.
  3. So(value, assertion, expected) – Asserts conditions (ShouldEqual, ShouldBeNil, etc.).
  4. Readable output – GoConvey enhances readability by showing structured test outputs.

Key Features of GoConvey

Nested Test Cases

GoConvey supports hierarchical test structures, making tests easier to read and maintain:

func TestMathOperations(t *testing.T) {
    Convey("Given a set of numbers", t, func() {
        a, b := 10, 20
 
        Convey("When adding them", func() {
            result := Add(a, b)
 
            Convey("Then the result should be correct", func() {
                So(result, ShouldEqual, 30)
            })
        })
 
        Convey("When subtracting them", func() {
            result := a - b
 
            Convey("Then the result should be correct", func() {
                So(result, ShouldEqual, -10)
            })
        })
    })
}

This structure ensures grouping of related tests under a common setup.

Assertions (So Function)

GoConvey provides rich assertions using the So function:

AssertionDescriptionExample
ShouldEqualChecks if values are equalSo(x, ShouldEqual, 5)
ShouldNotEqualChecks if values are not equalSo(x, ShouldNotEqual, 10)
ShouldBeNilChecks if value is nilSo(err, ShouldBeNil)
ShouldNotBeNilChecks if value is not nilSo(obj, ShouldNotBeNil)
ShouldBeTrueChecks if value is trueSo(flag, ShouldBeTrue)
ShouldBeFalseChecks if value is falseSo(flag, ShouldBeFalse)
ShouldContainChecks if a slice or string contains a valueSo([]int{1,2,3}, ShouldContain, 2)
ShouldResembleDeep comparison of slices, structs, mapsSo(slice1, ShouldResemble, slice2)

Example:

So(5, ShouldEqual, 5)         // Pass
So(10, ShouldNotEqual, 5)     // Pass
So(nil, ShouldBeNil)          // Pass
So([]int{1, 2, 3}, ShouldContain, 2) // Pass

Table-Driven Testing with GoConvey

For repetitive tests, GoConvey can be used in a table-driven manner.

func TestAddMultipleCases(t *testing.T) {
    testCases := []struct {
        a, b     int
        expected int
    }{
        {1, 2, 3},
        {5, 5, 10},
        {10, -10, 0},
    }
 
    Convey("Testing Add function with multiple cases", t, func() {
        for _, tc := range testCases {
            Convey("Adding two numbers", func() {
                So(Add(tc.a, tc.b), ShouldEqual, tc.expected)
            })
        }
    })
}

Handling Edge Cases and Error Conditions

Let’s modify our function to return an error if inputs are negative:

package mathutils
 
import "errors"
 
func SafeAdd(a, b int) (int, error) {
    if a < 0 || b < 0 {
        return 0, errors.New("negative numbers not allowed")
    }
    return a + b, nil
}

Now, let’s test it:

func TestSafeAdd(t *testing.T) {
    Convey("Given two numbers", t, func() {
        Convey("When both are positive", func() {
            result, err := SafeAdd(3, 5)
 
            Convey("Then it should return their sum without error", func() {
                So(result, ShouldEqual, 8)
                So(err, ShouldBeNil)
            })
        })
 
        Convey("When one is negative", func() {
            result, err := SafeAdd(-3, 5)
 
            Convey("Then it should return an error", func() {
                So(result, ShouldEqual, 0)
                So(err, ShouldNotBeNil)
                So(err.Error(), ShouldEqual, "negative numbers not allowed")
            })
        })
    })
}