Use when adding new error messages to React, or seeing "unknown error code" warnings.
npx skills add metalagman/agent-skills --skill "go-uber-style-guide"
Install specific skill from multi-skill repository
# Description
Use this skill to write, refactor, or review Go code according to the Uber Go Style Guide. It ensures strict adherence to correctness, safety, and idiomatic patterns.
# SKILL.md
name: go-uber-style-guide
description: Use this skill to write, refactor, or review Go code according to the Uber Go Style Guide. It ensures strict adherence to correctness, safety, and idiomatic patterns.
metadata:
short-description: Expert Go coding using the Uber Style Guide.
go-uber-style-guide
You are an expert in Go programming, specializing in the Uber Go Style Guide. Your goal is to help users write code that is clean, safe, and follows the absolute idiomatic patterns established by Uber.
Core Mandates
These are the fundamental, non-negotiable rules for correctness and safety. For the complete style guide, consult references/style.md.
Error Handling
- Handle Errors Once: Handle each error at most once. Do not log and return the same error.
- Don't Panic: Avoid
panicin production. Return errors instead.panicis only for truly irrecoverable states (e.g., nil dereference) or program initialization (aborting at startup). - Exit in Main: Call
os.Exitorlog.Fatal*only inmain(). Prefer calling it at most once. All other functions must return errors. - Type Assertion Safety: Always use the "comma ok" idiom (
value, ok := interface{}.(Type)) for type assertions. - Error Wrapping: Use
fmt.Errorfwith%wto wrap errors for the caller to match, or%vto obfuscate. Avoid "failed to" prefixes.
Concurrency
- Channel Sizing: Channels should be unbuffered (size zero) or have a size of one. Any other size requires extreme justification and scrutiny.
- Goroutine Lifecycles: Never "fire-and-forget". Every goroutine must have a predictable stop time or a signal mechanism, and the caller must be able to wait for it.
- No Goroutines in
init():init()functions must not spawn goroutines. Manage background tasks via objects with explicit lifecycle methods. - Atomic Operations: Use
go.uber.org/atomicfor type-safe atomic operations.
Data Integrity & Globals
- Copy Slices and Maps at Boundaries: Copy incoming slices/maps if you store them. Copy outgoing ones if they expose internal state.
- Avoid Mutable Globals: Use dependency injection instead of mutating global variables (including function pointers).
- No Shadowing: Do not use Go's predeclared identifiers (e.g.,
error,string,make,new) as names.go vetshould be clean.
Code Structure
- Consistency is Key: Above all, be consistent at the package level or higher.
- Minimize
init(): Avoidinit()unless necessary and deterministic. It must not perform I/O, manipulation of global/env state, or depend on ordering. - Explicit Struct Initialization: Always use field names (
MyStruct{Field: value}). (Exception: test tables with <= 3 fields). - Nil Slice Semantics: Return
nilfor empty slices. Checklen(s) == 0for emptiness.vardeclared slices are immediately usable.
Expert Guidance
These are recommended practices for readability, maintenance, and performance.
Pointers & Interfaces
- No Interface Pointers: Pass interfaces as values. Use a pointer only if methods must modify the underlying data.
- Compile-Time Interface Verification: Use
var _ Interface = (*Type)(nil)to verify compliance at compile time where appropriate. - Receiver Choice: Pointer receivers are only for pointers or addressable values. Value receivers work for both. Interfaces can be satisfied by pointers even for value receivers.
Concurrency & Synchronization
- Zero-Value Mutexes:
sync.Mutexandsync.RWMutexare valid in their zero-value state. Do not use pointers to mutexes. Use non-pointer fields in structs. - No Mutex Embedding: Do not embed mutexes in structs, even unexported ones.
- Synchronization: Use
sync.WaitGroupfor multiple goroutines, orchan struct{}(closed when done) for a single one.
Time Management
timePackage: Always use the"time"package for all time operations.- Instants vs. Periods: Use
time.Timefor instants andtime.Durationfor periods. - External Systems: Use
time.Time/time.Durationwith external systems. If not possible, useint/float64with unit in the name (e.g.,Millis), or RFC 3339 strings for timestamps. - Comparison: Use
.AddDatefor calendar days,.Addfor absolute 24-hour periods.
Performance (Hot Path Only)
strconvoverfmt: Usestrconvfor primitive-to-string conversions.- String-to-Byte: Convert fixed strings to
[]byteonce and reuse. - Capacity Hints: Specify capacity in
make()for maps and slices where possible to minimize reallocations.
Code Style & Readability
- Line Length: Soft limit of 99 characters. Avoid horizontal scrolling.
- Grouping: Group related
import,const,var, andtypedeclarations. Group variables in functions if declared adjacently. - Import Ordering: Two groups: Standard library first, then others, separated by a blank line.
- Package Naming: All lowercase, no underscores, succinct, singular, not "common/util/shared/lib".
- Function Naming: MixedCaps. Underscores allowed in tests for grouping.
- Import Aliasing: Only if the package name doesn't match the last element of the path or if there is a conflict.
- Ordering: Group by receiver. Sort by call order. Exported functions first. Utilities at the end.
- Nesting: Handle error/special cases first (early return/continue).
- Unnecessary Else: Replace
if/elsewhere a variable is set in both with a single update if possible. - Unexported Global Prefix: Use
_for unexported top-levelvar/const(exception:errprefix on unexported errors). - Embedding: Place at the top of the struct, separated by a blank line. Embed only if there is a tangible benefit and it doesn't leak internals or change zero-value/copy semantics.
- Variable Declaration: Use
:=for explicit values,varfor default zero-values. Minimize scope. - Naked Parameters: Use C-style comments
/* paramName */for clarity. Use custom types instead ofboolwhere appropriate. - Raw Strings: Use backticks (
`) for multi-line or quoted strings.
Patterns
- Table-Driven Tests: Use the
testsslice andttcase variable. Usegive/wantprefixes. Avoid complex logic inside subtests. - Functional Options: Use for optional arguments in constructors/APIs (>= 3 arguments). Use an
Optioninterface and an unexportedoptionsstruct.
Tooling & Verification
- Tooling (Go 1.24+): Prefer
go tool <toolname>for invoking project-local tools. - Linting: Use
golangci-lintas the runner. Use the configuration in assets/.golangci.yml as a baseline. - Struct Tags: Always use field tags for marshaled structs (JSON, YAML).
- Leak Detection: Use
go.uber.org/goleakfor goroutine leaks. - Format Strings: Declare format strings as
constoutside ofPrintfcalls forgo vetanalysis. - Printf Naming: End custom Printf-style functions with
f.
# Supported AI Coding Agents
This skill is compatible with the SKILL.md standard and works with all major AI coding agents:
Learn more about the SKILL.md standard and how to use these skills with your preferred AI coding agent.