trellis-herbivore 0.1.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.
- package/bin/trellis.js +3 -0
- package/dist/cli/index.d.ts +3 -0
- package/dist/cli/index.d.ts.map +1 -0
- package/dist/cli/index.js +174 -0
- package/dist/cli/index.js.map +1 -0
- package/dist/commands/channel/adapters/claude.d.ts +38 -0
- package/dist/commands/channel/adapters/claude.d.ts.map +1 -0
- package/dist/commands/channel/adapters/claude.js +209 -0
- package/dist/commands/channel/adapters/claude.js.map +1 -0
- package/dist/commands/channel/adapters/codex.d.ts +77 -0
- package/dist/commands/channel/adapters/codex.d.ts.map +1 -0
- package/dist/commands/channel/adapters/codex.js +495 -0
- package/dist/commands/channel/adapters/codex.js.map +1 -0
- package/dist/commands/channel/adapters/index.d.ts +79 -0
- package/dist/commands/channel/adapters/index.d.ts.map +1 -0
- package/dist/commands/channel/adapters/index.js +109 -0
- package/dist/commands/channel/adapters/index.js.map +1 -0
- package/dist/commands/channel/adapters/types.d.ts +33 -0
- package/dist/commands/channel/adapters/types.d.ts.map +1 -0
- package/dist/commands/channel/adapters/types.js +2 -0
- package/dist/commands/channel/adapters/types.js.map +1 -0
- package/dist/commands/channel/agent-loader.d.ts +32 -0
- package/dist/commands/channel/agent-loader.d.ts.map +1 -0
- package/dist/commands/channel/agent-loader.js +154 -0
- package/dist/commands/channel/agent-loader.js.map +1 -0
- package/dist/commands/channel/context-loader.d.ts +26 -0
- package/dist/commands/channel/context-loader.d.ts.map +1 -0
- package/dist/commands/channel/context-loader.js +290 -0
- package/dist/commands/channel/context-loader.js.map +1 -0
- package/dist/commands/channel/context.d.ts +16 -0
- package/dist/commands/channel/context.d.ts.map +1 -0
- package/dist/commands/channel/context.js +83 -0
- package/dist/commands/channel/context.js.map +1 -0
- package/dist/commands/channel/create.d.ts +27 -0
- package/dist/commands/channel/create.d.ts.map +1 -0
- package/dist/commands/channel/create.js +39 -0
- package/dist/commands/channel/create.js.map +1 -0
- package/dist/commands/channel/dev-parse-trace.d.ts +14 -0
- package/dist/commands/channel/dev-parse-trace.d.ts.map +1 -0
- package/dist/commands/channel/dev-parse-trace.js +70 -0
- package/dist/commands/channel/dev-parse-trace.js.map +1 -0
- package/dist/commands/channel/index.d.ts +3 -0
- package/dist/commands/channel/index.d.ts.map +1 -0
- package/dist/commands/channel/index.js +496 -0
- package/dist/commands/channel/index.js.map +1 -0
- package/dist/commands/channel/kill.d.ts +7 -0
- package/dist/commands/channel/kill.d.ts.map +1 -0
- package/dist/commands/channel/kill.js +121 -0
- package/dist/commands/channel/kill.js.map +1 -0
- package/dist/commands/channel/list.d.ts +17 -0
- package/dist/commands/channel/list.d.ts.map +1 -0
- package/dist/commands/channel/list.js +233 -0
- package/dist/commands/channel/list.js.map +1 -0
- package/dist/commands/channel/messages.d.ts +16 -0
- package/dist/commands/channel/messages.d.ts.map +1 -0
- package/dist/commands/channel/messages.js +237 -0
- package/dist/commands/channel/messages.js.map +1 -0
- package/dist/commands/channel/rm.d.ts +27 -0
- package/dist/commands/channel/rm.d.ts.map +1 -0
- package/dist/commands/channel/rm.js +216 -0
- package/dist/commands/channel/rm.js.map +1 -0
- package/dist/commands/channel/run.d.ts +31 -0
- package/dist/commands/channel/run.d.ts.map +1 -0
- package/dist/commands/channel/run.js +137 -0
- package/dist/commands/channel/run.js.map +1 -0
- package/dist/commands/channel/send.d.ts +12 -0
- package/dist/commands/channel/send.d.ts.map +1 -0
- package/dist/commands/channel/send.js +24 -0
- package/dist/commands/channel/send.js.map +1 -0
- package/dist/commands/channel/spawn.d.ts +25 -0
- package/dist/commands/channel/spawn.d.ts.map +1 -0
- package/dist/commands/channel/spawn.js +192 -0
- package/dist/commands/channel/spawn.js.map +1 -0
- package/dist/commands/channel/store/events.d.ts +39 -0
- package/dist/commands/channel/store/events.d.ts.map +1 -0
- package/dist/commands/channel/store/events.js +87 -0
- package/dist/commands/channel/store/events.js.map +1 -0
- package/dist/commands/channel/store/filter.d.ts +3 -0
- package/dist/commands/channel/store/filter.d.ts.map +1 -0
- package/dist/commands/channel/store/filter.js +2 -0
- package/dist/commands/channel/store/filter.js.map +1 -0
- package/dist/commands/channel/store/lock.d.ts +23 -0
- package/dist/commands/channel/store/lock.d.ts.map +1 -0
- package/dist/commands/channel/store/lock.js +99 -0
- package/dist/commands/channel/store/lock.js.map +1 -0
- package/dist/commands/channel/store/paths.d.ts +63 -0
- package/dist/commands/channel/store/paths.d.ts.map +1 -0
- package/dist/commands/channel/store/paths.js +246 -0
- package/dist/commands/channel/store/paths.js.map +1 -0
- package/dist/commands/channel/store/schema.d.ts +27 -0
- package/dist/commands/channel/store/schema.d.ts.map +1 -0
- package/dist/commands/channel/store/schema.js +34 -0
- package/dist/commands/channel/store/schema.js.map +1 -0
- package/dist/commands/channel/store/thread-state.d.ts +5 -0
- package/dist/commands/channel/store/thread-state.d.ts.map +1 -0
- package/dist/commands/channel/store/thread-state.js +16 -0
- package/dist/commands/channel/store/thread-state.js.map +1 -0
- package/dist/commands/channel/store/watch.d.ts +19 -0
- package/dist/commands/channel/store/watch.d.ts.map +1 -0
- package/dist/commands/channel/store/watch.js +130 -0
- package/dist/commands/channel/store/watch.js.map +1 -0
- package/dist/commands/channel/supervisor/inbox.d.ts +25 -0
- package/dist/commands/channel/supervisor/inbox.d.ts.map +1 -0
- package/dist/commands/channel/supervisor/inbox.js +99 -0
- package/dist/commands/channel/supervisor/inbox.js.map +1 -0
- package/dist/commands/channel/supervisor/shutdown.d.ts +66 -0
- package/dist/commands/channel/supervisor/shutdown.d.ts.map +1 -0
- package/dist/commands/channel/supervisor/shutdown.js +143 -0
- package/dist/commands/channel/supervisor/shutdown.js.map +1 -0
- package/dist/commands/channel/supervisor/stdout.d.ts +49 -0
- package/dist/commands/channel/supervisor/stdout.d.ts.map +1 -0
- package/dist/commands/channel/supervisor/stdout.js +107 -0
- package/dist/commands/channel/supervisor/stdout.js.map +1 -0
- package/dist/commands/channel/supervisor.d.ts +47 -0
- package/dist/commands/channel/supervisor.d.ts.map +1 -0
- package/dist/commands/channel/supervisor.js +283 -0
- package/dist/commands/channel/supervisor.js.map +1 -0
- package/dist/commands/channel/text-body.d.ts +13 -0
- package/dist/commands/channel/text-body.d.ts.map +1 -0
- package/dist/commands/channel/text-body.js +47 -0
- package/dist/commands/channel/text-body.js.map +1 -0
- package/dist/commands/channel/threads.d.ts +39 -0
- package/dist/commands/channel/threads.d.ts.map +1 -0
- package/dist/commands/channel/threads.js +106 -0
- package/dist/commands/channel/threads.js.map +1 -0
- package/dist/commands/channel/title.d.ts +12 -0
- package/dist/commands/channel/title.d.ts.map +1 -0
- package/dist/commands/channel/title.js +24 -0
- package/dist/commands/channel/title.js.map +1 -0
- package/dist/commands/channel/wait.d.ts +18 -0
- package/dist/commands/channel/wait.d.ts.map +1 -0
- package/dist/commands/channel/wait.js +76 -0
- package/dist/commands/channel/wait.js.map +1 -0
- package/dist/commands/init.d.ts +57 -0
- package/dist/commands/init.d.ts.map +1 -0
- package/dist/commands/init.js +1466 -0
- package/dist/commands/init.js.map +1 -0
- package/dist/commands/mem.d.ts +234 -0
- package/dist/commands/mem.d.ts.map +1 -0
- package/dist/commands/mem.js +1869 -0
- package/dist/commands/mem.js.map +1 -0
- package/dist/commands/uninstall.d.ts +27 -0
- package/dist/commands/uninstall.d.ts.map +1 -0
- package/dist/commands/uninstall.js +339 -0
- package/dist/commands/uninstall.js.map +1 -0
- package/dist/commands/update.d.ts +72 -0
- package/dist/commands/update.d.ts.map +1 -0
- package/dist/commands/update.js +1926 -0
- package/dist/commands/update.js.map +1 -0
- package/dist/commands/upgrade.d.ts +28 -0
- package/dist/commands/upgrade.d.ts.map +1 -0
- package/dist/commands/upgrade.js +84 -0
- package/dist/commands/upgrade.js.map +1 -0
- package/dist/configurators/antigravity.d.ts +7 -0
- package/dist/configurators/antigravity.d.ts.map +1 -0
- package/dist/configurators/antigravity.js +19 -0
- package/dist/configurators/antigravity.js.map +1 -0
- package/dist/configurators/claude.d.ts +9 -0
- package/dist/configurators/claude.d.ts.map +1 -0
- package/dist/configurators/claude.js +72 -0
- package/dist/configurators/claude.js.map +1 -0
- package/dist/configurators/codebuddy.d.ts +10 -0
- package/dist/configurators/codebuddy.d.ts.map +1 -0
- package/dist/configurators/codebuddy.js +30 -0
- package/dist/configurators/codebuddy.js.map +1 -0
- package/dist/configurators/codex.d.ts +8 -0
- package/dist/configurators/codex.d.ts.map +1 -0
- package/dist/configurators/codex.js +87 -0
- package/dist/configurators/codex.js.map +1 -0
- package/dist/configurators/copilot.d.ts +10 -0
- package/dist/configurators/copilot.d.ts.map +1 -0
- package/dist/configurators/copilot.js +51 -0
- package/dist/configurators/copilot.js.map +1 -0
- package/dist/configurators/cursor.d.ts +10 -0
- package/dist/configurators/cursor.d.ts.map +1 -0
- package/dist/configurators/cursor.js +29 -0
- package/dist/configurators/cursor.js.map +1 -0
- package/dist/configurators/droid.d.ts +10 -0
- package/dist/configurators/droid.d.ts.map +1 -0
- package/dist/configurators/droid.js +30 -0
- package/dist/configurators/droid.js.map +1 -0
- package/dist/configurators/gemini.d.ts +16 -0
- package/dist/configurators/gemini.d.ts.map +1 -0
- package/dist/configurators/gemini.js +38 -0
- package/dist/configurators/gemini.js.map +1 -0
- package/dist/configurators/index.d.ts +65 -0
- package/dist/configurators/index.d.ts.map +1 -0
- package/dist/configurators/index.js +367 -0
- package/dist/configurators/index.js.map +1 -0
- package/dist/configurators/kilo.d.ts +7 -0
- package/dist/configurators/kilo.d.ts.map +1 -0
- package/dist/configurators/kilo.js +19 -0
- package/dist/configurators/kilo.js.map +1 -0
- package/dist/configurators/kiro.d.ts +8 -0
- package/dist/configurators/kiro.d.ts.map +1 -0
- package/dist/configurators/kiro.js +24 -0
- package/dist/configurators/kiro.js.map +1 -0
- package/dist/configurators/opencode.d.ts +14 -0
- package/dist/configurators/opencode.d.ts.map +1 -0
- package/dist/configurators/opencode.js +96 -0
- package/dist/configurators/opencode.js.map +1 -0
- package/dist/configurators/pi.d.ts +3 -0
- package/dist/configurators/pi.d.ts.map +1 -0
- package/dist/configurators/pi.js +45 -0
- package/dist/configurators/pi.js.map +1 -0
- package/dist/configurators/qoder.d.ts +11 -0
- package/dist/configurators/qoder.d.ts.map +1 -0
- package/dist/configurators/qoder.js +31 -0
- package/dist/configurators/qoder.js.map +1 -0
- package/dist/configurators/shared.d.ts +178 -0
- package/dist/configurators/shared.d.ts.map +1 -0
- package/dist/configurators/shared.js +538 -0
- package/dist/configurators/shared.js.map +1 -0
- package/dist/configurators/windsurf.d.ts +7 -0
- package/dist/configurators/windsurf.d.ts.map +1 -0
- package/dist/configurators/windsurf.js +19 -0
- package/dist/configurators/windsurf.js.map +1 -0
- package/dist/configurators/workflow.d.ts +29 -0
- package/dist/configurators/workflow.d.ts.map +1 -0
- package/dist/configurators/workflow.js +163 -0
- package/dist/configurators/workflow.js.map +1 -0
- package/dist/constants/paths.d.ts +70 -0
- package/dist/constants/paths.d.ts.map +1 -0
- package/dist/constants/paths.js +79 -0
- package/dist/constants/paths.js.map +1 -0
- package/dist/constants/version.d.ts +9 -0
- package/dist/constants/version.d.ts.map +1 -0
- package/dist/constants/version.js +15 -0
- package/dist/constants/version.js.map +1 -0
- package/dist/index.d.ts +9 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +9 -0
- package/dist/index.js.map +1 -0
- package/dist/migrations/index.d.ts +62 -0
- package/dist/migrations/index.d.ts.map +1 -0
- package/dist/migrations/index.js +187 -0
- package/dist/migrations/index.js.map +1 -0
- package/dist/migrations/manifests/0.1.9.json +30 -0
- package/dist/migrations/manifests/0.2.0.json +49 -0
- package/dist/migrations/manifests/0.2.12.json +9 -0
- package/dist/migrations/manifests/0.2.13.json +9 -0
- package/dist/migrations/manifests/0.2.14.json +175 -0
- package/dist/migrations/manifests/0.2.15.json +33 -0
- package/dist/migrations/manifests/0.3.0-beta.0.json +297 -0
- package/dist/migrations/manifests/0.3.0-beta.1.json +9 -0
- package/dist/migrations/manifests/0.3.0-beta.10.json +9 -0
- package/dist/migrations/manifests/0.3.0-beta.11.json +9 -0
- package/dist/migrations/manifests/0.3.0-beta.12.json +9 -0
- package/dist/migrations/manifests/0.3.0-beta.13.json +9 -0
- package/dist/migrations/manifests/0.3.0-beta.14.json +9 -0
- package/dist/migrations/manifests/0.3.0-beta.15.json +9 -0
- package/dist/migrations/manifests/0.3.0-beta.16.json +9 -0
- package/dist/migrations/manifests/0.3.0-beta.2.json +9 -0
- package/dist/migrations/manifests/0.3.0-beta.3.json +9 -0
- package/dist/migrations/manifests/0.3.0-beta.4.json +9 -0
- package/dist/migrations/manifests/0.3.0-beta.5.json +9 -0
- package/dist/migrations/manifests/0.3.0-beta.6.json +9 -0
- package/dist/migrations/manifests/0.3.0-beta.7.json +11 -0
- package/dist/migrations/manifests/0.3.0-beta.8.json +9 -0
- package/dist/migrations/manifests/0.3.0-beta.9.json +9 -0
- package/dist/migrations/manifests/0.3.0-rc.0.json +9 -0
- package/dist/migrations/manifests/0.3.0-rc.1.json +9 -0
- package/dist/migrations/manifests/0.3.0-rc.2.json +9 -0
- package/dist/migrations/manifests/0.3.0-rc.3.json +9 -0
- package/dist/migrations/manifests/0.3.0-rc.4.json +9 -0
- package/dist/migrations/manifests/0.3.0-rc.5.json +9 -0
- package/dist/migrations/manifests/0.3.0-rc.6.json +9 -0
- package/dist/migrations/manifests/0.3.0.json +11 -0
- package/dist/migrations/manifests/0.3.1.json +9 -0
- package/dist/migrations/manifests/0.3.10.json +9 -0
- package/dist/migrations/manifests/0.3.2.json +9 -0
- package/dist/migrations/manifests/0.3.3.json +9 -0
- package/dist/migrations/manifests/0.3.4.json +21 -0
- package/dist/migrations/manifests/0.3.5.json +9 -0
- package/dist/migrations/manifests/0.3.6.json +9 -0
- package/dist/migrations/manifests/0.3.7.json +9 -0
- package/dist/migrations/manifests/0.3.8.json +9 -0
- package/dist/migrations/manifests/0.3.9.json +9 -0
- package/dist/migrations/manifests/0.4.0-beta.1.json +228 -0
- package/dist/migrations/manifests/0.4.0-beta.10.json +9 -0
- package/dist/migrations/manifests/0.4.0-beta.2.json +9 -0
- package/dist/migrations/manifests/0.4.0-beta.3.json +9 -0
- package/dist/migrations/manifests/0.4.0-beta.4.json +9 -0
- package/dist/migrations/manifests/0.4.0-beta.5.json +9 -0
- package/dist/migrations/manifests/0.4.0-beta.6.json +9 -0
- package/dist/migrations/manifests/0.4.0-beta.7.json +9 -0
- package/dist/migrations/manifests/0.4.0-beta.8.json +34 -0
- package/dist/migrations/manifests/0.4.0-beta.9.json +9 -0
- package/dist/migrations/manifests/0.4.0-rc.0.json +9 -0
- package/dist/migrations/manifests/0.4.0-rc.1.json +9 -0
- package/dist/migrations/manifests/0.4.0.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.0.json +1646 -0
- package/dist/migrations/manifests/0.5.0-beta.1.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.10.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.11.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.12.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.13.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.14.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.15.json +116 -0
- package/dist/migrations/manifests/0.5.0-beta.16.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.17.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.18.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.19.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.2.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.3.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.4.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.5.json +222 -0
- package/dist/migrations/manifests/0.5.0-beta.6.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.7.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.8.json +9 -0
- package/dist/migrations/manifests/0.5.0-beta.9.json +48 -0
- package/dist/migrations/manifests/0.5.0-rc.0.json +9 -0
- package/dist/migrations/manifests/0.5.0-rc.1.json +9 -0
- package/dist/migrations/manifests/0.5.0-rc.2.json +9 -0
- package/dist/migrations/manifests/0.5.0-rc.3.json +9 -0
- package/dist/migrations/manifests/0.5.0-rc.4.json +9 -0
- package/dist/migrations/manifests/0.5.0-rc.5.json +9 -0
- package/dist/migrations/manifests/0.5.0-rc.6.json +9 -0
- package/dist/migrations/manifests/0.5.0-rc.7.json +9 -0
- package/dist/migrations/manifests/0.5.0.json +9 -0
- package/dist/migrations/manifests/0.5.1.json +9 -0
- package/dist/migrations/manifests/0.5.10.json +9 -0
- package/dist/migrations/manifests/0.5.11.json +16 -0
- package/dist/migrations/manifests/0.5.12.json +9 -0
- package/dist/migrations/manifests/0.5.13.json +9 -0
- package/dist/migrations/manifests/0.5.14.json +9 -0
- package/dist/migrations/manifests/0.5.15.json +9 -0
- package/dist/migrations/manifests/0.5.2.json +9 -0
- package/dist/migrations/manifests/0.5.3.json +9 -0
- package/dist/migrations/manifests/0.5.4.json +9 -0
- package/dist/migrations/manifests/0.5.5.json +9 -0
- package/dist/migrations/manifests/0.5.6.json +9 -0
- package/dist/migrations/manifests/0.5.7.json +16 -0
- package/dist/migrations/manifests/0.5.8.json +9 -0
- package/dist/migrations/manifests/0.5.9.json +9 -0
- package/dist/migrations/manifests/0.6.0-beta.0.json +16 -0
- package/dist/migrations/manifests/0.6.0-beta.1.json +9 -0
- package/dist/migrations/manifests/0.6.0-beta.10.json +9 -0
- package/dist/migrations/manifests/0.6.0-beta.11.json +9 -0
- package/dist/migrations/manifests/0.6.0-beta.12.json +9 -0
- package/dist/migrations/manifests/0.6.0-beta.13.json +9 -0
- package/dist/migrations/manifests/0.6.0-beta.14.json +9 -0
- package/dist/migrations/manifests/0.6.0-beta.2.json +9 -0
- package/dist/migrations/manifests/0.6.0-beta.3.json +9 -0
- package/dist/migrations/manifests/0.6.0-beta.4.json +9 -0
- package/dist/migrations/manifests/0.6.0-beta.5.json +9 -0
- package/dist/migrations/manifests/0.6.0-beta.6.json +16 -0
- package/dist/migrations/manifests/0.6.0-beta.7.json +9 -0
- package/dist/migrations/manifests/0.6.0-beta.8.json +9 -0
- package/dist/migrations/manifests/0.6.0-beta.9.json +9 -0
- package/dist/templates/claude/agents/trellis-check.md +114 -0
- package/dist/templates/claude/agents/trellis-implement.md +113 -0
- package/dist/templates/claude/agents/trellis-research.md +137 -0
- package/dist/templates/claude/index.d.ts +22 -0
- package/dist/templates/claude/index.d.ts.map +1 -0
- package/dist/templates/claude/index.js +46 -0
- package/dist/templates/claude/index.js.map +1 -0
- package/dist/templates/claude/settings.json +73 -0
- package/dist/templates/codebuddy/agents/trellis-check.md +109 -0
- package/dist/templates/codebuddy/agents/trellis-implement.md +110 -0
- package/dist/templates/codebuddy/agents/trellis-research.md +137 -0
- package/dist/templates/codebuddy/index.d.ts +15 -0
- package/dist/templates/codebuddy/index.d.ts.map +1 -0
- package/dist/templates/codebuddy/index.js +15 -0
- package/dist/templates/codebuddy/index.js.map +1 -0
- package/dist/templates/codebuddy/settings.json +59 -0
- package/dist/templates/codex/agents/trellis-check.toml +84 -0
- package/dist/templates/codex/agents/trellis-implement.toml +65 -0
- package/dist/templates/codex/agents/trellis-research.toml +73 -0
- package/dist/templates/codex/config.toml +35 -0
- package/dist/templates/codex/hooks/session-start.py +545 -0
- package/dist/templates/codex/hooks.json +15 -0
- package/dist/templates/codex/index.d.ts +39 -0
- package/dist/templates/codex/index.d.ts.map +1 -0
- package/dist/templates/codex/index.js +85 -0
- package/dist/templates/codex/index.js.map +1 -0
- package/dist/templates/codex/skills/before-dev/SKILL.md +40 -0
- package/dist/templates/codex/skills/brainstorm/SKILL.md +112 -0
- package/dist/templates/codex/skills/break-loop/SKILL.md +130 -0
- package/dist/templates/codex/skills/check/SKILL.md +98 -0
- package/dist/templates/codex/skills/check-cross-layer/SKILL.md +158 -0
- package/dist/templates/codex/skills/create-command/SKILL.md +101 -0
- package/dist/templates/codex/skills/finish-work/SKILL.md +90 -0
- package/dist/templates/codex/skills/improve-ut/SKILL.md +69 -0
- package/dist/templates/codex/skills/integrate-skill/SKILL.md +221 -0
- package/dist/templates/codex/skills/onboard/SKILL.md +363 -0
- package/dist/templates/codex/skills/record-session/SKILL.md +67 -0
- package/dist/templates/codex/skills/start/SKILL.md +64 -0
- package/dist/templates/codex/skills/update-spec/SKILL.md +335 -0
- package/dist/templates/common/bundled-skills/trellis-meta/SKILL.md +73 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/customize-local/add-project-local-conventions.md +83 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/customize-local/change-agents.md +54 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/customize-local/change-context-loading.md +84 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/customize-local/change-hooks.md +57 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/customize-local/change-skills-or-commands.md +78 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/customize-local/change-spec-structure.md +83 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/customize-local/change-task-lifecycle.md +90 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/customize-local/change-workflow.md +65 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/customize-local/overview.md +55 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/local-architecture/context-injection.md +68 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/local-architecture/generated-files.md +80 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/local-architecture/overview.md +51 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/local-architecture/spec-system.md +102 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/local-architecture/task-system.md +103 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/local-architecture/workflow.md +75 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/local-architecture/workspace-memory.md +71 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/platform-files/agents.md +80 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/platform-files/hooks-and-settings.md +69 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/platform-files/overview.md +59 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/platform-files/platform-map.md +74 -0
- package/dist/templates/common/bundled-skills/trellis-meta/references/platform-files/skills-and-commands.md +83 -0
- package/dist/templates/common/commands/continue.md +56 -0
- package/dist/templates/common/commands/finish-work.md +66 -0
- package/dist/templates/common/commands/start.md +59 -0
- package/dist/templates/common/index.d.ts +48 -0
- package/dist/templates/common/index.d.ts.map +1 -0
- package/dist/templates/common/index.js +104 -0
- package/dist/templates/common/index.js.map +1 -0
- package/dist/templates/common/skills/before-dev.md +35 -0
- package/dist/templates/common/skills/brainstorm.md +112 -0
- package/dist/templates/common/skills/break-loop.md +125 -0
- package/dist/templates/common/skills/check.md +93 -0
- package/dist/templates/common/skills/update-spec.md +351 -0
- package/dist/templates/copilot/hooks/session-start.py +547 -0
- package/dist/templates/copilot/hooks.json +19 -0
- package/dist/templates/copilot/index.d.ts +23 -0
- package/dist/templates/copilot/index.d.ts.map +1 -0
- package/dist/templates/copilot/index.js +54 -0
- package/dist/templates/copilot/index.js.map +1 -0
- package/dist/templates/copilot/prompts/before-dev.prompt.md +39 -0
- package/dist/templates/copilot/prompts/brainstorm.prompt.md +111 -0
- package/dist/templates/copilot/prompts/break-loop.prompt.md +129 -0
- package/dist/templates/copilot/prompts/check-cross-layer.prompt.md +157 -0
- package/dist/templates/copilot/prompts/check.prompt.md +97 -0
- package/dist/templates/copilot/prompts/create-command.prompt.md +116 -0
- package/dist/templates/copilot/prompts/finish-work.prompt.md +99 -0
- package/dist/templates/copilot/prompts/integrate-skill.prompt.md +223 -0
- package/dist/templates/copilot/prompts/onboard.prompt.md +362 -0
- package/dist/templates/copilot/prompts/parallel.prompt.md +204 -0
- package/dist/templates/copilot/prompts/record-session.prompt.md +66 -0
- package/dist/templates/copilot/prompts/start.prompt.md +63 -0
- package/dist/templates/copilot/prompts/update-spec.prompt.md +358 -0
- package/dist/templates/cursor/agents/trellis-check.md +108 -0
- package/dist/templates/cursor/agents/trellis-implement.md +109 -0
- package/dist/templates/cursor/agents/trellis-research.md +136 -0
- package/dist/templates/cursor/hooks.json +30 -0
- package/dist/templates/cursor/index.d.ts +13 -0
- package/dist/templates/cursor/index.d.ts.map +1 -0
- package/dist/templates/cursor/index.js +13 -0
- package/dist/templates/cursor/index.js.map +1 -0
- package/dist/templates/droid/droids/trellis-check.md +101 -0
- package/dist/templates/droid/droids/trellis-implement.md +102 -0
- package/dist/templates/droid/droids/trellis-research.md +137 -0
- package/dist/templates/droid/index.d.ts +15 -0
- package/dist/templates/droid/index.d.ts.map +1 -0
- package/dist/templates/droid/index.js +15 -0
- package/dist/templates/droid/index.js.map +1 -0
- package/dist/templates/droid/settings.json +59 -0
- package/dist/templates/extract.d.ts +40 -0
- package/dist/templates/extract.d.ts.map +1 -0
- package/dist/templates/extract.js +106 -0
- package/dist/templates/extract.js.map +1 -0
- package/dist/templates/gemini/agents/trellis-check.md +101 -0
- package/dist/templates/gemini/agents/trellis-implement.md +102 -0
- package/dist/templates/gemini/agents/trellis-research.md +136 -0
- package/dist/templates/gemini/index.d.ts +13 -0
- package/dist/templates/gemini/index.d.ts.map +1 -0
- package/dist/templates/gemini/index.js +13 -0
- package/dist/templates/gemini/index.js.map +1 -0
- package/dist/templates/gemini/settings.json +28 -0
- package/dist/templates/kiro/agents/trellis-check.json +26 -0
- package/dist/templates/kiro/agents/trellis-implement.json +26 -0
- package/dist/templates/kiro/agents/trellis-research.json +30 -0
- package/dist/templates/kiro/index.d.ts +18 -0
- package/dist/templates/kiro/index.d.ts.map +1 -0
- package/dist/templates/kiro/index.js +18 -0
- package/dist/templates/kiro/index.js.map +1 -0
- package/dist/templates/markdown/agents.md +21 -0
- package/dist/templates/markdown/gitignore.txt +15 -0
- package/dist/templates/markdown/index.d.ts +27 -0
- package/dist/templates/markdown/index.d.ts.map +1 -0
- package/dist/templates/markdown/index.js +52 -0
- package/dist/templates/markdown/index.js.map +1 -0
- package/dist/templates/markdown/spec/backend/database-guidelines.md.txt +51 -0
- package/dist/templates/markdown/spec/backend/directory-structure.md.txt +54 -0
- package/dist/templates/markdown/spec/backend/error-handling.md.txt +51 -0
- package/dist/templates/markdown/spec/backend/index.md.txt +38 -0
- package/dist/templates/markdown/spec/backend/logging-guidelines.md.txt +51 -0
- package/dist/templates/markdown/spec/backend/quality-guidelines.md.txt +51 -0
- package/dist/templates/markdown/spec/frontend/component-guidelines.md.txt +59 -0
- package/dist/templates/markdown/spec/frontend/directory-structure.md.txt +54 -0
- package/dist/templates/markdown/spec/frontend/hook-guidelines.md.txt +51 -0
- package/dist/templates/markdown/spec/frontend/index.md.txt +39 -0
- package/dist/templates/markdown/spec/frontend/quality-guidelines.md.txt +51 -0
- package/dist/templates/markdown/spec/frontend/state-management.md.txt +51 -0
- package/dist/templates/markdown/spec/frontend/type-safety.md.txt +51 -0
- package/dist/templates/markdown/spec/guides/code-reuse-thinking-guide.md.txt +223 -0
- package/dist/templates/markdown/spec/guides/cross-layer-thinking-guide.md.txt +259 -0
- package/dist/templates/markdown/spec/guides/cross-platform-thinking-guide.md.txt +595 -0
- package/dist/templates/markdown/spec/guides/index.md.txt +97 -0
- package/dist/templates/markdown/workspace-index.md +125 -0
- package/dist/templates/markdown/worktree.yaml.txt +58 -0
- package/dist/templates/opencode/agents/trellis-check.md +116 -0
- package/dist/templates/opencode/agents/trellis-implement.md +118 -0
- package/dist/templates/opencode/agents/trellis-research.md +145 -0
- package/dist/templates/opencode/lib/session-utils.js +521 -0
- package/dist/templates/opencode/lib/trellis-context.js +381 -0
- package/dist/templates/opencode/package.json +5 -0
- package/dist/templates/opencode/plugins/inject-subagent-context.js +513 -0
- package/dist/templates/opencode/plugins/inject-workflow-state.js +159 -0
- package/dist/templates/opencode/plugins/session-start.js +101 -0
- package/dist/templates/pi/agents/trellis-check.md +36 -0
- package/dist/templates/pi/agents/trellis-implement.md +41 -0
- package/dist/templates/pi/agents/trellis-research.md +25 -0
- package/dist/templates/pi/extensions/trellis/index.ts.txt +1174 -0
- package/dist/templates/pi/index.d.ts +5 -0
- package/dist/templates/pi/index.d.ts.map +1 -0
- package/dist/templates/pi/index.js +12 -0
- package/dist/templates/pi/index.js.map +1 -0
- package/dist/templates/pi/settings.json +21 -0
- package/dist/templates/qoder/agents/trellis-check.md +102 -0
- package/dist/templates/qoder/agents/trellis-implement.md +103 -0
- package/dist/templates/qoder/agents/trellis-research.md +137 -0
- package/dist/templates/qoder/index.d.ts +15 -0
- package/dist/templates/qoder/index.d.ts.map +1 -0
- package/dist/templates/qoder/index.js +15 -0
- package/dist/templates/qoder/index.js.map +1 -0
- package/dist/templates/qoder/settings.json +47 -0
- package/dist/templates/shared-hooks/index.d.ts +50 -0
- package/dist/templates/shared-hooks/index.d.ts.map +1 -0
- package/dist/templates/shared-hooks/index.js +89 -0
- package/dist/templates/shared-hooks/index.js.map +1 -0
- package/dist/templates/shared-hooks/inject-shell-session-context.py +183 -0
- package/dist/templates/shared-hooks/inject-subagent-context.py +771 -0
- package/dist/templates/shared-hooks/inject-workflow-state.py +363 -0
- package/dist/templates/shared-hooks/session-start.py +827 -0
- package/dist/templates/template-utils.d.ts +26 -0
- package/dist/templates/template-utils.d.ts.map +1 -0
- package/dist/templates/template-utils.js +60 -0
- package/dist/templates/template-utils.js.map +1 -0
- package/dist/templates/trellis/config.yaml +90 -0
- package/dist/templates/trellis/gitignore.txt +32 -0
- package/dist/templates/trellis/index.d.ts +52 -0
- package/dist/templates/trellis/index.d.ts.map +1 -0
- package/dist/templates/trellis/index.js +97 -0
- package/dist/templates/trellis/index.js.map +1 -0
- package/dist/templates/trellis/scripts/__init__.py +5 -0
- package/dist/templates/trellis/scripts/add_session.py +547 -0
- package/dist/templates/trellis/scripts/common/__init__.py +92 -0
- package/dist/templates/trellis/scripts/common/active_task.py +626 -0
- package/dist/templates/trellis/scripts/common/cli_adapter.py +811 -0
- package/dist/templates/trellis/scripts/common/config.py +445 -0
- package/dist/templates/trellis/scripts/common/developer.py +190 -0
- package/dist/templates/trellis/scripts/common/git.py +31 -0
- package/dist/templates/trellis/scripts/common/git_context.py +106 -0
- package/dist/templates/trellis/scripts/common/io.py +37 -0
- package/dist/templates/trellis/scripts/common/log.py +45 -0
- package/dist/templates/trellis/scripts/common/packages_context.py +238 -0
- package/dist/templates/trellis/scripts/common/paths.py +447 -0
- package/dist/templates/trellis/scripts/common/safe_commit.py +285 -0
- package/dist/templates/trellis/scripts/common/session_context.py +821 -0
- package/dist/templates/trellis/scripts/common/task_context.py +223 -0
- package/dist/templates/trellis/scripts/common/task_queue.py +188 -0
- package/dist/templates/trellis/scripts/common/task_store.py +698 -0
- package/dist/templates/trellis/scripts/common/task_utils.py +274 -0
- package/dist/templates/trellis/scripts/common/tasks.py +112 -0
- package/dist/templates/trellis/scripts/common/trellis_config.py +131 -0
- package/dist/templates/trellis/scripts/common/types.py +110 -0
- package/dist/templates/trellis/scripts/common/workflow_phase.py +212 -0
- package/dist/templates/trellis/scripts/get_context.py +16 -0
- package/dist/templates/trellis/scripts/get_developer.py +26 -0
- package/dist/templates/trellis/scripts/hooks/linear_sync.py +243 -0
- package/dist/templates/trellis/scripts/init_developer.py +51 -0
- package/dist/templates/trellis/scripts/task.py +500 -0
- package/dist/templates/trellis/tasks/.gitkeep +0 -0
- package/dist/templates/trellis/workflow.md +690 -0
- package/dist/types/ai-tools.d.ts +95 -0
- package/dist/types/ai-tools.d.ts.map +1 -0
- package/dist/types/ai-tools.js +280 -0
- package/dist/types/ai-tools.js.map +1 -0
- package/dist/types/migration.d.ts +125 -0
- package/dist/types/migration.d.ts.map +1 -0
- package/dist/types/migration.js +8 -0
- package/dist/types/migration.js.map +1 -0
- package/dist/utils/compare-versions.d.ts +12 -0
- package/dist/utils/compare-versions.d.ts.map +1 -0
- package/dist/utils/compare-versions.js +86 -0
- package/dist/utils/compare-versions.js.map +1 -0
- package/dist/utils/cwd-guard.d.ts +38 -0
- package/dist/utils/cwd-guard.d.ts.map +1 -0
- package/dist/utils/cwd-guard.js +62 -0
- package/dist/utils/cwd-guard.js.map +1 -0
- package/dist/utils/file-writer.d.ts +36 -0
- package/dist/utils/file-writer.d.ts.map +1 -0
- package/dist/utils/file-writer.js +203 -0
- package/dist/utils/file-writer.js.map +1 -0
- package/dist/utils/manifest-prune.d.ts +61 -0
- package/dist/utils/manifest-prune.d.ts.map +1 -0
- package/dist/utils/manifest-prune.js +136 -0
- package/dist/utils/manifest-prune.js.map +1 -0
- package/dist/utils/posix.d.ts +13 -0
- package/dist/utils/posix.d.ts.map +1 -0
- package/dist/utils/posix.js +15 -0
- package/dist/utils/posix.js.map +1 -0
- package/dist/utils/project-detector.d.ts +46 -0
- package/dist/utils/project-detector.d.ts.map +1 -0
- package/dist/utils/project-detector.js +666 -0
- package/dist/utils/project-detector.js.map +1 -0
- package/dist/utils/proxy.d.ts +25 -0
- package/dist/utils/proxy.d.ts.map +1 -0
- package/dist/utils/proxy.js +60 -0
- package/dist/utils/proxy.js.map +1 -0
- package/dist/utils/task-json.d.ts +13 -0
- package/dist/utils/task-json.d.ts.map +1 -0
- package/dist/utils/task-json.js +12 -0
- package/dist/utils/task-json.js.map +1 -0
- package/dist/utils/template-fetcher.d.ts +150 -0
- package/dist/utils/template-fetcher.d.ts.map +1 -0
- package/dist/utils/template-fetcher.js +907 -0
- package/dist/utils/template-fetcher.js.map +1 -0
- package/dist/utils/template-hash.d.ts +123 -0
- package/dist/utils/template-hash.d.ts.map +1 -0
- package/dist/utils/template-hash.js +334 -0
- package/dist/utils/template-hash.js.map +1 -0
- package/dist/utils/uninstall-scrubbers.d.ts +66 -0
- package/dist/utils/uninstall-scrubbers.d.ts.map +1 -0
- package/dist/utils/uninstall-scrubbers.js +342 -0
- package/dist/utils/uninstall-scrubbers.js.map +1 -0
- package/package.json +90 -0
|
@@ -0,0 +1,827 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
# -*- coding: utf-8 -*-
|
|
3
|
+
"""
|
|
4
|
+
Session Start Hook - Inject structured context
|
|
5
|
+
"""
|
|
6
|
+
from __future__ import annotations
|
|
7
|
+
|
|
8
|
+
# IMPORTANT: Suppress all warnings FIRST
|
|
9
|
+
import warnings
|
|
10
|
+
warnings.filterwarnings("ignore")
|
|
11
|
+
|
|
12
|
+
import json
|
|
13
|
+
import os
|
|
14
|
+
import re
|
|
15
|
+
import shlex
|
|
16
|
+
import subprocess
|
|
17
|
+
import sys
|
|
18
|
+
from io import StringIO
|
|
19
|
+
from pathlib import Path
|
|
20
|
+
|
|
21
|
+
|
|
22
|
+
def _normalize_windows_shell_path(path_str: str) -> str:
|
|
23
|
+
"""Normalize Unix-style shell paths to real Windows paths.
|
|
24
|
+
|
|
25
|
+
On Windows, shells like Git Bash / MSYS2 / Cygwin may report paths like
|
|
26
|
+
`/d/Users/...` or `/cygdrive/d/Users/...`. `Path.resolve()` will misinterpret
|
|
27
|
+
these as `D:/d/Users...` on drive D: (or similar), breaking repo root
|
|
28
|
+
detection.
|
|
29
|
+
|
|
30
|
+
This function is intentionally conservative: it only rewrites patterns that
|
|
31
|
+
unambiguously represent a drive letter mount.
|
|
32
|
+
"""
|
|
33
|
+
if not isinstance(path_str, str) or not path_str:
|
|
34
|
+
return path_str
|
|
35
|
+
|
|
36
|
+
# Only relevant on Windows; keep other platforms untouched.
|
|
37
|
+
if not sys.platform.startswith("win"):
|
|
38
|
+
return path_str
|
|
39
|
+
|
|
40
|
+
p = path_str.strip()
|
|
41
|
+
|
|
42
|
+
# Already a Windows drive path (C:\... or C:/...)
|
|
43
|
+
if re.match(r"^[A-Za-z]:[\/]", p):
|
|
44
|
+
return p
|
|
45
|
+
|
|
46
|
+
# MSYS/Git-Bash style: /c/Users/... or /d/Work/...
|
|
47
|
+
m = re.match(r"^/([A-Za-z])/(.*)", p)
|
|
48
|
+
if m:
|
|
49
|
+
drive, rest = m.group(1).upper(), m.group(2)
|
|
50
|
+
rest = rest.replace('/', '\\')
|
|
51
|
+
return f"{drive}:\\{rest}"
|
|
52
|
+
|
|
53
|
+
# Cygwin style: /cygdrive/c/Users/...
|
|
54
|
+
m = re.match(r"^/cygdrive/([A-Za-z])/(.*)", p)
|
|
55
|
+
if m:
|
|
56
|
+
drive, rest = m.group(1).upper(), m.group(2)
|
|
57
|
+
rest = rest.replace('/', '\\')
|
|
58
|
+
return f"{drive}:\\{rest}"
|
|
59
|
+
|
|
60
|
+
# WSL mounted drive (sometimes leaked into env): /mnt/c/Users/...
|
|
61
|
+
m = re.match(r"^/mnt/([A-Za-z])/(.*)", p)
|
|
62
|
+
if m:
|
|
63
|
+
drive, rest = m.group(1).upper(), m.group(2)
|
|
64
|
+
rest = rest.replace('/', '\\')
|
|
65
|
+
return f"{drive}:\\{rest}"
|
|
66
|
+
|
|
67
|
+
return path_str
|
|
68
|
+
|
|
69
|
+
|
|
70
|
+
FIRST_REPLY_NOTICE = """<first-reply-notice>
|
|
71
|
+
First visible reply: say once in Chinese that Trellis SessionStart context is loaded, then answer directly.
|
|
72
|
+
This notice is one-shot: do not repeat it after the first assistant reply in the same session.
|
|
73
|
+
</first-reply-notice>"""
|
|
74
|
+
|
|
75
|
+
# Force UTF-8 on stdin/stdout/stderr on Windows. Default codepage there is
|
|
76
|
+
# cp936 / cp1252 / etc. — non-ASCII content (Chinese task names, prd snippets)
|
|
77
|
+
# both in stdin (hook payload from host CLI) and stdout (our emitted blocks)
|
|
78
|
+
# raises UnicodeDecodeError / UnicodeEncodeError. Equivalent to `python -X utf8`
|
|
79
|
+
# but applied per-stream so we don't depend on host CLI's command wiring.
|
|
80
|
+
if sys.platform.startswith("win"):
|
|
81
|
+
import io as _io
|
|
82
|
+
for _stream_name in ("stdin", "stdout", "stderr"):
|
|
83
|
+
_stream = getattr(sys, _stream_name, None)
|
|
84
|
+
if _stream is None:
|
|
85
|
+
continue
|
|
86
|
+
if hasattr(_stream, "reconfigure"):
|
|
87
|
+
try:
|
|
88
|
+
_stream.reconfigure(encoding="utf-8", errors="replace") # type: ignore[union-attr]
|
|
89
|
+
except Exception:
|
|
90
|
+
pass
|
|
91
|
+
elif hasattr(_stream, "detach"):
|
|
92
|
+
try:
|
|
93
|
+
setattr(sys, _stream_name, _io.TextIOWrapper(_stream.detach(), encoding="utf-8", errors="replace"))
|
|
94
|
+
except Exception:
|
|
95
|
+
pass
|
|
96
|
+
|
|
97
|
+
|
|
98
|
+
|
|
99
|
+
def _has_curated_jsonl_entry(jsonl_path: Path) -> bool:
|
|
100
|
+
"""Return True iff jsonl has at least one row with a ``file`` field.
|
|
101
|
+
|
|
102
|
+
A freshly seeded jsonl only contains a ``{"_example": ...}`` row (no
|
|
103
|
+
``file`` key) — that is NOT "ready". Readiness requires at least one
|
|
104
|
+
curated entry. Matches the contract used by hook-inject and pull-based
|
|
105
|
+
sub-agent context loaders.
|
|
106
|
+
"""
|
|
107
|
+
try:
|
|
108
|
+
for line in jsonl_path.read_text(encoding="utf-8").splitlines():
|
|
109
|
+
line = line.strip()
|
|
110
|
+
if not line:
|
|
111
|
+
continue
|
|
112
|
+
try:
|
|
113
|
+
row = json.loads(line)
|
|
114
|
+
except json.JSONDecodeError:
|
|
115
|
+
continue
|
|
116
|
+
if isinstance(row, dict) and row.get("file"):
|
|
117
|
+
return True
|
|
118
|
+
except (OSError, UnicodeDecodeError):
|
|
119
|
+
return False
|
|
120
|
+
return False
|
|
121
|
+
|
|
122
|
+
|
|
123
|
+
def should_skip_injection() -> bool:
|
|
124
|
+
"""Check if any platform's non-interactive flag is set, or if Trellis
|
|
125
|
+
hooks are explicitly disabled via TRELLIS_HOOKS=0 / TRELLIS_DISABLE_HOOKS=1.
|
|
126
|
+
"""
|
|
127
|
+
if os.environ.get("TRELLIS_HOOKS") == "0":
|
|
128
|
+
return True
|
|
129
|
+
if os.environ.get("TRELLIS_DISABLE_HOOKS") == "1":
|
|
130
|
+
return True
|
|
131
|
+
non_interactive_vars = [
|
|
132
|
+
"CLAUDE_NON_INTERACTIVE",
|
|
133
|
+
"QODER_NON_INTERACTIVE",
|
|
134
|
+
"CODEBUDDY_NON_INTERACTIVE",
|
|
135
|
+
"FACTORY_NON_INTERACTIVE",
|
|
136
|
+
"CURSOR_NON_INTERACTIVE",
|
|
137
|
+
"GEMINI_NON_INTERACTIVE",
|
|
138
|
+
"KIRO_NON_INTERACTIVE",
|
|
139
|
+
"COPILOT_NON_INTERACTIVE",
|
|
140
|
+
]
|
|
141
|
+
return any(os.environ.get(var) == "1" for var in non_interactive_vars)
|
|
142
|
+
|
|
143
|
+
|
|
144
|
+
def read_file(path: Path, fallback: str = "") -> str:
|
|
145
|
+
try:
|
|
146
|
+
return path.read_text(encoding="utf-8")
|
|
147
|
+
except (FileNotFoundError, PermissionError):
|
|
148
|
+
return fallback
|
|
149
|
+
|
|
150
|
+
|
|
151
|
+
def _repo_relative(repo_root: Path, path: Path) -> str:
|
|
152
|
+
try:
|
|
153
|
+
return path.relative_to(repo_root).as_posix()
|
|
154
|
+
except ValueError:
|
|
155
|
+
return str(path)
|
|
156
|
+
|
|
157
|
+
|
|
158
|
+
def _run_git(repo_root: Path, args: list[str]) -> str:
|
|
159
|
+
try:
|
|
160
|
+
result = subprocess.run(
|
|
161
|
+
["git", *args],
|
|
162
|
+
capture_output=True,
|
|
163
|
+
text=True,
|
|
164
|
+
encoding="utf-8",
|
|
165
|
+
errors="replace",
|
|
166
|
+
timeout=3,
|
|
167
|
+
cwd=str(repo_root),
|
|
168
|
+
)
|
|
169
|
+
except (subprocess.TimeoutExpired, FileNotFoundError, PermissionError):
|
|
170
|
+
return ""
|
|
171
|
+
if result.returncode != 0:
|
|
172
|
+
return ""
|
|
173
|
+
return result.stdout.strip()
|
|
174
|
+
|
|
175
|
+
|
|
176
|
+
def _format_git_state(repo_root: Path) -> str:
|
|
177
|
+
branch = _run_git(repo_root, ["branch", "--show-current"]) or "(detached)"
|
|
178
|
+
dirty_lines = [
|
|
179
|
+
line for line in _run_git(repo_root, ["status", "--porcelain"]).splitlines()
|
|
180
|
+
if line.strip()
|
|
181
|
+
]
|
|
182
|
+
dirty_text = "clean" if not dirty_lines else f"dirty {len(dirty_lines)} paths"
|
|
183
|
+
return f"Git: branch {branch}; {dirty_text}."
|
|
184
|
+
|
|
185
|
+
|
|
186
|
+
def _detect_platform(input_data: dict) -> str | None:
|
|
187
|
+
if isinstance(input_data.get("cursor_version"), str):
|
|
188
|
+
return "cursor"
|
|
189
|
+
env_map = {
|
|
190
|
+
"CLAUDE_PROJECT_DIR": "claude",
|
|
191
|
+
"CURSOR_PROJECT_DIR": "cursor",
|
|
192
|
+
"CODEBUDDY_PROJECT_DIR": "codebuddy",
|
|
193
|
+
"FACTORY_PROJECT_DIR": "droid",
|
|
194
|
+
"GEMINI_PROJECT_DIR": "gemini",
|
|
195
|
+
"QODER_PROJECT_DIR": "qoder",
|
|
196
|
+
"KIRO_PROJECT_DIR": "kiro",
|
|
197
|
+
"COPILOT_PROJECT_DIR": "copilot",
|
|
198
|
+
}
|
|
199
|
+
for env_name, platform in env_map.items():
|
|
200
|
+
if os.environ.get(env_name):
|
|
201
|
+
return platform
|
|
202
|
+
script_parts = set(Path(sys.argv[0]).parts)
|
|
203
|
+
if ".claude" in script_parts:
|
|
204
|
+
return "claude"
|
|
205
|
+
if ".cursor" in script_parts:
|
|
206
|
+
return "cursor"
|
|
207
|
+
if ".codex" in script_parts:
|
|
208
|
+
return "codex"
|
|
209
|
+
if ".gemini" in script_parts:
|
|
210
|
+
return "gemini"
|
|
211
|
+
if ".qoder" in script_parts:
|
|
212
|
+
return "qoder"
|
|
213
|
+
if ".codebuddy" in script_parts:
|
|
214
|
+
return "codebuddy"
|
|
215
|
+
if ".factory" in script_parts:
|
|
216
|
+
return "droid"
|
|
217
|
+
if ".kiro" in script_parts:
|
|
218
|
+
return "kiro"
|
|
219
|
+
return None
|
|
220
|
+
|
|
221
|
+
|
|
222
|
+
def _resolve_context_key(trellis_dir: Path, input_data: dict) -> str | None:
|
|
223
|
+
scripts_dir = trellis_dir / "scripts"
|
|
224
|
+
if str(scripts_dir) not in sys.path:
|
|
225
|
+
sys.path.insert(0, str(scripts_dir))
|
|
226
|
+
from common.active_task import resolve_context_key # type: ignore[import-not-found]
|
|
227
|
+
|
|
228
|
+
return resolve_context_key(input_data, platform=_detect_platform(input_data))
|
|
229
|
+
|
|
230
|
+
|
|
231
|
+
def _persist_context_key_for_bash(context_key: str | None) -> None:
|
|
232
|
+
"""Expose Trellis session identity to later Claude Code Bash commands.
|
|
233
|
+
|
|
234
|
+
Claude Code SessionStart hooks can append exports to CLAUDE_ENV_FILE; those
|
|
235
|
+
variables are then available to Bash tools in the same conversation. Without
|
|
236
|
+
this bridge, `task.py start` has hook stdin during SessionStart but no
|
|
237
|
+
session identity when the AI later runs it as a normal shell command.
|
|
238
|
+
"""
|
|
239
|
+
if not context_key:
|
|
240
|
+
return
|
|
241
|
+
env_file = os.environ.get("CLAUDE_ENV_FILE")
|
|
242
|
+
if not env_file:
|
|
243
|
+
return
|
|
244
|
+
try:
|
|
245
|
+
with open(env_file, "a", encoding="utf-8") as handle:
|
|
246
|
+
handle.write(f"export TRELLIS_CONTEXT_ID={shlex.quote(context_key)}\n")
|
|
247
|
+
except OSError:
|
|
248
|
+
pass
|
|
249
|
+
|
|
250
|
+
|
|
251
|
+
def _resolve_active_task(trellis_dir: Path, input_data: dict):
|
|
252
|
+
scripts_dir = trellis_dir / "scripts"
|
|
253
|
+
if str(scripts_dir) not in sys.path:
|
|
254
|
+
sys.path.insert(0, str(scripts_dir))
|
|
255
|
+
from common.active_task import resolve_active_task # type: ignore[import-not-found]
|
|
256
|
+
|
|
257
|
+
return resolve_active_task(
|
|
258
|
+
trellis_dir.parent,
|
|
259
|
+
input_data,
|
|
260
|
+
platform=_detect_platform(input_data),
|
|
261
|
+
)
|
|
262
|
+
|
|
263
|
+
|
|
264
|
+
def run_script(script_path: Path, context_key: str | None = None) -> str:
|
|
265
|
+
try:
|
|
266
|
+
if script_path.suffix == ".py":
|
|
267
|
+
# Add PYTHONIOENCODING to force UTF-8 in subprocess
|
|
268
|
+
env = os.environ.copy()
|
|
269
|
+
env["PYTHONIOENCODING"] = "utf-8"
|
|
270
|
+
if context_key:
|
|
271
|
+
env["TRELLIS_CONTEXT_ID"] = context_key
|
|
272
|
+
cmd = [sys.executable, "-W", "ignore", str(script_path)]
|
|
273
|
+
else:
|
|
274
|
+
env = os.environ.copy()
|
|
275
|
+
if context_key:
|
|
276
|
+
env["TRELLIS_CONTEXT_ID"] = context_key
|
|
277
|
+
cmd = [str(script_path)]
|
|
278
|
+
|
|
279
|
+
result = subprocess.run(
|
|
280
|
+
cmd,
|
|
281
|
+
capture_output=True,
|
|
282
|
+
text=True,
|
|
283
|
+
encoding="utf-8",
|
|
284
|
+
errors="replace",
|
|
285
|
+
timeout=5,
|
|
286
|
+
cwd=script_path.parent.parent.parent,
|
|
287
|
+
env=env,
|
|
288
|
+
)
|
|
289
|
+
return result.stdout if result.returncode == 0 else "No context available"
|
|
290
|
+
except (subprocess.TimeoutExpired, FileNotFoundError, PermissionError):
|
|
291
|
+
return "No context available"
|
|
292
|
+
|
|
293
|
+
|
|
294
|
+
def _normalize_task_ref(task_ref: str) -> str:
|
|
295
|
+
normalized = task_ref.strip()
|
|
296
|
+
if not normalized:
|
|
297
|
+
return ""
|
|
298
|
+
|
|
299
|
+
path_obj = Path(normalized)
|
|
300
|
+
if path_obj.is_absolute():
|
|
301
|
+
return str(path_obj)
|
|
302
|
+
|
|
303
|
+
normalized = normalized.replace("\\", "/")
|
|
304
|
+
while normalized.startswith("./"):
|
|
305
|
+
normalized = normalized[2:]
|
|
306
|
+
|
|
307
|
+
if normalized.startswith("tasks/"):
|
|
308
|
+
return f".trellis/{normalized}"
|
|
309
|
+
|
|
310
|
+
return normalized
|
|
311
|
+
|
|
312
|
+
|
|
313
|
+
def _resolve_task_dir(trellis_dir: Path, task_ref: str) -> Path:
|
|
314
|
+
normalized = _normalize_task_ref(task_ref)
|
|
315
|
+
path_obj = Path(normalized)
|
|
316
|
+
if path_obj.is_absolute():
|
|
317
|
+
return path_obj
|
|
318
|
+
if normalized.startswith(".trellis/"):
|
|
319
|
+
return trellis_dir.parent / path_obj
|
|
320
|
+
return trellis_dir / "tasks" / path_obj
|
|
321
|
+
|
|
322
|
+
|
|
323
|
+
def _get_task_status(trellis_dir: Path, input_data: dict) -> str:
|
|
324
|
+
"""Return compact active-task status, artifact presence, and next action."""
|
|
325
|
+
active = _resolve_active_task(trellis_dir, input_data)
|
|
326
|
+
|
|
327
|
+
if not active.task_path:
|
|
328
|
+
return (
|
|
329
|
+
"Status: NO ACTIVE TASK\n"
|
|
330
|
+
"Next-Action: Classify the current turn before creating any Trellis task. "
|
|
331
|
+
"Simple conversation / small task asks only whether this turn should create a Trellis task. "
|
|
332
|
+
"Complex task asks whether task creation and planning are allowed."
|
|
333
|
+
)
|
|
334
|
+
|
|
335
|
+
task_ref = active.task_path
|
|
336
|
+
task_dir = _resolve_task_dir(trellis_dir, task_ref)
|
|
337
|
+
if active.stale or not task_dir.is_dir():
|
|
338
|
+
return (
|
|
339
|
+
f"Status: STALE POINTER\nTask: {task_ref}\n"
|
|
340
|
+
f"Next-Action: Run `python3 ./.trellis/scripts/task.py finish` to clear the stale pointer, "
|
|
341
|
+
"then ask the user what to work on next."
|
|
342
|
+
)
|
|
343
|
+
|
|
344
|
+
task_json_path = task_dir / "task.json"
|
|
345
|
+
task_data = {}
|
|
346
|
+
if task_json_path.is_file():
|
|
347
|
+
try:
|
|
348
|
+
task_data = json.loads(task_json_path.read_text(encoding="utf-8"))
|
|
349
|
+
except (json.JSONDecodeError, PermissionError):
|
|
350
|
+
pass
|
|
351
|
+
|
|
352
|
+
task_title = task_data.get("title", task_ref)
|
|
353
|
+
task_status = task_data.get("status", "unknown")
|
|
354
|
+
artifact_names = ("prd.md", "design.md", "implement.md", "implement.jsonl", "check.jsonl")
|
|
355
|
+
present = [name for name in artifact_names if (task_dir / name).is_file()]
|
|
356
|
+
if (task_dir / "research").is_dir():
|
|
357
|
+
present.append("research/")
|
|
358
|
+
present_line = ", ".join(present) if present else "(none)"
|
|
359
|
+
|
|
360
|
+
if task_status == "completed":
|
|
361
|
+
return (
|
|
362
|
+
f"Status: COMPLETED\nTask: {task_title}\n"
|
|
363
|
+
f"Present: {present_line}\n"
|
|
364
|
+
"Next-Action: Run `/trellis:finish-work`. If the working tree is dirty, return to Phase 3.4 first."
|
|
365
|
+
)
|
|
366
|
+
|
|
367
|
+
has_prd = (task_dir / "prd.md").is_file()
|
|
368
|
+
has_design = (task_dir / "design.md").is_file()
|
|
369
|
+
has_implement_plan = (task_dir / "implement.md").is_file()
|
|
370
|
+
implement_jsonl = task_dir / "implement.jsonl"
|
|
371
|
+
check_jsonl = task_dir / "check.jsonl"
|
|
372
|
+
jsonl_ready = (
|
|
373
|
+
(not implement_jsonl.is_file() or _has_curated_jsonl_entry(implement_jsonl))
|
|
374
|
+
and (not check_jsonl.is_file() or _has_curated_jsonl_entry(check_jsonl))
|
|
375
|
+
)
|
|
376
|
+
|
|
377
|
+
if task_status == "planning" and not has_prd:
|
|
378
|
+
return (
|
|
379
|
+
f"Status: PLANNING\nTask: {task_title}\n"
|
|
380
|
+
f"Present: {present_line}\n"
|
|
381
|
+
"Next-Action: Load `trellis-brainstorm` and write `prd.md`. Stay in planning."
|
|
382
|
+
)
|
|
383
|
+
|
|
384
|
+
if task_status == "planning":
|
|
385
|
+
missing_complex = [
|
|
386
|
+
name for name, exists in (
|
|
387
|
+
("design.md", has_design),
|
|
388
|
+
("implement.md", has_implement_plan),
|
|
389
|
+
)
|
|
390
|
+
if not exists
|
|
391
|
+
]
|
|
392
|
+
next_bits: list[str] = []
|
|
393
|
+
if missing_complex:
|
|
394
|
+
next_bits.append(
|
|
395
|
+
"Lightweight task can request start review with PRD-only; "
|
|
396
|
+
f"complex task must add {', '.join(missing_complex)} before start"
|
|
397
|
+
)
|
|
398
|
+
else:
|
|
399
|
+
next_bits.append("Planning artifacts are present; ask for review before `task.py start`")
|
|
400
|
+
if not jsonl_ready:
|
|
401
|
+
next_bits.append("curate `implement.jsonl` and `check.jsonl` before sub-agent mode start")
|
|
402
|
+
return (
|
|
403
|
+
f"Status: PLANNING\nTask: {task_title}\n"
|
|
404
|
+
f"Present: {present_line}\n"
|
|
405
|
+
f"Next-Action: {'; '.join(next_bits)}. Do not enter implementation until the user confirms start."
|
|
406
|
+
)
|
|
407
|
+
|
|
408
|
+
return (
|
|
409
|
+
f"Status: {str(task_status).upper()}\nTask: {task_title}\n"
|
|
410
|
+
f"Present: {present_line}\n"
|
|
411
|
+
"Next-Action: Follow the matching per-turn workflow-state. "
|
|
412
|
+
"Implementation/check context order is jsonl entries -> `prd.md` -> `design.md if present` -> `implement.md if present`."
|
|
413
|
+
)
|
|
414
|
+
|
|
415
|
+
|
|
416
|
+
def _load_trellis_config(trellis_dir: Path, input_data: dict) -> tuple:
|
|
417
|
+
"""Load Trellis config for session-start decisions.
|
|
418
|
+
|
|
419
|
+
Returns:
|
|
420
|
+
(is_mono, packages_dict, spec_scope, task_pkg, default_pkg)
|
|
421
|
+
"""
|
|
422
|
+
scripts_dir = trellis_dir / "scripts"
|
|
423
|
+
if str(scripts_dir) not in sys.path:
|
|
424
|
+
sys.path.insert(0, str(scripts_dir))
|
|
425
|
+
|
|
426
|
+
try:
|
|
427
|
+
from common.config import get_default_package, get_packages, get_spec_scope, is_monorepo # type: ignore[import-not-found]
|
|
428
|
+
from common.paths import get_current_task # type: ignore[import-not-found]
|
|
429
|
+
|
|
430
|
+
repo_root = trellis_dir.parent
|
|
431
|
+
is_mono = is_monorepo(repo_root)
|
|
432
|
+
packages = get_packages(repo_root) or {}
|
|
433
|
+
scope = get_spec_scope(repo_root)
|
|
434
|
+
|
|
435
|
+
# Get active task's package
|
|
436
|
+
task_pkg = None
|
|
437
|
+
current = get_current_task(
|
|
438
|
+
repo_root,
|
|
439
|
+
input_data,
|
|
440
|
+
platform=_detect_platform(input_data),
|
|
441
|
+
)
|
|
442
|
+
if current:
|
|
443
|
+
task_json = repo_root / current / "task.json"
|
|
444
|
+
if task_json.is_file():
|
|
445
|
+
try:
|
|
446
|
+
data = json.loads(task_json.read_text(encoding="utf-8"))
|
|
447
|
+
if isinstance(data, dict):
|
|
448
|
+
tp = data.get("package")
|
|
449
|
+
if isinstance(tp, str) and tp:
|
|
450
|
+
task_pkg = tp
|
|
451
|
+
except (json.JSONDecodeError, OSError):
|
|
452
|
+
pass
|
|
453
|
+
|
|
454
|
+
default_pkg = get_default_package(repo_root)
|
|
455
|
+
return is_mono, packages, scope, task_pkg, default_pkg
|
|
456
|
+
except Exception:
|
|
457
|
+
return False, {}, None, None, None
|
|
458
|
+
|
|
459
|
+
|
|
460
|
+
def _check_legacy_spec(trellis_dir: Path, is_mono: bool, packages: dict) -> str | None:
|
|
461
|
+
"""Check for legacy spec directory structure in monorepo.
|
|
462
|
+
|
|
463
|
+
Returns warning message if legacy structure detected, None otherwise.
|
|
464
|
+
"""
|
|
465
|
+
if not is_mono or not packages:
|
|
466
|
+
return None
|
|
467
|
+
|
|
468
|
+
spec_dir = trellis_dir / "spec"
|
|
469
|
+
if not spec_dir.is_dir():
|
|
470
|
+
return None
|
|
471
|
+
|
|
472
|
+
# Check for legacy flat spec dirs (spec/backend/, spec/frontend/ with index.md)
|
|
473
|
+
has_legacy = False
|
|
474
|
+
for legacy_name in ("backend", "frontend"):
|
|
475
|
+
legacy_dir = spec_dir / legacy_name
|
|
476
|
+
if legacy_dir.is_dir() and (legacy_dir / "index.md").is_file():
|
|
477
|
+
has_legacy = True
|
|
478
|
+
break
|
|
479
|
+
|
|
480
|
+
if not has_legacy:
|
|
481
|
+
return None
|
|
482
|
+
|
|
483
|
+
# Check which packages are missing spec/<pkg>/ directory
|
|
484
|
+
missing = [
|
|
485
|
+
name for name in sorted(packages.keys())
|
|
486
|
+
if not (spec_dir / name).is_dir()
|
|
487
|
+
]
|
|
488
|
+
|
|
489
|
+
if not missing:
|
|
490
|
+
return None # All packages have spec dirs
|
|
491
|
+
|
|
492
|
+
if len(missing) == len(packages):
|
|
493
|
+
return (
|
|
494
|
+
f"[!] Legacy spec structure detected: found `spec/backend/` or `spec/frontend/` "
|
|
495
|
+
f"but no package-scoped `spec/<package>/` directories.\n"
|
|
496
|
+
f"Monorepo packages: {', '.join(sorted(packages.keys()))}\n"
|
|
497
|
+
f"Please reorganize: `spec/backend/` -> `spec/<package>/backend/`"
|
|
498
|
+
)
|
|
499
|
+
return (
|
|
500
|
+
f"[!] Partial spec migration detected: packages {', '.join(missing)} "
|
|
501
|
+
f"still missing `spec/<pkg>/` directory.\n"
|
|
502
|
+
f"Please complete migration for all packages."
|
|
503
|
+
)
|
|
504
|
+
|
|
505
|
+
|
|
506
|
+
def _resolve_spec_scope(
|
|
507
|
+
is_mono: bool,
|
|
508
|
+
packages: dict,
|
|
509
|
+
scope,
|
|
510
|
+
task_pkg: str | None,
|
|
511
|
+
default_pkg: str | None,
|
|
512
|
+
) -> set | None:
|
|
513
|
+
"""Resolve which packages should have their specs injected.
|
|
514
|
+
|
|
515
|
+
Returns:
|
|
516
|
+
Set of package names to include, or None for full scan.
|
|
517
|
+
"""
|
|
518
|
+
if not is_mono or not packages:
|
|
519
|
+
return None # Single-repo: full scan
|
|
520
|
+
|
|
521
|
+
if scope is None:
|
|
522
|
+
return None # No scope configured: full scan
|
|
523
|
+
|
|
524
|
+
if isinstance(scope, str) and scope == "active_task":
|
|
525
|
+
if task_pkg and task_pkg in packages:
|
|
526
|
+
return {task_pkg}
|
|
527
|
+
if default_pkg and default_pkg in packages:
|
|
528
|
+
return {default_pkg}
|
|
529
|
+
return None # Fallback to full scan
|
|
530
|
+
|
|
531
|
+
if isinstance(scope, list):
|
|
532
|
+
valid = set()
|
|
533
|
+
for entry in scope:
|
|
534
|
+
if entry in packages:
|
|
535
|
+
valid.add(entry)
|
|
536
|
+
else:
|
|
537
|
+
print(
|
|
538
|
+
f"Warning: spec_scope contains unknown package: {entry}, ignoring",
|
|
539
|
+
file=sys.stderr,
|
|
540
|
+
)
|
|
541
|
+
|
|
542
|
+
if valid:
|
|
543
|
+
# Warn if active task is out of scope
|
|
544
|
+
if task_pkg and task_pkg not in valid:
|
|
545
|
+
print(
|
|
546
|
+
f"Warning: active task package '{task_pkg}' is out of configured spec_scope",
|
|
547
|
+
file=sys.stderr,
|
|
548
|
+
)
|
|
549
|
+
return valid
|
|
550
|
+
|
|
551
|
+
# All entries invalid: fallback chain
|
|
552
|
+
print(
|
|
553
|
+
"Warning: all spec_scope entries invalid, falling back to task/default/full",
|
|
554
|
+
file=sys.stderr,
|
|
555
|
+
)
|
|
556
|
+
if task_pkg and task_pkg in packages:
|
|
557
|
+
return {task_pkg}
|
|
558
|
+
if default_pkg and default_pkg in packages:
|
|
559
|
+
return {default_pkg}
|
|
560
|
+
return None # Full scan
|
|
561
|
+
|
|
562
|
+
return None # Unknown scope type: full scan
|
|
563
|
+
|
|
564
|
+
|
|
565
|
+
def _collect_spec_index_paths(trellis_dir: Path, allowed_pkgs: set | None) -> list[str]:
|
|
566
|
+
paths: list[str] = []
|
|
567
|
+
guides_index = trellis_dir / "spec" / "guides" / "index.md"
|
|
568
|
+
if guides_index.is_file():
|
|
569
|
+
paths.append(".trellis/spec/guides/index.md")
|
|
570
|
+
|
|
571
|
+
spec_dir = trellis_dir / "spec"
|
|
572
|
+
if not spec_dir.is_dir():
|
|
573
|
+
return paths
|
|
574
|
+
|
|
575
|
+
for sub in sorted(spec_dir.iterdir()):
|
|
576
|
+
if not sub.is_dir() or sub.name.startswith(".") or sub.name == "guides":
|
|
577
|
+
continue
|
|
578
|
+
|
|
579
|
+
index_file = sub / "index.md"
|
|
580
|
+
if index_file.is_file():
|
|
581
|
+
paths.append(f".trellis/spec/{sub.name}/index.md")
|
|
582
|
+
continue
|
|
583
|
+
|
|
584
|
+
if allowed_pkgs is not None and sub.name not in allowed_pkgs:
|
|
585
|
+
continue
|
|
586
|
+
for nested in sorted(sub.iterdir()):
|
|
587
|
+
if not nested.is_dir():
|
|
588
|
+
continue
|
|
589
|
+
nested_index = nested / "index.md"
|
|
590
|
+
if nested_index.is_file():
|
|
591
|
+
paths.append(f".trellis/spec/{sub.name}/{nested.name}/index.md")
|
|
592
|
+
|
|
593
|
+
return paths
|
|
594
|
+
|
|
595
|
+
|
|
596
|
+
def _build_compact_current_state(
|
|
597
|
+
trellis_dir: Path,
|
|
598
|
+
input_data: dict,
|
|
599
|
+
spec_index_paths: list[str],
|
|
600
|
+
) -> str:
|
|
601
|
+
repo_root = trellis_dir.parent
|
|
602
|
+
lines: list[str] = []
|
|
603
|
+
|
|
604
|
+
try:
|
|
605
|
+
from common.paths import get_active_journal_file, get_developer, get_tasks_dir, count_lines # type: ignore[import-not-found]
|
|
606
|
+
from common.tasks import iter_active_tasks # type: ignore[import-not-found]
|
|
607
|
+
except Exception:
|
|
608
|
+
get_active_journal_file = None # type: ignore[assignment]
|
|
609
|
+
get_developer = None # type: ignore[assignment]
|
|
610
|
+
get_tasks_dir = None # type: ignore[assignment]
|
|
611
|
+
count_lines = None # type: ignore[assignment]
|
|
612
|
+
iter_active_tasks = None # type: ignore[assignment]
|
|
613
|
+
|
|
614
|
+
developer = get_developer(repo_root) if get_developer else None
|
|
615
|
+
lines.append(f"Developer: {developer or '(not initialized)'}")
|
|
616
|
+
lines.append(_format_git_state(repo_root))
|
|
617
|
+
|
|
618
|
+
active = _resolve_active_task(trellis_dir, input_data)
|
|
619
|
+
if active.task_path:
|
|
620
|
+
task_dir = _resolve_task_dir(trellis_dir, active.task_path)
|
|
621
|
+
status = "unknown"
|
|
622
|
+
task_json = task_dir / "task.json"
|
|
623
|
+
if task_json.is_file():
|
|
624
|
+
try:
|
|
625
|
+
data = json.loads(task_json.read_text(encoding="utf-8"))
|
|
626
|
+
if isinstance(data, dict):
|
|
627
|
+
status = str(data.get("status") or "unknown")
|
|
628
|
+
except (json.JSONDecodeError, OSError):
|
|
629
|
+
pass
|
|
630
|
+
lines.append(f"Current task: {_repo_relative(repo_root, task_dir)}; status={status}.")
|
|
631
|
+
else:
|
|
632
|
+
lines.append("Current task: none.")
|
|
633
|
+
|
|
634
|
+
if get_tasks_dir and iter_active_tasks:
|
|
635
|
+
try:
|
|
636
|
+
task_count = sum(1 for _ in iter_active_tasks(get_tasks_dir(repo_root)))
|
|
637
|
+
lines.append(
|
|
638
|
+
f"Active tasks: {task_count} total. Use `python3 ./.trellis/scripts/task.py list --mine` only if needed."
|
|
639
|
+
)
|
|
640
|
+
except Exception:
|
|
641
|
+
pass
|
|
642
|
+
|
|
643
|
+
if get_active_journal_file and count_lines:
|
|
644
|
+
journal = get_active_journal_file(repo_root)
|
|
645
|
+
if journal:
|
|
646
|
+
lines.append(
|
|
647
|
+
f"Journal: {_repo_relative(repo_root, journal)}, {count_lines(journal)} / 2000 lines."
|
|
648
|
+
)
|
|
649
|
+
|
|
650
|
+
if spec_index_paths:
|
|
651
|
+
lines.append(f"Spec indexes: {len(spec_index_paths)} available.")
|
|
652
|
+
|
|
653
|
+
return "\n".join(lines)
|
|
654
|
+
|
|
655
|
+
|
|
656
|
+
def _extract_range(content: str, start_header: str, end_header: str) -> str:
|
|
657
|
+
"""Extract lines starting at `## start_header` up to (but excluding) `## end_header`.
|
|
658
|
+
|
|
659
|
+
Both parameters are full header lines WITHOUT the `## ` prefix (e.g. "Phase Index").
|
|
660
|
+
Returns empty string if start header is not found.
|
|
661
|
+
End header missing → extracts to end of file.
|
|
662
|
+
"""
|
|
663
|
+
lines = content.splitlines()
|
|
664
|
+
start: int | None = None
|
|
665
|
+
end: int = len(lines)
|
|
666
|
+
start_match = f"## {start_header}"
|
|
667
|
+
end_match = f"## {end_header}"
|
|
668
|
+
for i, line in enumerate(lines):
|
|
669
|
+
stripped = line.strip()
|
|
670
|
+
if start is None and stripped == start_match:
|
|
671
|
+
start = i
|
|
672
|
+
continue
|
|
673
|
+
if start is not None and stripped == end_match:
|
|
674
|
+
end = i
|
|
675
|
+
break
|
|
676
|
+
if start is None:
|
|
677
|
+
return ""
|
|
678
|
+
return "\n".join(lines[start:end]).rstrip()
|
|
679
|
+
|
|
680
|
+
|
|
681
|
+
_BREADCRUMB_TAG_RE = re.compile(
|
|
682
|
+
r"\[workflow-state:([A-Za-z0-9_-]+)\]\s*\n.*?\n\s*\[/workflow-state:\1\]",
|
|
683
|
+
re.DOTALL,
|
|
684
|
+
)
|
|
685
|
+
|
|
686
|
+
|
|
687
|
+
def _strip_breadcrumb_tag_blocks(content: str) -> str:
|
|
688
|
+
"""Remove `[workflow-state:STATUS]...[/workflow-state:STATUS]` blocks.
|
|
689
|
+
|
|
690
|
+
The tag blocks live inside `## Phase Index` (since v0.5.0-rc.0, when
|
|
691
|
+
they were colocated with their phase summaries) and are consumed by the
|
|
692
|
+
UserPromptSubmit hook (`inject-workflow-state.py`). The session-start
|
|
693
|
+
payload already covers the full step bodies, so re-inlining the
|
|
694
|
+
breadcrumbs here would just duplicate context.
|
|
695
|
+
"""
|
|
696
|
+
stripped = _BREADCRUMB_TAG_RE.sub("", content)
|
|
697
|
+
stripped = re.sub(r"<!--.*?-->", "", stripped, flags=re.DOTALL)
|
|
698
|
+
stripped = re.sub(r"^\[(?!/?workflow-state:)/?[^\]\n]+\]\s*\n?", "", stripped, flags=re.MULTILINE)
|
|
699
|
+
return re.sub(r"\n{3,}", "\n\n", stripped).strip()
|
|
700
|
+
|
|
701
|
+
|
|
702
|
+
def _build_workflow_overview(workflow_path: Path) -> str:
|
|
703
|
+
"""Inject only the compact Phase Index summary for SessionStart."""
|
|
704
|
+
content = read_file(workflow_path)
|
|
705
|
+
if not content:
|
|
706
|
+
return "No workflow.md found"
|
|
707
|
+
|
|
708
|
+
out_lines = [
|
|
709
|
+
"# Development Workflow - Session Summary",
|
|
710
|
+
"Full guide: .trellis/workflow.md. Step detail: `python3 ./.trellis/scripts/get_context.py --mode phase --step <X.Y>`.",
|
|
711
|
+
"",
|
|
712
|
+
]
|
|
713
|
+
|
|
714
|
+
phases = _extract_range(content, "Phase Index", "Phase 1: Plan")
|
|
715
|
+
if phases:
|
|
716
|
+
out_lines.append(_strip_breadcrumb_tag_blocks(phases).rstrip())
|
|
717
|
+
|
|
718
|
+
return "\n".join(out_lines).rstrip()
|
|
719
|
+
|
|
720
|
+
|
|
721
|
+
def main():
|
|
722
|
+
if should_skip_injection():
|
|
723
|
+
sys.exit(0)
|
|
724
|
+
|
|
725
|
+
try:
|
|
726
|
+
hook_input = json.loads(sys.stdin.read())
|
|
727
|
+
if not isinstance(hook_input, dict):
|
|
728
|
+
hook_input = {}
|
|
729
|
+
except (json.JSONDecodeError, ValueError):
|
|
730
|
+
hook_input = {}
|
|
731
|
+
|
|
732
|
+
# Try platform-specific env vars, hook cwd, fallback to cwd
|
|
733
|
+
project_dir_env_vars = [
|
|
734
|
+
"CLAUDE_PROJECT_DIR",
|
|
735
|
+
"QODER_PROJECT_DIR",
|
|
736
|
+
"CODEBUDDY_PROJECT_DIR",
|
|
737
|
+
"FACTORY_PROJECT_DIR",
|
|
738
|
+
"CURSOR_PROJECT_DIR",
|
|
739
|
+
"GEMINI_PROJECT_DIR",
|
|
740
|
+
"KIRO_PROJECT_DIR",
|
|
741
|
+
"COPILOT_PROJECT_DIR",
|
|
742
|
+
]
|
|
743
|
+
project_dir = None
|
|
744
|
+
for var in project_dir_env_vars:
|
|
745
|
+
val = os.environ.get(var)
|
|
746
|
+
if val:
|
|
747
|
+
project_dir = Path(_normalize_windows_shell_path(val)).resolve()
|
|
748
|
+
break
|
|
749
|
+
if project_dir is None:
|
|
750
|
+
project_dir = Path(_normalize_windows_shell_path(hook_input.get("cwd", "."))).resolve()
|
|
751
|
+
|
|
752
|
+
trellis_dir = project_dir / ".trellis"
|
|
753
|
+
context_key = _resolve_context_key(trellis_dir, hook_input)
|
|
754
|
+
_persist_context_key_for_bash(context_key)
|
|
755
|
+
|
|
756
|
+
# Load config for scope filtering and legacy detection
|
|
757
|
+
is_mono, packages, scope_config, task_pkg, default_pkg = _load_trellis_config(
|
|
758
|
+
trellis_dir,
|
|
759
|
+
hook_input,
|
|
760
|
+
)
|
|
761
|
+
allowed_pkgs = _resolve_spec_scope(is_mono, packages, scope_config, task_pkg, default_pkg)
|
|
762
|
+
|
|
763
|
+
output = StringIO()
|
|
764
|
+
|
|
765
|
+
spec_index_paths = _collect_spec_index_paths(trellis_dir, allowed_pkgs)
|
|
766
|
+
|
|
767
|
+
output.write("""<session-context>
|
|
768
|
+
Trellis compact SessionStart context. Use it to orient the session; load details on demand.
|
|
769
|
+
</session-context>
|
|
770
|
+
|
|
771
|
+
""")
|
|
772
|
+
output.write(FIRST_REPLY_NOTICE)
|
|
773
|
+
output.write("\n\n")
|
|
774
|
+
|
|
775
|
+
# Legacy migration warning
|
|
776
|
+
legacy_warning = _check_legacy_spec(trellis_dir, is_mono, packages)
|
|
777
|
+
if legacy_warning:
|
|
778
|
+
output.write(f"<migration-warning>\n{legacy_warning}\n</migration-warning>\n\n")
|
|
779
|
+
|
|
780
|
+
output.write("<current-state>\n")
|
|
781
|
+
output.write(_build_compact_current_state(trellis_dir, hook_input, spec_index_paths))
|
|
782
|
+
output.write("\n</current-state>\n\n")
|
|
783
|
+
|
|
784
|
+
output.write("<trellis-workflow>\n")
|
|
785
|
+
output.write(_build_workflow_overview(trellis_dir / "workflow.md"))
|
|
786
|
+
output.write("\n</trellis-workflow>\n\n")
|
|
787
|
+
|
|
788
|
+
output.write("<guidelines>\n")
|
|
789
|
+
output.write(
|
|
790
|
+
"Task context order for implementation/check: jsonl entries -> `prd.md` -> "
|
|
791
|
+
"`design.md if present` -> `implement.md if present`. Missing optional artifacts "
|
|
792
|
+
"are skipped for lightweight tasks.\n\n"
|
|
793
|
+
)
|
|
794
|
+
|
|
795
|
+
if spec_index_paths:
|
|
796
|
+
output.write("## Available indexes (read on demand)\n")
|
|
797
|
+
for p in spec_index_paths:
|
|
798
|
+
output.write(f"- {p}\n")
|
|
799
|
+
output.write("\n")
|
|
800
|
+
|
|
801
|
+
output.write(
|
|
802
|
+
"Discover more via: "
|
|
803
|
+
"`python3 ./.trellis/scripts/get_context.py --mode packages`\n"
|
|
804
|
+
)
|
|
805
|
+
output.write("</guidelines>\n\n")
|
|
806
|
+
|
|
807
|
+
# Check task status and inject structured tag
|
|
808
|
+
task_status = _get_task_status(trellis_dir, hook_input)
|
|
809
|
+
output.write(f"<task-status>\n{task_status}\n</task-status>\n\n")
|
|
810
|
+
|
|
811
|
+
output.write("""<ready>
|
|
812
|
+
Context loaded. Follow <task-status>. Load workflow/spec/task details only when needed.
|
|
813
|
+
</ready>""")
|
|
814
|
+
|
|
815
|
+
result = {
|
|
816
|
+
"hookSpecificOutput": {
|
|
817
|
+
"hookEventName": "SessionStart",
|
|
818
|
+
"additionalContext": output.getvalue(),
|
|
819
|
+
}
|
|
820
|
+
}
|
|
821
|
+
|
|
822
|
+
# Output JSON - stdout is already configured for UTF-8
|
|
823
|
+
print(json.dumps(result, ensure_ascii=False), flush=True)
|
|
824
|
+
|
|
825
|
+
|
|
826
|
+
if __name__ == "__main__":
|
|
827
|
+
main()
|