@dxos/observability 0.8.4-main.bc674ce → 0.8.4-main.bcb3aa67d6

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 (346) hide show
  1. package/README.md +14 -11
  2. package/dist/lib/browser/chunk-J5LGTIGS.mjs +10 -0
  3. package/dist/lib/browser/chunk-K4VFBKST.mjs +13 -0
  4. package/dist/lib/browser/chunk-K4VFBKST.mjs.map +7 -0
  5. package/dist/lib/browser/index.mjs +922 -278
  6. package/dist/lib/browser/index.mjs.map +4 -4
  7. package/dist/lib/browser/log-processor-FDLTDQEM.mjs +45 -0
  8. package/dist/lib/browser/log-processor-FDLTDQEM.mjs.map +7 -0
  9. package/dist/lib/browser/logs-ATTRIUTL.mjs +113 -0
  10. package/dist/lib/browser/logs-ATTRIUTL.mjs.map +7 -0
  11. package/dist/lib/browser/meta.json +1 -1
  12. package/dist/lib/browser/metrics-PKTV6IGF.mjs +130 -0
  13. package/dist/lib/browser/metrics-PKTV6IGF.mjs.map +7 -0
  14. package/dist/lib/browser/traces-browser-PUNCJ5KV.mjs +112 -0
  15. package/dist/lib/browser/traces-browser-PUNCJ5KV.mjs.map +7 -0
  16. package/dist/lib/node-esm/chunk-FEVP3MK4.mjs +15 -0
  17. package/dist/lib/node-esm/chunk-FEVP3MK4.mjs.map +7 -0
  18. package/dist/lib/node-esm/chunk-HSLMI22Q.mjs +11 -0
  19. package/dist/lib/node-esm/index.mjs +939 -293
  20. package/dist/lib/node-esm/index.mjs.map +4 -4
  21. package/dist/lib/node-esm/log-processor-TKJVJJSJ.mjs +46 -0
  22. package/dist/lib/node-esm/log-processor-TKJVJJSJ.mjs.map +7 -0
  23. package/dist/lib/node-esm/logs-7J45KLM7.mjs +114 -0
  24. package/dist/lib/node-esm/logs-7J45KLM7.mjs.map +7 -0
  25. package/dist/lib/node-esm/meta.json +1 -1
  26. package/dist/lib/node-esm/metrics-H7DDLYSG.mjs +131 -0
  27. package/dist/lib/node-esm/metrics-H7DDLYSG.mjs.map +7 -0
  28. package/dist/lib/node-esm/{chunk-WAGGA7IT.mjs → traces-KMTHMYFX.mjs} +8 -16
  29. package/dist/lib/node-esm/traces-KMTHMYFX.mjs.map +7 -0
  30. package/dist/types/src/cli-observability-secrets.json +3 -4
  31. package/dist/types/src/extensions/index.d.ts +3 -0
  32. package/dist/types/src/extensions/index.d.ts.map +1 -0
  33. package/dist/types/src/extensions/index.js +6 -0
  34. package/dist/types/src/extensions/index.js.map +1 -0
  35. package/dist/types/src/extensions/otel/extension.d.ts +23 -0
  36. package/dist/types/src/extensions/otel/extension.d.ts.map +1 -0
  37. package/dist/types/src/extensions/otel/extension.js +124 -0
  38. package/dist/types/src/extensions/otel/extension.js.map +1 -0
  39. package/dist/types/src/extensions/otel/index.d.ts +2 -0
  40. package/dist/types/src/extensions/otel/index.d.ts.map +1 -0
  41. package/dist/types/src/extensions/otel/index.js +5 -0
  42. package/dist/types/src/extensions/otel/index.js.map +1 -0
  43. package/dist/types/src/{otel → extensions/otel}/logs.d.ts +4 -3
  44. package/dist/types/src/extensions/otel/logs.d.ts.map +1 -0
  45. package/dist/types/src/{otel → extensions/otel}/logs.js +48 -18
  46. package/dist/types/src/extensions/otel/logs.js.map +1 -0
  47. package/dist/types/src/extensions/otel/metrics.d.ts.map +1 -0
  48. package/dist/types/src/{otel → extensions/otel}/metrics.js +2 -10
  49. package/dist/types/src/extensions/otel/metrics.js.map +1 -0
  50. package/dist/types/src/{otel → extensions/otel}/otel.d.ts +3 -3
  51. package/dist/types/src/extensions/otel/otel.d.ts.map +1 -0
  52. package/dist/types/src/extensions/otel/otel.js.map +1 -0
  53. package/dist/types/src/extensions/otel/traces-browser.d.ts.map +1 -0
  54. package/dist/types/src/extensions/otel/traces-browser.js +85 -0
  55. package/dist/types/src/extensions/otel/traces-browser.js.map +1 -0
  56. package/dist/types/src/extensions/otel/traces.d.ts.map +1 -0
  57. package/dist/types/src/{otel → extensions/otel}/traces.js +5 -13
  58. package/dist/types/src/extensions/otel/traces.js.map +1 -0
  59. package/dist/types/src/extensions/posthog/extension.d.ts +18 -0
  60. package/dist/types/src/extensions/posthog/extension.d.ts.map +1 -0
  61. package/dist/types/src/extensions/posthog/extension.js +142 -0
  62. package/dist/types/src/extensions/posthog/extension.js.map +1 -0
  63. package/dist/types/src/extensions/posthog/index.d.ts +2 -0
  64. package/dist/types/src/extensions/posthog/index.d.ts.map +1 -0
  65. package/dist/types/src/extensions/posthog/index.js +5 -0
  66. package/dist/types/src/extensions/posthog/index.js.map +1 -0
  67. package/dist/types/src/extensions/posthog/log-processor.d.ts +3 -0
  68. package/dist/types/src/extensions/posthog/log-processor.d.ts.map +1 -0
  69. package/dist/types/src/extensions/posthog/log-processor.js +45 -0
  70. package/dist/types/src/extensions/posthog/log-processor.js.map +1 -0
  71. package/dist/types/src/extensions/posthog/log-processor.test.d.ts +2 -0
  72. package/dist/types/src/extensions/posthog/log-processor.test.d.ts.map +1 -0
  73. package/dist/types/src/extensions/posthog/log-processor.test.js +146 -0
  74. package/dist/types/src/extensions/posthog/log-processor.test.js.map +1 -0
  75. package/dist/types/src/extensions/stub.d.ts +3 -0
  76. package/dist/types/src/extensions/stub.d.ts.map +1 -0
  77. package/dist/types/src/extensions/stub.js +16 -0
  78. package/dist/types/src/extensions/stub.js.map +1 -0
  79. package/dist/types/src/index.d.ts +3 -2
  80. package/dist/types/src/index.d.ts.map +1 -1
  81. package/dist/types/src/index.js +4 -3
  82. package/dist/types/src/index.js.map +1 -1
  83. package/dist/types/src/observability-extension.d.ts +74 -0
  84. package/dist/types/src/observability-extension.d.ts.map +1 -0
  85. package/dist/types/src/observability-extension.js +5 -0
  86. package/dist/types/src/observability-extension.js.map +1 -0
  87. package/dist/types/src/observability.d.ts +32 -110
  88. package/dist/types/src/observability.d.ts.map +1 -1
  89. package/dist/types/src/observability.js +177 -465
  90. package/dist/types/src/observability.js.map +1 -1
  91. package/dist/types/src/observability.test.d.ts +2 -0
  92. package/dist/types/src/observability.test.d.ts.map +1 -0
  93. package/dist/types/src/observability.test.js +312 -0
  94. package/dist/types/src/observability.test.js.map +1 -0
  95. package/dist/types/src/providers/client-observability.d.ts +11 -0
  96. package/dist/types/src/providers/client-observability.d.ts.map +1 -0
  97. package/dist/types/src/providers/client-observability.js +200 -0
  98. package/dist/types/src/providers/client-observability.js.map +1 -0
  99. package/dist/types/src/providers/index.d.ts +4 -0
  100. package/dist/types/src/providers/index.d.ts.map +1 -0
  101. package/dist/types/src/providers/index.js +7 -0
  102. package/dist/types/src/providers/index.js.map +1 -0
  103. package/dist/types/src/providers/ip-data.d.ts +5 -0
  104. package/dist/types/src/providers/ip-data.d.ts.map +1 -0
  105. package/dist/types/src/providers/ip-data.js +57 -0
  106. package/dist/types/src/providers/ip-data.js.map +1 -0
  107. package/dist/types/src/providers/storage.d.ts +3 -0
  108. package/dist/types/src/providers/storage.d.ts.map +1 -0
  109. package/dist/types/src/providers/storage.js +19 -0
  110. package/dist/types/src/providers/storage.js.map +1 -0
  111. package/dist/types/src/storage/browser.d.ts +19 -0
  112. package/dist/types/src/storage/browser.d.ts.map +1 -0
  113. package/dist/types/src/storage/browser.js +59 -0
  114. package/dist/types/src/storage/browser.js.map +1 -0
  115. package/dist/types/src/storage/index.d.ts +2 -0
  116. package/dist/types/src/storage/index.d.ts.map +1 -0
  117. package/dist/types/src/storage/index.js.map +1 -0
  118. package/dist/types/src/storage/node.d.ts +26 -0
  119. package/dist/types/src/storage/node.d.ts.map +1 -0
  120. package/dist/types/src/{helpers/node-observability.js → storage/node.js} +37 -42
  121. package/dist/types/src/storage/node.js.map +1 -0
  122. package/dist/types/src/storage/node.test.d.ts +2 -0
  123. package/dist/types/src/storage/node.test.d.ts.map +1 -0
  124. package/dist/types/src/storage/node.test.js +103 -0
  125. package/dist/types/src/storage/node.test.js.map +1 -0
  126. package/dist/types/tsconfig.tsbuildinfo +1 -1
  127. package/package.json +26 -47
  128. package/src/cli-observability-secrets.json +3 -4
  129. package/src/extensions/index.ts +6 -0
  130. package/src/extensions/otel/extension.ts +178 -0
  131. package/src/extensions/otel/index.ts +5 -0
  132. package/src/{otel → extensions/otel}/logs.ts +55 -23
  133. package/src/{otel → extensions/otel}/metrics.ts +2 -12
  134. package/src/{otel → extensions/otel}/otel.ts +3 -3
  135. package/src/extensions/otel/traces-browser.ts +110 -0
  136. package/src/{otel → extensions/otel}/traces.ts +10 -21
  137. package/src/extensions/posthog/extension.ts +174 -0
  138. package/src/extensions/posthog/index.ts +5 -0
  139. package/src/extensions/posthog/log-processor.test.ts +185 -0
  140. package/src/extensions/posthog/log-processor.ts +54 -0
  141. package/src/extensions/stub.ts +19 -0
  142. package/src/index.ts +4 -3
  143. package/src/observability-extension.ts +94 -0
  144. package/src/observability.test.ts +531 -0
  145. package/src/observability.ts +236 -577
  146. package/src/providers/client-observability.ts +253 -0
  147. package/src/providers/index.ts +7 -0
  148. package/src/providers/ip-data.ts +91 -0
  149. package/src/providers/storage.ts +23 -0
  150. package/src/storage/browser.ts +62 -0
  151. package/src/{segment → storage}/index.ts +0 -1
  152. package/src/storage/node.test.ts +130 -0
  153. package/src/{helpers/node-observability.ts → storage/node.ts} +40 -69
  154. package/dist/lib/browser/chunk-5ICT2XF2.mjs +0 -1
  155. package/dist/lib/browser/chunk-5LN7D6GM.mjs +0 -148
  156. package/dist/lib/browser/chunk-5LN7D6GM.mjs.map +0 -7
  157. package/dist/lib/browser/chunk-JJQT5TQH.mjs +0 -672
  158. package/dist/lib/browser/chunk-JJQT5TQH.mjs.map +0 -7
  159. package/dist/lib/browser/chunk-O4BG5MRL.mjs +0 -164
  160. package/dist/lib/browser/chunk-O4BG5MRL.mjs.map +0 -7
  161. package/dist/lib/browser/chunk-U6JWT3E2.mjs +0 -1
  162. package/dist/lib/browser/chunk-VL6LVQPU.mjs +0 -69
  163. package/dist/lib/browser/chunk-VL6LVQPU.mjs.map +0 -7
  164. package/dist/lib/browser/observability-XK652NZG.mjs +0 -11
  165. package/dist/lib/browser/observability-XK652NZG.mjs.map +0 -7
  166. package/dist/lib/browser/otel/traces-browser.mjs +0 -7
  167. package/dist/lib/browser/otel/traces-browser.mjs.map +0 -7
  168. package/dist/lib/browser/otel/traces.mjs +0 -7
  169. package/dist/lib/browser/otel/traces.mjs.map +0 -7
  170. package/dist/lib/browser/otel-DI4ASU7Y.mjs +0 -226
  171. package/dist/lib/browser/otel-DI4ASU7Y.mjs.map +0 -7
  172. package/dist/lib/browser/segment/browser.mjs +0 -9
  173. package/dist/lib/browser/segment/browser.mjs.map +0 -7
  174. package/dist/lib/browser/segment/index.mjs +0 -12
  175. package/dist/lib/browser/segment/index.mjs.map +0 -7
  176. package/dist/lib/browser/segment/node.mjs +0 -9
  177. package/dist/lib/browser/segment/node.mjs.map +0 -7
  178. package/dist/lib/browser/sentry/browser.mjs +0 -23
  179. package/dist/lib/browser/sentry/browser.mjs.map +0 -7
  180. package/dist/lib/browser/sentry/index.mjs +0 -24
  181. package/dist/lib/browser/sentry/index.mjs.map +0 -7
  182. package/dist/lib/browser/sentry/node.mjs +0 -23
  183. package/dist/lib/browser/sentry/node.mjs.map +0 -7
  184. package/dist/lib/browser/sentry-log-processor-W7LI6WXA.mjs +0 -146
  185. package/dist/lib/browser/sentry-log-processor-W7LI6WXA.mjs.map +0 -7
  186. package/dist/lib/node-esm/chunk-KKNY7TRV.mjs +0 -673
  187. package/dist/lib/node-esm/chunk-KKNY7TRV.mjs.map +0 -7
  188. package/dist/lib/node-esm/chunk-KLJGCUYA.mjs +0 -2
  189. package/dist/lib/node-esm/chunk-KLJGCUYA.mjs.map +0 -7
  190. package/dist/lib/node-esm/chunk-KVJTNW3F.mjs +0 -147
  191. package/dist/lib/node-esm/chunk-KVJTNW3F.mjs.map +0 -7
  192. package/dist/lib/node-esm/chunk-M4627SMT.mjs +0 -135
  193. package/dist/lib/node-esm/chunk-M4627SMT.mjs.map +0 -7
  194. package/dist/lib/node-esm/chunk-QUZL7LKE.mjs +0 -2
  195. package/dist/lib/node-esm/chunk-QUZL7LKE.mjs.map +0 -7
  196. package/dist/lib/node-esm/chunk-UIVXGEGJ.mjs +0 -62
  197. package/dist/lib/node-esm/chunk-UIVXGEGJ.mjs.map +0 -7
  198. package/dist/lib/node-esm/chunk-WAGGA7IT.mjs.map +0 -7
  199. package/dist/lib/node-esm/observability-CXQ3CZGB.mjs +0 -13
  200. package/dist/lib/node-esm/observability-CXQ3CZGB.mjs.map +0 -7
  201. package/dist/lib/node-esm/otel/traces-browser.mjs +0 -70
  202. package/dist/lib/node-esm/otel/traces-browser.mjs.map +0 -7
  203. package/dist/lib/node-esm/otel/traces.mjs +0 -8
  204. package/dist/lib/node-esm/otel/traces.mjs.map +0 -7
  205. package/dist/lib/node-esm/otel-JFXO22WD.mjs +0 -227
  206. package/dist/lib/node-esm/otel-JFXO22WD.mjs.map +0 -7
  207. package/dist/lib/node-esm/segment/browser.mjs +0 -92
  208. package/dist/lib/node-esm/segment/browser.mjs.map +0 -7
  209. package/dist/lib/node-esm/segment/index.mjs +0 -15
  210. package/dist/lib/node-esm/segment/index.mjs.map +0 -7
  211. package/dist/lib/node-esm/segment/node.mjs +0 -11
  212. package/dist/lib/node-esm/segment/node.mjs.map +0 -7
  213. package/dist/lib/node-esm/sentry/browser.mjs +0 -165
  214. package/dist/lib/node-esm/sentry/browser.mjs.map +0 -7
  215. package/dist/lib/node-esm/sentry/index.mjs +0 -25
  216. package/dist/lib/node-esm/sentry/index.mjs.map +0 -7
  217. package/dist/lib/node-esm/sentry/node.mjs +0 -24
  218. package/dist/lib/node-esm/sentry/node.mjs.map +0 -7
  219. package/dist/lib/node-esm/sentry-log-processor-W3SG4RQL.mjs +0 -147
  220. package/dist/lib/node-esm/sentry-log-processor-W3SG4RQL.mjs.map +0 -7
  221. package/dist/types/src/helpers/browser-observability.d.ts +0 -17
  222. package/dist/types/src/helpers/browser-observability.d.ts.map +0 -1
  223. package/dist/types/src/helpers/browser-observability.js +0 -138
  224. package/dist/types/src/helpers/browser-observability.js.map +0 -1
  225. package/dist/types/src/helpers/common.d.ts +0 -12
  226. package/dist/types/src/helpers/common.d.ts.map +0 -1
  227. package/dist/types/src/helpers/common.js +0 -23
  228. package/dist/types/src/helpers/common.js.map +0 -1
  229. package/dist/types/src/helpers/index.d.ts +0 -6
  230. package/dist/types/src/helpers/index.d.ts.map +0 -1
  231. package/dist/types/src/helpers/index.js +0 -9
  232. package/dist/types/src/helpers/index.js.map +0 -1
  233. package/dist/types/src/helpers/map-spaces.d.ts +0 -18
  234. package/dist/types/src/helpers/map-spaces.d.ts.map +0 -1
  235. package/dist/types/src/helpers/map-spaces.js +0 -36
  236. package/dist/types/src/helpers/map-spaces.js.map +0 -1
  237. package/dist/types/src/helpers/node-observability.d.ts +0 -24
  238. package/dist/types/src/helpers/node-observability.d.ts.map +0 -1
  239. package/dist/types/src/helpers/node-observability.js.map +0 -1
  240. package/dist/types/src/helpers/setup-telemetry-listeners.d.ts +0 -4
  241. package/dist/types/src/helpers/setup-telemetry-listeners.d.ts.map +0 -1
  242. package/dist/types/src/helpers/setup-telemetry-listeners.js +0 -94
  243. package/dist/types/src/helpers/setup-telemetry-listeners.js.map +0 -1
  244. package/dist/types/src/otel/index.d.ts +0 -5
  245. package/dist/types/src/otel/index.d.ts.map +0 -1
  246. package/dist/types/src/otel/index.js +0 -8
  247. package/dist/types/src/otel/index.js.map +0 -1
  248. package/dist/types/src/otel/logs.d.ts.map +0 -1
  249. package/dist/types/src/otel/logs.js.map +0 -1
  250. package/dist/types/src/otel/metrics.d.ts.map +0 -1
  251. package/dist/types/src/otel/metrics.js.map +0 -1
  252. package/dist/types/src/otel/otel.d.ts.map +0 -1
  253. package/dist/types/src/otel/otel.js.map +0 -1
  254. package/dist/types/src/otel/traces-browser.d.ts.map +0 -1
  255. package/dist/types/src/otel/traces-browser.js +0 -53
  256. package/dist/types/src/otel/traces-browser.js.map +0 -1
  257. package/dist/types/src/otel/traces.d.ts.map +0 -1
  258. package/dist/types/src/otel/traces.js.map +0 -1
  259. package/dist/types/src/segment/base.d.ts +0 -15
  260. package/dist/types/src/segment/base.d.ts.map +0 -1
  261. package/dist/types/src/segment/base.js +0 -51
  262. package/dist/types/src/segment/base.js.map +0 -1
  263. package/dist/types/src/segment/browser.d.ts +0 -15
  264. package/dist/types/src/segment/browser.d.ts.map +0 -1
  265. package/dist/types/src/segment/browser.js +0 -63
  266. package/dist/types/src/segment/browser.js.map +0 -1
  267. package/dist/types/src/segment/index.d.ts +0 -3
  268. package/dist/types/src/segment/index.d.ts.map +0 -1
  269. package/dist/types/src/segment/index.js +0 -6
  270. package/dist/types/src/segment/index.js.map +0 -1
  271. package/dist/types/src/segment/node.d.ts +0 -16
  272. package/dist/types/src/segment/node.d.ts.map +0 -1
  273. package/dist/types/src/segment/node.js +0 -84
  274. package/dist/types/src/segment/node.js.map +0 -1
  275. package/dist/types/src/segment/types.d.ts +0 -52
  276. package/dist/types/src/segment/types.d.ts.map +0 -1
  277. package/dist/types/src/segment/types.js +0 -18
  278. package/dist/types/src/segment/types.js.map +0 -1
  279. package/dist/types/src/sentry/browser.d.ts +0 -32
  280. package/dist/types/src/sentry/browser.d.ts.map +0 -1
  281. package/dist/types/src/sentry/browser.js +0 -110
  282. package/dist/types/src/sentry/browser.js.map +0 -1
  283. package/dist/types/src/sentry/index.d.ts +0 -3
  284. package/dist/types/src/sentry/index.d.ts.map +0 -1
  285. package/dist/types/src/sentry/index.js.map +0 -1
  286. package/dist/types/src/sentry/node.d.ts +0 -32
  287. package/dist/types/src/sentry/node.d.ts.map +0 -1
  288. package/dist/types/src/sentry/node.js +0 -106
  289. package/dist/types/src/sentry/node.js.map +0 -1
  290. package/dist/types/src/sentry/node.node.test.d.ts +0 -2
  291. package/dist/types/src/sentry/node.node.test.d.ts.map +0 -1
  292. package/dist/types/src/sentry/node.node.test.js +0 -32
  293. package/dist/types/src/sentry/node.node.test.js.map +0 -1
  294. package/dist/types/src/sentry/sentry-log-processor.d.ts +0 -9
  295. package/dist/types/src/sentry/sentry-log-processor.d.ts.map +0 -1
  296. package/dist/types/src/sentry/sentry-log-processor.js +0 -144
  297. package/dist/types/src/sentry/sentry-log-processor.js.map +0 -1
  298. package/dist/types/src/sentry/sentry.node.test.d.ts +0 -2
  299. package/dist/types/src/sentry/sentry.node.test.d.ts.map +0 -1
  300. package/dist/types/src/sentry/sentry.node.test.js +0 -28
  301. package/dist/types/src/sentry/sentry.node.test.js.map +0 -1
  302. package/dist/types/src/sentry/types.d.ts +0 -18
  303. package/dist/types/src/sentry/types.d.ts.map +0 -1
  304. package/dist/types/src/sentry/types.js +0 -4
  305. package/dist/types/src/sentry/types.js.map +0 -1
  306. package/dist/types/src/testing/index.d.ts +0 -2
  307. package/dist/types/src/testing/index.d.ts.map +0 -1
  308. package/dist/types/src/testing/index.js +0 -5
  309. package/dist/types/src/testing/index.js.map +0 -1
  310. package/dist/types/src/testing/testkit/browser.d.ts +0 -2
  311. package/dist/types/src/testing/testkit/browser.d.ts.map +0 -1
  312. package/dist/types/src/testing/testkit/browser.js +0 -7
  313. package/dist/types/src/testing/testkit/browser.js.map +0 -1
  314. package/dist/types/src/testing/testkit/index.d.ts +0 -2
  315. package/dist/types/src/testing/testkit/index.d.ts.map +0 -1
  316. package/dist/types/src/testing/testkit/index.js +0 -6
  317. package/dist/types/src/testing/testkit/index.js.map +0 -1
  318. package/src/helpers/browser-observability.ts +0 -178
  319. package/src/helpers/common.ts +0 -38
  320. package/src/helpers/index.ts +0 -9
  321. package/src/helpers/map-spaces.ts +0 -48
  322. package/src/helpers/setup-telemetry-listeners.ts +0 -109
  323. package/src/otel/index.ts +0 -8
  324. package/src/otel/traces-browser.ts +0 -62
  325. package/src/segment/base.ts +0 -69
  326. package/src/segment/browser.ts +0 -69
  327. package/src/segment/node.ts +0 -95
  328. package/src/segment/types.ts +0 -57
  329. package/src/sentry/browser.ts +0 -133
  330. package/src/sentry/index.ts +0 -6
  331. package/src/sentry/node.node.test.ts +0 -40
  332. package/src/sentry/node.ts +0 -126
  333. package/src/sentry/sentry-log-processor.ts +0 -176
  334. package/src/sentry/sentry.node.test.ts +0 -35
  335. package/src/sentry/types.ts +0 -22
  336. package/src/testing/index.ts +0 -5
  337. package/src/testing/testkit/browser.ts +0 -8
  338. package/src/testing/testkit/index.ts +0 -7
  339. package/src/testing/testkit/shims.d.ts +0 -5
  340. /package/dist/lib/browser/{chunk-5ICT2XF2.mjs.map → chunk-J5LGTIGS.mjs.map} +0 -0
  341. /package/dist/lib/{browser/chunk-U6JWT3E2.mjs.map → node-esm/chunk-HSLMI22Q.mjs.map} +0 -0
  342. /package/dist/types/src/{otel → extensions/otel}/metrics.d.ts +0 -0
  343. /package/dist/types/src/{otel → extensions/otel}/otel.js +0 -0
  344. /package/dist/types/src/{otel → extensions/otel}/traces-browser.d.ts +0 -0
  345. /package/dist/types/src/{otel → extensions/otel}/traces.d.ts +0 -0
  346. /package/dist/types/src/{sentry → storage}/index.js +0 -0
