@fifine/aim-studio 0.0.1

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 (289) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +159 -0
  3. package/bin/aim.js +3 -0
  4. package/dist/cli/index.d.ts +3 -0
  5. package/dist/cli/index.d.ts.map +1 -0
  6. package/dist/cli/index.js +89 -0
  7. package/dist/cli/index.js.map +1 -0
  8. package/dist/commands/init.d.ts +13 -0
  9. package/dist/commands/init.d.ts.map +1 -0
  10. package/dist/commands/init.js +513 -0
  11. package/dist/commands/init.js.map +1 -0
  12. package/dist/commands/update.d.ts +27 -0
  13. package/dist/commands/update.d.ts.map +1 -0
  14. package/dist/commands/update.js +1275 -0
  15. package/dist/commands/update.js.map +1 -0
  16. package/dist/configurators/claude.d.ts +32 -0
  17. package/dist/configurators/claude.d.ts.map +1 -0
  18. package/dist/configurators/claude.js +98 -0
  19. package/dist/configurators/claude.js.map +1 -0
  20. package/dist/configurators/index.d.ts +51 -0
  21. package/dist/configurators/index.d.ts.map +1 -0
  22. package/dist/configurators/index.js +113 -0
  23. package/dist/configurators/index.js.map +1 -0
  24. package/dist/configurators/shared.d.ts +12 -0
  25. package/dist/configurators/shared.d.ts.map +1 -0
  26. package/dist/configurators/shared.js +21 -0
  27. package/dist/configurators/shared.js.map +1 -0
  28. package/dist/configurators/workflow.d.ts +28 -0
  29. package/dist/configurators/workflow.d.ts.map +1 -0
  30. package/dist/configurators/workflow.js +147 -0
  31. package/dist/configurators/workflow.js.map +1 -0
  32. package/dist/constants/paths.d.ts +68 -0
  33. package/dist/constants/paths.d.ts.map +1 -0
  34. package/dist/constants/paths.js +77 -0
  35. package/dist/constants/paths.js.map +1 -0
  36. package/dist/constants/version.d.ts +9 -0
  37. package/dist/constants/version.d.ts.map +1 -0
  38. package/dist/constants/version.js +15 -0
  39. package/dist/constants/version.js.map +1 -0
  40. package/dist/index.d.ts +9 -0
  41. package/dist/index.d.ts.map +1 -0
  42. package/dist/index.js +9 -0
  43. package/dist/index.js.map +1 -0
  44. package/dist/migrations/index.d.ts +54 -0
  45. package/dist/migrations/index.d.ts.map +1 -0
  46. package/dist/migrations/index.js +160 -0
  47. package/dist/migrations/index.js.map +1 -0
  48. package/dist/migrations/manifests/0.0.1.json +9 -0
  49. package/dist/migrations/manifests/0.1.9.json +30 -0
  50. package/dist/migrations/manifests/0.2.0.json +49 -0
  51. package/dist/migrations/manifests/0.2.12.json +9 -0
  52. package/dist/migrations/manifests/0.2.13.json +9 -0
  53. package/dist/migrations/manifests/0.2.14.json +175 -0
  54. package/dist/migrations/manifests/0.2.15.json +33 -0
  55. package/dist/migrations/manifests/0.3.0-beta.0.json +278 -0
  56. package/dist/migrations/manifests/0.3.0-beta.1.json +9 -0
  57. package/dist/migrations/manifests/0.3.0-beta.10.json +9 -0
  58. package/dist/migrations/manifests/0.3.0-beta.11.json +9 -0
  59. package/dist/migrations/manifests/0.3.0-beta.12.json +9 -0
  60. package/dist/migrations/manifests/0.3.0-beta.13.json +9 -0
  61. package/dist/migrations/manifests/0.3.0-beta.14.json +9 -0
  62. package/dist/migrations/manifests/0.3.0-beta.15.json +9 -0
  63. package/dist/migrations/manifests/0.3.0-beta.16.json +9 -0
  64. package/dist/migrations/manifests/0.3.0-beta.2.json +9 -0
  65. package/dist/migrations/manifests/0.3.0-beta.3.json +9 -0
  66. package/dist/migrations/manifests/0.3.0-beta.4.json +9 -0
  67. package/dist/migrations/manifests/0.3.0-beta.5.json +9 -0
  68. package/dist/migrations/manifests/0.3.0-beta.6.json +9 -0
  69. package/dist/migrations/manifests/0.3.0-beta.7.json +11 -0
  70. package/dist/migrations/manifests/0.3.0-beta.8.json +9 -0
  71. package/dist/migrations/manifests/0.3.0-beta.9.json +9 -0
  72. package/dist/migrations/manifests/0.3.0-rc.0.json +9 -0
  73. package/dist/migrations/manifests/0.3.0-rc.1.json +9 -0
  74. package/dist/migrations/manifests/0.3.0-rc.2.json +9 -0
  75. package/dist/templates/CLAUDE.md +71 -0
  76. package/dist/templates/aim/gitignore.txt +29 -0
  77. package/dist/templates/aim/index.d.ts +49 -0
  78. package/dist/templates/aim/index.d.ts.map +1 -0
  79. package/dist/templates/aim/index.js +92 -0
  80. package/dist/templates/aim/index.js.map +1 -0
  81. package/dist/templates/aim/scripts/__init__.py +5 -0
  82. package/dist/templates/aim/scripts/add_session.py +392 -0
  83. package/dist/templates/aim/scripts/common/__init__.py +80 -0
  84. package/dist/templates/aim/scripts/common/cli_adapter.py +435 -0
  85. package/dist/templates/aim/scripts/common/developer.py +190 -0
  86. package/dist/templates/aim/scripts/common/git_context.py +383 -0
  87. package/dist/templates/aim/scripts/common/paths.py +347 -0
  88. package/dist/templates/aim/scripts/common/phase.py +253 -0
  89. package/dist/templates/aim/scripts/common/registry.py +366 -0
  90. package/dist/templates/aim/scripts/common/task_queue.py +255 -0
  91. package/dist/templates/aim/scripts/common/task_utils.py +178 -0
  92. package/dist/templates/aim/scripts/common/worktree.py +219 -0
  93. package/dist/templates/aim/scripts/create_bootstrap.py +290 -0
  94. package/dist/templates/aim/scripts/get_context.py +16 -0
  95. package/dist/templates/aim/scripts/get_developer.py +26 -0
  96. package/dist/templates/aim/scripts/init_developer.py +51 -0
  97. package/dist/templates/aim/scripts/multi_agent/__init__.py +5 -0
  98. package/dist/templates/aim/scripts/multi_agent/cleanup.py +403 -0
  99. package/dist/templates/aim/scripts/multi_agent/create_pr.py +329 -0
  100. package/dist/templates/aim/scripts/multi_agent/plan.py +233 -0
  101. package/dist/templates/aim/scripts/multi_agent/start.py +461 -0
  102. package/dist/templates/aim/scripts/multi_agent/status.py +817 -0
  103. package/dist/templates/aim/scripts/task.py +1068 -0
  104. package/dist/templates/aim/scripts-shell-archive/add-session.sh +384 -0
  105. package/dist/templates/aim/scripts-shell-archive/common/developer.sh +129 -0
  106. package/dist/templates/aim/scripts-shell-archive/common/git-context.sh +263 -0
  107. package/dist/templates/aim/scripts-shell-archive/common/paths.sh +208 -0
  108. package/dist/templates/aim/scripts-shell-archive/common/phase.sh +150 -0
  109. package/dist/templates/aim/scripts-shell-archive/common/registry.sh +247 -0
  110. package/dist/templates/aim/scripts-shell-archive/common/task-queue.sh +142 -0
  111. package/dist/templates/aim/scripts-shell-archive/common/task-utils.sh +151 -0
  112. package/dist/templates/aim/scripts-shell-archive/common/worktree.sh +128 -0
  113. package/dist/templates/aim/scripts-shell-archive/create-bootstrap.sh +299 -0
  114. package/dist/templates/aim/scripts-shell-archive/get-context.sh +7 -0
  115. package/dist/templates/aim/scripts-shell-archive/get-developer.sh +15 -0
  116. package/dist/templates/aim/scripts-shell-archive/init-developer.sh +34 -0
  117. package/dist/templates/aim/scripts-shell-archive/multi-agent/cleanup.sh +396 -0
  118. package/dist/templates/aim/scripts-shell-archive/multi-agent/create-pr.sh +241 -0
  119. package/dist/templates/aim/scripts-shell-archive/multi-agent/plan.sh +207 -0
  120. package/dist/templates/aim/scripts-shell-archive/multi-agent/start.sh +317 -0
  121. package/dist/templates/aim/scripts-shell-archive/multi-agent/status.sh +828 -0
  122. package/dist/templates/aim/scripts-shell-archive/task.sh +1204 -0
  123. package/dist/templates/aim/tasks/.gitkeep +0 -0
  124. package/dist/templates/aim/workflow.md +258 -0
  125. package/dist/templates/aim/worktree.yaml +47 -0
  126. package/dist/templates/claude/agents/check.md +122 -0
  127. package/dist/templates/claude/agents/debug.md +106 -0
  128. package/dist/templates/claude/agents/dispatch.md +230 -0
  129. package/dist/templates/claude/agents/implement.md +96 -0
  130. package/dist/templates/claude/agents/plan.md +396 -0
  131. package/dist/templates/claude/agents/research.md +120 -0
  132. package/dist/templates/claude/agents/story.md +53 -0
  133. package/dist/templates/claude/commands/aim/before-backend-dev.md +13 -0
  134. package/dist/templates/claude/commands/aim/before-frontend-dev.md +13 -0
  135. package/dist/templates/claude/commands/aim/break-loop.md +153 -0
  136. package/dist/templates/claude/commands/aim/check-backend.md +13 -0
  137. package/dist/templates/claude/commands/aim/check-cross-layer.md +153 -0
  138. package/dist/templates/claude/commands/aim/check-frontend.md +13 -0
  139. package/dist/templates/claude/commands/aim/check-story.md +59 -0
  140. package/dist/templates/claude/commands/aim/create-command.md +154 -0
  141. package/dist/templates/claude/commands/aim/export.md +187 -0
  142. package/dist/templates/claude/commands/aim/finish-work.md +104 -0
  143. package/dist/templates/claude/commands/aim/integrate-skill.md +219 -0
  144. package/dist/templates/claude/commands/aim/onboard.md +358 -0
  145. package/dist/templates/claude/commands/aim/parallel.md +217 -0
  146. package/dist/templates/claude/commands/aim/portrait.md +170 -0
  147. package/dist/templates/claude/commands/aim/record-session.md +92 -0
  148. package/dist/templates/claude/commands/aim/start.md +112 -0
  149. package/dist/templates/claude/commands/aim/story.md +140 -0
  150. package/dist/templates/claude/commands/aim/update-spec.md +285 -0
  151. package/dist/templates/claude/commands/aim/visualize.md +182 -0
  152. package/dist/templates/claude/commands/trellis/before-backend-dev.md +13 -0
  153. package/dist/templates/claude/commands/trellis/before-frontend-dev.md +13 -0
  154. package/dist/templates/claude/commands/trellis/break-loop.md +125 -0
  155. package/dist/templates/claude/commands/trellis/check-backend.md +13 -0
  156. package/dist/templates/claude/commands/trellis/check-cross-layer.md +153 -0
  157. package/dist/templates/claude/commands/trellis/check-frontend.md +13 -0
  158. package/dist/templates/claude/commands/trellis/create-command.md +154 -0
  159. package/dist/templates/claude/commands/trellis/finish-work.md +129 -0
  160. package/dist/templates/claude/commands/trellis/integrate-skill.md +219 -0
  161. package/dist/templates/claude/commands/trellis/onboard.md +358 -0
  162. package/dist/templates/claude/commands/trellis/parallel.md +193 -0
  163. package/dist/templates/claude/commands/trellis/record-session.md +62 -0
  164. package/dist/templates/claude/commands/trellis/start.md +280 -0
  165. package/dist/templates/claude/commands/trellis/update-spec.md +285 -0
  166. package/dist/templates/claude/hooks/inject-subagent-context.py +772 -0
  167. package/dist/templates/claude/hooks/ralph-loop.py +388 -0
  168. package/dist/templates/claude/hooks/session-start.py +142 -0
  169. package/dist/templates/claude/index.d.ts +54 -0
  170. package/dist/templates/claude/index.d.ts.map +1 -0
  171. package/dist/templates/claude/index.js +85 -0
  172. package/dist/templates/claude/index.js.map +1 -0
  173. package/dist/templates/claude/settings.json +41 -0
  174. package/dist/templates/extract.d.ts +68 -0
  175. package/dist/templates/extract.d.ts.map +1 -0
  176. package/dist/templates/extract.js +128 -0
  177. package/dist/templates/extract.js.map +1 -0
  178. package/dist/templates/markdown/agents.md +25 -0
  179. package/dist/templates/markdown/gitignore.txt +12 -0
  180. package/dist/templates/markdown/index.d.ts +32 -0
  181. package/dist/templates/markdown/index.d.ts.map +1 -0
  182. package/dist/templates/markdown/index.js +58 -0
  183. package/dist/templates/markdown/index.js.map +1 -0
  184. package/dist/templates/markdown/spec/backend/database-guidelines.md.txt +51 -0
  185. package/dist/templates/markdown/spec/backend/directory-structure.md.txt +54 -0
  186. package/dist/templates/markdown/spec/backend/error-handling.md.txt +51 -0
  187. package/dist/templates/markdown/spec/backend/index.md +40 -0
  188. package/dist/templates/markdown/spec/backend/index.md.txt +38 -0
  189. package/dist/templates/markdown/spec/backend/logging-guidelines.md.txt +51 -0
  190. package/dist/templates/markdown/spec/backend/quality-guidelines.md.txt +51 -0
  191. package/dist/templates/markdown/spec/backend/script-conventions.md +467 -0
  192. package/dist/templates/markdown/spec/frontend/component-guidelines.md.txt +59 -0
  193. package/dist/templates/markdown/spec/frontend/directory-structure.md.txt +54 -0
  194. package/dist/templates/markdown/spec/frontend/hook-guidelines.md.txt +51 -0
  195. package/dist/templates/markdown/spec/frontend/index.md.txt +39 -0
  196. package/dist/templates/markdown/spec/frontend/quality-guidelines.md.txt +51 -0
  197. package/dist/templates/markdown/spec/frontend/state-management.md.txt +51 -0
  198. package/dist/templates/markdown/spec/frontend/type-safety.md.txt +51 -0
  199. package/dist/templates/markdown/spec/guides/code-reuse-thinking-guide.md +118 -0
  200. package/dist/templates/markdown/spec/guides/code-reuse-thinking-guide.md.txt +92 -0
  201. package/dist/templates/markdown/spec/guides/cross-layer-thinking-guide.md.txt +94 -0
  202. package/dist/templates/markdown/spec/guides/cross-platform-thinking-guide.md +394 -0
  203. package/dist/templates/markdown/spec/guides/cross-platform-thinking-guide.md.txt +319 -0
  204. package/dist/templates/markdown/spec/guides/index.md.txt +89 -0
  205. package/dist/templates/markdown/spec/story/character.md.txt +95 -0
  206. package/dist/templates/markdown/spec/story/index.md.txt +31 -0
  207. package/dist/templates/markdown/spec/story/script.md.txt +313 -0
  208. package/dist/templates/markdown/spec/story/world.md.txt +92 -0
  209. package/dist/templates/markdown/workspace-index.md +123 -0
  210. package/dist/templates/markdown/worktree.yaml.txt +58 -0
  211. package/dist/templates/trellis/gitignore.txt +29 -0
  212. package/dist/templates/trellis/index.d.ts +49 -0
  213. package/dist/templates/trellis/index.d.ts.map +1 -0
  214. package/dist/templates/trellis/index.js +92 -0
  215. package/dist/templates/trellis/index.js.map +1 -0
  216. package/dist/templates/trellis/scripts/__init__.py +5 -0
  217. package/dist/templates/trellis/scripts/add_session.py +392 -0
  218. package/dist/templates/trellis/scripts/common/__init__.py +80 -0
  219. package/dist/templates/trellis/scripts/common/cli_adapter.py +435 -0
  220. package/dist/templates/trellis/scripts/common/developer.py +190 -0
  221. package/dist/templates/trellis/scripts/common/git_context.py +383 -0
  222. package/dist/templates/trellis/scripts/common/paths.py +347 -0
  223. package/dist/templates/trellis/scripts/common/phase.py +253 -0
  224. package/dist/templates/trellis/scripts/common/registry.py +366 -0
  225. package/dist/templates/trellis/scripts/common/task_queue.py +255 -0
  226. package/dist/templates/trellis/scripts/common/task_utils.py +178 -0
  227. package/dist/templates/trellis/scripts/common/worktree.py +219 -0
  228. package/dist/templates/trellis/scripts/create_bootstrap.py +290 -0
  229. package/dist/templates/trellis/scripts/get_context.py +16 -0
  230. package/dist/templates/trellis/scripts/get_developer.py +26 -0
  231. package/dist/templates/trellis/scripts/init_developer.py +51 -0
  232. package/dist/templates/trellis/scripts/multi_agent/__init__.py +5 -0
  233. package/dist/templates/trellis/scripts/multi_agent/cleanup.py +403 -0
  234. package/dist/templates/trellis/scripts/multi_agent/create_pr.py +329 -0
  235. package/dist/templates/trellis/scripts/multi_agent/plan.py +233 -0
  236. package/dist/templates/trellis/scripts/multi_agent/start.py +461 -0
  237. package/dist/templates/trellis/scripts/multi_agent/status.py +817 -0
  238. package/dist/templates/trellis/scripts/task.py +1056 -0
  239. package/dist/templates/trellis/scripts-shell-archive/add-session.sh +384 -0
  240. package/dist/templates/trellis/scripts-shell-archive/common/developer.sh +129 -0
  241. package/dist/templates/trellis/scripts-shell-archive/common/git-context.sh +263 -0
  242. package/dist/templates/trellis/scripts-shell-archive/common/paths.sh +208 -0
  243. package/dist/templates/trellis/scripts-shell-archive/common/phase.sh +150 -0
  244. package/dist/templates/trellis/scripts-shell-archive/common/registry.sh +247 -0
  245. package/dist/templates/trellis/scripts-shell-archive/common/task-queue.sh +142 -0
  246. package/dist/templates/trellis/scripts-shell-archive/common/task-utils.sh +151 -0
  247. package/dist/templates/trellis/scripts-shell-archive/common/worktree.sh +128 -0
  248. package/dist/templates/trellis/scripts-shell-archive/create-bootstrap.sh +299 -0
  249. package/dist/templates/trellis/scripts-shell-archive/get-context.sh +7 -0
  250. package/dist/templates/trellis/scripts-shell-archive/get-developer.sh +15 -0
  251. package/dist/templates/trellis/scripts-shell-archive/init-developer.sh +34 -0
  252. package/dist/templates/trellis/scripts-shell-archive/multi-agent/cleanup.sh +396 -0
  253. package/dist/templates/trellis/scripts-shell-archive/multi-agent/create-pr.sh +241 -0
  254. package/dist/templates/trellis/scripts-shell-archive/multi-agent/plan.sh +207 -0
  255. package/dist/templates/trellis/scripts-shell-archive/multi-agent/start.sh +317 -0
  256. package/dist/templates/trellis/scripts-shell-archive/multi-agent/status.sh +828 -0
  257. package/dist/templates/trellis/scripts-shell-archive/task.sh +1204 -0
  258. package/dist/templates/trellis/tasks/.gitkeep +0 -0
  259. package/dist/templates/trellis/workflow.md +416 -0
  260. package/dist/templates/trellis/worktree.yaml +47 -0
  261. package/dist/types/ai-tools.d.ts +48 -0
  262. package/dist/types/ai-tools.d.ts.map +1 -0
  263. package/dist/types/ai-tools.js +32 -0
  264. package/dist/types/ai-tools.js.map +1 -0
  265. package/dist/types/migration.d.ts +86 -0
  266. package/dist/types/migration.d.ts.map +1 -0
  267. package/dist/types/migration.js +8 -0
  268. package/dist/types/migration.js.map +1 -0
  269. package/dist/utils/compare-versions.d.ts +12 -0
  270. package/dist/utils/compare-versions.d.ts.map +1 -0
  271. package/dist/utils/compare-versions.js +76 -0
  272. package/dist/utils/compare-versions.js.map +1 -0
  273. package/dist/utils/file-writer.d.ts +23 -0
  274. package/dist/utils/file-writer.d.ts.map +1 -0
  275. package/dist/utils/file-writer.js +140 -0
  276. package/dist/utils/file-writer.js.map +1 -0
  277. package/dist/utils/project-detector.d.ts +16 -0
  278. package/dist/utils/project-detector.d.ts.map +1 -0
  279. package/dist/utils/project-detector.js +188 -0
  280. package/dist/utils/project-detector.js.map +1 -0
  281. package/dist/utils/template-fetcher.d.ts +51 -0
  282. package/dist/utils/template-fetcher.d.ts.map +1 -0
  283. package/dist/utils/template-fetcher.js +174 -0
  284. package/dist/utils/template-fetcher.js.map +1 -0
  285. package/dist/utils/template-hash.d.ts +78 -0
  286. package/dist/utils/template-hash.d.ts.map +1 -0
  287. package/dist/utils/template-hash.js +239 -0
  288. package/dist/utils/template-hash.js.map +1 -0
  289. package/package.json +87 -0
