unreal-engine-mcp-server 0.4.6 → 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 +269 -22
  29. package/CONTRIBUTING.md +140 -0
  30. package/README.md +166 -72
  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 -604
  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 +5475 -1627
  97. package/dist/tools/consolidated-tool-definitions.js +829 -482
  98. package/dist/tools/consolidated-tool-handlers.d.ts +2 -1
  99. package/dist/tools/consolidated-tool-handlers.js +211 -1009
  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 +45 -0
  161. package/dist/tools/logs.js +210 -0
  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 +195 -11
  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 -649
  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 -500
  316. package/src/tools/consolidated-tool-handlers.ts +272 -1122
  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 +219 -0
  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 +250 -13
  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 -572
  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,123 @@
1
+ import { cleanObject } from '../../utils/safe-json.js';
2
+ import { ITools } from '../../types/tool-interfaces.js';
3
+ import { executeAutomationRequest, requireNonEmptyString } from './common-handlers.js';
4
+
5
+ export async function handleEditorTools(action: string, args: any, tools: ITools) {
6
+ switch (action) {
7
+ case 'play': {
8
+ const res = await tools.editorTools.playInEditor(args.timeoutMs);
9
+ return cleanObject(res);
10
+ }
11
+ case 'stop':
12
+ case 'stop_pie': {
13
+ const res = await tools.editorTools.stopPlayInEditor();
14
+ return cleanObject(res);
15
+ }
16
+ case 'eject': {
17
+ const inPie = await tools.editorTools.isInPIE();
18
+ if (!inPie) {
19
+ return { success: false, error: 'NOT_IN_PIE', message: 'Cannot eject while not in PIE' };
20
+ }
21
+ return await executeAutomationRequest(tools, 'control_editor', { action: 'eject' });
22
+ }
23
+ case 'possess': {
24
+ const inPie = await tools.editorTools.isInPIE();
25
+ if (!inPie) {
26
+ return { success: false, error: 'NOT_IN_PIE', message: 'Cannot possess actor while not in PIE' };
27
+ }
28
+ return await executeAutomationRequest(tools, 'control_editor', args);
29
+ }
30
+ case 'pause': {
31
+ const res = await tools.editorTools.pausePlayInEditor();
32
+ return cleanObject(res);
33
+ }
34
+ case 'resume': {
35
+ const res = await tools.editorTools.resumePlayInEditor();
36
+ return cleanObject(res);
37
+ }
38
+ case 'screenshot': {
39
+ const res = await tools.editorTools.takeScreenshot(args.filename, args.resolution);
40
+ return cleanObject(res);
41
+ }
42
+ case 'console_command': {
43
+ const res = await tools.editorTools.executeConsoleCommand(args.command);
44
+ return cleanObject(res);
45
+ }
46
+ case 'set_camera': {
47
+ const res = await tools.editorTools.setViewportCamera(args.location, args.rotation);
48
+ return cleanObject(res);
49
+ }
50
+ case 'start_recording': {
51
+ // Use console command as fallback if bridge doesn't support it
52
+ const filename = args.filename || 'TestRecording';
53
+ await tools.editorTools.executeConsoleCommand(`DemoRec ${filename}`);
54
+ return { success: true, message: `Started recording to ${filename}`, action: 'start_recording' };
55
+ }
56
+ case 'stop_recording': {
57
+ await tools.editorTools.executeConsoleCommand('DemoStop');
58
+ return { success: true, message: 'Stopped recording', action: 'stop_recording' };
59
+ }
60
+ case 'step_frame': {
61
+ // Use console command for single frame advance
62
+ await tools.editorTools.executeConsoleCommand('r.SingleFrameAdvance 1');
63
+ return { success: true, message: 'Stepped frame', action: 'step_frame' };
64
+ }
65
+ case 'create_bookmark': {
66
+ const idx = parseInt(args.bookmarkName) || 0;
67
+ await tools.editorTools.executeConsoleCommand(`r.SetBookmark ${idx}`);
68
+ return { success: true, message: `Created bookmark ${idx}`, action: 'create_bookmark' };
69
+ }
70
+ case 'jump_to_bookmark': {
71
+ const idx = parseInt(args.bookmarkName) || 0;
72
+ await tools.editorTools.executeConsoleCommand(`r.JumpToBookmark ${idx}`);
73
+ return { success: true, message: `Jumped to bookmark ${idx}`, action: 'jump_to_bookmark' };
74
+ }
75
+ case 'set_preferences': {
76
+ const res = await tools.editorTools.setEditorPreferences(args.category, args.preferences);
77
+ return cleanObject(res);
78
+ }
79
+ case 'open_asset': {
80
+ const assetPath = requireNonEmptyString(args.assetPath || args.path, 'assetPath');
81
+ const res = await executeAutomationRequest(tools, 'control_editor', { action: 'open_asset', assetPath });
82
+ return cleanObject(res);
83
+ }
84
+ case 'execute_command': {
85
+ const command = requireNonEmptyString(args.command, 'command');
86
+ const res = await tools.editorTools.executeConsoleCommand(command);
87
+ return { ...cleanObject(res), action: 'execute_command' };
88
+ }
89
+ case 'set_camera_fov': {
90
+ await tools.editorTools.executeConsoleCommand(`fov ${args.fov}`);
91
+ return { success: true, message: `Set FOV to ${args.fov}`, action: 'set_camera_fov' };
92
+ }
93
+ case 'set_game_speed': {
94
+ await tools.editorTools.executeConsoleCommand(`slomo ${args.speed}`);
95
+ return { success: true, message: `Set game speed to ${args.speed}`, action: 'set_game_speed' };
96
+ }
97
+ case 'set_view_mode': {
98
+ const viewMode = requireNonEmptyString(args.viewMode, 'viewMode');
99
+ const validModes = [
100
+ 'Lit', 'Unlit', 'Wireframe', 'DetailLighting', 'LightingOnly', 'Reflections',
101
+ 'OptimizationViewmodes', 'ShaderComplexity', 'LightmapDensity', 'StationaryLightOverlap', 'LightComplexity',
102
+ 'PathTracing', 'Visualizer', 'LODColoration', 'CollisionPawn', 'CollisionVisibility'
103
+ ];
104
+ if (!validModes.includes(viewMode)) {
105
+ throw new Error(`unknown_viewmode: ${viewMode}. Must be one of: ${validModes.join(', ')}`);
106
+ }
107
+ await tools.editorTools.executeConsoleCommand(`viewmode ${viewMode}`);
108
+ return { success: true, message: `Set view mode to ${viewMode}`, action: 'set_view_mode' };
109
+ }
110
+ case 'set_viewport_resolution': {
111
+ await tools.editorTools.executeConsoleCommand(`r.SetRes ${args.width}x${args.height}`);
112
+ return { success: true, message: `Set viewport resolution to ${args.width}x${args.height}`, action: 'set_viewport_resolution' };
113
+ }
114
+ case 'set_viewport_realtime': {
115
+ const enabled = args.enabled !== undefined ? args.enabled : (args.realtime !== false);
116
+ // Use console command since interface doesn't have setViewportRealtime
117
+ await tools.editorTools.executeConsoleCommand(`r.ViewportRealtime ${enabled ? 1 : 0}`);
118
+ return { success: true, message: `Set viewport realtime to ${enabled}`, action: 'set_viewport_realtime' };
119
+ }
120
+ default:
121
+ return await executeAutomationRequest(tools, 'control_editor', args);
122
+ }
123
+ }
@@ -0,0 +1,220 @@
1
+ import { cleanObject } from '../../utils/safe-json.js';
2
+ import { ITools } from '../../types/tool-interfaces.js';
3
+ import { executeAutomationRequest, requireNonEmptyString } from './common-handlers.js';
4
+
5
+ function ensureActionAndSubAction(action: string, args: any) {
6
+ if (!args || typeof args !== 'object') return;
7
+ // Many callers pass the tool action as the action name (e.g. "niagara") and
8
+ // omit args.action; the native handler requires subAction.
9
+ if (!args.action) {
10
+ args.action = action;
11
+ }
12
+ if (!args.subAction) {
13
+ args.subAction = args.action;
14
+ }
15
+ }
16
+
17
+ function isNonEmptyString(val: unknown): val is string {
18
+ return typeof val === 'string' && val.trim().length > 0;
19
+ }
20
+
21
+ export async function handleEffectTools(action: string, args: any, tools: ITools) {
22
+ if (!args || typeof args !== 'object') {
23
+ args = {};
24
+ }
25
+
26
+ // Always ensure action/subAction are present before any routing.
27
+ ensureActionAndSubAction(action, args);
28
+
29
+ // Handle creation actions explicitly to use NiagaraTools helper
30
+ if (action === 'create_niagara_system') {
31
+ const res = await tools.niagaraTools.createSystem({
32
+ name: args.name,
33
+ savePath: args.savePath,
34
+ template: args.template
35
+ });
36
+ return cleanObject(res);
37
+ }
38
+ if (action === 'create_niagara_emitter') {
39
+ const res = await tools.niagaraTools.createEmitter({
40
+ name: args.name,
41
+ savePath: args.savePath,
42
+ systemPath: args.systemPath,
43
+ template: args.template
44
+ });
45
+ return cleanObject(res);
46
+ }
47
+
48
+ // Pre-process arguments for particle presets
49
+ if (args.action === 'particle') {
50
+ const presets: Record<string, string> = {
51
+ 'Default': '/StarterContent/Particles/P_Steam_Lit.P_Steam_Lit',
52
+ 'Smoke': '/StarterContent/Particles/P_Smoke.P_Smoke',
53
+ 'Fire': '/StarterContent/Particles/P_Fire.P_Fire',
54
+ 'Explosion': '/StarterContent/Particles/P_Explosion.P_Explosion',
55
+ };
56
+ // Check both preset and effectType fields
57
+ const key = args.preset || args.effectType;
58
+ if (key && presets[key]) {
59
+ args.preset = presets[key];
60
+ }
61
+ }
62
+
63
+ // Handle debug shapes (must happen before any automation request)
64
+ if (action === 'debug_shape' || args.action === 'debug_shape') {
65
+ // Map 'shape' to 'shapeType' if provided (schema uses 'shape', C++ uses 'shapeType')
66
+ if (args.shape && !args.shapeType) {
67
+ args.shapeType = args.shape;
68
+ }
69
+ requireNonEmptyString(args.shapeType, 'shapeType', 'Missing required parameter: shapeType');
70
+ args.action = 'debug_shape';
71
+ args.subAction = 'debug_shape';
72
+ return cleanObject(await executeAutomationRequest(tools, 'create_effect', args));
73
+ }
74
+
75
+ // Validate Niagara-related required parameters (keep errors explicit and early)
76
+ const subAction = String(args.subAction || '').trim();
77
+ if (subAction === 'niagara' || subAction === 'spawn_niagara') {
78
+ requireNonEmptyString(args.systemPath, 'systemPath', 'Missing required parameter: systemPath');
79
+ }
80
+
81
+ if (subAction === 'activate_niagara' || subAction === 'deactivate_niagara' || subAction === 'advance_simulation') {
82
+ const systemName = args.systemName ?? args.actorName;
83
+ requireNonEmptyString(systemName, 'systemName', 'Missing required parameter: systemName (or actorName)');
84
+ args.systemName = systemName;
85
+ }
86
+
87
+ if (subAction === 'set_niagara_parameter') {
88
+ const systemName = args.systemName ?? args.actorName;
89
+ requireNonEmptyString(systemName, 'systemName', 'Missing required parameter: systemName (or actorName)');
90
+ requireNonEmptyString(args.parameterName, 'parameterName', 'Missing required parameter: parameterName');
91
+ // parameterType is required for unambiguous native conversion; accept common aliases.
92
+ if (!isNonEmptyString(args.parameterType) && isNonEmptyString(args.type)) {
93
+ args.parameterType = args.type.charAt(0).toUpperCase() + args.type.slice(1);
94
+ }
95
+ requireNonEmptyString(args.parameterType, 'parameterType', 'Missing required parameter: parameterType');
96
+ args.systemName = systemName;
97
+ }
98
+
99
+ // Handle debug cleanup actions
100
+ if (action === 'clear_debug_shapes') {
101
+ return executeAutomationRequest(tools, action, args);
102
+ }
103
+ if (action === 'cleanup') {
104
+ args.action = 'cleanup';
105
+ args.subAction = 'cleanup';
106
+ return executeAutomationRequest(tools, 'create_effect', args);
107
+ }
108
+
109
+ // Map high-level actions to create_effect with subAction
110
+ const createActions = [
111
+ 'create_volumetric_fog',
112
+ 'create_particle_trail',
113
+ 'create_environment_effect',
114
+ 'create_impact_effect',
115
+ 'create_niagara_ribbon'
116
+ ];
117
+ if (createActions.includes(action)) {
118
+ args.action = action;
119
+ return executeAutomationRequest(tools, 'create_effect', args);
120
+ }
121
+
122
+ // Map simulation control actions
123
+ if (action === 'activate' || action === 'activate_effect') {
124
+ args.action = 'activate_niagara';
125
+ args.systemName = args.actorName || args.systemName;
126
+ requireNonEmptyString(args.systemName, 'systemName', 'Missing required parameter: systemName (or actorName)');
127
+ args.reset = true;
128
+ return executeAutomationRequest(tools, 'create_effect', args);
129
+ }
130
+ if (action === 'deactivate') {
131
+ args.action = 'deactivate_niagara';
132
+ args.systemName = args.actorName || args.systemName;
133
+ requireNonEmptyString(args.systemName, 'systemName', 'Missing required parameter: systemName (or actorName)');
134
+ return executeAutomationRequest(tools, 'create_effect', args);
135
+ }
136
+ if (action === 'reset') {
137
+ args.action = 'activate_niagara';
138
+ args.systemName = args.actorName || args.systemName;
139
+ requireNonEmptyString(args.systemName, 'systemName', 'Missing required parameter: systemName (or actorName)');
140
+ args.reset = true;
141
+ return executeAutomationRequest(tools, 'create_effect', args);
142
+ }
143
+ if (action === 'advance_simulation') {
144
+ args.action = 'advance_simulation';
145
+ args.systemName = args.actorName || args.systemName;
146
+ requireNonEmptyString(args.systemName, 'systemName', 'Missing required parameter: systemName (or actorName)');
147
+ return executeAutomationRequest(tools, 'create_effect', args);
148
+ }
149
+
150
+ // Map parameter setting
151
+ if (action === 'set_niagara_parameter') {
152
+ args.action = 'set_niagara_parameter';
153
+ // If actorName is provided, use it as systemName (which C++ expects for actor label)
154
+ if (args.actorName && !args.systemName) {
155
+ args.systemName = args.actorName;
156
+ }
157
+ // Map 'type' to 'parameterType' if provided and parameterType is missing
158
+ if (args.type && !args.parameterType) {
159
+ args.parameterType = args.type.charAt(0).toUpperCase() + args.type.slice(1);
160
+ }
161
+ requireNonEmptyString(args.systemName, 'systemName', 'Missing required parameter: systemName (or actorName)');
162
+ requireNonEmptyString(args.parameterName, 'parameterName', 'Missing required parameter: parameterName');
163
+ requireNonEmptyString(args.parameterType, 'parameterType', 'Missing required parameter: parameterType');
164
+ return executeAutomationRequest(tools, 'create_effect', args);
165
+ }
166
+
167
+ const res: any = await executeAutomationRequest(
168
+ tools,
169
+ 'create_effect',
170
+ args,
171
+ 'Automation bridge not available for effect creation operations'
172
+ );
173
+
174
+ const result = res?.result ?? res ?? {};
175
+ const topError = typeof res?.error === 'string' ? res.error : '';
176
+ const nestedError = typeof result.error === 'string' ? result.error : '';
177
+ const errorCode = (topError || nestedError).toUpperCase();
178
+
179
+ const topMessage = typeof res?.message === 'string' ? res.message : '';
180
+ const nestedMessage = typeof result.message === 'string' ? result.message : '';
181
+ const message = topMessage || nestedMessage || '';
182
+
183
+ const combined = `${topError} ${nestedError} ${message}`.toLowerCase();
184
+
185
+ if (
186
+ action === 'niagara' &&
187
+ (
188
+ errorCode === 'SYSTEM_NOT_FOUND' ||
189
+ combined.includes('niagara system not found') ||
190
+ combined.includes('system asset not found')
191
+ )
192
+ ) {
193
+ return cleanObject({
194
+ success: false,
195
+ error: 'SYSTEM_NOT_FOUND',
196
+ message: message || 'Niagara system not found',
197
+ systemPath: args.systemPath,
198
+ handled: true
199
+ });
200
+ }
201
+
202
+ // If we got here and it was a spawn_niagara failure, maybe try to be helpful about paths
203
+ if (action === 'spawn_niagara' && errorCode === 'SYSTEM_NOT_FOUND' && args.systemPath) {
204
+ // Check if path ends in .Name
205
+ const path = args.systemPath;
206
+ const name = path.split('/').pop();
207
+ if (name && !path.endsWith(`.${name}`)) {
208
+ // Retry with corrected path?
209
+ // We can't easily retry here without recursion, but we can hint in the message.
210
+ return cleanObject({
211
+ success: false,
212
+ error: 'SYSTEM_NOT_FOUND',
213
+ message: `Niagara System not found at ${path}. Did you mean ${path}.${name}?`,
214
+ systemPath: path
215
+ });
216
+ }
217
+ }
218
+
219
+ return cleanObject(res);
220
+ }
@@ -0,0 +1,183 @@
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 handleEnvironmentTools(action: string, args: any, tools: ITools) {
6
+ const envAction = String(action || '').toLowerCase();
7
+ switch (envAction) {
8
+ case 'create_landscape':
9
+ return cleanObject(await tools.landscapeTools.createLandscape({
10
+ name: args.name,
11
+ location: args.location,
12
+ sizeX: args.sizeX,
13
+ sizeY: args.sizeY,
14
+ quadsPerSection: args.quadsPerSection,
15
+ sectionsPerComponent: args.sectionsPerComponent,
16
+ componentCount: args.componentCount,
17
+ materialPath: args.materialPath,
18
+ enableWorldPartition: args.enableWorldPartition,
19
+ runtimeGrid: args.runtimeGrid,
20
+ isSpatiallyLoaded: args.isSpatiallyLoaded,
21
+ dataLayers: args.dataLayers
22
+ }));
23
+ case 'modify_heightmap':
24
+ return cleanObject(await tools.landscapeTools.modifyHeightmap({
25
+ landscapeName: args.landscapeName || args.name,
26
+ heightData: args.heightData,
27
+ minX: args.minX,
28
+ minY: args.minY,
29
+ maxX: args.maxX,
30
+ maxY: args.maxY,
31
+ updateNormals: args.updateNormals
32
+ }));
33
+ case 'sculpt':
34
+ case 'sculpt_landscape': {
35
+ // Default to 'Raise' tool if not specified
36
+ const tool = args.tool || 'Raise';
37
+ return cleanObject(await tools.landscapeTools.sculptLandscape({
38
+ landscapeName: args.landscapeName || args.name,
39
+ tool,
40
+ location: args.location,
41
+ radius: args.radius || 500,
42
+ strength: args.strength || 0.5
43
+ }));
44
+ }
45
+ case 'add_foliage': {
46
+ // Check if this is adding a foliage TYPE (has meshPath) or INSTANCES (has locations/position)
47
+ if (args.meshPath) {
48
+ // Derive a better default name from mesh path if not provided
49
+ const defaultName = args.meshPath.split('/').pop()?.split('.')[0] + '_Foliage_Type';
50
+ return cleanObject(await tools.foliageTools.addFoliageType({
51
+ name: args.foliageType || args.name || defaultName,
52
+ meshPath: args.meshPath,
53
+ density: args.density
54
+ }));
55
+ } else {
56
+ // Validate foliageType is provided
57
+ const foliageType = args.foliageType || args.foliageTypePath;
58
+ if (!foliageType) {
59
+ return cleanObject({
60
+ success: false,
61
+ error: 'INVALID_ARGUMENT',
62
+ message: 'add_foliage requires either: (1) meshPath to create a new foliage type, or (2) foliageType/foliageTypePath to place instances of an existing type. Example foliage assets: /Game/StarterContent/Props/SM_Bush, /Engine/BasicShapes/Sphere'
63
+ });
64
+ }
65
+
66
+ // Support location+radius to generate locations if explicit array not provided
67
+ let locations = args.locations;
68
+ if (!locations && args.location && args.radius) {
69
+ // Generate locations around the center point within radius
70
+ const center = args.location;
71
+ const radius = args.radius || 500;
72
+ const count = args.density || args.count || 10;
73
+ locations = [];
74
+ for (let i = 0; i < count; i++) {
75
+ const angle = Math.random() * Math.PI * 2;
76
+ const dist = Math.random() * radius;
77
+ locations.push({
78
+ x: (center.x || 0) + Math.cos(angle) * dist,
79
+ y: (center.y || 0) + Math.sin(angle) * dist,
80
+ z: center.z || 0
81
+ });
82
+ }
83
+ } else if (!locations && args.position) {
84
+ locations = [args.position];
85
+ }
86
+
87
+ // Validate we have locations to place
88
+ if (!locations || locations.length === 0) {
89
+ return cleanObject({
90
+ success: false,
91
+ error: 'INVALID_ARGUMENT',
92
+ message: 'add_foliage requires locations to place foliage instances. Provide: locations array, or location+radius, or position'
93
+ });
94
+ }
95
+
96
+ return cleanObject(await tools.foliageTools.addFoliage({
97
+ foliageType,
98
+ locations
99
+ }));
100
+ }
101
+ }
102
+
103
+ case 'add_foliage_instances':
104
+ return cleanObject(await tools.foliageTools.addFoliageInstances({
105
+ foliageType: args.foliageType || args.foliageTypePath || args.meshPath,
106
+ transforms: args.transforms || (args.locations ? args.locations.map((l: any) => ({ location: [l.x, l.y, l.z] })) : [])
107
+ }));
108
+ case 'paint_foliage':
109
+ return cleanObject(await tools.foliageTools.paintFoliage({
110
+ foliageType: args.foliageType || args.foliageTypePath,
111
+ position: args.position || args.location, // Handle both
112
+ brushSize: args.brushSize || args.radius,
113
+ paintDensity: args.density || args.strength, // Map strength/density
114
+ eraseMode: args.eraseMode
115
+ }));
116
+ case 'create_procedural_terrain':
117
+ return cleanObject(await tools.landscapeTools.createProceduralTerrain({
118
+ name: args.name,
119
+ location: args.location,
120
+ subdivisions: args.subdivisions,
121
+ settings: args.settings
122
+ }));
123
+ case 'create_procedural_foliage':
124
+ return cleanObject(await tools.foliageTools.createProceduralFoliage({
125
+ name: args.name,
126
+ foliageTypes: args.foliageTypes,
127
+ volumeName: args.volumeName,
128
+ bounds: args.bounds,
129
+ seed: args.seed,
130
+ tileSize: args.tileSize
131
+ }));
132
+
133
+ case 'bake_lightmap':
134
+ return cleanObject(await tools.lightingTools.buildLighting({
135
+ quality: (args.quality as any) || 'Preview',
136
+ buildOnlySelected: false,
137
+ buildReflectionCaptures: false
138
+ }));
139
+ case 'create_landscape_grass_type':
140
+ return cleanObject(await tools.landscapeTools.createLandscapeGrassType({
141
+ name: args.name,
142
+ // Prefer explicit meshPath used by tests, fall back to path/staticMesh for
143
+ // compatibility with older callers.
144
+ meshPath: args.meshPath || args.path || args.staticMesh,
145
+ path: args.path,
146
+ staticMesh: args.staticMesh
147
+ }));
148
+ case 'export_snapshot':
149
+ return cleanObject(await tools.environmentTools.exportSnapshot({
150
+ path: args.path,
151
+ filename: args.filename
152
+ }));
153
+ case 'import_snapshot':
154
+ return cleanObject(await tools.environmentTools.importSnapshot({
155
+ path: args.path,
156
+ filename: args.filename
157
+ }));
158
+ case 'set_landscape_material':
159
+ return cleanObject(await tools.landscapeTools.setLandscapeMaterial({
160
+ landscapeName: args.landscapeName || args.name,
161
+ materialPath: args.materialPath
162
+ }));
163
+ case 'generate_lods':
164
+ return cleanObject(await executeAutomationRequest(tools, 'build_environment', {
165
+ action: 'generate_lods',
166
+ assetPaths: args.assetPaths || args.assets || (args.path ? [args.path] : []),
167
+ numLODs: args.numLODs
168
+ }, 'Bridge unavailable'));
169
+ case 'delete': {
170
+ const names = Array.isArray(args.names)
171
+ ? args.names
172
+ : (Array.isArray(args.actors) ? args.actors : []);
173
+ if (args.name) {
174
+ names.push(args.name);
175
+ }
176
+ const res = await tools.environmentTools.cleanup({ names });
177
+ return cleanObject(res);
178
+ }
179
+ default:
180
+ const res = await executeAutomationRequest(tools, 'build_environment', args, 'Automation bridge not available for environment building operations');
181
+ return cleanObject(res);
182
+ }
183
+ }
@@ -0,0 +1,116 @@
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 handleGraphTools(toolName: string, action: string, args: any, tools: ITools) {
6
+ // Common validation
7
+ if (!args.assetPath && !args.blueprintPath && !args.systemPath) {
8
+ // Some actions might not need a path if they operate on "currently open" asset,
9
+ // but generally we want an asset path.
10
+ }
11
+
12
+ // Dispatch based on tool name
13
+ switch (toolName) {
14
+ case 'manage_blueprint_graph':
15
+ return handleBlueprintGraph(action, args, tools);
16
+ case 'manage_niagara_graph':
17
+ return handleNiagaraGraph(action, args, tools);
18
+ case 'manage_material_graph':
19
+ return handleMaterialGraph(action, args, tools);
20
+ case 'manage_behavior_tree':
21
+ return handleBehaviorTree(action, args, tools);
22
+ default:
23
+ return { success: false, error: 'UNKNOWN_TOOL', message: `Unknown graph tool: ${toolName}` };
24
+ }
25
+ }
26
+
27
+ async function handleBlueprintGraph(action: string, args: any, tools: ITools) {
28
+ const processedArgs = { ...args, subAction: action };
29
+
30
+ // Default graphName
31
+ if (!processedArgs.graphName) {
32
+ processedArgs.graphName = 'EventGraph';
33
+ }
34
+
35
+ // Fix Issue 1: Map FunctionCall to CallFunction
36
+ if (processedArgs.nodeType === 'FunctionCall') {
37
+ processedArgs.nodeType = 'CallFunction';
38
+ }
39
+
40
+ // Fix Issue 2 & 3: Map memberName to specific names based on nodeType
41
+ if (processedArgs.memberName) {
42
+ if (processedArgs.nodeType === 'VariableGet' || processedArgs.nodeType === 'VariableSet') {
43
+ if (!processedArgs.variableName) processedArgs.variableName = processedArgs.memberName;
44
+ } else if (processedArgs.nodeType === 'Event' || processedArgs.nodeType === 'CustomEvent' || (processedArgs.nodeType && processedArgs.nodeType.startsWith('K2Node_Event'))) {
45
+ if (!processedArgs.eventName) processedArgs.eventName = processedArgs.memberName;
46
+ // CustomEvent uses eventName (mapped to CustomFunctionName) or customEventName in some contexts,
47
+ // but C++ CustomEvent handler uses 'eventName' payload field.
48
+ } else if (processedArgs.nodeType === 'CallFunction' || processedArgs.nodeType === 'K2Node_CallFunction') {
49
+ // C++ uses 'memberName' for CallFunction, so this is fine.
50
+ }
51
+ }
52
+
53
+ // Fix Issue 5: Map memberClass/componentClass to targetClass for Cast nodes
54
+ if ((processedArgs.memberClass || processedArgs.componentClass) &&
55
+ (processedArgs.nodeType === 'Cast' || (processedArgs.nodeType && processedArgs.nodeType.startsWith('CastTo')))) {
56
+ if (!processedArgs.targetClass) processedArgs.targetClass = processedArgs.memberClass || processedArgs.componentClass;
57
+ }
58
+
59
+ // Fix Issue 6: Support connect_pins parameter mapping
60
+ // Input: nodeId, pinName, linkedTo (TargetNode.Pin)
61
+ if (action === 'connect_pins') {
62
+ // Map source
63
+ if (!processedArgs.fromNodeId && processedArgs.nodeId) {
64
+ processedArgs.fromNodeId = processedArgs.nodeId;
65
+ }
66
+ if (!processedArgs.fromPinName && processedArgs.pinName) {
67
+ processedArgs.fromPinName = processedArgs.pinName;
68
+ }
69
+
70
+ // Map target from linkedTo
71
+ if (!processedArgs.toNodeId && processedArgs.linkedTo) {
72
+ if (processedArgs.linkedTo.includes('.')) {
73
+ const parts = processedArgs.linkedTo.split('.');
74
+ processedArgs.toNodeId = parts[0];
75
+ processedArgs.toPinName = parts.slice(1).join('.');
76
+ }
77
+ }
78
+ }
79
+
80
+ // Support Node.Pin format for connect_pins (existing logic preserved/enhanced)
81
+ if (action === 'connect_pins') {
82
+ if (processedArgs.fromNodeId && processedArgs.fromNodeId.includes('.') && !processedArgs.fromPinName) {
83
+ const parts = processedArgs.fromNodeId.split('.');
84
+ processedArgs.fromNodeId = parts[0];
85
+ processedArgs.fromPinName = parts.slice(1).join('.');
86
+ }
87
+ if (processedArgs.toNodeId && processedArgs.toNodeId.includes('.') && !processedArgs.toPinName) {
88
+ const parts = processedArgs.toNodeId.split('.');
89
+ processedArgs.toNodeId = parts[0];
90
+ processedArgs.toPinName = parts.slice(1).join('.');
91
+ }
92
+ }
93
+
94
+ const res: any = await executeAutomationRequest(tools, 'manage_blueprint_graph', processedArgs, 'Automation bridge not available');
95
+ return cleanObject({ ...res, ...(res.result || {}) });
96
+ }
97
+
98
+ async function handleNiagaraGraph(action: string, args: any, tools: ITools) {
99
+ const payload = { ...args, subAction: action };
100
+ // Map systemPath to assetPath if missing
101
+ if (payload.systemPath && !payload.assetPath) {
102
+ payload.assetPath = payload.systemPath;
103
+ }
104
+ const res: any = await executeAutomationRequest(tools, 'manage_niagara_graph', payload, 'Automation bridge not available');
105
+ return cleanObject({ ...res, ...(res.result || {}) });
106
+ }
107
+
108
+ async function handleMaterialGraph(action: string, args: any, tools: ITools) {
109
+ const res: any = await executeAutomationRequest(tools, 'manage_material_graph', { ...args, subAction: action }, 'Automation bridge not available');
110
+ return cleanObject({ ...res, ...(res.result || {}) });
111
+ }
112
+
113
+ async function handleBehaviorTree(action: string, args: any, tools: ITools) {
114
+ const res: any = await executeAutomationRequest(tools, 'manage_behavior_tree', { ...args, subAction: action }, 'Automation bridge not available');
115
+ return cleanObject({ ...res, ...(res.result || {}) });
116
+ }
@@ -0,0 +1,28 @@
1
+ import { ITools } from '../../types/tool-interfaces.js';
2
+ import { cleanObject } from '../../utils/safe-json.js';
3
+ import { ResponseFactory } from '../../utils/response-factory.js';
4
+ import { InputTools } from '../input.js';
5
+
6
+ export async function handleInputTools(
7
+ action: string,
8
+ args: any,
9
+ tools: ITools
10
+ ) {
11
+ const inputTools = tools.inputTools as InputTools;
12
+ if (!inputTools) {
13
+ return ResponseFactory.error('Input tools not available');
14
+ }
15
+
16
+ switch (action) {
17
+ case 'create_input_action':
18
+ return cleanObject(await inputTools.createInputAction(args.name, args.path));
19
+ case 'create_input_mapping_context':
20
+ return cleanObject(await inputTools.createInputMappingContext(args.name, args.path));
21
+ case 'add_mapping':
22
+ return cleanObject(await inputTools.addMapping(args.contextPath, args.actionPath, args.key));
23
+ case 'remove_mapping':
24
+ return cleanObject(await inputTools.removeMapping(args.contextPath, args.actionPath));
25
+ default:
26
+ return ResponseFactory.error(`Unknown input action: ${action}`);
27
+ }
28
+ }