@stackmemoryai/stackmemory 0.3.17 → 0.3.19

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 (234) hide show
  1. package/dist/cli/claude-sm.js +51 -5
  2. package/dist/cli/claude-sm.js.map +2 -2
  3. package/dist/cli/codex-sm.js +52 -19
  4. package/dist/cli/codex-sm.js.map +2 -2
  5. package/dist/cli/commands/db.js +143 -0
  6. package/dist/cli/commands/db.js.map +7 -0
  7. package/dist/cli/commands/login.js +50 -0
  8. package/dist/cli/commands/login.js.map +7 -0
  9. package/dist/cli/commands/migrate.js +178 -0
  10. package/dist/cli/commands/migrate.js.map +7 -0
  11. package/dist/cli/commands/onboard.js +158 -2
  12. package/dist/cli/commands/onboard.js.map +2 -2
  13. package/dist/cli/commands/skills.js +15 -2
  14. package/dist/cli/commands/skills.js.map +2 -2
  15. package/dist/cli/index.js +118 -834
  16. package/dist/cli/index.js.map +3 -3
  17. package/dist/core/context/dual-stack-manager.js +1 -1
  18. package/dist/core/context/dual-stack-manager.js.map +1 -1
  19. package/dist/core/context/frame-database.js +1 -0
  20. package/dist/core/context/frame-database.js.map +2 -2
  21. package/dist/core/context/frame-manager.js +59 -2
  22. package/dist/core/context/frame-manager.js.map +2 -2
  23. package/dist/core/database/database-adapter.js +6 -1
  24. package/dist/core/database/database-adapter.js.map +2 -2
  25. package/dist/core/database/sqlite-adapter.js +60 -2
  26. package/dist/core/database/sqlite-adapter.js.map +2 -2
  27. package/dist/integrations/claude-code/subagent-client.js +106 -3
  28. package/dist/integrations/claude-code/subagent-client.js.map +2 -2
  29. package/dist/servers/railway/config.js +51 -0
  30. package/dist/servers/railway/config.js.map +7 -0
  31. package/dist/servers/railway/index-enhanced.js +156 -0
  32. package/dist/servers/railway/index-enhanced.js.map +7 -0
  33. package/dist/servers/railway/index.js +843 -82
  34. package/dist/servers/railway/index.js.map +3 -3
  35. package/dist/servers/railway/minimal.js +48 -3
  36. package/dist/servers/railway/minimal.js.map +2 -2
  37. package/dist/servers/railway/storage-test.js +455 -0
  38. package/dist/servers/railway/storage-test.js.map +7 -0
  39. package/dist/skills/claude-skills.js +13 -12
  40. package/dist/skills/claude-skills.js.map +2 -2
  41. package/dist/skills/recursive-agent-orchestrator.js +27 -18
  42. package/dist/skills/recursive-agent-orchestrator.js.map +2 -2
  43. package/dist/skills/unified-rlm-orchestrator.js.map +2 -2
  44. package/package.json +13 -21
  45. package/scripts/README-TESTING.md +186 -0
  46. package/scripts/analyze-cli-security.js +288 -0
  47. package/scripts/archive/add-phase-tasks-to-linear.js +163 -0
  48. package/scripts/archive/analyze-linear-duplicates.js +214 -0
  49. package/scripts/archive/analyze-remaining-duplicates.js +230 -0
  50. package/scripts/archive/analyze-sta-duplicates.js +292 -0
  51. package/scripts/archive/analyze-sta-graphql.js +399 -0
  52. package/scripts/archive/cancel-duplicate-tasks.ts +246 -0
  53. package/scripts/archive/check-all-duplicates.ts +419 -0
  54. package/scripts/archive/clean-duplicate-tasks.js +114 -0
  55. package/scripts/archive/cleanup-duplicate-tasks.ts +286 -0
  56. package/scripts/archive/create-phase-tasks.js +387 -0
  57. package/scripts/archive/delete-linear-duplicates.js +182 -0
  58. package/scripts/archive/delete-remaining-duplicates.js +158 -0
  59. package/scripts/archive/delete-sta-duplicates.js +201 -0
  60. package/scripts/archive/delete-sta-oauth.js +201 -0
  61. package/scripts/archive/export-sta-tasks.js +62 -0
  62. package/scripts/archive/install-auto-sync.js +266 -0
  63. package/scripts/archive/install-chromadb-hooks.sh +133 -0
  64. package/scripts/archive/install-enhanced-clear-hooks.sh +431 -0
  65. package/scripts/archive/install-post-task-hooks.sh +289 -0
  66. package/scripts/archive/install-stackmemory-hooks.sh +420 -0
  67. package/scripts/archive/merge-linear-duplicates-safe.ts +362 -0
  68. package/scripts/archive/merge-linear-duplicates.ts +180 -0
  69. package/scripts/archive/remove-sta-tasks.js +70 -0
  70. package/scripts/archive/setup-background-sync.sh +168 -0
  71. package/scripts/archive/setup-claude-auto-triggers.sh +181 -0
  72. package/scripts/archive/setup-claude-autostart.sh +305 -0
  73. package/scripts/archive/setup-git-hooks.sh +25 -0
  74. package/scripts/archive/setup-linear-oauth.sh +46 -0
  75. package/scripts/archive/setup-mcp.sh +113 -0
  76. package/scripts/archive/setup-railway-deployment.sh +81 -0
  77. package/scripts/auto-handoff.sh +262 -0
  78. package/scripts/background-sync-manager.js +416 -0
  79. package/scripts/benchmark-performance.ts +57 -0
  80. package/scripts/check-redis.ts +48 -0
  81. package/scripts/chromadb-auto-loader.sh +128 -0
  82. package/scripts/chromadb-context-loader.js +479 -0
  83. package/scripts/claude-chromadb-hook.js +460 -0
  84. package/scripts/claude-code-wrapper.sh +66 -0
  85. package/scripts/claude-linear-skill.js +455 -0
  86. package/scripts/claude-pre-commit.sh +302 -0
  87. package/scripts/claude-sm-autostart.js +532 -0
  88. package/scripts/claude-sm-setup.sh +367 -0
  89. package/scripts/claude-with-chromadb.sh +69 -0
  90. package/scripts/claude-worktree-manager.sh +323 -0
  91. package/scripts/claude-worktree-monitor.sh +371 -0
  92. package/scripts/claude-worktree-setup.sh +327 -0
  93. package/scripts/clean-linear-backlog.js +273 -0
  94. package/scripts/cleanup-old-sessions.sh +57 -0
  95. package/scripts/codex-wrapper.sh +88 -0
  96. package/scripts/create-sandbox.sh +269 -0
  97. package/scripts/debug-linear-update.js +174 -0
  98. package/scripts/delete-linear-tasks.js +167 -0
  99. package/scripts/deploy.sh +89 -0
  100. package/scripts/deployment/railway.sh +352 -0
  101. package/scripts/deployment/test-deployment.js +194 -0
  102. package/scripts/detect-and-rehydrate.js +162 -0
  103. package/scripts/detect-and-rehydrate.mjs +165 -0
  104. package/scripts/development/create-demo-tasks.js +143 -0
  105. package/scripts/development/debug-frame-test.js +16 -0
  106. package/scripts/development/demo-auto-sync.js +128 -0
  107. package/scripts/development/fix-all-imports.js +213 -0
  108. package/scripts/development/fix-imports.js +229 -0
  109. package/scripts/development/fix-lint-loop.cjs +103 -0
  110. package/scripts/development/fix-project-id.ts +161 -0
  111. package/scripts/development/fix-strict-mode-issues.ts +291 -0
  112. package/scripts/development/reorganize-structure.sh +228 -0
  113. package/scripts/development/test-persistence-direct.js +148 -0
  114. package/scripts/development/test-persistence.js +114 -0
  115. package/scripts/development/test-tasks.js +93 -0
  116. package/scripts/development/update-imports.js +212 -0
  117. package/scripts/fetch-linear-status.js +125 -0
  118. package/scripts/git-hooks/README.md +310 -0
  119. package/scripts/git-hooks/branch-context-manager.sh +342 -0
  120. package/scripts/git-hooks/post-checkout-stackmemory.sh +63 -0
  121. package/scripts/git-hooks/post-commit-stackmemory.sh +305 -0
  122. package/scripts/git-hooks/pre-commit-stackmemory.sh +275 -0
  123. package/scripts/hooks/cleanup-shell.sh +130 -0
  124. package/scripts/hooks/task-complete.sh +114 -0
  125. package/scripts/initialize.ts +129 -0
  126. package/scripts/install-claude-hooks-auto.js +104 -0
  127. package/scripts/install-claude-hooks.sh +133 -0
  128. package/scripts/install-global.sh +296 -0
  129. package/scripts/install.sh +235 -0
  130. package/scripts/linear-auto-sync.js +262 -0
  131. package/scripts/linear-auto-sync.sh +161 -0
  132. package/scripts/linear-sync-daemon.js +150 -0
  133. package/scripts/linear-task-review.js +237 -0
  134. package/scripts/list-linear-tasks.ts +178 -0
  135. package/scripts/mcp-proxy.js +66 -0
  136. package/scripts/opencode-wrapper.sh +85 -0
  137. package/scripts/publish-local.js +74 -0
  138. package/scripts/query-chromadb.ts +201 -0
  139. package/scripts/railway-env-setup.sh +39 -0
  140. package/scripts/reconcile-local-tasks.js +170 -0
  141. package/scripts/recreate-frames-db.js +89 -0
  142. package/scripts/setup/claude-integration.js +138 -0
  143. package/scripts/setup/configure-alias.js +125 -0
  144. package/scripts/setup/configure-codex-alias.js +161 -0
  145. package/scripts/setup/configure-opencode-alias.js +175 -0
  146. package/scripts/setup-claude-integration.js +204 -0
  147. package/scripts/setup-claude-integration.sh +183 -0
  148. package/scripts/setup-railway-deployment.sh +37 -0
  149. package/scripts/setup.sh +31 -0
  150. package/scripts/show-linear-summary.ts +172 -0
  151. package/scripts/stackmemory-auto-handoff.sh +231 -0
  152. package/scripts/stackmemory-daemon.sh +40 -0
  153. package/scripts/start-linear-sync-daemon.sh +141 -0
  154. package/scripts/start-temporal-paradox.sh +214 -0
  155. package/scripts/status.ts +159 -0
  156. package/scripts/sync-and-clean-tasks.js +258 -0
  157. package/scripts/sync-frames-from-railway.js +228 -0
  158. package/scripts/sync-linear-graphql.js +303 -0
  159. package/scripts/sync-linear-tasks.js +186 -0
  160. package/scripts/test-auto-triggers.sh +57 -0
  161. package/scripts/test-browser-mcp.js +74 -0
  162. package/scripts/test-chromadb-full.js +115 -0
  163. package/scripts/test-chromadb-hooks.sh +28 -0
  164. package/scripts/test-chromadb-sync.ts +245 -0
  165. package/scripts/test-cli-security.js +293 -0
  166. package/scripts/test-hooks-persistence.sh +220 -0
  167. package/scripts/test-installation-scenarios.sh +359 -0
  168. package/scripts/test-installation.sh +224 -0
  169. package/scripts/test-mcp.js +163 -0
  170. package/scripts/test-pre-publish-quick.sh +75 -0
  171. package/scripts/test-quality-gates.sh +263 -0
  172. package/scripts/test-railway-db.js +222 -0
  173. package/scripts/test-redis-storage.ts +490 -0
  174. package/scripts/test-rlm-basic.sh +122 -0
  175. package/scripts/test-rlm-comprehensive.sh +260 -0
  176. package/scripts/test-rlm-e2e.sh +268 -0
  177. package/scripts/test-rlm-simple.js +90 -0
  178. package/scripts/test-rlm.js +110 -0
  179. package/scripts/test-session-handoff.sh +165 -0
  180. package/scripts/test-shell-integration.sh +275 -0
  181. package/scripts/testing/ab-test-runner.ts +508 -0
  182. package/scripts/testing/collect-metrics.ts +457 -0
  183. package/scripts/testing/quick-effectiveness-demo.js +187 -0
  184. package/scripts/testing/real-performance-test.js +422 -0
  185. package/scripts/testing/run-effectiveness-tests.sh +176 -0
  186. package/scripts/testing/scripts/testing/ab-test-runner.js +363 -0
  187. package/scripts/testing/scripts/testing/collect-metrics.js +292 -0
  188. package/scripts/testing/simple-effectiveness-test.js +310 -0
  189. package/scripts/testing/src/core/context/context-bridge.js +253 -0
  190. package/scripts/testing/src/core/context/frame-manager.js +746 -0
  191. package/scripts/testing/src/core/context/shared-context-layer.js +437 -0
  192. package/scripts/testing/src/core/database/database-adapter.js +54 -0
  193. package/scripts/testing/src/core/errors/index.js +291 -0
  194. package/scripts/testing/src/core/errors/recovery.js +268 -0
  195. package/scripts/testing/src/core/monitoring/logger.js +145 -0
  196. package/scripts/testing/src/core/retrieval/context-retriever.js +516 -0
  197. package/scripts/testing/src/core/session/index.js +1 -0
  198. package/scripts/testing/src/core/session/session-manager.js +323 -0
  199. package/scripts/testing/src/core/trace/cli-trace-wrapper.js +140 -0
  200. package/scripts/testing/src/core/trace/db-trace-wrapper.js +251 -0
  201. package/scripts/testing/src/core/trace/debug-trace.js +398 -0
  202. package/scripts/testing/src/core/trace/index.js +120 -0
  203. package/scripts/testing/src/core/trace/linear-api-wrapper.js +204 -0
  204. package/scripts/update-linear-status.js +268 -0
  205. package/scripts/update-linear-tasks-fixed.js +284 -0
  206. package/scripts/verify-railway-schema.ts +35 -0
  207. package/templates/claude-hooks/hooks.json +5 -0
  208. package/templates/claude-hooks/on-clear.js +56 -0
  209. package/templates/claude-hooks/on-startup.js +56 -0
  210. package/templates/claude-hooks/tool-use-trace.js +67 -0
  211. package/dist/features/tui/components/analytics-panel.js +0 -157
  212. package/dist/features/tui/components/analytics-panel.js.map +0 -7
  213. package/dist/features/tui/components/frame-visualizer.js +0 -377
  214. package/dist/features/tui/components/frame-visualizer.js.map +0 -7
  215. package/dist/features/tui/components/pr-tracker.js +0 -135
  216. package/dist/features/tui/components/pr-tracker.js.map +0 -7
  217. package/dist/features/tui/components/session-monitor.js +0 -299
  218. package/dist/features/tui/components/session-monitor.js.map +0 -7
  219. package/dist/features/tui/components/subagent-fleet.js +0 -395
  220. package/dist/features/tui/components/subagent-fleet.js.map +0 -7
  221. package/dist/features/tui/components/task-board.js +0 -1139
  222. package/dist/features/tui/components/task-board.js.map +0 -7
  223. package/dist/features/tui/index.js +0 -408
  224. package/dist/features/tui/index.js.map +0 -7
  225. package/dist/features/tui/services/data-service.js +0 -641
  226. package/dist/features/tui/services/data-service.js.map +0 -7
  227. package/dist/features/tui/services/linear-task-reader.js +0 -102
  228. package/dist/features/tui/services/linear-task-reader.js.map +0 -7
  229. package/dist/features/tui/services/websocket-client.js +0 -162
  230. package/dist/features/tui/services/websocket-client.js.map +0 -7
  231. package/dist/features/tui/terminal-compat.js +0 -220
  232. package/dist/features/tui/terminal-compat.js.map +0 -7
  233. package/dist/features/tui/types.js +0 -1
  234. package/dist/features/tui/types.js.map +0 -7
