@dxos/app-framework 0.8.4-main.c4373fc → 0.8.4-main.c85a9c8dae

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 (481) hide show
  1. package/.storybook/main.mts +2 -4
  2. package/.storybook/preview.mts +2 -2
  3. package/dist/lib/browser/capability-2GL5JAGJ.mjs +37 -0
  4. package/dist/lib/browser/capability-2GL5JAGJ.mjs.map +7 -0
  5. package/dist/lib/browser/capability-7RLVE42K.mjs +34 -0
  6. package/dist/lib/browser/capability-7RLVE42K.mjs.map +7 -0
  7. package/dist/lib/browser/chunk-4CTRO67U.mjs +703 -0
  8. package/dist/lib/browser/chunk-4CTRO67U.mjs.map +7 -0
  9. package/dist/lib/browser/chunk-5RJNZV7K.mjs +44 -0
  10. package/dist/lib/browser/chunk-5RJNZV7K.mjs.map +7 -0
  11. package/dist/lib/browser/chunk-FHQTHCX7.mjs +8 -0
  12. package/dist/lib/browser/chunk-FHQTHCX7.mjs.map +7 -0
  13. package/dist/lib/browser/chunk-FNKT2QQ2.mjs +145 -0
  14. package/dist/lib/browser/chunk-FNKT2QQ2.mjs.map +7 -0
  15. package/dist/lib/browser/chunk-HE27PNNQ.mjs +824 -0
  16. package/dist/lib/browser/chunk-HE27PNNQ.mjs.map +7 -0
  17. package/dist/lib/browser/chunk-J5LGTIGS.mjs +10 -0
  18. package/dist/lib/browser/chunk-NPUEVX42.mjs +34 -0
  19. package/dist/lib/browser/chunk-NPUEVX42.mjs.map +7 -0
  20. package/dist/lib/browser/chunk-PKQT6C53.mjs +47 -0
  21. package/dist/lib/browser/chunk-PKQT6C53.mjs.map +7 -0
  22. package/dist/lib/browser/chunk-REORGDJT.mjs +80 -0
  23. package/dist/lib/browser/chunk-REORGDJT.mjs.map +7 -0
  24. package/dist/lib/browser/chunk-YAFEA4GV.mjs +1 -0
  25. package/dist/lib/browser/chunk-YNFPIQGB.mjs +76 -0
  26. package/dist/lib/browser/chunk-YNFPIQGB.mjs.map +7 -0
  27. package/dist/lib/browser/chunk-ZRWBPIZG.mjs +170 -0
  28. package/dist/lib/browser/chunk-ZRWBPIZG.mjs.map +7 -0
  29. package/dist/lib/browser/cli/index.mjs +89 -0
  30. package/dist/lib/browser/cli/index.mjs.map +7 -0
  31. package/dist/lib/browser/common/activation-events.mjs +23 -0
  32. package/dist/lib/browser/common/capabilities.mjs +45 -0
  33. package/dist/lib/browser/core/activation-event.mjs +20 -0
  34. package/dist/lib/browser/core/activation-event.mjs.map +7 -0
  35. package/dist/lib/browser/core/capability.mjs +28 -0
  36. package/dist/lib/browser/core/capability.mjs.map +7 -0
  37. package/dist/lib/browser/core/plugin-manager.mjs +15 -0
  38. package/dist/lib/browser/core/plugin-manager.mjs.map +7 -0
  39. package/dist/lib/browser/core/plugin.mjs +27 -0
  40. package/dist/lib/browser/core/plugin.mjs.map +7 -0
  41. package/dist/lib/browser/index.mjs +87 -148
  42. package/dist/lib/browser/index.mjs.map +4 -4
  43. package/dist/lib/browser/invoker-capability-BNLVNYHU.mjs +36 -0
  44. package/dist/lib/browser/invoker-capability-BNLVNYHU.mjs.map +7 -0
  45. package/dist/lib/browser/meta.json +1 -1
  46. package/dist/lib/browser/testing/index.mjs +81 -26
  47. package/dist/lib/browser/testing/index.mjs.map +4 -4
  48. package/dist/lib/browser/ui/index.mjs +45 -0
  49. package/dist/lib/browser/ui/index.mjs.map +7 -0
  50. package/dist/lib/node-esm/capability-CHIMU6LX.mjs +38 -0
  51. package/dist/lib/node-esm/capability-CHIMU6LX.mjs.map +7 -0
  52. package/dist/lib/node-esm/capability-EVZK4REM.mjs +35 -0
  53. package/dist/lib/node-esm/capability-EVZK4REM.mjs.map +7 -0
  54. package/dist/lib/node-esm/chunk-2A4PRBIX.mjs +171 -0
  55. package/dist/lib/node-esm/chunk-2A4PRBIX.mjs.map +7 -0
  56. package/dist/lib/node-esm/chunk-7CPNAEGV.mjs +704 -0
  57. package/dist/lib/node-esm/chunk-7CPNAEGV.mjs.map +7 -0
  58. package/dist/lib/node-esm/chunk-7OWSHPYK.mjs +48 -0
  59. package/dist/lib/node-esm/chunk-7OWSHPYK.mjs.map +7 -0
  60. package/dist/lib/node-esm/chunk-CJCQS2YL.mjs +146 -0
  61. package/dist/lib/node-esm/chunk-CJCQS2YL.mjs.map +7 -0
  62. package/dist/lib/node-esm/chunk-DTCHT2X2.mjs +825 -0
  63. package/dist/lib/node-esm/chunk-DTCHT2X2.mjs.map +7 -0
  64. package/dist/lib/node-esm/chunk-HSLMI22Q.mjs +11 -0
  65. package/dist/lib/node-esm/chunk-HSLMI22Q.mjs.map +7 -0
  66. package/dist/lib/node-esm/chunk-JAZVHID3.mjs +35 -0
  67. package/dist/lib/node-esm/chunk-JAZVHID3.mjs.map +7 -0
  68. package/dist/lib/node-esm/chunk-SB5ODNPX.mjs +77 -0
  69. package/dist/lib/node-esm/chunk-SB5ODNPX.mjs.map +7 -0
  70. package/dist/lib/node-esm/chunk-UEWJDI2L.mjs +10 -0
  71. package/dist/lib/node-esm/chunk-UEWJDI2L.mjs.map +7 -0
  72. package/dist/lib/node-esm/chunk-UFW652GS.mjs +81 -0
  73. package/dist/lib/node-esm/chunk-UFW652GS.mjs.map +7 -0
  74. package/dist/lib/node-esm/chunk-VUIUFIGT.mjs +45 -0
  75. package/dist/lib/node-esm/chunk-VUIUFIGT.mjs.map +7 -0
  76. package/dist/lib/node-esm/chunk-Z4TJPSMP.mjs +2 -0
  77. package/dist/lib/node-esm/chunk-Z4TJPSMP.mjs.map +7 -0
  78. package/dist/lib/node-esm/cli/index.mjs +90 -0
  79. package/dist/lib/node-esm/cli/index.mjs.map +7 -0
  80. package/dist/lib/node-esm/common/activation-events.mjs +24 -0
  81. package/dist/lib/node-esm/common/activation-events.mjs.map +7 -0
  82. package/dist/lib/node-esm/common/capabilities.mjs +46 -0
  83. package/dist/lib/node-esm/common/capabilities.mjs.map +7 -0
  84. package/dist/lib/node-esm/core/activation-event.mjs +21 -0
  85. package/dist/lib/node-esm/core/activation-event.mjs.map +7 -0
  86. package/dist/lib/node-esm/core/capability.mjs +29 -0
  87. package/dist/lib/node-esm/core/capability.mjs.map +7 -0
  88. package/dist/lib/node-esm/core/plugin-manager.mjs +16 -0
  89. package/dist/lib/node-esm/core/plugin-manager.mjs.map +7 -0
  90. package/dist/lib/node-esm/core/plugin.mjs +28 -0
  91. package/dist/lib/node-esm/core/plugin.mjs.map +7 -0
  92. package/dist/lib/node-esm/index.mjs +87 -148
  93. package/dist/lib/node-esm/index.mjs.map +4 -4
  94. package/dist/lib/node-esm/invoker-capability-VF6SP44V.mjs +37 -0
  95. package/dist/lib/node-esm/invoker-capability-VF6SP44V.mjs.map +7 -0
  96. package/dist/lib/node-esm/meta.json +1 -1
  97. package/dist/lib/node-esm/testing/index.mjs +81 -26
  98. package/dist/lib/node-esm/testing/index.mjs.map +4 -4
  99. package/dist/lib/node-esm/ui/index.mjs +46 -0
  100. package/dist/lib/node-esm/ui/index.mjs.map +7 -0
  101. package/dist/types/src/cli/cli.d.ts +39 -0
  102. package/dist/types/src/cli/cli.d.ts.map +1 -0
  103. package/dist/types/src/cli/index.d.ts +2 -0
  104. package/dist/types/src/cli/index.d.ts.map +1 -0
  105. package/dist/types/src/common/activation-events.d.ts +27 -0
  106. package/dist/types/src/common/activation-events.d.ts.map +1 -0
  107. package/dist/types/src/common/capabilities.d.ts +114 -200
  108. package/dist/types/src/common/capabilities.d.ts.map +1 -1
  109. package/dist/types/src/common/index.d.ts +4 -8
  110. package/dist/types/src/common/index.d.ts.map +1 -1
  111. package/dist/types/src/common/operations.d.ts +33 -0
  112. package/dist/types/src/common/operations.d.ts.map +1 -0
  113. package/dist/types/src/common/translations.d.ts +7 -7
  114. package/dist/types/src/common/translations.d.ts.map +1 -1
  115. package/dist/types/src/context.d.ts +5 -0
  116. package/dist/types/src/context.d.ts.map +1 -0
  117. package/dist/types/src/core/{events.d.ts → activation-event.d.ts} +6 -6
  118. package/dist/types/src/core/activation-event.d.ts.map +1 -0
  119. package/dist/types/src/core/capability-manager.d.ts +43 -0
  120. package/dist/types/src/core/capability-manager.d.ts.map +1 -0
  121. package/dist/types/src/core/capability-manager.test.d.ts +2 -0
  122. package/dist/types/src/core/capability-manager.test.d.ts.map +1 -0
  123. package/dist/types/src/core/capability.d.ts +150 -0
  124. package/dist/types/src/core/capability.d.ts.map +1 -0
  125. package/dist/types/src/core/index.d.ts +5 -4
  126. package/dist/types/src/core/index.d.ts.map +1 -1
  127. package/dist/types/src/core/plugin-manager.d.ts +66 -0
  128. package/dist/types/src/core/plugin-manager.d.ts.map +1 -0
  129. package/dist/types/src/core/plugin-manager.test.d.ts +2 -0
  130. package/dist/types/src/core/plugin-manager.test.d.ts.map +1 -0
  131. package/dist/types/src/core/plugin.d.ts +96 -34
  132. package/dist/types/src/core/plugin.d.ts.map +1 -1
  133. package/dist/types/src/index.d.ts +3 -4
  134. package/dist/types/src/index.d.ts.map +1 -1
  135. package/dist/types/src/plugin-operation/OperationPlugin.d.ts +3 -0
  136. package/dist/types/src/plugin-operation/OperationPlugin.d.ts.map +1 -0
  137. package/dist/types/src/plugin-operation/history/capability.d.ts +7 -0
  138. package/dist/types/src/plugin-operation/history/capability.d.ts.map +1 -0
  139. package/dist/types/src/plugin-operation/history/errors.d.ts +5 -0
  140. package/dist/types/src/plugin-operation/history/errors.d.ts.map +1 -0
  141. package/dist/types/src/plugin-operation/history/history-tracker.d.ts +18 -0
  142. package/dist/types/src/plugin-operation/history/history-tracker.d.ts.map +1 -0
  143. package/dist/types/src/plugin-operation/history/history-tracker.test.d.ts +2 -0
  144. package/dist/types/src/plugin-operation/history/history-tracker.test.d.ts.map +1 -0
  145. package/dist/types/src/plugin-operation/history/index.d.ts +6 -0
  146. package/dist/types/src/plugin-operation/history/index.d.ts.map +1 -0
  147. package/dist/types/src/plugin-operation/history/types.d.ts +13 -0
  148. package/dist/types/src/plugin-operation/history/types.d.ts.map +1 -0
  149. package/dist/types/src/plugin-operation/history/undo-mapping.d.ts +101 -0
  150. package/dist/types/src/plugin-operation/history/undo-mapping.d.ts.map +1 -0
  151. package/dist/types/src/plugin-operation/history/undo-registry.d.ts +23 -0
  152. package/dist/types/src/plugin-operation/history/undo-registry.d.ts.map +1 -0
  153. package/dist/types/src/plugin-operation/history/undo-registry.test.d.ts +2 -0
  154. package/dist/types/src/plugin-operation/history/undo-registry.test.d.ts.map +1 -0
  155. package/dist/types/src/plugin-operation/index.d.ts +3 -0
  156. package/dist/types/src/plugin-operation/index.d.ts.map +1 -0
  157. package/dist/types/src/plugin-operation/invoker-capability.d.ts +6 -0
  158. package/dist/types/src/plugin-operation/invoker-capability.d.ts.map +1 -0
  159. package/dist/types/src/plugin-operation/meta.d.ts +3 -0
  160. package/dist/types/src/plugin-operation/meta.d.ts.map +1 -0
  161. package/dist/types/src/plugin-operation/testing.d.ts +109 -0
  162. package/dist/types/src/plugin-operation/testing.d.ts.map +1 -0
  163. package/dist/types/src/plugin-runtime/RuntimePlugin.d.ts +3 -0
  164. package/dist/types/src/plugin-runtime/RuntimePlugin.d.ts.map +1 -0
  165. package/dist/types/src/plugin-runtime/capability.d.ts +6 -0
  166. package/dist/types/src/plugin-runtime/capability.d.ts.map +1 -0
  167. package/dist/types/src/plugin-runtime/index.d.ts +2 -0
  168. package/dist/types/src/plugin-runtime/index.d.ts.map +1 -0
  169. package/dist/types/src/plugin-runtime/meta.d.ts +3 -0
  170. package/dist/types/src/plugin-runtime/meta.d.ts.map +1 -0
  171. package/dist/types/src/testing/index.d.ts +1 -0
  172. package/dist/types/src/testing/index.d.ts.map +1 -1
  173. package/dist/types/src/testing/service.d.ts +8 -0
  174. package/dist/types/src/testing/service.d.ts.map +1 -0
  175. package/dist/types/src/testing/withPluginManager.d.ts +7 -6
  176. package/dist/types/src/testing/withPluginManager.d.ts.map +1 -1
  177. package/dist/types/src/ui/components/App/App.d.ts +8 -0
  178. package/dist/types/src/ui/components/App/App.d.ts.map +1 -0
  179. package/dist/types/src/{react/Surface.stories.d.ts → ui/components/App/App.stories.d.ts} +9 -7
  180. package/dist/types/src/ui/components/App/App.stories.d.ts.map +1 -0
  181. package/dist/types/src/ui/components/App/index.d.ts +2 -0
  182. package/dist/types/src/ui/components/App/index.d.ts.map +1 -0
  183. package/dist/types/src/{playground/playground.stories.d.ts → ui/components/PluginManager/PluginManagerContext.stories.d.ts} +5 -3
  184. package/dist/types/src/ui/components/PluginManager/PluginManagerContext.stories.d.ts.map +1 -0
  185. package/dist/types/src/ui/components/PluginManager/PluginManagerProvider.d.ts +10 -0
  186. package/dist/types/src/ui/components/PluginManager/PluginManagerProvider.d.ts.map +1 -0
  187. package/dist/types/src/ui/components/PluginManager/index.d.ts +2 -0
  188. package/dist/types/src/ui/components/PluginManager/index.d.ts.map +1 -0
  189. package/dist/types/src/ui/components/Surface/SurfaceComponent.d.ts +12 -0
  190. package/dist/types/src/ui/components/Surface/SurfaceComponent.d.ts.map +1 -0
  191. package/dist/types/src/{components/App.stories.d.ts → ui/components/Surface/SurfaceComponent.stories.d.ts} +1 -1
  192. package/dist/types/src/ui/components/Surface/SurfaceComponent.stories.d.ts.map +1 -0
  193. package/dist/types/src/ui/components/Surface/SurfaceInfo.d.ts +11 -0
  194. package/dist/types/src/ui/components/Surface/SurfaceInfo.d.ts.map +1 -0
  195. package/dist/types/src/ui/components/Surface/context.d.ts +5 -0
  196. package/dist/types/src/ui/components/Surface/context.d.ts.map +1 -0
  197. package/dist/types/src/ui/components/Surface/index.d.ts +20 -0
  198. package/dist/types/src/ui/components/Surface/index.d.ts.map +1 -0
  199. package/dist/types/src/ui/components/Surface/types.d.ts +96 -0
  200. package/dist/types/src/ui/components/Surface/types.d.ts.map +1 -0
  201. package/dist/types/src/ui/components/index.d.ts +4 -0
  202. package/dist/types/src/ui/components/index.d.ts.map +1 -0
  203. package/dist/types/src/ui/hooks/index.d.ts +7 -0
  204. package/dist/types/src/ui/hooks/index.d.ts.map +1 -0
  205. package/dist/types/src/{components → ui/hooks}/useApp.d.ts +18 -12
  206. package/dist/types/src/ui/hooks/useApp.d.ts.map +1 -0
  207. package/dist/types/src/ui/hooks/useCapabilities.d.ts +31 -0
  208. package/dist/types/src/ui/hooks/useCapabilities.d.ts.map +1 -0
  209. package/dist/types/src/{components → ui/hooks}/useLoading.d.ts +1 -2
  210. package/dist/types/src/ui/hooks/useLoading.d.ts.map +1 -0
  211. package/dist/types/src/ui/hooks/useOperationResolver.d.ts +19 -0
  212. package/dist/types/src/ui/hooks/useOperationResolver.d.ts.map +1 -0
  213. package/dist/types/src/ui/hooks/useSettingsState.d.ts +10 -0
  214. package/dist/types/src/ui/hooks/useSettingsState.d.ts.map +1 -0
  215. package/dist/types/src/ui/hooks/useSurface.d.ts +3 -0
  216. package/dist/types/src/ui/hooks/useSurface.d.ts.map +1 -0
  217. package/dist/types/src/ui/index.d.ts +3 -0
  218. package/dist/types/src/ui/index.d.ts.map +1 -0
  219. package/dist/types/tsconfig.tsbuildinfo +1 -1
  220. package/moon.yml +10 -6
  221. package/package.json +111 -47
  222. package/src/cli/cli.ts +107 -0
  223. package/src/{components → cli}/index.ts +1 -1
  224. package/src/common/activation-events.ts +44 -0
  225. package/src/common/capabilities.ts +174 -213
  226. package/src/common/index.ts +4 -8
  227. package/src/common/operations.ts +38 -0
  228. package/src/common/translations.ts +17 -9
  229. package/src/context.ts +9 -0
  230. package/src/core/{events.ts → activation-event.ts} +5 -5
  231. package/src/core/capability-manager.test.ts +151 -0
  232. package/src/core/capability-manager.ts +171 -0
  233. package/src/core/capability.ts +236 -0
  234. package/src/core/index.ts +5 -4
  235. package/src/core/plugin-manager.test.ts +845 -0
  236. package/src/core/plugin-manager.ts +715 -0
  237. package/src/core/plugin.ts +227 -39
  238. package/src/index.ts +3 -4
  239. package/src/plugin-operation/OperationPlugin.ts +25 -0
  240. package/src/plugin-operation/history/capability.ts +37 -0
  241. package/src/plugin-operation/history/errors.ts +11 -0
  242. package/src/plugin-operation/history/history-tracker.test.ts +380 -0
  243. package/src/plugin-operation/history/history-tracker.ts +129 -0
  244. package/src/plugin-operation/history/index.ts +9 -0
  245. package/src/plugin-operation/history/types.ts +17 -0
  246. package/src/plugin-operation/history/undo-mapping.ts +135 -0
  247. package/src/plugin-operation/history/undo-registry.test.ts +73 -0
  248. package/src/plugin-operation/history/undo-registry.ts +54 -0
  249. package/src/plugin-operation/index.ts +6 -0
  250. package/src/plugin-operation/invoker-capability.ts +40 -0
  251. package/src/plugin-operation/meta.ts +11 -0
  252. package/src/plugin-operation/testing.ts +174 -0
  253. package/src/plugin-runtime/RuntimePlugin.ts +20 -0
  254. package/src/plugin-runtime/capability.ts +53 -0
  255. package/src/{playground/logger → plugin-runtime}/index.ts +1 -1
  256. package/src/plugin-runtime/meta.ts +11 -0
  257. package/src/testing/index.ts +1 -0
  258. package/src/testing/service.ts +52 -0
  259. package/src/testing/withPluginManager.stories.tsx +7 -7
  260. package/src/testing/withPluginManager.tsx +33 -27
  261. package/src/ui/components/App/App.stories.tsx +92 -0
  262. package/src/{components → ui/components/App}/App.tsx +9 -11
  263. package/src/{playground/debug → ui/components/App}/index.ts +1 -1
  264. package/src/ui/components/PluginManager/PluginManagerContext.stories.tsx +183 -0
  265. package/src/{react → ui/components/PluginManager}/PluginManagerProvider.ts +3 -3
  266. package/src/ui/components/PluginManager/index.ts +5 -0
  267. package/src/ui/components/Surface/SurfaceComponent.stories.tsx +143 -0
  268. package/src/ui/components/Surface/SurfaceComponent.tsx +247 -0
  269. package/src/ui/components/Surface/SurfaceInfo.tsx +107 -0
  270. package/src/ui/components/Surface/context.ts +12 -0
  271. package/src/ui/components/Surface/index.ts +20 -0
  272. package/src/ui/components/Surface/types.ts +117 -0
  273. package/src/ui/components/index.ts +7 -0
  274. package/src/ui/hooks/index.ts +10 -0
  275. package/src/ui/hooks/useApp.tsx +213 -0
  276. package/src/ui/hooks/useCapabilities.ts +67 -0
  277. package/src/{components → ui/hooks}/useLoading.tsx +4 -6
  278. package/src/ui/hooks/useOperationResolver.ts +40 -0
  279. package/src/ui/hooks/useSettingsState.ts +26 -0
  280. package/src/ui/hooks/useSurface.ts +13 -0
  281. package/src/ui/index.ts +6 -0
  282. package/tsconfig.json +12 -21
  283. package/tsconfig.node.json +1 -3
  284. package/typedoc.json +2 -4
  285. package/.swc/plugins/linux_x86_64_19.0.0/fce1bdb8e20a094e4af08bad09cc81497ed0e2e7c51223b07d371063cca18429.wasmer-v7 +0 -0
  286. package/dist/lib/browser/app-graph-builder-77MY7KAY.mjs +0 -138
  287. package/dist/lib/browser/app-graph-builder-77MY7KAY.mjs.map +0 -7
  288. package/dist/lib/browser/chunk-4NFLRSTX.mjs +0 -467
  289. package/dist/lib/browser/chunk-4NFLRSTX.mjs.map +0 -7
  290. package/dist/lib/browser/chunk-OKF2PAWO.mjs +0 -1646
  291. package/dist/lib/browser/chunk-OKF2PAWO.mjs.map +0 -7
  292. package/dist/lib/browser/chunk-SCPE4ZO2.mjs +0 -35
  293. package/dist/lib/browser/chunk-SCPE4ZO2.mjs.map +0 -7
  294. package/dist/lib/browser/intent-dispatcher-Y4LZNOBC.mjs +0 -11
  295. package/dist/lib/browser/intent-resolver-XO5OYWVD.mjs +0 -39
  296. package/dist/lib/browser/intent-resolver-XO5OYWVD.mjs.map +0 -7
  297. package/dist/lib/browser/store-E3YSBPJQ.mjs +0 -30
  298. package/dist/lib/browser/store-E3YSBPJQ.mjs.map +0 -7
  299. package/dist/lib/browser/worker.mjs +0 -77
  300. package/dist/lib/node-esm/app-graph-builder-WK7WBM6I.mjs +0 -139
  301. package/dist/lib/node-esm/app-graph-builder-WK7WBM6I.mjs.map +0 -7
  302. package/dist/lib/node-esm/chunk-7DNZQ6H2.mjs +0 -1648
  303. package/dist/lib/node-esm/chunk-7DNZQ6H2.mjs.map +0 -7
  304. package/dist/lib/node-esm/chunk-AHCD4UYO.mjs +0 -468
  305. package/dist/lib/node-esm/chunk-AHCD4UYO.mjs.map +0 -7
  306. package/dist/lib/node-esm/chunk-ZX63QUGE.mjs +0 -37
  307. package/dist/lib/node-esm/chunk-ZX63QUGE.mjs.map +0 -7
  308. package/dist/lib/node-esm/intent-dispatcher-R4CCPBHO.mjs +0 -12
  309. package/dist/lib/node-esm/intent-resolver-NVYY6BQA.mjs +0 -40
  310. package/dist/lib/node-esm/intent-resolver-NVYY6BQA.mjs.map +0 -7
  311. package/dist/lib/node-esm/store-QA5DGTOS.mjs +0 -31
  312. package/dist/lib/node-esm/store-QA5DGTOS.mjs.map +0 -7
  313. package/dist/lib/node-esm/worker.mjs +0 -78
  314. package/dist/types/src/common/collaboration.d.ts +0 -20
  315. package/dist/types/src/common/collaboration.d.ts.map +0 -1
  316. package/dist/types/src/common/events.d.ts +0 -52
  317. package/dist/types/src/common/events.d.ts.map +0 -1
  318. package/dist/types/src/common/file.d.ts +0 -14
  319. package/dist/types/src/common/file.d.ts.map +0 -1
  320. package/dist/types/src/common/graph.d.ts +0 -21
  321. package/dist/types/src/common/graph.d.ts.map +0 -1
  322. package/dist/types/src/common/layout.d.ts +0 -279
  323. package/dist/types/src/common/layout.d.ts.map +0 -1
  324. package/dist/types/src/common/surface.d.ts +0 -59
  325. package/dist/types/src/common/surface.d.ts.map +0 -1
  326. package/dist/types/src/components/App.d.ts +0 -10
  327. package/dist/types/src/components/App.d.ts.map +0 -1
  328. package/dist/types/src/components/App.stories.d.ts.map +0 -1
  329. package/dist/types/src/components/DefaultFallback.d.ts +0 -8
  330. package/dist/types/src/components/DefaultFallback.d.ts.map +0 -1
  331. package/dist/types/src/components/index.d.ts +0 -2
  332. package/dist/types/src/components/index.d.ts.map +0 -1
  333. package/dist/types/src/components/useApp.d.ts.map +0 -1
  334. package/dist/types/src/components/useLoading.d.ts.map +0 -1
  335. package/dist/types/src/core/capabilities.d.ts +0 -117
  336. package/dist/types/src/core/capabilities.d.ts.map +0 -1
  337. package/dist/types/src/core/capabilities.test.d.ts +0 -2
  338. package/dist/types/src/core/capabilities.test.d.ts.map +0 -1
  339. package/dist/types/src/core/events.d.ts.map +0 -1
  340. package/dist/types/src/core/manager.d.ts +0 -126
  341. package/dist/types/src/core/manager.d.ts.map +0 -1
  342. package/dist/types/src/core/manager.test.d.ts +0 -2
  343. package/dist/types/src/core/manager.test.d.ts.map +0 -1
  344. package/dist/types/src/playground/debug/Debug.d.ts +0 -6
  345. package/dist/types/src/playground/debug/Debug.d.ts.map +0 -1
  346. package/dist/types/src/playground/debug/index.d.ts +0 -2
  347. package/dist/types/src/playground/debug/index.d.ts.map +0 -1
  348. package/dist/types/src/playground/debug/plugin.d.ts +0 -2
  349. package/dist/types/src/playground/debug/plugin.d.ts.map +0 -1
  350. package/dist/types/src/playground/generator/Main.d.ts +0 -6
  351. package/dist/types/src/playground/generator/Main.d.ts.map +0 -1
  352. package/dist/types/src/playground/generator/Toolbar.d.ts +0 -6
  353. package/dist/types/src/playground/generator/Toolbar.d.ts.map +0 -1
  354. package/dist/types/src/playground/generator/generator.d.ts +0 -7
  355. package/dist/types/src/playground/generator/generator.d.ts.map +0 -1
  356. package/dist/types/src/playground/generator/index.d.ts +0 -3
  357. package/dist/types/src/playground/generator/index.d.ts.map +0 -1
  358. package/dist/types/src/playground/generator/plugin.d.ts +0 -2
  359. package/dist/types/src/playground/generator/plugin.d.ts.map +0 -1
  360. package/dist/types/src/playground/layout/Layout.d.ts +0 -8
  361. package/dist/types/src/playground/layout/Layout.d.ts.map +0 -1
  362. package/dist/types/src/playground/layout/index.d.ts +0 -2
  363. package/dist/types/src/playground/layout/index.d.ts.map +0 -1
  364. package/dist/types/src/playground/layout/plugin.d.ts +0 -2
  365. package/dist/types/src/playground/layout/plugin.d.ts.map +0 -1
  366. package/dist/types/src/playground/logger/Toolbar.d.ts +0 -6
  367. package/dist/types/src/playground/logger/Toolbar.d.ts.map +0 -1
  368. package/dist/types/src/playground/logger/index.d.ts +0 -2
  369. package/dist/types/src/playground/logger/index.d.ts.map +0 -1
  370. package/dist/types/src/playground/logger/plugin.d.ts +0 -2
  371. package/dist/types/src/playground/logger/plugin.d.ts.map +0 -1
  372. package/dist/types/src/playground/logger/schema.d.ts +0 -13
  373. package/dist/types/src/playground/logger/schema.d.ts.map +0 -1
  374. package/dist/types/src/playground/playground.stories.d.ts.map +0 -1
  375. package/dist/types/src/plugin-intent/IntentPlugin.d.ts +0 -2
  376. package/dist/types/src/plugin-intent/IntentPlugin.d.ts.map +0 -1
  377. package/dist/types/src/plugin-intent/actions.d.ts +0 -36
  378. package/dist/types/src/plugin-intent/actions.d.ts.map +0 -1
  379. package/dist/types/src/plugin-intent/errors.d.ts +0 -16
  380. package/dist/types/src/plugin-intent/errors.d.ts.map +0 -1
  381. package/dist/types/src/plugin-intent/index.d.ts +0 -6
  382. package/dist/types/src/plugin-intent/index.d.ts.map +0 -1
  383. package/dist/types/src/plugin-intent/intent-dispatcher.d.ts +0 -139
  384. package/dist/types/src/plugin-intent/intent-dispatcher.d.ts.map +0 -1
  385. package/dist/types/src/plugin-intent/intent-dispatcher.test.d.ts +0 -2
  386. package/dist/types/src/plugin-intent/intent-dispatcher.test.d.ts.map +0 -1
  387. package/dist/types/src/plugin-intent/intent.d.ts +0 -63
  388. package/dist/types/src/plugin-intent/intent.d.ts.map +0 -1
  389. package/dist/types/src/plugin-intent/meta.d.ts +0 -3
  390. package/dist/types/src/plugin-intent/meta.d.ts.map +0 -1
  391. package/dist/types/src/plugin-settings/SettingsPlugin.d.ts +0 -2
  392. package/dist/types/src/plugin-settings/SettingsPlugin.d.ts.map +0 -1
  393. package/dist/types/src/plugin-settings/actions.d.ts +0 -25
  394. package/dist/types/src/plugin-settings/actions.d.ts.map +0 -1
  395. package/dist/types/src/plugin-settings/app-graph-builder.d.ts +0 -4
  396. package/dist/types/src/plugin-settings/app-graph-builder.d.ts.map +0 -1
  397. package/dist/types/src/plugin-settings/index.d.ts +0 -3
  398. package/dist/types/src/plugin-settings/index.d.ts.map +0 -1
  399. package/dist/types/src/plugin-settings/intent-resolver.d.ts +0 -4
  400. package/dist/types/src/plugin-settings/intent-resolver.d.ts.map +0 -1
  401. package/dist/types/src/plugin-settings/meta.d.ts +0 -3
  402. package/dist/types/src/plugin-settings/meta.d.ts.map +0 -1
  403. package/dist/types/src/plugin-settings/store.d.ts +0 -5
  404. package/dist/types/src/plugin-settings/store.d.ts.map +0 -1
  405. package/dist/types/src/plugin-settings/translations.d.ts +0 -11
  406. package/dist/types/src/plugin-settings/translations.d.ts.map +0 -1
  407. package/dist/types/src/react/ErrorBoundary.d.ts +0 -30
  408. package/dist/types/src/react/ErrorBoundary.d.ts.map +0 -1
  409. package/dist/types/src/react/IntentContext.d.ts +0 -8
  410. package/dist/types/src/react/IntentContext.d.ts.map +0 -1
  411. package/dist/types/src/react/PluginManagerProvider.d.ts +0 -10
  412. package/dist/types/src/react/PluginManagerProvider.d.ts.map +0 -1
  413. package/dist/types/src/react/Surface.d.ts +0 -12
  414. package/dist/types/src/react/Surface.d.ts.map +0 -1
  415. package/dist/types/src/react/Surface.stories.d.ts.map +0 -1
  416. package/dist/types/src/react/common.d.ts +0 -13
  417. package/dist/types/src/react/common.d.ts.map +0 -1
  418. package/dist/types/src/react/index.d.ts +0 -7
  419. package/dist/types/src/react/index.d.ts.map +0 -1
  420. package/dist/types/src/react/useCapabilities.d.ts +0 -13
  421. package/dist/types/src/react/useCapabilities.d.ts.map +0 -1
  422. package/dist/types/src/react/useIntentResolver.d.ts +0 -3
  423. package/dist/types/src/react/useIntentResolver.d.ts.map +0 -1
  424. package/dist/types/src/worker.d.ts +0 -4
  425. package/dist/types/src/worker.d.ts.map +0 -1
  426. package/src/common/collaboration.ts +0 -18
  427. package/src/common/events.ts +0 -79
  428. package/src/common/file.ts +0 -22
  429. package/src/common/graph.ts +0 -30
  430. package/src/common/layout.ts +0 -277
  431. package/src/common/surface.ts +0 -83
  432. package/src/components/App.stories.tsx +0 -33
  433. package/src/components/DefaultFallback.tsx +0 -26
  434. package/src/components/useApp.tsx +0 -164
  435. package/src/core/capabilities.test.ts +0 -136
  436. package/src/core/capabilities.ts +0 -259
  437. package/src/core/manager.test.ts +0 -516
  438. package/src/core/manager.ts +0 -604
  439. package/src/playground/debug/Debug.tsx +0 -39
  440. package/src/playground/debug/plugin.ts +0 -16
  441. package/src/playground/generator/Main.tsx +0 -71
  442. package/src/playground/generator/Toolbar.tsx +0 -47
  443. package/src/playground/generator/generator.ts +0 -48
  444. package/src/playground/generator/index.ts +0 -6
  445. package/src/playground/generator/plugin.ts +0 -22
  446. package/src/playground/layout/Layout.tsx +0 -33
  447. package/src/playground/layout/index.ts +0 -5
  448. package/src/playground/layout/plugin.ts +0 -18
  449. package/src/playground/logger/Toolbar.tsx +0 -30
  450. package/src/playground/logger/plugin.ts +0 -41
  451. package/src/playground/logger/schema.ts +0 -12
  452. package/src/playground/playground.stories.tsx +0 -46
  453. package/src/plugin-intent/IntentPlugin.ts +0 -20
  454. package/src/plugin-intent/actions.ts +0 -31
  455. package/src/plugin-intent/errors.ts +0 -40
  456. package/src/plugin-intent/index.ts +0 -9
  457. package/src/plugin-intent/intent-dispatcher.test.ts +0 -286
  458. package/src/plugin-intent/intent-dispatcher.ts +0 -337
  459. package/src/plugin-intent/intent.ts +0 -154
  460. package/src/plugin-intent/meta.ts +0 -10
  461. package/src/plugin-settings/SettingsPlugin.ts +0 -34
  462. package/src/plugin-settings/actions.ts +0 -25
  463. package/src/plugin-settings/app-graph-builder.ts +0 -160
  464. package/src/plugin-settings/index.ts +0 -6
  465. package/src/plugin-settings/intent-resolver.ts +0 -35
  466. package/src/plugin-settings/meta.ts +0 -10
  467. package/src/plugin-settings/store.ts +0 -33
  468. package/src/plugin-settings/translations.ts +0 -19
  469. package/src/react/ErrorBoundary.tsx +0 -56
  470. package/src/react/IntentContext.tsx +0 -35
  471. package/src/react/Surface.stories.tsx +0 -101
  472. package/src/react/Surface.tsx +0 -86
  473. package/src/react/common.ts +0 -13
  474. package/src/react/index.ts +0 -10
  475. package/src/react/useCapabilities.ts +0 -31
  476. package/src/react/useIntentResolver.ts +0 -22
  477. package/src/worker.ts +0 -11
  478. /package/dist/lib/browser/{intent-dispatcher-Y4LZNOBC.mjs.map → chunk-J5LGTIGS.mjs.map} +0 -0
  479. /package/dist/lib/browser/{worker.mjs.map → chunk-YAFEA4GV.mjs.map} +0 -0
  480. /package/dist/lib/{node-esm/intent-dispatcher-R4CCPBHO.mjs.map → browser/common/activation-events.mjs.map} +0 -0
  481. /package/dist/lib/{node-esm/worker.mjs.map → browser/common/capabilities.mjs.map} +0 -0
