@pipeline-builder/pipeline-core 3.1.5 → 3.2.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.
@@ -0,0 +1,35 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ Object.defineProperty(exports, "__esModule", { value: true });
5
+ exports.templateResolutionDurationMs = exports.templateResolutionsTotal = void 0;
6
+ exports.recordResolution = recordResolution;
7
+ function resolvePromClient() {
8
+ try {
9
+ // eslint-disable-next-line @typescript-eslint/no-require-imports
10
+ return require('prom-client');
11
+ }
12
+ catch {
13
+ return null;
14
+ }
15
+ }
16
+ const promClient = resolvePromClient();
17
+ function makeCounter(name, help, labelNames) {
18
+ if (!promClient)
19
+ return { inc: () => { } };
20
+ const existing = promClient.register.getSingleMetric(name);
21
+ return existing ?? new promClient.Counter({ name, help, labelNames });
22
+ }
23
+ function makeHistogram(name, help, labelNames, buckets) {
24
+ if (!promClient)
25
+ return { observe: () => { } };
26
+ const existing = promClient.register.getSingleMetric(name);
27
+ return existing ?? new promClient.Histogram({ name, help, labelNames, buckets });
28
+ }
29
+ exports.templateResolutionsTotal = makeCounter('pipeline_builder_template_resolutions_total', 'Total number of template resolution operations', ['outcome', 'doc']);
30
+ exports.templateResolutionDurationMs = makeHistogram('pipeline_builder_template_resolution_duration_ms', 'Template resolution duration in milliseconds', ['doc'], [1, 5, 10, 50, 100, 500, 1000]);
31
+ function recordResolution(doc, durationMs, success) {
32
+ exports.templateResolutionsTotal.inc({ outcome: success ? 'success' : 'error', doc });
33
+ exports.templateResolutionDurationMs.observe({ doc }, durationMs);
34
+ }
35
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWV0cmljcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy90ZW1wbGF0ZS9tZXRyaWNzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQSwrQ0FBK0M7QUFDL0Msc0NBQXNDOzs7QUFpRHRDLDRDQUdDO0FBdkNELFNBQVMsaUJBQWlCO0lBQ3hCLElBQUksQ0FBQztRQUNILGlFQUFpRTtRQUNqRSxPQUFPLE9BQU8sQ0FBQyxhQUFhLENBQUMsQ0FBQztJQUNoQyxDQUFDO0lBQUMsTUFBTSxDQUFDO1FBQ1AsT0FBTyxJQUFJLENBQUM7SUFDZCxDQUFDO0FBQ0gsQ0FBQztBQUVELE1BQU0sVUFBVSxHQUFHLGlCQUFpQixFQUFFLENBQUM7QUFFdkMsU0FBUyxXQUFXLENBQUMsSUFBWSxFQUFFLElBQVksRUFBRSxVQUFvQjtJQUNuRSxJQUFJLENBQUMsVUFBVTtRQUFFLE9BQU8sRUFBRSxHQUFHLEVBQUUsR0FBRyxFQUFFLEdBQWMsQ0FBQyxFQUFFLENBQUM7SUFDdEQsTUFBTSxRQUFRLEdBQUcsVUFBVSxDQUFDLFFBQVEsQ0FBQyxlQUFlLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDM0QsT0FBTyxRQUFRLElBQUksSUFBSSxVQUFVLENBQUMsT0FBTyxDQUFDLEVBQUUsSUFBSSxFQUFFLElBQUksRUFBRSxVQUFVLEVBQUUsQ0FBQyxDQUFDO0FBQ3hFLENBQUM7QUFFRCxTQUFTLGFBQWEsQ0FBQyxJQUFZLEVBQUUsSUFBWSxFQUFFLFVBQW9CLEVBQUUsT0FBaUI7SUFDeEYsSUFBSSxDQUFDLFVBQVU7UUFBRSxPQUFPLEVBQUUsT0FBTyxFQUFFLEdBQUcsRUFBRSxHQUFjLENBQUMsRUFBRSxDQUFDO0lBQzFELE1BQU0sUUFBUSxHQUFHLFVBQVUsQ0FBQyxRQUFRLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQzNELE9BQU8sUUFBUSxJQUFJLElBQUksVUFBVSxDQUFDLFNBQVMsQ0FBQyxFQUFFLElBQUksRUFBRSxJQUFJLEVBQUUsVUFBVSxFQUFFLE9BQU8sRUFBRSxDQUFDLENBQUM7QUFDbkYsQ0FBQztBQUVZLFFBQUEsd0JBQXdCLEdBQUcsV0FBVyxDQUNqRCw2Q0FBNkMsRUFDN0MsZ0RBQWdELEVBQ2hELENBQUMsU0FBUyxFQUFFLEtBQUssQ0FBQyxDQUNuQixDQUFDO0FBRVcsUUFBQSw0QkFBNEIsR0FBRyxhQUFhLENBQ3ZELGtEQUFrRCxFQUNsRCw4Q0FBOEMsRUFDOUMsQ0FBQyxLQUFLLENBQUMsRUFDUCxDQUFDLENBQUMsRUFBRSxDQUFDLEVBQUUsRUFBRSxFQUFFLEVBQUUsRUFBRSxHQUFHLEVBQUUsR0FBRyxFQUFFLElBQUksQ0FBQyxDQUMvQixDQUFDO0FBRUYsU0FBZ0IsZ0JBQWdCLENBQUMsR0FBMEIsRUFBRSxVQUFrQixFQUFFLE9BQWdCO0lBQy9GLGdDQUF3QixDQUFDLEdBQUcsQ0FBQyxFQUFFLE9BQU8sRUFBRSxPQUFPLENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsT0FBTyxFQUFFLEdBQUcsRUFBRSxDQUFDLENBQUM7SUFDOUUsb0NBQTRCLENBQUMsT0FBTyxDQUFDLEVBQUUsR0FBRyxFQUFFLEVBQUUsVUFBVSxDQUFDLENBQUM7QUFDNUQsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIENvcHlyaWdodCAyMDI2IFBpcGVsaW5lIEJ1aWxkZXIgQ29udHJpYnV0b3JzXG4vLyBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogQXBhY2hlLTIuMFxuXG4vKipcbiAqIFByb21ldGhldXMgbWV0cmljcyBmb3IgdGVtcGxhdGUgcmVzb2x1dGlvbi4gT3B0aW9uYWwg4oCUIHRoZSBtb2R1bGUgaXNcbiAqIG9ubHkgd2lyZWQgdXAgd2hlbiBgcHJvbS1jbGllbnRgIGlzIGFscmVhZHkgcHJlc2VudCBpbiB0aGUgaG9zdFxuICogcHJvY2VzcyAoYXMgaXMgdGhlIGNhc2UgZm9yIGFwaS1zZXJ2ZXIgYmFzZWQgc2VydmljZXMpLiBGYWxscyBiYWNrIHRvXG4gKiBuby1vcCBzdHVicyB3aGVuIHVuYXZhaWxhYmxlIHNvIHBpcGVsaW5lLWNvcmUgc3RheXMgdXNhYmxlIG91dHNpZGVcbiAqIHNlcnZlciBjb250ZXh0cy5cbiAqL1xuXG5pbnRlcmZhY2UgQ291bnRlciB7IGluYyhsYWJlbHM/OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+LCB2YWx1ZT86IG51bWJlcik6IHZvaWQgfVxuaW50ZXJmYWNlIEhpc3RvZ3JhbSB7IG9ic2VydmUobGFiZWxzOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+LCB2YWx1ZTogbnVtYmVyKTogdm9pZCB9XG5cbmZ1bmN0aW9uIHJlc29sdmVQcm9tQ2xpZW50KCk6IGFueSB8IG51bGwge1xuICB0cnkge1xuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tcmVxdWlyZS1pbXBvcnRzXG4gICAgcmV0dXJuIHJlcXVpcmUoJ3Byb20tY2xpZW50Jyk7XG4gIH0gY2F0Y2gge1xuICAgIHJldHVybiBudWxsO1xuICB9XG59XG5cbmNvbnN0IHByb21DbGllbnQgPSByZXNvbHZlUHJvbUNsaWVudCgpO1xuXG5mdW5jdGlvbiBtYWtlQ291bnRlcihuYW1lOiBzdHJpbmcsIGhlbHA6IHN0cmluZywgbGFiZWxOYW1lczogc3RyaW5nW10pOiBDb3VudGVyIHtcbiAgaWYgKCFwcm9tQ2xpZW50KSByZXR1cm4geyBpbmM6ICgpID0+IHsgLyogbm9vcCAqLyB9IH07XG4gIGNvbnN0IGV4aXN0aW5nID0gcHJvbUNsaWVudC5yZWdpc3Rlci5nZXRTaW5nbGVNZXRyaWMobmFtZSk7XG4gIHJldHVybiBleGlzdGluZyA/PyBuZXcgcHJvbUNsaWVudC5Db3VudGVyKHsgbmFtZSwgaGVscCwgbGFiZWxOYW1lcyB9KTtcbn1cblxuZnVuY3Rpb24gbWFrZUhpc3RvZ3JhbShuYW1lOiBzdHJpbmcsIGhlbHA6IHN0cmluZywgbGFiZWxOYW1lczogc3RyaW5nW10sIGJ1Y2tldHM6IG51bWJlcltdKTogSGlzdG9ncmFtIHtcbiAgaWYgKCFwcm9tQ2xpZW50KSByZXR1cm4geyBvYnNlcnZlOiAoKSA9PiB7IC8qIG5vb3AgKi8gfSB9O1xuICBjb25zdCBleGlzdGluZyA9IHByb21DbGllbnQucmVnaXN0ZXIuZ2V0U2luZ2xlTWV0cmljKG5hbWUpO1xuICByZXR1cm4gZXhpc3RpbmcgPz8gbmV3IHByb21DbGllbnQuSGlzdG9ncmFtKHsgbmFtZSwgaGVscCwgbGFiZWxOYW1lcywgYnVja2V0cyB9KTtcbn1cblxuZXhwb3J0IGNvbnN0IHRlbXBsYXRlUmVzb2x1dGlvbnNUb3RhbCA9IG1ha2VDb3VudGVyKFxuICAncGlwZWxpbmVfYnVpbGRlcl90ZW1wbGF0ZV9yZXNvbHV0aW9uc190b3RhbCcsXG4gICdUb3RhbCBudW1iZXIgb2YgdGVtcGxhdGUgcmVzb2x1dGlvbiBvcGVyYXRpb25zJyxcbiAgWydvdXRjb21lJywgJ2RvYyddLFxuKTtcblxuZXhwb3J0IGNvbnN0IHRlbXBsYXRlUmVzb2x1dGlvbkR1cmF0aW9uTXMgPSBtYWtlSGlzdG9ncmFtKFxuICAncGlwZWxpbmVfYnVpbGRlcl90ZW1wbGF0ZV9yZXNvbHV0aW9uX2R1cmF0aW9uX21zJyxcbiAgJ1RlbXBsYXRlIHJlc29sdXRpb24gZHVyYXRpb24gaW4gbWlsbGlzZWNvbmRzJyxcbiAgWydkb2MnXSxcbiAgWzEsIDUsIDEwLCA1MCwgMTAwLCA1MDAsIDEwMDBdLFxuKTtcblxuZXhwb3J0IGZ1bmN0aW9uIHJlY29yZFJlc29sdXRpb24oZG9jOiAncGlwZWxpbmUnIHwgJ3BsdWdpbicsIGR1cmF0aW9uTXM6IG51bWJlciwgc3VjY2VzczogYm9vbGVhbik6IHZvaWQge1xuICB0ZW1wbGF0ZVJlc29sdXRpb25zVG90YWwuaW5jKHsgb3V0Y29tZTogc3VjY2VzcyA/ICdzdWNjZXNzJyA6ICdlcnJvcicsIGRvYyB9KTtcbiAgdGVtcGxhdGVSZXNvbHV0aW9uRHVyYXRpb25Ncy5vYnNlcnZlKHsgZG9jIH0sIGR1cmF0aW9uTXMpO1xufVxuIl19
@@ -0,0 +1,8 @@
1
+ import type { Plugin } from '@pipeline-builder/pipeline-data';
2
+ export declare function isPluginTemplatableField(field: string): boolean;
3
+ /**
4
+ * Return a shallow clone of `plugin` with all `{{ ... }}` templates
5
+ * resolved against the given pipeline scope. Caller must pre-populate
6
+ * `pipelineScope` with `{ pipeline, plugin, env }` keys.
7
+ */
8
+ export declare function resolvePluginTemplates(plugin: Plugin, pipelineScope: Record<string, unknown>): Plugin;
@@ -0,0 +1,52 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ Object.defineProperty(exports, "__esModule", { value: true });
5
+ exports.isPluginTemplatableField = isPluginTemplatableField;
6
+ exports.resolvePluginTemplates = resolvePluginTemplates;
7
+ const index_1 = require("./index");
8
+ /**
9
+ * Fields inside a Plugin record that accept `{{ ... }}` templates.
10
+ *
11
+ * Pure-string leaves only. `name`, `version`, `pluginType`, `computeType`,
12
+ * `timeout`, `secrets`, `failureBehavior`, `requiredMetadata`, `requiredVars`
13
+ * stay literal. `metadata.*` is excluded because CDK-metadata values are
14
+ * structural, not user-interpolated.
15
+ */
16
+ const TEMPLATABLE_FIELDS = [
17
+ 'description',
18
+ 'commands', // string[]
19
+ 'installCommands', // string[]
20
+ 'env', // Record<string, string>
21
+ 'buildArgs', // Record<string, string>
22
+ ];
23
+ function isPluginTemplatableField(field) {
24
+ // `commands` / `installCommands` are arrays of strings → entries like 'commands[0]'
25
+ // `env` / `buildArgs` are objects → entries like 'env.STAGE'
26
+ return TEMPLATABLE_FIELDS.some(f => field === f || field.startsWith(`${f}[`) || field.startsWith(`${f}.`));
27
+ }
28
+ /**
29
+ * Return a shallow clone of `plugin` with all `{{ ... }}` templates
30
+ * resolved against the given pipeline scope. Caller must pre-populate
31
+ * `pipelineScope` with `{ pipeline, plugin, env }` keys.
32
+ */
33
+ function resolvePluginTemplates(plugin, pipelineScope) {
34
+ // Deep clone so mutations don't leak back to the caller's Plugin.
35
+ // Structural fields we care about are plain JSON; structuredClone is safe.
36
+ const clone = structuredClone(plugin);
37
+ const scope = {
38
+ ...pipelineScope,
39
+ plugin: { name: plugin.name, version: plugin.version, imageTag: plugin.imageTag },
40
+ env: plugin.env ?? {},
41
+ };
42
+ const { errors } = (0, index_1.resolveTemplates)(clone, scope, isPluginTemplatableField, 'plugin');
43
+ if (errors.length > 0) {
44
+ // First error wins — resolver errors should never be batched at synth time
45
+ // because a broken template is a programmer error, not a validation step.
46
+ const e = errors[0];
47
+ const msg = `Template resolution failed in plugin "${plugin.name}" at field '${e.field}': ${e.message}`;
48
+ throw new Error(msg);
49
+ }
50
+ return clone;
51
+ }
52
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGx1Z2luLXJlc29sdmVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3RlbXBsYXRlL3BsdWdpbi1yZXNvbHZlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsK0NBQStDO0FBQy9DLHNDQUFzQzs7QUFxQnRDLDREQUlDO0FBT0Qsd0RBdUJDO0FBcERELG1DQUEyQztBQUUzQzs7Ozs7OztHQU9HO0FBQ0gsTUFBTSxrQkFBa0IsR0FBRztJQUN6QixhQUFhO0lBQ2IsVUFBVSxFQUFFLFdBQVc7SUFDdkIsaUJBQWlCLEVBQUUsV0FBVztJQUM5QixLQUFLLEVBQUUseUJBQXlCO0lBQ2hDLFdBQVcsRUFBRSx5QkFBeUI7Q0FDOUIsQ0FBQztBQUVYLFNBQWdCLHdCQUF3QixDQUFDLEtBQWE7SUFDcEQsb0ZBQW9GO0lBQ3BGLDZEQUE2RDtJQUM3RCxPQUFPLGtCQUFrQixDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLEtBQUssS0FBSyxDQUFDLElBQUksS0FBSyxDQUFDLFVBQVUsQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLElBQUksS0FBSyxDQUFDLFVBQVUsQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQztBQUM3RyxDQUFDO0FBRUQ7Ozs7R0FJRztBQUNILFNBQWdCLHNCQUFzQixDQUNwQyxNQUFjLEVBQ2QsYUFBc0M7SUFFdEMsa0VBQWtFO0lBQ2xFLDJFQUEyRTtJQUMzRSxNQUFNLEtBQUssR0FBRyxlQUFlLENBQUMsTUFBTSxDQUFxQyxDQUFDO0lBRTFFLE1BQU0sS0FBSyxHQUFHO1FBQ1osR0FBRyxhQUFhO1FBQ2hCLE1BQU0sRUFBRSxFQUFFLElBQUksRUFBRSxNQUFNLENBQUMsSUFBSSxFQUFFLE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLENBQUMsUUFBUSxFQUFFO1FBQ2pGLEdBQUcsRUFBRSxNQUFNLENBQUMsR0FBRyxJQUFJLEVBQUU7S0FDdEIsQ0FBQztJQUVGLE1BQU0sRUFBRSxNQUFNLEVBQUUsR0FBRyxJQUFBLHdCQUFnQixFQUFDLEtBQUssRUFBRSxLQUFLLEVBQUUsd0JBQXdCLEVBQUUsUUFBUSxDQUFDLENBQUM7SUFDdEYsSUFBSSxNQUFNLENBQUMsTUFBTSxHQUFHLENBQUMsRUFBRSxDQUFDO1FBQ3RCLDJFQUEyRTtRQUMzRSwwRUFBMEU7UUFDMUUsTUFBTSxDQUFDLEdBQUcsTUFBTSxDQUFDLENBQUMsQ0FBRSxDQUFDO1FBQ3JCLE1BQU0sR0FBRyxHQUFHLHlDQUF5QyxNQUFNLENBQUMsSUFBSSxlQUFlLENBQUMsQ0FBQyxLQUFLLE1BQU0sQ0FBQyxDQUFDLE9BQU8sRUFBRSxDQUFDO1FBQ3hHLE1BQU0sSUFBSSxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDdkIsQ0FBQztJQUNELE9BQU8sS0FBSyxDQUFDO0FBQ2YsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIENvcHlyaWdodCAyMDI2IFBpcGVsaW5lIEJ1aWxkZXIgQ29udHJpYnV0b3JzXG4vLyBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogQXBhY2hlLTIuMFxuXG5pbXBvcnQgdHlwZSB7IFBsdWdpbiB9IGZyb20gJ0BwaXBlbGluZS1idWlsZGVyL3BpcGVsaW5lLWRhdGEnO1xuaW1wb3J0IHsgcmVzb2x2ZVRlbXBsYXRlcyB9IGZyb20gJy4vaW5kZXgnO1xuXG4vKipcbiAqIEZpZWxkcyBpbnNpZGUgYSBQbHVnaW4gcmVjb3JkIHRoYXQgYWNjZXB0IGB7eyAuLi4gfX1gIHRlbXBsYXRlcy5cbiAqXG4gKiBQdXJlLXN0cmluZyBsZWF2ZXMgb25seS4gYG5hbWVgLCBgdmVyc2lvbmAsIGBwbHVnaW5UeXBlYCwgYGNvbXB1dGVUeXBlYCxcbiAqIGB0aW1lb3V0YCwgYHNlY3JldHNgLCBgZmFpbHVyZUJlaGF2aW9yYCwgYHJlcXVpcmVkTWV0YWRhdGFgLCBgcmVxdWlyZWRWYXJzYFxuICogc3RheSBsaXRlcmFsLiBgbWV0YWRhdGEuKmAgaXMgZXhjbHVkZWQgYmVjYXVzZSBDREstbWV0YWRhdGEgdmFsdWVzIGFyZVxuICogc3RydWN0dXJhbCwgbm90IHVzZXItaW50ZXJwb2xhdGVkLlxuICovXG5jb25zdCBURU1QTEFUQUJMRV9GSUVMRFMgPSBbXG4gICdkZXNjcmlwdGlvbicsXG4gICdjb21tYW5kcycsIC8vIHN0cmluZ1tdXG4gICdpbnN0YWxsQ29tbWFuZHMnLCAvLyBzdHJpbmdbXVxuICAnZW52JywgLy8gUmVjb3JkPHN0cmluZywgc3RyaW5nPlxuICAnYnVpbGRBcmdzJywgLy8gUmVjb3JkPHN0cmluZywgc3RyaW5nPlxuXSBhcyBjb25zdDtcblxuZXhwb3J0IGZ1bmN0aW9uIGlzUGx1Z2luVGVtcGxhdGFibGVGaWVsZChmaWVsZDogc3RyaW5nKTogYm9vbGVhbiB7XG4gIC8vIGBjb21tYW5kc2AgLyBgaW5zdGFsbENvbW1hbmRzYCBhcmUgYXJyYXlzIG9mIHN0cmluZ3Mg4oaSIGVudHJpZXMgbGlrZSAnY29tbWFuZHNbMF0nXG4gIC8vIGBlbnZgIC8gYGJ1aWxkQXJnc2AgYXJlIG9iamVjdHMg4oaSIGVudHJpZXMgbGlrZSAnZW52LlNUQUdFJ1xuICByZXR1cm4gVEVNUExBVEFCTEVfRklFTERTLnNvbWUoZiA9PiBmaWVsZCA9PT0gZiB8fCBmaWVsZC5zdGFydHNXaXRoKGAke2Z9W2ApIHx8IGZpZWxkLnN0YXJ0c1dpdGgoYCR7Zn0uYCkpO1xufVxuXG4vKipcbiAqIFJldHVybiBhIHNoYWxsb3cgY2xvbmUgb2YgYHBsdWdpbmAgd2l0aCBhbGwgYHt7IC4uLiB9fWAgdGVtcGxhdGVzXG4gKiByZXNvbHZlZCBhZ2FpbnN0IHRoZSBnaXZlbiBwaXBlbGluZSBzY29wZS4gQ2FsbGVyIG11c3QgcHJlLXBvcHVsYXRlXG4gKiBgcGlwZWxpbmVTY29wZWAgd2l0aCBgeyBwaXBlbGluZSwgcGx1Z2luLCBlbnYgfWAga2V5cy5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHJlc29sdmVQbHVnaW5UZW1wbGF0ZXMoXG4gIHBsdWdpbjogUGx1Z2luLFxuICBwaXBlbGluZVNjb3BlOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPixcbik6IFBsdWdpbiB7XG4gIC8vIERlZXAgY2xvbmUgc28gbXV0YXRpb25zIGRvbid0IGxlYWsgYmFjayB0byB0aGUgY2FsbGVyJ3MgUGx1Z2luLlxuICAvLyBTdHJ1Y3R1cmFsIGZpZWxkcyB3ZSBjYXJlIGFib3V0IGFyZSBwbGFpbiBKU09OOyBzdHJ1Y3R1cmVkQ2xvbmUgaXMgc2FmZS5cbiAgY29uc3QgY2xvbmUgPSBzdHJ1Y3R1cmVkQ2xvbmUocGx1Z2luKSBhcyBQbHVnaW4gJiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcblxuICBjb25zdCBzY29wZSA9IHtcbiAgICAuLi5waXBlbGluZVNjb3BlLFxuICAgIHBsdWdpbjogeyBuYW1lOiBwbHVnaW4ubmFtZSwgdmVyc2lvbjogcGx1Z2luLnZlcnNpb24sIGltYWdlVGFnOiBwbHVnaW4uaW1hZ2VUYWcgfSxcbiAgICBlbnY6IHBsdWdpbi5lbnYgPz8ge30sXG4gIH07XG5cbiAgY29uc3QgeyBlcnJvcnMgfSA9IHJlc29sdmVUZW1wbGF0ZXMoY2xvbmUsIHNjb3BlLCBpc1BsdWdpblRlbXBsYXRhYmxlRmllbGQsICdwbHVnaW4nKTtcbiAgaWYgKGVycm9ycy5sZW5ndGggPiAwKSB7XG4gICAgLy8gRmlyc3QgZXJyb3Igd2lucyDigJQgcmVzb2x2ZXIgZXJyb3JzIHNob3VsZCBuZXZlciBiZSBiYXRjaGVkIGF0IHN5bnRoIHRpbWVcbiAgICAvLyBiZWNhdXNlIGEgYnJva2VuIHRlbXBsYXRlIGlzIGEgcHJvZ3JhbW1lciBlcnJvciwgbm90IGEgdmFsaWRhdGlvbiBzdGVwLlxuICAgIGNvbnN0IGUgPSBlcnJvcnNbMF0hO1xuICAgIGNvbnN0IG1zZyA9IGBUZW1wbGF0ZSByZXNvbHV0aW9uIGZhaWxlZCBpbiBwbHVnaW4gXCIke3BsdWdpbi5uYW1lfVwiIGF0IGZpZWxkICcke2UuZmllbGR9JzogJHtlLm1lc3NhZ2V9YDtcbiAgICB0aHJvdyBuZXcgRXJyb3IobXNnKTtcbiAgfVxuICByZXR1cm4gY2xvbmU7XG59XG4iXX0=
@@ -0,0 +1,49 @@
1
+ /**
2
+ * Tokenizer for the pipeline-builder template grammar.
3
+ *
4
+ * Grammar:
5
+ * Template := (Literal | Expr)*
6
+ * Expr := "{{" ws Path (ws Filter)? ws "}}"
7
+ * Path := Identifier ("." Identifier)*
8
+ * Identifier := [a-zA-Z_][a-zA-Z0-9_]{0,63}
9
+ * Filter := "|" ws "default" ws ":" ws Quoted
10
+ * Quoted := "'" ... "'" | "\"" ... "\""
11
+ *
12
+ * `{{{{` is the escape sequence for a literal `{{`.
13
+ */
14
+ export declare const MAX_FIELD_SIZE_BYTES: number;
15
+ export declare const MAX_PATH_DEPTH = 5;
16
+ export declare const MAX_IDENTIFIER_LENGTH = 64;
17
+ export interface SourcePosition {
18
+ line: number;
19
+ col: number;
20
+ }
21
+ export interface LiteralToken {
22
+ kind: 'literal';
23
+ value: string;
24
+ pos: SourcePosition;
25
+ }
26
+ export type CoerceKind = 'number' | 'bool' | 'json';
27
+ export interface ExprToken {
28
+ kind: 'expr';
29
+ path: string[];
30
+ defaultValue?: string;
31
+ coerce?: CoerceKind;
32
+ source: string;
33
+ pos: SourcePosition;
34
+ }
35
+ export type Token = LiteralToken | ExprToken;
36
+ export declare class TokenizerError extends Error {
37
+ readonly pos: SourcePosition;
38
+ constructor(message: string, pos: SourcePosition);
39
+ }
40
+ /**
41
+ * Tokenize a template source string. Never throws on valid input;
42
+ * throws `TokenizerError` with source position on malformed templates.
43
+ */
44
+ export declare function tokenize(source: string): Token[];
45
+ /**
46
+ * Convenience: returns true if the source contains any templates.
47
+ * Used as a fast-path to skip tokenization for strings that are obviously literal.
48
+ */
49
+ export declare function hasTemplate(source: string): boolean;
@@ -0,0 +1,237 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ Object.defineProperty(exports, "__esModule", { value: true });
5
+ exports.TokenizerError = exports.MAX_IDENTIFIER_LENGTH = exports.MAX_PATH_DEPTH = exports.MAX_FIELD_SIZE_BYTES = void 0;
6
+ exports.tokenize = tokenize;
7
+ exports.hasTemplate = hasTemplate;
8
+ /**
9
+ * Tokenizer for the pipeline-builder template grammar.
10
+ *
11
+ * Grammar:
12
+ * Template := (Literal | Expr)*
13
+ * Expr := "{{" ws Path (ws Filter)? ws "}}"
14
+ * Path := Identifier ("." Identifier)*
15
+ * Identifier := [a-zA-Z_][a-zA-Z0-9_]{0,63}
16
+ * Filter := "|" ws "default" ws ":" ws Quoted
17
+ * Quoted := "'" ... "'" | "\"" ... "\""
18
+ *
19
+ * `{{{{` is the escape sequence for a literal `{{`.
20
+ */
21
+ exports.MAX_FIELD_SIZE_BYTES = 4 * 1024;
22
+ exports.MAX_PATH_DEPTH = 5;
23
+ exports.MAX_IDENTIFIER_LENGTH = 64;
24
+ class TokenizerError extends Error {
25
+ pos;
26
+ constructor(message, pos) {
27
+ super(`${message} at line ${pos.line}, col ${pos.col}`);
28
+ this.pos = pos;
29
+ this.name = 'TokenizerError';
30
+ }
31
+ }
32
+ exports.TokenizerError = TokenizerError;
33
+ /**
34
+ * Tokenize a template source string. Never throws on valid input;
35
+ * throws `TokenizerError` with source position on malformed templates.
36
+ */
37
+ function tokenize(source) {
38
+ if (source.length > exports.MAX_FIELD_SIZE_BYTES) {
39
+ throw new TokenizerError(`Field exceeds max size of ${exports.MAX_FIELD_SIZE_BYTES} bytes`, { line: 1, col: 1 });
40
+ }
41
+ const tokens = [];
42
+ let i = 0;
43
+ let line = 1;
44
+ let col = 1;
45
+ let literalStart = 0;
46
+ let literalStartPos = { line: 1, col: 1 };
47
+ const advance = (n) => {
48
+ for (let k = 0; k < n; k++) {
49
+ if (source[i + k] === '\n') {
50
+ line++;
51
+ col = 1;
52
+ }
53
+ else {
54
+ col++;
55
+ }
56
+ }
57
+ i += n;
58
+ };
59
+ const flushLiteral = (endIdx) => {
60
+ if (endIdx > literalStart) {
61
+ // Unescape `{{{{` → `{{`
62
+ const raw = source.slice(literalStart, endIdx);
63
+ const value = raw.replace(/\{\{\{\{/g, '{{');
64
+ tokens.push({ kind: 'literal', value, pos: literalStartPos });
65
+ }
66
+ };
67
+ const startLiteral = () => {
68
+ literalStart = i;
69
+ literalStartPos = { line, col };
70
+ };
71
+ startLiteral();
72
+ while (i < source.length) {
73
+ // Escape: `{{{{` → literal `{{`
74
+ if (source.startsWith('{{{{', i)) {
75
+ advance(4);
76
+ continue;
77
+ }
78
+ // Expression start
79
+ if (source.startsWith('{{', i)) {
80
+ flushLiteral(i);
81
+ const exprPos = { line, col };
82
+ advance(2);
83
+ const expr = readExpression(source, i, line, col);
84
+ tokens.push({
85
+ kind: 'expr',
86
+ path: expr.path,
87
+ defaultValue: expr.defaultValue,
88
+ coerce: expr.coerce,
89
+ source: source.slice(exprPos.col - 1 === 0 ? i - 2 : (() => { return i - 2; })(), expr.endIdx),
90
+ pos: exprPos,
91
+ });
92
+ advance(expr.endIdx - i);
93
+ startLiteral();
94
+ continue;
95
+ }
96
+ // Stray `}}` outside an expression
97
+ if (source.startsWith('}}', i)) {
98
+ throw new TokenizerError("Unexpected '}}' outside expression", { line, col });
99
+ }
100
+ advance(1);
101
+ }
102
+ flushLiteral(i);
103
+ return tokens;
104
+ }
105
+ function readExpression(source, startIdx, startLine, startCol) {
106
+ let i = startIdx;
107
+ let line = startLine;
108
+ let col = startCol;
109
+ const skipWs = () => {
110
+ while (i < source.length && (source[i] === ' ' || source[i] === '\t')) {
111
+ i++;
112
+ col++;
113
+ }
114
+ };
115
+ const readIdentifier = () => {
116
+ const startI = i;
117
+ if (i >= source.length || !/[a-zA-Z_]/.test(source[i])) {
118
+ throw new TokenizerError('Expected identifier inside template expression', { line, col });
119
+ }
120
+ while (i < source.length && /[a-zA-Z0-9_]/.test(source[i])) {
121
+ i++;
122
+ col++;
123
+ }
124
+ const ident = source.slice(startI, i);
125
+ if (ident.length > exports.MAX_IDENTIFIER_LENGTH) {
126
+ throw new TokenizerError(`Identifier exceeds max length of ${exports.MAX_IDENTIFIER_LENGTH}`, { line, col: col - ident.length });
127
+ }
128
+ return ident;
129
+ };
130
+ const readQuoted = () => {
131
+ const quote = source[i];
132
+ if (quote !== '"' && quote !== "'") {
133
+ throw new TokenizerError('Expected quoted string', { line, col });
134
+ }
135
+ i++;
136
+ col++;
137
+ const parts = [];
138
+ while (i < source.length && source[i] !== quote) {
139
+ if (source[i] === '\\') {
140
+ const next = source[i + 1];
141
+ if (next === '\\' || next === quote) {
142
+ parts.push(next);
143
+ i += 2;
144
+ col += 2;
145
+ continue;
146
+ }
147
+ throw new TokenizerError(`Invalid escape '\\${next ?? ''}' in quoted string`, { line, col });
148
+ }
149
+ if (source[i] === '\n') {
150
+ throw new TokenizerError('Unterminated quoted string', { line, col });
151
+ }
152
+ parts.push(source[i]);
153
+ i++;
154
+ col++;
155
+ }
156
+ if (i >= source.length) {
157
+ throw new TokenizerError('Unterminated quoted string', { line, col });
158
+ }
159
+ i++;
160
+ col++; // consume closing quote
161
+ return parts.join('');
162
+ };
163
+ skipWs();
164
+ const path = [readIdentifier()];
165
+ while (i < source.length && source[i] === '.') {
166
+ i++;
167
+ col++;
168
+ path.push(readIdentifier());
169
+ if (path.length > exports.MAX_PATH_DEPTH) {
170
+ throw new TokenizerError(`Path depth exceeds max of ${exports.MAX_PATH_DEPTH}`, { line, col });
171
+ }
172
+ }
173
+ skipWs();
174
+ let defaultValue;
175
+ let coerce;
176
+ // Filters chain: `| default: 'x' | number`, `| bool`, `| json`, etc.
177
+ // - `default` takes a quoted argument — applied first (before coercion)
178
+ // - `number`, `bool`, `json` are argument-less coercion filters — applied last
179
+ while (source[i] === '|') {
180
+ i++;
181
+ col++;
182
+ skipWs();
183
+ if (source.startsWith('default', i)) {
184
+ if (defaultValue !== undefined) {
185
+ throw new TokenizerError("'default' filter may only appear once", { line, col });
186
+ }
187
+ i += 'default'.length;
188
+ col += 'default'.length;
189
+ skipWs();
190
+ if (source[i] !== ':') {
191
+ throw new TokenizerError("Expected ':' after 'default'", { line, col });
192
+ }
193
+ i++;
194
+ col++;
195
+ skipWs();
196
+ defaultValue = readQuoted();
197
+ skipWs();
198
+ continue;
199
+ }
200
+ let matchedCoerce = null;
201
+ for (const kind of ['number', 'bool', 'json']) {
202
+ if (source.startsWith(kind, i)) {
203
+ // Ensure identifier boundary — `numberish` must not match `number`
204
+ const next = source[i + kind.length];
205
+ if (!next || !/[a-zA-Z0-9_]/.test(next)) {
206
+ matchedCoerce = kind;
207
+ i += kind.length;
208
+ col += kind.length;
209
+ break;
210
+ }
211
+ }
212
+ }
213
+ if (!matchedCoerce) {
214
+ throw new TokenizerError("Unknown filter — supported: 'default', 'number', 'bool', 'json'", { line, col });
215
+ }
216
+ if (coerce) {
217
+ throw new TokenizerError('Only one coercion filter is allowed per expression', { line, col });
218
+ }
219
+ coerce = matchedCoerce;
220
+ skipWs();
221
+ }
222
+ if (!source.startsWith('}}', i)) {
223
+ throw new TokenizerError("Expected '}}'", { line, col });
224
+ }
225
+ i += 2;
226
+ return { path, defaultValue, coerce, endIdx: i };
227
+ }
228
+ /**
229
+ * Convenience: returns true if the source contains any templates.
230
+ * Used as a fast-path to skip tokenization for strings that are obviously literal.
231
+ */
232
+ function hasTemplate(source) {
233
+ // Not completely accurate (a lone `{{{{` would return true) but that's fine —
234
+ // tokenize() will produce a literal-only token list and round-trip correctly.
235
+ return source.includes('{{');
236
+ }
237
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tokenizer.js","sourceRoot":"","sources":["../../src/template/tokenizer.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\n/**\n * Tokenizer for the pipeline-builder template grammar.\n *\n * Grammar:\n *   Template   := (Literal | Expr)*\n *   Expr       := \"{{\" ws Path (ws Filter)? ws \"}}\"\n *   Path       := Identifier (\".\" Identifier)*\n *   Identifier := [a-zA-Z_][a-zA-Z0-9_]{0,63}\n *   Filter     := \"|\" ws \"default\" ws \":\" ws Quoted\n *   Quoted     := \"'\" ... \"'\" | \"\\\"\" ... \"\\\"\"\n *\n * `{{{{` is the escape sequence for a literal `{{`.\n */\n\nexport const MAX_FIELD_SIZE_BYTES = 4 * 1024;\nexport const MAX_PATH_DEPTH = 5;\nexport const MAX_IDENTIFIER_LENGTH = 64;\n\nexport interface SourcePosition {\n  line: number;\n  col: number;\n}\n\nexport interface LiteralToken {\n  kind: 'literal';\n  value: string;\n  pos: SourcePosition;\n}\n\nexport type CoerceKind = 'number' | 'bool' | 'json';\n\nexport interface ExprToken {\n  kind: 'expr';\n  path: string[]; // e.g. ['pipeline', 'metadata', 'env']\n  defaultValue?: string; // value for `| default: '...'` filter\n  coerce?: CoerceKind; // `| number`, `| bool`, `| json`\n  source: string; // original \"{{ ... }}\" text (for error messages)\n  pos: SourcePosition;\n}\n\nexport type Token = LiteralToken | ExprToken;\n\nexport class TokenizerError extends Error {\n  constructor(\n    message: string,\n    public readonly pos: SourcePosition,\n  ) {\n    super(`${message} at line ${pos.line}, col ${pos.col}`);\n    this.name = 'TokenizerError';\n  }\n}\n\n/**\n * Tokenize a template source string. Never throws on valid input;\n * throws `TokenizerError` with source position on malformed templates.\n */\nexport function tokenize(source: string): Token[] {\n  if (source.length > MAX_FIELD_SIZE_BYTES) {\n    throw new TokenizerError(\n      `Field exceeds max size of ${MAX_FIELD_SIZE_BYTES} bytes`,\n      { line: 1, col: 1 },\n    );\n  }\n\n  const tokens: Token[] = [];\n  let i = 0;\n  let line = 1;\n  let col = 1;\n  let literalStart = 0;\n  let literalStartPos: SourcePosition = { line: 1, col: 1 };\n\n  const advance = (n: number): void => {\n    for (let k = 0; k < n; k++) {\n      if (source[i + k] === '\\n') {\n        line++;\n        col = 1;\n      } else {\n        col++;\n      }\n    }\n    i += n;\n  };\n\n  const flushLiteral = (endIdx: number): void => {\n    if (endIdx > literalStart) {\n      // Unescape `{{{{` → `{{`\n      const raw = source.slice(literalStart, endIdx);\n      const value = raw.replace(/\\{\\{\\{\\{/g, '{{');\n      tokens.push({ kind: 'literal', value, pos: literalStartPos });\n    }\n  };\n\n  const startLiteral = (): void => {\n    literalStart = i;\n    literalStartPos = { line, col };\n  };\n\n  startLiteral();\n\n  while (i < source.length) {\n    // Escape: `{{{{` → literal `{{`\n    if (source.startsWith('{{{{', i)) {\n      advance(4);\n      continue;\n    }\n    // Expression start\n    if (source.startsWith('{{', i)) {\n      flushLiteral(i);\n      const exprPos: SourcePosition = { line, col };\n      advance(2);\n      const expr = readExpression(source, i, line, col);\n      tokens.push({\n        kind: 'expr',\n        path: expr.path,\n        defaultValue: expr.defaultValue,\n        coerce: expr.coerce,\n        source: source.slice(exprPos.col - 1 === 0 ? i - 2 : (() => { return i - 2; })(), expr.endIdx),\n        pos: exprPos,\n      });\n      advance(expr.endIdx - i);\n      startLiteral();\n      continue;\n    }\n    // Stray `}}` outside an expression\n    if (source.startsWith('}}', i)) {\n      throw new TokenizerError(\"Unexpected '}}' outside expression\", { line, col });\n    }\n    advance(1);\n  }\n\n  flushLiteral(i);\n  return tokens;\n}\n\ninterface ParsedExpr {\n  path: string[];\n  defaultValue?: string;\n  coerce?: CoerceKind;\n  endIdx: number;\n}\n\nfunction readExpression(\n  source: string,\n  startIdx: number,\n  startLine: number,\n  startCol: number,\n): ParsedExpr {\n  let i = startIdx;\n  let line = startLine;\n  let col = startCol;\n\n  const skipWs = (): void => {\n    while (i < source.length && (source[i] === ' ' || source[i] === '\\t')) {\n      i++;\n      col++;\n    }\n  };\n\n  const readIdentifier = (): string => {\n    const startI = i;\n    if (i >= source.length || !/[a-zA-Z_]/.test(source[i]!)) {\n      throw new TokenizerError(\n        'Expected identifier inside template expression',\n        { line, col },\n      );\n    }\n    while (i < source.length && /[a-zA-Z0-9_]/.test(source[i]!)) {\n      i++;\n      col++;\n    }\n    const ident = source.slice(startI, i);\n    if (ident.length > MAX_IDENTIFIER_LENGTH) {\n      throw new TokenizerError(\n        `Identifier exceeds max length of ${MAX_IDENTIFIER_LENGTH}`,\n        { line, col: col - ident.length },\n      );\n    }\n    return ident;\n  };\n\n  const readQuoted = (): string => {\n    const quote = source[i];\n    if (quote !== '\"' && quote !== \"'\") {\n      throw new TokenizerError('Expected quoted string', { line, col });\n    }\n    i++; col++;\n    const parts: string[] = [];\n    while (i < source.length && source[i] !== quote) {\n      if (source[i] === '\\\\') {\n        const next = source[i + 1];\n        if (next === '\\\\' || next === quote) {\n          parts.push(next);\n          i += 2;\n          col += 2;\n          continue;\n        }\n        throw new TokenizerError(\n          `Invalid escape '\\\\${next ?? ''}' in quoted string`,\n          { line, col },\n        );\n      }\n      if (source[i] === '\\n') {\n        throw new TokenizerError('Unterminated quoted string', { line, col });\n      }\n      parts.push(source[i]!);\n      i++; col++;\n    }\n    if (i >= source.length) {\n      throw new TokenizerError('Unterminated quoted string', { line, col });\n    }\n    i++; col++; // consume closing quote\n    return parts.join('');\n  };\n\n  skipWs();\n  const path: string[] = [readIdentifier()];\n  while (i < source.length && source[i] === '.') {\n    i++; col++;\n    path.push(readIdentifier());\n    if (path.length > MAX_PATH_DEPTH) {\n      throw new TokenizerError(\n        `Path depth exceeds max of ${MAX_PATH_DEPTH}`,\n        { line, col },\n      );\n    }\n  }\n  skipWs();\n\n  let defaultValue: string | undefined;\n  let coerce: CoerceKind | undefined;\n\n  // Filters chain: `| default: 'x' | number`, `| bool`, `| json`, etc.\n  // - `default` takes a quoted argument — applied first (before coercion)\n  // - `number`, `bool`, `json` are argument-less coercion filters — applied last\n  while (source[i] === '|') {\n    i++; col++;\n    skipWs();\n\n    if (source.startsWith('default', i)) {\n      if (defaultValue !== undefined) {\n        throw new TokenizerError(\"'default' filter may only appear once\", { line, col });\n      }\n      i += 'default'.length;\n      col += 'default'.length;\n      skipWs();\n      if (source[i] !== ':') {\n        throw new TokenizerError(\"Expected ':' after 'default'\", { line, col });\n      }\n      i++; col++;\n      skipWs();\n      defaultValue = readQuoted();\n      skipWs();\n      continue;\n    }\n\n    let matchedCoerce: CoerceKind | null = null;\n    for (const kind of ['number', 'bool', 'json'] as const) {\n      if (source.startsWith(kind, i)) {\n        // Ensure identifier boundary — `numberish` must not match `number`\n        const next = source[i + kind.length];\n        if (!next || !/[a-zA-Z0-9_]/.test(next)) {\n          matchedCoerce = kind;\n          i += kind.length;\n          col += kind.length;\n          break;\n        }\n      }\n    }\n\n    if (!matchedCoerce) {\n      throw new TokenizerError(\n        \"Unknown filter — supported: 'default', 'number', 'bool', 'json'\",\n        { line, col },\n      );\n    }\n    if (coerce) {\n      throw new TokenizerError('Only one coercion filter is allowed per expression', { line, col });\n    }\n    coerce = matchedCoerce;\n    skipWs();\n  }\n\n  if (!source.startsWith('}}', i)) {\n    throw new TokenizerError(\"Expected '}}'\", { line, col });\n  }\n  i += 2;\n  return { path, defaultValue, coerce, endIdx: i };\n}\n\n/**\n * Convenience: returns true if the source contains any templates.\n * Used as a fast-path to skip tokenization for strings that are obviously literal.\n */\nexport function hasTemplate(source: string): boolean {\n  // Not completely accurate (a lone `{{{{` would return true) but that's fine —\n  // tokenize() will produce a literal-only token list and round-trip correctly.\n  return source.includes('{{');\n}\n"]}
@@ -0,0 +1,17 @@
1
+ export interface TopoNode {
2
+ /** Unique key identifying this node (e.g. source field path) */
3
+ key: string;
4
+ /** Dependency keys that must resolve before this node */
5
+ deps: string[];
6
+ }
7
+ export interface TopoResult {
8
+ ordered: string[];
9
+ cycles: string[][];
10
+ }
11
+ /**
12
+ * Topologically sort a set of nodes by their declared dependency keys.
13
+ * Returns `ordered` in resolution order, plus any detected cycles.
14
+ * Nodes whose deps reference keys not in the node set are treated as
15
+ * depending on "external" values and ordered first (no cycle risk).
16
+ */
17
+ export declare function topoSort(nodes: TopoNode[]): TopoResult;
@@ -0,0 +1,91 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ Object.defineProperty(exports, "__esModule", { value: true });
5
+ exports.topoSort = topoSort;
6
+ /**
7
+ * Topologically sort a set of nodes by their declared dependency keys.
8
+ * Returns `ordered` in resolution order, plus any detected cycles.
9
+ * Nodes whose deps reference keys not in the node set are treated as
10
+ * depending on "external" values and ordered first (no cycle risk).
11
+ */
12
+ function topoSort(nodes) {
13
+ const keys = new Set(nodes.map(n => n.key));
14
+ const graph = new Map();
15
+ const inDegree = new Map();
16
+ // Init
17
+ for (const n of nodes) {
18
+ graph.set(n.key, new Set());
19
+ inDegree.set(n.key, 0);
20
+ }
21
+ // Self-loops short-circuit — detected before topo-sort so single-node
22
+ // cycles are reported even when Kahn's would otherwise accept them.
23
+ const selfLoops = [];
24
+ for (const n of nodes) {
25
+ if (n.deps.includes(n.key))
26
+ selfLoops.push([n.key, n.key]);
27
+ }
28
+ // Edges only between internal nodes (external deps resolve before pass starts)
29
+ for (const n of nodes) {
30
+ for (const dep of n.deps) {
31
+ if (!keys.has(dep))
32
+ continue;
33
+ if (dep === n.key)
34
+ continue; // self-dep → already captured above
35
+ // Edge: dep -> n.key (dep must come first)
36
+ if (!graph.get(dep).has(n.key)) {
37
+ graph.get(dep).add(n.key);
38
+ inDegree.set(n.key, (inDegree.get(n.key) ?? 0) + 1);
39
+ }
40
+ }
41
+ }
42
+ // Kahn's algorithm
43
+ const ordered = [];
44
+ const queue = [];
45
+ for (const [k, d] of inDegree)
46
+ if (d === 0)
47
+ queue.push(k);
48
+ while (queue.length) {
49
+ const k = queue.shift();
50
+ ordered.push(k);
51
+ for (const next of graph.get(k) ?? []) {
52
+ const d = (inDegree.get(next) ?? 0) - 1;
53
+ inDegree.set(next, d);
54
+ if (d === 0)
55
+ queue.push(next);
56
+ }
57
+ }
58
+ // Detect cycles: any node not in `ordered` is part of a cycle
59
+ const cycles = [...selfLoops];
60
+ if (ordered.length < nodes.length) {
61
+ const remaining = new Set(nodes.map(n => n.key).filter(k => !ordered.includes(k)));
62
+ const visited = new Set();
63
+ for (const start of remaining) {
64
+ if (visited.has(start))
65
+ continue;
66
+ const path = [];
67
+ const cycle = findCycle(start, graph, visited, path, remaining);
68
+ if (cycle)
69
+ cycles.push(cycle);
70
+ }
71
+ }
72
+ return { ordered, cycles };
73
+ }
74
+ function findCycle(start, graph, visited, path, remaining) {
75
+ if (path.includes(start)) {
76
+ const idx = path.indexOf(start);
77
+ return [...path.slice(idx), start];
78
+ }
79
+ if (visited.has(start) || !remaining.has(start))
80
+ return null;
81
+ visited.add(start);
82
+ path.push(start);
83
+ for (const next of graph.get(start) ?? []) {
84
+ const cycle = findCycle(next, graph, visited, path, remaining);
85
+ if (cycle)
86
+ return cycle;
87
+ }
88
+ path.pop();
89
+ return null;
90
+ }
91
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"topo-sort.js","sourceRoot":"","sources":["../../src/template/topo-sort.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nexport interface TopoNode {\n  /** Unique key identifying this node (e.g. source field path) */\n  key: string;\n  /** Dependency keys that must resolve before this node */\n  deps: string[];\n}\n\nexport interface TopoResult {\n  ordered: string[];\n  cycles: string[][];\n}\n\n/**\n * Topologically sort a set of nodes by their declared dependency keys.\n * Returns `ordered` in resolution order, plus any detected cycles.\n * Nodes whose deps reference keys not in the node set are treated as\n * depending on \"external\" values and ordered first (no cycle risk).\n */\nexport function topoSort(nodes: TopoNode[]): TopoResult {\n  const keys = new Set(nodes.map(n => n.key));\n  const graph = new Map<string, Set<string>>();\n  const inDegree = new Map<string, number>();\n\n  // Init\n  for (const n of nodes) {\n    graph.set(n.key, new Set());\n    inDegree.set(n.key, 0);\n  }\n\n  // Self-loops short-circuit — detected before topo-sort so single-node\n  // cycles are reported even when Kahn's would otherwise accept them.\n  const selfLoops: string[][] = [];\n  for (const n of nodes) {\n    if (n.deps.includes(n.key)) selfLoops.push([n.key, n.key]);\n  }\n\n  // Edges only between internal nodes (external deps resolve before pass starts)\n  for (const n of nodes) {\n    for (const dep of n.deps) {\n      if (!keys.has(dep)) continue;\n      if (dep === n.key) continue; // self-dep → already captured above\n      // Edge: dep -> n.key (dep must come first)\n      if (!graph.get(dep)!.has(n.key)) {\n        graph.get(dep)!.add(n.key);\n        inDegree.set(n.key, (inDegree.get(n.key) ?? 0) + 1);\n      }\n    }\n  }\n\n  // Kahn's algorithm\n  const ordered: string[] = [];\n  const queue: string[] = [];\n  for (const [k, d] of inDegree) if (d === 0) queue.push(k);\n  while (queue.length) {\n    const k = queue.shift()!;\n    ordered.push(k);\n    for (const next of graph.get(k) ?? []) {\n      const d = (inDegree.get(next) ?? 0) - 1;\n      inDegree.set(next, d);\n      if (d === 0) queue.push(next);\n    }\n  }\n\n  // Detect cycles: any node not in `ordered` is part of a cycle\n  const cycles: string[][] = [...selfLoops];\n  if (ordered.length < nodes.length) {\n    const remaining = new Set(nodes.map(n => n.key).filter(k => !ordered.includes(k)));\n    const visited = new Set<string>();\n    for (const start of remaining) {\n      if (visited.has(start)) continue;\n      const path: string[] = [];\n      const cycle = findCycle(start, graph, visited, path, remaining);\n      if (cycle) cycles.push(cycle);\n    }\n  }\n\n  return { ordered, cycles };\n}\n\nfunction findCycle(\n  start: string,\n  graph: Map<string, Set<string>>,\n  visited: Set<string>,\n  path: string[],\n  remaining: Set<string>,\n): string[] | null {\n  if (path.includes(start)) {\n    const idx = path.indexOf(start);\n    return [...path.slice(idx), start];\n  }\n  if (visited.has(start) || !remaining.has(start)) return null;\n  visited.add(start);\n  path.push(start);\n  for (const next of graph.get(start) ?? []) {\n    const cycle = findCycle(next, graph, visited, path, remaining);\n    if (cycle) return cycle;\n  }\n  path.pop();\n  return null;\n}\n"]}
@@ -0,0 +1,45 @@
1
+ import { ErrorCode } from '@pipeline-builder/api-core';
2
+ import { FieldPredicate } from './walker';
3
+ export interface TemplateError {
4
+ field: string;
5
+ line?: number;
6
+ col?: number;
7
+ code: ErrorCode;
8
+ message: string;
9
+ path?: string;
10
+ cycle?: string[];
11
+ }
12
+ export interface ValidationResult {
13
+ valid: boolean;
14
+ errors: TemplateError[];
15
+ }
16
+ /**
17
+ * Build a predicate that accepts a template path when its root is in the
18
+ * given allow-list. Used by the validator to check that every `{{ path }}`
19
+ * resolves to a known scope root (`pipeline`, `plugin`, `env`, …).
20
+ *
21
+ * Only the root segment is checked — we don't enforce exact nested-key
22
+ * existence here because metadata / vars keys are user-provided per pipeline
23
+ * and not known at plugin upload time.
24
+ */
25
+ export declare function allowedScopeRoots(roots: string[]): (path: string[]) => boolean;
26
+ /**
27
+ * Validate all template tokens in a document have well-formed paths and
28
+ * point at allowed scope roots. Returns all errors found — does NOT stop
29
+ * on first.
30
+ *
31
+ * Caller supplies `isTemplatable` (the schema allow-list) and
32
+ * `isKnownPath` (the scope-shape predicate). Template text is parsed
33
+ * fresh; use a TokenCache externally if repeated parsing is a concern.
34
+ */
35
+ export declare function validateTemplates<T extends object>(doc: T, isTemplatable: FieldPredicate, isKnownPath: (path: string[]) => boolean): ValidationResult;
36
+ /**
37
+ * Detect cycles in a self-referencing document. Caller supplies the
38
+ * templatable predicate and a function to extract the scope-path that
39
+ * each template field writes to (same value used as a key in deps).
40
+ *
41
+ * Example for pipeline.json: the field `metadata.env` writes to scope
42
+ * path `metadata.env`; a template `{{ metadata.region }}` in that field
43
+ * declares a dependency on `metadata.region`.
44
+ */
45
+ export declare function detectCycles<T extends object>(doc: T, isTemplatable: FieldPredicate, fieldToScopePath: (field: string) => string | null): TemplateError[];