@@ -0,0 +1,50 @@
1
+ import inquirer from "inquirer";
2
+ import chalk from "chalk";
3
+ import { homedir } from "os";
4
+ import { join } from "path";
5
+ import { existsSync, mkdirSync, writeFileSync, readFileSync } from "fs";
6
+ function registerLoginCommand(program) {
7
+ program.command("login").description("Login to hosted StackMemory (configure managed Postgres)").option("--open", "Open hosted signup/login page before prompting").action(async (options) => {
8
+ const cfgDir = join(homedir(), ".stackmemory");
9
+ if (!existsSync(cfgDir)) mkdirSync(cfgDir, { recursive: true });
10
+ if (options.open) {
11
+ try {
12
+ const signupUrl = "https://stackmemory.ai/hosted";
13
+ const mod = await import("open");
14
+ await mod.default(signupUrl);
15
+ } catch (e) {
16
+ console.log(chalk.yellow("Could not open browser automatically."));
17
+ }
18
+ }
19
+ const { databaseUrl } = await inquirer.prompt([
20
+ {
21
+ type: "password",
22
+ name: "databaseUrl",
23
+ message: "Paste your hosted DATABASE_URL (postgres://...)",
24
+ validate: (input) => input.startsWith("postgres://") || input.startsWith("postgresql://") ? true : "Must start with postgres:// or postgresql://"
25
+ }
26
+ ]);
27
+ const cfgPath = join(cfgDir, "config.json");
28
+ let cfg = {};
29
+ try {
30
+ if (existsSync(cfgPath)) cfg = JSON.parse(readFileSync(cfgPath, "utf-8"));
31
+ } catch {
32
+ }
33
+ cfg.database = { ...cfg.database || {}, mode: "hosted", url: databaseUrl };
34
+ writeFileSync(cfgPath, JSON.stringify(cfg, null, 2));
35
+ console.log(chalk.green("\u2713 Hosted database configured in ~/.stackmemory/config.json"));
36
+ try {
37
+ const envFile = join(cfgDir, "railway.env");
38
+ writeFileSync(envFile, `# StackMemory hosted DB
39
+ DATABASE_URL=${databaseUrl}
40
+ `);
41
+ console.log(chalk.green("\u2713 Saved DATABASE_URL to ~/.stackmemory/railway.env"));
42
+ } catch {
43
+ }
44
+ console.log(chalk.gray("Tip: export DATABASE_URL before starting the server."));
45
+ });
46
+ }
47
+ export {
48
+ registerLoginCommand
49
+ };
50
+ //# sourceMappingURL=login.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../src/cli/commands/login.ts"],
4
+ "sourcesContent": ["import { Command } from 'commander';\nimport inquirer from 'inquirer';\nimport chalk from 'chalk';\nimport { homedir } from 'os';\nimport { join } from 'path';\nimport { existsSync, mkdirSync, writeFileSync, readFileSync } from 'fs';\n\ninterface ConfigShape {\n version?: string;\n setupCompleted?: string;\n features?: any;\n paths?: any;\n database?: { mode?: 'local' | 'hosted'; url?: string };\n}\n\nexport function registerLoginCommand(program: Command): void {\n program\n .command('login')\n .description('Login to hosted StackMemory (configure managed Postgres)')\n .option('--open', 'Open hosted signup/login page before prompting')\n .action(async (options) => {\n const cfgDir = join(homedir(), '.stackmemory');\n if (!existsSync(cfgDir)) mkdirSync(cfgDir, { recursive: true });\n\n if (options.open) {\n try {\n const signupUrl = 'https://stackmemory.ai/hosted';\n const mod = await import('open');\n await mod.default(signupUrl);\n } catch (e) {\n console.log(chalk.yellow('Could not open browser automatically.'));\n }\n }\n\n const { databaseUrl } = await inquirer.prompt([\n {\n type: 'password',\n name: 'databaseUrl',\n message: 'Paste your hosted DATABASE_URL (postgres://...)',\n validate: (input: string) =>\n input.startsWith('postgres://') || input.startsWith('postgresql://')\n ? true\n : 'Must start with postgres:// or postgresql://',\n },\n ]);\n\n // Merge into config.json\n const cfgPath = join(cfgDir, 'config.json');\n let cfg: ConfigShape = {};\n try {\n if (existsSync(cfgPath)) cfg = JSON.parse(readFileSync(cfgPath, 'utf-8'));\n } catch {}\n cfg.database = { ...(cfg.database || {}), mode: 'hosted', url: databaseUrl };\n writeFileSync(cfgPath, JSON.stringify(cfg, null, 2));\n console.log(chalk.green('\u2713 Hosted database configured in ~/.stackmemory/config.json'));\n\n // Save env helper\n try {\n const envFile = join(cfgDir, 'railway.env');\n writeFileSync(envFile, `# StackMemory hosted DB\\nDATABASE_URL=${databaseUrl}\\n`);\n console.log(chalk.green('\u2713 Saved DATABASE_URL to ~/.stackmemory/railway.env'));\n } catch {}\n\n console.log(chalk.gray('Tip: export DATABASE_URL before starting the server.'));\n });\n}\n\n"],
5
+ "mappings": "AACA,OAAO,cAAc;AACrB,OAAO,WAAW;AAClB,SAAS,eAAe;AACxB,SAAS,YAAY;AACrB,SAAS,YAAY,WAAW,eAAe,oBAAoB;AAU5D,SAAS,qBAAqB,SAAwB;AAC3D,UACG,QAAQ,OAAO,EACf,YAAY,0DAA0D,EACtE,OAAO,UAAU,gDAAgD,EACjE,OAAO,OAAO,YAAY;AACzB,UAAM,SAAS,KAAK,QAAQ,GAAG,cAAc;AAC7C,QAAI,CAAC,WAAW,MAAM,EAAG,WAAU,QAAQ,EAAE,WAAW,KAAK,CAAC;AAE9D,QAAI,QAAQ,MAAM;AAChB,UAAI;AACF,cAAM,YAAY;AAClB,cAAM,MAAM,MAAM,OAAO,MAAM;AAC/B,cAAM,IAAI,QAAQ,SAAS;AAAA,MAC7B,SAAS,GAAG;AACV,gBAAQ,IAAI,MAAM,OAAO,uCAAuC,CAAC;AAAA,MACnE;AAAA,IACF;AAEA,UAAM,EAAE,YAAY,IAAI,MAAM,SAAS,OAAO;AAAA,MAC5C;AAAA,QACE,MAAM;AAAA,QACN,MAAM;AAAA,QACN,SAAS;AAAA,QACT,UAAU,CAAC,UACT,MAAM,WAAW,aAAa,KAAK,MAAM,WAAW,eAAe,IAC/D,OACA;AAAA,MACR;AAAA,IACF,CAAC;AAGD,UAAM,UAAU,KAAK,QAAQ,aAAa;AAC1C,QAAI,MAAmB,CAAC;AACxB,QAAI;AACF,UAAI,WAAW,OAAO,EAAG,OAAM,KAAK,MAAM,aAAa,SAAS,OAAO,CAAC;AAAA,IAC1E,QAAQ;AAAA,IAAC;AACT,QAAI,WAAW,EAAE,GAAI,IAAI,YAAY,CAAC,GAAI,MAAM,UAAU,KAAK,YAAY;AAC3E,kBAAc,SAAS,KAAK,UAAU,KAAK,MAAM,CAAC,CAAC;AACnD,YAAQ,IAAI,MAAM,MAAM,iEAA4D,CAAC;AAGrF,QAAI;AACF,YAAM,UAAU,KAAK,QAAQ,aAAa;AAC1C,oBAAc,SAAS;AAAA,eAAyC,WAAW;AAAA,CAAI;AAC/E,cAAQ,IAAI,MAAM,MAAM,yDAAoD,CAAC;AAAA,IAC/E,QAAQ;AAAA,IAAC;AAET,YAAQ,IAAI,MAAM,KAAK,sDAAsD,CAAC;AAAA,EAChF,CAAC;AACL;",
6
+ "names": []
7
+ }
@@ -0,0 +1,178 @@
1
+ #!/usr/bin/env node
2
+ import { Command } from "commander";
3
+ import { Pool } from "pg";
4
+ import Database from "better-sqlite3";
5
+ const MIGRATIONS = [
6
+ {
7
+ version: 1,
8
+ description: "base schema",
9
+ statements: [
10
+ // contexts
11
+ `CREATE TABLE IF NOT EXISTS contexts (id ${isPg() ? "BIGSERIAL" : "INTEGER PRIMARY KEY AUTOINCREMENT"} PRIMARY KEY, project_id TEXT NOT NULL, content TEXT NOT NULL, type TEXT DEFAULT 'general', ${isPg() ? "metadata JSONB DEFAULT '{}'::jsonb" : "metadata TEXT DEFAULT '{}'"}, created_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"}, updated_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"})`,
12
+ // api_keys
13
+ `CREATE TABLE IF NOT EXISTS api_keys (id ${isPg() ? "BIGSERIAL" : "INTEGER PRIMARY KEY AUTOINCREMENT"} PRIMARY KEY, key_hash TEXT UNIQUE NOT NULL, user_id TEXT NOT NULL, name TEXT, created_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"}, ${isPg() ? "last_used TIMESTAMPTZ" : "last_used DATETIME"}, revoked ${isPg() ? "BOOLEAN" : "BOOLEAN"} DEFAULT ${isPg() ? "false" : "0"})`,
14
+ // users with role
15
+ `CREATE TABLE IF NOT EXISTS users (id TEXT PRIMARY KEY, email TEXT, name TEXT, tier TEXT DEFAULT 'free', role TEXT DEFAULT 'user', created_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"}, updated_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"})`,
16
+ // projects
17
+ `CREATE TABLE IF NOT EXISTS projects (id TEXT PRIMARY KEY, name TEXT, is_public ${isPg() ? "BOOLEAN" : "BOOLEAN"} DEFAULT ${isPg() ? "false" : "0"}, created_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"}, updated_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"})`,
18
+ // project members
19
+ `CREATE TABLE IF NOT EXISTS project_members (project_id TEXT NOT NULL, user_id TEXT NOT NULL, role TEXT NOT NULL ${isPg() ? "" : "CHECK (role IN ('admin','owner','editor','viewer'))"}, created_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"}, PRIMARY KEY (project_id, user_id))`,
20
+ // indexes
21
+ `CREATE INDEX IF NOT EXISTS idx_contexts_project ON contexts(project_id)`,
22
+ `CREATE INDEX IF NOT EXISTS idx_api_keys_hash ON api_keys(key_hash)`,
23
+ `CREATE INDEX IF NOT EXISTS idx_users_email ON users(email)`,
24
+ `CREATE INDEX IF NOT EXISTS idx_project_members_user ON project_members(user_id)`
25
+ ]
26
+ },
27
+ {
28
+ version: 2,
29
+ description: "admin sessions",
30
+ statements: [
31
+ `CREATE TABLE IF NOT EXISTS admin_sessions (id TEXT PRIMARY KEY, user_id TEXT NOT NULL, created_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"}, expires_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} NOT NULL, user_agent TEXT, ip TEXT)`,
32
+ `CREATE INDEX IF NOT EXISTS idx_admin_sessions_user ON admin_sessions(user_id)`
33
+ ]
34
+ },
35
+ {
36
+ version: 3,
37
+ description: "role enums & checks",
38
+ statements: [
39
+ // PG enum upgrades; for SQLite CHECK already present
40
+ `CREATE TYPE user_role AS ENUM ('admin','user')`,
41
+ `CREATE TYPE member_role AS ENUM ('admin','owner','editor','viewer')`,
42
+ `ALTER TABLE users ALTER COLUMN role TYPE user_role USING role::user_role`,
43
+ `ALTER TABLE project_members ALTER COLUMN role TYPE member_role USING role::member_role`,
44
+ `ALTER TABLE project_members ADD CONSTRAINT project_members_role_check CHECK (role IN ('admin','owner','editor','viewer'))`,
45
+ `ALTER TABLE users ADD CONSTRAINT users_role_check CHECK (role IN ('admin','user'))`
46
+ ]
47
+ }
48
+ ];
49
+ function isPg() {
50
+ const url = process.env.DATABASE_URL || "";
51
+ return url.startsWith("postgres://") || url.startsWith("postgresql://");
52
+ }
53
+ async function connect() {
54
+ if (isPg()) {
55
+ const pool = new Pool({ connectionString: process.env.DATABASE_URL });
56
+ await pool.query(`CREATE TABLE IF NOT EXISTS railway_schema_version (version INTEGER PRIMARY KEY, applied_at TIMESTAMPTZ DEFAULT NOW(), description TEXT)`);
57
+ return { kind: "pg", pg: pool };
58
+ } else {
59
+ const path = process.env.DATABASE_URL || ".stackmemory/railway.db";
60
+ const db = new Database(path);
61
+ db.exec(`CREATE TABLE IF NOT EXISTS railway_schema_version (version INTEGER PRIMARY KEY, applied_at DATETIME DEFAULT CURRENT_TIMESTAMP, description TEXT)`);
62
+ return { kind: "sqlite", sqlite: db };
63
+ }
64
+ }
65
+ async function getCurrentVersion(m) {
66
+ if (m.kind === "pg") {
67
+ const r = await m.pg.query("SELECT COALESCE(MAX(version), 0) AS v FROM railway_schema_version");
68
+ return Number(r.rows[0]?.v || 0);
69
+ }
70
+ const row = m.sqlite.prepare("SELECT COALESCE(MAX(version), 0) AS v FROM railway_schema_version").get();
71
+ return Number(row?.v || 0);
72
+ }
73
+ async function listApplied(m) {
74
+ if (m.kind === "pg") {
75
+ const r = await m.pg.query("SELECT version, description, applied_at FROM railway_schema_version ORDER BY version ASC");
76
+ return r.rows.map((row) => ({ version: Number(row.version), description: row.description }));
77
+ }
78
+ const rows = m.sqlite.prepare("SELECT version, description, applied_at FROM railway_schema_version ORDER BY version ASC").all();
79
+ return rows.map((row) => ({ version: Number(row.version), description: row.description }));
80
+ }
81
+ async function applyTo(m, target) {
82
+ const current = await getCurrentVersion(m);
83
+ const pending = MIGRATIONS.filter((mig) => mig.version > current && mig.version <= target);
84
+ for (const mig of pending) {
85
+ if (m.kind === "pg") {
86
+ for (const s of mig.statements) {
87
+ try {
88
+ await m.pg.query(s);
89
+ } catch {
90
+ }
91
+ }
92
+ await m.pg.query("INSERT INTO railway_schema_version (version, description) VALUES ($1, $2) ON CONFLICT (version) DO NOTHING", [mig.version, mig.description]);
93
+ } else {
94
+ m.sqlite.exec("BEGIN");
95
+ try {
96
+ for (const s of mig.statements) {
97
+ try {
98
+ m.sqlite.exec(s);
99
+ } catch {
100
+ }
101
+ }
102
+ m.sqlite.prepare("INSERT OR IGNORE INTO railway_schema_version (version, description) VALUES (?, ?)").run(mig.version, mig.description);
103
+ m.sqlite.exec("COMMIT");
104
+ } catch {
105
+ m.sqlite.exec("ROLLBACK");
106
+ throw new Error(`Migration ${mig.version} failed`);
107
+ }
108
+ }
109
+ console.log(`Applied migration v${mig.version}: ${mig.description}`);
110
+ }
111
+ }
112
+ async function rollbackTo(m, target) {
113
+ const current = await getCurrentVersion(m);
114
+ if (target >= current) {
115
+ console.log("Nothing to rollback");
116
+ return;
117
+ }
118
+ if (m.kind === "pg") {
119
+ await m.pg.query("DELETE FROM railway_schema_version WHERE version > $1", [target]);
120
+ } else {
121
+ m.sqlite.prepare("DELETE FROM railway_schema_version WHERE version > ?").run(target);
122
+ }
123
+ console.log(`Rolled back schema version pointer from ${current} to ${target}`);
124
+ }
125
+ async function main() {
126
+ const program = new Command();
127
+ program.name("railway-migrate").description("Manage Railway server schema migrations").option("-d, --database <url>", "DATABASE_URL override");
128
+ program.command("list").description("List applied migrations").action(async () => {
129
+ if (program.opts().database) process.env.DATABASE_URL = program.opts().database;
130
+ const m = await connect();
131
+ const applied = await listApplied(m);
132
+ const current = await getCurrentVersion(m);
133
+ console.log("Current version:", current);
134
+ if (applied.length === 0) console.log("(no migrations applied)");
135
+ applied.forEach((a) => console.log(`v${a.version} - ${a.description}`));
136
+ process.exit(0);
137
+ });
138
+ program.command("status").description("Show current version and pending migrations").action(async () => {
139
+ if (program.opts().database) process.env.DATABASE_URL = program.opts().database;
140
+ const m = await connect();
141
+ const current = await getCurrentVersion(m);
142
+ const latest = Math.max(...MIGRATIONS.map((m2) => m2.version));
143
+ const pending = MIGRATIONS.filter((mig) => mig.version > current);
144
+ console.log("Current version:", current);
145
+ console.log("Latest available:", latest);
146
+ if (pending.length === 0) console.log("No pending migrations.");
147
+ else {
148
+ console.log("Pending:");
149
+ pending.forEach((p) => console.log(`- v${p.version} ${p.description}`));
150
+ }
151
+ process.exit(0);
152
+ });
153
+ program.command("apply").description("Apply migrations up to a target").option("--to <version|latest>", 'Target version (number or "latest")', "latest").action(async (cmd) => {
154
+ if (program.opts().database) process.env.DATABASE_URL = program.opts().database;
155
+ const m = await connect();
156
+ const latest = Math.max(...MIGRATIONS.map((m2) => m2.version));
157
+ const target = cmd.to === "latest" ? latest : parseInt(cmd.to, 10);
158
+ if (!Number.isFinite(target)) throw new Error("Invalid target");
159
+ await applyTo(m, target);
160
+ console.log("Done.");
161
+ process.exit(0);
162
+ });
163
+ program.command("rollback").description("Rollback schema version pointer (non-destructive)").option("--to <version>", "Target version number", "0").action(async (cmd) => {
164
+ if (program.opts().database) process.env.DATABASE_URL = program.opts().database;
165
+ const m = await connect();
166
+ const target = parseInt(cmd.to, 10);
167
+ if (!Number.isFinite(target)) throw new Error("Invalid target");
168
+ await rollbackTo(m, target);
169
+ console.log("Done.");
170
+ process.exit(0);
171
+ });
172
+ await program.parseAsync(process.argv);
173
+ }
174
+ main().catch((e) => {
175
+ console.error(e);
176
+ process.exit(1);
177
+ });
178
+ //# sourceMappingURL=migrate.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../src/cli/commands/migrate.ts"],
4
+ "sourcesContent": ["#!/usr/bin/env node\n/**\n * Railway schema migration CLI\n * Usage examples:\n * - DATABASE_URL=... tsx src/cli/commands/migrate.ts list\n * - DATABASE_URL=... tsx src/cli/commands/migrate.ts status\n * - DATABASE_URL=... tsx src/cli/commands/migrate.ts apply --to latest\n * - DATABASE_URL=... tsx src/cli/commands/migrate.ts rollback --to 2\n */\n\nimport { Command } from 'commander';\nimport { Pool } from 'pg';\nimport Database from 'better-sqlite3';\n\ntype DbKind = 'pg' | 'sqlite';\n\ninterface Migrator {\n kind: DbKind;\n pg?: Pool;\n sqlite?: Database.Database;\n}\n\nconst MIGRATIONS: Array<{ version: number; description: string; statements: string[] }> = [\n {\n version: 1,\n description: 'base schema',\n statements: [\n // contexts\n `CREATE TABLE IF NOT EXISTS contexts (id ${isPg() ? 'BIGSERIAL' : 'INTEGER PRIMARY KEY AUTOINCREMENT'} PRIMARY KEY, project_id TEXT NOT NULL, content TEXT NOT NULL, type TEXT DEFAULT 'general', ${isPg() ? \"metadata JSONB DEFAULT '{}'::jsonb\" : \"metadata TEXT DEFAULT '{}'\"}, created_at ${isPg() ? 'TIMESTAMPTZ' : 'DATETIME'} DEFAULT ${isPg() ? 'NOW()' : 'CURRENT_TIMESTAMP'}, updated_at ${isPg() ? 'TIMESTAMPTZ' : 'DATETIME'} DEFAULT ${isPg() ? 'NOW()' : 'CURRENT_TIMESTAMP'})`,\n // api_keys\n `CREATE TABLE IF NOT EXISTS api_keys (id ${isPg() ? 'BIGSERIAL' : 'INTEGER PRIMARY KEY AUTOINCREMENT'} PRIMARY KEY, key_hash TEXT UNIQUE NOT NULL, user_id TEXT NOT NULL, name TEXT, created_at ${isPg() ? 'TIMESTAMPTZ' : 'DATETIME'} DEFAULT ${isPg() ? 'NOW()' : 'CURRENT_TIMESTAMP'}, ${isPg() ? 'last_used TIMESTAMPTZ' : 'last_used DATETIME'}, revoked ${isPg() ? 'BOOLEAN' : 'BOOLEAN'} DEFAULT ${isPg() ? 'false' : '0'})`,\n // users with role\n `CREATE TABLE IF NOT EXISTS users (id TEXT PRIMARY KEY, email TEXT, name TEXT, tier TEXT DEFAULT 'free', role TEXT DEFAULT 'user', created_at ${isPg() ? 'TIMESTAMPTZ' : 'DATETIME'} DEFAULT ${isPg() ? 'NOW()' : 'CURRENT_TIMESTAMP'}, updated_at ${isPg() ? 'TIMESTAMPTZ' : 'DATETIME'} DEFAULT ${isPg() ? 'NOW()' : 'CURRENT_TIMESTAMP'})`,\n // projects\n `CREATE TABLE IF NOT EXISTS projects (id TEXT PRIMARY KEY, name TEXT, is_public ${isPg() ? 'BOOLEAN' : 'BOOLEAN'} DEFAULT ${isPg() ? 'false' : '0'}, created_at ${isPg() ? 'TIMESTAMPTZ' : 'DATETIME'} DEFAULT ${isPg() ? 'NOW()' : 'CURRENT_TIMESTAMP'}, updated_at ${isPg() ? 'TIMESTAMPTZ' : 'DATETIME'} DEFAULT ${isPg() ? 'NOW()' : 'CURRENT_TIMESTAMP'})`,\n // project members\n `CREATE TABLE IF NOT EXISTS project_members (project_id TEXT NOT NULL, user_id TEXT NOT NULL, role TEXT NOT NULL ${isPg() ? '' : \"CHECK (role IN ('admin','owner','editor','viewer'))\"}, created_at ${isPg() ? 'TIMESTAMPTZ' : 'DATETIME'} DEFAULT ${isPg() ? 'NOW()' : 'CURRENT_TIMESTAMP'}, PRIMARY KEY (project_id, user_id))`,\n // indexes\n `CREATE INDEX IF NOT EXISTS idx_contexts_project ON contexts(project_id)`,\n `CREATE INDEX IF NOT EXISTS idx_api_keys_hash ON api_keys(key_hash)`,\n `CREATE INDEX IF NOT EXISTS idx_users_email ON users(email)`,\n `CREATE INDEX IF NOT EXISTS idx_project_members_user ON project_members(user_id)`\n ],\n },\n {\n version: 2,\n description: 'admin sessions',\n statements: [\n `CREATE TABLE IF NOT EXISTS admin_sessions (id TEXT PRIMARY KEY, user_id TEXT NOT NULL, created_at ${isPg() ? 'TIMESTAMPTZ' : 'DATETIME'} DEFAULT ${isPg() ? 'NOW()' : 'CURRENT_TIMESTAMP'}, expires_at ${isPg() ? 'TIMESTAMPTZ' : 'DATETIME'} NOT NULL, user_agent TEXT, ip TEXT)`,\n `CREATE INDEX IF NOT EXISTS idx_admin_sessions_user ON admin_sessions(user_id)`\n ],\n },\n {\n version: 3,\n description: 'role enums & checks',\n statements: [\n // PG enum upgrades; for SQLite CHECK already present\n `CREATE TYPE user_role AS ENUM ('admin','user')`,\n `CREATE TYPE member_role AS ENUM ('admin','owner','editor','viewer')`,\n `ALTER TABLE users ALTER COLUMN role TYPE user_role USING role::user_role`,\n `ALTER TABLE project_members ALTER COLUMN role TYPE member_role USING role::member_role`,\n `ALTER TABLE project_members ADD CONSTRAINT project_members_role_check CHECK (role IN ('admin','owner','editor','viewer'))`,\n `ALTER TABLE users ADD CONSTRAINT users_role_check CHECK (role IN ('admin','user'))`\n ],\n },\n];\n\nfunction isPg(): boolean {\n const url = process.env.DATABASE_URL || '';\n return url.startsWith('postgres://') || url.startsWith('postgresql://');\n}\n\nasync function connect(): Promise<Migrator> {\n if (isPg()) {\n const pool = new Pool({ connectionString: process.env.DATABASE_URL });\n // ensure version table\n await pool.query(`CREATE TABLE IF NOT EXISTS railway_schema_version (version INTEGER PRIMARY KEY, applied_at TIMESTAMPTZ DEFAULT NOW(), description TEXT)`);\n return { kind: 'pg', pg: pool };\n } else {\n const path = process.env.DATABASE_URL || '.stackmemory/railway.db';\n const db = new Database(path);\n db.exec(`CREATE TABLE IF NOT EXISTS railway_schema_version (version INTEGER PRIMARY KEY, applied_at DATETIME DEFAULT CURRENT_TIMESTAMP, description TEXT)`);\n return { kind: 'sqlite', sqlite: db };\n }\n}\n\nasync function getCurrentVersion(m: Migrator): Promise<number> {\n if (m.kind === 'pg') {\n const r = await m.pg!.query('SELECT COALESCE(MAX(version), 0) AS v FROM railway_schema_version');\n return Number(r.rows[0]?.v || 0);\n }\n const row = m.sqlite!.prepare('SELECT COALESCE(MAX(version), 0) AS v FROM railway_schema_version').get() as any;\n return Number(row?.v || 0);\n}\n\nasync function listApplied(m: Migrator): Promise<Array<{ version: number; description: string }>> {\n if (m.kind === 'pg') {\n const r = await m.pg!.query('SELECT version, description, applied_at FROM railway_schema_version ORDER BY version ASC');\n return r.rows.map((row) => ({ version: Number(row.version), description: row.description }));\n }\n const rows = m.sqlite!.prepare('SELECT version, description, applied_at FROM railway_schema_version ORDER BY version ASC').all() as any[];\n return rows.map((row) => ({ version: Number(row.version), description: row.description }));\n}\n\nasync function applyTo(m: Migrator, target: number): Promise<void> {\n const current = await getCurrentVersion(m);\n const pending = MIGRATIONS.filter((mig) => mig.version > current && mig.version <= target);\n for (const mig of pending) {\n if (m.kind === 'pg') {\n for (const s of mig.statements) {\n try { await m.pg!.query(s); } catch {}\n }\n await m.pg!.query('INSERT INTO railway_schema_version (version, description) VALUES ($1, $2) ON CONFLICT (version) DO NOTHING', [mig.version, mig.description]);\n } else {\n m.sqlite!.exec('BEGIN');\n try {\n for (const s of mig.statements) {\n try { m.sqlite!.exec(s); } catch {}\n }\n m.sqlite!.prepare('INSERT OR IGNORE INTO railway_schema_version (version, description) VALUES (?, ?)').run(mig.version, mig.description);\n m.sqlite!.exec('COMMIT');\n } catch {\n m.sqlite!.exec('ROLLBACK');\n throw new Error(`Migration ${mig.version} failed`);\n }\n }\n console.log(`Applied migration v${mig.version}: ${mig.description}`);\n }\n}\n\nasync function rollbackTo(m: Migrator, target: number): Promise<void> {\n const current = await getCurrentVersion(m);\n if (target >= current) {\n console.log('Nothing to rollback');\n return;\n }\n // Soft rollback: move version pointer back; does not drop objects\n if (m.kind === 'pg') {\n await m.pg!.query('DELETE FROM railway_schema_version WHERE version > $1', [target]);\n } else {\n m.sqlite!.prepare('DELETE FROM railway_schema_version WHERE version > ?').run(target);\n }\n console.log(`Rolled back schema version pointer from ${current} to ${target}`);\n}\n\nasync function main() {\n const program = new Command();\n program\n .name('railway-migrate')\n .description('Manage Railway server schema migrations')\n .option('-d, --database <url>', 'DATABASE_URL override');\n\n program\n .command('list')\n .description('List applied migrations')\n .action(async () => {\n if (program.opts().database) process.env.DATABASE_URL = program.opts().database;\n const m = await connect();\n const applied = await listApplied(m);\n const current = await getCurrentVersion(m);\n console.log('Current version:', current);\n if (applied.length === 0) console.log('(no migrations applied)');\n applied.forEach((a) => console.log(`v${a.version} - ${a.description}`));\n process.exit(0);\n });\n\n program\n .command('status')\n .description('Show current version and pending migrations')\n .action(async () => {\n if (program.opts().database) process.env.DATABASE_URL = program.opts().database;\n const m = await connect();\n const current = await getCurrentVersion(m);\n const latest = Math.max(...MIGRATIONS.map((m) => m.version));\n const pending = MIGRATIONS.filter((mig) => mig.version > current);\n console.log('Current version:', current);\n console.log('Latest available:', latest);\n if (pending.length === 0) console.log('No pending migrations.');\n else {\n console.log('Pending:');\n pending.forEach((p) => console.log(`- v${p.version} ${p.description}`));\n }\n process.exit(0);\n });\n\n program\n .command('apply')\n .description('Apply migrations up to a target')\n .option('--to <version|latest>', 'Target version (number or \"latest\")', 'latest')\n .action(async (cmd) => {\n if (program.opts().database) process.env.DATABASE_URL = program.opts().database;\n const m = await connect();\n const latest = Math.max(...MIGRATIONS.map((m) => m.version));\n const target = cmd.to === 'latest' ? latest : parseInt(cmd.to, 10);\n if (!Number.isFinite(target)) throw new Error('Invalid target');\n await applyTo(m, target);\n console.log('Done.');\n process.exit(0);\n });\n\n program\n .command('rollback')\n .description('Rollback schema version pointer (non-destructive)')\n .option('--to <version>', 'Target version number', '0')\n .action(async (cmd) => {\n if (program.opts().database) process.env.DATABASE_URL = program.opts().database;\n const m = await connect();\n const target = parseInt(cmd.to, 10);\n if (!Number.isFinite(target)) throw new Error('Invalid target');\n await rollbackTo(m, target);\n console.log('Done.');\n process.exit(0);\n });\n\n await program.parseAsync(process.argv);\n}\n\nmain().catch((e) => {\n console.error(e);\n process.exit(1);\n});\n\n"],
5
+ "mappings": ";AAUA,SAAS,eAAe;AACxB,SAAS,YAAY;AACrB,OAAO,cAAc;AAUrB,MAAM,aAAoF;AAAA,EACxF;AAAA,IACE,SAAS;AAAA,IACT,aAAa;AAAA,IACb,YAAY;AAAA;AAAA,MAEV,2CAA2C,KAAK,IAAI,cAAc,mCAAmC,+FAA+F,KAAK,IAAI,uCAAuC,4BAA4B,gBAAgB,KAAK,IAAI,gBAAgB,UAAU,YAAY,KAAK,IAAI,UAAU,mBAAmB,gBAAgB,KAAK,IAAI,gBAAgB,UAAU,YAAY,KAAK,IAAI,UAAU,mBAAmB;AAAA;AAAA,MAE1d,2CAA2C,KAAK,IAAI,cAAc,mCAAmC,6FAA6F,KAAK,IAAI,gBAAgB,UAAU,YAAY,KAAK,IAAI,UAAU,mBAAmB,KAAK,KAAK,IAAI,0BAA0B,oBAAoB,aAAa,KAAK,IAAI,YAAY,SAAS,YAAY,KAAK,IAAI,UAAU,GAAG;AAAA;AAAA,MAEha,gJAAgJ,KAAK,IAAI,gBAAgB,UAAU,YAAY,KAAK,IAAI,UAAU,mBAAmB,gBAAgB,KAAK,IAAI,gBAAgB,UAAU,YAAY,KAAK,IAAI,UAAU,mBAAmB;AAAA;AAAA,MAE1U,kFAAkF,KAAK,IAAI,YAAY,SAAS,YAAY,KAAK,IAAI,UAAU,GAAG,gBAAgB,KAAK,IAAI,gBAAgB,UAAU,YAAY,KAAK,IAAI,UAAU,mBAAmB,gBAAgB,KAAK,IAAI,gBAAgB,UAAU,YAAY,KAAK,IAAI,UAAU,mBAAmB;AAAA;AAAA,MAE5V,mHAAmH,KAAK,IAAI,KAAK,qDAAqD,gBAAgB,KAAK,IAAI,gBAAgB,UAAU,YAAY,KAAK,IAAI,UAAU,mBAAmB;AAAA;AAAA,MAE3R;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAAA,EACF;AAAA,EACA;AAAA,IACE,SAAS;AAAA,IACT,aAAa;AAAA,IACb,YAAY;AAAA,MACV,qGAAqG,KAAK,IAAI,gBAAgB,UAAU,YAAY,KAAK,IAAI,UAAU,mBAAmB,gBAAgB,KAAK,IAAI,gBAAgB,UAAU;AAAA,MAC7O;AAAA,IACF;AAAA,EACF;AAAA,EACA;AAAA,IACE,SAAS;AAAA,IACT,aAAa;AAAA,IACb,YAAY;AAAA;AAAA,MAEV;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAAA,EACF;AACF;AAEA,SAAS,OAAgB;AACvB,QAAM,MAAM,QAAQ,IAAI,gBAAgB;AACxC,SAAO,IAAI,WAAW,aAAa,KAAK,IAAI,WAAW,eAAe;AACxE;AAEA,eAAe,UAA6B;AAC1C,MAAI,KAAK,GAAG;AACV,UAAM,OAAO,IAAI,KAAK,EAAE,kBAAkB,QAAQ,IAAI,aAAa,CAAC;AAEpE,UAAM,KAAK,MAAM,yIAAyI;AAC1J,WAAO,EAAE,MAAM,MAAM,IAAI,KAAK;AAAA,EAChC,OAAO;AACL,UAAM,OAAO,QAAQ,IAAI,gBAAgB;AACzC,UAAM,KAAK,IAAI,SAAS,IAAI;AAC5B,OAAG,KAAK,kJAAkJ;AAC1J,WAAO,EAAE,MAAM,UAAU,QAAQ,GAAG;AAAA,EACtC;AACF;AAEA,eAAe,kBAAkB,GAA8B;AAC7D,MAAI,EAAE,SAAS,MAAM;AACnB,UAAM,IAAI,MAAM,EAAE,GAAI,MAAM,mEAAmE;AAC/F,WAAO,OAAO,EAAE,KAAK,CAAC,GAAG,KAAK,CAAC;AAAA,EACjC;AACA,QAAM,MAAM,EAAE,OAAQ,QAAQ,mEAAmE,EAAE,IAAI;AACvG,SAAO,OAAO,KAAK,KAAK,CAAC;AAC3B;AAEA,eAAe,YAAY,GAAuE;AAChG,MAAI,EAAE,SAAS,MAAM;AACnB,UAAM,IAAI,MAAM,EAAE,GAAI,MAAM,0FAA0F;AACtH,WAAO,EAAE,KAAK,IAAI,CAAC,SAAS,EAAE,SAAS,OAAO,IAAI,OAAO,GAAG,aAAa,IAAI,YAAY,EAAE;AAAA,EAC7F;AACA,QAAM,OAAO,EAAE,OAAQ,QAAQ,0FAA0F,EAAE,IAAI;AAC/H,SAAO,KAAK,IAAI,CAAC,SAAS,EAAE,SAAS,OAAO,IAAI,OAAO,GAAG,aAAa,IAAI,YAAY,EAAE;AAC3F;AAEA,eAAe,QAAQ,GAAa,QAA+B;AACjE,QAAM,UAAU,MAAM,kBAAkB,CAAC;AACzC,QAAM,UAAU,WAAW,OAAO,CAAC,QAAQ,IAAI,UAAU,WAAW,IAAI,WAAW,MAAM;AACzF,aAAW,OAAO,SAAS;AACzB,QAAI,EAAE,SAAS,MAAM;AACnB,iBAAW,KAAK,IAAI,YAAY;AAC9B,YAAI;AAAE,gBAAM,EAAE,GAAI,MAAM,CAAC;AAAA,QAAG,QAAQ;AAAA,QAAC;AAAA,MACvC;AACA,YAAM,EAAE,GAAI,MAAM,8GAA8G,CAAC,IAAI,SAAS,IAAI,WAAW,CAAC;AAAA,IAChK,OAAO;AACL,QAAE,OAAQ,KAAK,OAAO;AACtB,UAAI;AACF,mBAAW,KAAK,IAAI,YAAY;AAC9B,cAAI;AAAE,cAAE,OAAQ,KAAK,CAAC;AAAA,UAAG,QAAQ;AAAA,UAAC;AAAA,QACpC;AACA,UAAE,OAAQ,QAAQ,mFAAmF,EAAE,IAAI,IAAI,SAAS,IAAI,WAAW;AACvI,UAAE,OAAQ,KAAK,QAAQ;AAAA,MACzB,QAAQ;AACN,UAAE,OAAQ,KAAK,UAAU;AACzB,cAAM,IAAI,MAAM,aAAa,IAAI,OAAO,SAAS;AAAA,MACnD;AAAA,IACF;AACA,YAAQ,IAAI,sBAAsB,IAAI,OAAO,KAAK,IAAI,WAAW,EAAE;AAAA,EACrE;AACF;AAEA,eAAe,WAAW,GAAa,QAA+B;AACpE,QAAM,UAAU,MAAM,kBAAkB,CAAC;AACzC,MAAI,UAAU,SAAS;AACrB,YAAQ,IAAI,qBAAqB;AACjC;AAAA,EACF;AAEA,MAAI,EAAE,SAAS,MAAM;AACnB,UAAM,EAAE,GAAI,MAAM,yDAAyD,CAAC,MAAM,CAAC;AAAA,EACrF,OAAO;AACL,MAAE,OAAQ,QAAQ,sDAAsD,EAAE,IAAI,MAAM;AAAA,EACtF;AACA,UAAQ,IAAI,2CAA2C,OAAO,OAAO,MAAM,EAAE;AAC/E;AAEA,eAAe,OAAO;AACpB,QAAM,UAAU,IAAI,QAAQ;AAC5B,UACG,KAAK,iBAAiB,EACtB,YAAY,yCAAyC,EACrD,OAAO,wBAAwB,uBAAuB;AAEzD,UACG,QAAQ,MAAM,EACd,YAAY,yBAAyB,EACrC,OAAO,YAAY;AAClB,QAAI,QAAQ,KAAK,EAAE,SAAU,SAAQ,IAAI,eAAe,QAAQ,KAAK,EAAE;AACvE,UAAM,IAAI,MAAM,QAAQ;AACxB,UAAM,UAAU,MAAM,YAAY,CAAC;AACnC,UAAM,UAAU,MAAM,kBAAkB,CAAC;AACzC,YAAQ,IAAI,oBAAoB,OAAO;AACvC,QAAI,QAAQ,WAAW,EAAG,SAAQ,IAAI,yBAAyB;AAC/D,YAAQ,QAAQ,CAAC,MAAM,QAAQ,IAAI,IAAI,EAAE,OAAO,MAAM,EAAE,WAAW,EAAE,CAAC;AACtE,YAAQ,KAAK,CAAC;AAAA,EAChB,CAAC;AAEH,UACG,QAAQ,QAAQ,EAChB,YAAY,6CAA6C,EACzD,OAAO,YAAY;AAClB,QAAI,QAAQ,KAAK,EAAE,SAAU,SAAQ,IAAI,eAAe,QAAQ,KAAK,EAAE;AACvE,UAAM,IAAI,MAAM,QAAQ;AACxB,UAAM,UAAU,MAAM,kBAAkB,CAAC;AACzC,UAAM,SAAS,KAAK,IAAI,GAAG,WAAW,IAAI,CAACA,OAAMA,GAAE,OAAO,CAAC;AAC3D,UAAM,UAAU,WAAW,OAAO,CAAC,QAAQ,IAAI,UAAU,OAAO;AAChE,YAAQ,IAAI,oBAAoB,OAAO;AACvC,YAAQ,IAAI,qBAAqB,MAAM;AACvC,QAAI,QAAQ,WAAW,EAAG,SAAQ,IAAI,wBAAwB;AAAA,SACzD;AACH,cAAQ,IAAI,UAAU;AACtB,cAAQ,QAAQ,CAAC,MAAM,QAAQ,IAAI,MAAM,EAAE,OAAO,IAAI,EAAE,WAAW,EAAE,CAAC;AAAA,IACxE;AACA,YAAQ,KAAK,CAAC;AAAA,EAChB,CAAC;AAEH,UACG,QAAQ,OAAO,EACf,YAAY,iCAAiC,EAC7C,OAAO,yBAAyB,uCAAuC,QAAQ,EAC/E,OAAO,OAAO,QAAQ;AACrB,QAAI,QAAQ,KAAK,EAAE,SAAU,SAAQ,IAAI,eAAe,QAAQ,KAAK,EAAE;AACvE,UAAM,IAAI,MAAM,QAAQ;AACxB,UAAM,SAAS,KAAK,IAAI,GAAG,WAAW,IAAI,CAACA,OAAMA,GAAE,OAAO,CAAC;AAC3D,UAAM,SAAS,IAAI,OAAO,WAAW,SAAS,SAAS,IAAI,IAAI,EAAE;AACjE,QAAI,CAAC,OAAO,SAAS,MAAM,EAAG,OAAM,IAAI,MAAM,gBAAgB;AAC9D,UAAM,QAAQ,GAAG,MAAM;AACvB,YAAQ,IAAI,OAAO;AACnB,YAAQ,KAAK,CAAC;AAAA,EAChB,CAAC;AAEH,UACG,QAAQ,UAAU,EAClB,YAAY,mDAAmD,EAC/D,OAAO,kBAAkB,yBAAyB,GAAG,EACrD,OAAO,OAAO,QAAQ;AACrB,QAAI,QAAQ,KAAK,EAAE,SAAU,SAAQ,IAAI,eAAe,QAAQ,KAAK,EAAE;AACvE,UAAM,IAAI,MAAM,QAAQ;AACxB,UAAM,SAAS,SAAS,IAAI,IAAI,EAAE;AAClC,QAAI,CAAC,OAAO,SAAS,MAAM,EAAG,OAAM,IAAI,MAAM,gBAAgB;AAC9D,UAAM,WAAW,GAAG,MAAM;AAC1B,YAAQ,IAAI,OAAO;AACnB,YAAQ,KAAK,CAAC;AAAA,EAChB,CAAC;AAEH,QAAM,QAAQ,WAAW,QAAQ,IAAI;AACvC;AAEA,KAAK,EAAE,MAAM,CAAC,MAAM;AAClB,UAAQ,MAAM,CAAC;AACf,UAAQ,KAAK,CAAC;AAChB,CAAC;",
6
+ "names": ["m"]
7
+ }
@@ -64,8 +64,51 @@ async function runOnboarding() {
64
64
  scanProjects: false,
65
65
  enableLinear: false,
66
66
  enableAnalytics: true,
67
- defaultContextPath: join(homedir(), ".stackmemory")
67
+ defaultContextPath: join(homedir(), ".stackmemory"),
68
+ storageMode: "local"
68
69
  };
