unreal-engine-mcp-server 0.4.6 → 0.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (438) hide show
  1. package/.env.example +26 -0
  2. package/.env.production +38 -7
  3. package/.eslintrc.json +0 -54
  4. package/.eslintrc.override.json +8 -0
  5. package/.github/ISSUE_TEMPLATE/bug_report.yml +94 -0
  6. package/.github/ISSUE_TEMPLATE/config.yml +8 -0
  7. package/.github/ISSUE_TEMPLATE/feature_request.yml +56 -0
  8. package/.github/copilot-instructions.md +478 -45
  9. package/.github/dependabot.yml +19 -0
  10. package/.github/labeler.yml +24 -0
  11. package/.github/labels.yml +70 -0
  12. package/.github/pull_request_template.md +42 -0
  13. package/.github/release-drafter.yml +148 -0
  14. package/.github/workflows/auto-merge.yml +38 -0
  15. package/.github/workflows/ci.yml +38 -0
  16. package/.github/workflows/dependency-review.yml +17 -0
  17. package/.github/workflows/gemini-issue-triage.yml +172 -0
  18. package/.github/workflows/greetings.yml +23 -0
  19. package/.github/workflows/labeler.yml +16 -0
  20. package/.github/workflows/links.yml +80 -0
  21. package/.github/workflows/pr-size-labeler.yml +137 -0
  22. package/.github/workflows/publish-mcp.yml +12 -7
  23. package/.github/workflows/release-drafter.yml +23 -0
  24. package/.github/workflows/release.yml +112 -0
  25. package/.github/workflows/semantic-pull-request.yml +35 -0
  26. package/.github/workflows/smoke-test.yml +36 -0
  27. package/.github/workflows/stale.yml +28 -0
  28. package/CHANGELOG.md +269 -22
  29. package/CONTRIBUTING.md +140 -0
  30. package/README.md +166 -72
  31. package/claude_desktop_config_example.json +7 -6
  32. package/dist/automation/bridge.d.ts +50 -0
  33. package/dist/automation/bridge.js +452 -0
  34. package/dist/automation/connection-manager.d.ts +23 -0
  35. package/dist/automation/connection-manager.js +107 -0
  36. package/dist/automation/handshake.d.ts +11 -0
  37. package/dist/automation/handshake.js +89 -0
  38. package/dist/automation/index.d.ts +3 -0
  39. package/dist/automation/index.js +3 -0
  40. package/dist/automation/message-handler.d.ts +12 -0
  41. package/dist/automation/message-handler.js +149 -0
  42. package/dist/automation/request-tracker.d.ts +25 -0
  43. package/dist/automation/request-tracker.js +98 -0
  44. package/dist/automation/types.d.ts +130 -0
  45. package/dist/automation/types.js +2 -0
  46. package/dist/cli.js +32 -5
  47. package/dist/config.d.ts +27 -0
  48. package/dist/config.js +60 -0
  49. package/dist/constants.d.ts +12 -0
  50. package/dist/constants.js +12 -0
  51. package/dist/graphql/resolvers.d.ts +268 -0
  52. package/dist/graphql/resolvers.js +743 -0
  53. package/dist/graphql/schema.d.ts +5 -0
  54. package/dist/graphql/schema.js +437 -0
  55. package/dist/graphql/server.d.ts +26 -0
  56. package/dist/graphql/server.js +115 -0
  57. package/dist/graphql/types.d.ts +7 -0
  58. package/dist/graphql/types.js +2 -0
  59. package/dist/handlers/resource-handlers.d.ts +20 -0
  60. package/dist/handlers/resource-handlers.js +180 -0
  61. package/dist/index.d.ts +31 -18
  62. package/dist/index.js +119 -604
  63. package/dist/prompts/index.js +4 -4
  64. package/dist/resources/actors.d.ts +17 -12
  65. package/dist/resources/actors.js +56 -76
  66. package/dist/resources/assets.d.ts +6 -14
  67. package/dist/resources/assets.js +115 -147
  68. package/dist/resources/levels.d.ts +13 -13
  69. package/dist/resources/levels.js +25 -34
  70. package/dist/server/resource-registry.d.ts +20 -0
  71. package/dist/server/resource-registry.js +37 -0
  72. package/dist/server/tool-registry.d.ts +23 -0
  73. package/dist/server/tool-registry.js +322 -0
  74. package/dist/server-setup.d.ts +21 -0
  75. package/dist/server-setup.js +111 -0
  76. package/dist/services/health-monitor.d.ts +34 -0
  77. package/dist/services/health-monitor.js +105 -0
  78. package/dist/services/metrics-server.d.ts +11 -0
  79. package/dist/services/metrics-server.js +105 -0
  80. package/dist/tools/actors.d.ts +147 -9
  81. package/dist/tools/actors.js +350 -311
  82. package/dist/tools/animation.d.ts +135 -4
  83. package/dist/tools/animation.js +510 -411
  84. package/dist/tools/assets.d.ts +117 -19
  85. package/dist/tools/assets.js +259 -284
  86. package/dist/tools/audio.d.ts +102 -42
  87. package/dist/tools/audio.js +272 -685
  88. package/dist/tools/base-tool.d.ts +17 -0
  89. package/dist/tools/base-tool.js +46 -0
  90. package/dist/tools/behavior-tree.d.ts +94 -0
  91. package/dist/tools/behavior-tree.js +39 -0
  92. package/dist/tools/blueprint/helpers.d.ts +29 -0
  93. package/dist/tools/blueprint/helpers.js +182 -0
  94. package/dist/tools/blueprint.d.ts +228 -118
  95. package/dist/tools/blueprint.js +685 -832
  96. package/dist/tools/consolidated-tool-definitions.d.ts +5475 -1627
  97. package/dist/tools/consolidated-tool-definitions.js +829 -482
  98. package/dist/tools/consolidated-tool-handlers.d.ts +2 -1
  99. package/dist/tools/consolidated-tool-handlers.js +211 -1009
  100. package/dist/tools/debug.d.ts +143 -85
  101. package/dist/tools/debug.js +234 -180
  102. package/dist/tools/dynamic-handler-registry.d.ts +11 -0
  103. package/dist/tools/dynamic-handler-registry.js +101 -0
  104. package/dist/tools/editor.d.ts +139 -18
  105. package/dist/tools/editor.js +239 -244
  106. package/dist/tools/engine.d.ts +10 -4
  107. package/dist/tools/engine.js +13 -5
  108. package/dist/tools/environment.d.ts +36 -0
  109. package/dist/tools/environment.js +267 -0
  110. package/dist/tools/foliage.d.ts +105 -14
  111. package/dist/tools/foliage.js +219 -331
  112. package/dist/tools/handlers/actor-handlers.d.ts +3 -0
  113. package/dist/tools/handlers/actor-handlers.js +232 -0
  114. package/dist/tools/handlers/animation-handlers.d.ts +3 -0
  115. package/dist/tools/handlers/animation-handlers.js +185 -0
  116. package/dist/tools/handlers/argument-helper.d.ts +16 -0
  117. package/dist/tools/handlers/argument-helper.js +80 -0
  118. package/dist/tools/handlers/asset-handlers.d.ts +3 -0
  119. package/dist/tools/handlers/asset-handlers.js +496 -0
  120. package/dist/tools/handlers/audio-handlers.d.ts +3 -0
  121. package/dist/tools/handlers/audio-handlers.js +166 -0
  122. package/dist/tools/handlers/blueprint-handlers.d.ts +4 -0
  123. package/dist/tools/handlers/blueprint-handlers.js +358 -0
  124. package/dist/tools/handlers/common-handlers.d.ts +14 -0
  125. package/dist/tools/handlers/common-handlers.js +56 -0
  126. package/dist/tools/handlers/editor-handlers.d.ts +3 -0
  127. package/dist/tools/handlers/editor-handlers.js +119 -0
  128. package/dist/tools/handlers/effect-handlers.d.ts +3 -0
  129. package/dist/tools/handlers/effect-handlers.js +171 -0
  130. package/dist/tools/handlers/environment-handlers.d.ts +3 -0
  131. package/dist/tools/handlers/environment-handlers.js +170 -0
  132. package/dist/tools/handlers/graph-handlers.d.ts +3 -0
  133. package/dist/tools/handlers/graph-handlers.js +90 -0
  134. package/dist/tools/handlers/input-handlers.d.ts +3 -0
  135. package/dist/tools/handlers/input-handlers.js +21 -0
  136. package/dist/tools/handlers/inspect-handlers.d.ts +3 -0
  137. package/dist/tools/handlers/inspect-handlers.js +383 -0
  138. package/dist/tools/handlers/level-handlers.d.ts +3 -0
  139. package/dist/tools/handlers/level-handlers.js +237 -0
  140. package/dist/tools/handlers/lighting-handlers.d.ts +3 -0
  141. package/dist/tools/handlers/lighting-handlers.js +144 -0
  142. package/dist/tools/handlers/performance-handlers.d.ts +3 -0
  143. package/dist/tools/handlers/performance-handlers.js +130 -0
  144. package/dist/tools/handlers/pipeline-handlers.d.ts +3 -0
  145. package/dist/tools/handlers/pipeline-handlers.js +110 -0
  146. package/dist/tools/handlers/sequence-handlers.d.ts +3 -0
  147. package/dist/tools/handlers/sequence-handlers.js +376 -0
  148. package/dist/tools/handlers/system-handlers.d.ts +4 -0
  149. package/dist/tools/handlers/system-handlers.js +506 -0
  150. package/dist/tools/input.d.ts +19 -0
  151. package/dist/tools/input.js +89 -0
  152. package/dist/tools/introspection.d.ts +103 -40
  153. package/dist/tools/introspection.js +425 -568
  154. package/dist/tools/landscape.d.ts +97 -36
  155. package/dist/tools/landscape.js +280 -409
  156. package/dist/tools/level.d.ts +130 -10
  157. package/dist/tools/level.js +639 -675
  158. package/dist/tools/lighting.d.ts +77 -38
  159. package/dist/tools/lighting.js +441 -943
  160. package/dist/tools/logs.d.ts +45 -0
  161. package/dist/tools/logs.js +210 -0
  162. package/dist/tools/materials.d.ts +91 -24
  163. package/dist/tools/materials.js +190 -118
  164. package/dist/tools/niagara.d.ts +149 -39
  165. package/dist/tools/niagara.js +232 -182
  166. package/dist/tools/performance.d.ts +27 -12
  167. package/dist/tools/performance.js +204 -122
  168. package/dist/tools/physics.d.ts +32 -77
  169. package/dist/tools/physics.js +171 -582
  170. package/dist/tools/property-dictionary.d.ts +13 -0
  171. package/dist/tools/property-dictionary.js +82 -0
  172. package/dist/tools/sequence.d.ts +73 -48
  173. package/dist/tools/sequence.js +196 -748
  174. package/dist/tools/tool-definition-utils.d.ts +59 -0
  175. package/dist/tools/tool-definition-utils.js +35 -0
  176. package/dist/tools/ui.d.ts +66 -34
  177. package/dist/tools/ui.js +134 -214
  178. package/dist/types/env.d.ts +0 -3
  179. package/dist/types/env.js +0 -7
  180. package/dist/types/tool-interfaces.d.ts +898 -0
  181. package/dist/types/tool-interfaces.js +2 -0
  182. package/dist/types/tool-types.d.ts +195 -11
  183. package/dist/types/tool-types.js +0 -4
  184. package/dist/unreal-bridge.d.ts +24 -131
  185. package/dist/unreal-bridge.js +364 -1506
  186. package/dist/utils/command-validator.d.ts +9 -0
  187. package/dist/utils/command-validator.js +67 -0
  188. package/dist/utils/elicitation.d.ts +1 -1
  189. package/dist/utils/elicitation.js +12 -15
  190. package/dist/utils/error-handler.d.ts +2 -51
  191. package/dist/utils/error-handler.js +11 -87
  192. package/dist/utils/ini-reader.d.ts +3 -0
  193. package/dist/utils/ini-reader.js +69 -0
  194. package/dist/utils/logger.js +9 -6
  195. package/dist/utils/normalize.d.ts +3 -0
  196. package/dist/utils/normalize.js +56 -0
  197. package/dist/utils/response-factory.d.ts +7 -0
  198. package/dist/utils/response-factory.js +33 -0
  199. package/dist/utils/response-validator.d.ts +3 -24
  200. package/dist/utils/response-validator.js +130 -81
  201. package/dist/utils/result-helpers.d.ts +4 -5
  202. package/dist/utils/result-helpers.js +15 -16
  203. package/dist/utils/safe-json.js +5 -11
  204. package/dist/utils/unreal-command-queue.d.ts +24 -0
  205. package/dist/utils/unreal-command-queue.js +120 -0
  206. package/dist/utils/validation.d.ts +0 -40
  207. package/dist/utils/validation.js +1 -78
  208. package/dist/wasm/index.d.ts +70 -0
  209. package/dist/wasm/index.js +535 -0
  210. package/docs/GraphQL-API.md +888 -0
  211. package/docs/Migration-Guide-v0.5.0.md +692 -0
  212. package/docs/Roadmap.md +53 -0
  213. package/docs/WebAssembly-Integration.md +628 -0
  214. package/docs/editor-plugin-extension.md +370 -0
  215. package/docs/handler-mapping.md +242 -0
  216. package/docs/native-automation-progress.md +128 -0
  217. package/docs/testing-guide.md +423 -0
  218. package/mcp-config-example.json +6 -6
  219. package/package.json +60 -27
  220. package/plugins/McpAutomationBridge/Config/FilterPlugin.ini +8 -0
  221. package/plugins/McpAutomationBridge/McpAutomationBridge.uplugin +64 -0
  222. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/McpAutomationBridge.Build.cs +189 -0
  223. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeGlobals.cpp +22 -0
  224. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeGlobals.h +30 -0
  225. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeHelpers.h +1983 -0
  226. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeModule.cpp +72 -0
  227. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeSettings.cpp +46 -0
  228. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridgeSubsystem.cpp +581 -0
  229. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AnimationHandlers.cpp +2394 -0
  230. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AssetQueryHandlers.cpp +300 -0
  231. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AssetWorkflowHandlers.cpp +2807 -0
  232. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_AudioHandlers.cpp +1087 -0
  233. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BehaviorTreeHandlers.cpp +488 -0
  234. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintCreationHandlers.cpp +643 -0
  235. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintCreationHandlers.h +31 -0
  236. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintGraphHandlers.cpp +1184 -0
  237. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintHandlers.cpp +5652 -0
  238. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_BlueprintHandlers_List.cpp +152 -0
  239. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_ControlHandlers.cpp +2614 -0
  240. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_DebugHandlers.cpp +42 -0
  241. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_EditorFunctionHandlers.cpp +1237 -0
  242. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_EffectHandlers.cpp +1701 -0
  243. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_EnvironmentHandlers.cpp +2145 -0
  244. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_FoliageHandlers.cpp +954 -0
  245. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_InputHandlers.cpp +209 -0
  246. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_InsightsHandlers.cpp +41 -0
  247. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LandscapeHandlers.cpp +1164 -0
  248. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LevelHandlers.cpp +762 -0
  249. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LightingHandlers.cpp +634 -0
  250. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_LogHandlers.cpp +136 -0
  251. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_MaterialGraphHandlers.cpp +494 -0
  252. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_NiagaraGraphHandlers.cpp +278 -0
  253. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_NiagaraHandlers.cpp +625 -0
  254. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_PerformanceHandlers.cpp +401 -0
  255. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_PipelineHandlers.cpp +67 -0
  256. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_ProcessRequest.cpp +735 -0
  257. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_PropertyHandlers.cpp +2634 -0
  258. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_RenderHandlers.cpp +189 -0
  259. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SCSHandlers.cpp +917 -0
  260. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SCSHandlers.h +39 -0
  261. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SequenceHandlers.cpp +2670 -0
  262. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_SequencerHandlers.cpp +519 -0
  263. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_TestHandlers.cpp +38 -0
  264. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_UiHandlers.cpp +668 -0
  265. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpAutomationBridge_WorldPartitionHandlers.cpp +346 -0
  266. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpBridgeWebSocket.cpp +1330 -0
  267. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpBridgeWebSocket.h +149 -0
  268. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Private/McpConnectionManager.cpp +783 -0
  269. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Public/McpAutomationBridgeSettings.h +115 -0
  270. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Public/McpAutomationBridgeSubsystem.h +796 -0
  271. package/plugins/McpAutomationBridge/Source/McpAutomationBridge/Public/McpConnectionManager.h +117 -0
  272. package/scripts/check-unreal-connection.mjs +19 -0
  273. package/scripts/clean-tmp.js +23 -0
  274. package/scripts/patch-wasm.js +26 -0
  275. package/scripts/run-all-tests.mjs +131 -0
  276. package/scripts/smoke-test.ts +94 -0
  277. package/scripts/sync-mcp-plugin.js +143 -0
  278. package/scripts/test-no-plugin-alternates.mjs +113 -0
  279. package/scripts/validate-server.js +46 -0
  280. package/scripts/verify-automation-bridge.js +200 -0
  281. package/server.json +57 -21
  282. package/src/automation/bridge.ts +558 -0
  283. package/src/automation/connection-manager.ts +130 -0
  284. package/src/automation/handshake.ts +99 -0
  285. package/src/automation/index.ts +2 -0
  286. package/src/automation/message-handler.ts +167 -0
  287. package/src/automation/request-tracker.ts +123 -0
  288. package/src/automation/types.ts +107 -0
  289. package/src/cli.ts +33 -6
  290. package/src/config.ts +73 -0
  291. package/src/constants.ts +12 -0
  292. package/src/graphql/resolvers.ts +1010 -0
  293. package/src/graphql/schema.ts +452 -0
  294. package/src/graphql/server.ts +154 -0
  295. package/src/graphql/types.ts +7 -0
  296. package/src/handlers/resource-handlers.ts +186 -0
  297. package/src/index.ts +152 -649
  298. package/src/prompts/index.ts +4 -4
  299. package/src/resources/actors.ts +58 -76
  300. package/src/resources/assets.ts +147 -134
  301. package/src/resources/levels.ts +28 -33
  302. package/src/server/resource-registry.ts +47 -0
  303. package/src/server/tool-registry.ts +354 -0
  304. package/src/server-setup.ts +148 -0
  305. package/src/services/health-monitor.ts +132 -0
  306. package/src/services/metrics-server.ts +142 -0
  307. package/src/tools/actors.ts +417 -322
  308. package/src/tools/animation.ts +671 -461
  309. package/src/tools/assets.ts +353 -289
  310. package/src/tools/audio.ts +323 -766
  311. package/src/tools/base-tool.ts +52 -0
  312. package/src/tools/behavior-tree.ts +45 -0
  313. package/src/tools/blueprint/helpers.ts +189 -0
  314. package/src/tools/blueprint.ts +787 -965
  315. package/src/tools/consolidated-tool-definitions.ts +993 -500
  316. package/src/tools/consolidated-tool-handlers.ts +272 -1122
  317. package/src/tools/debug.ts +292 -187
  318. package/src/tools/dynamic-handler-registry.ts +151 -0
  319. package/src/tools/editor.ts +309 -246
  320. package/src/tools/engine.ts +14 -3
  321. package/src/tools/environment.ts +287 -0
  322. package/src/tools/foliage.ts +314 -379
  323. package/src/tools/handlers/actor-handlers.ts +271 -0
  324. package/src/tools/handlers/animation-handlers.ts +237 -0
  325. package/src/tools/handlers/argument-helper.ts +142 -0
  326. package/src/tools/handlers/asset-handlers.ts +532 -0
  327. package/src/tools/handlers/audio-handlers.ts +194 -0
  328. package/src/tools/handlers/blueprint-handlers.ts +380 -0
  329. package/src/tools/handlers/common-handlers.ts +87 -0
  330. package/src/tools/handlers/editor-handlers.ts +123 -0
  331. package/src/tools/handlers/effect-handlers.ts +220 -0
  332. package/src/tools/handlers/environment-handlers.ts +183 -0
  333. package/src/tools/handlers/graph-handlers.ts +116 -0
  334. package/src/tools/handlers/input-handlers.ts +28 -0
  335. package/src/tools/handlers/inspect-handlers.ts +450 -0
  336. package/src/tools/handlers/level-handlers.ts +252 -0
  337. package/src/tools/handlers/lighting-handlers.ts +147 -0
  338. package/src/tools/handlers/performance-handlers.ts +132 -0
  339. package/src/tools/handlers/pipeline-handlers.ts +127 -0
  340. package/src/tools/handlers/sequence-handlers.ts +415 -0
  341. package/src/tools/handlers/system-handlers.ts +564 -0
  342. package/src/tools/input.ts +101 -0
  343. package/src/tools/introspection.ts +493 -584
  344. package/src/tools/landscape.ts +394 -489
  345. package/src/tools/level.ts +752 -694
  346. package/src/tools/lighting.ts +583 -984
  347. package/src/tools/logs.ts +219 -0
  348. package/src/tools/materials.ts +231 -121
  349. package/src/tools/niagara.ts +293 -168
  350. package/src/tools/performance.ts +320 -168
  351. package/src/tools/physics.ts +268 -613
  352. package/src/tools/property-dictionary.ts +98 -0
  353. package/src/tools/sequence.ts +255 -815
  354. package/src/tools/tool-definition-utils.ts +35 -0
  355. package/src/tools/ui.ts +207 -283
  356. package/src/types/env.ts +0 -10
  357. package/src/types/tool-interfaces.ts +250 -0
  358. package/src/types/tool-types.ts +250 -13
  359. package/src/unreal-bridge.ts +460 -1550
  360. package/src/utils/command-validator.ts +75 -0
  361. package/src/utils/elicitation.ts +10 -7
  362. package/src/utils/error-handler.ts +14 -90
  363. package/src/utils/ini-reader.ts +86 -0
  364. package/src/utils/logger.ts +8 -3
  365. package/src/utils/normalize.ts +60 -0
  366. package/src/utils/response-factory.ts +39 -0
  367. package/src/utils/response-validator.ts +176 -56
  368. package/src/utils/result-helpers.ts +21 -19
  369. package/src/utils/safe-json.ts +14 -11
  370. package/src/utils/unreal-command-queue.ts +152 -0
  371. package/src/utils/validation.ts +4 -1
  372. package/src/wasm/index.ts +838 -0
  373. package/test-server.mjs +100 -0
  374. package/tests/run-unreal-tool-tests.mjs +242 -14
  375. package/tests/test-animation.mjs +44 -0
  376. package/tests/test-asset-advanced.mjs +82 -0
  377. package/tests/test-asset-errors.mjs +35 -0
  378. package/tests/test-audio.mjs +219 -0
  379. package/tests/test-automation-timeouts.mjs +98 -0
  380. package/tests/test-behavior-tree.mjs +261 -0
  381. package/tests/test-blueprint-events.mjs +35 -0
  382. package/tests/test-blueprint-graph.mjs +79 -0
  383. package/tests/test-blueprint.mjs +577 -0
  384. package/tests/test-client-mode.mjs +86 -0
  385. package/tests/test-console-command.mjs +56 -0
  386. package/tests/test-control-actor.mjs +425 -0
  387. package/tests/test-control-editor.mjs +80 -0
  388. package/tests/test-extra-tools.mjs +38 -0
  389. package/tests/test-graphql.mjs +322 -0
  390. package/tests/test-inspect.mjs +72 -0
  391. package/tests/test-landscape.mjs +60 -0
  392. package/tests/test-manage-asset.mjs +438 -0
  393. package/tests/test-manage-level.mjs +70 -0
  394. package/tests/test-materials.mjs +356 -0
  395. package/tests/test-niagara.mjs +185 -0
  396. package/tests/test-no-inline-python.mjs +122 -0
  397. package/tests/test-plugin-handshake.mjs +82 -0
  398. package/tests/test-render.mjs +33 -0
  399. package/tests/test-runner.mjs +933 -0
  400. package/tests/test-search-assets.mjs +66 -0
  401. package/tests/test-sequence.mjs +68 -0
  402. package/tests/test-system.mjs +57 -0
  403. package/tests/test-wasm.mjs +193 -0
  404. package/tests/test-world-partition.mjs +215 -0
  405. package/tsconfig.json +3 -3
  406. package/wasm/Cargo.lock +363 -0
  407. package/wasm/Cargo.toml +42 -0
  408. package/wasm/LICENSE +21 -0
  409. package/wasm/README.md +253 -0
  410. package/wasm/src/dependency_resolver.rs +377 -0
  411. package/wasm/src/lib.rs +153 -0
  412. package/wasm/src/property_parser.rs +271 -0
  413. package/wasm/src/transform_math.rs +396 -0
  414. package/wasm/tests/integration.rs +109 -0
  415. package/.github/workflows/smithery-build.yml +0 -29
  416. package/dist/tools/build_environment_advanced.d.ts +0 -65
  417. package/dist/tools/build_environment_advanced.js +0 -633
  418. package/dist/tools/rc.d.ts +0 -110
  419. package/dist/tools/rc.js +0 -437
  420. package/dist/tools/visual.d.ts +0 -40
  421. package/dist/tools/visual.js +0 -282
  422. package/dist/utils/http.d.ts +0 -6
  423. package/dist/utils/http.js +0 -151
  424. package/dist/utils/python-output.d.ts +0 -18
  425. package/dist/utils/python-output.js +0 -290
  426. package/dist/utils/python.d.ts +0 -2
  427. package/dist/utils/python.js +0 -4
  428. package/dist/utils/stdio-redirect.d.ts +0 -2
  429. package/dist/utils/stdio-redirect.js +0 -20
  430. package/docs/unreal-tool-test-cases.md +0 -572
  431. package/smithery.yaml +0 -29
  432. package/src/tools/build_environment_advanced.ts +0 -732
  433. package/src/tools/rc.ts +0 -515
  434. package/src/tools/visual.ts +0 -281
  435. package/src/utils/http.ts +0 -187
  436. package/src/utils/python-output.ts +0 -351
  437. package/src/utils/python.ts +0 -3
  438. package/src/utils/stdio-redirect.ts +0 -18
