Skip to content

negrel/assert

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

23 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

gopher illustration

PkgGoDev Go Report Card

assert - Zero cost debug assertions.

This package provides zero cost debug assertions to your Go programs. It is based on the excellent github.com/stretchr/testify/assert package and provide the same API (minus t testing.T parameter).

Why?

This is a complete rewrite of debuggo that aims to be up to date and more maintainable.

It aims to provide the same API as github.com/stretchr/testify/assert.

  • Prints friendly, easy to read failure descriptions
  • Allows for very readable code
  • Optionally annotate each assertion with a message
  • No performance impact on production build (see benchmarks)

Debug assertions main use is to assert invariant that can't be encoded in the type systems. For example, a method that should be called only when mutex is locked:

package mypkg

import (
	"sync"
	"github.com/negrel/assert"
)

type myType struct {
	mu sync.Mutex
	// other fields...
}

// doWork perform internal work. Caller must hold mutex while calling this
// function.
func (mt *myType) doWork(k string) {
	assert.Locked(&mt.mu) // panic if assertions are enabled and mutex isn't locked

	// Do work...
}

How does it works?

Read my blog post about to understand how assert works and why it is designed that way.

Getting started

Here is our example program:

package main

import (
	"github.com/negrel/assert"
)

func main() {
	assert.True(false)
	println("Hello world!")
}

A simple go run . will simply print Hello world! as all assert functions are removed by the compiler.

Now, if we compile and run it with assertions enabled go run -tags assert ., it will output something like:

panic:
        Error Trace:    /home/anegrel/code/go/assert/example/main.go:8
                                                /usr/share/go/src/runtime/proc.go:267
                                                /usr/share/go/src/runtime/asm_amd64.s:1650
        Error:          Should be true


goroutine 1 [running]:
github.com/negrel/assert.Fail({0x568dc2, 0xe}, {0x0, 0x0, 0x0})
        /home/anegrel/code/go/assert/assertions.go:349 +0x168
github.com/negrel/assert.True(...)
        /home/anegrel/code/go/assert/assertions.go:754
main.main()
        /home/anegrel/code/go/assert/example/main.go:8 +0x27
exit status 2

Note that most go subcommands (build, run, test, ...) supports -tags flag. You may want to set GOFLAGS environment variable to -tags assert make it permanent and avoid specifying it on each command.

Benchmarks

As we've seen previously, assertions are hidden behind a compilation flag. If the flag is absent, all assertions functions will be empty/noop function that the compiler will optimize.

WITH -tags assert:

goos: linux
goarch: amd64
pkg: github.com/negrel/assert/tests
cpu: Intel(R) Core(TM) i5-8250U CPU @ 1.60GHz
BenchmarkSliceIndexWithoutBoundCheckAssertions
BenchmarkSliceIndexWithoutBoundCheckAssertions-8        728439501                1.407 ns/op
BenchmarkSliceIndexWithBoundCheckAssertions
BenchmarkSliceIndexWithBoundCheckAssertions-8           27423670                40.80 ns/op
PASS
ok      github.com/negrel/assert/tests  3.338s

WITHOUT -tags assert:

goos: linux
goarch: amd64
pkg: github.com/negrel/assert/tests
cpu: Intel(R) Core(TM) i5-8250U CPU @ 1.60GHz
BenchmarkSliceIndexWithoutBoundCheckAssertions
BenchmarkSliceIndexWithoutBoundCheckAssertions-8        772181695                1.399 ns/op
BenchmarkSliceIndexWithBoundCheckAssertions
BenchmarkSliceIndexWithBoundCheckAssertions-8           802181890                1.412 ns/op
PASS
ok      github.com/negrel/assert/tests  2.531s

However, keep in mind that assert may slightly increase binary size (~100 KiB) as it imports net/http and reflect.

Contributing

If you want to contribute to assert to add a feature or improve the code contact me at negrel.dev@protonmail.com, open an issue or make a pull request.

🌠 Show your support

Please give a ⭐ if this project helped you!

buy me a coffee

📜 License

MIT © Alexandre Negrel

About

0️⃣ Zero cost debug assertions for Go.

Topics

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages