apcore-js 0.5.0 → 0.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (314) hide show
  1. package/dist/acl.d.ts +27 -0
  2. package/dist/acl.d.ts.map +1 -0
  3. package/dist/acl.js +175 -0
  4. package/dist/acl.js.map +1 -0
  5. package/dist/async-task.d.ts +90 -0
  6. package/dist/async-task.d.ts.map +1 -0
  7. package/dist/async-task.js +215 -0
  8. package/dist/async-task.js.map +1 -0
  9. package/dist/bindings.d.ts +12 -0
  10. package/dist/bindings.d.ts.map +1 -0
  11. package/dist/bindings.js +185 -0
  12. package/dist/bindings.js.map +1 -0
  13. package/dist/cancel.d.ts +14 -0
  14. package/dist/cancel.d.ts.map +1 -0
  15. package/dist/cancel.js +27 -0
  16. package/dist/cancel.js.map +1 -0
  17. package/dist/config.d.ts +9 -0
  18. package/dist/config.d.ts.map +1 -0
  19. package/dist/config.js +23 -0
  20. package/dist/config.js.map +1 -0
  21. package/dist/context.d.ts +50 -0
  22. package/dist/context.d.ts.map +1 -0
  23. package/dist/context.js +87 -0
  24. package/dist/context.js.map +1 -0
  25. package/dist/decorator.d.ts +57 -0
  26. package/dist/decorator.d.ts.map +1 -0
  27. package/dist/decorator.js +74 -0
  28. package/dist/decorator.js.map +1 -0
  29. package/dist/errors.d.ts +215 -0
  30. package/dist/errors.d.ts.map +1 -0
  31. package/dist/errors.js +246 -0
  32. package/dist/errors.js.map +1 -0
  33. package/dist/executor.d.ts +67 -0
  34. package/dist/executor.d.ts.map +1 -0
  35. package/dist/executor.js +372 -0
  36. package/dist/executor.js.map +1 -0
  37. package/dist/extensions.d.ts +58 -0
  38. package/dist/extensions.d.ts.map +1 -0
  39. package/dist/extensions.js +220 -0
  40. package/dist/extensions.js.map +1 -0
  41. package/{src/index.ts → dist/index.d.ts} +3 -62
  42. package/dist/index.d.ts.map +1 -0
  43. package/dist/index.js +43 -0
  44. package/dist/index.js.map +1 -0
  45. package/dist/middleware/adapters.d.ts +18 -0
  46. package/dist/middleware/adapters.d.ts.map +1 -0
  47. package/dist/middleware/adapters.js +25 -0
  48. package/dist/middleware/adapters.js.map +1 -0
  49. package/dist/middleware/base.d.ts +10 -0
  50. package/dist/middleware/base.d.ts.map +1 -0
  51. package/dist/middleware/base.js +15 -0
  52. package/dist/middleware/base.js.map +1 -0
  53. package/{src/middleware/index.ts → dist/middleware/index.d.ts} +1 -0
  54. package/dist/middleware/index.d.ts.map +1 -0
  55. package/dist/middleware/index.js +5 -0
  56. package/dist/middleware/index.js.map +1 -0
  57. package/dist/middleware/logging.d.ts +25 -0
  58. package/dist/middleware/logging.d.ts.map +1 -0
  59. package/dist/middleware/logging.js +64 -0
  60. package/dist/middleware/logging.js.map +1 -0
  61. package/dist/middleware/manager.d.ts +21 -0
  62. package/dist/middleware/manager.d.ts.map +1 -0
  63. package/dist/middleware/manager.js +77 -0
  64. package/dist/middleware/manager.js.map +1 -0
  65. package/dist/module.d.ts +31 -0
  66. package/dist/module.d.ts.map +1 -0
  67. package/dist/module.js +12 -0
  68. package/dist/module.js.map +1 -0
  69. package/dist/observability/context-logger.d.ts +54 -0
  70. package/dist/observability/context-logger.d.ts.map +1 -0
  71. package/dist/observability/context-logger.js +151 -0
  72. package/dist/observability/context-logger.js.map +1 -0
  73. package/{src/observability/index.ts → dist/observability/index.d.ts} +1 -0
  74. package/dist/observability/index.d.ts.map +1 -0
  75. package/dist/observability/index.js +4 -0
  76. package/dist/observability/index.js.map +1 -0
  77. package/dist/observability/metrics.d.ts +30 -0
  78. package/dist/observability/metrics.d.ts.map +1 -0
  79. package/dist/observability/metrics.js +177 -0
  80. package/dist/observability/metrics.js.map +1 -0
  81. package/dist/observability/tracing.d.ts +62 -0
  82. package/dist/observability/tracing.d.ts.map +1 -0
  83. package/dist/observability/tracing.js +184 -0
  84. package/dist/observability/tracing.js.map +1 -0
  85. package/dist/registry/dependencies.d.ts +6 -0
  86. package/dist/registry/dependencies.d.ts.map +1 -0
  87. package/dist/registry/dependencies.js +83 -0
  88. package/dist/registry/dependencies.js.map +1 -0
  89. package/dist/registry/entry-point.d.ts +6 -0
  90. package/dist/registry/entry-point.d.ts.map +1 -0
  91. package/dist/registry/entry-point.js +55 -0
  92. package/dist/registry/entry-point.js.map +1 -0
  93. package/{src/registry/index.ts → dist/registry/index.d.ts} +1 -0
  94. package/dist/registry/index.d.ts.map +1 -0
  95. package/dist/registry/index.js +8 -0
  96. package/dist/registry/index.js.map +1 -0
  97. package/dist/registry/metadata.d.ts +9 -0
  98. package/dist/registry/metadata.d.ts.map +1 -0
  99. package/dist/registry/metadata.js +105 -0
  100. package/dist/registry/metadata.js.map +1 -0
  101. package/dist/registry/registry.d.ts +102 -0
  102. package/dist/registry/registry.d.ts.map +1 -0
  103. package/dist/registry/registry.js +534 -0
  104. package/dist/registry/registry.js.map +1 -0
  105. package/dist/registry/scanner.d.ts +7 -0
  106. package/dist/registry/scanner.d.ts.map +1 -0
  107. package/dist/registry/scanner.js +164 -0
  108. package/dist/registry/scanner.js.map +1 -0
  109. package/dist/registry/schema-export.d.ts +9 -0
  110. package/dist/registry/schema-export.d.ts.map +1 -0
  111. package/dist/registry/schema-export.js +132 -0
  112. package/dist/registry/schema-export.js.map +1 -0
  113. package/dist/registry/types.d.ts +29 -0
  114. package/dist/registry/types.d.ts.map +1 -0
  115. package/dist/registry/types.js +5 -0
  116. package/dist/registry/types.js.map +1 -0
  117. package/dist/registry/validation.d.ts +9 -0
  118. package/dist/registry/validation.d.ts.map +1 -0
  119. package/dist/registry/validation.js +33 -0
  120. package/dist/registry/validation.js.map +1 -0
  121. package/dist/schema/annotations.d.ts +8 -0
  122. package/dist/schema/annotations.d.ts.map +1 -0
  123. package/dist/schema/annotations.js +52 -0
  124. package/dist/schema/annotations.js.map +1 -0
  125. package/dist/schema/exporter.d.ts +13 -0
  126. package/dist/schema/exporter.d.ts.map +1 -0
  127. package/dist/schema/exporter.js +71 -0
  128. package/dist/schema/exporter.js.map +1 -0
  129. package/dist/schema/index.d.ts +9 -0
  130. package/dist/schema/index.d.ts.map +1 -0
  131. package/{src/schema/index.ts → dist/schema/index.js} +1 -7
  132. package/dist/schema/index.js.map +1 -0
  133. package/dist/schema/loader.d.ts +30 -0
  134. package/dist/schema/loader.d.ts.map +1 -0
  135. package/dist/schema/loader.js +260 -0
  136. package/dist/schema/loader.js.map +1 -0
  137. package/dist/schema/ref-resolver.d.ts +19 -0
  138. package/dist/schema/ref-resolver.d.ts.map +1 -0
  139. package/dist/schema/ref-resolver.js +212 -0
  140. package/dist/schema/ref-resolver.js.map +1 -0
  141. package/dist/schema/strict.d.ts +7 -0
  142. package/dist/schema/strict.d.ts.map +1 -0
  143. package/dist/schema/strict.js +127 -0
  144. package/dist/schema/strict.js.map +1 -0
  145. package/dist/schema/types.d.ts +53 -0
  146. package/dist/schema/types.d.ts.map +1 -0
  147. package/dist/schema/types.js +31 -0
  148. package/dist/schema/types.js.map +1 -0
  149. package/dist/schema/validator.d.ts +16 -0
  150. package/dist/schema/validator.d.ts.map +1 -0
  151. package/dist/schema/validator.js +71 -0
  152. package/dist/schema/validator.js.map +1 -0
  153. package/dist/trace-context.d.ts +35 -0
  154. package/dist/trace-context.d.ts.map +1 -0
  155. package/dist/trace-context.js +86 -0
  156. package/dist/trace-context.js.map +1 -0
  157. package/dist/utils/index.d.ts +11 -0
  158. package/dist/utils/index.d.ts.map +1 -0
  159. package/dist/utils/index.js +32 -0
  160. package/dist/utils/index.js.map +1 -0
  161. package/dist/utils/pattern.d.ts +5 -0
  162. package/dist/utils/pattern.d.ts.map +1 -0
  163. package/dist/utils/pattern.js +31 -0
  164. package/dist/utils/pattern.js.map +1 -0
  165. package/package.json +8 -2
  166. package/.claude/settings.local.json +0 -12
  167. package/.github/workflows/ci.yml +0 -39
  168. package/.gitmessage +0 -60
  169. package/.pre-commit-config.yaml +0 -28
  170. package/CHANGELOG.md +0 -214
  171. package/CLAUDE.md +0 -68
  172. package/apcore-logo.svg +0 -79
  173. package/planning/acl-system/overview.md +0 -54
  174. package/planning/acl-system/plan.md +0 -92
  175. package/planning/acl-system/state.json +0 -76
  176. package/planning/acl-system/tasks/acl-core.md +0 -226
  177. package/planning/acl-system/tasks/acl-rule.md +0 -92
  178. package/planning/acl-system/tasks/conditional-rules.md +0 -259
  179. package/planning/acl-system/tasks/pattern-matching.md +0 -152
  180. package/planning/acl-system/tasks/yaml-loading.md +0 -271
  181. package/planning/core-executor/overview.md +0 -53
  182. package/planning/core-executor/plan.md +0 -88
  183. package/planning/core-executor/state.json +0 -76
  184. package/planning/core-executor/tasks/async-support.md +0 -106
  185. package/planning/core-executor/tasks/execution-pipeline.md +0 -113
  186. package/planning/core-executor/tasks/redaction.md +0 -85
  187. package/planning/core-executor/tasks/safety-checks.md +0 -65
  188. package/planning/core-executor/tasks/setup.md +0 -75
  189. package/planning/decorator-bindings/overview.md +0 -62
  190. package/planning/decorator-bindings/plan.md +0 -104
  191. package/planning/decorator-bindings/state.json +0 -87
  192. package/planning/decorator-bindings/tasks/binding-directory.md +0 -79
  193. package/planning/decorator-bindings/tasks/binding-loader.md +0 -148
  194. package/planning/decorator-bindings/tasks/explicit-schemas.md +0 -85
  195. package/planning/decorator-bindings/tasks/function-module.md +0 -127
  196. package/planning/decorator-bindings/tasks/module-factory.md +0 -89
  197. package/planning/decorator-bindings/tasks/schema-modes.md +0 -142
  198. package/planning/middleware-system/overview.md +0 -48
  199. package/planning/middleware-system/plan.md +0 -102
  200. package/planning/middleware-system/state.json +0 -65
  201. package/planning/middleware-system/tasks/adapters.md +0 -170
  202. package/planning/middleware-system/tasks/base.md +0 -115
  203. package/planning/middleware-system/tasks/logging-middleware.md +0 -304
  204. package/planning/middleware-system/tasks/manager.md +0 -313
  205. package/planning/observability/overview.md +0 -53
  206. package/planning/observability/plan.md +0 -119
  207. package/planning/observability/state.json +0 -98
  208. package/planning/observability/tasks/context-logger.md +0 -201
  209. package/planning/observability/tasks/exporters.md +0 -121
  210. package/planning/observability/tasks/metrics-collector.md +0 -162
  211. package/planning/observability/tasks/metrics-middleware.md +0 -141
  212. package/planning/observability/tasks/obs-logging-middleware.md +0 -179
  213. package/planning/observability/tasks/span-model.md +0 -120
  214. package/planning/observability/tasks/tracing-middleware.md +0 -179
  215. package/planning/overview.md +0 -81
  216. package/planning/registry-system/overview.md +0 -57
  217. package/planning/registry-system/plan.md +0 -114
  218. package/planning/registry-system/state.json +0 -109
  219. package/planning/registry-system/tasks/dependencies.md +0 -157
  220. package/planning/registry-system/tasks/entry-point.md +0 -148
  221. package/planning/registry-system/tasks/metadata.md +0 -198
  222. package/planning/registry-system/tasks/registry-core.md +0 -323
  223. package/planning/registry-system/tasks/scanner.md +0 -172
  224. package/planning/registry-system/tasks/schema-export.md +0 -261
  225. package/planning/registry-system/tasks/types.md +0 -124
  226. package/planning/registry-system/tasks/validation.md +0 -177
  227. package/planning/schema-system/overview.md +0 -56
  228. package/planning/schema-system/plan.md +0 -121
  229. package/planning/schema-system/state.json +0 -98
  230. package/planning/schema-system/tasks/exporter.md +0 -153
  231. package/planning/schema-system/tasks/loader.md +0 -106
  232. package/planning/schema-system/tasks/ref-resolver.md +0 -133
  233. package/planning/schema-system/tasks/strict-mode.md +0 -140
  234. package/planning/schema-system/tasks/typebox-generation.md +0 -133
  235. package/planning/schema-system/tasks/types-and-annotations.md +0 -160
  236. package/planning/schema-system/tasks/validator.md +0 -149
  237. package/src/acl.ts +0 -200
  238. package/src/async-task.ts +0 -267
  239. package/src/bindings.ts +0 -207
  240. package/src/cancel.ts +0 -32
  241. package/src/config.ts +0 -24
  242. package/src/context.ts +0 -160
  243. package/src/decorator.ts +0 -110
  244. package/src/errors.ts +0 -429
  245. package/src/executor.ts +0 -493
  246. package/src/extensions.ts +0 -265
  247. package/src/middleware/adapters.ts +0 -54
  248. package/src/middleware/base.ts +0 -33
  249. package/src/middleware/logging.ts +0 -103
  250. package/src/middleware/manager.ts +0 -105
  251. package/src/module.ts +0 -43
  252. package/src/observability/context-logger.ts +0 -203
  253. package/src/observability/metrics.ts +0 -214
  254. package/src/observability/tracing.ts +0 -252
  255. package/src/registry/dependencies.ts +0 -99
  256. package/src/registry/entry-point.ts +0 -64
  257. package/src/registry/metadata.ts +0 -111
  258. package/src/registry/registry.ts +0 -580
  259. package/src/registry/scanner.ts +0 -168
  260. package/src/registry/schema-export.ts +0 -181
  261. package/src/registry/types.ts +0 -32
  262. package/src/registry/validation.ts +0 -38
  263. package/src/schema/annotations.ts +0 -68
  264. package/src/schema/exporter.ts +0 -90
  265. package/src/schema/loader.ts +0 -273
  266. package/src/schema/ref-resolver.ts +0 -244
  267. package/src/schema/strict.ts +0 -136
  268. package/src/schema/types.ts +0 -73
  269. package/src/schema/validator.ts +0 -82
  270. package/src/trace-context.ts +0 -102
  271. package/src/utils/index.ts +0 -5
  272. package/src/utils/pattern.ts +0 -30
  273. package/tests/async-task.test.ts +0 -335
  274. package/tests/helpers.ts +0 -30
  275. package/tests/integration/test-acl-safety.test.ts +0 -269
  276. package/tests/integration/test-binding-executor.test.ts +0 -194
  277. package/tests/integration/test-e2e-flow.test.ts +0 -117
  278. package/tests/integration/test-error-propagation.test.ts +0 -259
  279. package/tests/integration/test-middleware-chain.test.ts +0 -120
  280. package/tests/integration/test-observability-integration.test.ts +0 -438
  281. package/tests/observability/test-context-logger.test.ts +0 -123
  282. package/tests/observability/test-metrics.test.ts +0 -186
  283. package/tests/observability/test-tracing.test.ts +0 -303
  284. package/tests/registry/test-dependencies.test.ts +0 -70
  285. package/tests/registry/test-entry-point.test.ts +0 -133
  286. package/tests/registry/test-metadata.test.ts +0 -265
  287. package/tests/registry/test-registry.test.ts +0 -1397
  288. package/tests/registry/test-scanner.test.ts +0 -257
  289. package/tests/registry/test-schema-export.test.ts +0 -355
  290. package/tests/registry/test-validation.test.ts +0 -75
  291. package/tests/schema/test-annotations.test.ts +0 -137
  292. package/tests/schema/test-exporter.test.ts +0 -172
  293. package/tests/schema/test-loader.test.ts +0 -461
  294. package/tests/schema/test-ref-resolver.test.ts +0 -530
  295. package/tests/schema/test-strict.test.ts +0 -348
  296. package/tests/schema/test-validator.test.ts +0 -64
  297. package/tests/test-acl.test.ts +0 -423
  298. package/tests/test-bindings.test.ts +0 -227
  299. package/tests/test-cancel.test.ts +0 -71
  300. package/tests/test-config.test.ts +0 -76
  301. package/tests/test-context.test.ts +0 -266
  302. package/tests/test-decorator.test.ts +0 -173
  303. package/tests/test-errors.test.ts +0 -647
  304. package/tests/test-executor-stream.test.ts +0 -208
  305. package/tests/test-executor.test.ts +0 -252
  306. package/tests/test-extensions.test.ts +0 -310
  307. package/tests/test-logging-middleware.test.ts +0 -150
  308. package/tests/test-middleware-manager.test.ts +0 -185
  309. package/tests/test-middleware.test.ts +0 -86
  310. package/tests/test-trace-context.test.ts +0 -251
  311. package/tests/utils/test-pattern.test.ts +0 -109
  312. package/tsconfig.build.json +0 -8
  313. package/tsconfig.json +0 -20
  314. package/vitest.config.ts +0 -18
