@sarjallab09/figma-intelligence 1.0.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 (286) hide show
  1. package/LICENSE +26 -0
  2. package/README.md +327 -0
  3. package/bin/cli.js +859 -0
  4. package/design-bridge/.env.example +5 -0
  5. package/design-bridge/bridge.js +196 -0
  6. package/design-bridge/lib/assets.js +367 -0
  7. package/design-bridge/lib/prompt.js +85 -0
  8. package/design-bridge/lib/server.js +66 -0
  9. package/design-bridge/lib/stitch.js +37 -0
  10. package/design-bridge/lib/tokens.js +82 -0
  11. package/design-bridge/package-lock.json +579 -0
  12. package/design-bridge/package.json +19 -0
  13. package/figma-bridge-plugin/README.md +97 -0
  14. package/figma-bridge-plugin/anthropic-chat-runner.js +192 -0
  15. package/figma-bridge-plugin/bridge-relay.js +2363 -0
  16. package/figma-bridge-plugin/chat-runner.js +459 -0
  17. package/figma-bridge-plugin/code.js +1528 -0
  18. package/figma-bridge-plugin/codex-runner.js +505 -0
  19. package/figma-bridge-plugin/component-schemas.js +110 -0
  20. package/figma-bridge-plugin/content-context.js +869 -0
  21. package/figma-bridge-plugin/create-button.js +216 -0
  22. package/figma-bridge-plugin/gemini-cli-runner.js +291 -0
  23. package/figma-bridge-plugin/gemini-runner.js +187 -0
  24. package/figma-bridge-plugin/html-to-figma.js +927 -0
  25. package/figma-bridge-plugin/knowledge-hub/.gitkeep +0 -0
  26. package/figma-bridge-plugin/knowledge-hub/uspec-references/anatomy-spec.md +159 -0
  27. package/figma-bridge-plugin/knowledge-hub/uspec-references/api-spec.md +162 -0
  28. package/figma-bridge-plugin/knowledge-hub/uspec-references/color-spec.md +148 -0
  29. package/figma-bridge-plugin/knowledge-hub/uspec-references/full-spec-template.md +314 -0
  30. package/figma-bridge-plugin/knowledge-hub/uspec-references/property-spec.md +175 -0
  31. package/figma-bridge-plugin/knowledge-hub/uspec-references/screen-reader-spec.md +180 -0
  32. package/figma-bridge-plugin/knowledge-hub/uspec-references/structure-spec.md +165 -0
  33. package/figma-bridge-plugin/manifest.json +21 -0
  34. package/figma-bridge-plugin/package-lock.json +1936 -0
  35. package/figma-bridge-plugin/package.json +20 -0
  36. package/figma-bridge-plugin/perplexity-runner.js +188 -0
  37. package/figma-bridge-plugin/references/SKILL.md +178 -0
  38. package/figma-bridge-plugin/references/anatomy-spec.md +159 -0
  39. package/figma-bridge-plugin/references/api-spec.md +162 -0
  40. package/figma-bridge-plugin/references/color-spec.md +148 -0
  41. package/figma-bridge-plugin/references/full-spec-template.md +314 -0
  42. package/figma-bridge-plugin/references/property-spec.md +175 -0
  43. package/figma-bridge-plugin/references/screen-reader-spec.md +180 -0
  44. package/figma-bridge-plugin/references/structure-spec.md +165 -0
  45. package/figma-bridge-plugin/shared-prompt-config.js +604 -0
  46. package/figma-bridge-plugin/spec-helpers/build-table.js +269 -0
  47. package/figma-bridge-plugin/spec-helpers/classify-elements.js +189 -0
  48. package/figma-bridge-plugin/spec-helpers/index.js +35 -0
  49. package/figma-bridge-plugin/spec-helpers/parse-figma-link.js +49 -0
  50. package/figma-bridge-plugin/spec-helpers/position-markers.js +158 -0
  51. package/figma-bridge-plugin/stitch-auth.js +322 -0
  52. package/figma-bridge-plugin/stitch-runner.js +1427 -0
  53. package/figma-bridge-plugin/token-resolver.js +107 -0
  54. package/figma-bridge-plugin/ui.html +4467 -0
  55. package/figma-intelligence-layer/.env.example +39 -0
  56. package/figma-intelligence-layer/docs/local-image-generation.md +60 -0
  57. package/figma-intelligence-layer/examples/comfyui-workflow-template.example.json +101 -0
  58. package/figma-intelligence-layer/jest.config.js +14 -0
  59. package/figma-intelligence-layer/mcp-config.json +19 -0
  60. package/figma-intelligence-layer/package-lock.json +5892 -0
  61. package/figma-intelligence-layer/package.json +48 -0
  62. package/figma-intelligence-layer/scripts/setup-comfyui-local.sh +67 -0
  63. package/figma-intelligence-layer/scripts/start-comfyui.sh +33 -0
  64. package/figma-intelligence-layer/src/index.ts +2233 -0
  65. package/figma-intelligence-layer/src/shared/auto-layout-validator.ts +404 -0
  66. package/figma-intelligence-layer/src/shared/cache.ts +187 -0
  67. package/figma-intelligence-layer/src/shared/color-operations.ts +533 -0
  68. package/figma-intelligence-layer/src/shared/color-utils.ts +138 -0
  69. package/figma-intelligence-layer/src/shared/component-script-builder.ts +413 -0
  70. package/figma-intelligence-layer/src/shared/component-templates.ts +2767 -0
  71. package/figma-intelligence-layer/src/shared/concept-taxonomy.ts +694 -0
  72. package/figma-intelligence-layer/src/shared/decision-log.ts +128 -0
  73. package/figma-intelligence-layer/src/shared/design-system-context.ts +568 -0
  74. package/figma-intelligence-layer/src/shared/design-system-intelligence.ts +131 -0
  75. package/figma-intelligence-layer/src/shared/design-system-matcher.ts +184 -0
  76. package/figma-intelligence-layer/src/shared/design-system-normalizers.ts +196 -0
  77. package/figma-intelligence-layer/src/shared/design-system-tokens.ts +295 -0
  78. package/figma-intelligence-layer/src/shared/dtcg-validator.ts +530 -0
  79. package/figma-intelligence-layer/src/shared/enrichment-pipeline.ts +671 -0
  80. package/figma-intelligence-layer/src/shared/figma-bridge.ts +1408 -0
  81. package/figma-intelligence-layer/src/shared/font-config.ts +126 -0
  82. package/figma-intelligence-layer/src/shared/icon-catalog.ts +360 -0
  83. package/figma-intelligence-layer/src/shared/icon-fetch.ts +80 -0
  84. package/figma-intelligence-layer/src/shared/prototype-script-builder.ts +162 -0
  85. package/figma-intelligence-layer/src/shared/response-compression.ts +440 -0
  86. package/figma-intelligence-layer/src/shared/semantic-token-catalog.ts +324 -0
  87. package/figma-intelligence-layer/src/shared/token-binder.ts +505 -0
  88. package/figma-intelligence-layer/src/shared/token-math.ts +427 -0
  89. package/figma-intelligence-layer/src/shared/token-naming.ts +468 -0
  90. package/figma-intelligence-layer/src/shared/token-utils.ts +420 -0
  91. package/figma-intelligence-layer/src/shared/types.ts +346 -0
  92. package/figma-intelligence-layer/src/shared/typography-presets.ts +94 -0
  93. package/figma-intelligence-layer/src/shared/unsplash.ts +165 -0
  94. package/figma-intelligence-layer/src/shared/vision-client.ts +607 -0
  95. package/figma-intelligence-layer/src/shared/vision-provider-anthropic.ts +334 -0
  96. package/figma-intelligence-layer/src/shared/vision-provider-openai.ts +446 -0
  97. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/a11y-annotate-handler.ts +782 -0
  98. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/a11y-annotate-renderer.ts +496 -0
  99. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/a11y-annotation-kit.ts +230 -0
  100. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/colorblind-sim.ts +66 -0
  101. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/index.ts +810 -0
  102. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/keyboard-sr-order-analyzer.ts +1191 -0
  103. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/keyboard-sr-order-figma-page.ts +1346 -0
  104. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/keyboard-sr-order-handler.ts +148 -0
  105. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/vpat-figma-page.ts +499 -0
  106. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/vpat-report.ts +910 -0
  107. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/wcag-checker.ts +989 -0
  108. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/wcag-criteria.ts +1160 -0
  109. package/figma-intelligence-layer/src/tools/phase1-vision/design-from-ref/index.ts +424 -0
  110. package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/component-recognizer.ts +38 -0
  111. package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/ds-matcher.ts +111 -0
  112. package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/font-matcher.ts +114 -0
  113. package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/icon-resolver.ts +103 -0
  114. package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/index.ts +1060 -0
  115. package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/layout-segmenter.ts +18 -0
  116. package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/token-inferencer.ts +39 -0
  117. package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/vision-pipeline.ts +58 -0
  118. package/figma-intelligence-layer/src/tools/phase1-vision/sketch-to-design/index.ts +298 -0
  119. package/figma-intelligence-layer/src/tools/phase1-vision/visual-audit/index.ts +197 -0
  120. package/figma-intelligence-layer/src/tools/phase2-accuracy/component-audit/index.ts +494 -0
  121. package/figma-intelligence-layer/src/tools/phase2-accuracy/intent-translator/index.ts +356 -0
  122. package/figma-intelligence-layer/src/tools/phase2-accuracy/layout-intelligence/container-patterns.ts +123 -0
  123. package/figma-intelligence-layer/src/tools/phase2-accuracy/layout-intelligence/index.ts +663 -0
  124. package/figma-intelligence-layer/src/tools/phase2-accuracy/lint-rules/built-in-rules.yaml +56 -0
  125. package/figma-intelligence-layer/src/tools/phase2-accuracy/lint-rules/index.ts +614 -0
  126. package/figma-intelligence-layer/src/tools/phase2-accuracy/lint-rules/rule-engine.ts +113 -0
  127. package/figma-intelligence-layer/src/tools/phase2-accuracy/theme-generator/color-theory.ts +178 -0
  128. package/figma-intelligence-layer/src/tools/phase2-accuracy/theme-generator/index.ts +470 -0
  129. package/figma-intelligence-layer/src/tools/phase2-accuracy/variant-expander/index.ts +429 -0
  130. package/figma-intelligence-layer/src/tools/phase2-accuracy/variant-expander/token-override-maps.ts +226 -0
  131. package/figma-intelligence-layer/src/tools/phase3-generation/ai-image-insert/index.ts +535 -0
  132. package/figma-intelligence-layer/src/tools/phase3-generation/component-archaeologist/index.ts +660 -0
  133. package/figma-intelligence-layer/src/tools/phase3-generation/component-archaeologist/pattern-fingerprints.ts +209 -0
  134. package/figma-intelligence-layer/src/tools/phase3-generation/composition-builder/index.ts +540 -0
  135. package/figma-intelligence-layer/src/tools/phase3-generation/figma-animated-build.ts +391 -0
  136. package/figma-intelligence-layer/src/tools/phase3-generation/page-architect/index.ts +2019 -0
  137. package/figma-intelligence-layer/src/tools/phase3-generation/page-architect/screen-templates.ts +131 -0
  138. package/figma-intelligence-layer/src/tools/phase3-generation/prototype-map/index.ts +381 -0
  139. package/figma-intelligence-layer/src/tools/phase3-generation/prototype-wire/index.ts +565 -0
  140. package/figma-intelligence-layer/src/tools/phase3-generation/swarm-build/index.ts +764 -0
  141. package/figma-intelligence-layer/src/tools/phase3-generation/system-drift/index.ts +535 -0
  142. package/figma-intelligence-layer/src/tools/phase3-generation/unsplash-search/index.ts +84 -0
  143. package/figma-intelligence-layer/src/tools/phase3-generation/url-to-frame/index.ts +401 -0
  144. package/figma-intelligence-layer/src/tools/phase4-sync/animation-specifier/code-generators/css-animations.ts +68 -0
  145. package/figma-intelligence-layer/src/tools/phase4-sync/animation-specifier/code-generators/framer-motion.ts +78 -0
  146. package/figma-intelligence-layer/src/tools/phase4-sync/animation-specifier/code-generators/swift-animations.ts +93 -0
  147. package/figma-intelligence-layer/src/tools/phase4-sync/animation-specifier/index.ts +596 -0
  148. package/figma-intelligence-layer/src/tools/phase4-sync/ci-check/index.ts +462 -0
  149. package/figma-intelligence-layer/src/tools/phase4-sync/export-tokens/index.ts +1470 -0
  150. package/figma-intelligence-layer/src/tools/phase4-sync/generate-component-code/index.ts +829 -0
  151. package/figma-intelligence-layer/src/tools/phase4-sync/handoff-spec/index.ts +702 -0
  152. package/figma-intelligence-layer/src/tools/phase4-sync/icon-library-sync/index.ts +483 -0
  153. package/figma-intelligence-layer/src/tools/phase4-sync/sync-from-code/index.ts +501 -0
  154. package/figma-intelligence-layer/src/tools/phase4-sync/sync-from-code/storybook-parser.ts +106 -0
  155. package/figma-intelligence-layer/src/tools/phase4-sync/watch-docs/index.ts +676 -0
  156. package/figma-intelligence-layer/src/tools/phase4-sync/webhook-listener/index.ts +560 -0
  157. package/figma-intelligence-layer/src/tools/phase5-governance/apg-doc/index.ts +1043 -0
  158. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/component-detection.ts +620 -0
  159. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/anatomy.ts +331 -0
  160. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/color-tokens.ts +77 -0
  161. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/properties.ts +54 -0
  162. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/snapshot.ts +287 -0
  163. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/spacing.ts +71 -0
  164. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/states.ts +43 -0
  165. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/typography.ts +71 -0
  166. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/index.ts +221 -0
  167. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/_default.ts +166 -0
  168. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/accordion.ts +232 -0
  169. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/alert.ts +234 -0
  170. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/avatar-group.ts +270 -0
  171. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/avatar.ts +249 -0
  172. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/badge.ts +231 -0
  173. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/banner.ts +293 -0
  174. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/breadcrumb.ts +240 -0
  175. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/button.ts +243 -0
  176. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/calendar.ts +307 -0
  177. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/card.ts +143 -0
  178. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/checkbox.ts +227 -0
  179. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/chip.ts +233 -0
  180. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/combobox.ts +282 -0
  181. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/datepicker.ts +276 -0
  182. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/divider.ts +223 -0
  183. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/drawer.ts +255 -0
  184. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/dropdown-menu.ts +289 -0
  185. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/empty-state.ts +261 -0
  186. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/file-uploader.ts +290 -0
  187. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/form.ts +265 -0
  188. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/grid.ts +238 -0
  189. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/icon.ts +255 -0
  190. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/index.ts +128 -0
  191. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/inline-edit.ts +286 -0
  192. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/inline-message.ts +255 -0
  193. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/input.ts +330 -0
  194. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/link.ts +247 -0
  195. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/list.ts +250 -0
  196. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/menu.ts +247 -0
  197. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/modal.ts +144 -0
  198. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/navbar.ts +264 -0
  199. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/navigation.ts +251 -0
  200. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/number-input.ts +261 -0
  201. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/pagination.ts +248 -0
  202. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/popover.ts +270 -0
  203. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/progress.ts +251 -0
  204. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/radio.ts +142 -0
  205. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/range-slider.ts +282 -0
  206. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/rating.ts +250 -0
  207. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/search.ts +258 -0
  208. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/segmented-control.ts +265 -0
  209. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/select.ts +319 -0
  210. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/skeleton.ts +256 -0
  211. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/slider.ts +232 -0
  212. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/spinner.ts +239 -0
  213. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/status-dot.ts +252 -0
  214. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/stepper.ts +270 -0
  215. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/table.ts +244 -0
  216. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/tabs.ts +143 -0
  217. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/tag.ts +243 -0
  218. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/textarea.ts +259 -0
  219. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/time-picker.ts +293 -0
  220. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/toast.ts +144 -0
  221. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/toggle.ts +289 -0
  222. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/toolbar.ts +267 -0
  223. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/tooltip.ts +232 -0
  224. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/treeview.ts +257 -0
  225. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/typography.ts +319 -0
  226. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/legacy-compat.ts +121 -0
  227. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/renderers/anatomy-diagram.ts +430 -0
  228. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/renderers/figma-page.ts +312 -0
  229. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/renderers/json.ts +129 -0
  230. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/renderers/markdown.ts +78 -0
  231. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/renderers/visual-doc.ts +2333 -0
  232. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/accessibility.ts +100 -0
  233. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/anatomy.ts +32 -0
  234. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/color-tokens.ts +59 -0
  235. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/content-guidance.ts +18 -0
  236. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/design-tokens.ts +53 -0
  237. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/interaction-rules.ts +19 -0
  238. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/overview.ts +91 -0
  239. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/properties-api.ts +71 -0
  240. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/qa-criteria.ts +19 -0
  241. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/related-components.ts +110 -0
  242. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/responsive.ts +19 -0
  243. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/size-specs.ts +67 -0
  244. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/spacing-structure.ts +58 -0
  245. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/state-specs.ts +79 -0
  246. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/states.ts +50 -0
  247. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/type-hierarchy.ts +33 -0
  248. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/typography.ts +55 -0
  249. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/usage-guidelines.ts +73 -0
  250. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/variants.ts +81 -0
  251. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/types.ts +409 -0
  252. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec-sheet/index.ts +198 -0
  253. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec-sheet/renderer.ts +701 -0
  254. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec-sheet/types.ts +88 -0
  255. package/figma-intelligence-layer/src/tools/phase5-governance/decision-log/index.ts +135 -0
  256. package/figma-intelligence-layer/src/tools/phase5-governance/design-decision-log/index.ts +491 -0
  257. package/figma-intelligence-layer/src/tools/phase5-governance/ds-primitives/index.ts +416 -0
  258. package/figma-intelligence-layer/src/tools/phase5-governance/ds-scaffolder/index.ts +722 -0
  259. package/figma-intelligence-layer/src/tools/phase5-governance/ds-variables/index.ts +449 -0
  260. package/figma-intelligence-layer/src/tools/phase5-governance/health-report/index.ts +393 -0
  261. package/figma-intelligence-layer/src/tools/phase5-governance/taxonomy-docs/index.ts +406 -0
  262. package/figma-intelligence-layer/src/tools/phase5-governance/taxonomy-docs/renderers/figma-page.ts +292 -0
  263. package/figma-intelligence-layer/src/tools/phase5-governance/taxonomy-docs/renderers/json.ts +24 -0
  264. package/figma-intelligence-layer/src/tools/phase5-governance/taxonomy-docs/renderers/markdown.ts +172 -0
  265. package/figma-intelligence-layer/src/tools/phase5-governance/taxonomy-docs/renderers/naming-guide.ts +409 -0
  266. package/figma-intelligence-layer/src/tools/phase5-governance/token-analytics/index.ts +594 -0
  267. package/figma-intelligence-layer/src/tools/phase5-governance/token-docs/index.ts +710 -0
  268. package/figma-intelligence-layer/src/tools/phase5-governance/token-migrate/index.ts +458 -0
  269. package/figma-intelligence-layer/src/tools/phase5-governance/token-naming/index.ts +134 -0
  270. package/figma-intelligence-layer/tests/apg-doc.test.ts +101 -0
  271. package/figma-intelligence-layer/tests/design-system-context.test.ts +152 -0
  272. package/figma-intelligence-layer/tests/design-system-matcher.test.ts +144 -0
  273. package/figma-intelligence-layer/tests/figma-bridge.test.ts +83 -0
  274. package/figma-intelligence-layer/tests/generate-image-and-insert.test.ts +56 -0
  275. package/figma-intelligence-layer/tests/screen-cloner-regression.test.ts +69 -0
  276. package/figma-intelligence-layer/tests/smoke.test.ts +174 -0
  277. package/figma-intelligence-layer/tests/spec-generator.test.ts +127 -0
  278. package/figma-intelligence-layer/tests/token-migrate.test.ts +21 -0
  279. package/figma-intelligence-layer/tests/token-naming.test.ts +30 -0
  280. package/figma-intelligence-layer/tsconfig.json +19 -0
  281. package/package.json +35 -0
  282. package/scripts/clean-existing-chunks.js +179 -0
  283. package/scripts/connect-ai-tool.js +490 -0
  284. package/scripts/convert-hub-pdfs.js +425 -0
  285. package/scripts/figma-mcp-status.js +349 -0
  286. package/scripts/register-codex-mcp.js +96 -0