@@ -0,0 +1,845 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { afterEach, assert, describe, it } from '@effect/vitest';
6
+ import { type Atom, Registry } from '@effect-atom/atom-react';
7
+ import * as Duration from 'effect/Duration';
8
+ import * as Effect from 'effect/Effect';
9
+ import * as Fiber from 'effect/Fiber';
10
+ import * as Match from 'effect/Match';
11
+ import * as PubSub from 'effect/PubSub';
12
+ import * as Queue from 'effect/Queue';
13
+ import * as TestClock from 'effect/TestClock';
14
+
15
+ import { invariant } from '@dxos/invariant';
16
+ import { type LogConfig, type LogEntry, LogLevel, log } from '@dxos/log';
17
+
18
+ import { ActivationEvents } from '../common';
19
+
20
+ import * as ActivationEvent from './activation-event';
21
+ import * as Capability from './capability';
22
+ import type * as CapabilityManager from './capability-manager';
23
+ import * as Plugin from './plugin';
24
+ import * as PluginManager from './plugin-manager';
25
+
26
+ const String = Capability.make<{ string: string }>('dxos.org/test/string');
27
+ const Number = Capability.make<{ number: number }>('dxos.org/test/number');
28
+ const Total = Capability.make<{ total: number }>('dxos.org/test/total');
29
+
30
+ const CountEvent = ActivationEvent.make('dxos.org/test/count');
31
+ const FailEvent = ActivationEvent.make('dxos.org/test/fail');
32
+
33
+ const testMeta = { id: 'dxos.org/plugin/test', name: 'Test' };
34
+
35
+ // TODO(wittjosiah): Factor out?
36
+ const atomCounter = (registry: Registry.Registry, atom: Atom.Atom<any>) => {
37
+ let count = 0;
38
+ let initial = true;
39
+ const dispose = registry.subscribe(
40
+ atom,
41
+ () => {
42
+ if (initial) {
43
+ initial = false;
44
+ return;
45
+ }
46
+ count++;
47
+ },
48
+ { immediate: true },
49
+ );
50
+ return {
51
+ get count() {
52
+ return count;
53
+ },
54
+ [Symbol.dispose]: dispose,
55
+ };
56
+ };
57
+
58
+ describe('PluginManager', () => {
59
+ let plugins: Plugin.Plugin[] = [];
60
+ const pluginLoader = Effect.fn(function* (id: string) {
61
+ const plugin = plugins.find((plugin) => plugin.meta.id === id);
62
+ invariant(plugin, `Plugin not found: ${id}`);
63
+ return plugin;
64
+ });
65
+
66
+ afterEach(() => {
67
+ plugins = [];
68
+ });
69
+
70
+ it.effect('should be able to add and remove plugins', () =>
71
+ Effect.gen(function* () {
72
+ const Test = Plugin.make(Plugin.define(testMeta));
73
+ const testPlugin = Test();
74
+ plugins = [testPlugin];
75
+
76
+ const manager = PluginManager.make({ pluginLoader });
77
+ const added = yield* manager.add(testMeta.id);
78
+ assert.isTrue(added);
79
+ assert.deepStrictEqual(manager.getPlugins(), [testPlugin]);
80
+ const removed = manager.remove(testMeta.id);
81
+ assert.isTrue(removed);
82
+ assert.deepStrictEqual(manager.getPlugins(), []);
83
+ }),
84
+ );
85
+
86
+ it.effect('should support factory pattern with options', () =>
87
+ Effect.gen(function* () {
88
+ type TestPluginOptions = { count: number };
89
+ const TestPluginFactory = Plugin.define<TestPluginOptions>(testMeta).pipe(
90
+ Plugin.addModule((options: TestPluginOptions) => ({
91
+ id: 'Hello',
92
+ activatesOn: ActivationEvents.Startup,
93
+ activate: () => Effect.succeed(Capability.contributes(String, { string: `hello-${options.count}` })),
94
+ })),
95
+ Plugin.addModule({
96
+ id: 'World',
97
+ activatesOn: ActivationEvents.Startup,
98
+ activate: () => Effect.succeed(Capability.contributes(String, { string: 'world' })),
99
+ }),
100
+ Plugin.make,
101
+ );
102
+
103
+ const plugin = TestPluginFactory({ count: 5 });
104
+ plugins = [plugin];
105
+
106
+ const manager = PluginManager.make({ plugins: [plugin], core: [], pluginLoader });
107
+ yield* manager.enable(testMeta.id);
108
+ yield* manager.activate(ActivationEvents.Startup);
109
+ const strings = manager.capabilities.getAll(String);
110
+ assert.strictEqual(strings.length, 2);
111
+ assert.strictEqual(strings[0].string, 'hello-5');
112
+ assert.strictEqual(strings[1].string, 'world');
113
+ }),
114
+ );
115
+
116
+ it.effect('should be able to enable and disable plugins', () =>
117
+ Effect.gen(function* () {
118
+ const Test = Plugin.define(testMeta).pipe(
119
+ Plugin.addModule({
120
+ id: 'Hello',
121
+ activatesOn: ActivationEvents.Startup,
122
+ activate: () => Effect.succeed(Capability.contributes(String, { string: 'hello' })),
123
+ }),
124
+ Plugin.make,
125
+ );
126
+
127
+ const testPlugin = Test();
128
+ const manager = PluginManager.make({ plugins: [testPlugin], core: [], pluginLoader });
129
+ yield* manager.enable(testMeta.id);
130
+ assert.deepStrictEqual(manager.getEnabled(), [Test.meta.id]);
131
+ assert.deepStrictEqual(manager.getModules(), [testPlugin.modules[0]]);
132
+ yield* manager.disable(testMeta.id);
133
+ assert.deepStrictEqual(manager.getEnabled(), []);
134
+ assert.deepStrictEqual(manager.getModules(), []);
135
+ }),
136
+ );
137
+
138
+ it.effect('should be able to activate plugins', () =>
139
+ Effect.gen(function* () {
140
+ const Test = Plugin.define(testMeta).pipe(
141
+ Plugin.addModule({
142
+ id: 'Hello',
143
+ activatesOn: ActivationEvents.Startup,
144
+ activate: () => Effect.succeed(Capability.contributes(String, { string: 'hello' })),
145
+ }),
146
+ Plugin.make,
147
+ );
148
+
149
+ const testPlugin = Test();
150
+ const manager = PluginManager.make({ plugins: [testPlugin], pluginLoader });
151
+ yield* manager.enable(Test.meta.id);
152
+ assert.deepStrictEqual(manager.getPlugins(), [testPlugin]);
153
+ assert.deepStrictEqual(manager.getEnabled(), [Test.meta.id]);
154
+ assert.deepStrictEqual(manager.getModules(), [testPlugin.modules[0]]);
155
+ assert.deepStrictEqual(manager.getActive(), []);
156
+ assert.deepStrictEqual(manager.getEventsFired(), []);
157
+ yield* manager.activate(ActivationEvents.Startup);
158
+ assert.deepStrictEqual(manager.getActive(), [testPlugin.modules[0].id]);
159
+ assert.deepStrictEqual(manager.getEventsFired(), [ActivationEvents.Startup.id]);
160
+ }),
161
+ );
162
+
163
+ it.effect('should handle activate returning void', () =>
164
+ Effect.gen(function* () {
165
+ const Test = Plugin.define(testMeta).pipe(
166
+ Plugin.addModule({
167
+ id: 'NoCapabilities',
168
+ activatesOn: ActivationEvents.Startup,
169
+ activate: Effect.fnUntraced(function* () {}),
170
+ }),
171
+ Plugin.make,
172
+ );
173
+
174
+ const testPlugin = Test();
175
+ const manager = PluginManager.make({ plugins: [testPlugin], pluginLoader });
176
+ yield* manager.enable(Test.meta.id);
177
+
178
+ const result = yield* manager.activate(ActivationEvents.Startup);
179
+ assert.isTrue(result);
180
+ assert.deepStrictEqual(manager.getActive(), [testPlugin.modules[0].id]);
181
+ assert.strictEqual(manager.capabilities.getAll(String).length, 0);
182
+ }),
183
+ );
184
+
185
+ it.effect('should propagate errors thrown by module activate callbacks', () =>
186
+ Effect.gen(function* () {
187
+ plugins = [
188
+ Plugin.define(testMeta).pipe(
189
+ Plugin.addModule({
190
+ activatesOn: FailEvent,
191
+ id: 'Fail',
192
+ activate: () => Effect.fail(new Error('test')),
193
+ }),
194
+ Plugin.make,
195
+ )(),
196
+ ];
197
+
198
+ const manager = PluginManager.make({ pluginLoader });
199
+ yield* manager.add(testMeta.id);
200
+ const error = yield* Effect.flip(manager.activate(FailEvent));
201
+ assert.strictEqual(error.message, 'test');
202
+ }),
203
+ );
204
+
205
+ it.effect('should catch and log defects (synchronous throws) in module activation', () =>
206
+ Effect.gen(function* () {
207
+ const DefectEvent = ActivationEvent.make('dxos.org/test/defect');
208
+ const capturedErrors: LogEntry[] = [];
209
+ const removeProcessor = log.addProcessor((_config: LogConfig, entry: LogEntry) => {
210
+ if (entry.level === LogLevel.ERROR) {
211
+ capturedErrors.push(entry);
212
+ }
213
+ });
214
+
215
+ plugins = [
216
+ Plugin.define(testMeta).pipe(
217
+ Plugin.addModule({
218
+ activatesOn: DefectEvent,
219
+ id: 'DefectInEffectSync',
220
+ activate: () =>
221
+ Effect.sync(() => {
222
+ // This is a defect - a synchronous throw inside Effect.sync.
223
+ throw new Error('defect in Effect.sync');
224
+ }),
225
+ }),
226
+ Plugin.make,
227
+ )(),
228
+ ];
229
+
230
+ const manager = PluginManager.make({ pluginLoader });
231
+ yield* manager.add(testMeta.id);
232
+ const error = yield* Effect.flip(manager.activate(DefectEvent));
233
+
234
+ // Verify the error was caught and propagated.
235
+ assert.strictEqual(error.message, 'defect in Effect.sync');
236
+
237
+ // Verify the error was logged with isDefect: true.
238
+ const defectLog = capturedErrors.find(
239
+ (entry) =>
240
+ entry.message?.includes('module failed to activate') &&
241
+ entry.context?.module === 'dxos.org/plugin/test/module/DefectInEffectSync',
242
+ );
243
+ assert.isNotNull(defectLog, 'Expected error log for defect');
244
+ assert.strictEqual(defectLog?.context?.isDefect, true, 'Expected isDefect to be true for synchronous throw');
245
+
246
+ removeProcessor();
247
+ }),
248
+ );
249
+
250
+ it.effect('should catch and log defects when activate throws before returning Effect', () =>
251
+ Effect.gen(function* () {
252
+ const DefectEvent = ActivationEvent.make('dxos.org/test/defect-immediate');
253
+ const capturedErrors: LogEntry[] = [];
254
+ const removeProcessor = log.addProcessor((_config: LogConfig, entry: LogEntry) => {
255
+ if (entry.level === LogLevel.ERROR) {
256
+ capturedErrors.push(entry);
257
+ }
258
+ });
259
+
260
+ plugins = [
261
+ Plugin.define(testMeta).pipe(
262
+ Plugin.addModule({
263
+ activatesOn: DefectEvent,
264
+ id: 'DefectImmediate',
265
+ activate: () => {
266
+ // This throws immediately before even returning an Effect.
267
+ // This is the most severe type of defect.
268
+ throw new Error('immediate throw before Effect');
269
+ return Effect.succeed(undefined);
270
+ },
271
+ }),
272
+ Plugin.make,
273
+ )(),
274
+ ];
275
+
276
+ const manager = PluginManager.make({ pluginLoader });
277
+ yield* manager.add(testMeta.id);
278
+ const error = yield* Effect.flip(manager.activate(DefectEvent));
279
+
280
+ // Verify the error was caught and propagated.
281
+ assert.strictEqual(error.message, 'immediate throw before Effect');
282
+
283
+ // Verify the error was logged with isDefect: true.
284
+ const defectLog = capturedErrors.find(
285
+ (entry) =>
286
+ entry.message?.includes('module failed to activate') &&
287
+ entry.context?.module === 'dxos.org/plugin/test/module/DefectImmediate',
288
+ );
289
+ assert.isNotNull(defectLog, 'Expected error log for immediate defect');
290
+ assert.strictEqual(
291
+ defectLog?.context?.isDefect,
292
+ true,
293
+ 'Expected isDefect to be true for immediate throw before Effect',
294
+ );
295
+
296
+ removeProcessor();
297
+ }),
298
+ );
299
+
300
+ it.effect('should fire activation events', () =>
301
+ Effect.gen(function* () {
302
+ plugins = [
303
+ Plugin.define(testMeta).pipe(
304
+ Plugin.addModule({
305
+ id: 'Hello',
306
+ activatesOn: ActivationEvents.Startup,
307
+ activate: () => Effect.succeed(Capability.contributes(String, { string: 'hello' })),
308
+ }),
309
+ Plugin.addModule({
310
+ activatesOn: FailEvent,
311
+ id: 'Fail',
312
+ activate: () => Effect.fail(new Error('test')),
313
+ }),
314
+ Plugin.make,
315
+ )(),
316
+ ];
317
+
318
+ const manager = PluginManager.make({ pluginLoader });
319
+ const activating = yield* Effect.makeLatch(false);
320
+ const activated = yield* Effect.makeLatch(false);
321
+ const error = yield* Effect.makeLatch(false);
322
+
323
+ const activationFiber = PubSub.subscribe(manager.activation).pipe(
324
+ Effect.flatMap((queue) =>
325
+ Queue.take(queue).pipe(
326
+ Effect.flatMap(({ state }) =>
327
+ Match.value(state).pipe(
328
+ Match.when('activating', () => activating.open),
329
+ Match.when('activated', () => activated.open),
330
+ Match.when('error', () => error.open),
331
+ Match.orElse(() => Effect.succeed(undefined)),
332
+ ),
333
+ ),
334
+ Effect.forever,
335
+ ),
336
+ ),
337
+ Effect.scoped,
338
+ Effect.runFork,
339
+ );
340
+
341
+ yield* manager.add(testMeta.id);
342
+ yield* manager.activate(ActivationEvents.Startup);
343
+ yield* activating.await;
344
+ yield* activated.await;
345
+
346
+ const activating2 = yield* Effect.makeLatch(false);
347
+ const activationFiber2 = PubSub.subscribe(manager.activation).pipe(
348
+ Effect.flatMap((queue) =>
349
+ Queue.take(queue).pipe(
350
+ Effect.flatMap(({ state }) =>
351
+ Match.value(state).pipe(
352
+ Match.when('activating', () => activating2.open),
353
+ Match.orElse(() => Effect.succeed(undefined)),
354
+ ),
355
+ ),
356
+ Effect.forever,
357
+ ),
358
+ ),
359
+ Effect.scoped,
360
+ Effect.runFork,
361
+ );
362
+
363
+ yield* manager.activate(FailEvent).pipe(Effect.catchAll(() => Effect.succeed(false)));
364
+ yield* activating2.await;
365
+ yield* error.await;
366
+ yield* Fiber.interrupt(activationFiber);
367
+ yield* Fiber.interrupt(activationFiber2);
368
+ }),
369
+ );
370
+
371
+ it.effect('should be able to reset an activation event', () =>
372
+ Effect.gen(function* () {
373
+ let count = 0;
374
+ const Test = Plugin.define(testMeta).pipe(
375
+ Plugin.addModule({
376
+ id: 'Hello',
377
+ activatesOn: ActivationEvents.Startup,
378
+ activate: () => {
379
+ count++;
380
+ return Effect.succeed(Capability.contributes(String, { string: 'hello' }));
381
+ },
382
+ }),
383
+ Plugin.make,
384
+ );
385
+ const testPlugin = Test();
386
+ plugins = [testPlugin];
387
+
388
+ const manager = PluginManager.make({ pluginLoader });
389
+
390
+ {
391
+ yield* manager.add(testMeta.id);
392
+ const result = yield* manager.activate(ActivationEvents.Startup);
393
+ assert.isTrue(result);
394
+ assert.deepStrictEqual(manager.getActive(), [testPlugin.modules[0].id]);
395
+ assert.strictEqual(count, 1);
396
+ }
397
+
398
+ {
399
+ const result = yield* manager.activate(ActivationEvents.Startup);
400
+ assert.isFalse(result);
401
+ }
402
+
403
+ {
404
+ const result = yield* manager.reset(ActivationEvents.Startup);
405
+ assert.isTrue(result);
406
+ assert.strictEqual(count, 2);
407
+ }
408
+ }),
409
+ );
410
+
411
+ it.effect('should not fire an unknown event', () =>
412
+ Effect.gen(function* () {
413
+ const manager = PluginManager.make({ pluginLoader });
414
+ const UnknownEvent = ActivationEvent.make('unknown');
415
+ const result = yield* manager.activate(UnknownEvent);
416
+ assert.isFalse(result);
417
+ }),
418
+ );
419
+
420
+ it.effect('should be able to fire custom activation events', () =>
421
+ Effect.gen(function* () {
422
+ const Plugin1 = Plugin.define({ id: 'dxos.org/test/plugin-1', name: 'Plugin 1' }).pipe(
423
+ Plugin.addModule({
424
+ activatesOn: CountEvent,
425
+ id: 'Plugin1',
426
+ activate: () => Effect.succeed([Capability.contributes(Number, { number: 1 })]),
427
+ }),
428
+ Plugin.make,
429
+ );
430
+ const Plugin2 = Plugin.define({ id: 'dxos.org/test/plugin-2', name: 'Plugin 2' }).pipe(
431
+ Plugin.addModule({
432
+ activatesOn: CountEvent,
433
+ id: 'Plugin2',
434
+ activate: () => Effect.succeed([Capability.contributes(Number, { number: 2 })]),
435
+ }),
436
+ Plugin.make,
437
+ );
438
+ const Plugin3 = Plugin.define({ id: 'dxos.org/test/plugin-3', name: 'Plugin 3' }).pipe(
439
+ Plugin.addModule({
440
+ activatesOn: CountEvent,
441
+ id: 'Plugin3',
442
+ activate: () => Effect.succeed([Capability.contributes(Number, { number: 3 })]),
443
+ }),
444
+ Plugin.make,
445
+ );
446
+ const plugin1 = Plugin1();
447
+ const plugin2 = Plugin2();
448
+ const plugin3 = Plugin3();
449
+ plugins = [plugin1, plugin2, plugin3];
450
+
451
+ const manager = PluginManager.make({ pluginLoader });
452
+ assert.deepStrictEqual(manager.getActive(), []);
453
+ assert.strictEqual(manager.capabilities.getAll(Number).length, 0);
454
+
455
+ yield* manager.add(Plugin1.meta.id);
456
+ yield* manager.activate(CountEvent);
457
+ assert.deepStrictEqual(manager.getActive(), [plugin1.modules[0].id]);
458
+ assert.strictEqual(manager.capabilities.getAll(Number).length, 1);
459
+
460
+ yield* manager.add(Plugin2.meta.id);
461
+ yield* manager.activate(CountEvent);
462
+ assert.deepStrictEqual(manager.getActive(), [plugin1.modules[0].id, plugin2.modules[0].id]);
463
+ assert.strictEqual(manager.capabilities.getAll(Number).length, 2);
464
+
465
+ yield* manager.add(Plugin3.meta.id);
466
+ yield* manager.activate(CountEvent);
467
+ assert.deepStrictEqual(manager.getActive(), [
468
+ plugin1.modules[0].id,
469
+ plugin2.modules[0].id,
470
+ plugin3.modules[0].id,
471
+ ]);
472
+ assert.strictEqual(manager.capabilities.getAll(Number).length, 3);
473
+ }),
474
+ );
475
+
476
+ it.effect('should only activate modules after all activatation events have been fired', () =>
477
+ Effect.gen(function* () {
478
+ const Test = Plugin.define(testMeta).pipe(
479
+ Plugin.addModule({
480
+ activatesOn: ActivationEvent.allOf(ActivationEvents.Startup, CountEvent),
481
+ id: 'Hello',
482
+ activate: () => {
483
+ return Effect.succeed(Capability.contributes(String, { string: 'hello' }));
484
+ },
485
+ }),
486
+ Plugin.make,
487
+ );
488
+ const testPlugin = Test();
489
+ plugins = [testPlugin];
490
+
491
+ const manager = PluginManager.make({ pluginLoader });
492
+ assert.deepStrictEqual(manager.getActive(), []);
493
+ assert.strictEqual(manager.capabilities.getAll(String).length, 0);
494
+
495
+ yield* manager.add(testMeta.id);
496
+ yield* manager.activate(ActivationEvents.Startup);
497
+ assert.deepStrictEqual(manager.getActive(), []);
498
+ assert.strictEqual(manager.capabilities.getAll(String).length, 0);
499
+
500
+ yield* manager.activate(CountEvent);
501
+ assert.deepStrictEqual(manager.getActive(), [testPlugin.modules[0].id]);
502
+ assert.strictEqual(manager.capabilities.getAll(String).length, 1);
503
+ }),
504
+ );
505
+
506
+ it.effect('should only activate modules once when multiple activatation events have been fired', () =>
507
+ Effect.gen(function* () {
508
+ let count = 0;
509
+ const Test = Plugin.define(testMeta).pipe(
510
+ Plugin.addModule({
511
+ id: 'Hello',
512
+ activatesOn: ActivationEvent.oneOf(ActivationEvents.Startup, CountEvent),
513
+ activate: () => {
514
+ count++;
515
+ return Effect.succeed(Capability.contributes(String, { string: 'hello' }));
516
+ },
517
+ }),
518
+ Plugin.make,
519
+ );
520
+ const testPlugin = Test();
521
+ plugins = [testPlugin];
522
+
523
+ const manager = PluginManager.make({ pluginLoader });
524
+ assert.deepStrictEqual(manager.getActive(), []);
525
+ assert.strictEqual(manager.capabilities.getAll(String).length, 0);
526
+ assert.strictEqual(count, 0);
527
+
528
+ yield* manager.add(testMeta.id);
529
+ yield* manager.activate(CountEvent);
530
+ assert.deepStrictEqual(manager.getActive(), [testPlugin.modules[0].id]);
531
+ assert.strictEqual(manager.capabilities.getAll(String).length, 1);
532
+ assert.strictEqual(count, 1);
533
+
534
+ yield* manager.activate(ActivationEvents.Startup);
535
+ assert.deepStrictEqual(manager.getActive(), [testPlugin.modules[0].id]);
536
+ assert.strictEqual(manager.capabilities.getAll(String).length, 1);
537
+ assert.strictEqual(count, 1);
538
+ }),
539
+ );
540
+
541
+ it.effect('should be able to disable and re-enable an active plugin', () =>
542
+ Effect.gen(function* () {
543
+ const state = { total: 0 };
544
+ const computeTotal = (capabilityManager: CapabilityManager.CapabilityManager) => {
545
+ const numbers = capabilityManager.getAll(Number);
546
+ state.total = numbers.reduce((acc: number, n: { number: number }) => acc + n.number, 0);
547
+ };
548
+
549
+ const Count = Plugin.define({ id: 'dxos.org/test/count', name: 'Count' }).pipe(
550
+ Plugin.addModule({
551
+ id: 'Count',
552
+ activatesOn: ActivationEvents.Startup,
553
+ activatesBefore: [CountEvent],
554
+ activate: Effect.fnUntraced(function* () {
555
+ const capabilityManager = yield* Capability.Service;
556
+ computeTotal(capabilityManager);
557
+ return Capability.contributes(Total, state);
558
+ }),
559
+ }),
560
+ Plugin.make,
561
+ );
562
+
563
+ const Test = Plugin.define(testMeta).pipe(
564
+ Plugin.addModule({
565
+ activatesOn: CountEvent,
566
+ id: 'Test1',
567
+ activate: () => Effect.succeed(Capability.contributes(Number, { number: 1 })),
568
+ }),
569
+ Plugin.addModule({
570
+ id: 'Test2',
571
+ activatesOn: CountEvent,
572
+ activate: () => Effect.succeed(Capability.contributes(Number, { number: 2 })),
573
+ }),
574
+ Plugin.addModule({
575
+ id: 'Test3',
576
+ activatesOn: CountEvent,
577
+ activate: () => Effect.succeed(Capability.contributes(Number, { number: 3 })),
578
+ }),
579
+ Plugin.make,
580
+ );
581
+ const countPlugin = Count();
582
+ const testPlugin = Test();
583
+ plugins = [countPlugin, testPlugin];
584
+
585
+ const manager = PluginManager.make({ pluginLoader });
586
+ {
587
+ yield* manager.add(Test.meta.id);
588
+ yield* manager.add(Count.meta.id);
589
+ yield* manager.activate(ActivationEvents.Startup);
590
+ assert.deepStrictEqual(manager.getActive(), [
591
+ ...testPlugin.modules.map((m) => m.id),
592
+ countPlugin.modules[0].id,
593
+ ]);
594
+ assert.deepStrictEqual(manager.getPendingReset(), []);
595
+
596
+ const totals = manager.capabilities.getAll(Total);
597
+ assert.strictEqual(totals.length, 1);
598
+ assert.strictEqual(totals[0].total, 6);
599
+ }
600
+
601
+ {
602
+ yield* manager.disable(Test.meta.id);
603
+ assert.deepStrictEqual(manager.getActive(), [countPlugin.modules[0].id]);
604
+ assert.deepStrictEqual(manager.getPendingReset(), []);
605
+
606
+ const totals = manager.capabilities.getAll(Total);
607
+ assert.strictEqual(totals.length, 1);
608
+ // Total doesn't change because it is not reactive.
609
+ assert.strictEqual(totals[0].total, 6);
610
+ }
611
+
612
+ {
613
+ yield* manager.enable(Test.meta.id);
614
+ assert.deepStrictEqual(manager.getActive(), [
615
+ countPlugin.modules[0].id,
616
+ ...testPlugin.modules.map((m) => m.id),
617
+ ]);
618
+ assert.deepStrictEqual(manager.getPendingReset(), []);
619
+
620
+ const totals = manager.capabilities.getAll(Total);
621
+ assert.strictEqual(totals.length, 1);
622
+ assert.strictEqual(totals[0].total, 6);
623
+ }
624
+ }),
625
+ );
626
+
627
+ it.effect('should be reactive', () =>
628
+ Effect.gen(function* () {
629
+ const Plugin1 = Plugin.define({ id: 'dxos.org/test/plugin-1', name: 'Plugin 1' }).pipe(
630
+ Plugin.addModule({
631
+ activatesOn: CountEvent,
632
+ id: 'Plugin1',
633
+ activate: () => Effect.succeed([Capability.contributes(Number, { number: 1 })]),
634
+ }),
635
+ Plugin.make,
636
+ );
637
+ const Plugin2 = Plugin.define({ id: 'dxos.org/test/plugin-2', name: 'Plugin 2' }).pipe(
638
+ Plugin.addModule({
639
+ activatesOn: CountEvent,
640
+ id: 'Plugin2',
641
+ activate: () => Effect.succeed([Capability.contributes(Number, { number: 2 })]),
642
+ }),
643
+ Plugin.make,
644
+ );
645
+ const Plugin3 = Plugin.define({ id: 'dxos.org/test/plugin-3', name: 'Plugin 3' }).pipe(
646
+ Plugin.addModule({
647
+ activatesOn: CountEvent,
648
+ id: 'Plugin3',
649
+ activate: () => Effect.succeed([Capability.contributes(Number, { number: 3 })]),
650
+ }),
651
+ Plugin.make,
652
+ );
653
+ plugins = [Plugin1(), Plugin2(), Plugin3()];
654
+
655
+ const registry = Registry.make();
656
+ const manager = PluginManager.make({ pluginLoader, registry });
657
+ using pluginUpdates = atomCounter(registry, manager.plugins);
658
+ using enabledUpdates = atomCounter(registry, manager.enabled);
659
+ using modulesUpdates = atomCounter(registry, manager.modules);
660
+ using activeUpdates = atomCounter(registry, manager.active);
661
+ using eventsFiredUpdates = atomCounter(registry, manager.eventsFired);
662
+ using pendingResetUpdates = atomCounter(registry, manager.pendingReset);
663
+ assert.strictEqual(pluginUpdates.count, 0);
664
+ assert.strictEqual(enabledUpdates.count, 0);
665
+ assert.strictEqual(modulesUpdates.count, 0);
666
+ assert.strictEqual(activeUpdates.count, 0);
667
+ assert.strictEqual(eventsFiredUpdates.count, 0);
668
+ assert.strictEqual(pendingResetUpdates.count, 0);
669
+
670
+ yield* manager.add(Plugin1.meta.id);
671
+ assert.strictEqual(pluginUpdates.count, 1);
672
+ assert.strictEqual(enabledUpdates.count, 1);
673
+ assert.strictEqual(modulesUpdates.count, 1);
674
+ assert.strictEqual(activeUpdates.count, 0);
675
+ assert.strictEqual(eventsFiredUpdates.count, 0);
676
+ assert.strictEqual(pendingResetUpdates.count, 0);
677
+
678
+ yield* manager.activate(CountEvent);
679
+ assert.strictEqual(pluginUpdates.count, 1);
680
+ assert.strictEqual(enabledUpdates.count, 1);
681
+ assert.strictEqual(modulesUpdates.count, 1);
682
+ assert.strictEqual(activeUpdates.count, 1);
683
+ assert.strictEqual(eventsFiredUpdates.count, 1);
684
+ assert.strictEqual(pendingResetUpdates.count, 0);
685
+
686
+ yield* manager.add(Plugin2.meta.id);
687
+ assert.strictEqual(pluginUpdates.count, 2);
688
+ assert.strictEqual(enabledUpdates.count, 2);
689
+ assert.strictEqual(modulesUpdates.count, 2);
690
+ assert.strictEqual(activeUpdates.count, 2);
691
+ assert.strictEqual(eventsFiredUpdates.count, 1);
692
+ assert.strictEqual(pendingResetUpdates.count, 2);
693
+
694
+ yield* manager.activate(CountEvent);
695
+ assert.strictEqual(pluginUpdates.count, 2);
696
+ assert.strictEqual(enabledUpdates.count, 2);
697
+ assert.strictEqual(modulesUpdates.count, 2);
698
+ assert.strictEqual(activeUpdates.count, 2);
699
+ assert.strictEqual(eventsFiredUpdates.count, 1);
700
+ assert.strictEqual(pendingResetUpdates.count, 2);
701
+
702
+ yield* manager.add(Plugin3.meta.id);
703
+ assert.strictEqual(pluginUpdates.count, 3);
704
+ assert.strictEqual(enabledUpdates.count, 3);
705
+ assert.strictEqual(modulesUpdates.count, 3);
706
+ assert.strictEqual(activeUpdates.count, 3);
707
+ assert.strictEqual(eventsFiredUpdates.count, 1);
708
+ assert.strictEqual(pendingResetUpdates.count, 4);
709
+
710
+ yield* manager.reset(CountEvent);
711
+ assert.strictEqual(pluginUpdates.count, 3);
712
+ assert.strictEqual(enabledUpdates.count, 3);
713
+ assert.strictEqual(modulesUpdates.count, 3);
714
+ // Starts at 3, plus deactivates 3, plus activates 3.
715
+ assert.strictEqual(activeUpdates.count, 9);
716
+ assert.strictEqual(eventsFiredUpdates.count, 1);
717
+ assert.strictEqual(pendingResetUpdates.count, 4);
718
+
719
+ yield* manager.disable(Plugin1.meta.id);
720
+ assert.strictEqual(pluginUpdates.count, 3);
721
+ assert.strictEqual(enabledUpdates.count, 4);
722
+ assert.strictEqual(modulesUpdates.count, 4);
723
+ assert.strictEqual(activeUpdates.count, 10);
724
+ assert.strictEqual(eventsFiredUpdates.count, 1);
725
+ assert.strictEqual(pendingResetUpdates.count, 4);
726
+
727
+ manager.remove(Plugin1.meta.id);
728
+ assert.strictEqual(pluginUpdates.count, 4);
729
+ assert.strictEqual(enabledUpdates.count, 4);
730
+ assert.strictEqual(modulesUpdates.count, 4);
731
+ assert.strictEqual(activeUpdates.count, 10);
732
+ assert.strictEqual(eventsFiredUpdates.count, 1);
733
+ assert.strictEqual(pendingResetUpdates.count, 4);
734
+
735
+ yield* manager.reset(CountEvent);
736
+ assert.strictEqual(pluginUpdates.count, 4);
737
+ assert.strictEqual(enabledUpdates.count, 4);
738
+ assert.strictEqual(modulesUpdates.count, 4);
739
+ // Starts at 10, plus deactivates 2, plus activates 2.
740
+ assert.strictEqual(activeUpdates.count, 14);
741
+ assert.strictEqual(eventsFiredUpdates.count, 1);
742
+ assert.strictEqual(pendingResetUpdates.count, 4);
743
+ }),
744
+ );
745
+
746
+ it.effect('should log a warning when a module takes too long to activate', () =>
747
+ Effect.gen(function* () {
748
+ const capturedWarnings: LogEntry[] = [];
749
+ const removeProcessor = log.addProcessor((_config: LogConfig, entry: LogEntry) => {
750
+ if (entry.level === LogLevel.WARN) {
751
+ capturedWarnings.push(entry);
752
+ }
753
+ });
754
+
755
+ const SlowEvent = ActivationEvent.make('dxos.org/test/slow');
756
+ const SlowPlugin = Plugin.define({ id: 'dxos.org/test/slow-plugin', name: 'Slow Plugin' }).pipe(
757
+ Plugin.addModule({
758
+ id: 'SlowModule',
759
+ activatesOn: SlowEvent,
760
+ activate: Effect.fnUntraced(function* () {
761
+ // Simulate a slow activation that takes 15 seconds.
762
+ yield* Effect.sleep(Duration.seconds(15));
763
+ return Capability.contributes(String, { string: 'slow' });
764
+ }),
765
+ }),
766
+ Plugin.make,
767
+ );
768
+
769
+ const slowPlugin = SlowPlugin();
770
+ plugins = [slowPlugin];
771
+
772
+ const manager = PluginManager.make({ pluginLoader });
773
+ yield* manager.add(SlowPlugin.meta.id);
774
+
775
+ // Fork the activation so we can control time with TestClock.
776
+ const activationFiber = yield* Effect.fork(manager.activate(SlowEvent));
777
+
778
+ // Advance time past the 10 second warning threshold.
779
+ yield* TestClock.adjust(Duration.seconds(11));
780
+
781
+ // Check that the warning was logged.
782
+ assert.isTrue(
783
+ capturedWarnings.some((entry) => entry.message?.includes('module is taking a long time to activate')),
784
+ 'Expected a warning about slow module activation',
785
+ );
786
+
787
+ // Advance time to let the module finish activating.
788
+ yield* TestClock.adjust(Duration.seconds(5));
789
+ yield* Fiber.join(activationFiber);
790
+
791
+ removeProcessor();
792
+ }),
793
+ );
794
+
795
+ it.effect('should prevent concurrent loads of the same module via semaphore', () =>
796
+ Effect.gen(function* () {
797
+ // Two different events that both can trigger the same module.
798
+ const EventA = ActivationEvent.make('dxos.org/test/event-a');
799
+ const EventB = ActivationEvent.make('dxos.org/test/event-b');
800
+
801
+ let activateCallCount = 0;
802
+ const ConcurrentPlugin = Plugin.define({ id: 'dxos.org/test/concurrent-plugin', name: 'Concurrent Plugin' }).pipe(
803
+ Plugin.addModule({
804
+ id: 'ConcurrentModule',
805
+ // Module activates on either event - this allows two different events to race.
806
+ activatesOn: ActivationEvent.oneOf(EventA, EventB),
807
+ activate: Effect.fnUntraced(function* () {
808
+ activateCallCount++;
809
+ // Simulate slow activation to create window for race condition.
810
+ yield* Effect.sleep(Duration.seconds(5));
811
+ return Capability.contributes(String, { string: 'concurrent' });
812
+ }),
813
+ }),
814
+ Plugin.make,
815
+ );
816
+
817
+ const concurrentPlugin = ConcurrentPlugin();
818
+ plugins = [concurrentPlugin];
819
+
820
+ const manager = PluginManager.make({ pluginLoader });
821
+ yield* manager.add(ConcurrentPlugin.meta.id);
822
+
823
+ // Fork two concurrent activations with DIFFERENT events.
824
+ // Both events trigger the same module, so both will try to call _loadModule.
825
+ // Without the semaphore, both would start loading the same module.
826
+ const fiber1 = yield* Effect.fork(manager.activate(EventA));
827
+ const fiber2 = yield* Effect.fork(manager.activate(EventB));
828
+
829
+ // Advance time to let both activations complete.
830
+ yield* TestClock.adjust(Duration.seconds(6));
831
+
832
+ yield* Fiber.join(fiber1);
833
+ yield* Fiber.join(fiber2);
834
+
835
+ // The semaphore should ensure the module's activate function is only called once,
836
+ // even when two different events race to load the same module.
837
+ assert.strictEqual(activateCallCount, 1, 'module activate should only be called once due to semaphore');
838
+
839
+ // Verify the capability was contributed.
840
+ const strings = manager.capabilities.getAll(String);
841
+ assert.isTrue(strings.length >= 1, 'capability should be contributed');
842
+ assert.strictEqual(strings[0].string, 'concurrent');
843
+ }),
844
+ );
845
+ });