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,71 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect } from 'vitest';
|
|
2
|
-
import { Type } from '@sinclair/typebox';
|
|
3
|
-
import { CancelToken, ExecutionCancelledError } from '../src/cancel.js';
|
|
4
|
-
import { Context } from '../src/context.js';
|
|
5
|
-
import { Executor } from '../src/executor.js';
|
|
6
|
-
import { FunctionModule } from '../src/decorator.js';
|
|
7
|
-
import { Registry } from '../src/registry/registry.js';
|
|
8
|
-
|
|
9
|
-
describe('CancelToken', () => {
|
|
10
|
-
it('is initially not cancelled', () => {
|
|
11
|
-
const token = new CancelToken();
|
|
12
|
-
expect(token.isCancelled).toBe(false);
|
|
13
|
-
});
|
|
14
|
-
|
|
15
|
-
it('sets flag after cancel()', () => {
|
|
16
|
-
const token = new CancelToken();
|
|
17
|
-
token.cancel();
|
|
18
|
-
expect(token.isCancelled).toBe(true);
|
|
19
|
-
});
|
|
20
|
-
|
|
21
|
-
it('check() does nothing when not cancelled', () => {
|
|
22
|
-
const token = new CancelToken();
|
|
23
|
-
expect(() => token.check()).not.toThrow();
|
|
24
|
-
});
|
|
25
|
-
|
|
26
|
-
it('check() throws ExecutionCancelledError when cancelled', () => {
|
|
27
|
-
const token = new CancelToken();
|
|
28
|
-
token.cancel();
|
|
29
|
-
expect(() => token.check()).toThrow(ExecutionCancelledError);
|
|
30
|
-
});
|
|
31
|
-
|
|
32
|
-
it('reset() clears cancellation', () => {
|
|
33
|
-
const token = new CancelToken();
|
|
34
|
-
token.cancel();
|
|
35
|
-
expect(token.isCancelled).toBe(true);
|
|
36
|
-
token.reset();
|
|
37
|
-
expect(token.isCancelled).toBe(false);
|
|
38
|
-
expect(() => token.check()).not.toThrow();
|
|
39
|
-
});
|
|
40
|
-
});
|
|
41
|
-
|
|
42
|
-
describe('Executor cancellation', () => {
|
|
43
|
-
it('respects cancelled token before execution', async () => {
|
|
44
|
-
const registry = new Registry();
|
|
45
|
-
const mod = new FunctionModule({
|
|
46
|
-
execute: () => ({ result: 'ok' }),
|
|
47
|
-
moduleId: 'test.module',
|
|
48
|
-
inputSchema: Type.Object({}),
|
|
49
|
-
outputSchema: Type.Object({ result: Type.String() }),
|
|
50
|
-
description: 'Simple module',
|
|
51
|
-
});
|
|
52
|
-
registry.register('test.module', mod);
|
|
53
|
-
|
|
54
|
-
const executor = new Executor({ registry });
|
|
55
|
-
const token = new CancelToken();
|
|
56
|
-
token.cancel();
|
|
57
|
-
|
|
58
|
-
const ctx = new Context(
|
|
59
|
-
'trace-1',
|
|
60
|
-
null,
|
|
61
|
-
[],
|
|
62
|
-
executor,
|
|
63
|
-
null,
|
|
64
|
-
null,
|
|
65
|
-
{},
|
|
66
|
-
token,
|
|
67
|
-
);
|
|
68
|
-
|
|
69
|
-
await expect(executor.call('test.module', {}, ctx)).rejects.toThrow(ExecutionCancelledError);
|
|
70
|
-
});
|
|
71
|
-
});
|
|
@@ -1,76 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect } from 'vitest';
|
|
2
|
-
import { Config } from '../src/config.js';
|
|
3
|
-
|
|
4
|
-
describe('Config', () => {
|
|
5
|
-
it('creates with provided data', () => {
|
|
6
|
-
const cfg = new Config({ name: 'test' });
|
|
7
|
-
expect(cfg.get('name')).toBe('test');
|
|
8
|
-
});
|
|
9
|
-
|
|
10
|
-
it('creates with no arguments', () => {
|
|
11
|
-
const cfg = new Config();
|
|
12
|
-
expect(cfg.get('anything')).toBeUndefined();
|
|
13
|
-
});
|
|
14
|
-
|
|
15
|
-
it('returns various value types', () => {
|
|
16
|
-
const cfg = new Config({
|
|
17
|
-
str: 'hello',
|
|
18
|
-
num: 42,
|
|
19
|
-
bool: true,
|
|
20
|
-
arr: [1, 2, 3],
|
|
21
|
-
obj: { nested: true },
|
|
22
|
-
nil: null,
|
|
23
|
-
});
|
|
24
|
-
expect(cfg.get('str')).toBe('hello');
|
|
25
|
-
expect(cfg.get('num')).toBe(42);
|
|
26
|
-
expect(cfg.get('bool')).toBe(true);
|
|
27
|
-
expect(cfg.get('arr')).toEqual([1, 2, 3]);
|
|
28
|
-
expect(cfg.get('obj')).toEqual({ nested: true });
|
|
29
|
-
expect(cfg.get('nil')).toBeNull();
|
|
30
|
-
});
|
|
31
|
-
|
|
32
|
-
it('traverses nested objects with dot-path', () => {
|
|
33
|
-
const cfg = new Config({
|
|
34
|
-
database: {
|
|
35
|
-
host: 'db.example.com',
|
|
36
|
-
port: 5432,
|
|
37
|
-
credentials: { user: 'admin', password: 'secret' },
|
|
38
|
-
},
|
|
39
|
-
});
|
|
40
|
-
expect(cfg.get('database.host')).toBe('db.example.com');
|
|
41
|
-
expect(cfg.get('database.port')).toBe(5432);
|
|
42
|
-
expect(cfg.get('database.credentials.user')).toBe('admin');
|
|
43
|
-
});
|
|
44
|
-
|
|
45
|
-
it('returns nested object for partial path', () => {
|
|
46
|
-
const cfg = new Config({ a: { b: { c: 'deep' } } });
|
|
47
|
-
expect(cfg.get('a.b')).toEqual({ c: 'deep' });
|
|
48
|
-
});
|
|
49
|
-
|
|
50
|
-
it('returns undefined when key missing and no default', () => {
|
|
51
|
-
const cfg = new Config({ x: 1 });
|
|
52
|
-
expect(cfg.get('y')).toBeUndefined();
|
|
53
|
-
});
|
|
54
|
-
|
|
55
|
-
it('returns default value when key missing', () => {
|
|
56
|
-
const cfg = new Config({ x: 1 });
|
|
57
|
-
expect(cfg.get('y', 'fallback')).toBe('fallback');
|
|
58
|
-
expect(cfg.get('y', 42)).toBe(42);
|
|
59
|
-
});
|
|
60
|
-
|
|
61
|
-
it('returns default when dot-path partially exists', () => {
|
|
62
|
-
const cfg = new Config({ a: { b: 1 } });
|
|
63
|
-
expect(cfg.get('a.c', 'default')).toBe('default');
|
|
64
|
-
expect(cfg.get('a.b.c.d', 'deep-default')).toBe('deep-default');
|
|
65
|
-
});
|
|
66
|
-
|
|
67
|
-
it('returns default when traversal hits non-object', () => {
|
|
68
|
-
const cfg = new Config({ a: 'string-value' });
|
|
69
|
-
expect(cfg.get('a.b', 'default')).toBe('default');
|
|
70
|
-
});
|
|
71
|
-
|
|
72
|
-
it('returns default when traversal hits null', () => {
|
|
73
|
-
const cfg = new Config({ a: null });
|
|
74
|
-
expect(cfg.get('a.b', 'default')).toBe('default');
|
|
75
|
-
});
|
|
76
|
-
});
|
|
@@ -1,266 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect } from 'vitest';
|
|
2
|
-
import { Context, createIdentity } from '../src/context.js';
|
|
3
|
-
import type { Identity } from '../src/context.js';
|
|
4
|
-
|
|
5
|
-
describe('createIdentity', () => {
|
|
6
|
-
it('creates identity with defaults', () => {
|
|
7
|
-
const id = createIdentity('user1');
|
|
8
|
-
expect(id.id).toBe('user1');
|
|
9
|
-
expect(id.type).toBe('user');
|
|
10
|
-
expect(id.roles).toEqual([]);
|
|
11
|
-
expect(id.attrs).toEqual({});
|
|
12
|
-
});
|
|
13
|
-
|
|
14
|
-
it('creates identity with all fields', () => {
|
|
15
|
-
const id = createIdentity('admin1', 'admin', ['superuser'], { org: 'acme' });
|
|
16
|
-
expect(id.id).toBe('admin1');
|
|
17
|
-
expect(id.type).toBe('admin');
|
|
18
|
-
expect(id.roles).toEqual(['superuser']);
|
|
19
|
-
expect(id.attrs).toEqual({ org: 'acme' });
|
|
20
|
-
});
|
|
21
|
-
|
|
22
|
-
it('returns frozen object', () => {
|
|
23
|
-
const id = createIdentity('u1');
|
|
24
|
-
expect(Object.isFrozen(id)).toBe(true);
|
|
25
|
-
expect(Object.isFrozen(id.roles)).toBe(true);
|
|
26
|
-
expect(Object.isFrozen(id.attrs)).toBe(true);
|
|
27
|
-
});
|
|
28
|
-
});
|
|
29
|
-
|
|
30
|
-
describe('Context.create()', () => {
|
|
31
|
-
it('creates context with unique traceId', () => {
|
|
32
|
-
const ctx1 = Context.create();
|
|
33
|
-
const ctx2 = Context.create();
|
|
34
|
-
expect(ctx1.traceId).toBeDefined();
|
|
35
|
-
expect(ctx2.traceId).toBeDefined();
|
|
36
|
-
expect(ctx1.traceId).not.toBe(ctx2.traceId);
|
|
37
|
-
});
|
|
38
|
-
|
|
39
|
-
it('has null callerId by default', () => {
|
|
40
|
-
const ctx = Context.create();
|
|
41
|
-
expect(ctx.callerId).toBeNull();
|
|
42
|
-
});
|
|
43
|
-
|
|
44
|
-
it('has empty callChain by default', () => {
|
|
45
|
-
const ctx = Context.create();
|
|
46
|
-
expect(ctx.callChain).toEqual([]);
|
|
47
|
-
});
|
|
48
|
-
|
|
49
|
-
it('accepts executor and identity', () => {
|
|
50
|
-
const identity = createIdentity('u1', 'admin');
|
|
51
|
-
const executor = { name: 'test-executor' };
|
|
52
|
-
const ctx = Context.create(executor, identity);
|
|
53
|
-
expect(ctx.executor).toBe(executor);
|
|
54
|
-
expect(ctx.identity).toBe(identity);
|
|
55
|
-
});
|
|
56
|
-
|
|
57
|
-
it('defaults identity to null', () => {
|
|
58
|
-
const ctx = Context.create();
|
|
59
|
-
expect(ctx.identity).toBeNull();
|
|
60
|
-
});
|
|
61
|
-
|
|
62
|
-
it('defaults executor to null', () => {
|
|
63
|
-
const ctx = Context.create();
|
|
64
|
-
expect(ctx.executor).toBeNull();
|
|
65
|
-
});
|
|
66
|
-
|
|
67
|
-
it('defaults data to empty object', () => {
|
|
68
|
-
const ctx = Context.create();
|
|
69
|
-
expect(ctx.data).toEqual({});
|
|
70
|
-
});
|
|
71
|
-
|
|
72
|
-
it('accepts custom data', () => {
|
|
73
|
-
const ctx = Context.create(null, null, { key: 'value' });
|
|
74
|
-
expect(ctx.data).toEqual({ key: 'value' });
|
|
75
|
-
});
|
|
76
|
-
|
|
77
|
-
it('has null redactedInputs by default', () => {
|
|
78
|
-
const ctx = Context.create();
|
|
79
|
-
expect(ctx.redactedInputs).toBeNull();
|
|
80
|
-
});
|
|
81
|
-
});
|
|
82
|
-
|
|
83
|
-
describe('Context.child()', () => {
|
|
84
|
-
it('preserves traceId from parent', () => {
|
|
85
|
-
const parent = Context.create();
|
|
86
|
-
const child = parent.child('module.a');
|
|
87
|
-
expect(child.traceId).toBe(parent.traceId);
|
|
88
|
-
});
|
|
89
|
-
|
|
90
|
-
it('sets callerId to null when parent callChain is empty', () => {
|
|
91
|
-
const parent = Context.create();
|
|
92
|
-
const child = parent.child('module.a');
|
|
93
|
-
expect(child.callerId).toBeNull();
|
|
94
|
-
});
|
|
95
|
-
|
|
96
|
-
it('sets callerId to last element of parent callChain', () => {
|
|
97
|
-
const parent = Context.create();
|
|
98
|
-
const child1 = parent.child('module.a');
|
|
99
|
-
expect(child1.callChain).toEqual(['module.a']);
|
|
100
|
-
|
|
101
|
-
const child2 = child1.child('module.b');
|
|
102
|
-
expect(child2.callerId).toBe('module.a');
|
|
103
|
-
expect(child2.callChain).toEqual(['module.a', 'module.b']);
|
|
104
|
-
});
|
|
105
|
-
|
|
106
|
-
it('builds up callChain through multiple levels', () => {
|
|
107
|
-
const root = Context.create();
|
|
108
|
-
const c1 = root.child('a');
|
|
109
|
-
const c2 = c1.child('b');
|
|
110
|
-
const c3 = c2.child('c');
|
|
111
|
-
expect(c3.callChain).toEqual(['a', 'b', 'c']);
|
|
112
|
-
expect(c3.callerId).toBe('b');
|
|
113
|
-
});
|
|
114
|
-
|
|
115
|
-
it('shares data reference with parent', () => {
|
|
116
|
-
const parent = Context.create(null, null, { shared: true });
|
|
117
|
-
const child = parent.child('mod');
|
|
118
|
-
expect(child.data).toBe(parent.data);
|
|
119
|
-
|
|
120
|
-
child.data['newKey'] = 'newValue';
|
|
121
|
-
expect(parent.data['newKey']).toBe('newValue');
|
|
122
|
-
});
|
|
123
|
-
|
|
124
|
-
it('preserves executor from parent', () => {
|
|
125
|
-
const executor = { id: 'exec' };
|
|
126
|
-
const parent = Context.create(executor);
|
|
127
|
-
const child = parent.child('mod');
|
|
128
|
-
expect(child.executor).toBe(executor);
|
|
129
|
-
});
|
|
130
|
-
|
|
131
|
-
it('preserves identity from parent', () => {
|
|
132
|
-
const identity = createIdentity('u1', 'admin', ['role1']);
|
|
133
|
-
const parent = Context.create(null, identity);
|
|
134
|
-
const child = parent.child('mod');
|
|
135
|
-
expect(child.identity).toBe(identity);
|
|
136
|
-
});
|
|
137
|
-
|
|
138
|
-
it('resets redactedInputs to null', () => {
|
|
139
|
-
const parent = Context.create();
|
|
140
|
-
parent.redactedInputs = { field: '***' };
|
|
141
|
-
const child = parent.child('mod');
|
|
142
|
-
expect(child.redactedInputs).toBeNull();
|
|
143
|
-
});
|
|
144
|
-
|
|
145
|
-
it('does not modify parent callChain', () => {
|
|
146
|
-
const parent = Context.create();
|
|
147
|
-
const chainBefore = [...parent.callChain];
|
|
148
|
-
parent.child('mod.a');
|
|
149
|
-
expect(parent.callChain).toEqual(chainBefore);
|
|
150
|
-
});
|
|
151
|
-
});
|
|
152
|
-
|
|
153
|
-
describe('Context.toJSON() / Context.fromJSON()', () => {
|
|
154
|
-
it('round-trips context with identity', () => {
|
|
155
|
-
const identity = createIdentity('user-42', 'admin', ['superuser', 'editor'], { org: 'acme' });
|
|
156
|
-
const executor = { name: 'test-executor' };
|
|
157
|
-
const original = new Context(
|
|
158
|
-
'trace-abc',
|
|
159
|
-
'caller-1',
|
|
160
|
-
['mod.a', 'mod.b'],
|
|
161
|
-
executor,
|
|
162
|
-
identity,
|
|
163
|
-
{ password: '***' },
|
|
164
|
-
{ transient: 'value' },
|
|
165
|
-
);
|
|
166
|
-
|
|
167
|
-
const serialized = original.toJSON();
|
|
168
|
-
const restored = Context.fromJSON(serialized);
|
|
169
|
-
|
|
170
|
-
expect(restored.traceId).toBe(original.traceId);
|
|
171
|
-
expect(restored.callerId).toBe(original.callerId);
|
|
172
|
-
expect(restored.callChain).toEqual(['mod.a', 'mod.b']);
|
|
173
|
-
expect(restored.identity).not.toBeNull();
|
|
174
|
-
expect(restored.identity!.id).toBe('user-42');
|
|
175
|
-
expect(restored.identity!.type).toBe('admin');
|
|
176
|
-
expect([...restored.identity!.roles]).toEqual(['superuser', 'editor']);
|
|
177
|
-
expect({ ...restored.identity!.attrs }).toEqual({ org: 'acme' });
|
|
178
|
-
expect(restored.redactedInputs).toEqual({ password: '***' });
|
|
179
|
-
expect(restored.data).toEqual({ transient: 'value' });
|
|
180
|
-
});
|
|
181
|
-
|
|
182
|
-
it('round-trips context without identity', () => {
|
|
183
|
-
const original = new Context('trace-xyz', null, [], null, null, null);
|
|
184
|
-
const serialized = original.toJSON();
|
|
185
|
-
const restored = Context.fromJSON(serialized);
|
|
186
|
-
|
|
187
|
-
expect(restored.traceId).toBe('trace-xyz');
|
|
188
|
-
expect(restored.callerId).toBeNull();
|
|
189
|
-
expect(restored.callChain).toEqual([]);
|
|
190
|
-
expect(restored.identity).toBeNull();
|
|
191
|
-
expect(restored.redactedInputs).toBeNull();
|
|
192
|
-
});
|
|
193
|
-
|
|
194
|
-
it('excludes executor from toJSON output but includes data', () => {
|
|
195
|
-
const ctx = new Context('trace-1', null, [], 'my-executor', null, null, { key: 'included' });
|
|
196
|
-
const serialized = ctx.toJSON();
|
|
197
|
-
|
|
198
|
-
expect(serialized).not.toHaveProperty('executor');
|
|
199
|
-
expect(serialized).toHaveProperty('data');
|
|
200
|
-
expect(serialized.data).toEqual({ key: 'included' });
|
|
201
|
-
});
|
|
202
|
-
|
|
203
|
-
it('re-injects executor via fromJSON', () => {
|
|
204
|
-
const ctx = new Context('trace-2', null, [], 'original-exec');
|
|
205
|
-
const serialized = ctx.toJSON();
|
|
206
|
-
const newExecutor = { name: 'new-executor' };
|
|
207
|
-
const restored = Context.fromJSON(serialized, newExecutor);
|
|
208
|
-
|
|
209
|
-
expect(restored.executor).toBe(newExecutor);
|
|
210
|
-
});
|
|
211
|
-
|
|
212
|
-
it('defaults executor to null when not provided to fromJSON', () => {
|
|
213
|
-
const serialized = { traceId: 't1', callerId: null, callChain: [], identity: null, redactedInputs: null };
|
|
214
|
-
const restored = Context.fromJSON(serialized);
|
|
215
|
-
expect(restored.executor).toBeNull();
|
|
216
|
-
});
|
|
217
|
-
|
|
218
|
-
it('toJSON returns copies, not references', () => {
|
|
219
|
-
const identity = createIdentity('u1', 'user', ['r1'], { k: 'v' });
|
|
220
|
-
const ctx = new Context('t1', null, ['a', 'b'], null, identity, { field: 'val' }, { shared: true });
|
|
221
|
-
const serialized = ctx.toJSON();
|
|
222
|
-
|
|
223
|
-
// Mutate serialized copies
|
|
224
|
-
(serialized.callChain as string[]).push('mutated');
|
|
225
|
-
(serialized.identity as Record<string, unknown>).id = 'mutated';
|
|
226
|
-
(serialized.redactedInputs as Record<string, unknown>).extra = true;
|
|
227
|
-
(serialized.data as Record<string, unknown>).extra = true;
|
|
228
|
-
|
|
229
|
-
// Originals unchanged
|
|
230
|
-
expect(ctx.callChain).toEqual(['a', 'b']);
|
|
231
|
-
expect(ctx.identity!.id).toBe('u1');
|
|
232
|
-
expect(ctx.redactedInputs).toEqual({ field: 'val' });
|
|
233
|
-
expect(ctx.data).toEqual({ shared: true });
|
|
234
|
-
});
|
|
235
|
-
|
|
236
|
-
it('toJSON excludes internal keys starting with _', () => {
|
|
237
|
-
const ctx = Context.create();
|
|
238
|
-
ctx.data['visible'] = 'yes';
|
|
239
|
-
ctx.data['_tracing_spans'] = [1, 2, 3];
|
|
240
|
-
ctx.data['_internal'] = 42;
|
|
241
|
-
const serialized = ctx.toJSON();
|
|
242
|
-
const data = serialized.data as Record<string, unknown>;
|
|
243
|
-
expect(data['visible']).toBe('yes');
|
|
244
|
-
expect(data['_tracing_spans']).toBeUndefined();
|
|
245
|
-
expect(data['_internal']).toBeUndefined();
|
|
246
|
-
});
|
|
247
|
-
|
|
248
|
-
it('fromJSON handles null roles and attrs gracefully', () => {
|
|
249
|
-
const json = {
|
|
250
|
-
traceId: 'test-id',
|
|
251
|
-
callerId: null,
|
|
252
|
-
callChain: [],
|
|
253
|
-
identity: {
|
|
254
|
-
id: 'user-1',
|
|
255
|
-
type: 'user',
|
|
256
|
-
roles: null,
|
|
257
|
-
attrs: null,
|
|
258
|
-
},
|
|
259
|
-
data: {},
|
|
260
|
-
};
|
|
261
|
-
const ctx = Context.fromJSON(json);
|
|
262
|
-
expect(ctx.identity).toBeDefined();
|
|
263
|
-
expect(ctx.identity!.roles).toEqual([]);
|
|
264
|
-
expect(ctx.identity!.attrs).toEqual({});
|
|
265
|
-
});
|
|
266
|
-
});
|
|
@@ -1,173 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect } from 'vitest';
|
|
2
|
-
import { Type } from '@sinclair/typebox';
|
|
3
|
-
import { FunctionModule, module, normalizeResult, makeAutoId } from '../src/decorator.js';
|
|
4
|
-
import { Context, createIdentity } from '../src/context.js';
|
|
5
|
-
import { Registry } from '../src/registry/registry.js';
|
|
6
|
-
|
|
7
|
-
const inputSchema = Type.Object({ name: Type.String() });
|
|
8
|
-
const outputSchema = Type.Object({ greeting: Type.String() });
|
|
9
|
-
|
|
10
|
-
describe('FunctionModule', () => {
|
|
11
|
-
it('wraps an execute function', async () => {
|
|
12
|
-
const fm = new FunctionModule({
|
|
13
|
-
execute: (inputs) => ({ greeting: `Hello, ${inputs['name']}!` }),
|
|
14
|
-
moduleId: 'greet.hello',
|
|
15
|
-
inputSchema,
|
|
16
|
-
outputSchema,
|
|
17
|
-
description: 'Says hello',
|
|
18
|
-
});
|
|
19
|
-
const ctx = Context.create(null, createIdentity('test-user'));
|
|
20
|
-
const result = await fm.execute({ name: 'World' }, ctx);
|
|
21
|
-
expect(result).toEqual({ greeting: 'Hello, World!' });
|
|
22
|
-
});
|
|
23
|
-
|
|
24
|
-
it('exposes correct properties', () => {
|
|
25
|
-
const fm = new FunctionModule({
|
|
26
|
-
execute: () => ({}),
|
|
27
|
-
moduleId: 'test.props',
|
|
28
|
-
inputSchema,
|
|
29
|
-
outputSchema,
|
|
30
|
-
description: 'Test props',
|
|
31
|
-
documentation: 'Some docs',
|
|
32
|
-
tags: ['tag1', 'tag2'],
|
|
33
|
-
version: '2.0.0',
|
|
34
|
-
});
|
|
35
|
-
expect(fm.moduleId).toBe('test.props');
|
|
36
|
-
expect(fm.description).toBe('Test props');
|
|
37
|
-
expect(fm.documentation).toBe('Some docs');
|
|
38
|
-
expect(fm.tags).toEqual(['tag1', 'tag2']);
|
|
39
|
-
expect(fm.version).toBe('2.0.0');
|
|
40
|
-
});
|
|
41
|
-
|
|
42
|
-
it('uses sensible defaults', () => {
|
|
43
|
-
const fm = new FunctionModule({
|
|
44
|
-
execute: () => ({}),
|
|
45
|
-
moduleId: 'test.defaults',
|
|
46
|
-
inputSchema,
|
|
47
|
-
outputSchema,
|
|
48
|
-
});
|
|
49
|
-
expect(fm.description).toBe('Module test.defaults');
|
|
50
|
-
expect(fm.documentation).toBeNull();
|
|
51
|
-
expect(fm.tags).toBeNull();
|
|
52
|
-
expect(fm.version).toBe('1.0.0');
|
|
53
|
-
expect(fm.annotations).toBeNull();
|
|
54
|
-
expect(fm.metadata).toBeNull();
|
|
55
|
-
expect(fm.examples).toBeNull();
|
|
56
|
-
});
|
|
57
|
-
|
|
58
|
-
it('normalizes null return value', async () => {
|
|
59
|
-
const fm = new FunctionModule({
|
|
60
|
-
execute: () => null as unknown as Record<string, unknown>,
|
|
61
|
-
moduleId: 'test.normalize',
|
|
62
|
-
inputSchema,
|
|
63
|
-
outputSchema,
|
|
64
|
-
});
|
|
65
|
-
const ctx = Context.create(null, createIdentity('test-user'));
|
|
66
|
-
const result = await fm.execute({}, ctx);
|
|
67
|
-
expect(result).toEqual({});
|
|
68
|
-
});
|
|
69
|
-
});
|
|
70
|
-
|
|
71
|
-
describe('normalizeResult', () => {
|
|
72
|
-
it('null returns empty object', () => {
|
|
73
|
-
expect(normalizeResult(null)).toEqual({});
|
|
74
|
-
});
|
|
75
|
-
|
|
76
|
-
it('undefined returns empty object', () => {
|
|
77
|
-
expect(normalizeResult(undefined)).toEqual({});
|
|
78
|
-
});
|
|
79
|
-
|
|
80
|
-
it('plain object passes through', () => {
|
|
81
|
-
const obj = { a: 1, b: 'two' };
|
|
82
|
-
expect(normalizeResult(obj)).toBe(obj);
|
|
83
|
-
});
|
|
84
|
-
|
|
85
|
-
it('string is wrapped in { result }', () => {
|
|
86
|
-
expect(normalizeResult('hello')).toEqual({ result: 'hello' });
|
|
87
|
-
});
|
|
88
|
-
|
|
89
|
-
it('number is wrapped in { result }', () => {
|
|
90
|
-
expect(normalizeResult(42)).toEqual({ result: 42 });
|
|
91
|
-
});
|
|
92
|
-
|
|
93
|
-
it('boolean is wrapped in { result }', () => {
|
|
94
|
-
expect(normalizeResult(true)).toEqual({ result: true });
|
|
95
|
-
});
|
|
96
|
-
|
|
97
|
-
it('array is wrapped in { result }', () => {
|
|
98
|
-
expect(normalizeResult([1, 2, 3])).toEqual({ result: [1, 2, 3] });
|
|
99
|
-
});
|
|
100
|
-
});
|
|
101
|
-
|
|
102
|
-
describe('module() factory', () => {
|
|
103
|
-
it('creates FunctionModule with correct properties', () => {
|
|
104
|
-
const fm = module({
|
|
105
|
-
id: 'factory.test',
|
|
106
|
-
inputSchema,
|
|
107
|
-
outputSchema,
|
|
108
|
-
description: 'Factory module',
|
|
109
|
-
execute: () => ({ greeting: 'hi' }),
|
|
110
|
-
});
|
|
111
|
-
expect(fm).toBeInstanceOf(FunctionModule);
|
|
112
|
-
expect(fm.moduleId).toBe('factory.test');
|
|
113
|
-
expect(fm.description).toBe('Factory module');
|
|
114
|
-
});
|
|
115
|
-
|
|
116
|
-
it('generates auto ID when not provided', () => {
|
|
117
|
-
const fm = module({
|
|
118
|
-
inputSchema,
|
|
119
|
-
outputSchema,
|
|
120
|
-
execute: () => ({}),
|
|
121
|
-
});
|
|
122
|
-
expect(fm.moduleId).toBe('anonymous');
|
|
123
|
-
});
|
|
124
|
-
|
|
125
|
-
it('passes through optional fields', () => {
|
|
126
|
-
const fm = module({
|
|
127
|
-
id: 'opts.check',
|
|
128
|
-
inputSchema,
|
|
129
|
-
outputSchema,
|
|
130
|
-
description: 'desc',
|
|
131
|
-
documentation: 'docs here',
|
|
132
|
-
tags: ['t1'],
|
|
133
|
-
version: '3.0.0',
|
|
134
|
-
metadata: { key: 'val' },
|
|
135
|
-
execute: () => ({}),
|
|
136
|
-
});
|
|
137
|
-
expect(fm.documentation).toBe('docs here');
|
|
138
|
-
expect(fm.tags).toEqual(['t1']);
|
|
139
|
-
expect(fm.version).toBe('3.0.0');
|
|
140
|
-
expect(fm.metadata).toEqual({ key: 'val' });
|
|
141
|
-
});
|
|
142
|
-
|
|
143
|
-
it('auto-registers with registry', () => {
|
|
144
|
-
const registry = new Registry();
|
|
145
|
-
const fm = module({
|
|
146
|
-
id: 'auto.registered',
|
|
147
|
-
inputSchema,
|
|
148
|
-
outputSchema,
|
|
149
|
-
execute: () => ({ ok: true }),
|
|
150
|
-
registry,
|
|
151
|
-
});
|
|
152
|
-
expect(registry.has('auto.registered')).toBe(true);
|
|
153
|
-
expect(registry.get('auto.registered')).toBe(fm);
|
|
154
|
-
});
|
|
155
|
-
});
|
|
156
|
-
|
|
157
|
-
describe('makeAutoId', () => {
|
|
158
|
-
it('lowercases and replaces non-alphanumeric', () => {
|
|
159
|
-
expect(makeAutoId('Hello World')).toBe('hello_world');
|
|
160
|
-
});
|
|
161
|
-
|
|
162
|
-
it('preserves dots', () => {
|
|
163
|
-
expect(makeAutoId('my.module.name')).toBe('my.module.name');
|
|
164
|
-
});
|
|
165
|
-
|
|
166
|
-
it('prefixes digit-leading segments', () => {
|
|
167
|
-
expect(makeAutoId('2fast.4you')).toBe('_2fast._4you');
|
|
168
|
-
});
|
|
169
|
-
|
|
170
|
-
it('handles valid IDs unchanged', () => {
|
|
171
|
-
expect(makeAutoId('valid_id')).toBe('valid_id');
|
|
172
|
-
});
|
|
173
|
-
});
|