| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152 |
- #!/usr/bin/env python3
- """
- 自动检查生成的SKILL.md是否通过Phase 4质量标准。
- 对照通过标准表格逐项检查,输出通过/不通过和具体原因。
- 用法:
- python3 quality_check.py <SKILL.md路径>
- 示例:
- python3 quality_check.py .claude/skills/elon-musk-perspective/SKILL.md
- """
- import sys
- import re
- from pathlib import Path
- def check_mental_models(content: str) -> tuple[bool, str]:
- """检查心智模型数量(3-7个)"""
- # 匹配 ### 模型N: 或 ### N. 等模式
- models = re.findall(r'^###\s+(?:模型|Model|心智模型)\s*\d', content, re.MULTILINE)
- if not models:
- # fallback: 数「### 」开头的行在心智模型section中
- in_section = False
- count = 0
- for line in content.split('\n'):
- if re.match(r'^##\s+.*心智模型|Mental Model', line, re.IGNORECASE):
- in_section = True
- continue
- if in_section and re.match(r'^##\s+', line) and '心智模型' not in line:
- break
- if in_section and re.match(r'^###\s+', line):
- count += 1
- if count > 0:
- passed = 3 <= count <= 7
- return passed, f"{count}个心智模型 {'✅' if passed else '❌ (应为3-7个)'}"
- count = len(models)
- if count == 0:
- return False, "未检测到心智模型section"
- passed = 3 <= count <= 7
- return passed, f"{count}个心智模型 {'✅' if passed else '❌ (应为3-7个)'}"
- def check_limitations(content: str) -> tuple[bool, str]:
- """检查每个模型是否有局限性"""
- has_limitation = bool(re.search(r'局限|失效|不适用|盲区|limitation|blind spot', content, re.IGNORECASE))
- return has_limitation, "有局限性标注 ✅" if has_limitation else "❌ 未找到局限性描述"
- def check_expression_dna(content: str) -> tuple[bool, str]:
- """检查表达DNA辨识度"""
- dna_section = bool(re.search(r'表达DNA|Expression DNA|表达风格', content, re.IGNORECASE))
- if not dna_section:
- return False, "❌ 未找到表达DNA section"
- # 检查是否有具体的风格描述(句式、词汇等)
- style_markers = len(re.findall(r'句式|词汇|语气|幽默|节奏|确定性|引用|口头禅', content))
- passed = style_markers >= 3
- return passed, f"表达DNA特征: {style_markers}项 {'✅' if passed else '❌ (应≥3项)'}"
- def check_honest_boundary(content: str) -> tuple[bool, str]:
- """检查诚实边界(至少3条)"""
- # 找诚实边界section
- boundary_match = re.search(r'(?:##\s+.*诚实边界|## Honest Boundary)(.*?)(?=\n##\s|\Z)', content, re.DOTALL | re.IGNORECASE)
- if not boundary_match:
- return False, "❌ 未找到诚实边界section"
- boundary_text = boundary_match.group(1)
- # 计算列表项
- items = re.findall(r'^[-*]\s+', boundary_text, re.MULTILINE)
- count = len(items)
- passed = count >= 3
- return passed, f"诚实边界: {count}条 {'✅' if passed else '❌ (应≥3条)'}"
- def check_tensions(content: str) -> tuple[bool, str]:
- """检查内在张力(至少2对)"""
- tension_markers = len(re.findall(r'张力|矛盾|tension|paradox|一方面.*另一方面|既.*又', content, re.IGNORECASE))
- passed = tension_markers >= 2
- return passed, f"内在张力: {tension_markers}处 {'✅' if passed else '❌ (应≥2处)'}"
- def check_primary_sources(content: str) -> tuple[bool, str]:
- """检查一手来源占比"""
- # 找调研来源section
- source_section = re.search(r'(?:##\s+.*来源|## Source|## Reference)(.*?)(?=\n##\s|\Z)', content, re.DOTALL | re.IGNORECASE)
- if not source_section:
- return True, "未找到来源section(跳过检查)"
- source_text = source_section.group(1)
- primary = len(re.findall(r'一手|primary|本人著作|原始', source_text, re.IGNORECASE))
- secondary = len(re.findall(r'二手|secondary|转述|评论', source_text, re.IGNORECASE))
- total = primary + secondary
- if total == 0:
- return True, "未标记来源类型(跳过检查)"
- ratio = primary / total
- passed = ratio > 0.5
- return passed, f"一手来源占比: {primary}/{total} ({ratio:.0%}) {'✅' if passed else '❌ (应>50%)'}"
- def main():
- if len(sys.argv) < 2:
- print("用法: python3 quality_check.py <SKILL.md路径>")
- sys.exit(1)
- skill_path = Path(sys.argv[1])
- if not skill_path.exists():
- print(f"❌ 文件不存在: {skill_path}")
- sys.exit(1)
- content = skill_path.read_text(encoding='utf-8')
- checks = [
- ("心智模型数量", check_mental_models),
- ("模型局限性", check_limitations),
- ("表达DNA辨识度", check_expression_dna),
- ("诚实边界", check_honest_boundary),
- ("内在张力", check_tensions),
- ("一手来源占比", check_primary_sources),
- ]
- print(f"质量检查: {skill_path.name}")
- print("=" * 50)
- passed_count = 0
- total = len(checks)
- for name, check_fn in checks:
- passed, detail = check_fn(content)
- status = "✅ PASS" if passed else "❌ FAIL"
- print(f" {name:<12} {status} {detail}")
- if passed:
- passed_count += 1
- print("=" * 50)
- print(f"结果: {passed_count}/{total} 通过")
- if passed_count == total:
- print("🎉 全部通过,可以交付")
- elif passed_count >= total - 1:
- print("⚠️ 基本通过,建议修复不通过项后交付")
- else:
- print("❌ 多项不通过,建议回到Phase 2迭代")
- sys.exit(0 if passed_count == total else 1)
- if __name__ == '__main__':
- main()
|