quiver-cli 0.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 (281) hide show
  1. package/README.md +188 -0
  2. package/bin/quiver-cli.mjs +2 -0
  3. package/dist/cli.js +3074 -0
  4. package/package.json +55 -0
  5. package/template/.agents/AGENTS.md +25 -0
  6. package/template/.agents/commands/cp.md +116 -0
  7. package/template/.agents/commands/next-setup.md +1064 -0
  8. package/template/.agents/commands/tf-readme.md +38 -0
  9. package/template/.agents/config.json +60 -0
  10. package/template/.agents/skills/agent-browser/SKILL.md +55 -0
  11. package/template/.agents/skills/apps/skybridge/SKILL.md +46 -0
  12. package/template/.agents/skills/apps/skybridge/references/architecture.md +175 -0
  13. package/template/.agents/skills/apps/skybridge/references/copy-template.md +24 -0
  14. package/template/.agents/skills/apps/skybridge/references/csp.md +33 -0
  15. package/template/.agents/skills/apps/skybridge/references/deploy.md +33 -0
  16. package/template/.agents/skills/apps/skybridge/references/discover.md +84 -0
  17. package/template/.agents/skills/apps/skybridge/references/download-file.md +77 -0
  18. package/template/.agents/skills/apps/skybridge/references/fetch-and-render-data.md +151 -0
  19. package/template/.agents/skills/apps/skybridge/references/oauth.md +115 -0
  20. package/template/.agents/skills/apps/skybridge/references/open-external-links.md +71 -0
  21. package/template/.agents/skills/apps/skybridge/references/prompt-llm.md +20 -0
  22. package/template/.agents/skills/apps/skybridge/references/publish.md +19 -0
  23. package/template/.agents/skills/apps/skybridge/references/run-locally.md +51 -0
  24. package/template/.agents/skills/apps/skybridge/references/state-and-context.md +151 -0
  25. package/template/.agents/skills/apps/skybridge/references/ui-guidelines.md +205 -0
  26. package/template/.agents/skills/code/cleanup/SKILL.md +26 -0
  27. package/template/.agents/skills/code/vercel-react-best-practices/AGENTS.md +3810 -0
  28. package/template/.agents/skills/code/vercel-react-best-practices/README.md +123 -0
  29. package/template/.agents/skills/code/vercel-react-best-practices/SKILL.md +149 -0
  30. package/template/.agents/skills/code/vercel-react-best-practices/metadata.json +15 -0
  31. package/template/.agents/skills/code/vercel-react-best-practices/rules/_sections.md +46 -0
  32. package/template/.agents/skills/code/vercel-react-best-practices/rules/_template.md +28 -0
  33. package/template/.agents/skills/code/vercel-react-best-practices/rules/advanced-effect-event-deps.md +56 -0
  34. package/template/.agents/skills/code/vercel-react-best-practices/rules/advanced-event-handler-refs.md +55 -0
  35. package/template/.agents/skills/code/vercel-react-best-practices/rules/advanced-init-once.md +42 -0
  36. package/template/.agents/skills/code/vercel-react-best-practices/rules/advanced-use-latest.md +39 -0
  37. package/template/.agents/skills/code/vercel-react-best-practices/rules/async-api-routes.md +38 -0
  38. package/template/.agents/skills/code/vercel-react-best-practices/rules/async-cheap-condition-before-await.md +37 -0
  39. package/template/.agents/skills/code/vercel-react-best-practices/rules/async-defer-await.md +82 -0
  40. package/template/.agents/skills/code/vercel-react-best-practices/rules/async-dependencies.md +51 -0
  41. package/template/.agents/skills/code/vercel-react-best-practices/rules/async-parallel.md +28 -0
  42. package/template/.agents/skills/code/vercel-react-best-practices/rules/async-suspense-boundaries.md +99 -0
  43. package/template/.agents/skills/code/vercel-react-best-practices/rules/bundle-analyzable-paths.md +63 -0
  44. package/template/.agents/skills/code/vercel-react-best-practices/rules/bundle-barrel-imports.md +60 -0
  45. package/template/.agents/skills/code/vercel-react-best-practices/rules/bundle-conditional.md +31 -0
  46. package/template/.agents/skills/code/vercel-react-best-practices/rules/bundle-defer-third-party.md +49 -0
  47. package/template/.agents/skills/code/vercel-react-best-practices/rules/bundle-dynamic-imports.md +35 -0
  48. package/template/.agents/skills/code/vercel-react-best-practices/rules/bundle-preload.md +50 -0
  49. package/template/.agents/skills/code/vercel-react-best-practices/rules/client-event-listeners.md +74 -0
  50. package/template/.agents/skills/code/vercel-react-best-practices/rules/client-localstorage-schema.md +71 -0
  51. package/template/.agents/skills/code/vercel-react-best-practices/rules/client-passive-event-listeners.md +48 -0
  52. package/template/.agents/skills/code/vercel-react-best-practices/rules/client-swr-dedup.md +56 -0
  53. package/template/.agents/skills/code/vercel-react-best-practices/rules/js-batch-dom-css.md +107 -0
  54. package/template/.agents/skills/code/vercel-react-best-practices/rules/js-cache-function-results.md +80 -0
  55. package/template/.agents/skills/code/vercel-react-best-practices/rules/js-cache-property-access.md +28 -0
  56. package/template/.agents/skills/code/vercel-react-best-practices/rules/js-cache-storage.md +70 -0
  57. package/template/.agents/skills/code/vercel-react-best-practices/rules/js-combine-iterations.md +32 -0
  58. package/template/.agents/skills/code/vercel-react-best-practices/rules/js-early-exit.md +50 -0
  59. package/template/.agents/skills/code/vercel-react-best-practices/rules/js-flatmap-filter.md +60 -0
  60. package/template/.agents/skills/code/vercel-react-best-practices/rules/js-hoist-regexp.md +45 -0
  61. package/template/.agents/skills/code/vercel-react-best-practices/rules/js-index-maps.md +37 -0
  62. package/template/.agents/skills/code/vercel-react-best-practices/rules/js-length-check-first.md +49 -0
  63. package/template/.agents/skills/code/vercel-react-best-practices/rules/js-min-max-loop.md +82 -0
  64. package/template/.agents/skills/code/vercel-react-best-practices/rules/js-request-idle-callback.md +105 -0
  65. package/template/.agents/skills/code/vercel-react-best-practices/rules/js-set-map-lookups.md +24 -0
  66. package/template/.agents/skills/code/vercel-react-best-practices/rules/js-tosorted-immutable.md +57 -0
  67. package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-activity.md +26 -0
  68. package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-animate-svg-wrapper.md +47 -0
  69. package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-conditional-render.md +40 -0
  70. package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-content-visibility.md +38 -0
  71. package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-hoist-jsx.md +46 -0
  72. package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-hydration-no-flicker.md +82 -0
  73. package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-hydration-suppress-warning.md +30 -0
  74. package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-resource-hints.md +85 -0
  75. package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-script-defer-async.md +68 -0
  76. package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-svg-precision.md +28 -0
  77. package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-usetransition-loading.md +75 -0
  78. package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-defer-reads.md +39 -0
  79. package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-dependencies.md +45 -0
  80. package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-derived-state-no-effect.md +40 -0
  81. package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-derived-state.md +29 -0
  82. package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-functional-setstate.md +74 -0
  83. package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-lazy-state-init.md +58 -0
  84. package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-memo-with-default-value.md +38 -0
  85. package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-memo.md +44 -0
  86. package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-move-effect-to-event.md +45 -0
  87. package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-no-inline-components.md +82 -0
  88. package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-simple-expression-in-memo.md +35 -0
  89. package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-split-combined-hooks.md +64 -0
  90. package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-transitions.md +40 -0
  91. package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-use-deferred-value.md +59 -0
  92. package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-use-ref-transient-values.md +73 -0
  93. package/template/.agents/skills/code/vercel-react-best-practices/rules/server-after-nonblocking.md +73 -0
  94. package/template/.agents/skills/code/vercel-react-best-practices/rules/server-auth-actions.md +96 -0
  95. package/template/.agents/skills/code/vercel-react-best-practices/rules/server-cache-lru.md +41 -0
  96. package/template/.agents/skills/code/vercel-react-best-practices/rules/server-cache-react.md +76 -0
  97. package/template/.agents/skills/code/vercel-react-best-practices/rules/server-dedup-props.md +65 -0
  98. package/template/.agents/skills/code/vercel-react-best-practices/rules/server-hoist-static-io.md +149 -0
  99. package/template/.agents/skills/code/vercel-react-best-practices/rules/server-no-shared-module-state.md +50 -0
  100. package/template/.agents/skills/code/vercel-react-best-practices/rules/server-parallel-fetching.md +83 -0
  101. package/template/.agents/skills/code/vercel-react-best-practices/rules/server-parallel-nested-fetching.md +34 -0
  102. package/template/.agents/skills/code/vercel-react-best-practices/rules/server-serialization.md +38 -0
  103. package/template/.agents/skills/data/prisma-cli/SKILL.md +247 -0
  104. package/template/.agents/skills/data/prisma-cli/references/db-execute.md +78 -0
  105. package/template/.agents/skills/data/prisma-cli/references/db-pull.md +185 -0
  106. package/template/.agents/skills/data/prisma-cli/references/db-push.md +148 -0
  107. package/template/.agents/skills/data/prisma-cli/references/db-seed.md +188 -0
  108. package/template/.agents/skills/data/prisma-cli/references/debug.md +46 -0
  109. package/template/.agents/skills/data/prisma-cli/references/dev.md +157 -0
  110. package/template/.agents/skills/data/prisma-cli/references/format.md +48 -0
  111. package/template/.agents/skills/data/prisma-cli/references/generate.md +173 -0
  112. package/template/.agents/skills/data/prisma-cli/references/init.md +136 -0
  113. package/template/.agents/skills/data/prisma-cli/references/mcp.md +38 -0
  114. package/template/.agents/skills/data/prisma-cli/references/migrate-deploy.md +127 -0
  115. package/template/.agents/skills/data/prisma-cli/references/migrate-dev.md +145 -0
  116. package/template/.agents/skills/data/prisma-cli/references/migrate-diff.md +89 -0
  117. package/template/.agents/skills/data/prisma-cli/references/migrate-reset.md +78 -0
  118. package/template/.agents/skills/data/prisma-cli/references/migrate-resolve.md +57 -0
  119. package/template/.agents/skills/data/prisma-cli/references/migrate-status.md +65 -0
  120. package/template/.agents/skills/data/prisma-cli/references/studio.md +137 -0
  121. package/template/.agents/skills/data/prisma-cli/references/validate.md +53 -0
  122. package/template/.agents/skills/data/prisma-client-api/SKILL.md +216 -0
  123. package/template/.agents/skills/data/prisma-client-api/references/client-methods.md +223 -0
  124. package/template/.agents/skills/data/prisma-client-api/references/constructor.md +208 -0
  125. package/template/.agents/skills/data/prisma-client-api/references/filters.md +256 -0
  126. package/template/.agents/skills/data/prisma-client-api/references/model-queries.md +281 -0
  127. package/template/.agents/skills/data/prisma-client-api/references/query-options.md +276 -0
  128. package/template/.agents/skills/data/prisma-client-api/references/raw-queries.md +194 -0
  129. package/template/.agents/skills/data/prisma-client-api/references/relations.md +308 -0
  130. package/template/.agents/skills/data/prisma-client-api/references/transactions.md +184 -0
  131. package/template/.agents/skills/design/impeccable/SKILL.md +176 -0
  132. package/template/.agents/skills/design/impeccable/reference/adapt.md +311 -0
  133. package/template/.agents/skills/design/impeccable/reference/animate.md +201 -0
  134. package/template/.agents/skills/design/impeccable/reference/audit.md +133 -0
  135. package/template/.agents/skills/design/impeccable/reference/bolder.md +113 -0
  136. package/template/.agents/skills/design/impeccable/reference/brand.md +108 -0
  137. package/template/.agents/skills/design/impeccable/reference/clarify.md +288 -0
  138. package/template/.agents/skills/design/impeccable/reference/codex.md +105 -0
  139. package/template/.agents/skills/design/impeccable/reference/colorize.md +257 -0
  140. package/template/.agents/skills/design/impeccable/reference/craft.md +123 -0
  141. package/template/.agents/skills/design/impeccable/reference/critique.md +767 -0
  142. package/template/.agents/skills/design/impeccable/reference/delight.md +302 -0
  143. package/template/.agents/skills/design/impeccable/reference/distill.md +111 -0
  144. package/template/.agents/skills/design/impeccable/reference/document.md +429 -0
  145. package/template/.agents/skills/design/impeccable/reference/extract.md +69 -0
  146. package/template/.agents/skills/design/impeccable/reference/harden.md +347 -0
  147. package/template/.agents/skills/design/impeccable/reference/init.md +172 -0
  148. package/template/.agents/skills/design/impeccable/reference/interaction-design.md +189 -0
  149. package/template/.agents/skills/design/impeccable/reference/layout.md +161 -0
  150. package/template/.agents/skills/design/impeccable/reference/live.md +718 -0
  151. package/template/.agents/skills/design/impeccable/reference/onboard.md +234 -0
  152. package/template/.agents/skills/design/impeccable/reference/optimize.md +258 -0
  153. package/template/.agents/skills/design/impeccable/reference/overdrive.md +130 -0
  154. package/template/.agents/skills/design/impeccable/reference/polish.md +241 -0
  155. package/template/.agents/skills/design/impeccable/reference/product.md +60 -0
  156. package/template/.agents/skills/design/impeccable/reference/quieter.md +99 -0
  157. package/template/.agents/skills/design/impeccable/reference/shape.md +165 -0
  158. package/template/.agents/skills/design/impeccable/reference/typeset.md +279 -0
  159. package/template/.agents/skills/design/impeccable/scripts/cleanup-deprecated.mjs +284 -0
  160. package/template/.agents/skills/design/impeccable/scripts/command-metadata.json +94 -0
  161. package/template/.agents/skills/design/impeccable/scripts/context-signals.mjs +225 -0
  162. package/template/.agents/skills/design/impeccable/scripts/context.mjs +270 -0
  163. package/template/.agents/skills/design/impeccable/scripts/critique-storage.mjs +242 -0
  164. package/template/.agents/skills/design/impeccable/scripts/design-parser.mjs +835 -0
  165. package/template/.agents/skills/design/impeccable/scripts/detect-csp.mjs +198 -0
  166. package/template/.agents/skills/design/impeccable/scripts/detect.mjs +21 -0
  167. package/template/.agents/skills/design/impeccable/scripts/detector/browser/injected/index.mjs +1733 -0
  168. package/template/.agents/skills/design/impeccable/scripts/detector/cli/main.mjs +244 -0
  169. package/template/.agents/skills/design/impeccable/scripts/detector/detect-antipatterns-browser.js +4551 -0
  170. package/template/.agents/skills/design/impeccable/scripts/detector/detect-antipatterns.mjs +43 -0
  171. package/template/.agents/skills/design/impeccable/scripts/detector/engines/browser/detect-url.mjs +252 -0
  172. package/template/.agents/skills/design/impeccable/scripts/detector/engines/regex/detect-text.mjs +535 -0
  173. package/template/.agents/skills/design/impeccable/scripts/detector/engines/static-html/css-cascade.mjs +986 -0
  174. package/template/.agents/skills/design/impeccable/scripts/detector/engines/static-html/detect-html.mjs +208 -0
  175. package/template/.agents/skills/design/impeccable/scripts/detector/engines/visual/screenshot-contrast.mjs +189 -0
  176. package/template/.agents/skills/design/impeccable/scripts/detector/findings.mjs +12 -0
  177. package/template/.agents/skills/design/impeccable/scripts/detector/node/file-system.mjs +198 -0
  178. package/template/.agents/skills/design/impeccable/scripts/detector/profile/profiler.mjs +166 -0
  179. package/template/.agents/skills/design/impeccable/scripts/detector/registry/antipatterns.mjs +419 -0
  180. package/template/.agents/skills/design/impeccable/scripts/detector/rules/checks.mjs +2316 -0
  181. package/template/.agents/skills/design/impeccable/scripts/detector/shared/color.mjs +124 -0
  182. package/template/.agents/skills/design/impeccable/scripts/detector/shared/constants.mjs +101 -0
  183. package/template/.agents/skills/design/impeccable/scripts/detector/shared/page.mjs +7 -0
  184. package/template/.agents/skills/design/impeccable/scripts/impeccable-paths.mjs +126 -0
  185. package/template/.agents/skills/design/impeccable/scripts/is-generated.mjs +69 -0
  186. package/template/.agents/skills/design/impeccable/scripts/live-accept.mjs +812 -0
  187. package/template/.agents/skills/design/impeccable/scripts/live-browser-session.js +123 -0
  188. package/template/.agents/skills/design/impeccable/scripts/live-browser.js +10316 -0
  189. package/template/.agents/skills/design/impeccable/scripts/live-commit-manual-edits.mjs +1241 -0
  190. package/template/.agents/skills/design/impeccable/scripts/live-complete.mjs +75 -0
  191. package/template/.agents/skills/design/impeccable/scripts/live-completion.mjs +19 -0
  192. package/template/.agents/skills/design/impeccable/scripts/live-copy-edit-agent.mjs +683 -0
  193. package/template/.agents/skills/design/impeccable/scripts/live-discard-manual-edits.mjs +51 -0
  194. package/template/.agents/skills/design/impeccable/scripts/live-event-validation.mjs +136 -0
  195. package/template/.agents/skills/design/impeccable/scripts/live-inject.mjs +557 -0
  196. package/template/.agents/skills/design/impeccable/scripts/live-insert-ui.mjs +458 -0
  197. package/template/.agents/skills/design/impeccable/scripts/live-insert.mjs +272 -0
  198. package/template/.agents/skills/design/impeccable/scripts/live-manual-edit-evidence.mjs +363 -0
  199. package/template/.agents/skills/design/impeccable/scripts/live-manual-edits-buffer.mjs +152 -0
  200. package/template/.agents/skills/design/impeccable/scripts/live-poll.mjs +379 -0
  201. package/template/.agents/skills/design/impeccable/scripts/live-resume.mjs +94 -0
  202. package/template/.agents/skills/design/impeccable/scripts/live-server.mjs +2322 -0
  203. package/template/.agents/skills/design/impeccable/scripts/live-session-store.mjs +289 -0
  204. package/template/.agents/skills/design/impeccable/scripts/live-status.mjs +61 -0
  205. package/template/.agents/skills/design/impeccable/scripts/live-svelte-component.mjs +826 -0
  206. package/template/.agents/skills/design/impeccable/scripts/live-sveltekit-adapter.mjs +274 -0
  207. package/template/.agents/skills/design/impeccable/scripts/live-ui-core.mjs +179 -0
  208. package/template/.agents/skills/design/impeccable/scripts/live-wrap.mjs +894 -0
  209. package/template/.agents/skills/design/impeccable/scripts/live.mjs +246 -0
  210. package/template/.agents/skills/design/impeccable/scripts/modern-screenshot.umd.js +14 -0
  211. package/template/.agents/skills/design/impeccable/scripts/palette.mjs +633 -0
  212. package/template/.agents/skills/design/impeccable/scripts/pin.mjs +214 -0
  213. package/template/.agents/skills/design/shadcn/SKILL.md +242 -0
  214. package/template/.agents/skills/design/shadcn/agents/openai.yml +5 -0
  215. package/template/.agents/skills/design/shadcn/assets/shadcn-small.png +0 -0
  216. package/template/.agents/skills/design/shadcn/assets/shadcn.png +0 -0
  217. package/template/.agents/skills/design/shadcn/cli.md +257 -0
  218. package/template/.agents/skills/design/shadcn/customization.md +202 -0
  219. package/template/.agents/skills/design/shadcn/evals/evals.json +47 -0
  220. package/template/.agents/skills/design/shadcn/mcp.md +94 -0
  221. package/template/.agents/skills/design/shadcn/rules/base-vs-radix.md +306 -0
  222. package/template/.agents/skills/design/shadcn/rules/composition.md +195 -0
  223. package/template/.agents/skills/design/shadcn/rules/forms.md +192 -0
  224. package/template/.agents/skills/design/shadcn/rules/icons.md +101 -0
  225. package/template/.agents/skills/design/shadcn/rules/styling.md +162 -0
  226. package/template/.agents/skills/find-skills/SKILL.md +142 -0
  227. package/template/.agents/skills/integrations/langfuse/SKILL.md +142 -0
  228. package/template/.agents/skills/integrations/langfuse/references/cli.md +52 -0
  229. package/template/.agents/skills/integrations/langfuse/references/error-analysis.md +100 -0
  230. package/template/.agents/skills/integrations/langfuse/references/instrumentation.md +134 -0
  231. package/template/.agents/skills/integrations/langfuse/references/judge-calibration.md +288 -0
  232. package/template/.agents/skills/integrations/langfuse/references/prompt-migration.md +234 -0
  233. package/template/.agents/skills/integrations/langfuse/references/sdk-upgrade.md +175 -0
  234. package/template/.agents/skills/integrations/langfuse/references/skill-feedback.md +52 -0
  235. package/template/.agents/skills/integrations/langfuse/references/user-feedback.md +88 -0
  236. package/template/.agents/skills/integrations/posthog/SKILL.md +102 -0
  237. package/template/.agents/skills/integrations/posthog/references/error-tracking-alerts.md +63 -0
  238. package/template/.agents/skills/integrations/posthog/references/error-tracking-assigning-issues.md +77 -0
  239. package/template/.agents/skills/integrations/posthog/references/error-tracking-fingerprints.md +57 -0
  240. package/template/.agents/skills/integrations/posthog/references/error-tracking-monitoring.md +140 -0
  241. package/template/.agents/skills/integrations/posthog/references/error-tracking-nextjs.md +490 -0
  242. package/template/.agents/skills/integrations/posthog/references/error-tracking-source-maps.md +45 -0
  243. package/template/.agents/skills/integrations/posthog/references/feature-flags-best-practices.md +139 -0
  244. package/template/.agents/skills/integrations/posthog/references/feature-flags-react.md +302 -0
  245. package/template/.agents/skills/integrations/posthog/references/identify-users.md +202 -0
  246. package/template/.agents/skills/integrations/posthog/references/integration-example.md +706 -0
  247. package/template/.agents/skills/integrations/posthog/references/integration-nextjs.md +385 -0
  248. package/template/.agents/skills/integrations/posthog/references/integration-step-1-begin.md +43 -0
  249. package/template/.agents/skills/integrations/posthog/references/integration-step-2-edit.md +37 -0
  250. package/template/.agents/skills/integrations/posthog/references/integration-step-3-revise.md +22 -0
  251. package/template/.agents/skills/integrations/posthog/references/integration-step-4-conclude.md +38 -0
  252. package/template/.agents/skills/integrations/posthog/references/llm-analytics-anthropic.md +200 -0
  253. package/template/.agents/skills/integrations/posthog/references/llm-analytics-basics.md +62 -0
  254. package/template/.agents/skills/integrations/posthog/references/llm-analytics-costs.md +197 -0
  255. package/template/.agents/skills/integrations/posthog/references/llm-analytics-manual-capture.md +397 -0
  256. package/template/.agents/skills/integrations/posthog/references/llm-analytics-traces.md +98 -0
  257. package/template/.agents/skills/integrations/posthog/references/llm-analytics-vercel-ai.md +120 -0
  258. package/template/.agents/skills/repo/repo-ci/SKILL.md +265 -0
  259. package/template/.agents/skills/repo/repo-init-next-js/SKILL.md +129 -0
  260. package/template/.agents/skills/repo/repo-init-next-js/references/file-contents.md +800 -0
  261. package/template/.agents/skills/repo/repo-init-next-js/scripts/setup.sh +47 -0
  262. package/template/.agents/skills/repo/repo-init-node/SKILL.md +196 -0
  263. package/template/.agents/skills/skill-creator/LICENSE.txt +202 -0
  264. package/template/.agents/skills/skill-creator/SKILL.md +485 -0
  265. package/template/.agents/skills/skill-creator/agents/analyzer.md +274 -0
  266. package/template/.agents/skills/skill-creator/agents/comparator.md +202 -0
  267. package/template/.agents/skills/skill-creator/agents/grader.md +223 -0
  268. package/template/.agents/skills/skill-creator/assets/eval_review.html +146 -0
  269. package/template/.agents/skills/skill-creator/eval-viewer/generate_review.py +471 -0
  270. package/template/.agents/skills/skill-creator/eval-viewer/viewer.html +1325 -0
  271. package/template/.agents/skills/skill-creator/references/schemas.md +430 -0
  272. package/template/.agents/skills/skill-creator/scripts/__init__.py +0 -0
  273. package/template/.agents/skills/skill-creator/scripts/aggregate_benchmark.py +401 -0
  274. package/template/.agents/skills/skill-creator/scripts/generate_report.py +326 -0
  275. package/template/.agents/skills/skill-creator/scripts/improve_description.py +247 -0
  276. package/template/.agents/skills/skill-creator/scripts/package_skill.py +136 -0
  277. package/template/.agents/skills/skill-creator/scripts/quick_validate.py +103 -0
  278. package/template/.agents/skills/skill-creator/scripts/run_eval.py +310 -0
  279. package/template/.agents/skills/skill-creator/scripts/run_loop.py +328 -0
  280. package/template/.agents/skills/skill-creator/scripts/utils.py +47 -0
  281. package/template/.agents/upstreams.json +80 -0
