@shepai/cli 1.148.0 → 1.149.0-pr464.a84e893

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 (199) hide show
  1. package/dist/src/presentation/web/app/api/agent-events/route.js +1 -1
  2. package/dist/src/presentation/web/app/api/sessions/route.d.ts.map +1 -1
  3. package/dist/src/presentation/web/app/api/sessions/route.js +2 -268
  4. package/dist/src/presentation/web/app/api/sessions-batch/route.d.ts +17 -0
  5. package/dist/src/presentation/web/app/api/sessions-batch/route.d.ts.map +1 -0
  6. package/dist/src/presentation/web/app/api/sessions-batch/route.js +61 -0
  7. package/dist/src/presentation/web/components/common/control-center-drawer/feature-drawer-client.d.ts.map +1 -1
  8. package/dist/src/presentation/web/components/common/control-center-drawer/feature-drawer-client.js +18 -2
  9. package/dist/src/presentation/web/components/common/feature-node/feature-sessions-dropdown.d.ts +1 -1
  10. package/dist/src/presentation/web/components/common/feature-node/feature-sessions-dropdown.d.ts.map +1 -1
  11. package/dist/src/presentation/web/components/common/feature-node/feature-sessions-dropdown.js +15 -73
  12. package/dist/src/presentation/web/components/common/feature-node/feature-sessions-dropdown.stories.d.ts.map +1 -1
  13. package/dist/src/presentation/web/components/common/feature-node/feature-sessions-dropdown.stories.js +18 -17
  14. package/dist/src/presentation/web/components/features/control-center/control-center-inner.d.ts.map +1 -1
  15. package/dist/src/presentation/web/components/features/control-center/control-center-inner.js +18 -0
  16. package/dist/src/presentation/web/components/features/control-center/control-center.d.ts.map +1 -1
  17. package/dist/src/presentation/web/components/features/control-center/control-center.js +2 -1
  18. package/dist/src/presentation/web/components/features/control-center/use-control-center-state.d.ts.map +1 -1
  19. package/dist/src/presentation/web/components/features/control-center/use-control-center-state.js +4 -1
  20. package/dist/src/presentation/web/components/layouts/app-sidebar/app-sidebar.d.ts.map +1 -1
  21. package/dist/src/presentation/web/components/layouts/app-sidebar/app-sidebar.js +32 -33
  22. package/dist/src/presentation/web/hooks/sessions-provider.d.ts +12 -0
  23. package/dist/src/presentation/web/hooks/sessions-provider.d.ts.map +1 -0
  24. package/dist/src/presentation/web/hooks/sessions-provider.js +57 -0
  25. package/dist/src/presentation/web/hooks/use-deploy-action.d.ts.map +1 -1
  26. package/dist/src/presentation/web/hooks/use-deploy-action.js +8 -54
  27. package/dist/src/presentation/web/lib/session-scanner.d.ts +27 -0
  28. package/dist/src/presentation/web/lib/session-scanner.d.ts.map +1 -0
  29. package/dist/src/presentation/web/lib/session-scanner.js +255 -0
  30. package/dist/tsconfig.build.tsbuildinfo +1 -1
  31. package/package.json +1 -1
  32. package/web/.next/BUILD_ID +1 -1
  33. package/web/.next/app-path-routes-manifest.json +1 -0
  34. package/web/.next/build-manifest.json +2 -2
  35. package/web/.next/fallback-build-manifest.json +2 -2
  36. package/web/.next/prerender-manifest.json +3 -3
  37. package/web/.next/required-server-files.js +3 -3
  38. package/web/.next/required-server-files.json +3 -3
  39. package/web/.next/routes-manifest.json +6 -0
  40. package/web/.next/server/app/(dashboard)/@drawer/adopt/page/server-reference-manifest.json +28 -28
  41. package/web/.next/server/app/(dashboard)/@drawer/adopt/page.js.nft.json +1 -1
  42. package/web/.next/server/app/(dashboard)/@drawer/adopt/page_client-reference-manifest.js +1 -1
  43. package/web/.next/server/app/(dashboard)/@drawer/create/page/server-reference-manifest.json +28 -28
  44. package/web/.next/server/app/(dashboard)/@drawer/create/page.js.nft.json +1 -1
  45. package/web/.next/server/app/(dashboard)/@drawer/create/page_client-reference-manifest.js +1 -1
  46. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page/server-reference-manifest.json +36 -36
  47. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page.js.nft.json +1 -1
  48. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page_client-reference-manifest.js +1 -1
  49. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page/server-reference-manifest.json +36 -36
  50. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page.js.nft.json +1 -1
  51. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page_client-reference-manifest.js +1 -1
  52. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page/server-reference-manifest.json +26 -26
  53. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page.js.nft.json +1 -1
  54. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page_client-reference-manifest.js +1 -1
  55. package/web/.next/server/app/(dashboard)/create/page/server-reference-manifest.json +28 -28
  56. package/web/.next/server/app/(dashboard)/create/page.js.nft.json +1 -1
  57. package/web/.next/server/app/(dashboard)/create/page_client-reference-manifest.js +1 -1
  58. package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page/server-reference-manifest.json +36 -36
  59. package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page.js.nft.json +1 -1
  60. package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page_client-reference-manifest.js +1 -1
  61. package/web/.next/server/app/(dashboard)/feature/[featureId]/page/server-reference-manifest.json +36 -36
  62. package/web/.next/server/app/(dashboard)/feature/[featureId]/page.js.nft.json +1 -1
  63. package/web/.next/server/app/(dashboard)/feature/[featureId]/page_client-reference-manifest.js +1 -1
  64. package/web/.next/server/app/(dashboard)/page/server-reference-manifest.json +26 -26
  65. package/web/.next/server/app/(dashboard)/page.js.nft.json +1 -1
  66. package/web/.next/server/app/(dashboard)/page_client-reference-manifest.js +1 -1
  67. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page/server-reference-manifest.json +26 -26
  68. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page.js.nft.json +1 -1
  69. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page_client-reference-manifest.js +1 -1
  70. package/web/.next/server/app/_global-error.html +2 -2
  71. package/web/.next/server/app/_global-error.rsc +1 -1
  72. package/web/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +1 -1
  73. package/web/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  74. package/web/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  75. package/web/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  76. package/web/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  77. package/web/.next/server/app/_not-found/page/server-reference-manifest.json +3 -3
  78. package/web/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  79. package/web/.next/server/app/api/attachments/preview/route.js.nft.json +1 -1
  80. package/web/.next/server/app/api/evidence/route.js.nft.json +1 -1
  81. package/web/.next/server/app/api/graph-data/route.js.nft.json +1 -1
  82. package/web/.next/server/app/api/sessions/route.js +2 -3
  83. package/web/.next/server/app/api/sessions/route.js.nft.json +1 -1
  84. package/web/.next/server/app/api/sessions-batch/route/app-paths-manifest.json +3 -0
  85. package/web/.next/server/app/api/sessions-batch/route/build-manifest.json +11 -0
  86. package/web/.next/server/app/api/sessions-batch/route/server-reference-manifest.json +4 -0
  87. package/web/.next/server/app/api/sessions-batch/route.js +7 -0
  88. package/web/.next/server/app/api/sessions-batch/route.js.map +5 -0
  89. package/web/.next/server/app/api/sessions-batch/route.js.nft.json +1 -0
  90. package/web/.next/server/app/api/sessions-batch/route_client-reference-manifest.js +2 -0
  91. package/web/.next/server/app/settings/page/server-reference-manifest.json +8 -8
  92. package/web/.next/server/app/settings/page.js.nft.json +1 -1
  93. package/web/.next/server/app/settings/page_client-reference-manifest.js +1 -1
  94. package/web/.next/server/app/skills/page/server-reference-manifest.json +8 -8
  95. package/web/.next/server/app/skills/page.js.nft.json +1 -1
  96. package/web/.next/server/app/skills/page_client-reference-manifest.js +1 -1
  97. package/web/.next/server/app/tools/page/server-reference-manifest.json +8 -8
  98. package/web/.next/server/app/tools/page.js.nft.json +1 -1
  99. package/web/.next/server/app/tools/page_client-reference-manifest.js +1 -1
  100. package/web/.next/server/app/version/page/server-reference-manifest.json +3 -3
  101. package/web/.next/server/app/version/page_client-reference-manifest.js +1 -1
  102. package/web/.next/server/app-paths-manifest.json +1 -0
  103. package/web/.next/server/chunks/403f9_next_dist_esm_build_templates_app-route_4d623b8e.js +1 -1
  104. package/web/.next/server/chunks/403f9_next_dist_esm_build_templates_app-route_4d623b8e.js.map +1 -1
  105. package/web/.next/server/chunks/744ca_web__next-internal_server_app_api_sessions-batch_route_actions_4859f283.js +3 -0
  106. package/web/.next/server/chunks/[root-of-the-server]__0d33c29e._.js +3 -0
  107. package/web/.next/server/chunks/[root-of-the-server]__0d33c29e._.js.map +1 -0
  108. package/web/.next/server/chunks/[root-of-the-server]__2f61738a._.js +3 -0
  109. package/web/.next/server/chunks/[root-of-the-server]__2f61738a._.js.map +1 -0
  110. package/web/.next/server/chunks/[root-of-the-server]__a402b567._.js +1 -1
  111. package/web/.next/server/chunks/ssr/744ca_web_components_common_control-center-drawer_create-drawer-client_tsx_5e26fc0a._.js +1 -1
  112. package/web/.next/server/chunks/ssr/744ca_web_components_common_control-center-drawer_create-drawer-client_tsx_5e26fc0a._.js.map +1 -1
  113. package/web/.next/server/chunks/ssr/[root-of-the-server]__2138fa7e._.js +2 -2
  114. package/web/.next/server/chunks/ssr/[root-of-the-server]__29580090._.js +1 -1
  115. package/web/.next/server/chunks/ssr/[root-of-the-server]__29580090._.js.map +1 -1
  116. package/web/.next/server/chunks/ssr/[root-of-the-server]__357d99f9._.js +1 -1
  117. package/web/.next/server/chunks/ssr/[root-of-the-server]__3ef34e4c._.js +1 -1
  118. package/web/.next/server/chunks/ssr/[root-of-the-server]__43f51aa6._.js +1 -1
  119. package/web/.next/server/chunks/ssr/[root-of-the-server]__43f51aa6._.js.map +1 -1
  120. package/web/.next/server/chunks/ssr/[root-of-the-server]__815546bd._.js +1 -1
  121. package/web/.next/server/chunks/ssr/[root-of-the-server]__815546bd._.js.map +1 -1
  122. package/web/.next/server/chunks/ssr/[root-of-the-server]__aad040c0._.js +3 -3
  123. package/web/.next/server/chunks/ssr/[root-of-the-server]__aad040c0._.js.map +1 -1
  124. package/web/.next/server/chunks/ssr/[root-of-the-server]__c094882b._.js +1 -1
  125. package/web/.next/server/chunks/ssr/[root-of-the-server]__c094882b._.js.map +1 -1
  126. package/web/.next/server/chunks/ssr/[root-of-the-server]__d48c5b11._.js +1 -1
  127. package/web/.next/server/chunks/ssr/[root-of-the-server]__d48c5b11._.js.map +1 -1
  128. package/web/.next/server/chunks/ssr/[root-of-the-server]__dac5dbf1._.js +1 -1
  129. package/web/.next/server/chunks/ssr/[root-of-the-server]__dac5dbf1._.js.map +1 -1
  130. package/web/.next/server/chunks/ssr/[root-of-the-server]__fae8b355._.js +1 -1
  131. package/web/.next/server/chunks/ssr/[root-of-the-server]__fae8b355._.js.map +1 -1
  132. package/web/.next/server/chunks/ssr/_05c23ad9._.js +1 -1
  133. package/web/.next/server/chunks/ssr/_05c23ad9._.js.map +1 -1
  134. package/web/.next/server/chunks/ssr/_0c5f56e3._.js +2 -2
  135. package/web/.next/server/chunks/ssr/_0c5f56e3._.js.map +1 -1
  136. package/web/.next/server/chunks/ssr/_16eb4fec._.js +1 -1
  137. package/web/.next/server/chunks/ssr/_16eb4fec._.js.map +1 -1
  138. package/web/.next/server/chunks/ssr/_1b719e7f._.js +1 -1
  139. package/web/.next/server/chunks/ssr/_1b719e7f._.js.map +1 -1
  140. package/web/.next/server/chunks/ssr/_37e8548b._.js +1 -1
  141. package/web/.next/server/chunks/ssr/_37e8548b._.js.map +1 -1
  142. package/web/.next/server/chunks/ssr/_55d763e2._.js +1 -1
  143. package/web/.next/server/chunks/ssr/_55d763e2._.js.map +1 -1
  144. package/web/.next/server/chunks/ssr/_6256a985._.js +1 -1
  145. package/web/.next/server/chunks/ssr/_6256a985._.js.map +1 -1
  146. package/web/.next/server/chunks/ssr/_64bdfc6f._.js +2 -2
  147. package/web/.next/server/chunks/ssr/_64bdfc6f._.js.map +1 -1
  148. package/web/.next/server/chunks/ssr/_8c875922._.js +4 -0
  149. package/web/.next/server/chunks/ssr/_8c875922._.js.map +1 -0
  150. package/web/.next/server/chunks/ssr/_8fcc39d4._.js +1 -1
  151. package/web/.next/server/chunks/ssr/_b06fea81._.js +3 -0
  152. package/web/.next/server/chunks/ssr/{_fe63a7f9._.js.map → _b06fea81._.js.map} +1 -1
  153. package/web/.next/server/chunks/ssr/_b71645b4._.js +1 -1
  154. package/web/.next/server/chunks/ssr/_b71645b4._.js.map +1 -1
  155. package/web/.next/server/chunks/ssr/_d8575088._.js +1 -1
  156. package/web/.next/server/chunks/ssr/_d8575088._.js.map +1 -1
  157. package/web/.next/server/chunks/ssr/b1a17_presentation_web_components_features_settings_settings-page-client_tsx_6ed9d5f8._.js +1 -1
  158. package/web/.next/server/chunks/ssr/b1a17_presentation_web_components_features_settings_settings-page-client_tsx_6ed9d5f8._.js.map +1 -1
  159. package/web/.next/server/chunks/ssr/{src_presentation_web_7b2fda40._.js → src_presentation_web_3095acd1._.js} +2 -2
  160. package/web/.next/server/chunks/ssr/{src_presentation_web_7b2fda40._.js.map → src_presentation_web_3095acd1._.js.map} +1 -1
  161. package/web/.next/server/chunks/ssr/src_presentation_web__next-internal_server_app_skills_page_actions_1b176e3c.js +1 -1
  162. package/web/.next/server/chunks/ssr/src_presentation_web__next-internal_server_app_skills_page_actions_1b176e3c.js.map +1 -1
  163. package/web/.next/server/chunks/ssr/src_presentation_web__next-internal_server_app_tools_page_actions_bd9f0dda.js +1 -1
  164. package/web/.next/server/chunks/ssr/src_presentation_web__next-internal_server_app_tools_page_actions_bd9f0dda.js.map +1 -1
  165. package/web/.next/server/chunks/ssr/src_presentation_web_app_actions_open-ide_ts_baaca5d5._.js +1 -1
  166. package/web/.next/server/chunks/ssr/src_presentation_web_components_e599bb8c._.js +1 -1
  167. package/web/.next/server/chunks/ssr/src_presentation_web_components_e599bb8c._.js.map +1 -1
  168. package/web/.next/server/chunks/ssr/src_presentation_web_components_features_control-center_7ac3562e._.js +1 -1
  169. package/web/.next/server/chunks/ssr/src_presentation_web_components_features_control-center_7ac3562e._.js.map +1 -1
  170. package/web/.next/server/pages/500.html +2 -2
  171. package/web/.next/server/server-reference-manifest.js +1 -1
  172. package/web/.next/server/server-reference-manifest.json +44 -44
  173. package/web/.next/static/chunks/{04869f1d3f5d9071.js → 1b1f15f424679455.js} +1 -1
  174. package/web/.next/static/chunks/{9dad6769d10a32df.js → 23c3e86ce0e24e00.js} +1 -1
  175. package/web/.next/static/chunks/36e2fd941864a90e.js +1 -0
  176. package/web/.next/static/chunks/{c731682077fbac4f.js → 3a4705f59cfd89a9.js} +1 -1
  177. package/web/.next/static/chunks/{6f76e63ead3fac2e.js → 51dee46741af1e4b.js} +1 -1
  178. package/web/.next/static/chunks/{48850e202dd814ac.js → 55683e8c2c1639b7.js} +1 -1
  179. package/web/.next/static/chunks/616c959907e661c9.js +1 -0
  180. package/web/.next/static/chunks/{21e82fee1a7e1668.js → 62501e23f82d4115.js} +1 -1
  181. package/web/.next/static/chunks/{7c5131e33516a325.js → 862d6fa007e83cb5.js} +1 -1
  182. package/web/.next/static/chunks/8e2039e569c81a6d.js +1 -0
  183. package/web/.next/static/chunks/a26e8c3319a6b723.js +5 -0
  184. package/web/.next/static/chunks/f946e2a33ee3b18e.js +2 -0
  185. package/web/.next/server/chunks/403f9_next_dist_esm_build_templates_app-route_ff60e4a5.js +0 -3
  186. package/web/.next/server/chunks/403f9_next_dist_esm_build_templates_app-route_ff60e4a5.js.map +0 -1
  187. package/web/.next/server/chunks/[externals]__448264a3._.js +0 -3
  188. package/web/.next/server/chunks/ssr/_4533d6f8._.js +0 -4
  189. package/web/.next/server/chunks/ssr/_4533d6f8._.js.map +0 -1
  190. package/web/.next/server/chunks/ssr/_fe63a7f9._.js +0 -3
  191. package/web/.next/static/chunks/0137d4850cab3c45.js +0 -5
  192. package/web/.next/static/chunks/063a24b49d9818a0.js +0 -1
  193. package/web/.next/static/chunks/682563e4503cbd58.js +0 -1
  194. package/web/.next/static/chunks/683b1d85e789c2eb.js +0 -2
  195. package/web/.next/static/chunks/d62ae5e449d87057.js +0 -1
  196. /package/web/.next/server/chunks/{[externals]__448264a3._.js.map → 744ca_web__next-internal_server_app_api_sessions-batch_route_actions_4859f283.js.map} +0 -0
  197. /package/web/.next/static/{zYKuE1zbe1UWwAJv5EVwg → QdGWt290JhTjWmj28lHyG}/_buildManifest.js +0 -0
  198. /package/web/.next/static/{zYKuE1zbe1UWwAJv5EVwg → QdGWt290JhTjWmj28lHyG}/_clientMiddlewareManifest.json +0 -0
  199. /package/web/.next/static/{zYKuE1zbe1UWwAJv5EVwg → QdGWt290JhTjWmj28lHyG}/_ssgManifest.js +0 -0
