apcore-js 0.5.0 → 0.7.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 (319) hide show
  1. package/README.md +1 -1
  2. package/dist/acl.d.ts +27 -0
  3. package/dist/acl.d.ts.map +1 -0
  4. package/dist/acl.js +175 -0
  5. package/dist/acl.js.map +1 -0
  6. package/dist/approval.d.ts +85 -0
  7. package/dist/approval.d.ts.map +1 -0
  8. package/dist/approval.js +73 -0
  9. package/dist/approval.js.map +1 -0
  10. package/dist/async-task.d.ts +90 -0
  11. package/dist/async-task.d.ts.map +1 -0
  12. package/dist/async-task.js +215 -0
  13. package/dist/async-task.js.map +1 -0
  14. package/dist/bindings.d.ts +12 -0
  15. package/dist/bindings.d.ts.map +1 -0
  16. package/dist/bindings.js +185 -0
  17. package/dist/bindings.js.map +1 -0
  18. package/dist/cancel.d.ts +14 -0
  19. package/dist/cancel.d.ts.map +1 -0
  20. package/dist/cancel.js +27 -0
  21. package/dist/cancel.js.map +1 -0
  22. package/dist/config.d.ts +9 -0
  23. package/dist/config.d.ts.map +1 -0
  24. package/dist/config.js +23 -0
  25. package/dist/config.js.map +1 -0
  26. package/dist/context.d.ts +50 -0
  27. package/dist/context.d.ts.map +1 -0
  28. package/dist/context.js +87 -0
  29. package/dist/context.js.map +1 -0
  30. package/dist/decorator.d.ts +57 -0
  31. package/dist/decorator.d.ts.map +1 -0
  32. package/dist/decorator.js +74 -0
  33. package/dist/decorator.js.map +1 -0
  34. package/dist/errors.d.ts +204 -0
  35. package/dist/errors.d.ts.map +1 -0
  36. package/dist/errors.js +364 -0
  37. package/dist/errors.js.map +1 -0
  38. package/dist/executor.d.ts +82 -0
  39. package/dist/executor.d.ts.map +1 -0
  40. package/dist/executor.js +489 -0
  41. package/dist/executor.js.map +1 -0
  42. package/dist/extensions.d.ts +58 -0
  43. package/dist/extensions.d.ts.map +1 -0
  44. package/dist/extensions.js +239 -0
  45. package/dist/extensions.js.map +1 -0
  46. package/{src/index.ts → dist/index.d.ts} +6 -63
  47. package/dist/index.d.ts.map +1 -0
  48. package/dist/index.js +45 -0
  49. package/dist/index.js.map +1 -0
  50. package/dist/middleware/adapters.d.ts +18 -0
  51. package/dist/middleware/adapters.d.ts.map +1 -0
  52. package/dist/middleware/adapters.js +25 -0
  53. package/dist/middleware/adapters.js.map +1 -0
  54. package/dist/middleware/base.d.ts +10 -0
  55. package/dist/middleware/base.d.ts.map +1 -0
  56. package/dist/middleware/base.js +15 -0
  57. package/dist/middleware/base.js.map +1 -0
  58. package/{src/middleware/index.ts → dist/middleware/index.d.ts} +1 -0
  59. package/dist/middleware/index.d.ts.map +1 -0
  60. package/dist/middleware/index.js +5 -0
  61. package/dist/middleware/index.js.map +1 -0
  62. package/dist/middleware/logging.d.ts +25 -0
  63. package/dist/middleware/logging.d.ts.map +1 -0
  64. package/dist/middleware/logging.js +64 -0
  65. package/dist/middleware/logging.js.map +1 -0
  66. package/dist/middleware/manager.d.ts +21 -0
  67. package/dist/middleware/manager.d.ts.map +1 -0
  68. package/dist/middleware/manager.js +77 -0
  69. package/dist/middleware/manager.js.map +1 -0
  70. package/dist/module.d.ts +31 -0
  71. package/dist/module.d.ts.map +1 -0
  72. package/dist/module.js +12 -0
  73. package/dist/module.js.map +1 -0
  74. package/dist/observability/context-logger.d.ts +54 -0
  75. package/dist/observability/context-logger.d.ts.map +1 -0
  76. package/dist/observability/context-logger.js +151 -0
  77. package/dist/observability/context-logger.js.map +1 -0
  78. package/{src/observability/index.ts → dist/observability/index.d.ts} +1 -0
  79. package/dist/observability/index.d.ts.map +1 -0
  80. package/dist/observability/index.js +4 -0
  81. package/dist/observability/index.js.map +1 -0
  82. package/dist/observability/metrics.d.ts +30 -0
  83. package/dist/observability/metrics.d.ts.map +1 -0
  84. package/dist/observability/metrics.js +177 -0
  85. package/dist/observability/metrics.js.map +1 -0
  86. package/dist/observability/tracing.d.ts +62 -0
  87. package/dist/observability/tracing.d.ts.map +1 -0
  88. package/dist/observability/tracing.js +184 -0
  89. package/dist/observability/tracing.js.map +1 -0
  90. package/dist/registry/dependencies.d.ts +6 -0
  91. package/dist/registry/dependencies.d.ts.map +1 -0
  92. package/dist/registry/dependencies.js +83 -0
  93. package/dist/registry/dependencies.js.map +1 -0
  94. package/dist/registry/entry-point.d.ts +6 -0
  95. package/dist/registry/entry-point.d.ts.map +1 -0
  96. package/dist/registry/entry-point.js +55 -0
  97. package/dist/registry/entry-point.js.map +1 -0
  98. package/{src/registry/index.ts → dist/registry/index.d.ts} +1 -0
  99. package/dist/registry/index.d.ts.map +1 -0
  100. package/dist/registry/index.js +8 -0
  101. package/dist/registry/index.js.map +1 -0
  102. package/dist/registry/metadata.d.ts +9 -0
  103. package/dist/registry/metadata.d.ts.map +1 -0
  104. package/dist/registry/metadata.js +105 -0
  105. package/dist/registry/metadata.js.map +1 -0
  106. package/dist/registry/registry.d.ts +102 -0
  107. package/dist/registry/registry.d.ts.map +1 -0
  108. package/dist/registry/registry.js +534 -0
  109. package/dist/registry/registry.js.map +1 -0
  110. package/dist/registry/scanner.d.ts +7 -0
  111. package/dist/registry/scanner.d.ts.map +1 -0
  112. package/dist/registry/scanner.js +164 -0
  113. package/dist/registry/scanner.js.map +1 -0
  114. package/dist/registry/schema-export.d.ts +9 -0
  115. package/dist/registry/schema-export.d.ts.map +1 -0
  116. package/dist/registry/schema-export.js +132 -0
  117. package/dist/registry/schema-export.js.map +1 -0
  118. package/dist/registry/types.d.ts +29 -0
  119. package/dist/registry/types.d.ts.map +1 -0
  120. package/dist/registry/types.js +5 -0
  121. package/dist/registry/types.js.map +1 -0
  122. package/dist/registry/validation.d.ts +9 -0
  123. package/dist/registry/validation.d.ts.map +1 -0
  124. package/dist/registry/validation.js +33 -0
  125. package/dist/registry/validation.js.map +1 -0
  126. package/dist/schema/annotations.d.ts +8 -0
  127. package/dist/schema/annotations.d.ts.map +1 -0
  128. package/dist/schema/annotations.js +52 -0
  129. package/dist/schema/annotations.js.map +1 -0
  130. package/dist/schema/exporter.d.ts +13 -0
  131. package/dist/schema/exporter.d.ts.map +1 -0
  132. package/dist/schema/exporter.js +71 -0
  133. package/dist/schema/exporter.js.map +1 -0
  134. package/dist/schema/index.d.ts +9 -0
  135. package/dist/schema/index.d.ts.map +1 -0
  136. package/{src/schema/index.ts → dist/schema/index.js} +1 -7
  137. package/dist/schema/index.js.map +1 -0
  138. package/dist/schema/loader.d.ts +30 -0
  139. package/dist/schema/loader.d.ts.map +1 -0
  140. package/dist/schema/loader.js +260 -0
  141. package/dist/schema/loader.js.map +1 -0
  142. package/dist/schema/ref-resolver.d.ts +19 -0
  143. package/dist/schema/ref-resolver.d.ts.map +1 -0
  144. package/dist/schema/ref-resolver.js +212 -0
  145. package/dist/schema/ref-resolver.js.map +1 -0
  146. package/dist/schema/strict.d.ts +7 -0
  147. package/dist/schema/strict.d.ts.map +1 -0
  148. package/dist/schema/strict.js +127 -0
  149. package/dist/schema/strict.js.map +1 -0
  150. package/dist/schema/types.d.ts +53 -0
  151. package/dist/schema/types.d.ts.map +1 -0
  152. package/dist/schema/types.js +31 -0
  153. package/dist/schema/types.js.map +1 -0
  154. package/dist/schema/validator.d.ts +16 -0
  155. package/dist/schema/validator.d.ts.map +1 -0
  156. package/dist/schema/validator.js +71 -0
  157. package/dist/schema/validator.js.map +1 -0
  158. package/dist/trace-context.d.ts +35 -0
  159. package/dist/trace-context.d.ts.map +1 -0
  160. package/dist/trace-context.js +86 -0
  161. package/dist/trace-context.js.map +1 -0
  162. package/dist/utils/index.d.ts +11 -0
  163. package/dist/utils/index.d.ts.map +1 -0
  164. package/dist/utils/index.js +32 -0
  165. package/dist/utils/index.js.map +1 -0
  166. package/dist/utils/pattern.d.ts +5 -0
  167. package/dist/utils/pattern.d.ts.map +1 -0
  168. package/dist/utils/pattern.js +31 -0
  169. package/dist/utils/pattern.js.map +1 -0
  170. package/package.json +24 -3
  171. package/.claude/settings.local.json +0 -12
  172. package/.github/workflows/ci.yml +0 -39
  173. package/.gitmessage +0 -60
  174. package/.pre-commit-config.yaml +0 -28
  175. package/CHANGELOG.md +0 -214
  176. package/CLAUDE.md +0 -68
  177. package/apcore-logo.svg +0 -79
  178. package/planning/acl-system/overview.md +0 -54
  179. package/planning/acl-system/plan.md +0 -92
  180. package/planning/acl-system/state.json +0 -76
  181. package/planning/acl-system/tasks/acl-core.md +0 -226
  182. package/planning/acl-system/tasks/acl-rule.md +0 -92
  183. package/planning/acl-system/tasks/conditional-rules.md +0 -259
  184. package/planning/acl-system/tasks/pattern-matching.md +0 -152
  185. package/planning/acl-system/tasks/yaml-loading.md +0 -271
  186. package/planning/core-executor/overview.md +0 -53
  187. package/planning/core-executor/plan.md +0 -88
  188. package/planning/core-executor/state.json +0 -76
  189. package/planning/core-executor/tasks/async-support.md +0 -106
  190. package/planning/core-executor/tasks/execution-pipeline.md +0 -113
  191. package/planning/core-executor/tasks/redaction.md +0 -85
  192. package/planning/core-executor/tasks/safety-checks.md +0 -65
  193. package/planning/core-executor/tasks/setup.md +0 -75
  194. package/planning/decorator-bindings/overview.md +0 -62
  195. package/planning/decorator-bindings/plan.md +0 -104
  196. package/planning/decorator-bindings/state.json +0 -87
  197. package/planning/decorator-bindings/tasks/binding-directory.md +0 -79
  198. package/planning/decorator-bindings/tasks/binding-loader.md +0 -148
  199. package/planning/decorator-bindings/tasks/explicit-schemas.md +0 -85
  200. package/planning/decorator-bindings/tasks/function-module.md +0 -127
  201. package/planning/decorator-bindings/tasks/module-factory.md +0 -89
  202. package/planning/decorator-bindings/tasks/schema-modes.md +0 -142
  203. package/planning/middleware-system/overview.md +0 -48
  204. package/planning/middleware-system/plan.md +0 -102
  205. package/planning/middleware-system/state.json +0 -65
  206. package/planning/middleware-system/tasks/adapters.md +0 -170
  207. package/planning/middleware-system/tasks/base.md +0 -115
  208. package/planning/middleware-system/tasks/logging-middleware.md +0 -304
  209. package/planning/middleware-system/tasks/manager.md +0 -313
  210. package/planning/observability/overview.md +0 -53
  211. package/planning/observability/plan.md +0 -119
  212. package/planning/observability/state.json +0 -98
  213. package/planning/observability/tasks/context-logger.md +0 -201
  214. package/planning/observability/tasks/exporters.md +0 -121
  215. package/planning/observability/tasks/metrics-collector.md +0 -162
  216. package/planning/observability/tasks/metrics-middleware.md +0 -141
  217. package/planning/observability/tasks/obs-logging-middleware.md +0 -179
  218. package/planning/observability/tasks/span-model.md +0 -120
  219. package/planning/observability/tasks/tracing-middleware.md +0 -179
  220. package/planning/overview.md +0 -81
  221. package/planning/registry-system/overview.md +0 -57
  222. package/planning/registry-system/plan.md +0 -114
  223. package/planning/registry-system/state.json +0 -109
  224. package/planning/registry-system/tasks/dependencies.md +0 -157
  225. package/planning/registry-system/tasks/entry-point.md +0 -148
  226. package/planning/registry-system/tasks/metadata.md +0 -198
  227. package/planning/registry-system/tasks/registry-core.md +0 -323
  228. package/planning/registry-system/tasks/scanner.md +0 -172
  229. package/planning/registry-system/tasks/schema-export.md +0 -261
  230. package/planning/registry-system/tasks/types.md +0 -124
  231. package/planning/registry-system/tasks/validation.md +0 -177
  232. package/planning/schema-system/overview.md +0 -56
  233. package/planning/schema-system/plan.md +0 -121
  234. package/planning/schema-system/state.json +0 -98
  235. package/planning/schema-system/tasks/exporter.md +0 -153
  236. package/planning/schema-system/tasks/loader.md +0 -106
  237. package/planning/schema-system/tasks/ref-resolver.md +0 -133
  238. package/planning/schema-system/tasks/strict-mode.md +0 -140
  239. package/planning/schema-system/tasks/typebox-generation.md +0 -133
  240. package/planning/schema-system/tasks/types-and-annotations.md +0 -160
  241. package/planning/schema-system/tasks/validator.md +0 -149
  242. package/src/acl.ts +0 -200
  243. package/src/async-task.ts +0 -267
  244. package/src/bindings.ts +0 -207
  245. package/src/cancel.ts +0 -32
  246. package/src/config.ts +0 -24
  247. package/src/context.ts +0 -160
  248. package/src/decorator.ts +0 -110
  249. package/src/errors.ts +0 -429
  250. package/src/executor.ts +0 -493
  251. package/src/extensions.ts +0 -265
  252. package/src/middleware/adapters.ts +0 -54
  253. package/src/middleware/base.ts +0 -33
  254. package/src/middleware/logging.ts +0 -103
  255. package/src/middleware/manager.ts +0 -105
  256. package/src/module.ts +0 -43
  257. package/src/observability/context-logger.ts +0 -203
  258. package/src/observability/metrics.ts +0 -214
  259. package/src/observability/tracing.ts +0 -252
  260. package/src/registry/dependencies.ts +0 -99
  261. package/src/registry/entry-point.ts +0 -64
  262. package/src/registry/metadata.ts +0 -111
  263. package/src/registry/registry.ts +0 -580
  264. package/src/registry/scanner.ts +0 -168
  265. package/src/registry/schema-export.ts +0 -181
  266. package/src/registry/types.ts +0 -32
  267. package/src/registry/validation.ts +0 -38
  268. package/src/schema/annotations.ts +0 -68
  269. package/src/schema/exporter.ts +0 -90
  270. package/src/schema/loader.ts +0 -273
  271. package/src/schema/ref-resolver.ts +0 -244
  272. package/src/schema/strict.ts +0 -136
  273. package/src/schema/types.ts +0 -73
  274. package/src/schema/validator.ts +0 -82
  275. package/src/trace-context.ts +0 -102
  276. package/src/utils/index.ts +0 -5
  277. package/src/utils/pattern.ts +0 -30
  278. package/tests/async-task.test.ts +0 -335
  279. package/tests/helpers.ts +0 -30
  280. package/tests/integration/test-acl-safety.test.ts +0 -269
  281. package/tests/integration/test-binding-executor.test.ts +0 -194
  282. package/tests/integration/test-e2e-flow.test.ts +0 -117
  283. package/tests/integration/test-error-propagation.test.ts +0 -259
  284. package/tests/integration/test-middleware-chain.test.ts +0 -120
  285. package/tests/integration/test-observability-integration.test.ts +0 -438
  286. package/tests/observability/test-context-logger.test.ts +0 -123
  287. package/tests/observability/test-metrics.test.ts +0 -186
  288. package/tests/observability/test-tracing.test.ts +0 -303
  289. package/tests/registry/test-dependencies.test.ts +0 -70
  290. package/tests/registry/test-entry-point.test.ts +0 -133
  291. package/tests/registry/test-metadata.test.ts +0 -265
  292. package/tests/registry/test-registry.test.ts +0 -1397
  293. package/tests/registry/test-scanner.test.ts +0 -257
  294. package/tests/registry/test-schema-export.test.ts +0 -355
  295. package/tests/registry/test-validation.test.ts +0 -75
  296. package/tests/schema/test-annotations.test.ts +0 -137
  297. package/tests/schema/test-exporter.test.ts +0 -172
  298. package/tests/schema/test-loader.test.ts +0 -461
  299. package/tests/schema/test-ref-resolver.test.ts +0 -530
  300. package/tests/schema/test-strict.test.ts +0 -348
  301. package/tests/schema/test-validator.test.ts +0 -64
  302. package/tests/test-acl.test.ts +0 -423
  303. package/tests/test-bindings.test.ts +0 -227
  304. package/tests/test-cancel.test.ts +0 -71
  305. package/tests/test-config.test.ts +0 -76
  306. package/tests/test-context.test.ts +0 -266
  307. package/tests/test-decorator.test.ts +0 -173
  308. package/tests/test-errors.test.ts +0 -647
  309. package/tests/test-executor-stream.test.ts +0 -208
  310. package/tests/test-executor.test.ts +0 -252
  311. package/tests/test-extensions.test.ts +0 -310
  312. package/tests/test-logging-middleware.test.ts +0 -150
  313. package/tests/test-middleware-manager.test.ts +0 -185
  314. package/tests/test-middleware.test.ts +0 -86
  315. package/tests/test-trace-context.test.ts +0 -251
  316. package/tests/utils/test-pattern.test.ts +0 -109
  317. package/tsconfig.build.json +0 -8
  318. package/tsconfig.json +0 -20
  319. 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
- });