claudecode-omc 5.6.8 → 5.11.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 (216) hide show
  1. package/.local/settings/settings.json +8 -0
  2. package/.local/skills/prompt-optimizer/SKILL.md +262 -19
  3. package/.omc-curation/ecc-selection.json +80 -0
  4. package/.omc-curation/governance.json +116 -0
  5. package/.omc-curation/sources.lock.json +30 -0
  6. package/README.md +78 -4
  7. package/bundled/manifest.json +6 -5
  8. package/bundled/upstream/anthropic-skills/.omc-source/bundle.json +18 -0
  9. package/bundled/upstream/anthropic-skills/.omc-source/provenance.json +399 -0
  10. package/bundled/upstream/anthropic-skills/skills/claude-api/SKILL.md +18 -17
  11. package/bundled/upstream/anthropic-skills/skills/claude-api/curl/examples.md +9 -9
  12. package/bundled/upstream/anthropic-skills/skills/claude-api/curl/managed-agents.md +4 -4
  13. package/bundled/upstream/anthropic-skills/skills/claude-api/go/managed-agents/README.md +2 -2
  14. package/bundled/upstream/anthropic-skills/skills/claude-api/java/claude-api.md +2 -2
  15. package/bundled/upstream/anthropic-skills/skills/claude-api/java/managed-agents/README.md +2 -2
  16. package/bundled/upstream/anthropic-skills/skills/claude-api/php/claude-api.md +10 -10
  17. package/bundled/upstream/anthropic-skills/skills/claude-api/php/managed-agents/README.md +2 -2
  18. package/bundled/upstream/anthropic-skills/skills/claude-api/python/claude-api/README.md +16 -16
  19. package/bundled/upstream/anthropic-skills/skills/claude-api/python/claude-api/batches.md +3 -3
  20. package/bundled/upstream/anthropic-skills/skills/claude-api/python/claude-api/files-api.md +3 -3
  21. package/bundled/upstream/anthropic-skills/skills/claude-api/python/claude-api/streaming.md +7 -7
  22. package/bundled/upstream/anthropic-skills/skills/claude-api/python/claude-api/tool-use.md +19 -19
  23. package/bundled/upstream/anthropic-skills/skills/claude-api/python/managed-agents/README.md +3 -3
  24. package/bundled/upstream/anthropic-skills/skills/claude-api/ruby/claude-api.md +4 -4
  25. package/bundled/upstream/anthropic-skills/skills/claude-api/ruby/managed-agents/README.md +2 -2
  26. package/bundled/upstream/anthropic-skills/skills/claude-api/shared/error-codes.md +5 -5
  27. package/bundled/upstream/anthropic-skills/skills/claude-api/shared/live-sources.md +3 -1
  28. package/bundled/upstream/anthropic-skills/skills/claude-api/shared/managed-agents-api-reference.md +10 -4
  29. package/bundled/upstream/anthropic-skills/skills/claude-api/shared/managed-agents-core.md +19 -1
  30. package/bundled/upstream/anthropic-skills/skills/claude-api/shared/managed-agents-environments.md +6 -2
  31. package/bundled/upstream/anthropic-skills/skills/claude-api/shared/managed-agents-multiagent.md +1 -1
  32. package/bundled/upstream/anthropic-skills/skills/claude-api/shared/managed-agents-onboarding.md +3 -3
  33. package/bundled/upstream/anthropic-skills/skills/claude-api/shared/managed-agents-overview.md +3 -2
  34. package/bundled/upstream/anthropic-skills/skills/claude-api/shared/managed-agents-self-hosted-sandboxes.md +173 -0
  35. package/bundled/upstream/anthropic-skills/skills/claude-api/shared/managed-agents-tools.md +10 -4
  36. package/bundled/upstream/anthropic-skills/skills/claude-api/shared/model-migration.md +113 -13
  37. package/bundled/upstream/anthropic-skills/skills/claude-api/shared/models.md +14 -11
  38. package/bundled/upstream/anthropic-skills/skills/claude-api/shared/prompt-caching.md +2 -2
  39. package/bundled/upstream/anthropic-skills/skills/claude-api/shared/tool-use-concepts.md +4 -4
  40. package/bundled/upstream/anthropic-skills/skills/claude-api/typescript/claude-api/README.md +15 -15
  41. package/bundled/upstream/anthropic-skills/skills/claude-api/typescript/claude-api/batches.md +2 -2
  42. package/bundled/upstream/anthropic-skills/skills/claude-api/typescript/claude-api/files-api.md +1 -1
  43. package/bundled/upstream/anthropic-skills/skills/claude-api/typescript/claude-api/streaming.md +5 -5
  44. package/bundled/upstream/anthropic-skills/skills/claude-api/typescript/claude-api/tool-use.md +15 -15
  45. package/bundled/upstream/anthropic-skills/skills/claude-api/typescript/managed-agents/README.md +3 -3
  46. package/bundled/upstream/ecc/.omc-source/bundle.json +2 -1
  47. package/bundled/upstream/ecc/.omc-source/last-plan-apply.json +108 -24
  48. package/bundled/upstream/ecc/.omc-source/manifests/.claude-plugin/marketplace.json +3 -3
  49. package/bundled/upstream/ecc/.omc-source/provenance.json +563 -0
  50. package/bundled/upstream/ecc/agents/marketing-agent.md +159 -0
  51. package/bundled/upstream/ecc/agents/react-build-resolver.md +215 -0
  52. package/bundled/upstream/ecc/agents/react-reviewer.md +167 -0
  53. package/bundled/upstream/ecc/agents/typescript-reviewer.md +3 -0
  54. package/bundled/upstream/ecc/commands/harness-audit.md +17 -10
  55. package/bundled/upstream/ecc/commands/marketing-campaign.md +129 -0
  56. package/bundled/upstream/ecc/commands/react-build.md +187 -0
  57. package/bundled/upstream/ecc/commands/react-review.md +170 -0
  58. package/bundled/upstream/ecc/commands/react-test.md +265 -0
  59. package/bundled/upstream/ecc/skills/benchmark-optimization-loop/SKILL.md +69 -0
  60. package/bundled/upstream/ecc/skills/blender-motion-state-inspection/SKILL.md +164 -0
  61. package/bundled/upstream/ecc/skills/canary-watch/SKILL.md +9 -1
  62. package/bundled/upstream/ecc/skills/continuous-learning-v2/hooks/observe.sh +31 -9
  63. package/bundled/upstream/ecc/skills/continuous-learning-v2/scripts/detect-project.sh +38 -4
  64. package/bundled/upstream/ecc/skills/continuous-learning-v2/scripts/instinct-cli.py +319 -12
  65. package/bundled/upstream/ecc/skills/data-throughput-accelerator/SKILL.md +72 -0
  66. package/bundled/upstream/ecc/skills/dynamic-workflow-mode/SKILL.md +123 -0
  67. package/bundled/upstream/ecc/skills/frontend-a11y/SKILL.md +446 -0
  68. package/bundled/upstream/ecc/skills/ito-basket-compare/SKILL.md +63 -0
  69. package/bundled/upstream/ecc/skills/ito-data-atlas-agent/SKILL.md +63 -0
  70. package/bundled/upstream/ecc/skills/ito-market-intelligence/SKILL.md +60 -0
  71. package/bundled/upstream/ecc/skills/ito-trade-planner/SKILL.md +67 -0
  72. package/bundled/upstream/ecc/skills/latency-critical-systems/SKILL.md +73 -0
  73. package/bundled/upstream/ecc/skills/marketing-campaign/SKILL.md +113 -0
  74. package/bundled/upstream/ecc/skills/nextjs-turbopack/SKILL.md +13 -0
  75. package/bundled/upstream/ecc/skills/parallel-execution-optimizer/SKILL.md +72 -0
  76. package/bundled/upstream/ecc/skills/prediction-market-oracle-research/SKILL.md +63 -0
  77. package/bundled/upstream/ecc/skills/prediction-market-risk-review/SKILL.md +60 -0
  78. package/bundled/upstream/ecc/skills/react-patterns/SKILL.md +341 -0
  79. package/bundled/upstream/ecc/skills/react-performance/SKILL.md +574 -0
  80. package/bundled/upstream/ecc/skills/react-testing/SKILL.md +423 -0
  81. package/bundled/upstream/ecc/skills/recsys-pipeline-architect/SKILL.md +114 -0
  82. package/bundled/upstream/ecc/skills/recursive-decision-ledger/SKILL.md +79 -0
  83. package/bundled/upstream/ecc/skills/social-publisher/SKILL.md +115 -0
  84. package/bundled/upstream/ecc/skills/team-agent-orchestration/SKILL.md +110 -0
  85. package/bundled/upstream/ecc/skills/uncloud/SKILL.md +343 -0
  86. package/bundled/upstream/ecc/skills/windows-desktop-e2e/SKILL.md +99 -0
  87. package/bundled/upstream/impeccable/.omc-source/bundle.json +20 -0
  88. package/bundled/upstream/impeccable/.omc-source/provenance.json +105 -0
  89. package/bundled/upstream/impeccable/agents/impeccable-manual-edit-applier.md +97 -0
  90. package/bundled/upstream/impeccable/skills/impeccable/SKILL.md +168 -0
  91. package/bundled/upstream/impeccable/skills/impeccable/reference/adapt.md +311 -0
  92. package/bundled/upstream/impeccable/skills/impeccable/reference/animate.md +201 -0
  93. package/bundled/upstream/impeccable/skills/impeccable/reference/audit.md +133 -0
  94. package/bundled/upstream/impeccable/skills/impeccable/reference/bolder.md +113 -0
  95. package/bundled/upstream/impeccable/skills/impeccable/reference/brand.md +108 -0
  96. package/bundled/upstream/impeccable/skills/impeccable/reference/clarify.md +288 -0
  97. package/bundled/upstream/impeccable/skills/impeccable/reference/codex.md +105 -0
  98. package/bundled/upstream/impeccable/skills/impeccable/reference/colorize.md +257 -0
  99. package/bundled/upstream/impeccable/skills/impeccable/reference/craft.md +123 -0
  100. package/bundled/upstream/impeccable/skills/impeccable/reference/critique.md +767 -0
  101. package/bundled/upstream/impeccable/skills/impeccable/reference/delight.md +302 -0
  102. package/bundled/upstream/impeccable/skills/impeccable/reference/distill.md +111 -0
  103. package/bundled/upstream/impeccable/skills/impeccable/reference/document.md +429 -0
  104. package/bundled/upstream/impeccable/skills/impeccable/reference/extract.md +69 -0
  105. package/bundled/upstream/impeccable/skills/impeccable/reference/harden.md +347 -0
  106. package/bundled/upstream/impeccable/skills/impeccable/reference/hooks.md +88 -0
  107. package/bundled/upstream/impeccable/skills/impeccable/reference/init.md +172 -0
  108. package/bundled/upstream/impeccable/skills/impeccable/reference/interaction-design.md +189 -0
  109. package/bundled/upstream/impeccable/skills/impeccable/reference/layout.md +161 -0
  110. package/bundled/upstream/impeccable/skills/impeccable/reference/live.md +718 -0
  111. package/bundled/upstream/impeccable/skills/impeccable/reference/onboard.md +234 -0
  112. package/bundled/upstream/impeccable/skills/impeccable/reference/optimize.md +258 -0
  113. package/bundled/upstream/impeccable/skills/impeccable/reference/overdrive.md +130 -0
  114. package/bundled/upstream/impeccable/skills/impeccable/reference/polish.md +241 -0
  115. package/bundled/upstream/impeccable/skills/impeccable/reference/product.md +60 -0
  116. package/bundled/upstream/impeccable/skills/impeccable/reference/quieter.md +99 -0
  117. package/bundled/upstream/impeccable/skills/impeccable/reference/shape.md +165 -0
  118. package/bundled/upstream/impeccable/skills/impeccable/reference/typeset.md +279 -0
  119. package/bundled/upstream/impeccable/skills/impeccable/scripts/command-metadata.json +94 -0
  120. package/bundled/upstream/impeccable/skills/impeccable/scripts/context-signals.mjs +225 -0
  121. package/bundled/upstream/impeccable/skills/impeccable/scripts/context.mjs +280 -0
  122. package/bundled/upstream/impeccable/skills/impeccable/scripts/critique-storage.mjs +242 -0
  123. package/bundled/upstream/impeccable/skills/impeccable/scripts/detect-csp.mjs +198 -0
  124. package/bundled/upstream/impeccable/skills/impeccable/scripts/detect.mjs +21 -0
  125. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/browser/injected/index.mjs +1735 -0
  126. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/cli/main.mjs +244 -0
  127. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/detect-antipatterns-browser.js +4907 -0
  128. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/detect-antipatterns.mjs +43 -0
  129. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/engines/browser/detect-url.mjs +252 -0
  130. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/engines/regex/detect-text.mjs +552 -0
  131. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/engines/static-html/css-cascade.mjs +1013 -0
  132. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/engines/static-html/detect-html.mjs +208 -0
  133. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/engines/visual/screenshot-contrast.mjs +189 -0
  134. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/findings.mjs +12 -0
  135. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/node/file-system.mjs +198 -0
  136. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/profile/profiler.mjs +166 -0
  137. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/registry/antipatterns.mjs +419 -0
  138. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/rules/checks.mjs +2671 -0
  139. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/shared/color.mjs +124 -0
  140. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/shared/constants.mjs +101 -0
  141. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/shared/page.mjs +7 -0
  142. package/bundled/upstream/impeccable/skills/impeccable/scripts/hook-admin.mjs +574 -0
  143. package/bundled/upstream/impeccable/skills/impeccable/scripts/hook-before-edit.mjs +473 -0
  144. package/bundled/upstream/impeccable/skills/impeccable/scripts/hook-lib.mjs +1286 -0
  145. package/bundled/upstream/impeccable/skills/impeccable/scripts/hook.mjs +61 -0
  146. package/bundled/upstream/impeccable/skills/impeccable/scripts/lib/design-parser.mjs +835 -0
  147. package/bundled/upstream/impeccable/skills/impeccable/scripts/lib/impeccable-paths.mjs +126 -0
  148. package/bundled/upstream/impeccable/skills/impeccable/scripts/lib/is-generated.mjs +69 -0
  149. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/browser-script-parts.mjs +49 -0
  150. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/completion.mjs +19 -0
  151. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/event-validation.mjs +137 -0
  152. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/insert-ui.mjs +458 -0
  153. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/manual-apply.mjs +939 -0
  154. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/manual-edit-routes.mjs +357 -0
  155. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/manual-edits-buffer.mjs +152 -0
  156. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/session-store.mjs +289 -0
  157. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/svelte-component.mjs +826 -0
  158. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/sveltekit-adapter.mjs +274 -0
  159. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/ui-core.mjs +180 -0
  160. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/vocabulary.mjs +36 -0
  161. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-accept.mjs +812 -0
  162. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-browser-dom.js +146 -0
  163. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-browser-session.js +123 -0
  164. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-browser.js +11086 -0
  165. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-commit-manual-edits.mjs +1241 -0
  166. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-complete.mjs +75 -0
  167. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-copy-edit-agent.mjs +683 -0
  168. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-discard-manual-edits.mjs +51 -0
  169. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-inject.mjs +583 -0
  170. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-insert.mjs +272 -0
  171. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-manual-edit-evidence.mjs +363 -0
  172. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-poll.mjs +379 -0
  173. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-resume.mjs +94 -0
  174. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-server.mjs +1134 -0
  175. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-status.mjs +61 -0
  176. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-wrap.mjs +894 -0
  177. package/bundled/upstream/impeccable/skills/impeccable/scripts/live.mjs +246 -0
  178. package/bundled/upstream/impeccable/skills/impeccable/scripts/modern-screenshot.umd.js +14 -0
  179. package/bundled/upstream/impeccable/skills/impeccable/scripts/palette.mjs +633 -0
  180. package/bundled/upstream/impeccable/skills/impeccable/scripts/pin.mjs +214 -0
  181. package/bundled/upstream/oh-my-claudecode/.omc-source/bundle.json +2 -1
  182. package/bundled/upstream/oh-my-claudecode/.omc-source/provenance.json +116 -0
  183. package/bundled/upstream/oh-my-claudecode/skills/autopilot/SKILL.md +7 -0
  184. package/bundled/upstream/oh-my-claudecode/skills/cancel/SKILL.md +1 -0
  185. package/bundled/upstream/oh-my-claudecode/skills/deep-interview/SKILL.md +39 -5
  186. package/bundled/upstream/oh-my-claudecode/skills/hud/SKILL.md +1 -0
  187. package/bundled/upstream/oh-my-claudecode/skills/local-build-reminder/SKILL.md +78 -0
  188. package/bundled/upstream/oh-my-claudecode/skills/omc-doctor/SKILL.md +1 -1
  189. package/bundled/upstream/oh-my-claudecode/skills/omc-setup/SKILL.md +26 -10
  190. package/bundled/upstream/oh-my-claudecode/skills/omc-setup/phases/01-install-claude-md.md +3 -3
  191. package/bundled/upstream/oh-my-claudecode/skills/omc-setup/phases/02-configure.md +6 -4
  192. package/bundled/upstream/oh-my-claudecode/skills/omc-setup/phases/03-integrations.md +1 -1
  193. package/bundled/upstream/oh-my-claudecode/skills/omc-setup/phases/04-welcome.md +2 -2
  194. package/bundled/upstream/oh-my-claudecode/skills/omc-teams/SKILL.md +6 -6
  195. package/bundled/upstream/oh-my-claudecode/skills/plan/SKILL.md +44 -32
  196. package/bundled/upstream/oh-my-claudecode/skills/ralph/SKILL.md +45 -21
  197. package/bundled/upstream/oh-my-claudecode/skills/ralplan/SKILL.md +1 -1
  198. package/bundled/upstream/oh-my-claudecode/skills/self-improve/SKILL.md +7 -0
  199. package/bundled/upstream/oh-my-claudecode/skills/self-improve/scripts/resolve-paths.mjs +39 -15
  200. package/bundled/upstream/oh-my-claudecode/skills/team/SKILL.md +132 -90
  201. package/bundled/upstream/oh-my-claudecode/skills/ultragoal/SKILL.md +93 -0
  202. package/bundled/upstream/oh-my-claudecode/skills/ultraqa/SKILL.md +28 -13
  203. package/bundled/upstream/oh-my-claudecode/skills/ultrawork/SKILL.md +7 -0
  204. package/bundled/upstream/superpowers/.omc-source/bundle.json +2 -1
  205. package/bundled/upstream/superpowers/.omc-source/provenance.json +63 -0
  206. package/package.json +2 -1
  207. package/src/catalog/source-catalog.js +10 -4
  208. package/src/cli/index.js +4 -0
  209. package/src/cli/plan.js +14 -2
  210. package/src/cli/setup.js +52 -13
  211. package/src/cli/skill.js +1 -1
  212. package/src/cli/source.js +271 -14
  213. package/src/config/sources.js +82 -1
  214. package/src/merge/content-patch.js +88 -0
  215. package/templates/merge-config.json +1 -8
  216. package/bundled/upstream/ecc/skills/strategic-compact/suggest-compact.sh +0 -54
