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
package/README.md
CHANGED
package/dist/acl.d.ts
ADDED
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ACL (Access Control List) types and implementation for apcore.
|
|
3
|
+
*/
|
|
4
|
+
import type { Context } from './context.js';
|
|
5
|
+
export interface ACLRule {
|
|
6
|
+
callers: string[];
|
|
7
|
+
targets: string[];
|
|
8
|
+
effect: string;
|
|
9
|
+
description: string;
|
|
10
|
+
conditions?: Record<string, unknown> | null;
|
|
11
|
+
}
|
|
12
|
+
export declare class ACL {
|
|
13
|
+
private _rules;
|
|
14
|
+
private _defaultEffect;
|
|
15
|
+
private _yamlPath;
|
|
16
|
+
debug: boolean;
|
|
17
|
+
constructor(rules: ACLRule[], defaultEffect?: string);
|
|
18
|
+
static load(yamlPath: string): ACL;
|
|
19
|
+
check(callerId: string | null, targetId: string, context?: Context | null): boolean;
|
|
20
|
+
private _matchPattern;
|
|
21
|
+
private _matchesRule;
|
|
22
|
+
private _checkConditions;
|
|
23
|
+
addRule(rule: ACLRule): void;
|
|
24
|
+
removeRule(callers: string[], targets: string[]): boolean;
|
|
25
|
+
reload(): void;
|
|
26
|
+
}
|
|
27
|
+
//# sourceMappingURL=acl.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"acl.d.ts","sourceRoot":"","sources":["../src/acl.ts"],"names":[],"mappings":"AAAA;;GAEG;AAGH,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,cAAc,CAAC;AAQ5C,MAAM,WAAW,OAAO;IACtB,OAAO,EAAE,MAAM,EAAE,CAAC;IAClB,OAAO,EAAE,MAAM,EAAE,CAAC;IAClB,MAAM,EAAE,MAAM,CAAC;IACf,WAAW,EAAE,MAAM,CAAC;IACpB,UAAU,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,IAAI,CAAC;CAC7C;AAsCD,qBAAa,GAAG;IACd,OAAO,CAAC,MAAM,CAAY;IAC1B,OAAO,CAAC,cAAc,CAAS;IAC/B,OAAO,CAAC,SAAS,CAAuB;IACxC,KAAK,EAAE,OAAO,CAAS;gBAEX,KAAK,EAAE,OAAO,EAAE,EAAE,aAAa,GAAE,MAAe;IAQ5D,MAAM,CAAC,IAAI,CAAC,QAAQ,EAAE,MAAM,GAAG,GAAG;IAqClC,KAAK,CAAC,QAAQ,EAAE,MAAM,GAAG,IAAI,EAAE,QAAQ,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,OAAO,GAAG,IAAI,GAAG,OAAO;IAYnF,OAAO,CAAC,aAAa;IAQrB,OAAO,CAAC,YAAY;IAcpB,OAAO,CAAC,gBAAgB;IAmCxB,OAAO,CAAC,IAAI,EAAE,OAAO,GAAG,IAAI;IAI5B,UAAU,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE,OAAO,EAAE,MAAM,EAAE,GAAG,OAAO;IAczD,MAAM,IAAI,IAAI;CAQf"}
|
package/dist/acl.js
ADDED
|
@@ -0,0 +1,175 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ACL (Access Control List) types and implementation for apcore.
|
|
3
|
+
*/
|
|
4
|
+
import yaml from 'js-yaml';
|
|
5
|
+
import { ACLRuleError, ConfigNotFoundError } from './errors.js';
|
|
6
|
+
import { matchPattern } from './utils/pattern.js';
|
|
7
|
+
// Lazy-load Node.js built-in modules for browser compatibility
|
|
8
|
+
let _nodeFs = null;
|
|
9
|
+
try {
|
|
10
|
+
_nodeFs = await import('node:fs');
|
|
11
|
+
}
|
|
12
|
+
catch { /* browser environment */ }
|
|
13
|
+
function parseAclRule(rawRule, index) {
|
|
14
|
+
if (typeof rawRule !== 'object' || rawRule === null || Array.isArray(rawRule)) {
|
|
15
|
+
throw new ACLRuleError(`Rule ${index} must be a mapping, got ${typeof rawRule}`);
|
|
16
|
+
}
|
|
17
|
+
const ruleObj = rawRule;
|
|
18
|
+
for (const key of ['callers', 'targets', 'effect']) {
|
|
19
|
+
if (!(key in ruleObj)) {
|
|
20
|
+
throw new ACLRuleError(`Rule ${index} missing required key '${key}'`);
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
const effect = ruleObj['effect'];
|
|
24
|
+
if (effect !== 'allow' && effect !== 'deny') {
|
|
25
|
+
throw new ACLRuleError(`Rule ${index} has invalid effect '${effect}', must be 'allow' or 'deny'`);
|
|
26
|
+
}
|
|
27
|
+
const callers = ruleObj['callers'];
|
|
28
|
+
if (!Array.isArray(callers)) {
|
|
29
|
+
throw new ACLRuleError(`Rule ${index} 'callers' must be a list, got ${typeof callers}`);
|
|
30
|
+
}
|
|
31
|
+
const targets = ruleObj['targets'];
|
|
32
|
+
if (!Array.isArray(targets)) {
|
|
33
|
+
throw new ACLRuleError(`Rule ${index} 'targets' must be a list, got ${typeof targets}`);
|
|
34
|
+
}
|
|
35
|
+
return {
|
|
36
|
+
callers: callers,
|
|
37
|
+
targets: targets,
|
|
38
|
+
effect,
|
|
39
|
+
description: ruleObj['description'] ?? '',
|
|
40
|
+
conditions: ruleObj['conditions'] ?? null,
|
|
41
|
+
};
|
|
42
|
+
}
|
|
43
|
+
export class ACL {
|
|
44
|
+
_rules;
|
|
45
|
+
_defaultEffect;
|
|
46
|
+
_yamlPath = null;
|
|
47
|
+
debug = false;
|
|
48
|
+
constructor(rules, defaultEffect = 'deny') {
|
|
49
|
+
if (defaultEffect !== 'allow' && defaultEffect !== 'deny') {
|
|
50
|
+
throw new ACLRuleError(`Invalid default_effect '${defaultEffect}', must be 'allow' or 'deny'`);
|
|
51
|
+
}
|
|
52
|
+
this._rules = [...rules];
|
|
53
|
+
this._defaultEffect = defaultEffect;
|
|
54
|
+
}
|
|
55
|
+
static load(yamlPath) {
|
|
56
|
+
const { existsSync, readFileSync } = _nodeFs;
|
|
57
|
+
if (!existsSync(yamlPath)) {
|
|
58
|
+
throw new ConfigNotFoundError(yamlPath);
|
|
59
|
+
}
|
|
60
|
+
let data;
|
|
61
|
+
try {
|
|
62
|
+
const content = readFileSync(yamlPath, 'utf-8');
|
|
63
|
+
data = yaml.load(content);
|
|
64
|
+
}
|
|
65
|
+
catch (e) {
|
|
66
|
+
if (e instanceof ConfigNotFoundError)
|
|
67
|
+
throw e;
|
|
68
|
+
throw new ACLRuleError(`Invalid YAML in ${yamlPath}: ${e}`);
|
|
69
|
+
}
|
|
70
|
+
if (typeof data !== 'object' || data === null || Array.isArray(data)) {
|
|
71
|
+
throw new ACLRuleError(`ACL config must be a mapping, got ${typeof data}`);
|
|
72
|
+
}
|
|
73
|
+
const dataObj = data;
|
|
74
|
+
if (!('rules' in dataObj)) {
|
|
75
|
+
throw new ACLRuleError("ACL config missing required 'rules' key");
|
|
76
|
+
}
|
|
77
|
+
const rawRules = dataObj['rules'];
|
|
78
|
+
if (!Array.isArray(rawRules)) {
|
|
79
|
+
throw new ACLRuleError(`'rules' must be a list, got ${typeof rawRules}`);
|
|
80
|
+
}
|
|
81
|
+
const defaultEffect = dataObj['default_effect'] ?? 'deny';
|
|
82
|
+
const rules = rawRules.map((raw, i) => parseAclRule(raw, i));
|
|
83
|
+
const acl = new ACL(rules, defaultEffect);
|
|
84
|
+
acl._yamlPath = yamlPath;
|
|
85
|
+
return acl;
|
|
86
|
+
}
|
|
87
|
+
check(callerId, targetId, context) {
|
|
88
|
+
const effectiveCaller = callerId === null ? '@external' : callerId;
|
|
89
|
+
for (const rule of this._rules) {
|
|
90
|
+
if (this._matchesRule(rule, effectiveCaller, targetId, context ?? null)) {
|
|
91
|
+
return rule.effect === 'allow';
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
return this._defaultEffect === 'allow';
|
|
95
|
+
}
|
|
96
|
+
_matchPattern(pattern, value, context) {
|
|
97
|
+
if (pattern === '@external')
|
|
98
|
+
return value === '@external';
|
|
99
|
+
if (pattern === '@system') {
|
|
100
|
+
return context !== null && context.identity !== null && context.identity.type === 'system';
|
|
101
|
+
}
|
|
102
|
+
return matchPattern(pattern, value);
|
|
103
|
+
}
|
|
104
|
+
_matchesRule(rule, caller, target, context) {
|
|
105
|
+
const callerMatch = rule.callers.some((p) => this._matchPattern(p, caller, context));
|
|
106
|
+
if (!callerMatch)
|
|
107
|
+
return false;
|
|
108
|
+
const targetMatch = rule.targets.some((p) => this._matchPattern(p, target, context));
|
|
109
|
+
if (!targetMatch)
|
|
110
|
+
return false;
|
|
111
|
+
if (rule.conditions != null) {
|
|
112
|
+
if (!this._checkConditions(rule.conditions, context))
|
|
113
|
+
return false;
|
|
114
|
+
}
|
|
115
|
+
return true;
|
|
116
|
+
}
|
|
117
|
+
_checkConditions(conditions, context) {
|
|
118
|
+
if (context === null)
|
|
119
|
+
return false;
|
|
120
|
+
if ('identity_types' in conditions) {
|
|
121
|
+
const types = conditions['identity_types'];
|
|
122
|
+
if (!Array.isArray(types)) {
|
|
123
|
+
console.warn('[apcore:acl] identity_types condition must be an array');
|
|
124
|
+
return false;
|
|
125
|
+
}
|
|
126
|
+
if (context.identity === null || !types.includes(context.identity.type))
|
|
127
|
+
return false;
|
|
128
|
+
}
|
|
129
|
+
if ('roles' in conditions) {
|
|
130
|
+
const roles = conditions['roles'];
|
|
131
|
+
if (!Array.isArray(roles)) {
|
|
132
|
+
console.warn('[apcore:acl] roles condition must be an array');
|
|
133
|
+
return false;
|
|
134
|
+
}
|
|
135
|
+
if (context.identity === null)
|
|
136
|
+
return false;
|
|
137
|
+
const identityRoles = new Set(context.identity.roles);
|
|
138
|
+
if (!roles.some((r) => identityRoles.has(r)))
|
|
139
|
+
return false;
|
|
140
|
+
}
|
|
141
|
+
if ('max_call_depth' in conditions) {
|
|
142
|
+
const maxDepth = conditions['max_call_depth'];
|
|
143
|
+
if (typeof maxDepth !== 'number') {
|
|
144
|
+
console.warn('[apcore:acl] max_call_depth condition must be a number');
|
|
145
|
+
return false;
|
|
146
|
+
}
|
|
147
|
+
if (context.callChain.length > maxDepth)
|
|
148
|
+
return false;
|
|
149
|
+
}
|
|
150
|
+
return true;
|
|
151
|
+
}
|
|
152
|
+
addRule(rule) {
|
|
153
|
+
this._rules.unshift(rule);
|
|
154
|
+
}
|
|
155
|
+
removeRule(callers, targets) {
|
|
156
|
+
for (let i = 0; i < this._rules.length; i++) {
|
|
157
|
+
const rule = this._rules[i];
|
|
158
|
+
if (JSON.stringify(rule.callers) === JSON.stringify(callers) &&
|
|
159
|
+
JSON.stringify(rule.targets) === JSON.stringify(targets)) {
|
|
160
|
+
this._rules.splice(i, 1);
|
|
161
|
+
return true;
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
return false;
|
|
165
|
+
}
|
|
166
|
+
reload() {
|
|
167
|
+
if (this._yamlPath === null) {
|
|
168
|
+
throw new ACLRuleError('Cannot reload: ACL was not loaded from a YAML file');
|
|
169
|
+
}
|
|
170
|
+
const reloaded = ACL.load(this._yamlPath);
|
|
171
|
+
this._rules = reloaded._rules;
|
|
172
|
+
this._defaultEffect = reloaded._defaultEffect;
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
//# sourceMappingURL=acl.js.map
|
package/dist/acl.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"acl.js","sourceRoot":"","sources":["../src/acl.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Approval system: interfaces, data types, built-in handlers.
|
|
3
|
+
*
|
|
4
|
+
* Provides a pluggable gate at Executor Step 4.5, between ACL enforcement
|
|
5
|
+
* and input validation. When a module declares requiresApproval=true and
|
|
6
|
+
* an ApprovalHandler is configured, the handler is invoked before execution.
|
|
7
|
+
*/
|
|
8
|
+
import type { Context } from './context.js';
|
|
9
|
+
import type { ModuleAnnotations } from './module.js';
|
|
10
|
+
/**
|
|
11
|
+
* Carries invocation context to the approval handler.
|
|
12
|
+
*/
|
|
13
|
+
export interface ApprovalRequest {
|
|
14
|
+
readonly moduleId: string;
|
|
15
|
+
readonly arguments: Record<string, unknown>;
|
|
16
|
+
readonly context: Context;
|
|
17
|
+
readonly annotations: ModuleAnnotations;
|
|
18
|
+
readonly description: string | null;
|
|
19
|
+
readonly tags: readonly string[];
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Create a frozen ApprovalRequest.
|
|
23
|
+
*/
|
|
24
|
+
export declare function createApprovalRequest(options: {
|
|
25
|
+
moduleId: string;
|
|
26
|
+
arguments: Record<string, unknown>;
|
|
27
|
+
context: Context;
|
|
28
|
+
annotations: ModuleAnnotations;
|
|
29
|
+
description?: string | null;
|
|
30
|
+
tags?: string[];
|
|
31
|
+
}): ApprovalRequest;
|
|
32
|
+
/**
|
|
33
|
+
* Carries the approval handler's decision.
|
|
34
|
+
*/
|
|
35
|
+
export interface ApprovalResult {
|
|
36
|
+
readonly status: 'approved' | 'rejected' | 'timeout' | 'pending';
|
|
37
|
+
readonly approvedBy: string | null;
|
|
38
|
+
readonly reason: string | null;
|
|
39
|
+
readonly approvalId: string | null;
|
|
40
|
+
readonly metadata: Record<string, unknown> | null;
|
|
41
|
+
}
|
|
42
|
+
/**
|
|
43
|
+
* Create a frozen ApprovalResult.
|
|
44
|
+
*/
|
|
45
|
+
export declare function createApprovalResult(options: {
|
|
46
|
+
status: 'approved' | 'rejected' | 'timeout' | 'pending';
|
|
47
|
+
approvedBy?: string | null;
|
|
48
|
+
reason?: string | null;
|
|
49
|
+
approvalId?: string | null;
|
|
50
|
+
metadata?: Record<string, unknown> | null;
|
|
51
|
+
}): ApprovalResult;
|
|
52
|
+
/**
|
|
53
|
+
* Protocol for pluggable approval handlers.
|
|
54
|
+
*
|
|
55
|
+
* Implementations receive an ApprovalRequest and return an ApprovalResult.
|
|
56
|
+
* Both methods are asynchronous.
|
|
57
|
+
*/
|
|
58
|
+
export interface ApprovalHandler {
|
|
59
|
+
requestApproval(request: ApprovalRequest): Promise<ApprovalResult>;
|
|
60
|
+
checkApproval(approvalId: string): Promise<ApprovalResult>;
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* Built-in handler that always rejects. Safe default for enforcement.
|
|
64
|
+
*/
|
|
65
|
+
export declare class AlwaysDenyHandler implements ApprovalHandler {
|
|
66
|
+
requestApproval(_request: ApprovalRequest): Promise<ApprovalResult>;
|
|
67
|
+
checkApproval(_approvalId: string): Promise<ApprovalResult>;
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* Built-in handler that always approves. For testing and development.
|
|
71
|
+
*/
|
|
72
|
+
export declare class AutoApproveHandler implements ApprovalHandler {
|
|
73
|
+
requestApproval(_request: ApprovalRequest): Promise<ApprovalResult>;
|
|
74
|
+
checkApproval(_approvalId: string): Promise<ApprovalResult>;
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Built-in handler that delegates to a user-provided async callback.
|
|
78
|
+
*/
|
|
79
|
+
export declare class CallbackApprovalHandler implements ApprovalHandler {
|
|
80
|
+
private _callback;
|
|
81
|
+
constructor(callback: (request: ApprovalRequest) => Promise<ApprovalResult>);
|
|
82
|
+
requestApproval(request: ApprovalRequest): Promise<ApprovalResult>;
|
|
83
|
+
checkApproval(_approvalId: string): Promise<ApprovalResult>;
|
|
84
|
+
}
|
|
85
|
+
//# sourceMappingURL=approval.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"approval.d.ts","sourceRoot":"","sources":["../src/approval.ts"],"names":[],"mappings":"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"}
|
package/dist/approval.js
ADDED
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Approval system: interfaces, data types, built-in handlers.
|
|
3
|
+
*
|
|
4
|
+
* Provides a pluggable gate at Executor Step 4.5, between ACL enforcement
|
|
5
|
+
* and input validation. When a module declares requiresApproval=true and
|
|
6
|
+
* an ApprovalHandler is configured, the handler is invoked before execution.
|
|
7
|
+
*/
|
|
8
|
+
/**
|
|
9
|
+
* Create a frozen ApprovalRequest.
|
|
10
|
+
*/
|
|
11
|
+
export function createApprovalRequest(options) {
|
|
12
|
+
return Object.freeze({
|
|
13
|
+
moduleId: options.moduleId,
|
|
14
|
+
arguments: options.arguments,
|
|
15
|
+
context: options.context,
|
|
16
|
+
annotations: options.annotations,
|
|
17
|
+
description: options.description ?? null,
|
|
18
|
+
tags: Object.freeze([...(options.tags ?? [])]),
|
|
19
|
+
});
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Create a frozen ApprovalResult.
|
|
23
|
+
*/
|
|
24
|
+
export function createApprovalResult(options) {
|
|
25
|
+
return Object.freeze({
|
|
26
|
+
status: options.status,
|
|
27
|
+
approvedBy: options.approvedBy ?? null,
|
|
28
|
+
reason: options.reason ?? null,
|
|
29
|
+
approvalId: options.approvalId ?? null,
|
|
30
|
+
metadata: options.metadata ?? null,
|
|
31
|
+
});
|
|
32
|
+
}
|
|
33
|
+
/**
|
|
34
|
+
* Built-in handler that always rejects. Safe default for enforcement.
|
|
35
|
+
*/
|
|
36
|
+
export class AlwaysDenyHandler {
|
|
37
|
+
async requestApproval(_request) {
|
|
38
|
+
return createApprovalResult({ status: 'rejected', reason: 'Always denied' });
|
|
39
|
+
}
|
|
40
|
+
async checkApproval(_approvalId) {
|
|
41
|
+
return createApprovalResult({ status: 'rejected', reason: 'Always denied' });
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Built-in handler that always approves. For testing and development.
|
|
46
|
+
*/
|
|
47
|
+
export class AutoApproveHandler {
|
|
48
|
+
async requestApproval(_request) {
|
|
49
|
+
return createApprovalResult({ status: 'approved', approvedBy: 'auto' });
|
|
50
|
+
}
|
|
51
|
+
async checkApproval(_approvalId) {
|
|
52
|
+
return createApprovalResult({ status: 'approved', approvedBy: 'auto' });
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* Built-in handler that delegates to a user-provided async callback.
|
|
57
|
+
*/
|
|
58
|
+
export class CallbackApprovalHandler {
|
|
59
|
+
_callback;
|
|
60
|
+
constructor(callback) {
|
|
61
|
+
this._callback = callback;
|
|
62
|
+
}
|
|
63
|
+
async requestApproval(request) {
|
|
64
|
+
return this._callback(request);
|
|
65
|
+
}
|
|
66
|
+
async checkApproval(_approvalId) {
|
|
67
|
+
return createApprovalResult({
|
|
68
|
+
status: 'rejected',
|
|
69
|
+
reason: 'Phase B not supported by callback handler',
|
|
70
|
+
});
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
//# sourceMappingURL=approval.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"approval.js","sourceRoot":"","sources":["../src/approval.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAiBH;;GAEG;AACH,MAAM,UAAU,qBAAqB,CAAC,OAOrC;IACC,OAAO,MAAM,CAAC,MAAM,CAAC;QACnB,QAAQ,EAAE,OAAO,CAAC,QAAQ;QAC1B,SAAS,EAAE,OAAO,CAAC,SAAS;QAC5B,OAAO,EAAE,OAAO,CAAC,OAAO;QACxB,WAAW,EAAE,OAAO,CAAC,WAAW;QAChC,WAAW,EAAE,OAAO,CAAC,WAAW,IAAI,IAAI;QACxC,IAAI,EAAE,MAAM,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,IAAI,EAAE,CAAC,CAAC,CAAC;KAC/C,CAAC,CAAC;AACL,CAAC;AAaD;;GAEG;AACH,MAAM,UAAU,oBAAoB,CAAC,OAMpC;IACC,OAAO,MAAM,CAAC,MAAM,CAAC;QACnB,MAAM,EAAE,OAAO,CAAC,MAAM;QACtB,UAAU,EAAE,OAAO,CAAC,UAAU,IAAI,IAAI;QACtC,MAAM,EAAE,OAAO,CAAC,MAAM,IAAI,IAAI;QAC9B,UAAU,EAAE,OAAO,CAAC,UAAU,IAAI,IAAI;QACtC,QAAQ,EAAE,OAAO,CAAC,QAAQ,IAAI,IAAI;KACnC,CAAC,CAAC;AACL,CAAC;AAaD;;GAEG;AACH,MAAM,OAAO,iBAAiB;IAC5B,KAAK,CAAC,eAAe,CAAC,QAAyB;QAC7C,OAAO,oBAAoB,CAAC,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,eAAe,EAAE,CAAC,CAAC;IAC/E,CAAC;IAED,KAAK,CAAC,aAAa,CAAC,WAAmB;QACrC,OAAO,oBAAoB,CAAC,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,eAAe,EAAE,CAAC,CAAC;IAC/E,CAAC;CACF;AAED;;GAEG;AACH,MAAM,OAAO,kBAAkB;IAC7B,KAAK,CAAC,eAAe,CAAC,QAAyB;QAC7C,OAAO,oBAAoB,CAAC,EAAE,MAAM,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,EAAE,CAAC,CAAC;IAC1E,CAAC;IAED,KAAK,CAAC,aAAa,CAAC,WAAmB;QACrC,OAAO,oBAAoB,CAAC,EAAE,MAAM,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,EAAE,CAAC,CAAC;IAC1E,CAAC;CACF;AAED;;GAEG;AACH,MAAM,OAAO,uBAAuB;IAC1B,SAAS,CAAwD;IAEzE,YAAY,QAA+D;QACzE,IAAI,CAAC,SAAS,GAAG,QAAQ,CAAC;IAC5B,CAAC;IAED,KAAK,CAAC,eAAe,CAAC,OAAwB;QAC5C,OAAO,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;IACjC,CAAC;IAED,KAAK,CAAC,aAAa,CAAC,WAAmB;QACrC,OAAO,oBAAoB,CAAC;YAC1B,MAAM,EAAE,UAAU;YAClB,MAAM,EAAE,2CAA2C;SACpD,CAAC,CAAC;IACL,CAAC;CACF"}
|
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Async task manager for background module execution.
|
|
3
|
+
*/
|
|
4
|
+
import type { Context } from './context.js';
|
|
5
|
+
import type { Executor } from './executor.js';
|
|
6
|
+
export declare enum TaskStatus {
|
|
7
|
+
PENDING = "pending",
|
|
8
|
+
RUNNING = "running",
|
|
9
|
+
COMPLETED = "completed",
|
|
10
|
+
FAILED = "failed",
|
|
11
|
+
CANCELLED = "cancelled"
|
|
12
|
+
}
|
|
13
|
+
export interface TaskInfo {
|
|
14
|
+
readonly taskId: string;
|
|
15
|
+
readonly moduleId: string;
|
|
16
|
+
readonly status: TaskStatus;
|
|
17
|
+
readonly submittedAt: number;
|
|
18
|
+
readonly startedAt: number | null;
|
|
19
|
+
readonly completedAt: number | null;
|
|
20
|
+
readonly result: Record<string, unknown> | null;
|
|
21
|
+
readonly error: string | null;
|
|
22
|
+
}
|
|
23
|
+
/**
|
|
24
|
+
* Manages background execution of modules via Promises.
|
|
25
|
+
*
|
|
26
|
+
* Uses a simple counter-based concurrency limiter instead of a semaphore.
|
|
27
|
+
*/
|
|
28
|
+
export declare class AsyncTaskManager {
|
|
29
|
+
private readonly _executor;
|
|
30
|
+
private readonly _maxConcurrent;
|
|
31
|
+
private readonly _maxTasks;
|
|
32
|
+
private readonly _tasks;
|
|
33
|
+
private _runningCount;
|
|
34
|
+
private readonly _waitQueue;
|
|
35
|
+
constructor(executor: Executor, maxConcurrent?: number, maxTasks?: number);
|
|
36
|
+
/**
|
|
37
|
+
* Submit a module for background execution.
|
|
38
|
+
*
|
|
39
|
+
* Returns the generated task_id immediately.
|
|
40
|
+
*/
|
|
41
|
+
submit(moduleId: string, inputs: Record<string, unknown>, context?: Context | null): string;
|
|
42
|
+
/**
|
|
43
|
+
* Return the TaskInfo for a task, or null if not found.
|
|
44
|
+
*/
|
|
45
|
+
getStatus(taskId: string): TaskInfo | null;
|
|
46
|
+
/**
|
|
47
|
+
* Return the result of a completed task.
|
|
48
|
+
*
|
|
49
|
+
* Throws if the task is not found or not in COMPLETED status.
|
|
50
|
+
*/
|
|
51
|
+
getResult(taskId: string): Record<string, unknown>;
|
|
52
|
+
/**
|
|
53
|
+
* Cancel a running or pending task.
|
|
54
|
+
*
|
|
55
|
+
* Sets the cancelled flag and updates status immediately.
|
|
56
|
+
* The underlying execution may still be in-flight but its result
|
|
57
|
+
* will be discarded when it completes.
|
|
58
|
+
*
|
|
59
|
+
* Returns true if the task was successfully marked as cancelled.
|
|
60
|
+
*/
|
|
61
|
+
cancel(taskId: string): boolean;
|
|
62
|
+
/**
|
|
63
|
+
* Return all tasks, optionally filtered by status.
|
|
64
|
+
*/
|
|
65
|
+
listTasks(status?: TaskStatus): TaskInfo[];
|
|
66
|
+
/**
|
|
67
|
+
* Remove terminal-state tasks older than maxAgeSeconds seconds.
|
|
68
|
+
*
|
|
69
|
+
* Terminal states: COMPLETED, FAILED, CANCELLED.
|
|
70
|
+
* Returns the number of tasks removed.
|
|
71
|
+
*/
|
|
72
|
+
cleanup(maxAgeSeconds?: number): number;
|
|
73
|
+
/**
|
|
74
|
+
* Cancel all pending and running tasks.
|
|
75
|
+
*/
|
|
76
|
+
shutdown(): void;
|
|
77
|
+
/**
|
|
78
|
+
* Acquire a concurrency slot. Resolves when a slot is available.
|
|
79
|
+
*/
|
|
80
|
+
private _acquireSlot;
|
|
81
|
+
/**
|
|
82
|
+
* Release a concurrency slot and notify the next waiter.
|
|
83
|
+
*/
|
|
84
|
+
private _releaseSlot;
|
|
85
|
+
/**
|
|
86
|
+
* Enqueue the task for execution under the concurrency limit.
|
|
87
|
+
*/
|
|
88
|
+
private _enqueue;
|
|
89
|
+
}
|
|
90
|
+
//# sourceMappingURL=async-task.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"async-task.d.ts","sourceRoot":"","sources":["../src/async-task.ts"],"names":[],"mappings":"AAAA;;GAEG;AAGH,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,cAAc,CAAC;AAC5C,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AAE9C,oBAAY,UAAU;IACpB,OAAO,YAAY;IACnB,OAAO,YAAY;IACnB,SAAS,cAAc;IACvB,MAAM,WAAW;IACjB,SAAS,cAAc;CACxB;AAED,MAAM,WAAW,QAAQ;IACvB,QAAQ,CAAC,MAAM,EAAE,MAAM,CAAC;IACxB,QAAQ,CAAC,QAAQ,EAAE,MAAM,CAAC;IAC1B,QAAQ,CAAC,MAAM,EAAE,UAAU,CAAC;IAC5B,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAC;IAC7B,QAAQ,CAAC,SAAS,EAAE,MAAM,GAAG,IAAI,CAAC;IAClC,QAAQ,CAAC,WAAW,EAAE,MAAM,GAAG,IAAI,CAAC;IACpC,QAAQ,CAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,IAAI,CAAC;IAChD,QAAQ,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI,CAAC;CAC/B;AAWD;;;;GAIG;AACH,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAW;IACrC,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAS;IACxC,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAS;IACnC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAwC;IAC/D,OAAO,CAAC,aAAa,CAAa;IAClC,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAyB;gBAExC,QAAQ,EAAE,QAAQ,EAAE,aAAa,GAAE,MAAW,EAAE,QAAQ,GAAE,MAAa;IAMnF;;;;OAIG;IACH,MAAM,CACJ,QAAQ,EAAE,MAAM,EAChB,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC/B,OAAO,CAAC,EAAE,OAAO,GAAG,IAAI,GACvB,MAAM;IAiCT;;OAEG;IACH,SAAS,CAAC,MAAM,EAAE,MAAM,GAAG,QAAQ,GAAG,IAAI;IAK1C;;;;OAIG;IACH,SAAS,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC;IAWlD;;;;;;;;OAQG;IACH,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO;IAe/B;;OAEG;IACH,SAAS,CAAC,MAAM,CAAC,EAAE,UAAU,GAAG,QAAQ,EAAE;IAM1C;;;;;OAKG;IACH,OAAO,CAAC,aAAa,GAAE,MAAa,GAAG,MAAM;IAiB7C;;OAEG;IACH,QAAQ,IAAI,IAAI;IAQhB;;OAEG;IACH,OAAO,CAAC,YAAY;IAapB;;OAEG;IACH,OAAO,CAAC,YAAY;IAQpB;;OAEG;IACH,OAAO,CAAC,QAAQ;CAgDjB"}
|