unreal-engine-mcp-server 0.5.4 → 0.5.6
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/CHANGELOG.md +350 -0
- package/dist/automation/bridge.d.ts.map +1 -0
- package/dist/automation/bridge.js +5 -4
- package/dist/automation/bridge.js.map +1 -0
- package/dist/automation/connection-manager.d.ts.map +1 -0
- package/dist/automation/connection-manager.js.map +1 -0
- package/dist/automation/handshake.d.ts.map +1 -0
- package/dist/automation/handshake.js.map +1 -0
- package/dist/automation/index.d.ts.map +1 -0
- package/dist/automation/index.js.map +1 -0
- package/dist/automation/message-handler.d.ts.map +1 -0
- package/dist/automation/message-handler.js.map +1 -0
- package/dist/automation/request-tracker.d.ts.map +1 -0
- package/dist/automation/request-tracker.js.map +1 -0
- package/dist/automation/types.d.ts +7 -0
- package/dist/automation/types.d.ts.map +1 -0
- package/dist/automation/types.js.map +1 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +6 -4
- package/dist/cli.js.map +1 -0
- package/dist/config/class-aliases.d.ts.map +1 -0
- package/dist/config/class-aliases.js.map +1 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js.map +1 -0
- package/dist/constants.d.ts.map +1 -0
- package/dist/constants.js.map +1 -0
- package/dist/graphql/loaders.d.ts.map +1 -0
- package/dist/graphql/loaders.js.map +1 -0
- package/dist/graphql/resolvers.d.ts +174 -69
- package/dist/graphql/resolvers.d.ts.map +1 -0
- package/dist/graphql/resolvers.js +82 -67
- package/dist/graphql/resolvers.js.map +1 -0
- package/dist/graphql/schema.d.ts.map +1 -0
- package/dist/graphql/schema.js.map +1 -0
- package/dist/graphql/server.d.ts.map +1 -0
- package/dist/graphql/server.js.map +1 -0
- package/dist/graphql/types.d.ts.map +1 -0
- package/dist/graphql/types.js.map +1 -0
- package/dist/handlers/resource-handlers.d.ts.map +1 -0
- package/dist/handlers/resource-handlers.js.map +1 -0
- package/dist/index.d.ts +2 -1
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +70 -9
- package/dist/index.js.map +1 -0
- package/dist/resources/actors.d.ts +7 -4
- package/dist/resources/actors.d.ts.map +1 -0
- package/dist/resources/actors.js +15 -12
- package/dist/resources/actors.js.map +1 -0
- package/dist/resources/assets.d.ts +43 -2
- package/dist/resources/assets.d.ts.map +1 -0
- package/dist/resources/assets.js +21 -12
- package/dist/resources/assets.js.map +1 -0
- package/dist/resources/levels.d.ts.map +1 -0
- package/dist/resources/levels.js +7 -5
- package/dist/resources/levels.js.map +1 -0
- package/dist/schemas/index.d.ts +4 -0
- package/dist/schemas/index.d.ts.map +1 -0
- package/dist/schemas/index.js +4 -0
- package/dist/schemas/index.js.map +1 -0
- package/dist/schemas/parser.d.ts +20 -0
- package/dist/schemas/parser.d.ts.map +1 -0
- package/dist/schemas/parser.js +61 -0
- package/dist/schemas/parser.js.map +1 -0
- package/dist/schemas/primitives.d.ts +221 -0
- package/dist/schemas/primitives.d.ts.map +1 -0
- package/dist/schemas/primitives.js +115 -0
- package/dist/schemas/primitives.js.map +1 -0
- package/dist/schemas/responses.d.ts +362 -0
- package/dist/schemas/responses.d.ts.map +1 -0
- package/dist/schemas/responses.js +252 -0
- package/dist/schemas/responses.js.map +1 -0
- package/dist/server/resource-registry.d.ts.map +1 -0
- package/dist/server/resource-registry.js.map +1 -0
- package/dist/server/tool-registry.d.ts.map +1 -0
- package/dist/server/tool-registry.js +22 -17
- package/dist/server/tool-registry.js.map +1 -0
- package/dist/server-setup.d.ts.map +1 -0
- package/dist/server-setup.js.map +1 -0
- package/dist/services/health-monitor.d.ts +1 -1
- package/dist/services/health-monitor.d.ts.map +1 -0
- package/dist/services/health-monitor.js +4 -3
- package/dist/services/health-monitor.js.map +1 -0
- package/dist/services/metrics-server.d.ts.map +1 -0
- package/dist/services/metrics-server.js.map +1 -0
- package/dist/tools/actors.d.ts +27 -27
- package/dist/tools/actors.d.ts.map +1 -0
- package/dist/tools/actors.js +14 -10
- package/dist/tools/actors.js.map +1 -0
- package/dist/tools/animation.d.ts +15 -23
- package/dist/tools/animation.d.ts.map +1 -0
- package/dist/tools/animation.js +17 -13
- package/dist/tools/animation.js.map +1 -0
- package/dist/tools/assets.d.ts.map +1 -0
- package/dist/tools/assets.js +18 -12
- package/dist/tools/assets.js.map +1 -0
- package/dist/tools/audio.d.ts +10 -10
- package/dist/tools/audio.d.ts.map +1 -0
- package/dist/tools/audio.js.map +1 -0
- package/dist/tools/base-tool.d.ts.map +1 -0
- package/dist/tools/base-tool.js.map +1 -0
- package/dist/tools/behavior-tree.d.ts +24 -24
- package/dist/tools/behavior-tree.d.ts.map +1 -0
- package/dist/tools/behavior-tree.js.map +1 -0
- package/dist/tools/blueprint.d.ts +14 -3
- package/dist/tools/blueprint.d.ts.map +1 -0
- package/dist/tools/blueprint.js +5 -3
- package/dist/tools/blueprint.js.map +1 -0
- package/dist/tools/consolidated-tool-definitions.d.ts +32 -32
- package/dist/tools/consolidated-tool-definitions.d.ts.map +1 -0
- package/dist/tools/consolidated-tool-definitions.js.map +1 -0
- package/dist/tools/consolidated-tool-handlers.d.ts +1 -1
- package/dist/tools/consolidated-tool-handlers.d.ts.map +1 -0
- package/dist/tools/consolidated-tool-handlers.js +26 -21
- package/dist/tools/consolidated-tool-handlers.js.map +1 -0
- package/dist/tools/debug.d.ts +25 -7
- package/dist/tools/debug.d.ts.map +1 -0
- package/dist/tools/debug.js +3 -1
- package/dist/tools/debug.js.map +1 -0
- package/dist/tools/dynamic-handler-registry.d.ts +1 -1
- package/dist/tools/dynamic-handler-registry.d.ts.map +1 -0
- package/dist/tools/dynamic-handler-registry.js +3 -1
- package/dist/tools/dynamic-handler-registry.js.map +1 -0
- package/dist/tools/editor.d.ts.map +1 -0
- package/dist/tools/editor.js +8 -6
- package/dist/tools/editor.js.map +1 -0
- package/dist/tools/engine.d.ts +1 -1
- package/dist/tools/engine.d.ts.map +1 -0
- package/dist/tools/engine.js +4 -2
- package/dist/tools/engine.js.map +1 -0
- package/dist/tools/environment.d.ts.map +1 -0
- package/dist/tools/environment.js +4 -3
- package/dist/tools/environment.js.map +1 -0
- package/dist/tools/foliage.d.ts.map +1 -0
- package/dist/tools/foliage.js +8 -8
- package/dist/tools/foliage.js.map +1 -0
- package/dist/tools/handlers/actor-handlers.d.ts +2 -1
- package/dist/tools/handlers/actor-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/actor-handlers.js +56 -33
- package/dist/tools/handlers/actor-handlers.js.map +1 -0
- package/dist/tools/handlers/animation-handlers.d.ts +2 -1
- package/dist/tools/handlers/animation-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/animation-handlers.js +74 -67
- package/dist/tools/handlers/animation-handlers.js.map +1 -0
- package/dist/tools/handlers/argument-helper.d.ts +24 -4
- package/dist/tools/handlers/argument-helper.d.ts.map +1 -0
- package/dist/tools/handlers/argument-helper.js +139 -4
- package/dist/tools/handlers/argument-helper.js.map +1 -0
- package/dist/tools/handlers/asset-handlers.d.ts +2 -1
- package/dist/tools/handlers/asset-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/asset-handlers.js +155 -94
- package/dist/tools/handlers/asset-handlers.js.map +1 -0
- package/dist/tools/handlers/audio-handlers.d.ts +2 -1
- package/dist/tools/handlers/audio-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/audio-handlers.js +82 -80
- package/dist/tools/handlers/audio-handlers.js.map +1 -0
- package/dist/tools/handlers/blueprint-handlers.d.ts +3 -5
- package/dist/tools/handlers/blueprint-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/blueprint-handlers.js +150 -142
- package/dist/tools/handlers/blueprint-handlers.js.map +1 -0
- package/dist/tools/handlers/common-handlers.d.ts +2 -3
- package/dist/tools/handlers/common-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/common-handlers.js.map +1 -0
- package/dist/tools/handlers/editor-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/editor-handlers.js +12 -2
- package/dist/tools/handlers/editor-handlers.js.map +1 -0
- package/dist/tools/handlers/effect-handlers.d.ts +2 -1
- package/dist/tools/handlers/effect-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/effect-handlers.js +70 -68
- package/dist/tools/handlers/effect-handlers.js.map +1 -0
- package/dist/tools/handlers/environment-handlers.d.ts +2 -1
- package/dist/tools/handlers/environment-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/environment-handlers.js +86 -74
- package/dist/tools/handlers/environment-handlers.js.map +1 -0
- package/dist/tools/handlers/graph-handlers.d.ts +1 -1
- package/dist/tools/handlers/graph-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/graph-handlers.js +63 -2
- package/dist/tools/handlers/graph-handlers.js.map +1 -0
- package/dist/tools/handlers/input-handlers.d.ts +2 -5
- package/dist/tools/handlers/input-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/input-handlers.js +5 -4
- package/dist/tools/handlers/input-handlers.js.map +1 -0
- package/dist/tools/handlers/inspect-handlers.d.ts +2 -1
- package/dist/tools/handlers/inspect-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/inspect-handlers.js +61 -37
- package/dist/tools/handlers/inspect-handlers.js.map +1 -0
- package/dist/tools/handlers/level-handlers.d.ts +2 -2
- package/dist/tools/handlers/level-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/level-handlers.js +43 -39
- package/dist/tools/handlers/level-handlers.js.map +1 -0
- package/dist/tools/handlers/lighting-handlers.d.ts +12 -1
- package/dist/tools/handlers/lighting-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/lighting-handlers.js +90 -47
- package/dist/tools/handlers/lighting-handlers.js.map +1 -0
- package/dist/tools/handlers/performance-handlers.d.ts +2 -1
- package/dist/tools/handlers/performance-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/performance-handlers.js +55 -40
- package/dist/tools/handlers/performance-handlers.js.map +1 -0
- package/dist/tools/handlers/pipeline-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/pipeline-handlers.js.map +1 -0
- package/dist/tools/handlers/sequence-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/sequence-handlers.js.map +1 -0
- package/dist/tools/handlers/system-handlers.d.ts +3 -2
- package/dist/tools/handlers/system-handlers.d.ts.map +1 -0
- package/dist/tools/handlers/system-handlers.js +105 -52
- package/dist/tools/handlers/system-handlers.js.map +1 -0
- package/dist/tools/input.d.ts.map +1 -0
- package/dist/tools/input.js +3 -1
- package/dist/tools/input.js.map +1 -0
- package/dist/tools/introspection.d.ts +14 -14
- package/dist/tools/introspection.d.ts.map +1 -0
- package/dist/tools/introspection.js +54 -45
- package/dist/tools/introspection.js.map +1 -0
- package/dist/tools/landscape.d.ts.map +1 -0
- package/dist/tools/landscape.js +15 -13
- package/dist/tools/landscape.js.map +1 -0
- package/dist/tools/level.d.ts.map +1 -0
- package/dist/tools/level.js +3 -2
- package/dist/tools/level.js.map +1 -0
- package/dist/tools/lighting.d.ts +32 -59
- package/dist/tools/lighting.d.ts.map +1 -0
- package/dist/tools/lighting.js +56 -19
- package/dist/tools/lighting.js.map +1 -0
- package/dist/tools/logs.d.ts.map +1 -0
- package/dist/tools/logs.js +2 -1
- package/dist/tools/logs.js.map +1 -0
- package/dist/tools/materials.d.ts +42 -14
- package/dist/tools/materials.d.ts.map +1 -0
- package/dist/tools/materials.js +15 -9
- package/dist/tools/materials.js.map +1 -0
- package/dist/tools/niagara.d.ts +63 -39
- package/dist/tools/niagara.d.ts.map +1 -0
- package/dist/tools/niagara.js +43 -33
- package/dist/tools/niagara.js.map +1 -0
- package/dist/tools/performance.d.ts +12 -11
- package/dist/tools/performance.d.ts.map +1 -0
- package/dist/tools/performance.js +3 -2
- package/dist/tools/performance.js.map +1 -0
- package/dist/tools/physics.d.ts +37 -20
- package/dist/tools/physics.d.ts.map +1 -0
- package/dist/tools/physics.js +37 -30
- package/dist/tools/physics.js.map +1 -0
- package/dist/tools/property-dictionary.d.ts.map +1 -0
- package/dist/tools/property-dictionary.js.map +1 -0
- package/dist/tools/sequence.d.ts +1 -1
- package/dist/tools/sequence.d.ts.map +1 -0
- package/dist/tools/sequence.js +8 -4
- package/dist/tools/sequence.js.map +1 -0
- package/dist/tools/tool-definition-utils.d.ts.map +1 -0
- package/dist/tools/tool-definition-utils.js.map +1 -0
- package/dist/tools/ui.d.ts +11 -11
- package/dist/tools/ui.d.ts.map +1 -0
- package/dist/tools/ui.js +7 -3
- package/dist/tools/ui.js.map +1 -0
- package/dist/types/automation-responses.d.ts.map +1 -0
- package/dist/types/automation-responses.js.map +1 -0
- package/dist/types/env.d.ts.map +1 -0
- package/dist/types/env.js.map +1 -0
- package/dist/types/handler-types.d.ts +112 -3
- package/dist/types/handler-types.d.ts.map +1 -0
- package/dist/types/handler-types.js.map +1 -0
- package/dist/types/tool-interfaces.d.ts +39 -21
- package/dist/types/tool-interfaces.d.ts.map +1 -0
- package/dist/types/tool-interfaces.js.map +1 -0
- package/dist/types/tool-types.d.ts +8 -8
- package/dist/types/tool-types.d.ts.map +1 -0
- package/dist/types/tool-types.js.map +1 -0
- package/dist/unreal-bridge.d.ts +8 -6
- package/dist/unreal-bridge.d.ts.map +1 -0
- package/dist/unreal-bridge.js +16 -3
- package/dist/unreal-bridge.js.map +1 -0
- package/dist/utils/command-validator.d.ts.map +1 -0
- package/dist/utils/command-validator.js.map +1 -0
- package/dist/utils/elicitation.d.ts +2 -5
- package/dist/utils/elicitation.d.ts.map +1 -0
- package/dist/utils/elicitation.js +3 -2
- package/dist/utils/elicitation.js.map +1 -0
- package/dist/utils/error-handler.d.ts.map +1 -0
- package/dist/utils/error-handler.js.map +1 -0
- package/dist/utils/ini-reader.d.ts +1 -1
- package/dist/utils/ini-reader.d.ts.map +1 -0
- package/dist/utils/ini-reader.js.map +1 -0
- package/dist/utils/logger.d.ts +4 -4
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js.map +1 -0
- package/dist/utils/normalize.d.ts +2 -2
- package/dist/utils/normalize.d.ts.map +1 -0
- package/dist/utils/normalize.js +4 -3
- package/dist/utils/normalize.js.map +1 -0
- package/dist/utils/path-security.d.ts.map +1 -0
- package/dist/utils/path-security.js.map +1 -0
- package/dist/utils/response-factory.d.ts +2 -2
- package/dist/utils/response-factory.d.ts.map +1 -0
- package/dist/utils/response-factory.js +3 -1
- package/dist/utils/response-factory.js.map +1 -0
- package/dist/utils/response-validator.d.ts +4 -4
- package/dist/utils/response-validator.d.ts.map +1 -0
- package/dist/utils/response-validator.js +31 -23
- package/dist/utils/response-validator.js.map +1 -0
- package/dist/utils/result-helpers.d.ts.map +1 -0
- package/dist/utils/result-helpers.js.map +1 -0
- package/dist/utils/safe-json.d.ts.map +1 -0
- package/dist/utils/safe-json.js.map +1 -0
- package/dist/utils/unreal-command-queue.d.ts +2 -2
- package/dist/utils/unreal-command-queue.d.ts.map +1 -0
- package/dist/utils/unreal-command-queue.js +4 -3
- package/dist/utils/unreal-command-queue.js.map +1 -0
- package/dist/utils/validation.d.ts +1 -1
- package/dist/utils/validation.d.ts.map +1 -0
- package/dist/utils/validation.js.map +1 -0
- package/dist/wasm/index.d.ts +2 -2
- package/dist/wasm/index.d.ts.map +1 -0
- package/dist/wasm/index.js +11 -7
- package/dist/wasm/index.js.map +1 -0
- package/package.json +12 -34
- package/server.json +2 -2
- package/.dockerignore +0 -57
- package/.env.example +0 -26
- package/.env.production +0 -61
- package/.eslintrc.json +0 -0
- package/.eslintrc.override.json +0 -8
- package/.github/ISSUE_TEMPLATE/bug_report.yml +0 -94
- package/.github/ISSUE_TEMPLATE/config.yml +0 -8
- package/.github/ISSUE_TEMPLATE/feature_request.yml +0 -56
- package/.github/copilot-instructions.md +0 -478
- package/.github/dependabot.yml +0 -19
- package/.github/labeler.yml +0 -24
- package/.github/labels.yml +0 -70
- package/.github/pull_request_template.md +0 -42
- package/.github/release-drafter-config.yml +0 -51
- package/.github/workflows/auto-merge.yml +0 -38
- package/.github/workflows/ci.yml +0 -38
- package/.github/workflows/dependency-review.yml +0 -17
- package/.github/workflows/gemini-issue-triage.yml +0 -172
- package/.github/workflows/greetings.yml +0 -27
- package/.github/workflows/labeler.yml +0 -17
- package/.github/workflows/links.yml +0 -80
- package/.github/workflows/pr-size-labeler.yml +0 -137
- package/.github/workflows/publish-mcp.yml +0 -79
- package/.github/workflows/release-drafter.yml +0 -24
- package/.github/workflows/release.yml +0 -112
- package/.github/workflows/semantic-pull-request.yml +0 -35
- package/.github/workflows/smoke-test.yml +0 -36
- package/.github/workflows/stale.yml +0 -28
- package/CONTRIBUTING.md +0 -140
- package/Dockerfile +0 -37
- package/GEMINI.md +0 -115
- package/Public/Plugin_setup_guide.mp4 +0 -0
- package/Public/icon.png +0 -0
- package/claude_desktop_config_example.json +0 -15
- package/dist/types/responses.d.ts +0 -249
- package/dist/types/responses.js +0 -2
- package/docs/GraphQL-API.md +0 -888
- package/docs/Migration-Guide-v0.5.0.md +0 -684
- package/docs/Roadmap.md +0 -53
- package/docs/WebAssembly-Integration.md +0 -628
- package/docs/editor-plugin-extension.md +0 -370
- package/docs/handler-mapping.md +0 -249
- package/docs/native-automation-progress.md +0 -128
- package/docs/testing-guide.md +0 -423
- package/eslint.config.mjs +0 -68
- package/mcp-config-example.json +0 -14
- package/plugins/McpAutomationBridge/Config/FilterPlugin.ini +0 -8
- package/plugins/McpAutomationBridge/McpAutomationBridge.uplugin +0 -64
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/McpAutomationBridge.Build.cs +0 -189
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeGlobals.cpp +0 -22
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeGlobals.h +0 -30
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeHelpers.h +0 -1983
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeModule.cpp +0 -72
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeSettings.cpp +0 -46
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeSubsystem.cpp +0 -846
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AnimationHandlers.cpp +0 -2393
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AssetQueryHandlers.cpp +0 -300
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AssetWorkflowHandlers.cpp +0 -2807
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AudioHandlers.cpp +0 -1087
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BehaviorTreeHandlers.cpp +0 -488
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintCreationHandlers.cpp +0 -643
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintCreationHandlers.h +0 -31
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintGraphHandlers.cpp +0 -1094
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintHandlers.cpp +0 -5750
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintHandlers_List.cpp +0 -152
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_ControlHandlers.cpp +0 -2614
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_DebugHandlers.cpp +0 -42
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_EditorFunctionHandlers.cpp +0 -1237
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_EffectHandlers.cpp +0 -1725
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_EnvironmentHandlers.cpp +0 -2265
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_FoliageHandlers.cpp +0 -954
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_InputHandlers.cpp +0 -209
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_InsightsHandlers.cpp +0 -41
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LandscapeHandlers.cpp +0 -1164
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LevelHandlers.cpp +0 -762
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LightingHandlers.cpp +0 -663
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LogHandlers.cpp +0 -136
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_MaterialGraphHandlers.cpp +0 -494
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_NiagaraGraphHandlers.cpp +0 -278
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_NiagaraHandlers.cpp +0 -625
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_PerformanceHandlers.cpp +0 -401
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_PipelineHandlers.cpp +0 -67
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_ProcessRequest.cpp +0 -472
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_PropertyHandlers.cpp +0 -2634
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_RenderHandlers.cpp +0 -189
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SCSHandlers.cpp +0 -917
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SCSHandlers.h +0 -39
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SequenceHandlers.cpp +0 -2706
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SequencerHandlers.cpp +0 -519
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_TestHandlers.cpp +0 -38
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_UiHandlers.cpp +0 -668
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_WorldPartitionHandlers.cpp +0 -346
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpBridgeWebSocket.cpp +0 -1345
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpBridgeWebSocket.h +0 -149
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpConnectionManager.cpp +0 -782
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Public/McpAutomationBridgeSettings.h +0 -115
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Public/McpAutomationBridgeSubsystem.h +0 -796
- package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Public/McpConnectionManager.h +0 -117
- package/scripts/check-unreal-connection.mjs +0 -19
- package/scripts/clean-tmp.js +0 -23
- package/scripts/patch-wasm.js +0 -26
- package/scripts/run-all-tests.mjs +0 -136
- package/scripts/smoke-test.ts +0 -94
- package/scripts/sync-mcp-plugin.js +0 -143
- package/scripts/test-no-plugin-alternates.mjs +0 -113
- package/scripts/validate-server.js +0 -46
- package/scripts/verify-automation-bridge.js +0 -200
- package/src/automation/bridge.ts +0 -630
- package/src/automation/connection-manager.ts +0 -148
- package/src/automation/handshake.ts +0 -99
- package/src/automation/index.ts +0 -2
- package/src/automation/message-handler.ts +0 -192
- package/src/automation/request-tracker.ts +0 -155
- package/src/automation/types.ts +0 -108
- package/src/cli.ts +0 -34
- package/src/config/class-aliases.ts +0 -65
- package/src/config.ts +0 -73
- package/src/constants.ts +0 -29
- package/src/graphql/loaders.ts +0 -244
- package/src/graphql/resolvers.ts +0 -1008
- package/src/graphql/schema.ts +0 -452
- package/src/graphql/server.ts +0 -156
- package/src/graphql/types.ts +0 -10
- package/src/handlers/resource-handlers.ts +0 -186
- package/src/index.ts +0 -243
- package/src/resources/actors.ts +0 -127
- package/src/resources/assets.ts +0 -286
- package/src/resources/levels.ts +0 -68
- package/src/server/resource-registry.ts +0 -47
- package/src/server/tool-registry.ts +0 -354
- package/src/server-setup.ts +0 -114
- package/src/services/health-monitor.ts +0 -132
- package/src/services/metrics-server.ts +0 -176
- package/src/tools/actors.ts +0 -564
- package/src/tools/animation.ts +0 -941
- package/src/tools/assets.ts +0 -394
- package/src/tools/audio.ts +0 -499
- package/src/tools/base-tool.ts +0 -52
- package/src/tools/behavior-tree.ts +0 -45
- package/src/tools/blueprint.ts +0 -940
- package/src/tools/consolidated-tool-definitions.ts +0 -1256
- package/src/tools/consolidated-tool-handlers.ts +0 -302
- package/src/tools/debug.ts +0 -622
- package/src/tools/dynamic-handler-registry.ts +0 -33
- package/src/tools/editor.ts +0 -435
- package/src/tools/engine.ts +0 -43
- package/src/tools/environment.ts +0 -281
- package/src/tools/foliage.ts +0 -596
- package/src/tools/handlers/actor-handlers.ts +0 -244
- package/src/tools/handlers/animation-handlers.ts +0 -237
- package/src/tools/handlers/argument-helper.ts +0 -142
- package/src/tools/handlers/asset-handlers.ts +0 -550
- package/src/tools/handlers/audio-handlers.ts +0 -194
- package/src/tools/handlers/blueprint-handlers.ts +0 -380
- package/src/tools/handlers/common-handlers.ts +0 -108
- package/src/tools/handlers/editor-handlers.ts +0 -124
- package/src/tools/handlers/effect-handlers.ts +0 -224
- package/src/tools/handlers/environment-handlers.ts +0 -183
- package/src/tools/handlers/graph-handlers.ts +0 -117
- package/src/tools/handlers/input-handlers.ts +0 -28
- package/src/tools/handlers/inspect-handlers.ts +0 -450
- package/src/tools/handlers/level-handlers.ts +0 -253
- package/src/tools/handlers/lighting-handlers.ts +0 -151
- package/src/tools/handlers/performance-handlers.ts +0 -132
- package/src/tools/handlers/pipeline-handlers.ts +0 -194
- package/src/tools/handlers/sequence-handlers.ts +0 -438
- package/src/tools/handlers/system-handlers.ts +0 -564
- package/src/tools/input.ts +0 -160
- package/src/tools/introspection.ts +0 -689
- package/src/tools/landscape.ts +0 -649
- package/src/tools/level.ts +0 -989
- package/src/tools/lighting.ts +0 -1052
- package/src/tools/logs.ts +0 -219
- package/src/tools/materials.ts +0 -295
- package/src/tools/niagara.ts +0 -485
- package/src/tools/performance.ts +0 -661
- package/src/tools/physics.ts +0 -679
- package/src/tools/property-dictionary.ts +0 -98
- package/src/tools/sequence.ts +0 -385
- package/src/tools/tool-definition-utils.ts +0 -35
- package/src/tools/ui.ts +0 -452
- package/src/types/automation-responses.ts +0 -119
- package/src/types/env.ts +0 -17
- package/src/types/handler-types.ts +0 -442
- package/src/types/responses.ts +0 -355
- package/src/types/tool-interfaces.ts +0 -250
- package/src/types/tool-types.ts +0 -575
- package/src/unreal-bridge.ts +0 -693
- package/src/utils/command-validator.ts +0 -139
- package/src/utils/elicitation.ts +0 -132
- package/src/utils/error-handler.ts +0 -287
- package/src/utils/ini-reader.ts +0 -86
- package/src/utils/logger.ts +0 -35
- package/src/utils/normalize.test.ts +0 -162
- package/src/utils/normalize.ts +0 -146
- package/src/utils/path-security.ts +0 -43
- package/src/utils/response-factory.ts +0 -44
- package/src/utils/response-validator.ts +0 -395
- package/src/utils/result-helpers.ts +0 -195
- package/src/utils/safe-json.test.ts +0 -90
- package/src/utils/safe-json.ts +0 -70
- package/src/utils/unreal-command-queue.ts +0 -166
- package/src/utils/validation.test.ts +0 -184
- package/src/utils/validation.ts +0 -312
- package/src/wasm/index.ts +0 -838
- package/test-server.mjs +0 -100
- package/tests/test-animation.mjs +0 -369
- package/tests/test-asset-advanced.mjs +0 -82
- package/tests/test-asset-graph.mjs +0 -311
- package/tests/test-audio.mjs +0 -417
- package/tests/test-automation-timeouts.mjs +0 -98
- package/tests/test-behavior-tree.mjs +0 -444
- package/tests/test-blueprint-graph.mjs +0 -410
- package/tests/test-blueprint.mjs +0 -577
- package/tests/test-client-mode.mjs +0 -86
- package/tests/test-console-command.mjs +0 -56
- package/tests/test-control-actor.mjs +0 -425
- package/tests/test-control-editor.mjs +0 -112
- package/tests/test-graphql.mjs +0 -372
- package/tests/test-input.mjs +0 -349
- package/tests/test-inspect.mjs +0 -302
- package/tests/test-landscape.mjs +0 -316
- package/tests/test-lighting.mjs +0 -428
- package/tests/test-manage-asset.mjs +0 -438
- package/tests/test-manage-level.mjs +0 -89
- package/tests/test-materials.mjs +0 -356
- package/tests/test-niagara.mjs +0 -185
- package/tests/test-no-inline-python.mjs +0 -122
- package/tests/test-performance.mjs +0 -539
- package/tests/test-plugin-handshake.mjs +0 -82
- package/tests/test-runner.mjs +0 -993
- package/tests/test-sequence.mjs +0 -104
- package/tests/test-system.mjs +0 -96
- package/tests/test-wasm.mjs +0 -283
- package/tests/test-world-partition.mjs +0 -215
- package/tsconfig.json +0 -56
- package/vitest.config.ts +0 -35
- package/wasm/Cargo.lock +0 -363
- package/wasm/Cargo.toml +0 -42
- package/wasm/LICENSE +0 -21
- package/wasm/README.md +0 -253
- package/wasm/src/dependency_resolver.rs +0 -377
- package/wasm/src/lib.rs +0 -153
- package/wasm/src/property_parser.rs +0 -271
- package/wasm/src/transform_math.rs +0 -396
- package/wasm/tests/integration.rs +0 -109
|
@@ -1,395 +0,0 @@
|
|
|
1
|
-
import Ajv, { ValidateFunction } from 'ajv';
|
|
2
|
-
import { Logger } from './logger.js';
|
|
3
|
-
import { cleanObject } from './safe-json.js';
|
|
4
|
-
import { wasmIntegration } from '../wasm/index.js';
|
|
5
|
-
|
|
6
|
-
const log = new Logger('ResponseValidator');
|
|
7
|
-
|
|
8
|
-
function isRecord(value: unknown): value is Record<string, unknown> {
|
|
9
|
-
return !!value && typeof value === 'object' && !Array.isArray(value);
|
|
10
|
-
}
|
|
11
|
-
|
|
12
|
-
function normalizeText(text: string): string {
|
|
13
|
-
return text.replace(/\s+/g, ' ').trim();
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
function buildSummaryText(toolName: string, payload: unknown): string {
|
|
17
|
-
if (typeof payload === 'string') {
|
|
18
|
-
const normalized = payload.trim();
|
|
19
|
-
return normalized || `${toolName} responded`;
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
if (typeof payload === 'number' || typeof payload === 'bigint' || typeof payload === 'boolean') {
|
|
23
|
-
return `${toolName} responded: ${payload}`;
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
if (!isRecord(payload)) {
|
|
27
|
-
return `${toolName} responded`;
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
// Recursively flatten data/result wrappers into effective payload
|
|
31
|
-
const effectivePayload: Record<string, any> = { ...(payload as object) };
|
|
32
|
-
|
|
33
|
-
const flattenWrappers = (obj: Record<string, any>, depth = 0): void => {
|
|
34
|
-
if (depth > 5) return; // Prevent infinite loops
|
|
35
|
-
if (isRecord(obj.data)) {
|
|
36
|
-
Object.assign(obj, obj.data);
|
|
37
|
-
delete obj.data;
|
|
38
|
-
flattenWrappers(obj, depth + 1);
|
|
39
|
-
}
|
|
40
|
-
if (isRecord(obj.result)) {
|
|
41
|
-
Object.assign(obj, obj.result);
|
|
42
|
-
delete obj.result;
|
|
43
|
-
flattenWrappers(obj, depth + 1);
|
|
44
|
-
}
|
|
45
|
-
};
|
|
46
|
-
flattenWrappers(effectivePayload);
|
|
47
|
-
|
|
48
|
-
const parts: string[] = [];
|
|
49
|
-
const addedKeys = new Set<string>();
|
|
50
|
-
|
|
51
|
-
// Keys to skip (internal/redundant)
|
|
52
|
-
const skipKeys = new Set(['requestId', 'type', 'data', 'result', 'warnings']);
|
|
53
|
-
|
|
54
|
-
// Helper to format a value for display
|
|
55
|
-
const formatValue = (val: unknown): string => {
|
|
56
|
-
if (val === null || val === undefined) return '';
|
|
57
|
-
if (typeof val === 'string') return val.length > 150 ? val.slice(0, 150) + '...' : val;
|
|
58
|
-
if (typeof val === 'number' || typeof val === 'boolean') return String(val);
|
|
59
|
-
|
|
60
|
-
// Handle arrays - show items with names/paths
|
|
61
|
-
if (Array.isArray(val)) {
|
|
62
|
-
if (val.length === 0) return '[] (0)';
|
|
63
|
-
const items = val.slice(0, 30).map(v => {
|
|
64
|
-
if (isRecord(v)) {
|
|
65
|
-
// Try common identifier fields
|
|
66
|
-
return v.name || v.path || v.id || v.nodeId || v.nodeName || v.className ||
|
|
67
|
-
v.displayName || v.type || v.assetPath || v.objectPath ||
|
|
68
|
-
JSON.stringify(v).slice(0, 50);
|
|
69
|
-
}
|
|
70
|
-
return String(v);
|
|
71
|
-
});
|
|
72
|
-
const suffix = val.length > 30 ? `, ... (+${val.length - 30} more)` : '';
|
|
73
|
-
return `[${items.join(', ')}${suffix}] (${val.length})`;
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
// Handle transform-like objects (location/rotation/scale)
|
|
77
|
-
if (isRecord(val)) {
|
|
78
|
-
const keys = Object.keys(val);
|
|
79
|
-
// Check if it looks like a 3D vector/transform
|
|
80
|
-
if (keys.some(k => ['x', 'y', 'z', 'pitch', 'yaw', 'roll'].includes(k))) {
|
|
81
|
-
const x = val.x ?? val.pitch ?? 0;
|
|
82
|
-
const y = val.y ?? val.yaw ?? 0;
|
|
83
|
-
const z = val.z ?? val.roll ?? 0;
|
|
84
|
-
return `[${x}, ${y}, ${z}]`;
|
|
85
|
-
}
|
|
86
|
-
// Generic object - show key=value pairs
|
|
87
|
-
const entries = Object.entries(val).slice(0, 8);
|
|
88
|
-
const formatted = entries.map(([k, v]) => {
|
|
89
|
-
const vStr = typeof v === 'object' ? JSON.stringify(v).slice(0, 40) : String(v);
|
|
90
|
-
return `${k}=${vStr}`;
|
|
91
|
-
});
|
|
92
|
-
return `{ ${formatted.join(', ')}${keys.length > 8 ? ' ...' : ''} }`;
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
return String(val);
|
|
96
|
-
};
|
|
97
|
-
|
|
98
|
-
// Process all keys in priority order
|
|
99
|
-
// 1. First add 'success' and 'error' at the start
|
|
100
|
-
for (const key of ['success', 'error']) {
|
|
101
|
-
if (effectivePayload[key] !== undefined && !addedKeys.has(key)) {
|
|
102
|
-
const formatted = formatValue(effectivePayload[key]);
|
|
103
|
-
if (formatted) {
|
|
104
|
-
parts.push(`${key}: ${formatted}`);
|
|
105
|
-
addedKeys.add(key);
|
|
106
|
-
}
|
|
107
|
-
}
|
|
108
|
-
}
|
|
109
|
-
|
|
110
|
-
// 2. Then add ALL other keys dynamically
|
|
111
|
-
let hasArrays = false;
|
|
112
|
-
for (const [key, val] of Object.entries(effectivePayload)) {
|
|
113
|
-
if (addedKeys.has(key)) continue;
|
|
114
|
-
if (skipKeys.has(key)) continue;
|
|
115
|
-
if (val === undefined || val === null) continue;
|
|
116
|
-
if (typeof val === 'string' && val.trim() === '') continue;
|
|
117
|
-
|
|
118
|
-
// Skip 'message' for now - handle later to avoid duplication
|
|
119
|
-
if (key === 'message') continue;
|
|
120
|
-
|
|
121
|
-
// Track if we have arrays (to skip duplicate count/totalCount later)
|
|
122
|
-
if (Array.isArray(val) && val.length > 0) hasArrays = true;
|
|
123
|
-
|
|
124
|
-
// Skip count/totalCount if we already have arrays showing counts
|
|
125
|
-
if ((key === 'count' || key === 'totalCount') && hasArrays) continue;
|
|
126
|
-
|
|
127
|
-
const formatted = formatValue(val);
|
|
128
|
-
if (formatted) {
|
|
129
|
-
parts.push(`${key}: ${formatted}`);
|
|
130
|
-
addedKeys.add(key);
|
|
131
|
-
}
|
|
132
|
-
}
|
|
133
|
-
|
|
134
|
-
// 3. Handle message last - but skip if it duplicates existing info
|
|
135
|
-
const message = typeof effectivePayload.message === 'string' ? normalizeText(effectivePayload.message) : '';
|
|
136
|
-
if (message && message.toLowerCase() !== 'success') {
|
|
137
|
-
// Skip if message duplicates count info
|
|
138
|
-
const isDuplicateInfo = /^(found|listed|retrieved|got|loaded|created|deleted|saved|spawned)\s+\d+/i.test(message) ||
|
|
139
|
-
/Folders:\s*\[/.test(message) ||
|
|
140
|
-
/\d+\s+(assets?|folders?|items?|actors?|components?)\s+(and|in|at)/i.test(message);
|
|
141
|
-
|
|
142
|
-
// Also skip if message content is already represented in parts
|
|
143
|
-
const messageInParts = parts.some(p => p.toLowerCase().includes(message.toLowerCase().slice(0, 30)));
|
|
144
|
-
|
|
145
|
-
if (!isDuplicateInfo && !messageInParts) {
|
|
146
|
-
parts.push(message);
|
|
147
|
-
}
|
|
148
|
-
}
|
|
149
|
-
|
|
150
|
-
// 4. Warnings at end
|
|
151
|
-
const warnings = Array.isArray(effectivePayload.warnings) ? effectivePayload.warnings : [];
|
|
152
|
-
if (warnings.length > 0) {
|
|
153
|
-
parts.push(`Warnings: ${warnings.map((w: any) => typeof w === 'string' ? w : JSON.stringify(w)).join('; ')}`);
|
|
154
|
-
}
|
|
155
|
-
|
|
156
|
-
return parts.length > 0 ? parts.join(' | ') : `${toolName} responded`;
|
|
157
|
-
}
|
|
158
|
-
|
|
159
|
-
/**
|
|
160
|
-
* Response Validator for MCP Tool Outputs
|
|
161
|
-
* Validates tool responses against their defined output schemas
|
|
162
|
-
*/
|
|
163
|
-
export class ResponseValidator {
|
|
164
|
-
// Ajv instance - using Ajv.default for ESM/CJS interop
|
|
165
|
-
private ajv: Ajv.default;
|
|
166
|
-
private validators: Map<string, ValidateFunction> = new Map();
|
|
167
|
-
|
|
168
|
-
constructor() {
|
|
169
|
-
// Ajv exports differ between ESM and CJS - handle both patterns
|
|
170
|
-
const AjvClass = (Ajv as unknown as { default: typeof Ajv.default }).default ?? Ajv.default;
|
|
171
|
-
this.ajv = new AjvClass({
|
|
172
|
-
allErrors: true,
|
|
173
|
-
verbose: true,
|
|
174
|
-
strict: true // Enforce strict schema validation
|
|
175
|
-
});
|
|
176
|
-
}
|
|
177
|
-
|
|
178
|
-
/**
|
|
179
|
-
* Register a tool's output schema for validation
|
|
180
|
-
*/
|
|
181
|
-
registerSchema(toolName: string, outputSchema: any) {
|
|
182
|
-
if (!outputSchema) {
|
|
183
|
-
log.warn(`No output schema defined for tool: ${toolName}`);
|
|
184
|
-
return;
|
|
185
|
-
}
|
|
186
|
-
|
|
187
|
-
try {
|
|
188
|
-
const validator = this.ajv.compile(outputSchema);
|
|
189
|
-
this.validators.set(toolName, validator);
|
|
190
|
-
// Demote per-tool schema registration to debug to reduce log noise
|
|
191
|
-
log.debug(`Registered output schema for tool: ${toolName}`);
|
|
192
|
-
} catch (_error) {
|
|
193
|
-
log.error(`Failed to compile output schema for ${toolName}:`, _error);
|
|
194
|
-
}
|
|
195
|
-
}
|
|
196
|
-
|
|
197
|
-
/**
|
|
198
|
-
* Validate a tool's response against its schema
|
|
199
|
-
*/
|
|
200
|
-
async validateResponse(toolName: string, response: any): Promise<{
|
|
201
|
-
valid: boolean;
|
|
202
|
-
errors?: string[];
|
|
203
|
-
structuredContent?: any;
|
|
204
|
-
}> {
|
|
205
|
-
const validator = this.validators.get(toolName);
|
|
206
|
-
|
|
207
|
-
if (!validator) {
|
|
208
|
-
log.debug(`No validator found for tool: ${toolName}`);
|
|
209
|
-
return { valid: true }; // Pass through if no schema defined
|
|
210
|
-
}
|
|
211
|
-
|
|
212
|
-
// Extract structured content from response
|
|
213
|
-
let structuredContent = response;
|
|
214
|
-
|
|
215
|
-
// If response has MCP format with content array
|
|
216
|
-
if (response.content && Array.isArray(response.content)) {
|
|
217
|
-
// Try to extract structured data from text content
|
|
218
|
-
const textContent = response.content.find((c: any) => c.type === 'text');
|
|
219
|
-
if (textContent?.text) {
|
|
220
|
-
const rawText = String(textContent.text);
|
|
221
|
-
const trimmed = rawText.trim();
|
|
222
|
-
const looksLikeJson = trimmed.startsWith('{') || trimmed.startsWith('[');
|
|
223
|
-
|
|
224
|
-
if (looksLikeJson) {
|
|
225
|
-
try {
|
|
226
|
-
// Check if text is JSON - use WASM for high-performance parsing (5-8x faster)
|
|
227
|
-
structuredContent = await wasmIntegration.parseProperties(rawText);
|
|
228
|
-
} catch {
|
|
229
|
-
// If JSON parsing fails, fall back to using the full response without
|
|
230
|
-
// emitting noisy parse errors for plain-text messages.
|
|
231
|
-
structuredContent = response;
|
|
232
|
-
}
|
|
233
|
-
} else {
|
|
234
|
-
// Plain-text summary or error message; keep the original structured
|
|
235
|
-
// response object instead of attempting JSON parsing.
|
|
236
|
-
structuredContent = response;
|
|
237
|
-
}
|
|
238
|
-
}
|
|
239
|
-
}
|
|
240
|
-
|
|
241
|
-
const valid = validator(structuredContent);
|
|
242
|
-
|
|
243
|
-
if (!valid) {
|
|
244
|
-
const errors = validator.errors?.map((err: any) =>
|
|
245
|
-
`${err.instancePath || 'root'}: ${err.message}`
|
|
246
|
-
);
|
|
247
|
-
|
|
248
|
-
log.warn(`Response validation failed for ${toolName}:`, errors);
|
|
249
|
-
|
|
250
|
-
return {
|
|
251
|
-
valid: false,
|
|
252
|
-
errors,
|
|
253
|
-
structuredContent
|
|
254
|
-
};
|
|
255
|
-
}
|
|
256
|
-
|
|
257
|
-
return {
|
|
258
|
-
valid: true,
|
|
259
|
-
structuredContent
|
|
260
|
-
};
|
|
261
|
-
}
|
|
262
|
-
|
|
263
|
-
/**
|
|
264
|
-
* Wrap a tool response with validation and MCP-compliant content shape.
|
|
265
|
-
*
|
|
266
|
-
* MCP tools/call responses must contain a `content` array. Many internal
|
|
267
|
-
* handlers return structured JSON objects (e.g., { success, message, ... }).
|
|
268
|
-
* This wrapper serializes such objects into a single text block while keeping
|
|
269
|
-
* existing `content` responses intact.
|
|
270
|
-
*/
|
|
271
|
-
async wrapResponse(toolName: string, response: any): Promise<any> {
|
|
272
|
-
// Ensure response is safe to serialize first
|
|
273
|
-
try {
|
|
274
|
-
if (response && typeof response === 'object') {
|
|
275
|
-
JSON.stringify(response);
|
|
276
|
-
}
|
|
277
|
-
} catch (_error) {
|
|
278
|
-
log.error(`Response for ${toolName} contains circular references, cleaning...`);
|
|
279
|
-
response = cleanObject(response);
|
|
280
|
-
}
|
|
281
|
-
|
|
282
|
-
// If handler already returned MCP content, keep it as-is (still validate)
|
|
283
|
-
const alreadyMcpShaped = response && typeof response === 'object' && Array.isArray(response.content);
|
|
284
|
-
|
|
285
|
-
// Choose the payload to validate: if already MCP-shaped, validate the
|
|
286
|
-
// structured content extracted from text; otherwise validate the object directly.
|
|
287
|
-
const validation = await this.validateResponse(toolName, response);
|
|
288
|
-
const structuredPayload = validation.structuredContent;
|
|
289
|
-
|
|
290
|
-
if (!validation.valid) {
|
|
291
|
-
log.warn(`Tool ${toolName} response validation failed:`, validation.errors);
|
|
292
|
-
}
|
|
293
|
-
|
|
294
|
-
// If it's already MCP-shaped, return as-is (optionally append validation meta)
|
|
295
|
-
if (alreadyMcpShaped) {
|
|
296
|
-
if (structuredPayload !== undefined && response && typeof response === 'object' && (response as any).structuredContent === undefined) {
|
|
297
|
-
try {
|
|
298
|
-
(response as any).structuredContent = structuredPayload && typeof structuredPayload === 'object'
|
|
299
|
-
? cleanObject(structuredPayload)
|
|
300
|
-
: structuredPayload;
|
|
301
|
-
} catch { }
|
|
302
|
-
}
|
|
303
|
-
// Promote failure semantics to top-level isError when obvious
|
|
304
|
-
try {
|
|
305
|
-
const sc: any = (response as any).structuredContent || structuredPayload || {};
|
|
306
|
-
const hasExplicitFailure = (typeof sc.success === 'boolean' && sc.success === false) || (typeof sc.error === 'string' && sc.error.length > 0);
|
|
307
|
-
if (hasExplicitFailure && (response as any).isError !== true) {
|
|
308
|
-
(response as any).isError = true;
|
|
309
|
-
}
|
|
310
|
-
} catch { }
|
|
311
|
-
if (!validation.valid) {
|
|
312
|
-
try {
|
|
313
|
-
(response as any)._validation = { valid: false, errors: validation.errors };
|
|
314
|
-
} catch { }
|
|
315
|
-
}
|
|
316
|
-
return response;
|
|
317
|
-
}
|
|
318
|
-
|
|
319
|
-
// Otherwise, wrap structured result into MCP content
|
|
320
|
-
const summarySource = structuredPayload !== undefined ? structuredPayload : response;
|
|
321
|
-
let text = buildSummaryText(toolName, summarySource);
|
|
322
|
-
if (!text || !text.trim()) {
|
|
323
|
-
text = buildSummaryText(toolName, response);
|
|
324
|
-
}
|
|
325
|
-
|
|
326
|
-
const wrapped = {
|
|
327
|
-
content: [
|
|
328
|
-
{ type: 'text', text }
|
|
329
|
-
]
|
|
330
|
-
} as any;
|
|
331
|
-
|
|
332
|
-
// Surface a top-level success flag when available so clients and test
|
|
333
|
-
// harnesses do not have to infer success from the absence of isError.
|
|
334
|
-
try {
|
|
335
|
-
if (structuredPayload && typeof (structuredPayload as any).success === 'boolean') {
|
|
336
|
-
(wrapped as any).success = Boolean((structuredPayload as any).success);
|
|
337
|
-
} else if (response && typeof (response as any).success === 'boolean') {
|
|
338
|
-
(wrapped as any).success = Boolean((response as any).success);
|
|
339
|
-
}
|
|
340
|
-
} catch { }
|
|
341
|
-
|
|
342
|
-
if (structuredPayload !== undefined) {
|
|
343
|
-
try {
|
|
344
|
-
wrapped.structuredContent = structuredPayload && typeof structuredPayload === 'object'
|
|
345
|
-
? cleanObject(structuredPayload)
|
|
346
|
-
: structuredPayload;
|
|
347
|
-
} catch {
|
|
348
|
-
wrapped.structuredContent = structuredPayload;
|
|
349
|
-
}
|
|
350
|
-
} else if (response && typeof response === 'object') {
|
|
351
|
-
try {
|
|
352
|
-
wrapped.structuredContent = cleanObject(response);
|
|
353
|
-
} catch {
|
|
354
|
-
wrapped.structuredContent = response;
|
|
355
|
-
}
|
|
356
|
-
}
|
|
357
|
-
|
|
358
|
-
// Promote failure semantics to top-level isError when obvious
|
|
359
|
-
try {
|
|
360
|
-
const sc: any = wrapped.structuredContent || {};
|
|
361
|
-
const hasExplicitFailure = (typeof sc.success === 'boolean' && sc.success === false) || (typeof sc.error === 'string' && sc.error.length > 0);
|
|
362
|
-
if (hasExplicitFailure) {
|
|
363
|
-
wrapped.isError = true;
|
|
364
|
-
}
|
|
365
|
-
} catch { }
|
|
366
|
-
|
|
367
|
-
if (!validation.valid) {
|
|
368
|
-
wrapped._validation = { valid: false, errors: validation.errors };
|
|
369
|
-
}
|
|
370
|
-
|
|
371
|
-
// Mark explicit error when success is false to avoid false positives in
|
|
372
|
-
// clients that check only for the absence of isError.
|
|
373
|
-
try {
|
|
374
|
-
const s = (wrapped as any).success;
|
|
375
|
-
if (typeof s === 'boolean' && s === false) {
|
|
376
|
-
(wrapped as any).isError = true;
|
|
377
|
-
}
|
|
378
|
-
} catch { }
|
|
379
|
-
|
|
380
|
-
return wrapped;
|
|
381
|
-
}
|
|
382
|
-
|
|
383
|
-
/**
|
|
384
|
-
* Get validation statistics
|
|
385
|
-
*/
|
|
386
|
-
getStats() {
|
|
387
|
-
return {
|
|
388
|
-
totalSchemas: this.validators.size,
|
|
389
|
-
tools: Array.from(this.validators.keys())
|
|
390
|
-
};
|
|
391
|
-
}
|
|
392
|
-
}
|
|
393
|
-
|
|
394
|
-
// Singleton instance
|
|
395
|
-
export const responseValidator = new ResponseValidator();
|
|
@@ -1,195 +0,0 @@
|
|
|
1
|
-
// Generic result interpretation for automation bridge responses
|
|
2
|
-
export interface InterpretedStandardResult {
|
|
3
|
-
success: boolean;
|
|
4
|
-
message: string;
|
|
5
|
-
error?: string;
|
|
6
|
-
warnings?: string[];
|
|
7
|
-
details?: string[];
|
|
8
|
-
payload: Record<string, unknown>;
|
|
9
|
-
cleanText?: string;
|
|
10
|
-
rawText: string;
|
|
11
|
-
raw: unknown;
|
|
12
|
-
}
|
|
13
|
-
|
|
14
|
-
/** Interprets automation bridge responses into a standard format */
|
|
15
|
-
export function interpretStandardResult(
|
|
16
|
-
response: unknown,
|
|
17
|
-
defaults: { successMessage: string; failureMessage: string }
|
|
18
|
-
): InterpretedStandardResult {
|
|
19
|
-
// Handle automation bridge response format
|
|
20
|
-
const payload = (response && typeof response === 'object' ? response : {}) as Record<string, unknown>;
|
|
21
|
-
const success = payload.success === true;
|
|
22
|
-
const rawText = typeof payload.message === 'string' ? payload.message :
|
|
23
|
-
typeof payload.output === 'string' ? payload.output :
|
|
24
|
-
String(payload.result ?? '');
|
|
25
|
-
|
|
26
|
-
const messageFromPayload = typeof payload.message === 'string' ? payload.message.trim() : '';
|
|
27
|
-
const errorFromPayload = typeof payload.error === 'string' ? payload.error.trim() : '';
|
|
28
|
-
|
|
29
|
-
const message = messageFromPayload || (success ? defaults.successMessage : defaults.failureMessage);
|
|
30
|
-
const error = success ? undefined : errorFromPayload || messageFromPayload || defaults.failureMessage;
|
|
31
|
-
|
|
32
|
-
return {
|
|
33
|
-
success,
|
|
34
|
-
message,
|
|
35
|
-
error,
|
|
36
|
-
warnings: coerceStringArray(payload.warnings),
|
|
37
|
-
details: coerceStringArray(payload.details),
|
|
38
|
-
payload,
|
|
39
|
-
cleanText: rawText || undefined,
|
|
40
|
-
rawText,
|
|
41
|
-
raw: response
|
|
42
|
-
};
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
/** Cleans result text by removing tags */
|
|
46
|
-
export function cleanResultText(
|
|
47
|
-
text: string | undefined,
|
|
48
|
-
options: { tag?: string; defaultValue?: string } = {}
|
|
49
|
-
): string | undefined {
|
|
50
|
-
const { defaultValue } = options;
|
|
51
|
-
if (!text) {
|
|
52
|
-
return defaultValue;
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
const cleaned = text.trim();
|
|
56
|
-
if (cleaned.length > 0) {
|
|
57
|
-
return cleaned;
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
return defaultValue;
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
export function bestEffortInterpretedText(
|
|
64
|
-
interpreted: Pick<InterpretedStandardResult, 'cleanText' | 'rawText'>,
|
|
65
|
-
defaultValue?: string
|
|
66
|
-
): string | undefined {
|
|
67
|
-
const cleaned = interpreted.cleanText?.trim();
|
|
68
|
-
if (cleaned) {
|
|
69
|
-
return cleaned;
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
const raw = interpreted.rawText?.trim?.();
|
|
73
|
-
if (raw) {
|
|
74
|
-
return raw;
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
return defaultValue;
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
export function coerceString(value: unknown): string | undefined {
|
|
81
|
-
if (typeof value === 'string') {
|
|
82
|
-
const trimmed = value.trim();
|
|
83
|
-
return trimmed.length > 0 ? trimmed : undefined;
|
|
84
|
-
}
|
|
85
|
-
return undefined;
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
export function coerceStringArray(value: unknown): string[] | undefined {
|
|
89
|
-
if (!Array.isArray(value)) {
|
|
90
|
-
return undefined;
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
const items: string[] = [];
|
|
94
|
-
for (const entry of value) {
|
|
95
|
-
if (typeof entry !== 'string') {
|
|
96
|
-
continue;
|
|
97
|
-
}
|
|
98
|
-
const trimmed = entry.trim();
|
|
99
|
-
if (!trimmed) {
|
|
100
|
-
continue;
|
|
101
|
-
}
|
|
102
|
-
items.push(trimmed);
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
return items.length > 0 ? items : undefined;
|
|
106
|
-
}
|
|
107
|
-
|
|
108
|
-
export function coerceBoolean(value: unknown, defaultValue?: boolean): boolean | undefined {
|
|
109
|
-
if (typeof value === 'boolean') {
|
|
110
|
-
return value;
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
if (typeof value === 'string') {
|
|
114
|
-
const normalized = value.trim().toLowerCase();
|
|
115
|
-
if (['true', '1', 'yes', 'on'].includes(normalized)) {
|
|
116
|
-
return true;
|
|
117
|
-
}
|
|
118
|
-
if (['false', '0', 'no', 'off'].includes(normalized)) {
|
|
119
|
-
return false;
|
|
120
|
-
}
|
|
121
|
-
}
|
|
122
|
-
|
|
123
|
-
if (typeof value === 'number') {
|
|
124
|
-
if (value === 1) {
|
|
125
|
-
return true;
|
|
126
|
-
}
|
|
127
|
-
if (value === 0) {
|
|
128
|
-
return false;
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
return defaultValue;
|
|
133
|
-
}
|
|
134
|
-
|
|
135
|
-
export function coerceNumber(value: unknown): number | undefined {
|
|
136
|
-
if (typeof value === 'number' && Number.isFinite(value)) {
|
|
137
|
-
return value;
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
if (typeof value === 'string') {
|
|
141
|
-
const parsed = Number(value);
|
|
142
|
-
if (Number.isFinite(parsed)) {
|
|
143
|
-
return parsed;
|
|
144
|
-
}
|
|
145
|
-
}
|
|
146
|
-
|
|
147
|
-
return undefined;
|
|
148
|
-
}
|
|
149
|
-
|
|
150
|
-
export function coerceVector3(value: unknown): [number, number, number] | undefined {
|
|
151
|
-
const toNumber = (entry: unknown): number | undefined => {
|
|
152
|
-
if (typeof entry === 'number' && Number.isFinite(entry)) {
|
|
153
|
-
return entry;
|
|
154
|
-
}
|
|
155
|
-
if (typeof entry === 'string') {
|
|
156
|
-
const parsed = Number(entry.trim());
|
|
157
|
-
if (Number.isFinite(parsed)) {
|
|
158
|
-
return parsed;
|
|
159
|
-
}
|
|
160
|
-
}
|
|
161
|
-
return undefined;
|
|
162
|
-
};
|
|
163
|
-
|
|
164
|
-
const fromArray = (arr: unknown[]): [number, number, number] | undefined => {
|
|
165
|
-
if (arr.length !== 3) {
|
|
166
|
-
return undefined;
|
|
167
|
-
}
|
|
168
|
-
const mapped = arr.map(toNumber);
|
|
169
|
-
if (mapped.every((entry): entry is number => typeof entry === 'number')) {
|
|
170
|
-
return mapped as [number, number, number];
|
|
171
|
-
}
|
|
172
|
-
return undefined;
|
|
173
|
-
};
|
|
174
|
-
|
|
175
|
-
if (Array.isArray(value)) {
|
|
176
|
-
const parsed = fromArray(value);
|
|
177
|
-
if (parsed) {
|
|
178
|
-
return parsed;
|
|
179
|
-
}
|
|
180
|
-
}
|
|
181
|
-
|
|
182
|
-
if (value && typeof value === 'object') {
|
|
183
|
-
const obj = value as Record<string, unknown>;
|
|
184
|
-
const candidate = fromArray([obj.x, obj.y, obj.z]);
|
|
185
|
-
if (candidate) {
|
|
186
|
-
return candidate;
|
|
187
|
-
}
|
|
188
|
-
const alternate = fromArray([obj.pitch, obj.yaw, obj.roll]);
|
|
189
|
-
if (alternate) {
|
|
190
|
-
return alternate;
|
|
191
|
-
}
|
|
192
|
-
}
|
|
193
|
-
|
|
194
|
-
return undefined;
|
|
195
|
-
}
|
|
@@ -1,90 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Unit tests for safe-json utility
|
|
3
|
-
*/
|
|
4
|
-
import { describe, it, expect } from 'vitest';
|
|
5
|
-
import { cleanObject } from './safe-json.js';
|
|
6
|
-
|
|
7
|
-
describe('cleanObject', () => {
|
|
8
|
-
it('removes undefined values', () => {
|
|
9
|
-
const input = { a: 1, b: undefined, c: 3 };
|
|
10
|
-
const result = cleanObject(input);
|
|
11
|
-
expect(result).toEqual({ a: 1, c: 3 });
|
|
12
|
-
expect('b' in result).toBe(false);
|
|
13
|
-
});
|
|
14
|
-
|
|
15
|
-
it('preserves null values (only removes undefined)', () => {
|
|
16
|
-
const input = { a: 1, b: null, c: 3 };
|
|
17
|
-
const result = cleanObject(input);
|
|
18
|
-
// cleanObject preserves null, only removes undefined
|
|
19
|
-
expect(result).toEqual({ a: 1, b: null, c: 3 });
|
|
20
|
-
});
|
|
21
|
-
|
|
22
|
-
it('handles nested objects', () => {
|
|
23
|
-
const input = { a: 1, nested: { b: 2, c: undefined } };
|
|
24
|
-
const result = cleanObject(input);
|
|
25
|
-
expect(result.nested).toEqual({ b: 2 });
|
|
26
|
-
});
|
|
27
|
-
|
|
28
|
-
it('handles arrays with undefined (preserves null)', () => {
|
|
29
|
-
const input = { arr: [1, undefined, 3, null] };
|
|
30
|
-
const result = cleanObject(input);
|
|
31
|
-
// Arrays preserve their structure, undefined becomes undefined in array
|
|
32
|
-
expect(result.arr).toEqual([1, undefined, 3, null]);
|
|
33
|
-
});
|
|
34
|
-
|
|
35
|
-
it('preserves falsy values that are not null/undefined', () => {
|
|
36
|
-
const input = { a: 0, b: '', c: false };
|
|
37
|
-
const result = cleanObject(input);
|
|
38
|
-
expect(result).toEqual({ a: 0, b: '', c: false });
|
|
39
|
-
});
|
|
40
|
-
|
|
41
|
-
it('handles empty objects', () => {
|
|
42
|
-
expect(cleanObject({})).toEqual({});
|
|
43
|
-
});
|
|
44
|
-
|
|
45
|
-
it('handles primitive inputs', () => {
|
|
46
|
-
expect(cleanObject('string')).toBe('string');
|
|
47
|
-
expect(cleanObject(42)).toBe(42);
|
|
48
|
-
expect(cleanObject(true)).toBe(true);
|
|
49
|
-
});
|
|
50
|
-
|
|
51
|
-
it('respects max depth limit', () => {
|
|
52
|
-
// Create deeply nested object
|
|
53
|
-
let deep: any = { value: 'deep' };
|
|
54
|
-
for (let i = 0; i < 15; i++) {
|
|
55
|
-
deep = { nested: deep };
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
// Should not throw when hitting depth limit
|
|
59
|
-
expect(() => cleanObject(deep, 10)).not.toThrow();
|
|
60
|
-
});
|
|
61
|
-
|
|
62
|
-
it('handles circular reference prevention at max depth', () => {
|
|
63
|
-
const obj: any = { a: 1 };
|
|
64
|
-
obj.self = obj; // circular reference
|
|
65
|
-
|
|
66
|
-
// Should not throw - depth limiting should prevent infinite recursion
|
|
67
|
-
expect(() => cleanObject(obj, 5)).not.toThrow();
|
|
68
|
-
});
|
|
69
|
-
|
|
70
|
-
it('handles Date objects (converts to empty object)', () => {
|
|
71
|
-
const date = new Date('2024-01-01');
|
|
72
|
-
const input = { created: date };
|
|
73
|
-
const result = cleanObject(input);
|
|
74
|
-
// Date objects are processed as objects using Object.keys, which returns []
|
|
75
|
-
expect(result.created).toEqual({});
|
|
76
|
-
});
|
|
77
|
-
|
|
78
|
-
it('handles empty arrays', () => {
|
|
79
|
-
const input = { arr: [] };
|
|
80
|
-
const result = cleanObject(input);
|
|
81
|
-
expect(result.arr).toEqual([]);
|
|
82
|
-
});
|
|
83
|
-
|
|
84
|
-
it('handles NaN values', () => {
|
|
85
|
-
const input = { a: 1, b: NaN };
|
|
86
|
-
const result = cleanObject(input);
|
|
87
|
-
// NaN handling depends on implementation
|
|
88
|
-
expect(result.a).toBe(1);
|
|
89
|
-
});
|
|
90
|
-
});
|