@@ -0,0 +1,253 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import * as Effect from 'effect/Effect';
6
+
7
+ import { Event, scheduleTaskInterval } from '@dxos/async';
8
+ import { type Client, type ClientServices } from '@dxos/client';
9
+ import { type Space } from '@dxos/client/echo';
10
+ import { DeviceKind } from '@dxos/client/halo';
11
+ import { Context } from '@dxos/context';
12
+ import { invariant } from '@dxos/invariant';
13
+ import { log } from '@dxos/log';
14
+ import { ConnectionState, type NetworkStatus, Platform } from '@dxos/protocols/proto/dxos/client/services';
15
+
16
+ import { type DataProvider } from '../observability';
17
+
18
+ const SPACE_METRICS_MIN_INTERVAL = 1000 * 60 * 10; // 10 minutes
19
+ const NETWORK_METRICS_MIN_INTERVAL = 1000 * 60 * 10; // 10 minutes
20
+ const RUNTIME_METRICS_MIN_INTERVAL = 1000 * 60 * 10; // 10 minutes
21
+
22
+ // TODO(wittjosiah): Improve privacy of telemetry identifiers.
23
+ // - Identifier should be generated client-side with no attachment to identity.
24
+ // - Identifier can then be reset by user.
25
+ // - Identifier can be synced via HALO to allow for correlation of events bewteen devices.
26
+ // - Identifier should also be stored outside of HALO such that it is available immediately on startup.
27
+ /** Subscribes to identity and device changes and sets observability tags accordingly. */
28
+ export const identityProvider = (clientServices: Partial<ClientServices>): DataProvider =>
29
+ Effect.fn(function* (observability) {
30
+ // TODO(wittjosiah): RPC subscribe returns void; cleanup requires upstream API change.
31
+ clientServices.IdentityService!.queryIdentity().subscribe((idqr) => {
32
+ if (!idqr?.identity?.did) {
33
+ return;
34
+ }
35
+
36
+ observability.identify(idqr.identity.did);
37
+ observability.setTags({ did: idqr.identity.did });
38
+ });
39
+
40
+ // TODO(wittjosiah): RPC subscribe returns void; cleanup requires upstream API change.
41
+ clientServices.DevicesService!.queryDevices().subscribe((dqr) => {
42
+ if (!dqr?.devices || dqr.devices.length === 0) {
43
+ return;
44
+ }
45
+
46
+ const thisDevice = dqr.devices.find((device) => device.kind === DeviceKind.CURRENT);
47
+ if (!thisDevice) {
48
+ return;
49
+ }
50
+
51
+ observability.setTags({ deviceKey: thisDevice.deviceKey.truncate() });
52
+ if (thisDevice.profile?.label) {
53
+ observability.setTags({ deviceProfile: thisDevice.profile.label });
54
+ }
55
+ });
56
+ });
57
+
58
+ /** Periodically publishes network connection and buffer metrics. */
59
+ export const networkMetricsProvider = (clientServices: Partial<ClientServices>): DataProvider =>
60
+ Effect.fn(function* (observability) {
61
+ const ctx = new Context();
62
+ let lastNetworkStatus: NetworkStatus | undefined;
63
+
64
+ // TODO(nf): support type in debounce()
65
+ const updateSignalMetrics = new Event<NetworkStatus>().debounce(NETWORK_METRICS_MIN_INTERVAL);
66
+ updateSignalMetrics.on(ctx, async () => {
67
+ log('send signal metrics');
68
+ (lastNetworkStatus?.signaling as NetworkStatus.Signal[])?.forEach(({ server, state }) => {
69
+ observability.metrics.gauge('dxos.client.network.signal.connectionState', state, { server });
70
+ });
71
+
72
+ let swarmCount = 0;
73
+ const connectionStates = new Map<string, number>();
74
+ for (const state in ConnectionState) {
75
+ connectionStates.set(state, 0);
76
+ }
77
+
78
+ let totalReadBufferSize = 0;
79
+ let totalWriteBufferSize = 0;
80
+ let totalChannelBufferSize = 0;
81
+
82
+ lastNetworkStatus?.connectionInfo?.forEach((connectionInfo) => {
83
+ swarmCount++;
84
+
85
+ for (const conn of connectionInfo.connections ?? []) {
86
+ connectionStates.set(conn.state, (connectionStates.get(conn.state) ?? 0) + 1);
87
+ totalReadBufferSize += conn.readBufferSize ?? 0;
88
+ totalWriteBufferSize += conn.writeBufferSize ?? 0;
89
+ for (const stream of conn.streams ?? []) {
90
+ totalChannelBufferSize += stream.writeBufferSize ?? 0;
91
+ }
92
+ }
93
+
94
+ observability.metrics.gauge('dxos.client.network.swarm.count', swarmCount);
95
+ for (const state in ConnectionState) {
96
+ observability.metrics.gauge('dxos.client.network.connection.count', connectionStates.get(state) ?? 0, {
97
+ state,
98
+ });
99
+ }
100
+ observability.metrics.gauge('dxos.client.network.totalReadBufferSize', totalReadBufferSize);
101
+ observability.metrics.gauge('dxos.client.network.totalWriteBufferSize', totalWriteBufferSize);
102
+ observability.metrics.gauge('dxos.client.network.totalChannelBufferSize', totalChannelBufferSize);
103
+ });
104
+ });
105
+
106
+ clientServices.NetworkService!.queryStatus().subscribe((networkStatus) => {
107
+ lastNetworkStatus = networkStatus;
108
+ updateSignalMetrics.emit();
109
+ });
110
+
111
+ scheduleTaskInterval(ctx, async () => updateSignalMetrics.emit(), NETWORK_METRICS_MIN_INTERVAL);
112
+
113
+ return async () => {
114
+ await ctx.dispose();
115
+ };
116
+ });
117
+
118
+ /** Periodically publishes platform and heap memory metrics. */
119
+ export const runtimeMetricsProvider = (clientServices: Partial<ClientServices>): DataProvider =>
120
+ Effect.fn(function* (observability) {
121
+ const ctx = new Context();
122
+ const platform = yield* Effect.promise(() => clientServices.SystemService!.getPlatform());
123
+ invariant(platform, 'platform is required');
124
+
125
+ observability.setTags({
126
+ platformType: Platform.PLATFORM_TYPE[platform.type as number].toLowerCase(),
127
+ platform: platform.platform,
128
+ arch: platform.arch,
129
+ runtime: platform.runtime,
130
+ });
131
+
132
+ scheduleTaskInterval(
133
+ ctx,
134
+ async () => {
135
+ if (clientServices.constructor.name === 'WorkerClientServices') {
136
+ const memory = (window.performance as any).memory;
137
+ if (memory) {
138
+ observability.metrics.gauge('dxos.client.runtime.heapTotal', memory.totalJSHeapSize);
139
+ observability.metrics.gauge('dxos.client.runtime.heapUsed', memory.usedJSHeapSize);
140
+ observability.metrics.gauge('dxos.client.runtime.heapSizeLimit', memory.jsHeapSizeLimit);
141
+ }
142
+ }
143
+
144
+ clientServices.SystemService?.getPlatform()
145
+ .then((platform) => {
146
+ if (platform.memory) {
147
+ observability.metrics.gauge('dxos.client.services.runtime.rss', platform.memory.rss);
148
+ observability.metrics.gauge('dxos.client.services.runtime.heapTotal', platform.memory.heapTotal);
149
+ observability.metrics.gauge('dxos.client.services.runtime.heapUsed', platform.memory.heapUsed);
150
+ }
151
+ })
152
+ .catch((error) => log('platform error', { error }));
153
+ },
154
+ RUNTIME_METRICS_MIN_INTERVAL,
155
+ );
156
+
157
+ return async () => {
158
+ await ctx.dispose();
159
+ };
160
+ });
161
+
162
+ /** Periodically publishes space membership, object count, and pipeline progress metrics. */
163
+ export const spacesMetricsProvider = (client: Client): DataProvider =>
164
+ Effect.fn(function* (observability) {
165
+ const ctx = new Context();
166
+ // TODO(nf): update subscription on new spaces
167
+ const spaces = client.spaces.get();
168
+ const subscriptions = new Map<string, { unsubscribe: () => void }>();
169
+ ctx.onDispose(() => subscriptions.forEach((subscription) => subscription.unsubscribe()));
170
+
171
+ const updateSpaceMetrics = new Event<Space>().debounce(SPACE_METRICS_MIN_INTERVAL);
172
+ updateSpaceMetrics.on(ctx, async () => {
173
+ log('send space metrics');
174
+ for (const data of mapSpaces(spaces, { truncateKeys: true })) {
175
+ observability.metrics.gauge('dxos.client.space.members', data.members, { key: data.key });
176
+ observability.metrics.gauge('dxos.client.space.objects', data.objects, { key: data.key });
177
+ observability.metrics.gauge('dxos.client.space.epoch', data.epoch, { key: data.key });
178
+ observability.metrics.gauge('dxos.client.space.currentDataMutations', data.currentDataMutations, {
179
+ key: data.key,
180
+ });
181
+ }
182
+ });
183
+
184
+ const subscribeToSpaceUpdate = (space: Space) =>
185
+ space.pipeline.subscribe({
186
+ next: () => {
187
+ updateSpaceMetrics.emit();
188
+ },
189
+ });
190
+
191
+ spaces.forEach((space) => {
192
+ subscriptions.set(space.id, subscribeToSpaceUpdate(space));
193
+ });
194
+
195
+ client.spaces.subscribe({
196
+ next: async (spaces) => {
197
+ spaces
198
+ .filter((space) => !subscriptions.has(space.id))
199
+ .forEach((space) => {
200
+ subscriptions.set(space.id, subscribeToSpaceUpdate(space));
201
+ });
202
+ },
203
+ });
204
+
205
+ scheduleTaskInterval(ctx, async () => updateSpaceMetrics.emit(), SPACE_METRICS_MIN_INTERVAL);
206
+
207
+ return async () => {
208
+ await ctx.dispose();
209
+ };
210
+ });
211
+
212
+ type MapSpacesOptions = {
213
+ verbose?: boolean;
214
+ truncateKeys?: boolean;
215
+ };
216
+
217
+ const mapSpaces = (spaces: Space[], options: MapSpacesOptions = { verbose: false, truncateKeys: false }) => {
218
+ return spaces.map((space) => {
219
+ // TODO(burdon): Factor out.
220
+ // TODO(burdon): Agent needs to restart before `ready` is available.
221
+ const { open, ready } = space.internal.data.metrics ?? {};
222
+ const startup = open && ready && ready.getTime() - open.getTime();
223
+
224
+ // TODO(burdon): Get feeds from client-services if verbose (factor out from devtools/diagnostics).
225
+ // const host = client.services.services.DevtoolsHost!;
226
+ const pipeline = space.internal.data.pipeline;
227
+ const startDataMutations = pipeline?.currentEpoch?.subject.assertion.timeframe.totalMessages() ?? 0;
228
+ const epoch = pipeline?.currentEpoch?.subject.assertion.number;
229
+ // const appliedEpoch = pipeline?.appliedEpoch?.subject.assertion.number;
230
+ const currentDataMutations = pipeline?.currentDataTimeframe?.totalMessages() ?? 0;
231
+ const totalDataMutations = pipeline?.targetDataTimeframe?.totalMessages() ?? 0;
232
+
233
+ return {
234
+ // TODO(nf): truncate keys for DD?
235
+ key: space.key.truncate(),
236
+ open: space.isOpen,
237
+ members: space.members.get().length,
238
+ objects: space.internal.db.coreDatabase.getAllObjectIds().length,
239
+ startup,
240
+ epoch,
241
+ // appliedEpoch,
242
+ startDataMutations,
243
+ currentDataMutations,
244
+ totalDataMutations,
245
+
246
+ // TODO(burdon): Negative?
247
+ progress: (
248
+ Math.min(Math.abs((currentDataMutations - startDataMutations) / (totalDataMutations - startDataMutations)), 1) *
249
+ 100
250
+ ).toFixed(0),
251
+ };
252
+ });
253
+ };
@@ -0,0 +1,7 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ export * as Client from './client-observability';
6
+ export * as IPData from './ip-data';
7
+ export * as Storage from './storage';
@@ -0,0 +1,91 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import * as FetchHttpClient from '@effect/platform/FetchHttpClient';
6
+ import * as HttpClient from '@effect/platform/HttpClient';
7
+ import * as HttpClientRequest from '@effect/platform/HttpClientRequest';
8
+ import * as Effect from 'effect/Effect';
9
+ import * as Schema from 'effect/Schema';
10
+ // NOTE: localStorage is not available in web workers.
11
+ import * as localForage from 'localforage';
12
+
13
+ import { type Config } from '@dxos/config';
14
+ import { log } from '@dxos/log';
15
+
16
+ import { type DataProvider } from '../observability';
17
+
18
+ const IP_DATA_CACHE_TIMEOUT = 6 * 60 * 60 * 1000; // 6 hours
19
+
20
+ const IPData = Schema.Struct({
21
+ city: Schema.String,
22
+ region: Schema.String,
23
+ country: Schema.String,
24
+ latitude: Schema.optional(Schema.Number),
25
+ longitude: Schema.optional(Schema.Number),
26
+ });
27
+ type IPData = Schema.Schema.Type<typeof IPData>;
28
+
29
+ type CachedIPData = {
30
+ data: IPData;
31
+ timestamp: number;
32
+ };
33
+
34
+ const getIPData = Effect.fn(function* (config: Config) {
35
+ const httpClient = yield* HttpClient.HttpClient;
36
+
37
+ // Disable tracing to avoid CORS errors from traceparent header on cross-origin requests.
38
+ const httpClientNoTrace = httpClient.pipe(HttpClient.withTracerDisabledWhen(() => true));
39
+
40
+ // Check cache first.
41
+ const cachedData = yield* Effect.promise(() => localForage.getItem<CachedIPData>('dxos:observability:ipdata'));
42
+ if (cachedData && cachedData.timestamp > Date.now() - IP_DATA_CACHE_TIMEOUT) {
43
+ return cachedData.data;
44
+ }
45
+
46
+ // Fetch data if not cached.
47
+ const IPDATA_API_KEY = config.get('runtime.app.env.DX_IPDATA_API_KEY');
48
+ if (IPDATA_API_KEY) {
49
+ const data = yield* HttpClientRequest.get(`https://api.ipdata.co?api-key=${IPDATA_API_KEY}`).pipe(
50
+ httpClientNoTrace.execute,
51
+ Effect.flatMap((res) => res.json),
52
+ Effect.flatMap(Schema.decodeUnknown(IPData)),
53
+ );
54
+
55
+ // Cache data.
56
+ yield* Effect.promise(() =>
57
+ localForage.setItem('dxos:observability:ipdata', {
58
+ data,
59
+ timestamp: Date.now(),
60
+ }),
61
+ );
62
+
63
+ return data;
64
+ }
65
+ });
66
+
67
+ /** Fetches IP geolocation data and sets city/region/country tags on the observability instance. */
68
+ export const provider =
69
+ (config: Config): DataProvider =>
70
+ (observability) =>
71
+ Effect.gen(function* () {
72
+ const ipData = yield* getIPData(config);
73
+ if (!ipData) {
74
+ return;
75
+ }
76
+
77
+ observability.setTags({
78
+ city: ipData.city,
79
+ region: ipData.region,
80
+ country: ipData.country,
81
+ latitude: ipData.latitude,
82
+ longitude: ipData.longitude,
83
+ });
84
+ }).pipe(
85
+ Effect.provide(FetchHttpClient.layer),
86
+ Effect.catchAll((err) =>
87
+ Effect.sync(() => {
88
+ log.verbose('ipdata fetch failed', { err });
89
+ }),
90
+ ),
91
+ );
@@ -0,0 +1,23 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import * as Duration from 'effect/Duration';
6
+ import * as Effect from 'effect/Effect';
7
+ import * as Fiber from 'effect/Fiber';
8
+ import * as Schedule from 'effect/Schedule';
9
+
10
+ import { type DataProvider } from '../observability';
11
+
12
+ export const provider: DataProvider = Effect.fn(function* (observability) {
13
+ if (typeof navigator !== 'undefined' && navigator.storage?.estimate) {
14
+ const action = Effect.gen(function* () {
15
+ const storageEstimate = yield* Effect.tryPromise(() => navigator.storage.estimate());
16
+ storageEstimate.usage && observability.metrics.gauge('storageUsage', storageEstimate.usage);
17
+ storageEstimate.quota && observability.metrics.gauge('storageQuota', storageEstimate.quota);
18
+ });
19
+
20
+ const fiber = action.pipe(Effect.repeat(Schedule.fixed(Duration.hours(1))), Effect.runFork);
21
+ return () => Effect.runSync(Fiber.interrupt(fiber));
22
+ }
23
+ });
@@ -0,0 +1,62 @@
1
+ //
2
+ // Copyright 2022 DXOS.org
3
+ //
4
+
5
+ // NOTE: localStorage is not available in web workers.
6
+ import * as localForage from 'localforage';
7
+
8
+ import { log } from '@dxos/log';
9
+ import { compositeKey } from '@dxos/util';
10
+
11
+ const OBSERVABILITY_DISABLED_KEY = 'observability-disabled';
12
+ const OBSERVABILITY_GROUP_KEY = 'observability-group';
13
+
14
+ /** No-op in browser contexts. */
15
+ export const showObservabilityBanner = () => {
16
+ log.warn('showObservabilityBanner is not supported in browser contexts.');
17
+ };
18
+
19
+ /**
20
+ * @param namespace - localForage key prefix used to scope the observability state in browser storage.
21
+ */
22
+ export const isObservabilityDisabled = async (namespace: string): Promise<boolean> => {
23
+ try {
24
+ return (await localForage.getItem(compositeKey(namespace, OBSERVABILITY_DISABLED_KEY))) === 'true';
25
+ } catch (err) {
26
+ log.catch('Failed to check if observability is disabled, assuming it is', err);
27
+ return true;
28
+ }
29
+ };
30
+
31
+ /**
32
+ * @param namespace - localForage key prefix used to scope the observability state in browser storage.
33
+ */
34
+ export const storeObservabilityDisabled = async (namespace: string, value: boolean) => {
35
+ try {
36
+ await localForage.setItem(compositeKey(namespace, OBSERVABILITY_DISABLED_KEY), String(value));
37
+ } catch (err) {
38
+ log.catch('Failed to store observability disabled', err);
39
+ }
40
+ };
41
+
42
+ /**
43
+ * @param namespace - localForage key prefix used to scope the observability state in browser storage.
44
+ */
45
+ export const getObservabilityGroup = async (namespace: string): Promise<string | undefined> => {
46
+ try {
47
+ return (await localForage.getItem(compositeKey(namespace, OBSERVABILITY_GROUP_KEY))) ?? undefined;
48
+ } catch (err) {
49
+ log.catch('Failed to get observability group', err);
50
+ }
51
+ };
52
+
53
+ /**
54
+ * @param namespace - localForage key prefix used to scope the observability state in browser storage.
55
+ */
56
+ export const storeObservabilityGroup = async (namespace: string, value: string) => {
57
+ try {
58
+ await localForage.setItem(compositeKey(namespace, OBSERVABILITY_GROUP_KEY), value);
59
+ } catch (err) {
60
+ log.catch('Failed to store observability group', err);
61
+ }
62
+ };
@@ -3,4 +3,3 @@
3
3
  //
