From golang-skills
Guides Go error handling strategy: sentinel vs typed vs opaque errors, wrapping with %w vs %v, and when to log vs return. Invoked automatically when working with Go errors.
How this skill is triggered — by the user, by Claude, or both
Slash command
/golang-skills:go-error-handlingThis skill is limited to the following tools:
The summary Claude sees in its skill listing — used to decide when to auto-load this skill
> Compatibility: `errors.Is`, `errors.As`, and `%w` wrapping require Go 1.13+; structured logging examples may use `log/slog` from Go 1.21+.
Compatibility:
errors.Is,errors.As, and%wwrapping require Go 1.13+; structured logging examples may uselog/slogfrom Go 1.21+.
scripts/check-errors.sh - Run when checking string-based error matching, bare error propagation, and log-and-return patterns.scripts/check-errors-ast.go - Implementation helper invoked by check-errors.sh; patch this when changing error-flow analysis behavior.references/ERROR-FLOW.md - Read when deciding where to handle, wrap, log, or return errors.references/ERROR-TYPES.md - Read when choosing sentinel errors, typed errors, or opaque errors.references/WRAPPING.md - Read when choosing %w versus %v or crossing package boundaries.In Go, errors are values — they are created by code and consumed by code.
%v to avoid leaking internals%wfmt.Errorf("...: %w", err)Default: wrap with %w and place it at the end of the format string.
Never return concrete error types from exported functions — a concrete nil
pointer can become a non-nil interface:
// Bad: Concrete type can cause subtle bugs
func Bad() *os.PathError { /*...*/ }
// Good: Always return the error interface
func Good() error { /*...*/ }
Error strings should not be capitalized and should not end with punctuation. Exception: exported names, proper nouns, or acronyms.
// Bad
err := fmt.Errorf("Something bad happened.")
// Good
err := fmt.Errorf("something bad happened")
For displayed messages (logs, test failures, API responses), capitalization is appropriate.
When a function returns an error, callers must treat all non-error return values as unspecified unless explicitly documented.
Tip: Functions taking a context.Context should usually return an error
so callers can determine if the context was cancelled.
When encountering an error, make a deliberate choice — do not discard
with _:
log.Fatal or panicTo intentionally ignore: add a comment explaining why.
n, _ := b.Write(p) // never returns a non-nil error
For related concurrent operations, use
errgroup:
g, ctx := errgroup.WithContext(ctx)
g.Go(func() error { return task1(ctx) })
g.Go(func() error { return task2(ctx) })
if err := g.Wait(); err != nil { return err }
Don't return -1, nil, or empty string to signal errors. Use multiple
returns:
// Bad: In-band error value
func Lookup(key string) int // returns -1 for missing
// Good: Explicit error or ok value
func Lookup(key string) (string, bool)
This prevents callers from writing Parse(Lookup(key)) — it causes a
compile-time error since Lookup(key) has 2 outputs.
Handle errors before normal code. Early returns keep the happy path unindented:
// Good: Error first, normal code unindented
if err != nil {
return err
}
// normal code
Handle errors once — either log or return, never both:
Error encountered?
├─ Caller can act on it? → Return (with context via %w)
├─ Top of call chain? → Log and handle
└─ Neither? → Log at appropriate level, continue
Advisory: Recommended best practice.
| Caller needs to match? | Message type | Use |
|---|---|---|
| No | static | errors.New("message") |
| No | dynamic | fmt.Errorf("msg: %v", val) |
| Yes | static | var ErrFoo = errors.New("...") |
| Yes | dynamic | custom error type |
Default: Wrap with fmt.Errorf("...: %w", err). Escalate to sentinels for
errors.Is(), to custom types for errors.As().
Advisory: Recommended best practice.
%v: At system boundaries, for logging, to hide internal details%w: To preserve error chain for errors.Is/errors.AsKey rules: Place %w at the end. Add context callers don't have. If
annotation adds nothing, return err directly.
Validation: After implementing error handling, run
bash scripts/check-errors.shto detect common anti-patterns. Then rungo vet ./...to catch additional issues.
ErrFoo) or custom error typeserrors.Is/errors.As or writing error-checking helpersnpx claudepluginhub cxuu/golang-skills --plugin golang-skillsProvides Go error handling patterns: basic errors, wrapping with %w, sentinel errors, custom types, errors.Is, and Unwrap for robust apps.
Idiomatic Go error handling — creation, wrapping with %w, errors.Is/As/Join, custom types, sentinels, panic/recover, slog logging, HTTP middleware, and samber/oops for production errors. Apply when creating, wrapping, inspecting, or logging errors in Go codebases.
Implements and reviews Go error handling: creates/wraps errors with fmt.Errorf, checks via errors.Is/As/Join, uses sentinels/custom types, logs with slog. Audits codebases and PRs.