@tagma/sdk 0.6.3 → 0.6.5
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 +8 -5
- package/dist/dag.test.d.ts +2 -0
- package/dist/dag.test.d.ts.map +1 -0
- package/dist/dag.test.js +42 -0
- package/dist/dag.test.js.map +1 -0
- package/dist/engine-ports.test.d.ts +2 -0
- package/dist/engine-ports.test.d.ts.map +1 -0
- package/dist/engine-ports.test.js +378 -0
- package/dist/engine-ports.test.js.map +1 -0
- package/dist/engine.d.ts.map +1 -1
- package/dist/engine.js +194 -21
- package/dist/engine.js.map +1 -1
- package/dist/pipeline-runner.d.ts.map +1 -1
- package/dist/pipeline-runner.js +3 -0
- package/dist/pipeline-runner.js.map +1 -1
- package/dist/ports.d.ts +118 -0
- package/dist/ports.d.ts.map +1 -0
- package/dist/ports.js +365 -0
- package/dist/ports.js.map +1 -0
- package/dist/ports.test.d.ts +2 -0
- package/dist/ports.test.d.ts.map +1 -0
- package/dist/ports.test.js +262 -0
- package/dist/ports.test.js.map +1 -0
- package/dist/prompt-doc.d.ts +35 -1
- package/dist/prompt-doc.d.ts.map +1 -1
- package/dist/prompt-doc.js +110 -0
- package/dist/prompt-doc.js.map +1 -1
- package/dist/prompt-doc.test.d.ts +2 -0
- package/dist/prompt-doc.test.d.ts.map +1 -0
- package/dist/prompt-doc.test.js +145 -0
- package/dist/prompt-doc.test.js.map +1 -0
- package/dist/runner.d.ts +17 -0
- package/dist/runner.d.ts.map +1 -1
- package/dist/runner.js +171 -8
- package/dist/runner.js.map +1 -1
- package/dist/runner.test.d.ts +2 -0
- package/dist/runner.test.d.ts.map +1 -0
- package/dist/runner.test.js +119 -0
- package/dist/runner.test.js.map +1 -0
- package/dist/schema-ports.test.d.ts +2 -0
- package/dist/schema-ports.test.d.ts.map +1 -0
- package/dist/schema-ports.test.js +219 -0
- package/dist/schema-ports.test.js.map +1 -0
- package/dist/schema.d.ts.map +1 -1
- package/dist/schema.js +8 -0
- package/dist/schema.js.map +1 -1
- package/dist/sdk.d.ts +3 -1
- package/dist/sdk.d.ts.map +1 -1
- package/dist/sdk.js +5 -1
- package/dist/sdk.js.map +1 -1
- package/dist/validate-raw-ports.test.d.ts +2 -0
- package/dist/validate-raw-ports.test.d.ts.map +1 -0
- package/dist/validate-raw-ports.test.js +157 -0
- package/dist/validate-raw-ports.test.js.map +1 -0
- package/dist/validate-raw.d.ts.map +1 -1
- package/dist/validate-raw.js +141 -0
- package/dist/validate-raw.js.map +1 -1
- package/package.json +2 -7
- package/src/dag.test.ts +56 -0
- package/src/engine-ports.test.ts +404 -0
- package/src/engine.ts +231 -24
- package/src/pipeline-runner.ts +3 -0
- package/src/ports.test.ts +301 -0
- package/src/ports.ts +442 -0
- package/src/prompt-doc.test.ts +174 -0
- package/src/prompt-doc.ts +121 -1
- package/src/runner.test.ts +142 -0
- package/src/runner.ts +198 -8
- package/src/schema-ports.test.ts +236 -0
- package/src/schema.ts +8 -0
- package/src/sdk.ts +14 -0
- package/src/validate-raw-ports.test.ts +198 -0
- package/src/validate-raw.ts +155 -1
|
@@ -0,0 +1,262 @@
|
|
|
1
|
+
import { describe, expect, test } from 'bun:test';
|
|
2
|
+
import { extractInputReferences, extractTaskOutputs, resolveTaskInputs, substituteInputs, } from './ports';
|
|
3
|
+
const PERMS = { read: true, write: false, execute: false };
|
|
4
|
+
function task(overrides) {
|
|
5
|
+
return {
|
|
6
|
+
name: overrides.id,
|
|
7
|
+
permissions: PERMS,
|
|
8
|
+
...overrides,
|
|
9
|
+
};
|
|
10
|
+
}
|
|
11
|
+
// ─── substituteInputs ────────────────────────────────────────────────
|
|
12
|
+
describe('substituteInputs', () => {
|
|
13
|
+
test('replaces single placeholder with string value', () => {
|
|
14
|
+
const { text, unresolved } = substituteInputs('hello {{inputs.name}}', { name: 'world' });
|
|
15
|
+
expect(text).toBe('hello world');
|
|
16
|
+
expect(unresolved).toEqual([]);
|
|
17
|
+
});
|
|
18
|
+
test('allows optional whitespace inside braces', () => {
|
|
19
|
+
const { text } = substituteInputs('{{ inputs.name }} / {{inputs.name}}', { name: 'x' });
|
|
20
|
+
expect(text).toBe('x / x');
|
|
21
|
+
});
|
|
22
|
+
test('stringifies number / boolean values verbatim', () => {
|
|
23
|
+
const { text } = substituteInputs('n={{inputs.n}} b={{inputs.b}}', { n: 42, b: true });
|
|
24
|
+
expect(text).toBe('n=42 b=true');
|
|
25
|
+
});
|
|
26
|
+
test('JSON-stringifies object values', () => {
|
|
27
|
+
const { text } = substituteInputs('payload={{inputs.p}}', {
|
|
28
|
+
p: { a: 1, b: 'x' },
|
|
29
|
+
});
|
|
30
|
+
expect(text).toBe('payload={"a":1,"b":"x"}');
|
|
31
|
+
});
|
|
32
|
+
test('renders unknown placeholder empty and reports it', () => {
|
|
33
|
+
const { text, unresolved } = substituteInputs('hello {{inputs.missing}}', {});
|
|
34
|
+
expect(text).toBe('hello ');
|
|
35
|
+
expect(unresolved).toEqual(['missing']);
|
|
36
|
+
});
|
|
37
|
+
test('renders null / undefined as empty and reports', () => {
|
|
38
|
+
const { text, unresolved } = substituteInputs('a={{inputs.a}} b={{inputs.b}}', {
|
|
39
|
+
a: null,
|
|
40
|
+
b: undefined,
|
|
41
|
+
});
|
|
42
|
+
expect(text).toBe('a= b=');
|
|
43
|
+
expect([...unresolved].sort()).toEqual(['a', 'b']);
|
|
44
|
+
});
|
|
45
|
+
test('leaves malformed placeholders alone', () => {
|
|
46
|
+
const { text } = substituteInputs('{{inputs.a.b}} {{inputs.}}', { a: 'x' });
|
|
47
|
+
expect(text).toBe('{{inputs.a.b}} {{inputs.}}');
|
|
48
|
+
});
|
|
49
|
+
test('handles circular objects without throwing', () => {
|
|
50
|
+
const obj = { self: null };
|
|
51
|
+
obj.self = obj;
|
|
52
|
+
const { text, unresolved } = substituteInputs('{{inputs.x}}', { x: obj });
|
|
53
|
+
expect(text).toBe('');
|
|
54
|
+
expect(unresolved).toEqual(['x']);
|
|
55
|
+
});
|
|
56
|
+
});
|
|
57
|
+
describe('extractInputReferences', () => {
|
|
58
|
+
test('returns unique referenced names', () => {
|
|
59
|
+
const refs = extractInputReferences('get {{inputs.city}} for id={{inputs.id}} and {{inputs.city}} again');
|
|
60
|
+
expect(refs.sort()).toEqual(['city', 'id']);
|
|
61
|
+
});
|
|
62
|
+
test('returns empty for text without placeholders', () => {
|
|
63
|
+
expect(extractInputReferences('no placeholders here')).toEqual([]);
|
|
64
|
+
});
|
|
65
|
+
});
|
|
66
|
+
// ─── resolveTaskInputs ────────────────────────────────────────────────
|
|
67
|
+
const cityPort = { name: 'city', type: 'string', required: true };
|
|
68
|
+
const idPort = { name: 'id', type: 'number', required: true };
|
|
69
|
+
describe('resolveTaskInputs', () => {
|
|
70
|
+
test('no declared inputs → ready with empty map', () => {
|
|
71
|
+
const t = task({ id: 'downstream', command: 'echo' });
|
|
72
|
+
const res = resolveTaskInputs(t, new Map(), []);
|
|
73
|
+
expect(res).toEqual({ kind: 'ready', inputs: {}, missingOptional: [] });
|
|
74
|
+
});
|
|
75
|
+
test('matches inputs by name across upstream outputs', () => {
|
|
76
|
+
const t = task({
|
|
77
|
+
id: 'downstream',
|
|
78
|
+
command: 'echo',
|
|
79
|
+
ports: { inputs: [cityPort, idPort] },
|
|
80
|
+
});
|
|
81
|
+
const upstream = new Map([
|
|
82
|
+
['t.prompt', { city: 'Shanghai' }],
|
|
83
|
+
['t.other', { id: 42 }],
|
|
84
|
+
]);
|
|
85
|
+
const res = resolveTaskInputs(t, upstream, ['t.prompt', 't.other']);
|
|
86
|
+
expect(res.kind).toBe('ready');
|
|
87
|
+
if (res.kind !== 'ready')
|
|
88
|
+
return;
|
|
89
|
+
expect(res.inputs).toEqual({ city: 'Shanghai', id: 42 });
|
|
90
|
+
});
|
|
91
|
+
test('required missing blocks with a readable reason', () => {
|
|
92
|
+
const t = task({
|
|
93
|
+
id: 'downstream',
|
|
94
|
+
command: 'echo',
|
|
95
|
+
ports: { inputs: [cityPort, idPort] },
|
|
96
|
+
});
|
|
97
|
+
const res = resolveTaskInputs(t, new Map(), ['t.x']);
|
|
98
|
+
expect(res.kind).toBe('blocked');
|
|
99
|
+
if (res.kind !== 'blocked')
|
|
100
|
+
return;
|
|
101
|
+
expect([...res.missingRequired].sort()).toEqual(['city', 'id']);
|
|
102
|
+
expect(res.reason).toMatch(/city.*id|id.*city/);
|
|
103
|
+
});
|
|
104
|
+
test('optional missing yields ready but reports missingOptional', () => {
|
|
105
|
+
const optional = { name: 'note', type: 'string' };
|
|
106
|
+
const t = task({
|
|
107
|
+
id: 'downstream',
|
|
108
|
+
command: 'echo',
|
|
109
|
+
ports: { inputs: [optional] },
|
|
110
|
+
});
|
|
111
|
+
const res = resolveTaskInputs(t, new Map(), []);
|
|
112
|
+
expect(res.kind).toBe('ready');
|
|
113
|
+
if (res.kind !== 'ready')
|
|
114
|
+
return;
|
|
115
|
+
expect(res.inputs).toEqual({});
|
|
116
|
+
expect(res.missingOptional).toEqual(['note']);
|
|
117
|
+
});
|
|
118
|
+
test('applies default for missing optional', () => {
|
|
119
|
+
const optional = { name: 'note', type: 'string', default: 'n/a' };
|
|
120
|
+
const t = task({
|
|
121
|
+
id: 'd',
|
|
122
|
+
command: 'echo',
|
|
123
|
+
ports: { inputs: [optional] },
|
|
124
|
+
});
|
|
125
|
+
const res = resolveTaskInputs(t, new Map(), []);
|
|
126
|
+
expect(res.kind).toBe('ready');
|
|
127
|
+
if (res.kind !== 'ready')
|
|
128
|
+
return;
|
|
129
|
+
expect(res.inputs).toEqual({ note: 'n/a' });
|
|
130
|
+
});
|
|
131
|
+
test('ambiguous multi-upstream match blocks unless disambiguated', () => {
|
|
132
|
+
const t = task({
|
|
133
|
+
id: 'd',
|
|
134
|
+
command: 'echo',
|
|
135
|
+
ports: { inputs: [cityPort] },
|
|
136
|
+
});
|
|
137
|
+
const upstream = new Map([
|
|
138
|
+
['t.a', { city: 'Shanghai' }],
|
|
139
|
+
['t.b', { city: 'Beijing' }],
|
|
140
|
+
]);
|
|
141
|
+
const res = resolveTaskInputs(t, upstream, ['t.a', 't.b']);
|
|
142
|
+
expect(res.kind).toBe('blocked');
|
|
143
|
+
if (res.kind !== 'blocked')
|
|
144
|
+
return;
|
|
145
|
+
expect(res.ambiguous.length).toBe(1);
|
|
146
|
+
expect(res.ambiguous[0].port).toBe('city');
|
|
147
|
+
expect([...res.ambiguous[0].producers].sort()).toEqual(['t.a', 't.b']);
|
|
148
|
+
});
|
|
149
|
+
test('explicit fully-qualified "from" wins over name-match ambiguity', () => {
|
|
150
|
+
const explicit = {
|
|
151
|
+
name: 'city',
|
|
152
|
+
type: 'string',
|
|
153
|
+
required: true,
|
|
154
|
+
from: 't.b.city',
|
|
155
|
+
};
|
|
156
|
+
const t = task({
|
|
157
|
+
id: 'd',
|
|
158
|
+
command: 'echo',
|
|
159
|
+
ports: { inputs: [explicit] },
|
|
160
|
+
});
|
|
161
|
+
const upstream = new Map([
|
|
162
|
+
['t.a', { city: 'Shanghai' }],
|
|
163
|
+
['t.b', { city: 'Beijing' }],
|
|
164
|
+
]);
|
|
165
|
+
const res = resolveTaskInputs(t, upstream, ['t.a', 't.b']);
|
|
166
|
+
expect(res.kind).toBe('ready');
|
|
167
|
+
if (res.kind !== 'ready')
|
|
168
|
+
return;
|
|
169
|
+
expect(res.inputs).toEqual({ city: 'Beijing' });
|
|
170
|
+
});
|
|
171
|
+
test('coerces numeric strings to number type', () => {
|
|
172
|
+
const t = task({
|
|
173
|
+
id: 'd',
|
|
174
|
+
command: 'echo',
|
|
175
|
+
ports: { inputs: [idPort] },
|
|
176
|
+
});
|
|
177
|
+
const upstream = new Map([['t.a', { id: '42' }]]);
|
|
178
|
+
const res = resolveTaskInputs(t, upstream, ['t.a']);
|
|
179
|
+
expect(res.kind).toBe('ready');
|
|
180
|
+
if (res.kind !== 'ready')
|
|
181
|
+
return;
|
|
182
|
+
expect(res.inputs.id).toBe(42);
|
|
183
|
+
});
|
|
184
|
+
test('flags type-coercion failures as blocked', () => {
|
|
185
|
+
const t = task({
|
|
186
|
+
id: 'd',
|
|
187
|
+
command: 'echo',
|
|
188
|
+
ports: { inputs: [idPort] },
|
|
189
|
+
});
|
|
190
|
+
const upstream = new Map([['t.a', { id: 'nope' }]]);
|
|
191
|
+
const res = resolveTaskInputs(t, upstream, ['t.a']);
|
|
192
|
+
expect(res.kind).toBe('blocked');
|
|
193
|
+
if (res.kind !== 'blocked')
|
|
194
|
+
return;
|
|
195
|
+
expect(res.typeErrors.length).toBe(1);
|
|
196
|
+
expect(res.typeErrors[0].port).toBe('id');
|
|
197
|
+
});
|
|
198
|
+
test('enforces enum membership', () => {
|
|
199
|
+
const colorPort = {
|
|
200
|
+
name: 'color',
|
|
201
|
+
type: 'enum',
|
|
202
|
+
enum: ['red', 'green'],
|
|
203
|
+
required: true,
|
|
204
|
+
};
|
|
205
|
+
const t = task({
|
|
206
|
+
id: 'd',
|
|
207
|
+
command: 'echo',
|
|
208
|
+
ports: { inputs: [colorPort] },
|
|
209
|
+
});
|
|
210
|
+
const upstream = new Map([['t.a', { color: 'blue' }]]);
|
|
211
|
+
const res = resolveTaskInputs(t, upstream, ['t.a']);
|
|
212
|
+
expect(res.kind).toBe('blocked');
|
|
213
|
+
if (res.kind !== 'blocked')
|
|
214
|
+
return;
|
|
215
|
+
expect(res.typeErrors[0].port).toBe('color');
|
|
216
|
+
});
|
|
217
|
+
});
|
|
218
|
+
// ─── extractTaskOutputs ──────────────────────────────────────────────
|
|
219
|
+
describe('extractTaskOutputs', () => {
|
|
220
|
+
const outputs = [
|
|
221
|
+
{ name: 'city', type: 'string' },
|
|
222
|
+
{ name: 'temp', type: 'number' },
|
|
223
|
+
];
|
|
224
|
+
test('no declared outputs → empty map, null diagnostic', () => {
|
|
225
|
+
const r = extractTaskOutputs(undefined, 'anything', null);
|
|
226
|
+
expect(r.outputs).toEqual({});
|
|
227
|
+
expect(r.diagnostic).toBeNull();
|
|
228
|
+
});
|
|
229
|
+
test('parses last-line JSON object as source record', () => {
|
|
230
|
+
const stdout = 'some log\nmore log\n{"city":"Shanghai","temp":23}\n';
|
|
231
|
+
const r = extractTaskOutputs({ outputs }, stdout, null);
|
|
232
|
+
expect(r.outputs).toEqual({ city: 'Shanghai', temp: 23 });
|
|
233
|
+
expect(r.diagnostic).toBeNull();
|
|
234
|
+
});
|
|
235
|
+
test('falls back to whole-source JSON when last line is a closing brace', () => {
|
|
236
|
+
const stdout = '{\n "city": "Shanghai",\n "temp": 23\n}\n';
|
|
237
|
+
const r = extractTaskOutputs({ outputs }, stdout, null);
|
|
238
|
+
expect(r.outputs).toEqual({ city: 'Shanghai', temp: 23 });
|
|
239
|
+
});
|
|
240
|
+
test('prefers normalizedOutput over stdout when provided', () => {
|
|
241
|
+
const stdout = '{"city":"Wrong","temp":0}';
|
|
242
|
+
const normalized = '{"city":"Shanghai","temp":23}';
|
|
243
|
+
const r = extractTaskOutputs({ outputs }, stdout, normalized);
|
|
244
|
+
expect(r.outputs).toEqual({ city: 'Shanghai', temp: 23 });
|
|
245
|
+
});
|
|
246
|
+
test('reports missing keys as diagnostic, keeps resolved keys', () => {
|
|
247
|
+
const r = extractTaskOutputs({ outputs }, '{"city":"Shanghai"}', null);
|
|
248
|
+
expect(r.outputs).toEqual({ city: 'Shanghai' });
|
|
249
|
+
expect(r.diagnostic).toContain('missing key "temp"');
|
|
250
|
+
});
|
|
251
|
+
test('reports coercion failure and skips bad port', () => {
|
|
252
|
+
const r = extractTaskOutputs({ outputs }, '{"city":"Shanghai","temp":"not-a-number"}', null);
|
|
253
|
+
expect(r.outputs).toEqual({ city: 'Shanghai' });
|
|
254
|
+
expect(r.diagnostic).toContain('"temp"');
|
|
255
|
+
});
|
|
256
|
+
test('reports diagnostic when no JSON can be parsed', () => {
|
|
257
|
+
const r = extractTaskOutputs({ outputs }, 'plain text output\nnothing json\n', null);
|
|
258
|
+
expect(r.outputs).toEqual({});
|
|
259
|
+
expect(r.diagnostic).toContain('could not find a final-line JSON object');
|
|
260
|
+
});
|
|
261
|
+
});
|
|
262
|
+
//# sourceMappingURL=ports.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ports.test.js","sourceRoot":"","sources":["../src/ports.test.ts"],"names":[],"mappings":"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"}
|
package/dist/prompt-doc.d.ts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import type {
|
|
1
|
+
import type { PortDef, PromptContextBlock, PromptDocument } from './types';
|
|
2
2
|
/**
|
|
3
3
|
* Build a fresh `PromptDocument` from a raw task string.
|
|
4
4
|
* Middlewares receive this from the engine and push context blocks onto
|
|
@@ -33,4 +33,38 @@ export declare function serializePromptDocument(doc: PromptDocument): string;
|
|
|
33
33
|
* appended to `contexts`, preserving immutability of `doc`.
|
|
34
34
|
*/
|
|
35
35
|
export declare function appendContext(doc: PromptDocument, block: PromptContextBlock): PromptDocument;
|
|
36
|
+
/**
|
|
37
|
+
* Helper: return a new document with the given block PREPENDED. The
|
|
38
|
+
* engine uses this to place port-related context blocks (`[Inputs]`,
|
|
39
|
+
* `[Output Format]`) at the top of the document so middlewares that
|
|
40
|
+
* assemble retrieval context against the task's inputs see them.
|
|
41
|
+
*/
|
|
42
|
+
export declare function prependContext(doc: PromptDocument, block: PromptContextBlock): PromptDocument;
|
|
43
|
+
/**
|
|
44
|
+
* Build an `[Inputs]` context block from a map of resolved port inputs.
|
|
45
|
+
* Each input is rendered on its own line as `name: <value>` with an
|
|
46
|
+
* optional trailing `# <description>` comment so the model has both the
|
|
47
|
+
* value and the reason it matters.
|
|
48
|
+
*
|
|
49
|
+
* The block is *only* useful for AI tasks; command tasks consume inputs
|
|
50
|
+
* through `{{inputs.X}}` substitution in their command line and do not
|
|
51
|
+
* need this context.
|
|
52
|
+
*
|
|
53
|
+
* Returns null when there are no inputs to render — callers can forward
|
|
54
|
+
* that nullish value to `prependContext` via an `if (block)` check so
|
|
55
|
+
* empty-input tasks don't grow a noise block in their prompt.
|
|
56
|
+
*/
|
|
57
|
+
export declare function renderInputsBlock(inputsDecl: readonly PortDef[] | undefined, values: Readonly<Record<string, unknown>>): PromptContextBlock | null;
|
|
58
|
+
/**
|
|
59
|
+
* Build an `[Output Format]` context block from a task's declared output
|
|
60
|
+
* ports. The block instructs the model to emit a final-line JSON object
|
|
61
|
+
* matching the declared schema so `extractTaskOutputs` can pick it up
|
|
62
|
+
* without fragile heuristics. Returns null when the task declares no
|
|
63
|
+
* outputs.
|
|
64
|
+
*
|
|
65
|
+
* The instruction is deliberately short and explicit — a terse "emit
|
|
66
|
+
* this object as JSON on the final line" beats a long schema dump
|
|
67
|
+
* because shorter prompts compose better with downstream middlewares.
|
|
68
|
+
*/
|
|
69
|
+
export declare function renderOutputSchemaBlock(outputsDecl: readonly PortDef[] | undefined): PromptContextBlock | null;
|
|
36
70
|
//# sourceMappingURL=prompt-doc.d.ts.map
|
package/dist/prompt-doc.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"prompt-doc.d.ts","sourceRoot":"","sources":["../src/prompt-doc.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,
|
|
1
|
+
{"version":3,"file":"prompt-doc.d.ts","sourceRoot":"","sources":["../src/prompt-doc.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,OAAO,EAAE,kBAAkB,EAAE,cAAc,EAAE,MAAM,SAAS,CAAC;AAE3E;;;;;GAKG;AACH,wBAAgB,wBAAwB,CAAC,IAAI,EAAE,MAAM,GAAG,cAAc,CAErE;AAED;;;;;;;;;;;;;;;;;;;;GAoBG;AACH,wBAAgB,uBAAuB,CAAC,GAAG,EAAE,cAAc,GAAG,MAAM,CAInE;AAED;;;GAGG;AACH,wBAAgB,aAAa,CAC3B,GAAG,EAAE,cAAc,EACnB,KAAK,EAAE,kBAAkB,GACxB,cAAc,CAEhB;AAED;;;;;GAKG;AACH,wBAAgB,cAAc,CAC5B,GAAG,EAAE,cAAc,EACnB,KAAK,EAAE,kBAAkB,GACxB,cAAc,CAEhB;AAED;;;;;;;;;;;;;GAaG;AACH,wBAAgB,iBAAiB,CAC/B,UAAU,EAAE,SAAS,OAAO,EAAE,GAAG,SAAS,EAC1C,MAAM,EAAE,QAAQ,CAAC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC,GACxC,kBAAkB,GAAG,IAAI,CAY3B;AAaD;;;;;;;;;;GAUG;AACH,wBAAgB,uBAAuB,CACrC,WAAW,EAAE,SAAS,OAAO,EAAE,GAAG,SAAS,GAC1C,kBAAkB,GAAG,IAAI,CAsB3B"}
|
package/dist/prompt-doc.js
CHANGED
|
@@ -41,4 +41,114 @@ export function serializePromptDocument(doc) {
|
|
|
41
41
|
export function appendContext(doc, block) {
|
|
42
42
|
return { contexts: [...doc.contexts, block], task: doc.task };
|
|
43
43
|
}
|
|
44
|
+
/**
|
|
45
|
+
* Helper: return a new document with the given block PREPENDED. The
|
|
46
|
+
* engine uses this to place port-related context blocks (`[Inputs]`,
|
|
47
|
+
* `[Output Format]`) at the top of the document so middlewares that
|
|
48
|
+
* assemble retrieval context against the task's inputs see them.
|
|
49
|
+
*/
|
|
50
|
+
export function prependContext(doc, block) {
|
|
51
|
+
return { contexts: [block, ...doc.contexts], task: doc.task };
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* Build an `[Inputs]` context block from a map of resolved port inputs.
|
|
55
|
+
* Each input is rendered on its own line as `name: <value>` with an
|
|
56
|
+
* optional trailing `# <description>` comment so the model has both the
|
|
57
|
+
* value and the reason it matters.
|
|
58
|
+
*
|
|
59
|
+
* The block is *only* useful for AI tasks; command tasks consume inputs
|
|
60
|
+
* through `{{inputs.X}}` substitution in their command line and do not
|
|
61
|
+
* need this context.
|
|
62
|
+
*
|
|
63
|
+
* Returns null when there are no inputs to render — callers can forward
|
|
64
|
+
* that nullish value to `prependContext` via an `if (block)` check so
|
|
65
|
+
* empty-input tasks don't grow a noise block in their prompt.
|
|
66
|
+
*/
|
|
67
|
+
export function renderInputsBlock(inputsDecl, values) {
|
|
68
|
+
if (!inputsDecl || inputsDecl.length === 0)
|
|
69
|
+
return null;
|
|
70
|
+
const lines = [];
|
|
71
|
+
for (const port of inputsDecl) {
|
|
72
|
+
if (!(port.name in values))
|
|
73
|
+
continue;
|
|
74
|
+
const raw = values[port.name];
|
|
75
|
+
const rendered = renderInputValue(raw);
|
|
76
|
+
const descr = port.description?.trim();
|
|
77
|
+
lines.push(descr ? `${port.name}: ${rendered} # ${descr}` : `${port.name}: ${rendered}`);
|
|
78
|
+
}
|
|
79
|
+
if (lines.length === 0)
|
|
80
|
+
return null;
|
|
81
|
+
return { label: 'Inputs', content: lines.join('\n') };
|
|
82
|
+
}
|
|
83
|
+
function renderInputValue(value) {
|
|
84
|
+
if (value === null || value === undefined)
|
|
85
|
+
return '';
|
|
86
|
+
if (typeof value === 'string')
|
|
87
|
+
return JSON.stringify(value);
|
|
88
|
+
if (typeof value === 'number' || typeof value === 'boolean')
|
|
89
|
+
return String(value);
|
|
90
|
+
try {
|
|
91
|
+
return JSON.stringify(value);
|
|
92
|
+
}
|
|
93
|
+
catch {
|
|
94
|
+
return String(value);
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* Build an `[Output Format]` context block from a task's declared output
|
|
99
|
+
* ports. The block instructs the model to emit a final-line JSON object
|
|
100
|
+
* matching the declared schema so `extractTaskOutputs` can pick it up
|
|
101
|
+
* without fragile heuristics. Returns null when the task declares no
|
|
102
|
+
* outputs.
|
|
103
|
+
*
|
|
104
|
+
* The instruction is deliberately short and explicit — a terse "emit
|
|
105
|
+
* this object as JSON on the final line" beats a long schema dump
|
|
106
|
+
* because shorter prompts compose better with downstream middlewares.
|
|
107
|
+
*/
|
|
108
|
+
export function renderOutputSchemaBlock(outputsDecl) {
|
|
109
|
+
if (!outputsDecl || outputsDecl.length === 0)
|
|
110
|
+
return null;
|
|
111
|
+
const lines = [];
|
|
112
|
+
lines.push('After your response, emit a single JSON object on the FINAL line with these keys:');
|
|
113
|
+
for (const port of outputsDecl) {
|
|
114
|
+
const descr = port.description?.trim();
|
|
115
|
+
const enumHint = port.type === 'enum' && port.enum?.length
|
|
116
|
+
? ` (one of: ${port.enum.map((v) => JSON.stringify(v)).join(', ')})`
|
|
117
|
+
: '';
|
|
118
|
+
lines.push(descr
|
|
119
|
+
? ` - ${port.name} (${port.type}${enumHint}): ${descr}`
|
|
120
|
+
: ` - ${port.name} (${port.type}${enumHint})`);
|
|
121
|
+
}
|
|
122
|
+
const example = buildExampleObject(outputsDecl);
|
|
123
|
+
lines.push('');
|
|
124
|
+
lines.push(`Example final line: ${JSON.stringify(example)}`);
|
|
125
|
+
return { label: 'Output Format', content: lines.join('\n') };
|
|
126
|
+
}
|
|
127
|
+
function buildExampleObject(outputsDecl) {
|
|
128
|
+
const example = {};
|
|
129
|
+
for (const port of outputsDecl) {
|
|
130
|
+
if (port.default !== undefined) {
|
|
131
|
+
example[port.name] = port.default;
|
|
132
|
+
continue;
|
|
133
|
+
}
|
|
134
|
+
switch (port.type) {
|
|
135
|
+
case 'string':
|
|
136
|
+
example[port.name] = '...';
|
|
137
|
+
break;
|
|
138
|
+
case 'number':
|
|
139
|
+
example[port.name] = 0;
|
|
140
|
+
break;
|
|
141
|
+
case 'boolean':
|
|
142
|
+
example[port.name] = false;
|
|
143
|
+
break;
|
|
144
|
+
case 'enum':
|
|
145
|
+
example[port.name] = port.enum?.[0] ?? '...';
|
|
146
|
+
break;
|
|
147
|
+
case 'json':
|
|
148
|
+
default:
|
|
149
|
+
example[port.name] = null;
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
return example;
|
|
153
|
+
}
|
|
44
154
|
//# sourceMappingURL=prompt-doc.js.map
|
package/dist/prompt-doc.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"prompt-doc.js","sourceRoot":"","sources":["../src/prompt-doc.ts"],"names":[],"mappings":"AAEA;;;;;GAKG;AACH,MAAM,UAAU,wBAAwB,CAAC,IAAY;IACnD,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC;AAChC,CAAC;AAED;;;;;;;;;;;;;;;;;;;;GAoBG;AACH,MAAM,UAAU,uBAAuB,CAAC,GAAmB;IACzD,IAAI,GAAG,CAAC,QAAQ,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,GAAG,CAAC,IAAI,CAAC;IAC/C,MAAM,MAAM,GAAG,GAAG,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,CAAC,KAAK,MAAM,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC;IACrE,OAAO,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,GAAG,CAAC,IAAI,EAAE,CAAC;AACjD,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,aAAa,CAC3B,GAAmB,EACnB,KAAyB;IAEzB,OAAO,EAAE,QAAQ,EAAE,CAAC,GAAG,GAAG,CAAC,QAAQ,EAAE,KAAK,CAAC,EAAE,IAAI,EAAE,GAAG,CAAC,IAAI,EAAE,CAAC;AAChE,CAAC"}
|
|
1
|
+
{"version":3,"file":"prompt-doc.js","sourceRoot":"","sources":["../src/prompt-doc.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"prompt-doc.test.d.ts","sourceRoot":"","sources":["../src/prompt-doc.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,145 @@
|
|
|
1
|
+
import { describe, expect, test } from 'bun:test';
|
|
2
|
+
import { appendContext, prependContext, promptDocumentFromString, renderInputsBlock, renderOutputSchemaBlock, serializePromptDocument, } from './prompt-doc';
|
|
3
|
+
// ─── renderInputsBlock ────────────────────────────────────────────────
|
|
4
|
+
describe('renderInputsBlock', () => {
|
|
5
|
+
test('returns null when no inputs declared', () => {
|
|
6
|
+
expect(renderInputsBlock(undefined, {})).toBeNull();
|
|
7
|
+
expect(renderInputsBlock([], { any: 'x' })).toBeNull();
|
|
8
|
+
});
|
|
9
|
+
test('returns null when declared inputs have no resolved values', () => {
|
|
10
|
+
const ports = [{ name: 'city', type: 'string' }];
|
|
11
|
+
// values missing entirely — block is noise, omit it
|
|
12
|
+
expect(renderInputsBlock(ports, {})).toBeNull();
|
|
13
|
+
});
|
|
14
|
+
test('renders name: value per declared input', () => {
|
|
15
|
+
const ports = [
|
|
16
|
+
{ name: 'city', type: 'string' },
|
|
17
|
+
{ name: 'id', type: 'number' },
|
|
18
|
+
];
|
|
19
|
+
const block = renderInputsBlock(ports, { city: 'Shanghai', id: 42 });
|
|
20
|
+
expect(block).not.toBeNull();
|
|
21
|
+
expect(block.label).toBe('Inputs');
|
|
22
|
+
expect(block.content).toBe('city: "Shanghai"\nid: 42');
|
|
23
|
+
});
|
|
24
|
+
test('appends # description comment when provided', () => {
|
|
25
|
+
const ports = [
|
|
26
|
+
{ name: 'city', type: 'string', description: 'Target city' },
|
|
27
|
+
];
|
|
28
|
+
const block = renderInputsBlock(ports, { city: 'Shanghai' });
|
|
29
|
+
expect(block.content).toBe('city: "Shanghai" # Target city');
|
|
30
|
+
});
|
|
31
|
+
test('preserves declaration order, not input-map iteration order', () => {
|
|
32
|
+
const ports = [
|
|
33
|
+
{ name: 'b', type: 'string' },
|
|
34
|
+
{ name: 'a', type: 'string' },
|
|
35
|
+
];
|
|
36
|
+
// Values object has 'a' first, 'b' second — block should still emit 'b' first.
|
|
37
|
+
const block = renderInputsBlock(ports, { a: 'x', b: 'y' });
|
|
38
|
+
expect(block.content).toBe('b: "y"\na: "x"');
|
|
39
|
+
});
|
|
40
|
+
test('skips ports whose values were not resolved', () => {
|
|
41
|
+
const ports = [
|
|
42
|
+
{ name: 'a', type: 'string' },
|
|
43
|
+
{ name: 'b', type: 'string' },
|
|
44
|
+
];
|
|
45
|
+
const block = renderInputsBlock(ports, { a: 'x' });
|
|
46
|
+
expect(block.content).toBe('a: "x"');
|
|
47
|
+
});
|
|
48
|
+
test('JSON-encodes non-primitive values', () => {
|
|
49
|
+
const ports = [{ name: 'payload', type: 'json' }];
|
|
50
|
+
const block = renderInputsBlock(ports, { payload: { a: 1, b: [2, 3] } });
|
|
51
|
+
expect(block.content).toBe('payload: {"a":1,"b":[2,3]}');
|
|
52
|
+
});
|
|
53
|
+
test('booleans render verbatim, not quoted', () => {
|
|
54
|
+
const ports = [{ name: 'flag', type: 'boolean' }];
|
|
55
|
+
const block = renderInputsBlock(ports, { flag: true });
|
|
56
|
+
expect(block.content).toBe('flag: true');
|
|
57
|
+
});
|
|
58
|
+
});
|
|
59
|
+
// ─── renderOutputSchemaBlock ──────────────────────────────────────────
|
|
60
|
+
describe('renderOutputSchemaBlock', () => {
|
|
61
|
+
test('returns null when no outputs declared', () => {
|
|
62
|
+
expect(renderOutputSchemaBlock(undefined)).toBeNull();
|
|
63
|
+
expect(renderOutputSchemaBlock([])).toBeNull();
|
|
64
|
+
});
|
|
65
|
+
test('instructs the model to emit final-line JSON', () => {
|
|
66
|
+
const ports = [{ name: 'city', type: 'string' }];
|
|
67
|
+
const block = renderOutputSchemaBlock(ports);
|
|
68
|
+
expect(block.label).toBe('Output Format');
|
|
69
|
+
expect(block.content).toMatch(/final line/i);
|
|
70
|
+
});
|
|
71
|
+
test('lists each port with its type', () => {
|
|
72
|
+
const ports = [
|
|
73
|
+
{ name: 'city', type: 'string', description: 'Target city' },
|
|
74
|
+
{ name: 'temp', type: 'number' },
|
|
75
|
+
];
|
|
76
|
+
const block = renderOutputSchemaBlock(ports);
|
|
77
|
+
expect(block.content).toContain('- city (string): Target city');
|
|
78
|
+
expect(block.content).toContain('- temp (number)');
|
|
79
|
+
});
|
|
80
|
+
test('includes enum values in the type hint', () => {
|
|
81
|
+
const ports = [
|
|
82
|
+
{ name: 'color', type: 'enum', enum: ['red', 'green', 'blue'] },
|
|
83
|
+
];
|
|
84
|
+
const block = renderOutputSchemaBlock(ports);
|
|
85
|
+
expect(block.content).toContain('color (enum (one of: "red", "green", "blue"))');
|
|
86
|
+
});
|
|
87
|
+
test('example object uses declared defaults when present', () => {
|
|
88
|
+
const ports = [
|
|
89
|
+
{ name: 'score', type: 'number', default: 0.5 },
|
|
90
|
+
{ name: 'note', type: 'string', default: 'n/a' },
|
|
91
|
+
];
|
|
92
|
+
const block = renderOutputSchemaBlock(ports);
|
|
93
|
+
// The example line is `Example final line: {"score":0.5,"note":"n/a"}`.
|
|
94
|
+
expect(block.content).toContain('"score":0.5');
|
|
95
|
+
expect(block.content).toContain('"note":"n/a"');
|
|
96
|
+
});
|
|
97
|
+
test('example uses type-appropriate placeholders when no default', () => {
|
|
98
|
+
const ports = [
|
|
99
|
+
{ name: 's', type: 'string' },
|
|
100
|
+
{ name: 'n', type: 'number' },
|
|
101
|
+
{ name: 'b', type: 'boolean' },
|
|
102
|
+
{ name: 'j', type: 'json' },
|
|
103
|
+
];
|
|
104
|
+
const block = renderOutputSchemaBlock(ports);
|
|
105
|
+
expect(block.content).toContain('"s":"..."');
|
|
106
|
+
expect(block.content).toContain('"n":0');
|
|
107
|
+
expect(block.content).toContain('"b":false');
|
|
108
|
+
expect(block.content).toContain('"j":null');
|
|
109
|
+
});
|
|
110
|
+
test('example uses first enum value when present', () => {
|
|
111
|
+
const ports = [
|
|
112
|
+
{ name: 'tier', type: 'enum', enum: ['low', 'high'] },
|
|
113
|
+
];
|
|
114
|
+
const block = renderOutputSchemaBlock(ports);
|
|
115
|
+
expect(block.content).toContain('"tier":"low"');
|
|
116
|
+
});
|
|
117
|
+
});
|
|
118
|
+
// ─── prependContext / appendContext ──────────────────────────────────
|
|
119
|
+
describe('prependContext / appendContext', () => {
|
|
120
|
+
const block = { label: 'X', content: 'x' };
|
|
121
|
+
test('prependContext puts block at front without mutating input', () => {
|
|
122
|
+
const doc = { contexts: [{ label: 'Y', content: 'y' }], task: 't' };
|
|
123
|
+
const next = prependContext(doc, block);
|
|
124
|
+
expect(next.contexts.map((c) => c.label)).toEqual(['X', 'Y']);
|
|
125
|
+
// Original untouched — immutability is part of the contract for
|
|
126
|
+
// middleware safety (the engine compares doc identity to detect
|
|
127
|
+
// changes in some paths).
|
|
128
|
+
expect(doc.contexts).toHaveLength(1);
|
|
129
|
+
expect(doc.contexts[0].label).toBe('Y');
|
|
130
|
+
expect(next.task).toBe('t');
|
|
131
|
+
});
|
|
132
|
+
test('appendContext puts block at end without mutating input', () => {
|
|
133
|
+
const doc = { contexts: [{ label: 'Y', content: 'y' }], task: 't' };
|
|
134
|
+
const next = appendContext(doc, block);
|
|
135
|
+
expect(next.contexts.map((c) => c.label)).toEqual(['Y', 'X']);
|
|
136
|
+
expect(doc.contexts).toHaveLength(1);
|
|
137
|
+
});
|
|
138
|
+
test('prepend + serialize produces [X] block above the task', () => {
|
|
139
|
+
const doc = promptDocumentFromString('do the thing');
|
|
140
|
+
const next = prependContext(doc, { label: 'Inputs', content: 'city: "Shanghai"' });
|
|
141
|
+
const text = serializePromptDocument(next);
|
|
142
|
+
expect(text).toBe('[Inputs]\ncity: "Shanghai"\n\ndo the thing');
|
|
143
|
+
});
|
|
144
|
+
});
|
|
145
|
+
//# sourceMappingURL=prompt-doc.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"prompt-doc.test.js","sourceRoot":"","sources":["../src/prompt-doc.test.ts"],"names":[],"mappings":"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"}
|