@@ -0,0 +1,357 @@
1
+ import { validateEvent } from './event-validation.mjs';
2
+ import {
3
+ countByPage as countPendingByPage,
4
+ readBuffer as readManualEditsBuffer,
5
+ removeEntries as removeManualEditEntries,
6
+ stageEntry as stageManualEditEntry,
7
+ truncateBuffer as truncateManualEditsBuffer,
8
+ } from './manual-edits-buffer.mjs';
9
+ import {
10
+ summarizeManualApplyFailures,
11
+ summarizeManualDiagnostics,
12
+ summarizeManualLogFile,
13
+ } from './manual-apply.mjs';
14
+ import { buildManualEditEvidence } from '../live-manual-edit-evidence.mjs';
15
+ import { commitManualEdits } from '../live-commit-manual-edits.mjs';
16
+
17
+ export function createManualEditRoutes({
18
+ getToken,
19
+ manualApply,
20
+ recordManualEditActivity,
21
+ getManualEditStatus,
22
+ chatAgentLikelyActive,
23
+ cwd = () => process.cwd(),
24
+ env = () => process.env,
25
+ } = {}) {
26
+ const projectCwd = () => typeof cwd === 'function' ? cwd() : cwd || process.cwd();
27
+ const currentEnv = () => typeof env === 'function' ? env() : env || process.env;
28
+
29
+ return function handleManualEditRoute(req, res, url) {
30
+ const p = url.pathname;
31
+
32
+ // Save stages entries; Apply commits the staged page batch through the
33
+ // local AI copy-edit runner.
34
+ if (p === '/manual-edit-stash' && req.method === 'POST') {
35
+ let body = '';
36
+ req.on('data', (c) => { body += c; });
37
+ req.on('end', () => {
38
+ let msg;
39
+ try { msg = JSON.parse(body); } catch {
40
+ sendJson(res, 400, { error: 'Invalid JSON' });
41
+ return;
42
+ }
43
+ if (msg.token !== getToken()) {
44
+ sendJson(res, 401, { error: 'Unauthorized' });
45
+ return;
46
+ }
47
+ const error = validateEvent({ ...msg, type: 'manual_edits' });
48
+ if (error) {
49
+ sendJson(res, 400, { error });
50
+ return;
51
+ }
52
+ try {
53
+ stageManualEditEntry(projectCwd(), {
54
+ id: msg.id,
55
+ pageUrl: msg.pageUrl,
56
+ element: msg.element,
57
+ ops: msg.ops,
58
+ });
59
+ } catch (err) {
60
+ sendJson(res, 500, { error: 'stash_write_failed', message: err.message });
61
+ return;
62
+ }
63
+ const { totalCount, perPage } = countPendingByPage(projectCwd());
64
+ const pendingCount = perPage[msg.pageUrl] || 0;
65
+ recordManualEditActivity('manual_edit_stashed', {
66
+ id: msg.id,
67
+ pageUrl: msg.pageUrl,
68
+ opCount: msg.ops.length,
69
+ pendingCount,
70
+ totalCount,
71
+ hintedFileCount: new Set((msg.ops || []).map((op) => summarizeManualLogFile(op.sourceHint?.file, projectCwd())).filter(Boolean)).size,
72
+ });
73
+ sendJson(res, 200, { ok: true, pendingCount, totalCount, perPage });
74
+ });
75
+ return true;
76
+ }
77
+
78
+ if (p === '/manual-edit-stash' && req.method === 'GET') {
79
+ const token = url.searchParams.get('token');
80
+ if (token !== getToken()) { res.writeHead(401); res.end('Unauthorized'); return true; }
81
+ const pageUrl = url.searchParams.get('pageUrl') || '';
82
+ const { totalCount, perPage } = countPendingByPage(projectCwd());
83
+ const buffer = readManualEditsBuffer(projectCwd());
84
+ const entriesForPage = pageUrl ? buffer.entries.filter((e) => e.pageUrl === pageUrl) : buffer.entries;
85
+ sendJson(res, 200, {
86
+ count: pageUrl ? (perPage[pageUrl] || 0) : totalCount,
87
+ totalCount,
88
+ perPage,
89
+ entries: entriesForPage,
90
+ });
91
+ return true;
92
+ }
93
+
94
+ if (p === '/manual-edit-commit' && req.method === 'POST') {
95
+ const token = url.searchParams.get('token');
96
+ if (token !== getToken()) { res.writeHead(401); res.end('Unauthorized'); return true; }
97
+ const pageUrl = url.searchParams.get('pageUrl');
98
+ const asyncMode = /^(1|true|yes)$/i.test(url.searchParams.get('async') || '');
99
+ const repairOnly = /^(1|true|yes)$/i.test(url.searchParams.get('repair') || '');
100
+ const existingTransaction = manualApply.readTransaction();
101
+ if (repairOnly && !existingTransaction) {
102
+ sendJson(res, 409, { error: 'manual_edit_repair_transaction_missing' });
103
+ return true;
104
+ }
105
+ const recoveredTransaction = repairOnly ? null : manualApply.rollbackTransaction({
106
+ pageUrl,
107
+ reason: 'manual_edit_commit_recovered_abandoned_transaction',
108
+ });
109
+ const before = getManualEditStatus();
110
+ const pendingCount = pageUrl ? (before.perPage[pageUrl] || 0) : before.totalCount;
111
+ recordManualEditActivity('manual_edit_commit_started', {
112
+ pageUrl,
113
+ repairOnly,
114
+ pendingCount,
115
+ totalCount: before.totalCount,
116
+ recoveredTransaction: recoveredTransaction ? {
117
+ id: recoveredTransaction.id,
118
+ reason: recoveredTransaction.reason,
119
+ skipped: recoveredTransaction.skipped,
120
+ rolledBackFiles: recoveredTransaction.rolledBackFiles,
121
+ rollbackFailures: summarizeManualDiagnostics(recoveredTransaction.rollbackFailures, projectCwd()),
122
+ } : null,
123
+ ...summarizePendingManualEditBatch(projectCwd(), pageUrl),
124
+ });
125
+ if (asyncMode) {
126
+ sendJson(res, 202, {
127
+ status: 'started',
128
+ pendingCount,
129
+ totalCount: before.totalCount,
130
+ perPage: before.perPage,
131
+ });
132
+ }
133
+ (async () => {
134
+ let result;
135
+ let routedProvider = 'subprocess';
136
+ let transaction = null;
137
+ let commitBatch = null;
138
+ try {
139
+ if (pendingCount > 0) {
140
+ const transactionBatch = buildManualEditEvidence({ cwd: projectCwd(), pageUrl });
141
+ commitBatch = transactionBatch;
142
+ if (!repairOnly && manualApply.countOps(transactionBatch) > 0) {
143
+ transaction = manualApply.writeTransaction({
144
+ pageUrl,
145
+ batch: transactionBatch,
146
+ });
147
+ } else if (repairOnly && existingTransaction) {
148
+ transaction = existingTransaction;
149
+ }
150
+ }
151
+ const envValue = currentEnv();
152
+ const requestedMode = (envValue.IMPECCABLE_LIVE_COPY_AGENT || 'auto').trim().toLowerCase();
153
+ const useChatRoute = requestedMode === 'chat'
154
+ || (requestedMode === 'auto' && chatAgentLikelyActive());
155
+ if (useChatRoute) {
156
+ routedProvider = 'chat';
157
+ const timeoutMs = Number(envValue.IMPECCABLE_LIVE_COPY_AGENT_TIMEOUT_MS || 120000);
158
+ result = await commitManualEdits({
159
+ cwd: projectCwd(),
160
+ pageUrl,
161
+ provider: 'chat',
162
+ env: envValue,
163
+ timeoutMs,
164
+ chatAvailable: chatAgentLikelyActive,
165
+ applyBatchToSource: (batch, context) => manualApply.pushBatchInChunksAndWait(batch, pageUrl, context),
166
+ repairOnly,
167
+ transactionId: transaction?.id || existingTransaction?.id || null,
168
+ batch: commitBatch,
169
+ });
170
+ } else {
171
+ const timeoutMs = Number(envValue.IMPECCABLE_LIVE_COPY_AGENT_TIMEOUT_MS || 120000);
172
+ const provider = ['codex', 'claude', 'mock'].includes(requestedMode) ? requestedMode : undefined;
173
+ result = await commitManualEdits({
174
+ cwd: projectCwd(),
175
+ pageUrl,
176
+ provider,
177
+ env: envValue,
178
+ timeoutMs,
179
+ chatAvailable: chatAgentLikelyActive,
180
+ repairOnly,
181
+ transactionId: transaction?.id || existingTransaction?.id || null,
182
+ batch: commitBatch,
183
+ });
184
+ }
185
+ } catch (err) {
186
+ if (transaction) {
187
+ manualApply.rollbackTransaction({
188
+ pageUrl,
189
+ reason: 'manual_edit_commit_exception',
190
+ });
191
+ }
192
+ const message = err.stderr?.toString?.() || err.message;
193
+ recordManualEditActivity('manual_edit_commit_failed', {
194
+ pageUrl,
195
+ provider: routedProvider,
196
+ error: 'manual_edit_commit_failed',
197
+ message,
198
+ transactionId: transaction?.id || null,
199
+ });
200
+ if (!asyncMode) {
201
+ sendJson(res, 500, {
202
+ error: 'manual_edit_commit_failed',
203
+ message,
204
+ });
205
+ }
206
+ return;
207
+ } finally {
208
+ if (transaction) {
209
+ const shouldKeepTransaction = result?.needsManualDecision === true;
210
+ if (!shouldKeepTransaction) manualApply.clearTransaction(transaction.id);
211
+ }
212
+ }
213
+ const { totalCount, perPage } = countPendingByPage(projectCwd());
214
+ if (result?.needsManualDecision) {
215
+ recordManualEditActivity('manual_edit_repair_needs_decision', {
216
+ pageUrl,
217
+ provider: routedProvider,
218
+ transactionId: transaction?.id || existingTransaction?.id || null,
219
+ repair: result.repair || null,
220
+ failed: summarizeManualApplyFailures(result.failed, projectCwd()),
221
+ files: Array.isArray(result.files) ? result.files.slice(0, 20).map((file) => summarizeManualLogFile(file, projectCwd())).filter(Boolean) : [],
222
+ remainingCount: pageUrl ? (perPage[pageUrl] || 0) : totalCount,
223
+ totalCount,
224
+ });
225
+ } else {
226
+ recordManualEditActivity('manual_edit_commit_done', {
227
+ pageUrl,
228
+ provider: routedProvider,
229
+ reason: result.reason || null,
230
+ repair: result.repair || null,
231
+ appliedCount: Array.isArray(result.applied) ? result.applied.length : 0,
232
+ failedCount: Array.isArray(result.failed) ? result.failed.length : 0,
233
+ failed: summarizeManualApplyFailures(result.failed, projectCwd()),
234
+ files: Array.isArray(result.files) ? result.files.slice(0, 20).map((file) => summarizeManualLogFile(file, projectCwd())).filter(Boolean) : [],
235
+ warnings: summarizeManualDiagnostics(result.warnings, projectCwd()),
236
+ rolledBackFiles: Array.isArray(result.rolledBackFiles) ? result.rolledBackFiles.slice(0, 20).map((file) => summarizeManualLogFile(file, projectCwd())).filter(Boolean) : [],
237
+ rollbackFailures: summarizeManualDiagnostics(result.rollbackFailures, projectCwd()),
238
+ unreportedFiles: Array.isArray(result.unreportedFiles) ? result.unreportedFiles.slice(0, 20).map((file) => summarizeManualLogFile(file, projectCwd())).filter(Boolean) : undefined,
239
+ noteCount: Array.isArray(result.notes) ? result.notes.length : 0,
240
+ cleared: result.cleared || 0,
241
+ remainingCount: pageUrl ? (perPage[pageUrl] || 0) : totalCount,
242
+ totalCount,
243
+ });
244
+ }
245
+ if (!asyncMode) {
246
+ sendJson(res, 200, { ...result, totalCount, perPage });
247
+ }
248
+ })();
249
+ return true;
250
+ }
251
+
252
+ if (p === '/manual-edit-repair-decision' && req.method === 'POST') {
253
+ let body = '';
254
+ req.on('data', (chunk) => { body += chunk; });
255
+ req.on('end', () => {
256
+ let payload = {};
257
+ try { payload = body ? JSON.parse(body) : {}; } catch {
258
+ sendJson(res, 400, { error: 'Invalid JSON' });
259
+ return;
260
+ }
261
+ const token = payload.token || url.searchParams.get('token');
262
+ if (token !== getToken()) { res.writeHead(401); res.end('Unauthorized'); return; }
263
+ const pageUrl = payload.pageUrl || url.searchParams.get('pageUrl') || null;
264
+ const action = String(payload.action || url.searchParams.get('action') || '').trim().toLowerCase();
265
+ if (action !== 'rollback') {
266
+ sendJson(res, 400, { error: 'unsupported_manual_edit_repair_decision', action });
267
+ return;
268
+ }
269
+ const rollback = manualApply.rollbackTransaction({
270
+ pageUrl,
271
+ reason: 'manual_edit_user_requested_rollback',
272
+ });
273
+ const { totalCount, perPage } = countPendingByPage(projectCwd());
274
+ const response = {
275
+ action,
276
+ pageUrl,
277
+ rollback,
278
+ remainingCount: pageUrl ? (perPage[pageUrl] || 0) : totalCount,
279
+ totalCount,
280
+ perPage,
281
+ };
282
+ recordManualEditActivity('manual_edit_repair_rollback_done', response);
283
+ sendJson(res, 200, response);
284
+ });
285
+ return true;
286
+ }
287
+
288
+ if (p === '/manual-edit-discard' && req.method === 'POST') {
289
+ const token = url.searchParams.get('token');
290
+ if (token !== getToken()) { res.writeHead(401); res.end('Unauthorized'); return true; }
291
+ const pageUrl = url.searchParams.get('pageUrl');
292
+ let discarded;
293
+ let discardedEntries = [];
294
+ let canceledApplyEvents = [];
295
+ let transactionRollback = null;
296
+ try {
297
+ const buffer = readManualEditsBuffer(projectCwd());
298
+ transactionRollback = manualApply.rollbackTransaction({
299
+ pageUrl,
300
+ reason: 'manual_edit_discarded',
301
+ });
302
+ if (pageUrl) {
303
+ discardedEntries = buffer.entries.filter((entry) => entry.pageUrl === pageUrl);
304
+ discarded = removeManualEditEntries(projectCwd(), (entry) => entry.pageUrl === pageUrl);
305
+ } else {
306
+ discardedEntries = buffer.entries;
307
+ discarded = truncateManualEditsBuffer(projectCwd());
308
+ }
309
+ canceledApplyEvents = manualApply.cancelPendingEvents(pageUrl);
310
+ } catch (err) {
311
+ sendJson(res, 500, { error: 'discard_failed', message: err.message });
312
+ return true;
313
+ }
314
+ const { totalCount, perPage } = countPendingByPage(projectCwd());
315
+ recordManualEditActivity('manual_edit_discarded', {
316
+ pageUrl,
317
+ discarded,
318
+ canceledApplyIds: canceledApplyEvents.map((event) => event.id),
319
+ transactionRollback: transactionRollback ? {
320
+ id: transactionRollback.id,
321
+ rolledBackFiles: transactionRollback.rolledBackFiles?.map((file) => summarizeManualLogFile(file, projectCwd())).filter(Boolean) || [],
322
+ rollbackFailures: summarizeManualDiagnostics(transactionRollback.rollbackFailures, projectCwd()),
323
+ skipped: transactionRollback.skipped,
324
+ } : undefined,
325
+ totalCount,
326
+ });
327
+ sendJson(res, 200, { discarded, entries: discardedEntries, canceledApplyEvents, totalCount, perPage });
328
+ return true;
329
+ }
330
+
331
+ if (p === '/manual-edit' && req.method === 'POST') {
332
+ sendJson(res, 410, { error: '/manual-edit is removed; use /manual-edit-stash and /manual-edit-commit for staged copy edits.' });
333
+ return true;
334
+ }
335
+
336
+ return false;
337
+ };
338
+ }
339
+
340
+ function sendJson(res, status, body) {
341
+ res.writeHead(status, { 'Content-Type': 'application/json' });
342
+ res.end(JSON.stringify(body));
343
+ }
344
+
345
+ function summarizePendingManualEditBatch(cwd, pageUrl = null) {
346
+ try {
347
+ const buffer = readManualEditsBuffer(cwd);
348
+ const entries = (buffer.entries || [])
349
+ .filter((entry) => !pageUrl || entry.pageUrl === pageUrl);
350
+ return {
351
+ pendingEntryCount: entries.length,
352
+ pendingOpCount: entries.reduce((sum, entry) => sum + (entry.ops?.length || 0), 0),
353
+ };
354
+ } catch (err) {
355
+ return { pendingSummaryError: err.message || String(err) };
356
+ }
357
+ }
@@ -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 '../lib/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
+ }