@zenalexa/unicli 0.224.1 → 0.225.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (256) hide show
  1. package/AGENTS.md +5 -5
  2. package/README.md +92 -92
  3. package/README.zh-CN.md +69 -69
  4. package/dist/adapters/_electron/desktop-shared.d.ts.map +1 -1
  5. package/dist/adapters/_electron/desktop-shared.js +2 -1
  6. package/dist/adapters/_electron/desktop-shared.js.map +1 -1
  7. package/dist/adapters/_electron/shared.d.ts +6 -0
  8. package/dist/adapters/_electron/shared.d.ts.map +1 -1
  9. package/dist/adapters/_electron/shared.js +9 -0
  10. package/dist/adapters/_electron/shared.js.map +1 -1
  11. package/dist/adapters/antigravity/extra.js +4 -1
  12. package/dist/adapters/antigravity/extra.js.map +1 -1
  13. package/dist/adapters/chatwise/extra.js +4 -1
  14. package/dist/adapters/chatwise/extra.js.map +1 -1
  15. package/dist/adapters/codex/codex.js +3 -1
  16. package/dist/adapters/codex/codex.js.map +1 -1
  17. package/dist/adapters/codex/extra.js +4 -1
  18. package/dist/adapters/codex/extra.js.map +1 -1
  19. package/dist/adapters/codex/projects.d.ts.map +1 -1
  20. package/dist/adapters/codex/projects.js +3 -1
  21. package/dist/adapters/codex/projects.js.map +1 -1
  22. package/dist/adapters/cursor/cursor.js +6 -1
  23. package/dist/adapters/cursor/cursor.js.map +1 -1
  24. package/dist/adapters/discord-app/discord-app.js +10 -1
  25. package/dist/adapters/discord-app/discord-app.js.map +1 -1
  26. package/dist/adapters/notion-app/notion-app.js +11 -1
  27. package/dist/adapters/notion-app/notion-app.js.map +1 -1
  28. package/dist/adapters/spotify/api.js +36 -6
  29. package/dist/adapters/spotify/api.js.map +1 -1
  30. package/dist/adapters/xiaohongshu/browser-state.d.ts +2 -1
  31. package/dist/adapters/xiaohongshu/browser-state.d.ts.map +1 -1
  32. package/dist/adapters/xiaohongshu/browser-state.js +56 -2
  33. package/dist/adapters/xiaohongshu/browser-state.js.map +1 -1
  34. package/dist/adapters/xiaohongshu/feed.d.ts +24 -0
  35. package/dist/adapters/xiaohongshu/feed.d.ts.map +1 -0
  36. package/dist/adapters/xiaohongshu/feed.js +82 -0
  37. package/dist/adapters/xiaohongshu/feed.js.map +1 -0
  38. package/dist/adapters/xueqiu/extra.js +2 -2
  39. package/dist/adapters/xueqiu/extra.js.map +1 -1
  40. package/dist/browser/cdp-client.d.ts +15 -1
  41. package/dist/browser/cdp-client.d.ts.map +1 -1
  42. package/dist/browser/cdp-client.js +45 -16
  43. package/dist/browser/cdp-client.js.map +1 -1
  44. package/dist/browser/daemon.js +29 -7
  45. package/dist/browser/daemon.js.map +1 -1
  46. package/dist/browser/launcher.d.ts.map +1 -1
  47. package/dist/browser/launcher.js +22 -8
  48. package/dist/browser/launcher.js.map +1 -1
  49. package/dist/browser/local-profiles.d.ts +2 -0
  50. package/dist/browser/local-profiles.d.ts.map +1 -1
  51. package/dist/browser/local-profiles.js +42 -2
  52. package/dist/browser/local-profiles.js.map +1 -1
  53. package/dist/browser/page.d.ts +2 -2
  54. package/dist/browser/page.d.ts.map +1 -1
  55. package/dist/browser/page.js +2 -2
  56. package/dist/browser/page.js.map +1 -1
  57. package/dist/browser/protocol.d.ts +7 -0
  58. package/dist/browser/protocol.d.ts.map +1 -1
  59. package/dist/browser/protocol.js +5 -0
  60. package/dist/browser/protocol.js.map +1 -1
  61. package/dist/cli.d.ts.map +1 -1
  62. package/dist/cli.js +4 -3
  63. package/dist/cli.js.map +1 -1
  64. package/dist/commands/architecture.d.ts +3 -1
  65. package/dist/commands/architecture.d.ts.map +1 -1
  66. package/dist/commands/architecture.js +13 -5
  67. package/dist/commands/architecture.js.map +1 -1
  68. package/dist/commands/browser/index.d.ts.map +1 -1
  69. package/dist/commands/browser/index.js +26 -4
  70. package/dist/commands/browser/index.js.map +1 -1
  71. package/dist/commands/delivery.d.ts.map +1 -1
  72. package/dist/commands/delivery.js.map +1 -1
  73. package/dist/commands/describe.d.ts.map +1 -1
  74. package/dist/commands/describe.js +104 -9
  75. package/dist/commands/describe.js.map +1 -1
  76. package/dist/commands/dispatch.d.ts.map +1 -1
  77. package/dist/commands/dispatch.js +3 -13
  78. package/dist/commands/dispatch.js.map +1 -1
  79. package/dist/commands/do.d.ts +16 -13
  80. package/dist/commands/do.d.ts.map +1 -1
  81. package/dist/commands/do.js +72 -18
  82. package/dist/commands/do.js.map +1 -1
  83. package/dist/commands/social.d.ts.map +1 -1
  84. package/dist/commands/social.js +3 -13
  85. package/dist/commands/social.js.map +1 -1
  86. package/dist/core/architecture-tree.d.ts +26 -6
  87. package/dist/core/architecture-tree.d.ts.map +1 -1
  88. package/dist/core/architecture-tree.js +184 -40
  89. package/dist/core/architecture-tree.js.map +1 -1
  90. package/dist/core/capability-matrix.d.ts +63 -0
  91. package/dist/core/capability-matrix.d.ts.map +1 -0
  92. package/dist/core/capability-matrix.js +316 -0
  93. package/dist/core/capability-matrix.js.map +1 -0
  94. package/dist/core/command-contract-lint.d.ts.map +1 -1
  95. package/dist/core/command-contract-lint.js +3 -1
  96. package/dist/core/command-contract-lint.js.map +1 -1
  97. package/dist/core/command-contract.d.ts +10 -3
  98. package/dist/core/command-contract.d.ts.map +1 -1
  99. package/dist/core/command-contract.js +87 -5
  100. package/dist/core/command-contract.js.map +1 -1
  101. package/dist/discovery/aliases.d.ts.map +1 -1
  102. package/dist/discovery/aliases.js +26 -0
  103. package/dist/discovery/aliases.js.map +1 -1
  104. package/dist/discovery/core-catalog.d.ts +1 -0
  105. package/dist/discovery/core-catalog.d.ts.map +1 -1
  106. package/dist/discovery/core-catalog.js +22 -5
  107. package/dist/discovery/core-catalog.js.map +1 -1
  108. package/dist/discovery/intents.d.ts +31 -4
  109. package/dist/discovery/intents.d.ts.map +1 -1
  110. package/dist/discovery/intents.js +166 -3
  111. package/dist/discovery/intents.js.map +1 -1
  112. package/dist/discovery/loader.d.ts.map +1 -1
  113. package/dist/discovery/loader.js +3 -0
  114. package/dist/discovery/loader.js.map +1 -1
  115. package/dist/discovery/search.d.ts.map +1 -1
  116. package/dist/discovery/search.js +10 -1
  117. package/dist/discovery/search.js.map +1 -1
  118. package/dist/engine/cascade.d.ts.map +1 -1
  119. package/dist/engine/cascade.js +21 -25
  120. package/dist/engine/cascade.js.map +1 -1
  121. package/dist/engine/chromium-cookies.d.ts +27 -0
  122. package/dist/engine/chromium-cookies.d.ts.map +1 -1
  123. package/dist/engine/chromium-cookies.js +53 -14
  124. package/dist/engine/chromium-cookies.js.map +1 -1
  125. package/dist/engine/cookie-capture.d.ts +30 -0
  126. package/dist/engine/cookie-capture.d.ts.map +1 -0
  127. package/dist/engine/cookie-capture.js +104 -0
  128. package/dist/engine/cookie-capture.js.map +1 -0
  129. package/dist/engine/cookie-extractor.d.ts.map +1 -1
  130. package/dist/engine/cookie-extractor.js +4 -12
  131. package/dist/engine/cookie-extractor.js.map +1 -1
  132. package/dist/engine/cookie-refresh.d.ts +59 -8
  133. package/dist/engine/cookie-refresh.d.ts.map +1 -1
  134. package/dist/engine/cookie-refresh.js +80 -58
  135. package/dist/engine/cookie-refresh.js.map +1 -1
  136. package/dist/engine/cookie-source.d.ts +110 -0
  137. package/dist/engine/cookie-source.d.ts.map +1 -0
  138. package/dist/engine/cookie-source.js +207 -0
  139. package/dist/engine/cookie-source.js.map +1 -0
  140. package/dist/engine/cookies.d.ts +37 -25
  141. package/dist/engine/cookies.d.ts.map +1 -1
  142. package/dist/engine/cookies.js +58 -157
  143. package/dist/engine/cookies.js.map +1 -1
  144. package/dist/engine/delivery/index.d.ts +2 -1
  145. package/dist/engine/delivery/index.d.ts.map +1 -1
  146. package/dist/engine/delivery/index.js +2 -1
  147. package/dist/engine/delivery/index.js.map +1 -1
  148. package/dist/engine/delivery/spec.d.ts +44 -0
  149. package/dist/engine/delivery/spec.d.ts.map +1 -0
  150. package/dist/engine/delivery/spec.js +82 -0
  151. package/dist/engine/delivery/spec.js.map +1 -0
  152. package/dist/engine/executor.d.ts +17 -6
  153. package/dist/engine/executor.d.ts.map +1 -1
  154. package/dist/engine/executor.js +131 -68
  155. package/dist/engine/executor.js.map +1 -1
  156. package/dist/engine/kernel/stages.d.ts.map +1 -1
  157. package/dist/engine/kernel/stages.js +14 -4
  158. package/dist/engine/kernel/stages.js.map +1 -1
  159. package/dist/engine/objective/catalog.d.ts +23 -0
  160. package/dist/engine/objective/catalog.d.ts.map +1 -0
  161. package/dist/engine/objective/catalog.js +42 -0
  162. package/dist/engine/objective/catalog.js.map +1 -0
  163. package/dist/engine/objective/delivery.d.ts +18 -0
  164. package/dist/engine/objective/delivery.d.ts.map +1 -0
  165. package/dist/engine/objective/delivery.js +64 -0
  166. package/dist/engine/objective/delivery.js.map +1 -0
  167. package/dist/engine/objective/index.d.ts +20 -0
  168. package/dist/engine/objective/index.d.ts.map +1 -0
  169. package/dist/engine/objective/index.js +20 -0
  170. package/dist/engine/objective/index.js.map +1 -0
  171. package/dist/engine/objective/media-playback.d.ts +17 -0
  172. package/dist/engine/objective/media-playback.d.ts.map +1 -0
  173. package/dist/engine/objective/media-playback.js +186 -0
  174. package/dist/engine/objective/media-playback.js.map +1 -0
  175. package/dist/engine/objective/output.d.ts +20 -0
  176. package/dist/engine/objective/output.d.ts.map +1 -0
  177. package/dist/engine/objective/output.js +88 -0
  178. package/dist/engine/objective/output.js.map +1 -0
  179. package/dist/engine/objective/planner.d.ts +17 -0
  180. package/dist/engine/objective/planner.d.ts.map +1 -0
  181. package/dist/engine/objective/planner.js +60 -0
  182. package/dist/engine/objective/planner.js.map +1 -0
  183. package/dist/engine/objective/types.d.ts +66 -0
  184. package/dist/engine/objective/types.d.ts.map +1 -0
  185. package/dist/engine/objective/types.js +16 -0
  186. package/dist/engine/objective/types.js.map +1 -0
  187. package/dist/engine/runtime.d.ts.map +1 -1
  188. package/dist/engine/runtime.js +9 -4
  189. package/dist/engine/runtime.js.map +1 -1
  190. package/dist/engine/step-observer.d.ts +62 -0
  191. package/dist/engine/step-observer.d.ts.map +1 -0
  192. package/dist/engine/step-observer.js +38 -0
  193. package/dist/engine/step-observer.js.map +1 -0
  194. package/dist/engine/steps/browser-helpers.d.ts.map +1 -1
  195. package/dist/engine/steps/browser-helpers.js +36 -8
  196. package/dist/engine/steps/browser-helpers.js.map +1 -1
  197. package/dist/engine/steps/fetch-text.d.ts.map +1 -1
  198. package/dist/engine/steps/fetch-text.js +106 -33
  199. package/dist/engine/steps/fetch-text.js.map +1 -1
  200. package/dist/engine/steps/fetch.d.ts +20 -0
  201. package/dist/engine/steps/fetch.d.ts.map +1 -1
  202. package/dist/engine/steps/fetch.js.map +1 -1
  203. package/dist/engine/steps/index.d.ts +2 -0
  204. package/dist/engine/steps/index.d.ts.map +1 -1
  205. package/dist/engine/steps/index.js +2 -0
  206. package/dist/engine/steps/index.js.map +1 -1
  207. package/dist/engine/steps/split-text.d.ts +26 -0
  208. package/dist/engine/steps/split-text.d.ts.map +1 -0
  209. package/dist/engine/steps/split-text.js +89 -0
  210. package/dist/engine/steps/split-text.js.map +1 -0
  211. package/dist/engine/steps/to-entries.d.ts +9 -0
  212. package/dist/engine/steps/to-entries.d.ts.map +1 -0
  213. package/dist/engine/steps/to-entries.js +27 -0
  214. package/dist/engine/steps/to-entries.js.map +1 -0
  215. package/dist/fast-path/handlers/discovery.d.ts.map +1 -1
  216. package/dist/fast-path/handlers/discovery.js +7 -7
  217. package/dist/fast-path/handlers/discovery.js.map +1 -1
  218. package/dist/manifest-compact.txt +3 -2
  219. package/dist/manifest.json +183 -17
  220. package/dist/mcp/tools.js +1 -1
  221. package/dist/mcp/tools.js.map +1 -1
  222. package/dist/output/auth-guidance.d.ts +17 -3
  223. package/dist/output/auth-guidance.d.ts.map +1 -1
  224. package/dist/output/auth-guidance.js +27 -3
  225. package/dist/output/auth-guidance.js.map +1 -1
  226. package/dist/output/error-map.d.ts.map +1 -1
  227. package/dist/output/error-map.js +4 -0
  228. package/dist/output/error-map.js.map +1 -1
  229. package/package.json +4 -2
  230. package/server.json +2 -2
  231. package/skills/unicli/SKILL.md +1 -1
  232. package/skills/unicli-claude-code/SKILL.md +1 -1
  233. package/skills/unicli-hermes/SKILL.md +1 -1
  234. package/src/adapters/12306/price.yaml +91 -0
  235. package/src/adapters/12306/stations.yaml +52 -0
  236. package/src/adapters/12306/trains.yaml +129 -0
  237. package/src/adapters/_electron/desktop-shared.ts +5 -1
  238. package/src/adapters/_electron/shared.ts +15 -0
  239. package/src/adapters/antigravity/extra.ts +10 -1
  240. package/src/adapters/chatwise/extra.ts +10 -1
  241. package/src/adapters/codex/codex.ts +6 -0
  242. package/src/adapters/codex/extra.ts +10 -1
  243. package/src/adapters/codex/projects.ts +9 -1
  244. package/src/adapters/cursor/cursor.ts +9 -0
  245. package/src/adapters/discord-app/discord-app.ts +16 -1
  246. package/src/adapters/macos/brightness.yaml +6 -3
  247. package/src/adapters/macos/calendar-list.yaml +9 -11
  248. package/src/adapters/macos/calendar-today.yaml +1 -1
  249. package/src/adapters/macos/safari-url.yaml +8 -4
  250. package/src/adapters/maoyan/hot.yaml +1 -1
  251. package/src/adapters/notion-app/notion-app.ts +17 -1
  252. package/src/adapters/spotify/api.ts +54 -8
  253. package/src/adapters/weibo/trending.yaml +2 -0
  254. package/src/adapters/xiaohongshu/browser-state.ts +59 -2
  255. package/src/adapters/xiaohongshu/feed.ts +103 -0
  256. package/src/adapters/xueqiu/extra.ts +5 -2
