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,117 @@
1
+ #pragma once
2
+
3
+ #include "CoreMinimal.h"
4
+ #include "Containers/Ticker.h"
5
+ #include "Dom/JsonObject.h"
6
+ #include "Templates/SharedPointer.h"
7
+ #include "Misc/ScopeLock.h"
8
+
9
+ class FMcpBridgeWebSocket;
10
+ class UMcpAutomationBridgeSettings;
11
+
12
+ /**
13
+ * Delegate for handling incoming automation requests.
14
+ * Params: RequestId, Action, Payload, RequestingSocket
15
+ */
16
+ DECLARE_DELEGATE_FourParams(FMcpMessageReceivedCallback, const FString&, const FString&, const TSharedPtr<FJsonObject>&, TSharedPtr<FMcpBridgeWebSocket>);
17
+
18
+ /**
19
+ * Manages WebSocket connections for the MCP Automation Bridge.
20
+ * Handles listening, connecting, reconnecting, heartbeats, and message dispatching.
21
+ */
22
+ class MCPAUTOMATIONBRIDGE_API FMcpConnectionManager : public TSharedFromThis<FMcpConnectionManager>
23
+ {
24
+ public:
25
+ FMcpConnectionManager();
26
+ ~FMcpConnectionManager();
27
+
28
+ void Initialize(const UMcpAutomationBridgeSettings* Settings);
29
+ void Start();
30
+ void Stop();
31
+
32
+ bool IsConnected() const;
33
+ bool IsBridgeActive() const { return bBridgeAvailable; }
34
+
35
+ bool SendRawMessage(const FString& Message);
36
+ void SendAutomationResponse(TSharedPtr<FMcpBridgeWebSocket> TargetSocket, const FString& RequestId, bool bSuccess, const FString& Message, const TSharedPtr<FJsonObject>& Result, const FString& ErrorCode);
37
+ void SendControlMessage(const TSharedPtr<FJsonObject>& Message);
38
+
39
+ void SetOnMessageReceived(FMcpMessageReceivedCallback InCallback);
40
+
41
+ // Request tracking helpers
42
+ int32 GetActiveSocketCount() const;
43
+ void RegisterRequestSocket(const FString& RequestId, TSharedPtr<FMcpBridgeWebSocket> Socket);
44
+
45
+ // Telemetry helpers
46
+ void StartRequestTelemetry(const FString& RequestId, const FString& Action);
47
+ void RecordAutomationTelemetry(const FString& RequestId, bool bSuccess, const FString& Message, const FString& ErrorCode);
48
+
49
+ bool Tick(float DeltaTime);
50
+
51
+ private:
52
+ void AttemptConnection();
53
+ void ForceReconnect(const FString& Reason, float ReconnectDelayOverride = -1.0f);
54
+
55
+ void HandleConnected(TSharedPtr<FMcpBridgeWebSocket> Socket);
56
+ void HandleClientConnected(TSharedPtr<FMcpBridgeWebSocket> ClientSocket);
57
+ void HandleConnectionError(TSharedPtr<FMcpBridgeWebSocket> Socket, const FString& Error);
58
+ void HandleServerConnectionError(const FString& Error);
59
+ void HandleClosed(TSharedPtr<FMcpBridgeWebSocket> Socket, int32 StatusCode, const FString& Reason, bool bWasClean);
60
+ void HandleMessage(TSharedPtr<FMcpBridgeWebSocket> Socket, const FString& Message);
61
+ void HandleHeartbeat(TSharedPtr<FMcpBridgeWebSocket> Socket);
62
+
63
+ void EmitAutomationTelemetrySummaryIfNeeded(double NowSeconds);
64
+
65
+ private:
66
+ TArray<TSharedPtr<FMcpBridgeWebSocket>> ActiveSockets;
67
+ TMap<FString, TSharedPtr<FMcpBridgeWebSocket>> PendingRequestsToSockets;
68
+ FTSTicker::FDelegateHandle TickerHandle;
69
+ FMcpMessageReceivedCallback OnMessageReceived;
70
+
71
+ // Configuration
72
+ FString EnvListenHost;
73
+ FString EnvListenPorts;
74
+ FString EndpointUrl;
75
+ FString CapabilityToken;
76
+ FString ServerName;
77
+ FString ServerVersion;
78
+ FString ActiveSessionId;
79
+
80
+ int32 ClientPort = 0;
81
+ float AutoReconnectDelaySeconds = 5.0f;
82
+ float HeartbeatTimeoutSeconds = 0.0f;
83
+
84
+ bool bRequireCapabilityToken = false;
85
+ bool bEnvListenPortsSet = false;
86
+ bool bHeartbeatTrackingEnabled = false;
87
+
88
+ // State
89
+ bool bBridgeAvailable = false;
90
+ bool bReconnectEnabled = true;
91
+ float TimeUntilReconnect = 0.0f;
92
+ double LastHeartbeatTimestamp = 0.0;
93
+
94
+ // Telemetry
95
+ struct FAutomationRequestTelemetry
96
+ {
97
+ FString Action;
98
+ double StartTimeSeconds = 0.0;
99
+ };
100
+
101
+ struct FAutomationActionStats
102
+ {
103
+ int32 SuccessCount = 0;
104
+ int32 FailureCount = 0;
105
+ double TotalSuccessDurationSeconds = 0.0;
106
+ double TotalFailureDurationSeconds = 0.0;
107
+ double LastDurationSeconds = 0.0;
108
+ double LastUpdatedSeconds = 0.0;
109
+ };
110
+
111
+ TMap<FString, FAutomationRequestTelemetry> ActiveRequestTelemetry;
112
+ TMap<FString, FAutomationActionStats> AutomationActionTelemetry;
113
+ double TelemetrySummaryIntervalSeconds = 120.0;
114
+ double LastTelemetrySummaryLogSeconds = 0.0;
115
+
116
+ mutable FCriticalSection PendingRequestsMutex;
117
+ };
@@ -0,0 +1,19 @@
1
+ #!/usr/bin/env node
2
+ import net from 'node:net';
3
+
4
+ const HOST = process.env.MCP_AUTOMATION_WS_HOST || '127.0.0.1';
5
+ const PORT = process.env.MCP_AUTOMATION_WS_PORT ? parseInt(process.env.MCP_AUTOMATION_WS_PORT) : 8091;
6
+ console.log(`Checking connection to Unreal Engine on port ${PORT}...`);
7
+
8
+ function check(host, port, timeoutMs = 2000) {
9
+ return new Promise((resolve) => {
10
+ const socket = new net.Socket();
11
+ const timer = setTimeout(() => { try { socket.destroy(); } catch {} resolve(false); }, timeoutMs);
12
+ socket.once('error', () => { clearTimeout(timer); resolve(false); });
13
+ socket.connect(port, host, () => { clearTimeout(timer); try { socket.destroy(); } catch {} resolve(true); });
14
+ });
15
+ }
16
+
17
+ const ok = await check(HOST, PORT);
18
+ console.log(ok ? `OK: ${HOST}:${PORT} accepting TCP` : `DOWN: ${HOST}:${PORT} not reachable`);
19
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,23 @@
1
+ import fs from 'fs';
2
+ import path from 'path';
3
+ import { fileURLToPath } from 'url';
4
+
5
+ const __filename = fileURLToPath(import.meta.url);
6
+ const __dirname = path.dirname(__filename);
7
+
8
+ const TMP_DIR = path.resolve(__dirname, '..', 'tmp');
9
+ if (!fs.existsSync(TMP_DIR)) {
10
+ console.log('No tmp directory present.');
11
+ process.exit(0);
12
+ }
13
+
14
+ for (const f of fs.readdirSync(TMP_DIR)) {
15
+ const p = path.join(TMP_DIR, f);
16
+ try {
17
+ fs.unlinkSync(p);
18
+ console.log('Removed', p);
19
+ } catch (e) {
20
+ console.warn('Failed to remove', p, e.message || e);
21
+ }
22
+ }
23
+ console.log('tmp cleanup complete.');
@@ -0,0 +1,26 @@
1
+ import fs from 'fs';
2
+ import path from 'path';
3
+ import { fileURLToPath } from 'url';
4
+
5
+ const __filename = fileURLToPath(import.meta.url);
6
+ const __dirname = path.dirname(__filename);
7
+
8
+ const wasmPkgPath = path.resolve(__dirname, '../src/wasm/pkg/unreal_mcp_wasm.js');
9
+
10
+ try {
11
+ if (fs.existsSync(wasmPkgPath)) {
12
+ let content = fs.readFileSync(wasmPkgPath, 'utf8');
13
+ if (content.includes('console.log(getObject(arg0));')) {
14
+ content = content.replace('console.log(getObject(arg0));', 'console.error(getObject(arg0));');
15
+ fs.writeFileSync(wasmPkgPath, content);
16
+ console.log('Successfully patched console.log to console.error in WASM bindings.');
17
+ } else {
18
+ console.log('WASM bindings already patched or console.log not found.');
19
+ }
20
+ } else {
21
+ console.warn('WASM binding file not found at:', wasmPkgPath);
22
+ }
23
+ } catch (error) {
24
+ console.error('Error patching WASM bindings:', error);
25
+ process.exit(1);
26
+ }
@@ -0,0 +1,131 @@
1
+ #!/usr/bin/env node
2
+ import { spawn } from 'node:child_process';
3
+
4
+ const tests = [
5
+ 'test:control_actor', // Pass
6
+ 'test:control_editor', // Pass
7
+ 'test:manage_level', // Pass
8
+ 'test:animation', // Pass
9
+ 'test:materials', // Pass
10
+ 'test:niagara', // Pass
11
+ 'test:landscape', // Pass
12
+ 'test:sequence', // Pass
13
+ 'test:system', // Pass
14
+ 'test:console_command', // Pass
15
+ 'test:inspect', // Pass
16
+ 'test:manage_asset', // Pass
17
+ 'test:blueprint', // Pass
18
+ 'test:blueprint_graph', // Pass
19
+ // 'test:graphql', // Pass
20
+ 'test:wasm:all', // Pass
21
+ 'test:no-inline-python',
22
+ 'test:plugin-handshake', // Pass
23
+ 'test:asset_advanced', // Pass
24
+ 'test:render', // Pass
25
+ 'test:world_partition' // Pass
26
+ ];
27
+
28
+ const isWindows = process.platform === 'win32';
29
+ const npmExecPath = process.env.npm_execpath;
30
+ const npmCommand = isWindows ? 'npm.cmd' : 'npm';
31
+
32
+ function spawnNpm(cmd) {
33
+ if (npmExecPath && npmExecPath.endsWith('.js')) {
34
+ return spawn(process.execPath, [npmExecPath, 'run', cmd], {
35
+ stdio: 'inherit',
36
+ cwd: process.cwd(),
37
+ env: process.env,
38
+ windowsHide: true
39
+ });
40
+ }
41
+ return spawn(npmCommand, ['run', cmd], {
42
+ stdio: 'inherit',
43
+ cwd: process.cwd(),
44
+ env: process.env,
45
+ windowsHide: true
46
+ });
47
+ }
48
+
49
+ function runOne(cmd) {
50
+ return new Promise((resolve) => {
51
+ const startedAt = Date.now();
52
+ const child = spawnNpm(cmd);
53
+
54
+ let errorMsg = '';
55
+ child.on('error', (err) => {
56
+ errorMsg = `Failed to launch npm run ${cmd}: ${err.message}`;
57
+ });
58
+
59
+ child.on('exit', (code, signal) => {
60
+ const durationMs = Date.now() - startedAt;
61
+ if (signal) {
62
+ resolve({ name: cmd, ok: false, code: null, signal, durationMs, error: `${cmd} terminated via signal ${signal}` });
63
+ } else {
64
+ resolve({ name: cmd, ok: code === 0, code, signal: null, durationMs, error: code === 0 ? null : (errorMsg || `${cmd} failed with code ${code}`) });
65
+ }
66
+ });
67
+
68
+ const forwardSignal = (sig) => {
69
+ if (child.killed) return;
70
+ child.kill(sig);
71
+ };
72
+
73
+ process.once('SIGTERM', forwardSignal);
74
+
75
+ child.once('exit', () => {
76
+ process.removeListener('SIGINT', forwardSignal);
77
+ process.removeListener('SIGTERM', forwardSignal);
78
+ });
79
+ });
80
+ }
81
+
82
+ (async () => {
83
+ // Pass 0: Build once
84
+ console.log('\n============================================================');
85
+ console.log('Running Pre-Test Build');
86
+ console.log('============================================================');
87
+
88
+ // Use runOne to execute the build command
89
+ // We can't use npm run build directly if it's not in the 'tests' array, so let's just trigger it.
90
+ const buildRes = await runOne('build');
91
+ if (!buildRes.ok) {
92
+ console.error(`\nBuild failed (code=${buildRes.code ?? 'n/a'}). Aborting tests.`);
93
+ process.exit(1);
94
+ }
95
+ console.log('Build completed successfully. Disabling auto-build for tests.');
96
+
97
+ // Set the environment variable to disable auto-build for subsequent tests
98
+ process.env.UNREAL_MCP_NO_AUTO_BUILD = '1';
99
+ // Force use of the just-built dist folder to prevent falling back to ts-node
100
+ // (which can happen if build:wasm updates timestamps in src/ after build:core)
101
+ process.env.UNREAL_MCP_FORCE_DIST = '1';
102
+
103
+ const results = [];
104
+ for (const t of tests) {
105
+ const res = await runOne(t);
106
+ results.push(res);
107
+ if (!res.ok) {
108
+ console.error(`\n${t} failed (code=${res.code ?? 'n/a'}, signal=${res.signal ?? 'n/a'}) after ${res.durationMs}ms`);
109
+ if (res.error) console.error(res.error);
110
+ }
111
+ }
112
+
113
+ // Summary
114
+ const passed = results.filter(r => r.ok).length;
115
+ const failed = results.length - passed;
116
+ console.log('\n============================================================');
117
+ console.log('Test Suite Summary');
118
+ console.log('============================================================');
119
+ for (const r of results) {
120
+ const status = r.ok ? 'PASSED' : 'FAILED';
121
+ console.log(`${status} - ${r.name} (${r.durationMs} ms)`);
122
+ }
123
+ console.log('------------------------------------------------------------');
124
+ console.log(`Total: ${results.length} Passed: ${passed} Failed: ${failed}`);
125
+ console.log('============================================================');
126
+
127
+ process.exit(failed > 0 ? 1 : 0);
128
+ })().catch((err) => {
129
+ console.error(err.message || err);
130
+ process.exit(1);
131
+ });
@@ -0,0 +1,94 @@
1
+
2
+ import { spawn } from 'child_process';
3
+ import path from 'path';
4
+ import { fileURLToPath } from 'url';
5
+
6
+ const __dirname = path.dirname(fileURLToPath(import.meta.url));
7
+ const serverPath = path.join(__dirname, '../dist/cli.js');
8
+
9
+ console.log('🚬 Running Smoke Test (Mock Mode)...');
10
+ console.log(`🔌 Server Path: ${serverPath}`);
11
+
12
+ const env = { ...process.env, MOCK_UNREAL_CONNECTION: 'true' };
13
+
14
+ const child = spawn('node', [serverPath], {
15
+ env,
16
+ stdio: ['pipe', 'pipe', 'inherit'] // pipe stdin/stdout, inherit stderr
17
+ });
18
+
19
+ const requests = [
20
+ {
21
+ jsonrpc: '2.0',
22
+ id: 1,
23
+ method: 'initialize',
24
+ params: {
25
+ protocolVersion: '2024-11-05',
26
+ capabilities: {},
27
+ clientInfo: { name: 'smoke-test', version: '1.0' }
28
+ }
29
+ },
30
+ {
31
+ jsonrpc: '2.0',
32
+ id: 2,
33
+ method: 'tools/list',
34
+ params: {}
35
+ }
36
+ ];
37
+
38
+ let buffer = '';
39
+ let passed = false;
40
+
41
+ child.stdout.on('data', (data) => {
42
+ const chunk = data.toString();
43
+ buffer += chunk;
44
+
45
+ // Try to parse JSON lines
46
+ const lines = buffer.split('\n');
47
+ buffer = lines.pop() || ''; // Keep the incomplete last line
48
+
49
+ for (const line of lines) {
50
+ if (!line.trim()) continue;
51
+ try {
52
+ const msg = JSON.parse(line);
53
+ console.log('Received:', JSON.stringify(msg).substring(0, 100) + '...');
54
+
55
+ if (msg.id === 1 && msg.result) {
56
+ console.log('✅ Initialize success');
57
+ // Send list tools request
58
+ child.stdin.write(JSON.stringify(requests[1]) + '\n');
59
+ }
60
+
61
+ if (msg.id === 2 && msg.result) {
62
+ console.log(`✅ Tools check success: Found ${msg.result.tools?.length || 0} tools`);
63
+ passed = true;
64
+ child.kill();
65
+ }
66
+
67
+ } catch (_e) {
68
+ // Ignore non-JSON output (logs)
69
+ }
70
+ }
71
+ });
72
+
73
+ child.on('exit', (_code) => {
74
+ if (passed) {
75
+ console.log('🎉 Smoke Test PASSED');
76
+ process.exit(0);
77
+ } else {
78
+ console.error('❌ Smoke Test FAILED - Server exited without passing checks');
79
+ process.exit(1);
80
+ }
81
+ });
82
+
83
+ // Start by sending initialize
84
+ console.log('Sending initialize...');
85
+ child.stdin.write(JSON.stringify(requests[0]) + '\n');
86
+
87
+ // Timeout safety
88
+ setTimeout(() => {
89
+ if (!passed) {
90
+ console.error('❌ Timeout waiting for smoke test');
91
+ child.kill();
92
+ process.exit(1);
93
+ }
94
+ }, 10000);
@@ -0,0 +1,143 @@
1
+ #!/usr/bin/env node
2
+ import fs from 'fs';
3
+ import path from 'path';
4
+ import { fileURLToPath } from 'url';
5
+
6
+ const __filename = fileURLToPath(import.meta.url);
7
+ const __dirname = path.dirname(__filename);
8
+
9
+ const SOURCE_RELATIVE = ['..', 'plugins', 'McpAutomationBridge'];
10
+ const sourceDir = path.resolve(__dirname, ...SOURCE_RELATIVE);
11
+
12
+ function showHelp() {
13
+ const scriptName = path.basename(process.argv[1]);
14
+ console.log(`Usage: node ${scriptName} [--engine <engine_plugins_dir>] [--project <project_plugins_dir>] [--dry-run] [--clean-engine] [--clean-project]`);
15
+ console.log('Copies the MCP Automation Bridge plugin from this repository into Engine and/or Project plugin directories.');
16
+ console.log('--engine <dir> Path to the Engine/Plugins directory (the script appends McpAutomationBridge automatically).');
17
+ console.log('--project <dir> Path to the Project/Plugins directory (the script appends McpAutomationBridge automatically).');
18
+ console.log('--dry-run Log the operations without modifying the filesystem.');
19
+ console.log('--clean-engine Remove the destination engine plugin directory before copying.');
20
+ console.log('--clean-project Remove the destination project plugin directory before copying.');
21
+ console.log('--help Show this message.');
22
+ }
23
+
24
+ function parseArgs(argv) {
25
+ const args = {
26
+ engine: null,
27
+ project: null,
28
+ dryRun: false,
29
+ cleanEngine: false,
30
+ cleanProject: false,
31
+ help: false
32
+ };
33
+
34
+ for (let i = 2; i < argv.length; i += 1) {
35
+ const token = argv[i];
36
+ switch (token) {
37
+ case '--engine':
38
+ args.engine = argv[++i];
39
+ break;
40
+ case '--project':
41
+ args.project = argv[++i];
42
+ break;
43
+ case '--dry-run':
44
+ args.dryRun = true;
45
+ break;
46
+ case '--clean-engine':
47
+ args.cleanEngine = true;
48
+ break;
49
+ case '--clean-project':
50
+ args.cleanProject = true;
51
+ break;
52
+ case '--help':
53
+ case '-h':
54
+ args.help = true;
55
+ break;
56
+ default:
57
+ console.error(`Unknown argument: ${token}`);
58
+ args.help = true;
59
+ break;
60
+ }
61
+ }
62
+
63
+ return args;
64
+ }
65
+
66
+ function ensureSourceExists() {
67
+ if (!fs.existsSync(sourceDir) || !fs.statSync(sourceDir).isDirectory()) {
68
+ throw new Error(`Source directory missing: ${sourceDir}`);
69
+ }
70
+ }
71
+
72
+ function resolveDestination(rootDir, label) {
73
+ if (!rootDir) {
74
+ return null;
75
+ }
76
+
77
+ const fullPath = path.resolve(rootDir, 'McpAutomationBridge');
78
+ if (!fs.existsSync(path.dirname(fullPath))) {
79
+ throw new Error(`Destination parent directory does not exist for ${label}: ${path.dirname(fullPath)}`);
80
+ }
81
+
82
+ return fullPath;
83
+ }
84
+
85
+ function removeDir(targetPath, dryRun) {
86
+ if (!fs.existsSync(targetPath)) {
87
+ return;
88
+ }
89
+
90
+ if (dryRun) {
91
+ console.log(`[dry-run] Would remove directory: ${targetPath}`);
92
+ return;
93
+ }
94
+
95
+ fs.rmSync(targetPath, { recursive: true, force: true });
96
+ console.log(`Removed existing directory: ${targetPath}`);
97
+ }
98
+
99
+ function copyDir(targetPath, dryRun) {
100
+ if (dryRun) {
101
+ console.log(`[dry-run] Would copy ${sourceDir} -> ${targetPath}`);
102
+ return;
103
+ }
104
+
105
+ fs.cpSync(sourceDir, targetPath, { recursive: true });
106
+ console.log(`Copied ${sourceDir} -> ${targetPath}`);
107
+ }
108
+
109
+ function main() {
110
+ const args = parseArgs(process.argv);
111
+
112
+ if (args.help || (!args.engine && !args.project)) {
113
+ showHelp();
114
+ return;
115
+ }
116
+
117
+ ensureSourceExists();
118
+
119
+ if (args.engine) {
120
+ const engineDest = resolveDestination(args.engine, 'engine');
121
+ if (args.cleanEngine) {
122
+ removeDir(engineDest, args.dryRun);
123
+ }
124
+ copyDir(engineDest, args.dryRun);
125
+ }
126
+
127
+ if (args.project) {
128
+ const projectDest = resolveDestination(args.project, 'project');
129
+ if (args.cleanProject) {
130
+ removeDir(projectDest, args.dryRun);
131
+ }
132
+ copyDir(projectDest, args.dryRun);
133
+ }
134
+
135
+ console.log('Done. Remember to clear the plugin Binaries/Intermediate folders if requested by Unreal.');
136
+ }
137
+
138
+ try {
139
+ main();
140
+ } catch (err) {
141
+ console.error(`Error: ${err.message}`);
142
+ process.exitCode = 1;
143
+ }
@@ -0,0 +1,113 @@
1
+ #!/usr/bin/env node
2
+ import fs from 'node:fs/promises';
3
+ import path from 'node:path';
4
+ import { fileURLToPath } from 'node:url';
5
+
6
+ const __dirname = path.dirname(fileURLToPath(import.meta.url));
7
+ const repoRoot = path.resolve(__dirname, '..');
8
+ const srcRoot = path.resolve(repoRoot, 'src');
9
+
10
+ function isPositionInsideLiteralOrComment(content, pos) {
11
+ let inSingle = false;
12
+ let inDouble = false;
13
+ let inBacktick = false;
14
+ let inLineComment = false;
15
+ let inBlockComment = false;
16
+
17
+ for (let i = 0; i < pos; i += 1) {
18
+ const ch = content[i];
19
+ const next = content[i + 1];
20
+
21
+ if (inLineComment) {
22
+ if (ch === '\n') inLineComment = false;
23
+ continue;
24
+ }
25
+ if (inBlockComment) {
26
+ if (ch === '*' && next === '/') {
27
+ inBlockComment = false; i += 1; continue;
28
+ }
29
+ continue;
30
+ }
31
+ if (inSingle) {
32
+ if (ch === "\\") { i += 1; continue; }
33
+ if (ch === "'") { inSingle = false; }
34
+ continue;
35
+ }
36
+ if (inDouble) {
37
+ if (ch === "\\") { i += 1; continue; }
38
+ if (ch === '"') { inDouble = false; }
39
+ continue;
40
+ }
41
+ if (inBacktick) {
42
+ if (ch === "\\") { i += 1; continue; }
43
+ if (ch === '`') { inBacktick = false; }
44
+ continue;
45
+ }
46
+
47
+ if (ch === '/' && next === '/') { inLineComment = true; i += 1; continue; }
48
+ if (ch === '/' && next === '*') { inBlockComment = true; i += 1; continue; }
49
+ if (ch === "'") { inSingle = true; continue; }
50
+ if (ch === '"') { inDouble = true; continue; }
51
+ if (ch === '`') { inBacktick = true; continue; }
52
+ }
53
+
54
+ return inSingle || inDouble || inBacktick || inLineComment || inBlockComment;
55
+ }
56
+
57
+ async function listTsFiles(dir) {
58
+ const out = [];
59
+ const entries = await fs.readdir(dir, { withFileTypes: true });
60
+ for (const ent of entries) {
61
+ if (ent.name === 'node_modules' || ent.name === 'dist' || ent.name === '.git') continue;
62
+ const full = path.join(dir, ent.name);
63
+ if (ent.isDirectory()) {
64
+ out.push(...(await listTsFiles(full)));
65
+ } else if (ent.isFile() && full.endsWith('.ts')) {
66
+ out.push(full);
67
+ }
68
+ }
69
+ return out;
70
+ }
71
+
72
+ function toLineCol(content, index) {
73
+ const before = content.slice(0, index);
74
+ const lines = before.split(/\r?\n/);
75
+ const line = lines.length;
76
+ const col = lines[lines.length - 1].length + 1;
77
+ return { line, col };
78
+ }
79
+
80
+ async function main() {
81
+ const pattern = /simulated\s*:/g;
82
+ const files = await listTsFiles(srcRoot);
83
+ const failures = [];
84
+
85
+ for (const f of files) {
86
+ const content = await fs.readFile(f, 'utf8');
87
+ let m;
88
+ while ((m = pattern.exec(content)) !== null) {
89
+ const idx = m.index;
90
+ if (isPositionInsideLiteralOrComment(content, idx)) continue;
91
+ const pos = toLineCol(content, idx);
92
+ const snippetLine = content.split(/\r?\n/)[pos.line - 1] ?? '';
93
+ failures.push({ file: f, line: pos.line, col: pos.col, snippet: snippetLine.trim() });
94
+ }
95
+ }
96
+
97
+ if (failures.length > 0) {
98
+ console.error('\nDetected per-tool "simulated" alternatives usages in TypeScript files:');
99
+ for (const fail of failures) {
100
+ console.error(`\n - ${path.relative(repoRoot, fail.file)}:${fail.line}:${fail.col} -> ${fail.snippet}`);
101
+ }
102
+ console.error('\nPer-tool simulated alternatives are disallowed. Replace simulated behavior with explicit plugin actions or return an explicit error (e.g. MISSING_ENGINE_PLUGINS).');
103
+ process.exitCode = 1;
104
+ return;
105
+ }
106
+
107
+ console.log('No per-tool plugin "simulated" alternatives found in TypeScript source files.');
108
+ }
109
+
110
+ main().catch((err) => {
111
+ console.error('Test failed with unexpected error:', err);
112
+ process.exitCode = 2;
113
+ });