@getrift/rift 0.1.0-beta.0 → 0.1.0-beta.10

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 (202) hide show
  1. package/README.md +72 -48
  2. package/dist/src/auth/keychain.d.ts +9 -0
  3. package/dist/src/auth/keychain.d.ts.map +1 -1
  4. package/dist/src/auth/keychain.js +37 -0
  5. package/dist/src/auth/keychain.js.map +1 -1
  6. package/dist/src/capture/recover-quarantine.d.ts +221 -0
  7. package/dist/src/capture/recover-quarantine.d.ts.map +1 -0
  8. package/dist/src/capture/recover-quarantine.js +453 -0
  9. package/dist/src/capture/recover-quarantine.js.map +1 -0
  10. package/dist/src/cli/commands/backfill.d.ts.map +1 -1
  11. package/dist/src/cli/commands/backfill.js +5 -2
  12. package/dist/src/cli/commands/backfill.js.map +1 -1
  13. package/dist/src/cli/commands/capture-recover.d.ts +12 -0
  14. package/dist/src/cli/commands/capture-recover.d.ts.map +1 -0
  15. package/dist/src/cli/commands/capture-recover.js +120 -0
  16. package/dist/src/cli/commands/capture-recover.js.map +1 -0
  17. package/dist/src/cli/commands/capture.d.ts.map +1 -1
  18. package/dist/src/cli/commands/capture.js +10 -4
  19. package/dist/src/cli/commands/capture.js.map +1 -1
  20. package/dist/src/cli/commands/feedback.d.ts.map +1 -1
  21. package/dist/src/cli/commands/feedback.js +6 -2
  22. package/dist/src/cli/commands/feedback.js.map +1 -1
  23. package/dist/src/cli/commands/hooks-install.d.ts +19 -0
  24. package/dist/src/cli/commands/hooks-install.d.ts.map +1 -0
  25. package/dist/src/cli/commands/hooks-install.js +103 -0
  26. package/dist/src/cli/commands/hooks-install.js.map +1 -0
  27. package/dist/src/cli/commands/mcp-install.js +5 -2
  28. package/dist/src/cli/commands/mcp-install.js.map +1 -1
  29. package/dist/src/cli/commands/onboard.d.ts +24 -0
  30. package/dist/src/cli/commands/onboard.d.ts.map +1 -1
  31. package/dist/src/cli/commands/onboard.js +200 -21
  32. package/dist/src/cli/commands/onboard.js.map +1 -1
  33. package/dist/src/cli/commands/rebuild.d.ts.map +1 -1
  34. package/dist/src/cli/commands/rebuild.js +6 -3
  35. package/dist/src/cli/commands/rebuild.js.map +1 -1
  36. package/dist/src/cli/commands/review.d.ts.map +1 -1
  37. package/dist/src/cli/commands/review.js +22 -7
  38. package/dist/src/cli/commands/review.js.map +1 -1
  39. package/dist/src/cli/commands/status.d.ts.map +1 -1
  40. package/dist/src/cli/commands/status.js +48 -2
  41. package/dist/src/cli/commands/status.js.map +1 -1
  42. package/dist/src/cli/commands/token-issue.d.ts.map +1 -1
  43. package/dist/src/cli/commands/token-issue.js +9 -1
  44. package/dist/src/cli/commands/token-issue.js.map +1 -1
  45. package/dist/src/cli/commands/triage.d.ts.map +1 -1
  46. package/dist/src/cli/commands/triage.js +7 -5
  47. package/dist/src/cli/commands/triage.js.map +1 -1
  48. package/dist/src/cli/commands/update.d.ts +26 -0
  49. package/dist/src/cli/commands/update.d.ts.map +1 -0
  50. package/dist/src/cli/commands/update.js +130 -0
  51. package/dist/src/cli/commands/update.js.map +1 -0
  52. package/dist/src/cli/default-config-path.d.ts +15 -0
  53. package/dist/src/cli/default-config-path.d.ts.map +1 -0
  54. package/dist/src/cli/default-config-path.js +27 -0
  55. package/dist/src/cli/default-config-path.js.map +1 -0
  56. package/dist/src/cli/hooks-writers/claude-code-policy-script.d.ts +25 -0
  57. package/dist/src/cli/hooks-writers/claude-code-policy-script.d.ts.map +1 -0
  58. package/dist/src/cli/hooks-writers/claude-code-policy-script.js +85 -0
  59. package/dist/src/cli/hooks-writers/claude-code-policy-script.js.map +1 -0
  60. package/dist/src/cli/hooks-writers/claude-code.d.ts +12 -0
  61. package/dist/src/cli/hooks-writers/claude-code.d.ts.map +1 -0
  62. package/dist/src/cli/hooks-writers/claude-code.js +228 -0
  63. package/dist/src/cli/hooks-writers/claude-code.js.map +1 -0
  64. package/dist/src/cli/hooks-writers/errors.d.ts +16 -0
  65. package/dist/src/cli/hooks-writers/errors.d.ts.map +1 -0
  66. package/dist/src/cli/hooks-writers/errors.js +24 -0
  67. package/dist/src/cli/hooks-writers/errors.js.map +1 -0
  68. package/dist/src/cli/hooks-writers/index.d.ts +13 -0
  69. package/dist/src/cli/hooks-writers/index.d.ts.map +1 -0
  70. package/dist/src/cli/hooks-writers/index.js +26 -0
  71. package/dist/src/cli/hooks-writers/index.js.map +1 -0
  72. package/dist/src/cli/hooks-writers/types.d.ts +27 -0
  73. package/dist/src/cli/hooks-writers/types.d.ts.map +1 -0
  74. package/dist/src/cli/hooks-writers/types.js +9 -0
  75. package/dist/src/cli/hooks-writers/types.js.map +1 -0
  76. package/dist/src/cli/http-client.d.ts +56 -1
  77. package/dist/src/cli/http-client.d.ts.map +1 -1
  78. package/dist/src/cli/http-client.js +140 -4
  79. package/dist/src/cli/http-client.js.map +1 -1
  80. package/dist/src/cli/index.d.ts.map +1 -1
  81. package/dist/src/cli/index.js +27 -6
  82. package/dist/src/cli/index.js.map +1 -1
  83. package/dist/src/cli/status/friend-header.d.ts.map +1 -1
  84. package/dist/src/cli/status/friend-header.js +117 -7
  85. package/dist/src/cli/status/friend-header.js.map +1 -1
  86. package/dist/src/ingestion/inbox-core/conversation-key.d.ts +2 -0
  87. package/dist/src/ingestion/inbox-core/conversation-key.d.ts.map +1 -0
  88. package/dist/src/ingestion/inbox-core/conversation-key.js +31 -0
  89. package/dist/src/ingestion/inbox-core/conversation-key.js.map +1 -0
  90. package/dist/src/ingestion/inbox-core/extensions.d.ts +3 -0
  91. package/dist/src/ingestion/inbox-core/extensions.d.ts.map +1 -0
  92. package/dist/src/ingestion/inbox-core/extensions.js +16 -0
  93. package/dist/src/ingestion/inbox-core/extensions.js.map +1 -0
  94. package/dist/src/ingestion/inbox-core/idempotency.d.ts +2 -0
  95. package/dist/src/ingestion/inbox-core/idempotency.d.ts.map +1 -0
  96. package/dist/src/ingestion/inbox-core/idempotency.js +22 -0
  97. package/dist/src/ingestion/inbox-core/idempotency.js.map +1 -0
  98. package/dist/src/ingestion/inbox-core/index.d.ts +19 -0
  99. package/dist/src/ingestion/inbox-core/index.d.ts.map +1 -0
  100. package/dist/src/ingestion/inbox-core/index.js +19 -0
  101. package/dist/src/ingestion/inbox-core/index.js.map +1 -0
  102. package/dist/src/ingestion/inbox-core/source-detection.d.ts +2 -0
  103. package/dist/src/ingestion/inbox-core/source-detection.d.ts.map +1 -0
  104. package/dist/src/ingestion/inbox-core/source-detection.js +23 -0
  105. package/dist/src/ingestion/inbox-core/source-detection.js.map +1 -0
  106. package/dist/src/ingestion/inbox-core/source-sniffer.d.ts +11 -0
  107. package/dist/src/ingestion/inbox-core/source-sniffer.d.ts.map +1 -0
  108. package/dist/src/ingestion/inbox-core/source-sniffer.js +69 -0
  109. package/dist/src/ingestion/inbox-core/source-sniffer.js.map +1 -0
  110. package/dist/src/ingestion/inbox-core/zip-sniffer.d.ts +70 -0
  111. package/dist/src/ingestion/inbox-core/zip-sniffer.d.ts.map +1 -0
  112. package/dist/src/ingestion/inbox-core/zip-sniffer.js +161 -0
  113. package/dist/src/ingestion/inbox-core/zip-sniffer.js.map +1 -0
  114. package/dist/src/ingestion/inbox-watcher.d.ts.map +1 -1
  115. package/dist/src/ingestion/inbox-watcher.js +34 -50
  116. package/dist/src/ingestion/inbox-watcher.js.map +1 -1
  117. package/dist/src/ingestion/indexer.d.ts +7 -0
  118. package/dist/src/ingestion/indexer.d.ts.map +1 -1
  119. package/dist/src/ingestion/indexer.js +36 -2
  120. package/dist/src/ingestion/indexer.js.map +1 -1
  121. package/dist/src/ingestion/metadata-extraction.d.ts +8 -5
  122. package/dist/src/ingestion/metadata-extraction.d.ts.map +1 -1
  123. package/dist/src/ingestion/metadata-extraction.js +24 -5
  124. package/dist/src/ingestion/metadata-extraction.js.map +1 -1
  125. package/dist/src/ingestion/skip-quarantine.d.ts +10 -0
  126. package/dist/src/ingestion/skip-quarantine.d.ts.map +1 -0
  127. package/dist/src/ingestion/skip-quarantine.js +35 -0
  128. package/dist/src/ingestion/skip-quarantine.js.map +1 -0
  129. package/dist/src/jobs/handlers/compact.d.ts.map +1 -1
  130. package/dist/src/jobs/handlers/compact.js +25 -4
  131. package/dist/src/jobs/handlers/compact.js.map +1 -1
  132. package/dist/src/jobs/handlers/ingest.d.ts.map +1 -1
  133. package/dist/src/jobs/handlers/ingest.js +49 -24
  134. package/dist/src/jobs/handlers/ingest.js.map +1 -1
  135. package/dist/src/jobs/handlers/reconcile.d.ts.map +1 -1
  136. package/dist/src/jobs/handlers/reconcile.js +30 -8
  137. package/dist/src/jobs/handlers/reconcile.js.map +1 -1
  138. package/dist/src/jobs/handlers/reindex.d.ts.map +1 -1
  139. package/dist/src/jobs/handlers/reindex.js +12 -2
  140. package/dist/src/jobs/handlers/reindex.js.map +1 -1
  141. package/dist/src/jobs/handlers/save.d.ts.map +1 -1
  142. package/dist/src/jobs/handlers/save.js +9 -2
  143. package/dist/src/jobs/handlers/save.js.map +1 -1
  144. package/dist/src/jobs/queue.d.ts +11 -0
  145. package/dist/src/jobs/queue.d.ts.map +1 -1
  146. package/dist/src/jobs/queue.js +18 -0
  147. package/dist/src/jobs/queue.js.map +1 -1
  148. package/dist/src/jobs/worker-entry.d.ts.map +1 -1
  149. package/dist/src/jobs/worker-entry.js +2 -0
  150. package/dist/src/jobs/worker-entry.js.map +1 -1
  151. package/dist/src/main.js +36 -4
  152. package/dist/src/main.js.map +1 -1
  153. package/dist/src/observability/embedding-events.d.ts +52 -0
  154. package/dist/src/observability/embedding-events.d.ts.map +1 -0
  155. package/dist/src/observability/embedding-events.js +149 -0
  156. package/dist/src/observability/embedding-events.js.map +1 -0
  157. package/dist/src/observability/index-events.d.ts +70 -0
  158. package/dist/src/observability/index-events.d.ts.map +1 -0
  159. package/dist/src/observability/index-events.js +148 -0
  160. package/dist/src/observability/index-events.js.map +1 -0
  161. package/dist/src/observability/tool-usage-stats.d.ts +7 -0
  162. package/dist/src/observability/tool-usage-stats.d.ts.map +1 -1
  163. package/dist/src/observability/tool-usage-stats.js +41 -5
  164. package/dist/src/observability/tool-usage-stats.js.map +1 -1
  165. package/dist/src/observability/tool-usage.d.ts +7 -7
  166. package/dist/src/observability/tool-usage.d.ts.map +1 -1
  167. package/dist/src/observability/tool-usage.js +78 -39
  168. package/dist/src/observability/tool-usage.js.map +1 -1
  169. package/dist/src/observability/version-check.d.ts +70 -0
  170. package/dist/src/observability/version-check.d.ts.map +1 -0
  171. package/dist/src/observability/version-check.js +197 -0
  172. package/dist/src/observability/version-check.js.map +1 -0
  173. package/dist/src/providers/ollama-embed.d.ts +2 -1
  174. package/dist/src/providers/ollama-embed.d.ts.map +1 -1
  175. package/dist/src/providers/ollama-embed.js +1 -0
  176. package/dist/src/providers/ollama-embed.js.map +1 -1
  177. package/dist/src/providers/openai-metadata-extraction.d.ts +3 -3
  178. package/dist/src/providers/openai-metadata-extraction.d.ts.map +1 -1
  179. package/dist/src/providers/openai-metadata-extraction.js +18 -3
  180. package/dist/src/providers/openai-metadata-extraction.js.map +1 -1
  181. package/dist/src/providers/stub.d.ts +2 -0
  182. package/dist/src/providers/stub.d.ts.map +1 -1
  183. package/dist/src/providers/stub.js +2 -0
  184. package/dist/src/providers/stub.js.map +1 -1
  185. package/dist/src/providers/types.d.ts +11 -0
  186. package/dist/src/providers/types.d.ts.map +1 -1
  187. package/dist/src/providers/voyage.d.ts +2 -1
  188. package/dist/src/providers/voyage.d.ts.map +1 -1
  189. package/dist/src/providers/voyage.js +1 -0
  190. package/dist/src/providers/voyage.js.map +1 -1
  191. package/dist/src/server/app.d.ts.map +1 -1
  192. package/dist/src/server/app.js +67 -1
  193. package/dist/src/server/app.js.map +1 -1
  194. package/dist/src/server/routes/friend-status.d.ts +202 -3
  195. package/dist/src/server/routes/friend-status.d.ts.map +1 -1
  196. package/dist/src/server/routes/friend-status.js +290 -7
  197. package/dist/src/server/routes/friend-status.js.map +1 -1
  198. package/dist/src/storage/rebuild.d.ts +14 -1
  199. package/dist/src/storage/rebuild.d.ts.map +1 -1
  200. package/dist/src/storage/rebuild.js +160 -34
  201. package/dist/src/storage/rebuild.js.map +1 -1
  202. package/package.json +1 -1