@@ -0,0 +1,459 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * chat-runner.js — Spawns a Claude CLI subprocess for chat messages.
4
+ * Uses the user's existing Claude Code subscription — no API key required.
5
+ *
6
+ * OPTIMIZED: Session persistence via --session-id / --resume.
7
+ * Second+ messages skip system prompt, MCP config, and conversation history.
8
+ * Per-request disk I/O eliminated (MCP config written once at startup).
9
+ * Prompt bloat removed (expandShortPrompt, AGENTS.md injection, skill files).
10
+ */
11
+
12
+ const { spawn } = require("child_process");
13
+ const { writeFileSync, unlinkSync, readFileSync, mkdirSync, existsSync } = require("fs");
14
+ const { tmpdir, homedir } = require("os");
15
+ const { join, resolve } = require("path");
16
+ const crypto = require("crypto");
17
+
18
+ const {
19
+ SYSTEM_PROMPT,
20
+ buildSystemPrompt,
21
+ buildChatPrompt,
22
+ buildDualOutputPrompt,
23
+ buildSkillAddendum,
24
+ detectActiveSkills,
25
+ REPO_DIR,
26
+ } = require("./shared-prompt-config");
27
+
28
+ const MCP_CONFIG_PATH = join(tmpdir(), "figma-intelligence-chat-mcp.json");
29
+ const CLAUDE_SETTINGS_PATH = join(homedir(), ".claude", "settings.json");
30
+
31
+ // Use the absolute claude binary path stored by setup.sh in launchd env.
32
+ // Falls back to "claude" when running interactively (it's on PATH then).
33
+ const CLAUDE_BIN = process.env.CLAUDE_BIN_PATH || "claude";
34
+
35
+ // ── Session Persistence ──────────────────────────────────────────────────────
36
+ // First message uses --session-id <uuid> (creates a new session).
37
+ // Subsequent messages use --resume <uuid> (reloads full conversation context).
38
+ // No need to re-send system prompt, MCP config, or conversation history on resume.
39
+ let activeSessionIds = { code: null, chat: null, dual: null };
40
+
41
+ function resetSession(mode) {
42
+ if (mode) {
43
+ activeSessionIds[mode] = null;
44
+ console.log(`[chat-runner] ${mode} session reset — next message starts a new session.`);
45
+ } else {
46
+ activeSessionIds = { code: null, chat: null };
47
+ console.log("[chat-runner] All sessions reset — next message starts a new session.");
48
+ }
49
+ }
50
+
51
+ // ── MCP Config (written once at startup) ─────────────────────────────────────
52
+
53
+ function getFigmaToken() {
54
+ try {
55
+ if (existsSync(CLAUDE_SETTINGS_PATH)) {
56
+ const settings = JSON.parse(readFileSync(CLAUDE_SETTINGS_PATH, "utf8"));
57
+ return settings?.mcpServers?.["figma-intelligence-layer"]?.env?.FIGMA_ACCESS_TOKEN || "";
58
+ }
59
+ } catch {}
60
+ return "";
61
+ }
62
+
63
+ function getClaudeSettings() {
64
+ try {
65
+ if (existsSync(CLAUDE_SETTINGS_PATH)) {
66
+ return JSON.parse(readFileSync(CLAUDE_SETTINGS_PATH, "utf8"));
67
+ }
68
+ } catch {}
69
+ return {};
70
+ }
71
+
72
+ function writeMcpConfig(bridgePort) {
73
+ const port = String(bridgePort || process.env.BRIDGE_PORT || "9001");
74
+ const figmaToken = getFigmaToken();
75
+ const settings = getClaudeSettings();
76
+ const existingServers =
77
+ settings?.mcpServers && typeof settings.mcpServers === "object"
78
+ ? settings.mcpServers
79
+ : {};
80
+
81
+ const existingFigmaEnv =
82
+ existingServers["figma-intelligence-layer"]?.env &&
83
+ typeof existingServers["figma-intelligence-layer"].env === "object"
84
+ ? existingServers["figma-intelligence-layer"].env
85
+ : {};
86
+
87
+ const designBridgeEnv =
88
+ existingServers["design-bridge"]?.env &&
89
+ typeof existingServers["design-bridge"].env === "object"
90
+ ? existingServers["design-bridge"].env
91
+ : {};
92
+
93
+ const config = {
94
+ mcpServers: {
95
+ "figma-intelligence-layer": {
96
+ type: "stdio",
97
+ command: "node",
98
+ args: [join(REPO_DIR, "figma-intelligence-layer", "dist", "index.js")],
99
+ env: {
100
+ ...(designBridgeEnv.UNSPLASH_ACCESS_KEY ? { UNSPLASH_ACCESS_KEY: designBridgeEnv.UNSPLASH_ACCESS_KEY } : {}),
101
+ ...(designBridgeEnv.PEXELS_API_KEY ? { PEXELS_API_KEY: designBridgeEnv.PEXELS_API_KEY } : {}),
102
+ ...(designBridgeEnv.STITCH_API_KEY ? { STITCH_API_KEY: designBridgeEnv.STITCH_API_KEY } : {}),
103
+ ...(designBridgeEnv.GOOGLE_CLOUD_PROJECT ? { GOOGLE_CLOUD_PROJECT: designBridgeEnv.GOOGLE_CLOUD_PROJECT } : {}),
104
+ ...(process.env.GEMINI_API_KEY ? { GEMINI_API_KEY: process.env.GEMINI_API_KEY } : {}),
105
+ ...(process.env.UNSPLASH_ACCESS_KEY ? { UNSPLASH_ACCESS_KEY: process.env.UNSPLASH_ACCESS_KEY } : {}),
106
+ ...existingFigmaEnv,
107
+ FIGMA_ACCESS_TOKEN: figmaToken,
108
+ FIGMA_BRIDGE_PORT: port,
109
+ ENABLE_DECISION_LOG: "true",
110
+ },
111
+ },
112
+ },
113
+ };
114
+ mkdirSync(tmpdir(), { recursive: true });
115
+ writeFileSync(MCP_CONFIG_PATH, JSON.stringify(config, null, 2));
116
+ console.log(`[chat-runner] MCP config written with FIGMA_BRIDGE_PORT=${port}`);
117
+ }
118
+
119
+ // Write initial config (will be rewritten with actual port once relay starts)
120
+ writeMcpConfig();
121
+
122
+ function getCleanEnv() {
123
+ const env = { ...process.env };
124
+ for (const key of Object.keys(env)) {
125
+ if (key === "CLAUDECODE" || key.startsWith("CLAUDE_CODE")) {
126
+ delete env[key];
127
+ }
128
+ }
129
+ delete env.ANTHROPIC_AUTH_TOKEN;
130
+ return env;
131
+ }
132
+
133
+ /**
134
+ * Process file attachments from the plugin chat UI.
135
+ * Images are written to temp files and referenced inline.
136
+ * Text/code files are appended inline to the message.
137
+ */
138
+ function processAttachments(attachments) {
139
+ const tempFiles = [];
140
+ let extraText = "";
141
+
142
+ for (const att of (attachments || [])) {
143
+ if (!att || !att.data) continue;
144
+
145
+ if (att.isImage) {
146
+ const match = att.data.match(/^data:([^;]+);base64,(.+)$/);
147
+ if (match) {
148
+ const mime = match[1];
149
+ const b64 = match[2];
150
+ const ext = mime.split("/")[1]?.split("+")[0] || "png";
151
+ const tmpPath = join(
152
+ tmpdir(),
153
+ `figma-chat-img-${Date.now()}-${Math.random().toString(36).slice(2)}.${ext}`
154
+ );
155
+ writeFileSync(tmpPath, Buffer.from(b64, "base64"));
156
+ tempFiles.push(tmpPath);
157
+ const label = att.name ? att.name : `image.${ext}`;
158
+ extraText += `\n\n[Image attached: ${label}]\nFile path: ${tmpPath}\nPlease read and analyze this image file to understand the design.`;
159
+ }
160
+ } else {
161
+ const label = att.name ? `\n\n--- Attached: ${att.name} ---\n` : "\n\n--- Attached file ---\n";
162
+ extraText += label + att.data;
163
+ }
164
+ }
165
+
166
+ return { imageArgs: [], extraText, tempFiles };
167
+ }
168
+
169
+ /**
170
+ * Spawn a claude CLI subprocess for a single chat message.
171
+ * Uses session persistence: first message creates session, subsequent messages resume it.
172
+ * Streams events back via onEvent callback.
173
+ * Returns the ChildProcess so the caller can kill it for abort.
174
+ */
175
+ const CLAUDE_DEFAULT_MODEL = "claude-opus-4-6";
176
+ const CLAUDE_VALID_MODELS = new Set(["claude-opus-4-6", "claude-sonnet-4-6", "claude-haiku-4-5-20251001"]);
177
+
178
+ function runClaude({ message, attachments, conversation, requestId, model, designSystemId, mode, frameworkConfig, onEvent }) {
179
+ const { imageArgs, extraText, tempFiles } = processAttachments(attachments);
180
+
181
+ const resolvedModel = CLAUDE_VALID_MODELS.has(model) ? model : CLAUDE_DEFAULT_MODEL;
182
+ const sessionMode = mode || "code";
183
+
184
+ const rawText = (message || "").trim() || (extraText ? "Please analyse the attached image(s) and help me create a Figma design based on them." : "");
185
+ const userText = rawText; // No more expandShortPrompt — Claude handles short prompts natively
186
+
187
+ // Detect active skills early — needed for session reset decision
188
+ const skills = (sessionMode === "code" || sessionMode === "dual") ? detectActiveSkills(userText) : [];
189
+
190
+ // Force new session when Component Doc Generator skill is detected.
191
+ // This ensures the system prompt includes the spec-type reference and
192
+ // tool restriction instructions, which won't be present on resumed sessions.
193
+ const hasDocGenSkill = skills.some(s => s.startsWith("Component Doc Generator:") && s !== "Component Doc Generator:all");
194
+ if (hasDocGenSkill && activeSessionIds[sessionMode]) {
195
+ console.log(`[chat-runner] Component Doc Generator detected — resetting ${sessionMode} session for fresh system prompt`);
196
+ activeSessionIds[sessionMode] = null;
197
+ }
198
+
199
+ const isFirstMessage = !activeSessionIds[sessionMode];
200
+
201
+ // Generate session ID on first message
202
+ if (isFirstMessage) {
203
+ activeSessionIds[sessionMode] = crypto.randomUUID();
204
+ console.log(`[chat-runner] New ${sessionMode} session: ${activeSessionIds[sessionMode]}`);
205
+ }
206
+
207
+ const currentSessionId = activeSessionIds[sessionMode];
208
+
209
+ // Build user message — no more conversation history, task guidance, or AGENTS.md injection
210
+ // Session persistence handles conversation context natively
211
+ const userMessage = `${userText}${extraText}`;
212
+
213
+ // Emit pre-flight progress
214
+ if (sessionMode === "code" || sessionMode === "dual") {
215
+ if (skills.length > 0) {
216
+ onEvent({ type: "phase_start", id: requestId, phase: `Skills: ${skills.join(" · ")}` });
217
+ }
218
+ const modeLabel = sessionMode === "dual" ? "Dual (Design + Code)" : "Code";
219
+ onEvent({ type: "phase_start", id: requestId, phase: `${modeLabel} · ${resolvedModel} · MCP: figma-intelligence-layer` });
220
+ } else {
221
+ onEvent({ type: "phase_start", id: requestId, phase: `Chat · ${resolvedModel}` });
222
+ }
223
+
224
+ let args;
225
+ if (isFirstMessage) {
226
+ // First message: create session with full config
227
+ let baseSystemPrompt;
228
+ if (sessionMode === "chat") {
229
+ baseSystemPrompt = buildChatPrompt();
230
+ } else if (sessionMode === "dual") {
231
+ baseSystemPrompt = buildDualOutputPrompt(designSystemId, frameworkConfig);
232
+ } else {
233
+ baseSystemPrompt = buildSystemPrompt(designSystemId);
234
+ }
235
+ const fullSystemPrompt = (sessionMode === "code" || sessionMode === "dual") ? baseSystemPrompt + buildSkillAddendum(skills) : baseSystemPrompt;
236
+ args = [
237
+ "--model", resolvedModel,
238
+ "--system-prompt", fullSystemPrompt,
239
+ "--print", userMessage,
240
+ "--output-format", "stream-json",
241
+ "--verbose",
242
+ "--dangerously-skip-permissions",
243
+ "--session-id", currentSessionId,
244
+ ];
245
+ // In code mode, load ONLY the figma-intelligence-layer MCP server.
246
+ // --strict-mcp-config ensures ONLY servers from --mcp-config are used,
247
+ // ignoring Pencil, design-bridge, and any other MCP servers from
248
+ // ~/.claude/settings.json or .vscode/mcp.json.
249
+ if (sessionMode === "code" || sessionMode === "dual") {
250
+ args.push("--mcp-config", MCP_CONFIG_PATH, "--strict-mcp-config");
251
+ }
252
+ } else {
253
+ // Subsequent messages: resume session — context already loaded.
254
+ // MUST re-pass --mcp-config and --strict-mcp-config on resume too,
255
+ // otherwise Claude re-discovers Pencil and other MCP servers.
256
+ args = [
257
+ "--print", userMessage,
258
+ "--output-format", "stream-json",
259
+ "--verbose",
260
+ "--dangerously-skip-permissions",
261
+ "--resume", currentSessionId,
262
+ ];
263
+ if (sessionMode === "code" || sessionMode === "dual") {
264
+ args.push("--mcp-config", MCP_CONFIG_PATH, "--strict-mcp-config");
265
+ }
266
+ console.log(`[chat-runner] Resuming ${sessionMode} session: ${currentSessionId}`);
267
+ }
268
+
269
+ const proc = spawn(CLAUDE_BIN, args, {
270
+ stdio: ["ignore", "pipe", "pipe"],
271
+ env: getCleanEnv(),
272
+ cwd: REPO_DIR,
273
+ });
274
+
275
+ // Clean up temp image files once the process exits
276
+ proc.on("close", () => {
277
+ for (const f of tempFiles) {
278
+ try { unlinkSync(f); } catch {}
279
+ }
280
+ });
281
+
282
+ let fullText = "";
283
+ let buffer = "";
284
+ let stderrOutput = "";
285
+
286
+ const eventState = {
287
+ pendingToolsByIndex: new Map(),
288
+ toolCallIdToName: new Map(),
289
+ };
290
+
291
+ proc.stdout.on("data", (chunk) => {
292
+ buffer += chunk.toString();
293
+ const lines = buffer.split("\n");
294
+ buffer = lines.pop() || "";
295
+
296
+ for (const line of lines) {
297
+ if (!line.trim()) continue;
298
+ try {
299
+ const parsed = JSON.parse(line);
300
+ handleStreamEvent(parsed, requestId, onEvent, (t) => { fullText += t; }, eventState);
301
+ } catch {}
302
+ }
303
+ });
304
+
305
+ proc.stderr.on("data", (chunk) => {
306
+ const text = chunk.toString().trim();
307
+ stderrOutput += text + "\n";
308
+ console.error("[claude chat]", text);
309
+ });
310
+
311
+ proc.on("close", (code) => {
312
+ if (buffer.trim()) {
313
+ try {
314
+ const parsed = JSON.parse(buffer);
315
+ handleStreamEvent(parsed, requestId, onEvent, (t) => { fullText += t; }, eventState);
316
+ } catch {}
317
+ }
318
+ for (const [, { name }] of eventState.pendingToolsByIndex) {
319
+ onEvent({ type: "tool_done", id: requestId, tool: name, isError: false });
320
+ }
321
+ eventState.pendingToolsByIndex.clear();
322
+ for (const [, name] of eventState.toolCallIdToName) {
323
+ onEvent({ type: "tool_done", id: requestId, tool: name, isError: false });
324
+ }
325
+ eventState.toolCallIdToName.clear();
326
+
327
+ if (code !== 0 && code !== null && fullText === "") {
328
+ const detail = stderrOutput.trim() || `exit code ${code}`;
329
+ // If session resume failed, reset and let next message start fresh
330
+ if (detail.includes("session") || detail.includes("resume")) {
331
+ console.error(`[chat-runner] Session resume failed, resetting ${sessionMode} session.`);
332
+ resetSession(sessionMode);
333
+ }
334
+ onEvent({
335
+ type: "error",
336
+ id: requestId,
337
+ error: `Claude error: ${detail}`,
338
+ });
339
+ }
340
+ onEvent({ type: "done", id: requestId, fullText });
341
+ });
342
+
343
+ proc.on("error", (err) => {
344
+ onEvent({ type: "error", id: requestId, error: `Failed to start claude: ${err.message}` });
345
+ onEvent({ type: "done", id: requestId, fullText: "" });
346
+ });
347
+
348
+ return proc;
349
+ }
350
+
351
+ /**
352
+ * Parse a single JSONL event from Claude CLI stream-json output.
353
+ */
354
+ function handleStreamEvent(parsed, requestId, onEvent, appendText, eventState) {
355
+ const { pendingToolsByIndex, toolCallIdToName } = eventState;
356
+
357
+ switch (parsed.type) {
358
+ case "assistant": {
359
+ if (parsed.message?.content) {
360
+ for (const block of parsed.message.content) {
361
+ if (block.type === "text" && block.text) {
362
+ for (const [, name] of toolCallIdToName) {
363
+ onEvent({ type: "tool_done", id: requestId, tool: name, isError: false });
364
+ }
365
+ toolCallIdToName.clear();
366
+ appendText(block.text);
367
+ onEvent({ type: "text_delta", id: requestId, delta: block.text });
368
+ } else if (block.type === "tool_use" && block.name) {
369
+ if (block.id) toolCallIdToName.set(block.id, block.name);
370
+ onEvent({ type: "tool_start", id: requestId, tool: block.name });
371
+ }
372
+ }
373
+ }
374
+ break;
375
+ }
376
+
377
+ case "content_block_start": {
378
+ const block = parsed.content_block;
379
+ if (!block) break;
380
+ if (block.type === "tool_use" && block.name) {
381
+ pendingToolsByIndex.set(parsed.index, { id: block.id || null, name: block.name });
382
+ if (block.id) toolCallIdToName.set(block.id, block.name);
383
+ onEvent({ type: "tool_start", id: requestId, tool: block.name });
384
+ }
385
+ break;
386
+ }
387
+
388
+ case "content_block_delta": {
389
+ if (parsed.delta?.type === "text_delta" && parsed.delta.text) {
390
+ if (pendingToolsByIndex.size > 0) {
391
+ for (const [, { name }] of pendingToolsByIndex) {
392
+ onEvent({ type: "tool_done", id: requestId, tool: name, isError: false });
393
+ }
394
+ pendingToolsByIndex.clear();
395
+ }
396
+ appendText(parsed.delta.text);
397
+ onEvent({ type: "text_delta", id: requestId, delta: parsed.delta.text });
398
+ }
399
+ break;
400
+ }
401
+
402
+ case "user": {
403
+ if (parsed.message?.content) {
404
+ for (const block of parsed.message.content) {
405
+ if (block.type === "tool_result") {
406
+ const name = toolCallIdToName.get(block.tool_use_id) || block.tool_use_id || "tool";
407
+ onEvent({ type: "tool_done", id: requestId, tool: name, isError: !!block.is_error });
408
+ if (block.tool_use_id) toolCallIdToName.delete(block.tool_use_id);
409
+ }
410
+ }
411
+ }
412
+ break;
413
+ }
414
+
415
+ case "tool_use": {
416
+ if (parsed.name) {
417
+ if (parsed.id) toolCallIdToName.set(parsed.id, parsed.name);
418
+ onEvent({ type: "tool_start", id: requestId, tool: parsed.name });
419
+ }
420
+ break;
421
+ }
422
+
423
+ case "tool_result": {
424
+ const toolName = toolCallIdToName.get(parsed.tool_use_id) || parsed.tool || parsed.name || "tool";
425
+ onEvent({ type: "tool_done", id: requestId, tool: toolName, isError: !!parsed.is_error });
426
+ if (parsed.tool_use_id) toolCallIdToName.delete(parsed.tool_use_id);
427
+ break;
428
+ }
429
+ }
430
+ }
431
+
432
+ /** Check if the claude CLI binary is available */
433
+ function isClaudeAvailable() {
434
+ return new Promise((resolve) => {
435
+ const proc = spawn(CLAUDE_BIN, ["--version"], { stdio: "pipe", env: getCleanEnv() });
436
+ proc.on("close", (code) => resolve(code === 0));
437
+ proc.on("error", () => resolve(false));
438
+ });
439
+ }
440
+
441
+ /** Returns { loggedIn: boolean, email: string|null } */
442
+ function getClaudeAuthInfo() {
443
+ return new Promise((resolve) => {
444
+ const proc = spawn(CLAUDE_BIN, ["auth", "status"], { stdio: "pipe", env: getCleanEnv() });
445
+ let output = "";
446
+ const timer = setTimeout(() => { proc.kill(); resolve({ loggedIn: false, email: null }); }, 5000);
447
+ proc.stdout?.on("data", (d) => { output += d.toString(); });
448
+ proc.stderr?.on("data", (d) => { output += d.toString(); });
449
+ proc.on("close", (code) => {
450
+ clearTimeout(timer);
451
+ const loggedIn = code === 0 || output.toLowerCase().includes("logged in");
452
+ const emailMatch = output.match(/[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}/);
453
+ resolve({ loggedIn, email: emailMatch ? emailMatch[0] : null });
454
+ });
455
+ proc.on("error", () => { clearTimeout(timer); resolve({ loggedIn: false, email: null }); });
456
+ });
457
+ }
458
+
459
+ module.exports = { runClaude, resetSession, isClaudeAvailable, getClaudeAuthInfo, writeMcpConfig };