70
+ const { storageMode } = await inquirer.prompt([
71
+ {
72
+ type: "list",
73
+ name: "storageMode",
74
+ message: "Where should StackMemory store data?",
75
+ choices: [
76
+ { name: "Local (free, SQLite in ~/.stackmemory)", value: "local" },
77
+ { name: "Hosted (paid, managed Postgres)", value: "hosted" }
78
+ ],
79
+ default: "local"
80
+ }
81
+ ]);
82
+ config.storageMode = storageMode;
83
+ if (storageMode === "hosted") {
84
+ const { hasAccount } = await inquirer.prompt([
85
+ {
86
+ type: "confirm",
87
+ name: "hasAccount",
88
+ message: "Do you already have a hosted account & connection string?",
89
+ default: false
90
+ }
91
+ ]);
92
+ if (!hasAccount) {
93
+ try {
94
+ const signupUrl = "https://stackmemory.ai/hosted";
95
+ console.log(chalk.gray(`Opening signup page: ${signupUrl}`));
96
+ const mod = await import("open");
97
+ await mod.default(signupUrl);
98
+ } catch (e) {
99
+ console.log(chalk.yellow("Could not open browser automatically. Please sign up and obtain your DATABASE_URL."));
100
+ }
101
+ }
102
+ const { databaseUrl } = await inquirer.prompt([
103
+ {
104
+ type: "password",
105
+ name: "databaseUrl",
106
+ message: "Paste your DATABASE_URL (postgres://...)",
107
+ validate: (input) => input.startsWith("postgres://") || input.startsWith("postgresql://") ? true : "Must start with postgres:// or postgresql://"
108
+ }
109
+ ]);
110
+ config.databaseUrl = databaseUrl;
111
+ }
69
112
  if (setupType === "basic") {
70
113
  const basicAnswers = await inquirer.prompt([
71
114
  {
@@ -299,12 +342,28 @@ async function applyConfiguration(config) {
299
342
  },
300
343
  paths: {
301
344
  default: config.defaultContextPath
345
+ },
346
+ database: {
347
+ mode: config.storageMode,
348
+ ...config.databaseUrl ? { url: config.databaseUrl } : {}
302
349
  }
303
350
  };
304
351
  writeFileSync(
305
352
  join(configPath, "config.json"),
306
353
  JSON.stringify(mainConfig, null, 2)
307
354
  );
355
+ if (config.storageMode === "hosted" && config.databaseUrl) {
356
+ try {
357
+ const envFile = join(configPath, "railway.env");
358
+ writeFileSync(envFile, `# StackMemory hosted DB
359
+ DATABASE_URL=${config.databaseUrl}
360
+ `);
361
+ console.log(chalk.green(" \u2713 Saved hosted DB settings to ~/.stackmemory/railway.env"));
362
+ console.log(chalk.gray(" Tip: export DATABASE_URL from this file in your shell profile."));
363
+ } catch (e) {
364
+ console.log(chalk.yellow(" \u26A0 Could not write hosted DB env file"));
365
+ }
366
+ }
308
367
  const binPath = "/usr/local/bin/claude-sm";
309
368
  const sourcePath = join(configPath, "bin", "stackmemory");
310
369
  try {
@@ -337,6 +396,101 @@ exec stackmemory "$@"
337
396
  chalk.yellow(" \u26A0 Could not create claude-sm symlink (may need sudo)")
338
397
  );
339
398
  }
399
+ const codexBinPath = "/usr/local/bin/codex-sm";
400
+ const codexSourcePath = join(configPath, "bin", "codex-sm");
401
+ try {
402
+ const codexWrapper = `#!/bin/bash
403
+ # Codex CLI wrapper with StackMemory integration
404
+ # Usage: codex-sm [--auto-sync] [--sync-interval=MINUTES] [args...]
405
+
406
+ # Flags
407
+ AUTO_SYNC=false
408
+ SYNC_INTERVAL=5
409
+ for arg in "$@"; do
410
+ case $arg in
411
+ --auto-sync)
412
+ AUTO_SYNC=true
413
+ shift
414
+ ;;
415
+ --sync-interval=*)
416
+ SYNC_INTERVAL="\${arg#*=}"
417
+ shift
418
+ ;;
419
+ esac
420
+ done
421
+
422
+ # Auto-initialize StackMemory if in git repo without it
423
+ if [ -d ".git" ] && [ ! -d ".stackmemory" ]; then
424
+ echo "\u{1F4E6} Initializing StackMemory for this project..."
425
+ stackmemory init --silent 2>/dev/null || true
426
+ fi
427
+
428
+ # Load existing context if available
429
+ if [ -d ".stackmemory" ]; then
430
+ echo "\u{1F9E0} Loading StackMemory context..."
431
+ stackmemory status --brief 2>/dev/null || true
432
+ fi
433
+
434
+ # Start Linear auto-sync in background if requested
435
+ SYNC_PID=""
436
+ if [ "$AUTO_SYNC" = true ] && [ -n "$LINEAR_API_KEY" ]; then
437
+ echo "\u{1F504} Starting Linear auto-sync (${SYNC_INTERVAL}min intervals)..."
438
+ (
439
+ while true; do
440
+ sleep $((SYNC_INTERVAL * 60))
441
+ if [ -d ".stackmemory" ]; then
442
+ stackmemory linear sync --quiet 2>/dev/null || true
443
+ fi
444
+ done
445
+ ) &
446
+ SYNC_PID=$!
447
+ fi
448
+
449
+ cleanup() {
450
+ echo ""
451
+ echo "\u{1F4DD} Saving StackMemory context..."
452
+
453
+ # Kill auto-sync if running
454
+ if [ -n "$SYNC_PID" ] && kill -0 $SYNC_PID 2>/dev/null; then
455
+ echo "\u{1F6D1} Stopping auto-sync..."
456
+ kill $SYNC_PID 2>/dev/null || true
457
+ fi
458
+
459
+ # Save project status and final sync
460
+ if [ -d ".stackmemory" ]; then
461
+ stackmemory status 2>/dev/null
462
+ if [ -n "$LINEAR_API_KEY" ]; then
463
+ echo "\u{1F504} Final Linear sync..."
464
+ stackmemory linear sync 2>/dev/null
465
+ fi
466
+ echo "\u2705 StackMemory context saved"
467
+ fi
468
+ }
469
+
470
+ trap cleanup EXIT INT TERM
471
+
472
+ # Run Codex CLI
473
+ if command -v codex &> /dev/null; then
474
+ codex "$@"
475
+ elif command -v codex-cli &> /dev/null; then
476
+ codex-cli "$@"
477
+ else
478
+ echo "\u274C Codex CLI not found. Please install it first."
479
+ echo " See: https://github.com/openai/codex-cli"
480
+ exit 1
481
+ fi
482
+ `;
483
+ writeFileSync(codexSourcePath, codexWrapper);
484
+ execFileSync("chmod", ["+x", codexSourcePath]);
485
+ if (!existsSync(codexBinPath)) {
486
+ execFileSync("ln", ["-s", codexSourcePath, codexBinPath]);
487
+ console.log(chalk.green(" \u2713 Created codex-sm command"));
488
+ }
489
+ } catch (error) {
490
+ console.log(
491
+ chalk.yellow(" \u26A0 Could not create codex-sm symlink (may need sudo)")
492
+ );
493
+ }
340
494
  }
341
495
  function showNextSteps(config) {
342
496
  console.log(chalk.cyan("\u{1F389} Next Steps:\n"));
@@ -359,8 +513,10 @@ function showNextSteps(config) {
359
513
  }
360
514
  console.log("3. Use with Claude:");
361
515
  console.log(chalk.gray(" claude-sm # Or use stackmemory directly\n"));
516
+ console.log("4. Use with Codex:");
517
+ console.log(chalk.gray(" codex-sm # Codex + StackMemory integration\n"));
362
518
  if (config.enableLinear) {
363
- console.log("4. Sync with Linear:");
519
+ console.log("5. Sync with Linear:");
364
520
  console.log(chalk.gray(" stackmemory linear sync\n"));
365
521
  }
366
522
  console.log("For more help:");
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/cli/commands/onboard.ts"],
4
- "sourcesContent": ["#!/usr/bin/env node\n/**\n * StackMemory Onboarding CLI\n * Interactive setup for new StackMemory installations\n */\n\nimport { Command } from 'commander';\nimport inquirer from 'inquirer';\nimport chalk from 'chalk';\nimport { homedir } from 'os';\nimport { join } from 'path';\nimport { existsSync, mkdirSync, writeFileSync } from 'fs';\nimport { execSync, execFileSync } from 'child_process';\nimport { WorktreeManager } from '../../core/worktree/worktree-manager.js';\nimport { ProjectManager } from '../../core/projects/project-manager.js';\nimport { logger } from '../../core/monitoring/logger.js';\n\ninterface OnboardingConfig {\n setupType: 'basic' | 'advanced';\n enableWorktrees: boolean;\n worktreeIsolation: boolean;\n enableProjects: boolean;\n scanProjects: boolean;\n enableLinear: boolean;\n linearApiKey?: string;\n enableAnalytics: boolean;\n defaultContextPath: string;\n}\n\nexport function registerOnboardingCommand(program: Command): void {\n program\n .command('onboard')\n .alias('setup')\n .description('Interactive setup for StackMemory')\n .option('--reset', 'Reset all configurations and start fresh')\n .action(async (options) => {\n console.log(chalk.cyan('\\n\uD83D\uDE80 Welcome to StackMemory Setup!\\n'));\n\n // Check if already configured\n const configPath = join(homedir(), '.stackmemory');\n if (existsSync(configPath) && !options.reset) {\n const { proceed } = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'proceed',\n message:\n 'StackMemory is already configured. Do you want to reconfigure?',\n default: false,\n },\n ]);\n\n if (!proceed) {\n console.log(chalk.yellow('\\nSetup cancelled.'));\n return;\n }\n }\n\n try {\n const config = await runOnboarding();\n await applyConfiguration(config);\n\n console.log(\n chalk.green('\\n\u2705 StackMemory setup completed successfully!\\n')\n );\n showNextSteps(config);\n } catch (error: unknown) {\n logger.error('Onboarding failed', error as Error);\n console.error(\n chalk.red('\\n\u274C Setup failed:'),\n (error as Error).message\n );\n process.exit(1);\n }\n });\n}\n\nasync function runOnboarding(): Promise<OnboardingConfig> {\n // Basic or Advanced Setup\n const { setupType } = await inquirer.prompt([\n {\n type: 'list',\n name: 'setupType',\n message: 'Choose your setup type:',\n choices: [\n { name: 'Basic (Recommended for most users)', value: 'basic' },\n { name: 'Advanced (Full customization)', value: 'advanced' },\n ],\n default: 'basic',\n },\n ]);\n\n let config: OnboardingConfig = {\n setupType,\n enableWorktrees: false,\n worktreeIsolation: true,\n enableProjects: true,\n scanProjects: false,\n enableLinear: false,\n enableAnalytics: true,\n defaultContextPath: join(homedir(), '.stackmemory'),\n };\n\n if (setupType === 'basic') {\n // Basic setup with sensible defaults\n const basicAnswers = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'enableWorktrees',\n message:\n 'Enable Git worktree support? (Recommended for multi-branch workflows)',\n default: false,\n },\n {\n type: 'confirm',\n name: 'scanProjects',\n message: 'Scan and organize your existing projects?',\n default: true,\n },\n {\n type: 'confirm',\n name: 'enableLinear',\n message: 'Connect to Linear for task management?',\n default: false,\n },\n ]);\n\n config = { ...config, ...basicAnswers };\n\n if (basicAnswers.enableLinear) {\n const { linearApiKey } = await inquirer.prompt([\n {\n type: 'password',\n name: 'linearApiKey',\n message: 'Enter your Linear API key:',\n validate: (input: string) =>\n input.length > 0 || 'API key is required',\n },\n ]);\n config.linearApiKey = linearApiKey;\n }\n } else {\n // Advanced setup with all options\n const advancedAnswers = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'enableWorktrees',\n message: 'Enable Git worktree support?',\n default: false,\n },\n ]);\n\n if (advancedAnswers.enableWorktrees) {\n const worktreeAnswers = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'worktreeIsolation',\n message: 'Isolate contexts between worktrees? (Recommended)',\n default: true,\n },\n {\n type: 'confirm',\n name: 'autoDetect',\n message: 'Auto-detect worktrees when switching directories?',\n default: true,\n },\n {\n type: 'confirm',\n name: 'shareGlobal',\n message: 'Share global context across worktrees?',\n default: false,\n },\n {\n type: 'number',\n name: 'syncInterval',\n message: 'Context sync interval in minutes (0 to disable):',\n default: 15,\n validate: (input: number) => input >= 0 || 'Must be 0 or positive',\n },\n ]);\n\n config = { ...config, ...advancedAnswers, ...worktreeAnswers };\n }\n\n const projectAnswers = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'enableProjects',\n message: 'Enable automatic project management?',\n default: true,\n },\n ]);\n\n if (projectAnswers.enableProjects) {\n const projectDetailAnswers: any = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'scanProjects',\n message: 'Scan for existing projects now?',\n default: true,\n },\n {\n type: 'checkbox',\n name: 'scanPaths',\n message: 'Select directories to scan:',\n choices: [\n { name: '~/Dev', value: join(homedir(), 'Dev'), checked: true },\n { name: '~/dev', value: join(homedir(), 'dev'), checked: true },\n {\n name: '~/Projects',\n value: join(homedir(), 'Projects'),\n checked: true,\n },\n {\n name: '~/projects',\n value: join(homedir(), 'projects'),\n checked: true,\n },\n { name: '~/Work', value: join(homedir(), 'Work'), checked: false },\n { name: '~/code', value: join(homedir(), 'code'), checked: true },\n {\n name: '~/Documents/GitHub',\n value: join(homedir(), 'Documents/GitHub'),\n checked: false,\n },\n ],\n when: (): boolean => projectDetailAnswers.scanProjects,\n },\n ]);\n\n config = { ...config, ...projectAnswers, ...projectDetailAnswers };\n }\n\n const integrationAnswers = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'enableLinear',\n message: 'Enable Linear integration?',\n default: false,\n },\n {\n type: 'password',\n name: 'linearApiKey',\n message: 'Linear API key:',\n when: (answers: any) => answers.enableLinear,\n validate: (input: string) => input.length > 0 || 'API key is required',\n },\n {\n type: 'confirm',\n name: 'enableAnalytics',\n message: 'Enable usage analytics? (Local only)',\n default: true,\n },\n {\n type: 'input',\n name: 'defaultContextPath',\n message: 'Default context storage path:',\n default: join(homedir(), '.stackmemory'),\n validate: (input: string) => input.length > 0 || 'Path is required',\n },\n ]);\n\n config = { ...config, ...integrationAnswers };\n }\n\n return config;\n}\n\nasync function applyConfiguration(config: OnboardingConfig): Promise<void> {\n const configPath = join(homedir(), '.stackmemory');\n\n // Create base directory structure\n console.log(chalk.gray('\\nCreating directory structure...'));\n const dirs = [\n configPath,\n join(configPath, 'contexts'),\n join(configPath, 'projects'),\n join(configPath, 'worktrees'),\n join(configPath, 'bin'),\n join(configPath, 'logs'),\n join(configPath, 'analytics'),\n ];\n\n for (const dir of dirs) {\n if (!existsSync(dir)) {\n mkdirSync(dir, { recursive: true });\n }\n }\n\n // Configure git worktree support\n if (config.enableWorktrees) {\n console.log(chalk.gray('Configuring worktree support...'));\n const worktreeManager = WorktreeManager.getInstance();\n worktreeManager.saveConfig({\n enabled: true,\n autoDetect: true,\n isolateContexts: config.worktreeIsolation,\n shareGlobalContext: false,\n syncInterval: 15,\n });\n\n // Detect current worktrees\n const worktrees = worktreeManager.detectWorktrees();\n if (worktrees.length > 0) {\n console.log(chalk.green(` \u2713 Found ${worktrees.length} worktree(s)`));\n worktrees.forEach((wt) => {\n console.log(chalk.gray(` - ${wt.branch} at ${wt.path}`));\n });\n }\n }\n\n // Scan and organize projects\n if (config.enableProjects && config.scanProjects) {\n console.log(chalk.gray('Scanning for projects...'));\n const projectManager = ProjectManager.getInstance();\n\n const scanPaths = (config as any).scanPaths || [\n join(homedir(), 'Dev'),\n join(homedir(), 'dev'),\n join(homedir(), 'Projects'),\n join(homedir(), 'projects'),\n join(homedir(), 'code'),\n ];\n\n await projectManager.scanAndCategorizeAllProjects(\n scanPaths.filter((p: string) => existsSync(p))\n );\n\n const projects = projectManager.getAllProjects();\n console.log(chalk.green(` \u2713 Found ${projects.length} project(s)`));\n\n // Show summary\n const byType: Record<string, number> = {};\n projects.forEach((p) => {\n byType[p.accountType] = (byType[p.accountType] || 0) + 1;\n });\n\n Object.entries(byType).forEach(([type, count]) => {\n console.log(chalk.gray(` - ${type}: ${count} project(s)`));\n });\n }\n\n // Configure Linear integration\n if (config.enableLinear && config.linearApiKey) {\n console.log(chalk.gray('Configuring Linear integration...'));\n const linearConfig = {\n apiKey: config.linearApiKey,\n autoSync: true,\n syncInterval: 300000, // 5 minutes\n };\n\n writeFileSync(\n join(configPath, 'linear-config.json'),\n JSON.stringify(linearConfig, null, 2)\n );\n console.log(chalk.green(' \u2713 Linear configured'));\n }\n\n // Save main configuration\n const mainConfig = {\n version: '1.0.0',\n setupCompleted: new Date().toISOString(),\n features: {\n worktrees: config.enableWorktrees,\n projects: config.enableProjects,\n linear: config.enableLinear,\n analytics: config.enableAnalytics,\n },\n paths: {\n default: config.defaultContextPath,\n },\n };\n\n writeFileSync(\n join(configPath, 'config.json'),\n JSON.stringify(mainConfig, null, 2)\n );\n\n // Create claude-sm symlink for easy access\n const binPath = '/usr/local/bin/claude-sm';\n const sourcePath = join(configPath, 'bin', 'stackmemory');\n\n try {\n // Create wrapper script\n const wrapperScript = `#!/bin/bash\n# StackMemory CLI wrapper with worktree support\nCURRENT_DIR=$(pwd)\n\n# Auto-detect worktree if enabled\nif [ -f ~/.stackmemory/worktree-config.json ]; then\n WORKTREE_ENABLED=$(grep '\"enabled\": true' ~/.stackmemory/worktree-config.json)\n if [ ! -z \"$WORKTREE_ENABLED\" ]; then\n # Check if we're in a git worktree\n if git worktree list &>/dev/null; then\n export SM_WORKTREE_PATH=\"$CURRENT_DIR\"\n fi\n fi\nfi\n\n# Run StackMemory with context\nexec stackmemory \"$@\"\n`;\n\n writeFileSync(sourcePath, wrapperScript);\n execFileSync('chmod', ['+x', sourcePath]);\n\n // Create symlink if it doesn't exist\n if (!existsSync(binPath)) {\n execFileSync('ln', ['-s', sourcePath, binPath]);\n console.log(chalk.green(' \u2713 Created claude-sm command'));\n }\n } catch (error: unknown) {\n console.log(\n chalk.yellow(' \u26A0 Could not create claude-sm symlink (may need sudo)')\n );\n }\n}\n\nfunction showNextSteps(config: OnboardingConfig): void {\n console.log(chalk.cyan('\uD83C\uDF89 Next Steps:\\n'));\n\n console.log('1. Initialize StackMemory in your project:');\n console.log(chalk.gray(' cd your-project'));\n console.log(chalk.gray(' stackmemory init\\n'));\n\n if (config.enableWorktrees) {\n console.log('2. Create a new worktree:');\n console.log(\n chalk.gray(\n ' git worktree add -b feature/new-feature ../project-feature'\n )\n );\n console.log(chalk.gray(' cd ../project-feature'));\n console.log(\n chalk.gray(\n ' stackmemory status # Isolated context for this worktree\\n'\n )\n );\n }\n\n console.log('3. Use with Claude:');\n console.log(chalk.gray(' claude-sm # Or use stackmemory directly\\n'));\n\n if (config.enableLinear) {\n console.log('4. Sync with Linear:');\n console.log(chalk.gray(' stackmemory linear sync\\n'));\n }\n\n console.log('For more help:');\n console.log(chalk.gray(' stackmemory --help'));\n console.log(chalk.gray(' stackmemory projects --help'));\n if (config.enableWorktrees) {\n console.log(chalk.gray(' stackmemory worktree --help'));\n }\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["#!/usr/bin/env node\n/**\n * StackMemory Onboarding CLI\n * Interactive setup for new StackMemory installations\n */\n\nimport { Command } from 'commander';\nimport inquirer from 'inquirer';\nimport chalk from 'chalk';\nimport { homedir } from 'os';\nimport { join } from 'path';\nimport { existsSync, mkdirSync, writeFileSync } from 'fs';\nimport { execSync, execFileSync } from 'child_process';\nimport { WorktreeManager } from '../../core/worktree/worktree-manager.js';\nimport { ProjectManager } from '../../core/projects/project-manager.js';\nimport { logger } from '../../core/monitoring/logger.js';\n\ninterface OnboardingConfig {\n setupType: 'basic' | 'advanced';\n enableWorktrees: boolean;\n worktreeIsolation: boolean;\n enableProjects: boolean;\n scanProjects: boolean;\n enableLinear: boolean;\n linearApiKey?: string;\n enableAnalytics: boolean;\n defaultContextPath: string;\n storageMode: 'local' | 'hosted';\n databaseUrl?: string;\n}\n\nexport function registerOnboardingCommand(program: Command): void {\n program\n .command('onboard')\n .alias('setup')\n .description('Interactive setup for StackMemory')\n .option('--reset', 'Reset all configurations and start fresh')\n .action(async (options) => {\n console.log(chalk.cyan('\\n\uD83D\uDE80 Welcome to StackMemory Setup!\\n'));\n\n // Check if already configured\n const configPath = join(homedir(), '.stackmemory');\n if (existsSync(configPath) && !options.reset) {\n const { proceed } = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'proceed',\n message:\n 'StackMemory is already configured. Do you want to reconfigure?',\n default: false,\n },\n ]);\n\n if (!proceed) {\n console.log(chalk.yellow('\\nSetup cancelled.'));\n return;\n }\n }\n\n try {\n const config = await runOnboarding();\n await applyConfiguration(config);\n\n console.log(\n chalk.green('\\n\u2705 StackMemory setup completed successfully!\\n')\n );\n showNextSteps(config);\n } catch (error: unknown) {\n logger.error('Onboarding failed', error as Error);\n console.error(\n chalk.red('\\n\u274C Setup failed:'),\n (error as Error).message\n );\n process.exit(1);\n }\n });\n}\n\nasync function runOnboarding(): Promise<OnboardingConfig> {\n // Basic or Advanced Setup\n const { setupType } = await inquirer.prompt([\n {\n type: 'list',\n name: 'setupType',\n message: 'Choose your setup type:',\n choices: [\n { name: 'Basic (Recommended for most users)', value: 'basic' },\n { name: 'Advanced (Full customization)', value: 'advanced' },\n ],\n default: 'basic',\n },\n ]);\n\n let config: OnboardingConfig = {\n setupType,\n enableWorktrees: false,\n worktreeIsolation: true,\n enableProjects: true,\n scanProjects: false,\n enableLinear: false,\n enableAnalytics: true,\n defaultContextPath: join(homedir(), '.stackmemory'),\n storageMode: 'local',\n };\n\n // Choose storage mode\n const { storageMode } = await inquirer.prompt([\n {\n type: 'list',\n name: 'storageMode',\n message: 'Where should StackMemory store data?',\n choices: [\n { name: 'Local (free, SQLite in ~/.stackmemory)', value: 'local' },\n { name: 'Hosted (paid, managed Postgres)', value: 'hosted' },\n ],\n default: 'local',\n },\n ]);\n config.storageMode = storageMode;\n\n if (storageMode === 'hosted') {\n const { hasAccount } = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'hasAccount',\n message: 'Do you already have a hosted account & connection string?',\n default: false,\n },\n ]);\n\n if (!hasAccount) {\n try {\n const signupUrl = 'https://stackmemory.ai/hosted';\n console.log(chalk.gray(`Opening signup page: ${signupUrl}`));\n // Lazy load open to avoid ES module issues\n const mod = await import('open');\n await mod.default(signupUrl);\n } catch (e) {\n console.log(chalk.yellow('Could not open browser automatically. Please sign up and obtain your DATABASE_URL.'));\n }\n }\n\n const { databaseUrl } = await inquirer.prompt([\n {\n type: 'password',\n name: 'databaseUrl',\n message: 'Paste your DATABASE_URL (postgres://...)',\n validate: (input: string) =>\n input.startsWith('postgres://') || input.startsWith('postgresql://')\n ? true\n : 'Must start with postgres:// or postgresql://',\n },\n ]);\n config.databaseUrl = databaseUrl;\n }\n\n if (setupType === 'basic') {\n // Basic setup with sensible defaults\n const basicAnswers = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'enableWorktrees',\n message:\n 'Enable Git worktree support? (Recommended for multi-branch workflows)',\n default: false,\n },\n {\n type: 'confirm',\n name: 'scanProjects',\n message: 'Scan and organize your existing projects?',\n default: true,\n },\n {\n type: 'confirm',\n name: 'enableLinear',\n message: 'Connect to Linear for task management?',\n default: false,\n },\n ]);\n\n config = { ...config, ...basicAnswers };\n\n if (basicAnswers.enableLinear) {\n const { linearApiKey } = await inquirer.prompt([\n {\n type: 'password',\n name: 'linearApiKey',\n message: 'Enter your Linear API key:',\n validate: (input: string) =>\n input.length > 0 || 'API key is required',\n },\n ]);\n config.linearApiKey = linearApiKey;\n }\n } else {\n // Advanced setup with all options\n const advancedAnswers = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'enableWorktrees',\n message: 'Enable Git worktree support?',\n default: false,\n },\n ]);\n\n if (advancedAnswers.enableWorktrees) {\n const worktreeAnswers = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'worktreeIsolation',\n message: 'Isolate contexts between worktrees? (Recommended)',\n default: true,\n },\n {\n type: 'confirm',\n name: 'autoDetect',\n message: 'Auto-detect worktrees when switching directories?',\n default: true,\n },\n {\n type: 'confirm',\n name: 'shareGlobal',\n message: 'Share global context across worktrees?',\n default: false,\n },\n {\n type: 'number',\n name: 'syncInterval',\n message: 'Context sync interval in minutes (0 to disable):',\n default: 15,\n validate: (input: number) => input >= 0 || 'Must be 0 or positive',\n },\n ]);\n\n config = { ...config, ...advancedAnswers, ...worktreeAnswers };\n }\n\n const projectAnswers = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'enableProjects',\n message: 'Enable automatic project management?',\n default: true,\n },\n ]);\n\n if (projectAnswers.enableProjects) {\n const projectDetailAnswers: any = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'scanProjects',\n message: 'Scan for existing projects now?',\n default: true,\n },\n {\n type: 'checkbox',\n name: 'scanPaths',\n message: 'Select directories to scan:',\n choices: [\n { name: '~/Dev', value: join(homedir(), 'Dev'), checked: true },\n { name: '~/dev', value: join(homedir(), 'dev'), checked: true },\n {\n name: '~/Projects',\n value: join(homedir(), 'Projects'),\n checked: true,\n },\n {\n name: '~/projects',\n value: join(homedir(), 'projects'),\n checked: true,\n },\n { name: '~/Work', value: join(homedir(), 'Work'), checked: false },\n { name: '~/code', value: join(homedir(), 'code'), checked: true },\n {\n name: '~/Documents/GitHub',\n value: join(homedir(), 'Documents/GitHub'),\n checked: false,\n },\n ],\n when: (): boolean => projectDetailAnswers.scanProjects,\n },\n ]);\n\n config = { ...config, ...projectAnswers, ...projectDetailAnswers };\n }\n\n const integrationAnswers = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'enableLinear',\n message: 'Enable Linear integration?',\n default: false,\n },\n {\n type: 'password',\n name: 'linearApiKey',\n message: 'Linear API key:',\n when: (answers: any) => answers.enableLinear,\n validate: (input: string) => input.length > 0 || 'API key is required',\n },\n {\n type: 'confirm',\n name: 'enableAnalytics',\n message: 'Enable usage analytics? (Local only)',\n default: true,\n },\n {\n type: 'input',\n name: 'defaultContextPath',\n message: 'Default context storage path:',\n default: join(homedir(), '.stackmemory'),\n validate: (input: string) => input.length > 0 || 'Path is required',\n },\n ]);\n\n config = { ...config, ...integrationAnswers };\n }\n\n return config;\n}\n\nasync function applyConfiguration(config: OnboardingConfig): Promise<void> {\n const configPath = join(homedir(), '.stackmemory');\n\n // Create base directory structure\n console.log(chalk.gray('\\nCreating directory structure...'));\n const dirs = [\n configPath,\n join(configPath, 'contexts'),\n join(configPath, 'projects'),\n join(configPath, 'worktrees'),\n join(configPath, 'bin'),\n join(configPath, 'logs'),\n join(configPath, 'analytics'),\n ];\n\n for (const dir of dirs) {\n if (!existsSync(dir)) {\n mkdirSync(dir, { recursive: true });\n }\n }\n\n // Configure git worktree support\n if (config.enableWorktrees) {\n console.log(chalk.gray('Configuring worktree support...'));\n const worktreeManager = WorktreeManager.getInstance();\n worktreeManager.saveConfig({\n enabled: true,\n autoDetect: true,\n isolateContexts: config.worktreeIsolation,\n shareGlobalContext: false,\n syncInterval: 15,\n });\n\n // Detect current worktrees\n const worktrees = worktreeManager.detectWorktrees();\n if (worktrees.length > 0) {\n console.log(chalk.green(` \u2713 Found ${worktrees.length} worktree(s)`));\n worktrees.forEach((wt) => {\n console.log(chalk.gray(` - ${wt.branch} at ${wt.path}`));\n });\n }\n }\n\n // Scan and organize projects\n if (config.enableProjects && config.scanProjects) {\n console.log(chalk.gray('Scanning for projects...'));\n const projectManager = ProjectManager.getInstance();\n\n const scanPaths = (config as any).scanPaths || [\n join(homedir(), 'Dev'),\n join(homedir(), 'dev'),\n join(homedir(), 'Projects'),\n join(homedir(), 'projects'),\n join(homedir(), 'code'),\n ];\n\n await projectManager.scanAndCategorizeAllProjects(\n scanPaths.filter((p: string) => existsSync(p))\n );\n\n const projects = projectManager.getAllProjects();\n console.log(chalk.green(` \u2713 Found ${projects.length} project(s)`));\n\n // Show summary\n const byType: Record<string, number> = {};\n projects.forEach((p) => {\n byType[p.accountType] = (byType[p.accountType] || 0) + 1;\n });\n\n Object.entries(byType).forEach(([type, count]) => {\n console.log(chalk.gray(` - ${type}: ${count} project(s)`));\n });\n }\n\n // Configure Linear integration\n if (config.enableLinear && config.linearApiKey) {\n console.log(chalk.gray('Configuring Linear integration...'));\n const linearConfig = {\n apiKey: config.linearApiKey,\n autoSync: true,\n syncInterval: 300000, // 5 minutes\n };\n\n writeFileSync(\n join(configPath, 'linear-config.json'),\n JSON.stringify(linearConfig, null, 2)\n );\n console.log(chalk.green(' \u2713 Linear configured'));\n }\n\n // Save main configuration\n const mainConfig = {\n version: '1.0.0',\n setupCompleted: new Date().toISOString(),\n features: {\n worktrees: config.enableWorktrees,\n projects: config.enableProjects,\n linear: config.enableLinear,\n analytics: config.enableAnalytics,\n },\n paths: {\n default: config.defaultContextPath,\n },\n database: {\n mode: config.storageMode,\n ...(config.databaseUrl ? { url: config.databaseUrl } : {}),\n },\n };\n\n writeFileSync(\n join(configPath, 'config.json'),\n JSON.stringify(mainConfig, null, 2)\n );\n\n // If hosted selected, persist a simple env file suggestion\n if (config.storageMode === 'hosted' && config.databaseUrl) {\n try {\n const envFile = join(configPath, 'railway.env');\n writeFileSync(envFile, `# StackMemory hosted DB\\nDATABASE_URL=${config.databaseUrl}\\n`);\n console.log(chalk.green(' \u2713 Saved hosted DB settings to ~/.stackmemory/railway.env'));\n console.log(chalk.gray(' Tip: export DATABASE_URL from this file in your shell profile.'));\n } catch (e) {\n console.log(chalk.yellow(' \u26A0 Could not write hosted DB env file'));\n }\n }\n\n // Create claude-sm symlink for easy access\n const binPath = '/usr/local/bin/claude-sm';\n const sourcePath = join(configPath, 'bin', 'stackmemory');\n\n try {\n // Create wrapper script\n const wrapperScript = `#!/bin/bash\n# StackMemory CLI wrapper with worktree support\nCURRENT_DIR=$(pwd)\n\n# Auto-detect worktree if enabled\nif [ -f ~/.stackmemory/worktree-config.json ]; then\n WORKTREE_ENABLED=$(grep '\"enabled\": true' ~/.stackmemory/worktree-config.json)\n if [ ! -z \"$WORKTREE_ENABLED\" ]; then\n # Check if we're in a git worktree\n if git worktree list &>/dev/null; then\n export SM_WORKTREE_PATH=\"$CURRENT_DIR\"\n fi\n fi\nfi\n\n# Run StackMemory with context\nexec stackmemory \"$@\"\n`;\n\n writeFileSync(sourcePath, wrapperScript);\n execFileSync('chmod', ['+x', sourcePath]);\n\n // Create symlink if it doesn't exist\n if (!existsSync(binPath)) {\n execFileSync('ln', ['-s', sourcePath, binPath]);\n console.log(chalk.green(' \u2713 Created claude-sm command'));\n }\n } catch (error: unknown) {\n console.log(\n chalk.yellow(' \u26A0 Could not create claude-sm symlink (may need sudo)')\n );\n }\n\n // Create codex-sm symlink for easy access\n const codexBinPath = '/usr/local/bin/codex-sm';\n const codexSourcePath = join(configPath, 'bin', 'codex-sm');\n\n try {\n // Create Codex wrapper script (based on scripts/codex-wrapper.sh)\n const codexWrapper = `#!/bin/bash\n# Codex CLI wrapper with StackMemory integration\n# Usage: codex-sm [--auto-sync] [--sync-interval=MINUTES] [args...]\n\n# Flags\nAUTO_SYNC=false\nSYNC_INTERVAL=5\nfor arg in \"$@\"; do\n case $arg in\n --auto-sync)\n AUTO_SYNC=true\n shift\n ;;\n --sync-interval=*)\n SYNC_INTERVAL=\"\\${arg#*=}\"\n shift\n ;;\n esac\ndone\n\n# Auto-initialize StackMemory if in git repo without it\nif [ -d \".git\" ] && [ ! -d \".stackmemory\" ]; then\n echo \"\uD83D\uDCE6 Initializing StackMemory for this project...\"\n stackmemory init --silent 2>/dev/null || true\nfi\n\n# Load existing context if available\nif [ -d \".stackmemory\" ]; then\n echo \"\uD83E\uDDE0 Loading StackMemory context...\"\n stackmemory status --brief 2>/dev/null || true\nfi\n\n# Start Linear auto-sync in background if requested\nSYNC_PID=\"\"\nif [ \"$AUTO_SYNC\" = true ] && [ -n \"$LINEAR_API_KEY\" ]; then\n echo \"\uD83D\uDD04 Starting Linear auto-sync (${SYNC_INTERVAL}min intervals)...\"\n (\n while true; do\n sleep $((SYNC_INTERVAL * 60))\n if [ -d \".stackmemory\" ]; then\n stackmemory linear sync --quiet 2>/dev/null || true\n fi\n done\n ) &\n SYNC_PID=$!\nfi\n\ncleanup() {\n echo \"\"\n echo \"\uD83D\uDCDD Saving StackMemory context...\"\n\n # Kill auto-sync if running\n if [ -n \"$SYNC_PID\" ] && kill -0 $SYNC_PID 2>/dev/null; then\n echo \"\uD83D\uDED1 Stopping auto-sync...\"\n kill $SYNC_PID 2>/dev/null || true\n fi\n\n # Save project status and final sync\n if [ -d \".stackmemory\" ]; then\n stackmemory status 2>/dev/null\n if [ -n \"$LINEAR_API_KEY\" ]; then\n echo \"\uD83D\uDD04 Final Linear sync...\"\n stackmemory linear sync 2>/dev/null\n fi\n echo \"\u2705 StackMemory context saved\"\n fi\n}\n\ntrap cleanup EXIT INT TERM\n\n# Run Codex CLI\nif command -v codex &> /dev/null; then\n codex \"$@\"\nelif command -v codex-cli &> /dev/null; then\n codex-cli \"$@\"\nelse\n echo \"\u274C Codex CLI not found. Please install it first.\"\n echo \" See: https://github.com/openai/codex-cli\"\n exit 1\nfi\n`;\n\n writeFileSync(codexSourcePath, codexWrapper);\n execFileSync('chmod', ['+x', codexSourcePath]);\n\n // Create symlink if it doesn't exist\n if (!existsSync(codexBinPath)) {\n execFileSync('ln', ['-s', codexSourcePath, codexBinPath]);\n console.log(chalk.green(' \u2713 Created codex-sm command'));\n }\n } catch (error: unknown) {\n console.log(\n chalk.yellow(' \u26A0 Could not create codex-sm symlink (may need sudo)')\n );\n }\n}\n\nfunction showNextSteps(config: OnboardingConfig): void {\n console.log(chalk.cyan('\uD83C\uDF89 Next Steps:\\n'));\n\n console.log('1. Initialize StackMemory in your project:');\n console.log(chalk.gray(' cd your-project'));\n console.log(chalk.gray(' stackmemory init\\n'));\n\n if (config.enableWorktrees) {\n console.log('2. Create a new worktree:');\n console.log(\n chalk.gray(\n ' git worktree add -b feature/new-feature ../project-feature'\n )\n );\n console.log(chalk.gray(' cd ../project-feature'));\n console.log(\n chalk.gray(\n ' stackmemory status # Isolated context for this worktree\\n'\n )\n );\n }\n\n console.log('3. Use with Claude:');\n console.log(chalk.gray(' claude-sm # Or use stackmemory directly\\n'));\n\n console.log('4. Use with Codex:');\n console.log(chalk.gray(' codex-sm # Codex + StackMemory integration\\n'));\n\n if (config.enableLinear) {\n console.log('5. Sync with Linear:');\n console.log(chalk.gray(' stackmemory linear sync\\n'));\n }\n\n console.log('For more help:');\n console.log(chalk.gray(' stackmemory --help'));\n console.log(chalk.gray(' stackmemory projects --help'));\n if (config.enableWorktrees) {\n console.log(chalk.gray(' stackmemory worktree --help'));\n }\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }