29 KiB
Semantic Commit
큰 변경사항을 의미 있는 최소 단위로 분할하여, 시맨틱한 커밋 메시지와 함께 순차적으로 커밋합니다. 외부 도구에 의존하지 않고 git 표준 명령어만 사용합니다.
사용법
/semantic-commit [옵션]
옵션
--dry-run: 실제 커밋은 하지 않고, 제안되는 커밋 분할만 표시--lang <언어>: 커밋 메시지 언어를 강제 지정(en, ko)--max-commits <수>: 최대 커밋 수 지정(기본값: 10)
기본 예시
# 현재 변경사항을 분석해서 논리적 단위로 커밋
/semantic-commit
# 분할안만 확인 (실제 커밋 없음)
/semantic-commit --dry-run
# 영어로 커밋 메시지 생성
/semantic-commit --lang en
# 한국어로 커밋 메시지 생성
/semantic-commit --lang ko
# 최대 5 개 커밋으로 분할
/semantic-commit --max-commits 5
동작 플로
- 변경 분석:
git diff HEAD로 전체 변경사항 취득 - 파일 분류: 변경된 파일을 논리적으로 그룹화
- 커밋 제안: 각 그룹에 대해 시맨틱한 커밋 메시지 생성
- 순차 실행: 사용자 확인 후 각 그룹을 순차적으로 커밋
변경 분할의 핵심 기능
"큰 변경사항" 탐지
다음 조건으로 큰 변경사항으로 탐지:
- 변경 파일 수: 5 개 이상의 파일 변경
- 변경 라인 수: 100 라인 이상의 변경
- 복수 기능: 2 개 이상의 기능 영역에 걸친 변경
- 혼재 패턴: feat + fix + docs 가 혼재
# 변경 규모 분석
CHANGED_FILES=$(git diff HEAD --name-only | wc -l)
CHANGED_LINES=$(git diff HEAD --stat | tail -1 | grep -o '[0-9]\+ insertions\|[0-9]\+ deletions' | awk '{sum+=$1} END {print sum}')
if [ $CHANGED_FILES -ge 5 ] || [ $CHANGED_LINES -ge 100 ]; then
echo "큰 변경사항 탐지: 분할을 권장"
fi
"의미 있는 최소 단위"로의 분할 전략
1. 기능 경계를 통한 분할
# 디렉터리 구조에서 기능 단위 특정
git diff HEAD --name-only | cut -d'/' -f1-2 | sort | uniq
# → src/auth, src/api, components/ui 등
2. 변경 종별을 통한 분리
# 신규 파일 vs 기존 파일 수정
git diff HEAD --name-status | grep '^A' # 신규 파일
git diff HEAD --name-status | grep '^M' # 수정 파일
git diff HEAD --name-status | grep '^D' # 삭제 파일
3. 의존 관계 분석
# 임포트 관계 변경 탐지
git diff HEAD | grep -E '^[+-].*import|^[+-].*require' | \
cut -d' ' -f2- | sort | uniq
파일 단위의 상세 분석
# 변경된 파일 목록 취득
git diff HEAD --name-only
# 각 파일의 변경 내용을 개별 분석
git diff HEAD -- <file>
# 파일의 변경 타입 판정
git diff HEAD --name-status | while read status file; do
case $status in
A) echo "$file: 신규 생성" ;;
M) echo "$file: 수정" ;;
D) echo "$file: 삭제" ;;
R*) echo "$file: 이름 변경" ;;
esac
done
논리적 그룹화 기준
-
기능 단위: 동일 기능 관련 파일
src/auth/하위 파일 → 인증 기능components/하위 파일 → UI 컴포넌트
-
변경 종별: 같은 종류의 변경
- 테스트 파일만 →
test: - 문서만 →
docs: - 설정 파일만 →
chore:
- 테스트 파일만 →
-
의존 관계: 상호 관련 파일
- 모델 + 마이그레이션
- 컴포넌트 + 스타일
-
변경 규모: 적절한 커밋 크기 유지
- 1 커밋당 10 파일 이하
- 관련성 높은 파일 그룹화
출력 예시
$ /semantic-commit
변경 분석 중...
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
탐지된 변경사항:
• src/auth/login.ts (수정)
• src/auth/register.ts (신규)
• src/auth/types.ts (수정)
• tests/auth.test.ts (신규)
• docs/authentication.md (신규)
제안되는 커밋 분할:
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
커밋 1/3
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
메시지: feat: implement user registration and login system
포함되는 파일:
• src/auth/login.ts
• src/auth/register.ts
• src/auth/types.ts
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
커밋 2/3
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
메시지: test: add comprehensive tests for authentication system
포함되는 파일:
• tests/auth.test.ts
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
커밋 3/3
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
메시지: docs: add authentication system documentation
포함되는 파일:
• docs/authentication.md
이 분할안으로 커밋을 실행하시겠습니까? (y/n/edit):
실행 시 선택지
y: 제안된 커밋 분할로 실행n: 취소edit: 커밋 메시지를 개별 편집merge <번호 1> <번호 2>: 지정한 커밋을 합치기split <번호>: 지정한 커밋을 더 분할
Dry Run 모드
$ /semantic-commit --dry-run
변경 분석 중... (DRY RUN)
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
[커밋 분할 제안 표시]
ℹ️ DRY RUN 모드: 실제 커밋은 실행되지 않습니다
💡 실행하려면 --dry-run 옵션을 제거하고 다시 실행하세요
스마트 분석 기능
1. 프로젝트 구조 이해
package.json,Cargo.toml,pom.xml등에서 프로젝트 종류 판정- 폴더 구조에서 기능 단위 추측
2. 변경 패턴 인식
# 버그 수정 패턴 탐지
- "fix", "bug", "error" 등의 키워드
- 예외 처리 추가
- 조건 분기 수정
# 새 기능 패턴 탐지
- 새 파일 생성
- 새 메서드 추가
- API 엔드포인트 추가
3. 의존 관계 분석
- import 문의 변경
- 타입 정의 추가/수정
- 설정 파일과의 연관성
기술적 구현
Git 표준 명령어를 통한 순차 커밋 구현
1. 전처리: 현재 상태 저장
# 스테이징되지 않은 변경사항이 있으면 일단 리셋
git reset HEAD
git status --porcelain > /tmp/original_state.txt
# 작업 브랜치 확인
CURRENT_BRANCH=$(git branch --show-current)
echo "작업 중인 브랜치: $CURRENT_BRANCH"
2. 그룹별 순차 커밋 실행
# 분할 계획 읽기
while IFS= read -r commit_plan; do
group_num=$(echo "$commit_plan" | cut -d':' -f1)
files=$(echo "$commit_plan" | cut -d':' -f2- | tr ' ' '\n')
echo "=== 커밋 $group_num 실행 ==="
# 해당 파일만 스테이징
echo "$files" | while read file; do
if [ -f "$file" ]; then
git add "$file"
echo "스테이징: $file"
fi
done
# 스테이징 상태 확인
staged_files=$(git diff --staged --name-only)
if [ -z "$staged_files" ]; then
echo "경고: 스테이징된 파일이 없습니다"
continue
fi
# 커밋 메시지 생성 (LLM 분석)
commit_msg=$(generate_commit_message_for_staged_files)
# 사용자 확인
echo "제안 커밋 메시지: $commit_msg"
echo "스테이징된 파일:"
echo "$staged_files"
read -p "이 커밋을 실행하시겠습니까? (y/n): " confirm
if [ "$confirm" = "y" ]; then
# 커밋 실행
git commit -m "$commit_msg"
echo "✅ 커밋 $group_num 완료"
else
# 스테이징 취소
git reset HEAD
echo "❌ 커밋 $group_num 건너뛰기"
fi
done < /tmp/commit_plan.txt
3. 에러 처리와 롤백
# 프리커밋 훅 실패 시 처리
commit_with_retry() {
local commit_msg="$1"
local max_retries=2
local retry_count=0
while [ $retry_count -lt $max_retries ]; do
if git commit -m "$commit_msg"; then
echo "✅ 커밋 성공"
return 0
else
echo "❌ 커밋 실패 (시도 $((retry_count + 1))/$max_retries)"
# 프리커밋 훅의 자동 수정 적용
if git diff --staged --quiet; then
echo "프리커밋 훅이 변경사항을 자동 수정했습니다"
git add -u
fi
retry_count=$((retry_count + 1))
fi
done
echo "❌ 커밋에 실패했습니다. 수동으로 확인하세요."
return 1
}
# 중단에서 복구
resume_from_failure() {
echo "중단된 커밋 처리를 감지했습니다"
echo "현재 스테이징 상태:"
git status --porcelain
read -p "처리를 계속하시겠습니까? (y/n): " resume
if [ "$resume" = "y" ]; then
# 마지막 커밋 위치에서 재개
last_commit=$(git log --oneline -1 --pretty=format:"%s")
echo "마지막 커밋: $last_commit"
else
# 완전 리셋
git reset HEAD
echo "처리를 리셋했습니다"
fi
}
4. 완료 후 검증
# 모든 변경사항이 커밋되었는지 확인
remaining_changes=$(git status --porcelain | wc -l)
if [ $remaining_changes -eq 0 ]; then
echo "✅ 모든 변경사항이 커밋되었습니다"
else
echo "⚠️ 커밋되지 않은 변경사항이 남아있습니다:"
git status --short
fi
# 커밋 이력 표시
echo "생성된 커밋:"
git log --oneline -n 10 --graph
5. 자동 푸시 억제
# 주의: 자동 푸시는 하지 않음
echo "📝 주의: 자동 푸시는 실행되지 않습니다"
echo "필요시 다음 명령어로 푸시하세요:"
echo " git push origin $CURRENT_BRANCH"
분할 알고리즘 상세
단계 1: 초기 분석
# 모든 변경 파일 취득 및 분류
git diff HEAD --name-status | while read status file; do
echo "$status:$file"
done > /tmp/changes.txt
# 기능 디렉터리별 변경 통계
git diff HEAD --name-only | cut -d'/' -f1-2 | sort | uniq -c
단계 2: 기능 경계를 통한 초기 그룹화
# 디렉터리 기반 그룹화
GROUPS=$(git diff HEAD --name-only | cut -d'/' -f1-2 | sort | uniq)
for group in $GROUPS; do
echo "=== 그룹: $group ==="
git diff HEAD --name-only | grep "^$group" | head -10
done
단계 3: 변경 내용의 유사성 분석
# 각 파일의 변경 타입 분석
git diff HEAD --name-only | while read file; do
# 새 함수/클래스 추가 탐지
NEW_FUNCTIONS=$(git diff HEAD -- "$file" | grep -c '^+.*function\|^+.*class\|^+.*def ')
# 버그 수정 패턴 탐지
BUG_FIXES=$(git diff HEAD -- "$file" | grep -c '^+.*fix\|^+.*bug\|^-.*error')
# 테스트 파일 판정
if [[ "$file" =~ test|spec ]]; then
echo "$file: TEST"
elif [ $NEW_FUNCTIONS -gt 0 ]; then
echo "$file: FEAT"
elif [ $BUG_FIXES -gt 0 ]; then
echo "$file: FIX"
else
echo "$file: REFACTOR"
fi
done
단계 4: 의존 관계를 통한 조정
# import 관계 분석
git diff HEAD | grep -E '^[+-].*import|^[+-].*from.*import' | \
while read line; do
echo "$line" | sed 's/^[+-]//' | awk '{print $2}'
done | sort | uniq > /tmp/imports.txt
# 관련 파일 그룹화
git diff HEAD --name-only | while read file; do
basename=$(basename "$file" .js .ts .py)
related=$(git diff HEAD --name-only | grep "$basename" | grep -v "^$file$")
if [ -n "$related" ]; then
echo "관련 파일 그룹: $file <-> $related"
fi
done
단계 5: 커밋 크기 최적화
# 그룹 크기 조정
MAX_FILES_PER_COMMIT=8
current_group=1
file_count=0
git diff HEAD --name-only | while read file; do
if [ $file_count -ge $MAX_FILES_PER_COMMIT ]; then
current_group=$((current_group + 1))
file_count=0
fi
echo "커밋 $current_group: $file"
file_count=$((file_count + 1))
done
단계 6: 최종 그룹 결정
# 분할 결과 검증
for group in $(seq 1 $current_group); do
files=$(grep "커밋 $group:" /tmp/commit_plan.txt | cut -d':' -f2-)
lines=$(echo "$files" | xargs git diff HEAD -- | wc -l)
echo "커밋 $group: $(echo "$files" | wc -w) 파일, $lines 라인 변경"
done
Conventional Commits 준수
기본 형식
<type>[optional scope]: <description>
[optional body]
[optional footer(s)]
표준 타입
필수 타입:
feat: 새 기능 (사용자에게 보이는 기능 추가)fix: 버그 수정
선택적 타입:
build: 빌드 시스템이나 외부 의존관계 변경chore: 기타 변경사항 (릴리즈에 영향하지 않음)ci: CI 설정 파일이나 스크립트 변경docs: 문서만의 변경style: 코드 의미에 영향하지 않는 변경 (공백, 포맷, 세미콜론 등)refactor: 버그 수정이나 기능 추가를 동반하지 않는 코드 변경perf: 성능 개선test: 테스트 추가나 수정
스코프 (선택적)
변경의 영향 범위 표시:
feat(api): add user authentication endpoint
fix(ui): resolve button alignment issue
docs(readme): update installation instructions
Breaking Change
API 의 파괴적 변경이 있는 경우:
feat!: change user API response format
BREAKING CHANGE: user response now includes additional metadata
또는
feat(api)!: change authentication flow
프로젝트 규약의 자동 감지
중요: 프로젝트 고유 규약이 존재하면 그것을 우선합니다.
1. CommitLint 설정 확인
다음 파일에서 설정을 자동 감지:
commitlint.config.jscommitlint.config.mjscommitlint.config.cjscommitlint.config.ts.commitlintrc.js.commitlintrc.json.commitlintrc.yml.commitlintrc.yamlpackage.json의commitlint섹션
# 설정 파일 예시 확인
cat commitlint.config.mjs
cat .commitlintrc.json
grep -A 10 '"commitlint"' package.json
2. 커스텀 타입 감지
프로젝트 고유 타입 예시:
// commitlint.config.mjs
export default {
extends: ["@commitlint/config-conventional"],
rules: {
"type-enum": [
2,
"always",
[
"feat",
"fix",
"docs",
"style",
"refactor",
"test",
"chore",
"wip", // 작업 중
"hotfix", // 긴급 수정
"release", // 릴리즈
"deps", // 의존성 업데이트
"config", // 설정 변경
],
],
},
};
3. 언어 설정 감지
// 프로젝트가 한국어 메시지를 사용하는 경우
export default {
rules: {
"subject-case": [0], // 한국어 대응을 위해 비활성화
"subject-max-length": [2, "always", 72], // 한국어는 문자 수 제한 조정
},
};
자동 분석 흐름
-
설정 파일 검색
find . -name "commitlint.config.*" -o -name ".commitlintrc.*" | head -1 -
기존 커밋 분석
git log --oneline -50 --pretty=format:"%s" -
사용 타입 통계
git log --oneline -100 --pretty=format:"%s" | \ grep -oE '^[a-z]+(\([^)]+\))?' | \ sort | uniq -c | sort -nr
프로젝트 규약 예시
Angular 스타일
feat(scope): add new feature
fix(scope): fix bug
docs(scope): update documentation
Gitmoji 병용 스타일
✨ feat: add user registration
🐛 fix: resolve login issue
📚 docs: update API docs
한국어 프로젝트
feat: 사용자 등록 기능 추가
fix: 로그인 처리 버그 수정
docs: API 문서 업데이트
언어 판정
이 명령어에서 완결되는 언어 판정 로직:
-
CommitLint 설정에서 언어 설정 확인
# subject-case 규칙이 비활성화되면 한국어로 판정 grep -E '"subject-case".*\[0\]|subject-case.*0' commitlint.config.* -
git log 분석을 통한 자동 판정
# 최근 20 개 커밋의 언어 분석 git log --oneline -20 --pretty=format:"%s" | \ grep -E '^[가-힣]' | wc -l # 50% 이상이 한국어면 한국어 모드 -
프로젝트 파일의 언어 설정
# README.md 언어 확인 head -10 README.md | grep -E '[가-힣]' | wc -l # package.json 의 description 확인 grep -E '"description".*[가-힣]' package.json -
변경 파일 내 주석·문자열 분석
# 변경된 파일의 주석 언어 확인 git diff HEAD | grep -E '^[+-].*//.*[가-힣]' | wc -l
판정 알고리즘
# 언어 판정 점수 계산
KOREAN_SCORE=0
# 1. CommitLint 설정 (+3 점)
if grep -q '"subject-case".*\[0\]' commitlint.config.* 2>/dev/null; then
KOREAN_SCORE=$((KOREAN_SCORE + 3))
fi
# 2. git log 분석 (최대 +2 점)
KOREAN_COMMITS=$(git log --oneline -20 --pretty=format:"%s" | \
grep -cE '[가-힣]' 2>/dev/null || echo 0)
if [ $KOREAN_COMMITS -gt 10 ]; then
KOREAN_SCORE=$((KOREAN_SCORE + 2))
elif [ $KOREAN_COMMITS -gt 5 ]; then
KOREAN_SCORE=$((KOREAN_SCORE + 1))
fi
# 3. README.md 확인 (+1 점)
if head -5 README.md 2>/dev/null | grep -qE '[가-힣]'; then
KOREAN_SCORE=$((KOREAN_SCORE + 1))
fi
# 4. 변경 파일 내용 확인 (+1 점)
if git diff HEAD 2>/dev/null | grep -qE '^[+-].*[가-힣]'; then
KOREAN_SCORE=$((KOREAN_SCORE + 1))
fi
# 판정: 3 점 이상이면 한국어 모드
if [ $KOREAN_SCORE -ge 3 ]; then
LANGUAGE="ko"
else
LANGUAGE="en"
fi
설정 파일 자동 읽기
실행 시 동작
명령어 실행 시 다음 순서로 설정 확인:
-
CommitLint 설정 파일 검색
# 다음 순서로 검색하여 처음 찾은 파일 사용 commitlint.config.mjs commitlint.config.js commitlint.config.cjs commitlint.config.ts .commitlintrc.js .commitlintrc.json .commitlintrc.yml .commitlintrc.yaml package.json (commitlint 섹션) -
설정 내용 분석
- 사용 가능한 타입 목록 추출
- 스코프 제한 확인
- 메시지 길이 제한 취득
- 언어 설정 확인
-
기존 커밋 이력 분석
# 최근 커밋에서 사용 패턴 학습 git log --oneline -100 --pretty=format:"%s" | \ head -20
설정 예시 분석
표준 commitlint.config.mjs:
export default {
extends: ["@commitlint/config-conventional"],
rules: {
"type-enum": [
2,
"always",
["feat", "fix", "docs", "style", "refactor", "perf", "test", "chore"],
],
"scope-enum": [2, "always", ["api", "ui", "core", "auth", "db"]],
},
};
한국어 대응 설정:
export default {
extends: ["@commitlint/config-conventional"],
rules: {
"subject-case": [0], // 한국어를 위해 비활성화
"subject-max-length": [2, "always", 72],
"type-enum": [
2,
"always",
["feat", "fix", "docs", "style", "refactor", "test", "chore"],
],
},
};
커스텀 타입 포함 설정:
export default {
extends: ["@commitlint/config-conventional"],
rules: {
"type-enum": [
2,
"always",
[
"feat",
"fix",
"docs",
"style",
"refactor",
"test",
"chore",
"wip", // Work in Progress
"hotfix", // 긴급 수정
"release", // 릴리즈 준비
"deps", // 의존성 업데이트
"config", // 설정 변경
],
],
},
};
폴백 동작
설정 파일을 찾을 수 없는 경우:
-
git log 분석을 통한 자동 추측
# 최근 100 개 커밋에서 타입 추출 git log --oneline -100 --pretty=format:"%s" | \ grep -oE '^[a-z]+(\([^)]+\))?' | \ sort | uniq -c | sort -nr -
Conventional Commits 표준을 기본 사용
feat, fix, docs, style, refactor, perf, test, chore, build, ci -
언어 판정
- 한국어 커밋이 50% 이상 → 한국어 모드
- 그 외 → 영어 모드
전제 조건
- Git 저장소 내에서 실행
- 커밋되지 않은 변경사항이 존재할 것
- 스테이징된 변경사항은 일시적으로 리셋됩니다
주의사항
- 자동 푸시 없음: 커밋 후
git push는 수동 실행 - 브랜치 생성 없음: 현재 브랜치에서 커밋
- 백업 권장: 중요한 변경 전에는
git stash로 백업
프로젝트 규약의 우선순위
커밋 메시지 생성 시 우선순위:
-
CommitLint 설정 (최우선)
commitlint.config.*파일 설정- 커스텀 타입이나 스코프 제한
- 메시지 길이나 케이스 제한
-
기존 커밋 이력 (2 순위)
- 실제 사용된 타입 통계
- 메시지 언어(한국어/영어)
- 스코프 사용 패턴
-
프로젝트 종류 (3 순위)
package.json→ Node.js 프로젝트Cargo.toml→ Rust 프로젝트pom.xml→ Java 프로젝트
-
Conventional Commits 표준 (폴백)
- 설정을 찾을 수 없는 경우의 표준 동작
규약 감지 실례
Monorepo 에서 scope 자동 감지:
# packages/ 폴더에서 scope 추측
ls packages/ | head -10
# → api, ui, core, auth 등을 scope 로 제안
프레임워크별 규약:
// Angular 프로젝트
{
'scope-enum': [2, 'always', [
'animations', 'common', 'core', 'forms', 'http', 'platform-browser',
'platform-server', 'router', 'service-worker', 'upgrade'
]]
}
// React 프로젝트
{
'scope-enum': [2, 'always', [
'components', 'hooks', 'utils', 'types', 'styles', 'api'
]]
}
기업·팀 고유 규약:
// 한국 기업에서 자주 보이는 패턴
{
'type-enum': [2, 'always', [
'feat', 'fix', 'docs', 'style', 'refactor', 'test', 'chore',
'wip', // 작업 중(풀 리퀘스트용)
'hotfix', // 긴급 수정
'release' // 릴리즈 준비
]],
'subject-case': [0], // 한국어 대응
'subject-max-length': [2, 'always', 72] // 한국어는 길게 설정
}
베스트 프랙티스
- 프로젝트 규약 준수: 기존 설정이나 패턴을 따름
- 작은 변경 단위: 1 개 커밋은 1 개의 논리적 변경
- 명확한 메시지: 무엇을 변경했는지가 명확
- 관련성 중시: 기능적으로 관련된 파일을 그룹화
- 테스트 분리: 테스트 파일은 별도 커밋으로
- 설정 파일 활용: CommitLint 를 도입해 팀 전체 규약 통일
실제 분할 예시(Before/After)
예시 1: 대규모 인증 시스템 추가
Before(하나의 거대한 커밋):
# 변경된 파일(15 개 파일, 850 라인 변경)
src/auth/login.js # 신규 생성
src/auth/register.js # 신규 생성
src/auth/password.js # 신규 생성
src/auth/types.js # 신규 생성
src/api/auth-routes.js # 신규 생성
src/middleware/auth.js # 신규 생성
src/database/migrations/001_users.sql # 신규 생성
src/database/models/user.js # 신규 생성
tests/auth/login.test.js # 신규 생성
tests/auth/register.test.js # 신규 생성
tests/api/auth-routes.test.js # 신규 생성
docs/authentication.md # 신규 생성
package.json # 의존성 추가
README.md # 사용법 추가
.env.example # 환경변수 예시 추가
# 기존의 문제 있는 커밋
feat: implement complete user authentication system with login, registration, password reset, API routes, database models, tests and documentation
After(의미 있는 5 개 커밋으로 분할):
# 커밋 1: 데이터베이스 기반
feat(db): add user model and authentication schema
포함 파일:
- src/database/migrations/001_users.sql
- src/database/models/user.js
- src/auth/types.js
이유: 데이터베이스 구조는 다른 기능의 기반이므로 먼저 커밋
# 커밋 2: 인증 로직
feat(auth): implement core authentication functionality
포함 파일:
- src/auth/login.js
- src/auth/register.js
- src/auth/password.js
- src/middleware/auth.js
이유: 인증의 핵심 비즈니스 로직을 한 번에 커밋
# 커밋 3: API 엔드포인트
feat(api): add authentication API routes
포함 파일:
- src/api/auth-routes.js
이유: API 레이어는 인증 로직에 의존하므로 뒤에 커밋
# 커밋 4: 포괄적 테스트
test(auth): add comprehensive authentication tests
포함 파일:
- tests/auth/login.test.js
- tests/auth/register.test.js
- tests/api/auth-routes.test.js
이유: 구현 완료 후 테스트를 일괄 추가
# 커밋 5: 설정과 문서
docs(auth): add authentication documentation and configuration
포함 파일:
- docs/authentication.md
- package.json
- README.md
- .env.example
이유: 문서와 설정은 마지막에 함께 커밋
예시 2: 버그 수정과 리팩터링 혼재
Before(혼재된 문제 있는 커밋):
# 변경된 파일(8 개 파일, 320 라인 변경)
src/user/service.js # 버그 수정 + 리팩터링
src/user/validator.js # 신규 생성(리팩터링)
src/auth/middleware.js # 버그 수정
src/api/user-routes.js # 버그 수정 + 에러 처리 개선
tests/user.test.js # 테스트 추가
tests/auth.test.js # 버그 수정 테스트 추가
docs/user-api.md # 문서 업데이트
package.json # 의존성 업데이트
# 문제 있는 커밋
fix: resolve user validation bugs and refactor validation logic with improved error handling
After(종류별로 3 개 커밋으로 분할):
# 커밋 1: 긴급 버그 수정
fix: resolve user validation and authentication bugs
포함 파일:
- src/user/service.js(버그 수정 부분만)
- src/auth/middleware.js
- tests/auth.test.js(버그 수정 테스트만)
이유: 프로덕션 환경에 영향을 주는 버그는 최우선 수정
# 커밋 2: 검증 로직 리팩터링
refactor: extract and improve user validation logic
포함 파일:
- src/user/service.js(리팩터링 부분)
- src/user/validator.js
- src/api/user-routes.js
- tests/user.test.js
이유: 구조 개선은 기능 단위로 함께 커밋
# 커밋 3: 문서와 의존성 업데이트
chore: update documentation and dependencies
포함 파일:
- docs/user-api.md
- package.json
이유: 개발 환경 정비는 마지막에 함께 커밋
예시 3: 여러 기능 동시 개발
Before(기능 횡단 거대 커밋):
# 변경된 파일(12 개 파일, 600 라인 변경)
src/user/profile.js # 새 기능 A
src/user/avatar.js # 새 기능 A
src/notification/email.js # 새 기능 B
src/notification/sms.js # 새 기능 B
src/api/profile-routes.js # 새 기능 A 용 API
src/api/notification-routes.js # 새 기능 B 용 API
src/dashboard/widgets.js # 새 기능 C
src/dashboard/charts.js # 새 기능 C
tests/profile.test.js # 새 기능 A 용 테스트
tests/notification.test.js # 새 기능 B 용 테스트
tests/dashboard.test.js # 새 기능 C 용 테스트
package.json # 전체 기능 의존성
# 문제 있는 커밋
feat: add user profile management, notification system and dashboard widgets
After(기능별로 4 개 커밋으로 분할):
# 커밋 1: 사용자 프로필 기능
feat(profile): add user profile management
포함 파일:
- src/user/profile.js
- src/user/avatar.js
- src/api/profile-routes.js
- tests/profile.test.js
이유: 프로필 기능은 독립된 기능 단위
# 커밋 2: 알림 시스템
feat(notification): implement email and SMS notifications
포함 파일:
- src/notification/email.js
- src/notification/sms.js
- src/api/notification-routes.js
- tests/notification.test.js
이유: 알림 기능은 독립된 기능 단위
# 커밋 3: 대시보드 위젯
feat(dashboard): add interactive widgets and charts
포함 파일:
- src/dashboard/widgets.js
- src/dashboard/charts.js
- tests/dashboard.test.js
이유: 대시보드 기능은 독립된 기능 단위
# 커밋 4: 의존성과 인프라 업데이트
chore: update dependencies for new features
포함 파일:
- package.json
이유: 공통 의존성 업데이트는 마지막에 함께
분할 효과 비교
| 항목 | Before(거대 커밋) | After(적절한 분할) |
|---|---|---|
| 리뷰성 | ❌ 매우 어려움 | ✅ 각 커밋이 작아 리뷰 가능 |
| 버그 추적 | ❌ 문제 위치 파악 어려움 | ✅ 문제 있는 커밋 즉시 파악 |
| 되돌리기 | ❌ 전체를 되돌려야 함 | ✅ 문제 부분만 선택적 되돌리기 |
| 병렬 개발 | ❌ 충돌 발생 쉬움 | ✅ 기능별로 병합 용이 |
| 배포 | ❌ 전체 기능 일괄 배포 | ✅ 단계적 배포 가능 |
트러블슈팅
커밋 실패 시
- 프리커밋 훅 확인
- 의존성 해결
- 개별 파일로 재시도
분할이 적절하지 않은 경우
--max-commits옵션으로 조정- 수동
edit모드 사용 - 더 세밀한 단위로 재실행