@@ -0,0 +1,207 @@
1
+ /**
2
+ * @owner src::engine::cookie-source
3
+ * @does Multi-source cookie acquisition that surfaces the REAL cause of a
4
+ * miss (keychain denied / corrupt file / v20 encryption / CDP
5
+ * unavailable) as a typed outcome, instead of collapsing every
6
+ * failure to null.
7
+ * @needs node:fs, node:path, ./chromium-cookies (lazy), ./cookie-extractor (lazy)
8
+ * @feeds src::engine::cookies (loadCookies/loadCookiesWithCDP/acquireCookies
9
+ * projections), src::engine::executor (auth error detail)
10
+ * @breaks never throws from loadCookiesWithDiagnostics — failures become
11
+ * CookieLoadOutcome {status:"error", reasons}; readDiskCookies is total
12
+ * @invariants exactly one of loaded|absent|error; "absent" ⇒ no source errored
13
+ * (genuinely not logged in); "error" ⇒ ≥1 source had a real failure
14
+ * @side-effects readDiskCookies reads fs; default sources read browser DB / CDP
15
+ * @perf disk O(file); browser tries installed browsers in order, stops on hit
16
+ * @concurrency stateless; sources own their own IO
17
+ * @test tests/unit/engine/cookie-source.test.ts
18
+ * @stability experimental
19
+ * @since 2026-05-30
20
+ */
21
+ import { readFileSync, existsSync } from "node:fs";
22
+ import { join } from "node:path";
23
+ const SITE_RE = /^[a-zA-Z0-9._-]+$/;
24
+ export function cookieDir() {
25
+ return (process.env.UNICLI_COOKIE_DIR ??
26
+ join(process.env.HOME ?? "~", ".unicli", "cookies"));
27
+ }
28
+ /**
29
+ * Read the on-disk cookie file, distinguishing absent from corrupt. The old
30
+ * loadCookies collapsed a truncated / wrong-shaped file to null ("no auth"),
31
+ * hiding the real cause; this returns a typed corrupt with a detail.
32
+ */
33
+ export function readDiskCookies(site) {
34
+ if (!SITE_RE.test(site)) {
35
+ return { kind: "corrupt", detail: `invalid site name "${site}"` };
36
+ }
37
+ const path = join(cookieDir(), `${site}.json`);
38
+ if (!existsSync(path))
39
+ return { kind: "absent" };
40
+ let raw;
41
+ try {
42
+ raw = readFileSync(path, "utf-8");
43
+ }
44
+ catch (err) {
45
+ return {
46
+ kind: "corrupt",
47
+ detail: `cannot read ${path}: ${err instanceof Error ? err.message : String(err)}`,
48
+ };
49
+ }
50
+ let parsed;
51
+ try {
52
+ parsed = JSON.parse(raw);
53
+ }
54
+ catch {
55
+ return { kind: "corrupt", detail: `${path} is not valid JSON` };
56
+ }
57
+ if (typeof parsed !== "object" || parsed === null || Array.isArray(parsed)) {
58
+ return { kind: "corrupt", detail: `${path} is not a {name: value} object` };
59
+ }
60
+ return { kind: "ok", cookies: parsed };
61
+ }
62
+ /** Resolve the cookie domain the same way the legacy loader did. */
63
+ export function resolveCookieDomain(site, domain) {
64
+ let resolved = domain ?? site.replace(/_/g, ".");
65
+ if (!resolved.includes("."))
66
+ resolved = `${resolved}.com`;
67
+ return resolved;
68
+ }
69
+ async function defaultReadBrowser(domain) {
70
+ let mod;
71
+ try {
72
+ mod = await import("./chromium-cookies.js");
73
+ }
74
+ catch (err) {
75
+ return {
76
+ kind: "error",
77
+ reasons: [
78
+ {
79
+ source: "browser",
80
+ code: "module_load_failed",
81
+ detail: err instanceof Error ? err.message : String(err),
82
+ },
83
+ ],
84
+ };
85
+ }
86
+ const installed = mod.detectInstalledBrowsers();
87
+ if (installed.length === 0)
88
+ return { kind: "none" };
89
+ const reasons = [];
90
+ for (const browser of installed) {
91
+ try {
92
+ const record = mod.readCookiesAsRecord({ browser, domain });
93
+ if (Object.keys(record).length > 0)
94
+ return { kind: "ok", cookies: record };
95
+ }
96
+ catch (err) {
97
+ const code = err instanceof mod.ChromiumCookieError
98
+ ? err.code
99
+ : "browser_read_failed";
100
+ reasons.push({
101
+ source: "browser",
102
+ code,
103
+ detail: err instanceof Error ? err.message : String(err),
104
+ });
105
+ }
106
+ }
107
+ return reasons.length > 0 ? { kind: "error", reasons } : { kind: "none" };
108
+ }
109
+ async function defaultReadCdp(domain) {
110
+ const { extractCookiesViaCDP } = await import("./cookie-extractor.js");
111
+ return extractCookiesViaCDP(domain);
112
+ }
113
+ export const defaultCookieSources = {
114
+ readDisk: readDiskCookies,
115
+ readBrowser: defaultReadBrowser,
116
+ readCdp: defaultReadCdp,
117
+ };
118
+ /**
119
+ * Acquire cookies across disk → browser → CDP, collecting the real cause of
120
+ * each source's failure. Never throws: a miss with no underlying fault is
121
+ * "absent"; a miss where a source actually errored (Keychain denial, corrupt
122
+ * file, v20 encryption, CDP down) is "error" with the reasons.
123
+ *
124
+ * Domain precedence and the disk-first / browser / CDP order match the legacy
125
+ * loadCookiesWithCDP exactly, so the projection wrappers preserve behavior.
126
+ */
127
+ export async function loadCookiesWithDiagnostics(site, domain, sources = defaultCookieSources, opts = {}) {
128
+ const reasons = [];
129
+ // Refresh-after-401 skips disk: the on-disk cookies are exactly the stale
130
+ // ones that just failed, so re-acquisition must go straight to the live
131
+ // browser / CDP sources.
132
+ if (!opts.skipDisk) {
133
+ const disk = sources.readDisk(site);
134
+ if (disk.kind === "ok") {
135
+ return { status: "loaded", source: "disk", cookies: disk.cookies };
136
+ }
137
+ if (disk.kind === "corrupt") {
138
+ reasons.push({
139
+ source: "disk",
140
+ code: "corrupt_file",
141
+ detail: disk.detail,
142
+ });
143
+ }
144
+ }
145
+ const cookieDomain = resolveCookieDomain(site, domain);
146
+ if (process.env.UNICLI_COOKIE_NO_BROWSER !== "1") {
147
+ const browser = await sources.readBrowser(cookieDomain);
148
+ if (browser.kind === "ok" && Object.keys(browser.cookies).length > 0) {
149
+ return { status: "loaded", source: "browser", cookies: browser.cookies };
150
+ }
151
+ if (browser.kind === "error")
152
+ reasons.push(...browser.reasons);
153
+ }
154
+ try {
155
+ const cdp = await sources.readCdp(cookieDomain);
156
+ if (Object.keys(cdp).length > 0) {
157
+ return { status: "loaded", source: "cdp", cookies: cdp };
158
+ }
159
+ }
160
+ catch (err) {
161
+ reasons.push({
162
+ source: "cdp",
163
+ code: "cdp_unavailable",
164
+ detail: err instanceof Error ? err.message : String(err),
165
+ });
166
+ }
167
+ return reasons.length > 0
168
+ ? { status: "error", reasons }
169
+ : { status: "absent" };
170
+ }
171
+ /**
172
+ * Turn a non-loaded outcome into an agent-actionable message + suggestion.
173
+ * This is where the no-silent-failure win is spent: Keychain denial, v20
174
+ * encryption, and a corrupt file each get a DISTINCT next step instead of the
175
+ * legacy one-size-fits-all "run unicli auth setup".
176
+ */
177
+ export function describeCookieFailure(outcome, site, domain) {
178
+ const d = resolveCookieDomain(site, domain);
179
+ if (outcome.status === "absent") {
180
+ return {
181
+ message: `No cookies found for "${site}".`,
182
+ suggestion: `Sign in to https://${d} in Chrome, then run: unicli auth import ${site} --domain ${d}`,
183
+ retryable: false,
184
+ };
185
+ }
186
+ const codes = outcome.reasons.map((r) => `${r.source}:${r.code}`).join(", ");
187
+ const has = (code) => outcome.reasons.some((r) => r.code === code);
188
+ let suggestion;
189
+ if (has("keychain_denied")) {
190
+ suggestion = `Chrome cookie decryption was denied by the macOS Keychain. Grant access when prompted, or run: unicli auth import ${site} --domain ${d}`;
191
+ }
192
+ else if (has("encryption_unsupported")) {
193
+ suggestion = `Chrome's app-bound encryption (v20) blocks direct cookie reads. Start Chrome via "unicli browser start" (CDP path), or run: unicli auth import ${site} --domain ${d}`;
194
+ }
195
+ else if (has("corrupt_file")) {
196
+ suggestion = `The saved cookie file is unreadable. Re-import: unicli auth import ${site} --domain ${d}`;
197
+ }
198
+ else {
199
+ suggestion = `Could not acquire cookies (${codes}). Sign in to https://${d}, then run: unicli auth import ${site} --domain ${d}`;
200
+ }
201
+ return {
202
+ message: `Failed to load cookies for "${site}" — ${codes}.`,
203
+ suggestion,
204
+ retryable: false,
205
+ };
206
+ }
207
+ //# sourceMappingURL=cookie-source.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cookie-source.js","sourceRoot":"","sources":["../../src/engine/cookie-source.ts"],"names":[],"mappings":"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"}
@@ -1,12 +1,24 @@
1
1
  /**
2
- * Cookie file reader for authenticated adapters.
3
- *
4
- * Cookies are stored as JSON in ~/.unicli/cookies/<site>.json
5
- * Each file contains { "cookie_name": "value", ... }
2
+ * @owner src::engine::cookies
3
+ * @does Cookie front door for adapters — thin projections over the
4
+ * structured acquisition core in cookie-source.ts (disk read,
5
+ * header formatting, multi-source load, refresh).
6
+ * @needs ./cookie-source, ./cookie-extractor (saveCookies)
7
+ * @feeds adapters (loadCookiesWithCDP/formatCookieHeader), executor,
8
+ * dispatch/social (refreshCookiesFromBrowser), commands/auth
9
+ * @breaks loadCookies/loadCookiesWithCDP return null on miss (back-compat);
10
+ * acquireCookies returns a typed CookieLoadOutcome that names the
11
+ * real cause — callers that need the cause use acquireCookies
12
+ * @invariants loadCookiesWithCDP cookies === acquireCookies(...).cookies on load
13
+ * @side-effects acquireCookies persists browser/CDP cookies to disk (best-effort)
14
+ * @test tests/unit/engine/cookie-source.test.ts, cookie-refresh-format.test.ts
15
+ * @stability stable
16
+ * @since 2026-05-30
6
17
  */