@@ -0,0 +1,152 @@
1
+ /**
2
+ * Shared helpers for the pending-manual-edits buffer on disk.
3
+ *
4
+ * Location: .impeccable/live/pending-manual-edits.json (project-local).
5
+ * Schema: { version: 1, entries: [{ id, pageUrl, element, ops, stagedAt }] }
6
+ *
7
+ * Each entry corresponds to one Save action from the browser. Ops merge by
8
+ * (pageUrl, ref): if the user re-edits the same element before committing, the
9
+ * existing entry's `newText` is replaced and `originalText` is kept (it holds
10
+ * the real source state).
11
+ */
12
+
13
+ import fs from 'node:fs';
14
+ import path from 'node:path';
15
+ import { getLiveDir } from './impeccable-paths.mjs';
16
+
17
+ const BUFFER_VERSION = 1;
18
+ const BUFFER_FILENAME = 'pending-manual-edits.json';
19
+
20
+ export function getBufferPath(cwd = process.cwd()) {
21
+ return path.join(getLiveDir(cwd), BUFFER_FILENAME);
22
+ }
23
+
24
+ export function readBuffer(cwd = process.cwd()) {
25
+ return readBufferInternal(cwd, { strict: false });
26
+ }
27
+
28
+ export function readBufferStrict(cwd = process.cwd()) {
29
+ return readBufferInternal(cwd, { strict: true });
30
+ }
31
+
32
+ function readBufferInternal(cwd, { strict }) {
33
+ const filePath = getBufferPath(cwd);
34
+ try {
35
+ const raw = fs.readFileSync(filePath, 'utf-8');
36
+ const parsed = JSON.parse(raw);
37
+ if (!parsed || typeof parsed !== 'object' || !Array.isArray(parsed.entries)) {
38
+ if (strict) throw new Error('manual_edit_buffer_invalid_schema');
39
+ return { version: BUFFER_VERSION, entries: [] };
40
+ }
41
+ return { version: BUFFER_VERSION, entries: parsed.entries };
42
+ } catch (err) {
43
+ if (strict && err?.code !== 'ENOENT') {
44
+ throw new Error('manual_edit_buffer_unreadable: ' + (err.message || String(err)));
45
+ }
46
+ return { version: BUFFER_VERSION, entries: [] };
47
+ }
48
+ }
49
+
50
+ export function writeBuffer(cwd, buffer) {
51
+ const filePath = getBufferPath(cwd);
52
+ fs.mkdirSync(path.dirname(filePath), { recursive: true });
53
+ fs.writeFileSync(filePath, JSON.stringify({ version: BUFFER_VERSION, entries: buffer.entries }, null, 2));
54
+ }
55
+
56
+ /**
57
+ * Merge a new entry into the buffer. For each op in the new entry, if there's
58
+ * already a buffered op for the same (pageUrl, ref), update that op's newText
59
+ * and keep its original originalText (the true source state). Otherwise add
60
+ * the op (creating an entry if needed).
61
+ *
62
+ * Multiple ops in one Save are allowed; each is keyed by (pageUrl, ref).
63
+ */
64
+ export function stageEntry(cwd, newEntry) {
65
+ const buf = readBufferStrict(cwd);
66
+ const pageUrl = newEntry.pageUrl;
67
+ for (const newOp of newEntry.ops) {
68
+ let mergedIntoExisting = false;
69
+ for (const existing of buf.entries) {
70
+ if (existing.pageUrl !== pageUrl) continue;
71
+ const existingOpIdx = existing.ops.findIndex((op) => op.ref === newOp.ref);
72
+ if (existingOpIdx >= 0) {
73
+ // Keep the original source text but refresh the latest DOM/source evidence.
74
+ existing.ops[existingOpIdx] = {
75
+ ...newOp,
76
+ originalText: existing.ops[existingOpIdx].originalText,
77
+ newText: newOp.newText,
78
+ deleted: newOp.deleted || false,
79
+ };
80
+ if (newEntry.element) existing.element = newEntry.element;
81
+ existing.stagedAt = new Date().toISOString();
82
+ mergedIntoExisting = true;
83
+ break;
84
+ }
85
+ }
86
+ if (mergedIntoExisting) continue;
87
+ // No existing op for this (pageUrl, ref). Find or create an entry to hold it.
88
+ let entry = buf.entries.find((e) => e.pageUrl === pageUrl && e.id === newEntry.id);
89
+ if (!entry) {
90
+ entry = {
91
+ id: newEntry.id,
92
+ pageUrl,
93
+ element: newEntry.element,
94
+ ops: [],
95
+ stagedAt: new Date().toISOString(),
96
+ };
97
+ buf.entries.push(entry);
98
+ }
99
+ entry.ops.push(newOp);
100
+ entry.stagedAt = new Date().toISOString();
101
+ }
102
+ writeBuffer(cwd, buf);
103
+ return buf;
104
+ }
105
+
106
+ /**
107
+ * Remove entries matching a predicate. Returns count of removed *ops* (not
108
+ * entries) so callers report a unit consistent with truncateBuffer and the
109
+ * pill's per-page op count. Empty entries (no ops left) are also pruned.
110
+ */
111
+ export function removeEntries(cwd, predicate) {
112
+ const buf = readBuffer(cwd);
113
+ let removedOps = 0;
114
+ const kept = [];
115
+ for (const entry of buf.entries) {
116
+ if (predicate(entry)) {
117
+ removedOps += entry.ops?.length || 0;
118
+ } else if (entry.ops && entry.ops.length > 0) {
119
+ kept.push(entry);
120
+ }
121
+ }
122
+ buf.entries = kept;
123
+ writeBuffer(cwd, buf);
124
+ return removedOps;
125
+ }
126
+
127
+ /**
128
+ * Count by page for the counter UI. Returns { totalCount, perPage: {[pageUrl]: count} }.
129
+ */
130
+ export function countByPage(cwd = process.cwd()) {
131
+ const buf = readBuffer(cwd);
132
+ const perPage = {};
133
+ let totalCount = 0;
134
+ for (const entry of buf.entries) {
135
+ const n = entry.ops.length;
136
+ perPage[entry.pageUrl] = (perPage[entry.pageUrl] || 0) + n;
137
+ totalCount += n;
138
+ }
139
+ return { totalCount, perPage };
140
+ }
141
+
142
+ /**
143
+ * Truncate the buffer to empty (used by discard-all). Returns the count of
144
+ * removed ops.
145
+ */
146
+ export function truncateBuffer(cwd) {
147
+ const buf = readBuffer(cwd);
148
+ let removed = 0;
149
+ for (const entry of buf.entries) removed += entry.ops.length;
150
+ writeBuffer(cwd, { version: BUFFER_VERSION, entries: [] });
151
+ return removed;
152
+ }
@@ -0,0 +1,379 @@
1
+ /**
2
+ * CLI client for the live variant mode poll/reply protocol.
3
+ *
4
+ * Usage:
5
+ * npx impeccable poll # Block until browser event, print JSON
6
+ * npx impeccable poll --stream # Experimental: keep polling; one JSON line per event
7
+ * npx impeccable poll --timeout=600000 # Custom timeout (ms); default is long-poll friendly
8
+ * npx impeccable poll --reply <id> done # Reply "done" to event <id>
9
+ * npx impeccable poll --reply <id> error "msg" # Reply with error
10
+ */
11
+
12
+ import { execFileSync } from 'node:child_process';
13
+ import path from 'node:path';
14
+ import { fileURLToPath } from 'node:url';
15
+ import { completionAckForAcceptResult, completionTypeForAcceptResult } from './live-completion.mjs';
16
+ import { readLiveServerInfo } from './impeccable-paths.mjs';
17
+
18
+ // Node's built-in fetch (undici under the hood) enforces a 300s headers
19
+ // timeout that can't be lowered per-request. We cap each request below
20
+ // that ceiling and loop in `pollOnce` to synthesize a long poll without
21
+ // depending on the standalone undici package.
22
+ export const PER_REQUEST_TIMEOUT_MS = 270_000;
23
+ export const DEFAULT_EVENT_LEASE_MS = 600_000;
24
+
25
+ const EVENT_TYPES_NEEDING_AGENT_REPLY = new Set(['generate', 'steer', 'manual_edit_apply']);
26
+
27
+ function readServerInfo() {
28
+ const record = readLiveServerInfo(process.cwd());
29
+ if (!record) {
30
+ console.error('No running live server found. Start one with: npx impeccable live');
31
+ process.exit(1);
32
+ }
33
+ return record.info;
34
+ }
35
+
36
+ export function buildPollReplyPayload(token, { id, type, message, file, data }) {
37
+ return { token, id, type, message, file, data };
38
+ }
39
+
40
+ export function manualApplyPollBanner(event = {}) {
41
+ const id = event.id || 'EVENT_ID';
42
+ return [
43
+ `Manual Apply action required: edit source, then reply with \`live-poll.mjs --reply ${id} done --data '<json>'\`.`,
44
+ 'The JSON data must include status, appliedEntryIds, failed, files, and notes; summary counters are only a recovery fallback.',
45
+ 'Do not run live-commit-manual-edits.mjs for this leased event.',
46
+ 'Do not poll again before replying.',
47
+ ].join('\n') + '\n';
48
+ }
49
+
50
+ /**
51
+ * Parse `--reply <id> <status> [--file path] [--data '<json>'] [message]` argv
52
+ * into a reply object. Returns null when `--reply` is absent. Throws (code
53
+ * INVALID_REPLY_ARGS) when the reply shape is missing its event id/status and
54
+ * INVALID_DATA_JSON when `--data` is present but not valid JSON.
55
+ */
56
+ export function parseReplyArgs(args) {
57
+ const replyIdx = args.indexOf('--reply');
58
+ if (replyIdx === -1) return null;
59
+ const id = args[replyIdx + 1];
60
+ const status = args[replyIdx + 2];
61
+ validateReplyArgs({ id, status });
62
+ const fileIdx = args.indexOf('--file');
63
+ const file = fileIdx !== -1 && fileIdx + 1 < args.length ? args[fileIdx + 1] : undefined;
64
+ const dataIdx = args.indexOf('--data');
65
+ let data;
66
+ if (dataIdx !== -1 && dataIdx + 1 < args.length) {
67
+ try {
68
+ data = JSON.parse(args[dataIdx + 1]);
69
+ } catch (err) {
70
+ const wrapped = new Error('--data must be valid JSON: ' + err.message);
71
+ wrapped.code = 'INVALID_DATA_JSON';
72
+ throw wrapped;
73
+ }
74
+ }
75
+ const message = args.find((a, i) =>
76
+ i > replyIdx + 2
77
+ && !a.startsWith('--')
78
+ && i !== fileIdx + 1
79
+ && i !== dataIdx + 1
80
+ ) || undefined;
81
+ return { id, type: status, message, file, data };
82
+ }
83
+
84
+ function validateReplyArgs({ id, status }) {
85
+ const usage = "Usage: npx impeccable poll --reply <id> <status> [--file path] [--data '<json>'] [message]";
86
+ if (!id || id.startsWith('--')) {
87
+ const err = new Error(`${usage}\nMissing event id after --reply.`);
88
+ err.code = 'INVALID_REPLY_ARGS';
89
+ throw err;
90
+ }
91
+ if (['done', 'error', 'complete', 'discard', 'discarded'].includes(id)) {
92
+ const err = new Error(`${usage}\nThe value after --reply must be the event id, not the status ${JSON.stringify(id)}. Use --reply EVENT_ID ${id}.`);
93
+ err.code = 'INVALID_REPLY_ARGS';
94
+ throw err;
95
+ }
96
+ if (!status || status.startsWith('--')) {
97
+ const err = new Error(`${usage}\nMissing reply status after event id ${JSON.stringify(id)}.`);
98
+ err.code = 'INVALID_REPLY_ARGS';
99
+ throw err;
100
+ }
101
+ }
102
+
103
+ export function requiresAgentReply(event) {
104
+ return EVENT_TYPES_NEEDING_AGENT_REPLY.has(event?.type);
105
+ }
106
+
107
+ export async function postReply(base, token, reply) {
108
+ const res = await fetch(`${base}/poll`, {
109
+ method: 'POST',
110
+ headers: { 'Content-Type': 'application/json' },
111
+ body: JSON.stringify(buildPollReplyPayload(token, reply)),
112
+ });
113
+ if (!res.ok) {
114
+ const body = await res.json().catch(() => ({}));
115
+ const parts = [body.error || res.statusText, body.reason, body.hint].filter(Boolean);
116
+ throw new Error(parts.join(': '));
117
+ }
118
+ }
119
+
120
+ export async function fetchServerStatus(base, token) {
121
+ const res = await fetch(`${base}/status?token=${token}`);
122
+ if (res.status === 401) {
123
+ const err = new Error('Authentication failed. The server token may have changed.');
124
+ err.code = 'AUTH_FAILED';
125
+ throw err;
126
+ }
127
+ if (!res.ok) {
128
+ throw new Error(`Status failed: ${res.status} ${res.statusText}`);
129
+ }
130
+ return res.json();
131
+ }
132
+
133
+ export function isEventPending(status, eventId) {
134
+ return (status.pendingEvents || []).some((entry) => entry.id === eventId);
135
+ }
136
+
137
+ export async function waitForEventAck(base, token, eventId, {
138
+ pollIntervalMs = 400,
139
+ maxWaitMs = 600_000,
140
+ } = {}) {
141
+ const deadline = Date.now() + maxWaitMs;
142
+ while (Date.now() < deadline) {
143
+ const status = await fetchServerStatus(base, token);
144
+ if (!isEventPending(status, eventId)) return true;
145
+ await new Promise((resolve) => setTimeout(resolve, pollIntervalMs));
146
+ }
147
+ return false;
148
+ }
149
+
150
+ export async function fetchNextEvent(base, token, { totalDeadline } = {}) {
151
+ while (true) {
152
+ if (totalDeadline && Date.now() >= totalDeadline) {
153
+ return { type: 'timeout' };
154
+ }
155
+
156
+ const remaining = totalDeadline
157
+ ? totalDeadline - Date.now()
158
+ : PER_REQUEST_TIMEOUT_MS;
159
+ const slice = Math.min(Math.max(remaining, 1000), PER_REQUEST_TIMEOUT_MS);
160
+ const res = await fetch(`${base}/poll?token=${token}&timeout=${slice}&leaseMs=${DEFAULT_EVENT_LEASE_MS}`);
161
+
162
+ if (res.status === 401) {
163
+ const err = new Error('Authentication failed. The server token may have changed.');
164
+ err.code = 'AUTH_FAILED';
165
+ throw err;
166
+ }
167
+
168
+ if (!res.ok) {
169
+ throw new Error(`Poll failed: ${res.status} ${res.statusText}`);
170
+ }
171
+
172
+ const next = await res.json();
173
+ if (next?.type === 'timeout') {
174
+ if (totalDeadline && Date.now() < totalDeadline) continue;
175
+ if (!totalDeadline) continue;
176
+ return next;
177
+ }
178
+ return next;
179
+ }
180
+ }
181
+
182
+ export async function augmentEventWithAcceptHandling(event, base, token) {
183
+ if (event.type !== 'accept' && event.type !== 'discard') return event;
184
+
185
+ const __dirname = path.dirname(fileURLToPath(import.meta.url));
186
+ const acceptScript = path.join(__dirname, 'live-accept.mjs');
187
+ const scriptArgs = buildAcceptScriptArgs(event);
188
+
189
+ try {
190
+ const out = execFileSync(
191
+ 'node',
192
+ [acceptScript, ...scriptArgs],
193
+ { encoding: 'utf-8', cwd: process.cwd(), timeout: 30_000 },
194
+ );
195
+ event._acceptResult = JSON.parse(out.trim());
196
+ } catch (err) {
197
+ event._acceptResult = { handled: false, mode: 'error', error: err.message };
198
+ }
199
+
200
+ const completionType = completionTypeForAcceptResult(event.type, event._acceptResult);
201
+ try {
202
+ await postReply(base, token, {
203
+ id: event.id,
204
+ type: completionType,
205
+ message: event._acceptResult?.error,
206
+ file: event._acceptResult?.file,
207
+ data: event._acceptResult?.carbonize === true ? { carbonize: true } : undefined,
208
+ });
209
+ } catch (err) {
210
+ event._completionAck = { ok: false, error: err.message };
211
+ }
212
+ if (!event._completionAck) {
213
+ event._completionAck = completionAckForAcceptResult(event.id, completionType, event._acceptResult);
214
+ }
215
+
216
+ return event;
217
+ }
218
+
219
+ export function buildAcceptScriptArgs(event) {
220
+ const scriptArgs = event.type === 'discard'
221
+ ? ['--id', String(event.id), '--discard']
222
+ : ['--id', String(event.id), '--variant', String(event.variantId)];
223
+ if (event.pageUrl) scriptArgs.push('--page-url', String(event.pageUrl));
224
+ if (event.type === 'accept' && event.paramValues && Object.keys(event.paramValues).length > 0) {
225
+ scriptArgs.push('--param-values', JSON.stringify(event.paramValues));
226
+ }
227
+ return scriptArgs;
228
+ }
229
+
230
+ export function writeCarbonizeBanner(event) {
231
+ if (event.type === 'manual_edit_apply') {
232
+ process.stderr.write('\n' + manualApplyPollBanner(event) + '\n');
233
+ }
234
+ if (event._acceptResult?.carbonize === true) {
235
+ process.stderr.write('\n⚠ Carbonize cleanup REQUIRED before next poll. After cleanup, run live-complete.mjs --id ' + event.id + '. See reference/live.md "Required after accept".\n\n');
236
+ }
237
+ }
238
+
239
+ export function printPollEvent(event) {
240
+ console.log(JSON.stringify(event));
241
+ }
242
+
243
+ export async function runPollOnce(base, token, { totalTimeout = 600_000 } = {}) {
244
+ const deadline = Date.now() + totalTimeout;
245
+ const event = await fetchNextEvent(base, token, { totalDeadline: deadline });
246
+ await augmentEventWithAcceptHandling(event, base, token);
247
+ writeCarbonizeBanner(event);
248
+ printPollEvent(event);
249
+ return event;
250
+ }
251
+
252
+ export async function runPollStream(base, token, {
253
+ ackTimeoutMs = 600_000,
254
+ ackPollIntervalMs = 400,
255
+ shouldContinue = () => true,
256
+ } = {}) {
257
+ process.stderr.write('[impeccable-poll] stream mode: one JSON object per line on stdout; use --reply while this process stays running\n');
258
+
259
+ while (shouldContinue()) {
260
+ const event = await fetchNextEvent(base, token);
261
+ await augmentEventWithAcceptHandling(event, base, token);
262
+ writeCarbonizeBanner(event);
263
+ printPollEvent(event);
264
+
265
+ if (event.type === 'exit') return event;
266
+
267
+ if (requiresAgentReply(event)) {
268
+ const acked = await waitForEventAck(base, token, event.id, {
269
+ pollIntervalMs: ackPollIntervalMs,
270
+ maxWaitMs: ackTimeoutMs,
271
+ });
272
+ if (!acked) {
273
+ const err = new Error(`Timed out waiting for --reply on event ${event.id}`);
274
+ err.code = 'ACK_TIMEOUT';
275
+ throw err;
276
+ }
277
+ }
278
+ }
279
+
280
+ return null;
281
+ }
282
+
283
+ function handlePollError(err) {
284
+ if (err.code === 'AUTH_FAILED') {
285
+ console.error(err.message);
286
+ console.error('Try restarting: npx impeccable live stop && npx impeccable live');
287
+ process.exit(1);
288
+ }
289
+ if (err.cause?.code === 'ECONNREFUSED') {
290
+ console.error('Live server not running. Start one with: npx impeccable live');
291
+ process.exit(1);
292
+ }
293
+ if (err.code === 'ACK_TIMEOUT') {
294
+ console.error(err.message);
295
+ process.exit(1);
296
+ }
297
+ console.error('Poll failed:', err.message);
298
+ process.exit(1);
299
+ }
300
+
301
+ export async function pollCli() {
302
+ const args = process.argv.slice(2);
303
+
304
+ if (args.includes('--help') || args.includes('-h')) {
305
+ console.log(`Usage: impeccable poll [options]
306
+
307
+ Wait for a browser event from the live variant server, or reply to one.
308
+
309
+ Modes:
310
+ poll Block until a browser event arrives, print JSON, exit
311
+ poll --stream Keep polling; print one JSON line per event (see live.md)
312
+ poll --reply <id> done Reply "done" to event <id> (replace or insert generate)
313
+ poll --reply <id> steer_done Reply after handling a steer event (unlocks Steer bar)
314
+ poll --reply <id> error "msg" Reply with an error message
315
+ poll --reply <id> done --data '<json>'
316
+ Reply with a structured JSON result (manual_edit_apply)
317
+
318
+ Options:
319
+ --timeout=MS One-shot poll timeout in ms (default: 600000). Ignored in --stream mode
320
+ --ack-timeout=MS Stream mode: max wait for --reply after generate/steer (default: 600000)
321
+ --file PATH Attach a source file path to the reply (generate/steer flow)
322
+ --data JSON Attach a JSON result object to the reply (manual_edit_apply flow). Must be valid JSON
323
+ --help Show this help message
324
+
325
+ Harness note:
326
+ Default one-shot mode is the portable contract for Claude Code, Codex, and Cursor.
327
+ --stream is experimental for harnesses with fast incremental stdout; do not use on Cursor.`);
328
+ process.exit(0);
329
+ }
330
+
331
+ const info = readServerInfo();
332
+ const base = `http://localhost:${info.port}`;
333
+
334
+ // Reply mode: npx impeccable poll --reply <id> <status> [--file path] [--data '<json>'] [message]
335
+ if (args.includes('--reply')) {
336
+ let reply;
337
+ try {
338
+ reply = parseReplyArgs(args);
339
+ } catch (err) {
340
+ console.error(err.message);
341
+ process.exit(1);
342
+ }
343
+
344
+ try {
345
+ await postReply(base, info.token, reply);
346
+ } catch (err) {
347
+ if (err.cause?.code === 'ECONNREFUSED') {
348
+ console.error('Live server not running. Start one with: npx impeccable live');
349
+ } else {
350
+ console.error('Reply failed:', err.message);
351
+ }
352
+ process.exit(1);
353
+ }
354
+ return;
355
+ }
356
+
357
+ const streamMode = args.includes('--stream');
358
+ const ackTimeoutArg = args.find((a) => a.startsWith('--ack-timeout='));
359
+ const ackTimeoutMs = ackTimeoutArg ? parseInt(ackTimeoutArg.split('=')[1], 10) : 600_000;
360
+
361
+ try {
362
+ if (streamMode) {
363
+ await runPollStream(base, info.token, { ackTimeoutMs });
364
+ return;
365
+ }
366
+
367
+ const timeoutArg = args.find((a) => a.startsWith('--timeout='));
368
+ const totalTimeout = timeoutArg ? parseInt(timeoutArg.split('=')[1], 10) : 600_000;
369
+ await runPollOnce(base, info.token, { totalTimeout });
370
+ } catch (err) {
371
+ handlePollError(err);
372
+ }
373
+ }
374
+
375
+ // Auto-execute when run directly
376
+ const _running = process.argv[1];
377
+ if (_running?.endsWith('live-poll.mjs') || _running?.endsWith('live-poll.mjs/')) {
378
+ pollCli();
379
+ }
@@ -0,0 +1,94 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * Recover the next agent action from the durable live-session journal.
4
+ */
5
+
6
+ import { createLiveSessionStore } from './live-session-store.mjs';
7
+
8
+ function manualApplyReplyCommand(eventOrId = 'EVENT_ID') {
9
+ const id = typeof eventOrId === 'string' ? eventOrId : eventOrId?.id || 'EVENT_ID';
10
+ return `live-poll.mjs --reply ${id} done --data '<json>'`;
11
+ }
12
+
13
+ export function manualApplyResumeHint(event = {}) {
14
+ const summary = event.manualApplySummary || summarizeManualApplyEvent(event);
15
+ const parts = [];
16
+ if (summary.pageUrl) parts.push(`page ${summary.pageUrl}`);
17
+ if (summary.chunk) parts.push(`chunk ${summary.chunk.index}/${summary.chunk.total}`);
18
+ if (Number.isFinite(summary.opCount)) parts.push(`${summary.opCount} op(s)`);
19
+ if (Number.isFinite(summary.entryCount)) parts.push(`${summary.entryCount} entr${summary.entryCount === 1 ? 'y' : 'ies'}`);
20
+ if (summary.files?.length) parts.push(`likely files: ${summary.files.join(', ')}`);
21
+ const scope = parts.length ? ` (${parts.join(', ')})` : '';
22
+ return `Manual Apply pending${scope}. If you have not already leased it, run live-poll.mjs. Apply the source edits from the manual_edit_apply batch, then reply with ${manualApplyReplyCommand(event.id)}. Polling only leases this work item; it does not commit source edits. Do not run live-commit-manual-edits.mjs for this leased event. Do not poll again before replying.`;
23
+ }
24
+
25
+ function summarizeManualApplyEvent(event = {}) {
26
+ const entries = Array.isArray(event.batch?.entries) ? event.batch.entries : [];
27
+ const opCount = entries.reduce((sum, entry) => sum + (Array.isArray(entry.ops) ? entry.ops.length : 0), 0);
28
+ return {
29
+ pageUrl: event.pageUrl || null,
30
+ chunk: event.chunk || null,
31
+ entryCount: entries.length,
32
+ opCount,
33
+ files: collectManualApplyFiles(event.batch),
34
+ };
35
+ }
36
+
37
+ function collectManualApplyFiles(batch) {
38
+ const files = [];
39
+ for (const entry of batch?.entries || []) {
40
+ for (const op of entry.ops || []) files.push(op.sourceHint?.file);
41
+ }
42
+ for (const candidate of batch?.candidates || []) {
43
+ files.push(candidate.sourceHint?.relativeFile, candidate.sourceHint?.file);
44
+ for (const item of candidate.textMatches || []) files.push(item.file);
45
+ for (const item of candidate.objectKeyMatches || []) files.push(item.file);
46
+ for (const item of candidate.locatorMatches || []) files.push(item.file);
47
+ for (const item of candidate.contextTextMatches || []) files.push(item.file);
48
+ }
49
+ return [...new Set(files.filter((file) => typeof file === 'string' && file.length > 0))].sort();
50
+ }
51
+
52
+ function parseArgs(argv) {
53
+ const out = { id: null };
54
+ for (let i = 0; i < argv.length; i++) {
55
+ const arg = argv[i];
56
+ if (arg === '--id') out.id = argv[++i];
57
+ else if (arg.startsWith('--id=')) out.id = arg.slice('--id='.length);
58
+ else if (arg === '--help' || arg === '-h') out.help = true;
59
+ }
60
+ return out;
61
+ }
62
+
63
+ export async function resumeCli() {
64
+ const args = parseArgs(process.argv.slice(2));
65
+ if (args.help) {
66
+ console.log(`Usage: node live-resume.mjs [--id SESSION_ID]\n\nPrint the active durable session checkpoint and the next safe agent action.`);
67
+ return;
68
+ }
69
+
70
+ const store = createLiveSessionStore({ cwd: process.cwd(), sessionId: args.id || undefined });
71
+ const snapshot = args.id ? store.getSnapshot(args.id) : store.listActiveSessions()[0] || null;
72
+ if (!snapshot) {
73
+ console.log(JSON.stringify({ active: false, nextAction: 'No active durable live session found.' }, null, 2));
74
+ return;
75
+ }
76
+
77
+ const pending = snapshot.pendingEvent || null;
78
+ const nextAction = pending
79
+ ? pending.type === 'manual_edit_apply'
80
+ ? manualApplyResumeHint(pending)
81
+ : `Run live-poll.mjs, handle ${pending.type} ${pending.id}, then acknowledge with live-poll.mjs --reply ${pending.id} done.`
82
+ : snapshot.phase === 'carbonize_required'
83
+ ? `Finish carbonize cleanup${snapshot.sourceFile ? ` in ${snapshot.sourceFile}` : ''}, then run live-complete.mjs --id ${snapshot.id}.`
84
+ : snapshot.phase === 'accept_requested'
85
+ ? `Run live-complete.mjs --id ${snapshot.id} after verifying the accepted variant is written.`
86
+ : `Inspect ${snapshot.id}; no pending agent event is currently queued.`;
87
+
88
+ console.log(JSON.stringify({ active: true, snapshot, pendingEvent: pending, nextAction }, null, 2));
89
+ }
90
+
91
+ const _running = process.argv[1];
92
+ if (_running?.endsWith('live-resume.mjs') || _running?.endsWith('live-resume.mjs/')) {
93
+ resumeCli();
94
+ }