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.
- package/.claude/agents/check.md +122 -0
- package/.claude/agents/debug.md +106 -0
- package/.claude/agents/dispatch.md +214 -0
- package/.claude/agents/implement.md +96 -0
- package/.claude/agents/plan.md +396 -0
- package/.claude/agents/research.md +120 -0
- package/.claude/commands/evolve/merge.md +80 -0
- package/.claude/commands/trellis/before-backend-dev.md +13 -0
- package/.claude/commands/trellis/before-frontend-dev.md +13 -0
- package/.claude/commands/trellis/break-loop.md +107 -0
- package/.claude/commands/trellis/check-backend.md +13 -0
- package/.claude/commands/trellis/check-cross-layer.md +153 -0
- package/.claude/commands/trellis/check-frontend.md +13 -0
- package/.claude/commands/trellis/create-command.md +154 -0
- package/.claude/commands/trellis/finish-work.md +129 -0
- package/.claude/commands/trellis/integrate-skill.md +219 -0
- package/.claude/commands/trellis/onboard.md +358 -0
- package/.claude/commands/trellis/parallel.md +193 -0
- package/.claude/commands/trellis/record-session.md +62 -0
- package/.claude/commands/trellis/start.md +280 -0
- package/.claude/commands/trellis/update-spec.md +213 -0
- package/.claude/hooks/inject-subagent-context.py +758 -0
- package/.claude/hooks/ralph-loop.py +374 -0
- package/.claude/hooks/session-start.py +126 -0
- package/.claude/settings.json +41 -0
- package/.claude/skills/deepagents-guide/SKILL.md +428 -0
- package/.cursor/commands/trellis-before-backend-dev.md +13 -0
- package/.cursor/commands/trellis-before-frontend-dev.md +13 -0
- package/.cursor/commands/trellis-break-loop.md +107 -0
- package/.cursor/commands/trellis-check-backend.md +13 -0
- package/.cursor/commands/trellis-check-cross-layer.md +153 -0
- package/.cursor/commands/trellis-check-frontend.md +13 -0
- package/.cursor/commands/trellis-create-command.md +154 -0
- package/.cursor/commands/trellis-finish-work.md +129 -0
- package/.cursor/commands/trellis-integrate-skill.md +219 -0
- package/.cursor/commands/trellis-onboard.md +358 -0
- package/.cursor/commands/trellis-record-session.md +62 -0
- package/.cursor/commands/trellis-start.md +156 -0
- package/.cursor/commands/trellis-update-spec.md +213 -0
- package/.env.example +11 -0
- package/.husky/pre-commit +1 -0
- package/.mcp.json +8 -0
- package/.trellis/.template-hashes.json +65 -0
- package/.trellis/.version +1 -0
- package/.trellis/scripts/add-session.sh +384 -0
- package/.trellis/scripts/common/developer.sh +129 -0
- package/.trellis/scripts/common/git-context.sh +263 -0
- package/.trellis/scripts/common/paths.sh +208 -0
- package/.trellis/scripts/common/phase.sh +150 -0
- package/.trellis/scripts/common/registry.sh +247 -0
- package/.trellis/scripts/common/task-queue.sh +142 -0
- package/.trellis/scripts/common/task-utils.sh +151 -0
- package/.trellis/scripts/common/worktree.sh +128 -0
- package/.trellis/scripts/create-bootstrap.sh +299 -0
- package/.trellis/scripts/get-context.sh +7 -0
- package/.trellis/scripts/get-developer.sh +15 -0
- package/.trellis/scripts/init-developer.sh +34 -0
- package/.trellis/scripts/multi-agent/cleanup.sh +396 -0
- package/.trellis/scripts/multi-agent/create-pr.sh +241 -0
- package/.trellis/scripts/multi-agent/plan.sh +207 -0
- package/.trellis/scripts/multi-agent/start.sh +310 -0
- package/.trellis/scripts/multi-agent/status.sh +828 -0
- package/.trellis/scripts/task.sh +1118 -0
- package/.trellis/spec/backend/deepagents-guide.md +337 -0
- package/.trellis/spec/backend/directory-structure.md +126 -0
- package/.trellis/spec/backend/examples/skills/deepagents-guide/README.md +11 -0
- package/.trellis/spec/backend/examples/skills/deepagents-guide/agent.js.template +20 -0
- package/.trellis/spec/backend/examples/skills/deepagents-guide/skills-config.js.template +13 -0
- package/.trellis/spec/backend/examples/skills/deepagents-guide/subagent.js.template +19 -0
- package/.trellis/spec/backend/hook-guidelines.md +178 -0
- package/.trellis/spec/backend/index.md +36 -0
- package/.trellis/spec/backend/quality-guidelines.md +201 -0
- package/.trellis/spec/backend/state-management.md +76 -0
- package/.trellis/spec/backend/tool-guidelines.md +144 -0
- package/.trellis/spec/backend/type-safety.md +71 -0
- package/.trellis/spec/guides/code-reuse-thinking-guide.md +92 -0
- package/.trellis/spec/guides/cross-layer-thinking-guide.md +94 -0
- package/.trellis/spec/guides/index.md +79 -0
- package/.trellis/tasks/archive/02-02-evolving-skills/prd.md +61 -0
- package/.trellis/tasks/archive/02-02-evolving-skills/task.json +29 -0
- package/.trellis/tasks/archive/2026-02/00-bootstrap-guidelines/prd.md +86 -0
- package/.trellis/tasks/archive/2026-02/00-bootstrap-guidelines/task.json +27 -0
- package/.trellis/tasks/archive/2026-02/02-02-skills-system/check.jsonl +3 -0
- package/.trellis/tasks/archive/2026-02/02-02-skills-system/debug.jsonl +2 -0
- package/.trellis/tasks/archive/2026-02/02-02-skills-system/implement.jsonl +5 -0
- package/.trellis/tasks/archive/2026-02/02-02-skills-system/prd.md +33 -0
- package/.trellis/tasks/archive/2026-02/02-02-skills-system/task.json +41 -0
- package/.trellis/workflow.md +407 -0
- package/.trellis/workspace/index.md +123 -0
- package/.trellis/workspace/pony/index.md +40 -0
- package/.trellis/workspace/pony/journal-1.md +7 -0
- package/.trellis/worktree.yaml +47 -0
- package/AGENTS.md +18 -0
- package/CLAUDE.md +292 -0
- package/README.md +134 -0
- package/agents/deepspider.md +142 -0
- package/docs/DEBUG.md +42 -0
- package/docs/GUIDE.md +334 -0
- package/docs/PROMPT.md +60 -0
- package/docs/USAGE.md +226 -0
- package/eslint.config.js +51 -0
- package/package.json +78 -0
- package/requirements-crypto.txt +14 -0
- package/src/agent/index.js +97 -0
- package/src/agent/logger.js +164 -0
- package/src/agent/middleware/filterTools.js +64 -0
- package/src/agent/middleware/report.js +79 -0
- package/src/agent/prompts/system.js +315 -0
- package/src/agent/run.js +575 -0
- package/src/agent/skills/anti-detect/SKILL.md +28 -0
- package/src/agent/skills/anti-detect/evolved.md +12 -0
- package/src/agent/skills/captcha/SKILL.md +37 -0
- package/src/agent/skills/captcha/evolved.md +12 -0
- package/src/agent/skills/config.js +30 -0
- package/src/agent/skills/crawler/SKILL.md +9 -0
- package/src/agent/skills/crawler/evolved.md +16 -0
- package/src/agent/skills/dynamic-analysis/SKILL.md +91 -0
- package/src/agent/skills/dynamic-analysis/evolved.md +12 -0
- package/src/agent/skills/env/SKILL.md +72 -0
- package/src/agent/skills/env/evolved.md +12 -0
- package/src/agent/skills/evolve.js +79 -0
- package/src/agent/skills/general/SKILL.md +12 -0
- package/src/agent/skills/general/evolved.md +12 -0
- package/src/agent/skills/js2python/SKILL.md +30 -0
- package/src/agent/skills/js2python/evolved.md +13 -0
- package/src/agent/skills/report/SKILL.md +21 -0
- package/src/agent/skills/report/evolved.md +12 -0
- package/src/agent/skills/sandbox/SKILL.md +22 -0
- package/src/agent/skills/sandbox/evolved.md +16 -0
- package/src/agent/skills/static-analysis/SKILL.md +93 -0
- package/src/agent/skills/static-analysis/evolved.md +12 -0
- package/src/agent/skills/xpath/SKILL.md +119 -0
- package/src/agent/subagents/anti-detect.js +45 -0
- package/src/agent/subagents/captcha.js +51 -0
- package/src/agent/subagents/crawler.js +138 -0
- package/src/agent/subagents/dynamic.js +64 -0
- package/src/agent/subagents/env-agent.js +82 -0
- package/src/agent/subagents/index.js +37 -0
- package/src/agent/subagents/js2python.js +72 -0
- package/src/agent/subagents/sandbox.js +55 -0
- package/src/agent/subagents/static.js +66 -0
- package/src/agent/tools/analysis.js +135 -0
- package/src/agent/tools/analyzer.js +85 -0
- package/src/agent/tools/anti-detect.js +89 -0
- package/src/agent/tools/antidebug.js +64 -0
- package/src/agent/tools/async.js +43 -0
- package/src/agent/tools/browser.js +324 -0
- package/src/agent/tools/captcha.js +223 -0
- package/src/agent/tools/capture.js +179 -0
- package/src/agent/tools/correlate.js +303 -0
- package/src/agent/tools/crawler.js +116 -0
- package/src/agent/tools/cryptohook.js +80 -0
- package/src/agent/tools/debug.js +246 -0
- package/src/agent/tools/deobfuscator.js +90 -0
- package/src/agent/tools/env.js +83 -0
- package/src/agent/tools/envdump.js +92 -0
- package/src/agent/tools/evolve.js +164 -0
- package/src/agent/tools/extract.js +114 -0
- package/src/agent/tools/extractor.js +54 -0
- package/src/agent/tools/file.js +224 -0
- package/src/agent/tools/hook.js +84 -0
- package/src/agent/tools/hookManager.js +178 -0
- package/src/agent/tools/index.js +137 -0
- package/src/agent/tools/nodejs.js +101 -0
- package/src/agent/tools/patch.js +46 -0
- package/src/agent/tools/preprocess.js +71 -0
- package/src/agent/tools/profile.js +122 -0
- package/src/agent/tools/python.js +627 -0
- package/src/agent/tools/report.js +124 -0
- package/src/agent/tools/runtime.js +132 -0
- package/src/agent/tools/sandbox.js +79 -0
- package/src/agent/tools/store.js +73 -0
- package/src/agent/tools/trace.js +74 -0
- package/src/agent/tools/tracing.js +201 -0
- package/src/agent/tools/utils.js +51 -0
- package/src/agent/tools/verify.js +184 -0
- package/src/agent/tools/webcrack.js +109 -0
- package/src/analyzer/ASTAnalyzer.js +387 -0
- package/src/analyzer/CallStackAnalyzer.js +379 -0
- package/src/analyzer/Deobfuscator.js +289 -0
- package/src/analyzer/EncryptionAnalyzer.js +99 -0
- package/src/analyzer/index.js +22 -0
- package/src/browser/EnvBridge.js +186 -0
- package/src/browser/cdp.js +168 -0
- package/src/browser/client.js +197 -0
- package/src/browser/collector.js +444 -0
- package/src/browser/collectors/RequestCryptoLinker.js +109 -0
- package/src/browser/collectors/ResponseSearcher.js +107 -0
- package/src/browser/collectors/ScriptCollector.js +158 -0
- package/src/browser/collectors/index.js +26 -0
- package/src/browser/defaultHooks.js +932 -0
- package/src/browser/hooks/crypto.js +55 -0
- package/src/browser/hooks/index.js +64 -0
- package/src/browser/hooks/native.js +9 -0
- package/src/browser/hooks/network.js +33 -0
- package/src/browser/index.js +42 -0
- package/src/browser/interceptors/NetworkInterceptor.js +116 -0
- package/src/browser/interceptors/ScriptInterceptor.js +76 -0
- package/src/browser/interceptors/index.js +6 -0
- package/src/browser/ui/analysisPanel.js +1782 -0
- package/src/browser/ui/confirmDialog.js +158 -0
- package/src/browser/ui/panel.html +152 -0
- package/src/browser/ui/selector.js +170 -0
- package/src/config/index.js +5 -0
- package/src/config/paths.js +71 -0
- package/src/config/patterns/crypto.js +36 -0
- package/src/config/profiles/chrome.json +71 -0
- package/src/config/profiles/firefox.json +44 -0
- package/src/config/profiles/safari.json +38 -0
- package/src/core/EnvMonitor.js +200 -0
- package/src/core/PatchGenerator.js +278 -0
- package/src/core/Sandbox.js +181 -0
- package/src/env/AntiAntiDebug.js +111 -0
- package/src/env/AsyncHook.js +68 -0
- package/src/env/BrowserAPIList.js +265 -0
- package/src/env/CookieHook.js +48 -0
- package/src/env/CryptoHook.js +205 -0
- package/src/env/EnvCodeGenerator.js +157 -0
- package/src/env/EnvDumper.js +356 -0
- package/src/env/EnvExtractor.js +220 -0
- package/src/env/HookBase.js +618 -0
- package/src/env/NetworkHook.js +159 -0
- package/src/env/modules/bom/history.js +29 -0
- package/src/env/modules/bom/location.js +26 -0
- package/src/env/modules/bom/navigator.js +70 -0
- package/src/env/modules/bom/screen.js +26 -0
- package/src/env/modules/bom/storage.js +23 -0
- package/src/env/modules/dom/document.js +110 -0
- package/src/env/modules/dom/event.js +51 -0
- package/src/env/modules/index.js +34 -0
- package/src/env/modules/webapi/fetch.js +46 -0
- package/src/env/modules/webapi/url.js +47 -0
- package/src/env/modules/webapi/xhr.js +48 -0
- package/src/index.js +27 -0
- package/src/mcp/server.js +89 -0
- package/src/store/DataStore.js +708 -0
- package/src/store/Store.js +158 -0
- package/src/store/Validator.js +24 -0
- package/test/analyze.test.js +90 -0
- package/test/envdump.test.js +74 -0
- package/test/flow.test.js +90 -0
- package/test/hooks.test.js +138 -0
- package/test/plugin.test.js +35 -0
- package/test/refactor-full.test.js +30 -0
- package/test/refactor.test.js +21 -0
- package/test/samples/obfuscated.js +61 -0
- package/test/samples/original.js +66 -0
- package/test/samples/v10_eval_chain.js +52 -0
- package/test/samples/v11_bytecode_vm.js +81 -0
- package/test/samples/v12_polymorphic.js +69 -0
- package/test/samples/v1_ob_basic.js +98 -0
- package/test/samples/v2_ob_advanced.js +99 -0
- package/test/samples/v3_jjencode.js +77 -0
- package/test/samples/v4_aaencode.js +73 -0
- package/test/samples/v5_control_flow.js +86 -0
- package/test/samples/v6_string_encryption.js +71 -0
- package/test/samples/v7_jsvmp.js +83 -0
- package/test/samples/v8_anti_debug.js +79 -0
- package/test/samples/v9_proxy_trap.js +49 -0
- package/test/samples.test.js +96 -0
- 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
|
+
}
|