unreal-engine-mcp-server 0.4.7 → 0.5.1

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 (454) 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-config.yml +51 -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 +27 -0
  19. package/.github/workflows/labeler.yml +17 -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 +13 -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 +338 -31
  29. package/CONTRIBUTING.md +140 -0
  30. package/GEMINI.md +115 -0
  31. package/Public/Plugin_setup_guide.mp4 +0 -0
  32. package/README.md +189 -128
  33. package/claude_desktop_config_example.json +7 -6
  34. package/dist/automation/bridge.d.ts +50 -0
  35. package/dist/automation/bridge.js +452 -0
  36. package/dist/automation/connection-manager.d.ts +23 -0
  37. package/dist/automation/connection-manager.js +107 -0
  38. package/dist/automation/handshake.d.ts +11 -0
  39. package/dist/automation/handshake.js +89 -0
  40. package/dist/automation/index.d.ts +3 -0
  41. package/dist/automation/index.js +3 -0
  42. package/dist/automation/message-handler.d.ts +12 -0
  43. package/dist/automation/message-handler.js +149 -0
  44. package/dist/automation/request-tracker.d.ts +25 -0
  45. package/dist/automation/request-tracker.js +98 -0
  46. package/dist/automation/types.d.ts +130 -0
  47. package/dist/automation/types.js +2 -0
  48. package/dist/cli.js +32 -5
  49. package/dist/config.d.ts +26 -0
  50. package/dist/config.js +59 -0
  51. package/dist/constants.d.ts +16 -0
  52. package/dist/constants.js +16 -0
  53. package/dist/graphql/loaders.d.ts +64 -0
  54. package/dist/graphql/loaders.js +117 -0
  55. package/dist/graphql/resolvers.d.ts +268 -0
  56. package/dist/graphql/resolvers.js +746 -0
  57. package/dist/graphql/schema.d.ts +5 -0
  58. package/dist/graphql/schema.js +437 -0
  59. package/dist/graphql/server.d.ts +26 -0
  60. package/dist/graphql/server.js +117 -0
  61. package/dist/graphql/types.d.ts +9 -0
  62. package/dist/graphql/types.js +2 -0
  63. package/dist/handlers/resource-handlers.d.ts +20 -0
  64. package/dist/handlers/resource-handlers.js +180 -0
  65. package/dist/index.d.ts +33 -18
  66. package/dist/index.js +130 -619
  67. package/dist/resources/actors.d.ts +17 -12
  68. package/dist/resources/actors.js +56 -76
  69. package/dist/resources/assets.d.ts +6 -14
  70. package/dist/resources/assets.js +115 -147
  71. package/dist/resources/levels.d.ts +13 -13
  72. package/dist/resources/levels.js +25 -34
  73. package/dist/server/resource-registry.d.ts +20 -0
  74. package/dist/server/resource-registry.js +37 -0
  75. package/dist/server/tool-registry.d.ts +23 -0
  76. package/dist/server/tool-registry.js +322 -0
  77. package/dist/server-setup.d.ts +20 -0
  78. package/dist/server-setup.js +71 -0
  79. package/dist/services/health-monitor.d.ts +34 -0
  80. package/dist/services/health-monitor.js +105 -0
  81. package/dist/services/metrics-server.d.ts +11 -0
  82. package/dist/services/metrics-server.js +105 -0
  83. package/dist/tools/actors.d.ts +163 -9
  84. package/dist/tools/actors.js +356 -311
  85. package/dist/tools/animation.d.ts +135 -4
  86. package/dist/tools/animation.js +510 -411
  87. package/dist/tools/assets.d.ts +75 -29
  88. package/dist/tools/assets.js +265 -284
  89. package/dist/tools/audio.d.ts +102 -42
  90. package/dist/tools/audio.js +272 -685
  91. package/dist/tools/base-tool.d.ts +17 -0
  92. package/dist/tools/base-tool.js +46 -0
  93. package/dist/tools/behavior-tree.d.ts +94 -0
  94. package/dist/tools/behavior-tree.js +39 -0
  95. package/dist/tools/blueprint.d.ts +208 -126
  96. package/dist/tools/blueprint.js +685 -832
  97. package/dist/tools/consolidated-tool-definitions.d.ts +5462 -1781
  98. package/dist/tools/consolidated-tool-definitions.js +829 -496
  99. package/dist/tools/consolidated-tool-handlers.d.ts +2 -1
  100. package/dist/tools/consolidated-tool-handlers.js +198 -1027
  101. package/dist/tools/debug.d.ts +143 -85
  102. package/dist/tools/debug.js +234 -180
  103. package/dist/tools/dynamic-handler-registry.d.ts +13 -0
  104. package/dist/tools/dynamic-handler-registry.js +23 -0
  105. package/dist/tools/editor.d.ts +30 -83
  106. package/dist/tools/editor.js +247 -244
  107. package/dist/tools/engine.d.ts +10 -4
  108. package/dist/tools/engine.js +13 -5
  109. package/dist/tools/environment.d.ts +30 -0
  110. package/dist/tools/environment.js +267 -0
  111. package/dist/tools/foliage.d.ts +65 -99
  112. package/dist/tools/foliage.js +221 -331
  113. package/dist/tools/handlers/actor-handlers.d.ts +3 -0
  114. package/dist/tools/handlers/actor-handlers.js +227 -0
  115. package/dist/tools/handlers/animation-handlers.d.ts +3 -0
  116. package/dist/tools/handlers/animation-handlers.js +185 -0
  117. package/dist/tools/handlers/argument-helper.d.ts +16 -0
  118. package/dist/tools/handlers/argument-helper.js +80 -0
  119. package/dist/tools/handlers/asset-handlers.d.ts +3 -0
  120. package/dist/tools/handlers/asset-handlers.js +496 -0
  121. package/dist/tools/handlers/audio-handlers.d.ts +3 -0
  122. package/dist/tools/handlers/audio-handlers.js +166 -0
  123. package/dist/tools/handlers/blueprint-handlers.d.ts +4 -0
  124. package/dist/tools/handlers/blueprint-handlers.js +358 -0
  125. package/dist/tools/handlers/common-handlers.d.ts +14 -0
  126. package/dist/tools/handlers/common-handlers.js +56 -0
  127. package/dist/tools/handlers/editor-handlers.d.ts +3 -0
  128. package/dist/tools/handlers/editor-handlers.js +119 -0
  129. package/dist/tools/handlers/effect-handlers.d.ts +3 -0
  130. package/dist/tools/handlers/effect-handlers.js +171 -0
  131. package/dist/tools/handlers/environment-handlers.d.ts +3 -0
  132. package/dist/tools/handlers/environment-handlers.js +170 -0
  133. package/dist/tools/handlers/graph-handlers.d.ts +3 -0
  134. package/dist/tools/handlers/graph-handlers.js +90 -0
  135. package/dist/tools/handlers/input-handlers.d.ts +3 -0
  136. package/dist/tools/handlers/input-handlers.js +21 -0
  137. package/dist/tools/handlers/inspect-handlers.d.ts +3 -0
  138. package/dist/tools/handlers/inspect-handlers.js +383 -0
  139. package/dist/tools/handlers/level-handlers.d.ts +3 -0
  140. package/dist/tools/handlers/level-handlers.js +237 -0
  141. package/dist/tools/handlers/lighting-handlers.d.ts +3 -0
  142. package/dist/tools/handlers/lighting-handlers.js +144 -0
  143. package/dist/tools/handlers/performance-handlers.d.ts +3 -0
  144. package/dist/tools/handlers/performance-handlers.js +130 -0
  145. package/dist/tools/handlers/pipeline-handlers.d.ts +3 -0
  146. package/dist/tools/handlers/pipeline-handlers.js +110 -0
  147. package/dist/tools/handlers/sequence-handlers.d.ts +3 -0
  148. package/dist/tools/handlers/sequence-handlers.js +376 -0
  149. package/dist/tools/handlers/system-handlers.d.ts +4 -0
  150. package/dist/tools/handlers/system-handlers.js +506 -0
  151. package/dist/tools/input.d.ts +19 -0
  152. package/dist/tools/input.js +89 -0
  153. package/dist/tools/introspection.d.ts +103 -40
  154. package/dist/tools/introspection.js +425 -568
  155. package/dist/tools/landscape.d.ts +54 -93
  156. package/dist/tools/landscape.js +284 -409
  157. package/dist/tools/level.d.ts +66 -27
  158. package/dist/tools/level.js +647 -675
  159. package/dist/tools/lighting.d.ts +77 -38
  160. package/dist/tools/lighting.js +445 -943
  161. package/dist/tools/logs.d.ts +3 -3
  162. package/dist/tools/logs.js +5 -57
  163. package/dist/tools/materials.d.ts +91 -24
  164. package/dist/tools/materials.js +194 -118
  165. package/dist/tools/niagara.d.ts +149 -39
  166. package/dist/tools/niagara.js +267 -182
  167. package/dist/tools/performance.d.ts +27 -13
  168. package/dist/tools/performance.js +203 -122
  169. package/dist/tools/physics.d.ts +32 -77
  170. package/dist/tools/physics.js +175 -582
  171. package/dist/tools/property-dictionary.d.ts +13 -0
  172. package/dist/tools/property-dictionary.js +82 -0
  173. package/dist/tools/sequence.d.ts +85 -60
  174. package/dist/tools/sequence.js +208 -747
  175. package/dist/tools/tool-definition-utils.d.ts +59 -0
  176. package/dist/tools/tool-definition-utils.js +35 -0
  177. package/dist/tools/ui.d.ts +64 -34
  178. package/dist/tools/ui.js +134 -214
  179. package/dist/types/automation-responses.d.ts +115 -0
  180. package/dist/types/automation-responses.js +2 -0
  181. package/dist/types/env.d.ts +0 -3
  182. package/dist/types/env.js +0 -7
  183. package/dist/types/responses.d.ts +249 -0
  184. package/dist/types/responses.js +2 -0
  185. package/dist/types/tool-interfaces.d.ts +898 -0
  186. package/dist/types/tool-interfaces.js +2 -0
  187. package/dist/types/tool-types.d.ts +183 -19
  188. package/dist/types/tool-types.js +0 -4
  189. package/dist/unreal-bridge.d.ts +24 -131
  190. package/dist/unreal-bridge.js +364 -1506
  191. package/dist/utils/command-validator.d.ts +9 -0
  192. package/dist/utils/command-validator.js +68 -0
  193. package/dist/utils/elicitation.d.ts +1 -1
  194. package/dist/utils/elicitation.js +12 -15
  195. package/dist/utils/error-handler.d.ts +2 -51
  196. package/dist/utils/error-handler.js +11 -87
  197. package/dist/utils/ini-reader.d.ts +3 -0
  198. package/dist/utils/ini-reader.js +69 -0
  199. package/dist/utils/logger.js +9 -6
  200. package/dist/utils/normalize.d.ts +3 -0
  201. package/dist/utils/normalize.js +56 -0
  202. package/dist/utils/path-security.d.ts +2 -0
  203. package/dist/utils/path-security.js +24 -0
  204. package/dist/utils/response-factory.d.ts +7 -0
  205. package/dist/utils/response-factory.js +27 -0
  206. package/dist/utils/response-validator.d.ts +3 -24
  207. package/dist/utils/response-validator.js +130 -81
  208. package/dist/utils/result-helpers.d.ts +4 -5
  209. package/dist/utils/result-helpers.js +15 -16
  210. package/dist/utils/safe-json.js +5 -11
  211. package/dist/utils/unreal-command-queue.d.ts +24 -0
  212. package/dist/utils/unreal-command-queue.js +120 -0
  213. package/dist/utils/validation.d.ts +0 -40
  214. package/dist/utils/validation.js +1 -78
  215. package/dist/wasm/index.d.ts +70 -0
  216. package/dist/wasm/index.js +535 -0
  217. package/docs/GraphQL-API.md +888 -0
  218. package/docs/Migration-Guide-v0.5.0.md +684 -0
  219. package/docs/Roadmap.md +53 -0
  220. package/docs/WebAssembly-Integration.md +628 -0
  221. package/docs/editor-plugin-extension.md +370 -0
  222. package/docs/handler-mapping.md +242 -0
  223. package/docs/native-automation-progress.md +128 -0
  224. package/docs/testing-guide.md +423 -0
  225. package/mcp-config-example.json +6 -6
  226. package/package.json +67 -28
  227. package/plugins/McpAutomationBridge/Config/FilterPlugin.ini +8 -0
  228. package/plugins/McpAutomationBridge/McpAutomationBridge.uplugin +64 -0
  229. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/McpAutomationBridge.Build.cs +189 -0
  230. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeGlobals.cpp +22 -0
  231. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeGlobals.h +30 -0
  232. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeHelpers.h +1983 -0
  233. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeModule.cpp +72 -0
  234. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeSettings.cpp +46 -0
  235. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeSubsystem.cpp +581 -0
  236. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AnimationHandlers.cpp +2394 -0
  237. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AssetQueryHandlers.cpp +300 -0
  238. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AssetWorkflowHandlers.cpp +2807 -0
  239. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AudioHandlers.cpp +1087 -0
  240. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BehaviorTreeHandlers.cpp +488 -0
  241. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintCreationHandlers.cpp +643 -0
  242. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintCreationHandlers.h +31 -0
  243. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintGraphHandlers.cpp +1184 -0
  244. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintHandlers.cpp +5652 -0
  245. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintHandlers_List.cpp +152 -0
  246. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_ControlHandlers.cpp +2614 -0
  247. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_DebugHandlers.cpp +42 -0
  248. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_EditorFunctionHandlers.cpp +1237 -0
  249. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_EffectHandlers.cpp +1701 -0
  250. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_EnvironmentHandlers.cpp +2145 -0
  251. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_FoliageHandlers.cpp +954 -0
  252. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_InputHandlers.cpp +209 -0
  253. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_InsightsHandlers.cpp +41 -0
  254. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LandscapeHandlers.cpp +1164 -0
  255. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LevelHandlers.cpp +762 -0
  256. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LightingHandlers.cpp +634 -0
  257. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LogHandlers.cpp +136 -0
  258. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_MaterialGraphHandlers.cpp +494 -0
  259. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_NiagaraGraphHandlers.cpp +278 -0
  260. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_NiagaraHandlers.cpp +625 -0
  261. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_PerformanceHandlers.cpp +401 -0
  262. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_PipelineHandlers.cpp +67 -0
  263. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_ProcessRequest.cpp +735 -0
  264. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_PropertyHandlers.cpp +2634 -0
  265. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_RenderHandlers.cpp +189 -0
  266. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SCSHandlers.cpp +917 -0
  267. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SCSHandlers.h +39 -0
  268. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SequenceHandlers.cpp +2670 -0
  269. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SequencerHandlers.cpp +519 -0
  270. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_TestHandlers.cpp +38 -0
  271. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_UiHandlers.cpp +668 -0
  272. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_WorldPartitionHandlers.cpp +346 -0
  273. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpBridgeWebSocket.cpp +1330 -0
  274. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpBridgeWebSocket.h +149 -0
  275. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpConnectionManager.cpp +783 -0
  276. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Public/McpAutomationBridgeSettings.h +115 -0
  277. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Public/McpAutomationBridgeSubsystem.h +796 -0
  278. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Public/McpConnectionManager.h +117 -0
  279. package/scripts/check-unreal-connection.mjs +19 -0
  280. package/scripts/clean-tmp.js +23 -0
  281. package/scripts/patch-wasm.js +26 -0
  282. package/scripts/run-all-tests.mjs +136 -0
  283. package/scripts/smoke-test.ts +94 -0
  284. package/scripts/sync-mcp-plugin.js +143 -0
  285. package/scripts/test-no-plugin-alternates.mjs +113 -0
  286. package/scripts/validate-server.js +46 -0
  287. package/scripts/verify-automation-bridge.js +200 -0
  288. package/server.json +58 -21
  289. package/src/automation/bridge.ts +558 -0
  290. package/src/automation/connection-manager.ts +130 -0
  291. package/src/automation/handshake.ts +99 -0
  292. package/src/automation/index.ts +2 -0
  293. package/src/automation/message-handler.ts +167 -0
  294. package/src/automation/request-tracker.ts +123 -0
  295. package/src/automation/types.ts +107 -0
  296. package/src/cli.ts +33 -6
  297. package/src/config.ts +73 -0
  298. package/src/constants.ts +19 -0
  299. package/src/graphql/loaders.ts +244 -0
  300. package/src/graphql/resolvers.ts +1008 -0
  301. package/src/graphql/schema.ts +452 -0
  302. package/src/graphql/server.ts +156 -0
  303. package/src/graphql/types.ts +10 -0
  304. package/src/handlers/resource-handlers.ts +186 -0
  305. package/src/index.ts +166 -664
  306. package/src/resources/actors.ts +58 -76
  307. package/src/resources/assets.ts +148 -134
  308. package/src/resources/levels.ts +28 -33
  309. package/src/server/resource-registry.ts +47 -0
  310. package/src/server/tool-registry.ts +354 -0
  311. package/src/server-setup.ts +114 -0
  312. package/src/services/health-monitor.ts +132 -0
  313. package/src/services/metrics-server.ts +142 -0
  314. package/src/tools/actors.ts +426 -323
  315. package/src/tools/animation.ts +672 -461
  316. package/src/tools/assets.ts +364 -289
  317. package/src/tools/audio.ts +323 -766
  318. package/src/tools/base-tool.ts +52 -0
  319. package/src/tools/behavior-tree.ts +45 -0
  320. package/src/tools/blueprint.ts +792 -970
  321. package/src/tools/consolidated-tool-definitions.ts +993 -515
  322. package/src/tools/consolidated-tool-handlers.ts +258 -1146
  323. package/src/tools/debug.ts +292 -187
  324. package/src/tools/dynamic-handler-registry.ts +33 -0
  325. package/src/tools/editor.ts +329 -253
  326. package/src/tools/engine.ts +14 -3
  327. package/src/tools/environment.ts +281 -0
  328. package/src/tools/foliage.ts +330 -392
  329. package/src/tools/handlers/actor-handlers.ts +265 -0
  330. package/src/tools/handlers/animation-handlers.ts +237 -0
  331. package/src/tools/handlers/argument-helper.ts +142 -0
  332. package/src/tools/handlers/asset-handlers.ts +532 -0
  333. package/src/tools/handlers/audio-handlers.ts +194 -0
  334. package/src/tools/handlers/blueprint-handlers.ts +380 -0
  335. package/src/tools/handlers/common-handlers.ts +87 -0
  336. package/src/tools/handlers/editor-handlers.ts +123 -0
  337. package/src/tools/handlers/effect-handlers.ts +220 -0
  338. package/src/tools/handlers/environment-handlers.ts +183 -0
  339. package/src/tools/handlers/graph-handlers.ts +116 -0
  340. package/src/tools/handlers/input-handlers.ts +28 -0
  341. package/src/tools/handlers/inspect-handlers.ts +450 -0
  342. package/src/tools/handlers/level-handlers.ts +252 -0
  343. package/src/tools/handlers/lighting-handlers.ts +147 -0
  344. package/src/tools/handlers/performance-handlers.ts +132 -0
  345. package/src/tools/handlers/pipeline-handlers.ts +127 -0
  346. package/src/tools/handlers/sequence-handlers.ts +415 -0
  347. package/src/tools/handlers/system-handlers.ts +564 -0
  348. package/src/tools/input.ts +101 -0
  349. package/src/tools/introspection.ts +493 -584
  350. package/src/tools/landscape.ts +418 -507
  351. package/src/tools/level.ts +786 -708
  352. package/src/tools/lighting.ts +588 -984
  353. package/src/tools/logs.ts +9 -57
  354. package/src/tools/materials.ts +237 -121
  355. package/src/tools/niagara.ts +335 -168
  356. package/src/tools/performance.ts +320 -169
  357. package/src/tools/physics.ts +274 -613
  358. package/src/tools/property-dictionary.ts +98 -0
  359. package/src/tools/sequence.ts +276 -820
  360. package/src/tools/tool-definition-utils.ts +35 -0
  361. package/src/tools/ui.ts +205 -283
  362. package/src/types/automation-responses.ts +119 -0
  363. package/src/types/env.ts +0 -10
  364. package/src/types/responses.ts +355 -0
  365. package/src/types/tool-interfaces.ts +250 -0
  366. package/src/types/tool-types.ts +243 -21
  367. package/src/unreal-bridge.ts +460 -1550
  368. package/src/utils/command-validator.ts +76 -0
  369. package/src/utils/elicitation.ts +10 -7
  370. package/src/utils/error-handler.ts +14 -90
  371. package/src/utils/ini-reader.ts +86 -0
  372. package/src/utils/logger.ts +8 -3
  373. package/src/utils/normalize.test.ts +162 -0
  374. package/src/utils/normalize.ts +60 -0
  375. package/src/utils/path-security.ts +43 -0
  376. package/src/utils/response-factory.ts +44 -0
  377. package/src/utils/response-validator.ts +176 -56
  378. package/src/utils/result-helpers.ts +21 -19
  379. package/src/utils/safe-json.test.ts +90 -0
  380. package/src/utils/safe-json.ts +14 -11
  381. package/src/utils/unreal-command-queue.ts +152 -0
  382. package/src/utils/validation.test.ts +184 -0
  383. package/src/utils/validation.ts +4 -1
  384. package/src/wasm/index.ts +838 -0
  385. package/test-server.mjs +100 -0
  386. package/tests/run-unreal-tool-tests.mjs +242 -14
  387. package/tests/test-animation.mjs +369 -0
  388. package/tests/test-asset-advanced.mjs +82 -0
  389. package/tests/test-asset-errors.mjs +35 -0
  390. package/tests/test-asset-graph.mjs +311 -0
  391. package/tests/test-audio.mjs +417 -0
  392. package/tests/test-automation-timeouts.mjs +98 -0
  393. package/tests/test-behavior-tree.mjs +444 -0
  394. package/tests/test-blueprint-graph.mjs +410 -0
  395. package/tests/test-blueprint.mjs +577 -0
  396. package/tests/test-client-mode.mjs +86 -0
  397. package/tests/test-console-command.mjs +56 -0
  398. package/tests/test-control-actor.mjs +425 -0
  399. package/tests/test-control-editor.mjs +112 -0
  400. package/tests/test-graphql.mjs +372 -0
  401. package/tests/test-input.mjs +349 -0
  402. package/tests/test-inspect.mjs +302 -0
  403. package/tests/test-landscape.mjs +316 -0
  404. package/tests/test-lighting.mjs +428 -0
  405. package/tests/test-manage-asset.mjs +438 -0
  406. package/tests/test-manage-level.mjs +89 -0
  407. package/tests/test-materials.mjs +356 -0
  408. package/tests/test-niagara.mjs +185 -0
  409. package/tests/test-no-inline-python.mjs +122 -0
  410. package/tests/test-performance.mjs +539 -0
  411. package/tests/test-plugin-handshake.mjs +82 -0
  412. package/tests/test-runner.mjs +933 -0
  413. package/tests/test-sequence.mjs +104 -0
  414. package/tests/test-system.mjs +96 -0
  415. package/tests/test-wasm.mjs +283 -0
  416. package/tests/test-world-partition.mjs +215 -0
  417. package/tsconfig.json +3 -3
  418. package/vitest.config.ts +35 -0
  419. package/wasm/Cargo.lock +363 -0
  420. package/wasm/Cargo.toml +42 -0
  421. package/wasm/LICENSE +21 -0
  422. package/wasm/README.md +253 -0
  423. package/wasm/src/dependency_resolver.rs +377 -0
  424. package/wasm/src/lib.rs +153 -0
  425. package/wasm/src/property_parser.rs +271 -0
  426. package/wasm/src/transform_math.rs +396 -0
  427. package/wasm/tests/integration.rs +109 -0
  428. package/.github/workflows/smithery-build.yml +0 -29
  429. package/dist/prompts/index.d.ts +0 -21
  430. package/dist/prompts/index.js +0 -217
  431. package/dist/tools/build_environment_advanced.d.ts +0 -65
  432. package/dist/tools/build_environment_advanced.js +0 -633
  433. package/dist/tools/rc.d.ts +0 -110
  434. package/dist/tools/rc.js +0 -437
  435. package/dist/tools/visual.d.ts +0 -40
  436. package/dist/tools/visual.js +0 -282
  437. package/dist/utils/http.d.ts +0 -6
  438. package/dist/utils/http.js +0 -151
  439. package/dist/utils/python-output.d.ts +0 -18
  440. package/dist/utils/python-output.js +0 -290
  441. package/dist/utils/python.d.ts +0 -2
  442. package/dist/utils/python.js +0 -4
  443. package/dist/utils/stdio-redirect.d.ts +0 -2
  444. package/dist/utils/stdio-redirect.js +0 -20
  445. package/docs/unreal-tool-test-cases.md +0 -574
  446. package/smithery.yaml +0 -29
  447. package/src/prompts/index.ts +0 -249
  448. package/src/tools/build_environment_advanced.ts +0 -732
  449. package/src/tools/rc.ts +0 -515
  450. package/src/tools/visual.ts +0 -281
  451. package/src/utils/http.ts +0 -187
  452. package/src/utils/python-output.ts +0 -351
  453. package/src/utils/python.ts +0 -3
  454. package/src/utils/stdio-redirect.ts +0 -18
