sandstream-kit 1.0.0 → 1.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (86) hide show
  1. package/README.md +18 -13
  2. package/dist/cli.js +458 -2
  3. package/dist/cli.js.map +1 -1
  4. package/dist/database.d.ts +2 -2
  5. package/dist/database.js +9 -14
  6. package/dist/database.js.map +1 -1
  7. package/dist/lock.js +5 -3
  8. package/dist/lock.js.map +1 -1
  9. package/dist/memory/backup 2.d.ts +6 -0
  10. package/dist/memory/backup 2.js +80 -0
  11. package/dist/memory/backup 2.js.map +1 -0
  12. package/dist/memory/backup.d.ts +6 -0
  13. package/dist/memory/backup.js +80 -0
  14. package/dist/memory/backup.js.map +1 -0
  15. package/dist/memory/db 2.d.ts +40 -0
  16. package/dist/memory/db 2.js +233 -0
  17. package/dist/memory/db 2.js.map +1 -0
  18. package/dist/memory/db.d.ts +40 -0
  19. package/dist/memory/db.js +233 -0
  20. package/dist/memory/db.js.map +1 -0
  21. package/dist/memory/hook 2.d.ts +6 -0
  22. package/dist/memory/hook 2.js +51 -0
  23. package/dist/memory/hook 2.js.map +1 -0
  24. package/dist/memory/hook.d.ts +6 -0
  25. package/dist/memory/hook.js +51 -0
  26. package/dist/memory/hook.js.map +1 -0
  27. package/dist/memory/hook.test 2.d.ts +1 -0
  28. package/dist/memory/hook.test 2.js +35 -0
  29. package/dist/memory/hook.test 2.js.map +1 -0
  30. package/dist/memory/install 2.d.ts +8 -0
  31. package/dist/memory/install 2.js +72 -0
  32. package/dist/memory/install 2.js.map +1 -0
  33. package/dist/memory/install.d.ts +8 -0
  34. package/dist/memory/install.js +72 -0
  35. package/dist/memory/install.js.map +1 -0
  36. package/dist/memory/install.test 2.d.ts +1 -0
  37. package/dist/memory/install.test 2.js +59 -0
  38. package/dist/memory/install.test 2.js.map +1 -0
  39. package/dist/memory/pal 2.d.ts +47 -0
  40. package/dist/memory/pal 2.js +154 -0
  41. package/dist/memory/pal 2.js.map +1 -0
  42. package/dist/memory/pal.d.ts +47 -0
  43. package/dist/memory/pal.js +154 -0
  44. package/dist/memory/pal.js.map +1 -0
  45. package/dist/memory/parser.d.ts +25 -0
  46. package/dist/memory/parser.js +164 -0
  47. package/dist/memory/parser.js.map +1 -0
  48. package/dist/memory/project 2.d.ts +1 -0
  49. package/dist/memory/project 2.js +24 -0
  50. package/dist/memory/project 2.js.map +1 -0
  51. package/dist/memory/project.d.ts +1 -0
  52. package/dist/memory/project.js +24 -0
  53. package/dist/memory/project.js.map +1 -0
  54. package/dist/memory/scan 2.d.ts +15 -0
  55. package/dist/memory/scan 2.js +94 -0
  56. package/dist/memory/scan 2.js.map +1 -0
  57. package/dist/memory/scan.d.ts +15 -0
  58. package/dist/memory/scan.js +94 -0
  59. package/dist/memory/scan.js.map +1 -0
  60. package/dist/memory/scan.test 2.d.ts +1 -0
  61. package/dist/memory/scan.test 2.js +55 -0
  62. package/dist/memory/scan.test 2.js.map +1 -0
  63. package/dist/memory/shared 2.d.ts +33 -0
  64. package/dist/memory/shared 2.js +120 -0
  65. package/dist/memory/shared 2.js.map +1 -0
  66. package/dist/memory/shared.d.ts +33 -0
  67. package/dist/memory/shared.js +120 -0
  68. package/dist/memory/shared.js.map +1 -0
  69. package/dist/memory/threads 2.d.ts +37 -0
  70. package/dist/memory/threads 2.js +50 -0
  71. package/dist/memory/threads 2.js.map +1 -0
  72. package/dist/memory/threads.d.ts +37 -0
  73. package/dist/memory/threads.js +50 -0
  74. package/dist/memory/threads.js.map +1 -0
  75. package/dist/memory/threads.test 2.d.ts +1 -0
  76. package/dist/memory/threads.test 2.js +66 -0
  77. package/dist/memory/threads.test 2.js.map +1 -0
  78. package/dist/memory/types 2.d.ts +52 -0
  79. package/dist/memory/types 2.js +5 -0
  80. package/dist/memory/types 2.js.map +1 -0
  81. package/dist/memory/types.d.ts +52 -0
  82. package/dist/memory/types.js +5 -0
  83. package/dist/memory/types.js.map +1 -0
  84. package/dist/secrets-pull.d.ts +1 -1
  85. package/dist/secrets-pull.js +1 -1
  86. package/package.json +1 -1
