apcore-js 0.4.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 (306) 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} +14 -59
  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} +2 -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 -183
  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/bindings.ts +0 -207
  239. package/src/config.ts +0 -24
  240. package/src/context.ts +0 -78
  241. package/src/decorator.ts +0 -110
  242. package/src/errors.ts +0 -425
  243. package/src/executor.ts +0 -475
  244. package/src/middleware/adapters.ts +0 -54
  245. package/src/middleware/base.ts +0 -33
  246. package/src/middleware/logging.ts +0 -103
  247. package/src/middleware/manager.ts +0 -105
  248. package/src/module.ts +0 -43
  249. package/src/observability/context-logger.ts +0 -203
  250. package/src/observability/metrics.ts +0 -214
  251. package/src/observability/tracing.ts +0 -188
  252. package/src/registry/dependencies.ts +0 -99
  253. package/src/registry/entry-point.ts +0 -64
  254. package/src/registry/metadata.ts +0 -111
  255. package/src/registry/registry.ts +0 -360
  256. package/src/registry/scanner.ts +0 -168
  257. package/src/registry/schema-export.ts +0 -181
  258. package/src/registry/types.ts +0 -32
  259. package/src/registry/validation.ts +0 -38
  260. package/src/schema/annotations.ts +0 -68
  261. package/src/schema/exporter.ts +0 -90
  262. package/src/schema/loader.ts +0 -273
  263. package/src/schema/ref-resolver.ts +0 -244
  264. package/src/schema/strict.ts +0 -136
  265. package/src/schema/types.ts +0 -73
  266. package/src/schema/validator.ts +0 -82
  267. package/src/utils/index.ts +0 -5
  268. package/src/utils/pattern.ts +0 -30
  269. package/tests/helpers.ts +0 -30
  270. package/tests/integration/test-acl-safety.test.ts +0 -269
  271. package/tests/integration/test-binding-executor.test.ts +0 -194
  272. package/tests/integration/test-e2e-flow.test.ts +0 -117
  273. package/tests/integration/test-error-propagation.test.ts +0 -259
  274. package/tests/integration/test-middleware-chain.test.ts +0 -120
  275. package/tests/integration/test-observability-integration.test.ts +0 -438
  276. package/tests/observability/test-context-logger.test.ts +0 -123
  277. package/tests/observability/test-metrics.test.ts +0 -186
  278. package/tests/observability/test-tracing.test.ts +0 -131
  279. package/tests/registry/test-dependencies.test.ts +0 -70
  280. package/tests/registry/test-entry-point.test.ts +0 -133
  281. package/tests/registry/test-metadata.test.ts +0 -265
  282. package/tests/registry/test-registry.test.ts +0 -1008
  283. package/tests/registry/test-scanner.test.ts +0 -257
  284. package/tests/registry/test-schema-export.test.ts +0 -355
  285. package/tests/registry/test-validation.test.ts +0 -75
  286. package/tests/schema/test-annotations.test.ts +0 -137
  287. package/tests/schema/test-exporter.test.ts +0 -172
  288. package/tests/schema/test-loader.test.ts +0 -461
  289. package/tests/schema/test-ref-resolver.test.ts +0 -530
  290. package/tests/schema/test-strict.test.ts +0 -348
  291. package/tests/schema/test-validator.test.ts +0 -64
  292. package/tests/test-acl.test.ts +0 -423
  293. package/tests/test-bindings.test.ts +0 -227
  294. package/tests/test-config.test.ts +0 -76
  295. package/tests/test-context.test.ts +0 -151
  296. package/tests/test-decorator.test.ts +0 -173
  297. package/tests/test-errors.test.ts +0 -647
  298. package/tests/test-executor-stream.test.ts +0 -208
  299. package/tests/test-executor.test.ts +0 -252
  300. package/tests/test-logging-middleware.test.ts +0 -150
  301. package/tests/test-middleware-manager.test.ts +0 -185
  302. package/tests/test-middleware.test.ts +0 -86
  303. package/tests/utils/test-pattern.test.ts +0 -109
  304. package/tsconfig.build.json +0 -8
  305. package/tsconfig.json +0 -20
  306. 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,131 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import { Context } from '../../src/context.js';