@@ -0,0 +1,415 @@
1
+ import { cleanObject } from '../../utils/safe-json.js';
2
+ import { ITools } from '../../types/tool-interfaces.js';
3
+ import { executeAutomationRequest, requireNonEmptyString } from './common-handlers.js';
4
+
5
+ const managedSequences = new Set<string>();
6
+ const deletedSequences = new Set<string>();
7
+
8
+ function normalizeSequencePath(path: unknown): string | undefined {
9
+ if (typeof path !== 'string') return undefined;
10
+ const trimmed = path.trim();
11
+ return trimmed.length > 0 ? trimmed : undefined;
12
+ }
13
+
14
+ function markSequenceCreated(path: unknown) {
15
+ const norm = normalizeSequencePath(path);
16
+ if (!norm) return;
17
+ deletedSequences.delete(norm);
18
+ managedSequences.add(norm);
19
+ }
20
+
21
+ function markSequenceDeleted(path: unknown) {
22
+ const norm = normalizeSequencePath(path);
23
+ if (!norm) return;
24
+ managedSequences.delete(norm);
25
+ deletedSequences.add(norm);
26
+ }
27
+
28
+
29
+
30
+ export async function handleSequenceTools(action: string, args: any, tools: ITools) {
31
+ const seqAction = String(action || '').trim();
32
+ switch (seqAction) {
33
+ case 'create': {
34
+ const name = requireNonEmptyString(args.name, 'name', 'Missing required parameter: name');
35
+ const res = await tools.sequenceTools.create({ name, path: args.path });
36
+
37
+ let sequencePath: string | undefined;
38
+ if (res && (res as any).result && typeof (res as any).result.sequencePath === 'string') {
39
+ sequencePath = (res as any).result.sequencePath;
40
+ } else if (typeof args.path === 'string' && args.path.trim().length > 0) {
41
+ const basePath = args.path.trim().replace(/\/$/, '');
42
+ sequencePath = `${basePath}/${name}`;
43
+ }
44
+ if (sequencePath && res && (res as any).success !== false) {
45
+ markSequenceCreated(sequencePath);
46
+ }
47
+
48
+ const errorCode = String((res && (res as any).error) || '').toUpperCase();
49
+ const msgLower = String((res && (res as any).message) || '').toLowerCase();
50
+ if (res && (res as any).success === false && (errorCode === 'FACTORY_NOT_AVAILABLE' || msgLower.includes('ulevelsequencefactorynew not available'))) {
51
+ const path = sequencePath || (typeof args.path === 'string' ? args.path : undefined);
52
+ return cleanObject({
53
+ success: false,
54
+ error: 'FACTORY_NOT_AVAILABLE',
55
+ message: (res as any).message || 'Sequence creation failed: factory not available',
56
+ action: 'create',
57
+ name,
58
+ path,
59
+ sequencePath,
60
+ handled: true
61
+ });
62
+ }
63
+
64
+ return cleanObject(res);
65
+ }
66
+ case 'open': {
67
+ const path = requireNonEmptyString(args.path, 'path', 'Missing required parameter: path');
68
+ const res = await tools.sequenceTools.open({ path });
69
+ return cleanObject(res);
70
+ }
71
+ case 'add_camera': {
72
+ const res = await tools.sequenceTools.addCamera({ spawnable: args.spawnable, path: args.path });
73
+ return cleanObject(res);
74
+ }
75
+ case 'add_actor': {
76
+ const actorName = requireNonEmptyString(args.actorName, 'actorName', 'Missing required parameter: actorName');
77
+ const path = typeof args.path === 'string' ? args.path.trim() : '';
78
+ const payload = {
79
+ ...args,
80
+ actorName,
81
+ path: path || args.path,
82
+ subAction: 'add_actor'
83
+ };
84
+
85
+ const res = await executeAutomationRequest(tools, 'manage_sequence', payload);
86
+
87
+ const errorCode = String((res && (res as any).error) || '').toUpperCase();
88
+ const msgLower = String((res && (res as any).message) || '').toLowerCase();
89
+
90
+ if (res && (res as any).success === false && path) {
91
+ const isInvalidSequence = errorCode === 'INVALID_SEQUENCE' || msgLower.includes('sequence_add_actor requires a sequence path') || msgLower.includes('sequence not found');
92
+ if (isInvalidSequence) {
93
+ return cleanObject({
94
+ success: false,
95
+ error: 'NOT_FOUND',
96
+ message: (res as any).message || 'Sequence not found',
97
+ action: 'add_actor',
98
+ path,
99
+ actorName
100
+ });
101
+ }
102
+ }
103
+
104
+ const results = res && (res as any).result && Array.isArray((res as any).result.results)
105
+ ? (res as any).result.results as any[]
106
+ : undefined;
107
+ if (results && results.length) {
108
+ const failed = results.find((item) => item && item.success === false && typeof item.error === 'string');
109
+ if (failed) {
110
+ const errText = String(failed.error).toLowerCase();
111
+ if (errText.includes('actor not found')) {
112
+ return cleanObject({
113
+ success: false,
114
+ error: 'NOT_FOUND',
115
+ message: failed.error,
116
+ action: 'add_actor',
117
+ path: path || undefined,
118
+ actorName
119
+ });
120
+ }
121
+ }
122
+ }
123
+
124
+ return cleanObject(res);
125
+ }
126
+ case 'add_actors': {
127
+ const actorNames: string[] = Array.isArray(args.actorNames) ? args.actorNames : [];
128
+ const res = await tools.sequenceTools.addActors({ actorNames, path: args.path });
129
+ const errorCode = String((res && (res as any).error) || '').toUpperCase();
130
+ const msgLower = String((res && (res as any).message) || '').toLowerCase();
131
+ if (actorNames.length === 0 && res && (res as any).success === false && errorCode === 'INVALID_ARGUMENT') {
132
+ return cleanObject({
133
+ success: false,
134
+ error: 'INVALID_ARGUMENT',
135
+ message: (res as any).message || 'Invalid argument: actorNames required',
136
+ action: 'add_actors',
137
+ actorNames
138
+ });
139
+ }
140
+ if (res && (res as any).success === false && msgLower.includes('actor not found')) {
141
+ return cleanObject({
142
+ success: false,
143
+ error: 'NOT_FOUND',
144
+ message: (res as any).message || 'Actor not found',
145
+ action: 'add_actors',
146
+ actorNames
147
+ });
148
+ }
149
+ return cleanObject(res);
150
+ }
151
+ case 'remove_actors': {
152
+ const actorNames: string[] = Array.isArray(args.actorNames) ? args.actorNames : [];
153
+ const res = await tools.sequenceTools.removeActors({ actorNames, path: args.path });
154
+ return cleanObject(res);
155
+ }
156
+ case 'get_bindings': {
157
+ const path = typeof args.path === 'string' ? args.path : undefined;
158
+ const res = await tools.sequenceTools.getBindings({ path });
159
+ return cleanObject(res);
160
+ }
161
+ case 'add_keyframe': {
162
+ const path = typeof args.path === 'string' ? args.path.trim() : '';
163
+ const actorName = typeof args.actorName === 'string' ? args.actorName : undefined;
164
+ const property = typeof args.property === 'string' ? args.property : undefined;
165
+ const frame = typeof args.frame === 'number' ? args.frame : Number(args.frame);
166
+
167
+ const payload = {
168
+ ...args,
169
+ path: path || args.path,
170
+ actorName,
171
+ property,
172
+ frame,
173
+ subAction: 'add_keyframe'
174
+ };
175
+
176
+ // Fix: Map common property names to internal names
177
+ if (property === 'Location') {
178
+ payload.property = 'Transform';
179
+ payload.value = { location: args.value };
180
+ } else if (property === 'Rotation') {
181
+ payload.property = 'Transform';
182
+ payload.value = { rotation: args.value };
183
+ } else if (property === 'Scale') {
184
+ payload.property = 'Transform';
185
+ payload.value = { scale: args.value };
186
+ }
187
+
188
+ const res = await executeAutomationRequest(tools, 'manage_sequence', payload);
189
+ const errorCode = String((res && (res as any).error) || '').toUpperCase();
190
+ const msgLower = String((res && (res as any).message) || '').toLowerCase();
191
+
192
+ // Keep explicit INVALID_ARGUMENT for missing frame as a real error
193
+ if (errorCode === 'INVALID_ARGUMENT' || msgLower.includes('frame number is required')) {
194
+ return cleanObject(res);
195
+ }
196
+
197
+ if (res && (res as any).success === false) {
198
+ const isBindingIssue = errorCode === 'BINDING_NOT_FOUND' || msgLower.includes('binding not found');
199
+ const isUnsupported = errorCode === 'UNSUPPORTED_PROPERTY' || msgLower.includes('unsupported property') || msgLower.includes('invalid_sequence_type');
200
+ const isInvalidSeq = errorCode === 'INVALID_SEQUENCE' || msgLower.includes('sequence not found') || msgLower.includes('requires a sequence path');
201
+
202
+ if (path && isInvalidSeq) {
203
+ return cleanObject({
204
+ success: false,
205
+ error: 'NOT_FOUND',
206
+ message: (res as any).message || 'Sequence not found',
207
+ action: 'add_keyframe',
208
+ path,
209
+ actorName,
210
+ property,
211
+ frame
212
+ });
213
+ }
214
+
215
+ // Preserve plugin-provided failure for binding / unsupported-property cases
216
+ if (path && (isBindingIssue || isUnsupported)) {
217
+ return cleanObject(res);
218
+ }
219
+ }
220
+
221
+ return cleanObject(res);
222
+ }
223
+ case 'add_spawnable_from_class': {
224
+ const className = requireNonEmptyString(args.className, 'className', 'Missing required parameter: className');
225
+ const res = await tools.sequenceTools.addSpawnableFromClass({ className, path: args.path });
226
+ return cleanObject(res);
227
+ }
228
+ case 'play': {
229
+ const res = await tools.sequenceTools.play({ path: args.path, startTime: args.startTime, loopMode: args.loopMode });
230
+ return cleanObject(res);
231
+ }
232
+ case 'pause': {
233
+ const res = await tools.sequenceTools.pause({ path: args.path });
234
+ return cleanObject(res);
235
+ }
236
+ case 'stop': {
237
+ const res = await tools.sequenceTools.stop({ path: args.path });
238
+ return cleanObject(res);
239
+ }
240
+ case 'set_properties': {
241
+ const res = await tools.sequenceTools.setSequenceProperties({
242
+ path: args.path,
243
+ frameRate: args.frameRate,
244
+ lengthInFrames: args.lengthInFrames,
245
+ playbackStart: args.playbackStart,
246
+ playbackEnd: args.playbackEnd
247
+ });
248
+ return cleanObject(res);
249
+ }
250
+ case 'get_properties': {
251
+ const path = typeof args.path === 'string' ? args.path : undefined;
252
+ const res = await tools.sequenceTools.getSequenceProperties({ path });
253
+ return cleanObject(res);
254
+ }
255
+ case 'set_playback_speed': {
256
+ const speed = Number(args.speed);
257
+ if (!Number.isFinite(speed) || speed <= 0) {
258
+ throw new Error('Invalid speed: must be a positive number');
259
+ }
260
+ // Try setting speed
261
+ let res = await tools.sequenceTools.setPlaybackSpeed({ speed, path: args.path });
262
+
263
+ // Fix: Auto-open if editor not open
264
+ const errorCode = String((res && (res as any).error) || '').toUpperCase();
265
+ if ((!res || res.success === false) && errorCode === 'EDITOR_NOT_OPEN' && args.path) {
266
+ // Attempt to open the sequence
267
+ await tools.sequenceTools.open({ path: args.path });
268
+
269
+ // Wait a short moment for editor to initialize on game thread
270
+ await new Promise(resolve => setTimeout(resolve, 1000));
271
+
272
+ // Retry
273
+ res = await tools.sequenceTools.setPlaybackSpeed({ speed, path: args.path });
274
+ }
275
+
276
+ return cleanObject(res);
277
+ }
278
+ case 'list': {
279
+ const res = await tools.sequenceTools.list({ path: args.path });
280
+ return cleanObject(res);
281
+ }
282
+ case 'duplicate': {
283
+ const path = requireNonEmptyString(args.path, 'path', 'Missing required parameter: path');
284
+ const destDir = requireNonEmptyString(args.destinationPath, 'destinationPath', 'Missing required parameter: destinationPath');
285
+ const newName = requireNonEmptyString(args.newName || path.split('/').pop(), 'newName', 'Missing required parameter: newName');
286
+ const baseDir = destDir.replace(/\/$/, '');
287
+ const destPath = `${baseDir}/${newName}`;
288
+ const res = await tools.sequenceTools.duplicate({ path, destinationPath: destPath });
289
+ return cleanObject(res);
290
+ }
291
+ case 'rename': {
292
+ const path = requireNonEmptyString(args.path, 'path', 'Missing required parameter: path');
293
+ const newName = requireNonEmptyString(args.newName, 'newName', 'Missing required parameter: newName');
294
+ const res = await tools.sequenceTools.rename({ path, newName });
295
+ const errorCode = String((res && (res as any).error) || '').toUpperCase();
296
+ const msgLower = String((res && (res as any).message) || '').toLowerCase();
297
+ if (res && (res as any).success === false && (errorCode === 'OPERATION_FAILED' || msgLower.includes('failed to rename sequence'))) {
298
+ // Return actual failure, not best-effort success - rename is a destructive operation
299
+ return cleanObject({
300
+ success: false,
301
+ error: 'OPERATION_FAILED',
302
+ message: (res as any).message || 'Failed to rename sequence',
303
+ action: 'rename',
304
+ path,
305
+ newName
306
+ });
307
+ }
308
+ return cleanObject(res);
309
+ }
310
+ case 'delete': {
311
+ const path = requireNonEmptyString(args.path, 'path', 'Missing required parameter: path');
312
+ const res = await tools.sequenceTools.deleteSequence({ path });
313
+
314
+ if (res && (res as any).success !== false) {
315
+ markSequenceDeleted(path);
316
+ }
317
+ return cleanObject(res);
318
+ }
319
+ case 'get_metadata': {
320
+ const res = await tools.sequenceTools.getMetadata({ path: args.path });
321
+ return cleanObject(res);
322
+ }
323
+ case 'set_metadata': {
324
+ const path = requireNonEmptyString(args.path, 'path', 'Missing required parameter: path');
325
+ const metadata = (args.metadata && typeof args.metadata === 'object') ? args.metadata : {};
326
+ const res = await executeAutomationRequest(tools, 'set_metadata', { assetPath: path, metadata });
327
+ return cleanObject(res);
328
+ }
329
+ case 'add_track': {
330
+ // Forward add_track to the C++ plugin - it requires MovieScene API
331
+ const path = typeof args.path === 'string' ? args.path.trim() : '';
332
+ const trackType = typeof args.trackType === 'string' ? args.trackType : '';
333
+ const trackName = typeof args.trackName === 'string' ? args.trackName : '';
334
+ const actorName = typeof args.actorName === 'string' ? args.actorName : undefined;
335
+
336
+ // Fix: Check if actor is bound before adding track
337
+ if (actorName) {
338
+ const bindingsRes = await tools.sequenceTools.getBindings({ path });
339
+ if (bindingsRes && bindingsRes.success) {
340
+ const bindings = bindingsRes.bindings || [];
341
+ const isBound = bindings.some((b: any) => b.name === actorName);
342
+ if (!isBound) {
343
+ return cleanObject({
344
+ success: false,
345
+ error: 'BINDING_NOT_FOUND',
346
+ message: `Actor '${actorName}' is not bound to this sequence. Please call 'add_actor' first.`,
347
+ action: 'add_track',
348
+ path,
349
+ actorName
350
+ });
351
+ }
352
+ }
353
+ }
354
+
355
+ const payload = {
356
+ ...args,
357
+ path: path || args.path,
358
+ trackType,
359
+ trackName,
360
+ actorName,
361
+ subAction: 'add_track'
362
+ };
363
+
364
+ const res = await executeAutomationRequest(tools, 'manage_sequence', payload);
365
+ return cleanObject(res);
366
+ }
367
+ case 'add_section': {
368
+ // Forward add_section to C++
369
+ const payload = { ...args, subAction: 'add_section' };
370
+ return cleanObject(await executeAutomationRequest(tools, 'manage_sequence', payload));
371
+ }
372
+ case 'remove_track': {
373
+ // Forward remove_track to C++
374
+ const payload = { ...args, subAction: 'remove_track' };
375
+ return cleanObject(await executeAutomationRequest(tools, 'manage_sequence', payload));
376
+ }
377
+ case 'set_track_muted': {
378
+ const payload = { ...args, subAction: 'set_track_muted' };
379
+ return cleanObject(await executeAutomationRequest(tools, 'manage_sequence', payload));
380
+ }
381
+ case 'set_track_solo': {
382
+ const payload = { ...args, subAction: 'set_track_solo' };
383
+ return cleanObject(await executeAutomationRequest(tools, 'manage_sequence', payload));
384
+ }
385
+ case 'set_track_locked': {
386
+ const payload = { ...args, subAction: 'set_track_locked' };
387
+ return cleanObject(await executeAutomationRequest(tools, 'manage_sequence', payload));
388
+ }
389
+ case 'list_tracks': {
390
+ const path = requireNonEmptyString(args.path, 'path', 'Missing required parameter: path');
391
+ const res = await tools.sequenceTools.listTracks({ path });
392
+ return cleanObject(res);
393
+ }
394
+ case 'set_work_range': {
395
+ const start = Number(args.start);
396
+ const end = Number(args.end);
397
+ // Validate start/end are numbers
398
+ if (!Number.isFinite(start)) throw new Error('Invalid start: must be a number');
399
+ if (!Number.isFinite(end)) throw new Error('Invalid end: must be a number');
400
+
401
+ const res = await tools.sequenceTools.setWorkRange({
402
+ path: args.path,
403
+ start,
404
+ end
405
+ });
406
+ return cleanObject(res);
407
+ }
408
+ default:
409
+ // Ensure subAction is set for compatibility with C++ handler expectations
410
+ if (args.action && !args.subAction) {
411
+ args.subAction = args.action;
412
+ }
413
+ return await executeAutomationRequest(tools, 'manage_sequence', args);
414
+ }
415
+ }