@sarjallab09/figma-intelligence 1.1.0 → 1.2.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.
- package/README.md +67 -36
- package/dist/bin/cli.js +2 -0
- package/dist/design-bridge/bridge.js +2 -0
- package/dist/figma-bridge-plugin/bridge-relay.js +2 -0
- package/dist/figma-bridge-plugin/code.js +1 -0
- package/{figma-bridge-plugin → dist/figma-bridge-plugin}/package-lock.json +0 -3
- package/dist/figma-bridge-plugin/ui.html +4970 -0
- package/dist/figma-intelligence-layer/dist/index.js +2 -0
- package/dist/scripts/clean-existing-chunks.js +2 -0
- package/dist/scripts/connect-ai-tool.js +2 -0
- package/dist/scripts/convert-hub-pdfs.js +2 -0
- package/dist/scripts/figma-mcp-status.js +2 -0
- package/dist/scripts/register-codex-mcp.js +2 -0
- package/dist/scripts/test-copilot-chat.js +2 -0
- package/package.json +11 -8
- package/bin/cli.js +0 -859
- package/design-bridge/bridge.js +0 -196
- package/design-bridge/lib/assets.js +0 -367
- package/design-bridge/lib/prompt.js +0 -85
- package/design-bridge/lib/server.js +0 -66
- package/design-bridge/lib/stitch.js +0 -37
- package/design-bridge/lib/tokens.js +0 -82
- package/design-bridge/package-lock.json +0 -579
- package/figma-bridge-plugin/README.md +0 -97
- package/figma-bridge-plugin/anthropic-chat-runner.js +0 -192
- package/figma-bridge-plugin/bridge-relay.js +0 -2505
- package/figma-bridge-plugin/chat-runner.js +0 -485
- package/figma-bridge-plugin/code.js +0 -1534
- package/figma-bridge-plugin/codex-runner.js +0 -505
- package/figma-bridge-plugin/component-schemas.js +0 -110
- package/figma-bridge-plugin/content-context.js +0 -869
- package/figma-bridge-plugin/create-button.js +0 -216
- package/figma-bridge-plugin/gemini-cli-runner.js +0 -291
- package/figma-bridge-plugin/gemini-runner.js +0 -187
- package/figma-bridge-plugin/html-to-figma.js +0 -927
- package/figma-bridge-plugin/knowledge-hub/.gitkeep +0 -0
- package/figma-bridge-plugin/knowledge-hub/uspec-references/anatomy-spec.md +0 -159
- package/figma-bridge-plugin/knowledge-hub/uspec-references/api-spec.md +0 -162
- package/figma-bridge-plugin/knowledge-hub/uspec-references/color-spec.md +0 -148
- package/figma-bridge-plugin/knowledge-hub/uspec-references/full-spec-template.md +0 -314
- package/figma-bridge-plugin/knowledge-hub/uspec-references/property-spec.md +0 -175
- package/figma-bridge-plugin/knowledge-hub/uspec-references/screen-reader-spec.md +0 -180
- package/figma-bridge-plugin/knowledge-hub/uspec-references/structure-spec.md +0 -165
- package/figma-bridge-plugin/perplexity-runner.js +0 -188
- package/figma-bridge-plugin/references/SKILL.md +0 -178
- package/figma-bridge-plugin/references/anatomy-spec.md +0 -159
- package/figma-bridge-plugin/references/api-spec.md +0 -162
- package/figma-bridge-plugin/references/color-spec.md +0 -148
- package/figma-bridge-plugin/references/full-spec-template.md +0 -314
- package/figma-bridge-plugin/references/property-spec.md +0 -175
- package/figma-bridge-plugin/references/screen-reader-spec.md +0 -180
- package/figma-bridge-plugin/references/structure-spec.md +0 -165
- package/figma-bridge-plugin/shared-prompt-config.js +0 -645
- package/figma-bridge-plugin/spec-helpers/build-table.js +0 -269
- package/figma-bridge-plugin/spec-helpers/classify-elements.js +0 -189
- package/figma-bridge-plugin/spec-helpers/index.js +0 -35
- package/figma-bridge-plugin/spec-helpers/parse-figma-link.js +0 -49
- package/figma-bridge-plugin/spec-helpers/position-markers.js +0 -158
- package/figma-bridge-plugin/stitch-auth.js +0 -322
- package/figma-bridge-plugin/stitch-runner.js +0 -1427
- package/figma-bridge-plugin/token-resolver.js +0 -107
- package/figma-bridge-plugin/ui.html +0 -4542
- package/figma-intelligence-layer/.env.example +0 -39
- package/figma-intelligence-layer/docs/local-image-generation.md +0 -60
- package/figma-intelligence-layer/examples/comfyui-workflow-template.example.json +0 -101
- package/figma-intelligence-layer/jest.config.js +0 -14
- package/figma-intelligence-layer/mcp-config.json +0 -19
- package/figma-intelligence-layer/package-lock.json +0 -5892
- package/figma-intelligence-layer/scripts/setup-comfyui-local.sh +0 -67
- package/figma-intelligence-layer/scripts/start-comfyui.sh +0 -33
- package/figma-intelligence-layer/src/index.ts +0 -2233
- package/figma-intelligence-layer/src/shared/auto-layout-validator.ts +0 -404
- package/figma-intelligence-layer/src/shared/cache.ts +0 -187
- package/figma-intelligence-layer/src/shared/color-operations.ts +0 -533
- package/figma-intelligence-layer/src/shared/color-utils.ts +0 -138
- package/figma-intelligence-layer/src/shared/component-script-builder.ts +0 -413
- package/figma-intelligence-layer/src/shared/component-templates.ts +0 -2767
- package/figma-intelligence-layer/src/shared/concept-taxonomy.ts +0 -694
- package/figma-intelligence-layer/src/shared/decision-log.ts +0 -128
- package/figma-intelligence-layer/src/shared/design-system-context.ts +0 -568
- package/figma-intelligence-layer/src/shared/design-system-intelligence.ts +0 -131
- package/figma-intelligence-layer/src/shared/design-system-matcher.ts +0 -184
- package/figma-intelligence-layer/src/shared/design-system-normalizers.ts +0 -196
- package/figma-intelligence-layer/src/shared/design-system-tokens.ts +0 -295
- package/figma-intelligence-layer/src/shared/dtcg-validator.ts +0 -530
- package/figma-intelligence-layer/src/shared/enrichment-pipeline.ts +0 -671
- package/figma-intelligence-layer/src/shared/figma-bridge.ts +0 -1418
- package/figma-intelligence-layer/src/shared/font-config.ts +0 -126
- package/figma-intelligence-layer/src/shared/icon-catalog.ts +0 -360
- package/figma-intelligence-layer/src/shared/icon-fetch.ts +0 -80
- package/figma-intelligence-layer/src/shared/prototype-script-builder.ts +0 -162
- package/figma-intelligence-layer/src/shared/response-compression.ts +0 -440
- package/figma-intelligence-layer/src/shared/semantic-token-catalog.ts +0 -324
- package/figma-intelligence-layer/src/shared/token-binder.ts +0 -505
- package/figma-intelligence-layer/src/shared/token-math.ts +0 -427
- package/figma-intelligence-layer/src/shared/token-naming.ts +0 -468
- package/figma-intelligence-layer/src/shared/token-utils.ts +0 -420
- package/figma-intelligence-layer/src/shared/types.ts +0 -346
- package/figma-intelligence-layer/src/shared/typography-presets.ts +0 -94
- package/figma-intelligence-layer/src/shared/unsplash.ts +0 -165
- package/figma-intelligence-layer/src/shared/vision-client.ts +0 -607
- package/figma-intelligence-layer/src/shared/vision-provider-anthropic.ts +0 -334
- package/figma-intelligence-layer/src/shared/vision-provider-openai.ts +0 -446
- package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/a11y-annotate-handler.ts +0 -782
- package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/a11y-annotate-renderer.ts +0 -496
- package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/a11y-annotation-kit.ts +0 -230
- package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/colorblind-sim.ts +0 -66
- package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/index.ts +0 -810
- package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/keyboard-sr-order-analyzer.ts +0 -1191
- package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/keyboard-sr-order-figma-page.ts +0 -1346
- package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/keyboard-sr-order-handler.ts +0 -148
- package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/vpat-figma-page.ts +0 -499
- package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/vpat-report.ts +0 -910
- package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/wcag-checker.ts +0 -989
- package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/wcag-criteria.ts +0 -1160
- package/figma-intelligence-layer/src/tools/phase1-vision/design-from-ref/index.ts +0 -424
- package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/component-recognizer.ts +0 -38
- package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/ds-matcher.ts +0 -111
- package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/font-matcher.ts +0 -114
- package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/icon-resolver.ts +0 -103
- package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/index.ts +0 -1060
- package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/layout-segmenter.ts +0 -18
- package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/token-inferencer.ts +0 -39
- package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/vision-pipeline.ts +0 -58
- package/figma-intelligence-layer/src/tools/phase1-vision/sketch-to-design/index.ts +0 -298
- package/figma-intelligence-layer/src/tools/phase1-vision/visual-audit/index.ts +0 -197
- package/figma-intelligence-layer/src/tools/phase2-accuracy/component-audit/index.ts +0 -494
- package/figma-intelligence-layer/src/tools/phase2-accuracy/intent-translator/index.ts +0 -356
- package/figma-intelligence-layer/src/tools/phase2-accuracy/layout-intelligence/container-patterns.ts +0 -123
- package/figma-intelligence-layer/src/tools/phase2-accuracy/layout-intelligence/index.ts +0 -663
- package/figma-intelligence-layer/src/tools/phase2-accuracy/lint-rules/built-in-rules.yaml +0 -56
- package/figma-intelligence-layer/src/tools/phase2-accuracy/lint-rules/index.ts +0 -614
- package/figma-intelligence-layer/src/tools/phase2-accuracy/lint-rules/rule-engine.ts +0 -113
- package/figma-intelligence-layer/src/tools/phase2-accuracy/theme-generator/color-theory.ts +0 -178
- package/figma-intelligence-layer/src/tools/phase2-accuracy/theme-generator/index.ts +0 -470
- package/figma-intelligence-layer/src/tools/phase2-accuracy/variant-expander/index.ts +0 -429
- package/figma-intelligence-layer/src/tools/phase2-accuracy/variant-expander/token-override-maps.ts +0 -226
- package/figma-intelligence-layer/src/tools/phase3-generation/ai-image-insert/index.ts +0 -535
- package/figma-intelligence-layer/src/tools/phase3-generation/component-archaeologist/index.ts +0 -660
- package/figma-intelligence-layer/src/tools/phase3-generation/component-archaeologist/pattern-fingerprints.ts +0 -209
- package/figma-intelligence-layer/src/tools/phase3-generation/composition-builder/index.ts +0 -540
- package/figma-intelligence-layer/src/tools/phase3-generation/figma-animated-build.ts +0 -391
- package/figma-intelligence-layer/src/tools/phase3-generation/page-architect/index.ts +0 -2019
- package/figma-intelligence-layer/src/tools/phase3-generation/page-architect/screen-templates.ts +0 -131
- package/figma-intelligence-layer/src/tools/phase3-generation/prototype-map/index.ts +0 -381
- package/figma-intelligence-layer/src/tools/phase3-generation/prototype-wire/index.ts +0 -565
- package/figma-intelligence-layer/src/tools/phase3-generation/swarm-build/index.ts +0 -764
- package/figma-intelligence-layer/src/tools/phase3-generation/system-drift/index.ts +0 -535
- package/figma-intelligence-layer/src/tools/phase3-generation/unsplash-search/index.ts +0 -84
- package/figma-intelligence-layer/src/tools/phase3-generation/url-to-frame/index.ts +0 -401
- package/figma-intelligence-layer/src/tools/phase4-sync/animation-specifier/code-generators/css-animations.ts +0 -68
- package/figma-intelligence-layer/src/tools/phase4-sync/animation-specifier/code-generators/framer-motion.ts +0 -78
- package/figma-intelligence-layer/src/tools/phase4-sync/animation-specifier/code-generators/swift-animations.ts +0 -93
- package/figma-intelligence-layer/src/tools/phase4-sync/animation-specifier/index.ts +0 -596
- package/figma-intelligence-layer/src/tools/phase4-sync/ci-check/index.ts +0 -462
- package/figma-intelligence-layer/src/tools/phase4-sync/export-tokens/index.ts +0 -1470
- package/figma-intelligence-layer/src/tools/phase4-sync/generate-component-code/index.ts +0 -829
- package/figma-intelligence-layer/src/tools/phase4-sync/handoff-spec/index.ts +0 -702
- package/figma-intelligence-layer/src/tools/phase4-sync/icon-library-sync/index.ts +0 -483
- package/figma-intelligence-layer/src/tools/phase4-sync/sync-from-code/index.ts +0 -501
- package/figma-intelligence-layer/src/tools/phase4-sync/sync-from-code/storybook-parser.ts +0 -106
- package/figma-intelligence-layer/src/tools/phase4-sync/watch-docs/index.ts +0 -676
- package/figma-intelligence-layer/src/tools/phase4-sync/webhook-listener/index.ts +0 -560
- package/figma-intelligence-layer/src/tools/phase5-governance/apg-doc/index.ts +0 -1043
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/component-detection.ts +0 -620
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/anatomy.ts +0 -331
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/color-tokens.ts +0 -77
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/properties.ts +0 -54
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/snapshot.ts +0 -287
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/spacing.ts +0 -71
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/states.ts +0 -43
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/typography.ts +0 -71
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/index.ts +0 -221
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/_default.ts +0 -166
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/accordion.ts +0 -232
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/alert.ts +0 -234
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/avatar-group.ts +0 -270
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/avatar.ts +0 -249
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/badge.ts +0 -231
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/banner.ts +0 -293
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/breadcrumb.ts +0 -240
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/button.ts +0 -243
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/calendar.ts +0 -307
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/card.ts +0 -143
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/checkbox.ts +0 -227
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/chip.ts +0 -233
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/combobox.ts +0 -282
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/datepicker.ts +0 -276
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/divider.ts +0 -223
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/drawer.ts +0 -255
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/dropdown-menu.ts +0 -289
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/empty-state.ts +0 -261
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/file-uploader.ts +0 -290
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/form.ts +0 -265
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/grid.ts +0 -238
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/icon.ts +0 -255
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/index.ts +0 -128
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/inline-edit.ts +0 -286
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/inline-message.ts +0 -255
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/input.ts +0 -330
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/link.ts +0 -247
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/list.ts +0 -250
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/menu.ts +0 -247
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/modal.ts +0 -144
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/navbar.ts +0 -264
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/navigation.ts +0 -251
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/number-input.ts +0 -261
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/pagination.ts +0 -248
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/popover.ts +0 -270
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/progress.ts +0 -251
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/radio.ts +0 -142
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/range-slider.ts +0 -282
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/rating.ts +0 -250
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/search.ts +0 -258
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/segmented-control.ts +0 -265
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/select.ts +0 -319
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/skeleton.ts +0 -256
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/slider.ts +0 -232
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/spinner.ts +0 -239
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/status-dot.ts +0 -252
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/stepper.ts +0 -270
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/table.ts +0 -244
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/tabs.ts +0 -143
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/tag.ts +0 -243
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/textarea.ts +0 -259
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/time-picker.ts +0 -293
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/toast.ts +0 -144
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/toggle.ts +0 -289
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/toolbar.ts +0 -267
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/tooltip.ts +0 -232
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/treeview.ts +0 -257
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/typography.ts +0 -319
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/legacy-compat.ts +0 -121
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/renderers/anatomy-diagram.ts +0 -430
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/renderers/figma-page.ts +0 -312
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/renderers/json.ts +0 -129
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/renderers/markdown.ts +0 -78
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/renderers/visual-doc.ts +0 -2333
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/accessibility.ts +0 -100
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/anatomy.ts +0 -32
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/color-tokens.ts +0 -59
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/content-guidance.ts +0 -18
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/design-tokens.ts +0 -53
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/interaction-rules.ts +0 -19
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/overview.ts +0 -91
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/properties-api.ts +0 -71
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/qa-criteria.ts +0 -19
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/related-components.ts +0 -110
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/responsive.ts +0 -19
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/size-specs.ts +0 -67
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/spacing-structure.ts +0 -58
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/state-specs.ts +0 -79
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/states.ts +0 -50
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/type-hierarchy.ts +0 -33
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/typography.ts +0 -55
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/usage-guidelines.ts +0 -73
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/variants.ts +0 -81
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/types.ts +0 -409
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec-sheet/index.ts +0 -198
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec-sheet/renderer.ts +0 -701
- package/figma-intelligence-layer/src/tools/phase5-governance/component-spec-sheet/types.ts +0 -88
- package/figma-intelligence-layer/src/tools/phase5-governance/decision-log/index.ts +0 -135
- package/figma-intelligence-layer/src/tools/phase5-governance/design-decision-log/index.ts +0 -491
- package/figma-intelligence-layer/src/tools/phase5-governance/ds-primitives/index.ts +0 -416
- package/figma-intelligence-layer/src/tools/phase5-governance/ds-scaffolder/index.ts +0 -722
- package/figma-intelligence-layer/src/tools/phase5-governance/ds-variables/index.ts +0 -449
- package/figma-intelligence-layer/src/tools/phase5-governance/health-report/index.ts +0 -393
- package/figma-intelligence-layer/src/tools/phase5-governance/taxonomy-docs/index.ts +0 -406
- package/figma-intelligence-layer/src/tools/phase5-governance/taxonomy-docs/renderers/figma-page.ts +0 -292
- package/figma-intelligence-layer/src/tools/phase5-governance/taxonomy-docs/renderers/json.ts +0 -24
- package/figma-intelligence-layer/src/tools/phase5-governance/taxonomy-docs/renderers/markdown.ts +0 -172
- package/figma-intelligence-layer/src/tools/phase5-governance/taxonomy-docs/renderers/naming-guide.ts +0 -409
- package/figma-intelligence-layer/src/tools/phase5-governance/token-analytics/index.ts +0 -594
- package/figma-intelligence-layer/src/tools/phase5-governance/token-docs/index.ts +0 -710
- package/figma-intelligence-layer/src/tools/phase5-governance/token-migrate/index.ts +0 -458
- package/figma-intelligence-layer/src/tools/phase5-governance/token-naming/index.ts +0 -134
- package/figma-intelligence-layer/tests/apg-doc.test.ts +0 -101
- package/figma-intelligence-layer/tests/design-system-context.test.ts +0 -152
- package/figma-intelligence-layer/tests/design-system-matcher.test.ts +0 -144
- package/figma-intelligence-layer/tests/figma-bridge.test.ts +0 -83
- package/figma-intelligence-layer/tests/generate-image-and-insert.test.ts +0 -56
- package/figma-intelligence-layer/tests/screen-cloner-regression.test.ts +0 -69
- package/figma-intelligence-layer/tests/smoke.test.ts +0 -174
- package/figma-intelligence-layer/tests/spec-generator.test.ts +0 -127
- package/figma-intelligence-layer/tests/token-migrate.test.ts +0 -21
- package/figma-intelligence-layer/tests/token-naming.test.ts +0 -30
- package/figma-intelligence-layer/tsconfig.json +0 -19
- package/scripts/clean-existing-chunks.js +0 -179
- package/scripts/connect-ai-tool.js +0 -490
- package/scripts/convert-hub-pdfs.js +0 -425
- package/scripts/figma-mcp-status.js +0 -349
- package/scripts/register-codex-mcp.js +0 -96
- /package/{design-bridge → dist/design-bridge}/.env.example +0 -0
- /package/{design-bridge → dist/design-bridge}/package.json +0 -0
- /package/{figma-bridge-plugin → dist/figma-bridge-plugin}/manifest.json +0 -0
- /package/{figma-bridge-plugin → dist/figma-bridge-plugin}/package.json +0 -0
- /package/{figma-intelligence-layer → dist/figma-intelligence-layer}/package.json +0 -0
|
@@ -1,216 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
const WebSocket = require("ws");
|
|
3
|
-
|
|
4
|
-
const code = `
|
|
5
|
-
// ── Create Button Component with all variant properties ──
|
|
6
|
-
|
|
7
|
-
var page = figma.currentPage;
|
|
8
|
-
|
|
9
|
-
// Helper to create a single button variant
|
|
10
|
-
async function createButtonVariant(opts) {
|
|
11
|
-
var frame = figma.createFrame();
|
|
12
|
-
frame.name = "Size=" + opts.size + ", Style=" + opts.style + ", State=" + opts.state + ", IconPosition=" + opts.iconPos;
|
|
13
|
-
frame.layoutMode = "HORIZONTAL";
|
|
14
|
-
frame.primaryAxisSizingMode = "AUTO";
|
|
15
|
-
frame.counterAxisSizingMode = "AUTO";
|
|
16
|
-
frame.primaryAxisAlignItems = "CENTER";
|
|
17
|
-
frame.counterAxisAlignItems = "CENTER";
|
|
18
|
-
frame.itemSpacing = 8;
|
|
19
|
-
frame.cornerRadius = opts.radius;
|
|
20
|
-
|
|
21
|
-
// Padding by size
|
|
22
|
-
var padH = opts.size === "sm" ? 12 : opts.size === "md" ? 16 : 20;
|
|
23
|
-
var padV = opts.size === "sm" ? 6 : opts.size === "md" ? 10 : 14;
|
|
24
|
-
frame.paddingLeft = padH;
|
|
25
|
-
frame.paddingRight = padH;
|
|
26
|
-
frame.paddingTop = padV;
|
|
27
|
-
frame.paddingBottom = padV;
|
|
28
|
-
|
|
29
|
-
// Colors by style & state
|
|
30
|
-
var bgColor, textColor, borderColor, bgOpacity;
|
|
31
|
-
bgOpacity = 1;
|
|
32
|
-
|
|
33
|
-
if (opts.style === "Primary") {
|
|
34
|
-
bgColor = { r: 0.15, g: 0.39, b: 0.92 }; // #2563EB
|
|
35
|
-
textColor = { r: 1, g: 1, b: 1 };
|
|
36
|
-
borderColor = null;
|
|
37
|
-
} else if (opts.style === "Secondary") {
|
|
38
|
-
bgColor = { r: 0.93, g: 0.95, b: 1 }; // #EDF0FF
|
|
39
|
-
textColor = { r: 0.15, g: 0.39, b: 0.92 };
|
|
40
|
-
borderColor = null;
|
|
41
|
-
} else if (opts.style === "Outline") {
|
|
42
|
-
bgColor = { r: 1, g: 1, b: 1 };
|
|
43
|
-
textColor = { r: 0.15, g: 0.39, b: 0.92 };
|
|
44
|
-
borderColor = { r: 0.15, g: 0.39, b: 0.92 };
|
|
45
|
-
} else if (opts.style === "Ghost") {
|
|
46
|
-
bgColor = { r: 1, g: 1, b: 1 };
|
|
47
|
-
textColor = { r: 0.15, g: 0.39, b: 0.92 };
|
|
48
|
-
bgOpacity = 0;
|
|
49
|
-
borderColor = null;
|
|
50
|
-
} else if (opts.style === "Destructive") {
|
|
51
|
-
bgColor = { r: 0.86, g: 0.15, b: 0.15 }; // #DC2626
|
|
52
|
-
textColor = { r: 1, g: 1, b: 1 };
|
|
53
|
-
borderColor = null;
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
// State modifications
|
|
57
|
-
if (opts.state === "Hover") {
|
|
58
|
-
bgColor = { r: bgColor.r * 0.85, g: bgColor.g * 0.85, b: bgColor.b * 0.85 };
|
|
59
|
-
} else if (opts.state === "Pressed") {
|
|
60
|
-
bgColor = { r: bgColor.r * 0.7, g: bgColor.g * 0.7, b: bgColor.b * 0.7 };
|
|
61
|
-
} else if (opts.state === "Disabled") {
|
|
62
|
-
bgOpacity = 0.4;
|
|
63
|
-
} else if (opts.state === "Loading") {
|
|
64
|
-
bgOpacity = 0.7;
|
|
65
|
-
} else if (opts.state === "Focused") {
|
|
66
|
-
// Add focus ring effect later
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
frame.fills = [{ type: "SOLID", color: bgColor, opacity: bgOpacity }];
|
|
70
|
-
|
|
71
|
-
if (borderColor) {
|
|
72
|
-
frame.strokes = [{ type: "SOLID", color: borderColor }];
|
|
73
|
-
frame.strokeWeight = 1.5;
|
|
74
|
-
frame.strokeAlign = "INSIDE";
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
// Icon size by button size
|
|
78
|
-
var iconSize = opts.size === "sm" ? 16 : opts.size === "md" ? 20 : 24;
|
|
79
|
-
|
|
80
|
-
// Leading icon
|
|
81
|
-
if (opts.iconPos === "Left" || opts.iconPos === "Both") {
|
|
82
|
-
var leadIcon = figma.createRectangle();
|
|
83
|
-
leadIcon.name = "Icon-Left";
|
|
84
|
-
leadIcon.resize(iconSize, iconSize);
|
|
85
|
-
leadIcon.fills = [{ type: "SOLID", color: textColor }];
|
|
86
|
-
leadIcon.cornerRadius = 3;
|
|
87
|
-
leadIcon.opacity = 0.85;
|
|
88
|
-
frame.appendChild(leadIcon);
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
// Label
|
|
92
|
-
var fontSize = opts.size === "sm" ? 12 : opts.size === "md" ? 14 : 16;
|
|
93
|
-
var fontStyle = opts.size === "lg" ? "Bold" : "Semi Bold";
|
|
94
|
-
|
|
95
|
-
await figma.loadFontAsync({ family: "Inter", style: fontStyle });
|
|
96
|
-
var label = figma.createText();
|
|
97
|
-
label.name = "Label";
|
|
98
|
-
label.fontName = { family: "Inter", style: fontStyle };
|
|
99
|
-
label.characters = "Button";
|
|
100
|
-
label.fontSize = fontSize;
|
|
101
|
-
label.fills = [{ type: "SOLID", color: textColor }];
|
|
102
|
-
frame.appendChild(label);
|
|
103
|
-
|
|
104
|
-
// Loading spinner (just an indicator rect for Loading state)
|
|
105
|
-
if (opts.state === "Loading") {
|
|
106
|
-
label.opacity = 0;
|
|
107
|
-
var spinner = figma.createEllipse();
|
|
108
|
-
spinner.name = "Spinner";
|
|
109
|
-
spinner.resize(iconSize, iconSize);
|
|
110
|
-
spinner.fills = [];
|
|
111
|
-
spinner.strokes = [{ type: "SOLID", color: textColor }];
|
|
112
|
-
spinner.strokeWeight = 2;
|
|
113
|
-
spinner.dashPattern = [6, 4];
|
|
114
|
-
frame.insertChild(frame.children.length, spinner);
|
|
115
|
-
}
|
|
116
|
-
|
|
117
|
-
// Trailing icon
|
|
118
|
-
if (opts.iconPos === "Right" || opts.iconPos === "Both") {
|
|
119
|
-
var trailIcon = figma.createRectangle();
|
|
120
|
-
trailIcon.name = "Icon-Right";
|
|
121
|
-
trailIcon.resize(iconSize, iconSize);
|
|
122
|
-
trailIcon.fills = [{ type: "SOLID", color: textColor }];
|
|
123
|
-
trailIcon.cornerRadius = 3;
|
|
124
|
-
trailIcon.opacity = 0.85;
|
|
125
|
-
frame.appendChild(trailIcon);
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
return frame;
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
// ── Generate all variant combos ──
|
|
132
|
-
var sizes = ["sm", "md", "lg"];
|
|
133
|
-
var styles = ["Primary", "Secondary", "Outline", "Ghost", "Destructive"];
|
|
134
|
-
var states = ["Default", "Hover", "Pressed", "Focused", "Disabled", "Loading"];
|
|
135
|
-
var iconPositions = ["None", "Left", "Right", "Both"];
|
|
136
|
-
|
|
137
|
-
var radiusMap = { sm: 6, md: 8, lg: 10 };
|
|
138
|
-
var allVariants = [];
|
|
139
|
-
|
|
140
|
-
for (var si = 0; si < sizes.length; si++) {
|
|
141
|
-
for (var sti = 0; sti < styles.length; sti++) {
|
|
142
|
-
for (var sta = 0; sta < states.length; sta++) {
|
|
143
|
-
for (var ip = 0; ip < iconPositions.length; ip++) {
|
|
144
|
-
allVariants.push({
|
|
145
|
-
size: sizes[si],
|
|
146
|
-
style: styles[sti],
|
|
147
|
-
state: states[sta],
|
|
148
|
-
iconPos: iconPositions[ip],
|
|
149
|
-
radius: radiusMap[sizes[si]],
|
|
150
|
-
});
|
|
151
|
-
}
|
|
152
|
-
}
|
|
153
|
-
}
|
|
154
|
-
}
|
|
155
|
-
|
|
156
|
-
// Create variants in batches to avoid timeout
|
|
157
|
-
var components = [];
|
|
158
|
-
for (var i = 0; i < allVariants.length; i++) {
|
|
159
|
-
var v = allVariants[i];
|
|
160
|
-
var frame = await createButtonVariant(v);
|
|
161
|
-
page.appendChild(frame);
|
|
162
|
-
var comp = figma.createComponentFromNode(frame);
|
|
163
|
-
components.push(comp);
|
|
164
|
-
}
|
|
165
|
-
|
|
166
|
-
// Combine into a component set
|
|
167
|
-
if (components.length > 1) {
|
|
168
|
-
var set = figma.combineAsVariants(components, page);
|
|
169
|
-
set.name = "Button";
|
|
170
|
-
set.description = "Button component with Size, Style, State, and IconPosition variant properties.\\n\\nSizes: sm, md, lg\\nStyles: Primary, Secondary, Outline, Ghost, Destructive\\nStates: Default, Hover, Pressed, Focused, Disabled, Loading\\nIcon Position: None, Left, Right, Both";
|
|
171
|
-
set.layoutMode = "VERTICAL";
|
|
172
|
-
set.primaryAxisSizingMode = "AUTO";
|
|
173
|
-
set.counterAxisSizingMode = "AUTO";
|
|
174
|
-
set.itemSpacing = 16;
|
|
175
|
-
set.paddingLeft = 24;
|
|
176
|
-
set.paddingRight = 24;
|
|
177
|
-
set.paddingTop = 24;
|
|
178
|
-
set.paddingBottom = 24;
|
|
179
|
-
set.x = 200;
|
|
180
|
-
set.y = 200;
|
|
181
|
-
figma.viewport.scrollAndZoomIntoView([set]);
|
|
182
|
-
|
|
183
|
-
return {
|
|
184
|
-
componentSetId: set.id,
|
|
185
|
-
name: set.name,
|
|
186
|
-
variantCount: components.length,
|
|
187
|
-
properties: {
|
|
188
|
-
Size: sizes,
|
|
189
|
-
Style: styles,
|
|
190
|
-
State: states,
|
|
191
|
-
IconPosition: iconPositions,
|
|
192
|
-
},
|
|
193
|
-
};
|
|
194
|
-
}
|
|
195
|
-
|
|
196
|
-
return { error: "Not enough variants created" };
|
|
197
|
-
`;
|
|
198
|
-
|
|
199
|
-
const ws = new WebSocket("ws://localhost:9001");
|
|
200
|
-
ws.on("open", () => {
|
|
201
|
-
console.log("Connected to relay, sending button creation command...");
|
|
202
|
-
console.log("Creating " + 3*5*6*4 + " variants (3 sizes x 5 styles x 6 states x 4 icon positions)...");
|
|
203
|
-
ws.send(JSON.stringify({ id: "btn-gen", method: "execute", params: { code } }));
|
|
204
|
-
});
|
|
205
|
-
ws.on("message", (data) => {
|
|
206
|
-
const resp = JSON.parse(data.toString());
|
|
207
|
-
if (resp.error) {
|
|
208
|
-
console.error("Error:", resp.error);
|
|
209
|
-
} else {
|
|
210
|
-
console.log("Success:", JSON.stringify(resp.result, null, 2));
|
|
211
|
-
}
|
|
212
|
-
ws.close();
|
|
213
|
-
process.exit(0);
|
|
214
|
-
});
|
|
215
|
-
ws.on("error", (e) => { console.error("WS Error:", e.message); process.exit(1); });
|
|
216
|
-
setTimeout(() => { console.log("Timeout - 360 variants may take a while. Check Figma."); process.exit(0); }, 120000);
|
|
@@ -1,291 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
/**
|
|
3
|
-
* gemini-cli-runner.js — Spawns a Google Gemini CLI subprocess for chat messages.
|
|
4
|
-
* Uses the user's existing Google One AI Premium / Gemini Advanced subscription.
|
|
5
|
-
* No API key required — authenticates via Google OAuth (browser flow).
|
|
6
|
-
*
|
|
7
|
-
* Install: npm install -g @google/gemini-cli
|
|
8
|
-
* Auth: gemini auth login (or just run `gemini` — it prompts on first use)
|
|
9
|
-
*
|
|
10
|
-
* Emits the same event shape as chat-runner.js so bridge-relay can use either.
|
|
11
|
-
*
|
|
12
|
-
* OPTIMIZED: Prompt bloat removed (expandShortPrompt, conversation history, skill files).
|
|
13
|
-
*/
|
|
14
|
-
|
|
15
|
-
const { spawn, spawnSync, execSync } = require("child_process");
|
|
16
|
-
const { readFileSync, writeFileSync, existsSync, mkdirSync } = require("fs");
|
|
17
|
-
const { homedir, platform } = require("os");
|
|
18
|
-
const { join, resolve } = require("path");
|
|
19
|
-
|
|
20
|
-
const GEMINI_BIN = process.env.GEMINI_BIN_PATH || "gemini";
|
|
21
|
-
const REPO_DIR = resolve(__dirname, "..");
|
|
22
|
-
const CLAUDE_SETTINGS_PATH = join(homedir(), ".claude", "settings.json");
|
|
23
|
-
const MCP_BUILD_PATH = join(REPO_DIR, "figma-intelligence-layer", "dist", "index.js");
|
|
24
|
-
const GEMINI_SETTINGS_DIR = join(homedir(), ".gemini");
|
|
25
|
-
const GEMINI_SETTINGS_PATH = join(GEMINI_SETTINGS_DIR, "settings.json");
|
|
26
|
-
|
|
27
|
-
const GEMINI_AUTH_CANDIDATES = [
|
|
28
|
-
join(homedir(), ".gemini", "oauth_creds.json"),
|
|
29
|
-
join(homedir(), ".gemini", "credentials.json"),
|
|
30
|
-
join(homedir(), ".gemini", "auth.json"),
|
|
31
|
-
join(homedir(), ".config", "google", "application_default_credentials.json"),
|
|
32
|
-
];
|
|
33
|
-
|
|
34
|
-
// Map plugin model tier names to Gemini model IDs
|
|
35
|
-
const MODEL_MAP = {
|
|
36
|
-
opus: "gemini-2.5-pro",
|
|
37
|
-
sonnet: "gemini-2.5-flash",
|
|
38
|
-
haiku: "gemini-1.5-flash-8b",
|
|
39
|
-
"gemini-2.5-pro": "gemini-2.5-pro",
|
|
40
|
-
"gemini-2.5-flash": "gemini-2.5-flash",
|
|
41
|
-
"gemini-2.0-flash": "gemini-2.0-flash",
|
|
42
|
-
"gemini-1.5-flash-8b":"gemini-1.5-flash-8b",
|
|
43
|
-
};
|
|
44
|
-
|
|
45
|
-
const {
|
|
46
|
-
SYSTEM_PROMPT,
|
|
47
|
-
buildSystemPrompt,
|
|
48
|
-
buildChatPrompt,
|
|
49
|
-
buildSkillAddendum,
|
|
50
|
-
detectActiveSkills,
|
|
51
|
-
} = require("./shared-prompt-config");
|
|
52
|
-
|
|
53
|
-
// ── Helpers ────────────────────────────────────────────────────────────────
|
|
54
|
-
|
|
55
|
-
function getCleanEnv() {
|
|
56
|
-
const env = { ...process.env };
|
|
57
|
-
for (const key of Object.keys(env)) {
|
|
58
|
-
if (key === "CLAUDECODE" || key.startsWith("CLAUDE_CODE")) delete env[key];
|
|
59
|
-
}
|
|
60
|
-
delete env.ANTHROPIC_AUTH_TOKEN;
|
|
61
|
-
return env;
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
function getFigmaTokenFromClaudeSettings() {
|
|
65
|
-
try {
|
|
66
|
-
if (existsSync(CLAUDE_SETTINGS_PATH)) {
|
|
67
|
-
const settings = JSON.parse(readFileSync(CLAUDE_SETTINGS_PATH, "utf8"));
|
|
68
|
-
return settings?.mcpServers?.["figma-intelligence-layer"]?.env?.FIGMA_ACCESS_TOKEN || "";
|
|
69
|
-
}
|
|
70
|
-
} catch {}
|
|
71
|
-
return "";
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
// ── MCP Registration ────────────────────────────────────────────────────────
|
|
75
|
-
|
|
76
|
-
function ensureGeminiCliMcpRegistered() {
|
|
77
|
-
if (!existsSync(MCP_BUILD_PATH)) return;
|
|
78
|
-
|
|
79
|
-
const figmaToken = getFigmaTokenFromClaudeSettings();
|
|
80
|
-
if (!figmaToken) return;
|
|
81
|
-
|
|
82
|
-
try {
|
|
83
|
-
mkdirSync(GEMINI_SETTINGS_DIR, { recursive: true });
|
|
84
|
-
|
|
85
|
-
let settings = {};
|
|
86
|
-
if (existsSync(GEMINI_SETTINGS_PATH)) {
|
|
87
|
-
try { settings = JSON.parse(readFileSync(GEMINI_SETTINGS_PATH, "utf8")); } catch {}
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
const existing = settings?.mcpServers?.["figma-intelligence-layer"];
|
|
91
|
-
if (
|
|
92
|
-
existing &&
|
|
93
|
-
existing.command === "node" &&
|
|
94
|
-
Array.isArray(existing.args) &&
|
|
95
|
-
existing.args[0] === MCP_BUILD_PATH &&
|
|
96
|
-
existing.env?.FIGMA_ACCESS_TOKEN === figmaToken
|
|
97
|
-
) {
|
|
98
|
-
return;
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
if (!settings.mcpServers) settings.mcpServers = {};
|
|
102
|
-
settings.mcpServers["figma-intelligence-layer"] = {
|
|
103
|
-
command: "node",
|
|
104
|
-
args: [MCP_BUILD_PATH],
|
|
105
|
-
env: {
|
|
106
|
-
FIGMA_ACCESS_TOKEN: figmaToken,
|
|
107
|
-
FIGMA_BRIDGE_PORT: "9001",
|
|
108
|
-
ENABLE_DECISION_LOG: "true",
|
|
109
|
-
},
|
|
110
|
-
};
|
|
111
|
-
|
|
112
|
-
writeFileSync(GEMINI_SETTINGS_PATH, JSON.stringify(settings, null, 2));
|
|
113
|
-
console.log("[gemini-cli-runner] Registered figma-intelligence-layer MCP in ~/.gemini/settings.json");
|
|
114
|
-
} catch (err) {
|
|
115
|
-
console.error("[gemini-cli-runner] Could not update ~/.gemini/settings.json:", err.message);
|
|
116
|
-
}
|
|
117
|
-
}
|
|
118
|
-
|
|
119
|
-
ensureGeminiCliMcpRegistered();
|
|
120
|
-
|
|
121
|
-
// ── Auth Detection ──────────────────────────────────────────────────────────
|
|
122
|
-
|
|
123
|
-
function isGeminiCliAvailable() {
|
|
124
|
-
return new Promise((resolve) => {
|
|
125
|
-
const proc = spawn(GEMINI_BIN, ["--version"], { stdio: "pipe", env: getCleanEnv() });
|
|
126
|
-
proc.on("close", (code) => resolve(code === 0));
|
|
127
|
-
proc.on("error", () => resolve(false));
|
|
128
|
-
});
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
async function getGeminiCliAuthInfo() {
|
|
132
|
-
if (platform() === "darwin") {
|
|
133
|
-
try {
|
|
134
|
-
execSync(
|
|
135
|
-
'security find-generic-password -s "gemini-cli-oauth" -a "main-account" 2>/dev/null',
|
|
136
|
-
{ stdio: "pipe" }
|
|
137
|
-
);
|
|
138
|
-
let email = null;
|
|
139
|
-
try {
|
|
140
|
-
const raw = execSync(
|
|
141
|
-
'security find-generic-password -s "gemini-cli-oauth" -a "main-account" -w 2>/dev/null',
|
|
142
|
-
{ stdio: "pipe" }
|
|
143
|
-
).toString().trim();
|
|
144
|
-
const parsed = JSON.parse(Buffer.from(raw, "base64").toString("utf8") || raw);
|
|
145
|
-
email = parsed?.email || parsed?.client_email || parsed?.user?.email || null;
|
|
146
|
-
} catch {}
|
|
147
|
-
return { loggedIn: true, email };
|
|
148
|
-
} catch {}
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
const oauthCredsPath = join(homedir(), ".gemini", "oauth_creds.json");
|
|
152
|
-
if (existsSync(oauthCredsPath)) {
|
|
153
|
-
try {
|
|
154
|
-
const data = JSON.parse(readFileSync(oauthCredsPath, "utf8"));
|
|
155
|
-
const email = data.email || data.client_email || data.user?.email || null;
|
|
156
|
-
if (data.access_token || data.refresh_token || data.client_email || email || data.token) {
|
|
157
|
-
return { loggedIn: true, email };
|
|
158
|
-
}
|
|
159
|
-
} catch {}
|
|
160
|
-
}
|
|
161
|
-
|
|
162
|
-
for (const p of GEMINI_AUTH_CANDIDATES) {
|
|
163
|
-
if (existsSync(p)) {
|
|
164
|
-
try {
|
|
165
|
-
const data = JSON.parse(readFileSync(p, "utf8"));
|
|
166
|
-
const email = data.email || data.client_email || data.user?.email || null;
|
|
167
|
-
if (data.access_token || data.refresh_token || data.client_email || email || data.token) {
|
|
168
|
-
return { loggedIn: true, email };
|
|
169
|
-
}
|
|
170
|
-
} catch {}
|
|
171
|
-
}
|
|
172
|
-
}
|
|
173
|
-
|
|
174
|
-
return { loggedIn: false, email: null };
|
|
175
|
-
}
|
|
176
|
-
|
|
177
|
-
// ── Chat Runner ─────────────────────────────────────────────────────────────
|
|
178
|
-
|
|
179
|
-
function runGeminiCli({ message, attachments, conversation, requestId, model, designSystemId, mode, onEvent }) {
|
|
180
|
-
const rawText = (message || "").trim() || "Please help with the Figma design.";
|
|
181
|
-
const userText = rawText; // No more expandShortPrompt
|
|
182
|
-
|
|
183
|
-
// Process text attachments
|
|
184
|
-
let extraText = "";
|
|
185
|
-
for (const att of (attachments || [])) {
|
|
186
|
-
if (!att?.data) continue;
|
|
187
|
-
if (att.isImage) {
|
|
188
|
-
extraText += `\n\n[Image attached: ${att.name || "image"} — visual context noted]`;
|
|
189
|
-
} else {
|
|
190
|
-
const label = att.name ? `\n\n--- Attached: ${att.name} ---\n` : "\n\n--- Attached file ---\n";
|
|
191
|
-
extraText += label + att.data;
|
|
192
|
-
}
|
|
193
|
-
}
|
|
194
|
-
|
|
195
|
-
// Slimmed prompt: system prompt + user message (no task guidance, no history, no AGENTS.md)
|
|
196
|
-
const sessionMode = mode || "code";
|
|
197
|
-
const basePrompt = sessionMode === "chat" ? buildChatPrompt() : buildSystemPrompt(designSystemId);
|
|
198
|
-
const skills = detectActiveSkills(userText);
|
|
199
|
-
const systemPrompt = sessionMode === "code" ? basePrompt + buildSkillAddendum(skills) : basePrompt;
|
|
200
|
-
const fullMessage = `${systemPrompt}\n\n---\n\n${userText}${extraText}`;
|
|
201
|
-
const geminiModel = MODEL_MAP[model] || "gemini-2.5-flash";
|
|
202
|
-
|
|
203
|
-
// In chat mode, skip MCP server loading for faster startup (~10s saved)
|
|
204
|
-
const isChatOnly = sessionMode === "chat";
|
|
205
|
-
const args = [
|
|
206
|
-
"--model", geminiModel,
|
|
207
|
-
"--yolo",
|
|
208
|
-
...(isChatOnly ? ["--sandbox", "--allowed-mcp-server-names", ""] : []),
|
|
209
|
-
"-p", fullMessage,
|
|
210
|
-
];
|
|
211
|
-
|
|
212
|
-
const proc = spawn(GEMINI_BIN, args, {
|
|
213
|
-
stdio: ["ignore", "pipe", "pipe"],
|
|
214
|
-
env: getCleanEnv(),
|
|
215
|
-
cwd: REPO_DIR,
|
|
216
|
-
});
|
|
217
|
-
|
|
218
|
-
let fullText = "";
|
|
219
|
-
let stderrOutput = "";
|
|
220
|
-
let lastActivity = Date.now();
|
|
221
|
-
let killed = false;
|
|
222
|
-
const TIMEOUT_MS = 45_000;
|
|
223
|
-
|
|
224
|
-
const timeoutCheck = setInterval(() => {
|
|
225
|
-
if (Date.now() - lastActivity > TIMEOUT_MS) {
|
|
226
|
-
clearInterval(timeoutCheck);
|
|
227
|
-
killed = true;
|
|
228
|
-
try { proc.kill("SIGTERM"); } catch {}
|
|
229
|
-
onEvent({ type: "error", id: requestId, error: "Gemini CLI timed out (45s with no response). Your authentication may have expired — run 'gemini auth login' to re-authenticate." });
|
|
230
|
-
onEvent({ type: "done", id: requestId, fullText: "" });
|
|
231
|
-
}
|
|
232
|
-
}, 5_000);
|
|
233
|
-
|
|
234
|
-
proc.stdout.on("data", (chunk) => {
|
|
235
|
-
lastActivity = Date.now();
|
|
236
|
-
const text = chunk.toString();
|
|
237
|
-
fullText += text;
|
|
238
|
-
onEvent({ type: "text_delta", id: requestId, delta: text });
|
|
239
|
-
});
|
|
240
|
-
|
|
241
|
-
proc.stderr.on("data", (chunk) => {
|
|
242
|
-
lastActivity = Date.now();
|
|
243
|
-
const text = chunk.toString().trim();
|
|
244
|
-
if (text) {
|
|
245
|
-
stderrOutput += text + "\n";
|
|
246
|
-
console.error("[gemini-cli chat]", text);
|
|
247
|
-
|
|
248
|
-
// Detect fatal errors from Gemini API and fail fast instead of retrying
|
|
249
|
-
if (!killed && /MODEL_CAPACITY_EXHAUSTED|exhausted your capacity|ModelNotFoundError|Requested entity was not found/i.test(text)) {
|
|
250
|
-
killed = true;
|
|
251
|
-
clearInterval(timeoutCheck);
|
|
252
|
-
try { proc.kill("SIGTERM"); } catch {}
|
|
253
|
-
const isCapacity = /capacity/i.test(text);
|
|
254
|
-
const errMsg = isCapacity
|
|
255
|
-
? `Gemini model "${geminiModel}" is at capacity. Try switching to a different model (e.g. Gemini 2.5 Flash or Gemini 2.0 Flash).`
|
|
256
|
-
: `Gemini model "${geminiModel}" is not available. Try switching to Gemini 2.5 Flash.`;
|
|
257
|
-
onEvent({ type: "error", id: requestId, error: errMsg });
|
|
258
|
-
onEvent({ type: "done", id: requestId, fullText: "" });
|
|
259
|
-
}
|
|
260
|
-
}
|
|
261
|
-
});
|
|
262
|
-
|
|
263
|
-
proc.on("close", (code) => {
|
|
264
|
-
clearInterval(timeoutCheck);
|
|
265
|
-
if (killed) return; // Already reported error
|
|
266
|
-
if (code !== 0 && code !== null && fullText === "") {
|
|
267
|
-
const detail = stderrOutput.trim().split("\n").pop() || `exit code ${code}`;
|
|
268
|
-
onEvent({
|
|
269
|
-
type: "error",
|
|
270
|
-
id: requestId,
|
|
271
|
-
error: `Gemini CLI error: ${detail}`,
|
|
272
|
-
});
|
|
273
|
-
}
|
|
274
|
-
onEvent({ type: "done", id: requestId, fullText });
|
|
275
|
-
});
|
|
276
|
-
|
|
277
|
-
proc.on("error", (err) => {
|
|
278
|
-
clearInterval(timeoutCheck);
|
|
279
|
-
if (killed) return;
|
|
280
|
-
onEvent({
|
|
281
|
-
type: "error",
|
|
282
|
-
id: requestId,
|
|
283
|
-
error: `Failed to start gemini CLI: ${err.message}. Install with: npm install -g @google/gemini-cli`,
|
|
284
|
-
});
|
|
285
|
-
onEvent({ type: "done", id: requestId, fullText: "" });
|
|
286
|
-
});
|
|
287
|
-
|
|
288
|
-
return proc;
|
|
289
|
-
}
|
|
290
|
-
|
|
291
|
-
module.exports = { runGeminiCli, isGeminiCliAvailable, getGeminiCliAuthInfo, ensureGeminiCliMcpRegistered };
|
|
@@ -1,187 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
/**
|
|
3
|
-
* gemini-runner.js — Streams a chat response from the Google Gemini API.
|
|
4
|
-
* Emits the same event shape as chat-runner.js so bridge-relay can use either.
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
const https = require("https");
|
|
8
|
-
const { EventEmitter } = require("events");
|
|
9
|
-
const { buildSystemPrompt, buildChatPrompt, buildSkillAddendum, detectActiveSkills } = require("./shared-prompt-config");
|
|
10
|
-
|
|
11
|
-
// Map the plugin's Opus/Sonnet/Haiku tier names to Gemini model IDs
|
|
12
|
-
const MODEL_MAP = {
|
|
13
|
-
opus: "gemini-2.5-pro",
|
|
14
|
-
sonnet: "gemini-2.0-flash",
|
|
15
|
-
haiku: "gemini-1.5-flash-8b",
|
|
16
|
-
"gemini-2.5-pro": "gemini-2.5-pro",
|
|
17
|
-
"gemini-2.5-flash": "gemini-2.5-flash",
|
|
18
|
-
"gemini-2.0-flash": "gemini-2.0-flash",
|
|
19
|
-
"gemini-1.5-flash-8b": "gemini-1.5-flash-8b",
|
|
20
|
-
};
|
|
21
|
-
|
|
22
|
-
// Model-aware output token limits
|
|
23
|
-
const MAX_TOKENS = {
|
|
24
|
-
"gemini-2.5-pro": 65536,
|
|
25
|
-
"gemini-2.5-flash": 65536,
|
|
26
|
-
"gemini-2.0-flash": 8192,
|
|
27
|
-
"gemini-1.5-flash-8b": 8192,
|
|
28
|
-
};
|
|
29
|
-
|
|
30
|
-
// Temperature per mode
|
|
31
|
-
const TEMPERATURE = {
|
|
32
|
-
code: 0.2,
|
|
33
|
-
chat: 0.7,
|
|
34
|
-
dual: 0.3,
|
|
35
|
-
};
|
|
36
|
-
|
|
37
|
-
function formatConversationHistory(conversation) {
|
|
38
|
-
if (!Array.isArray(conversation) || !conversation.length) return "";
|
|
39
|
-
const lines = conversation
|
|
40
|
-
.filter((entry) => entry && (entry.role === "user" || entry.role === "assistant") && entry.text)
|
|
41
|
-
.map((entry) => `${entry.role === "assistant" ? "Assistant" : "User"}: ${entry.text}`);
|
|
42
|
-
return lines.length ? `Conversation so far:\n${lines.join("\n\n")}\n\n---\n\n` : "";
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
/**
|
|
46
|
-
* Spawn a Gemini streaming generate request.
|
|
47
|
-
* Returns an EventEmitter-like object with a .kill() method.
|
|
48
|
-
*/
|
|
49
|
-
function runGemini({ message, attachments, conversation, requestId, apiKey, model, designSystemId, mode, onEvent }) {
|
|
50
|
-
const emitter = new EventEmitter();
|
|
51
|
-
|
|
52
|
-
// Fail fast if no API key
|
|
53
|
-
if (!apiKey) {
|
|
54
|
-
onEvent({ type: "error", id: requestId, error: "No Gemini API key configured. Go to Settings and enter your Google AI Studio API key, or install the Gemini CLI (npm install -g @google/gemini-cli) and run 'gemini auth login'." });
|
|
55
|
-
onEvent({ type: "done", id: requestId, fullText: "" });
|
|
56
|
-
emitter.kill = () => {};
|
|
57
|
-
return emitter;
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
// Process text attachments inline
|
|
61
|
-
let extraText = "";
|
|
62
|
-
for (const att of (attachments || [])) {
|
|
63
|
-
if (!att?.data) continue;
|
|
64
|
-
if (att.isImage) {
|
|
65
|
-
const label = att.name || "image";
|
|
66
|
-
extraText += `\n\n[Image attached: ${label}] (image analysis not available in this mode)`;
|
|
67
|
-
} else {
|
|
68
|
-
const label = att.name ? `\n\n--- Attached: ${att.name} ---\n` : "\n\n--- Attached file ---\n";
|
|
69
|
-
extraText += label + att.data;
|
|
70
|
-
}
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
const userText = (message || "").trim() || "Please help with the Figma design.";
|
|
74
|
-
const historyText = formatConversationHistory(conversation);
|
|
75
|
-
const fullMessage = historyText + userText + extraText;
|
|
76
|
-
|
|
77
|
-
const geminiModel = MODEL_MAP[model] || "gemini-2.0-flash";
|
|
78
|
-
|
|
79
|
-
const bodyObj = {
|
|
80
|
-
system_instruction: { parts: [{ text: (() => { const sm = (mode || "code"); const base = sm === "chat" ? buildChatPrompt() : buildSystemPrompt(designSystemId); return sm === "code" ? base + buildSkillAddendum(detectActiveSkills(userText)) : base; })() }] },
|
|
81
|
-
contents: [{ role: "user", parts: [{ text: fullMessage }] }],
|
|
82
|
-
generationConfig: {
|
|
83
|
-
maxOutputTokens: MAX_TOKENS[geminiModel] || 8192,
|
|
84
|
-
temperature: TEMPERATURE[mode || "code"] || 0.4,
|
|
85
|
-
},
|
|
86
|
-
};
|
|
87
|
-
const body = JSON.stringify(bodyObj);
|
|
88
|
-
|
|
89
|
-
const path =
|
|
90
|
-
`/v1beta/models/${geminiModel}:streamGenerateContent?alt=sse&key=${encodeURIComponent(apiKey)}`;
|
|
91
|
-
|
|
92
|
-
let aborted = false;
|
|
93
|
-
let req = null;
|
|
94
|
-
|
|
95
|
-
function abort() {
|
|
96
|
-
aborted = true;
|
|
97
|
-
if (req) { try { req.destroy(); } catch {} }
|
|
98
|
-
emitter.emit("close", null);
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
try {
|
|
102
|
-
req = https.request(
|
|
103
|
-
{
|
|
104
|
-
hostname: "generativelanguage.googleapis.com",
|
|
105
|
-
path,
|
|
106
|
-
method: "POST",
|
|
107
|
-
headers: {
|
|
108
|
-
"Content-Type": "application/json",
|
|
109
|
-
"Content-Length": Buffer.byteLength(body),
|
|
110
|
-
},
|
|
111
|
-
},
|
|
112
|
-
(res) => {
|
|
113
|
-
if (res.statusCode !== 200) {
|
|
114
|
-
let errBody = "";
|
|
115
|
-
res.on("data", (d) => { errBody += d.toString(); });
|
|
116
|
-
res.on("end", () => {
|
|
117
|
-
let msg = errBody;
|
|
118
|
-
try { msg = JSON.parse(errBody).error?.message || errBody; } catch {}
|
|
119
|
-
onEvent({ type: "error", id: requestId, error: `Gemini: ${msg}` });
|
|
120
|
-
onEvent({ type: "done", id: requestId, fullText: "" });
|
|
121
|
-
emitter.emit("close", 1);
|
|
122
|
-
});
|
|
123
|
-
return;
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
let buffer = "";
|
|
127
|
-
let fullText = "";
|
|
128
|
-
|
|
129
|
-
res.on("data", (chunk) => {
|
|
130
|
-
if (aborted) return;
|
|
131
|
-
buffer += chunk.toString();
|
|
132
|
-
const lines = buffer.split("\n");
|
|
133
|
-
buffer = lines.pop() || "";
|
|
134
|
-
|
|
135
|
-
for (const line of lines) {
|
|
136
|
-
if (!line.startsWith("data: ")) continue;
|
|
137
|
-
const data = line.slice(6).trim();
|
|
138
|
-
try {
|
|
139
|
-
const parsed = JSON.parse(data);
|
|
140
|
-
const text = parsed.candidates?.[0]?.content?.parts?.[0]?.text;
|
|
141
|
-
if (text) {
|
|
142
|
-
fullText += text;
|
|
143
|
-
onEvent({ type: "text_delta", id: requestId, delta: text });
|
|
144
|
-
}
|
|
145
|
-
} catch {}
|
|
146
|
-
}
|
|
147
|
-
});
|
|
148
|
-
|
|
149
|
-
res.on("end", () => {
|
|
150
|
-
if (!aborted) {
|
|
151
|
-
onEvent({ type: "done", id: requestId, fullText });
|
|
152
|
-
}
|
|
153
|
-
emitter.emit("close", 0);
|
|
154
|
-
});
|
|
155
|
-
|
|
156
|
-
res.on("error", (err) => {
|
|
157
|
-
if (!aborted) {
|
|
158
|
-
onEvent({ type: "error", id: requestId, error: `Gemini stream error: ${err.message}` });
|
|
159
|
-
onEvent({ type: "done", id: requestId, fullText: "" });
|
|
160
|
-
}
|
|
161
|
-
emitter.emit("close", 1);
|
|
162
|
-
});
|
|
163
|
-
}
|
|
164
|
-
);
|
|
165
|
-
|
|
166
|
-
req.on("error", (err) => {
|
|
167
|
-
if (!aborted) {
|
|
168
|
-
onEvent({ type: "error", id: requestId, error: `Gemini connection error: ${err.message}` });
|
|
169
|
-
onEvent({ type: "done", id: requestId, fullText: "" });
|
|
170
|
-
}
|
|
171
|
-
emitter.emit("close", 1);
|
|
172
|
-
});
|
|
173
|
-
|
|
174
|
-
req.write(body);
|
|
175
|
-
req.end();
|
|
176
|
-
|
|
177
|
-
} catch (err) {
|
|
178
|
-
onEvent({ type: "error", id: requestId, error: `Gemini error: ${err.message}` });
|
|
179
|
-
onEvent({ type: "done", id: requestId, fullText: "" });
|
|
180
|
-
emitter.emit("close", 1);
|
|
181
|
-
}
|
|
182
|
-
|
|
183
|
-
emitter.kill = abort;
|
|
184
|
-
return emitter;
|
|
185
|
-
}
|
|
186
|
-
|
|
187
|
-
module.exports = { runGemini };
|