unreal-engine-mcp-server 0.4.7 → 0.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (438) hide show
  1. package/.env.example +26 -0
  2. package/.env.production +38 -7
  3. package/.eslintrc.json +0 -54
  4. package/.eslintrc.override.json +8 -0
  5. package/.github/ISSUE_TEMPLATE/bug_report.yml +94 -0
  6. package/.github/ISSUE_TEMPLATE/config.yml +8 -0
  7. package/.github/ISSUE_TEMPLATE/feature_request.yml +56 -0
  8. package/.github/copilot-instructions.md +478 -45
  9. package/.github/dependabot.yml +19 -0
  10. package/.github/labeler.yml +24 -0
  11. package/.github/labels.yml +70 -0
  12. package/.github/pull_request_template.md +42 -0
  13. package/.github/release-drafter.yml +148 -0
  14. package/.github/workflows/auto-merge.yml +38 -0
  15. package/.github/workflows/ci.yml +38 -0
  16. package/.github/workflows/dependency-review.yml +17 -0
  17. package/.github/workflows/gemini-issue-triage.yml +172 -0
  18. package/.github/workflows/greetings.yml +23 -0
  19. package/.github/workflows/labeler.yml +16 -0
  20. package/.github/workflows/links.yml +80 -0
  21. package/.github/workflows/pr-size-labeler.yml +137 -0
  22. package/.github/workflows/publish-mcp.yml +12 -7
  23. package/.github/workflows/release-drafter.yml +23 -0
  24. package/.github/workflows/release.yml +112 -0
  25. package/.github/workflows/semantic-pull-request.yml +35 -0
  26. package/.github/workflows/smoke-test.yml +36 -0
  27. package/.github/workflows/stale.yml +28 -0
  28. package/CHANGELOG.md +267 -31
  29. package/CONTRIBUTING.md +140 -0
  30. package/README.md +166 -71
  31. package/claude_desktop_config_example.json +7 -6
  32. package/dist/automation/bridge.d.ts +50 -0
  33. package/dist/automation/bridge.js +452 -0
  34. package/dist/automation/connection-manager.d.ts +23 -0
  35. package/dist/automation/connection-manager.js +107 -0
  36. package/dist/automation/handshake.d.ts +11 -0
  37. package/dist/automation/handshake.js +89 -0
  38. package/dist/automation/index.d.ts +3 -0
  39. package/dist/automation/index.js +3 -0
  40. package/dist/automation/message-handler.d.ts +12 -0
  41. package/dist/automation/message-handler.js +149 -0
  42. package/dist/automation/request-tracker.d.ts +25 -0
  43. package/dist/automation/request-tracker.js +98 -0
  44. package/dist/automation/types.d.ts +130 -0
  45. package/dist/automation/types.js +2 -0
  46. package/dist/cli.js +32 -5
  47. package/dist/config.d.ts +27 -0
  48. package/dist/config.js +60 -0
  49. package/dist/constants.d.ts +12 -0
  50. package/dist/constants.js +12 -0
  51. package/dist/graphql/resolvers.d.ts +268 -0
  52. package/dist/graphql/resolvers.js +743 -0
  53. package/dist/graphql/schema.d.ts +5 -0
  54. package/dist/graphql/schema.js +437 -0
  55. package/dist/graphql/server.d.ts +26 -0
  56. package/dist/graphql/server.js +115 -0
  57. package/dist/graphql/types.d.ts +7 -0
  58. package/dist/graphql/types.js +2 -0
  59. package/dist/handlers/resource-handlers.d.ts +20 -0
  60. package/dist/handlers/resource-handlers.js +180 -0
  61. package/dist/index.d.ts +31 -18
  62. package/dist/index.js +119 -619
  63. package/dist/prompts/index.js +4 -4
  64. package/dist/resources/actors.d.ts +17 -12
  65. package/dist/resources/actors.js +56 -76
  66. package/dist/resources/assets.d.ts +6 -14
  67. package/dist/resources/assets.js +115 -147
  68. package/dist/resources/levels.d.ts +13 -13
  69. package/dist/resources/levels.js +25 -34
  70. package/dist/server/resource-registry.d.ts +20 -0
  71. package/dist/server/resource-registry.js +37 -0
  72. package/dist/server/tool-registry.d.ts +23 -0
  73. package/dist/server/tool-registry.js +322 -0
  74. package/dist/server-setup.d.ts +21 -0
  75. package/dist/server-setup.js +111 -0
  76. package/dist/services/health-monitor.d.ts +34 -0
  77. package/dist/services/health-monitor.js +105 -0
  78. package/dist/services/metrics-server.d.ts +11 -0
  79. package/dist/services/metrics-server.js +105 -0
  80. package/dist/tools/actors.d.ts +147 -9
  81. package/dist/tools/actors.js +350 -311
  82. package/dist/tools/animation.d.ts +135 -4
  83. package/dist/tools/animation.js +510 -411
  84. package/dist/tools/assets.d.ts +117 -19
  85. package/dist/tools/assets.js +259 -284
  86. package/dist/tools/audio.d.ts +102 -42
  87. package/dist/tools/audio.js +272 -685
  88. package/dist/tools/base-tool.d.ts +17 -0
  89. package/dist/tools/base-tool.js +46 -0
  90. package/dist/tools/behavior-tree.d.ts +94 -0
  91. package/dist/tools/behavior-tree.js +39 -0
  92. package/dist/tools/blueprint/helpers.d.ts +29 -0
  93. package/dist/tools/blueprint/helpers.js +182 -0
  94. package/dist/tools/blueprint.d.ts +228 -118
  95. package/dist/tools/blueprint.js +685 -832
  96. package/dist/tools/consolidated-tool-definitions.d.ts +5462 -1781
  97. package/dist/tools/consolidated-tool-definitions.js +829 -496
  98. package/dist/tools/consolidated-tool-handlers.d.ts +2 -1
  99. package/dist/tools/consolidated-tool-handlers.js +211 -1026
  100. package/dist/tools/debug.d.ts +143 -85
  101. package/dist/tools/debug.js +234 -180
  102. package/dist/tools/dynamic-handler-registry.d.ts +11 -0
  103. package/dist/tools/dynamic-handler-registry.js +101 -0
  104. package/dist/tools/editor.d.ts +139 -18
  105. package/dist/tools/editor.js +239 -244
  106. package/dist/tools/engine.d.ts +10 -4
  107. package/dist/tools/engine.js +13 -5
  108. package/dist/tools/environment.d.ts +36 -0
  109. package/dist/tools/environment.js +267 -0
  110. package/dist/tools/foliage.d.ts +105 -14
  111. package/dist/tools/foliage.js +219 -331
  112. package/dist/tools/handlers/actor-handlers.d.ts +3 -0
  113. package/dist/tools/handlers/actor-handlers.js +232 -0
  114. package/dist/tools/handlers/animation-handlers.d.ts +3 -0
  115. package/dist/tools/handlers/animation-handlers.js +185 -0
  116. package/dist/tools/handlers/argument-helper.d.ts +16 -0
  117. package/dist/tools/handlers/argument-helper.js +80 -0
  118. package/dist/tools/handlers/asset-handlers.d.ts +3 -0
  119. package/dist/tools/handlers/asset-handlers.js +496 -0
  120. package/dist/tools/handlers/audio-handlers.d.ts +3 -0
  121. package/dist/tools/handlers/audio-handlers.js +166 -0
  122. package/dist/tools/handlers/blueprint-handlers.d.ts +4 -0
  123. package/dist/tools/handlers/blueprint-handlers.js +358 -0
  124. package/dist/tools/handlers/common-handlers.d.ts +14 -0
  125. package/dist/tools/handlers/common-handlers.js +56 -0
  126. package/dist/tools/handlers/editor-handlers.d.ts +3 -0
  127. package/dist/tools/handlers/editor-handlers.js +119 -0
  128. package/dist/tools/handlers/effect-handlers.d.ts +3 -0
  129. package/dist/tools/handlers/effect-handlers.js +171 -0
  130. package/dist/tools/handlers/environment-handlers.d.ts +3 -0
  131. package/dist/tools/handlers/environment-handlers.js +170 -0
  132. package/dist/tools/handlers/graph-handlers.d.ts +3 -0
  133. package/dist/tools/handlers/graph-handlers.js +90 -0
  134. package/dist/tools/handlers/input-handlers.d.ts +3 -0
  135. package/dist/tools/handlers/input-handlers.js +21 -0
  136. package/dist/tools/handlers/inspect-handlers.d.ts +3 -0
  137. package/dist/tools/handlers/inspect-handlers.js +383 -0
  138. package/dist/tools/handlers/level-handlers.d.ts +3 -0
  139. package/dist/tools/handlers/level-handlers.js +237 -0
  140. package/dist/tools/handlers/lighting-handlers.d.ts +3 -0
  141. package/dist/tools/handlers/lighting-handlers.js +144 -0
  142. package/dist/tools/handlers/performance-handlers.d.ts +3 -0
  143. package/dist/tools/handlers/performance-handlers.js +130 -0
  144. package/dist/tools/handlers/pipeline-handlers.d.ts +3 -0
  145. package/dist/tools/handlers/pipeline-handlers.js +110 -0
  146. package/dist/tools/handlers/sequence-handlers.d.ts +3 -0
  147. package/dist/tools/handlers/sequence-handlers.js +376 -0
  148. package/dist/tools/handlers/system-handlers.d.ts +4 -0
  149. package/dist/tools/handlers/system-handlers.js +506 -0
  150. package/dist/tools/input.d.ts +19 -0
  151. package/dist/tools/input.js +89 -0
  152. package/dist/tools/introspection.d.ts +103 -40
  153. package/dist/tools/introspection.js +425 -568
  154. package/dist/tools/landscape.d.ts +97 -36
  155. package/dist/tools/landscape.js +280 -409
  156. package/dist/tools/level.d.ts +130 -10
  157. package/dist/tools/level.js +639 -675
  158. package/dist/tools/lighting.d.ts +77 -38
  159. package/dist/tools/lighting.js +441 -943
  160. package/dist/tools/logs.d.ts +3 -3
  161. package/dist/tools/logs.js +5 -57
  162. package/dist/tools/materials.d.ts +91 -24
  163. package/dist/tools/materials.js +190 -118
  164. package/dist/tools/niagara.d.ts +149 -39
  165. package/dist/tools/niagara.js +232 -182
  166. package/dist/tools/performance.d.ts +27 -12
  167. package/dist/tools/performance.js +204 -122
  168. package/dist/tools/physics.d.ts +32 -77
  169. package/dist/tools/physics.js +171 -582
  170. package/dist/tools/property-dictionary.d.ts +13 -0
  171. package/dist/tools/property-dictionary.js +82 -0
  172. package/dist/tools/sequence.d.ts +73 -48
  173. package/dist/tools/sequence.js +196 -748
  174. package/dist/tools/tool-definition-utils.d.ts +59 -0
  175. package/dist/tools/tool-definition-utils.js +35 -0
  176. package/dist/tools/ui.d.ts +66 -34
  177. package/dist/tools/ui.js +134 -214
  178. package/dist/types/env.d.ts +0 -3
  179. package/dist/types/env.js +0 -7
  180. package/dist/types/tool-interfaces.d.ts +898 -0
  181. package/dist/types/tool-interfaces.js +2 -0
  182. package/dist/types/tool-types.d.ts +183 -19
  183. package/dist/types/tool-types.js +0 -4
  184. package/dist/unreal-bridge.d.ts +24 -131
  185. package/dist/unreal-bridge.js +364 -1506
  186. package/dist/utils/command-validator.d.ts +9 -0
  187. package/dist/utils/command-validator.js +67 -0
  188. package/dist/utils/elicitation.d.ts +1 -1
  189. package/dist/utils/elicitation.js +12 -15
  190. package/dist/utils/error-handler.d.ts +2 -51
  191. package/dist/utils/error-handler.js +11 -87
  192. package/dist/utils/ini-reader.d.ts +3 -0
  193. package/dist/utils/ini-reader.js +69 -0
  194. package/dist/utils/logger.js +9 -6
  195. package/dist/utils/normalize.d.ts +3 -0
  196. package/dist/utils/normalize.js +56 -0
  197. package/dist/utils/response-factory.d.ts +7 -0
  198. package/dist/utils/response-factory.js +33 -0
  199. package/dist/utils/response-validator.d.ts +3 -24
  200. package/dist/utils/response-validator.js +130 -81
  201. package/dist/utils/result-helpers.d.ts +4 -5
  202. package/dist/utils/result-helpers.js +15 -16
  203. package/dist/utils/safe-json.js +5 -11
  204. package/dist/utils/unreal-command-queue.d.ts +24 -0
  205. package/dist/utils/unreal-command-queue.js +120 -0
  206. package/dist/utils/validation.d.ts +0 -40
  207. package/dist/utils/validation.js +1 -78
  208. package/dist/wasm/index.d.ts +70 -0
  209. package/dist/wasm/index.js +535 -0
  210. package/docs/GraphQL-API.md +888 -0
  211. package/docs/Migration-Guide-v0.5.0.md +692 -0
  212. package/docs/Roadmap.md +53 -0
  213. package/docs/WebAssembly-Integration.md +628 -0
  214. package/docs/editor-plugin-extension.md +370 -0
  215. package/docs/handler-mapping.md +242 -0
  216. package/docs/native-automation-progress.md +128 -0
  217. package/docs/testing-guide.md +423 -0
  218. package/mcp-config-example.json +6 -6
  219. package/package.json +60 -27
  220. package/plugins/McpAutomationBridge/Config/FilterPlugin.ini +8 -0
  221. package/plugins/McpAutomationBridge/McpAutomationBridge.uplugin +64 -0
  222. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/McpAutomationBridge.Build.cs +189 -0
  223. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeGlobals.cpp +22 -0
  224. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeGlobals.h +30 -0
  225. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeHelpers.h +1983 -0
  226. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeModule.cpp +72 -0
  227. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeSettings.cpp +46 -0
  228. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeSubsystem.cpp +581 -0
  229. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AnimationHandlers.cpp +2394 -0
  230. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AssetQueryHandlers.cpp +300 -0
  231. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AssetWorkflowHandlers.cpp +2807 -0
  232. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AudioHandlers.cpp +1087 -0
  233. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BehaviorTreeHandlers.cpp +488 -0
  234. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintCreationHandlers.cpp +643 -0
  235. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintCreationHandlers.h +31 -0
  236. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintGraphHandlers.cpp +1184 -0
  237. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintHandlers.cpp +5652 -0
  238. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintHandlers_List.cpp +152 -0
  239. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_ControlHandlers.cpp +2614 -0
  240. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_DebugHandlers.cpp +42 -0
  241. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_EditorFunctionHandlers.cpp +1237 -0
  242. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_EffectHandlers.cpp +1701 -0
  243. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_EnvironmentHandlers.cpp +2145 -0
  244. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_FoliageHandlers.cpp +954 -0
  245. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_InputHandlers.cpp +209 -0
  246. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_InsightsHandlers.cpp +41 -0
  247. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LandscapeHandlers.cpp +1164 -0
  248. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LevelHandlers.cpp +762 -0
  249. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LightingHandlers.cpp +634 -0
  250. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LogHandlers.cpp +136 -0
  251. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_MaterialGraphHandlers.cpp +494 -0
  252. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_NiagaraGraphHandlers.cpp +278 -0
  253. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_NiagaraHandlers.cpp +625 -0
  254. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_PerformanceHandlers.cpp +401 -0
  255. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_PipelineHandlers.cpp +67 -0
  256. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_ProcessRequest.cpp +735 -0
  257. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_PropertyHandlers.cpp +2634 -0
  258. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_RenderHandlers.cpp +189 -0
  259. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SCSHandlers.cpp +917 -0
  260. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SCSHandlers.h +39 -0
  261. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SequenceHandlers.cpp +2670 -0
  262. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SequencerHandlers.cpp +519 -0
  263. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_TestHandlers.cpp +38 -0
  264. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_UiHandlers.cpp +668 -0
  265. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_WorldPartitionHandlers.cpp +346 -0
  266. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpBridgeWebSocket.cpp +1330 -0
  267. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpBridgeWebSocket.h +149 -0
  268. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpConnectionManager.cpp +783 -0
  269. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Public/McpAutomationBridgeSettings.h +115 -0
  270. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Public/McpAutomationBridgeSubsystem.h +796 -0
  271. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Public/McpConnectionManager.h +117 -0
  272. package/scripts/check-unreal-connection.mjs +19 -0
  273. package/scripts/clean-tmp.js +23 -0
  274. package/scripts/patch-wasm.js +26 -0
  275. package/scripts/run-all-tests.mjs +131 -0
  276. package/scripts/smoke-test.ts +94 -0
  277. package/scripts/sync-mcp-plugin.js +143 -0
  278. package/scripts/test-no-plugin-alternates.mjs +113 -0
  279. package/scripts/validate-server.js +46 -0
  280. package/scripts/verify-automation-bridge.js +200 -0
  281. package/server.json +57 -21
  282. package/src/automation/bridge.ts +558 -0
  283. package/src/automation/connection-manager.ts +130 -0
  284. package/src/automation/handshake.ts +99 -0
  285. package/src/automation/index.ts +2 -0
  286. package/src/automation/message-handler.ts +167 -0
  287. package/src/automation/request-tracker.ts +123 -0
  288. package/src/automation/types.ts +107 -0
  289. package/src/cli.ts +33 -6
  290. package/src/config.ts +73 -0
  291. package/src/constants.ts +12 -0
  292. package/src/graphql/resolvers.ts +1010 -0
  293. package/src/graphql/schema.ts +452 -0
  294. package/src/graphql/server.ts +154 -0
  295. package/src/graphql/types.ts +7 -0
  296. package/src/handlers/resource-handlers.ts +186 -0
  297. package/src/index.ts +152 -663
  298. package/src/prompts/index.ts +4 -4
  299. package/src/resources/actors.ts +58 -76
  300. package/src/resources/assets.ts +147 -134
  301. package/src/resources/levels.ts +28 -33
  302. package/src/server/resource-registry.ts +47 -0
  303. package/src/server/tool-registry.ts +354 -0
  304. package/src/server-setup.ts +148 -0
  305. package/src/services/health-monitor.ts +132 -0
  306. package/src/services/metrics-server.ts +142 -0
  307. package/src/tools/actors.ts +417 -322
  308. package/src/tools/animation.ts +671 -461
  309. package/src/tools/assets.ts +353 -289
  310. package/src/tools/audio.ts +323 -766
  311. package/src/tools/base-tool.ts +52 -0
  312. package/src/tools/behavior-tree.ts +45 -0
  313. package/src/tools/blueprint/helpers.ts +189 -0
  314. package/src/tools/blueprint.ts +787 -965
  315. package/src/tools/consolidated-tool-definitions.ts +993 -515
  316. package/src/tools/consolidated-tool-handlers.ts +272 -1139
  317. package/src/tools/debug.ts +292 -187
  318. package/src/tools/dynamic-handler-registry.ts +151 -0
  319. package/src/tools/editor.ts +309 -246
  320. package/src/tools/engine.ts +14 -3
  321. package/src/tools/environment.ts +287 -0
  322. package/src/tools/foliage.ts +314 -379
  323. package/src/tools/handlers/actor-handlers.ts +271 -0
  324. package/src/tools/handlers/animation-handlers.ts +237 -0
  325. package/src/tools/handlers/argument-helper.ts +142 -0
  326. package/src/tools/handlers/asset-handlers.ts +532 -0
  327. package/src/tools/handlers/audio-handlers.ts +194 -0
  328. package/src/tools/handlers/blueprint-handlers.ts +380 -0
  329. package/src/tools/handlers/common-handlers.ts +87 -0
  330. package/src/tools/handlers/editor-handlers.ts +123 -0
  331. package/src/tools/handlers/effect-handlers.ts +220 -0
  332. package/src/tools/handlers/environment-handlers.ts +183 -0
  333. package/src/tools/handlers/graph-handlers.ts +116 -0
  334. package/src/tools/handlers/input-handlers.ts +28 -0
  335. package/src/tools/handlers/inspect-handlers.ts +450 -0
  336. package/src/tools/handlers/level-handlers.ts +252 -0
  337. package/src/tools/handlers/lighting-handlers.ts +147 -0
  338. package/src/tools/handlers/performance-handlers.ts +132 -0
  339. package/src/tools/handlers/pipeline-handlers.ts +127 -0
  340. package/src/tools/handlers/sequence-handlers.ts +415 -0
  341. package/src/tools/handlers/system-handlers.ts +564 -0
  342. package/src/tools/input.ts +101 -0
  343. package/src/tools/introspection.ts +493 -584
  344. package/src/tools/landscape.ts +394 -489
  345. package/src/tools/level.ts +752 -694
  346. package/src/tools/lighting.ts +583 -984
  347. package/src/tools/logs.ts +9 -57
  348. package/src/tools/materials.ts +231 -121
  349. package/src/tools/niagara.ts +293 -168
  350. package/src/tools/performance.ts +320 -168
  351. package/src/tools/physics.ts +268 -613
  352. package/src/tools/property-dictionary.ts +98 -0
  353. package/src/tools/sequence.ts +255 -815
  354. package/src/tools/tool-definition-utils.ts +35 -0
  355. package/src/tools/ui.ts +207 -283
  356. package/src/types/env.ts +0 -10
  357. package/src/types/tool-interfaces.ts +250 -0
  358. package/src/types/tool-types.ts +243 -21
  359. package/src/unreal-bridge.ts +460 -1550
  360. package/src/utils/command-validator.ts +75 -0
  361. package/src/utils/elicitation.ts +10 -7
  362. package/src/utils/error-handler.ts +14 -90
  363. package/src/utils/ini-reader.ts +86 -0
  364. package/src/utils/logger.ts +8 -3
  365. package/src/utils/normalize.ts +60 -0
  366. package/src/utils/response-factory.ts +39 -0
  367. package/src/utils/response-validator.ts +176 -56
  368. package/src/utils/result-helpers.ts +21 -19
  369. package/src/utils/safe-json.ts +14 -11
  370. package/src/utils/unreal-command-queue.ts +152 -0
  371. package/src/utils/validation.ts +4 -1
  372. package/src/wasm/index.ts +838 -0
  373. package/test-server.mjs +100 -0
  374. package/tests/run-unreal-tool-tests.mjs +242 -14
  375. package/tests/test-animation.mjs +44 -0
  376. package/tests/test-asset-advanced.mjs +82 -0
  377. package/tests/test-asset-errors.mjs +35 -0
  378. package/tests/test-audio.mjs +219 -0
  379. package/tests/test-automation-timeouts.mjs +98 -0
  380. package/tests/test-behavior-tree.mjs +261 -0
  381. package/tests/test-blueprint-events.mjs +35 -0
  382. package/tests/test-blueprint-graph.mjs +79 -0
  383. package/tests/test-blueprint.mjs +577 -0
  384. package/tests/test-client-mode.mjs +86 -0
  385. package/tests/test-console-command.mjs +56 -0
  386. package/tests/test-control-actor.mjs +425 -0
  387. package/tests/test-control-editor.mjs +80 -0
  388. package/tests/test-extra-tools.mjs +38 -0
  389. package/tests/test-graphql.mjs +322 -0
  390. package/tests/test-inspect.mjs +72 -0
  391. package/tests/test-landscape.mjs +60 -0
  392. package/tests/test-manage-asset.mjs +438 -0
  393. package/tests/test-manage-level.mjs +70 -0
  394. package/tests/test-materials.mjs +356 -0
  395. package/tests/test-niagara.mjs +185 -0
  396. package/tests/test-no-inline-python.mjs +122 -0
  397. package/tests/test-plugin-handshake.mjs +82 -0
  398. package/tests/test-render.mjs +33 -0
  399. package/tests/test-runner.mjs +933 -0
  400. package/tests/test-search-assets.mjs +66 -0
  401. package/tests/test-sequence.mjs +68 -0
  402. package/tests/test-system.mjs +57 -0
  403. package/tests/test-wasm.mjs +193 -0
  404. package/tests/test-world-partition.mjs +215 -0
  405. package/tsconfig.json +3 -3
  406. package/wasm/Cargo.lock +363 -0
  407. package/wasm/Cargo.toml +42 -0
  408. package/wasm/LICENSE +21 -0
  409. package/wasm/README.md +253 -0
  410. package/wasm/src/dependency_resolver.rs +377 -0
  411. package/wasm/src/lib.rs +153 -0
  412. package/wasm/src/property_parser.rs +271 -0
  413. package/wasm/src/transform_math.rs +396 -0
  414. package/wasm/tests/integration.rs +109 -0
  415. package/.github/workflows/smithery-build.yml +0 -29
  416. package/dist/tools/build_environment_advanced.d.ts +0 -65
  417. package/dist/tools/build_environment_advanced.js +0 -633
  418. package/dist/tools/rc.d.ts +0 -110
  419. package/dist/tools/rc.js +0 -437
  420. package/dist/tools/visual.d.ts +0 -40
  421. package/dist/tools/visual.js +0 -282
  422. package/dist/utils/http.d.ts +0 -6
  423. package/dist/utils/http.js +0 -151
  424. package/dist/utils/python-output.d.ts +0 -18
  425. package/dist/utils/python-output.js +0 -290
  426. package/dist/utils/python.d.ts +0 -2
  427. package/dist/utils/python.js +0 -4
  428. package/dist/utils/stdio-redirect.d.ts +0 -2
  429. package/dist/utils/stdio-redirect.js +0 -20
  430. package/docs/unreal-tool-test-cases.md +0 -574
  431. package/smithery.yaml +0 -29
  432. package/src/tools/build_environment_advanced.ts +0 -732
  433. package/src/tools/rc.ts +0 -515
  434. package/src/tools/visual.ts +0 -281
  435. package/src/utils/http.ts +0 -187
  436. package/src/utils/python-output.ts +0 -351
  437. package/src/utils/python.ts +0 -3
  438. package/src/utils/stdio-redirect.ts +0 -18