@@ -0,0 +1,383 @@
1
+ #!/usr/bin/env python3
2
+ # -*- coding: utf-8 -*-
3
+ """
4
+ Git and Session Context utilities.
5
+
6
+ Provides:
7
+ output_json - Output context in JSON format
8
+ output_text - Output context in text format
9
+ """
10
+
11
+ from __future__ import annotations
12
+
13
+ import sys
14
+
15
+ # IMPORTANT: Force stdout to use UTF-8 on Windows
16
+ # This fixes UnicodeEncodeError when outputting non-ASCII characters
17
+ if sys.platform == "win32":
18
+ import io as _io
19
+ if hasattr(sys.stdout, "reconfigure"):
20
+ sys.stdout.reconfigure(encoding="utf-8", errors="replace") # type: ignore[union-attr]
21
+ elif hasattr(sys.stdout, "detach"):
22
+ sys.stdout = _io.TextIOWrapper(sys.stdout.detach(), encoding="utf-8", errors="replace") # type: ignore[union-attr]
23
+
24
+ import json
25
+ import subprocess
26
+ from pathlib import Path
27
+
28
+ from .paths import (
29
+ DIR_SCRIPTS,
30
+ DIR_SPEC,
31
+ DIR_TASKS,
32
+ DIR_WORKFLOW,
33
+ DIR_WORKSPACE,
34
+ FILE_TASK_JSON,
35
+ count_lines,
36
+ get_active_journal_file,
37
+ get_current_task,
38
+ get_developer,
39
+ get_repo_root,
40
+ get_tasks_dir,
41
+ )
42
+
43
+ # =============================================================================
44
+ # Helper Functions
45
+ # =============================================================================
46
+
47
+
48
+ def _run_git_command(args: list[str], cwd: Path | None = None) -> tuple[int, str, str]:
49
+ """Run a git command and return (returncode, stdout, stderr).
50
+
51
+ Uses UTF-8 encoding with -c i18n.logOutputEncoding=UTF-8 to ensure
52
+ consistent output across all platforms (Windows, macOS, Linux).
53
+ """
54
+ try:
55
+ # Force git to output UTF-8 for consistent cross-platform behavior
56
+ git_args = ["git", "-c", "i18n.logOutputEncoding=UTF-8"] + args
57
+ result = subprocess.run(
58
+ git_args,
59
+ cwd=cwd,
60
+ capture_output=True,
61
+ text=True,
62
+ encoding="utf-8",
63
+ errors="replace",
64
+ )
65
+ return result.returncode, result.stdout, result.stderr
66
+ except Exception as e:
67
+ return 1, "", str(e)
68
+
69
+
70
+ def _read_json_file(path: Path) -> dict | None:
71
+ """Read and parse a JSON file."""
72
+ try:
73
+ return json.loads(path.read_text(encoding="utf-8"))
74
+ except (FileNotFoundError, json.JSONDecodeError, OSError):
75
+ return None
76
+
77
+
78
+ # =============================================================================
79
+ # JSON Output
80
+ # =============================================================================
81
+
82
+
83
+ def get_context_json(repo_root: Path | None = None) -> dict:
84
+ """Get context as a dictionary.
85
+
86
+ Args:
87
+ repo_root: Repository root path. Defaults to auto-detected.
88
+
89
+ Returns:
90
+ Context dictionary.
91
+ """
92
+ if repo_root is None:
93
+ repo_root = get_repo_root()
94
+
95
+ developer = get_developer(repo_root)
96
+ tasks_dir = get_tasks_dir(repo_root)
97
+ journal_file = get_active_journal_file(repo_root)
98
+
99
+ journal_lines = 0
100
+ journal_relative = ""
101
+ if journal_file and developer:
102
+ journal_lines = count_lines(journal_file)
103
+ journal_relative = (
104
+ f"{DIR_WORKFLOW}/{DIR_WORKSPACE}/{developer}/{journal_file.name}"
105
+ )
106
+
107
+ # Git info
108
+ _, branch_out, _ = _run_git_command(["branch", "--show-current"], cwd=repo_root)
109
+ branch = branch_out.strip() or "unknown"
110
+
111
+ _, status_out, _ = _run_git_command(["status", "--porcelain"], cwd=repo_root)
112
+ git_status_count = len([line for line in status_out.splitlines() if line.strip()])
113
+ is_clean = git_status_count == 0
114
+
115
+ # Recent commits
116
+ _, log_out, _ = _run_git_command(["log", "--oneline", "-5"], cwd=repo_root)
117
+ commits = []
118
+ for line in log_out.splitlines():
119
+ if line.strip():
120
+ parts = line.split(" ", 1)
121
+ if len(parts) >= 2:
122
+ commits.append({"hash": parts[0], "message": parts[1]})
123
+ elif len(parts) == 1:
124
+ commits.append({"hash": parts[0], "message": ""})
125
+
126
+ # Tasks
127
+ tasks = []
128
+ if tasks_dir.is_dir():
129
+ for d in tasks_dir.iterdir():
130
+ if d.is_dir() and d.name != "archive":
131
+ task_json_path = d / FILE_TASK_JSON
132
+ if task_json_path.is_file():
133
+ data = _read_json_file(task_json_path)
134
+ if data:
135
+ tasks.append(
136
+ {
137
+ "dir": d.name,
138
+ "name": data.get("name") or data.get("id") or "unknown",
139
+ "status": data.get("status", "unknown"),
140
+ }
141
+ )
142
+
143
+ return {
144
+ "developer": developer or "",
145
+ "git": {
146
+ "branch": branch,
147
+ "isClean": is_clean,
148
+ "uncommittedChanges": git_status_count,
149
+ "recentCommits": commits,
150
+ },
151
+ "tasks": {
152
+ "active": tasks,
153
+ "directory": f"{DIR_WORKFLOW}/{DIR_TASKS}",
154
+ },
155
+ "journal": {
156
+ "file": journal_relative,
157
+ "lines": journal_lines,
158
+ "nearLimit": journal_lines > 1800,
159
+ },
160
+ }
161
+
162
+
163
+ def output_json(repo_root: Path | None = None) -> None:
164
+ """Output context in JSON format.
165
+
166
+ Args:
167
+ repo_root: Repository root path. Defaults to auto-detected.
168
+ """
169
+ context = get_context_json(repo_root)
170
+ print(json.dumps(context, indent=2, ensure_ascii=False))
171
+
172
+
173
+ # =============================================================================
174
+ # Text Output
175
+ # =============================================================================
176
+
177
+
178
+ def get_context_text(repo_root: Path | None = None) -> str:
179
+ """Get context as formatted text.
180
+
181
+ Args:
182
+ repo_root: Repository root path. Defaults to auto-detected.
183
+
184
+ Returns:
185
+ Formatted text output.
186
+ """
187
+ if repo_root is None:
188
+ repo_root = get_repo_root()
189
+
190
+ lines = []
191
+ lines.append("========================================")
192
+ lines.append("SESSION CONTEXT")
193
+ lines.append("========================================")
194
+ lines.append("")
195
+
196
+ developer = get_developer(repo_root)
197
+
198
+ # Developer section
199
+ lines.append("## DEVELOPER")
200
+ if not developer:
201
+ lines.append(
202
+ f"ERROR: Not initialized. Run: python3 ./{DIR_WORKFLOW}/{DIR_SCRIPTS}/init_developer.py <name>"
203
+ )
204
+ return "\n".join(lines)
205
+
206
+ lines.append(f"Name: {developer}")
207
+ lines.append("")
208
+
209
+ # Git status
210
+ lines.append("## GIT STATUS")
211
+ _, branch_out, _ = _run_git_command(["branch", "--show-current"], cwd=repo_root)
212
+ branch = branch_out.strip() or "unknown"
213
+ lines.append(f"Branch: {branch}")
214
+
215
+ _, status_out, _ = _run_git_command(["status", "--porcelain"], cwd=repo_root)
216
+ status_lines = [line for line in status_out.splitlines() if line.strip()]
217
+ status_count = len(status_lines)
218
+
219
+ if status_count == 0:
220
+ lines.append("Working directory: Clean")
221
+ else:
222
+ lines.append(f"Working directory: {status_count} uncommitted change(s)")
223
+ lines.append("")
224
+ lines.append("Changes:")
225
+ _, short_out, _ = _run_git_command(["status", "--short"], cwd=repo_root)
226
+ for line in short_out.splitlines()[:10]:
227
+ lines.append(line)
228
+ lines.append("")
229
+
230
+ # Recent commits
231
+ lines.append("## RECENT COMMITS")
232
+ _, log_out, _ = _run_git_command(["log", "--oneline", "-5"], cwd=repo_root)
233
+ if log_out.strip():
234
+ for line in log_out.splitlines():
235
+ lines.append(line)
236
+ else:
237
+ lines.append("(no commits)")
238
+ lines.append("")
239
+
240
+ # Current task
241
+ lines.append("## CURRENT TASK")
242
+ current_task = get_current_task(repo_root)
243
+ if current_task:
244
+ current_task_dir = repo_root / current_task
245
+ task_json_path = current_task_dir / FILE_TASK_JSON
246
+ lines.append(f"Path: {current_task}")
247
+
248
+ if task_json_path.is_file():
249
+ data = _read_json_file(task_json_path)
250
+ if data:
251
+ t_name = data.get("name") or data.get("id") or "unknown"
252
+ t_status = data.get("status", "unknown")
253
+ t_created = data.get("createdAt", "unknown")
254
+ t_desc = data.get("description", "")
255
+
256
+ lines.append(f"Name: {t_name}")
257
+ lines.append(f"Status: {t_status}")
258
+ lines.append(f"Created: {t_created}")
259
+ if t_desc:
260
+ lines.append(f"Description: {t_desc}")
261
+
262
+ # Check for prd.md
263
+ prd_file = current_task_dir / "prd.md"
264
+ if prd_file.is_file():
265
+ lines.append("")
266
+ lines.append("[!] This task has prd.md - read it for task details")
267
+ else:
268
+ lines.append("(none)")
269
+ lines.append("")
270
+
271
+ # Active tasks
272
+ lines.append("## ACTIVE TASKS")
273
+ tasks_dir = get_tasks_dir(repo_root)
274
+ task_count = 0
275
+
276
+ if tasks_dir.is_dir():
277
+ for d in sorted(tasks_dir.iterdir()):
278
+ if d.is_dir() and d.name != "archive":
279
+ dir_name = d.name
280
+ t_json = d / FILE_TASK_JSON
281
+ status = "unknown"
282
+ assignee = "-"
283
+
284
+ if t_json.is_file():
285
+ data = _read_json_file(t_json)
286
+ if data:
287
+ status = data.get("status", "unknown")
288
+ assignee = data.get("assignee", "-")
289
+
290
+ lines.append(f"- {dir_name}/ ({status}) @{assignee}")
291
+ task_count += 1
292
+
293
+ if task_count == 0:
294
+ lines.append("(no active tasks)")
295
+ lines.append(f"Total: {task_count} active task(s)")
296
+ lines.append("")
297
+
298
+ # My tasks
299
+ lines.append("## MY TASKS (Assigned to me)")
300
+ my_task_count = 0
301
+
302
+ if tasks_dir.is_dir():
303
+ for d in sorted(tasks_dir.iterdir()):
304
+ if d.is_dir() and d.name != "archive":
305
+ t_json = d / FILE_TASK_JSON
306
+ if t_json.is_file():
307
+ data = _read_json_file(t_json)
308
+ if data:
309
+ assignee = data.get("assignee", "")
310
+ status = data.get("status", "planning")
311
+
312
+ if assignee == developer and status != "done":
313
+ title = data.get("title") or data.get("name") or "unknown"
314
+ priority = data.get("priority", "P2")
315
+ lines.append(f"- [{priority}] {title} ({status})")
316
+ my_task_count += 1
317
+
318
+ if my_task_count == 0:
319
+ lines.append("(no tasks assigned to you)")
320
+ lines.append("")
321
+
322
+ # Journal file
323
+ lines.append("## JOURNAL FILE")
324
+ journal_file = get_active_journal_file(repo_root)
325
+ if journal_file:
326
+ journal_lines = count_lines(journal_file)
327
+ relative = f"{DIR_WORKFLOW}/{DIR_WORKSPACE}/{developer}/{journal_file.name}"
328
+ lines.append(f"Active file: {relative}")
329
+ lines.append(f"Line count: {journal_lines} / 2000")
330
+ if journal_lines > 1800:
331
+ lines.append("[!] WARNING: Approaching 2000 line limit!")
332
+ else:
333
+ lines.append("No journal file found")
334
+ lines.append("")
335
+
336
+ # Paths
337
+ lines.append("## PATHS")
338
+ lines.append(f"Workspace: {DIR_WORKFLOW}/{DIR_WORKSPACE}/{developer}/")
339
+ lines.append(f"Tasks: {DIR_WORKFLOW}/{DIR_TASKS}/")
340
+ lines.append(f"Spec: {DIR_WORKFLOW}/{DIR_SPEC}/")
341
+ lines.append("")
342
+
343
+ lines.append("========================================")
344
+
345
+ return "\n".join(lines)
346
+
347
+
348
+ def output_text(repo_root: Path | None = None) -> None:
349
+ """Output context in text format.
350
+
351
+ Args:
352
+ repo_root: Repository root path. Defaults to auto-detected.
353
+ """
354
+ print(get_context_text(repo_root))
355
+
356
+
357
+ # =============================================================================
358
+ # Main Entry
359
+ # =============================================================================
360
+
361
+
362
+ def main() -> None:
363
+ """CLI entry point."""
364
+ import argparse
365
+
366
+ parser = argparse.ArgumentParser(description="Get Session Context for AI Agent")
367
+ parser.add_argument(
368
+ "--json",
369
+ "-j",
370
+ action="store_true",
371
+ help="Output context in JSON format",
372
+ )
373
+
374
+ args = parser.parse_args()
375
+
376
+ if args.json:
377
+ output_json()
378
+ else:
379
+ output_text()
380
+
381
+
382
+ if __name__ == "__main__":
383
+ main()