@@ -0,0 +1,136 @@
1
+ #include "McpAutomationBridgeSubsystem.h"
2
+ #include "McpAutomationBridgeHelpers.h"
3
+ #include "McpAutomationBridgeGlobals.h"
4
+ #include "Misc/OutputDevice.h"
5
+ #include "Async/Async.h"
6
+
7
+ // Define a custom output device to capture logs and stream them via the bridge
8
+ class FMcpLogOutputDevice : public FOutputDevice
9
+ {
10
+ public:
11
+ FMcpLogOutputDevice(UMcpAutomationBridgeSubsystem* InSubsystem)
12
+ : Subsystem(InSubsystem)
13
+ {
14
+ }
15
+
16
+ virtual void Serialize(const TCHAR* V, ELogVerbosity::Type Verbosity, const class FName& Category) override
17
+ {
18
+ if (!Subsystem || !Subsystem->IsValidLowLevel())
19
+ {
20
+ return;
21
+ }
22
+
23
+ // Filter out very verbose logs if needed, but for now allow all
24
+ // Prevent infinite recursion if our own logging causes more logging
25
+ // Filter out highly verbose categories that clutter test output
26
+ // Use string comparison to be robust against FName issues
27
+ FString CategoryStr = Category.ToString();
28
+
29
+ if (Category == LogMcpAutomationBridgeSubsystem.GetCategoryName() ||
30
+ CategoryStr == TEXT("LogRHI") ||
31
+ CategoryStr == TEXT("LogEOSSDK") ||
32
+ CategoryStr == TEXT("LogCsvProfiler"))
33
+ {
34
+ return;
35
+ }
36
+
37
+ // Filter specific noisy warnings
38
+ if (Verbosity == ELogVerbosity::Warning && CategoryStr == TEXT("LogSlateStyle"))
39
+ {
40
+ // "Missing Resource from 'ProfileVisualizerStyle'" is a known engine warning during 'show collision'
41
+ if (FString(V).Contains(TEXT("Missing Resource from 'ProfileVisualizerStyle'")))
42
+ {
43
+ return;
44
+ }
45
+ }
46
+
47
+ if (CategoryStr == TEXT("LogStats"))
48
+ {
49
+ // "There is no thread with id" is noise during stat commands
50
+ if (FString(V).Contains(TEXT("There is no thread with id")))
51
+ {
52
+ return;
53
+ }
54
+ }
55
+
56
+ FString VerbosityString;
57
+ switch (Verbosity)
58
+ {
59
+ case ELogVerbosity::Fatal: VerbosityString = TEXT("Fatal"); break;
60
+ case ELogVerbosity::Error: VerbosityString = TEXT("Error"); break;
61
+ case ELogVerbosity::Warning: VerbosityString = TEXT("Warning"); break;
62
+ case ELogVerbosity::Display: VerbosityString = TEXT("Display"); break;
63
+ case ELogVerbosity::Log: VerbosityString = TEXT("Log"); break;
64
+ case ELogVerbosity::Verbose: VerbosityString = TEXT("Verbose"); break;
65
+ case ELogVerbosity::VeryVerbose: VerbosityString = TEXT("VeryVerbose"); break;
66
+ default: VerbosityString = TEXT("Log"); break;
67
+ }
68
+
69
+ FString Message = FString(V);
70
+ FString CategoryString = Category.ToString();
71
+
72
+ // Dispatch to game thread to ensure safe socket sending if not already there
73
+ // Actually, SendRawMessage might be thread safe, but let's be safe.
74
+ // Copy data for lambda capture
75
+ const FString PayloadJson = FString::Printf(TEXT("{\"event\":\"log\",\"category\":\"%s\",\"verbosity\":\"%s\",\"message\":\"%s\"}"),
76
+ *CategoryString, *VerbosityString, *Message.ReplaceCharWithEscapedChar());
77
+
78
+ // Use a weak pointer to the subsystem to avoid crashing if it's destroyed
79
+ TWeakObjectPtr<UMcpAutomationBridgeSubsystem> WeakSubsystem(Subsystem);
80
+
81
+ AsyncTask(ENamedThreads::GameThread, [WeakSubsystem, PayloadJson]()
82
+ {
83
+ if (UMcpAutomationBridgeSubsystem* StrongSubsystem = WeakSubsystem.Get())
84
+ {
85
+ StrongSubsystem->SendRawMessage(PayloadJson);
86
+ }
87
+ });
88
+ }
89
+
90
+ private:
91
+ UMcpAutomationBridgeSubsystem* Subsystem;
92
+ };
93
+
94
+ bool UMcpAutomationBridgeSubsystem::HandleLogAction(const FString& RequestId, const FString& Action, const TSharedPtr<FJsonObject>& Payload, TSharedPtr<FMcpBridgeWebSocket> RequestingSocket)
95
+ {
96
+ if (Action != TEXT("manage_logs"))
97
+ {
98
+ return false;
99
+ }
100
+
101
+ if (!Payload.IsValid())
102
+ {
103
+ SendAutomationError(RequestingSocket, RequestId, TEXT("Missing payload."), TEXT("INVALID_PAYLOAD"));
104
+ return true;
105
+ }
106
+
107
+ FString SubAction = Payload->GetStringField(TEXT("subAction"));
108
+
109
+ if (SubAction == TEXT("subscribe"))
110
+ {
111
+ if (!LogCaptureDevice.IsValid())
112
+ {
113
+ LogCaptureDevice = MakeShared<FMcpLogOutputDevice>(this);
114
+ GLog->AddOutputDevice(LogCaptureDevice.Get());
115
+ UE_LOG(LogMcpAutomationBridgeSubsystem, Display, TEXT("Log streaming enabled by client request."));
116
+ }
117
+
118
+ SendAutomationResponse(RequestingSocket, RequestId, true, TEXT("Subscribed to editor logs."));
119
+ return true;
120
+ }
121
+ else if (SubAction == TEXT("unsubscribe"))
122
+ {
123
+ if (LogCaptureDevice.IsValid())
124
+ {
125
+ GLog->RemoveOutputDevice(LogCaptureDevice.Get());
126
+ LogCaptureDevice.Reset();
127
+ UE_LOG(LogMcpAutomationBridgeSubsystem, Display, TEXT("Log streaming disabled by client request."));
128
+ }
129
+
130
+ SendAutomationResponse(RequestingSocket, RequestId, true, TEXT("Unsubscribed from editor logs."));
131
+ return true;
132
+ }
133
+
134
+ SendAutomationError(RequestingSocket, RequestId, TEXT("Unknown subAction."), TEXT("INVALID_SUBACTION"));
135
+ return true;
136
+ }
@@ -0,0 +1,494 @@
1
+ #include "McpAutomationBridgeGlobals.h"
2
+ #include "McpAutomationBridgeHelpers.h"
3
+ #include "McpAutomationBridgeSubsystem.h"
4
+
5
+ #if WITH_EDITOR
6
+ #include "EdGraph/EdGraph.h"
7
+ #include "EdGraph/EdGraphSchema.h"
8
+ #include "Materials/Material.h"
9
+ #include "Materials/MaterialExpression.h"
10
+ #include "Materials/MaterialExpressionAdd.h"
11
+ #include "Materials/MaterialExpressionConstant.h"
12
+ #include "Materials/MaterialExpressionConstant3Vector.h"
13
+ #include "Materials/MaterialExpressionMultiply.h"
14
+ #include "Materials/MaterialExpressionScalarParameter.h"
15
+ #include "Materials/MaterialExpressionTextureSample.h"
16
+ #include "Materials/MaterialExpressionVectorParameter.h"
17
+ #endif
18
+
19
+ bool UMcpAutomationBridgeSubsystem::HandleMaterialGraphAction(
20
+ const FString &RequestId, const FString &Action,
21
+ const TSharedPtr<FJsonObject> &Payload,
22
+ TSharedPtr<FMcpBridgeWebSocket> Socket) {
23
+ if (Action != TEXT("manage_material_graph")) {
24
+ return false;
25
+ }
26
+
27
+ #if WITH_EDITOR
28
+ if (!Payload.IsValid()) {
29
+ SendAutomationError(Socket, RequestId, TEXT("Missing payload."),
30
+ TEXT("INVALID_PAYLOAD"));
31
+ return true;
32
+ }
33
+
34
+ FString AssetPath;
35
+ if (!Payload->TryGetStringField(TEXT("assetPath"), AssetPath) ||
36
+ AssetPath.IsEmpty()) {
37
+ SendAutomationError(Socket, RequestId, TEXT("Missing 'assetPath'."),
38
+ TEXT("INVALID_ARGUMENT"));
39
+ return true;
40
+ }
41
+
42
+ UMaterial *Material = LoadObject<UMaterial>(nullptr, *AssetPath);
43
+ if (!Material) {
44
+ SendAutomationError(Socket, RequestId, TEXT("Could not load Material."),
45
+ TEXT("ASSET_NOT_FOUND"));
46
+ return true;
47
+ }
48
+
49
+ FString SubAction;
50
+ if (!Payload->TryGetStringField(TEXT("subAction"), SubAction) ||
51
+ SubAction.IsEmpty()) {
52
+ SendAutomationError(Socket, RequestId,
53
+ TEXT("Missing 'subAction' for manage_material_graph"),
54
+ TEXT("INVALID_ARGUMENT"));
55
+ return true;
56
+ }
57
+
58
+ auto FindExpressionByIdOrName =
59
+ [&](const FString &IdOrName) -> UMaterialExpression * {
60
+ if (IdOrName.IsEmpty()) {
61
+ return nullptr;
62
+ }
63
+
64
+ const FString Needle = IdOrName.TrimStartAndEnd();
65
+ for (UMaterialExpression *Expr : Material->GetExpressions()) {
66
+ if (!Expr) {
67
+ continue;
68
+ }
69
+ if (Expr->MaterialExpressionGuid.ToString() == Needle) {
70
+ return Expr;
71
+ }
72
+ if (Expr->GetName() == Needle) {
73
+ return Expr;
74
+ }
75
+ // Some callers may pass a full object path.
76
+ if (Expr->GetPathName() == Needle) {
77
+ return Expr;
78
+ }
79
+
80
+ // Also check ParameterName if it's a parameter node
81
+ if (UMaterialExpressionParameter *ParamExpr =
82
+ Cast<UMaterialExpressionParameter>(Expr)) {
83
+ if (ParamExpr->ParameterName.ToString() == Needle) {
84
+ return Expr;
85
+ }
86
+ }
87
+ }
88
+ return nullptr;
89
+ };
90
+
91
+ if (SubAction == TEXT("add_node")) {
92
+ FString NodeType;
93
+ Payload->TryGetStringField(TEXT("nodeType"), NodeType);
94
+ float X = 0.0f;
95
+ float Y = 0.0f;
96
+ Payload->TryGetNumberField(TEXT("x"), X);
97
+ Payload->TryGetNumberField(TEXT("y"), Y);
98
+
99
+ UClass *ExpressionClass = nullptr;
100
+ if (NodeType == TEXT("TextureSample"))
101
+ ExpressionClass = UMaterialExpressionTextureSample::StaticClass();
102
+ else if (NodeType == TEXT("VectorParameter") ||
103
+ NodeType == TEXT("ConstantVectorParameter"))
104
+ ExpressionClass = UMaterialExpressionVectorParameter::StaticClass();
105
+ else if (NodeType == TEXT("ScalarParameter") ||
106
+ NodeType == TEXT("ConstantScalarParameter"))
107
+ ExpressionClass = UMaterialExpressionScalarParameter::StaticClass();
108
+ else if (NodeType == TEXT("Add"))
109
+ ExpressionClass = UMaterialExpressionAdd::StaticClass();
110
+ else if (NodeType == TEXT("Multiply"))
111
+ ExpressionClass = UMaterialExpressionMultiply::StaticClass();
112
+ else if (NodeType == TEXT("Constant") || NodeType == TEXT("Float") ||
113
+ NodeType == TEXT("Scalar"))
114
+ ExpressionClass = UMaterialExpressionConstant::StaticClass();
115
+ else if (NodeType == TEXT("Constant3Vector") ||
116
+ NodeType == TEXT("ConstantVector") || NodeType == TEXT("Color") ||
117
+ NodeType == TEXT("Vector3"))
118
+ ExpressionClass = UMaterialExpressionConstant3Vector::StaticClass();
119
+ else {
120
+ // Try resolve class by full path or partial name
121
+ ExpressionClass = ResolveClassByName(NodeType);
122
+ // Also try with MaterialExpression prefix
123
+ if (!ExpressionClass ||
124
+ !ExpressionClass->IsChildOf(UMaterialExpression::StaticClass())) {
125
+ FString PrefixedName =
126
+ FString::Printf(TEXT("MaterialExpression%s"), *NodeType);
127
+ ExpressionClass = ResolveClassByName(PrefixedName);
128
+ }
129
+ if (!ExpressionClass ||
130
+ !ExpressionClass->IsChildOf(UMaterialExpression::StaticClass())) {
131
+ // Provide helpful error with available types
132
+ SendAutomationError(
133
+ Socket, RequestId,
134
+ FString::Printf(
135
+ TEXT("Unknown node type: %s. Available types: TextureSample, "
136
+ "VectorParameter, ScalarParameter, Add, Multiply, "
137
+ "Constant, Constant3Vector, "
138
+ "Color, ConstantVectorParameter. Or use full class name "
139
+ "like 'MaterialExpressionLerp'."),
140
+ *NodeType),
141
+ TEXT("UNKNOWN_TYPE"));
142
+ return true;
143
+ }
144
+ }
145
+
146
+ UMaterialExpression *NewExpr = NewObject<UMaterialExpression>(
147
+ Material, ExpressionClass, NAME_None, RF_Transactional);
148
+ if (NewExpr) {
149
+ NewExpr->MaterialExpressionEditorX = (int32)X;
150
+ NewExpr->MaterialExpressionEditorY = (int32)Y;
151
+ #if WITH_EDITORONLY_DATA
152
+ if (Material->GetEditorOnlyData()) {
153
+ Material->GetEditorOnlyData()->ExpressionCollection.Expressions.Add(
154
+ NewExpr);
155
+ }
156
+ #endif
157
+
158
+ // If parameter, set name
159
+ FString ParamName;
160
+ if (Payload->TryGetStringField(TEXT("name"), ParamName)) {
161
+ if (UMaterialExpressionParameter *ParamExpr =
162
+ Cast<UMaterialExpressionParameter>(NewExpr)) {
163
+ ParamExpr->ParameterName = FName(*ParamName);
164
+ }
165
+ }
166
+
167
+ Material->PostEditChange();
168
+ Material->MarkPackageDirty();
169
+
170
+ TSharedPtr<FJsonObject> Result = MakeShared<FJsonObject>();
171
+ Result->SetStringField(TEXT("nodeId"),
172
+ NewExpr->MaterialExpressionGuid.ToString());
173
+ SendAutomationResponse(Socket, RequestId, true, TEXT("Node added."),
174
+ Result);
175
+ } else {
176
+ SendAutomationError(Socket, RequestId,
177
+ TEXT("Failed to create expression."),
178
+ TEXT("CREATE_FAILED"));
179
+ }
180
+ return true;
181
+ } else if (SubAction == TEXT("remove_node")) {
182
+ FString NodeId;
183
+ Payload->TryGetStringField(TEXT("nodeId"), NodeId);
184
+
185
+ if (NodeId.IsEmpty()) {
186
+ SendAutomationError(Socket, RequestId, TEXT("Missing 'nodeId'."),
187
+ TEXT("INVALID_ARGUMENT"));
188
+ return true;
189
+ }
190
+
191
+ UMaterialExpression *TargetExpr = FindExpressionByIdOrName(NodeId);
192
+
193
+ if (TargetExpr) {
194
+ #if WITH_EDITORONLY_DATA
195
+ if (Material->GetEditorOnlyData()) {
196
+ Material->GetEditorOnlyData()->ExpressionCollection.Expressions.Remove(
197
+ TargetExpr);
198
+ }
199
+ #endif
200
+ Material->PostEditChange();
201
+ Material->MarkPackageDirty();
202
+ SendAutomationResponse(Socket, RequestId, true, TEXT("Node removed."));
203
+ } else {
204
+ SendAutomationError(Socket, RequestId, TEXT("Node not found."),
205
+ TEXT("NODE_NOT_FOUND"));
206
+ }
207
+ return true;
208
+ } else if (SubAction == TEXT("connect_nodes") ||
209
+ SubAction == TEXT("connect_pins")) {
210
+ // Material graph connections are complex because inputs are structs on the
211
+ // expression, not EdGraph pins We need to find the target expression and
212
+ // set its input
213
+ FString SourceNodeId, TargetNodeId, InputName;
214
+ Payload->TryGetStringField(TEXT("sourceNodeId"), SourceNodeId);
215
+ Payload->TryGetStringField(TEXT("targetNodeId"), TargetNodeId);
216
+ Payload->TryGetStringField(TEXT("inputName"), InputName);
217
+
218
+ UMaterialExpression *SourceExpr = FindExpressionByIdOrName(SourceNodeId);
219
+
220
+ if (!SourceExpr) {
221
+ SendAutomationError(Socket, RequestId, TEXT("Source node not found."),
222
+ TEXT("NODE_NOT_FOUND"));
223
+ return true;
224
+ }
225
+
226
+ // Target could be another expression OR the main material node (if
227
+ // TargetNodeId is empty or "Main")
228
+ if (TargetNodeId.IsEmpty() || TargetNodeId == TEXT("Main")) {
229
+ bool bFound = false;
230
+ #if WITH_EDITORONLY_DATA
231
+ if (InputName == TEXT("BaseColor")) {
232
+ Material->GetEditorOnlyData()->BaseColor.Expression = SourceExpr;
233
+ bFound = true;
234
+ } else if (InputName == TEXT("EmissiveColor")) {
235
+ Material->GetEditorOnlyData()->EmissiveColor.Expression = SourceExpr;
236
+ bFound = true;
237
+ } else if (InputName == TEXT("Roughness")) {
238
+ Material->GetEditorOnlyData()->Roughness.Expression = SourceExpr;
239
+ bFound = true;
240
+ } else if (InputName == TEXT("Metallic")) {
241
+ Material->GetEditorOnlyData()->Metallic.Expression = SourceExpr;
242
+ bFound = true;
243
+ } else if (InputName == TEXT("Specular")) {
244
+ Material->GetEditorOnlyData()->Specular.Expression = SourceExpr;
245
+ bFound = true;
246
+ } else if (InputName == TEXT("Normal")) {
247
+ Material->GetEditorOnlyData()->Normal.Expression = SourceExpr;
248
+ bFound = true;
249
+ } else if (InputName == TEXT("Opacity")) {
250
+ Material->GetEditorOnlyData()->Opacity.Expression = SourceExpr;
251
+ bFound = true;
252
+ } else if (InputName == TEXT("OpacityMask")) {
253
+ Material->GetEditorOnlyData()->OpacityMask.Expression = SourceExpr;
254
+ bFound = true;
255
+ } else if (InputName == TEXT("AmbientOcclusion")) {
256
+ Material->GetEditorOnlyData()->AmbientOcclusion.Expression = SourceExpr;
257
+ bFound = true;
258
+ } else if (InputName == TEXT("SubsurfaceColor")) {
259
+ Material->GetEditorOnlyData()->SubsurfaceColor.Expression = SourceExpr;
260
+ bFound = true;
261
+ }
262
+ #endif
263
+
264
+ if (bFound) {
265
+ Material->PostEditChange();
266
+ Material->MarkPackageDirty();
267
+ SendAutomationResponse(Socket, RequestId, true,
268
+ TEXT("Connected to main material node."));
269
+ } else {
270
+ SendAutomationError(
271
+ Socket, RequestId,
272
+ FString::Printf(TEXT("Unknown input on main node: %s"), *InputName),
273
+ TEXT("INVALID_PIN"));
274
+ }
275
+ return true;
276
+ } else {
277
+ UMaterialExpression *TargetExpr = FindExpressionByIdOrName(TargetNodeId);
278
+
279
+ if (TargetExpr) {
280
+ // We have to iterate properties to find the FExpressionInput
281
+ FProperty *Prop =
282
+ TargetExpr->GetClass()->FindPropertyByName(FName(*InputName));
283
+ if (Prop) {
284
+ if (FStructProperty *StructProp = CastField<FStructProperty>(Prop)) {
285
+ if (StructProp->Struct->GetFName() ==
286
+ FName("ExpressionInput")) // Note: FExpressionInput struct name
287
+ // check
288
+ {
289
+ FExpressionInput *InputPtr =
290
+ StructProp->ContainerPtrToValuePtr<FExpressionInput>(
291
+ TargetExpr);
292
+ if (InputPtr) {
293
+ InputPtr->Expression = SourceExpr;
294
+ Material->PostEditChange();
295
+ Material->MarkPackageDirty();
296
+ SendAutomationResponse(Socket, RequestId, true,
297
+ TEXT("Nodes connected."));
298
+ return true;
299
+ }
300
+ }
301
+ }
302
+ // Also handle FColorMaterialInput, FScalarMaterialInput,
303
+ // FVectorMaterialInput which inherit FExpressionInput Just check if
304
+ // it has 'Expression' member? No, reflection doesn't work that way
305
+ // easily. In 5.6, inputs are usually typed. Fallback: check known
306
+ // input names for common nodes or generic implementation Since we
307
+ // can't easily genericize this without iteration or casting, we might
308
+ // fail if property isn't direct FExpressionInput. But typically they
309
+ // are FExpressionInput derived.
310
+ }
311
+
312
+ SendAutomationError(
313
+ Socket, RequestId,
314
+ FString::Printf(TEXT("Input pin '%s' not found or not compatible."),
315
+ *InputName),
316
+ TEXT("PIN_NOT_FOUND"));
317
+ } else {
318
+ SendAutomationError(Socket, RequestId, TEXT("Target node not found."),
319
+ TEXT("NODE_NOT_FOUND"));
320
+ }
321
+ return true;
322
+ }
323
+ } else if (SubAction == TEXT("break_connections")) {
324
+ FString NodeId;
325
+ Payload->TryGetStringField(TEXT("nodeId"), NodeId);
326
+ FString
327
+ PinName; // If provided, break specific pin. If empty, break all inputs?
328
+ Payload->TryGetStringField(TEXT("pinName"), PinName);
329
+
330
+ // Check if main node
331
+ if (NodeId.IsEmpty() || NodeId == TEXT("Main")) {
332
+ // Disconnect from main material node
333
+ if (!PinName.IsEmpty()) {
334
+ bool bFound = false;
335
+ #if WITH_EDITORONLY_DATA
336
+ if (PinName == TEXT("BaseColor")) {
337
+ Material->GetEditorOnlyData()->BaseColor.Expression = nullptr;
338
+ bFound = true;
339
+ } else if (PinName == TEXT("EmissiveColor")) {
340
+ Material->GetEditorOnlyData()->EmissiveColor.Expression = nullptr;
341
+ bFound = true;
342
+ } else if (PinName == TEXT("Roughness")) {
343
+ Material->GetEditorOnlyData()->Roughness.Expression = nullptr;
344
+ bFound = true;
345
+ } else if (PinName == TEXT("Metallic")) {
346
+ Material->GetEditorOnlyData()->Metallic.Expression = nullptr;
347
+ bFound = true;
348
+ } else if (PinName == TEXT("Specular")) {
349
+ Material->GetEditorOnlyData()->Specular.Expression = nullptr;
350
+ bFound = true;
351
+ } else if (PinName == TEXT("Normal")) {
352
+ Material->GetEditorOnlyData()->Normal.Expression = nullptr;
353
+ bFound = true;
354
+ } else if (PinName == TEXT("Opacity")) {
355
+ Material->GetEditorOnlyData()->Opacity.Expression = nullptr;
356
+ bFound = true;
357
+ } else if (PinName == TEXT("OpacityMask")) {
358
+ Material->GetEditorOnlyData()->OpacityMask.Expression = nullptr;
359
+ bFound = true;
360
+ } else if (PinName == TEXT("AmbientOcclusion")) {
361
+ Material->GetEditorOnlyData()->AmbientOcclusion.Expression = nullptr;
362
+ bFound = true;
363
+ } else if (PinName == TEXT("SubsurfaceColor")) {
364
+ Material->GetEditorOnlyData()->SubsurfaceColor.Expression = nullptr;
365
+ bFound = true;
366
+ }
367
+ #endif
368
+
369
+ if (bFound) {
370
+ Material->PostEditChange();
371
+ Material->MarkPackageDirty();
372
+ SendAutomationResponse(Socket, RequestId, true,
373
+ TEXT("Disconnected from main material pin."));
374
+ return true;
375
+ } else {
376
+ SendAutomationError(
377
+ Socket, RequestId,
378
+ FString::Printf(TEXT("Unknown or unsupported pin: %s"), *PinName),
379
+ TEXT("INVALID_PIN"));
380
+ return true;
381
+ }
382
+ }
383
+ }
384
+
385
+ UMaterialExpression *TargetExpr = FindExpressionByIdOrName(NodeId);
386
+
387
+ if (TargetExpr) {
388
+ // Disconnect all inputs of this node if no specific pin name
389
+ // Since GetInputs() is not available, we skip generic breaking for now.
390
+ // We can implement breaking for specific pin if needed via property
391
+ // reflection.
392
+
393
+ // For now, just acknowledge but warn.
394
+ Material->PostEditChange();
395
+ Material->MarkPackageDirty();
396
+ SendAutomationResponse(
397
+ Socket, RequestId, true,
398
+ TEXT("Node disconnection partial (generic inputs not cleared)."));
399
+ return true;
400
+ }
401
+
402
+ SendAutomationError(Socket, RequestId, TEXT("Node not found."),
403
+ TEXT("NODE_NOT_FOUND"));
404
+ return true;
405
+ } else if (SubAction == TEXT("get_node_details")) {
406
+ FString NodeId;
407
+ Payload->TryGetStringField(TEXT("nodeId"), NodeId);
408
+
409
+ UMaterialExpression *TargetExpr = nullptr;
410
+ auto AllExpressions = Material->GetExpressions();
411
+
412
+ // If nodeId provided, try to find that specific node
413
+ if (!NodeId.IsEmpty()) {
414
+ TargetExpr = FindExpressionByIdOrName(NodeId);
415
+ }
416
+
417
+ if (TargetExpr) {
418
+ TSharedPtr<FJsonObject> Result = MakeShared<FJsonObject>();
419
+ Result->SetStringField(TEXT("nodeType"),
420
+ TargetExpr->GetClass()->GetName());
421
+ Result->SetStringField(TEXT("desc"), TargetExpr->Desc);
422
+ Result->SetNumberField(TEXT("x"), TargetExpr->MaterialExpressionEditorX);
423
+ Result->SetNumberField(TEXT("y"), TargetExpr->MaterialExpressionEditorY);
424
+
425
+ SendAutomationResponse(Socket, RequestId, true,
426
+ TEXT("Node details retrieved."), Result);
427
+ } else {
428
+ // List all available nodes to help the user
429
+ TSharedPtr<FJsonObject> Result = MakeShared<FJsonObject>();
430
+ TArray<TSharedPtr<FJsonValue>> NodeList;
431
+
432
+ for (int32 i = 0; i < AllExpressions.Num(); ++i) {
433
+ UMaterialExpression *Expr = AllExpressions[i];
434
+ TSharedPtr<FJsonObject> NodeInfo = MakeShared<FJsonObject>();
435
+ NodeInfo->SetStringField(TEXT("nodeId"),
436
+ Expr->MaterialExpressionGuid.ToString());
437
+ NodeInfo->SetStringField(TEXT("nodeType"), Expr->GetClass()->GetName());
438
+ NodeInfo->SetNumberField(TEXT("index"), i);
439
+ if (!Expr->Desc.IsEmpty()) {
440
+ NodeInfo->SetStringField(TEXT("desc"), Expr->Desc);
441
+ }
442
+ NodeList.Add(MakeShared<FJsonValueObject>(NodeInfo));
443
+ }
444
+
445
+ Result->SetArrayField(TEXT("availableNodes"), NodeList);
446
+ Result->SetNumberField(TEXT("nodeCount"), AllExpressions.Num());
447
+
448
+ FString Message =
449
+ NodeId.IsEmpty()
450
+ ? FString::Printf(
451
+ TEXT("No nodeId provided. Material has %d nodes."),
452
+ AllExpressions.Num())
453
+ : FString::Printf(
454
+ TEXT("Node '%s' not found. Material has %d nodes."),
455
+ *NodeId, AllExpressions.Num());
456
+
457
+ SendAutomationResponse(Socket, RequestId, false, Message, Result,
458
+ TEXT("NODE_NOT_FOUND"));
459
+ }
460
+ return true;
461
+ }
462
+
463
+ SendAutomationError(
464
+ Socket, RequestId,
465
+ FString::Printf(TEXT("Unknown subAction: %s"), *SubAction),
466
+ TEXT("INVALID_SUBACTION"));
467
+ return true;
468
+ #else
469
+ SendAutomationError(Socket, RequestId, TEXT("Editor only."),
470
+ TEXT("EDITOR_ONLY"));
471
+ return true;
472
+ #endif
473
+ }
474
+
475
+ bool UMcpAutomationBridgeSubsystem::HandleAddMaterialTextureSample(
476
+ const FString &RequestId, const FString &Action,
477
+ const TSharedPtr<FJsonObject> &Payload,
478
+ TSharedPtr<FMcpBridgeWebSocket> Socket) {
479
+ return false;
480
+ }
481
+
482
+ bool UMcpAutomationBridgeSubsystem::HandleAddMaterialExpression(
483
+ const FString &RequestId, const FString &Action,
484
+ const TSharedPtr<FJsonObject> &Payload,
485
+ TSharedPtr<FMcpBridgeWebSocket> Socket) {
486
+ return false;
487
+ }
488
+
489
+ bool UMcpAutomationBridgeSubsystem::HandleCreateMaterialNodes(
490
+ const FString &RequestId, const FString &Action,
491
+ const TSharedPtr<FJsonObject> &Payload,
492
+ TSharedPtr<FMcpBridgeWebSocket> Socket) {
493
+ return false;
494
+ }