@@ -0,0 +1,194 @@
1
+ import { ITools } from '../../types/tool-interfaces.js';
2
+ import { cleanObject } from '../../utils/safe-json.js';
3
+ import { requireNonEmptyString } from './common-handlers.js';
4
+
5
+ function toVec3Array(v: any): [number, number, number] | undefined {
6
+ if (!v || typeof v !== 'object') return undefined;
7
+ const x = Number(v.x);
8
+ const y = Number(v.y);
9
+ const z = Number(v.z);
10
+ if (!Number.isFinite(x) || !Number.isFinite(y) || !Number.isFinite(z)) return undefined;
11
+ return [x, y, z];
12
+ }
13
+
14
+ function toRotArray(r: any): [number, number, number] | undefined {
15
+ if (!r || typeof r !== 'object') return undefined;
16
+ const pitch = Number(r.pitch);
17
+ const yaw = Number(r.yaw);
18
+ const roll = Number(r.roll);
19
+ if (!Number.isFinite(pitch) || !Number.isFinite(yaw) || !Number.isFinite(roll)) return undefined;
20
+ return [pitch, yaw, roll];
21
+ }
22
+
23
+ function getTimeoutMs(): number {
24
+ const envDefault = Number(process.env.MCP_AUTOMATION_REQUEST_TIMEOUT_MS ?? '120000');
25
+ return Number.isFinite(envDefault) && envDefault > 0 ? envDefault : 120000;
26
+ }
27
+
28
+ export async function handleAudioTools(
29
+ action: string,
30
+ args: any,
31
+ tools: ITools
32
+ ) {
33
+ switch (action) {
34
+ case 'create_sound_cue':
35
+ requireNonEmptyString(args?.name, 'name', 'Missing required parameter: name');
36
+ requireNonEmptyString(args?.wavePath ?? args?.soundPath, 'soundPath', 'Missing required parameter: soundPath (or wavePath)');
37
+ return cleanObject(await tools.audioTools.createSoundCue({
38
+ name: args.name,
39
+ // MCP schema uses soundPath; AudioTools uses wavePath.
40
+ wavePath: args.wavePath ?? args.soundPath,
41
+ savePath: args.savePath,
42
+ settings: args.settings
43
+ }));
44
+
45
+ case 'play_sound_at_location':
46
+ requireNonEmptyString(args?.soundPath, 'soundPath', 'Missing required parameter: soundPath');
47
+ return cleanObject(await tools.audioTools.playSoundAtLocation({
48
+ soundPath: args.soundPath,
49
+ location: toVec3Array(args.location) ?? [0, 0, 0],
50
+ rotation: toRotArray(args.rotation),
51
+ volume: args.volume,
52
+ pitch: args.pitch,
53
+ startTime: args.startTime,
54
+ attenuationPath: args.attenuationPath,
55
+ concurrencyPath: args.concurrencyPath
56
+ }));
57
+
58
+ case 'play_sound_2d':
59
+ requireNonEmptyString(args?.soundPath, 'soundPath', 'Missing required parameter: soundPath');
60
+ return cleanObject(await tools.audioTools.playSound2D({
61
+ soundPath: args.soundPath,
62
+ volume: args.volume,
63
+ pitch: args.pitch,
64
+ startTime: args.startTime
65
+ }));
66
+
67
+ case 'create_audio_component':
68
+ requireNonEmptyString(args?.actorName, 'actorName', 'Missing required parameter: actorName');
69
+ requireNonEmptyString(args?.componentName, 'componentName', 'Missing required parameter: componentName');
70
+ requireNonEmptyString(args?.soundPath, 'soundPath', 'Missing required parameter: soundPath');
71
+ return cleanObject(await tools.audioTools.createAudioComponent({
72
+ actorName: args.actorName,
73
+ componentName: args.componentName,
74
+ soundPath: args.soundPath,
75
+ autoPlay: args.autoPlay,
76
+ is3D: args.is3D
77
+ }));
78
+
79
+ case 'set_sound_attenuation':
80
+ requireNonEmptyString(args?.name, 'name', 'Missing required parameter: name');
81
+ return cleanObject(await tools.audioTools.setSoundAttenuation({
82
+ name: args.name,
83
+ innerRadius: args.innerRadius,
84
+ falloffDistance: args.falloffDistance,
85
+ attenuationShape: args.attenuationShape,
86
+ falloffMode: args.falloffMode
87
+ }));
88
+
89
+ case 'create_sound_class':
90
+ requireNonEmptyString(args?.name, 'name', 'Missing required parameter: name');
91
+ return cleanObject(await tools.audioTools.createSoundClass({
92
+ name: args.name,
93
+ parentClass: args.parentClass,
94
+ properties: args.properties
95
+ }));
96
+
97
+ case 'create_sound_mix':
98
+ requireNonEmptyString(args?.name, 'name', 'Missing required parameter: name');
99
+ return cleanObject(await tools.audioTools.createSoundMix({
100
+ name: args.name,
101
+ classAdjusters: args.classAdjusters
102
+ }));
103
+
104
+ case 'push_sound_mix':
105
+ requireNonEmptyString(args?.mixName ?? args?.name, 'mixName', 'Missing required parameter: mixName (or name)');
106
+ return cleanObject(await tools.audioTools.pushSoundMix({
107
+ mixName: args.mixName ?? args.name
108
+ }));
109
+
110
+ case 'pop_sound_mix':
111
+ requireNonEmptyString(args?.mixName ?? args?.name, 'mixName', 'Missing required parameter: mixName (or name)');
112
+ return cleanObject(await tools.audioTools.popSoundMix({
113
+ mixName: args.mixName ?? args.name
114
+ }));
115
+
116
+ case 'create_ambient_sound':
117
+ requireNonEmptyString(args?.soundPath, 'soundPath', 'Missing required parameter: soundPath');
118
+ return cleanObject(await tools.audioTools.createAmbientSound({
119
+ soundPath: args.soundPath,
120
+ location: toVec3Array(args.location) ?? [0, 0, 0],
121
+ volume: args.volume,
122
+ pitch: args.pitch,
123
+ startTime: args.startTime,
124
+ attenuationPath: args.attenuationPath,
125
+ concurrencyPath: args.concurrencyPath
126
+ }));
127
+
128
+ case 'create_reverb_zone':
129
+ requireNonEmptyString(args?.name, 'name', 'Missing required parameter: name');
130
+ return cleanObject(await tools.audioTools.createReverbZone({
131
+ name: args.name,
132
+ location: toVec3Array(args.location) ?? [0, 0, 0],
133
+ size: toVec3Array(args.size) ?? [0, 0, 0],
134
+ reverbEffect: args.reverbEffect,
135
+ volume: args.volume,
136
+ fadeTime: args.fadeTime
137
+ }));
138
+
139
+ case 'enable_audio_analysis':
140
+ return cleanObject(await tools.audioTools.enableAudioAnalysis({
141
+ enabled: args.enabled,
142
+ fftSize: args.fftSize,
143
+ outputType: args.outputType
144
+ }));
145
+
146
+ case 'fade_sound':
147
+ requireNonEmptyString(args?.soundName, 'soundName', 'Missing required parameter: soundName');
148
+ return cleanObject(await tools.audioTools.fadeSound({
149
+ soundName: args.soundName,
150
+ targetVolume: args.targetVolume,
151
+ fadeTime: args.fadeTime,
152
+ fadeType: args.fadeType
153
+ }));
154
+
155
+ case 'set_doppler_effect':
156
+ return cleanObject(await tools.audioTools.setDopplerEffect({
157
+ enabled: args.enabled,
158
+ scale: args.scale
159
+ }));
160
+
161
+ case 'set_audio_occlusion':
162
+ return cleanObject(await tools.audioTools.setAudioOcclusion({
163
+ enabled: args.enabled,
164
+ lowPassFilterFrequency: args.lowPassFilterFrequency,
165
+ volumeAttenuation: args.volumeAttenuation
166
+ }));
167
+
168
+ case 'spawn_sound_at_location':
169
+ // Direct pass-through to C++ handler
170
+ return cleanObject(await tools.automationBridge?.sendAutomationRequest('spawn_sound_at_location', args, { timeoutMs: getTimeoutMs() }));
171
+
172
+ case 'play_sound_attached':
173
+ return cleanObject(await tools.automationBridge?.sendAutomationRequest('play_sound_attached', args, { timeoutMs: getTimeoutMs() }));
174
+
175
+ case 'set_sound_mix_class_override':
176
+ return cleanObject(await tools.automationBridge?.sendAutomationRequest('set_sound_mix_class_override', args, { timeoutMs: getTimeoutMs() }));
177
+
178
+ case 'clear_sound_mix_class_override':
179
+ return cleanObject(await tools.automationBridge?.sendAutomationRequest('clear_sound_mix_class_override', args, { timeoutMs: getTimeoutMs() }));
180
+
181
+ case 'set_base_sound_mix':
182
+ return cleanObject(await tools.automationBridge?.sendAutomationRequest('set_base_sound_mix', args, { timeoutMs: getTimeoutMs() }));
183
+
184
+ case 'prime_sound':
185
+ return cleanObject(await tools.automationBridge?.sendAutomationRequest('prime_sound', args, { timeoutMs: getTimeoutMs() }));
186
+
187
+ case 'fade_sound_in':
188
+ case 'fade_sound_out':
189
+ return cleanObject(await tools.automationBridge?.sendAutomationRequest(action, args, { timeoutMs: getTimeoutMs() }));
190
+
191
+ default:
192
+ return cleanObject({ success: false, error: 'UNKNOWN_ACTION', message: `Unknown audio action: ${action}` });
193
+ }
194
+ }
@@ -0,0 +1,380 @@
1
+ import { cleanObject } from '../../utils/safe-json.js';
2
+ import { ITools } from '../../types/tool-interfaces.js';
3
+ import { executeAutomationRequest } from './common-handlers.js';
4
+
5
+ export async function handleBlueprintTools(action: string, args: any, tools: ITools) {
6
+ switch (action) {
7
+ case 'create': {
8
+ // Support 'path' or 'blueprintPath' argument by splitting it into name and savePath if not provided
9
+ let name = args.name;
10
+ let savePath = args.savePath;
11
+ const pathArg = args.path || args.blueprintPath;
12
+
13
+ if (pathArg) {
14
+ const parts = pathArg.split('/');
15
+ const extractName = parts.pop(); // The last part is the name
16
+
17
+ // If name wasn't provided, use the extracted name
18
+ if (!name) {
19
+ name = extractName;
20
+ }
21
+
22
+ // If savePath wasn't provided, use the extracted path
23
+ if (!savePath) {
24
+ savePath = parts.join('/');
25
+ }
26
+ }
27
+
28
+ if (!savePath) savePath = '/Game';
29
+
30
+ const res = await tools.blueprintTools.createBlueprint({
31
+ name: name,
32
+ blueprintType: args.blueprintType,
33
+ savePath: savePath,
34
+ parentClass: args.parentClass,
35
+ properties: args.properties,
36
+ timeoutMs: args.timeoutMs,
37
+ waitForCompletion: args.waitForCompletion,
38
+ waitForCompletionTimeoutMs: args.waitForCompletionTimeoutMs
39
+ });
40
+ return cleanObject(res);
41
+ }
42
+ case 'ensure_exists': {
43
+ const res = await tools.blueprintTools.waitForBlueprint(args.name || args.blueprintPath || args.path, args.timeoutMs);
44
+ return cleanObject(res);
45
+ }
46
+ case 'add_variable': {
47
+ const res = await tools.blueprintTools.addVariable({
48
+ blueprintName: args.name || args.blueprintPath || args.path,
49
+ variableName: args.variableName,
50
+ variableType: args.variableType,
51
+ defaultValue: args.defaultValue,
52
+ category: args.category,
53
+ isReplicated: args.isReplicated,
54
+ isPublic: args.isPublic,
55
+ variablePinType: args.variablePinType,
56
+ timeoutMs: args.timeoutMs,
57
+ waitForCompletion: args.waitForCompletion,
58
+ waitForCompletionTimeoutMs: args.waitForCompletionTimeoutMs
59
+ });
60
+ return cleanObject(res);
61
+ }
62
+ case 'set_variable_metadata': {
63
+ const res = await tools.blueprintTools.setVariableMetadata({
64
+ blueprintName: args.name || args.blueprintPath || args.path,
65
+ variableName: args.variableName,
66
+ metadata: args.metadata,
67
+ timeoutMs: args.timeoutMs
68
+ });
69
+ return cleanObject(res);
70
+ }
71
+ case 'remove_variable': {
72
+ const res = await tools.blueprintTools.removeVariable({
73
+ blueprintName: args.name || args.blueprintPath || args.path,
74
+ variableName: args.variableName,
75
+ timeoutMs: args.timeoutMs,
76
+ waitForCompletion: args.waitForCompletion,
77
+ waitForCompletionTimeoutMs: args.waitForCompletionTimeoutMs
78
+ });
79
+ return cleanObject(res);
80
+ }
81
+ case 'rename_variable': {
82
+ const res = await tools.blueprintTools.renameVariable({
83
+ blueprintName: args.name || args.blueprintPath || args.path,
84
+ oldName: args.oldName,
85
+ newName: args.newName,
86
+ timeoutMs: args.timeoutMs,
87
+ waitForCompletion: args.waitForCompletion,
88
+ waitForCompletionTimeoutMs: args.waitForCompletionTimeoutMs
89
+ });
90
+ return cleanObject(res);
91
+ }
92
+ case 'set_metadata': {
93
+ const assetPathRaw = typeof args.assetPath === 'string' ? args.assetPath.trim() : '';
94
+ const blueprintPathRaw = typeof args.blueprintPath === 'string' ? args.blueprintPath.trim() : '';
95
+ const nameRaw = typeof args.name === 'string' ? args.name.trim() : '';
96
+ const savePathRaw = typeof args.savePath === 'string' ? args.savePath.trim() : '';
97
+
98
+ let assetPath = assetPathRaw;
99
+ if (!assetPath) {
100
+ if (blueprintPathRaw) {
101
+ assetPath = blueprintPathRaw;
102
+ } else if (nameRaw && savePathRaw) {
103
+ const base = savePathRaw.replace(/\/$/, '');
104
+ assetPath = `${base}/${nameRaw}`;
105
+ }
106
+ }
107
+ if (!assetPath) {
108
+ throw new Error('Invalid parameters: assetPath or blueprintPath or name+savePath required for set_metadata');
109
+ }
110
+
111
+ const metadata = (args.metadata && typeof args.metadata === 'object') ? args.metadata : {};
112
+ const res = await executeAutomationRequest(tools, 'set_metadata', { assetPath, metadata });
113
+ return cleanObject(res);
114
+ }
115
+ case 'add_event': {
116
+ const blueprintName = args.blueprintPath || args.path || args.name;
117
+ const usedNameForBlueprint = !args.blueprintPath && !args.path && args.name;
118
+
119
+ const res: any = await tools.blueprintTools.addEvent({
120
+ blueprintName: blueprintName,
121
+ eventType: args.eventType,
122
+ customEventName: args.customEventName || (!usedNameForBlueprint ? args.name : undefined),
123
+ parameters: args.parameters,
124
+ timeoutMs: args.timeoutMs,
125
+ waitForCompletion: args.waitForCompletion,
126
+ waitForCompletionTimeoutMs: args.waitForCompletionTimeoutMs
127
+ });
128
+
129
+ if (res && res.success === false) {
130
+ const msg = (res.message || '').toLowerCase();
131
+ if (msg.includes('already exists') || msg.includes('duplicate')) {
132
+ return cleanObject({
133
+ success: false,
134
+ error: 'EVENT_ALREADY_EXISTS',
135
+ message: res.message || 'Event already exists',
136
+ blueprintName
137
+ });
138
+ }
139
+ }
140
+ return cleanObject(res);
141
+ }
142
+ case 'remove_event': {
143
+ const res = await tools.blueprintTools.removeEvent({
144
+ blueprintName: args.name || args.blueprintPath || args.path,
145
+ eventName: args.eventName,
146
+ customEventName: args.customEventName,
147
+ timeoutMs: args.timeoutMs,
148
+ waitForCompletion: args.waitForCompletion,
149
+ waitForCompletionTimeoutMs: args.waitForCompletionTimeoutMs
150
+ });
151
+ return cleanObject(res);
152
+ }
153
+ case 'add_function': {
154
+ // Prioritize explicit path for blueprint, allowing 'name' to be function name
155
+ const blueprintName = args.blueprintPath || args.path || args.name;
156
+ const usedNameForBlueprint = !args.blueprintPath && !args.path && args.name;
157
+
158
+ const res = await tools.blueprintTools.addFunction({
159
+ blueprintName: blueprintName,
160
+ functionName: args.functionName || args.memberName || (!usedNameForBlueprint ? args.name : undefined),
161
+ inputs: args.inputs,
162
+ outputs: args.outputs,
163
+ isPublic: args.isPublic,
164
+ category: args.category,
165
+ timeoutMs: args.timeoutMs,
166
+ waitForCompletion: args.waitForCompletion,
167
+ waitForCompletionTimeoutMs: args.waitForCompletionTimeoutMs
168
+ });
169
+ return cleanObject(res);
170
+ }
171
+ case 'add_component': {
172
+ const res = await tools.blueprintTools.addComponent({
173
+ blueprintName: args.name || args.blueprintPath || args.path,
174
+ componentType: args.componentType || args.componentClass,
175
+ componentName: args.componentName,
176
+ attachTo: args.attachTo,
177
+ transform: args.transform,
178
+ properties: args.properties,
179
+ compile: args.applyAndSave,
180
+ save: args.applyAndSave,
181
+ timeoutMs: args.timeoutMs,
182
+ waitForCompletion: args.waitForCompletion,
183
+ waitForCompletionTimeoutMs: args.waitForCompletionTimeoutMs
184
+ });
185
+ return cleanObject(res);
186
+ }
187
+ case 'modify_scs': {
188
+ const res = await tools.blueprintTools.modifyConstructionScript({
189
+ blueprintPath: args.name || args.blueprintPath || args.path,
190
+ operations: args.operations,
191
+ compile: args.applyAndSave,
192
+ save: args.applyAndSave,
193
+ timeoutMs: args.timeoutMs,
194
+ waitForCompletion: args.waitForCompletion,
195
+ waitForCompletionTimeoutMs: args.waitForCompletionTimeoutMs
196
+ });
197
+ return cleanObject(res);
198
+ }
199
+ case 'set_scs_transform': {
200
+ const res = await tools.blueprintTools.setSCSComponentTransform({
201
+ blueprintPath: args.name || args.blueprintPath || args.path,
202
+ componentName: args.componentName,
203
+ location: args.location,
204
+ rotation: args.rotation,
205
+ scale: args.scale,
206
+ timeoutMs: args.timeoutMs
207
+ });
208
+ return cleanObject(res);
209
+ }
210
+ case 'add_construction_script': {
211
+ const res = await tools.blueprintTools.addConstructionScript({
212
+ blueprintName: args.name || args.blueprintPath || args.path,
213
+ scriptName: args.scriptName,
214
+ timeoutMs: args.timeoutMs,
215
+ waitForCompletion: args.waitForCompletion,
216
+ waitForCompletionTimeoutMs: args.waitForCompletionTimeoutMs
217
+ });
218
+ return cleanObject(res);
219
+ }
220
+ case 'add_node': {
221
+ if ((args.nodeType === 'CallFunction' || args.nodeType === 'K2Node_CallFunction') && !args.functionName && !args.memberName) {
222
+ throw new Error('CallFunction node requires functionName parameter');
223
+ }
224
+
225
+ // Map common node aliases to K2Node types
226
+ const nodeAliases: Record<string, string> = {
227
+ 'CallFunction': 'K2Node_CallFunction',
228
+ 'VariableGet': 'K2Node_VariableGet',
229
+ 'VariableSet': 'K2Node_VariableSet',
230
+ 'If': 'K2Node_IfThenElse',
231
+ 'Branch': 'K2Node_IfThenElse',
232
+ 'Switch': 'K2Node_Switch',
233
+ 'Select': 'K2Node_Select',
234
+ 'Cast': 'K2Node_DynamicCast',
235
+ 'CustomEvent': 'K2Node_CustomEvent',
236
+ 'Event': 'K2Node_Event',
237
+ 'MakeArray': 'K2Node_MakeArray',
238
+ 'ForEach': 'K2Node_ForEachElementInEnum' // Note: ForEachLoop is a macro, this is different
239
+ };
240
+
241
+ const resolvedNodeType = nodeAliases[args.nodeType] || args.nodeType;
242
+
243
+ // Validation for Event nodes
244
+ if ((resolvedNodeType === 'K2Node_Event' || resolvedNodeType === 'K2Node_CustomEvent') && !args.eventName && !args.customEventName && !args.name) {
245
+ // Allow 'name' as fallback for customEventName/eventName
246
+ if (!args.eventName) args.eventName = args.name;
247
+
248
+ if (!args.eventName) {
249
+ throw new Error(`${resolvedNodeType} requires eventName (or customEventName) parameter`);
250
+ }
251
+ }
252
+
253
+ const res = await tools.blueprintTools.addNode({
254
+ blueprintName: args.name || args.blueprintPath || args.path,
255
+ nodeType: resolvedNodeType,
256
+ graphName: args.graphName,
257
+ functionName: args.functionName,
258
+ variableName: args.variableName,
259
+ nodeName: args.nodeName,
260
+ eventName: args.eventName || args.customEventName,
261
+ memberClass: args.memberClass,
262
+ posX: args.posX,
263
+ posY: args.posY,
264
+ timeoutMs: args.timeoutMs
265
+ });
266
+ return cleanObject(res);
267
+ }
268
+ case 'add_scs_component': {
269
+ const res = await tools.blueprintTools.addSCSComponent({
270
+ blueprintPath: args.name || args.blueprintPath || args.path,
271
+ componentClass: args.componentClass || args.componentType,
272
+ componentName: args.componentName,
273
+ meshPath: args.meshPath,
274
+ materialPath: args.materialPath,
275
+ timeoutMs: args.timeoutMs
276
+ });
277
+ return cleanObject(res);
278
+ }
279
+ case 'reparent_scs_component': {
280
+ const res = await tools.blueprintTools.reparentSCSComponent({
281
+ blueprintPath: args.name || args.blueprintPath || args.path,
282
+ componentName: args.componentName,
283
+ newParent: args.newParent,
284
+ timeoutMs: args.timeoutMs
285
+ });
286
+ return cleanObject(res);
287
+ }
288
+ case 'set_scs_property': {
289
+ const res = await tools.blueprintTools.setSCSComponentProperty({
290
+ blueprintPath: args.name || args.blueprintPath || args.path,
291
+ componentName: args.componentName,
292
+ propertyName: args.propertyName,
293
+ propertyValue: args.propertyValue,
294
+ timeoutMs: args.timeoutMs
295
+ });
296
+ return cleanObject(res);
297
+ }
298
+ case 'remove_scs_component': {
299
+ const res = await tools.blueprintTools.removeSCSComponent({
300
+ blueprintPath: args.name || args.blueprintPath || args.path,
301
+ componentName: args.componentName,
302
+ timeoutMs: args.timeoutMs
303
+ });
304
+ return cleanObject(res);
305
+ }
306
+ case 'get_scs': {
307
+ const res = await tools.blueprintTools.getBlueprintSCS({
308
+ blueprintPath: args.name || args.blueprintPath || args.path,
309
+ timeoutMs: args.timeoutMs
310
+ });
311
+ return cleanObject(res);
312
+ }
313
+ case 'set_default': {
314
+ const res = await tools.blueprintTools.setBlueprintDefault({
315
+ blueprintName: args.name || args.blueprintPath || args.path,
316
+ propertyName: args.propertyName,
317
+ value: args.value
318
+ });
319
+ return cleanObject(res);
320
+ }
321
+ case 'compile': {
322
+ const res = await tools.blueprintTools.compileBlueprint({
323
+ blueprintName: args.name || args.blueprintPath || args.path,
324
+ saveAfterCompile: args.saveAfterCompile
325
+ });
326
+ return cleanObject(res);
327
+ }
328
+ case 'probe_handle': {
329
+ const res = await tools.blueprintTools.probeSubobjectDataHandle({
330
+ componentClass: args.componentClass
331
+ });
332
+ return cleanObject(res);
333
+ }
334
+ case 'get': {
335
+ const res = await tools.blueprintTools.getBlueprintInfo({
336
+ blueprintPath: args.name || args.blueprintPath || args.path,
337
+ timeoutMs: args.timeoutMs
338
+ });
339
+ return cleanObject(res);
340
+ }
341
+ case 'connect_pins':
342
+ case 'break_pin_links':
343
+ case 'delete_node':
344
+ case 'create_reroute_node':
345
+ case 'set_node_property':
346
+ case 'get_node_details':
347
+ case 'get_pin_details':
348
+ case 'get_graph_details': {
349
+ const processedArgs = {
350
+ ...args,
351
+ subAction: action,
352
+ blueprintPath: args.blueprintPath || args.path || args.name
353
+ };
354
+ const res = await executeAutomationRequest(tools, 'manage_blueprint_graph', processedArgs, 'Automation bridge not available for blueprint graph operations');
355
+ return cleanObject(res);
356
+ }
357
+ default: {
358
+ // Translate applyAndSave to compile/save flags for modify_scs action
359
+ const processedArgs = { ...args };
360
+ if (args.action === 'modify_scs' && args.applyAndSave === true) {
361
+ processedArgs.compile = true;
362
+ processedArgs.save = true;
363
+ }
364
+ const res = await executeAutomationRequest(tools, 'manage_blueprint', processedArgs, 'Automation bridge not available for blueprint operations');
365
+ return cleanObject(res);
366
+ }
367
+ }
368
+ }
369
+
370
+ export async function handleBlueprintGet(args: any, tools: ITools) {
371
+ const res = await executeAutomationRequest(tools, 'blueprint_get', args, 'Automation bridge not available for blueprint operations');
372
+ if (res && res.success) {
373
+ return cleanObject({
374
+ ...res,
375
+ blueprint: args.blueprintPath || args.path || args.name || 'Unknown Blueprint',
376
+ message: res.message || 'Blueprint fetched'
377
+ });
378
+ }
379
+ return cleanObject(res);
380
+ }
@@ -0,0 +1,87 @@
1
+ import { ITools } from '../../types/tool-interfaces.js';
2
+
3
+ export function ensureArgsPresent(args: any) {
4
+ if (args === null || args === undefined) {
5
+ throw new Error('Invalid arguments: null or undefined');
6
+ }
7
+ }
8
+
9
+ export function requireAction(args: any): string {
10
+ ensureArgsPresent(args);
11
+ const action = args.action;
12
+ if (typeof action !== 'string' || action.trim() === '') {
13
+ throw new Error('Missing required parameter: action');
14
+ }
15
+ return action;
16
+ }
17
+
18
+ export function requireNonEmptyString(value: any, field: string, message?: string): string {
19
+ if (typeof value !== 'string' || value.trim() === '') {
20
+ throw new Error(message ?? `Invalid ${field}: must be a non-empty string`);
21
+ }
22
+ return value;
23
+ }
24
+
25
+ export async function executeAutomationRequest(
26
+ tools: ITools,
27
+ toolName: string,
28
+ args: any,
29
+ errorMessage: string = 'Automation bridge not available',
30
+ options: { timeoutMs?: number } = {}
31
+ ) {
32
+ const automationBridge = tools.automationBridge;
33
+ // If the bridge is missing or not a function, we can't proceed with automation requests
34
+ if (!automationBridge || typeof automationBridge.sendAutomationRequest !== 'function') {
35
+ throw new Error(errorMessage);
36
+ }
37
+
38
+ if (!automationBridge.isConnected()) {
39
+ throw new Error(`Automation bridge is not connected to Unreal Engine. Please check if the editor is running and the plugin is enabled. Action: ${toolName}`);
40
+ }
41
+
42
+ return await automationBridge.sendAutomationRequest(toolName, args, options);
43
+ }
44
+
45
+ /**
46
+ * Normalize location to [x, y, z] array format
47
+ * Accepts both {x,y,z} object and [x,y,z] array formats
48
+ */
49
+ export function normalizeLocation(location: any): [number, number, number] | undefined {
50
+ if (!location) return undefined;
51
+
52
+ // Already array format
53
+ if (Array.isArray(location) && location.length >= 3) {
54
+ return [Number(location[0]) || 0, Number(location[1]) || 0, Number(location[2]) || 0];
55
+ }
56
+
57
+ // Object format {x, y, z}
58
+ if (typeof location === 'object' && ('x' in location || 'y' in location || 'z' in location)) {
59
+ return [Number(location.x) || 0, Number(location.y) || 0, Number(location.z) || 0];
60
+ }
61
+
62
+ return undefined;
63
+ }
64
+
65
+ /**
66
+ * Normalize rotation to {pitch, yaw, roll} object format
67
+ * Accepts both {pitch,yaw,roll} object and [pitch,yaw,roll] array formats
68
+ */
69
+ export function normalizeRotation(rotation: any): { pitch: number; yaw: number; roll: number } | undefined {
70
+ if (!rotation) return undefined;
71
+
72
+ // Array format [pitch, yaw, roll]
73
+ if (Array.isArray(rotation) && rotation.length >= 3) {
74
+ return { pitch: Number(rotation[0]) || 0, yaw: Number(rotation[1]) || 0, roll: Number(rotation[2]) || 0 };
75
+ }
76
+
77
+ // Already object format
78
+ if (typeof rotation === 'object') {
79
+ return {
80
+ pitch: Number(rotation.pitch) || 0,
81
+ yaw: Number(rotation.yaw) || 0,
82
+ roll: Number(rotation.roll) || 0
83
+ };
84
+ }
85
+
86
+ return undefined;
87
+ }