package/README.md CHANGED
@@ -252,6 +252,24 @@ Production credentials are gated behind explicit env-switching and short-lived e
252
252
  - Supply-chain findings auto-append to `.kit-audit.jsonl` (one JSON line per finding) for SIEM ingest
253
253
  - Releases ship with SLSA provenance (`npm publish --provenance`), CycloneDX + SPDX SBOMs on every GitHub release, cosign-signed Docker images, and weekly OpenSSF Scorecard
254
254
 
255
+ ## Memory
256
+
257
+ `kit memory` gives an agent a local-first, deterministic second brain — it stores
258
+ your raw conversation history and searches it *before answering*, so it pulls
259
+ receipts instead of guessing. SQLite + FTS5, two hooks, no vectors, no model calls.
260
+ A private personal tier (encrypted backup so a stolen laptop doesn't lose your
261
+ context) plus a curated, area-organized **shared** tier that travels with the repo
262
+ and is reviewed like code.
263
+
264
+ ```bash
265
+ kit memory install && kit memory index
266
+ kit memory search "what did we decide about X" # project-scoped recall
267
+ kit memory area stripe # shared: how we built it, status, security
268
+ ```
269
+
270
+ Full reference: [`docs/MEMORY.md`](docs/MEMORY.md). Schema + two-hook design
271
+ credited to [cloudctx](https://github.com/chadptk1238/cloudctx) (MIT).
272
+
255
273
  ## Lock Files
256
274
 
257
275
  kit uses lock files in `.kit/` to track exact versions of skills and tools:
@@ -584,19 +602,6 @@ For Cline, add the same config to your `cline_mcp_settings.json`.
584
602
  }
