From 212895f6b45f3dd618da8fc2f8961761f55bc469 Mon Sep 17 00:00:00 2001 From: htlee Date: Sun, 1 Mar 2026 17:27:02 +0900 Subject: [PATCH 01/11] =?UTF-8?q?docs:=20sync-team-workflow=EC=97=90=20env?= =?UTF-8?q?=20=EB=8F=99=EA=B8=B0=ED=99=94=20+=20v1.5.0=20=EC=8A=A4?= =?UTF-8?q?=ED=82=AC/=EC=97=90=EC=9D=B4=EC=A0=84=ED=8A=B8=20=EB=AA=A9?= =?UTF-8?q?=EB=A1=9D=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .claude/skills/sync-team-workflow/SKILL.md | 14 ++++++++++++++ 1 file changed, 14 insertions(+) diff --git a/.claude/skills/sync-team-workflow/SKILL.md b/.claude/skills/sync-team-workflow/SKILL.md index 2d51512..5c07617 100644 --- a/.claude/skills/sync-team-workflow/SKILL.md +++ b/.claude/skills/sync-team-workflow/SKILL.md @@ -44,12 +44,22 @@ curl -sf "${GITEA_URL}/gc/template-react-ts/raw/branch/develop/.editorconfig" ``` .claude/rules/team-policy.md .claude/rules/git-workflow.md +.claude/rules/release-notes-guide.md +.claude/rules/subagent-policy.md .claude/rules/code-style.md (타입별) .claude/rules/naming.md (타입별) .claude/rules/testing.md (타입별) ``` +#### 4-1b. 에이전트 파일 (덮어쓰기) +``` +.claude/agents/explorer.md +.claude/agents/implementer.md +.claude/agents/reviewer.md +``` + #### 4-2. settings.json (부분 갱신) +- `env`: 글로벌 최신으로 교체 (CLAUDE_BOT_TOKEN 등 팀 공통 환경변수) - `deny` 목록: 글로벌 최신으로 교체 - `allow` 목록: 기존 사용자 커스텀 유지 + 글로벌 기본값 병합 - `hooks`: init-project SKILL.md의 hooks JSON 블록을 참조하여 교체 (없으면 추가) @@ -63,6 +73,10 @@ curl -sf "${GITEA_URL}/gc/template-react-ts/raw/branch/develop/.editorconfig" .claude/skills/fix-issue/SKILL.md .claude/skills/sync-team-workflow/SKILL.md .claude/skills/init-project/SKILL.md +.claude/skills/push/SKILL.md +.claude/skills/mr/SKILL.md +.claude/skills/release/SKILL.md +.claude/skills/version/SKILL.md ``` #### 4-4. Git Hooks (덮어쓰기 + 실행 권한) -- 2.45.2 From ee54083dcf6f73c5ed9ae2ddc5a98245961e63f7 Mon Sep 17 00:00:00 2001 From: htlee Date: Sun, 1 Mar 2026 17:27:13 +0900 Subject: [PATCH 02/11] =?UTF-8?q?docs:=20init-project=EC=97=90=20env(CLAUD?= =?UTF-8?q?E=5FBOT=5FTOKEN)=20=EC=95=88=EB=82=B4=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .claude/skills/init-project/SKILL.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.claude/skills/init-project/SKILL.md b/.claude/skills/init-project/SKILL.md index eb8d47f..ec3da04 100644 --- a/.claude/skills/init-project/SKILL.md +++ b/.claude/skills/init-project/SKILL.md @@ -45,7 +45,7 @@ curl -sf "${GITEA_URL}/gc/template-react-ts/raw/branch/develop/.editorconfig" ### 3. .claude/ 디렉토리 구성 이미 팀 표준 파일이 존재하면 건너뜀. 없는 경우 위의 URL 패턴으로 Gitea에서 다운로드: -- `.claude/settings.json` — 프로젝트 타입별 표준 권한 설정 + hooks 섹션 (4단계 참조) +- `.claude/settings.json` — 프로젝트 타입별 표준 권한 설정 + env(CLAUDE_BOT_TOKEN 등) + hooks 섹션 (4단계 참조) - `.claude/rules/` — 팀 규칙 파일 (team-policy, git-workflow, code-style, naming, testing) - `.claude/skills/` — 팀 스킬 (create-mr, fix-issue, sync-team-workflow, init-project) -- 2.45.2 From b766e9d47be912e28ceb118bda8b6e87aa62c4d5 Mon Sep 17 00:00:00 2001 From: htlee Date: Sun, 1 Mar 2026 18:04:48 +0900 Subject: [PATCH 03/11] =?UTF-8?q?feat:=20/mr=20=EC=8A=A4=ED=82=AC=EC=97=90?= =?UTF-8?q?=20=EB=B4=87=20=EC=8A=B9=EC=9D=B8+=EB=A8=B8=EC=A7=80=20?= =?UTF-8?q?=EC=98=B5=EC=85=98=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .claude/skills/mr/SKILL.md | 47 +++++++++++++++++++++++++++++++++++++- 1 file changed, 46 insertions(+), 1 deletion(-) diff --git a/.claude/skills/mr/SKILL.md b/.claude/skills/mr/SKILL.md index 7b3b317..9ac5448 100644 --- a/.claude/skills/mr/SKILL.md +++ b/.claude/skills/mr/SKILL.md @@ -174,8 +174,52 @@ curl -X POST "https://{host}/api/v1/repos/{owner}/{repo}/pulls" \ }' ``` -### 7. 결과 출력 +### 7. PR 승인 + 머지 (선택) +**사용자 확인** (AskUserQuestion): +- **질문**: "MR을 어떻게 처리하시겠습니까?" +- 옵션 1: 리뷰 대기 (추천) — MR만 생성, 리뷰어 지정 후 수동 머지 +- 옵션 2: 승인 + 머지 — claude-bot으로 즉시 승인하고 머지 + +**승인 + 머지 선택 시**: + +```bash +# CLAUDE_BOT_TOKEN 확인 +if [ -z "$CLAUDE_BOT_TOKEN" ]; then + echo "CLAUDE_BOT_TOKEN이 설정되지 않아 자동 승인이 불가합니다. MR만 생성합니다." + # MR URL 출력 후 종료 +fi + +# 1. claude-bot이 PR 리뷰 승인 +curl -X POST "https://{host}/api/v1/repos/{owner}/{repo}/pulls/{pr_number}/reviews" \ + -H "Authorization: token ${CLAUDE_BOT_TOKEN}" \ + -H "Content-Type: application/json" \ + -d '{"event": "APPROVED", "body": "MR 승인 (via /mr skill)"}' + +# 2. 머지 실행 +curl -X POST "https://{host}/api/v1/repos/{owner}/{repo}/pulls/{pr_number}/merge" \ + -H "Authorization: token ${GITEA_TOKEN}" \ + -H "Content-Type: application/json" \ + -d '{"Do": "merge", "merge_message_field": "MR 제목", "delete_branch_after_merge": true}' +``` + +⚠️ `delete_branch_after_merge: true` — feature 브랜치는 머지 후 삭제한다 (develop/main과 달리). + +### 8. 결과 출력 + +**승인 + 머지한 경우**: +``` +✅ MR 머지 완료 + 브랜치: feature/my-branch → develop + MR: https://gitea.gc-si.dev/gc/my-project/pulls/42 + 커밋: 3건, 파일: 5개 변경 + 릴리즈 노트: docs/RELEASE-NOTES.md [Unreleased] 갱신됨 + PR 승인: claude-bot ✅ + 머지: 완료 ✅ + 브랜치 삭제: feature/my-branch ✅ +``` + +**리뷰 대기 선택 시**: ``` ✅ MR 생성 완료 브랜치: feature/my-branch → develop @@ -190,6 +234,7 @@ curl -X POST "https://{host}/api/v1/repos/{owner}/{repo}/pulls" \ - `GITEA_TOKEN`: Gitea API 접근 토큰 - 없으면: "Gitea 토큰이 필요합니다. Settings → Applications에서 생성하세요" 안내 +- `CLAUDE_BOT_TOKEN`: claude-bot PR 승인용 토큰 (선택, 없으면 자동 승인 건너뜀) ## 기존 /create-mr과의 차이 -- 2.45.2 From fdef842a14ad8a8048ebcf4e9b9fece5657f805b Mon Sep 17 00:00:00 2001 From: htlee Date: Sun, 1 Mar 2026 18:05:07 +0900 Subject: [PATCH 04/11] =?UTF-8?q?docs:=20workflow-version.json=20/mr=20?= =?UTF-8?q?=EB=B4=87=20=EC=8A=B9=EC=9D=B8=20=EC=98=B5=EC=85=98=20=EB=B0=98?= =?UTF-8?q?=EC=98=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- workflow-version.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/workflow-version.json b/workflow-version.json index 56a8f81..fba9683 100644 --- a/workflow-version.json +++ b/workflow-version.json @@ -9,7 +9,7 @@ "date": "2026-03-01", "description": "2계층 릴리즈 노트 + 권한 기반 스킬 체계 + Claude 봇 PR 승인", "items": [ - "/mr: 권한 확인 + docs/RELEASE-NOTES.md [Unreleased] 자동 갱신 + 초안 첨삭", + "/mr: 권한 확인 + docs/RELEASE-NOTES.md [Unreleased] 자동 갱신 + 초안 첨삭 + 봇 승인/머지 옵션", "/release: 권한 확인(admin) + 날짜 압축 + 봇 PR 승인/머지 + 미기록 커밋 감지", "/version: RELEASE-NOTES.md 기반 docs/VERSION-HISTORY.md 생성 (신규 스킬)", "/push: Gitea API 권한 확인 추가", -- 2.45.2 From ca39b31b8ca111f55b9d7cde62dc70bbe71292c7 Mon Sep 17 00:00:00 2001 From: htlee Date: Sun, 1 Mar 2026 18:27:13 +0900 Subject: [PATCH 05/11] =?UTF-8?q?fix:=20sync-team-workflow=20settings.json?= =?UTF-8?q?=20=EC=86=8C=EC=8A=A4=20=EA=B2=BD=EB=A1=9C=20=EB=AA=85=EC=8B=9C?= =?UTF-8?q?=20+=20custom=5Fpre=5Fcommit=20=ED=94=8C=EB=9E=98=EA=B7=B8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .claude/skills/sync-team-workflow/SKILL.md | 14 ++++++++++++++ 1 file changed, 14 insertions(+) diff --git a/.claude/skills/sync-team-workflow/SKILL.md b/.claude/skills/sync-team-workflow/SKILL.md index 5c07617..536ad71 100644 --- a/.claude/skills/sync-team-workflow/SKILL.md +++ b/.claude/skills/sync-team-workflow/SKILL.md @@ -59,6 +59,13 @@ curl -sf "${GITEA_URL}/gc/template-react-ts/raw/branch/develop/.editorconfig" ``` #### 4-2. settings.json (부분 갱신) + +⚠️ settings.json은 **타입별 템플릿**에서 다운로드 (template-common에는 없음): +```bash +curl -sf "${GITEA_URL}/gc/template-${PROJECT_TYPE}/raw/branch/develop/.claude/settings.json" +``` + +다운로드한 최신 settings.json과 로컬 settings.json을 비교하여 부분 갱신: - `env`: 글로벌 최신으로 교체 (CLAUDE_BOT_TOKEN 등 팀 공통 환경변수) - `deny` 목록: 글로벌 최신으로 교체 - `allow` 목록: 기존 사용자 커스텀 유지 + 글로벌 기본값 병합 @@ -80,6 +87,13 @@ curl -sf "${GITEA_URL}/gc/template-react-ts/raw/branch/develop/.editorconfig" ``` #### 4-4. Git Hooks (덮어쓰기 + 실행 권한) + +`commit-msg`, `post-checkout`은 **항상 팀 표준으로 교체** (팀 커뮤니케이션 규칙 + 인프라). + +`pre-commit`은 `.claude/workflow-version.json`의 `custom_pre_commit` 플래그를 확인: +- `"custom_pre_commit": true` → pre-commit 건너뜀 (프로젝트 커스텀 유지), "⚠️ pre-commit은 프로젝트 커스텀 유지" 로그 +- 플래그 없거나 false → 팀 표준으로 교체 + ```bash chmod +x .githooks/* ``` -- 2.45.2 From a138b97c325ef67e46e42ae9a6fdd883f6c8c2a4 Mon Sep 17 00:00:00 2001 From: htlee Date: Sun, 1 Mar 2026 18:27:16 +0900 Subject: [PATCH 06/11] =?UTF-8?q?feat:=20init-project=20pre-commit=20?= =?UTF-8?q?=ED=9B=85=20=EA=B2=80=EC=A6=9D=20+=20custom=5Fpre=5Fcommit=20?= =?UTF-8?q?=EC=A7=80=EC=9B=90?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .claude/skills/init-project/SKILL.md | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/.claude/skills/init-project/SKILL.md b/.claude/skills/init-project/SKILL.md index ec3da04..a7bd3cc 100644 --- a/.claude/skills/init-project/SKILL.md +++ b/.claude/skills/init-project/SKILL.md @@ -165,6 +165,13 @@ git config core.hooksPath .githooks chmod +x .githooks/* ``` +**pre-commit 훅 검증**: `.githooks/pre-commit`을 실행하여 빌드 검증이 정상 동작하는지 확인. +에러 발생 시 (예: 모노레포가 아닌 특수 구조, 빌드 명령 불일치 등): +1. 프로젝트에 맞게 `.githooks/pre-commit`을 커스텀 수정 +2. `.claude/workflow-version.json`에 `"custom_pre_commit": true` 추가 +3. 이후 `/sync-team-workflow` 실행 시 pre-commit은 덮어쓰지 않고 보존됨 + (`commit-msg`, `post-checkout`은 항상 팀 표준으로 동기화) + ### 6. 프로젝트 타입별 추가 설정 #### java-maven -- 2.45.2 From bdd02e689d19183c2fc1aaa107bd402c08cd219c Mon Sep 17 00:00:00 2001 From: htlee Date: Fri, 6 Mar 2026 14:18:43 +0900 Subject: [PATCH 07/11] =?UTF-8?q?feat:=20v1.6.0=20=ED=95=B4=EC=8B=9C=20?= =?UTF-8?q?=EA=B8=B0=EB=B0=98=20=EB=8F=99=EA=B8=B0=ED=99=94=20+=20?= =?UTF-8?q?=EC=9E=90=EB=8F=99=20=EC=B5=9C=EC=8B=A0=ED=99=94=20=EA=B0=95?= =?UTF-8?q?=EC=A0=9C?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - push/mr/release: SHA256 해시 체크 Step 0.5 추가 - sync-team-workflow: 전면 재작성 (항상 서버 기준 동기화) - init-project: 마지막에 sync 자동 호출 + .gitignore 팀 관리 경로 - .claude/scripts/ 파일 template-common에 분리 저장 - update-hash.sh: 관리자용 해시 갱신 스크립트 - workflow-version.json: v1.6.0 + content_hashes 필드 --- .claude/scripts/on-commit.sh | 14 ++ .claude/scripts/on-post-compact.sh | 23 +++ .claude/scripts/on-pre-compact.sh | 8 + .claude/skills/init-project/SKILL.md | 89 +++------- .claude/skills/mr/SKILL.md | 25 +++ .claude/skills/push/SKILL.md | 25 +++ .claude/skills/release/SKILL.md | 25 +++ .claude/skills/sync-team-workflow/SKILL.md | 194 +++++++++++++-------- update-hash.sh | 115 ++++++++++++ workflow-version.json | 33 +++- 10 files changed, 409 insertions(+), 142 deletions(-) create mode 100644 .claude/scripts/on-commit.sh create mode 100644 .claude/scripts/on-post-compact.sh create mode 100644 .claude/scripts/on-pre-compact.sh create mode 100644 update-hash.sh diff --git a/.claude/scripts/on-commit.sh b/.claude/scripts/on-commit.sh new file mode 100644 index 0000000..f473403 --- /dev/null +++ b/.claude/scripts/on-commit.sh @@ -0,0 +1,14 @@ +#!/bin/bash +INPUT=$(cat) +COMMAND=$(echo "$INPUT" | python3 -c "import sys,json;print(json.load(sys.stdin).get('tool_input',{}).get('command',''))" 2>/dev/null || echo "") +if echo "$COMMAND" | grep -qE 'git commit'; then + cat </dev/null || echo "") +if [ -z "$CWD" ]; then + CWD=$(pwd) +fi +PROJECT_HASH=$(echo "$CWD" | sed 's|/|-|g') +MEMORY_DIR="$HOME/.claude/projects/$PROJECT_HASH/memory" +CONTEXT="" +if [ -f "$MEMORY_DIR/MEMORY.md" ]; then + SUMMARY=$(head -100 "$MEMORY_DIR/MEMORY.md" | python3 -c "import sys;print(sys.stdin.read().replace('\\\\','\\\\\\\\').replace('\"','\\\\\"').replace('\n','\\\\n'))" 2>/dev/null) + CONTEXT="컨텍스트가 압축되었습니다.\\n\\n[세션 요약]\\n${SUMMARY}" +fi +if [ -f "$MEMORY_DIR/project-snapshot.md" ]; then + SNAP=$(head -50 "$MEMORY_DIR/project-snapshot.md" | python3 -c "import sys;print(sys.stdin.read().replace('\\\\','\\\\\\\\').replace('\"','\\\\\"').replace('\n','\\\\n'))" 2>/dev/null) + CONTEXT="${CONTEXT}\\n\\n[프로젝트 최신 상태]\\n${SNAP}" +fi +if [ -n "$CONTEXT" ]; then + CONTEXT="${CONTEXT}\\n\\n위 내용을 참고하여 작업을 이어가세요. 상세 내용은 memory/ 디렉토리의 각 파일을 참조하세요." + echo "{\"hookSpecificOutput\":{\"additionalContext\":\"${CONTEXT}\"}}" +else + echo "{\"hookSpecificOutput\":{\"additionalContext\":\"컨텍스트가 압축되었습니다. memory 파일이 없으므로 사용자에게 이전 작업 내용을 확인하세요.\"}}" +fi diff --git a/.claude/scripts/on-pre-compact.sh b/.claude/scripts/on-pre-compact.sh new file mode 100644 index 0000000..3f52f09 --- /dev/null +++ b/.claude/scripts/on-pre-compact.sh @@ -0,0 +1,8 @@ +#!/bin/bash +# PreCompact hook: systemMessage만 지원 (hookSpecificOutput 사용 불가) +INPUT=$(cat) +cat </dev/null || echo "") -if [ -z "$CWD" ]; then - CWD=$(pwd) -fi -PROJECT_HASH=$(echo "$CWD" | sed 's|/|-|g') -MEMORY_DIR="$HOME/.claude/projects/$PROJECT_HASH/memory" -CONTEXT="" -if [ -f "$MEMORY_DIR/MEMORY.md" ]; then - SUMMARY=$(head -100 "$MEMORY_DIR/MEMORY.md" | python3 -c "import sys;print(sys.stdin.read().replace('\\\\','\\\\\\\\').replace('\"','\\\\\"').replace('\n','\\\\n'))" 2>/dev/null) - CONTEXT="컨텍스트가 압축되었습니다.\\n\\n[세션 요약]\\n${SUMMARY}" -fi -if [ -f "$MEMORY_DIR/project-snapshot.md" ]; then - SNAP=$(head -50 "$MEMORY_DIR/project-snapshot.md" | python3 -c "import sys;print(sys.stdin.read().replace('\\\\','\\\\\\\\').replace('\"','\\\\\"').replace('\n','\\\\n'))" 2>/dev/null) - CONTEXT="${CONTEXT}\\n\\n[프로젝트 최신 상태]\\n${SNAP}" -fi -if [ -n "$CONTEXT" ]; then - CONTEXT="${CONTEXT}\\n\\n위 내용을 참고하여 작업을 이어가세요. 상세 내용은 memory/ 디렉토리의 각 파일을 참조하세요." - echo "{\"hookSpecificOutput\":{\"additionalContext\":\"${CONTEXT}\"}}" -else - echo "{\"hookSpecificOutput\":{\"additionalContext\":\"컨텍스트가 압축되었습니다. memory 파일이 없으므로 사용자에게 이전 작업 내용을 확인하세요.\"}}" -fi -``` - -- `.claude/scripts/on-commit.sh`: - -```bash -#!/bin/bash -INPUT=$(cat) -COMMAND=$(echo "$INPUT" | python3 -c "import sys,json;print(json.load(sys.stdin).get('tool_input',{}).get('command',''))" 2>/dev/null || echo "") -if echo "$COMMAND" | grep -qE 'git commit'; then - cat </dev/null) +PROJECT_TYPE=$(python3 -c "import json; print(json.load(open('.claude/workflow-version.json')).get('project_type', ''))" 2>/dev/null) + +# 서버 해시 조회 +SERVER_VER=$(curl -sf --max-time 5 "${GITEA_URL}/gc/template-common/raw/branch/develop/workflow-version.json") +SERVER_HASH=$(echo "$SERVER_VER" | python3 -c "import sys,json; print(json.load(sys.stdin).get('content_hashes',{}).get('${PROJECT_TYPE}',''))" 2>/dev/null) + +# 로컬 해시 계산 +LOCAL_HASH=$(find .claude/rules .claude/agents .claude/scripts .githooks \ + .claude/skills/push .claude/skills/mr .claude/skills/create-mr \ + .claude/skills/release .claude/skills/version .claude/skills/fix-issue \ + -type f 2>/dev/null | sort | xargs cat 2>/dev/null | shasum -a 256 | cut -d' ' -f1) +``` + +**비교 결과 처리**: +- **서버 조회 실패** (`SERVER_HASH` 비어있음): "⚠️ 서버 연결 불가, 워크플로우 체크를 건너뜁니다" 경고 후 다음 단계 진행 +- **일치** (`LOCAL_HASH == SERVER_HASH`): 다음 단계 진행 +- **불일치**: "⚠️ 팀 워크플로우가 최신이 아닙니다. 동기화를 실행합니다..." 출력 → **sync-team-workflow 절차를 자동 실행** → 완료 후 원래 작업 계속 + ### 1. 사전 검증 ```bash diff --git a/.claude/skills/push/SKILL.md b/.claude/skills/push/SKILL.md index 7cbdceb..72e7e73 100644 --- a/.claude/skills/push/SKILL.md +++ b/.claude/skills/push/SKILL.md @@ -30,6 +30,31 @@ CAN_PUSH=$(echo "$PERMISSIONS" | python3 -c "import sys,json; print(json.load(sy - `CAN_PUSH`가 `False`이면: "push 권한이 필요합니다. 프로젝트 관리자에게 요청하세요." 안내 후 종료 +### 0.5. 팀 워크플로우 최신화 확인 + +`.claude/workflow-version.json`이 존재하지 않으면 이 단계를 건너뛴다 (팀 프로젝트가 아닌 경우). + +```bash +# 로컬 설정 읽기 +GITEA_URL=$(python3 -c "import json; print(json.load(open('.claude/workflow-version.json')).get('gitea_url', 'https://gitea.gc-si.dev'))" 2>/dev/null) +PROJECT_TYPE=$(python3 -c "import json; print(json.load(open('.claude/workflow-version.json')).get('project_type', ''))" 2>/dev/null) + +# 서버 해시 조회 +SERVER_VER=$(curl -sf --max-time 5 "${GITEA_URL}/gc/template-common/raw/branch/develop/workflow-version.json") +SERVER_HASH=$(echo "$SERVER_VER" | python3 -c "import sys,json; print(json.load(sys.stdin).get('content_hashes',{}).get('${PROJECT_TYPE}',''))" 2>/dev/null) + +# 로컬 해시 계산 +LOCAL_HASH=$(find .claude/rules .claude/agents .claude/scripts .githooks \ + .claude/skills/push .claude/skills/mr .claude/skills/create-mr \ + .claude/skills/release .claude/skills/version .claude/skills/fix-issue \ + -type f 2>/dev/null | sort | xargs cat 2>/dev/null | shasum -a 256 | cut -d' ' -f1) +``` + +**비교 결과 처리**: +- **서버 조회 실패** (`SERVER_HASH` 비어있음): "⚠️ 서버 연결 불가, 워크플로우 체크를 건너뜁니다" 경고 후 다음 단계 진행 +- **일치** (`LOCAL_HASH == SERVER_HASH`): 다음 단계 진행 +- **불일치**: "⚠️ 팀 워크플로우가 최신이 아닙니다. 동기화를 실행합니다..." 출력 → **sync-team-workflow 절차를 자동 실행** → 완료 후 원래 작업 계속 + ### 1. 현재 상태 수집 ```bash diff --git a/.claude/skills/release/SKILL.md b/.claude/skills/release/SKILL.md index 8369a99..89a105e 100644 --- a/.claude/skills/release/SKILL.md +++ b/.claude/skills/release/SKILL.md @@ -29,6 +29,31 @@ IS_ADMIN=$(echo "$PERMISSIONS" | python3 -c "import sys,json; print(json.load(sy - `IS_ADMIN`이 `False`이면: "릴리즈는 프로젝트 관리자만 실행할 수 있습니다." 안내 후 종료 +### 0.5. 팀 워크플로우 최신화 확인 + +`.claude/workflow-version.json`이 존재하지 않으면 이 단계를 건너뛴다 (팀 프로젝트가 아닌 경우). + +```bash +# 로컬 설정 읽기 +GITEA_URL=$(python3 -c "import json; print(json.load(open('.claude/workflow-version.json')).get('gitea_url', 'https://gitea.gc-si.dev'))" 2>/dev/null) +PROJECT_TYPE=$(python3 -c "import json; print(json.load(open('.claude/workflow-version.json')).get('project_type', ''))" 2>/dev/null) + +# 서버 해시 조회 +SERVER_VER=$(curl -sf --max-time 5 "${GITEA_URL}/gc/template-common/raw/branch/develop/workflow-version.json") +SERVER_HASH=$(echo "$SERVER_VER" | python3 -c "import sys,json; print(json.load(sys.stdin).get('content_hashes',{}).get('${PROJECT_TYPE}',''))" 2>/dev/null) + +# 로컬 해시 계산 +LOCAL_HASH=$(find .claude/rules .claude/agents .claude/scripts .githooks \ + .claude/skills/push .claude/skills/mr .claude/skills/create-mr \ + .claude/skills/release .claude/skills/version .claude/skills/fix-issue \ + -type f 2>/dev/null | sort | xargs cat 2>/dev/null | shasum -a 256 | cut -d' ' -f1) +``` + +**비교 결과 처리**: +- **서버 조회 실패** (`SERVER_HASH` 비어있음): "⚠️ 서버 연결 불가, 워크플로우 체크를 건너뜁니다" 경고 후 다음 단계 진행 +- **일치** (`LOCAL_HASH == SERVER_HASH`): 다음 단계 진행 +- **불일치**: "⚠️ 팀 워크플로우가 최신이 아닙니다. 동기화를 실행합니다..." 출력 → **sync-team-workflow 절차를 자동 실행** → 완료 후 원래 작업 계속 + ### 1. 사전 검증 - 커밋되지 않은 변경 사항이 있으면 경고 ("먼저 /push로 커밋하세요") diff --git a/.claude/skills/sync-team-workflow/SKILL.md b/.claude/skills/sync-team-workflow/SKILL.md index 536ad71..5aa6441 100644 --- a/.claude/skills/sync-team-workflow/SKILL.md +++ b/.claude/skills/sync-team-workflow/SKILL.md @@ -3,123 +3,163 @@ name: sync-team-workflow description: 팀 글로벌 워크플로우를 현재 프로젝트에 동기화합니다 --- -팀 글로벌 워크플로우의 최신 버전을 현재 프로젝트에 적용합니다. +팀 글로벌 워크플로우의 최신 파일을 서버에서 다운로드하여 로컬에 적용합니다. +호출 시 항상 서버 기준으로 전체 동기화합니다 (버전 비교 없음). ## 수행 절차 -### 1. 글로벌 버전 조회 -Gitea API로 template-common 리포의 workflow-version.json 조회: +### 1. 사전 조건 확인 + +`.claude/workflow-version.json` 존재 확인: +- 없으면 → "/init-project를 먼저 실행해주세요" 안내 후 종료 + +설정 읽기: ```bash GITEA_URL=$(python3 -c "import json; print(json.load(open('.claude/workflow-version.json')).get('gitea_url', 'https://gitea.gc-si.dev'))" 2>/dev/null || echo "https://gitea.gc-si.dev") - -curl -sf "${GITEA_URL}/gc/template-common/raw/branch/develop/workflow-version.json" +PROJECT_TYPE=$(python3 -c "import json; print(json.load(open('.claude/workflow-version.json')).get('project_type', ''))" 2>/dev/null || echo "") ``` -### 2. 버전 비교 -로컬 `.claude/workflow-version.json`의 `applied_global_version` 필드와 비교: -- 버전 일치 → "최신 버전입니다" 안내 후 종료 -- 버전 불일치 → 미적용 변경 항목 추출하여 표시 - -### 3. 프로젝트 타입 감지 -자동 감지 순서: -1. `.claude/workflow-version.json`의 `project_type` 필드 확인 -2. 없으면: `pom.xml` → java-maven, `build.gradle` → java-gradle, `package.json` → react-ts +프로젝트 타입이 비어있으면 자동 감지: +1. `pom.xml` → java-maven +2. `build.gradle` / `build.gradle.kts` → java-gradle +3. `package.json` + `tsconfig.json` → react-ts +4. 감지 실패 → 사용자에게 선택 요청 ### Gitea 파일 다운로드 URL 패턴 -⚠️ Gitea raw 파일은 반드시 **web raw URL**을 사용해야 합니다 (`/api/v1/` 경로 사용 불가): +⚠️ Gitea raw 파일은 반드시 **web raw URL** 사용: ```bash -GITEA_URL="${GITEA_URL:-https://gitea.gc-si.dev}" # common 파일: ${GITEA_URL}/gc/template-common/raw/branch/develop/<파일경로> -# 타입별 파일: ${GITEA_URL}/gc/template-<타입>/raw/branch/develop/<파일경로> -# 예시: -curl -sf "${GITEA_URL}/gc/template-common/raw/branch/develop/.claude/rules/team-policy.md" -curl -sf "${GITEA_URL}/gc/template-react-ts/raw/branch/develop/.editorconfig" +# 타입별 파일: ${GITEA_URL}/gc/template-${PROJECT_TYPE}/raw/branch/develop/<파일경로> ``` -### 4. 파일 다운로드 및 적용 -위의 URL 패턴으로 해당 타입 + common 템플릿 파일 다운로드: +### 2. 디렉토리 준비 -#### 4-1. 규칙 파일 (덮어쓰기) -팀 규칙은 로컬 수정 불가 — 항상 글로벌 최신으로 교체: +필요한 디렉토리가 없으면 생성: +```bash +mkdir -p .claude/rules .claude/agents .claude/scripts +mkdir -p .claude/skills/push .claude/skills/mr .claude/skills/create-mr +mkdir -p .claude/skills/release .claude/skills/version .claude/skills/fix-issue +mkdir -p .githooks +``` + +### 3. 서버 파일 다운로드 + 적용 + +각 파일을 `curl -sf` 로 다운로드하여 프로젝트 루트의 동일 경로에 저장. +다운로드 실패한 파일은 경고 출력 후 건너뜀. + +#### 3-1. template-common 파일 (덮어쓰기) + +**규칙 파일**: ``` .claude/rules/team-policy.md .claude/rules/git-workflow.md .claude/rules/release-notes-guide.md .claude/rules/subagent-policy.md -.claude/rules/code-style.md (타입별) -.claude/rules/naming.md (타입별) -.claude/rules/testing.md (타입별) ``` -#### 4-1b. 에이전트 파일 (덮어쓰기) +**에이전트 파일**: ``` .claude/agents/explorer.md .claude/agents/implementer.md .claude/agents/reviewer.md ``` -#### 4-2. settings.json (부분 갱신) - -⚠️ settings.json은 **타입별 템플릿**에서 다운로드 (template-common에는 없음): -```bash -curl -sf "${GITEA_URL}/gc/template-${PROJECT_TYPE}/raw/branch/develop/.claude/settings.json" +**스킬 파일 (6종)**: ``` - -다운로드한 최신 settings.json과 로컬 settings.json을 비교하여 부분 갱신: -- `env`: 글로벌 최신으로 교체 (CLAUDE_BOT_TOKEN 등 팀 공통 환경변수) -- `deny` 목록: 글로벌 최신으로 교체 -- `allow` 목록: 기존 사용자 커스텀 유지 + 글로벌 기본값 병합 -- `hooks`: init-project SKILL.md의 hooks JSON 블록을 참조하여 교체 (없으면 추가) - - SessionStart(compact) → on-post-compact.sh - - PreCompact → on-pre-compact.sh - - PostToolUse(Bash) → on-commit.sh - -#### 4-3. 스킬 파일 (덮어쓰기) -``` -.claude/skills/create-mr/SKILL.md -.claude/skills/fix-issue/SKILL.md -.claude/skills/sync-team-workflow/SKILL.md -.claude/skills/init-project/SKILL.md .claude/skills/push/SKILL.md .claude/skills/mr/SKILL.md +.claude/skills/create-mr/SKILL.md .claude/skills/release/SKILL.md .claude/skills/version/SKILL.md +.claude/skills/fix-issue/SKILL.md ``` -#### 4-4. Git Hooks (덮어쓰기 + 실행 권한) - -`commit-msg`, `post-checkout`은 **항상 팀 표준으로 교체** (팀 커뮤니케이션 규칙 + 인프라). - -`pre-commit`은 `.claude/workflow-version.json`의 `custom_pre_commit` 플래그를 확인: -- `"custom_pre_commit": true` → pre-commit 건너뜀 (프로젝트 커스텀 유지), "⚠️ pre-commit은 프로젝트 커스텀 유지" 로그 -- 플래그 없거나 false → 팀 표준으로 교체 - -```bash -chmod +x .githooks/* -``` - -#### 4-5. Hook 스크립트 갱신 -init-project SKILL.md의 코드 블록에서 최신 스크립트를 추출하여 덮어쓰기: +**Hook 스크립트**: ``` .claude/scripts/on-pre-compact.sh .claude/scripts/on-post-compact.sh .claude/scripts/on-commit.sh ``` -실행 권한 부여: `chmod +x .claude/scripts/*.sh` -### 5. 로컬 버전 업데이트 -`.claude/workflow-version.json` 갱신: -```json -{ - "applied_global_version": "새버전", - "applied_date": "오늘날짜", - "project_type": "감지된타입", - "gitea_url": "https://gitea.gc-si.dev" -} +**Git Hooks** (commit-msg, post-checkout은 항상 교체): +``` +.githooks/commit-msg +.githooks/post-checkout ``` +다운로드 예시: +```bash +curl -sf "${GITEA_URL}/gc/template-common/raw/branch/develop/.claude/rules/team-policy.md" -o ".claude/rules/team-policy.md" +``` + +#### 3-2. template-{type} 파일 (타입별 덮어쓰기) + +``` +.claude/rules/code-style.md +.claude/rules/naming.md +.claude/rules/testing.md +``` + +**pre-commit hook**: +`.claude/workflow-version.json`의 `custom_pre_commit` 플래그 확인: +- `"custom_pre_commit": true` → pre-commit 건너뜀, "⚠️ pre-commit은 프로젝트 커스텀 유지" 로그 +- 플래그 없거나 false → `.githooks/pre-commit` 교체 + +다운로드 예시: +```bash +curl -sf "${GITEA_URL}/gc/template-${PROJECT_TYPE}/raw/branch/develop/.claude/rules/code-style.md" -o ".claude/rules/code-style.md" +``` + +#### 3-3. 실행 권한 부여 +```bash +chmod +x .githooks/* 2>/dev/null +chmod +x .claude/scripts/*.sh 2>/dev/null +``` + +### 4. settings.json 부분 머지 + +⚠️ settings.json은 **타입별 템플릿**에서 다운로드 (template-common에는 없음): +```bash +SERVER_SETTINGS=$(curl -sf "${GITEA_URL}/gc/template-${PROJECT_TYPE}/raw/branch/develop/.claude/settings.json") +``` + +다운로드한 최신 settings.json과 로컬 `.claude/settings.json`을 비교하여 부분 갱신: +- `env`: 서버 최신으로 교체 +- `deny` 목록: 서버 최신으로 교체 +- `allow` 목록: 기존 사용자 커스텀 유지 + 서버 기본값 병합 +- `hooks`: 서버 최신으로 교체 + +### 5. workflow-version.json 갱신 + +서버의 최신 `workflow-version.json` 조회: +```bash +SERVER_VER=$(curl -sf "${GITEA_URL}/gc/template-common/raw/branch/develop/workflow-version.json") +SERVER_VERSION=$(echo "$SERVER_VER" | python3 -c "import sys,json; print(json.load(sys.stdin).get('version',''))") +``` + +`.claude/workflow-version.json` 업데이트: +```json +{ + "applied_global_version": "<서버 version>", + "applied_date": "<현재날짜>", + "project_type": "<프로젝트타입>", + "gitea_url": "" +} +``` +기존 필드(`custom_pre_commit` 등)는 보존. + ### 6. 변경 보고 -- `git diff`로 변경 내역 확인 -- 업데이트된 파일 목록 출력 -- 변경 로그(글로벌 workflow-version.json의 changes) 표시 -- 필요한 추가 조치 안내 (빌드 확인, 의존성 업데이트 등) + +- 다운로드/갱신된 파일 목록 출력 +- 서버 `workflow-version.json`의 `changes` 중 최신 항목 표시 +- 결과 형태: +``` +✅ 팀 워크플로우 동기화 완료 + 버전: v1.6.0 + 갱신 파일: 22개 (rules 7, agents 3, skills 6, scripts 3, hooks 3) + settings.json: 부분 갱신 (env, deny, hooks) +``` + +## 필요 환경변수 + +없음 (Gitea raw URL은 인증 불필요) diff --git a/update-hash.sh b/update-hash.sh new file mode 100644 index 0000000..8dcf8a5 --- /dev/null +++ b/update-hash.sh @@ -0,0 +1,115 @@ +#!/bin/bash +#============================================================================== +# update-hash.sh — 팀 워크플로우 content_hashes 갱신 스크립트 +# +# template-common 리포 루트에서 실행한다. +# common 파일 + 각 타입별 파일을 결합하여 SHA256 해시를 계산하고 +# workflow-version.json의 content_hashes 필드를 갱신한다. +# +# 사용법: +# cd template-common && bash update-hash.sh +# +# 의존성: curl, python3, shasum +#============================================================================== +set -e + +# Gitea URL 읽기 +GITEA_URL=$(python3 -c "import json; print(json.load(open('workflow-version.json')).get('gitea_url', 'https://gitea.gc-si.dev'))" 2>/dev/null) +if [ -z "$GITEA_URL" ]; then + echo "ERROR: workflow-version.json에서 gitea_url을 읽을 수 없습니다." + exit 1 +fi + +TYPES=("java-maven" "java-gradle" "react-ts") +TYPE_SPECIFIC_FILES=(".claude/rules/code-style.md" ".claude/rules/naming.md" ".claude/rules/testing.md" ".githooks/pre-commit") + +# 임시 디렉토리 +TMPDIR=$(mktemp -d) +trap "rm -rf $TMPDIR" EXIT + +echo "=== 팀 워크플로우 해시 갱신 ===" +echo "Gitea URL: $GITEA_URL" +echo "" + +# 1. common 관리 파일을 임시 디렉토리에 복사 +MANAGED_DIRS=".claude/rules .claude/agents .claude/scripts .githooks" +MANAGED_SKILLS=".claude/skills/push .claude/skills/mr .claude/skills/create-mr .claude/skills/release .claude/skills/version .claude/skills/fix-issue" + +mkdir -p "$TMPDIR/common" +for dir in $MANAGED_DIRS $MANAGED_SKILLS; do + if [ -d "$dir" ]; then + mkdir -p "$TMPDIR/common/$dir" + find "$dir" -type f -exec sh -c 'cp "$1" "'"$TMPDIR/common"'/$1"' _ {} \; + fi +done + +COMMON_COUNT=$(cd "$TMPDIR/common" && find . -type f | wc -l | tr -d ' ') +echo "Common 파일: ${COMMON_COUNT}개" + +# 2. 각 타입별 해시 계산 +declare -A HASHES + +for TYPE in "${TYPES[@]}"; do + echo "" + echo "--- $TYPE ---" + + # common 파일 복사 + rm -rf "$TMPDIR/combined" + cp -r "$TMPDIR/common" "$TMPDIR/combined" + + # 타입별 파일 다운로드 + TYPE_COUNT=0 + for FILE in "${TYPE_SPECIFIC_FILES[@]}"; do + URL="${GITEA_URL}/gc/template-${TYPE}/raw/branch/develop/${FILE}" + TARGET="$TMPDIR/combined/$FILE" + mkdir -p "$(dirname "$TARGET")" + + if curl -sf --max-time 10 "$URL" -o "$TARGET" 2>/dev/null; then + echo " ✓ $FILE" + TYPE_COUNT=$((TYPE_COUNT + 1)) + else + echo " ✗ $FILE (다운로드 실패, 건너뜀)" + fi + done + + # 해시 계산 (find | sort | xargs cat | shasum — 로컬 계산과 동일한 방식) + HASH=$(cd "$TMPDIR/combined" && find .claude/rules .claude/agents .claude/scripts .githooks \ + .claude/skills/push .claude/skills/mr .claude/skills/create-mr \ + .claude/skills/release .claude/skills/version .claude/skills/fix-issue \ + -type f 2>/dev/null | sort | xargs cat 2>/dev/null | shasum -a 256 | cut -d' ' -f1) + + HASHES[$TYPE]=$HASH + echo " 타입별 파일: ${TYPE_COUNT}개 추가" + echo " 해시: $HASH" +done + +# 3. workflow-version.json 갱신 +echo "" +echo "=== workflow-version.json 갱신 ===" + +# Python으로 직접 갱신 (bash 변수 전달) +python3 -c " +import json + +hashes = { +$(for TYPE in "${TYPES[@]}"; do + echo " '${TYPE}': '${HASHES[$TYPE]}'," +done) +} + +with open('workflow-version.json', 'r') as f: + data = json.load(f) + +data['content_hashes'] = hashes + +with open('workflow-version.json', 'w') as f: + json.dump(data, f, indent=2, ensure_ascii=False) + f.write('\n') + +print('content_hashes 갱신 완료:') +for t, h in hashes.items(): + print(f' {t}: {h}') +" + +echo "" +echo "✅ 완료. workflow-version.json을 커밋하세요." diff --git a/workflow-version.json b/workflow-version.json index fba9683..d2baa10 100644 --- a/workflow-version.json +++ b/workflow-version.json @@ -1,9 +1,38 @@ { - "version": "1.5.0", - "updated": "2026-03-01", + "version": "1.6.0", + "updated": "2026-03-06", "gitea_url": "https://gitea.gc-si.dev", "nexus_url": "https://nexus.gc-si.dev", + "content_hashes": { + "java-maven": "", + "java-gradle": "", + "react-ts": "" + }, "changes": [ + { + "version": "1.6.0", + "date": "2026-03-06", + "description": "해시 기반 동기화 + 자동 최신화 강제", + "items": [ + "push/mr/release: SHA256 해시 체크 → 불일치 시 자동 동기화", + "sync-team-workflow: 해시 기반 전면 재작성 (버전 비교 제거, 항상 서버 기준 동기화)", + "init-project: 마지막에 sync 자동 호출, .gitignore에 팀 관리 경로 추가", + "팀 관리 파일 .gitignore 처리 (리포 커밋 방지, rollback 위험 제거)", + ".claude/scripts/ 파일을 template-common에 분리 저장", + "update-hash.sh: 관리자용 해시 갱신 스크립트 추가", + "적용: 프로젝트에서 /sync-team-workflow 실행" + ], + "affected_files": [ + ".claude/skills/sync-team-workflow/SKILL.md", + ".claude/skills/push/SKILL.md", + ".claude/skills/mr/SKILL.md", + ".claude/skills/release/SKILL.md", + ".claude/skills/init-project/SKILL.md", + ".claude/scripts/*", + "workflow-version.json", + "update-hash.sh" + ] + }, { "version": "1.5.0", "date": "2026-03-01", -- 2.45.2 From 013fd6dc8890f7624dbbeabbd1cf09f52aa3bdcb Mon Sep 17 00:00:00 2001 From: htlee Date: Fri, 6 Mar 2026 14:21:04 +0900 Subject: [PATCH 08/11] =?UTF-8?q?chore:=20content=5Fhashes=20=EA=B0=B1?= =?UTF-8?q?=EC=8B=A0=20+=20update-hash.sh=20bash3=20=ED=98=B8=ED=99=98=20?= =?UTF-8?q?=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - 3개 타입별 SHA256 해시 계산 완료 - declare -A → 파일 기반 방식으로 bash3(macOS) 호환 수정 - sha256sum 폴백 추가 (Linux 서버 호환) --- update-hash.sh | 67 +++++++++++++++++++++++++++---------------- workflow-version.json | 6 ++-- 2 files changed, 45 insertions(+), 28 deletions(-) diff --git a/update-hash.sh b/update-hash.sh index 8dcf8a5..01d0492 100644 --- a/update-hash.sh +++ b/update-hash.sh @@ -9,10 +9,20 @@ # 사용법: # cd template-common && bash update-hash.sh # -# 의존성: curl, python3, shasum +# 의존성: curl, python3, shasum (macOS) 또는 sha256sum (Linux) #============================================================================== set -e +# shasum 호환 (macOS: shasum, Linux: sha256sum) +if command -v shasum &>/dev/null; then + SHA_CMD="shasum -a 256" +elif command -v sha256sum &>/dev/null; then + SHA_CMD="sha256sum" +else + echo "ERROR: shasum 또는 sha256sum이 필요합니다." + exit 1 +fi + # Gitea URL 읽기 GITEA_URL=$(python3 -c "import json; print(json.load(open('workflow-version.json')).get('gitea_url', 'https://gitea.gc-si.dev'))" 2>/dev/null) if [ -z "$GITEA_URL" ]; then @@ -20,12 +30,12 @@ if [ -z "$GITEA_URL" ]; then exit 1 fi -TYPES=("java-maven" "java-gradle" "react-ts") -TYPE_SPECIFIC_FILES=(".claude/rules/code-style.md" ".claude/rules/naming.md" ".claude/rules/testing.md" ".githooks/pre-commit") +TYPES="java-maven java-gradle react-ts" +TYPE_SPECIFIC_FILES=".claude/rules/code-style.md .claude/rules/naming.md .claude/rules/testing.md .githooks/pre-commit" # 임시 디렉토리 -TMPDIR=$(mktemp -d) -trap "rm -rf $TMPDIR" EXIT +WORK_TMPDIR=$(mktemp -d) +trap "rm -rf $WORK_TMPDIR" EXIT echo "=== 팀 워크플로우 해시 갱신 ===" echo "Gitea URL: $GITEA_URL" @@ -35,33 +45,35 @@ echo "" MANAGED_DIRS=".claude/rules .claude/agents .claude/scripts .githooks" MANAGED_SKILLS=".claude/skills/push .claude/skills/mr .claude/skills/create-mr .claude/skills/release .claude/skills/version .claude/skills/fix-issue" -mkdir -p "$TMPDIR/common" +mkdir -p "$WORK_TMPDIR/common" for dir in $MANAGED_DIRS $MANAGED_SKILLS; do if [ -d "$dir" ]; then - mkdir -p "$TMPDIR/common/$dir" - find "$dir" -type f -exec sh -c 'cp "$1" "'"$TMPDIR/common"'/$1"' _ {} \; + mkdir -p "$WORK_TMPDIR/common/$dir" + find "$dir" -type f -exec sh -c 'cp "$1" "'"$WORK_TMPDIR/common"'/$1"' _ {} \; fi done -COMMON_COUNT=$(cd "$TMPDIR/common" && find . -type f | wc -l | tr -d ' ') +COMMON_COUNT=$(cd "$WORK_TMPDIR/common" && find . -type f | wc -l | tr -d ' ') echo "Common 파일: ${COMMON_COUNT}개" -# 2. 각 타입별 해시 계산 -declare -A HASHES +# 2. 각 타입별 해시 계산 → 파일로 저장 +HASH_FILE="$WORK_TMPDIR/hashes.json" +echo "{" > "$HASH_FILE" -for TYPE in "${TYPES[@]}"; do +FIRST=true +for TYPE in $TYPES; do echo "" echo "--- $TYPE ---" # common 파일 복사 - rm -rf "$TMPDIR/combined" - cp -r "$TMPDIR/common" "$TMPDIR/combined" + rm -rf "$WORK_TMPDIR/combined" + cp -r "$WORK_TMPDIR/common" "$WORK_TMPDIR/combined" # 타입별 파일 다운로드 TYPE_COUNT=0 - for FILE in "${TYPE_SPECIFIC_FILES[@]}"; do + for FILE in $TYPE_SPECIFIC_FILES; do URL="${GITEA_URL}/gc/template-${TYPE}/raw/branch/develop/${FILE}" - TARGET="$TMPDIR/combined/$FILE" + TARGET="$WORK_TMPDIR/combined/$FILE" mkdir -p "$(dirname "$TARGET")" if curl -sf --max-time 10 "$URL" -o "$TARGET" 2>/dev/null; then @@ -73,29 +85,34 @@ for TYPE in "${TYPES[@]}"; do done # 해시 계산 (find | sort | xargs cat | shasum — 로컬 계산과 동일한 방식) - HASH=$(cd "$TMPDIR/combined" && find .claude/rules .claude/agents .claude/scripts .githooks \ + HASH=$(cd "$WORK_TMPDIR/combined" && find .claude/rules .claude/agents .claude/scripts .githooks \ .claude/skills/push .claude/skills/mr .claude/skills/create-mr \ .claude/skills/release .claude/skills/version .claude/skills/fix-issue \ - -type f 2>/dev/null | sort | xargs cat 2>/dev/null | shasum -a 256 | cut -d' ' -f1) + -type f 2>/dev/null | sort | xargs cat 2>/dev/null | $SHA_CMD | cut -d' ' -f1) + + if [ "$FIRST" = true ]; then + FIRST=false + else + echo "," >> "$HASH_FILE" + fi + printf ' "%s": "%s"' "$TYPE" "$HASH" >> "$HASH_FILE" - HASHES[$TYPE]=$HASH echo " 타입별 파일: ${TYPE_COUNT}개 추가" echo " 해시: $HASH" done +echo "" >> "$HASH_FILE" +echo "}" >> "$HASH_FILE" + # 3. workflow-version.json 갱신 echo "" echo "=== workflow-version.json 갱신 ===" -# Python으로 직접 갱신 (bash 변수 전달) python3 -c " import json -hashes = { -$(for TYPE in "${TYPES[@]}"; do - echo " '${TYPE}': '${HASHES[$TYPE]}'," -done) -} +with open('$HASH_FILE', 'r') as f: + hashes = json.load(f) with open('workflow-version.json', 'r') as f: data = json.load(f) diff --git a/workflow-version.json b/workflow-version.json index d2baa10..863448e 100644 --- a/workflow-version.json +++ b/workflow-version.json @@ -4,9 +4,9 @@ "gitea_url": "https://gitea.gc-si.dev", "nexus_url": "https://nexus.gc-si.dev", "content_hashes": { - "java-maven": "", - "java-gradle": "", - "react-ts": "" + "java-maven": "1456f972fb250ba60fa6331d5fad81155c9992bf80947867e34a7178293513db", + "java-gradle": "0f7afc70942888cd2f98e01ef2eef0fcb815c0a68704d27b970a528081710d22", + "react-ts": "2a3a5b6ce9e2c6080c79b78be92c4967ed2418c3efff6082038b3f270c3b7e18" }, "changes": [ { -- 2.45.2 From 303d4dff684fc15ab48bb1fa150950482f093847 Mon Sep 17 00:00:00 2001 From: htlee Date: Fri, 6 Mar 2026 15:30:48 +0900 Subject: [PATCH 09/11] =?UTF-8?q?feat:=20v1.6.1=20custom=5Fpre=5Fcommit=20?= =?UTF-8?q?=ED=94=84=EB=A1=9C=EC=A0=9D=ED=8A=B8=20=ED=95=B4=EC=8B=9C=20?= =?UTF-8?q?=EB=B6=84=EA=B8=B0=20=EC=A7=80=EC=9B=90?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - content_hashes_custom_precommit 필드 추가 (pre-commit 제외 해시) - push/mr/release Step 0.5: custom_pre_commit 플래그에 따라 해시 비교 분기 - update-hash.sh: HASH_NO_PC_FILE 초기화 버그 수정 + 두 세트 해시 동시 계산 --- .claude/skills/mr/SKILL.md | 26 +++++++++++++++++++------- .claude/skills/push/SKILL.md | 26 +++++++++++++++++++------- .claude/skills/release/SKILL.md | 26 +++++++++++++++++++------- update-hash.sh | 25 +++++++++++++++++++++++-- workflow-version.json | 31 +++++++++++++++++++++++++++---- 5 files changed, 107 insertions(+), 27 deletions(-) diff --git a/.claude/skills/mr/SKILL.md b/.claude/skills/mr/SKILL.md index 27c8c27..cb218b1 100644 --- a/.claude/skills/mr/SKILL.md +++ b/.claude/skills/mr/SKILL.md @@ -38,16 +38,28 @@ CAN_PUSH=$(echo "$PERMISSIONS" | python3 -c "import sys,json; print(json.load(sy # 로컬 설정 읽기 GITEA_URL=$(python3 -c "import json; print(json.load(open('.claude/workflow-version.json')).get('gitea_url', 'https://gitea.gc-si.dev'))" 2>/dev/null) PROJECT_TYPE=$(python3 -c "import json; print(json.load(open('.claude/workflow-version.json')).get('project_type', ''))" 2>/dev/null) +CUSTOM_PRECOMMIT=$(python3 -c "import json; print(json.load(open('.claude/workflow-version.json')).get('custom_pre_commit', False))" 2>/dev/null) -# 서버 해시 조회 +# 서버 해시 조회 (custom_pre_commit이면 pre-commit 제외 해시 사용) SERVER_VER=$(curl -sf --max-time 5 "${GITEA_URL}/gc/template-common/raw/branch/develop/workflow-version.json") -SERVER_HASH=$(echo "$SERVER_VER" | python3 -c "import sys,json; print(json.load(sys.stdin).get('content_hashes',{}).get('${PROJECT_TYPE}',''))" 2>/dev/null) +if [ "$CUSTOM_PRECOMMIT" = "True" ]; then + SERVER_HASH=$(echo "$SERVER_VER" | python3 -c "import sys,json; print(json.load(sys.stdin).get('content_hashes_custom_precommit',{}).get('${PROJECT_TYPE}',''))" 2>/dev/null) +else + SERVER_HASH=$(echo "$SERVER_VER" | python3 -c "import sys,json; print(json.load(sys.stdin).get('content_hashes',{}).get('${PROJECT_TYPE}',''))" 2>/dev/null) +fi -# 로컬 해시 계산 -LOCAL_HASH=$(find .claude/rules .claude/agents .claude/scripts .githooks \ - .claude/skills/push .claude/skills/mr .claude/skills/create-mr \ - .claude/skills/release .claude/skills/version .claude/skills/fix-issue \ - -type f 2>/dev/null | sort | xargs cat 2>/dev/null | shasum -a 256 | cut -d' ' -f1) +# 로컬 해시 계산 (custom_pre_commit이면 .githooks/pre-commit 제외) +if [ "$CUSTOM_PRECOMMIT" = "True" ]; then + LOCAL_HASH=$(find .claude/rules .claude/agents .claude/scripts .githooks \ + .claude/skills/push .claude/skills/mr .claude/skills/create-mr \ + .claude/skills/release .claude/skills/version .claude/skills/fix-issue \ + -type f ! -path '.githooks/pre-commit' 2>/dev/null | sort | xargs cat 2>/dev/null | shasum -a 256 | cut -d' ' -f1) +else + LOCAL_HASH=$(find .claude/rules .claude/agents .claude/scripts .githooks \ + .claude/skills/push .claude/skills/mr .claude/skills/create-mr \ + .claude/skills/release .claude/skills/version .claude/skills/fix-issue \ + -type f 2>/dev/null | sort | xargs cat 2>/dev/null | shasum -a 256 | cut -d' ' -f1) +fi ``` **비교 결과 처리**: diff --git a/.claude/skills/push/SKILL.md b/.claude/skills/push/SKILL.md index 72e7e73..b9b7203 100644 --- a/.claude/skills/push/SKILL.md +++ b/.claude/skills/push/SKILL.md @@ -38,16 +38,28 @@ CAN_PUSH=$(echo "$PERMISSIONS" | python3 -c "import sys,json; print(json.load(sy # 로컬 설정 읽기 GITEA_URL=$(python3 -c "import json; print(json.load(open('.claude/workflow-version.json')).get('gitea_url', 'https://gitea.gc-si.dev'))" 2>/dev/null) PROJECT_TYPE=$(python3 -c "import json; print(json.load(open('.claude/workflow-version.json')).get('project_type', ''))" 2>/dev/null) +CUSTOM_PRECOMMIT=$(python3 -c "import json; print(json.load(open('.claude/workflow-version.json')).get('custom_pre_commit', False))" 2>/dev/null) -# 서버 해시 조회 +# 서버 해시 조회 (custom_pre_commit이면 pre-commit 제외 해시 사용) SERVER_VER=$(curl -sf --max-time 5 "${GITEA_URL}/gc/template-common/raw/branch/develop/workflow-version.json") -SERVER_HASH=$(echo "$SERVER_VER" | python3 -c "import sys,json; print(json.load(sys.stdin).get('content_hashes',{}).get('${PROJECT_TYPE}',''))" 2>/dev/null) +if [ "$CUSTOM_PRECOMMIT" = "True" ]; then + SERVER_HASH=$(echo "$SERVER_VER" | python3 -c "import sys,json; print(json.load(sys.stdin).get('content_hashes_custom_precommit',{}).get('${PROJECT_TYPE}',''))" 2>/dev/null) +else + SERVER_HASH=$(echo "$SERVER_VER" | python3 -c "import sys,json; print(json.load(sys.stdin).get('content_hashes',{}).get('${PROJECT_TYPE}',''))" 2>/dev/null) +fi -# 로컬 해시 계산 -LOCAL_HASH=$(find .claude/rules .claude/agents .claude/scripts .githooks \ - .claude/skills/push .claude/skills/mr .claude/skills/create-mr \ - .claude/skills/release .claude/skills/version .claude/skills/fix-issue \ - -type f 2>/dev/null | sort | xargs cat 2>/dev/null | shasum -a 256 | cut -d' ' -f1) +# 로컬 해시 계산 (custom_pre_commit이면 .githooks/pre-commit 제외) +if [ "$CUSTOM_PRECOMMIT" = "True" ]; then + LOCAL_HASH=$(find .claude/rules .claude/agents .claude/scripts .githooks \ + .claude/skills/push .claude/skills/mr .claude/skills/create-mr \ + .claude/skills/release .claude/skills/version .claude/skills/fix-issue \ + -type f ! -path '.githooks/pre-commit' 2>/dev/null | sort | xargs cat 2>/dev/null | shasum -a 256 | cut -d' ' -f1) +else + LOCAL_HASH=$(find .claude/rules .claude/agents .claude/scripts .githooks \ + .claude/skills/push .claude/skills/mr .claude/skills/create-mr \ + .claude/skills/release .claude/skills/version .claude/skills/fix-issue \ + -type f 2>/dev/null | sort | xargs cat 2>/dev/null | shasum -a 256 | cut -d' ' -f1) +fi ``` **비교 결과 처리**: diff --git a/.claude/skills/release/SKILL.md b/.claude/skills/release/SKILL.md index 89a105e..4d99a21 100644 --- a/.claude/skills/release/SKILL.md +++ b/.claude/skills/release/SKILL.md @@ -37,16 +37,28 @@ IS_ADMIN=$(echo "$PERMISSIONS" | python3 -c "import sys,json; print(json.load(sy # 로컬 설정 읽기 GITEA_URL=$(python3 -c "import json; print(json.load(open('.claude/workflow-version.json')).get('gitea_url', 'https://gitea.gc-si.dev'))" 2>/dev/null) PROJECT_TYPE=$(python3 -c "import json; print(json.load(open('.claude/workflow-version.json')).get('project_type', ''))" 2>/dev/null) +CUSTOM_PRECOMMIT=$(python3 -c "import json; print(json.load(open('.claude/workflow-version.json')).get('custom_pre_commit', False))" 2>/dev/null) -# 서버 해시 조회 +# 서버 해시 조회 (custom_pre_commit이면 pre-commit 제외 해시 사용) SERVER_VER=$(curl -sf --max-time 5 "${GITEA_URL}/gc/template-common/raw/branch/develop/workflow-version.json") -SERVER_HASH=$(echo "$SERVER_VER" | python3 -c "import sys,json; print(json.load(sys.stdin).get('content_hashes',{}).get('${PROJECT_TYPE}',''))" 2>/dev/null) +if [ "$CUSTOM_PRECOMMIT" = "True" ]; then + SERVER_HASH=$(echo "$SERVER_VER" | python3 -c "import sys,json; print(json.load(sys.stdin).get('content_hashes_custom_precommit',{}).get('${PROJECT_TYPE}',''))" 2>/dev/null) +else + SERVER_HASH=$(echo "$SERVER_VER" | python3 -c "import sys,json; print(json.load(sys.stdin).get('content_hashes',{}).get('${PROJECT_TYPE}',''))" 2>/dev/null) +fi -# 로컬 해시 계산 -LOCAL_HASH=$(find .claude/rules .claude/agents .claude/scripts .githooks \ - .claude/skills/push .claude/skills/mr .claude/skills/create-mr \ - .claude/skills/release .claude/skills/version .claude/skills/fix-issue \ - -type f 2>/dev/null | sort | xargs cat 2>/dev/null | shasum -a 256 | cut -d' ' -f1) +# 로컬 해시 계산 (custom_pre_commit이면 .githooks/pre-commit 제외) +if [ "$CUSTOM_PRECOMMIT" = "True" ]; then + LOCAL_HASH=$(find .claude/rules .claude/agents .claude/scripts .githooks \ + .claude/skills/push .claude/skills/mr .claude/skills/create-mr \ + .claude/skills/release .claude/skills/version .claude/skills/fix-issue \ + -type f ! -path '.githooks/pre-commit' 2>/dev/null | sort | xargs cat 2>/dev/null | shasum -a 256 | cut -d' ' -f1) +else + LOCAL_HASH=$(find .claude/rules .claude/agents .claude/scripts .githooks \ + .claude/skills/push .claude/skills/mr .claude/skills/create-mr \ + .claude/skills/release .claude/skills/version .claude/skills/fix-issue \ + -type f 2>/dev/null | sort | xargs cat 2>/dev/null | shasum -a 256 | cut -d' ' -f1) +fi ``` **비교 결과 처리**: diff --git a/update-hash.sh b/update-hash.sh index 01d0492..43438e1 100644 --- a/update-hash.sh +++ b/update-hash.sh @@ -58,7 +58,9 @@ echo "Common 파일: ${COMMON_COUNT}개" # 2. 각 타입별 해시 계산 → 파일로 저장 HASH_FILE="$WORK_TMPDIR/hashes.json" +HASH_NO_PC_FILE="$WORK_TMPDIR/hashes_no_pc.json" echo "{" > "$HASH_FILE" +echo "{" > "$HASH_NO_PC_FILE" FIRST=true for TYPE in $TYPES; do @@ -84,25 +86,36 @@ for TYPE in $TYPES; do fi done - # 해시 계산 (find | sort | xargs cat | shasum — 로컬 계산과 동일한 방식) + # 해시 계산 — 전체 (find | sort | xargs cat | shasum) HASH=$(cd "$WORK_TMPDIR/combined" && find .claude/rules .claude/agents .claude/scripts .githooks \ .claude/skills/push .claude/skills/mr .claude/skills/create-mr \ .claude/skills/release .claude/skills/version .claude/skills/fix-issue \ -type f 2>/dev/null | sort | xargs cat 2>/dev/null | $SHA_CMD | cut -d' ' -f1) + # 해시 계산 — pre-commit 제외 (custom_pre_commit 프로젝트용) + HASH_NO_PC=$(cd "$WORK_TMPDIR/combined" && find .claude/rules .claude/agents .claude/scripts .githooks \ + .claude/skills/push .claude/skills/mr .claude/skills/create-mr \ + .claude/skills/release .claude/skills/version .claude/skills/fix-issue \ + -type f ! -path '.githooks/pre-commit' 2>/dev/null | sort | xargs cat 2>/dev/null | $SHA_CMD | cut -d' ' -f1) + if [ "$FIRST" = true ]; then FIRST=false else echo "," >> "$HASH_FILE" + echo "," >> "$HASH_NO_PC_FILE" fi printf ' "%s": "%s"' "$TYPE" "$HASH" >> "$HASH_FILE" + printf ' "%s": "%s"' "$TYPE" "$HASH_NO_PC" >> "$HASH_NO_PC_FILE" echo " 타입별 파일: ${TYPE_COUNT}개 추가" - echo " 해시: $HASH" + echo " 해시(전체): $HASH" + echo " 해시(pre-commit 제외): $HASH_NO_PC" done echo "" >> "$HASH_FILE" echo "}" >> "$HASH_FILE" +echo "" >> "$HASH_NO_PC_FILE" +echo "}" >> "$HASH_NO_PC_FILE" # 3. workflow-version.json 갱신 echo "" @@ -114,10 +127,14 @@ import json with open('$HASH_FILE', 'r') as f: hashes = json.load(f) +with open('$HASH_NO_PC_FILE', 'r') as f: + hashes_no_pc = json.load(f) + with open('workflow-version.json', 'r') as f: data = json.load(f) data['content_hashes'] = hashes +data['content_hashes_custom_precommit'] = hashes_no_pc with open('workflow-version.json', 'w') as f: json.dump(data, f, indent=2, ensure_ascii=False) @@ -126,6 +143,10 @@ with open('workflow-version.json', 'w') as f: print('content_hashes 갱신 완료:') for t, h in hashes.items(): print(f' {t}: {h}') +print() +print('content_hashes_custom_precommit 갱신 완료:') +for t, h in hashes_no_pc.items(): + print(f' {t}: {h}') " echo "" diff --git a/workflow-version.json b/workflow-version.json index 863448e..34a4d36 100644 --- a/workflow-version.json +++ b/workflow-version.json @@ -1,14 +1,37 @@ { - "version": "1.6.0", + "version": "1.6.1", "updated": "2026-03-06", "gitea_url": "https://gitea.gc-si.dev", "nexus_url": "https://nexus.gc-si.dev", "content_hashes": { - "java-maven": "1456f972fb250ba60fa6331d5fad81155c9992bf80947867e34a7178293513db", - "java-gradle": "0f7afc70942888cd2f98e01ef2eef0fcb815c0a68704d27b970a528081710d22", - "react-ts": "2a3a5b6ce9e2c6080c79b78be92c4967ed2418c3efff6082038b3f270c3b7e18" + "java-maven": "", + "java-gradle": "", + "react-ts": "" + }, + "content_hashes_custom_precommit": { + "java-maven": "", + "java-gradle": "", + "react-ts": "" }, "changes": [ + { + "version": "1.6.1", + "date": "2026-03-06", + "description": "custom_pre_commit 프로젝트 해시 불일치 해결", + "items": [ + "content_hashes_custom_precommit: pre-commit 제외 해시 필드 추가", + "push/mr/release Step 0.5: custom_pre_commit 플래그에 따라 해시 비교 분기", + "update-hash.sh: 두 세트 해시 동시 계산 (전체 + pre-commit 제외)", + "적용: 자동 (기존 v1.6.0 프로젝트는 /push 시 해시 차이로 자동 동기화)" + ], + "affected_files": [ + ".claude/skills/push/SKILL.md", + ".claude/skills/mr/SKILL.md", + ".claude/skills/release/SKILL.md", + "update-hash.sh", + "workflow-version.json" + ] + }, { "version": "1.6.0", "date": "2026-03-06", -- 2.45.2 From aaed4d31ea88cd424de0ca2985d1e7027e10c2fc Mon Sep 17 00:00:00 2001 From: htlee Date: Fri, 6 Mar 2026 15:31:13 +0900 Subject: [PATCH 10/11] =?UTF-8?q?chore:=20v1.6.1=20content=5Fhashes=20+=20?= =?UTF-8?q?content=5Fhashes=5Fcustom=5Fprecommit=20=EA=B0=B1=EC=8B=A0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- workflow-version.json | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/workflow-version.json b/workflow-version.json index 34a4d36..3559302 100644 --- a/workflow-version.json +++ b/workflow-version.json @@ -4,14 +4,14 @@ "gitea_url": "https://gitea.gc-si.dev", "nexus_url": "https://nexus.gc-si.dev", "content_hashes": { - "java-maven": "", - "java-gradle": "", - "react-ts": "" + "java-maven": "1901f5fc63056422b9b1f3e77d6bf1ba2d57f052c2a6d4c6ed76534602c160db", + "java-gradle": "ba49d863bf0eb5e0c5216584527a7170260bb9dd961b943f920ee47925d0159f", + "react-ts": "9c4a91614b576b314b10431edda3f411fd82d0a7e00a80c9116f70f64f34cde6" }, "content_hashes_custom_precommit": { - "java-maven": "", - "java-gradle": "", - "react-ts": "" + "java-maven": "97467e5ce10682e354e1946851e23e41eba445ee8b16ade2c78faa8287fd4f7d", + "java-gradle": "97467e5ce10682e354e1946851e23e41eba445ee8b16ade2c78faa8287fd4f7d", + "react-ts": "f3f062592541f3fb29591b2e107640e4a03765d674c5afbe6276af0e43c6b819" }, "changes": [ { -- 2.45.2 From 227259fc5cc6e852310ca19550a1970b8f6068cc Mon Sep 17 00:00:00 2001 From: htlee Date: Fri, 6 Mar 2026 15:33:11 +0900 Subject: [PATCH 11/11] =?UTF-8?q?docs:=20init-project=EC=97=90=20GITEA=5FT?= =?UTF-8?q?OKEN=20=EC=83=9D=EC=84=B1=20=EC=95=88=EB=82=B4=20=EC=A0=88?= =?UTF-8?q?=EC=B0=A8=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - Step 3.5: Gitea 토큰 생성 가이드 (GUI 권한 설정 포함) - 토큰 유효성 API 검증 + settings.local.json 자동 저장 - CLAUDE_BOT_TOKEN은 settings.json env에서 자동 배포 --- .claude/skills/init-project/SKILL.md | 80 ++++++++++++++++++++++++++++ 1 file changed, 80 insertions(+) diff --git a/.claude/skills/init-project/SKILL.md b/.claude/skills/init-project/SKILL.md index 1056c34..d0df8e5 100644 --- a/.claude/skills/init-project/SKILL.md +++ b/.claude/skills/init-project/SKILL.md @@ -49,6 +49,86 @@ curl -sf "${GITEA_URL}/gc/template-react-ts/raw/branch/develop/.editorconfig" ⚠️ 팀 규칙(.claude/rules/), 에이전트(.claude/agents/), 스킬 6종, 스크립트는 12단계(sync-team-workflow)에서 자동 다운로드된다. 여기서는 settings.json만 설정한다. +### 3.5. Gitea 토큰 설정 + +**CLAUDE_BOT_TOKEN** (팀 공용): `settings.json`의 `env` 필드에 이미 포함되어 있음 (3단계에서 설정됨). 별도 조치 불필요. + +**GITEA_TOKEN** (개인): `/push`, `/mr`, `/release` 등 Git 스킬에 필요한 개인 토큰. + +```bash +# 현재 GITEA_TOKEN 설정 여부 확인 +if [ -z "$GITEA_TOKEN" ]; then + echo "GITEA_TOKEN 미설정" +fi +``` + +**GITEA_TOKEN이 없는 경우**, 다음 안내를 **AskUserQuestion**으로 표시: + +**질문**: "GITEA_TOKEN이 설정되지 않았습니다. Gitea 개인 토큰을 생성하시겠습니까?" +- 옵션 1: 토큰 생성 안내 보기 (추천) +- 옵션 2: 이미 있음 (토큰 입력) +- 옵션 3: 나중에 하기 + +**토큰 생성 안내 선택 시**, 다음 내용을 표시: + +``` +📋 Gitea 토큰 생성 방법: + +1. 브라우저에서 접속: + https://gitea.gc-si.dev/user/settings/applications + +2. "Manage Access Tokens" 섹션에서 "Generate New Token" 클릭 + +3. 입력: + - Token Name: "claude-code" (자유롭게 지정) + - Repository and Organization Access: ✅ All (public, private, and limited) + +4. Select permissions (아래 4개만 설정, 나머지는 No Access 유지): + + ┌─────────────────┬──────────────────┬──────────────────────────────┐ + │ 항목 │ 권한 │ 용도 │ + ├─────────────────┼──────────────────┼──────────────────────────────┤ + │ issue │ Read and Write │ /fix-issue 이슈 조회/코멘트 │ + │ organization │ Read │ gc 조직 리포 접근 │ + │ repository │ Read and Write │ /push, /mr, /release API 호출 │ + │ user │ Read │ API 사용자 인증 확인 │ + └─────────────────┴──────────────────┴──────────────────────────────┘ + +5. "Generate Token" 클릭 → ⚠️ 토큰이 한 번만 표시됩니다! 반드시 복사하세요. +``` + +표시 후 **AskUserQuestion**: "생성한 토큰을 입력하세요" +- 옵션 1: 토큰 입력 (Other로 입력) +- 옵션 2: 나중에 하기 + +**토큰 입력 시**: + +1. Gitea API로 유효성 검증: +```bash +curl -sf "https://gitea.gc-si.dev/api/v1/user" \ + -H "Authorization: token <입력된 토큰>" +``` +- 성공: `✅ () 인증 확인` 출력 +- 실패: `❌ 토큰이 유효하지 않습니다. 다시 확인해주세요.` 출력 → 재입력 요청 + +2. `.claude/settings.local.json`에 저장 (이 파일은 .gitignore에 포함, 리포 커밋 안됨): +```json +{ + "env": { + "GITEA_TOKEN": "<입력된 토큰>" + } +} +``` + +기존 `settings.local.json`이 있으면 `env.GITEA_TOKEN`만 추가/갱신. + +**나중에 하기 선택 시**: 경고 표시 후 다음 단계로 진행: +``` +⚠️ GITEA_TOKEN 없이는 /push, /mr, /release 스킬을 사용할 수 없습니다. + 나중에 토큰을 생성하면 .claude/settings.local.json에 다음을 추가하세요: + { "env": { "GITEA_TOKEN": "your-token-here" } } +``` + ### 4. Hook 스크립트 설정 ⚠️ `.claude/scripts/` 스크립트 파일은 12단계(sync-team-workflow)에서 서버로부터 자동 다운로드된다. -- 2.45.2