18
+ import { type CookieLoadOutcome } from "./cookie-source.js";
7
19
  /**
8
- * Load cookies for a site from disk.
9
- * Returns null if file doesn't exist or is malformed.
20
+ * Load cookies for a site from disk. Returns null when the file is absent OR
21
+ * unreadable; callers needing to tell those apart use `acquireCookies`.
10
22
  */
11
23
  export declare function loadCookies(site: string): Record<string, string> | null;
12
24
  /**
@@ -14,32 +26,27 @@ export declare function loadCookies(site: string): Record<string, string> | null
14
26
  * Example: "SESSDATA=abc; bili_jct=def"
15
27
  */
16
28
  export declare function formatCookieHeader(cookies: Record<string, string>): string;
17
- /**
18
- * Validate that a cookie file has all required keys.
19
- */
29
+ /** Validate that a cookie file has all required keys. */
20
30
  export declare function validateCookies(site: string, requiredKeys: string[]): {
21
31
  valid: boolean;
22
32
  missing: string[];
23
33
  };
34
+ /** Get the cookie directory path (for display in auth commands). */
35
+ export declare function getCookieDir(): string;
24
36
  /**
25
- * Get the cookie directory path (for display in auth commands).
37
+ * Acquire cookies across disk browser CDP, returning the structured outcome
38
+ * (loaded / absent / error+reasons). On a non-disk load, persists to
39
+ * ~/.unicli/cookies for offline reuse. This is the front door for callers that
40
+ * need to surface WHY acquisition failed (Keychain denial, v20 encryption,
41
+ * corrupt file) instead of a bare null.
26
42
  */
