aiwcli 0.10.2 → 0.11.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (249) hide show
  1. package/bin/run.js +1 -1
  2. package/dist/commands/clear.d.ts +11 -6
  3. package/dist/commands/clear.js +229 -381
  4. package/dist/commands/init/index.d.ts +1 -17
  5. package/dist/commands/init/index.js +22 -107
  6. package/dist/lib/gitignore-manager.d.ts +32 -0
  7. package/dist/lib/gitignore-manager.js +141 -2
  8. package/dist/lib/template-installer.d.ts +7 -12
  9. package/dist/lib/template-installer.js +69 -193
  10. package/dist/lib/template-settings-reconstructor.d.ts +35 -0
  11. package/dist/lib/template-settings-reconstructor.js +130 -0
  12. package/dist/templates/CLAUDE.md +8 -8
  13. package/dist/templates/_shared/.claude/commands/handoff-resume.md +64 -0
  14. package/dist/templates/_shared/.claude/commands/handoff.md +16 -10
  15. package/dist/templates/_shared/.claude/settings.json +7 -7
  16. package/dist/templates/_shared/hooks-ts/_utils/git-state.ts +2 -0
  17. package/dist/templates/_shared/hooks-ts/archive_plan.ts +159 -0
  18. package/dist/templates/_shared/hooks-ts/context_monitor.ts +147 -0
  19. package/dist/templates/_shared/hooks-ts/file-suggestion.ts +130 -0
  20. package/dist/templates/_shared/hooks-ts/pre_compact.ts +49 -0
  21. package/dist/templates/_shared/hooks-ts/session_end.ts +104 -0
  22. package/dist/templates/_shared/hooks-ts/session_start.ts +144 -0
  23. package/dist/templates/_shared/hooks-ts/task_create_capture.ts +48 -0
  24. package/dist/templates/_shared/hooks-ts/task_update_capture.ts +74 -0
  25. package/dist/templates/_shared/hooks-ts/user_prompt_submit.ts +83 -0
  26. package/dist/templates/_shared/lib-ts/CLAUDE.md +318 -0
  27. package/dist/templates/_shared/lib-ts/base/atomic-write.ts +138 -0
  28. package/dist/templates/_shared/lib-ts/base/constants.ts +306 -0
  29. package/dist/templates/_shared/lib-ts/base/git-state.ts +58 -0
  30. package/dist/templates/_shared/lib-ts/base/hook-utils.ts +439 -0
  31. package/dist/templates/_shared/lib-ts/base/inference.ts +252 -0
  32. package/dist/templates/_shared/lib-ts/base/logger.ts +250 -0
  33. package/dist/templates/_shared/lib-ts/base/state-io.ts +116 -0
  34. package/dist/templates/_shared/lib-ts/base/stop-words.ts +184 -0
  35. package/dist/templates/_shared/lib-ts/base/subprocess-utils.ts +162 -0
  36. package/dist/templates/_shared/lib-ts/base/utils.ts +184 -0
  37. package/dist/templates/_shared/lib-ts/context/context-formatter.ts +438 -0
  38. package/dist/templates/_shared/lib-ts/context/context-selector.ts +515 -0
  39. package/dist/templates/_shared/lib-ts/context/context-store.ts +707 -0
  40. package/dist/templates/_shared/lib-ts/context/plan-manager.ts +316 -0
  41. package/dist/templates/_shared/lib-ts/context/task-tracker.ts +185 -0
  42. package/dist/templates/_shared/lib-ts/handoff/document-generator.ts +216 -0
  43. package/dist/templates/_shared/lib-ts/handoff/handoff-reader.ts +159 -0
  44. package/dist/templates/_shared/lib-ts/package.json +21 -0
  45. package/dist/templates/_shared/lib-ts/templates/formatters.ts +104 -0
  46. package/dist/templates/_shared/{lib/templates/plan_context.py → lib-ts/templates/plan-context.ts} +14 -22
  47. package/dist/templates/_shared/lib-ts/tsconfig.json +13 -0
  48. package/dist/templates/_shared/lib-ts/types.ts +164 -0
  49. package/dist/templates/_shared/scripts/resolve_context.ts +24 -0
  50. package/dist/templates/_shared/scripts/resume_handoff.ts +321 -0
  51. package/dist/templates/_shared/scripts/save_handoff.ts +359 -0
  52. package/dist/templates/_shared/scripts/status_line.ts +733 -0
  53. package/dist/templates/cc-native/.claude/settings.json +175 -185
  54. package/dist/templates/cc-native/TEMPLATE-SCHEMA.md +15 -17
  55. package/dist/templates/cc-native/_cc-native/agents/ARCH-EVOLUTION.md +63 -0
  56. package/dist/templates/cc-native/_cc-native/agents/ARCH-PATTERNS.md +62 -0
  57. package/dist/templates/cc-native/_cc-native/agents/ARCH-STRUCTURE.md +63 -0
  58. package/dist/templates/cc-native/_cc-native/agents/{ASSUMPTION-CHAIN-TRACER.md → ASSUMPTION-TRACER.md} +6 -10
  59. package/dist/templates/cc-native/_cc-native/agents/CLARITY-AUDITOR.md +6 -10
  60. package/dist/templates/cc-native/_cc-native/agents/CLAUDE.md +74 -3
  61. package/dist/templates/cc-native/_cc-native/agents/COMPLETENESS-FEASIBILITY.md +67 -0
  62. package/dist/templates/cc-native/_cc-native/agents/COMPLETENESS-GAPS.md +71 -0
  63. package/dist/templates/cc-native/_cc-native/agents/COMPLETENESS-ORDERING.md +63 -0
  64. package/dist/templates/cc-native/_cc-native/agents/CONSTRAINT-VALIDATOR.md +73 -0
  65. package/dist/templates/cc-native/_cc-native/agents/DESIGN-ADR-VALIDATOR.md +62 -0
  66. package/dist/templates/cc-native/_cc-native/agents/DESIGN-SCALE-MATCHER.md +65 -0
  67. package/dist/templates/cc-native/_cc-native/agents/DEVILS-ADVOCATE.md +6 -9
  68. package/dist/templates/cc-native/_cc-native/agents/DOCUMENTATION-PHILOSOPHY.md +87 -0
  69. package/dist/templates/cc-native/_cc-native/agents/HANDOFF-READINESS.md +5 -9
  70. package/dist/templates/cc-native/_cc-native/agents/{HIDDEN-COMPLEXITY-DETECTOR.md → HIDDEN-COMPLEXITY.md} +6 -10
  71. package/dist/templates/cc-native/_cc-native/agents/INCREMENTAL-DELIVERY.md +67 -0
  72. package/dist/templates/cc-native/_cc-native/agents/PLAN-ORCHESTRATOR.md +91 -18
  73. package/dist/templates/cc-native/_cc-native/agents/RISK-DEPENDENCY.md +63 -0
  74. package/dist/templates/cc-native/_cc-native/agents/RISK-FMEA.md +67 -0
  75. package/dist/templates/cc-native/_cc-native/agents/RISK-PREMORTEM.md +72 -0
  76. package/dist/templates/cc-native/_cc-native/agents/RISK-REVERSIBILITY.md +75 -0
  77. package/dist/templates/cc-native/_cc-native/agents/SCOPE-BOUNDARY.md +78 -0
  78. package/dist/templates/cc-native/_cc-native/agents/SIMPLICITY-GUARDIAN.md +5 -9
  79. package/dist/templates/cc-native/_cc-native/agents/SKEPTIC.md +16 -12
  80. package/dist/templates/cc-native/_cc-native/agents/TESTDRIVEN-BEHAVIOR-AUDITOR.md +62 -0
  81. package/dist/templates/cc-native/_cc-native/agents/TESTDRIVEN-CHARACTERIZATION.md +72 -0
  82. package/dist/templates/cc-native/_cc-native/agents/TESTDRIVEN-FIRST-VALIDATOR.md +62 -0
  83. package/dist/templates/cc-native/_cc-native/agents/TESTDRIVEN-PYRAMID-ANALYZER.md +62 -0
  84. package/dist/templates/cc-native/_cc-native/agents/TRADEOFF-COSTS.md +68 -0
  85. package/dist/templates/cc-native/_cc-native/agents/TRADEOFF-STAKEHOLDERS.md +66 -0
  86. package/dist/templates/cc-native/_cc-native/agents/VERIFY-COVERAGE.md +75 -0
  87. package/dist/templates/cc-native/_cc-native/agents/VERIFY-STRENGTH.md +70 -0
  88. package/dist/templates/cc-native/_cc-native/hooks/CLAUDE.md +109 -135
  89. package/dist/templates/cc-native/_cc-native/hooks/add_plan_context.ts +119 -0
  90. package/dist/templates/cc-native/_cc-native/hooks/cc-native-plan-review.ts +921 -0
  91. package/dist/templates/cc-native/_cc-native/hooks/plan_questions_early.ts +61 -0
  92. package/dist/templates/cc-native/_cc-native/lib-ts/aggregate-agents.ts +157 -0
  93. package/dist/templates/cc-native/_cc-native/lib-ts/artifacts.ts +709 -0
  94. package/dist/templates/cc-native/_cc-native/lib-ts/cc-native-state.ts +199 -0
  95. package/dist/templates/cc-native/_cc-native/lib-ts/cli-output-parser.ts +124 -0
  96. package/dist/templates/cc-native/_cc-native/lib-ts/config.ts +57 -0
  97. package/dist/templates/cc-native/_cc-native/lib-ts/constants.ts +83 -0
  98. package/dist/templates/cc-native/_cc-native/lib-ts/debug.ts +80 -0
  99. package/dist/templates/cc-native/_cc-native/lib-ts/index.ts +119 -0
  100. package/dist/templates/cc-native/_cc-native/lib-ts/json-parser.ts +162 -0
  101. package/dist/templates/cc-native/_cc-native/lib-ts/nul +3 -0
  102. package/dist/templates/cc-native/_cc-native/lib-ts/orchestrator.ts +249 -0
  103. package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/agent.ts +155 -0
  104. package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/codex.ts +130 -0
  105. package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/gemini.ts +106 -0
  106. package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/index.ts +10 -0
  107. package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/types.ts +23 -0
  108. package/dist/templates/cc-native/_cc-native/lib-ts/state.ts +243 -0
  109. package/dist/templates/cc-native/_cc-native/lib-ts/tsconfig.json +18 -0
  110. package/dist/templates/cc-native/_cc-native/lib-ts/types.ts +310 -0
  111. package/dist/templates/cc-native/_cc-native/lib-ts/verdict.ts +72 -0
  112. package/dist/templates/cc-native/_cc-native/plan-review.config.json +12 -16
  113. package/oclif.manifest.json +1 -1
  114. package/package.json +1 -1
  115. package/dist/lib/template-merger.d.ts +0 -47
  116. package/dist/lib/template-merger.js +0 -162
  117. package/dist/templates/_shared/hooks/__init__.py +0 -16
  118. package/dist/templates/_shared/hooks/__pycache__/__init__.cpython-313.pyc +0 -0
  119. package/dist/templates/_shared/hooks/__pycache__/archive_plan.cpython-313.pyc +0 -0
  120. package/dist/templates/_shared/hooks/__pycache__/context_enforcer.cpython-313.pyc +0 -0
  121. package/dist/templates/_shared/hooks/__pycache__/context_monitor.cpython-313.pyc +0 -0
  122. package/dist/templates/_shared/hooks/__pycache__/file-suggestion.cpython-313.pyc +0 -0
  123. package/dist/templates/_shared/hooks/__pycache__/pre_compact.cpython-313.pyc +0 -0
  124. package/dist/templates/_shared/hooks/__pycache__/session_end.cpython-313.pyc +0 -0
  125. package/dist/templates/_shared/hooks/__pycache__/session_start.cpython-313.pyc +0 -0
  126. package/dist/templates/_shared/hooks/__pycache__/task_create_atomicity.cpython-313.pyc +0 -0
  127. package/dist/templates/_shared/hooks/__pycache__/task_create_capture.cpython-313.pyc +0 -0
  128. package/dist/templates/_shared/hooks/__pycache__/task_update_capture.cpython-313.pyc +0 -0
  129. package/dist/templates/_shared/hooks/__pycache__/user_prompt_submit.cpython-313.pyc +0 -0
  130. package/dist/templates/_shared/hooks/archive_plan.py +0 -169
  131. package/dist/templates/_shared/hooks/context_monitor.py +0 -270
  132. package/dist/templates/_shared/hooks/file-suggestion.py +0 -215
  133. package/dist/templates/_shared/hooks/pre_compact.py +0 -104
  134. package/dist/templates/_shared/hooks/session_end.py +0 -173
  135. package/dist/templates/_shared/hooks/session_start.py +0 -206
  136. package/dist/templates/_shared/hooks/task_create_capture.py +0 -108
  137. package/dist/templates/_shared/hooks/task_update_capture.py +0 -145
  138. package/dist/templates/_shared/hooks/user_prompt_submit.py +0 -139
  139. package/dist/templates/_shared/lib/__init__.py +0 -1
  140. package/dist/templates/_shared/lib/__pycache__/__init__.cpython-313.pyc +0 -0
  141. package/dist/templates/_shared/lib/base/__init__.py +0 -65
  142. package/dist/templates/_shared/lib/base/__pycache__/__init__.cpython-313.pyc +0 -0
  143. package/dist/templates/_shared/lib/base/__pycache__/atomic_write.cpython-313.pyc +0 -0
  144. package/dist/templates/_shared/lib/base/__pycache__/constants.cpython-313.pyc +0 -0
  145. package/dist/templates/_shared/lib/base/__pycache__/hook_utils.cpython-313.pyc +0 -0
  146. package/dist/templates/_shared/lib/base/__pycache__/inference.cpython-313.pyc +0 -0
  147. package/dist/templates/_shared/lib/base/__pycache__/logger.cpython-313.pyc +0 -0
  148. package/dist/templates/_shared/lib/base/__pycache__/stop_words.cpython-313.pyc +0 -0
  149. package/dist/templates/_shared/lib/base/__pycache__/subprocess_utils.cpython-313.pyc +0 -0
  150. package/dist/templates/_shared/lib/base/__pycache__/utils.cpython-313.pyc +0 -0
  151. package/dist/templates/_shared/lib/base/atomic_write.py +0 -180
  152. package/dist/templates/_shared/lib/base/constants.py +0 -358
  153. package/dist/templates/_shared/lib/base/hook_utils.py +0 -341
  154. package/dist/templates/_shared/lib/base/inference.py +0 -318
  155. package/dist/templates/_shared/lib/base/logger.py +0 -291
  156. package/dist/templates/_shared/lib/base/stop_words.py +0 -213
  157. package/dist/templates/_shared/lib/base/subprocess_utils.py +0 -46
  158. package/dist/templates/_shared/lib/base/utils.py +0 -242
  159. package/dist/templates/_shared/lib/context/__init__.py +0 -102
  160. package/dist/templates/_shared/lib/context/__pycache__/__init__.cpython-313.pyc +0 -0
  161. package/dist/templates/_shared/lib/context/__pycache__/cache.cpython-313.pyc +0 -0
  162. package/dist/templates/_shared/lib/context/__pycache__/context_extractor.cpython-313.pyc +0 -0
  163. package/dist/templates/_shared/lib/context/__pycache__/context_formatter.cpython-313.pyc +0 -0
  164. package/dist/templates/_shared/lib/context/__pycache__/context_manager.cpython-313.pyc +0 -0
  165. package/dist/templates/_shared/lib/context/__pycache__/context_selector.cpython-313.pyc +0 -0
  166. package/dist/templates/_shared/lib/context/__pycache__/context_store.cpython-313.pyc +0 -0
  167. package/dist/templates/_shared/lib/context/__pycache__/discovery.cpython-313.pyc +0 -0
  168. package/dist/templates/_shared/lib/context/__pycache__/event_log.cpython-313.pyc +0 -0
  169. package/dist/templates/_shared/lib/context/__pycache__/plan_archive.cpython-313.pyc +0 -0
  170. package/dist/templates/_shared/lib/context/__pycache__/plan_manager.cpython-313.pyc +0 -0
  171. package/dist/templates/_shared/lib/context/__pycache__/task_sync.cpython-313.pyc +0 -0
  172. package/dist/templates/_shared/lib/context/__pycache__/task_tracker.cpython-313.pyc +0 -0
  173. package/dist/templates/_shared/lib/context/context_formatter.py +0 -317
  174. package/dist/templates/_shared/lib/context/context_selector.py +0 -508
  175. package/dist/templates/_shared/lib/context/context_store.py +0 -653
  176. package/dist/templates/_shared/lib/context/plan_manager.py +0 -204
  177. package/dist/templates/_shared/lib/context/task_tracker.py +0 -188
  178. package/dist/templates/_shared/lib/handoff/__init__.py +0 -22
  179. package/dist/templates/_shared/lib/handoff/__pycache__/__init__.cpython-313.pyc +0 -0
  180. package/dist/templates/_shared/lib/handoff/__pycache__/document_generator.cpython-313.pyc +0 -0
  181. package/dist/templates/_shared/lib/handoff/document_generator.py +0 -278
  182. package/dist/templates/_shared/lib/templates/README.md +0 -206
  183. package/dist/templates/_shared/lib/templates/__init__.py +0 -36
  184. package/dist/templates/_shared/lib/templates/__pycache__/__init__.cpython-313.pyc +0 -0
  185. package/dist/templates/_shared/lib/templates/__pycache__/formatters.cpython-313.pyc +0 -0
  186. package/dist/templates/_shared/lib/templates/__pycache__/persona_questions.cpython-313.pyc +0 -0
  187. package/dist/templates/_shared/lib/templates/__pycache__/plan_context.cpython-313.pyc +0 -0
  188. package/dist/templates/_shared/lib/templates/formatters.py +0 -146
  189. package/dist/templates/_shared/scripts/__pycache__/save_handoff.cpython-313.pyc +0 -0
  190. package/dist/templates/_shared/scripts/__pycache__/status_line.cpython-313.pyc +0 -0
  191. package/dist/templates/_shared/scripts/save_handoff.py +0 -357
  192. package/dist/templates/_shared/scripts/status_line.py +0 -701
  193. package/dist/templates/cc-native/.claude/commands/cc-native/fresh-perspective.md +0 -8
  194. package/dist/templates/cc-native/.windsurf/workflows/cc-native/fresh-perspective.md +0 -8
  195. package/dist/templates/cc-native/MIGRATION.md +0 -86
  196. package/dist/templates/cc-native/_cc-native/agents/ACCESSIBILITY-TESTER.md +0 -79
  197. package/dist/templates/cc-native/_cc-native/agents/ARCHITECT-REVIEWER.md +0 -48
  198. package/dist/templates/cc-native/_cc-native/agents/CODE-REVIEWER.md +0 -70
  199. package/dist/templates/cc-native/_cc-native/agents/COMPLETENESS-CHECKER.md +0 -59
  200. package/dist/templates/cc-native/_cc-native/agents/CONTEXT-EXTRACTOR.md +0 -92
  201. package/dist/templates/cc-native/_cc-native/agents/DOCUMENTATION-REVIEWER.md +0 -51
  202. package/dist/templates/cc-native/_cc-native/agents/FEASIBILITY-ANALYST.md +0 -57
  203. package/dist/templates/cc-native/_cc-native/agents/FRESH-PERSPECTIVE.md +0 -54
  204. package/dist/templates/cc-native/_cc-native/agents/INCENTIVE-MAPPER.md +0 -61
  205. package/dist/templates/cc-native/_cc-native/agents/PENETRATION-TESTER.md +0 -79
  206. package/dist/templates/cc-native/_cc-native/agents/PERFORMANCE-ENGINEER.md +0 -75
  207. package/dist/templates/cc-native/_cc-native/agents/PRECEDENT-FINDER.md +0 -70
  208. package/dist/templates/cc-native/_cc-native/agents/REVERSIBILITY-ANALYST.md +0 -61
  209. package/dist/templates/cc-native/_cc-native/agents/RISK-ASSESSOR.md +0 -58
  210. package/dist/templates/cc-native/_cc-native/agents/SECOND-ORDER-ANALYST.md +0 -61
  211. package/dist/templates/cc-native/_cc-native/agents/STAKEHOLDER-ADVOCATE.md +0 -55
  212. package/dist/templates/cc-native/_cc-native/agents/TRADE-OFF-ILLUMINATOR.md +0 -204
  213. package/dist/templates/cc-native/_cc-native/hooks/__pycache__/add_plan_context.cpython-313.pyc +0 -0
  214. package/dist/templates/cc-native/_cc-native/hooks/__pycache__/cc-native-plan-review.cpython-313.pyc +0 -0
  215. package/dist/templates/cc-native/_cc-native/hooks/__pycache__/mark_questions_asked.cpython-313.pyc +0 -0
  216. package/dist/templates/cc-native/_cc-native/hooks/__pycache__/plan_accepted.cpython-313.pyc +0 -0
  217. package/dist/templates/cc-native/_cc-native/hooks/__pycache__/plan_questions_early.cpython-313.pyc +0 -0
  218. package/dist/templates/cc-native/_cc-native/hooks/__pycache__/suggest-fresh-perspective.cpython-313.pyc +0 -0
  219. package/dist/templates/cc-native/_cc-native/hooks/add_plan_context.py +0 -130
  220. package/dist/templates/cc-native/_cc-native/hooks/cc-native-plan-review.py +0 -869
  221. package/dist/templates/cc-native/_cc-native/hooks/plan_questions_early.py +0 -81
  222. package/dist/templates/cc-native/_cc-native/hooks/suggest-fresh-perspective.py +0 -340
  223. package/dist/templates/cc-native/_cc-native/lib/CLAUDE.md +0 -265
  224. package/dist/templates/cc-native/_cc-native/lib/__init__.py +0 -53
  225. package/dist/templates/cc-native/_cc-native/lib/__pycache__/__init__.cpython-313.pyc +0 -0
  226. package/dist/templates/cc-native/_cc-native/lib/__pycache__/atomic_write.cpython-313.pyc +0 -0
  227. package/dist/templates/cc-native/_cc-native/lib/__pycache__/constants.cpython-313.pyc +0 -0
  228. package/dist/templates/cc-native/_cc-native/lib/__pycache__/debug.cpython-313.pyc +0 -0
  229. package/dist/templates/cc-native/_cc-native/lib/__pycache__/orchestrator.cpython-313.pyc +0 -0
  230. package/dist/templates/cc-native/_cc-native/lib/__pycache__/state.cpython-313.pyc +0 -0
  231. package/dist/templates/cc-native/_cc-native/lib/__pycache__/utils.cpython-313.pyc +0 -0
  232. package/dist/templates/cc-native/_cc-native/lib/constants.py +0 -45
  233. package/dist/templates/cc-native/_cc-native/lib/debug.py +0 -139
  234. package/dist/templates/cc-native/_cc-native/lib/orchestrator.py +0 -362
  235. package/dist/templates/cc-native/_cc-native/lib/reviewers/__init__.py +0 -28
  236. package/dist/templates/cc-native/_cc-native/lib/reviewers/__pycache__/__init__.cpython-313.pyc +0 -0
  237. package/dist/templates/cc-native/_cc-native/lib/reviewers/__pycache__/agent.cpython-313.pyc +0 -0
  238. package/dist/templates/cc-native/_cc-native/lib/reviewers/__pycache__/base.cpython-313.pyc +0 -0
  239. package/dist/templates/cc-native/_cc-native/lib/reviewers/__pycache__/codex.cpython-313.pyc +0 -0
  240. package/dist/templates/cc-native/_cc-native/lib/reviewers/__pycache__/gemini.cpython-313.pyc +0 -0
  241. package/dist/templates/cc-native/_cc-native/lib/reviewers/agent.py +0 -215
  242. package/dist/templates/cc-native/_cc-native/lib/reviewers/base.py +0 -88
  243. package/dist/templates/cc-native/_cc-native/lib/reviewers/codex.py +0 -124
  244. package/dist/templates/cc-native/_cc-native/lib/reviewers/gemini.py +0 -108
  245. package/dist/templates/cc-native/_cc-native/lib/state.py +0 -268
  246. package/dist/templates/cc-native/_cc-native/lib/utils.py +0 -1027
  247. package/dist/templates/cc-native/_cc-native/scripts/__pycache__/aggregate_agents.cpython-313.pyc +0 -0
  248. package/dist/templates/cc-native/_cc-native/scripts/aggregate_agents.py +0 -168
  249. package/dist/templates/cc-native/_cc-native/workflows/fresh-perspective.md +0 -134
