AgentSkillsCN

atomic-git-commits

编写原子化、可人工审核的Git提交——每次提交只包含一项逻辑变更,并附上清晰的常规提交信息

SKILL.md
--- frontmatter
name: atomic-git-commits
description: "Write atomic, human-reviewable git commits — one logical change per commit with clear conventional messages"
version: 1.0.4

Atomic Git Commits

Write atomic, human-reviewable git commits. Each commit should represent one logical change that can be understood, reviewed, and reverted independently.

What Makes a Commit Atomic

An atomic commit:

  • Contains one logical change — not "fix bug and also refactor and update deps"
  • Builds successfully at every point in history — no broken intermediate states
  • Can be reverted independently without side effects on unrelated code
  • Has a clear, descriptive message that explains the why, not just the what

Commit Message Format

Use Conventional Commits with scope:

code
<type>(<scope>): <subject>

<body — optional, explains WHY>

Co-Authored-By: Name <email>

Types

TypeWhen to Use
featNew feature or capability
fixBug fix
refactorCode restructuring without behavior change
docsDocumentation only
testAdding or updating tests
choreBuild, CI, dependency updates
perfPerformance improvement

Subject Line Rules

  • Imperative mood: "add feature" not "added feature" or "adds feature"
  • Under 70 characters — details go in the body
  • Lowercase first word after the colon
  • No trailing period

Examples

code
feat(web): add session timeout with configurable TTL

refactor(web): replace stateless logger with wide event Logger class

fix(cli): handle 429 rate limit with exponential backoff

docs: add server functions middleware documentation

Splitting Work into Atomic Commits

Strategy: Layer by Dependency

When refactoring, commit in dependency order so each commit compiles:

code
1. feat(shared): add new validation types        ← foundation
2. refactor(web): update middleware to use types  ← consumer
3. feat(web): enrich auth middleware with logger  ← integration

Strategy: Separate Mechanism from Policy

code
1. refactor(web): extract Logger class            ← mechanism (the tool)
2. refactor(web): wire Logger into middleware      ← policy (how it's used)
3. feat(web): add user identity to wide events    ← enrichment

Strategy: Infrastructure then Features

code
1. chore(web): add rate limiting middleware       ← infrastructure
2. feat(api): apply rate limits to publish route  ← feature using it

What NOT to Do

Don't mix concerns

code
# Bad — two unrelated changes in one commit
git commit -m "fix login bug and update README"

# Good — separate commits
git commit -m "fix(auth): validate email before session create"
git commit -m "docs: update README with auth setup instructions"

Don't commit broken intermediate states

code
# Bad — breaks the build between commits
commit 1: "remove old auth module"       ← build breaks here
commit 2: "add new auth module"          ← build works again

# Good — atomic swap
commit 1: "refactor(auth): replace old auth with better-auth"

Don't commit generated files with source changes

code
# Bad — noise in the diff
commit: "feat(web): add dashboard route"
  + src/routes/dashboard.tsx          ← your code
  + src/routeTree.gen.ts              ← auto-generated noise

# Good — separate or .gitignore generated files
commit: "feat(web): add dashboard route"
  + src/routes/dashboard.tsx

Staging Specific Files

Always stage explicitly — avoid git add -A which can catch secrets or generated files:

bash
# Stage specific files for this logical change
git add src/lib/logger.ts src/lib/middleware/types.ts

# Commit with heredoc for multi-line messages
git commit -m "$(cat <<'EOF'
refactor(web): replace stateless logger with wide event Logger class

Introduce a Logger class that accumulates context throughout a request
lifecycle and flushes a single canonical log line.

Co-Authored-By: Name <email>
EOF
)"

Review Checklist

Before each commit, verify:

  • git diff --staged shows only changes related to this commit's purpose
  • The commit message accurately describes the change
  • The project builds after this commit
  • No unrelated formatting, whitespace, or import changes snuck in
  • No secrets, .env files, or large binaries are staged
  • The commit can be understood in isolation during code review

Interactive Rebase for Cleanup

If you made messy WIP commits, clean up before pushing:

bash
# Squash/reorder last N commits
git rebase -i HEAD~N

# In the editor:
pick abc1234 feat(web): add Logger class
squash def5678 wip: fix typo in Logger     ← squash into previous
pick ghi9012 refactor(web): wire Logger into middleware
drop jkl3456 debug: add console.log        ← drop entirely

Important: Only rebase commits that haven't been pushed. Never rebase shared history.

Practical Example: Refactoring a Logger

Given a task to refactor a stateless logger into a wide event Logger class that touches 5 files, split into 3 atomic commits:

CommitFilesPurpose
1. refactor: replace logger with Logger classlogger.ts, types.tsCore abstraction + type update
2. refactor: simplify logging middlewarewith-logging.tsMain consumer of new Logger
3. feat: enrich wide events with user identityauth.ts, index.tsAuth integration + re-export

Each commit:

  • Compiles independently
  • Contains one logical change
  • Can be reviewed in isolation
  • Tells a clear story when read in sequence