27
- export declare function getCookieDir(): string;
43
+ export declare function acquireCookies(site: string, domain?: string, opts?: {
44
+ skipDisk?: boolean;
45
+ }): Promise<CookieLoadOutcome>;
28
46
  /**
29
- * Load cookies with multi-source fallback.
30
- *
31
- * Precedence (each falls through silently to the next on miss):
32
- * 1. ~/.unicli/cookies/<site>.json — explicit user import
33
- * 2. browser local DB (Chrome/Arc/Dia/…) — direct SQLite read, no browser launch
34
- * 3. CDP — connects to Chrome debug port (legacy)
35
- *
36
- * The browser disk source is the new default for `strategy: cookie` adapters:
37
- * it works whether the browser is open or closed, never opens a new tab, and
38
- * needs neither extension nor daemon. Successful reads are persisted to
39
- * ~/.unicli/cookies for offline reuse.
40
- *
41
- * Set `UNICLI_COOKIE_NO_BROWSER=1` to skip the browser-disk step (e.g., in CI
42
- * where the macOS Keychain prompt would block).
47
+ * Load cookies with multi-source fallback (disk → browser DB → CDP). Returns
48
+ * null on any miss. Behavior-compatible with every existing adapter consumer;
49
+ * the structured cause is available via `acquireCookies`.
43
50
  */
