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,272 @@
1
+ /**
2
+ * CLI helper: find an anchor element in source and splice an insert-variant
3
+ * wrapper before or after it (no original variant — net-new content).
4
+ *
5
+ * Usage:
6
+ * node live-insert.mjs --id SESSION_ID --count N --position after \
7
+ * --classes "hero" --tag section [--file path]
8
+ */
9
+
10
+ import fs from 'node:fs';
11
+ import path from 'node:path';
12
+ import { isGeneratedFile } from './is-generated.mjs';
13
+ import {
14
+ buildSearchQueries,
15
+ findElement,
16
+ findAllElements,
17
+ filterByText,
18
+ findFileWithQuery,
19
+ detectCommentSyntax,
20
+ detectStyleMode,
21
+ buildCssAuthoring,
22
+ buildCssSelectorPrefixExamples,
23
+ } from './live-wrap.mjs';
24
+ import {
25
+ buildSvelteComponentCssAuthoring,
26
+ scaffoldSvelteComponentInsertSession,
27
+ shouldUseSvelteComponentInjection,
28
+ } from './live-svelte-component.mjs';
29
+
30
+ const INSERT_POSITIONS = new Set(['before', 'after']);
31
+
32
+ export function isInsertPosition(value) {
33
+ return INSERT_POSITIONS.has(value);
34
+ }
35
+
36
+ export function computeInsertLine(startLine, endLine, position) {
37
+ return position === 'before' ? startLine : endLine + 1;
38
+ }
39
+
40
+ export function buildInsertWrapperLines({ id, count, indent, commentSyntax, isJsx }) {
41
+ const styleContents = isJsx ? 'style={{ display: "contents" }}' : 'style="display: contents"';
42
+ const attrs =
43
+ 'data-impeccable-variants="' + id + '" ' +
44
+ 'data-impeccable-mode="insert" ' +
45
+ 'data-impeccable-variant-count="' + count + '" ' +
46
+ styleContents;
47
+
48
+ if (isJsx) {
49
+ return [
50
+ indent + '<div ' + attrs + '>',
51
+ indent + ' ' + commentSyntax.open + ' impeccable-variants-start ' + id + ' ' + commentSyntax.close,
52
+ indent + ' ' + commentSyntax.open + ' Variants: insert below this line ' + commentSyntax.close,
53
+ indent + ' ' + commentSyntax.open + ' impeccable-variants-end ' + id + ' ' + commentSyntax.close,
54
+ indent + '</div>',
55
+ ];
56
+ }
57
+
58
+ return [
59
+ indent + commentSyntax.open + ' impeccable-variants-start ' + id + ' ' + commentSyntax.close,
60
+ indent + '<div ' + attrs + '>',
61
+ indent + ' ' + commentSyntax.open + ' Variants: insert below this line ' + commentSyntax.close,
62
+ indent + '</div>',
63
+ indent + commentSyntax.open + ' impeccable-variants-end ' + id + ' ' + commentSyntax.close,
64
+ ];
65
+ }
66
+
67
+ function argVal(args, flag) {
68
+ const idx = args.indexOf(flag);
69
+ return idx !== -1 && idx + 1 < args.length ? args[idx + 1] : null;
70
+ }
71
+
72
+ function resolveElementMatch({ lines, queries, tag, text }) {
73
+ if (text) {
74
+ const candidates = [];
75
+ for (const q of queries) {
76
+ const all = findAllElements(lines, q, tag);
77
+ for (const c of all) {
78
+ if (!candidates.some((x) => x.startLine === c.startLine)) candidates.push(c);
79
+ }
80
+ if (candidates.length === 1) break;
81
+ }
82
+ if (candidates.length === 0) return { error: 'element_not_found' };
83
+ if (candidates.length === 1) return { match: candidates[0] };
84
+ const filtered = filterByText(candidates, lines, text);
85
+ if (filtered.length === 1) return { match: filtered[0] };
86
+ if (filtered.length === 0) return { match: candidates[0] };
87
+ return { error: 'element_ambiguous', candidates: filtered };
88
+ }
89
+
90
+ for (const q of queries) {
91
+ const match = findElement(lines, q, tag);
92
+ if (match) return { match };
93
+ }
94
+ return { error: 'element_not_found' };
95
+ }
96
+
97
+ export async function insertCli() {
98
+ const args = process.argv.slice(2);
99
+
100
+ if (args.includes('--help') || args.includes('-h')) {
101
+ console.log(`Usage: node live-insert.mjs [options]
102
+
103
+ Find an anchor element in source and splice an insert-variant wrapper.
104
+
105
+ Required:
106
+ --id ID Session ID for the variant wrapper
107
+ --count N Number of expected variants (1-8)
108
+ --position POS before | after (relative to the anchor element)
109
+
110
+ Element identification (at least one required):
111
+ --element-id ID HTML id attribute of the anchor element
112
+ --classes A,B,C Comma-separated CSS class names
113
+ --tag TAG Tag name (div, section, etc.)
114
+ --query TEXT Fallback: raw text to search for
115
+
116
+ Optional:
117
+ --file PATH Source file to search in (skips auto-detection)
118
+ --text TEXT Anchor textContent for disambiguation (~80 chars)
119
+
120
+ Output (JSON):
121
+ { mode: "insert", file, position, insertLine, commentSyntax, styleMode, styleTag, cssAuthoring }`);
122
+ process.exit(0);
123
+ }
124
+
125
+ const id = argVal(args, '--id');
126
+ const count = parseInt(argVal(args, '--count') || '3', 10);
127
+ const position = argVal(args, '--position');
128
+ const elementId = argVal(args, '--element-id');
129
+ const classes = argVal(args, '--classes');
130
+ const tag = argVal(args, '--tag');
131
+ const query = argVal(args, '--query');
132
+ const filePath = argVal(args, '--file');
133
+ const text = argVal(args, '--text');
134
+
135
+ if (!id) { console.error('Missing --id'); process.exit(1); }
136
+ if (!position) { console.error('Missing --position (before | after)'); process.exit(1); }
137
+ if (!isInsertPosition(position)) { console.error('Invalid --position: ' + position); process.exit(1); }
138
+ if (!elementId && !classes && !query) {
139
+ console.error('Need at least one of: --element-id, --classes, --query');
140
+ process.exit(1);
141
+ }
142
+
143
+ const queries = buildSearchQueries(elementId, classes, tag, query);
144
+ const genOpts = { cwd: process.cwd() };
145
+
146
+ let targetFile = filePath;
147
+ if (!targetFile) {
148
+ for (const q of queries) {
149
+ targetFile = findFileWithQuery(q, process.cwd(), genOpts);
150
+ if (targetFile) break;
151
+ }
152
+ if (!targetFile) {
153
+ let generatedHit = null;
154
+ for (const q of queries) {
155
+ generatedHit = findFileWithQuery(q, process.cwd(), { ...genOpts, includeGenerated: true });
156
+ if (generatedHit) break;
157
+ }
158
+ console.error(JSON.stringify({
159
+ error: generatedHit ? 'element_not_in_source' : 'element_not_found',
160
+ fallback: 'agent-driven',
161
+ hint: 'See "Handle fallback" in live.md.',
162
+ }));
163
+ process.exit(1);
164
+ }
165
+ } else if (isGeneratedFile(targetFile, genOpts)) {
166
+ console.error(JSON.stringify({
167
+ error: 'file_is_generated',
168
+ fallback: 'agent-driven',
169
+ file: path.relative(process.cwd(), path.resolve(process.cwd(), targetFile)),
170
+ }));
171
+ process.exit(1);
172
+ }
173
+
174
+ const content = fs.readFileSync(targetFile, 'utf-8');
175
+ const lines = content.split('\n');
176
+ const resolved = resolveElementMatch({ lines, queries, tag, text });
177
+
178
+ if (resolved.error === 'element_ambiguous') {
179
+ console.error(JSON.stringify({
180
+ error: 'element_ambiguous',
181
+ fallback: 'agent-driven',
182
+ file: path.relative(process.cwd(), targetFile),
183
+ candidates: resolved.candidates.map((c) => ({
184
+ startLine: c.startLine + 1,
185
+ endLine: c.endLine + 1,
186
+ })),
187
+ }));
188
+ process.exit(1);
189
+ }
190
+ if (!resolved.match) {
191
+ console.error(JSON.stringify({ error: 'element_not_found', fallback: 'agent-driven' }));
192
+ process.exit(1);
193
+ }
194
+
195
+ const { startLine, endLine } = resolved.match;
196
+ const commentSyntax = detectCommentSyntax(targetFile);
197
+ const styleMode = detectStyleMode(targetFile);
198
+ const isJsx = commentSyntax.open === '{/*';
199
+ const spliceIndex = computeInsertLine(startLine, endLine, position);
200
+ const relTargetFile = path.relative(process.cwd(), targetFile).split(path.sep).join('/');
201
+
202
+ if (shouldUseSvelteComponentInjection(targetFile)) {
203
+ const session = scaffoldSvelteComponentInsertSession({
204
+ id,
205
+ count,
206
+ sourceFile: relTargetFile,
207
+ insertLine: spliceIndex + 1,
208
+ position,
209
+ anchorStartLine: startLine + 1,
210
+ anchorEndLine: endLine + 1,
211
+ anchorLines: lines.slice(startLine, endLine + 1),
212
+ cwd: process.cwd(),
213
+ });
214
+ console.log(JSON.stringify({
215
+ mode: 'insert',
216
+ position,
217
+ file: session.manifestFile,
218
+ sourceFile: relTargetFile,
219
+ previewMode: 'svelte-component',
220
+ componentDir: session.componentDir,
221
+ propContract: session.propContract,
222
+ insertLine: 1,
223
+ sourceInsertLine: spliceIndex + 1,
224
+ anchorStartLine: startLine + 1,
225
+ anchorEndLine: endLine + 1,
226
+ commentSyntax,
227
+ styleMode: 'svelte-component',
228
+ styleTag: null,
229
+ cssSelectorPrefixExamples: [],
230
+ cssAuthoring: buildSvelteComponentCssAuthoring(count),
231
+ }));
232
+ return;
233
+ }
234
+
235
+ const indent = lines[spliceIndex]?.match(/^(\s*)/)?.[1]
236
+ ?? lines[startLine]?.match(/^(\s*)/)?.[1]
237
+ ?? '';
238
+
239
+ const wrapperLines = buildInsertWrapperLines({
240
+ id,
241
+ count,
242
+ indent,
243
+ commentSyntax,
244
+ isJsx,
245
+ });
246
+
247
+ const newLines = [
248
+ ...lines.slice(0, spliceIndex),
249
+ ...wrapperLines,
250
+ ...lines.slice(spliceIndex),
251
+ ];
252
+ fs.writeFileSync(targetFile, newLines.join('\n'), 'utf-8');
253
+
254
+ const insertLine = spliceIndex + 3;
255
+
256
+ console.log(JSON.stringify({
257
+ mode: 'insert',
258
+ position,
259
+ file: relTargetFile,
260
+ insertLine: insertLine + 1,
261
+ commentSyntax,
262
+ styleMode: styleMode.mode,
263
+ styleTag: styleMode.styleTag,
264
+ cssSelectorPrefixExamples: buildCssSelectorPrefixExamples(styleMode.mode, count),
265
+ cssAuthoring: buildCssAuthoring(styleMode, count),
266
+ }));
267
+ }
268
+
269
+ const _running = process.argv[1];
270
+ if (_running?.endsWith('live-insert.mjs') || _running?.endsWith('live-insert.mjs/')) {
271
+ insertCli();
272
+ }
@@ -0,0 +1,363 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * Collect evidence for pending live copy edits.
4
+ *
5
+ * This module intentionally does not edit source files and does not choose a
6
+ * winner. It gathers staged browser edits, rendered context, framework source
7
+ * hints, and likely source candidates so the AI copy-edit batch runner can make
8
+ * source changes with full repo context.
9
+ */
10
+
11
+ import fs from 'node:fs';
12
+ import path from 'node:path';
13
+ import { isGeneratedFile } from './is-generated.mjs';
14
+ import { readBuffer, getBufferPath } from './live-manual-edits-buffer.mjs';
15
+
16
+ const EVIDENCE_VERSION = 1;
17
+ const TEXT_EXTENSIONS = new Set(['.html', '.jsx', '.tsx', '.vue', '.svelte', '.astro', '.js', '.mjs', '.ts']);
18
+ const SEARCH_DIRS = ['src', 'app', 'pages', 'components', 'public', 'views', 'templates', 'site', 'lib', 'data'];
19
+ const STRONG_LITERAL_MATCH_LIMIT = 8;
20
+ const WEAK_LITERAL_MATCH_LIMIT = 4;
21
+ const OBJECT_KEY_MATCH_LIMIT = 8;
22
+ const LOCATOR_MATCH_LIMIT = 4;
23
+ const CONTEXT_MATCH_LIMIT = 8;
24
+ const CONTEXT_MATCH_PER_HINT = 2;
25
+ const SKIP_DIRS = new Set([
26
+ 'node_modules',
27
+ '.git',
28
+ '.impeccable',
29
+ '.astro',
30
+ '.next',
31
+ '.nuxt',
32
+ '.svelte-kit',
33
+ 'dist',
34
+ 'build',
35
+ 'out',
36
+ 'coverage',
37
+ ]);
38
+
39
+ export function buildManualEditEvidence({ cwd = process.cwd(), pageUrl = null } = {}) {
40
+ const buffer = readBuffer(cwd);
41
+ const entries = pageUrl
42
+ ? buffer.entries.filter((entry) => entry.pageUrl === pageUrl)
43
+ : buffer.entries;
44
+ const opCount = countOps(entries);
45
+
46
+ if (opCount === 0) {
47
+ return {
48
+ pageUrl,
49
+ count: 0,
50
+ entries: [],
51
+ ops: [],
52
+ candidates: [],
53
+ };
54
+ }
55
+
56
+ const searchFiles = collectSearchFiles(cwd);
57
+ const ops = flattenOps(entries);
58
+ const candidates = ops.map((op) => buildCandidatesForOp(op, cwd, searchFiles));
59
+ return {
60
+ version: EVIDENCE_VERSION,
61
+ pageUrl: pageUrl || null,
62
+ count: opCount,
63
+ entries,
64
+ ops,
65
+ context: {
66
+ cwd,
67
+ bufferPath: path.relative(cwd, getBufferPath(cwd)),
68
+ totalEntries: entries.length,
69
+ totalOps: opCount,
70
+ },
71
+ candidates,
72
+ };
73
+ }
74
+
75
+ function countOps(entries) {
76
+ let count = 0;
77
+ for (const entry of entries) count += Array.isArray(entry.ops) ? entry.ops.length : 0;
78
+ return count;
79
+ }
80
+
81
+ function flattenOps(entries) {
82
+ const out = [];
83
+ for (const entry of entries) {
84
+ const contextHintsByRef = buildContextHintsByRef(entry);
85
+ for (const op of entry.ops || []) {
86
+ out.push({
87
+ entryId: entry.id,
88
+ pageUrl: entry.pageUrl,
89
+ ref: op.ref,
90
+ contextRef: op.contextRef || null,
91
+ tag: op.tag,
92
+ elementId: op.elementId || null,
93
+ classes: Array.isArray(op.classes) ? op.classes : [],
94
+ originalText: op.originalText,
95
+ newText: op.newText,
96
+ deleted: op.deleted === true,
97
+ sourceHint: op.sourceHint || null,
98
+ leaf: op.leaf || null,
99
+ nearbyEditableTexts: Array.isArray(op.nearbyEditableTexts) ? op.nearbyEditableTexts : [],
100
+ container: op.container || null,
101
+ contextHints: contextHintsByRef.get(op.ref) || [],
102
+ });
103
+ }
104
+ }
105
+ return out;
106
+ }
107
+
108
+ function buildContextHintsByRef(entry) {
109
+ const map = new Map();
110
+ for (const op of entry.ops || []) {
111
+ const hints = new Set();
112
+ const add = (value) => {
113
+ const text = normalizeText(decodeBasicHtml(String(value || '')));
114
+ if (text.length < 3 || text.length > 160) return;
115
+ if (text === normalizeText(op.originalText) || text === normalizeText(op.newText)) return;
116
+ hints.add(text);
117
+ };
118
+
119
+ for (const item of op.nearbyEditableTexts || []) {
120
+ add(typeof item === 'string' ? item : item?.text);
121
+ }
122
+ const outer = typeof entry.element?.outerHTML === 'string' ? entry.element.outerHTML : '';
123
+ for (const match of outer.matchAll(/data-impeccable-original-text="([^"]*)"/g)) add(match[1]);
124
+ if (typeof entry.element?.textContent === 'string') {
125
+ for (const chunk of entry.element.textContent.split(/\s{2,}|\n|\t/)) add(chunk);
126
+ }
127
+ map.set(op.ref, [...hints].slice(0, 16));
128
+ }
129
+ return map;
130
+ }
131
+
132
+ function buildCandidatesForOp(op, cwd, searchFiles) {
133
+ const originalText = String(op.originalText || '');
134
+ const contextNeedles = op.contextHints || [];
135
+ return {
136
+ entryId: op.entryId,
137
+ ref: op.ref,
138
+ originalText,
139
+ sourceHint: analyzeSourceHint(op, cwd),
140
+ textMatches: originalText ? findLiteralMatches(searchFiles, originalText, { max: literalMatchLimit(originalText) }) : [],
141
+ objectKeyMatches: originalText ? findObjectKeyMatches(searchFiles, originalText, { max: OBJECT_KEY_MATCH_LIMIT }) : [],
142
+ locatorMatches: findLocatorMatches(searchFiles, op, { max: LOCATOR_MATCH_LIMIT }),
143
+ contextTextMatches: findContextMatches(searchFiles, contextNeedles, { maxPerHint: CONTEXT_MATCH_PER_HINT, max: CONTEXT_MATCH_LIMIT }),
144
+ };
145
+ }
146
+
147
+ function literalMatchLimit(text) {
148
+ return isWeakSourceNeedle(text) ? WEAK_LITERAL_MATCH_LIMIT : STRONG_LITERAL_MATCH_LIMIT;
149
+ }
150
+
151
+ function isWeakSourceNeedle(text) {
152
+ const normalized = normalizeText(text);
153
+ return normalized.length < 4 || /^[\d.,+\-%\s]+$/.test(normalized);
154
+ }
155
+
156
+ function analyzeSourceHint(op, cwd) {
157
+ const hint = normalizeSourceHint(op.sourceHint);
158
+ if (!hint.file) return null;
159
+ const file = path.resolve(cwd, hint.file);
160
+ const relativeFile = path.relative(cwd, file);
161
+ if (!isPathInsideOrEqual(cwd, file)) {
162
+ return { ...hint, status: 'outside_cwd', relativeFile: hint.file };
163
+ }
164
+ if (!fs.existsSync(file)) {
165
+ return { ...hint, status: 'file_missing', relativeFile };
166
+ }
167
+ if (isGeneratedFile(file, { cwd })) {
168
+ return { ...hint, status: 'generated', relativeFile };
169
+ }
170
+
171
+ const content = fs.readFileSync(file, 'utf-8');
172
+ const lines = content.split('\n');
173
+ const line = hint.line || 1;
174
+ const start = Math.max(0, line - 4);
175
+ const end = Math.min(lines.length, line + 3);
176
+ const windowText = lines.slice(start, end).join('\n');
177
+ const containsOriginalText = typeof op.originalText === 'string' && windowText.includes(op.originalText);
178
+ return {
179
+ ...hint,
180
+ status: containsOriginalText ? 'ok' : 'text_not_found_near_hint',
181
+ relativeFile,
182
+ excerpt: lines.slice(start, end).map((text, index) => ({
183
+ line: start + index + 1,
184
+ text: text.slice(0, 240),
185
+ })),
186
+ };
187
+ }
188
+
189
+ function normalizeSourceHint(hint) {
190
+ if (!hint || typeof hint !== 'object') return {};
191
+ let line = Number.isFinite(Number(hint.line)) ? Number(hint.line) : null;
192
+ let column = Number.isFinite(Number(hint.column)) ? Number(hint.column) : null;
193
+ if ((!line || !column) && typeof hint.loc === 'string') {
194
+ const match = hint.loc.match(/^(\d+)(?::(\d+))?/);
195
+ if (match) {
196
+ line = Number(match[1]);
197
+ if (match[2]) column = Number(match[2]);
198
+ }
199
+ }
200
+ return {
201
+ file: typeof hint.file === 'string' ? hint.file : '',
202
+ loc: typeof hint.loc === 'string' ? hint.loc : '',
203
+ line,
204
+ column,
205
+ };
206
+ }
207
+
208
+ function collectSearchFiles(cwd) {
209
+ const out = [];
210
+ const seenDirs = new Set();
211
+ const seenFiles = new Set();
212
+ for (const dir of SEARCH_DIRS) {
213
+ scanDir(path.join(cwd, dir), cwd, seenDirs, seenFiles, out, 0);
214
+ }
215
+ scanRootFiles(cwd, seenFiles, out);
216
+ return out;
217
+ }
218
+
219
+ function scanDir(dir, cwd, seenDirs, seenFiles, out, depth) {
220
+ if (depth > 7 || !fs.existsSync(dir)) return;
221
+ let realDir;
222
+ try { realDir = fs.realpathSync(dir); } catch { return; }
223
+ if (seenDirs.has(realDir)) return;
224
+ seenDirs.add(realDir);
225
+
226
+ let entries;
227
+ try { entries = fs.readdirSync(dir, { withFileTypes: true }); } catch { return; }
228
+ for (const entry of entries) {
229
+ const fullPath = path.join(dir, entry.name);
230
+ if (entry.isDirectory()) {
231
+ if (SKIP_DIRS.has(entry.name)) continue;
232
+ scanDir(fullPath, cwd, seenDirs, seenFiles, out, depth + 1);
233
+ continue;
234
+ }
235
+ if (!entry.isFile() || !TEXT_EXTENSIONS.has(path.extname(entry.name).toLowerCase())) continue;
236
+ maybeAddSearchFile(fullPath, cwd, seenFiles, out);
237
+ }
238
+ }
239
+
240
+ function scanRootFiles(cwd, seenFiles, out) {
241
+ let entries;
242
+ try { entries = fs.readdirSync(cwd, { withFileTypes: true }); } catch { return; }
243
+ for (const entry of entries) {
244
+ if (!entry.isFile() || !TEXT_EXTENSIONS.has(path.extname(entry.name).toLowerCase())) continue;
245
+ maybeAddSearchFile(path.join(cwd, entry.name), cwd, seenFiles, out);
246
+ }
247
+ }
248
+
249
+ function maybeAddSearchFile(file, cwd, seenFiles, out) {
250
+ let realFile;
251
+ try { realFile = fs.realpathSync(file); } catch { return; }
252
+ if (seenFiles.has(realFile)) return;
253
+ seenFiles.add(realFile);
254
+ if (isGeneratedFile(file, { cwd })) return;
255
+ let content;
256
+ try { content = fs.readFileSync(file, 'utf-8'); } catch { return; }
257
+ out.push({ file, relativeFile: path.relative(cwd, file), content, lines: content.split('\n') });
258
+ }
259
+
260
+ function findLiteralMatches(searchFiles, needle, { max }) {
261
+ return findMatches(searchFiles, needle, { kind: 'text', max });
262
+ }
263
+
264
+ function findObjectKeyMatches(searchFiles, text, { max }) {
265
+ const re = new RegExp('(["\\\'`])' + escapeRegExp(text) + '\\1(?=\\s*:)', 'g');
266
+ const out = [];
267
+ for (const file of searchFiles) {
268
+ for (const match of file.content.matchAll(re)) {
269
+ out.push(matchForIndex(file, match.index, 'object_key', text));
270
+ if (out.length >= max) return out;
271
+ }
272
+ }
273
+ return out;
274
+ }
275
+
276
+ function findLocatorMatches(searchFiles, op, { max }) {
277
+ const needles = [];
278
+ if (op.elementId) needles.push({ kind: 'id', needle: op.elementId });
279
+ for (const cls of op.classes || []) {
280
+ if (cls) needles.push({ kind: 'class', needle: cls });
281
+ }
282
+ if (op.tag) needles.push({ kind: 'tag', needle: '<' + op.tag });
283
+
284
+ const out = [];
285
+ const seen = new Set();
286
+ for (const { kind, needle } of needles) {
287
+ for (const match of findMatches(searchFiles, needle, { kind, max })) {
288
+ const key = match.file + ':' + match.line + ':' + kind + ':' + needle;
289
+ if (seen.has(key)) continue;
290
+ seen.add(key);
291
+ out.push({ ...match, needle });
292
+ if (out.length >= max) return out;
293
+ }
294
+ }
295
+ return out;
296
+ }
297
+
298
+ function findContextMatches(searchFiles, hints, { maxPerHint, max }) {
299
+ const out = [];
300
+ const seen = new Set();
301
+ for (const hint of hints || []) {
302
+ for (const match of findMatches(searchFiles, hint, { kind: 'context', max: maxPerHint })) {
303
+ const key = match.file + ':' + match.line + ':' + hint;
304
+ if (seen.has(key)) continue;
305
+ seen.add(key);
306
+ out.push({ ...match, needle: hint });
307
+ if (out.length >= max) return out;
308
+ }
309
+ }
310
+ return out;
311
+ }
312
+
313
+ function findMatches(searchFiles, needle, { kind, max }) {
314
+ const text = String(needle || '');
315
+ if (!text) return [];
316
+ const out = [];
317
+ for (const file of searchFiles) {
318
+ let index = 0;
319
+ while (out.length < max) {
320
+ index = file.content.indexOf(text, index);
321
+ if (index === -1) break;
322
+ out.push(matchForIndex(file, index, kind, text));
323
+ index += Math.max(1, text.length);
324
+ }
325
+ if (out.length >= max) break;
326
+ }
327
+ return out;
328
+ }
329
+
330
+ function matchForIndex(file, index, kind, needle) {
331
+ const line = file.content.slice(0, index).split('\n').length;
332
+ const lineText = file.lines[line - 1] || '';
333
+ return {
334
+ kind,
335
+ file: file.relativeFile,
336
+ line,
337
+ needle,
338
+ excerpt: lineText.trim().slice(0, 240),
339
+ };
340
+ }
341
+
342
+ function isPathInsideOrEqual(cwd, file) {
343
+ const rel = path.relative(path.resolve(cwd), path.resolve(file));
344
+ return rel === '' || (!rel.startsWith('..') && !path.isAbsolute(rel));
345
+ }
346
+
347
+ function normalizeText(value) {
348
+ return String(value || '').replace(/\s+/g, ' ').trim();
349
+ }
350
+
351
+ function decodeBasicHtml(value) {
352
+ return value
353
+ .replace(/&quot;/g, '"')
354
+ .replace(/&#39;/g, "'")
355
+ .replace(/&apos;/g, "'")
356
+ .replace(/&amp;/g, '&')
357
+ .replace(/&lt;/g, '<')
358
+ .replace(/&gt;/g, '>');
359
+ }
360
+
361
+ function escapeRegExp(value) {
362
+ return String(value).replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
363
+ }