@@ -0,0 +1,255 @@
1
+ /**
2
+ * Shared session scanning logic used by both /api/sessions and /api/sessions-batch.
3
+ *
4
+ * Scans Claude Code and Cursor session directories for JSONL session files,
5
+ * parsing headers to extract preview, message count, and timestamps.
6
+ */
7
+ import { createHash } from 'node:crypto';
8
+ import { homedir } from 'node:os';
9
+ import { join } from 'node:path';
10
+ import { readdir, stat } from 'node:fs/promises';
11
+ // ── Path encoding helpers ─────────────────────────────────────────────
12
+ /**
13
+ * Claude Code encodes paths by replacing '/', '\', '.' with '-'.
14
+ * e.g. /home/user/.shep/repos/abc → -home-user--shep-repos-abc
15
+ */
16
+ function claudeEncodePath(p) {
17
+ return p.replace(/[/\\.]/g, '-');
18
+ }
19
+ /**
20
+ * Cursor encodes paths by stripping the leading '/', removing dots,
21
+ * and replacing '/' and '\' with '-'.
22
+ * e.g. /home/user/.shep/repos/abc → home-user-shep-repos-abc
23
+ */
24
+ function cursorEncodePath(p) {
25
+ return p.replace(/^\//, '').replace(/\./g, '').replace(/[/\\]/g, '-');
26
+ }
27
+ // ── Shared helpers ────────────────────────────────────────────────────
28
+ function extractText(content) {
29
+ if (typeof content === 'string')
30
+ return content;
31
+ if (Array.isArray(content)) {
32
+ for (const block of content) {
33
+ if (typeof block === 'object' && block !== null) {
34
+ const b = block;
35
+ if (b.type === 'text' && typeof b.text === 'string')
36
+ return b.text;
37
+ }
38
+ }
39
+ }
40
+ return null;
41
+ }
42
+ const PREVIEW_READ_BYTES = 8_192; // 8KB is enough for first few messages
43
+ // ── Claude Code session scanner ───────────────────────────────────────
44
+ async function collectJsonlFiles(projectDir) {
45
+ let entries;
46
+ try {
47
+ entries = await readdir(projectDir);
48
+ }
49
+ catch {
50
+ return [];
51
+ }
52
+ const jsonlFiles = entries.filter((e) => e.endsWith('.jsonl'));
53
+ const fileInfos = await Promise.allSettled(jsonlFiles.map(async (name) => {
54
+ const filePath = join(projectDir, name);
55
+ const s = await stat(filePath);
56
+ return { name, filePath, mtime: s.mtime.getTime() };
57
+ }));
58
+ return fileInfos
59
+ .filter((r) => r.status === 'fulfilled')
60
+ .map((r) => r.value);
61
+ }
62
+ async function parseClaudeSession(filePath, fileName, mtime, repositoryPath) {
63
+ const { createReadStream } = await import('node:fs');
64
+ const id = fileName.replace('.jsonl', '');
65
+ let preview = null;
66
+ let firstTimestamp = null;
67
+ let messageCount = 0;
68
+ const head = await new Promise((resolve) => {
69
+ const chunks = [];
70
+ let size = 0;
71
+ const stream = createReadStream(filePath, { end: PREVIEW_READ_BYTES - 1 });
72
+ stream.on('data', (chunk) => {
73
+ chunks.push(chunk);
74
+ size += chunk.length;
75
+ });
76
+ stream.on('end', () => resolve(Buffer.concat(chunks, size).toString('utf-8')));
77
+ stream.on('error', () => resolve(''));
78
+ });
79
+ if (!head)
80
+ return null;
81
+ const lines = head.split('\n').filter((l) => l.trim());
82
+ for (const line of lines) {
83
+ try {
84
+ const entry = JSON.parse(line);
85
+ if (entry.type === 'user' || entry.type === 'assistant') {
86
+ const role = entry.message?.role;
87
+ if (role === 'user' || role === 'assistant') {
88
+ messageCount++;
89
+ if (entry.timestamp) {
90
+ firstTimestamp ??= entry.timestamp;
91
+ }
92
+ if (role === 'user' && preview === null) {
93
+ preview = extractText(entry.message?.content);
94
+ }
95
+ }
96
+ }
97
+ }
98
+ catch {
99
+ break;
100
+ }
101
+ }
102
+ if (messageCount === 0)
103
+ return null;
104
+ const mtimeIso = new Date(mtime).toISOString();
105
+ return {
106
+ id,
107
+ agentType: 'claude-code',
108
+ preview,
109
+ messageCount,
110
+ firstMessageAt: firstTimestamp,
111
+ lastMessageAt: mtimeIso,
112
+ createdAt: firstTimestamp ?? mtimeIso,
113
+ projectPath: repositoryPath,
114
+ filePath,
115
+ _mtime: mtime,
116
+ };
117
+ }
118
+ export async function scanClaudeSessions(repositoryPath, limit, includeWorktrees = false) {
119
+ const dirName = claudeEncodePath(repositoryPath);
120
+ const projectsRoot = join(homedir(), '.claude', 'projects');
121
+ const primaryDir = join(projectsRoot, dirName);
122
+ let allFiles = await collectJsonlFiles(primaryDir);
123
+ if (includeWorktrees) {
124
+ try {
125
+ const allDirs = await readdir(projectsRoot);
126
+ const prefixMatches = allDirs.filter((d) => d !== dirName && d.startsWith(dirName));
127
+ const normalizedRepoPath = repositoryPath.replace(/\\/g, '/');
128
+ const repoHash = createHash('sha256').update(normalizedRepoPath).digest('hex').slice(0, 16);
129
+ const shepHome = join(homedir(), '.shep').replace(/\\/g, '/');
130
+ const shepWorktreePrefix = claudeEncodePath(join(shepHome, 'repos', repoHash));
131
+ const shepMatches = allDirs.filter((d) => d.startsWith(shepWorktreePrefix) && !prefixMatches.includes(d) && d !== dirName);
132
+ const worktreeDirs = [...prefixMatches, ...shepMatches];
133
+ const worktreeResults = await Promise.all(worktreeDirs.map((d) => collectJsonlFiles(join(projectsRoot, d))));
134
+ for (const files of worktreeResults) {
135
+ allFiles = allFiles.concat(files);
136
+ }
137
+ }
138
+ catch {
139
+ // projectsRoot doesn't exist — no sessions at all
140
+ }
141
+ }
142
+ const valid = allFiles.sort((a, b) => b.mtime - a.mtime).slice(0, limit);
143
+ const results = await Promise.allSettled(valid.map(async (fi) => parseClaudeSession(fi.filePath, fi.name, fi.mtime, repositoryPath)));
144
+ return results
145
+ .filter((r) => r.status === 'fulfilled')
146
+ .map((r) => r.value)
147
+ .filter((s) => s !== null);
148
+ }
149
+ // ── Cursor session scanner ────────────────────────────────────────────
150
+ async function parseCursorSession(filePath, fileName, mtime, repositoryPath) {
151
+ const { createReadStream } = await import('node:fs');
152
+ const id = fileName.replace('.jsonl', '');
153
+ const head = await new Promise((resolve) => {
154
+ const chunks = [];
155
+ let size = 0;
156
+ const stream = createReadStream(filePath, { end: PREVIEW_READ_BYTES - 1 });
157
+ stream.on('data', (chunk) => {
158
+ chunks.push(chunk);
159
+ size += chunk.length;
160
+ });
161
+ stream.on('end', () => resolve(Buffer.concat(chunks, size).toString('utf-8')));
162
+ stream.on('error', () => resolve(''));
163
+ });
164
+ if (!head)
165
+ return null;
166
+ let preview = null;
167
+ let messageCount = 0;
168
+ const lines = head.split('\n').filter((l) => l.trim());
169
+ for (const line of lines) {
170
+ try {
171
+ const entry = JSON.parse(line);
172
+ if (entry.role === 'user' || entry.role === 'assistant') {
173
+ messageCount++;
174
+ if (entry.role === 'user' && preview === null) {
175
+ preview = extractText(entry.message?.content);
176
+ }
177
+ }
178
+ }
179
+ catch {
180
+ break;
181
+ }
182
+ }
183
+ if (messageCount === 0)
184
+ return null;
185
+ const mtimeIso = new Date(mtime).toISOString();
186
+ return {
187
+ id,
188
+ agentType: 'cursor',
189
+ preview,
190
+ messageCount,
191
+ firstMessageAt: mtimeIso,
192
+ lastMessageAt: mtimeIso,
193
+ createdAt: mtimeIso,
194
+ projectPath: repositoryPath,
195
+ filePath,
196
+ _mtime: mtime,
197
+ };
198
+ }
199
+ export async function scanCursorSessions(repositoryPath, limit) {
200
+ const dirName = cursorEncodePath(repositoryPath);
201
+ const transcriptsDir = join(homedir(), '.cursor', 'projects', dirName, 'agent-transcripts');
202
+ let entries;
203
+ try {
204
+ entries = await readdir(transcriptsDir);
205
+ }
206
+ catch {
207
+ return [];
208
+ }
209
+ const fileInfos = await Promise.allSettled(entries.map(async (entry) => {
210
+ const entryPath = join(transcriptsDir, entry);
211
+ const s = await stat(entryPath);
212
+ if (s.isFile() && entry.endsWith('.jsonl')) {
213
+ return { name: entry, filePath: entryPath, mtime: s.mtime.getTime() };
214
+ }
215
+ if (s.isDirectory()) {
216
+ const jsonlPath = join(entryPath, `${entry}.jsonl`);
217
+ try {
218
+ const jsonlStat = await stat(jsonlPath);
219
+ return {
220
+ name: `${entry}.jsonl`,
221
+ filePath: jsonlPath,
222
+ mtime: jsonlStat.mtime.getTime(),
223
+ };
224
+ }
225
+ catch {
226
+ return null;
227
+ }
228
+ }
229
+ return null;
230
+ }));
231
+ const valid = fileInfos
232
+ .filter((r) => r.status === 'fulfilled')
233
+ .map((r) => r.value)
234
+ .filter((v) => v !== null)
235
+ .sort((a, b) => b.mtime - a.mtime)
236
+ .slice(0, limit);
237
+ const results = await Promise.allSettled(valid.map(async (fi) => parseCursorSession(fi.filePath, fi.name, fi.mtime, repositoryPath)));
238
+ return results
239
+ .filter((r) => r.status === 'fulfilled')
240
+ .map((r) => r.value)
241
+ .filter((s) => s !== null);
242
+ }
243
+ /**
244
+ * Scan sessions for a single repository path from all providers.
245
+ * Merges and sorts by recency.
246
+ */
247
+ export async function scanSessionsForPath(repositoryPath, limit, includeWorktrees = false) {
248
+ const [claudeSessions, cursorSessions] = await Promise.all([
249
+ scanClaudeSessions(repositoryPath, limit, includeWorktrees),
250
+ scanCursorSessions(repositoryPath, limit),
251
+ ]);
252
+ return [...claudeSessions, ...cursorSessions]
253
+ .sort((a, b) => b._mtime - a._mtime)
254
+ .slice(0, Math.min(limit, 50));
255
+ }