4
4
 
5
5
  export * from './node';
6
- export * from './types';
@@ -0,0 +1,130 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { existsSync } from 'node:fs';
6
+ import { mkdir, readFile, rm, writeFile } from 'node:fs/promises';
7
+ import { tmpdir } from 'node:os';
8
+ import { join } from 'node:path';
9
+
10
+ import yaml from 'js-yaml';
11
+ import { validate as validateUuid } from 'uuid';
12
+ import { afterEach, beforeEach, describe, expect, test, vi } from 'vitest';
13
+
14
+ import {
15
+ type PersistentObservabilityState,
16
+ getObservabilityGroup,
17
+ isObservabilityDisabled,
18
+ showObservabilityBanner,
19
+ storeObservabilityDisabled,
20
+ storeObservabilityGroup,
21
+ } from './node';
22
+
23
+ let configDir: string;
24
+
25
+ beforeEach(async () => {
26
+ configDir = join(tmpdir(), `observability-test-${Date.now()}-${Math.random().toString(36).slice(2)}`);
27
+ // Clean env vars.
28
+ delete process.env.DX_DISABLE_OBSERVABILITY;
29
+ delete process.env.DX_OBSERVABILITY_GROUP;
30
+ });
31
+
32
+ afterEach(async () => {
33
+ if (existsSync(configDir)) {
34
+ await rm(configDir, { recursive: true, force: true });
35
+ }
36
+ });
37
+
38
+ describe('node storage', () => {
39
+ test('creates config directory if missing', async () => {
40
+ expect(existsSync(configDir)).toBe(false);
41
+ await isObservabilityDisabled(configDir);
42
+ expect(existsSync(configDir)).toBe(true);
43
+ });
44
+
45
+ test('initializes state with UUID and writes observability.yml', async () => {
46
+ await isObservabilityDisabled(configDir);
47
+ const content = await readFile(join(configDir, 'observability.yml'), 'utf-8');
48
+ const state = yaml.load(content) as PersistentObservabilityState;
49
+ expect(validateUuid(state.installationId)).toBe(true);
50
+ expect(state.disabled).toBe(false);
51
+ });
52
+
53
+ test('isObservabilityDisabled returns false by default', async () => {
54
+ const disabled = await isObservabilityDisabled(configDir);
55
+ expect(disabled).toBe(false);
56
+ });
57
+
58
+ test('storeObservabilityDisabled persists to file', async () => {
59
+ await storeObservabilityDisabled(configDir, true);
60
+ const content = await readFile(join(configDir, 'observability.yml'), 'utf-8');
61
+ const state = yaml.load(content) as PersistentObservabilityState;
62
+ expect(state.disabled).toBe(true);
63
+ });
64
+
65
+ test('isObservabilityDisabled reads persisted value', async () => {
66
+ await storeObservabilityDisabled(configDir, true);
67
+ const disabled = await isObservabilityDisabled(configDir);
68
+ expect(disabled).toBe(true);
69
+ });
70
+
71
+ test('getObservabilityGroup returns undefined by default', async () => {
72
+ const group = await getObservabilityGroup(configDir);
73
+ expect(group).toBeUndefined();
74
+ });
75
+
76
+ test('storeObservabilityGroup persists and reads back', async () => {
77
+ await storeObservabilityGroup(configDir, 'beta-testers');
78
+ const group = await getObservabilityGroup(configDir);
79
+ expect(group).toBe('beta-testers');
80
+ });
81
+
82
+ test('respects DX_DISABLE_OBSERVABILITY env var', async () => {
83
+ process.env.DX_DISABLE_OBSERVABILITY = 'true';
84
+ const disabled = await isObservabilityDisabled(configDir);
85
+ expect(disabled).toBe(true);
86
+ });
87
+
88
+ test('respects DX_OBSERVABILITY_GROUP env var on initial creation', async () => {
89
+ process.env.DX_OBSERVABILITY_GROUP = 'alpha';
90
+ const group = await getObservabilityGroup(configDir);
91
+ expect(group).toBe('alpha');
92
+ });
93
+
94
+ test('validates UUID in existing state file', async () => {
95
+ await mkdir(configDir, { recursive: true });
96
+ const state: PersistentObservabilityState = {
97
+ installationId: '550e8400-e29b-41d4-a716-446655440000',
98
+ disabled: false,
99
+ };
100
+ await writeFile(join(configDir, 'observability.yml'), yaml.dump(state), 'utf-8');
101
+ const disabled = await isObservabilityDisabled(configDir);
102
+ expect(disabled).toBe(false);
103
+ });
104
+
105
+ test('reinitializes if UUID is invalid', async () => {
106
+ await mkdir(configDir, { recursive: true });
107
+ const badState = { installationId: 'not-a-uuid', disabled: true };
108
+ await writeFile(join(configDir, 'observability.yml'), yaml.dump(badState), 'utf-8');
109
+
110
+ // Should reinitialize with a fresh UUID.
111
+ await isObservabilityDisabled(configDir);
112
+ const content = await readFile(join(configDir, 'observability.yml'), 'utf-8');
113
+ const state = yaml.load(content) as PersistentObservabilityState;
114
+ expect(validateUuid(state.installationId)).toBe(true);
115
+ // Reinitialized state uses env var defaults, not the persisted bad state.
116
+ expect(state.disabled).toBe(false);
117
+ });
118
+
119
+ test('showObservabilityBanner prints once then is silent', async () => {
120
+ await mkdir(configDir, { recursive: true });
121
+ const bannerCb = vi.fn();
122
+
123
+ await showObservabilityBanner(configDir, bannerCb);
124
+ expect(bannerCb).toHaveBeenCalledTimes(1);
125
+
126
+ bannerCb.mockClear();
127
+ await showObservabilityBanner(configDir, bannerCb);
128
+ expect(bannerCb).not.toHaveBeenCalled();
129
+ });
130
+ });