plugins/languages/golang/skills/error/SKILL.md
Go 错误处理规范——禁止单行 if err、必须记录日志(lazygophers/log 或 slog)、禁止包装错误、errors.Join 聚合、sentinel error 模式、初始化用 log.Fatalf、参数校验用 validate tag 禁手写 if。处理 Go error、写 if err 块、设计错误类型、做参数校验时触发。
npx skillsauth add lazygophers/ccplugin golang-errorInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
if err != nil {} 块,禁单行。log.Errorf("err:%v", err)。fmt.Errorf("%w", err)、不用 errors.Wrap,直接返回原始 err。log.Fatalf:init() 或启动期失败用 Fatalf,不用 panic。data, err := os.ReadFile(path)
if err != nil {
log.Errorf("err:%v", err)
return nil, err
}
err = eg.Wait()
if err != nil {
log.Errorf("err:%v", err)
return err
}
不写 if err := eg.Wait(); err != nil {},错误变量必须显式赋值后判断。
var errs []error
for _, item := range items {
err := processItem(item)
if err != nil {
log.Errorf("err:%v", err)
errs = append(errs, err)
}
}
if len(errs) > 0 {
return errors.Join(errs...)
}
var (
ErrNotFound = errors.New("not found")
ErrForbidden = errors.New("forbidden")
ErrBadRequest = errors.New("bad request")
)
判断方式(按项目约定择一):
// 直接比较 sentinel
if err == ErrNotFound { return nil, err }
// 自定义错误码
if xerror.CheckCode(err, CodeNotFound) { return nil, err }
// 类型判定函数
if IsNotFoundErr(err) { return nil, err }
file, err := os.Open(path)
if err != nil {
log.Errorf("err:%v", err)
return nil, err
}
defer file.Close()
type AddUserReq struct {
Username string `json:"username" validate:"required"`
Email string `json:"email" validate:"required,email"`
Age uint8 `json:"age" validate:"gte=0"`
}
validate tag 声明约束,校验器在中间件/拦截层统一执行if req.X == "" / if req.Y == 0 / if req.Z == nil 校验strings.TrimSpace 预处理请求字段import "github.com/lazygophers/log"
log.Infof("user registered: %s", username)
log.Warnf("cache miss for key: %s", key)
log.Errorf("err:%v", err)
log.Fatalf("failed to load config: %v", err)
import "log/slog"
slog.Info("user registered", "username", name)
slog.Error("operation failed", "err", err, "user_id", uid)
slog 在 2026 已是事实标准(go-test-coverage、service observability 主流)。新仓库优先用 slog + JSON handler。
func init() {
config, err := loadConfig()
if err != nil {
log.Errorf("err:%v", err)
log.Fatalf("failed to load config")
}
}
| AI 借口 | 实际应验证 | | --- | --- | | "单行 if err 更简洁" | 所有 error 多行? | | "fmt.Errorf 加上下文更好" | 禁止包装、直接 return? | | "errors.Is/As 更现代" | 用项目约定的判断方式? | | "panic 快速失败" | 业务用 return error、初始化用 Fatalf? | | "日志太多了" | 每个 error 分支都有日志? | | "logrus/zap 更强" | 用 lazygophers/log 或 slog? |
log.Errorf("err:%v", err) 格式fmt.Errorf/errors.Wrap 包装if errif err := f(); err != nil 内联声明errors.Joinerrors.New 定义development
Go 数据库规范——GORM Model 命名 ModelXxx、表名单数、枚举 uint8 + 常量、索引 idx_ 前缀 + deleted_at leading column、禁 time.Time 统一 int64 unix、禁指针/nullable 字段、TEXT/BLOB/JSON 禁 default、AutoMigrate 禁改主键。设计 DB model、写 GORM tag、建索引、做 migration 审查时触发。
development
Go HTTP API 规范——响应始终 200 + body code 字段、路由 /api/* 全 POST 单段 <Action><Model>、中间件逐路由注册禁 Group(prefix,mw...)、handler 仅返回 (rsp,error)、认证走 header。设计 HTTP API、写路由/handler/中间件时触发。
development
Go 项目结构规范——三层架构(API → Impl → State)、全局状态模式、internal/ 私有包、cmd/ 仅 main.go、go.work 多模块、禁止 Repository 接口和 DI 容器、struct 公共字段开头全 omitempty、handler var rsp 顶声明、禁 legacy migration。设计项目骨架、新建目录、组织包、做架构评审时触发。
development
Go 命名规范——Id/Uid 字段(非 ID)、IsActive/HasMFA 布尔前缀、CreatedAt 时间字段、接收者统一用 p、包名全小写无下划线、泛型类型参数描述性命名、集合字段 xxx_list 禁 xxxs 复数、Enum 0 值 XxxNil 禁 Unknown、禁 Status 统一 State、Set/Update 语义区分。定义结构体字段、函数、变量、包、接收者名、泛型、枚举时触发。