@@ -0,0 +1,162 @@
1
+ /**
2
+ * Subprocess environment utilities.
3
+ * See SPEC.md §5.10
4
+ */
5
+
6
+ import { execFile, execSync } from "node:child_process";
7
+
8
+ /**
9
+ * Check if this is an internal subprocess call.
10
+ * All hooks should check this and return early to prevent recursion.
11
+ */
12
+ export function isInternalCall(): boolean {
13
+ return process.env.AIWCLI_INTERNAL_CALL === "true";
14
+ }
15
+
16
+ /**
17
+ * Get environment for internal subprocess calls.
18
+ * Returns a copy of process.env with AIWCLI_INTERNAL_CALL=true.
19
+ */
20
+ export function getInternalSubprocessEnv(): Record<string, string | undefined> {
21
+ return {
22
+ ...process.env,
23
+ AIWCLI_INTERNAL_CALL: "true",
24
+ };
25
+ }
26
+
27
+ /**
28
+ * Find an executable on the system PATH.
29
+ * Uses `where` on Windows, `which` on Unix.
30
+ * On Windows, prefers .cmd/.exe over extensionless shims since
31
+ * execFileSync cannot spawn extensionless shell scripts.
32
+ * Returns the first match or null if not found.
33
+ */
34
+ export function findExecutable(name: string): null | string {
35
+ try {
36
+ const cmd = process.platform === "win32" ? `where ${name}` : `which ${name}`;
37
+ const lines = execSync(cmd, { encoding: "utf-8", stdio: ["pipe", "pipe", "pipe"] })
38
+ .trim()
39
+ .split(/\r?\n/)
40
+ .map((l) => l.trim())
41
+ .filter(Boolean);
42
+
43
+ if (lines.length === 0) return null;
44
+
45
+ // On Windows, `where` may return an extensionless shim first (e.g. npm creates
46
+ // both `claude` and `claude.cmd`). execFileSync can't spawn the extensionless
47
+ // one, so prefer .cmd or .exe.
48
+ if (process.platform === "win32") {
49
+ const preferred = lines.find((l) => /\.(cmd|exe)$/i.test(l));
50
+ return preferred ?? lines[0] ?? null;
51
+ }
52
+
53
+ return lines[0] ?? null;
54
+ } catch {
55
+ return null;
56
+ }
57
+ }
58
+
59
+ /**
60
+ * Type guard for Node.js child_process exec errors.
61
+ * ExecSync throws objects with these extra properties on non-zero exit or timeout.
62
+ */
63
+ export interface ExecSyncError {
64
+ killed: boolean;
65
+ message: string;
66
+ signal: null | string;
67
+ status: null | number;
68
+ stderr: Buffer | string;
69
+ stdout: Buffer | string;
70
+ }
71
+
72
+ /** Check if an unknown error is an ExecSync error with process info. */
73
+ export function isExecSyncError(e: unknown): e is ExecSyncError {
74
+ return (
75
+ typeof e === "object" &&
76
+ e !== null &&
77
+ "killed" in e &&
78
+ "signal" in e
79
+ );
80
+ }
81
+
82
+ // ---------------------------------------------------------------------------
83
+ // Async Subprocess Execution
84
+ // ---------------------------------------------------------------------------
85
+
86
+ /**
87
+ * Result from an async subprocess execution.
88
+ * Never throws — callers inspect fields to determine outcome.
89
+ */
90
+ export interface ExecResult {
91
+ exitCode: number;
92
+ killed: boolean;
93
+ signal: null | string;
94
+ stderr: string;
95
+ stdout: string;
96
+ }
97
+
98
+ /** Options for execFileAsync. */
99
+ export interface ExecAsyncOptions {
100
+ /** Environment variables for the child process. */
101
+ env?: Record<string, string | undefined>;
102
+ /** Data piped to the child's stdin. */
103
+ input?: string;
104
+ /** Maximum bytes on stdout/stderr. Default: 10 MB. */
105
+ maxBuffer?: number;
106
+ /** Timeout in milliseconds (not seconds). */
107
+ timeout?: number;
108
+ }
109
+
110
+ /**
111
+ * Async subprocess execution that does NOT block the event loop.
112
+ * Drop-in replacement for execFileSync in Promise-based parallel patterns.
113
+ *
114
+ * Returns ExecResult on both success and non-zero exit.
115
+ * On timeout: result.killed = true, result.signal = "SIGTERM".
116
+ * On spawn failure: result.exitCode = -1, result.stderr contains error.
117
+ */
118
+ export function execFileAsync(
119
+ file: string,
120
+ args: string[],
121
+ options?: ExecAsyncOptions,
122
+ ): Promise<ExecResult> {
123
+ return new Promise((resolve) => {
124
+ const child = execFile(
125
+ file,
126
+ args,
127
+ {
128
+ encoding: "utf-8",
129
+ timeout: options?.timeout ?? 0,
130
+ env: options?.env as NodeJS.ProcessEnv,
131
+ maxBuffer: options?.maxBuffer ?? 10 * 1024 * 1024,
132
+ },
133
+ (error, stdout, stderr) => {
134
+ if (error) {
135
+ // execFile callback error includes process exit info
136
+ const errObj = error as unknown as Record<string, unknown>;
137
+ resolve({
138
+ stdout: String(stdout ?? ""),
139
+ stderr: String(stderr ?? ""),
140
+ exitCode: typeof errObj.code === "number" ? errObj.code : (error as any).status ?? 1,
141
+ killed: Boolean(errObj.killed),
142
+ signal: typeof errObj.signal === "string" ? errObj.signal : null,
143
+ });
144
+ } else {
145
+ resolve({
146
+ stdout: String(stdout ?? ""),
147
+ stderr: String(stderr ?? ""),
148
+ exitCode: 0,
149
+ killed: false,
150
+ signal: null,
151
+ });
152
+ }
153
+ },
154
+ );
155
+
156
+ // Pipe input to stdin if provided
157
+ if (options?.input !== null && options?.input !== undefined && child.stdin) {
158
+ child.stdin.write(options.input);
159
+ child.stdin.end();
160
+ }
161
+ });
162
+ }
@@ -0,0 +1,184 @@
1
+ /**
2
+ * Core utilities for shared context management.
3
+ * See SPEC.md §14.2, §14.3
4
+ */
5
+
6
+ import { sanitizeTitle } from "./constants.js";
7
+ import { logDebug, logError, logWarn } from "./logger.js";
8
+ import { STOP_WORDS } from "./stop-words.js";
9
+
10
+ /**
11
+ * Print to stderr. For terminal-only UX messages, not diagnostics.
12
+ */
13
+ export function eprint(...args: any[]): void {
14
+ process.stderr.write(args.map(String).join(" ") + "\n");
15
+ }
16
+
17
+ /**
18
+ * Get current local datetime as Date.
19
+ */
20
+ export function nowLocal(): Date {
21
+ return new Date();
22
+ }
23
+
24
+ /**
25
+ * Get current time as ISO 8601 string (local time, no timezone suffix).
26
+ * Matches Python datetime.now().isoformat() behavior.
27
+ */
28
+ export function nowIso(): string {
29
+ const d = new Date();
30
+ const year = d.getFullYear();
31
+ const month = String(d.getMonth() + 1).padStart(2, "0");
32
+ const day = String(d.getDate()).padStart(2, "0");
33
+ const hours = String(d.getHours()).padStart(2, "0");
34
+ const minutes = String(d.getMinutes()).padStart(2, "0");
35
+ const seconds = String(d.getSeconds()).padStart(2, "0");
36
+ const ms = String(d.getMilliseconds()).padStart(3, "0");
37
+ return `${year}-${month}-${day}T${hours}:${minutes}:${seconds}.${ms}`;
38
+ }
39
+
40
+ /**
41
+ * Format datetime for display.
42
+ * Returns "YYYY-MM-DD HH:MM:SS"
43
+ * See SPEC.md §14.3
44
+ */
45
+ export function formatTimestamp(dt?: Date): string {
46
+ const d = dt ?? nowLocal();
47
+ const year = d.getFullYear();
48
+ const month = String(d.getMonth() + 1).padStart(2, "0");
49
+ const day = String(d.getDate()).padStart(2, "0");
50
+ const hours = String(d.getHours()).padStart(2, "0");
51
+ const minutes = String(d.getMinutes()).padStart(2, "0");
52
+ const seconds = String(d.getSeconds()).padStart(2, "0");
53
+ return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`;
54
+ }
55
+
56
+ /**
57
+ * Parse ISO 8601 timestamp string.
58
+ * Returns null if parsing fails.
59
+ * See SPEC.md §14.3
60
+ */
61
+ export function parseIsoTimestamp(isoStr: string): Date | null {
62
+ try {
63
+ const normalized = isoStr.replace("Z", "+00:00");
64
+ const d = new Date(normalized);
65
+ if (isNaN(d.getTime())) return null;
66
+ return d;
67
+ } catch {
68
+ return null;
69
+ }
70
+ }
71
+
72
+ /**
73
+ * Clean text for stop-word matching in slug generation.
74
+ * Strips apostrophes (i'm -> im), removes punctuation, normalizes whitespace.
75
+ * See SPEC.md §14.2
76
+ */
77
+ export function cleanTextForSlug(text: string): string {
78
+ if (!text) return "";
79
+ let result = text.toLowerCase();
80
+ result = result.replaceAll('\'', ""); // i'm -> im, you're -> youre
81
+ result = result.replaceAll(/[^a-z0-9\s]/g, " "); // punctuation -> spaces
82
+ result = result.replaceAll(/\s+/g, " ").trim();
83
+ return result;
84
+ }
85
+
86
+ /**
87
+ * Generate a slug from text using AI inference with stop-word fallbacks.
88
+ * Pipeline: AI inference → stop-word post-filter → stop-word fallback → word-length fallback.
89
+ * Reusable by both context ID generation and plan archival.
90
+ * See SPEC.md §14.2
91
+ */
92
+ export function generateSlug(
93
+ text: string,
94
+ maxLen = 150,
95
+ fallbackSlug = "context",
96
+ ): string {
97
+ if (!text || !text.trim()) return fallbackSlug;
98
+
99
+ let slug: null | string = null;
100
+ const cleanedText = cleanTextForSlug(text);
101
+
102
+ // Tier 1: AI inference via generateContextIdSlug (sync — uses execFileSync)
103
+ try {
104
+ // eslint-disable-next-line @typescript-eslint/no-require-imports, no-undef
105
+ const { generateContextIdSlug } = require("./inference.js");
106
+ const aiSlug = generateContextIdSlug(text);
107
+ if (aiSlug) {
108
+ const filteredWords = aiSlug
109
+ .split(/\s+/)
110
+ .filter(
111
+ (w: string) => !STOP_WORDS.has(w.toLowerCase()) && w.length > 1,
112
+ );
113
+ if (filteredWords.length >= 5) {
114
+ slug = sanitizeTitle(filteredWords.join(" "), maxLen);
115
+ } else {
116
+ logDebug(
117
+ "utils",
118
+ `AI slug too generic after stop-word filter (${filteredWords.length} words remain), using fallback`,
119
+ );
120
+ }
121
+ }
122
+ } catch (error: any) {
123
+ logWarn("utils", `AI slug generation failed, using fallback: ${error}`);
124
+ }
125
+
126
+ // Tier 2: Stop-word filtering on cleaned text
127
+ if (!slug) {
128
+ const words = cleanedText
129
+ .split(/\s+/)
130
+ .filter((w) => !STOP_WORDS.has(w) && w.length > 1)
131
+ .slice(0, 12);
132
+ slug = words.length >= 3
133
+ ? sanitizeTitle(words.join(" "), maxLen)
134
+ : sanitizeTitle(
135
+ cleanedText.split(/\s+/).filter((w) => w.length > 2).slice(0, 6).join(" "),
136
+ maxLen,
137
+ ) || fallbackSlug;
138
+ }
139
+
140
+ return slug;
141
+ }
142
+
143
+ /**
144
+ * Generate a context ID from a summary string.
145
+ * Format: YYMMDD-HHMM-slug
146
+ * Delegates slug generation to generateSlug().
147
+ * See SPEC.md §14.2
148
+ */
149
+ export function generateContextId(
150
+ summary: string,
151
+ existingIds?: Set<string>,
152
+ ): string {
153
+ const now = new Date();
154
+ const yy = String(now.getFullYear()).slice(2);
155
+ const mm = String(now.getMonth() + 1).padStart(2, "0");
156
+ const dd = String(now.getDate()).padStart(2, "0");
157
+ const hh = String(now.getHours()).padStart(2, "0");
158
+ const min = String(now.getMinutes()).padStart(2, "0");
159
+ const timestamp = `${yy}${mm}${dd}-${hh}${min}`;
160
+
161
+ let baseId: string;
162
+
163
+ try {
164
+ const slug = generateSlug(summary);
165
+ baseId = `${timestamp}-${slug}`;
166
+ } catch (error: any) {
167
+ logError(
168
+ "utils",
169
+ `Context ID generation failed entirely, using timestamp: ${error}`,
170
+ );
171
+ baseId = `${timestamp}-context`;
172
+ }
173
+
174
+ if (!existingIds || !existingIds.has(baseId)) {
175
+ return baseId;
176
+ }
177
+
178
+ let counter = 2;
179
+ while (existingIds.has(`${baseId}-${counter}`)) {
180
+ counter++;
181
+ }
182
+
183
+ return `${baseId}-${counter}`;
184
+ }