deepspider 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.
Files changed (261) hide show
  1. package/.claude/agents/check.md +122 -0
  2. package/.claude/agents/debug.md +106 -0
  3. package/.claude/agents/dispatch.md +214 -0
  4. package/.claude/agents/implement.md +96 -0
  5. package/.claude/agents/plan.md +396 -0
  6. package/.claude/agents/research.md +120 -0
  7. package/.claude/commands/evolve/merge.md +80 -0
  8. package/.claude/commands/trellis/before-backend-dev.md +13 -0
  9. package/.claude/commands/trellis/before-frontend-dev.md +13 -0
  10. package/.claude/commands/trellis/break-loop.md +107 -0
  11. package/.claude/commands/trellis/check-backend.md +13 -0
  12. package/.claude/commands/trellis/check-cross-layer.md +153 -0
  13. package/.claude/commands/trellis/check-frontend.md +13 -0
  14. package/.claude/commands/trellis/create-command.md +154 -0
  15. package/.claude/commands/trellis/finish-work.md +129 -0
  16. package/.claude/commands/trellis/integrate-skill.md +219 -0
  17. package/.claude/commands/trellis/onboard.md +358 -0
  18. package/.claude/commands/trellis/parallel.md +193 -0
  19. package/.claude/commands/trellis/record-session.md +62 -0
  20. package/.claude/commands/trellis/start.md +280 -0
  21. package/.claude/commands/trellis/update-spec.md +213 -0
  22. package/.claude/hooks/inject-subagent-context.py +758 -0
  23. package/.claude/hooks/ralph-loop.py +374 -0
  24. package/.claude/hooks/session-start.py +126 -0
  25. package/.claude/settings.json +41 -0
  26. package/.claude/skills/deepagents-guide/SKILL.md +428 -0
  27. package/.cursor/commands/trellis-before-backend-dev.md +13 -0
  28. package/.cursor/commands/trellis-before-frontend-dev.md +13 -0
  29. package/.cursor/commands/trellis-break-loop.md +107 -0
  30. package/.cursor/commands/trellis-check-backend.md +13 -0
  31. package/.cursor/commands/trellis-check-cross-layer.md +153 -0
  32. package/.cursor/commands/trellis-check-frontend.md +13 -0
  33. package/.cursor/commands/trellis-create-command.md +154 -0
  34. package/.cursor/commands/trellis-finish-work.md +129 -0
  35. package/.cursor/commands/trellis-integrate-skill.md +219 -0
  36. package/.cursor/commands/trellis-onboard.md +358 -0
  37. package/.cursor/commands/trellis-record-session.md +62 -0
  38. package/.cursor/commands/trellis-start.md +156 -0
  39. package/.cursor/commands/trellis-update-spec.md +213 -0
  40. package/.env.example +11 -0
  41. package/.husky/pre-commit +1 -0
  42. package/.mcp.json +8 -0
  43. package/.trellis/.template-hashes.json +65 -0
  44. package/.trellis/.version +1 -0
  45. package/.trellis/scripts/add-session.sh +384 -0
  46. package/.trellis/scripts/common/developer.sh +129 -0
  47. package/.trellis/scripts/common/git-context.sh +263 -0
  48. package/.trellis/scripts/common/paths.sh +208 -0
  49. package/.trellis/scripts/common/phase.sh +150 -0
  50. package/.trellis/scripts/common/registry.sh +247 -0
  51. package/.trellis/scripts/common/task-queue.sh +142 -0
  52. package/.trellis/scripts/common/task-utils.sh +151 -0
  53. package/.trellis/scripts/common/worktree.sh +128 -0
  54. package/.trellis/scripts/create-bootstrap.sh +299 -0
  55. package/.trellis/scripts/get-context.sh +7 -0
  56. package/.trellis/scripts/get-developer.sh +15 -0
  57. package/.trellis/scripts/init-developer.sh +34 -0
  58. package/.trellis/scripts/multi-agent/cleanup.sh +396 -0
  59. package/.trellis/scripts/multi-agent/create-pr.sh +241 -0
  60. package/.trellis/scripts/multi-agent/plan.sh +207 -0
  61. package/.trellis/scripts/multi-agent/start.sh +310 -0
  62. package/.trellis/scripts/multi-agent/status.sh +828 -0
  63. package/.trellis/scripts/task.sh +1118 -0
  64. package/.trellis/spec/backend/deepagents-guide.md +337 -0
  65. package/.trellis/spec/backend/directory-structure.md +126 -0
  66. package/.trellis/spec/backend/examples/skills/deepagents-guide/README.md +11 -0
  67. package/.trellis/spec/backend/examples/skills/deepagents-guide/agent.js.template +20 -0
  68. package/.trellis/spec/backend/examples/skills/deepagents-guide/skills-config.js.template +13 -0
  69. package/.trellis/spec/backend/examples/skills/deepagents-guide/subagent.js.template +19 -0
  70. package/.trellis/spec/backend/hook-guidelines.md +178 -0
  71. package/.trellis/spec/backend/index.md +36 -0
  72. package/.trellis/spec/backend/quality-guidelines.md +201 -0
  73. package/.trellis/spec/backend/state-management.md +76 -0
  74. package/.trellis/spec/backend/tool-guidelines.md +144 -0
  75. package/.trellis/spec/backend/type-safety.md +71 -0
  76. package/.trellis/spec/guides/code-reuse-thinking-guide.md +92 -0
  77. package/.trellis/spec/guides/cross-layer-thinking-guide.md +94 -0
  78. package/.trellis/spec/guides/index.md +79 -0
  79. package/.trellis/tasks/archive/02-02-evolving-skills/prd.md +61 -0
  80. package/.trellis/tasks/archive/02-02-evolving-skills/task.json +29 -0
  81. package/.trellis/tasks/archive/2026-02/00-bootstrap-guidelines/prd.md +86 -0
  82. package/.trellis/tasks/archive/2026-02/00-bootstrap-guidelines/task.json +27 -0
  83. package/.trellis/tasks/archive/2026-02/02-02-skills-system/check.jsonl +3 -0
  84. package/.trellis/tasks/archive/2026-02/02-02-skills-system/debug.jsonl +2 -0
  85. package/.trellis/tasks/archive/2026-02/02-02-skills-system/implement.jsonl +5 -0
  86. package/.trellis/tasks/archive/2026-02/02-02-skills-system/prd.md +33 -0
  87. package/.trellis/tasks/archive/2026-02/02-02-skills-system/task.json +41 -0
  88. package/.trellis/workflow.md +407 -0
  89. package/.trellis/workspace/index.md +123 -0
  90. package/.trellis/workspace/pony/index.md +40 -0
  91. package/.trellis/workspace/pony/journal-1.md +7 -0
  92. package/.trellis/worktree.yaml +47 -0
  93. package/AGENTS.md +18 -0
  94. package/CLAUDE.md +292 -0
  95. package/README.md +134 -0
  96. package/agents/deepspider.md +142 -0
  97. package/docs/DEBUG.md +42 -0
  98. package/docs/GUIDE.md +334 -0
  99. package/docs/PROMPT.md +60 -0
  100. package/docs/USAGE.md +226 -0
  101. package/eslint.config.js +51 -0
  102. package/package.json +78 -0
  103. package/requirements-crypto.txt +14 -0
  104. package/src/agent/index.js +97 -0
  105. package/src/agent/logger.js +164 -0
  106. package/src/agent/middleware/filterTools.js +64 -0
  107. package/src/agent/middleware/report.js +79 -0
  108. package/src/agent/prompts/system.js +315 -0
  109. package/src/agent/run.js +575 -0
  110. package/src/agent/skills/anti-detect/SKILL.md +28 -0
  111. package/src/agent/skills/anti-detect/evolved.md +12 -0
  112. package/src/agent/skills/captcha/SKILL.md +37 -0
  113. package/src/agent/skills/captcha/evolved.md +12 -0
  114. package/src/agent/skills/config.js +30 -0
  115. package/src/agent/skills/crawler/SKILL.md +9 -0
  116. package/src/agent/skills/crawler/evolved.md +16 -0
  117. package/src/agent/skills/dynamic-analysis/SKILL.md +91 -0
  118. package/src/agent/skills/dynamic-analysis/evolved.md +12 -0
  119. package/src/agent/skills/env/SKILL.md +72 -0
  120. package/src/agent/skills/env/evolved.md +12 -0
  121. package/src/agent/skills/evolve.js +79 -0
  122. package/src/agent/skills/general/SKILL.md +12 -0
  123. package/src/agent/skills/general/evolved.md +12 -0
  124. package/src/agent/skills/js2python/SKILL.md +30 -0
  125. package/src/agent/skills/js2python/evolved.md +13 -0
  126. package/src/agent/skills/report/SKILL.md +21 -0
  127. package/src/agent/skills/report/evolved.md +12 -0
  128. package/src/agent/skills/sandbox/SKILL.md +22 -0
  129. package/src/agent/skills/sandbox/evolved.md +16 -0
  130. package/src/agent/skills/static-analysis/SKILL.md +93 -0
  131. package/src/agent/skills/static-analysis/evolved.md +12 -0
  132. package/src/agent/skills/xpath/SKILL.md +119 -0
  133. package/src/agent/subagents/anti-detect.js +45 -0
  134. package/src/agent/subagents/captcha.js +51 -0
  135. package/src/agent/subagents/crawler.js +138 -0
  136. package/src/agent/subagents/dynamic.js +64 -0
  137. package/src/agent/subagents/env-agent.js +82 -0
  138. package/src/agent/subagents/index.js +37 -0
  139. package/src/agent/subagents/js2python.js +72 -0
  140. package/src/agent/subagents/sandbox.js +55 -0
  141. package/src/agent/subagents/static.js +66 -0
  142. package/src/agent/tools/analysis.js +135 -0
  143. package/src/agent/tools/analyzer.js +85 -0
  144. package/src/agent/tools/anti-detect.js +89 -0
  145. package/src/agent/tools/antidebug.js +64 -0
  146. package/src/agent/tools/async.js +43 -0
  147. package/src/agent/tools/browser.js +324 -0
  148. package/src/agent/tools/captcha.js +223 -0
  149. package/src/agent/tools/capture.js +179 -0
  150. package/src/agent/tools/correlate.js +303 -0
  151. package/src/agent/tools/crawler.js +116 -0
  152. package/src/agent/tools/cryptohook.js +80 -0
  153. package/src/agent/tools/debug.js +246 -0
  154. package/src/agent/tools/deobfuscator.js +90 -0
  155. package/src/agent/tools/env.js +83 -0
  156. package/src/agent/tools/envdump.js +92 -0
  157. package/src/agent/tools/evolve.js +164 -0
  158. package/src/agent/tools/extract.js +114 -0
  159. package/src/agent/tools/extractor.js +54 -0
  160. package/src/agent/tools/file.js +224 -0
  161. package/src/agent/tools/hook.js +84 -0
  162. package/src/agent/tools/hookManager.js +178 -0
  163. package/src/agent/tools/index.js +137 -0
  164. package/src/agent/tools/nodejs.js +101 -0
  165. package/src/agent/tools/patch.js +46 -0
  166. package/src/agent/tools/preprocess.js +71 -0
  167. package/src/agent/tools/profile.js +122 -0
  168. package/src/agent/tools/python.js +627 -0
  169. package/src/agent/tools/report.js +124 -0
  170. package/src/agent/tools/runtime.js +132 -0
  171. package/src/agent/tools/sandbox.js +79 -0
  172. package/src/agent/tools/store.js +73 -0
  173. package/src/agent/tools/trace.js +74 -0
  174. package/src/agent/tools/tracing.js +201 -0
  175. package/src/agent/tools/utils.js +51 -0
  176. package/src/agent/tools/verify.js +184 -0
  177. package/src/agent/tools/webcrack.js +109 -0
  178. package/src/analyzer/ASTAnalyzer.js +387 -0
  179. package/src/analyzer/CallStackAnalyzer.js +379 -0
  180. package/src/analyzer/Deobfuscator.js +289 -0
  181. package/src/analyzer/EncryptionAnalyzer.js +99 -0
  182. package/src/analyzer/index.js +22 -0
  183. package/src/browser/EnvBridge.js +186 -0
  184. package/src/browser/cdp.js +168 -0
  185. package/src/browser/client.js +197 -0
  186. package/src/browser/collector.js +444 -0
  187. package/src/browser/collectors/RequestCryptoLinker.js +109 -0
  188. package/src/browser/collectors/ResponseSearcher.js +107 -0
  189. package/src/browser/collectors/ScriptCollector.js +158 -0
  190. package/src/browser/collectors/index.js +26 -0
  191. package/src/browser/defaultHooks.js +932 -0
  192. package/src/browser/hooks/crypto.js +55 -0
  193. package/src/browser/hooks/index.js +64 -0
  194. package/src/browser/hooks/native.js +9 -0
  195. package/src/browser/hooks/network.js +33 -0
  196. package/src/browser/index.js +42 -0
  197. package/src/browser/interceptors/NetworkInterceptor.js +116 -0
  198. package/src/browser/interceptors/ScriptInterceptor.js +76 -0
  199. package/src/browser/interceptors/index.js +6 -0
  200. package/src/browser/ui/analysisPanel.js +1782 -0
  201. package/src/browser/ui/confirmDialog.js +158 -0
  202. package/src/browser/ui/panel.html +152 -0
  203. package/src/browser/ui/selector.js +170 -0
  204. package/src/config/index.js +5 -0
  205. package/src/config/paths.js +71 -0
  206. package/src/config/patterns/crypto.js +36 -0
  207. package/src/config/profiles/chrome.json +71 -0
  208. package/src/config/profiles/firefox.json +44 -0
  209. package/src/config/profiles/safari.json +38 -0
  210. package/src/core/EnvMonitor.js +200 -0
  211. package/src/core/PatchGenerator.js +278 -0
  212. package/src/core/Sandbox.js +181 -0
  213. package/src/env/AntiAntiDebug.js +111 -0
  214. package/src/env/AsyncHook.js +68 -0
  215. package/src/env/BrowserAPIList.js +265 -0
  216. package/src/env/CookieHook.js +48 -0
  217. package/src/env/CryptoHook.js +205 -0
  218. package/src/env/EnvCodeGenerator.js +157 -0
  219. package/src/env/EnvDumper.js +356 -0
  220. package/src/env/EnvExtractor.js +220 -0
  221. package/src/env/HookBase.js +618 -0
  222. package/src/env/NetworkHook.js +159 -0
  223. package/src/env/modules/bom/history.js +29 -0
  224. package/src/env/modules/bom/location.js +26 -0
  225. package/src/env/modules/bom/navigator.js +70 -0
  226. package/src/env/modules/bom/screen.js +26 -0
  227. package/src/env/modules/bom/storage.js +23 -0
  228. package/src/env/modules/dom/document.js +110 -0
  229. package/src/env/modules/dom/event.js +51 -0
  230. package/src/env/modules/index.js +34 -0
  231. package/src/env/modules/webapi/fetch.js +46 -0
  232. package/src/env/modules/webapi/url.js +47 -0
  233. package/src/env/modules/webapi/xhr.js +48 -0
  234. package/src/index.js +27 -0
  235. package/src/mcp/server.js +89 -0
  236. package/src/store/DataStore.js +708 -0
  237. package/src/store/Store.js +158 -0
  238. package/src/store/Validator.js +24 -0
  239. package/test/analyze.test.js +90 -0
  240. package/test/envdump.test.js +74 -0
  241. package/test/flow.test.js +90 -0
  242. package/test/hooks.test.js +138 -0
  243. package/test/plugin.test.js +35 -0
  244. package/test/refactor-full.test.js +30 -0
  245. package/test/refactor.test.js +21 -0
  246. package/test/samples/obfuscated.js +61 -0
  247. package/test/samples/original.js +66 -0
  248. package/test/samples/v10_eval_chain.js +52 -0
  249. package/test/samples/v11_bytecode_vm.js +81 -0
  250. package/test/samples/v12_polymorphic.js +69 -0
  251. package/test/samples/v1_ob_basic.js +98 -0
  252. package/test/samples/v2_ob_advanced.js +99 -0
  253. package/test/samples/v3_jjencode.js +77 -0
  254. package/test/samples/v4_aaencode.js +73 -0
  255. package/test/samples/v5_control_flow.js +86 -0
  256. package/test/samples/v6_string_encryption.js +71 -0
  257. package/test/samples/v7_jsvmp.js +83 -0
  258. package/test/samples/v8_anti_debug.js +79 -0
  259. package/test/samples/v9_proxy_trap.js +49 -0
  260. package/test/samples.test.js +96 -0
  261. package/test/webcrack.test.js +55 -0
