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,43 @@
1
+ #!/usr/bin/env node
2
+
3
+ /**
4
+ * Anti-Pattern Detector for Impeccable
5
+ * Copyright (c) 2026 Paul Bakaus
6
+ * SPDX-License-Identifier: Apache-2.0
7
+ *
8
+ * Public API facade. Runtime engines live under cli/engine/engines/.
9
+ */
10
+
11
+ import { detectCli } from './cli/main.mjs';
12
+
13
+ export { ANTIPATTERNS, RULE_ENGINE_SUPPORT, getAntipattern, getRulesForCategory, getRuleEngineSupport } from './registry/antipatterns.mjs';
14
+ export { SAFE_TAGS, BORDER_SAFE_TAGS, OVERUSED_FONTS, GENERIC_FONTS, KNOWN_SERIF_FONTS } from './shared/constants.mjs';
15
+ export { isNeutralColor, parseRgb, relativeLuminance, contrastRatio, parseGradientColors, hasChroma, getHue, colorToHex } from './shared/color.mjs';
16
+ export { isFullPage } from './shared/page.mjs';
17
+ export {
18
+ checkElementBorders,
19
+ checkElementMotion,
20
+ checkElementGlow,
21
+ checkPageTypography,
22
+ checkPageLayout,
23
+ checkHtmlPatterns,
24
+ } from './rules/checks.mjs';
25
+ export { createDetectorProfile, summarizeDetectorProfile } from './profile/profiler.mjs';
26
+ export { detectHtml } from './engines/static-html/detect-html.mjs';
27
+ export { detectUrl, createBrowserDetector } from './engines/browser/detect-url.mjs';
28
+ export { detectText, extractStyleBlocks, extractCSSinJS } from './engines/regex/detect-text.mjs';
29
+ export {
30
+ walkDir,
31
+ SCANNABLE_EXTENSIONS,
32
+ SKIP_DIRS,
33
+ buildImportGraph,
34
+ resolveImport,
35
+ detectFrameworkConfig,
36
+ isPortListening,
37
+ FRAMEWORK_CONFIGS,
38
+ } from './node/file-system.mjs';
39
+ export { formatFindings, detectCli } from './cli/main.mjs';
40
+
41
+ const isMainModule = process.argv[1]?.endsWith('detect-antipatterns.mjs') ||
42
+ process.argv[1]?.endsWith('detect-antipatterns.mjs/');
43
+ if (isMainModule) detectCli();
@@ -0,0 +1,252 @@
1
+ import fs from 'node:fs';
2
+ import path from 'node:path';
3
+ import { fileURLToPath } from 'node:url';
4
+
5
+ import { finding } from '../../findings.mjs';
6
+ import { filterByProviders } from '../../registry/antipatterns.mjs';
7
+ import { profileFindingsAsync, profileStep, profileStepAsync } from '../../profile/profiler.mjs';
8
+ import { captureVisualContrastCandidate } from '../visual/screenshot-contrast.mjs';
9
+
10
+ async function runVisualContrastFallback(page, serializedGroups, options, profile, target) {
11
+ if (options?.visualContrast === false) return [];
12
+ const maxCandidates = Number.isFinite(options?.visualContrastMaxCandidates)
13
+ ? options.visualContrastMaxCandidates
14
+ : 12;
15
+ const scrollOffscreen = options?.visualContrastScrollOffscreen !== false;
16
+ const existingLowContrastSelectors = new Set(
17
+ serializedGroups
18
+ .filter(group => group.findings?.some(f => f.type === 'low-contrast'))
19
+ .map(group => group.selector)
20
+ .filter(Boolean)
21
+ );
22
+
23
+ let browserAnalyses = [];
24
+ const findings = [];
25
+ if (options?.visualContrastBrowser !== false) {
26
+ const browserFindings = await profileFindingsAsync(profile, {
27
+ engine: 'browser',
28
+ phase: 'visual-contrast',
29
+ ruleId: 'browser-fallback',
30
+ target,
31
+ }, async () => {
32
+ browserAnalyses = await page.evaluate(async ({ maxCandidates, scrollOffscreen }) => {
33
+ if (typeof window.impeccableAnalyzeVisualContrast !== 'function') return [];
34
+ return window.impeccableAnalyzeVisualContrast({ maxCandidates, scrollOffscreen });
35
+ }, { maxCandidates, scrollOffscreen });
36
+ return browserAnalyses
37
+ .filter(result => result.finding && !existingLowContrastSelectors.has(result.selector))
38
+ .map(result => result.finding);
39
+ });
40
+ findings.push(...browserFindings);
41
+ }
42
+
43
+ let candidates = browserAnalyses.length > 0 ? browserAnalyses : [];
44
+ if (candidates.length === 0) {
45
+ candidates = await profileStepAsync(profile, {
46
+ engine: 'browser',
47
+ phase: 'visual-contrast',
48
+ ruleId: 'collect-candidates',
49
+ target,
50
+ }, () => page.evaluate(({ maxCandidates }) => {
51
+ if (typeof window.impeccableCollectVisualContrastCandidates !== 'function') return [];
52
+ return window.impeccableCollectVisualContrastCandidates({ maxCandidates });
53
+ }, { maxCandidates }));
54
+ }
55
+
56
+ const viewport = options?.viewport || { width: 1280, height: 800 };
57
+ const browserResolvedSelectors = new Set(
58
+ browserAnalyses
59
+ .filter(result => result.status === 'fail' || result.status === 'pass')
60
+ .map(result => result.selector)
61
+ .filter(Boolean)
62
+ );
63
+ const filtered = candidates.filter(candidate =>
64
+ !existingLowContrastSelectors.has(candidate.selector) &&
65
+ !browserResolvedSelectors.has(candidate.selector)
66
+ );
67
+ if (options?.visualContrastPixel === false) return findings;
68
+ for (const candidate of filtered) {
69
+ const result = await profileFindingsAsync(profile, {
70
+ engine: 'browser',
71
+ phase: 'visual-contrast',
72
+ ruleId: 'pixel-diff',
73
+ target,
74
+ }, async () => {
75
+ const finding = await captureVisualContrastCandidate(page, candidate, viewport);
76
+ return finding ? [finding] : [];
77
+ });
78
+ findings.push(...result);
79
+ }
80
+ return findings;
81
+ }
82
+
83
+ // ---------------------------------------------------------------------------
84
+ // Puppeteer detection (for URLs)
85
+ // ---------------------------------------------------------------------------
86
+
87
+ async function detectUrl(url, options = {}) {
88
+ const profile = options?.profile;
89
+ const waitUntil = options?.waitUntil || 'networkidle0';
90
+ const settleMs = Number.isFinite(options?.settleMs) ? options.settleMs : 0;
91
+ const viewport = options?.viewport || { width: 1280, height: 800 };
92
+ const externalBrowser = options?.browser || null;
93
+ let puppeteer;
94
+ if (!externalBrowser) {
95
+ try {
96
+ puppeteer = await profileStepAsync(profile, {
97
+ engine: 'browser',
98
+ phase: 'setup',
99
+ ruleId: 'import-puppeteer',
100
+ target: url,
101
+ }, () => import('puppeteer'));
102
+ } catch {
103
+ throw new Error('puppeteer is required for URL scanning. Install: npm install puppeteer');
104
+ }
105
+ }
106
+
107
+ // Read the browser detection script — reuse it instead of reimplementing
108
+ const browserScriptPath = path.resolve(
109
+ path.dirname(fileURLToPath(import.meta.url)),
110
+ '..',
111
+ '..',
112
+ 'detect-antipatterns-browser.js'
113
+ );
114
+ let browserScript;
115
+ try {
116
+ browserScript = profileStep(profile, {
117
+ engine: 'browser',
118
+ phase: 'setup',
119
+ ruleId: 'read-browser-script',
120
+ target: url,
121
+ }, () => fs.readFileSync(browserScriptPath, 'utf-8'));
122
+ } catch {
123
+ throw new Error(`Browser script not found at ${browserScriptPath}`);
124
+ }
125
+
126
+ // CI runners (GitHub Actions Ubuntu) block unprivileged user namespaces, so
127
+ // Chrome can't initialize its sandbox there. Disable the sandbox only when
128
+ // running in CI; local users keep the default hardened launch.
129
+ const launchArgs = process.env.CI ? ['--no-sandbox', '--disable-setuid-sandbox'] : [];
130
+ const browser = externalBrowser || await profileStepAsync(profile, {
131
+ engine: 'browser',
132
+ phase: 'load',
133
+ ruleId: 'launch-browser',
134
+ target: url,
135
+ }, () => puppeteer.default.launch({ headless: true, args: launchArgs }));
136
+ const page = await profileStepAsync(profile, {
137
+ engine: 'browser',
138
+ phase: 'load',
139
+ ruleId: 'new-page',
140
+ target: url,
141
+ }, () => browser.newPage());
142
+ let results = [];
143
+ try {
144
+ await profileStepAsync(profile, {
145
+ engine: 'browser',
146
+ phase: 'load',
147
+ ruleId: 'set-viewport',
148
+ target: url,
149
+ }, () => page.setViewport(viewport));
150
+ await profileStepAsync(profile, {
151
+ engine: 'browser',
152
+ phase: 'load',
153
+ ruleId: `goto:${waitUntil}`,
154
+ target: url,
155
+ }, () => page.goto(url, { waitUntil, timeout: 30000 }));
156
+ if (settleMs > 0) {
157
+ await profileStepAsync(profile, {
158
+ engine: 'browser',
159
+ phase: 'load',
160
+ ruleId: 'settle',
161
+ target: url,
162
+ }, () => new Promise(resolve => setTimeout(resolve, settleMs)));
163
+ }
164
+
165
+ // Inject the browser detection script and collect results
166
+ await profileStepAsync(profile, {
167
+ engine: 'browser',
168
+ phase: 'scan',
169
+ ruleId: 'configure-pure-detect',
170
+ target: url,
171
+ }, () => page.evaluate(() => {
172
+ window.__IMPECCABLE_CONFIG__ = {
173
+ ...(window.__IMPECCABLE_CONFIG__ || {}),
174
+ autoScan: false,
175
+ };
176
+ }));
177
+ await profileStepAsync(profile, {
178
+ engine: 'browser',
179
+ phase: 'scan',
180
+ ruleId: 'inject-browser-script',
181
+ target: url,
182
+ }, () => page.evaluate(browserScript));
183
+ let serializedGroups = [];
184
+ results = await profileFindingsAsync(profile, {
185
+ engine: 'browser',
186
+ phase: 'scan',
187
+ ruleId: 'browser-scan',
188
+ target: url,
189
+ }, async () => {
190
+ serializedGroups = await page.evaluate(() => {
191
+ if (!window.impeccableDetect) return [];
192
+ return window.impeccableDetect({ decorate: false, serialize: true });
193
+ });
194
+ return serializedGroups.flatMap(({ findings }) =>
195
+ findings.map(f => ({ id: f.type, snippet: f.detail }))
196
+ );
197
+ });
198
+ const visualFindings = await runVisualContrastFallback(page, serializedGroups, options, profile, url);
199
+ results.push(...visualFindings);
200
+ } finally {
201
+ await profileStepAsync(profile, {
202
+ engine: 'browser',
203
+ phase: 'load',
204
+ ruleId: 'close-page',
205
+ target: url,
206
+ }, () => page.close().catch(() => {}));
207
+ if (!externalBrowser) {
208
+ await profileStepAsync(profile, {
209
+ engine: 'browser',
210
+ phase: 'load',
211
+ ruleId: 'close-browser',
212
+ target: url,
213
+ }, () => browser.close());
214
+ }
215
+ }
216
+ return filterByProviders(results.map(f => finding(f.id, url, f.snippet)), options.providers);
217
+ }
218
+
219
+ async function createBrowserDetector(options = {}) {
220
+ let puppeteer;
221
+ try {
222
+ puppeteer = await import('puppeteer');
223
+ } catch {
224
+ throw new Error('puppeteer is required for URL scanning. Install: npm install puppeteer');
225
+ }
226
+ const launchArgs = options.launchArgs || (process.env.CI ? ['--no-sandbox', '--disable-setuid-sandbox'] : []);
227
+ const browser = options.browser || await puppeteer.default.launch({
228
+ headless: options.headless ?? true,
229
+ args: launchArgs,
230
+ });
231
+ const ownsBrowser = !options.browser;
232
+ const defaults = {
233
+ waitUntil: options.waitUntil || 'load',
234
+ settleMs: Number.isFinite(options.settleMs) ? options.settleMs : 100,
235
+ viewport: options.viewport || { width: 1280, height: 800 },
236
+ };
237
+ return {
238
+ browser,
239
+ async detectUrl(url, scanOptions = {}) {
240
+ return detectUrl(url, {
241
+ ...defaults,
242
+ ...scanOptions,
243
+ browser,
244
+ });
245
+ },
246
+ async close() {
247
+ if (ownsBrowser) await browser.close().catch(() => {});
248
+ },
249
+ };
250
+ }
251
+
252
+ export { runVisualContrastFallback, detectUrl, createBrowserDetector };