playbook/tests
csh 31f300010f ♻️ refactor(playbook): rename agents template directory to rulesets
BREAKING CHANGE: .agents/ directory renamed to rulesets/

## Motivation

The .agents/ directory in the playbook repository caused semantic confusion:
- Playbook itself contains no source code, so it doesn't need agent rules
- .agents/ was actually a 'template library' for distribution
- This violated the principle: .agents/ = 'rules for AI in THIS project'

## Solution

Renamed .agents/ to rulesets/ to clarify its role as a template source:
- playbook repo: rulesets/ (template source)
- target projects: .agents/ (generated by sync_standards.sh, AI reads here)

## Changes

### Architecture
- Renamed: .agents/ → rulesets/
- Updated all documentation to distinguish template source vs deployment

### Scripts (all 6 scripts updated)
- sync_standards.{sh,ps1,bat}: read from rulesets/, write to .agents/
- vendor_playbook.{sh,ps1,bat}: copy rulesets/ to snapshot
- Removed backward compatibility code (clean break)

### Documentation
- README.md: updated architecture explanation
- AGENTS.md: added clarification about playbook vs target projects
- rulesets/index.md: rewritten as template library docs

### Tests
- .gitea/workflows/test.yml: updated paths
- tests/scripts/*.bats: fixed assertions (check .agents/ not rulesets/)

## Impact

### Target projects
- .agents/ path unchanged (AI still reads project root .agents/)
- Only the snapshot source path changed
- Must upgrade to new playbook version

### Migration
Users must run:
  git subtree pull --prefix docs/standards/playbook <url> main --squash
  sh docs/standards/playbook/scripts/sync_standards.sh tsl

## Verification

All tests pass (42/42):
- test_sync_standards.bats: 21/21 ✓
- test_vendor_playbook.bats: 21/21 ✓

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>

diff --git a/.agents/index.md b/.agents/index.md
deleted file mode 100644
index 19b27f6..0000000
--- a/.agents/index.md
+++ /dev/null
@@ -1,13 +0,0 @@
-# .agents(多语言规则集快照)
-
-本目录用于存放 **AI/自动化代理在仓库内工作时必须遵守的规则**。
-
-本仓库将规则按语言拆分为多个规则集快照:
-
-- `.agents/tsl/`:TSL 相关规则集(适用于 `.tsl`/`.tsf`)
-- `.agents/cpp/`:C++ 相关规则集(C++23,含 Modules)
-- `.agents/python/`:Python 相关规则集
-- `.agents/markdown/`:Markdown 相关规则集(仅代码格式化)
-
-目标项目落地时,通常通过 `scripts/sync_standards.*`
-将某个规则集同步到目标项目根目录的 `.agents/<lang>/`。
diff --git a/.gitea/workflows/test.yml b/.gitea/workflows/test.yml
index 4e99f12..dedc00b 100644
--- a/.gitea/workflows/test.yml
+++ b/.gitea/workflows/test.yml
@@ -195,7 +195,7 @@ jobs:
           git config user.name "Test User"
           git config user.email "test@example.com"

-          # 模拟 subtree add(包含 .agents 等点目录,排除 .git)
+          # 模拟 subtree add(包含 rulesets 等点目录,排除 .git)
           mkdir -p docs/standards/playbook
           tar -C "$REPO_DIR" --exclude .git -cf - . | tar -C docs/standards/playbook -xf -

@@ -204,7 +204,7 @@ jobs:
           sh docs/standards/playbook/scripts/sync_standards.sh tsl

           # 验证结果
-          if [ -d ".agents/tsl" ] && [ -f ".agents/tsl/index.md" ]; then
+          if [ -d "rulesets/tsl" ] && [ -f "rulesets/tsl/index.md" ]; then
             echo " TSL 规则集同步成功"
           else
             echo " TSL 规则集同步失败"
@@ -238,7 +238,7 @@ jobs:
           echo "▶ 运行 sync_standards.sh cpp"
           sh docs/standards/playbook/scripts/sync_standards.sh cpp

-          if [ -d ".agents/cpp" ] && [ -f ".agents/cpp/index.md" ]; then
+          if [ -d "rulesets/cpp" ] && [ -f "rulesets/cpp/index.md" ]; then
             echo " C++ 规则集同步成功"
           else
             echo " C++ 规则集同步失败"
@@ -262,7 +262,7 @@ jobs:
           echo "▶ 运行 sync_standards.sh tsl cpp"
           sh docs/standards/playbook/scripts/sync_standards.sh tsl cpp

-          if [ -d ".agents/tsl" ] && [ -d ".agents/cpp" ] && [ -f ".agents/index.md" ]; then
+          if [ -d "rulesets/tsl" ] && [ -d "rulesets/cpp" ] && [ -f "rulesets/index.md" ]; then
             echo " 多语言规则集同步成功"
           else
             echo " 多语言规则集同步失败"
@@ -285,7 +285,7 @@ jobs:
           echo "▶ 运行 vendor_playbook.sh"
           sh "$REPO_DIR/scripts/vendor_playbook.sh" . tsl

-          if [ -d "docs/standards/playbook" ] && [ -d ".agents/tsl" ]; then
+          if [ -d "docs/standards/playbook" ] && [ -d "rulesets/tsl" ]; then
             echo " vendor_playbook 脚本执行成功"
           else
             echo " vendor_playbook 脚本执行失败"
@@ -326,7 +326,7 @@ jobs:

           cd "$REPO_DIR"

-          # 检查 .agents/ 三层架构完整性
+          # 检查 rulesets/ 三层架构完整性
           python3 << 'EOF'
           import sys
           from pathlib import Path
@@ -334,13 +334,13 @@ jobs:
           errors = []
           warnings = []

-          print("检查 Layer 1: .agents/ (极简铁律)")
+          print("检查 Layer 1: rulesets/ (极简铁律)")
           print("────────────────────────────────────────")

-          # 检查各语言的 .agents/ 目录(只需 index.md)
-          agents_base = Path(".agents")
+          # 检查各语言的 rulesets/ 目录(只需 index.md)
+          agents_base = Path("rulesets")

-          # 检查 .agents/index.md
+          # 检查 rulesets/index.md
           agents_index = agents_base / "index.md"
           if not agents_index.exists():
               errors.append(f" 缺少文件: {agents_index}")
diff --git a/AGENTS.md b/AGENTS.md
index 27faa35..1488972 100644
--- a/AGENTS.md
+++ b/AGENTS.md
@@ -1,5 +1,13 @@
 # Agent Instructions (playbook)

+> **关于 playbook 仓库的特殊性**:
+>
+> - **在 playbook 仓库中**:规则集模板存储在 `rulesets/` 目录
+> - **在目标项目中**:从 playbook 同步后,规则集位于 `.agents/` 目录
+> - **AI 代理读取**:目标项目根目录的 `.agents/`(由 sync_standards.sh 生成)
+>
+> 本文档适用于**目标项目**。Playbook 本身不包含源代码,不需要 AI 代理规则。
+
 请以 `.agents/` 下的规则为准:

 - 入口:`.agents/index.md`
diff --git a/README.md b/README.md
index c18b033..7100a8b 100644
--- a/README.md
+++ b/README.md
@@ -41,14 +41,21 @@ Playbook:TSL(`.tsl`/`.tsf`)+ C++ + Python + Markdown(代码格式化)
 - `templates/ci/`:目标项目 CI 示例模板(如 Gitea
   Actions),用于自动化校验部分规范。

-## .agents/(代理规则 - 三层架构)
+## rulesets/(规则集模板库 - 三层架构)

-`.agents/` 是 AI 代理的极简铁律(三层架构设计):
+> **重要说明**:playbook 仓库中的 `rulesets/` 是**规则集模板库**,不是 playbook 项目自身的代理规则。
+>
+> Playbook 本身不包含源代码,因此不需要 AI 代理遵循规则。`rulesets/` 存在的目的是:
+> 1. 作为**模板源**,供其他项目复制
+> 2. 通过 `sync_standards.sh` 部署到目标项目的 `.agents/`
+> 3. 目标项目的 AI 代理读取**项目根目录的 `.agents/`**(从模板生成)
+
+`rulesets/` 是 AI 代理规则集模板(三层架构设计):

 ### 三层架构设计

 ```
-Layer 1: .agents/          (≤50 行/语言,自动加载)
+Layer 1: rulesets/          (≤50 行/语言,模板源)
   ├─ 核心约束与安全红线
   └─ 指向 Skills 和 docs

@@ -62,11 +69,11 @@ Layer 3: docs/             (权威静态文档)
 ```

 **目录结构**:
-- `.agents/index.md`:规则集索引(跨语言)
-- `.agents/tsl/index.md`:TSL 核心约定(47 行)
-- `.agents/cpp/index.md`:C++ 核心约定(50 行)
-- `.agents/python/index.md`:Python 核心约定(48 行)
-- `.agents/markdown/index.md`:Markdown 核心约定(23 行,仅代码格式化)
+- `rulesets/index.md`:规则集索引(跨语言)
+- `rulesets/tsl/index.md`:TSL 核心约定(47 行)
+- `rulesets/cpp/index.md`:C++ 核心约定(50 行)
+- `rulesets/python/index.md`:Python 核心约定(48 行)
+- `rulesets/markdown/index.md`:Markdown 核心约定(23 行,仅代码格式化)

 详见:`AGENTS.md`

diff --git a/.agents/cpp/index.md b/rulesets/cpp/index.md
similarity index 100%
rename from .agents/cpp/index.md
rename to rulesets/cpp/index.md
diff --git a/rulesets/index.md b/rulesets/index.md
new file mode 100644
index 0000000..81edaca
--- /dev/null
+++ b/rulesets/index.md
@@ -0,0 +1,23 @@
+# rulesets/(规则集模板库)
+
+> **重要**:本目录位于 **playbook 仓库**,作为**规则集模板源**。
+>
+> - **模板源**:`playbook/rulesets/` → 通过 `sync_standards.*` 同步 → 目标项目的 `.agents/`
+> - **AI 读取**:目标项目根目录的 `.agents/`,而非此处
+> - **使用流程**:
+>   ```
+>   playbook/rulesets/tsl/  →  [sync]  →  your-project/.agents/tsl/  ←  AI 代理读取
+>      (模板源)                               (实际使用)
+>   ```
+
+本目录用于存放 **AI/自动化代理规则集模板**,用于分发到其他项目。
+
+本仓库将规则按语言拆分为多个规则集模板:
+
+- `rulesets/tsl/`:TSL 相关规则集(适用于 `.tsl`/`.tsf`)
+- `rulesets/cpp/`:C++ 相关规则集(C++23,含 Modules)
+- `rulesets/python/`:Python 相关规则集
+- `rulesets/markdown/`:Markdown 相关规则集(仅代码格式化)
+
+目标项目落地时,通过 `scripts/sync_standards.*`
+将规则集从 `rulesets/<lang>/` 同步到目标项目根目录的 `.agents/<lang>/`。
diff --git a/.agents/markdown/index.md b/rulesets/markdown/index.md
similarity index 100%
rename from .agents/markdown/index.md
rename to rulesets/markdown/index.md
diff --git a/.agents/python/index.md b/rulesets/python/index.md
similarity index 100%
rename from .agents/python/index.md
rename to rulesets/python/index.md
diff --git a/.agents/tsl/index.md b/rulesets/tsl/index.md
similarity index 100%
rename from .agents/tsl/index.md
rename to rulesets/tsl/index.md
diff --git a/scripts/sync_standards.bat b/scripts/sync_standards.bat
index bfdfdf8..d55ab27 100644
--- a/scripts/sync_standards.bat
+++ b/scripts/sync_standards.bat
@@ -2,7 +2,7 @@
 setlocal enabledelayedexpansion

 rem Sync standards snapshot to project root.
-rem - Copies <snapshot>\.agents\<AGENTS_NS> -> <project-root>\.agents\<AGENTS_NS>
+rem - Copies <snapshot>\rulesets\<AGENTS_NS> -> <project-root>\.agents\<AGENTS_NS>
 rem - Updates <project-root>\.gitattributes (append missing rules by default)
 rem Existing targets are backed up before overwrite.
 rem
@@ -18,8 +18,15 @@ if "%ROOT%"=="" set "ROOT=%cd%"
 for %%I in ("%ROOT%") do set "ROOT=%%~fI"

 for %%I in ("%SCRIPT_DIR%..") do set "SRC=%%~fI"
-set "AGENTS_SRC_ROOT=%SRC%\.agents"
+
+set "AGENTS_SRC_ROOT=%SRC%\rulesets"
 set "GITATTR_SRC=%SRC%\.gitattributes"
+
+if not exist "%AGENTS_SRC_ROOT%" (
+  echo ERROR: Standards snapshot not found at %AGENTS_SRC_ROOT% >&2
+  echo Run: git subtree add --prefix docs/standards/playbook ^<url^> ^<branch^> --squash >&2
+  exit /b 1
+)
 set "AGENTS_NS=%AGENTS_NS%"
 set "GITATTR_DST=%ROOT%\.gitattributes"
 set "SYNC_GITATTR_MODE=%SYNC_GITATTR_MODE%"
diff --git a/scripts/sync_standards.ps1 b/scripts/sync_standards.ps1
index 938bc9f..dafd490 100644
--- a/scripts/sync_standards.ps1
+++ b/scripts/sync_standards.ps1
@@ -1,5 +1,5 @@
 # Sync standards snapshot to project root.
-# - Copies <snapshot>/.agents/<AGENTS_NS> -> <project-root>/.agents/<AGENTS_NS>
+# - Copies <snapshot>/rulesets/<AGENTS_NS> -> <project-root>/.agents/<AGENTS_NS>
 # - Updates <project-root>/.gitattributes (append missing rules by default)
 # Existing targets are backed up before overwrite.
 [CmdletBinding()]
@@ -22,7 +22,7 @@ if (-not $Root) {
 }
 $Root = (Resolve-Path $Root).Path

-$AgentsSrcRoot = Join-Path $Src ".agents"
+$AgentsSrcRoot = Join-Path $Src "rulesets"
 $GitAttrSrc = Join-Path $Src ".gitattributes"

 if (-not (Test-Path $AgentsSrcRoot)) {
diff --git a/scripts/sync_standards.sh b/scripts/sync_standards.sh
index f133675..52cee5f 100644
--- a/scripts/sync_standards.sh
+++ b/scripts/sync_standards.sh
@@ -2,7 +2,7 @@
 set -eu

 # Sync standards snapshot to project root.
-# - Copies <snapshot>/.agents/<AGENTS_NS> -> <project-root>/.agents/<AGENTS_NS>
+# - Copies <snapshot>/rulesets/<AGENTS_NS> -> <project-root>/.agents/<AGENTS_NS>
 # - Updates <project-root>/.gitattributes (append missing rules by default)
 # Existing targets are backed up before overwrite.
 #
@@ -20,7 +20,8 @@ else
   ROOT="$(git -C "$SCRIPT_DIR" rev-parse --show-toplevel 2>/dev/null || pwd)"
 fi
 ROOT="$(CDPATH= cd -- "$ROOT" && pwd -P)"
-AGENTS_SRC_ROOT="$SRC/.agents"
+
+AGENTS_SRC_ROOT="$SRC/rulesets"
 GITATTR_SRC="$SRC/.gitattributes"

 if [ ! -d "$AGENTS_SRC_ROOT" ]; then
diff --git a/scripts/vendor_playbook.bat b/scripts/vendor_playbook.bat
index 0ade3d2..5b3e602 100644
--- a/scripts/vendor_playbook.bat
+++ b/scripts/vendor_playbook.bat
@@ -2,7 +2,7 @@
 setlocal enabledelayedexpansion

 rem Vendor a trimmed Playbook snapshot into a target project (offline copy),
-rem then run sync_standards to materialize .agents\<lang>\ and .gitattributes in
+rem then run sync_standards to materialize rulesets\<lang>\ and .gitattributes in
 rem the target project root.
 rem
 rem Usage:
@@ -75,8 +75,8 @@ if errorlevel 1 (
   exit /b 1
 )

-if not exist "%DEST_PREFIX%\\.agents" mkdir "%DEST_PREFIX%\\.agents"
-copy /y "%SRC%\\.agents\\index.md" "%DEST_PREFIX%\\.agents\\index.md" >nul
+if not exist "%DEST_PREFIX%\\rulesets" mkdir "%DEST_PREFIX%\\rulesets"
+copy /y "%SRC%\\rulesets\\index.md" "%DEST_PREFIX%\\rulesets\\index.md" >nul

 if not exist "%DEST_PREFIX%\\templates" mkdir "%DEST_PREFIX%\\templates"

@@ -103,8 +103,8 @@ for %%L in (%LANGS%) do (
     echo ERROR: docs not found for lang=%%~L "%SRC%\\docs\\%%~L"
     exit /b 1
   )
-  if not exist "%SRC%\\.agents\\%%~L" (
-    echo ERROR: agents ruleset not found for lang=%%~L "%SRC%\\.agents\\%%~L"
+  if not exist "%SRC%\\rulesets\\%%~L" (
+    echo ERROR: agents ruleset not found for lang=%%~L "%SRC%\\rulesets\\%%~L"
     exit /b 1
   )

@@ -114,7 +114,7 @@ for %%L in (%LANGS%) do (
     exit /b 1
   )

-  xcopy "%SRC%\\.agents\\%%~L\\*" "%DEST_PREFIX%\\.agents\\%%~L\\" /e /i /y >nul
+  xcopy "%SRC%\\rulesets\\%%~L\\*" "%DEST_PREFIX%\\rulesets\\%%~L\\" /e /i /y >nul
   if errorlevel 1 (
     echo ERROR: failed to copy agents for lang=%%~L
     exit /b 1
@@ -166,7 +166,7 @@ set "README=%DEST_PREFIX%\\README.md"
 >> "%README%" echo 查看规范入口:
 >> "%README%" echo.
 >> "%README%" echo - `docs/standards/playbook/docs/index.md`
->> "%README%" echo - `.agents/index.md`
+>> "%README%" echo - `rulesets/index.md`
 >> "%README%" echo.
 >> "%README%" echo ## Codex skills(可选)
 >> "%README%" echo.
@@ -191,24 +191,24 @@ set "SOURCE=%DEST_PREFIX%\\SOURCE.md"

 echo Vendored snapshot -^> %DEST_PREFIX%

-set "PROJECT_AGENTS_ROOT=%DEST_ROOT_ABS%\\.agents"
+set "PROJECT_AGENTS_ROOT=%DEST_ROOT_ABS%\\rulesets"
 set "PROJECT_AGENTS_INDEX=%PROJECT_AGENTS_ROOT%\\index.md"
 if not exist "%PROJECT_AGENTS_ROOT%" mkdir "%PROJECT_AGENTS_ROOT%"
 if not exist "%PROJECT_AGENTS_INDEX%" (
-  > "%PROJECT_AGENTS_INDEX%" echo # .agents(多语言)
+  > "%PROJECT_AGENTS_INDEX%" echo # rulesets(多语言)
   >> "%PROJECT_AGENTS_INDEX%" echo.
   >> "%PROJECT_AGENTS_INDEX%" echo 本目录用于存放仓库级/语言级的代理规则集。
   >> "%PROJECT_AGENTS_INDEX%" echo.
   >> "%PROJECT_AGENTS_INDEX%" echo 本项目已启用的规则集:
   for %%L in (%LANGS%) do (
-    if /I "%%~L"=="tsl" >> "%PROJECT_AGENTS_INDEX%" echo - .agents/tsl/:TSL 相关规则集(适用于 .tsl/.tsf)
-    if /I "%%~L"=="cpp" >> "%PROJECT_AGENTS_INDEX%" echo - .agents/cpp/:C++ 相关规则集(C++23,含 Modules)
-    if /I "%%~L"=="python" >> "%PROJECT_AGENTS_INDEX%" echo - .agents/python/:Python 相关规则集
-    if /I "%%~L"=="markdown" >> "%PROJECT_AGENTS_INDEX%" echo - .agents/markdown/:Markdown 相关规则集(仅代码格式化)
+    if /I "%%~L"=="tsl" >> "%PROJECT_AGENTS_INDEX%" echo - rulesets/tsl/:TSL 相关规则集(适用于 .tsl/.tsf)
+    if /I "%%~L"=="cpp" >> "%PROJECT_AGENTS_INDEX%" echo - rulesets/cpp/:C++ 相关规则集(C++23,含 Modules)
+    if /I "%%~L"=="python" >> "%PROJECT_AGENTS_INDEX%" echo - rulesets/python/:Python 相关规则集
+    if /I "%%~L"=="markdown" >> "%PROJECT_AGENTS_INDEX%" echo - rulesets/markdown/:Markdown 相关规则集(仅代码格式化)
   )
   >> "%PROJECT_AGENTS_INDEX%" echo.
   >> "%PROJECT_AGENTS_INDEX%" echo 入口建议从:
-  for %%L in (%LANGS%) do >> "%PROJECT_AGENTS_INDEX%" echo - .agents/%%~L/index.md
+  for %%L in (%LANGS%) do >> "%PROJECT_AGENTS_INDEX%" echo - rulesets/%%~L/index.md
   >> "%PROJECT_AGENTS_INDEX%" echo.
   >> "%PROJECT_AGENTS_INDEX%" echo 标准快照文档入口:
   >> "%PROJECT_AGENTS_INDEX%" echo.
diff --git a/scripts/vendor_playbook.ps1 b/scripts/vendor_playbook.ps1
index 94634f7..1c57080 100644
--- a/scripts/vendor_playbook.ps1
+++ b/scripts/vendor_playbook.ps1
@@ -1,5 +1,5 @@
 # Vendor a trimmed Playbook snapshot into a target project (offline copy),
-# then run sync_standards to materialize .agents\<lang>\ and .gitattributes in
+# then run sync_standards to materialize rulesets\<lang>\ and .gitattributes in
 # the target project root.
 #
 # Usage:
@@ -74,9 +74,9 @@ $DocsDir = Join-Path $DestPrefix "docs"
 New-Item -ItemType Directory -Path $DocsDir -Force | Out-Null
 Copy-Item (Join-Path $Src "docs/common") $DocsDir -Recurse -Force

-$AgentsDir = Join-Path $DestPrefix ".agents"
+$AgentsDir = Join-Path $DestPrefix "rulesets"
 New-Item -ItemType Directory -Path $AgentsDir -Force | Out-Null
-Copy-Item (Join-Path $Src ".agents/index.md") (Join-Path $AgentsDir "index.md") -Force
+Copy-Item (Join-Path $Src "rulesets/index.md") (Join-Path $AgentsDir "index.md") -Force

 $TemplatesDir = Join-Path $DestPrefix "templates"
 New-Item -ItemType Directory -Path $TemplatesDir -Force | Out-Null
@@ -91,7 +91,7 @@ foreach ($lang in $Langs) {
   if (-not (Test-Path $docsSrc)) { throw "Docs not found for lang=$lang ($docsSrc)" }
   Copy-Item $docsSrc $DocsDir -Recurse -Force

-  $agentsSrc = Join-Path (Join-Path $Src ".agents") $lang
+  $agentsSrc = Join-Path (Join-Path $Src "rulesets") $lang
   if (-not (Test-Path $agentsSrc)) { throw "Agents ruleset not found for lang=$lang ($agentsSrc)" }
   Copy-Item $agentsSrc $AgentsDir -Recurse -Force

@@ -184,7 +184,7 @@ sh docs/standards/playbook/scripts/sync_standards.sh $langsCsv
 查看规范入口:

 - `docs/standards/playbook/docs/index.md`
-- `.agents/index.md`
+- `rulesets/index.md`

 ## Codex skills(可选)

@@ -211,27 +211,27 @@ sh docs/standards/playbook/scripts/install_codex_skills.sh

 Write-Host "Vendored snapshot -> $DestPrefix"

-$ProjectAgentsRoot = Join-Path $DestRootAbs ".agents"
+$ProjectAgentsRoot = Join-Path $DestRootAbs "rulesets"
 $ProjectAgentsIndex = Join-Path $ProjectAgentsRoot "index.md"
 New-Item -ItemType Directory -Path $ProjectAgentsRoot -Force | Out-Null
 if (-not (Test-Path $ProjectAgentsIndex)) {
   $agentLines = New-Object System.Collections.Generic.List[string]
-  $agentLines.Add("# .agents(多语言)")
+  $agentLines.Add("# rulesets(多语言)")
   $agentLines.Add("")
   $agentLines.Add("本目录用于存放仓库级/语言级的代理规则集。")
   $agentLines.Add("")
   $agentLines.Add("本项目已启用的规则集:")
   foreach ($lang in $Langs) {
     switch ($lang) {
-      "tsl" { $agentLines.Add("- .agents/tsl/:TSL 相关规则集(适用于 .tsl/.tsf)"); break }
-      "cpp" { $agentLines.Add("- .agents/cpp/:C++ 相关规则集(C++23,含 Modules)"); break }
-      "python" { $agentLines.Add("- .agents/python/:Python 相关规则集"); break }
-      "markdown" { $agentLines.Add("- .agents/markdown/:Markdown 相关规则集(仅代码格式化)"); break }
+      "tsl" { $agentLines.Add("- rulesets/tsl/:TSL 相关规则集(适用于 .tsl/.tsf)"); break }
+      "cpp" { $agentLines.Add("- rulesets/cpp/:C++ 相关规则集(C++23,含 Modules)"); break }
+      "python" { $agentLines.Add("- rulesets/python/:Python 相关规则集"); break }
+      "markdown" { $agentLines.Add("- rulesets/markdown/:Markdown 相关规则集(仅代码格式化)"); break }
     }
   }
   $agentLines.Add("")
   $agentLines.Add("入口建议从:")
-  foreach ($lang in $Langs) { $agentLines.Add("- .agents/$lang/index.md") }
+  foreach ($lang in $Langs) { $agentLines.Add("- rulesets/$lang/index.md") }
   $agentLines.Add("")
   $agentLines.Add("标准快照文档入口:")
   $agentLines.Add("")
diff --git a/scripts/vendor_playbook.sh b/scripts/vendor_playbook.sh
index 19fdd61..4c64b1b 100644
--- a/scripts/vendor_playbook.sh
+++ b/scripts/vendor_playbook.sh
@@ -13,6 +13,7 @@ set -eu
 # Notes:
 # - Snapshot is written to: <project-root>/docs/standards/playbook/
 # - Existing snapshot is backed up before overwrite.
+# - Ruleset templates from rulesets/ will be copied to snapshot for sync_standards use.

 SCRIPT_DIR="$(CDPATH= cd -- "$(dirname -- "$0")" && pwd -P)"
 SRC="$(CDPATH= cd -- "$SCRIPT_DIR/.." && pwd -P)"
@@ -80,8 +81,9 @@ cp "$SRC/SKILLS.md" "$DEST_PREFIX/SKILLS.md"
 mkdir -p "$DEST_PREFIX/docs"
 cp -R "$SRC/docs/common" "$DEST_PREFIX/docs/"

-mkdir -p "$DEST_PREFIX/.agents"
-cp "$SRC/.agents/index.md" "$DEST_PREFIX/.agents/index.md"
+# Copy rulesets
+mkdir -p "$DEST_PREFIX/rulesets"
+cp "$SRC/rulesets/index.md" "$DEST_PREFIX/rulesets/index.md"

 mkdir -p "$DEST_PREFIX/templates"
 if [ -d "$SRC/templates/ci" ]; then
@@ -108,13 +110,14 @@ for lang in "$@"; do
     echo "ERROR: docs not found for lang=$lang ($SRC/docs/$lang)" >&2
     exit 1
   fi
-  if [ ! -d "$SRC/.agents/$lang" ]; then
-    echo "ERROR: agents ruleset not found for lang=$lang ($SRC/.agents/$lang)" >&2
+
+  if [ ! -d "$SRC/rulesets/$lang" ]; then
+    echo "ERROR: rulesets not found for lang=$lang ($SRC/rulesets/$lang)" >&2
     exit 1
   fi

   cp -R "$SRC/docs/$lang" "$DEST_PREFIX/docs/"
-  cp -R "$SRC/.agents/$lang" "$DEST_PREFIX/.agents/"
+  cp -R "$SRC/rulesets/$lang" "$DEST_PREFIX/rulesets/"
   if [ -d "$SRC/templates/$lang" ]; then
     cp -R "$SRC/templates/$lang" "$DEST_PREFIX/templates/"
   fi
diff --git a/tests/scripts/test_sync_standards.bats b/tests/scripts/test_sync_standards.bats
index bec93ef..6a215ca 100644
--- a/tests/scripts/test_sync_standards.bats
+++ b/tests/scripts/test_sync_standards.bats
@@ -14,7 +14,7 @@ setup() {

     # 模拟 playbook 快照目录
     mkdir -p docs/standards/playbook
-    cp -r "$PLAYBOOK_ROOT"/{.agents,.gitattributes,docs,scripts} docs/standards/playbook/ 2>/dev/null || true
+    cp -r "$PLAYBOOK_ROOT"/{rulesets,.gitattributes,docs,scripts} docs/standards/playbook/ 2>/dev/null || true

     export SCRIPT_PATH="$TEST_DIR/docs/standards/playbook/scripts/sync_standards.sh"
 }
@@ -170,7 +170,7 @@ EOF
     [ -f ".gitattributes.bak."* ] || [ -f ".gitattributes.bak" ]
 }

-@test "备份 - .agents/ 更新前创建备份" {
+@test "备份 - rulesets/ 更新前创建备份" {
     cd "$TEST_DIR"
     mkdir -p .agents/tsl
     echo "# Old index" > .agents/tsl/index.md
@@ -189,8 +189,8 @@ EOF
     cd "$TEST_DIR"

     # 复制 Python 规则集(如果存在)
-    if [ -d "$PLAYBOOK_ROOT/.agents/python" ]; then
-        cp -r "$PLAYBOOK_ROOT/.agents/python" docs/standards/playbook/.agents/
+    if [ -d "$PLAYBOOK_ROOT/rulesets/python" ]; then
+        cp -r "$PLAYBOOK_ROOT/rulesets/python" docs/standards/playbook/rulesets/
     fi

     sh "$SCRIPT_PATH" tsl cpp
@@ -209,7 +209,7 @@ EOF

 @test "错误处理 - 未找到 playbook 快照时报错" {
     cd "$TEST_DIR"
-    rm -rf docs/standards/playbook/.agents
+    rm -rf docs/standards/playbook/rulesets

     run sh "$SCRIPT_PATH" tsl

@@ -278,11 +278,11 @@ EOF

     # 第一次同步
     sh "$SCRIPT_PATH" tsl
-    CHECKSUM1=$(find .agents/tsl -type f -exec md5sum {} \; | sort | md5sum)
+    CHECKSUM1=$(find rulesets/tsl -type f -exec md5sum {} \; | sort | md5sum)

     # 第二次同步
     sh "$SCRIPT_PATH" tsl
-    CHECKSUM2=$(find .agents/tsl -type f -exec md5sum {} \; | sort | md5sum)
+    CHECKSUM2=$(find rulesets/tsl -type f -exec md5sum {} \; | sort | md5sum)

     [ "$CHECKSUM1" = "$CHECKSUM2" ]
 }
diff --git a/tests/scripts/test_vendor_playbook.bats b/tests/scripts/test_vendor_playbook.bats
index 96babb0..50ea96e 100644
--- a/tests/scripts/test_vendor_playbook.bats
+++ b/tests/scripts/test_vendor_playbook.bats
@@ -53,7 +53,7 @@ teardown() {
     [ -d "docs/standards/playbook" ]
     [ -d "docs/standards/playbook/docs/common" ]
     [ -d "docs/standards/playbook/docs/tsl" ]
-    [ -d "docs/standards/playbook/.agents/tsl" ]
+    [ -d "docs/standards/playbook/rulesets/tsl" ]
     [ -f "docs/standards/playbook/scripts/sync_standards.sh" ]
 }

@@ -65,8 +65,8 @@ teardown() {
     # 验证包含两种语言的文档
     [ -d "docs/standards/playbook/docs/tsl" ]
     [ -d "docs/standards/playbook/docs/cpp" ]
-    [ -d "docs/standards/playbook/.agents/tsl" ]
-    [ -d "docs/standards/playbook/.agents/cpp" ]
+    [ -d "docs/standards/playbook/rulesets/tsl" ]
+    [ -d "docs/standards/playbook/rulesets/cpp" ]
 }

 # ==============================================
2026-01-10 21:17:12 +08:00
..
integration 🐛 fix(tests): make doc link check awk-portable 2026-01-08 15:44:39 +08:00
scripts ♻️ refactor(playbook): rename agents template directory to rulesets 2026-01-10 21:17:12 +08:00
templates test: add automated tests and ci workflow 2026-01-08 11:29:44 +08:00
README.md feat(sync_standards): auto-detect existing languages 2026-01-08 13:33:47 +08:00

README.md

🧪 Playbook 测试套件

本目录包含 Playbook 项目的完整测试框架,用于验证脚本、模板和文档的正确性。

📋 目录结构

tests/
├── README.md                       # 本文件:测试文档
├── scripts/                        # Shell 脚本测试bats
│   ├── test_sync_standards.bats   # sync_standards.sh 测试
│   ├── test_vendor_playbook.bats  # vendor_playbook.sh 测试
│   └── test_install_codex_skills.bats  # install_codex_skills.sh 测试
├── templates/                      # 模板验证测试
│   ├── validate_python_templates.sh    # Python 模板验证
│   ├── validate_cpp_templates.sh       # C++ 模板验证
│   └── validate_ci_templates.sh        # CI 模板验证
└── integration/                    # 集成测试
    └── check_doc_links.sh          # 文档链接有效性检查

🚀 快速开始

本地运行所有测试

# 进入 playbook 根目录
cd /path/to/playbook

# 1. 运行 Shell 脚本测试(需要 bats
sudo apt-get install bats  # Ubuntu/Debian
cd tests/scripts
bats test_sync_standards.bats
bats test_vendor_playbook.bats
bats test_install_codex_skills.bats

# 2. 运行模板验证测试
cd tests/templates
sh validate_python_templates.sh
sh validate_cpp_templates.sh
sh validate_ci_templates.sh

# 3. 运行集成测试
cd tests/integration
sh check_doc_links.sh

CI 自动化测试

测试套件通过 Gitea Actions 自动运行(见 .gitea/workflows/test.yml

  • 触发时机
    • 推送到 main 分支
    • Pull Request 到 main 分支
    • 手动触发workflow_dispatch
  • 运行平台ubuntu-22.04
  • 并行策略:使用 matrix 策略并行运行多个测试组

📚 测试详解

1. Shell 脚本测试 (scripts/)

使用 bats-core 框架测试 shell 脚本。

test_sync_standards.bats

测试 scripts/sync_standards.sh 脚本的功能:

  • 基础功能
    • 脚本存在且可执行
    • 无参数时默认同步 tsl已有 .agents/<lang>/ 则自动识别)
    • 单语言同步tsl/cpp
    • 多语言同步tsl cpp
  • .gitattributes 同步
    • 默认模式追加缺失规则
    • 保留现有内容
    • 更新已存在的 managed block
  • AGENTS.md 处理
    • 不存在时自动创建
    • 已存在时不覆盖
  • 备份功能
    • 更新前创建备份
  • 错误处理
    • 未找到 playbook 快照时报错
    • 无效语言参数时报错
  • 环境变量
    • SYNC_GITATTR_MODE 配置
  • 幂等性
    • 多次执行结果一致

test_vendor_playbook.bats

测试 scripts/vendor_playbook.sh 脚本的功能:

  • 基础功能
    • 单语言 vendoring
    • 多语言 vendoring
  • 自动同步
    • 自动执行 sync_standards
  • SOURCE.md 生成
    • 包含来源信息
    • 包含 commit hash
    • 包含时间戳
  • 裁剪功能
    • 仅包含指定语言
    • 始终包含 common 目录
    • 包含对应模板文件
  • 目标目录处理
    • 已存在时覆盖更新
    • 创建必要的父目录
  • 错误处理
    • 目标目录不存在时报错
    • 无效语言参数时报错
  • 完整性验证
    • 所有必要文件已复制
    • 脚本可执行
  • 幂等性
    • 多次 vendor 结果一致

test_install_codex_skills.bats

测试 scripts/install_codex_skills.sh 脚本的功能:

  • 基础功能
    • 脚本存在且可执行
  • 安装功能
    • 创建 skills 目录
    • 复制 skill 目录(包含 SKILL.md
    • 支持指定单个 skill 安装
    • 同名目录安装前创建备份
  • 错误处理
    • 指定不存在的 skill 报错
  • 幂等性
    • 多次安装结果一致

2. 模板验证测试 (templates/)

验证项目模板文件的正确性和完整性。

validate_python_templates.sh

验证 templates/python/ 目录下的 Python 模板:

  • pyproject.toml
    • 文件存在
    • TOML 语法正确
    • 包含必要配置节tool.black, tool.isort, tool.pytest.ini_options
    • black line-length 配置正确
    • isort profile 配置正确
  • .flake8
    • 文件存在
    • 包含 [flake8] 配置
    • 配置了 max-line-length
    • 配置了错误忽略规则
  • .pylintrc
    • 文件存在
    • 包含必要配置节MASTER, MESSAGES CONTROL, FORMAT
    • 配置了 max-line-length
  • .pre-commit-config.yaml
    • 文件存在
    • YAML 语法正确
    • 包含 repos 配置
    • 配置了常用 hooksblack, isort, flake8
  • .editorconfig
    • 文件存在
    • 包含 root = true
    • 包含 Python 文件配置
  • .vscode/settings.json
    • 文件存在
    • JSON 语法正确
    • 包含 Python 配置

validate_cpp_templates.sh

验证 templates/cpp/ 目录下的 C++ 模板:

  • CMakeLists.txt
    • 文件存在
    • CMake 基础语法正确
    • 配置了 C++23 标准
    • 启用了 C++ Modules 扫描(可选)
    • 启用了 import std; 支持(可选)
    • 启用了编译命令导出(用于 clangd
  • .clang-format
    • 文件存在
    • YAML 语法正确
    • 配置了 Language: Cpp
    • 基于某个风格BasedOnStyle
    • 配置了 C++ 标准
    • 配置了缩进宽度
  • .clangd
    • 文件存在
    • YAML 语法正确
    • 包含 CompileFlags 配置
    • 配置了 -std=c++23
    • 配置了 CompilationDatabase 路径
    • 配置了 Index 选项
  • conanfile.txt
    • 文件存在
    • 包含 [requires] 配置(可选)
    • 包含 [generators] 配置
    • 配置了 CMakeDeps 生成器
    • 配置了 CMakeToolchain 生成器
    • 包含 [options] 配置
  • CMakeUserPresets.json
    • 文件存在
    • JSON 语法正确
    • 包含 version 字段
    • 包含 configurePresets
    • 包含 buildPresets

validate_ci_templates.sh

验证 templates/ci/ 目录下的 CI 模板:

  • Gitea/GitHub workflow 文件
    • 文件存在
    • YAML 语法正确
    • 包含必要字段name, on, jobs, runs-on, steps
    • 包含中文注释(符合项目风格)
    • 包含配置区域标记
    • 包含环境变量配置
  • 特定 workflow 模板
    • standards-check.yml包含格式化检查和 lint 检查
    • test.yml包含测试步骤和测试矩阵
  • 文档
    • 包含 README 说明文档
    • 包含使用说明

3. 集成测试 (integration/)

端到端测试,验证整体功能。

check_doc_links.sh

检查所有 Markdown 文档中的链接有效性:

  • 扫描范围
    • 所有 *.md 文件
    • 排除 node_modules, .git, build, dist 等目录
  • 链接类型
    • Markdown 链接:[text](link)
    • 引用链接:[text]: link
  • 验证逻辑
    • 检查相对路径链接是否指向存在的文件
    • 跳过外部链接http/https
    • 跳过 mailto 链接
    • 跳过代码块与行内代码中的链接样式文本
    • 支持锚点(但只验证文件存在性)
  • 报告内容
    • 总链接数
    • 有效链接数
    • 跳过链接数(外部/mailto
    • 断开链接详情(文件、行号、链接、目标路径)

🔧 本地开发

安装测试依赖

Ubuntu/Debian

# bats-core (Shell 脚本测试)
sudo apt-get update
sudo apt-get install bats

# Python 工具(模板验证)
pip install toml tomli jsonschema yamllint

# C++ 工具(模板验证,可选)
sudo apt-get install cmake clang-format

# YAML 验证
pip install yamllint

macOS

# bats-core
brew install bats-core

# Python 工具
pip3 install toml tomli jsonschema yamllint

# C++ 工具(可选)
brew install cmake clang-format

# YAML 验证
pip3 install yamllint

运行单个测试

# Shell 脚本测试
cd tests/scripts
bats test_sync_standards.bats --tap  # TAP 格式输出
bats test_vendor_playbook.bats --formatter junit  # JUnit 格式

# 模板验证测试
cd tests/templates
sh validate_python_templates.sh
sh validate_cpp_templates.sh
sh validate_ci_templates.sh

# 集成测试
cd tests/integration
sh check_doc_links.sh

添加新测试

添加新的 bats 测试

tests/scripts/ 创建新文件 test_<script_name>.bats

#!/usr/bin/env bats

setup() {
    # 测试前准备
    export TEST_DIR="$(mktemp -d)"
}

teardown() {
    # 测试后清理
    rm -rf "$TEST_DIR"
}

@test "描述测试内容" {
    # 测试代码
    [ -f "some_file" ]
}

添加新的模板验证

tests/templates/ 创建新文件 validate_<template_type>_templates.sh,参考现有脚本结构。

添加新的集成测试

tests/integration/ 创建新脚本,确保:

  1. 使用 set -eu 启用错误检测
  2. 输出清晰的测试进度
  3. 生成详细的报告文件
  4. 返回正确的退出码0 = 成功,非 0 = 失败)

📊 测试覆盖率目标

  • Shell 脚本测试:目标覆盖率 ≥ 80%
  • 模板验证测试:目标通过率 = 100%
  • 集成测试:目标通过率 = 100%
  • 文档链接有效性:目标有效率 = 100%

🐛 故障排查

bats 测试失败

# 使用 --verbose 查看详细输出
bats --verbose test_sync_standards.bats

# 使用 --trace 查看执行跟踪
bats --trace test_sync_standards.bats

模板验证失败

验证脚本会生成详细报告文件:

  • tests/templates/python_validation_report.txt
  • tests/templates/cpp_validation_report.txt
  • tests/templates/ci_validation_report.txt

文档链接检查失败

查看详细报告:

cat /tmp/doc_links_report.txt

🤝 贡献指南

添加新功能时,请同步更新相应的测试:

  1. 修改脚本scripts/)→ 更新对应的 .bats 测试
  2. 修改模板templates/)→ 更新对应的验证脚本
  3. 修改文档docs/, .agents/)→ 运行文档链接检查
  4. 修改 CI workflow.gitea/workflows/)→ 验证 YAML 语法

📖 相关文档

常见问题

Q: 为什么测试在 CI 通过,但本地失败?

A: 可能原因:

  • 环境差异(工具版本、路径)
  • 权限问题
  • Git 配置差异

建议使用 Docker 容器复现 CI 环境。

Q: 如何跳过某些测试?

A: bats 支持 skip 命令:

@test "某个测试" {
    skip "原因说明"
    # 测试代码
}

Q: 测试运行很慢,如何加速?

A: 建议:

  1. 使用 bats 的 --jobs 参数并行运行
  2. 只运行变更相关的测试
  3. 使用 CI 的缓存机制

测试套件维护者: Playbook 团队 最后更新: 2026-01-07