44
51
  export declare function loadCookiesWithCDP(site: string, domain?: string): Promise<Record<string, string> | null>;
45
52
  export interface CookieRefreshResult {
@@ -51,5 +58,10 @@ export interface CookieRefreshResult {
51
58
  cookies?: string[];
52
59
  suggestion?: string;
53
60
  }
61
+ /**
62
+ * Re-acquire cookies from the live browser / CDP after an auth failure. Skips
63
+ * the on-disk file (those are the stale cookies that just failed) and reports
64
+ * the real cause on failure rather than a generic message.
65
+ */
54
66
  export declare function refreshCookiesFromBrowser(site: string, domain?: string): Promise<CookieRefreshResult>;
55
67
  //# sourceMappingURL=cookies.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"cookies.d.ts","sourceRoot":"","sources":["../../src/engine/cookies.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAgBH;;;GAGG;AACH,wBAAgB,WAAW,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,IAAI,CAmBvE;AAED;;;GAGG;AACH,wBAAgB,kBAAkB,CAAC,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,MAAM,CAI1E;AAED;;GAEG;AACH,wBAAgB,eAAe,CAC7B,IAAI,EAAE,MAAM,EACZ,YAAY,EAAE,MAAM,EAAE,GACrB;IAAE,KAAK,EAAE,OAAO,CAAC;IAAC,OAAO,EAAE,MAAM,EAAE,CAAA;CAAE,CAMvC;AAED;;GAEG;AACH,wBAAgB,YAAY,IAAI,MAAM,CAErC;AAED;;;;;;;;;;;;;;;GAeG;AACH,wBAAsB,kBAAkB,CACtC,IAAI,EAAE,MAAM,EACZ,MAAM,CAAC,EAAE,MAAM,GACd,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,IAAI,CAAC,CAwCxC;AAED,MAAM,WAAW,mBAAmB;IAClC,EAAE,EAAE,OAAO,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;IACf,MAAM,CAAC,EAAE,SAAS,GAAG,KAAK,CAAC;IAC3B,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,OAAO,CAAC,EAAE,MAAM,EAAE,CAAC;IACnB,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB;AAUD,wBAAsB,yBAAyB,CAC7C,IAAI,EAAE,MAAM,EACZ,MAAM,CAAC,EAAE,MAAM,GACd,OAAO,CAAC,mBAAmB,CAAC,CA2D9B"}
1
+ {"version":3,"file":"cookies.d.ts","sourceRoot":"","sources":["../../src/engine/cookies.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;GAgBG;AAEH,OAAO,EAML,KAAK,iBAAiB,EACvB,MAAM,oBAAoB,CAAC;AAG5B;;;GAGG;AACH,wBAAgB,WAAW,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,IAAI,CAGvE;AAED;;;GAGG;AACH,wBAAgB,kBAAkB,CAAC,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,MAAM,CAI1E;AAED,yDAAyD;AACzD,wBAAgB,eAAe,CAC7B,IAAI,EAAE,MAAM,EACZ,YAAY,EAAE,MAAM,EAAE,GACrB;IAAE,KAAK,EAAE,OAAO,CAAC;IAAC,OAAO,EAAE,MAAM,EAAE,CAAA;CAAE,CAKvC;AAED,oEAAoE;AACpE,wBAAgB,YAAY,IAAI,MAAM,CAErC;AAED;;;;;;GAMG;AACH,wBAAsB,cAAc,CAClC,IAAI,EAAE,MAAM,EACZ,MAAM,CAAC,EAAE,MAAM,EACf,IAAI,GAAE;IAAE,QAAQ,CAAC,EAAE,OAAO,CAAA;CAAO,GAChC,OAAO,CAAC,iBAAiB,CAAC,CAgB5B;AAED;;;;GAIG;AACH,wBAAsB,kBAAkB,CACtC,IAAI,EAAE,MAAM,EACZ,MAAM,CAAC,EAAE,MAAM,GACd,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,IAAI,CAAC,CAGxC;AAED,MAAM,WAAW,mBAAmB;IAClC,EAAE,EAAE,OAAO,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;IACf,MAAM,CAAC,EAAE,SAAS,GAAG,KAAK,CAAC;IAC3B,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,OAAO,CAAC,EAAE,MAAM,EAAE,CAAC;IACnB,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB;AAED;;;;GAIG;AACH,wBAAsB,yBAAyB,CAC7C,IAAI,EAAE,MAAM,EACZ,MAAM,CAAC,EAAE,MAAM,GACd,OAAO,CAAC,mBAAmB,CAAC,CA+B9B"}
@@ -1,39 +1,29 @@
1
1
  /**
2
- * Cookie file reader for authenticated adapters.
3
- *
4
- * Cookies are stored as JSON in ~/.unicli/cookies/<site>.json
5
- * Each file contains { "cookie_name": "value", ... }
2
+ * @owner src::engine::cookies
3
+ * @does Cookie front door for adapters — thin projections over the
4
+ * structured acquisition core in cookie-source.ts (disk read,
5
+ * header formatting, multi-source load, refresh).
6
+ * @needs ./cookie-source, ./cookie-extractor (saveCookies)
7
+ * @feeds adapters (loadCookiesWithCDP/formatCookieHeader), executor,
8
+ * dispatch/social (refreshCookiesFromBrowser), commands/auth
9
+ * @breaks loadCookies/loadCookiesWithCDP return null on miss (back-compat);
10
+ * acquireCookies returns a typed CookieLoadOutcome that names the
11
+ * real cause — callers that need the cause use acquireCookies
12
+ * @invariants loadCookiesWithCDP cookies === acquireCookies(...).cookies on load
13
+ * @side-effects acquireCookies persists browser/CDP cookies to disk (best-effort)
14
+ * @test tests/unit/engine/cookie-source.test.ts, cookie-refresh-format.test.ts
15
+ * @stability stable
16
+ * @since 2026-05-30
6
17
  */
7
- import { readFileSync, existsSync } from "node:fs";
8
- import { join } from "node:path";
9
- import { extractCookiesViaCDP, saveCookies as saveCookiesToDisk, } from "./cookie-extractor.js";
10
- function cookieDir() {
11
- return (process.env.UNICLI_COOKIE_DIR ??
12
- join(process.env.HOME ?? "~", ".unicli", "cookies"));
13
- }
18
+ import { cookieDir, describeCookieFailure, loadCookiesWithDiagnostics, readDiskCookies, resolveCookieDomain, } from "./cookie-source.js";
19
+ import { saveCookies as saveCookiesToDisk } from "./cookie-extractor.js";
14
20
  /**
15
- * Load cookies for a site from disk.
16
- * Returns null if file doesn't exist or is malformed.
21
+ * Load cookies for a site from disk. Returns null when the file is absent OR
22
+ * unreadable; callers needing to tell those apart use `acquireCookies`.
17
23
  */
18
24
  export function loadCookies(site) {
19
- if (!/^[a-zA-Z0-9._-]+$/.test(site))
20
- return null;
21
- const path = join(cookieDir(), `${site}.json`);
22
- if (!existsSync(path))
23
- return null;
24
- try {
25
- const raw = readFileSync(path, "utf-8");
26
- const parsed = JSON.parse(raw);
27
- if (typeof parsed !== "object" ||
28
- parsed === null ||
29
- Array.isArray(parsed)) {
30
- return null;
31
- }
32
- return parsed;
33
- }
34
- catch {
35
- return null;
36
- }
25
+ const read = readDiskCookies(site);
26
+ return read.kind === "ok" ? read.cookies : null;
37
27
  }
38
28
  /**
39
29
  * Format cookies as an HTTP Cookie header value.
@@ -44,9 +34,7 @@ export function formatCookieHeader(cookies) {
44
34
  .map(([k, v]) => `${k}=${v}`)
45
35
  .join("; ");
46
36
  }
47
- /**
48
- * Validate that a cookie file has all required keys.
49
- */
37
+ /** Validate that a cookie file has all required keys. */
50
38
  export function validateCookies(site, requiredKeys) {
51
39
  const cookies = loadCookies(site);
52
40
  if (!cookies)
@@ -54,77 +42,44 @@ export function validateCookies(site, requiredKeys) {
54
42
  const missing = requiredKeys.filter((k) => !(k in cookies));
55
43
  return { valid: missing.length === 0, missing };
56
44
  }
57
- /**
58
- * Get the cookie directory path (for display in auth commands).
59
- */
45
+ /** Get the cookie directory path (for display in auth commands). */
60
46
  export function getCookieDir() {
61
47
  return cookieDir();
62
48
  }
63
49
  /**
64
- * Load cookies with multi-source fallback.
65
- *
66
- * Precedence (each falls through silently to the next on miss):
67
- * 1. ~/.unicli/cookies/<site>.json — explicit user import
68
- * 2. browser local DB (Chrome/Arc/Dia/…) direct SQLite read, no browser launch
69
- * 3. CDP — connects to Chrome debug port (legacy)
70
- *
71
- * The browser disk source is the new default for `strategy: cookie` adapters:
72
- * it works whether the browser is open or closed, never opens a new tab, and
73
- * needs neither extension nor daemon. Successful reads are persisted to
74
- * ~/.unicli/cookies for offline reuse.
75
- *
76
- * Set `UNICLI_COOKIE_NO_BROWSER=1` to skip the browser-disk step (e.g., in CI
77
- * where the macOS Keychain prompt would block).
50
+ * Acquire cookies across disk → browser → CDP, returning the structured outcome
51
+ * (loaded / absent / error+reasons). On a non-disk load, persists to
52
+ * ~/.unicli/cookies for offline reuse. This is the front door for callers that
53
+ * need to surface WHY acquisition failed (Keychain denial, v20 encryption,
54
+ * corrupt file) instead of a bare null.
78
55
  */
79
- export async function loadCookiesWithCDP(site, domain) {
80
- // 1. ~/.unicli/cookies first
81
- const diskCookies = loadCookies(site);
82
- if (diskCookies)
83
- return diskCookies;
84
- // Resolve the cookie domain once for both browser and CDP paths.
85
- let cookieDomain = domain ?? site.replace(/_/g, ".");
86
- if (!cookieDomain.includes(".")) {
87
- cookieDomain = `${cookieDomain}.com`;
88
- }
89
- // 2. Direct browser disk read (no launch, no CDP, no extension).
90
- if (process.env.UNICLI_COOKIE_NO_BROWSER !== "1") {
91
- const browserCookies = await loadFromInstalledBrowser(cookieDomain);
92
- if (browserCookies && Object.keys(browserCookies).length > 0) {
93
- try {
94
- saveCookiesToDisk(site, browserCookies);
95
- }
96
- catch {
97
- // Non-fatal — caller still gets the cookies.
98
- }
99
- return browserCookies;
56
+ export async function acquireCookies(site, domain, opts = {}) {
57
+ const outcome = await loadCookiesWithDiagnostics(site, domain, undefined, opts);
58
+ if (outcome.status === "loaded" && outcome.source !== "disk") {
59
+ try {
60
+ saveCookiesToDisk(site, outcome.cookies);
100
61
  }
101
- }
102
- // 3. CDP fallback for users running Chrome with --remote-debugging-port.
103
- try {
104
- const cdpCookies = await extractCookiesViaCDP(cookieDomain);
105
- if (Object.keys(cdpCookies).length > 0) {
106
- try {
107
- saveCookiesToDisk(site, cdpCookies);
108
- }
109
- catch {
110
- // Non-fatal: disk write failed but we still have the cookies
111
- }
112
- return cdpCookies;
62
+ catch {
63
+ // REASON: persistence is best-effort; the cookies are already usable for
64
+ // this run, and the next run re-acquires from the same live source.
113
65
  }
114
66
  }
115
- catch {
116
- // CDP not available — fall through
117
- }
118
- return null;
67
+ return outcome;
119
68
  }
120
- function persistCookies(site, cookies) {
121
- try {
122
- saveCookiesToDisk(site, cookies);
123
- }
124
- catch {
125
- // Non-fatal: callers can still retry because browser/CDP remains readable.
126
- }
69
+ /**
70
+ * Load cookies with multi-source fallback (disk → browser DB → CDP). Returns
71
+ * null on any miss. Behavior-compatible with every existing adapter consumer;
72
+ * the structured cause is available via `acquireCookies`.
73
+ */
74
+ export async function loadCookiesWithCDP(site, domain) {
75
+ const outcome = await acquireCookies(site, domain);
76
+ return outcome.status === "loaded" ? outcome.cookies : null;
127
77
  }
78
+ /**
79
+ * Re-acquire cookies from the live browser / CDP after an auth failure. Skips
80
+ * the on-disk file (those are the stale cookies that just failed) and reports
81
+ * the real cause on failure rather than a generic message.
82
+ */
128
83
  export async function refreshCookiesFromBrowser(site, domain) {
129
84
  if (!/^[a-zA-Z0-9._-]+$/.test(site)) {
130
85
  return {
@@ -134,77 +89,23 @@ export async function refreshCookiesFromBrowser(site, domain) {
134
89
  suggestion: "Site names must contain only letters, digits, dot, dash, or underscore.",
135
90
  };
136
91
  }
137
- let cookieDomain = domain ?? site.replace(/_/g, ".");
138
- if (!cookieDomain.includes(".")) {
139
- cookieDomain = `${cookieDomain}.com`;
140
- }
141
- if (process.env.UNICLI_COOKIE_NO_BROWSER !== "1") {
142
- const browserCookies = await loadFromInstalledBrowser(cookieDomain);
143
- if (browserCookies && Object.keys(browserCookies).length > 0) {
144
- persistCookies(site, browserCookies);
145
- return {
146
- ok: true,
147
- site,
148
- domain: cookieDomain,
149
- source: "browser",
150
- cookieCount: Object.keys(browserCookies).length,
151
- cookies: Object.keys(browserCookies),
152
- };
153
- }
154
- }
155
- try {
156
- const cdpCookies = await extractCookiesViaCDP(cookieDomain);
157
- if (Object.keys(cdpCookies).length > 0) {
158
- persistCookies(site, cdpCookies);
159
- return {
160
- ok: true,
161
- site,
162
- domain: cookieDomain,
163
- source: "cdp",
164
- cookieCount: Object.keys(cdpCookies).length,
165
- cookies: Object.keys(cdpCookies),
166
- };
167
- }
168
- }
169
- catch {
92
+ const cookieDomain = resolveCookieDomain(site, domain);
93
+ const outcome = await acquireCookies(site, domain, { skipDisk: true });
94
+ if (outcome.status === "loaded") {
170
95
  return {
171
- ok: false,
96
+ ok: true,
172
97
  site,
173
98
  domain: cookieDomain,
174
- suggestion: `Open https://${cookieDomain} in the browser, complete login or challenge, then run: unicli auth import ${site} --domain ${cookieDomain}`,
99
+ source: outcome.source === "cdp" ? "cdp" : "browser",
100
+ cookieCount: Object.keys(outcome.cookies).length,
101
+ cookies: Object.keys(outcome.cookies),
175
102
  };
176
103
  }
177
104
  return {
178
105
  ok: false,
179
106
  site,
180
107
  domain: cookieDomain,
181
- suggestion: `No cookies found for ${cookieDomain}. Sign in to https://${cookieDomain}, then run: unicli auth import ${site} --domain ${cookieDomain}`,
108
+ suggestion: describeCookieFailure(outcome, site, domain).suggestion,
182
109
  };
183
110
  }
184
- /**
185
- * Try each installed Chromium browser in priority order until one yields
186
- * cookies for the domain. Failures (browser not installed, Keychain denied,
187
- * unsupported encryption) fall through silently — the next source handles it.
188
- */
189
- async function loadFromInstalledBrowser(domain) {
190
- let mod;
191
- try {
192
- mod = await import("./chromium-cookies.js");
193
- }
194
- catch {
195
- return null;
196
- }
197
- const installed = mod.detectInstalledBrowsers();
198
- for (const browser of installed) {
199
- try {
200
- const record = mod.readCookiesAsRecord({ browser, domain });
201
- if (Object.keys(record).length > 0)
202
- return record;
203
- }
204
- catch {
205
- // Move on to the next browser.
206
- }
207
- }
208
- return null;
209
- }
210
111
  //# sourceMappingURL=cookies.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"cookies.js","sourceRoot":"","sources":["../../src/engine/cookies.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"cookies.js","sourceRoot":"","sources":["../../src/engine/cookies.ts"],"names":[],"mappings":"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"}
@@ -1,7 +1,7 @@
1
1
  /**
2
2
  * @owner src/engine/delivery/index.ts
3
3
  * @does Exposes the internal objective-level delivery kernel.
4
- * @needs src/engine/delivery/planner.ts, src/engine/delivery/repair.ts, src/engine/delivery/session.ts, src/engine/delivery/trajectory.ts, src/engine/delivery/types.ts
4
+ * @needs src/engine/delivery/planner.ts, src/engine/delivery/repair.ts, src/engine/delivery/session.ts, src/engine/delivery/spec.ts, src/engine/delivery/trajectory.ts, src/engine/delivery/types.ts
5
5
  * @feeds tests and future command surfaces that orchestrate objective delivery.
6
6
  * @breaks Missing exports hide the delivery state machine from integration layers.
7
7
  * @invariants exports are side-effect free and preserve the delivery module boundary.
@@ -15,6 +15,7 @@
15
15
  export * from "./planner.js";
16
16
  export * from "./repair.js";
17
17
  export * from "./session.js";
18
+ export * from "./spec.js";
18
19
  export * from "./trajectory.js";
19
20
  export * from "./types.js";
20
21
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/engine/delivery/index.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;GAaG;AAEH,cAAc,cAAc,CAAC;AAC7B,cAAc,aAAa,CAAC;AAC5B,cAAc,cAAc,CAAC;AAC7B,cAAc,iBAAiB,CAAC;AAChC,cAAc,YAAY,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/engine/delivery/index.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;GAaG;AAEH,cAAc,cAAc,CAAC;AAC7B,cAAc,aAAa,CAAC;AAC5B,cAAc,cAAc,CAAC;AAC7B,cAAc,WAAW,CAAC;AAC1B,cAAc,iBAAiB,CAAC;AAChC,cAAc,YAAY,CAAC"}
@@ -1,7 +1,7 @@
1
1
  /**
2
2
  * @owner src/engine/delivery/index.ts
3
3
  * @does Exposes the internal objective-level delivery kernel.
4
- * @needs src/engine/delivery/planner.ts, src/engine/delivery/repair.ts, src/engine/delivery/session.ts, src/engine/delivery/trajectory.ts, src/engine/delivery/types.ts
4
+ * @needs src/engine/delivery/planner.ts, src/engine/delivery/repair.ts, src/engine/delivery/session.ts, src/engine/delivery/spec.ts, src/engine/delivery/trajectory.ts, src/engine/delivery/types.ts
5
5
  * @feeds tests and future command surfaces that orchestrate objective delivery.
6
6
  * @breaks Missing exports hide the delivery state machine from integration layers.
7
7
  * @invariants exports are side-effect free and preserve the delivery module boundary.
@@ -15,6 +15,7 @@
15
15
  export * from "./planner.js";
16
16
  export * from "./repair.js";
17
17
  export * from "./session.js";
18
+ export * from "./spec.js";
18
19
  export * from "./trajectory.js";
19
20
  export * from "./types.js";
20
21
  //# sourceMappingURL=index.js.map