585
603
  ```
586
604
 
587
- ## OpenRouter API Key Setup
588
-
589
- kit uses OpenRouter for AI model access via the OpenCode CLI. To set up:
590
-
591
- 1. Get your API key from [OpenRouter](https://openrouter.ai/keys)
592
- 2. Add it to your `.env.local` file:
593
- ```bash
594
- OPENROUTER_API_KEY=<your-openrouter-key>
595
- ```
596
- 3. The key is automatically loaded by `opencode.json`
597
-
598
- **Security Note:** Never commit your API key to git. It's configured as an environment variable in `opencode.json` and should only exist in `.env.local`.
599
-
600
605
  ## Community & Support
601
606
 
602
607
  ### Getting Help
package/dist/cli.js CHANGED
@@ -1,8 +1,8 @@
1
1
  #!/usr/bin/env node
2
- import { readFileSync } from "node:fs";
2
+ import { readFileSync, existsSync } from "node:fs";
3
3
  import { writeFile, access, mkdir } from "node:fs/promises";
4
4
  import { fileURLToPath } from "node:url";
5
- import { resolve, dirname, join } from "node:path";
5
+ import { resolve, dirname, join, basename } from "node:path";
6
6
  import { loadConfig, resolveActiveEnvironment } from "./config.js";
7
7
  import { checkTools } from "./check-tools.js";
8
8
  import { checkServices } from "./check-services.js";
@@ -68,6 +68,16 @@ import { listServices, openService } from "./open.js";
68
68
  import { gatherProjectContext } from "./context.js";
69
69
  import { runTriage, listTriageTools } from "./triage.js";
70
70
  import { parsePkgSpec, installPkg } from "./pkg.js";
71
+ import { openMemoryDb, getStats, getMemoryDbPath, searchMessages } from "./memory/db.js";
72
+ import { indexClaudeTranscripts, getClaudeProjectsDir } from "./memory/parser.js";
73
+ import { getCurrentProjectRoot } from "./memory/project.js";
74
+ import { scanDbForSecrets } from "./memory/scan.js";
75
+ import { backupEncrypted, restoreEncrypted } from "./memory/backup.js";
76
+ import { shareEntry, listAreas, queryArea, getSharedPath, } from "./memory/shared.js";
77
+ import { userPromptSubmitReminder, runSessionEndIndex } from "./memory/hook.js";
78
+ import { installMemoryHooks, uninstallMemoryHooks, getClaudeSettingsPath, } from "./memory/install.js";
79
+ import { palAdd, palList, palDone, palSnooze, palAutoVerify, importLegacyLedger, } from "./memory/pal.js";
80
+ import { saveThread, listThreads, removeThread, latestSessionId, resolveThread, } from "./memory/threads.js";
71
81
  const KIT_FILE = ".kit.toml";
72
82
  const __dirname = dirname(fileURLToPath(import.meta.url));
73
83
  const KIT_VERSION = JSON.parse(readFileSync(join(__dirname, "..", "package.json"), "utf-8")).version;
@@ -3391,6 +3401,21 @@ function cmdVersion() {
3391
3401
  }
3392
3402
  const COMMAND_HELP = {
3393
3403
  check: "Check status of all tools, services, secrets, and lock files",
3404
+ memory: "Local conversation memory — index transcripts + show stats",
3405
+ "memory index": "Index ~/.claude transcripts into the SQLite memory store",
3406
+ "memory search": "Full-text search memory (current project; --global for all)",
3407
+ "memory stats": "Show what the local memory store contains",
3408
+ "memory install": "Wire UserPromptSubmit + SessionEnd hooks into ~/.claude/settings.json",
3409
+ "memory scan": "Scan the memory store for stored secrets (exit 1 if any found)",
3410
+ "memory backup": "Encrypted backup of the memory store (AES-256-GCM; KIT_MEMORY_PASSPHRASE)",
3411
+ "memory restore": "Restore an encrypted memory backup (e.g. on a new machine)",
3412
+ "memory share": "Promote a curated, secret-scanned entry to the shared (team) memory",
3413
+ "memory areas": "List shared responsibility areas (stripe, whatsapp, …)",
3414
+ "memory area": "Show shared entries for one area (decisions, how-built, status, security)",
3415
+ "memory pal": "Pending action ledger — list/add/done/snooze/verify/import 'blocked-on-you' items",
3416
+ "memory save": "Bookmark the current session as a named copilot",
3417
+ "memory threads": "List saved copilots (current project; --global for all)",
3418
+ "memory resume": "Print the resume command for a saved copilot (by name or number)",
3394
3419
  init: "Detect stack, generate .kit.toml, and run full setup",
3395
3420
  upgrade: "Update lock files from .kit.toml",
3396
3421
  install: "Install missing tools via mise",
@@ -3858,6 +3883,434 @@ async function showSkippedCommitBanner() {
3858
3883
  /* banner is best-effort — never break the CLI */
3859
3884
  }
3860
3885
  }
3886
+ /**
3887
+ * `kit memory` — local conversation memory (SQLite + FTS5). Deterministic and
3888
+ * zero-LLM: it stores raw transcripts and searches them; it never calls a model.
3889
+ */
3890
+ async function cmdMemory() {
3891
+ const subcommand = process.argv[3];
3892
+ const jsonMode = hasFlag(process.argv, "--json");
3893
+ if (!subcommand || subcommand === "--help" || subcommand === "-h") {
3894
+ console.log("kit memory — local conversation memory (SQLite + FTS5)");
3895
+ console.log("\nUsage:");
3896
+ console.log(" kit memory index Index ~/.claude transcripts into the memory store");
3897
+ console.log(" kit memory search <query> Search memory (current project; --global for all)");
3898
+ console.log(" kit memory stats Show what the memory store contains");
3899
+ console.log(" kit memory install Wire the 2 hooks into ~/.claude/settings.json");
3900
+ console.log(" kit memory uninstall Remove the hooks");
3901
+ console.log(" kit memory pal [list|add|done|snooze|verify|import] Pending action ledger");
3902
+ console.log(" kit memory save <name> Bookmark the current session as a named copilot");
3903
+ console.log(" kit memory threads List saved copilots (--global for all)");
3904
+ console.log(" kit memory resume <name|n> Print the resume command for a saved copilot");
3905
+ console.log(" kit memory forget <name> Remove a saved copilot");
3906
+ console.log(" kit memory scan Scan the store for stored secrets");
3907
+ console.log(" kit memory backup <file> Encrypted backup (set KIT_MEMORY_PASSPHRASE)");
3908
+ console.log(" kit memory restore <file> Restore an encrypted backup (new machine)");
3909
+ console.log(" kit memory share … Promote a curated entry to shared (team) memory");
3910
+ console.log(" kit memory areas List shared responsibility areas");
3911
+ console.log(" kit memory area <name> Show shared entries for one area");
3912
+ return true;
3913
+ }
3914
+ if (subcommand === "index") {
3915
+ const db = openMemoryDb();
3916
+ const t0 = Date.now();
3917
+ const res = indexClaudeTranscripts(db);
3918
+ const ms = Date.now() - t0;
3919
+ db.close();
3920
+ if (jsonMode) {
3921
+ console.log(JSON.stringify({ ...res, ms }));
3922
+ return true;
3923
+ }
3924
+ console.log(`${c.green}✓${c.reset} indexed ${c.bold}${res.messages}${c.reset} messages + ${res.toolUses} tool-uses from ${res.files} sessions ${c.dim}(${ms}ms)${c.reset}`);
3925
+ console.log(`${c.dim}source: ${getClaudeProjectsDir()}${c.reset}`);
3926
+ return true;
3927
+ }
3928
+ if (subcommand === "stats") {
3929
+ const db = openMemoryDb();
3930
+ const s = getStats(db);
3931
+ db.close();
3932
+ if (jsonMode) {
3933
+ console.log(JSON.stringify(s));
3934
+ return true;
3935
+ }
3936
+ console.log(`${c.bold}kit memory${c.reset} ${c.dim}${s.dbPath}${c.reset}`);
3937
+ console.log(` sessions ${s.sessions}`);
3938
+ console.log(` messages ${s.messages}`);
3939
+ console.log(` tool-uses ${s.toolUses}`);
3940
+ console.log(` pending ${s.pendingOpen} ${c.dim}(open action items)${c.reset}`);
3941
+ console.log(` size ${Math.round(s.sizeBytes / 1024)} KB`);
3942
+ return true;
3943
+ }
3944
+ if (subcommand === "search") {
3945
+ const terms = process.argv.slice(4).filter((a) => !a.startsWith("--"));
3946
+ const query = terms.join(" ").trim();
3947
+ if (!query) {
3948
+ console.error(`${c.red}usage: kit memory search <query> [--global] [--project=<path>] [--limit=N]${c.reset}`);
3949
+ return false;
3950
+ }
3951
+ const limit = Number(flagValue(process.argv, "--limit") ?? "20") || 20;
3952
+ const projectPath = hasFlag(process.argv, "--global")
3953
+ ? undefined
3954
+ : (flagValue(process.argv, "--project") ?? getCurrentProjectRoot());
3955
+ const db = openMemoryDb();
3956
+ const hits = searchMessages(db, query, { limit, projectPath });
3957
+ db.close();
3958
+ if (jsonMode) {
3959
+ console.log(JSON.stringify(hits));
3960
+ return true;
3961
+ }
3962
+ const scope = projectPath ? `${c.dim}in ${projectPath}${c.reset}` : `${c.dim}(global)${c.reset}`;
3963
+ if (!hits.length) {
3964
+ console.log(`${c.dim}no matches for "${query}" ${projectPath ?? "(global)"}${c.reset}`);
3965
+ return true;
3966
+ }
3967
+ console.log(`${c.bold}${hits.length}${c.reset} match(es) ${scope}`);
3968
+ for (const h of hits) {
3969
+ const snippet = (h.content ?? "").replace(/\s+/g, " ").slice(0, 120);
3970
+ console.log(` ${c.dim}${h.timestamp ?? "?"}${c.reset} ${c.bold}${h.role ?? h.uuid ?? ""}${c.reset} ${snippet}`);
3971
+ }
3972
+ return true;
3973
+ }
3974
+ if (subcommand === "hook") {
3975
+ // Internal: invoked by Claude Code hooks. Fail-open — never block.
3976
+ const event = process.argv[4];
3977
+ if (event === "user-prompt-submit") {
3978
+ const text = userPromptSubmitReminder();
3979
+ if (text)
3980
+ console.log(text);
3981
+ return true;
3982
+ }
3983
+ if (event === "session-end") {
3984
+ runSessionEndIndex();
3985
+ return true;
3986
+ }
3987
+ console.error(`${c.red}Unknown hook event: ${event ?? "(none)"}${c.reset}`);
3988
+ return false;
3989
+ }
3990
+ if (subcommand === "install") {
3991
+ const { added, alreadyPresent } = installMemoryHooks();
3992
+ for (const e of added)
3993
+ console.log(`${c.green}✓${c.reset} installed ${e} hook`);
3994
+ for (const e of alreadyPresent)
3995
+ console.log(`${c.dim}• ${e} hook already present${c.reset}`);
3996
+ console.log(`${c.dim}settings: ${getClaudeSettingsPath()}${c.reset}`);
3997
+ return true;
3998
+ }
3999
+ if (subcommand === "uninstall") {
4000
+ const { removed } = uninstallMemoryHooks();
4001
+ if (removed.length) {
4002
+ for (const e of removed)
4003
+ console.log(`${c.green}✓${c.reset} removed ${e} hook`);
4004
+ }
4005
+ else {
4006
+ console.log(`${c.dim}no kit memory hooks were installed${c.reset}`);
4007
+ }
4008
+ return true;
4009
+ }
4010
+ if (subcommand === "share") {
4011
+ const area = flagValue(process.argv, "--area");
4012
+ const title = flagValue(process.argv, "--title");
4013
+ const kind = (flagValue(process.argv, "--kind") ?? "note");
4014
+ const body = flagValue(process.argv, "--body") ?? "";
4015
+ const ref = flagValue(process.argv, "--ref");
4016
+ if (!area || !title) {
4017
+ console.error(`${c.red}usage: kit memory share --area <a> --title <t> [--kind decision|convention|how-built|status|security|note] [--body <b>] [--ref <r>]${c.reset}`);
4018
+ return false;
4019
+ }
4020
+ const root = getCurrentProjectRoot();
4021
+ try {
4022
+ const e = shareEntry(root, { area, kind, title, body, refs: ref ? [ref] : [] }, new Date().toISOString());
4023
+ console.log(`${c.green}✓${c.reset} shared ${c.bold}${e.id}${c.reset} to area ${c.bold}${area}${c.reset} ${c.dim}(${getSharedPath(root)})${c.reset}`);
4024
+ console.log(`${c.dim}commit .kit/shared/memory.jsonl + open a PR — shared memory is reviewed like code${c.reset}`);
4025
+ }
4026
+ catch (err) {
4027
+ console.error(`${c.red}${err.message}${c.reset}`);
4028
+ return false;
4029
+ }
4030
+ return true;
4031
+ }
4032
+ if (subcommand === "areas") {
4033
+ const areas = listAreas(getCurrentProjectRoot());
4034
+ if (jsonMode) {
4035
+ console.log(JSON.stringify(areas));
4036
+ return true;
4037
+ }
4038
+ if (!areas.length) {
4039
+ console.log(`${c.dim}no shared areas yet — add one with kit memory share${c.reset}`);
4040
+ return true;
4041
+ }
4042
+ console.log(`${c.bold}${areas.length}${c.reset} responsibility area(s):`);
4043
+ for (const a of areas) {
4044
+ console.log(` ${c.bold}${a.area}${c.reset} ${c.dim}· ${a.count} entr${a.count === 1 ? "y" : "ies"}${c.reset}`);
4045
+ }
4046
+ return true;
4047
+ }
4048
+ if (subcommand === "area") {
4049
+ const name = process.argv[4];
4050
+ if (!name) {
4051
+ console.error(`${c.red}usage: kit memory area <name>${c.reset}`);
4052
+ return false;
4053
+ }
4054
+ const entries = queryArea(getCurrentProjectRoot(), name);
4055
+ if (jsonMode) {
4056
+ console.log(JSON.stringify(entries));
4057
+ return true;
4058
+ }
4059
+ if (!entries.length) {
4060
+ console.log(`${c.dim}no shared memory for area '${name}'${c.reset}`);
4061
+ return true;
4062
+ }
4063
+ console.log(`${c.bold}${name}${c.reset} ${c.dim}· ${entries.length} entr${entries.length === 1 ? "y" : "ies"}${c.reset}`);
4064
+ for (const e of entries) {
4065
+ const prov = `${e.author}${e.source_ref ? ` @${e.source_ref}` : ""}`;
4066
+ console.log(` ${c.bold}[${e.kind}]${c.reset} ${e.title} ${c.dim}— ${prov}${c.reset}`);
4067
+ if (e.body)
4068
+ console.log(` ${e.body}`);
4069
+ if (e.refs.length)
4070
+ console.log(` ${c.dim}refs: ${e.refs.join(", ")}${c.reset}`);
4071
+ }
4072
+ return true;
4073
+ }
4074
+ if (subcommand === "scan") {
4075
+ const db = openMemoryDb();
4076
+ const findings = scanDbForSecrets(db);
4077
+ db.close();
4078
+ if (jsonMode) {
4079
+ console.log(JSON.stringify(findings));
4080
+ return !findings.some((f) => f.confidence === "high");
4081
+ }
4082
+ if (!findings.length) {
4083
+ console.log(`${c.green}✓${c.reset} no stored secrets found in the memory store`);
4084
+ return true;
4085
+ }
4086
+ const high = findings.filter((f) => f.confidence === "high");
4087
+ const heuristic = findings.filter((f) => f.confidence === "heuristic");
4088
+ const times = (n) => (n > 1 ? ` ×${n}` : "");
4089
+ if (high.length) {
4090
+ console.log(`${c.red}⚠ ${high.length} high-confidence secret(s):${c.reset}`);
4091
+ for (const f of high) {
4092
+ const proj = f.projects.length ? `${c.bold}[${f.projects.join(", ")}]${c.reset}${c.dim} · ` : "";
4093
+ console.log(` ${c.bold}${f.label}${c.reset} ${c.dim}${f.preview}${times(f.count)} · ${proj}${f.sample}${c.reset}`);
4094
+ }
4095
+ }
4096
+ else {
4097
+ console.log(`${c.green}✓${c.reset} no high-confidence secrets`);
4098
+ }
4099
+ if (heuristic.length) {
4100
+ const showAll = hasFlag(process.argv, "--all");
4101
+ if (showAll) {
4102
+ console.log(`${c.dim}${heuristic.length} heuristic match(es) (KEY=value patterns — usually env vars / paths):${c.reset}`);
4103
+ for (const f of heuristic) {
4104
+ console.log(` ${c.dim}${f.label} ${f.preview}${times(f.count)} · ${f.sample}${c.reset}`);
4105
+ }
4106
+ }
4107
+ else {
4108
+ console.log(`${c.dim}+ ${heuristic.length} heuristic match(es) (likely env vars / paths) — run with --all to see${c.reset}`);
4109
+ }
4110
+ }
4111
+ return high.length === 0; // exit non-zero only on high-confidence findings
4112
+ }
4113
+ if (subcommand === "backup") {
4114
+ const out = process.argv[4];
4115
+ const pass = process.env.KIT_MEMORY_PASSPHRASE ?? flagValue(process.argv, "--passphrase");
4116
+ if (!out) {
4117
+ console.error(`${c.red}usage: kit memory backup <file> (set KIT_MEMORY_PASSPHRASE)${c.reset}`);
4118
+ return false;
4119
+ }
4120
+ if (!pass) {
4121
+ console.error(`${c.red}set KIT_MEMORY_PASSPHRASE (or --passphrase) — the key is never stored${c.reset}`);
4122
+ return false;
4123
+ }
4124
+ try {
4125
+ backupEncrypted(pass, getMemoryDbPath(), out);
4126
+ }
4127
+ catch (err) {
4128
+ console.error(`${c.red}${err.message}${c.reset}`);
4129
+ return false;
4130
+ }
4131
+ console.log(`${c.green}✓${c.reset} encrypted backup → ${out} ${c.dim}(AES-256-GCM · scrypt)${c.reset}`);
4132
+ return true;
4133
+ }
4134
+ if (subcommand === "restore") {
4135
+ const inFile = process.argv[4];
4136
+ const pass = process.env.KIT_MEMORY_PASSPHRASE ?? flagValue(process.argv, "--passphrase");
4137
+ if (!inFile) {
4138
+ console.error(`${c.red}usage: kit memory restore <file> [--to <path>] [--force]${c.reset}`);
4139
+ return false;
4140
+ }
4141
+ if (!pass) {
4142
+ console.error(`${c.red}set KIT_MEMORY_PASSPHRASE (or --passphrase)${c.reset}`);
4143
+ return false;
4144
+ }
4145
+ const dest = flagValue(process.argv, "--to") ?? getMemoryDbPath();
4146
+ if (existsSync(dest) && !hasFlag(process.argv, "--force")) {
4147
+ console.error(`${c.red}${dest} exists — pass --force to overwrite${c.reset}`);
4148
+ return false;
4149
+ }
4150
+ try {
4151
+ restoreEncrypted(pass, inFile, dest);
4152
+ }
4153
+ catch {
4154
+ console.error(`${c.red}restore failed — wrong passphrase or corrupt backup${c.reset}`);
4155
+ return false;
4156
+ }
4157
+ console.log(`${c.green}✓${c.reset} restored → ${dest}`);
4158
+ return true;
4159
+ }
4160
+ if (subcommand === "save") {
4161
+ const name = process.argv.slice(4).filter((a) => !a.startsWith("--")).join(" ").trim();
4162
+ if (!name) {
4163
+ console.error(`${c.red}usage: kit memory save <name> [--session=<id>]${c.reset}`);
4164
+ return false;
4165
+ }
4166
+ const root = getCurrentProjectRoot();
4167
+ const db = openMemoryDb();
4168
+ const sessionId = flagValue(process.argv, "--session") ?? latestSessionId(db, { projectPath: root });
4169
+ if (!sessionId) {
4170
+ db.close();
4171
+ console.error(`${c.red}no session found for ${root} — index first or pass --session=<id>${c.reset}`);
4172
+ return false;
4173
+ }
4174
+ saveThread(db, { name, sessionId, projectPath: root });
4175
+ db.close();
4176
+ console.log(`${c.green}✓${c.reset} saved copilot ${c.bold}${name}${c.reset} ${c.dim}→ ${sessionId}${c.reset}`);
4177
+ return true;
4178
+ }
4179
+ if (subcommand === "threads") {
4180
+ const projectPath = hasFlag(process.argv, "--global") ? undefined : getCurrentProjectRoot();
4181
+ const db = openMemoryDb();
4182
+ const list = listThreads(db, { projectPath });
4183
+ db.close();
4184
+ if (jsonMode) {
4185
+ console.log(JSON.stringify(list));
4186
+ return true;
4187
+ }
4188
+ if (!list.length) {
4189
+ console.log(`${c.dim}no saved copilots${projectPath ? ` in ${projectPath}` : ""}${c.reset}`);
4190
+ return true;
4191
+ }
4192
+ const scope = projectPath ? `${c.dim}in ${projectPath}${c.reset}` : `${c.dim}(global)${c.reset}`;
4193
+ console.log(`${c.bold}${list.length}${c.reset} saved copilot(s) ${scope}:`);
4194
+ list.forEach((t, i) => {
4195
+ console.log(` ${c.bold}${i + 1}${c.reset}. ${t.name} ${c.dim}${t.session_id}${c.reset}`);
4196
+ });
4197
+ console.log(`${c.dim}resume with: kit memory resume <name|number>${c.reset}`);
4198
+ return true;
4199
+ }
4200
+ if (subcommand === "resume") {
4201
+ const ref = process.argv[4];
4202
+ if (!ref) {
4203
+ console.error(`${c.red}usage: kit memory resume <name|number>${c.reset}`);
4204
+ return false;
4205
+ }
4206
+ const projectPath = hasFlag(process.argv, "--global") ? undefined : getCurrentProjectRoot();
4207
+ const db = openMemoryDb();
4208
+ const t = resolveThread(db, ref, { projectPath });
4209
+ db.close();
4210
+ if (!t) {
4211
+ console.error(`${c.red}no saved copilot '${ref}'${c.reset}`);
4212
+ return false;
4213
+ }
4214
+ console.log(`${c.bold}${t.name}${c.reset} — run:`);
4215
+ console.log(` claude --resume ${t.session_id}`);
4216
+ return true;
4217
+ }
4218
+ if (subcommand === "forget") {
4219
+ const name = process.argv.slice(4).filter((a) => !a.startsWith("--")).join(" ").trim();
4220
+ if (!name) {
4221
+ console.error(`${c.red}usage: kit memory forget <name>${c.reset}`);
4222
+ return false;
4223
+ }
4224
+ const db = openMemoryDb();
4225
+ const ok = removeThread(db, name);
4226
+ db.close();
4227
+ console.log(ok ? `${c.green}✓${c.reset} forgot ${name}` : `${c.dim}no copilot '${name}'${c.reset}`);
4228
+ return true;
4229
+ }
4230
+ if (subcommand === "pal") {
4231
+ const action = process.argv[4] && !process.argv[4].startsWith("--") ? process.argv[4] : "list";
4232
+ const db = openMemoryDb();
4233
+ try {
4234
+ if (action === "list") {
4235
+ const scope = hasFlag(process.argv, "--global")
4236
+ ? undefined
4237
+ : basename(getCurrentProjectRoot());
4238
+ const items = palList(db, { scope });
4239
+ if (jsonMode) {
4240
+ console.log(JSON.stringify(items));
4241
+ return true;
4242
+ }
4243
+ if (!items.length) {
4244
+ console.log(`${c.dim}no open action items${c.reset}`);
4245
+ return true;
4246
+ }
4247
+ console.log(`${c.bold}${items.length}${c.reset} open action item(s):`);
4248
+ for (const p of items) {
4249
+ const tag = p.kind === "auto" ? ` ${c.dim}· auto${c.reset}` : "";
4250
+ const scope = p.scope ? ` ${c.dim}[${p.scope}]${c.reset}` : "";
4251
+ console.log(` ${c.bold}${p.id}${c.reset} ${p.title}${scope}${tag}`);
4252
+ }
4253
+ return true;
4254
+ }
4255
+ if (action === "add") {
4256
+ const title = process.argv.slice(5).filter((a) => !a.startsWith("--")).join(" ").trim();
4257
+ if (!title) {
4258
+ console.error(`${c.red}usage: kit memory pal add <title> [--verify=<cmd>] [--scope=<s>]${c.reset}`);
4259
+ return false;
4260
+ }
4261
+ const id = palAdd(db, {
4262
+ title,
4263
+ verifyCmd: flagValue(process.argv, "--verify"),
4264
+ scope: flagValue(process.argv, "--scope") ?? basename(getCurrentProjectRoot()),
4265
+ });
4266
+ console.log(`${c.green}✓${c.reset} added ${c.bold}${id}${c.reset}`);
4267
+ return true;
4268
+ }
4269
+ if (action === "done") {
4270
+ const id = process.argv[5];
4271
+ if (!id) {
4272
+ console.error(`${c.red}usage: kit memory pal done <id>${c.reset}`);
4273
+ return false;
4274
+ }
4275
+ console.log(palDone(db, id)
4276
+ ? `${c.green}✓${c.reset} closed ${id}`
4277
+ : `${c.dim}${id} not found or already closed${c.reset}`);
4278
+ return true;
4279
+ }
4280
+ if (action === "snooze") {
4281
+ const id = process.argv[5];
4282
+ const days = Number(process.argv[6] ?? "7") || 7;
4283
+ if (!id) {
4284
+ console.error(`${c.red}usage: kit memory pal snooze <id> [days]${c.reset}`);
4285
+ return false;
4286
+ }
4287
+ console.log(palSnooze(db, id, days)
4288
+ ? `${c.green}✓${c.reset} snoozed ${id} for ${days}d`
4289
+ : `${c.dim}${id} not found${c.reset}`);
4290
+ return true;
4291
+ }
4292
+ if (action === "verify") {
4293
+ const r = palAutoVerify(db);
4294
+ console.log(`${c.dim}checked ${r.checked} · closed ${r.closed.length} · reopened ${r.reopened.length}${c.reset}`);
4295
+ return true;
4296
+ }
4297
+ if (action === "import") {
4298
+ const r = importLegacyLedger(db);
4299
+ console.log(`${c.green}✓${c.reset} imported ${r.imported} item(s) from the legacy ledger`);
4300
+ return true;
4301
+ }
4302
+ console.error(`${c.red}Unknown pal action: ${action}${c.reset}`);
4303
+ console.error("Use: kit memory pal [list|add|done|snooze|verify|import]");
4304
+ return false;
4305
+ }
4306
+ finally {
4307
+ db.close();
4308
+ }
4309
+ }
4310
+ console.error(`${c.red}Unknown memory subcommand: ${subcommand}${c.reset}`);
4311
+ console.error("Use: kit memory index | search <query> | stats | install | uninstall | pal");
4312
+ return false;
4313
+ }
3861
4314
  async function main() {
3862
4315
  const args = process.argv.slice(2);
3863
4316
  const command = args[0];
@@ -4032,6 +4485,9 @@ async function main() {
4032
4485
  case "team":
4033
4486
  ok = await cmdTeam();
4034
4487
  break;
4488
+ case "memory":
4489
+ ok = await cmdMemory();
4490
+ break;
4035
4491
  default: {
4036
4492
  console.error(`Unknown command: ${command}`);
4037
4493
  const { didYouMean } = await import("./utils/didYouMean.js");