@@ -0,0 +1,150 @@
1
+ #!/bin/bash
2
+ # =============================================================================
3
+ # Phase Management Utilities
4
+ # =============================================================================
5
+ # Centralized phase tracking for multi-agent pipeline
6
+ #
7
+ # Usage:
8
+ # source common/phase.sh
9
+ #
10
+ # get_current_phase "$task_json" # Returns current phase number
11
+ # get_total_phases "$task_json" # Returns total phase count
12
+ # get_phase_action "$task_json" "$phase" # Returns action name for phase
13
+ # get_phase_info "$task_json" # Returns "N/M (action)" format
14
+ # set_phase "$task_json" "$phase" # Sets current_phase
15
+ # advance_phase "$task_json" # Advances to next phase
16
+ # get_phase_for_action "$task_json" "$action" # Returns phase number for action
17
+ # =============================================================================
18
+
19
+ # Get current phase number
20
+ get_current_phase() {
21
+ local task_json="$1"
22
+ if [ ! -f "$task_json" ]; then
23
+ echo "0"
24
+ return
25
+ fi
26
+ jq -r '.current_phase // 0' "$task_json"
27
+ }
28
+
29
+ # Get total number of phases
30
+ get_total_phases() {
31
+ local task_json="$1"
32
+ if [ ! -f "$task_json" ]; then
33
+ echo "0"
34
+ return
35
+ fi
36
+ jq -r '.next_action | length // 0' "$task_json"
37
+ }
38
+
39
+ # Get action name for a specific phase
40
+ get_phase_action() {
41
+ local task_json="$1"
42
+ local phase="$2"
43
+ if [ ! -f "$task_json" ]; then
44
+ echo "unknown"
45
+ return
46
+ fi
47
+ jq -r --argjson phase "$phase" '.next_action[] | select(.phase == $phase) | .action // "unknown"' "$task_json"
48
+ }
49
+
50
+ # Get formatted phase info: "N/M (action)"
51
+ get_phase_info() {
52
+ local task_json="$1"
53
+ if [ ! -f "$task_json" ]; then
54
+ echo "N/A"
55
+ return
56
+ fi
57
+
58
+ local current_phase=$(get_current_phase "$task_json")
59
+ local total_phases=$(get_total_phases "$task_json")
60
+ local action_name=$(get_phase_action "$task_json" "$current_phase")
61
+
62
+ if [ "$current_phase" = "0" ] || [ "$current_phase" = "null" ]; then
63
+ echo "0/${total_phases} (pending)"
64
+ else
65
+ echo "${current_phase}/${total_phases} (${action_name})"
66
+ fi
67
+ }
68
+
69
+ # Set current phase to a specific value
70
+ set_phase() {
71
+ local task_json="$1"
72
+ local phase="$2"
73
+
74
+ if [ ! -f "$task_json" ]; then
75
+ echo "Error: task.json not found: $task_json" >&2
76
+ return 1
77
+ fi
78
+
79
+ jq --argjson phase "$phase" '.current_phase = $phase' "$task_json" > "${task_json}.tmp"
80
+ mv "${task_json}.tmp" "$task_json"
81
+ }
82
+
83
+ # Advance to next phase
84
+ advance_phase() {
85
+ local task_json="$1"
86
+
87
+ if [ ! -f "$task_json" ]; then
88
+ echo "Error: task.json not found: $task_json" >&2
89
+ return 1
90
+ fi
91
+
92
+ local current=$(get_current_phase "$task_json")
93
+ local total=$(get_total_phases "$task_json")
94
+ local next=$((current + 1))
95
+
96
+ if [ "$next" -gt "$total" ]; then
97
+ echo "Warning: Already at final phase" >&2
98
+ return 0
99
+ fi
100
+
101
+ set_phase "$task_json" "$next"
102
+ }
103
+
104
+ # Get phase number for a specific action name
105
+ get_phase_for_action() {
106
+ local task_json="$1"
107
+ local action="$2"
108
+
109
+ if [ ! -f "$task_json" ]; then
110
+ echo "0"
111
+ return
112
+ fi
113
+
114
+ jq -r --arg action "$action" '.next_action[] | select(.action == $action) | .phase // 0' "$task_json"
115
+ }
116
+
117
+ # Map subagent type to action name
118
+ # Used by hooks to determine which action a subagent corresponds to
119
+ map_subagent_to_action() {
120
+ local subagent_type="$1"
121
+
122
+ case "$subagent_type" in
123
+ implement) echo "implement" ;;
124
+ check) echo "check" ;;
125
+ debug) echo "debug" ;;
126
+ research) echo "research" ;;
127
+ # finish uses check agent but is a different action
128
+ *) echo "$subagent_type" ;;
129
+ esac
130
+ }
131
+
132
+ # Check if a phase is completed (current_phase > phase)
133
+ is_phase_completed() {
134
+ local task_json="$1"
135
+ local phase="$2"
136
+
137
+ local current=$(get_current_phase "$task_json")
138
+ [ "$current" -gt "$phase" ]
139
+ }
140
+
141
+ # Check if we're at a specific action
142
+ is_current_action() {
143
+ local task_json="$1"
144
+ local action="$2"
145
+
146
+ local current=$(get_current_phase "$task_json")
147
+ local action_phase=$(get_phase_for_action "$task_json" "$action")
148
+
149
+ [ "$current" = "$action_phase" ]
150
+ }
@@ -0,0 +1,247 @@
1
+ #!/bin/bash
2
+ # Registry utility functions for multi-agent pipeline
3
+ #
4
+ # Usage: source this file in other scripts
5
+ # source "$(dirname "$0")/common/registry.sh"
6
+ #
7
+ # Provides:
8
+ # registry_get_file - Get registry file path
9
+ # registry_get_agent_by_id - Find agent by ID
10
+ # registry_get_agent_by_worktree - Find agent by worktree path
11
+ # registry_get_task_dir - Get task dir for a worktree
12
+ # registry_remove_by_id - Remove agent by ID
13
+ # registry_remove_by_worktree - Remove agent by worktree path
14
+ # registry_add_agent - Add agent to registry
15
+
16
+ # Ensure dependencies are loaded
17
+ if ! type get_repo_root &>/dev/null; then
18
+ echo "Error: paths.sh must be sourced before registry.sh" >&2
19
+ exit 1
20
+ fi
21
+
22
+ if ! type get_agents_dir &>/dev/null; then
23
+ echo "Error: developer.sh must be sourced before registry.sh" >&2
24
+ exit 1
25
+ fi
26
+
27
+ # =============================================================================
28
+ # Registry File Access
29
+ # =============================================================================
30
+
31
+ # Get registry file path
32
+ # Args: [repo_root]
33
+ # Returns: path to registry.json
34
+ registry_get_file() {
35
+ local repo_root="${1:-$(get_repo_root)}"
36
+ local agents_dir=$(get_agents_dir "$repo_root")
37
+ echo "${agents_dir}/registry.json"
38
+ }
39
+
40
+ # Ensure registry file exists with valid structure
41
+ # Args: [repo_root]
42
+ _ensure_registry() {
43
+ local repo_root="${1:-$(get_repo_root)}"
44
+ local registry_file=$(registry_get_file "$repo_root")
45
+ local agents_dir=$(dirname "$registry_file")
46
+
47
+ mkdir -p "$agents_dir"
48
+
49
+ if [[ ! -f "$registry_file" ]]; then
50
+ echo '{"agents":[]}' > "$registry_file"
51
+ fi
52
+ }
53
+
54
+ # =============================================================================
55
+ # Agent Lookup
56
+ # =============================================================================
57
+
58
+ # Get agent by ID
59
+ # Args: agent_id, [repo_root]
60
+ # Returns: agent JSON object (compact), or empty if not found
61
+ registry_get_agent_by_id() {
62
+ local agent_id="$1"
63
+ local repo_root="${2:-$(get_repo_root)}"
64
+ local registry_file=$(registry_get_file "$repo_root")
65
+
66
+ if [[ ! -f "$registry_file" ]]; then
67
+ return 1
68
+ fi
69
+
70
+ local agent=$(jq -c --arg id "$agent_id" '.agents[] | select(.id == $id)' "$registry_file" 2>/dev/null)
71
+
72
+ if [[ -n "$agent" ]] && [[ "$agent" != "null" ]]; then
73
+ echo "$agent"
74
+ return 0
75
+ fi
76
+
77
+ return 1
78
+ }
79
+
80
+ # Get agent by worktree path
81
+ # Args: worktree_path, [repo_root]
82
+ # Returns: agent JSON object (compact), or empty if not found
83
+ registry_get_agent_by_worktree() {
84
+ local worktree_path="$1"
85
+ local repo_root="${2:-$(get_repo_root)}"
86
+ local registry_file=$(registry_get_file "$repo_root")
87
+
88
+ if [[ ! -f "$registry_file" ]]; then
89
+ return 1
90
+ fi
91
+
92
+ local agent=$(jq -c --arg path "$worktree_path" '.agents[] | select(.worktree_path == $path)' "$registry_file" 2>/dev/null)
93
+
94
+ if [[ -n "$agent" ]] && [[ "$agent" != "null" ]]; then
95
+ echo "$agent"
96
+ return 0
97
+ fi
98
+
99
+ return 1
100
+ }
101
+
102
+ # Search agent by ID or task_dir containing search term
103
+ # Args: search_term, [repo_root]
104
+ # Returns: first matching agent JSON object (compact), or empty if not found
105
+ registry_search_agent() {
106
+ local search="$1"
107
+ local repo_root="${2:-$(get_repo_root)}"
108
+ local registry_file=$(registry_get_file "$repo_root")
109
+
110
+ if [[ ! -f "$registry_file" ]]; then
111
+ return 1
112
+ fi
113
+
114
+ local agent=$(jq -c --arg search "$search" \
115
+ '[.agents[] | select(.id == $search or (.task_dir | contains($search)))] | first' \
116
+ "$registry_file" 2>/dev/null)
117
+
118
+ if [[ -n "$agent" ]] && [[ "$agent" != "null" ]]; then
119
+ echo "$agent"
120
+ return 0
121
+ fi
122
+
123
+ return 1
124
+ }
125
+
126
+ # Get task directory for a worktree
127
+ # Args: worktree_path, [repo_root]
128
+ # Returns: task_dir value, or empty if not found
129
+ registry_get_task_dir() {
130
+ local worktree_path="$1"
131
+ local repo_root="${2:-$(get_repo_root)}"
132
+ local registry_file=$(registry_get_file "$repo_root")
133
+
134
+ if [[ ! -f "$registry_file" ]]; then
135
+ return 1
136
+ fi
137
+
138
+ local task_dir=$(jq -r --arg path "$worktree_path" \
139
+ '.agents[] | select(.worktree_path == $path) | .task_dir' \
140
+ "$registry_file" 2>/dev/null)
141
+
142
+ if [[ -n "$task_dir" ]] && [[ "$task_dir" != "null" ]]; then
143
+ echo "$task_dir"
144
+ return 0
145
+ fi
146
+
147
+ return 1
148
+ }
149
+
150
+ # =============================================================================
151
+ # Agent Modification
152
+ # =============================================================================
153
+
154
+ # Remove agent by ID
155
+ # Args: agent_id, [repo_root]
156
+ # Returns: 0 on success
157
+ registry_remove_by_id() {
158
+ local agent_id="$1"
159
+ local repo_root="${2:-$(get_repo_root)}"
160
+ local registry_file=$(registry_get_file "$repo_root")
161
+
162
+ if [[ ! -f "$registry_file" ]]; then
163
+ return 0
164
+ fi
165
+
166
+ local updated=$(jq --arg id "$agent_id" \
167
+ '.agents = [.agents[] | select(.id != $id)]' \
168
+ "$registry_file")
169
+
170
+ echo "$updated" | jq '.' > "$registry_file"
171
+ return 0
172
+ }
173
+
174
+ # Remove agent by worktree path
175
+ # Args: worktree_path, [repo_root]
176
+ # Returns: 0 on success
177
+ registry_remove_by_worktree() {
178
+ local worktree_path="$1"
179
+ local repo_root="${2:-$(get_repo_root)}"
180
+ local registry_file=$(registry_get_file "$repo_root")
181
+
182
+ if [[ ! -f "$registry_file" ]]; then
183
+ return 0
184
+ fi
185
+
186
+ local updated=$(jq --arg path "$worktree_path" \
187
+ '.agents = [.agents[] | select(.worktree_path != $path)]' \
188
+ "$registry_file")
189
+
190
+ echo "$updated" | jq '.' > "$registry_file"
191
+ return 0
192
+ }
193
+
194
+ # Add agent to registry (replaces if same ID exists)
195
+ # Args: agent_id, worktree_path, pid, task_dir, [repo_root]
196
+ # Returns: 0 on success
197
+ registry_add_agent() {
198
+ local agent_id="$1"
199
+ local worktree_path="$2"
200
+ local pid="$3"
201
+ local task_dir="$4"
202
+ local repo_root="${5:-$(get_repo_root)}"
203
+
204
+ _ensure_registry "$repo_root"
205
+ local registry_file=$(registry_get_file "$repo_root")
206
+
207
+ local started_at=$(date -Iseconds)
208
+
209
+ # Remove existing agent with same ID
210
+ local registry=$(jq --arg id "$agent_id" \
211
+ '.agents = [.agents[] | select(.id != $id)]' \
212
+ "$registry_file")
213
+
214
+ # Create new agent record
215
+ local new_agent=$(jq -n \
216
+ --arg id "$agent_id" \
217
+ --arg worktree "$worktree_path" \
218
+ --arg pid "$pid" \
219
+ --arg started_at "$started_at" \
220
+ --arg task_dir "$task_dir" \
221
+ '{
222
+ id: $id,
223
+ worktree_path: $worktree,
224
+ pid: ($pid | tonumber),
225
+ started_at: $started_at,
226
+ task_dir: $task_dir
227
+ }')
228
+
229
+ # Add to registry
230
+ echo "$registry" | jq --argjson agent "$new_agent" '.agents += [$agent]' > "$registry_file"
231
+ return 0
232
+ }
233
+
234
+ # List all agents
235
+ # Args: [repo_root]
236
+ # Returns: JSON array of agents
237
+ registry_list_agents() {
238
+ local repo_root="${1:-$(get_repo_root)}"
239
+ local registry_file=$(registry_get_file "$repo_root")
240
+
241
+ if [[ ! -f "$registry_file" ]]; then
242
+ echo '[]'
243
+ return 0
244
+ fi
245
+
246
+ jq '.agents' "$registry_file"
247
+ }
@@ -0,0 +1,142 @@
1
+ #!/bin/bash
2
+ # Task queue utility functions
3
+ #
4
+ # Usage: source this file in other scripts
5
+ # source "$(dirname "$0")/common/task-queue.sh"
6
+ #
7
+ # Provides:
8
+ # list_pending_tasks - List tasks with pending status
9
+ # get_task_stats - Get P0/P1/P2/P3 counts
10
+
11
+ # Ensure paths.sh is loaded
12
+ if ! type get_repo_root &>/dev/null; then
13
+ echo "Error: paths.sh must be sourced before task-queue.sh" >&2
14
+ exit 1
15
+ fi
16
+
17
+ # =============================================================================
18
+ # Public Functions
19
+ # =============================================================================
20
+
21
+ # List tasks by status
22
+ # Args: [filter_status]
23
+ # Output: formatted list to stdout
24
+ list_tasks_by_status() {
25
+ local filter_status="${1:-}"
26
+ local repo_root="${2:-$(get_repo_root)}"
27
+
28
+ local tasks_dir=$(get_tasks_dir "$repo_root")
29
+
30
+ if [[ ! -d "$tasks_dir" ]]; then
31
+ return 0
32
+ fi
33
+
34
+ for d in "$tasks_dir"/*/; do
35
+ if [[ -d "$d" ]] && [[ "$(basename "$d")" != "archive" ]]; then
36
+ local task_json="$d/$FILE_TASK_JSON"
37
+ if [[ -f "$task_json" ]]; then
38
+ local id=$(jq -r '.id' "$task_json")
39
+ local title=$(jq -r '.title // .name' "$task_json")
40
+ local priority=$(jq -r '.priority // "P2"' "$task_json")
41
+ local status=$(jq -r '.status // "planning"' "$task_json")
42
+ local assignee=$(jq -r '.assignee // "-"' "$task_json")
43
+
44
+ # Apply filter
45
+ if [[ -n "$filter_status" ]] && [[ "$status" != "$filter_status" ]]; then
46
+ continue
47
+ fi
48
+
49
+ echo "$priority|$id|$title|$status|$assignee"
50
+ fi
51
+ fi
52
+ done
53
+ }
54
+
55
+ # List pending tasks
56
+ list_pending_tasks() {
57
+ list_tasks_by_status "planning" "$@"
58
+ }
59
+
60
+ # List tasks assigned to a specific developer
61
+ # Args: developer_name, [filter_status], [repo_root]
62
+ # Output: formatted list to stdout
63
+ list_tasks_by_assignee() {
64
+ local assignee="$1"
65
+ local filter_status="${2:-}"
66
+ local repo_root="${3:-$(get_repo_root)}"
67
+
68
+ local tasks_dir=$(get_tasks_dir "$repo_root")
69
+
70
+ if [[ ! -d "$tasks_dir" ]]; then
71
+ return 0
72
+ fi
73
+
74
+ for d in "$tasks_dir"/*/; do
75
+ if [[ -d "$d" ]] && [[ "$(basename "$d")" != "archive" ]]; then
76
+ local task_json="$d/$FILE_TASK_JSON"
77
+ if [[ -f "$task_json" ]]; then
78
+ local id=$(jq -r '.id' "$task_json")
79
+ local title=$(jq -r '.title // .name' "$task_json")
80
+ local priority=$(jq -r '.priority // "P2"' "$task_json")
81
+ local status=$(jq -r '.status // "planning"' "$task_json")
82
+ local task_assignee=$(jq -r '.assignee // "-"' "$task_json")
83
+
84
+ # Apply assignee filter
85
+ if [[ "$task_assignee" != "$assignee" ]]; then
86
+ continue
87
+ fi
88
+
89
+ # Apply status filter
90
+ if [[ -n "$filter_status" ]] && [[ "$status" != "$filter_status" ]]; then
91
+ continue
92
+ fi
93
+
94
+ echo "$priority|$id|$title|$status|$task_assignee"
95
+ fi
96
+ fi
97
+ done
98
+ }
99
+
100
+ # List my tasks (current developer)
101
+ # Args: [filter_status], [repo_root]
102
+ list_my_tasks() {
103
+ local filter_status="${1:-}"
104
+ local repo_root="${2:-$(get_repo_root)}"
105
+ local developer=$(get_developer "$repo_root")
106
+
107
+ if [[ -z "$developer" ]]; then
108
+ echo "Error: Developer not set" >&2
109
+ return 1
110
+ fi
111
+
112
+ list_tasks_by_assignee "$developer" "$filter_status" "$repo_root"
113
+ }
114
+
115
+ # Get task statistics
116
+ # Output: "P0:N P1:N P2:N P3:N Total:N"
117
+ get_task_stats() {
118
+ local repo_root="${1:-$(get_repo_root)}"
119
+ local tasks_dir=$(get_tasks_dir "$repo_root")
120
+
121
+ local p0=0 p1=0 p2=0 p3=0 total=0
122
+
123
+ if [[ -d "$tasks_dir" ]]; then
124
+ for d in "$tasks_dir"/*/; do
125
+ if [[ -d "$d" ]] && [[ "$(basename "$d")" != "archive" ]]; then
126
+ local task_json="$d/$FILE_TASK_JSON"
127
+ if [[ -f "$task_json" ]]; then
128
+ local priority=$(jq -r '.priority // "P2"' "$task_json" 2>/dev/null)
129
+ case "$priority" in
130
+ P0) ((p0++)) ;;
131
+ P1) ((p1++)) ;;
132
+ P2) ((p2++)) ;;
133
+ P3) ((p3++)) ;;
134
+ esac
135
+ ((total++))
136
+ fi
137
+ fi
138
+ done
139
+ fi
140
+
141
+ echo "P0:$p0 P1:$p1 P2:$p2 P3:$p3 Total:$total"
142
+ }
@@ -0,0 +1,151 @@
1
+ #!/bin/bash
2
+ # Task utility functions
3
+ #
4
+ # Usage: source this file in other scripts
5
+ # source "$(dirname "$0")/common/task-utils.sh"
6
+ #
7
+ # Provides:
8
+ # is_safe_task_path - Validate task path is safe to operate on
9
+ # find_task_by_name - Find task directory by name
10
+ # archive_task_dir - Archive task to monthly directory
11
+
12
+ # Ensure dependencies are loaded
13
+ if ! type get_repo_root &>/dev/null; then
14
+ echo "Error: paths.sh must be sourced before task-utils.sh" >&2
15
+ exit 1
16
+ fi
17
+
18
+ # =============================================================================
19
+ # Path Safety
20
+ # =============================================================================
21
+
22
+ # Check if a relative task path is safe to operate on
23
+ # Args: task_path (relative), repo_root
24
+ # Returns: 0 if safe, 1 if dangerous
25
+ # Outputs: error message to stderr if unsafe
26
+ is_safe_task_path() {
27
+ local task_path="$1"
28
+ local repo_root="${2:-$(get_repo_root)}"
29
+
30
+ # Check empty or null
31
+ if [[ -z "$task_path" ]] || [[ "$task_path" = "null" ]]; then
32
+ echo "Error: empty or null task path" >&2
33
+ return 1
34
+ fi
35
+
36
+ # Reject absolute paths
37
+ if [[ "$task_path" = /* ]]; then
38
+ echo "Error: absolute path not allowed: $task_path" >&2
39
+ return 1
40
+ fi
41
+
42
+ # Reject ".", "..", paths starting with "./" or "../", or containing ".."
43
+ if [[ "$task_path" = "." ]] || [[ "$task_path" = ".." ]] || \
44
+ [[ "$task_path" = "./" ]] || [[ "$task_path" == ./* ]] || \
45
+ [[ "$task_path" == *".."* ]]; then
46
+ echo "Error: path traversal not allowed: $task_path" >&2
47
+ return 1
48
+ fi
49
+
50
+ # Final check: ensure resolved path is not the repo root
51
+ local abs_path="${repo_root}/${task_path}"
52
+ if [[ -e "$abs_path" ]]; then
53
+ local resolved=$(realpath "$abs_path" 2>/dev/null)
54
+ local root_resolved=$(realpath "$repo_root" 2>/dev/null)
55
+ if [[ "$resolved" = "$root_resolved" ]]; then
56
+ echo "Error: path resolves to repo root: $task_path" >&2
57
+ return 1
58
+ fi
59
+ fi
60
+
61
+ return 0
62
+ }
63
+
64
+ # =============================================================================
65
+ # Task Lookup
66
+ # =============================================================================
67
+
68
+ # Find task directory by name (exact or suffix match)
69
+ # Args: task_name, tasks_dir
70
+ # Returns: absolute path to task directory, or empty if not found
71
+ find_task_by_name() {
72
+ local task_name="$1"
73
+ local tasks_dir="$2"
74
+
75
+ if [[ -z "$task_name" ]] || [[ -z "$tasks_dir" ]]; then
76
+ return 1
77
+ fi
78
+
79
+ # Try exact match first
80
+ local task_dir=$(find "$tasks_dir" -maxdepth 1 -type d -name "${task_name}" 2>/dev/null | head -1)
81
+
82
+ # Try suffix match (e.g., "my-task" matches "01-21-my-task")
83
+ if [[ -z "$task_dir" ]]; then
84
+ task_dir=$(find "$tasks_dir" -maxdepth 1 -type d -name "*-${task_name}" 2>/dev/null | head -1)
85
+ fi
86
+
87
+ if [[ -n "$task_dir" ]] && [[ -d "$task_dir" ]]; then
88
+ echo "$task_dir"
89
+ return 0
90
+ fi
91
+
92
+ return 1
93
+ }
94
+
95
+ # =============================================================================
96
+ # Archive Operations
97
+ # =============================================================================
98
+
99
+ # Archive a task directory to archive/{YYYY-MM}/
100
+ # Args: task_dir_abs, [repo_root]
101
+ # Returns: 0 on success, 1 on error
102
+ # Outputs: archive destination path
103
+ archive_task_dir() {
104
+ local task_dir_abs="$1"
105
+ local repo_root="${2:-$(get_repo_root)}"
106
+
107
+ if [[ ! -d "$task_dir_abs" ]]; then
108
+ echo "Error: task directory not found: $task_dir_abs" >&2
109
+ return 1
110
+ fi
111
+
112
+ # Get tasks directory (parent of the task)
113
+ local tasks_dir=$(dirname "$task_dir_abs")
114
+ local archive_dir="$tasks_dir/archive"
115
+ local year_month=$(date +%Y-%m)
116
+ local month_dir="$archive_dir/$year_month"
117
+
118
+ # Create archive directory
119
+ mkdir -p "$month_dir"
120
+
121
+ # Move task to archive
122
+ local task_name=$(basename "$task_dir_abs")
123
+ mv "$task_dir_abs" "$month_dir/"
124
+
125
+ # Output the destination
126
+ echo "$month_dir/$task_name"
127
+ return 0
128
+ }
129
+
130
+ # Complete archive workflow: archive directory
131
+ # Args: task_dir_abs, [repo_root]
132
+ # Returns: 0 on success
133
+ # Outputs: lines with status info
134
+ archive_task_complete() {
135
+ local task_dir_abs="$1"
136
+ local repo_root="${2:-$(get_repo_root)}"
137
+
138
+ if [[ ! -d "$task_dir_abs" ]]; then
139
+ echo "Error: task directory not found: $task_dir_abs" >&2
140
+ return 1
141
+ fi
142
+
143
+ # Archive the directory
144
+ local archive_dest
145
+ if archive_dest=$(archive_task_dir "$task_dir_abs" "$repo_root"); then
146
+ echo "archived_to:$archive_dest"
147
+ return 0
148
+ fi
149
+
150
+ return 1
151
+ }