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.
- package/README.md +1 -1
- package/dist/acl.d.ts +27 -0
- package/dist/acl.d.ts.map +1 -0
- package/dist/acl.js +175 -0
- package/dist/acl.js.map +1 -0
- package/dist/approval.d.ts +85 -0
- package/dist/approval.d.ts.map +1 -0
- package/dist/approval.js +73 -0
- package/dist/approval.js.map +1 -0
- package/dist/async-task.d.ts +90 -0
- package/dist/async-task.d.ts.map +1 -0
- package/dist/async-task.js +215 -0
- package/dist/async-task.js.map +1 -0
- package/dist/bindings.d.ts +12 -0
- package/dist/bindings.d.ts.map +1 -0
- package/dist/bindings.js +185 -0
- package/dist/bindings.js.map +1 -0
- package/dist/cancel.d.ts +14 -0
- package/dist/cancel.d.ts.map +1 -0
- package/dist/cancel.js +27 -0
- package/dist/cancel.js.map +1 -0
- package/dist/config.d.ts +9 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +23 -0
- package/dist/config.js.map +1 -0
- package/dist/context.d.ts +50 -0
- package/dist/context.d.ts.map +1 -0
- package/dist/context.js +87 -0
- package/dist/context.js.map +1 -0
- package/dist/decorator.d.ts +57 -0
- package/dist/decorator.d.ts.map +1 -0
- package/dist/decorator.js +74 -0
- package/dist/decorator.js.map +1 -0
- package/dist/errors.d.ts +204 -0
- package/dist/errors.d.ts.map +1 -0
- package/dist/errors.js +364 -0
- package/dist/errors.js.map +1 -0
- package/dist/executor.d.ts +82 -0
- package/dist/executor.d.ts.map +1 -0
- package/dist/executor.js +489 -0
- package/dist/executor.js.map +1 -0
- package/dist/extensions.d.ts +58 -0
- package/dist/extensions.d.ts.map +1 -0
- package/dist/extensions.js +239 -0
- package/dist/extensions.js.map +1 -0
- package/{src/index.ts → dist/index.d.ts} +6 -63
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +45 -0
- package/dist/index.js.map +1 -0
- package/dist/middleware/adapters.d.ts +18 -0
- package/dist/middleware/adapters.d.ts.map +1 -0
- package/dist/middleware/adapters.js +25 -0
- package/dist/middleware/adapters.js.map +1 -0
- package/dist/middleware/base.d.ts +10 -0
- package/dist/middleware/base.d.ts.map +1 -0
- package/dist/middleware/base.js +15 -0
- package/dist/middleware/base.js.map +1 -0
- package/{src/middleware/index.ts → dist/middleware/index.d.ts} +1 -0
- package/dist/middleware/index.d.ts.map +1 -0
- package/dist/middleware/index.js +5 -0
- package/dist/middleware/index.js.map +1 -0
- package/dist/middleware/logging.d.ts +25 -0
- package/dist/middleware/logging.d.ts.map +1 -0
- package/dist/middleware/logging.js +64 -0
- package/dist/middleware/logging.js.map +1 -0
- package/dist/middleware/manager.d.ts +21 -0
- package/dist/middleware/manager.d.ts.map +1 -0
- package/dist/middleware/manager.js +77 -0
- package/dist/middleware/manager.js.map +1 -0
- package/dist/module.d.ts +31 -0
- package/dist/module.d.ts.map +1 -0
- package/dist/module.js +12 -0
- package/dist/module.js.map +1 -0
- package/dist/observability/context-logger.d.ts +54 -0
- package/dist/observability/context-logger.d.ts.map +1 -0
- package/dist/observability/context-logger.js +151 -0
- package/dist/observability/context-logger.js.map +1 -0
- package/{src/observability/index.ts → dist/observability/index.d.ts} +1 -0
- package/dist/observability/index.d.ts.map +1 -0
- package/dist/observability/index.js +4 -0
- package/dist/observability/index.js.map +1 -0
- package/dist/observability/metrics.d.ts +30 -0
- package/dist/observability/metrics.d.ts.map +1 -0
- package/dist/observability/metrics.js +177 -0
- package/dist/observability/metrics.js.map +1 -0
- package/dist/observability/tracing.d.ts +62 -0
- package/dist/observability/tracing.d.ts.map +1 -0
- package/dist/observability/tracing.js +184 -0
- package/dist/observability/tracing.js.map +1 -0
- package/dist/registry/dependencies.d.ts +6 -0
- package/dist/registry/dependencies.d.ts.map +1 -0
- package/dist/registry/dependencies.js +83 -0
- package/dist/registry/dependencies.js.map +1 -0
- package/dist/registry/entry-point.d.ts +6 -0
- package/dist/registry/entry-point.d.ts.map +1 -0
- package/dist/registry/entry-point.js +55 -0
- package/dist/registry/entry-point.js.map +1 -0
- package/{src/registry/index.ts → dist/registry/index.d.ts} +1 -0
- package/dist/registry/index.d.ts.map +1 -0
- package/dist/registry/index.js +8 -0
- package/dist/registry/index.js.map +1 -0
- package/dist/registry/metadata.d.ts +9 -0
- package/dist/registry/metadata.d.ts.map +1 -0
- package/dist/registry/metadata.js +105 -0
- package/dist/registry/metadata.js.map +1 -0
- package/dist/registry/registry.d.ts +102 -0
- package/dist/registry/registry.d.ts.map +1 -0
- package/dist/registry/registry.js +534 -0
- package/dist/registry/registry.js.map +1 -0
- package/dist/registry/scanner.d.ts +7 -0
- package/dist/registry/scanner.d.ts.map +1 -0
- package/dist/registry/scanner.js +164 -0
- package/dist/registry/scanner.js.map +1 -0
- package/dist/registry/schema-export.d.ts +9 -0
- package/dist/registry/schema-export.d.ts.map +1 -0
- package/dist/registry/schema-export.js +132 -0
- package/dist/registry/schema-export.js.map +1 -0
- package/dist/registry/types.d.ts +29 -0
- package/dist/registry/types.d.ts.map +1 -0
- package/dist/registry/types.js +5 -0
- package/dist/registry/types.js.map +1 -0
- package/dist/registry/validation.d.ts +9 -0
- package/dist/registry/validation.d.ts.map +1 -0
- package/dist/registry/validation.js +33 -0
- package/dist/registry/validation.js.map +1 -0
- package/dist/schema/annotations.d.ts +8 -0
- package/dist/schema/annotations.d.ts.map +1 -0
- package/dist/schema/annotations.js +52 -0
- package/dist/schema/annotations.js.map +1 -0
- package/dist/schema/exporter.d.ts +13 -0
- package/dist/schema/exporter.d.ts.map +1 -0
- package/dist/schema/exporter.js +71 -0
- package/dist/schema/exporter.js.map +1 -0
- package/dist/schema/index.d.ts +9 -0
- package/dist/schema/index.d.ts.map +1 -0
- package/{src/schema/index.ts → dist/schema/index.js} +1 -7
- package/dist/schema/index.js.map +1 -0
- package/dist/schema/loader.d.ts +30 -0
- package/dist/schema/loader.d.ts.map +1 -0
- package/dist/schema/loader.js +260 -0
- package/dist/schema/loader.js.map +1 -0
- package/dist/schema/ref-resolver.d.ts +19 -0
- package/dist/schema/ref-resolver.d.ts.map +1 -0
- package/dist/schema/ref-resolver.js +212 -0
- package/dist/schema/ref-resolver.js.map +1 -0
- package/dist/schema/strict.d.ts +7 -0
- package/dist/schema/strict.d.ts.map +1 -0
- package/dist/schema/strict.js +127 -0
- package/dist/schema/strict.js.map +1 -0
- package/dist/schema/types.d.ts +53 -0
- package/dist/schema/types.d.ts.map +1 -0
- package/dist/schema/types.js +31 -0
- package/dist/schema/types.js.map +1 -0
- package/dist/schema/validator.d.ts +16 -0
- package/dist/schema/validator.d.ts.map +1 -0
- package/dist/schema/validator.js +71 -0
- package/dist/schema/validator.js.map +1 -0
- package/dist/trace-context.d.ts +35 -0
- package/dist/trace-context.d.ts.map +1 -0
- package/dist/trace-context.js +86 -0
- package/dist/trace-context.js.map +1 -0
- package/dist/utils/index.d.ts +11 -0
- package/dist/utils/index.d.ts.map +1 -0
- package/dist/utils/index.js +32 -0
- package/dist/utils/index.js.map +1 -0
- package/dist/utils/pattern.d.ts +5 -0
- package/dist/utils/pattern.d.ts.map +1 -0
- package/dist/utils/pattern.js +31 -0
- package/dist/utils/pattern.js.map +1 -0
- package/package.json +24 -3
- package/.claude/settings.local.json +0 -12
- package/.github/workflows/ci.yml +0 -39
- package/.gitmessage +0 -60
- package/.pre-commit-config.yaml +0 -28
- package/CHANGELOG.md +0 -214
- package/CLAUDE.md +0 -68
- package/apcore-logo.svg +0 -79
- package/planning/acl-system/overview.md +0 -54
- package/planning/acl-system/plan.md +0 -92
- package/planning/acl-system/state.json +0 -76
- package/planning/acl-system/tasks/acl-core.md +0 -226
- package/planning/acl-system/tasks/acl-rule.md +0 -92
- package/planning/acl-system/tasks/conditional-rules.md +0 -259
- package/planning/acl-system/tasks/pattern-matching.md +0 -152
- package/planning/acl-system/tasks/yaml-loading.md +0 -271
- package/planning/core-executor/overview.md +0 -53
- package/planning/core-executor/plan.md +0 -88
- package/planning/core-executor/state.json +0 -76
- package/planning/core-executor/tasks/async-support.md +0 -106
- package/planning/core-executor/tasks/execution-pipeline.md +0 -113
- package/planning/core-executor/tasks/redaction.md +0 -85
- package/planning/core-executor/tasks/safety-checks.md +0 -65
- package/planning/core-executor/tasks/setup.md +0 -75
- package/planning/decorator-bindings/overview.md +0 -62
- package/planning/decorator-bindings/plan.md +0 -104
- package/planning/decorator-bindings/state.json +0 -87
- package/planning/decorator-bindings/tasks/binding-directory.md +0 -79
- package/planning/decorator-bindings/tasks/binding-loader.md +0 -148
- package/planning/decorator-bindings/tasks/explicit-schemas.md +0 -85
- package/planning/decorator-bindings/tasks/function-module.md +0 -127
- package/planning/decorator-bindings/tasks/module-factory.md +0 -89
- package/planning/decorator-bindings/tasks/schema-modes.md +0 -142
- package/planning/middleware-system/overview.md +0 -48
- package/planning/middleware-system/plan.md +0 -102
- package/planning/middleware-system/state.json +0 -65
- package/planning/middleware-system/tasks/adapters.md +0 -170
- package/planning/middleware-system/tasks/base.md +0 -115
- package/planning/middleware-system/tasks/logging-middleware.md +0 -304
- package/planning/middleware-system/tasks/manager.md +0 -313
- package/planning/observability/overview.md +0 -53
- package/planning/observability/plan.md +0 -119
- package/planning/observability/state.json +0 -98
- package/planning/observability/tasks/context-logger.md +0 -201
- package/planning/observability/tasks/exporters.md +0 -121
- package/planning/observability/tasks/metrics-collector.md +0 -162
- package/planning/observability/tasks/metrics-middleware.md +0 -141
- package/planning/observability/tasks/obs-logging-middleware.md +0 -179
- package/planning/observability/tasks/span-model.md +0 -120
- package/planning/observability/tasks/tracing-middleware.md +0 -179
- package/planning/overview.md +0 -81
- package/planning/registry-system/overview.md +0 -57
- package/planning/registry-system/plan.md +0 -114
- package/planning/registry-system/state.json +0 -109
- package/planning/registry-system/tasks/dependencies.md +0 -157
- package/planning/registry-system/tasks/entry-point.md +0 -148
- package/planning/registry-system/tasks/metadata.md +0 -198
- package/planning/registry-system/tasks/registry-core.md +0 -323
- package/planning/registry-system/tasks/scanner.md +0 -172
- package/planning/registry-system/tasks/schema-export.md +0 -261
- package/planning/registry-system/tasks/types.md +0 -124
- package/planning/registry-system/tasks/validation.md +0 -177
- package/planning/schema-system/overview.md +0 -56
- package/planning/schema-system/plan.md +0 -121
- package/planning/schema-system/state.json +0 -98
- package/planning/schema-system/tasks/exporter.md +0 -153
- package/planning/schema-system/tasks/loader.md +0 -106
- package/planning/schema-system/tasks/ref-resolver.md +0 -133
- package/planning/schema-system/tasks/strict-mode.md +0 -140
- package/planning/schema-system/tasks/typebox-generation.md +0 -133
- package/planning/schema-system/tasks/types-and-annotations.md +0 -160
- package/planning/schema-system/tasks/validator.md +0 -149
- package/src/acl.ts +0 -200
- package/src/async-task.ts +0 -267
- package/src/bindings.ts +0 -207
- package/src/cancel.ts +0 -32
- package/src/config.ts +0 -24
- package/src/context.ts +0 -160
- package/src/decorator.ts +0 -110
- package/src/errors.ts +0 -429
- package/src/executor.ts +0 -493
- package/src/extensions.ts +0 -265
- package/src/middleware/adapters.ts +0 -54
- package/src/middleware/base.ts +0 -33
- package/src/middleware/logging.ts +0 -103
- package/src/middleware/manager.ts +0 -105
- package/src/module.ts +0 -43
- package/src/observability/context-logger.ts +0 -203
- package/src/observability/metrics.ts +0 -214
- package/src/observability/tracing.ts +0 -252
- package/src/registry/dependencies.ts +0 -99
- package/src/registry/entry-point.ts +0 -64
- package/src/registry/metadata.ts +0 -111
- package/src/registry/registry.ts +0 -580
- package/src/registry/scanner.ts +0 -168
- package/src/registry/schema-export.ts +0 -181
- package/src/registry/types.ts +0 -32
- package/src/registry/validation.ts +0 -38
- package/src/schema/annotations.ts +0 -68
- package/src/schema/exporter.ts +0 -90
- package/src/schema/loader.ts +0 -273
- package/src/schema/ref-resolver.ts +0 -244
- package/src/schema/strict.ts +0 -136
- package/src/schema/types.ts +0 -73
- package/src/schema/validator.ts +0 -82
- package/src/trace-context.ts +0 -102
- package/src/utils/index.ts +0 -5
- package/src/utils/pattern.ts +0 -30
- package/tests/async-task.test.ts +0 -335
- package/tests/helpers.ts +0 -30
- package/tests/integration/test-acl-safety.test.ts +0 -269
- package/tests/integration/test-binding-executor.test.ts +0 -194
- package/tests/integration/test-e2e-flow.test.ts +0 -117
- package/tests/integration/test-error-propagation.test.ts +0 -259
- package/tests/integration/test-middleware-chain.test.ts +0 -120
- package/tests/integration/test-observability-integration.test.ts +0 -438
- package/tests/observability/test-context-logger.test.ts +0 -123
- package/tests/observability/test-metrics.test.ts +0 -186
- package/tests/observability/test-tracing.test.ts +0 -303
- package/tests/registry/test-dependencies.test.ts +0 -70
- package/tests/registry/test-entry-point.test.ts +0 -133
- package/tests/registry/test-metadata.test.ts +0 -265
- package/tests/registry/test-registry.test.ts +0 -1397
- package/tests/registry/test-scanner.test.ts +0 -257
- package/tests/registry/test-schema-export.test.ts +0 -355
- package/tests/registry/test-validation.test.ts +0 -75
- package/tests/schema/test-annotations.test.ts +0 -137
- package/tests/schema/test-exporter.test.ts +0 -172
- package/tests/schema/test-loader.test.ts +0 -461
- package/tests/schema/test-ref-resolver.test.ts +0 -530
- package/tests/schema/test-strict.test.ts +0 -348
- package/tests/schema/test-validator.test.ts +0 -64
- package/tests/test-acl.test.ts +0 -423
- package/tests/test-bindings.test.ts +0 -227
- package/tests/test-cancel.test.ts +0 -71
- package/tests/test-config.test.ts +0 -76
- package/tests/test-context.test.ts +0 -266
- package/tests/test-decorator.test.ts +0 -173
- package/tests/test-errors.test.ts +0 -647
- package/tests/test-executor-stream.test.ts +0 -208
- package/tests/test-executor.test.ts +0 -252
- package/tests/test-extensions.test.ts +0 -310
- package/tests/test-logging-middleware.test.ts +0 -150
- package/tests/test-middleware-manager.test.ts +0 -185
- package/tests/test-middleware.test.ts +0 -86
- package/tests/test-trace-context.test.ts +0 -251
- package/tests/utils/test-pattern.test.ts +0 -109
- package/tsconfig.build.json +0 -8
- package/tsconfig.json +0 -20
- 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
|
-
});
|