Tiny (~100 LoC) Go assertion library focused on crystal-clear failure messages and thoughtful source context.
- 🔍 Crystal-clear failure messages with contextual values
- 📚 Rich source context showing the exact failure location
- 🛠 Tiny and free of dependencies (~100 lines of Go)
- 💡 Elegant, idiomatic Go API
- 🎯 Two-tier assertion system with build tag support
- ⚙️ Configurable source context behavior
Inspired by Tiger Style.
go get github.com/nikoksr/assert-go
import "github.com/nikoksr/assert-go"
func PaymentProcessing() {
payment := processPayment(PaymentRequest{
Amount: 99.99,
CustomerID: "cust_123",
Currency: "USD",
})
// Assert payment was processed successfully
assert.Assert(payment.Status == "completed", "payment should be completed",
// Optionally, add context to the panic
"payment_id", payment.ID,
"status", payment.Status,
"amount", payment.Amount,
"error", payment.Error,
"timestamp", payment.ProcessedAt,
)
}
On failure, you get:
Assertion failed at payment_test.go:43
Message: payment should be completed
Relevant values:
[payment_id]: "pmt_789"
[status]: "failed"
[amount]: 99.99
[error]: "insufficient_funds"
[timestamp]: "2024-12-06T15:04:05Z"
Source context:
37 | payment := processPayment(PaymentRequest{
38 | Amount: 99.99,
39 | CustomerID: "cust_123",
40 | Currency: "USD",
41 | })
42 |
→ 43 | assert.Assert(payment.Status == "completed", "payment should be completed",
44 | "payment_id", payment.ID,
45 | "status", payment.Status,
46 | "amount", payment.Amount,
47 | "error", payment.Error,
48 | "timestamp", payment.ProcessedAt,
49 | )
goroutine 1 [running]:
github.com/nikoksr/assert-go.PaymentProcessing(0xc00011c000)
/app/payment.go:43 +0x1b4
# ... regular Go stacktrace continues
The library provides two types of assertions:
Assert()
- Always active, meant for critical checks that should run in all environmentsDebug()
- Development-time assertions that can be disabled in production
Debug assertions are disabled by default. To enable them, use the assertdebug
build tag:
go test -tags assertdebug ./...
go run -tags assertdebug main.go
Example usage:
// This will only be evaluated when built with -tags assertdebug
assert.Debug(len(items) < 1000, "items list too large",
"current_length", len(items),
"max_allowed", 1000,
)
// This will always be evaluated regardless of build tags
assert.Assert(response != nil, "HTTP response cannot be nil",
"status_code", response.StatusCode,
)
You can configure the assertion behavior:
// Configure assertion behavior
assert.SetConfig(assert.Config{
// Enable/disable source context in error messages
IncludeSource: true,
// Number of context lines to show before and after the failing line
ContextLines: 5,
})
I initially shared the common view that assertions don't align well with Go's philosophy of explicit error handling. Reading TigerStyle's perspective on assertions made me reconsider this stance and experiment with them in my own code.
I've found that assertions serve a distinct and valuable purpose alongside traditional error handling. While I handle operational failures - like network issues or invalid user input - through error returns, I now use assertions to catch programmer mistakes that should never occur in correct code. I used to write sanity checks that would return errors, wondering why I'm burdening users with checks for conditions that should be impossible to be false anyway given my code's structure – like a logger that I initialized and passed down myself just three function calls earlier. It feels weird to check for nil because I know that I just initialized this logger, it feels weird to return these types of errors to users, but at the same time, I always had this urge of checking for the "impossible". These aren't cases where graceful error handling makes sense; they're cases where continuing execution would only mask a fundamental bug in my code.
I'm selective about where I use assertions. They belong in application code where I can make strong guarantees about internal state and invariants, particularly during system initialization. I don't use them in libraries or for validating application input - that's firmly error handling territory. But when I know something must be true for my program to be correct, assertions help me catch bugs early and prevent corrupted state from silently spreading.
What started as an experiment has become an essential part of how I write Go. At this point, my own experience has convinced me that thoughtful use of assertions makes my code more reliable and bugs easier to diagnose.
- Minimal: Single-purpose library that does one thing well
- Context over complexity: Rich debugging information without complex APIs
- Clear failures: Source context shows exactly where and why things went wrong
- Idiomatic Go: Feels natural in your Go codebase