Compare commits
6 Commits
feature/is
...
feature/is
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
85a59542f8 | ||
| 5830791355 | |||
|
|
b1610f14c5 | ||
| 1984065499 | |||
|
|
d912471d0e | ||
| 5f337e2ebc |
@@ -8,6 +8,11 @@
|
|||||||
- TASK: `TASK-...`
|
- TASK: `TASK-...`
|
||||||
- TEST: `TEST-...`
|
- TEST: `TEST-...`
|
||||||
|
|
||||||
|
## Ticket Stage
|
||||||
|
|
||||||
|
- Current stage: `Implemented` / `Integrated` / `Observed` / `Accepted`
|
||||||
|
- Previous stage evidence link:
|
||||||
|
|
||||||
## Main -> Verifier Directive Contract
|
## Main -> Verifier Directive Contract
|
||||||
|
|
||||||
- Scope: 대상 요구사항/코드/로그 경로
|
- Scope: 대상 요구사항/코드/로그 경로
|
||||||
@@ -30,9 +35,19 @@
|
|||||||
- [ ] `docs/commands.md`와 `docs/workflow.md` 트러블슈팅 선확인
|
- [ ] `docs/commands.md`와 `docs/workflow.md` 트러블슈팅 선확인
|
||||||
- [ ] `tea` 사용 (`gh` 미사용)
|
- [ ] `tea` 사용 (`gh` 미사용)
|
||||||
|
|
||||||
|
## Session Handover Gate
|
||||||
|
|
||||||
|
- [ ] `python3 scripts/session_handover_check.py --strict` 통과
|
||||||
|
- [ ] `workflow/session-handover.md` 최신 엔트리가 현재 브랜치/당일(UTC) 기준으로 갱신됨
|
||||||
|
- 최신 handover 엔트리 heading:
|
||||||
|
|
||||||
## Runtime Evidence
|
## Runtime Evidence
|
||||||
|
|
||||||
- 시스템 실제 구동 커맨드:
|
- 시스템 실제 구동 커맨드:
|
||||||
- 모니터링 로그 경로:
|
- 모니터링 로그 경로:
|
||||||
- 이상 징후/이슈 링크:
|
- 이상 징후/이슈 링크:
|
||||||
|
|
||||||
|
## Approval Gate
|
||||||
|
|
||||||
|
- [ ] Static Verifier approval comment linked
|
||||||
|
- [ ] Runtime Verifier approval comment linked
|
||||||
|
|||||||
38
.gitea/workflows/ci.yml
Normal file
38
.gitea/workflows/ci.yml
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
name: Gitea CI
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- main
|
||||||
|
- feature/**
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
test:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Set up Python
|
||||||
|
uses: actions/setup-python@v5
|
||||||
|
with:
|
||||||
|
python-version: "3.11"
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: pip install ".[dev]"
|
||||||
|
|
||||||
|
- name: Session handover gate
|
||||||
|
run: python3 scripts/session_handover_check.py --strict
|
||||||
|
|
||||||
|
- name: Validate governance assets
|
||||||
|
run: python3 scripts/validate_governance_assets.py
|
||||||
|
|
||||||
|
- name: Validate Ouroboros docs
|
||||||
|
run: python3 scripts/validate_ouroboros_docs.py
|
||||||
|
|
||||||
|
- name: Lint
|
||||||
|
run: ruff check src/ tests/
|
||||||
|
|
||||||
|
- name: Run tests with coverage
|
||||||
|
run: pytest -v --cov=src --cov-report=term-missing --cov-fail-under=80
|
||||||
3
.github/workflows/ci.yml
vendored
3
.github/workflows/ci.yml
vendored
@@ -21,6 +21,9 @@ jobs:
|
|||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: pip install ".[dev]"
|
run: pip install ".[dev]"
|
||||||
|
|
||||||
|
- name: Session handover gate
|
||||||
|
run: python3 scripts/session_handover_check.py --strict
|
||||||
|
|
||||||
- name: Validate governance assets
|
- name: Validate governance assets
|
||||||
run: python3 scripts/validate_governance_assets.py
|
run: python3 scripts/validate_governance_assets.py
|
||||||
|
|
||||||
|
|||||||
@@ -32,6 +32,16 @@ It is distinct from `docs/requirements-log.md`, which records **project/product
|
|||||||
(or in a dedicated policy doc) and reference it when working.
|
(or in a dedicated policy doc) and reference it when working.
|
||||||
- Keep entries short and concrete, with dates.
|
- Keep entries short and concrete, with dates.
|
||||||
|
|
||||||
|
5. **Session start handover gate**
|
||||||
|
- Before implementation/verification work, run `python3 scripts/session_handover_check.py --strict`.
|
||||||
|
- Keep `workflow/session-handover.md` updated with a same-day entry for the active branch.
|
||||||
|
- If the check fails, stop and fix handover artifacts first.
|
||||||
|
|
||||||
|
6. **Process-change-first execution gate**
|
||||||
|
- If process/governance change is required, merge the process ticket to the feature branch first.
|
||||||
|
- Do not start code/test edits for implementation tickets until process merge evidence is confirmed.
|
||||||
|
- Subagents must be constrained to read-only exploration until the process gate is satisfied.
|
||||||
|
|
||||||
## Change Control
|
## Change Control
|
||||||
|
|
||||||
- Changes to this file follow the same workflow as code changes.
|
- Changes to this file follow the same workflow as code changes.
|
||||||
@@ -50,3 +60,10 @@ It is distinct from `docs/requirements-log.md`, which records **project/product
|
|||||||
|
|
||||||
- All agents must pre-read `docs/commands.md` and `docs/workflow.md` troubleshooting before running Gitea issue/PR/comment commands.
|
- All agents must pre-read `docs/commands.md` and `docs/workflow.md` troubleshooting before running Gitea issue/PR/comment commands.
|
||||||
- `gh` CLI is prohibited for repository ticket/PR operations; use `tea` (or documented Gitea API fallback only).
|
- `gh` CLI is prohibited for repository ticket/PR operations; use `tea` (or documented Gitea API fallback only).
|
||||||
|
- Session start must pass `python3 scripts/session_handover_check.py --strict`, with branch-matched entry in `workflow/session-handover.md`.
|
||||||
|
|
||||||
|
### 2026-02-27
|
||||||
|
|
||||||
|
- Apply process-change-first as an execution gate: process ticket must be merged before implementation ticket coding.
|
||||||
|
- Handover entry must record concrete `next_ticket` and `process_gate_checked`; placeholders are not allowed in strict gate.
|
||||||
|
- Before process merge confirmation, all subagent tasks must remain read-only (analysis only).
|
||||||
|
|||||||
@@ -11,6 +11,16 @@
|
|||||||
- 기본 도구는 `tea`이며, `tea` 미지원 케이스만 Gitea API를 fallback으로 사용한다.
|
- 기본 도구는 `tea`이며, `tea` 미지원 케이스만 Gitea API를 fallback으로 사용한다.
|
||||||
- 실행 전 `docs/workflow.md`의 `Gitea CLI Formatting Troubleshooting`을 반드시 확인한다.
|
- 실행 전 `docs/workflow.md`의 `Gitea CLI Formatting Troubleshooting`을 반드시 확인한다.
|
||||||
|
|
||||||
|
## Session Handover Preflight (Mandatory)
|
||||||
|
|
||||||
|
- 세션 시작 직후(코드 변경 전) 아래 명령을 먼저 실행한다.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
python3 scripts/session_handover_check.py --strict
|
||||||
|
```
|
||||||
|
|
||||||
|
- 실패 시 `workflow/session-handover.md` 최신 엔트리를 보강한 뒤 재실행한다.
|
||||||
|
|
||||||
### tea CLI (Gitea Command Line Tool)
|
### tea CLI (Gitea Command Line Tool)
|
||||||
|
|
||||||
#### ❌ TTY Error - Interactive Confirmation Fails
|
#### ❌ TTY Error - Interactive Confirmation Fails
|
||||||
@@ -150,6 +160,9 @@ python -m src.main --mode=paper --dashboard
|
|||||||
# Runtime verification monitor (NOT_OBSERVED detection)
|
# Runtime verification monitor (NOT_OBSERVED detection)
|
||||||
bash scripts/runtime_verify_monitor.sh
|
bash scripts/runtime_verify_monitor.sh
|
||||||
|
|
||||||
|
# Session handover gate (must pass before implementation)
|
||||||
|
python3 scripts/session_handover_check.py --strict
|
||||||
|
|
||||||
# Follow runtime verification log
|
# Follow runtime verification log
|
||||||
tail -f data/overnight/runtime_verify_*.log
|
tail -f data/overnight/runtime_verify_*.log
|
||||||
|
|
||||||
|
|||||||
@@ -121,6 +121,7 @@ Control checks:
|
|||||||
- Verifier가 `Coverage Matrix`(`REQ/TASK/TEST` x `PASS/FAIL/NOT_OBSERVED`) 첨부
|
- Verifier가 `Coverage Matrix`(`REQ/TASK/TEST` x `PASS/FAIL/NOT_OBSERVED`) 첨부
|
||||||
- `NOT_OBSERVED` 항목 수가 0인지 확인(0이 아니면 Gate 실패)
|
- `NOT_OBSERVED` 항목 수가 0인지 확인(0이 아니면 Gate 실패)
|
||||||
- Runtime Verifier가 스테이징/실운영 모니터링 계획 승인
|
- Runtime Verifier가 스테이징/실운영 모니터링 계획 승인
|
||||||
|
- 정적 Verifier 승인 + Runtime Verifier 승인 2개 모두 확인
|
||||||
- 산출물: 수용 승인 레코드
|
- 산출물: 수용 승인 레코드
|
||||||
|
|
||||||
### Phase 5: Release and Post-Release Control
|
### Phase 5: Release and Post-Release Control
|
||||||
@@ -160,6 +161,15 @@ TPM 티켓 운영 규칙:
|
|||||||
- PM/TPM/Dev/Reviewer/Verifier/Runtime Verifier는 주요 의사결정 시점마다 PR 코멘트를 남겨 결정 근거를 추적 가능 상태로 유지한다.
|
- PM/TPM/Dev/Reviewer/Verifier/Runtime Verifier는 주요 의사결정 시점마다 PR 코멘트를 남겨 결정 근거를 추적 가능 상태로 유지한다.
|
||||||
- PM/TPM/Dev/Reviewer/Verifier/Runtime Verifier는 이슈/PR/코멘트 조작 전에 `docs/commands.md`와 `docs/workflow.md`의 Gitea 트러블슈팅 섹션을 선참조해야 한다.
|
- PM/TPM/Dev/Reviewer/Verifier/Runtime Verifier는 이슈/PR/코멘트 조작 전에 `docs/commands.md`와 `docs/workflow.md`의 Gitea 트러블슈팅 섹션을 선참조해야 한다.
|
||||||
- 저장소 협업에서 GitHub CLI(`gh`) 사용은 금지하며, Gitea 작업은 `tea`(필요 시 문서화된 API fallback)만 허용한다.
|
- 저장소 협업에서 GitHub CLI(`gh`) 사용은 금지하며, Gitea 작업은 `tea`(필요 시 문서화된 API fallback)만 허용한다.
|
||||||
|
- 재발 방지/운영 규칙 변경이 합의되면, 기능 구현 이전에 process 티켓을 먼저 생성/머지해야 한다.
|
||||||
|
- process 티켓 미반영 상태에서 구현 티켓 진행 시 TPM이 즉시 `BLOCKED` 처리한다.
|
||||||
|
|
||||||
|
티켓 성숙도 단계 (Mandatory):
|
||||||
|
- `Implemented`: 코드/문서 변경 완료
|
||||||
|
- `Integrated`: 호출 경로/파이프라인 연결 확인
|
||||||
|
- `Observed`: 런타임/실행 증적 확보
|
||||||
|
- `Accepted`: Verifier + Runtime Verifier 승인 완료
|
||||||
|
- 단계는 순차 전진만 허용되며, 단계 점프는 허용되지 않는다.
|
||||||
|
|
||||||
브랜치 운영 규칙:
|
브랜치 운영 규칙:
|
||||||
- TPM은 각 티켓에 대해 `ticket temp branch -> program feature branch` PR 경로를 지정한다.
|
- TPM은 각 티켓에 대해 `ticket temp branch -> program feature branch` PR 경로를 지정한다.
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ Doc-ID: DOC-OPS-002
|
|||||||
Version: 1.0.0
|
Version: 1.0.0
|
||||||
Status: active
|
Status: active
|
||||||
Owner: tpm
|
Owner: tpm
|
||||||
Updated: 2026-02-26
|
Updated: 2026-02-27
|
||||||
-->
|
-->
|
||||||
|
|
||||||
# 저장소 강제 설정 체크리스트
|
# 저장소 강제 설정 체크리스트
|
||||||
@@ -49,6 +49,7 @@ Updated: 2026-02-26
|
|||||||
- 이슈 연결(`Closes #N`) 존재
|
- 이슈 연결(`Closes #N`) 존재
|
||||||
- PR 본문에 `REQ-*`, `TASK-*`, `TEST-*` 매핑 표 존재
|
- PR 본문에 `REQ-*`, `TASK-*`, `TEST-*` 매핑 표 존재
|
||||||
- Main -> Verifier Directive Contract(범위/방법/합격/실패/미관측/증적 형식) 기재
|
- Main -> Verifier Directive Contract(범위/방법/합격/실패/미관측/증적 형식) 기재
|
||||||
|
- process-change-first 대상이면 process 티켓 PR이 선머지됨
|
||||||
- `src/core/risk_manager.py` 변경 없음
|
- `src/core/risk_manager.py` 변경 없음
|
||||||
- 주요 의사결정 체크포인트(DCP-01~04) 중 해당 단계 Main Agent 확인 기록 존재
|
- 주요 의사결정 체크포인트(DCP-01~04) 중 해당 단계 Main Agent 확인 기록 존재
|
||||||
- 주요 의사결정(리뷰 지적/수정 합의/검증 승인)에 대한 에이전트 PR 코멘트 존재
|
- 주요 의사결정(리뷰 지적/수정 합의/검증 승인)에 대한 에이전트 PR 코멘트 존재
|
||||||
@@ -57,11 +58,14 @@ Updated: 2026-02-26
|
|||||||
자동 점검:
|
자동 점검:
|
||||||
- 문서 검증 스크립트 통과
|
- 문서 검증 스크립트 통과
|
||||||
- 테스트 통과
|
- 테스트 통과
|
||||||
|
- `python3 scripts/session_handover_check.py --strict` 통과
|
||||||
- 개발 완료 시 시스템 구동/모니터링 증적 코멘트 존재
|
- 개발 완료 시 시스템 구동/모니터링 증적 코멘트 존재
|
||||||
- 이슈/PR 조작 전에 `docs/commands.md` 및 `docs/workflow.md` 트러블슈팅 확인 코멘트 존재
|
- 이슈/PR 조작 전에 `docs/commands.md` 및 `docs/workflow.md` 트러블슈팅 확인 코멘트 존재
|
||||||
- `gh` CLI 미사용, `tea` 사용 증적 존재
|
- `gh` CLI 미사용, `tea` 사용 증적 존재
|
||||||
- Verifier `Coverage Matrix` 첨부(PASS/FAIL/NOT_OBSERVED)
|
- Verifier `Coverage Matrix` 첨부(PASS/FAIL/NOT_OBSERVED)
|
||||||
- `NOT_OBSERVED` 항목 0 확인(0이 아니면 머지 금지)
|
- `NOT_OBSERVED` 항목 0 확인(0이 아니면 머지 금지)
|
||||||
|
- 티켓 단계 기록(`Implemented` -> `Integrated` -> `Observed` -> `Accepted`) 존재
|
||||||
|
- 정적 Verifier 승인 + Runtime Verifier 승인 2개 확인
|
||||||
|
|
||||||
## 5) 감사 추적
|
## 5) 감사 추적
|
||||||
|
|
||||||
|
|||||||
@@ -30,6 +30,19 @@ Gitea 이슈/PR/코멘트 작업 전에 모든 에이전트는 아래를 먼저
|
|||||||
- `tea` 실패 시 동일 명령 재시도 전에 원인/수정사항을 PR 코멘트에 남긴다.
|
- `tea` 실패 시 동일 명령 재시도 전에 원인/수정사항을 PR 코멘트에 남긴다.
|
||||||
- 필요한 경우에만 Gitea API(`localhost:3000`)를 fallback으로 사용한다.
|
- 필요한 경우에만 Gitea API(`localhost:3000`)를 fallback으로 사용한다.
|
||||||
|
|
||||||
|
## Session Handover Gate (Mandatory)
|
||||||
|
|
||||||
|
새 세션에서 구현/검증을 시작하기 전에 아래를 선행해야 한다.
|
||||||
|
|
||||||
|
1. `docs/workflow.md`, `docs/commands.md`, `docs/agent-constraints.md` 재확인
|
||||||
|
2. `workflow/session-handover.md`에 최신 세션 엔트리 추가
|
||||||
|
3. `python3 scripts/session_handover_check.py --strict` 통과 확인
|
||||||
|
|
||||||
|
강제 규칙:
|
||||||
|
- handover check 실패 상태에서 코드 수정/이슈 상태 전이/PR 생성 금지
|
||||||
|
- 최신 handover 엔트리는 현재 작업 브랜치를 명시해야 한다
|
||||||
|
- 최신 handover 엔트리는 당일(UTC) 날짜를 포함해야 한다
|
||||||
|
|
||||||
## Branch Strategy (Mandatory)
|
## Branch Strategy (Mandatory)
|
||||||
|
|
||||||
- Team operation default branch is the **program feature branch**, not `main`.
|
- Team operation default branch is the **program feature branch**, not `main`.
|
||||||
@@ -167,6 +180,41 @@ Use `run_in_background=True` for independent tasks that don't block subsequent w
|
|||||||
- `NOT_OBSERVED`는 운영상 `FAIL`과 동일하게 처리
|
- `NOT_OBSERVED`는 운영상 `FAIL`과 동일하게 처리
|
||||||
- `NOT_OBSERVED`가 하나라도 있으면 승인/머지 금지
|
- `NOT_OBSERVED`가 하나라도 있으면 승인/머지 금지
|
||||||
|
|
||||||
|
### Process-Change-First Rule (Mandatory)
|
||||||
|
|
||||||
|
재발 방지/운영 규칙 변경이 결정되면, 기능 구현 티켓보다 먼저 서버(feature branch)에 반영해야 한다.
|
||||||
|
|
||||||
|
- 순서: `process ticket merge` -> `implementation ticket start`
|
||||||
|
- process ticket 미반영 상태에서 기능 티켓 코딩/머지 금지
|
||||||
|
- 세션 전환 시에도 동일 규칙 유지
|
||||||
|
|
||||||
|
### Implementation Start Gate (Mandatory)
|
||||||
|
|
||||||
|
구현 티켓을 시작하기 전에 아래 3개를 모두 만족해야 한다.
|
||||||
|
|
||||||
|
1. `process ticket merge` 증적 확인 (feature branch 반영 커밋/PR)
|
||||||
|
2. `workflow/session-handover.md` 최신 엔트리에 `next_ticket`과 `process_gate_checked` 기록
|
||||||
|
3. `python3 scripts/session_handover_check.py --strict` 통과
|
||||||
|
|
||||||
|
강제 규칙:
|
||||||
|
- 위 3개 중 하나라도 불충족이면 코드/테스트 수정 금지
|
||||||
|
- 서브에이전트 지시도 동일하게 제한한다 (`process merged 확인 전 read-only 탐색만 허용`)
|
||||||
|
- 성급 착수 발견 시 구현 작업을 즉시 중단하고 handover/proces gate부터 복구한다
|
||||||
|
|
||||||
|
### Ticket Maturity Stages (Mandatory)
|
||||||
|
|
||||||
|
모든 티켓은 아래 4단계를 순서대로 통과해야 한다.
|
||||||
|
|
||||||
|
1. `Implemented`: 코드/문서 변경 완료
|
||||||
|
2. `Integrated`: 호출 경로/파이프라인 연결 완료
|
||||||
|
3. `Observed`: 런타임/실행 증적 확보 완료
|
||||||
|
4. `Accepted`: 정적 Verifier + Runtime Verifier 승인 완료
|
||||||
|
|
||||||
|
강제 규칙:
|
||||||
|
- 단계 점프 금지 (예: Implemented -> Accepted 금지)
|
||||||
|
- `Observed` 전에는 완료 선언 금지
|
||||||
|
- `Accepted` 전에는 머지 금지
|
||||||
|
|
||||||
## Code Review Checklist
|
## Code Review Checklist
|
||||||
|
|
||||||
**CRITICAL: Every PR review MUST verify plan-implementation consistency.**
|
**CRITICAL: Every PR review MUST verify plan-implementation consistency.**
|
||||||
@@ -204,3 +252,6 @@ Before approving any PR, the reviewer (human or agent) must check ALL of the fol
|
|||||||
- [ ] `gh` 명령을 사용하지 않고 `tea`(또는 허용된 Gitea API fallback)만 사용했다
|
- [ ] `gh` 명령을 사용하지 않고 `tea`(또는 허용된 Gitea API fallback)만 사용했다
|
||||||
- [ ] Main -> Verifier 지시가 Directive Contract 6개 항목을 모두 포함한다
|
- [ ] Main -> Verifier 지시가 Directive Contract 6개 항목을 모두 포함한다
|
||||||
- [ ] Verifier 결과에 `Coverage Matrix`(PASS/FAIL/NOT_OBSERVED)가 있고, `NOT_OBSERVED=0`이다
|
- [ ] Verifier 결과에 `Coverage Matrix`(PASS/FAIL/NOT_OBSERVED)가 있고, `NOT_OBSERVED=0`이다
|
||||||
|
- [ ] Process-change-first 대상이면 해당 process PR이 먼저 머지되었다
|
||||||
|
- [ ] 티켓 단계가 `Implemented -> Integrated -> Observed -> Accepted` 순서로 기록되었다
|
||||||
|
- [ ] 정적 Verifier와 Runtime Verifier 승인 코멘트가 모두 존재한다
|
||||||
|
|||||||
146
scripts/session_handover_check.py
Executable file
146
scripts/session_handover_check.py
Executable file
@@ -0,0 +1,146 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
"""Session handover preflight gate.
|
||||||
|
|
||||||
|
This script enforces a minimal handover record per working branch so that
|
||||||
|
new sessions cannot start implementation without reading the required docs
|
||||||
|
and recording current intent.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import argparse
|
||||||
|
import subprocess
|
||||||
|
import sys
|
||||||
|
from datetime import UTC, datetime
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
REQUIRED_DOCS = (
|
||||||
|
Path("docs/workflow.md"),
|
||||||
|
Path("docs/commands.md"),
|
||||||
|
Path("docs/agent-constraints.md"),
|
||||||
|
)
|
||||||
|
HANDOVER_LOG = Path("workflow/session-handover.md")
|
||||||
|
|
||||||
|
|
||||||
|
def _run_git(*args: str) -> str:
|
||||||
|
try:
|
||||||
|
return (
|
||||||
|
subprocess.check_output(["git", *args], stderr=subprocess.DEVNULL)
|
||||||
|
.decode("utf-8")
|
||||||
|
.strip()
|
||||||
|
)
|
||||||
|
except Exception:
|
||||||
|
return ""
|
||||||
|
|
||||||
|
|
||||||
|
def _current_branch() -> str:
|
||||||
|
branch = _run_git("branch", "--show-current")
|
||||||
|
if branch:
|
||||||
|
return branch
|
||||||
|
return _run_git("rev-parse", "--abbrev-ref", "HEAD")
|
||||||
|
|
||||||
|
|
||||||
|
def _latest_entry(text: str) -> str:
|
||||||
|
chunks = text.split("\n### ")
|
||||||
|
if not chunks:
|
||||||
|
return ""
|
||||||
|
if chunks[0].startswith("### "):
|
||||||
|
chunks[0] = chunks[0][4:]
|
||||||
|
latest = chunks[-1].strip()
|
||||||
|
if not latest:
|
||||||
|
return ""
|
||||||
|
if not latest.startswith("### "):
|
||||||
|
latest = f"### {latest}"
|
||||||
|
return latest
|
||||||
|
|
||||||
|
|
||||||
|
def _check_required_files(errors: list[str]) -> None:
|
||||||
|
for path in REQUIRED_DOCS:
|
||||||
|
if not path.exists():
|
||||||
|
errors.append(f"missing required document: {path}")
|
||||||
|
if not HANDOVER_LOG.exists():
|
||||||
|
errors.append(f"missing handover log: {HANDOVER_LOG}")
|
||||||
|
|
||||||
|
|
||||||
|
def _check_handover_entry(
|
||||||
|
*,
|
||||||
|
branch: str,
|
||||||
|
strict: bool,
|
||||||
|
errors: list[str],
|
||||||
|
) -> None:
|
||||||
|
if not HANDOVER_LOG.exists():
|
||||||
|
return
|
||||||
|
text = HANDOVER_LOG.read_text(encoding="utf-8")
|
||||||
|
latest = _latest_entry(text)
|
||||||
|
if not latest:
|
||||||
|
errors.append("handover log has no session entry")
|
||||||
|
return
|
||||||
|
|
||||||
|
required_tokens = (
|
||||||
|
"- branch:",
|
||||||
|
"- docs_checked:",
|
||||||
|
"- open_issues_reviewed:",
|
||||||
|
"- next_ticket:",
|
||||||
|
"- process_gate_checked:",
|
||||||
|
)
|
||||||
|
for token in required_tokens:
|
||||||
|
if token not in latest:
|
||||||
|
errors.append(f"latest handover entry missing token: {token}")
|
||||||
|
|
||||||
|
if strict:
|
||||||
|
today_utc = datetime.now(UTC).date().isoformat()
|
||||||
|
if today_utc not in latest:
|
||||||
|
errors.append(
|
||||||
|
f"latest handover entry must contain today's UTC date ({today_utc})"
|
||||||
|
)
|
||||||
|
branch_token = f"- branch: {branch}"
|
||||||
|
if branch_token not in latest:
|
||||||
|
errors.append(
|
||||||
|
"latest handover entry must target current branch "
|
||||||
|
f"({branch_token})"
|
||||||
|
)
|
||||||
|
if "- next_ticket: #TBD" in latest:
|
||||||
|
errors.append("latest handover entry must not use placeholder next_ticket (#TBD)")
|
||||||
|
if "merged_to_feature_branch=no" in latest:
|
||||||
|
errors.append(
|
||||||
|
"process gate indicates not merged; implementation must stay blocked "
|
||||||
|
"(merged_to_feature_branch=no)"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def main() -> int:
|
||||||
|
parser = argparse.ArgumentParser(
|
||||||
|
description="Validate session handover gate requirements."
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
"--strict",
|
||||||
|
action="store_true",
|
||||||
|
help="Enforce today-date and current-branch match on latest handover entry.",
|
||||||
|
)
|
||||||
|
args = parser.parse_args()
|
||||||
|
|
||||||
|
errors: list[str] = []
|
||||||
|
_check_required_files(errors)
|
||||||
|
|
||||||
|
branch = _current_branch()
|
||||||
|
if not branch:
|
||||||
|
errors.append("cannot resolve current git branch")
|
||||||
|
elif branch in {"main", "master"}:
|
||||||
|
errors.append(f"working branch must not be {branch}")
|
||||||
|
|
||||||
|
_check_handover_entry(branch=branch, strict=args.strict, errors=errors)
|
||||||
|
|
||||||
|
if errors:
|
||||||
|
print("[FAIL] session handover check failed")
|
||||||
|
for err in errors:
|
||||||
|
print(f"- {err}")
|
||||||
|
return 1
|
||||||
|
|
||||||
|
print("[OK] session handover check passed")
|
||||||
|
print(f"[OK] branch={branch}")
|
||||||
|
print(f"[OK] handover_log={HANDOVER_LOG}")
|
||||||
|
return 0
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
sys.exit(main())
|
||||||
@@ -22,6 +22,10 @@ def main() -> int:
|
|||||||
|
|
||||||
pr_template = Path(".gitea/PULL_REQUEST_TEMPLATE.md")
|
pr_template = Path(".gitea/PULL_REQUEST_TEMPLATE.md")
|
||||||
issue_template = Path(".gitea/ISSUE_TEMPLATE/runtime_verification.md")
|
issue_template = Path(".gitea/ISSUE_TEMPLATE/runtime_verification.md")
|
||||||
|
workflow_doc = Path("docs/workflow.md")
|
||||||
|
commands_doc = Path("docs/commands.md")
|
||||||
|
handover_script = Path("scripts/session_handover_check.py")
|
||||||
|
handover_log = Path("workflow/session-handover.md")
|
||||||
|
|
||||||
must_contain(
|
must_contain(
|
||||||
pr_template,
|
pr_template,
|
||||||
@@ -32,6 +36,8 @@ def main() -> int:
|
|||||||
"NOT_OBSERVED",
|
"NOT_OBSERVED",
|
||||||
"tea",
|
"tea",
|
||||||
"gh",
|
"gh",
|
||||||
|
"Session Handover Gate",
|
||||||
|
"session_handover_check.py --strict",
|
||||||
],
|
],
|
||||||
errors,
|
errors,
|
||||||
)
|
)
|
||||||
@@ -45,6 +51,35 @@ def main() -> int:
|
|||||||
],
|
],
|
||||||
errors,
|
errors,
|
||||||
)
|
)
|
||||||
|
must_contain(
|
||||||
|
workflow_doc,
|
||||||
|
[
|
||||||
|
"Session Handover Gate (Mandatory)",
|
||||||
|
"session_handover_check.py --strict",
|
||||||
|
],
|
||||||
|
errors,
|
||||||
|
)
|
||||||
|
must_contain(
|
||||||
|
commands_doc,
|
||||||
|
[
|
||||||
|
"Session Handover Preflight (Mandatory)",
|
||||||
|
"session_handover_check.py --strict",
|
||||||
|
],
|
||||||
|
errors,
|
||||||
|
)
|
||||||
|
must_contain(
|
||||||
|
handover_log,
|
||||||
|
[
|
||||||
|
"Session Handover Log",
|
||||||
|
"- branch:",
|
||||||
|
"- docs_checked:",
|
||||||
|
"- open_issues_reviewed:",
|
||||||
|
"- next_ticket:",
|
||||||
|
],
|
||||||
|
errors,
|
||||||
|
)
|
||||||
|
if not handover_script.exists():
|
||||||
|
errors.append(f"missing file: {handover_script}")
|
||||||
|
|
||||||
if errors:
|
if errors:
|
||||||
print("[FAIL] governance asset validation failed")
|
print("[FAIL] governance asset validation failed")
|
||||||
@@ -58,4 +93,3 @@ def main() -> int:
|
|||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
sys.exit(main())
|
sys.exit(main())
|
||||||
|
|
||||||
|
|||||||
43
workflow/session-handover.md
Normal file
43
workflow/session-handover.md
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
# Session Handover Log
|
||||||
|
|
||||||
|
목적: 세션 시작 시 인수인계 확인을 기록하고, 구현/검증 작업 시작 전에 공통 컨텍스트를 강제한다.
|
||||||
|
|
||||||
|
작성 규칙:
|
||||||
|
- 세션 시작마다 최신 엔트리를 맨 아래에 추가한다.
|
||||||
|
- `docs/workflow.md`, `docs/commands.md`, `docs/agent-constraints.md`를 먼저 확인한 뒤 기록한다.
|
||||||
|
- 각 엔트리는 현재 작업 브랜치 기준으로 작성한다.
|
||||||
|
|
||||||
|
템플릿:
|
||||||
|
|
||||||
|
```md
|
||||||
|
### YYYY-MM-DD | session=<id or short label>
|
||||||
|
- branch: <current-branch>
|
||||||
|
- docs_checked: docs/workflow.md, docs/commands.md, docs/agent-constraints.md
|
||||||
|
- open_issues_reviewed: #...
|
||||||
|
- next_ticket: #...
|
||||||
|
- process_gate_checked: process_ticket=#..., merged_to_feature_branch=yes|no|n/a
|
||||||
|
- risks_or_notes: ...
|
||||||
|
```
|
||||||
|
|
||||||
|
### 2026-02-27 | session=handover-gate-bootstrap
|
||||||
|
- branch: feature/v3-session-policy-stream
|
||||||
|
- docs_checked: docs/workflow.md, docs/commands.md, docs/agent-constraints.md
|
||||||
|
- open_issues_reviewed: #304, #305, #306
|
||||||
|
- next_ticket: #304
|
||||||
|
- risks_or_notes: 세션 시작 게이트를 문서/스크립트/CI로 강제 적용
|
||||||
|
|
||||||
|
### 2026-02-27 | session=codex-handover-start
|
||||||
|
- branch: feature/v3-session-policy-stream
|
||||||
|
- docs_checked: docs/workflow.md, docs/commands.md, docs/agent-constraints.md
|
||||||
|
- open_issues_reviewed: #306, #308, #309
|
||||||
|
- next_ticket: #304
|
||||||
|
- process_gate_checked: process_ticket=#306,#308 merged_to_feature_branch=yes
|
||||||
|
- risks_or_notes: 미추적 로컬 파일 존재(문서/DB/lock)로 커밋 범위 분리 필요
|
||||||
|
|
||||||
|
### 2026-02-27 | session=codex-process-gate-hardening
|
||||||
|
- branch: feature/issue-304-runtime-staged-exit-semantics
|
||||||
|
- docs_checked: docs/workflow.md, docs/commands.md, docs/agent-constraints.md
|
||||||
|
- open_issues_reviewed: #304, #305
|
||||||
|
- next_ticket: #304
|
||||||
|
- process_gate_checked: process_ticket=#306,#308 merged_to_feature_branch=yes
|
||||||
|
- risks_or_notes: process-change-first 실행 게이트를 문서+스크립트로 강화
|
||||||
Reference in New Issue
Block a user