unreal-engine-mcp-server 0.4.7 → 0.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (438) hide show
  1. package/.env.example +26 -0
  2. package/.env.production +38 -7
  3. package/.eslintrc.json +0 -54
  4. package/.eslintrc.override.json +8 -0
  5. package/.github/ISSUE_TEMPLATE/bug_report.yml +94 -0
  6. package/.github/ISSUE_TEMPLATE/config.yml +8 -0
  7. package/.github/ISSUE_TEMPLATE/feature_request.yml +56 -0
  8. package/.github/copilot-instructions.md +478 -45
  9. package/.github/dependabot.yml +19 -0
  10. package/.github/labeler.yml +24 -0
  11. package/.github/labels.yml +70 -0
  12. package/.github/pull_request_template.md +42 -0
  13. package/.github/release-drafter.yml +148 -0
  14. package/.github/workflows/auto-merge.yml +38 -0
  15. package/.github/workflows/ci.yml +38 -0
  16. package/.github/workflows/dependency-review.yml +17 -0
  17. package/.github/workflows/gemini-issue-triage.yml +172 -0
  18. package/.github/workflows/greetings.yml +23 -0
  19. package/.github/workflows/labeler.yml +16 -0
  20. package/.github/workflows/links.yml +80 -0
  21. package/.github/workflows/pr-size-labeler.yml +137 -0
  22. package/.github/workflows/publish-mcp.yml +12 -7
  23. package/.github/workflows/release-drafter.yml +23 -0
  24. package/.github/workflows/release.yml +112 -0
  25. package/.github/workflows/semantic-pull-request.yml +35 -0
  26. package/.github/workflows/smoke-test.yml +36 -0
  27. package/.github/workflows/stale.yml +28 -0
  28. package/CHANGELOG.md +267 -31
  29. package/CONTRIBUTING.md +140 -0
  30. package/README.md +166 -71
  31. package/claude_desktop_config_example.json +7 -6
  32. package/dist/automation/bridge.d.ts +50 -0
  33. package/dist/automation/bridge.js +452 -0
  34. package/dist/automation/connection-manager.d.ts +23 -0
  35. package/dist/automation/connection-manager.js +107 -0
  36. package/dist/automation/handshake.d.ts +11 -0
  37. package/dist/automation/handshake.js +89 -0
  38. package/dist/automation/index.d.ts +3 -0
  39. package/dist/automation/index.js +3 -0
  40. package/dist/automation/message-handler.d.ts +12 -0
  41. package/dist/automation/message-handler.js +149 -0
  42. package/dist/automation/request-tracker.d.ts +25 -0
  43. package/dist/automation/request-tracker.js +98 -0
  44. package/dist/automation/types.d.ts +130 -0
  45. package/dist/automation/types.js +2 -0
  46. package/dist/cli.js +32 -5
  47. package/dist/config.d.ts +27 -0
  48. package/dist/config.js +60 -0
  49. package/dist/constants.d.ts +12 -0
  50. package/dist/constants.js +12 -0
  51. package/dist/graphql/resolvers.d.ts +268 -0
  52. package/dist/graphql/resolvers.js +743 -0
  53. package/dist/graphql/schema.d.ts +5 -0
  54. package/dist/graphql/schema.js +437 -0
  55. package/dist/graphql/server.d.ts +26 -0
  56. package/dist/graphql/server.js +115 -0
  57. package/dist/graphql/types.d.ts +7 -0
  58. package/dist/graphql/types.js +2 -0
  59. package/dist/handlers/resource-handlers.d.ts +20 -0
  60. package/dist/handlers/resource-handlers.js +180 -0
  61. package/dist/index.d.ts +31 -18
  62. package/dist/index.js +119 -619
  63. package/dist/prompts/index.js +4 -4
  64. package/dist/resources/actors.d.ts +17 -12
  65. package/dist/resources/actors.js +56 -76
  66. package/dist/resources/assets.d.ts +6 -14
  67. package/dist/resources/assets.js +115 -147
  68. package/dist/resources/levels.d.ts +13 -13
  69. package/dist/resources/levels.js +25 -34
  70. package/dist/server/resource-registry.d.ts +20 -0
  71. package/dist/server/resource-registry.js +37 -0
  72. package/dist/server/tool-registry.d.ts +23 -0
  73. package/dist/server/tool-registry.js +322 -0
  74. package/dist/server-setup.d.ts +21 -0
  75. package/dist/server-setup.js +111 -0
  76. package/dist/services/health-monitor.d.ts +34 -0
  77. package/dist/services/health-monitor.js +105 -0
  78. package/dist/services/metrics-server.d.ts +11 -0
  79. package/dist/services/metrics-server.js +105 -0
  80. package/dist/tools/actors.d.ts +147 -9
  81. package/dist/tools/actors.js +350 -311
  82. package/dist/tools/animation.d.ts +135 -4
  83. package/dist/tools/animation.js +510 -411
  84. package/dist/tools/assets.d.ts +117 -19
  85. package/dist/tools/assets.js +259 -284
  86. package/dist/tools/audio.d.ts +102 -42
  87. package/dist/tools/audio.js +272 -685
  88. package/dist/tools/base-tool.d.ts +17 -0
  89. package/dist/tools/base-tool.js +46 -0
  90. package/dist/tools/behavior-tree.d.ts +94 -0
  91. package/dist/tools/behavior-tree.js +39 -0
  92. package/dist/tools/blueprint/helpers.d.ts +29 -0
  93. package/dist/tools/blueprint/helpers.js +182 -0
  94. package/dist/tools/blueprint.d.ts +228 -118
  95. package/dist/tools/blueprint.js +685 -832
  96. package/dist/tools/consolidated-tool-definitions.d.ts +5462 -1781
  97. package/dist/tools/consolidated-tool-definitions.js +829 -496
  98. package/dist/tools/consolidated-tool-handlers.d.ts +2 -1
  99. package/dist/tools/consolidated-tool-handlers.js +211 -1026
  100. package/dist/tools/debug.d.ts +143 -85
  101. package/dist/tools/debug.js +234 -180
  102. package/dist/tools/dynamic-handler-registry.d.ts +11 -0
  103. package/dist/tools/dynamic-handler-registry.js +101 -0
  104. package/dist/tools/editor.d.ts +139 -18
  105. package/dist/tools/editor.js +239 -244
  106. package/dist/tools/engine.d.ts +10 -4
  107. package/dist/tools/engine.js +13 -5
  108. package/dist/tools/environment.d.ts +36 -0
  109. package/dist/tools/environment.js +267 -0
  110. package/dist/tools/foliage.d.ts +105 -14
  111. package/dist/tools/foliage.js +219 -331
  112. package/dist/tools/handlers/actor-handlers.d.ts +3 -0
  113. package/dist/tools/handlers/actor-handlers.js +232 -0
  114. package/dist/tools/handlers/animation-handlers.d.ts +3 -0
  115. package/dist/tools/handlers/animation-handlers.js +185 -0
  116. package/dist/tools/handlers/argument-helper.d.ts +16 -0
  117. package/dist/tools/handlers/argument-helper.js +80 -0
  118. package/dist/tools/handlers/asset-handlers.d.ts +3 -0
  119. package/dist/tools/handlers/asset-handlers.js +496 -0
  120. package/dist/tools/handlers/audio-handlers.d.ts +3 -0
  121. package/dist/tools/handlers/audio-handlers.js +166 -0
  122. package/dist/tools/handlers/blueprint-handlers.d.ts +4 -0
  123. package/dist/tools/handlers/blueprint-handlers.js +358 -0
  124. package/dist/tools/handlers/common-handlers.d.ts +14 -0
  125. package/dist/tools/handlers/common-handlers.js +56 -0
  126. package/dist/tools/handlers/editor-handlers.d.ts +3 -0
  127. package/dist/tools/handlers/editor-handlers.js +119 -0
  128. package/dist/tools/handlers/effect-handlers.d.ts +3 -0
  129. package/dist/tools/handlers/effect-handlers.js +171 -0
  130. package/dist/tools/handlers/environment-handlers.d.ts +3 -0
  131. package/dist/tools/handlers/environment-handlers.js +170 -0
  132. package/dist/tools/handlers/graph-handlers.d.ts +3 -0
  133. package/dist/tools/handlers/graph-handlers.js +90 -0
  134. package/dist/tools/handlers/input-handlers.d.ts +3 -0
  135. package/dist/tools/handlers/input-handlers.js +21 -0
  136. package/dist/tools/handlers/inspect-handlers.d.ts +3 -0
  137. package/dist/tools/handlers/inspect-handlers.js +383 -0
  138. package/dist/tools/handlers/level-handlers.d.ts +3 -0
  139. package/dist/tools/handlers/level-handlers.js +237 -0
  140. package/dist/tools/handlers/lighting-handlers.d.ts +3 -0
  141. package/dist/tools/handlers/lighting-handlers.js +144 -0
  142. package/dist/tools/handlers/performance-handlers.d.ts +3 -0
  143. package/dist/tools/handlers/performance-handlers.js +130 -0
  144. package/dist/tools/handlers/pipeline-handlers.d.ts +3 -0
  145. package/dist/tools/handlers/pipeline-handlers.js +110 -0
  146. package/dist/tools/handlers/sequence-handlers.d.ts +3 -0
  147. package/dist/tools/handlers/sequence-handlers.js +376 -0
  148. package/dist/tools/handlers/system-handlers.d.ts +4 -0
  149. package/dist/tools/handlers/system-handlers.js +506 -0
  150. package/dist/tools/input.d.ts +19 -0
  151. package/dist/tools/input.js +89 -0
  152. package/dist/tools/introspection.d.ts +103 -40
  153. package/dist/tools/introspection.js +425 -568
  154. package/dist/tools/landscape.d.ts +97 -36
  155. package/dist/tools/landscape.js +280 -409
  156. package/dist/tools/level.d.ts +130 -10
  157. package/dist/tools/level.js +639 -675
  158. package/dist/tools/lighting.d.ts +77 -38
  159. package/dist/tools/lighting.js +441 -943
  160. package/dist/tools/logs.d.ts +3 -3
  161. package/dist/tools/logs.js +5 -57
  162. package/dist/tools/materials.d.ts +91 -24
  163. package/dist/tools/materials.js +190 -118
  164. package/dist/tools/niagara.d.ts +149 -39
  165. package/dist/tools/niagara.js +232 -182
  166. package/dist/tools/performance.d.ts +27 -12
  167. package/dist/tools/performance.js +204 -122
  168. package/dist/tools/physics.d.ts +32 -77
  169. package/dist/tools/physics.js +171 -582
  170. package/dist/tools/property-dictionary.d.ts +13 -0
  171. package/dist/tools/property-dictionary.js +82 -0
  172. package/dist/tools/sequence.d.ts +73 -48
  173. package/dist/tools/sequence.js +196 -748
  174. package/dist/tools/tool-definition-utils.d.ts +59 -0
  175. package/dist/tools/tool-definition-utils.js +35 -0
  176. package/dist/tools/ui.d.ts +66 -34
  177. package/dist/tools/ui.js +134 -214
  178. package/dist/types/env.d.ts +0 -3
  179. package/dist/types/env.js +0 -7
  180. package/dist/types/tool-interfaces.d.ts +898 -0
  181. package/dist/types/tool-interfaces.js +2 -0
  182. package/dist/types/tool-types.d.ts +183 -19
  183. package/dist/types/tool-types.js +0 -4
  184. package/dist/unreal-bridge.d.ts +24 -131
  185. package/dist/unreal-bridge.js +364 -1506
  186. package/dist/utils/command-validator.d.ts +9 -0
  187. package/dist/utils/command-validator.js +67 -0
  188. package/dist/utils/elicitation.d.ts +1 -1
  189. package/dist/utils/elicitation.js +12 -15
  190. package/dist/utils/error-handler.d.ts +2 -51
  191. package/dist/utils/error-handler.js +11 -87
  192. package/dist/utils/ini-reader.d.ts +3 -0
  193. package/dist/utils/ini-reader.js +69 -0
  194. package/dist/utils/logger.js +9 -6
  195. package/dist/utils/normalize.d.ts +3 -0
  196. package/dist/utils/normalize.js +56 -0
  197. package/dist/utils/response-factory.d.ts +7 -0
  198. package/dist/utils/response-factory.js +33 -0
  199. package/dist/utils/response-validator.d.ts +3 -24
  200. package/dist/utils/response-validator.js +130 -81
  201. package/dist/utils/result-helpers.d.ts +4 -5
  202. package/dist/utils/result-helpers.js +15 -16
  203. package/dist/utils/safe-json.js +5 -11
  204. package/dist/utils/unreal-command-queue.d.ts +24 -0
  205. package/dist/utils/unreal-command-queue.js +120 -0
  206. package/dist/utils/validation.d.ts +0 -40
  207. package/dist/utils/validation.js +1 -78
  208. package/dist/wasm/index.d.ts +70 -0
  209. package/dist/wasm/index.js +535 -0
  210. package/docs/GraphQL-API.md +888 -0
  211. package/docs/Migration-Guide-v0.5.0.md +692 -0
  212. package/docs/Roadmap.md +53 -0
  213. package/docs/WebAssembly-Integration.md +628 -0
  214. package/docs/editor-plugin-extension.md +370 -0
  215. package/docs/handler-mapping.md +242 -0
  216. package/docs/native-automation-progress.md +128 -0
  217. package/docs/testing-guide.md +423 -0
  218. package/mcp-config-example.json +6 -6
  219. package/package.json +60 -27
  220. package/plugins/McpAutomationBridge/Config/FilterPlugin.ini +8 -0
  221. package/plugins/McpAutomationBridge/McpAutomationBridge.uplugin +64 -0
  222. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/McpAutomationBridge.Build.cs +189 -0
  223. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeGlobals.cpp +22 -0
  224. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeGlobals.h +30 -0
  225. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeHelpers.h +1983 -0
  226. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeModule.cpp +72 -0
  227. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeSettings.cpp +46 -0
  228. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeSubsystem.cpp +581 -0
  229. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AnimationHandlers.cpp +2394 -0
  230. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AssetQueryHandlers.cpp +300 -0
  231. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AssetWorkflowHandlers.cpp +2807 -0
  232. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AudioHandlers.cpp +1087 -0
  233. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BehaviorTreeHandlers.cpp +488 -0
  234. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintCreationHandlers.cpp +643 -0
  235. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintCreationHandlers.h +31 -0
  236. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintGraphHandlers.cpp +1184 -0
  237. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintHandlers.cpp +5652 -0
  238. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintHandlers_List.cpp +152 -0
  239. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_ControlHandlers.cpp +2614 -0
  240. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_DebugHandlers.cpp +42 -0
  241. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_EditorFunctionHandlers.cpp +1237 -0
  242. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_EffectHandlers.cpp +1701 -0
  243. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_EnvironmentHandlers.cpp +2145 -0
  244. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_FoliageHandlers.cpp +954 -0
  245. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_InputHandlers.cpp +209 -0
  246. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_InsightsHandlers.cpp +41 -0
  247. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LandscapeHandlers.cpp +1164 -0
  248. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LevelHandlers.cpp +762 -0
  249. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LightingHandlers.cpp +634 -0
  250. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LogHandlers.cpp +136 -0
  251. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_MaterialGraphHandlers.cpp +494 -0
  252. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_NiagaraGraphHandlers.cpp +278 -0
  253. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_NiagaraHandlers.cpp +625 -0
  254. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_PerformanceHandlers.cpp +401 -0
  255. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_PipelineHandlers.cpp +67 -0
  256. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_ProcessRequest.cpp +735 -0
  257. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_PropertyHandlers.cpp +2634 -0
  258. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_RenderHandlers.cpp +189 -0
  259. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SCSHandlers.cpp +917 -0
  260. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SCSHandlers.h +39 -0
  261. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SequenceHandlers.cpp +2670 -0
  262. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SequencerHandlers.cpp +519 -0
  263. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_TestHandlers.cpp +38 -0
  264. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_UiHandlers.cpp +668 -0
  265. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_WorldPartitionHandlers.cpp +346 -0
  266. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpBridgeWebSocket.cpp +1330 -0
  267. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpBridgeWebSocket.h +149 -0
  268. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpConnectionManager.cpp +783 -0
  269. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Public/McpAutomationBridgeSettings.h +115 -0
  270. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Public/McpAutomationBridgeSubsystem.h +796 -0
  271. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Public/McpConnectionManager.h +117 -0
  272. package/scripts/check-unreal-connection.mjs +19 -0
  273. package/scripts/clean-tmp.js +23 -0
  274. package/scripts/patch-wasm.js +26 -0
  275. package/scripts/run-all-tests.mjs +131 -0
  276. package/scripts/smoke-test.ts +94 -0
  277. package/scripts/sync-mcp-plugin.js +143 -0
  278. package/scripts/test-no-plugin-alternates.mjs +113 -0
  279. package/scripts/validate-server.js +46 -0
  280. package/scripts/verify-automation-bridge.js +200 -0
  281. package/server.json +57 -21
  282. package/src/automation/bridge.ts +558 -0
  283. package/src/automation/connection-manager.ts +130 -0
  284. package/src/automation/handshake.ts +99 -0
  285. package/src/automation/index.ts +2 -0
  286. package/src/automation/message-handler.ts +167 -0
  287. package/src/automation/request-tracker.ts +123 -0
  288. package/src/automation/types.ts +107 -0
  289. package/src/cli.ts +33 -6
  290. package/src/config.ts +73 -0
  291. package/src/constants.ts +12 -0
  292. package/src/graphql/resolvers.ts +1010 -0
  293. package/src/graphql/schema.ts +452 -0
  294. package/src/graphql/server.ts +154 -0
  295. package/src/graphql/types.ts +7 -0
  296. package/src/handlers/resource-handlers.ts +186 -0
  297. package/src/index.ts +152 -663
  298. package/src/prompts/index.ts +4 -4
  299. package/src/resources/actors.ts +58 -76
  300. package/src/resources/assets.ts +147 -134
  301. package/src/resources/levels.ts +28 -33
  302. package/src/server/resource-registry.ts +47 -0
  303. package/src/server/tool-registry.ts +354 -0
  304. package/src/server-setup.ts +148 -0
  305. package/src/services/health-monitor.ts +132 -0
  306. package/src/services/metrics-server.ts +142 -0
  307. package/src/tools/actors.ts +417 -322
  308. package/src/tools/animation.ts +671 -461
  309. package/src/tools/assets.ts +353 -289
  310. package/src/tools/audio.ts +323 -766
  311. package/src/tools/base-tool.ts +52 -0
  312. package/src/tools/behavior-tree.ts +45 -0
  313. package/src/tools/blueprint/helpers.ts +189 -0
  314. package/src/tools/blueprint.ts +787 -965
  315. package/src/tools/consolidated-tool-definitions.ts +993 -515
  316. package/src/tools/consolidated-tool-handlers.ts +272 -1139
  317. package/src/tools/debug.ts +292 -187
  318. package/src/tools/dynamic-handler-registry.ts +151 -0
  319. package/src/tools/editor.ts +309 -246
  320. package/src/tools/engine.ts +14 -3
  321. package/src/tools/environment.ts +287 -0
  322. package/src/tools/foliage.ts +314 -379
  323. package/src/tools/handlers/actor-handlers.ts +271 -0
  324. package/src/tools/handlers/animation-handlers.ts +237 -0
  325. package/src/tools/handlers/argument-helper.ts +142 -0
  326. package/src/tools/handlers/asset-handlers.ts +532 -0
  327. package/src/tools/handlers/audio-handlers.ts +194 -0
  328. package/src/tools/handlers/blueprint-handlers.ts +380 -0
  329. package/src/tools/handlers/common-handlers.ts +87 -0
  330. package/src/tools/handlers/editor-handlers.ts +123 -0
  331. package/src/tools/handlers/effect-handlers.ts +220 -0
  332. package/src/tools/handlers/environment-handlers.ts +183 -0
  333. package/src/tools/handlers/graph-handlers.ts +116 -0
  334. package/src/tools/handlers/input-handlers.ts +28 -0
  335. package/src/tools/handlers/inspect-handlers.ts +450 -0
  336. package/src/tools/handlers/level-handlers.ts +252 -0
  337. package/src/tools/handlers/lighting-handlers.ts +147 -0
  338. package/src/tools/handlers/performance-handlers.ts +132 -0
  339. package/src/tools/handlers/pipeline-handlers.ts +127 -0
  340. package/src/tools/handlers/sequence-handlers.ts +415 -0
  341. package/src/tools/handlers/system-handlers.ts +564 -0
  342. package/src/tools/input.ts +101 -0
  343. package/src/tools/introspection.ts +493 -584
  344. package/src/tools/landscape.ts +394 -489
  345. package/src/tools/level.ts +752 -694
  346. package/src/tools/lighting.ts +583 -984
  347. package/src/tools/logs.ts +9 -57
  348. package/src/tools/materials.ts +231 -121
  349. package/src/tools/niagara.ts +293 -168
  350. package/src/tools/performance.ts +320 -168
  351. package/src/tools/physics.ts +268 -613
  352. package/src/tools/property-dictionary.ts +98 -0
  353. package/src/tools/sequence.ts +255 -815
  354. package/src/tools/tool-definition-utils.ts +35 -0
  355. package/src/tools/ui.ts +207 -283
  356. package/src/types/env.ts +0 -10
  357. package/src/types/tool-interfaces.ts +250 -0
  358. package/src/types/tool-types.ts +243 -21
  359. package/src/unreal-bridge.ts +460 -1550
  360. package/src/utils/command-validator.ts +75 -0
  361. package/src/utils/elicitation.ts +10 -7
  362. package/src/utils/error-handler.ts +14 -90
  363. package/src/utils/ini-reader.ts +86 -0
  364. package/src/utils/logger.ts +8 -3
  365. package/src/utils/normalize.ts +60 -0
  366. package/src/utils/response-factory.ts +39 -0
  367. package/src/utils/response-validator.ts +176 -56
  368. package/src/utils/result-helpers.ts +21 -19
  369. package/src/utils/safe-json.ts +14 -11
  370. package/src/utils/unreal-command-queue.ts +152 -0
  371. package/src/utils/validation.ts +4 -1
  372. package/src/wasm/index.ts +838 -0
  373. package/test-server.mjs +100 -0
  374. package/tests/run-unreal-tool-tests.mjs +242 -14
  375. package/tests/test-animation.mjs +44 -0
  376. package/tests/test-asset-advanced.mjs +82 -0
  377. package/tests/test-asset-errors.mjs +35 -0
  378. package/tests/test-audio.mjs +219 -0
  379. package/tests/test-automation-timeouts.mjs +98 -0
  380. package/tests/test-behavior-tree.mjs +261 -0
  381. package/tests/test-blueprint-events.mjs +35 -0
  382. package/tests/test-blueprint-graph.mjs +79 -0
  383. package/tests/test-blueprint.mjs +577 -0
  384. package/tests/test-client-mode.mjs +86 -0
  385. package/tests/test-console-command.mjs +56 -0
  386. package/tests/test-control-actor.mjs +425 -0
  387. package/tests/test-control-editor.mjs +80 -0
  388. package/tests/test-extra-tools.mjs +38 -0
  389. package/tests/test-graphql.mjs +322 -0
  390. package/tests/test-inspect.mjs +72 -0
  391. package/tests/test-landscape.mjs +60 -0
  392. package/tests/test-manage-asset.mjs +438 -0
  393. package/tests/test-manage-level.mjs +70 -0
  394. package/tests/test-materials.mjs +356 -0
  395. package/tests/test-niagara.mjs +185 -0
  396. package/tests/test-no-inline-python.mjs +122 -0
  397. package/tests/test-plugin-handshake.mjs +82 -0
  398. package/tests/test-render.mjs +33 -0
  399. package/tests/test-runner.mjs +933 -0
  400. package/tests/test-search-assets.mjs +66 -0
  401. package/tests/test-sequence.mjs +68 -0
  402. package/tests/test-system.mjs +57 -0
  403. package/tests/test-wasm.mjs +193 -0
  404. package/tests/test-world-partition.mjs +215 -0
  405. package/tsconfig.json +3 -3
  406. package/wasm/Cargo.lock +363 -0
  407. package/wasm/Cargo.toml +42 -0
  408. package/wasm/LICENSE +21 -0
  409. package/wasm/README.md +253 -0
  410. package/wasm/src/dependency_resolver.rs +377 -0
  411. package/wasm/src/lib.rs +153 -0
  412. package/wasm/src/property_parser.rs +271 -0
  413. package/wasm/src/transform_math.rs +396 -0
  414. package/wasm/tests/integration.rs +109 -0
  415. package/.github/workflows/smithery-build.yml +0 -29
  416. package/dist/tools/build_environment_advanced.d.ts +0 -65
  417. package/dist/tools/build_environment_advanced.js +0 -633
  418. package/dist/tools/rc.d.ts +0 -110
  419. package/dist/tools/rc.js +0 -437
  420. package/dist/tools/visual.d.ts +0 -40
  421. package/dist/tools/visual.js +0 -282
  422. package/dist/utils/http.d.ts +0 -6
  423. package/dist/utils/http.js +0 -151
  424. package/dist/utils/python-output.d.ts +0 -18
  425. package/dist/utils/python-output.js +0 -290
  426. package/dist/utils/python.d.ts +0 -2
  427. package/dist/utils/python.js +0 -4
  428. package/dist/utils/stdio-redirect.d.ts +0 -2
  429. package/dist/utils/stdio-redirect.js +0 -20
  430. package/docs/unreal-tool-test-cases.md +0 -574
  431. package/smithery.yaml +0 -29
  432. package/src/tools/build_environment_advanced.ts +0 -732
  433. package/src/tools/rc.ts +0 -515
  434. package/src/tools/visual.ts +0 -281
  435. package/src/utils/http.ts +0 -187
  436. package/src/utils/python-output.ts +0 -351
  437. package/src/utils/python.ts +0 -3
  438. package/src/utils/stdio-redirect.ts +0 -18
@@ -0,0 +1,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
+ }