3
- import {
4
- createSpan,
5
- InMemoryExporter,
6
- StdoutExporter,
7
- TracingMiddleware,
8
- } from '../../src/observability/tracing.js';
9
-
10
- describe('Span', () => {
11
- it('createSpan creates span with defaults', () => {
12
- const span = createSpan({
13
- traceId: 'trace-1',
14
- name: 'test.span',
15
- startTime: 100,
16
- });
17
- expect(span.traceId).toBe('trace-1');
18
- expect(span.name).toBe('test.span');
19
- expect(span.startTime).toBe(100);
20
- expect(span.spanId).toBeDefined();
21
- expect(span.parentSpanId).toBeNull();
22
- expect(span.status).toBe('ok');
23
- expect(span.events).toEqual([]);
24
- });
25
- });
26
-
27
- describe('InMemoryExporter', () => {
28
- it('collects and retrieves spans', () => {
29
- const exporter = new InMemoryExporter();
30
- const span = createSpan({ traceId: 't1', name: 'test', startTime: 0 });
31
- exporter.export(span);
32
- expect(exporter.getSpans()).toHaveLength(1);
33
- expect(exporter.getSpans()[0].traceId).toBe('t1');
34
- });
35
-
36
- it('respects max_spans limit', () => {
37
- const exporter = new InMemoryExporter(3);
38
- for (let i = 0; i < 5; i++) {
39
- exporter.export(createSpan({ traceId: `t${i}`, name: 'test', startTime: i }));
40
- }
41
- const spans = exporter.getSpans();
42
- expect(spans).toHaveLength(3);
43
- expect(spans[0].traceId).toBe('t2');
44
- });
45
-
46
- it('clear removes all spans', () => {
47
- const exporter = new InMemoryExporter();
48
- exporter.export(createSpan({ traceId: 't1', name: 'test', startTime: 0 }));
49
- exporter.clear();
50
- expect(exporter.getSpans()).toHaveLength(0);
51
- });
52
- });
53
-
54
- describe('TracingMiddleware', () => {
55
- it('creates and exports spans on success', () => {
56
- const exporter = new InMemoryExporter();
57
- const mw = new TracingMiddleware(exporter);
58
- const ctx = Context.create();
59
-
60
- mw.before('mod.a', {}, ctx);
61
- mw.after('mod.a', {}, { result: 'ok' }, ctx);
62
-
63
- const spans = exporter.getSpans();
64
- expect(spans).toHaveLength(1);
65
- expect(spans[0].name).toBe('apcore.module.execute');
66
- expect(spans[0].status).toBe('ok');
67
- expect(spans[0].attributes['moduleId']).toBe('mod.a');
68
- });
69
-
70
- it('creates error spans', () => {
71
- const exporter = new InMemoryExporter();
72
- const mw = new TracingMiddleware(exporter);
73
- const ctx = Context.create();
74
-
75
- mw.before('mod.err', {}, ctx);
76
- mw.onError('mod.err', {}, new Error('fail'), ctx);
77
-
78
- const spans = exporter.getSpans();
79
- expect(spans).toHaveLength(1);
80
- expect(spans[0].status).toBe('error');
81
- });
82
-
83
- it('supports nested spans with parent chain', () => {
84
- const exporter = new InMemoryExporter();
85
- const mw = new TracingMiddleware(exporter);
86
- const ctx = Context.create();
87
-
88
- mw.before('mod.outer', {}, ctx);
89
- mw.before('mod.inner', {}, ctx);
90
- mw.after('mod.inner', {}, {}, ctx);
91
- mw.after('mod.outer', {}, {}, ctx);
92
-
93
- const spans = exporter.getSpans();
94
- expect(spans).toHaveLength(2);
95
- // inner span has outer as parent
96
- expect(spans[0].parentSpanId).toBe(spans[1].spanId);
97
- });
98
-
99
- it('off strategy does not export', () => {
100
- const exporter = new InMemoryExporter();
101
- const mw = new TracingMiddleware(exporter, 1.0, 'off');
102
- const ctx = Context.create();
103
-
104
- mw.before('mod.a', {}, ctx);
105
- mw.after('mod.a', {}, {}, ctx);
106
-
107
- expect(exporter.getSpans()).toHaveLength(0);
108
- });
109
-
110
- it('error_first exports errors even when not sampled', () => {
111
- const exporter = new InMemoryExporter();
112
- const mw = new TracingMiddleware(exporter, 0.0, 'error_first');
113
- const ctx = Context.create();
114
-
115
- mw.before('mod.a', {}, ctx);
116
- mw.onError('mod.a', {}, new Error('fail'), ctx);
117
-
118
- expect(exporter.getSpans()).toHaveLength(1);
119
- });
120
-
121
- it('throws on invalid sampling rate', () => {
122
- const exporter = new InMemoryExporter();
123
- expect(() => new TracingMiddleware(exporter, -0.1)).toThrow();
124
- expect(() => new TracingMiddleware(exporter, 1.5)).toThrow();
125
- });
126
-
127
- it('throws on invalid sampling strategy', () => {
128
- const exporter = new InMemoryExporter();
129
- expect(() => new TracingMiddleware(exporter, 1.0, 'invalid')).toThrow();
130
- });
131
- });
@@ -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
- });
@@ -1,133 +0,0 @@
1
- import { describe, it, expect, beforeEach, afterEach } from 'vitest';
2
- import { mkdtempSync, writeFileSync, rmSync } from 'node:fs';
3
- import { join } from 'node:path';
4
- import { tmpdir } from 'node:os';
5
- import { pathToFileURL } from 'node:url';
6
- import { snakeToPascal, resolveEntryPoint } from '../../src/registry/entry-point.js';
7
- import { ModuleLoadError } from '../../src/errors.js';
8
-
9
- function validModuleSource(name: string): string {
10
- return `{
11
- inputSchema: { type: 'object' },
12
- outputSchema: { type: 'object' },
13
- description: '${name} module',
14
- execute: function() { return {}; },
15
- }`;
16
- }
17
-
18
- describe('snakeToPascal', () => {
19
- it('returns empty string for empty input', () => {
20
- expect(snakeToPascal('')).toBe('');
21
- });
22
-
23
- it('capitalises a single word', () => {
24
- expect(snakeToPascal('hello')).toBe('Hello');
25
- });
26
-
27
- it('converts two-word snake_case', () => {
28
- expect(snakeToPascal('hello_world')).toBe('HelloWorld');
29
- });
30
-
31
- it('converts multi-word snake_case', () => {
32
- expect(snakeToPascal('my_module_name')).toBe('MyModuleName');
33
- });
34
-
35
- it('capitalises a word with no underscores', () => {
36
- expect(snakeToPascal('already')).toBe('Already');
37
- });
38
- });
39
-
40
- describe('resolveEntryPoint', () => {
41
- let tmpDir: string;
42
-
43
- beforeEach(() => {
44
- tmpDir = mkdtempSync(join(tmpdir(), 'apcore-entry-test-'));
45
- });
46
-
47
- afterEach(() => {
48
- rmSync(tmpDir, { recursive: true, force: true });
49
- });
50
-
51
- function fileUrl(filePath: string): string {
52
- return pathToFileURL(filePath).href;
53
- }
54
-
55
- it('throws ModuleLoadError for a non-existent file path', async () => {
56
- const badPath = fileUrl(join(tmpDir, 'does_not_exist.js'));
57
- await expect(resolveEntryPoint(badPath)).rejects.toThrow(ModuleLoadError);
58
- });
59
-
60
- it('finds default export that satisfies isModuleClass', async () => {
61
- const filePath = join(tmpDir, 'default_mod.mjs');
62
- writeFileSync(filePath, `const mod = ${validModuleSource('Default')};\nexport default mod;\n`);
63
- const result = await resolveEntryPoint(fileUrl(filePath));
64
- expect(result).toBeDefined();
65
- expect((result as Record<string, unknown>)['description']).toBe('Default module');
66
- });
67
-
68
- it('finds a single named export that satisfies isModuleClass', async () => {
69
- const filePath = join(tmpDir, 'named_mod.mjs');
70
- writeFileSync(filePath, `export const MyModule = ${validModuleSource('Named')};\n`);
71
- const result = await resolveEntryPoint(fileUrl(filePath));
72
- expect(result).toBeDefined();
73
- expect((result as Record<string, unknown>)['description']).toBe('Named module');
74
- });
75
-
76
- it('throws ModuleLoadError when no export satisfies isModuleClass', async () => {
77
- const filePath = join(tmpDir, 'no_valid.mjs');
78
- writeFileSync(filePath, `export const notAModule = { foo: 'bar' };\nexport const alsoNot = 42;\n`);
79
- await expect(resolveEntryPoint(fileUrl(filePath))).rejects.toThrow(ModuleLoadError);
80
- await expect(resolveEntryPoint(fileUrl(filePath))).rejects.toThrow(/No Module subclass found/);
81
- });
82
-
83
- it('throws ModuleLoadError when multiple exports satisfy isModuleClass', async () => {
84
- const filePath = join(tmpDir, 'ambiguous.mjs');
85
- writeFileSync(filePath, `export const ModuleA = ${validModuleSource('A')};\nexport const ModuleB = ${validModuleSource('B')};\n`);
86
- await expect(resolveEntryPoint(fileUrl(filePath))).rejects.toThrow(ModuleLoadError);
87
- await expect(resolveEntryPoint(fileUrl(filePath))).rejects.toThrow(/Ambiguous entry point/);
88
- });
89
-
90
- it('uses meta entry_point to find a specific named class', async () => {
91
- const filePath = join(tmpDir, 'meta_override.mjs');
92
- writeFileSync(filePath, `export const Alpha = ${validModuleSource('Alpha')};\nexport const Beta = ${validModuleSource('Beta')};\n`);
93
- const meta = { entry_point: 'some.path:Beta' };
94
- const result = await resolveEntryPoint(fileUrl(filePath), meta);
95
- expect((result as Record<string, unknown>)['description']).toBe('Beta module');
96
- });
97
-
98
- it('throws ModuleLoadError when meta entry_point class does not exist', async () => {
99
- const filePath = join(tmpDir, 'meta_missing.mjs');
100
- writeFileSync(filePath, `export const Alpha = ${validModuleSource('Alpha')};\n`);
101
- const meta = { entry_point: 'mod:NonExistent' };
102
- await expect(resolveEntryPoint(fileUrl(filePath), meta)).rejects.toThrow(ModuleLoadError);
103
- await expect(resolveEntryPoint(fileUrl(filePath), meta)).rejects.toThrow(/Entry point class 'NonExistent' not found/);
104
- });
105
-
106
- it('prefers default export over named exports when both are valid', async () => {
107
- const filePath = join(tmpDir, 'default_priority.mjs');
108
- writeFileSync(filePath, `const def = ${validModuleSource('DefaultPriority')};\nexport default def;\nexport const Named = ${validModuleSource('NamedSibling')};\n`);
109
- const result = await resolveEntryPoint(fileUrl(filePath));
110
- expect((result as Record<string, unknown>)['description']).toBe('DefaultPriority module');
111
- });
112
-
113
- it('falls back to auto-infer when meta is null', async () => {
114
- const filePath = join(tmpDir, 'null_meta.mjs');
115
- writeFileSync(filePath, `export const OnlyModule = ${validModuleSource('OnlyMeta')};\n`);
116
- const result = await resolveEntryPoint(fileUrl(filePath), null);
117
- expect((result as Record<string, unknown>)['description']).toBe('OnlyMeta module');
118
- });
119
-
120
- it('ignores exports missing execute function', async () => {
121
- const filePath = join(tmpDir, 'missing_execute.mjs');
122
- writeFileSync(filePath, `export const Incomplete = {\n inputSchema: { type: 'object' },\n outputSchema: { type: 'object' },\n description: 'no execute',\n};\nexport const Complete = ${validModuleSource('Complete')};\n`);
123
- const result = await resolveEntryPoint(fileUrl(filePath));
124
- expect((result as Record<string, unknown>)['description']).toBe('Complete module');
125
- });
126
-
127
- it('ignores primitive and null exports', async () => {
128
- const filePath = join(tmpDir, 'primitives.mjs');
129
- writeFileSync(filePath, `export const aNumber = 42;\nexport const aString = 'hello';\nexport const aNull = null;\nexport const Valid = ${validModuleSource('Valid')};\n`);
130
- const result = await resolveEntryPoint(fileUrl(filePath));
131
- expect((result as Record<string, unknown>)['description']).toBe('Valid module');
132
- });
133
- });