package/README.md CHANGED
@@ -1,6 +1,8 @@
1
1
  # Rift
2
2
 
3
- **Mémoire personnelle locale pour tes IA.** Rift capture en local tes sessions Claude Code et Codex CLI, les triage, les indexe, et expose une mémoire interrogeable via MCP — pour que Claude Desktop, Claude Code, Codex et Cursor retrouvent ce que tu leur as déjà dit.
3
+ **Mémoire personnelle locale pour tes IA.**
4
+
5
+ Rift capture en local tes sessions Claude Code et Codex CLI, les triage, les indexe, et expose une mémoire interrogeable via MCP — pour que Claude Desktop, Claude Code, Codex et Cursor retrouvent ce que tu leur as déjà dit.
4
6
 
5
7
  Tout reste sur ta machine, sauf les *embeddings* (envoyés à Voyage AI sous une clé qui t'appartient).
6
8
 
@@ -11,33 +13,42 @@ Tout reste sur ta machine, sauf les *embeddings* (envoyés à Voyage AI sous une
11
13
  > **La capture démarre maintenant.**
12
14
  > Au premier lancement, Rift pose un *watermark* sur tes sessions Claude Code et Codex CLI existantes : **elles ne sont pas réingérées rétroactivement**. Seules les sessions à partir de l'installation comptent.
13
15
  >
14
- > Si tu veux que tes conversations historiques (sessions Claude Code locales antérieures, exports `.zip` de claude.ai / chatgpt.com / gemini.google.com / grok.com) soient *aussi* indexées, c'est une **prestation séparée** — voir la section [Backfill historique (payant)](#backfill-historique-payant) plus bas.
16
+ > Si tu veux que ton corpus passé soit *aussi* indexé (sessions Claude Code locales antérieures + exports `.zip` claude.ai / chatgpt.com / gemini.google.com / grok.com), c'est une **prestation séparée** — voir [Backfill historique (payant)](#backfill-historique-payant) plus bas.
15
17
 
16
18
  ---
17
19
 
18
20
  ## Prérequis
19
21
 
20
- - macOS 12.3 ou plus récent.
21
- - Node.js 20.19+ (vérifie avec `node --version`).
22
- - npm (livré avec Node).
23
- - [Codex CLI](https://github.com/openai/codex) installé et authentifié (`codex login`). Rift fait passer le triage et l'extraction de métadonnées par ton CLI Codex local — pas besoin de clé OpenAI séparée.
24
- - Une clé Voyage AI (création gratuite sur [voyageai.com](https://www.voyageai.com/)). Sert *uniquement* aux embeddings ; chaque user a sa propre clé.
22
+ À avoir sur ton Mac avant l'install :
23
+
24
+ | | Quoi | Comment |
25
+ |---|---|---|
26
+ | 1 | **macOS 12.3+** | Pomme propos de ce Mac" |
27
+ | 2 | **Node.js 20.19+** | `node --version` dans Ghostty. Si manquant : [nodejs.org](https://nodejs.org/en/download) (LTS) |
28
+ | 3 | **Codex CLI authentifié** | `codex --version`, puis `codex login` si tu n'as jamais loggé |
29
+ | 4 | **Une clé Voyage AI** | Je te l'envoie en DM (c'est un projet dédié sur mon compte Voyage). Range-la dans 1Password, on la collera pendant l'onboard |
30
+
31
+ Codex CLI fait tourner le triage et l'extraction de métadonnées — pas besoin de clé OpenAI séparée. La clé Voyage sert *uniquement* aux embeddings.
25
32
 
26
33
  ## Installation (1 commande)
27
34
 
35
+ Dans Ghostty, tape :
36
+
28
37
  ```sh
29
38
  curl -fsSL https://getrift.dev/install | bash
30
39
  ```
31
40
 
32
41
  Ce que fait le script :
33
42
 
34
- - vérifie les prérequis (macOS, Node, codex CLI…),
35
- - installe globalement le paquet npm `@getrift/rift@beta`,
36
- - crée les dossiers de données (`~/Library/Application Support/Rift/`),
37
- - pose un agent launchd (`com.getrift.daemon`) qui démarre Rift à la session,
38
- - attend que `/health` réponde sur `127.0.0.1:3577`.
43
+ - vérifie les prérequis ;
44
+ - installe le paquet npm `@getrift/rift@beta` globalement ;
45
+ - crée tes dossiers de données sous `~/Library/Application Support/Rift/` ;
46
+ - pose un agent launchd qui lance Rift à chaque session ;
47
+ - attend que le daemon réponde sur le port local.
48
+
49
+ **Si on te demande ton mot de passe Mac (`Password:`)**, c'est normal — il faut écrire dans un dossier système pour rendre la commande `rift` disponible partout. Tape ton mot de passe (rien ne s'affiche pendant la frappe — c'est normal sur Unix), `Entrée`.
39
50
 
40
- Si l'installation échoue, le script restaure ton état précédent et imprime un identifiant de backup pour le diagnostic.
51
+ Si ça plante en cours de route, le script restaure ton état précédent et imprime un identifiant de backup. Tu peux me l'envoyer.
41
52
 
42
53
  ## Premier paramétrage
43
54
 
@@ -45,15 +56,20 @@ Si l'installation échoue, le script restaure ton état précédent et imprime u
45
56
  rift onboard
46
57
  ```
47
58
 
48
- L'assistant va :
59
+ L'assistant va te poser **5 ou 6 questions**, dans cet ordre :
49
60
 
50
- 1. te demander ta clé Voyage AI (collée depuis ton 1Password elle ne s'affiche pas) ;
51
- 2. valider la clé en faisant un embed test ;
52
- 3. faire un *preflight* sur ton Codex CLI (vérifie qu'il est authentifié) ;
53
- 4. lancer une première passe de capture (les sessions existantes sont watermarkées, pas ingérées) ;
54
- 5. proposer de brancher Rift comme serveur MCP dans Claude Desktop / Claude Code / Codex / Cursor.
61
+ 1. **La clé Voyage que je t'ai envoyée.** Colle-la depuis 1Password. Elle ne s'affiche pas pendant la frappe.
62
+ 2. **Validation.** Rift fait un embed test pour vérifier la clé.
63
+ 3. **Vérifie que Codex est authentifié.** Si ça échoue, lance `codex login` et relance `rift onboard`. Ou demande à Claude.
64
+ 4. **Première passe de capture.** Watermarke tes sessions existantes (ne les ingère pas).
65
+ 5. **Branchement MCP.** Te demande dans quels clients tu veux Rift comme serveur MCP Claude Desktop, Claude Code, Codex, Cursor. **Réponds `oui` à tous ceux que tu utilises.**
66
+ 6. **Garde-fou Claude Code (optionnel).** Si Claude Code est détecté, l'assistant te propose d'installer un *PreToolUse hook* qui force la récupération Rift à commencer par `rift_context_pack` (compact, budgété) avant tout `rift_search`. Évite que l'agent crame ton contexte avec des recherches larges. **Réponds `y`** si tu utilises Claude Code. Désactivable à tout moment via `RIFT_POLICY_DISABLED=1` ou en supprimant l'entrée de `~/.claude/settings.json`.
55
67
 
56
- Tu peux importer un export ChatGPT ou Claude.ai pendant l'onboarding (optionnel voir la section backfill).
68
+ À la fin, l'assistant te dit "Onboarding terminé" et te suggère ta première recherche.
69
+
70
+ > Si tu veux importer un export ChatGPT ou Claude.ai, l'assistant te le proposera. **Pour la beta : décline** — c'est de la backfill historique, voir la section payante plus bas.
71
+
72
+ > Si tu as déjà fait `rift onboard` avant cette beta, tu peux poser le hook après coup avec `rift hooks install --client claude-code`.
57
73
 
58
74
  ## Vérifier que ça tourne
59
75
 
@@ -61,10 +77,10 @@ Tu peux importer un export ChatGPT ou Claude.ai pendant l'onboarding (optionnel
61
77
  rift status
62
78
  ```
63
79
 
64
- Tu dois voir quelque chose comme :
80
+ Tu dois voir une sortie qui ressemble à ça (en anglais — le binaire est anglophone) :
65
81
 
66
82
  ```
67
- Rift v0.1.0-beta.0 · daemon: running (uptime 5m) · port 3577 ✓
83
+ Rift v0.1.0-beta.2 · daemon: running (uptime 5m) · port 3577 ✓
68
84
  Voyage: key valid (last 4 …xxxx, last embed 2m ago)
69
85
  Codex CLI: authed ✓
70
86
  Capture: last run 8m ago — 3 saved · 0 review · 0 failed
@@ -74,56 +90,64 @@ Feedback: relay off (local JSONL only)
74
90
  Next: nothing broken. Try rift search "<topic>".
75
91
  ```
76
92
 
93
+ Les ✓ partout = tout va bien. Si une ligne montre ✗ ou un *Next:* différent de "nothing broken", suis l'instruction qu'il imprime.
94
+
77
95
  ## Usage quotidien
78
96
 
79
97
  - **Rien à faire.** Le daemon tourne en tâche de fond, ingère tes nouvelles sessions Claude Code / Codex CLI toutes les heures, et expose la mémoire via MCP.
80
- - **Rechercher en CLI :** `rift search "comment on a structuré le pricing"`
81
- - **Forcer une passe maintenant :** `rift capture`
82
- - **Ouvrir une conversation MCP** (Claude Desktop, Cursor, etc.) — Rift apparaît dans la liste des outils. Tape *"recall what we decided about X"* et l'agent appellera `rift_search` / `rift_context_pack` automatiquement.
98
+ - **Tester depuis Claude Desktop ou Cursor.** Ouvre une conversation, demande par exemple : *"Rappelle-toi ce qu'on a discuté sur le pricing la semaine dernière, va voir dans Rift"* — Claude/Cursor appellera `rift_search` ou `rift_context_pack` automatiquement et t'affichera ce qu'il trouve.
99
+ - **Rechercher en ligne de commande :** `rift search "comment on a structuré le pricing"`
100
+ - **Forcer une passe de capture maintenant :** `rift capture`
83
101
 
84
102
  ## Vie privée
85
103
 
86
- - **Local par défaut.** Tes sessions, métadonnées, et la base vectorielle (LanceDB) restent sur ta machine.
87
- - **Voyage AI.** Seuls les *snippets* envoyés pour embedding partent vers Voyage. Ta clé t'appartient ; tu peux la révoquer côté Voyage à tout moment.
88
- - **Aucune télémétrie de contenu.** Rift n'envoie rien chez moi, sauf si tu actives explicitement le *feedback relay* (`rift feedback --kind=...`) et même là, c'est uniquement la note que tu écris, pas tes conversations.
89
- - **Désinstallation propre :** `rift uninstall` enlève le daemon, les entrées MCP, et (avec `--purge-data`) les données.
104
+ - **Local par défaut.** Tes sessions, métadonnées et la base vectorielle (LanceDB) restent sur ta machine.
105
+ - **Voyage AI.** Seuls les *snippets* envoyés pour embedding partent vers Voyage. Ta clé t'appartient tu peux la révoquer côté Voyage à tout moment, ce qui coupe l'envoi.
106
+ - **Aucune télémétrie de contenu.** Rift ne m'envoie rien, sauf si tu actives explicitement le *feedback relay* (`rift feedback --kind=...`). Et même là, c'est juste la note que tu écris, pas tes conversations.
107
+ - **Désinstallation propre.** `rift uninstall` enlève le daemon et les entrées MCP. `rift uninstall --purge-data` supprime aussi les données locales.
90
108
 
91
- Le contrat de vie privée détaillé arrive avec le paquet : `docs/feedback/PRIVACY.md`.
109
+ Le contrat de vie privée détaillé est dans `docs/feedback/PRIVACY.md` (livré avec le paquet).
92
110
 
93
111
  ## Backfill historique (payant)
94
112
 
95
- Le watermark protège ton historique pré-installation : Rift ne va pas inonder ta mémoire avec deux ans de conversations dont la moitié sont du débogage jetable. Mais si tu veux *vraiment* que ton corpus passé soit indexé — sessions Claude Code locales antérieures, exports `.zip` claude.ai, chatgpt.com, gemini.google.com, grok.com — c'est faisable, mais ça représente plusieurs heures de tri et de QA pour calibrer les filtres à ton usage.
113
+ Le watermark protège ton historique pré-installation : Rift ne va pas inonder ta mémoire avec deux ans de conversations dont la moitié sont du débogage jetable.
114
+
115
+ Mais si tu veux *vraiment* que ton corpus passé soit indexé, c'est faisable. Ce que ça représente côté boulot :
96
116
 
97
- **Tarif : 500 HT, prestation ponctuelle.**
117
+ - import + audit de tes exports web (claude.ai, chatgpt.com, gemini.google.com, grok.com) ;
118
+ - déwatermark ciblé des sessions Claude Code / Codex CLI qui valent le coup (filtré par date, projet, heuristique) ;
119
+ - tuning des règles de triage à ton usage réel sur ~1 semaine ;
120
+ - session de 60 min en visio pour valider ce qui est entré dans la mémoire et ce qui a été écarté.
98
121
 
99
- Ce que ça inclut :
122
+ **Tarifs (HT, prestation ponctuelle) :**
100
123
 
101
- - import de tes exports web (`rift import <export.zip>`) avec audit du contenu,
102
- - déwatermark ciblé des sessions Claude Code / Codex CLI valant le coup (filtré par date, projet, ou heuristique),
103
- - tuning des règles de triage à ton usage réel pendant une semaine,
104
- - une session de 60 min en visio pour valider ce qui est dans la mémoire vs ce qui a été écarté.
124
+ | Profondeur | Tarif |
125
+ |---|---|
126
+ | 1 mois de données | **1 000 €** |
127
+ | 6 mois de données | **3 000 €** |
128
+ | 1 an de données | **5 000 €** |
105
129
 
106
- Pour déclencher : il me faut un bon de commande / facture HT signée par ton boss. Une fois validé, on cale 1h de kickoff.
130
+ Pour déclencher : il me faut un **bon de commande / facture HT signée** par ton boss. Une fois validé, on cale 1h de kickoff. Tu m'envoies un message quand tu es prêt.
107
131
 
108
132
  ## En cas de pépin
109
133
 
110
- - **Le daemon ne répond pas :**
111
- ```sh
112
- launchctl kickstart -k gui/$UID/com.getrift.daemon
113
- ```
114
- - **Codex CLI auth périmé :** `codex login`
115
- - **Clé Voyage révoquée / changée :** `rift onboard --reconfigure-voyage`
116
- - **MCP pas branché dans un client :** `rift mcp install --client=claude-desktop` (ou `codex`, `claude-code`, `cursor`)
117
- - **Donner du feedback (utile) :** `rift feedback --kind=broke --with-status` — capture l'état du daemon et l'envoie en local (et en relay si tu l'as activé pendant l'onboard).
134
+ | Symptôme | Solution |
135
+ |---|---|
136
+ | `rift status` dit "daemon: unreachable" | `launchctl kickstart -k gui/$UID/com.getrift.daemon` |
137
+ | Codex CLI auth périmé | `codex login` |
138
+ | Clé Voyage révoquée ou changée | `rift onboard --reconfigure-voyage` |
139
+ | Rift n'apparaît pas dans Claude Desktop / Cursor | `rift mcp install --client=claude-desktop` (remplace par `codex`, `claude-code`, `cursor` selon le besoin) |
140
+ | Le garde-fou Claude Code te bloque trop souvent | `export RIFT_POLICY_DISABLED=1` dans le shell (désactive le hook le temps de la session), ou enlève l'entrée `rift-policy.mjs` de `~/.claude/settings.json` |
141
+ | Truc bizarre, tu veux me prévenir | `rift feedback --kind=broke --with-status` — capture l'état actuel et envoie le tout en local (+ chez moi si tu as activé le relay pendant l'onboard) |
118
142
 
119
143
  ## Désinstaller
120
144
 
121
145
  ```sh
122
- rift uninstall # enlève daemon + MCP
146
+ rift uninstall # enlève daemon + entrées MCP
123
147
  rift uninstall --purge-data # + supprime les données locales
124
148
  ```
125
149
 
126
- La clé Voyage reste dans `~/.rift.env` même après `--purge-data` — la commande affiche le `rm` exact pour la supprimer (ou tu peux la révoquer côté Voyage).
150
+ La clé Voyage reste dans `~/.rift.env` même après `--purge-data` — la commande affiche le `rm` exact pour la nettoyer, ou tu peux la révoquer côté Voyage.
127
151
 
128
152
  ## License
129
153
 
@@ -1,3 +1,12 @@
1
+ /**
2
+ * Typed error thrown when the macOS Keychain is locked. Caught by
3
+ * `readToken` (CLI read path) and `ensureToken` (CLI write path) to
4
+ * surface the `security unlock-keychain ~/Library/Keychains/login.keychain-db`
5
+ * recovery copy instead of the raw `security` CLI failure.
6
+ */
7
+ export declare class LockedKeychainError extends Error {
8
+ constructor(message: string);
9
+ }
1
10
  /**
2
11
  * Low-level Keychain operations.
3
12
  *
@@ -1 +1 @@
1
- {"version":3,"file":"keychain.d.ts","sourceRoot":"","sources":["../../../src/auth/keychain.ts"],"names":[],"mappings":"AAaA;;;;;GAKG;AACH,MAAM,WAAW,WAAW;IAC1B,IAAI,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAAC;IAC9C,KAAK,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IACrD,MAAM,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IACvC,iBAAiB,IAAI,OAAO,CAAC,MAAM,CAAC,CAAC;CACtC;AAgBD;;;;;;;;;;GAUG;AACH,eAAO,MAAM,WAAW,EAAE,WA0EzB,CAAC"}
1
+ {"version":3,"file":"keychain.d.ts","sourceRoot":"","sources":["../../../src/auth/keychain.ts"],"names":[],"mappings":"AAuBA;;;;;GAKG;AACH,qBAAa,mBAAoB,SAAQ,KAAK;gBAChC,OAAO,EAAE,MAAM;CAI5B;AAWD;;;;;GAKG;AACH,MAAM,WAAW,WAAW;IAC1B,IAAI,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAAC;IAC9C,KAAK,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IACrD,MAAM,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IACvC,iBAAiB,IAAI,OAAO,CAAC,MAAM,CAAC,CAAC;CACtC;AAgBD;;;;;;;;;;GAUG;AACH,eAAO,MAAM,WAAW,EAAE,WAoFzB,CAAC"}
@@ -8,6 +8,37 @@ const SERVICE = "com.getrift.daemon";
8
8
  * All other non-zero exits are real failures that must propagate.
9
9
  */
10
10
  const ERR_SEC_ITEM_NOT_FOUND = 44;
11
+ /**
12
+ * Substring emitted by the macOS `security` CLI when the login keychain
13
+ * is locked (typical over SSH or in CI without a GUI session). Locked-
14
+ * keychain failures are a distinct *class* — they're recoverable with
15
+ * `security unlock-keychain`, unlike permission denials or missing
16
+ * binaries — so callers benefit from a typed error to render a precise
17
+ * recovery hint.
18
+ */
19
+ const LOCKED_KEYCHAIN_SUBSTRING = "User interaction is not allowed";
20
+ /**
21
+ * Typed error thrown when the macOS Keychain is locked. Caught by
22
+ * `readToken` (CLI read path) and `ensureToken` (CLI write path) to
23
+ * surface the `security unlock-keychain ~/Library/Keychains/login.keychain-db`
24
+ * recovery copy instead of the raw `security` CLI failure.
25
+ */
26
+ export class LockedKeychainError extends Error {
27
+ constructor(message) {
28
+ super(message);
29
+ this.name = "LockedKeychainError";
30
+ }
31
+ }
32
+ function isLockedKeychain(err) {
33
+ if (!(err instanceof Error))
34
+ return false;
35
+ if (err.message.includes(LOCKED_KEYCHAIN_SUBSTRING))
36
+ return true;
37
+ // The substring lands on stderr, which `execFile` attaches to the
38
+ // error object as `stderr`. Belt + braces: check both surfaces.
39
+ const stderr = err.stderr;
40
+ return typeof stderr === "string" && stderr.includes(LOCKED_KEYCHAIN_SUBSTRING);
41
+ }
11
42
  function accountName(version) {
12
43
  return `AUTH_TOKEN_v${version}`;
13
44
  }
@@ -48,6 +79,9 @@ export const macKeychain = {
48
79
  catch (err) {
49
80
  if (isItemNotFound(err))
50
81
  return null;
82
+ if (isLockedKeychain(err)) {
83
+ throw new LockedKeychainError(err instanceof Error ? err.message : String(err));
84
+ }
51
85
  throw err;
52
86
  }
53
87
  },
@@ -106,6 +140,9 @@ export const macKeychain = {
106
140
  catch (err) {
107
141
  if (isItemNotFound(err))
108
142
  return 0; // No tokens issued yet.
143
+ if (isLockedKeychain(err)) {
144
+ throw new LockedKeychainError(err instanceof Error ? err.message : String(err));
145
+ }
109
146
  throw err;
110
147
  }
111
148
  },
@@ -1 +1 @@
1
- {"version":3,"file":"keychain.js","sourceRoot":"","sources":["../../../src/auth/keychain.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"keychain.js","sourceRoot":"","sources":["../../../src/auth/keychain.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,221 @@
1
+ import { z } from "zod";
2
+ /**
3
+ * Default byte budget per summarization chunk. Sits well below the
4
+ * 8 MB auto-capture threshold so the Codex CLI input never hits the
5
+ * same wall that caused the quarantine in the first place. 512 KB
6
+ * fits comfortably in stdin and leaves room for the prompt.
7
+ */
8
+ export declare const DEFAULT_RECOVERY_CHUNK_BYTES: number;
9
+ /** Minimum chunk size we allow — anything smaller risks turn-truncation. */
10
+ export declare const MIN_RECOVERY_CHUNK_BYTES: number;
11
+ export declare const PROVENANCE_TOPIC = "recovered_oversized_quarantine";
12
+ declare const QuarantineRecordSchema: z.ZodObject<{
13
+ source: z.ZodLiteral<"codex_cli">;
14
+ session_id: z.ZodString;
15
+ session_path: z.ZodString;
16
+ size_bytes: z.ZodNumber;
17
+ threshold_bytes: z.ZodNumber;
18
+ fingerprint: z.ZodString;
19
+ reason: z.ZodString;
20
+ quarantined_at: z.ZodString;
21
+ }, "strip", z.ZodTypeAny, {
22
+ source: "codex_cli";
23
+ size_bytes: number;
24
+ quarantined_at: string;
25
+ reason: string;
26
+ fingerprint: string;
27
+ session_id: string;
28
+ session_path: string;
29
+ threshold_bytes: number;
30
+ }, {
31
+ source: "codex_cli";
32
+ size_bytes: number;
33
+ quarantined_at: string;
34
+ reason: string;
35
+ fingerprint: string;
36
+ session_id: string;
37
+ session_path: string;
38
+ threshold_bytes: number;
39
+ }>;
40
+ export type QuarantineRecord = z.infer<typeof QuarantineRecordSchema> & {
41
+ /** Path to the quarantine record file itself. */
42
+ record_path: string;
43
+ };
44
+ declare const RecoveredEntrySchema: z.ZodObject<{
45
+ recovered_at: z.ZodString;
46
+ fingerprint: z.ZodString;
47
+ idempotency_key: z.ZodString;
48
+ chunks: z.ZodNumber;
49
+ session_path: z.ZodString;
50
+ size_bytes: z.ZodNumber;
51
+ }, "strip", z.ZodTypeAny, {
52
+ idempotency_key: string;
53
+ size_bytes: number;
54
+ fingerprint: string;
55
+ session_path: string;
56
+ recovered_at: string;
57
+ chunks: number;
58
+ }, {
59
+ idempotency_key: string;
60
+ size_bytes: number;
61
+ fingerprint: string;
62
+ session_path: string;
63
+ recovered_at: string;
64
+ chunks: number;
65
+ }>;
66
+ export type RecoveredEntry = z.infer<typeof RecoveredEntrySchema>;
67
+ declare const RecoveryStateSchema: z.ZodObject<{
68
+ recovered: z.ZodDefault<z.ZodRecord<z.ZodString, z.ZodObject<{
69
+ recovered_at: z.ZodString;
70
+ fingerprint: z.ZodString;
71
+ idempotency_key: z.ZodString;
72
+ chunks: z.ZodNumber;
73
+ session_path: z.ZodString;
74
+ size_bytes: z.ZodNumber;
75
+ }, "strip", z.ZodTypeAny, {
76
+ idempotency_key: string;
77
+ size_bytes: number;
78
+ fingerprint: string;
79
+ session_path: string;
80
+ recovered_at: string;
81
+ chunks: number;
82
+ }, {
83
+ idempotency_key: string;
84
+ size_bytes: number;
85
+ fingerprint: string;
86
+ session_path: string;
87
+ recovered_at: string;
88
+ chunks: number;
89
+ }>>>;
90
+ }, "strip", z.ZodTypeAny, {
91
+ recovered: Record<string, {
92
+ idempotency_key: string;
93
+ size_bytes: number;
94
+ fingerprint: string;
95
+ session_path: string;
96
+ recovered_at: string;
97
+ chunks: number;
98
+ }>;
99
+ }, {
100
+ recovered?: Record<string, {
101
+ idempotency_key: string;
102
+ size_bytes: number;
103
+ fingerprint: string;
104
+ session_path: string;
105
+ recovered_at: string;
106
+ chunks: number;
107
+ }> | undefined;
108
+ }>;
109
+ export type RecoveryState = z.infer<typeof RecoveryStateSchema>;
110
+ export type SkipReason = "already_recovered" | "missing_source" | "empty_session" | "summarize_failed" | "save_failed";
111
+ export interface RecoveryResult {
112
+ session_id: string;
113
+ fingerprint: string;
114
+ outcome: "recovered" | "skipped" | "failed";
115
+ reason?: SkipReason;
116
+ chunks?: number;
117
+ idempotency_key?: string;
118
+ error?: string;
119
+ }
120
+ export interface RecoveryReport {
121
+ total_records: number;
122
+ total_sessions: number;
123
+ recovered: number;
124
+ skipped: number;
125
+ missing_source: number;
126
+ failed: number;
127
+ results: RecoveryResult[];
128
+ }
129
+ export interface ChunkSummary {
130
+ summary: string;
131
+ topics: string[];
132
+ }
133
+ export interface ChunkSummarizer {
134
+ summarize(opts: {
135
+ chunk: string;
136
+ index: number;
137
+ total: number;
138
+ sessionId: string;
139
+ }): Promise<ChunkSummary>;
140
+ }
141
+ export interface RecoverQuarantineSaveInput {
142
+ sessionId: string;
143
+ fingerprint: string;
144
+ sessionPath: string;
145
+ sizeBytes: number;
146
+ thresholdBytes: number;
147
+ chunks: ChunkSummary[];
148
+ content: string;
149
+ summary: string;
150
+ topics: string[];
151
+ idempotencyKey: string;
152
+ }
153
+ export interface RecoverQuarantineDeps {
154
+ dataDir: string;
155
+ /** Override quarantine directory (defaults to `<dataDir>/quarantine`). */
156
+ quarantineDir?: string;
157
+ /**
158
+ * Chunk byte budget. Clamped to at least `MIN_RECOVERY_CHUNK_BYTES`.
159
+ * Defaults to `DEFAULT_RECOVERY_CHUNK_BYTES`.
160
+ */
161
+ chunkBytes?: number;
162
+ summarizer: ChunkSummarizer;
163
+ /**
164
+ * Save the recovered artifact through the daemon. Implementations
165
+ * should be idempotent w.r.t. `idempotencyKey` so a partial failure
166
+ * mid-batch doesn't double-insert when the recovery state write
167
+ * happened after a successful save (it doesn't here — state is
168
+ * persisted on success only — but the contract still holds).
169
+ */
170
+ saveFn: (input: RecoverQuarantineSaveInput) => Promise<void>;
171
+ /**
172
+ * Dry-run: discover, parse, and chunk only. The Codex CLI summarizer
173
+ * is NOT invoked (each chunk call costs real time/quota on real data),
174
+ * and saving + state persistence are skipped. Use this to plan a real
175
+ * run cheaply; the report's `chunks` count reflects what a real run
176
+ * would summarize.
177
+ */
178
+ dryRun?: boolean;
179
+ /** Optional clock for deterministic tests. */
180
+ now?: () => Date;
181
+ }
182
+ /**
183
+ * Read all Codex-CLI quarantine records under the configured directory.
184
+ * Malformed JSON / schema mismatches are skipped silently with a stderr
185
+ * note so one corrupt record can never block the rest of the batch.
186
+ */
187
+ export declare function loadQuarantineRecords(deps: Pick<RecoverQuarantineDeps, "dataDir" | "quarantineDir">): QuarantineRecord[];
188
+ /**
189
+ * Collapse multiple quarantine records that point at the same session
190
+ * (the same jsonl that hit the threshold at different sizes as it grew)
191
+ * down to one: the latest `quarantined_at`. The collapsed record still
192
+ * carries the recovery key, so reruns can mark every related record as
193
+ * already-recovered for that session.
194
+ */
195
+ export declare function groupBySession(records: QuarantineRecord[]): QuarantineRecord[];
196
+ export declare function loadRecoveryState(deps: Pick<RecoverQuarantineDeps, "dataDir">): RecoveryState;
197
+ export declare function saveRecoveryState(deps: Pick<RecoverQuarantineDeps, "dataDir">, state: RecoveryState): Promise<void>;
198
+ /**
199
+ * Split parser-emitted Codex content into byte-budgeted chunks at turn
200
+ * boundaries. The parser format is `User: …\n\nAssistant: …\n\n…`, so
201
+ * we split on `\n\n` immediately preceding a `User:` or `Assistant:`
202
+ * marker. A single turn larger than `maxBytes` becomes its own chunk
203
+ * (oversized but preserved — better than dropping signal).
204
+ */
205
+ export declare function chunkByTurns(content: string, maxBytes: number): string[];
206
+ export declare function runRecoverQuarantine(deps: RecoverQuarantineDeps): Promise<RecoveryReport>;
207
+ export declare class CodexCliChunkSummarizer implements ChunkSummarizer {
208
+ private readonly options;
209
+ constructor(options?: {
210
+ cwd?: string;
211
+ timeoutMs?: number;
212
+ });
213
+ summarize(opts: {
214
+ chunk: string;
215
+ index: number;
216
+ total: number;
217
+ sessionId: string;
218
+ }): Promise<ChunkSummary>;
219
+ }
220
+ export {};
221
+ //# sourceMappingURL=recover-quarantine.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"recover-quarantine.d.ts","sourceRoot":"","sources":["../../../src/capture/recover-quarantine.ts"],"names":[],"mappings":"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"}