GO
go-backend-best-practices
Go backend patterns from Uber style and standard layout
Install
mkdir -p .claude/skills/go-backend-best-practices && curl -L -o skill.zip "https://agentskills.codes/api/skills/download/16569" && unzip -o skill.zip -d .claude/skills/go-backend-best-practices && rm skill.zipInstalls to .claude/skills/go-backend-best-practices
Activation
This is the description your AI agent reads to decide when to run this skill — the better it matches your request, the more reliably it fires.
Go backend patterns from Uber style and standard layout55 charsno explicit “when” trigger
About this skill
Purpose
Apply Go backend patterns for building production-ready services.
Rules
1. Project Structure (Standard Layout)
layout: "cmd/{server/main.go} + internal/{handler/,service/,repository/,model/} + pkg/{shared/} + api/{openapi.yaml} + configs/ + scripts/"
directories:
cmd: Main applications (one per binary)
internal: Private application code
pkg: Library code safe for external use
api: API definitions (OpenAPI, protobuf)
configs: Configuration files
scripts: Build and CI scripts
Reference: guides/go-backend/project-layout.md
2. Error Handling (Uber Style)
principles:
- Wrap errors with context using %w
- Handle errors once (don't log AND return)
- Use sentinel errors for specific conditions
- Name error variables with Err prefix
patterns:
sentinel: "var ErrNotFound = errors.New(\"not found\")"
wrapping: "fmt.Errorf(\"getUser %s: %w\", id, err)"
checking: "errors.Is(err, ErrNotFound)"
Reference: guides/go-backend/uber-style.md
3. Concurrency (Uber Style)
channels:
size: "Use 0 (unbuffered) or 1 only"
larger: "Requires careful review"
goroutines:
never: fire-and-forget
always: wait for completion or manage lifecycle
patterns: "sync.WaitGroup + error channel for parallel work; context.Context for cancellation"
Reference: guides/go-backend/uber-style.md
4. HTTP Server
structure:
handler: HTTP layer (request/response)
service: Business logic
repository: Data access
patterns:
dependency_injection: "Handler struct with service field, constructor NewXxxHandler()"
router: "chi.NewRouter() with middleware (Logger, Recoverer) and versioned routes"
error_mapping: "errors.Is() to map domain errors to HTTP status codes"
Reference: guides/go-backend/uber-style.md
5. Dependency Injection
approach: constructor injection
avoid: global variables
pattern: "Struct with dependencies as fields, New* constructor functions, wire up in main()"
Reference: guides/go-backend/uber-style.md
6. Configuration
approach:
- Use environment variables
- Validate at startup
- Group related settings
pattern: "Config struct with nested typed configs, env struct tags, Parse at startup"
Reference: guides/go-backend/uber-style.md
7. Testing
patterns:
table_driven: for comprehensive coverage
interfaces: for mocking
parallel: for speed
tools: "gomock for mocks, cmp.Diff for assertions"
Reference: guides/go-backend/uber-style.md
8. Performance (Uber Style)
guidelines:
- Use strconv over fmt for conversions
- Pre-allocate slices with known capacity
- Avoid repeated string-to-byte conversions
- Copy slices/maps at boundaries
Reference: guides/go-backend/uber-style.md
Application
When writing Go backend code:
- Always use standard project layout
- Always wrap errors with context
- Never fire-and-forget goroutines
- Use constructor injection
- Use table-driven tests
- Handle errors once
- Copy data at boundaries
- Validate config at startup