@@ -1,186 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import { MetricsCollector, MetricsMiddleware } from '../../src/observability/metrics.js';
3
- import { Context } from '../../src/context.js';
4
- import { ModuleError } from '../../src/errors.js';
5
-
6
- describe('MetricsCollector', () => {
7
- it('increments counters', () => {
8
- const collector = new MetricsCollector();
9
- collector.increment('test_counter', { method: 'GET' });
10
- collector.increment('test_counter', { method: 'GET' });
11
- const snap = collector.snapshot();
12
- expect((snap['counters'] as Record<string, number>)['test_counter|method=GET']).toBe(2);
13
- });
14
-
15
- it('observes histogram values', () => {
16
- const collector = new MetricsCollector();
17
- collector.observe('test_hist', { module: 'a' }, 0.05);
18
- collector.observe('test_hist', { module: 'a' }, 0.5);
19
- const snap = collector.snapshot();
20
- const hists = snap['histograms'] as Record<string, Record<string, number>>;
21
- expect(hists['sums']['test_hist|module=a']).toBeCloseTo(0.55);
22
- expect(hists['counts']['test_hist|module=a']).toBe(2);
23
- });
24
-
25
- it('convenience methods work', () => {
26
- const collector = new MetricsCollector();
27
- collector.incrementCalls('mod.a', 'success');
28
- collector.incrementErrors('mod.a', 'TIMEOUT');
29
- collector.observeDuration('mod.a', 0.1);
30
-
31
- const snap = collector.snapshot();
32
- const counters = snap['counters'] as Record<string, number>;
33
- expect(counters['apcore_module_calls_total|module_id=mod.a,status=success']).toBe(1);
34
- expect(counters['apcore_module_errors_total|error_code=TIMEOUT,module_id=mod.a']).toBe(1);
35
- });
36
-
37
- it('reset clears all data', () => {
38
- const collector = new MetricsCollector();
39
- collector.incrementCalls('mod.a', 'success');
40
- collector.observeDuration('mod.a', 0.1);
41
- collector.reset();
42
- const snap = collector.snapshot();
43
- expect(Object.keys(snap['counters'] as Record<string, unknown>)).toHaveLength(0);
44
- });
45
-
46
- it('snapshot returns counters and histogram sub-keys', () => {
47
- const collector = new MetricsCollector();
48
- collector.increment('req_total', { status: '200' });
49
- collector.observe('req_duration', { route: '/health' }, 0.01);
50
- const snap = collector.snapshot();
51
- expect(snap).toHaveProperty('counters');
52
- expect(snap).toHaveProperty('histograms');
53
- const hists = snap['histograms'] as Record<string, unknown>;
54
- expect(hists).toHaveProperty('sums');
55
- expect(hists).toHaveProperty('counts');
56
- expect(hists).toHaveProperty('buckets');
57
- });
58
-
59
- it('exportPrometheus produces valid format', () => {
60
- const collector = new MetricsCollector();
61
- collector.incrementCalls('mod.a', 'success');
62
- collector.observeDuration('mod.a', 0.05);
63
- const output = collector.exportPrometheus();
64
- expect(output).toContain('# HELP');
65
- expect(output).toContain('# TYPE');
66
- expect(output).toContain('apcore_module_calls_total');
67
- expect(output).toContain('apcore_module_duration_seconds');
68
- });
69
-
70
- it('empty collector returns empty prometheus string', () => {
71
- const collector = new MetricsCollector();
72
- expect(collector.exportPrometheus()).toBe('');
73
- });
74
-
75
- it('accepts custom buckets and uses them for observations', () => {
76
- const collector = new MetricsCollector([0.1, 0.5, 1.0]);
77
- collector.observe('custom_hist', { op: 'read' }, 0.3);
78
- const snap = collector.snapshot();
79
- const hists = snap['histograms'] as Record<string, Record<string, number>>;
80
- // value 0.3 falls in the 0.5 bucket but not 0.1
81
- expect(hists['buckets']['custom_hist|op=read|0.1']).toBeUndefined();
82
- expect(hists['buckets']['custom_hist|op=read|0.5']).toBe(1);
83
- expect(hists['buckets']['custom_hist|op=read|Inf']).toBe(1);
84
- });
85
-
86
- it('exportPrometheus omits label braces when metric has no labels', () => {
87
- const collector = new MetricsCollector();
88
- // increment with empty labels so parseLabels receives '' and formatLabels receives {}
89
- collector.increment('no_label_counter', {});
90
- const output = collector.exportPrometheus();
91
- expect(output).toContain('no_label_counter 1');
92
- expect(output).not.toContain('no_label_counter{');
93
- });
94
- });
95
-
96
- describe('MetricsMiddleware', () => {
97
- it('records call metrics on success', () => {
98
- const collector = new MetricsCollector();
99
- const mw = new MetricsMiddleware(collector);
100
- const ctx = Context.create();
101
-
102
- mw.before('mod.a', {}, ctx);
103
- mw.after('mod.a', {}, { result: 'ok' }, ctx);
104
-
105
- const snap = collector.snapshot();
106
- const counters = snap['counters'] as Record<string, number>;
107
- expect(counters['apcore_module_calls_total|module_id=mod.a,status=success']).toBe(1);
108
- });
109
-
110
- it('records error metrics on failure with plain Error', () => {
111
- const collector = new MetricsCollector();
112
- const mw = new MetricsMiddleware(collector);
113
- const ctx = Context.create();
114
-
115
- mw.before('mod.a', {}, ctx);
116
- mw.onError('mod.a', {}, new Error('boom'), ctx);
117
-
118
- const snap = collector.snapshot();
119
- const counters = snap['counters'] as Record<string, number>;
120
- expect(counters['apcore_module_calls_total|module_id=mod.a,status=error']).toBe(1);
121
- expect(counters['apcore_module_errors_total|error_code=Error,module_id=mod.a']).toBe(1);
122
- });
123
-
124
- it('records error code from ModuleError.code instead of constructor name', () => {
125
- const collector = new MetricsCollector();
126
- const mw = new MetricsMiddleware(collector);
127
- const ctx = Context.create();
128
-
129
- mw.before('mod.b', {}, ctx);
130
- mw.onError('mod.b', {}, new ModuleError('CUSTOM_CODE', 'something went wrong'), ctx);
131
-
132
- const snap = collector.snapshot();
133
- const counters = snap['counters'] as Record<string, number>;
134
- expect(counters['apcore_module_errors_total|error_code=CUSTOM_CODE,module_id=mod.b']).toBe(1);
135
- });
136
-
137
- it('after() returns null without recording metrics when starts is undefined', () => {
138
- const collector = new MetricsCollector();
139
- const mw = new MetricsMiddleware(collector);
140
- const ctx = new Context('trace-id', null, [], null, null);
141
-
142
- const result = mw.after('mod.a', {}, { result: 'ok' }, ctx);
143
-
144
- expect(result).toBeNull();
145
- const snap = collector.snapshot();
146
- expect(Object.keys(snap['counters'] as Record<string, unknown>)).toHaveLength(0);
147
- });
148
-
149
- it('after() returns null without recording metrics when starts array is empty', () => {
150
- const collector = new MetricsCollector();
151
- const mw = new MetricsMiddleware(collector);
152
- const ctx = new Context('trace-id', null, [], null, null);
153
- ctx.data['_metrics_starts'] = [];
154
-
155
- const result = mw.after('mod.a', {}, { result: 'ok' }, ctx);
156
-
157
- expect(result).toBeNull();
158
- const snap = collector.snapshot();
159
- expect(Object.keys(snap['counters'] as Record<string, unknown>)).toHaveLength(0);
160
- });
161
-
162
- it('onError() returns null without recording metrics when starts is undefined', () => {
163
- const collector = new MetricsCollector();
164
- const mw = new MetricsMiddleware(collector);
165
- const ctx = new Context('trace-id', null, [], null, null);
166
-
167
- const result = mw.onError('mod.a', {}, new Error('boom'), ctx);
168
-
169
- expect(result).toBeNull();
170
- const snap = collector.snapshot();
171
- expect(Object.keys(snap['counters'] as Record<string, unknown>)).toHaveLength(0);
172
- });
173
-
174
- it('onError() returns null without recording metrics when starts array is empty', () => {
175
- const collector = new MetricsCollector();
176
- const mw = new MetricsMiddleware(collector);
177
- const ctx = new Context('trace-id', null, [], null, null);
178
- ctx.data['_metrics_starts'] = [];
179
-
180
- const result = mw.onError('mod.a', {}, new Error('boom'), ctx);
181
-
182
- expect(result).toBeNull();
183
- const snap = collector.snapshot();
184
- expect(Object.keys(snap['counters'] as Record<string, unknown>)).toHaveLength(0);
185
- });
186
- });
@@ -1,303 +0,0 @@
1
- import { describe, it, expect, vi, afterEach } from 'vitest';
2
- import { Context } from '../../src/context.js';
3
- import {
4
- createSpan,
5
- InMemoryExporter,
6
- OTLPExporter,
7
- StdoutExporter,
8
- TracingMiddleware,
9
- } from '../../src/observability/tracing.js';
10
-
11
- describe('Span', () => {
12
- it('createSpan creates span with defaults', () => {
13
- const span = createSpan({
14
- traceId: 'trace-1',
15
- name: 'test.span',
16
- startTime: 100,
17
- });
18
- expect(span.traceId).toBe('trace-1');
19
- expect(span.name).toBe('test.span');
20
- expect(span.startTime).toBe(100);
21
- expect(span.spanId).toBeDefined();
22
- expect(span.parentSpanId).toBeNull();
23
- expect(span.status).toBe('ok');
24
- expect(span.events).toEqual([]);
25
- });
26
- });
27
-
28
- describe('InMemoryExporter', () => {
29
- it('collects and retrieves spans', () => {
30
- const exporter = new InMemoryExporter();
31
- const span = createSpan({ traceId: 't1', name: 'test', startTime: 0 });
32
- exporter.export(span);
33
- expect(exporter.getSpans()).toHaveLength(1);
34
- expect(exporter.getSpans()[0].traceId).toBe('t1');
35
- });
36
-
37
- it('respects max_spans limit', () => {
38
- const exporter = new InMemoryExporter(3);
39
- for (let i = 0; i < 5; i++) {
40
- exporter.export(createSpan({ traceId: `t${i}`, name: 'test', startTime: i }));
41
- }
42
- const spans = exporter.getSpans();
43
- expect(spans).toHaveLength(3);
44
- expect(spans[0].traceId).toBe('t2');
45
- });
46
-
47
- it('clear removes all spans', () => {
48
- const exporter = new InMemoryExporter();
49
- exporter.export(createSpan({ traceId: 't1', name: 'test', startTime: 0 }));
50
- exporter.clear();
51
- expect(exporter.getSpans()).toHaveLength(0);
52
- });
53
- });
54
-
55
- describe('TracingMiddleware', () => {
56
- it('creates and exports spans on success', () => {
57
- const exporter = new InMemoryExporter();
58
- const mw = new TracingMiddleware(exporter);
59
- const ctx = Context.create();
60
-
61
- mw.before('mod.a', {}, ctx);
62
- mw.after('mod.a', {}, { result: 'ok' }, ctx);
63
-
64
- const spans = exporter.getSpans();
65
- expect(spans).toHaveLength(1);
66
- expect(spans[0].name).toBe('apcore.module.execute');
67
- expect(spans[0].status).toBe('ok');
68
- expect(spans[0].attributes['moduleId']).toBe('mod.a');
69
- });
70
-
71
- it('creates error spans', () => {
72
- const exporter = new InMemoryExporter();
73
- const mw = new TracingMiddleware(exporter);
74
- const ctx = Context.create();
75
-
76
- mw.before('mod.err', {}, ctx);
77
- mw.onError('mod.err', {}, new Error('fail'), ctx);
78
-
79
- const spans = exporter.getSpans();
80
- expect(spans).toHaveLength(1);
81
- expect(spans[0].status).toBe('error');
82
- });
83
-
84
- it('supports nested spans with parent chain', () => {
85
- const exporter = new InMemoryExporter();
86
- const mw = new TracingMiddleware(exporter);
87
- const ctx = Context.create();
88
-
89
- mw.before('mod.outer', {}, ctx);
90
- mw.before('mod.inner', {}, ctx);
91
- mw.after('mod.inner', {}, {}, ctx);
92
- mw.after('mod.outer', {}, {}, ctx);
93
-
94
- const spans = exporter.getSpans();
95
- expect(spans).toHaveLength(2);
96
- // inner span has outer as parent
97
- expect(spans[0].parentSpanId).toBe(spans[1].spanId);
98
- });
99
-
100
- it('off strategy does not export', () => {
101
- const exporter = new InMemoryExporter();
102
- const mw = new TracingMiddleware(exporter, 1.0, 'off');
103
- const ctx = Context.create();
104
-
105
- mw.before('mod.a', {}, ctx);
106
- mw.after('mod.a', {}, {}, ctx);
107
-
108
- expect(exporter.getSpans()).toHaveLength(0);
109
- });
110
-
111
- it('error_first exports errors even when not sampled', () => {
112
- const exporter = new InMemoryExporter();
113
- const mw = new TracingMiddleware(exporter, 0.0, 'error_first');
114
- const ctx = Context.create();
115
-
116
- mw.before('mod.a', {}, ctx);
117
- mw.onError('mod.a', {}, new Error('fail'), ctx);
118
-
119
- expect(exporter.getSpans()).toHaveLength(1);
120
- });
121
-
122
- it('throws on invalid sampling rate', () => {
123
- const exporter = new InMemoryExporter();
124
- expect(() => new TracingMiddleware(exporter, -0.1)).toThrow();
125
- expect(() => new TracingMiddleware(exporter, 1.5)).toThrow();
126
- });
127
-
128
- it('throws on invalid sampling strategy', () => {
129
- const exporter = new InMemoryExporter();
130
- expect(() => new TracingMiddleware(exporter, 1.0, 'invalid')).toThrow();
131
- });
132
-
133
- describe('setExporter', () => {
134
- it('rejects null exporter', () => {
135
- const mw = new TracingMiddleware(new InMemoryExporter());
136
- expect(() => mw.setExporter(null as any)).toThrow('exporter must implement SpanExporter interface');
137
- });
138
-
139
- it('rejects object without export method', () => {
140
- const mw = new TracingMiddleware(new InMemoryExporter());
141
- expect(() => mw.setExporter({} as any)).toThrow('exporter must implement SpanExporter interface');
142
- });
143
-
144
- it('accepts valid exporter', () => {
145
- const mw = new TracingMiddleware(new InMemoryExporter());
146
- const newExporter = { export: () => {} };
147
- expect(() => mw.setExporter(newExporter)).not.toThrow();
148
- });
149
- });
150
- });
151
-
152
- describe('OTLPExporter', () => {
153
- const originalFetch = globalThis.fetch;
154
-
155
- afterEach(() => {
156
- globalThis.fetch = originalFetch;
157
- });
158
-
159
- function makeTestSpan() {
160
- return createSpan({
161
- traceId: 'trace-abc',
162
- name: 'test.op',
163
- startTime: 1000,
164
- spanId: 'span-123',
165
- parentSpanId: 'parent-456',
166
- attributes: { foo: 'bar' },
167
- });
168
- }
169
-
170
- it('calls fetch with correct URL and payload shape', async () => {
171
- const mockFetch = vi.fn().mockResolvedValue(new Response('ok'));
172
- globalThis.fetch = mockFetch;
173
-
174
- const exporter = new OTLPExporter();
175
- const span = makeTestSpan();
176
- span.endTime = 2000;
177
- span.status = 'ok';
178
- exporter.export(span);
179
-
180
- // Wait for the fire-and-forget promise
181
- await vi.waitFor(() => expect(mockFetch).toHaveBeenCalledTimes(1));
182
-
183
- const [url, options] = mockFetch.mock.calls[0];
184
- expect(url).toBe('http://localhost:4318/v1/traces');
185
- expect(options.method).toBe('POST');
186
- expect(options.headers['Content-Type']).toBe('application/json');
187
-
188
- const body = JSON.parse(options.body);
189
- expect(body.resourceSpans).toHaveLength(1);
190
- expect(body.resourceSpans[0].resource.attributes[0].key).toBe('service.name');
191
- expect(body.resourceSpans[0].resource.attributes[0].value.stringValue).toBe('apcore');
192
-
193
- const exportedSpan = body.resourceSpans[0].scopeSpans[0].spans[0];
194
- expect(exportedSpan.traceId).toBe('trace-abc');
195
- expect(exportedSpan.spanId).toBe('span-123');
196
- expect(exportedSpan.parentSpanId).toBe('parent-456');
197
- expect(exportedSpan.name).toBe('test.op');
198
- expect(exportedSpan.status.code).toBe(1);
199
- expect(exportedSpan.attributes).toEqual([
200
- { key: 'foo', value: { stringValue: 'bar' } },
201
- ]);
202
- });
203
-
204
- it('uses default endpoint when none provided', () => {
205
- const mockFetch = vi.fn().mockResolvedValue(new Response('ok'));
206
- globalThis.fetch = mockFetch;
207
-
208
- const exporter = new OTLPExporter();
209
- exporter.export(makeTestSpan());
210
-
211
- expect(mockFetch).toHaveBeenCalledTimes(1);
212
- expect(mockFetch.mock.calls[0][0]).toBe('http://localhost:4318/v1/traces');
213
- });
214
-
215
- it('uses custom endpoint when provided', () => {
216
- const mockFetch = vi.fn().mockResolvedValue(new Response('ok'));
217
- globalThis.fetch = mockFetch;
218
-
219
- const exporter = new OTLPExporter({ endpoint: 'http://custom:9999/traces' });
220
- exporter.export(makeTestSpan());
221
-
222
- expect(mockFetch).toHaveBeenCalledTimes(1);
223
- expect(mockFetch.mock.calls[0][0]).toBe('http://custom:9999/traces');
224
- });
225
-
226
- it('includes custom headers in fetch call', () => {
227
- const mockFetch = vi.fn().mockResolvedValue(new Response('ok'));
228
- globalThis.fetch = mockFetch;
229
-
230
- const exporter = new OTLPExporter({
231
- headers: { 'X-Api-Key': 'secret-key', 'X-Custom': 'value' },
232
- });
233
- exporter.export(makeTestSpan());
234
-
235
- const headers = mockFetch.mock.calls[0][1].headers;
236
- expect(headers['X-Api-Key']).toBe('secret-key');
237
- expect(headers['X-Custom']).toBe('value');
238
- expect(headers['Content-Type']).toBe('application/json');
239
- });
240
-
241
- it('silently catches network errors', async () => {
242
- const mockFetch = vi.fn().mockRejectedValue(new Error('network down'));
243
- globalThis.fetch = mockFetch;
244
-
245
- const exporter = new OTLPExporter();
246
- // Should not throw
247
- expect(() => exporter.export(makeTestSpan())).not.toThrow();
248
-
249
- // Wait for the rejected promise to be caught
250
- await vi.waitFor(() => expect(mockFetch).toHaveBeenCalledTimes(1));
251
- });
252
-
253
- it('converts timestamps to nanoseconds in OTLP payload', async () => {
254
- const mockFetch = vi.fn().mockResolvedValue(new Response('ok'));
255
- globalThis.fetch = mockFetch;
256
-
257
- const exporter = new OTLPExporter();
258
- const span = createSpan({
259
- traceId: 'trace-ns',
260
- name: 'ns.test',
261
- startTime: 1700000000.123,
262
- spanId: 'span-ns',
263
- });
264
- span.endTime = 1700000002.456;
265
- exporter.export(span);
266
-
267
- await vi.waitFor(() => expect(mockFetch).toHaveBeenCalledTimes(1));
268
-
269
- const body = JSON.parse(mockFetch.mock.calls[0][1].body);
270
- const exportedSpan = body.resourceSpans[0].scopeSpans[0].spans[0];
271
-
272
- // startTime 1700000000.123 * 1_000_000_000 = 1700000000123000000
273
- const expectedStartNano = String(Math.round(1700000000.123 * 1_000_000_000));
274
- expect(exportedSpan.startTimeUnixNano).toBe(expectedStartNano);
275
-
276
- // endTime 1700000002.456 * 1_000_000_000 = 1700000002456000000
277
- const expectedEndNano = String(Math.round(1700000002.456 * 1_000_000_000));
278
- expect(exportedSpan.endTimeUnixNano).toBe(expectedEndNano);
279
- });
280
-
281
- it('omits endTimeUnixNano when endTime is null', async () => {
282
- const mockFetch = vi.fn().mockResolvedValue(new Response('ok'));
283
- globalThis.fetch = mockFetch;
284
-
285
- const exporter = new OTLPExporter();
286
- const span = createSpan({
287
- traceId: 'trace-no-end',
288
- name: 'no.end',
289
- startTime: 1700000000,
290
- spanId: 'span-no-end',
291
- });
292
- // endTime is null by default
293
- exporter.export(span);
294
-
295
- await vi.waitFor(() => expect(mockFetch).toHaveBeenCalledTimes(1));
296
-
297
- const body = JSON.parse(mockFetch.mock.calls[0][1].body);
298
- const exportedSpan = body.resourceSpans[0].scopeSpans[0].spans[0];
299
-
300
- expect(exportedSpan.startTimeUnixNano).toBe(String(Math.round(1700000000 * 1_000_000_000)));
301
- expect(exportedSpan.endTimeUnixNano).toBeUndefined();
302
- });
303
- });
@@ -1,70 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import { resolveDependencies } from '../../src/registry/dependencies.js';
3
- import { CircularDependencyError, ModuleLoadError } from '../../src/errors.js';
4
-
5
- describe('resolveDependencies', () => {
6
- it('returns empty for empty input', () => {
7
- expect(resolveDependencies([])).toEqual([]);
8
- });
9
-
10
- it('returns single module', () => {
11
- const result = resolveDependencies([['mod.a', []]]);
12
- expect(result).toEqual(['mod.a']);
13
- });
14
-
15
- it('resolves linear dependency chain', () => {
16
- const modules: Array<[string, Array<{ moduleId: string; optional: boolean; version: string | null }>]> = [
17
- ['mod.b', [{ moduleId: 'mod.a', optional: false, version: null }]],
18
- ['mod.a', []],
19
- ];
20
- const result = resolveDependencies(modules);
21
- expect(result.indexOf('mod.a')).toBeLessThan(result.indexOf('mod.b'));
22
- });
23
-
24
- it('resolves diamond dependency', () => {
25
- const modules: Array<[string, Array<{ moduleId: string; optional: boolean; version: string | null }>]> = [
26
- ['mod.d', [{ moduleId: 'mod.b', optional: false, version: null }, { moduleId: 'mod.c', optional: false, version: null }]],
27
- ['mod.b', [{ moduleId: 'mod.a', optional: false, version: null }]],
28
- ['mod.c', [{ moduleId: 'mod.a', optional: false, version: null }]],
29
- ['mod.a', []],
30
- ];
31
- const result = resolveDependencies(modules);
32
- expect(result.indexOf('mod.a')).toBeLessThan(result.indexOf('mod.b'));
33
- expect(result.indexOf('mod.a')).toBeLessThan(result.indexOf('mod.c'));
34
- expect(result.indexOf('mod.b')).toBeLessThan(result.indexOf('mod.d'));
35
- expect(result.indexOf('mod.c')).toBeLessThan(result.indexOf('mod.d'));
36
- });
37
-
38
- it('throws CircularDependencyError on cycle', () => {
39
- const modules: Array<[string, Array<{ moduleId: string; optional: boolean; version: string | null }>]> = [
40
- ['mod.a', [{ moduleId: 'mod.b', optional: false, version: null }]],
41
- ['mod.b', [{ moduleId: 'mod.a', optional: false, version: null }]],
42
- ];
43
- expect(() => resolveDependencies(modules)).toThrow(CircularDependencyError);
44
- });
45
-
46
- it('throws ModuleLoadError for missing required dependency', () => {
47
- const modules: Array<[string, Array<{ moduleId: string; optional: boolean; version: string | null }>]> = [
48
- ['mod.a', [{ moduleId: 'mod.missing', optional: false, version: null }]],
49
- ];
50
- expect(() => resolveDependencies(modules)).toThrow(ModuleLoadError);
51
- });
52
-
53
- it('skips optional missing dependencies', () => {
54
- const modules: Array<[string, Array<{ moduleId: string; optional: boolean; version: string | null }>]> = [
55
- ['mod.a', [{ moduleId: 'mod.missing', optional: true, version: null }]],
56
- ];
57
- const result = resolveDependencies(modules);
58
- expect(result).toEqual(['mod.a']);
59
- });
60
-
61
- it('independent modules in deterministic order', () => {
62
- const modules: Array<[string, Array<{ moduleId: string; optional: boolean; version: string | null }>]> = [
63
- ['mod.c', []],
64
- ['mod.a', []],
65
- ['mod.b', []],
66
- ];
67
- const result = resolveDependencies(modules);
68
- expect(result).toEqual(['mod.a', 'mod.b', 'mod.c']);
69
- });
70
- });