@peers-app/peers-sdk 0.16.5 → 0.17.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/dist/contracts/__tests__/builder.test.d.ts +1 -0
- package/dist/contracts/__tests__/builder.test.js +426 -0
- package/dist/contracts/__tests__/extract.test.d.ts +1 -0
- package/dist/contracts/__tests__/extract.test.js +145 -0
- package/dist/contracts/__tests__/integration.test.d.ts +1 -0
- package/dist/contracts/__tests__/integration.test.js +348 -0
- package/dist/contracts/__tests__/registry.test.d.ts +1 -0
- package/dist/contracts/__tests__/registry.test.js +324 -0
- package/dist/contracts/__tests__/validate.test.d.ts +1 -0
- package/dist/contracts/__tests__/validate.test.js +699 -0
- package/dist/contracts/builder.d.ts +102 -0
- package/dist/contracts/builder.js +216 -0
- package/dist/contracts/contract-providers.table.d.ts +40 -0
- package/dist/contracts/contract-providers.table.js +41 -0
- package/dist/contracts/contracts.table.d.ts +44 -0
- package/dist/contracts/contracts.table.js +44 -0
- package/dist/contracts/extract.d.ts +46 -0
- package/dist/contracts/extract.js +51 -0
- package/dist/contracts/index.d.ts +9 -0
- package/dist/contracts/index.js +31 -0
- package/dist/contracts/persistent-registry.d.ts +32 -0
- package/dist/contracts/persistent-registry.js +138 -0
- package/dist/contracts/registry.d.ts +58 -0
- package/dist/contracts/registry.js +155 -0
- package/dist/contracts/types.d.ts +108 -0
- package/dist/contracts/types.js +10 -0
- package/dist/contracts/validate.d.ts +24 -0
- package/dist/contracts/validate.js +274 -0
- package/dist/data/assistants.d.ts +5 -0
- package/dist/data/assistants.js +1 -0
- package/dist/data/package-versions.d.ts +2 -2
- package/dist/data/persistent-vars.d.ts +3 -0
- package/dist/data/persistent-vars.js +3 -0
- package/dist/data/tools.d.ts +5 -2
- package/dist/data/tools.js +1 -1
- package/dist/data/workflows.d.ts +1 -0
- package/dist/index.d.ts +1 -0
- package/dist/index.js +1 -0
- package/dist/package-loader/contract-package-loader.d.ts +23 -0
- package/dist/package-loader/contract-package-loader.js +65 -0
- package/dist/package-loader/index.d.ts +1 -0
- package/dist/package-loader/index.js +1 -0
- package/dist/package-loader/package-loader.d.ts +11 -0
- package/dist/package-loader/package-loader.js +59 -8
- package/dist/rpc-types.d.ts +7 -0
- package/dist/rpc-types.js +4 -0
- package/dist/types/workflow.d.ts +3 -0
- package/dist/types/workflow.js +1 -0
- package/package.json +1 -1
|
@@ -0,0 +1,348 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const zod_1 = require("zod");
|
|
4
|
+
const types_1 = require("../../data/orm/types");
|
|
5
|
+
const field_type_1 = require("../../types/field-type");
|
|
6
|
+
const utils_1 = require("../../utils");
|
|
7
|
+
const builder_1 = require("../builder");
|
|
8
|
+
const extract_1 = require("../extract");
|
|
9
|
+
const registry_1 = require("../registry");
|
|
10
|
+
const types_2 = require("../types");
|
|
11
|
+
beforeAll(() => {
|
|
12
|
+
(0, extract_1.setSchemaToFieldsFn)(types_1.schemaToFields);
|
|
13
|
+
});
|
|
14
|
+
/**
|
|
15
|
+
* End-to-end integration tests that exercise the full flow:
|
|
16
|
+
* definePackage -> register in registry -> check dependencies -> resolve -> swap provider.
|
|
17
|
+
*/
|
|
18
|
+
describe("Package Contracts — integration", () => {
|
|
19
|
+
let registry;
|
|
20
|
+
beforeEach(() => {
|
|
21
|
+
registry = new registry_1.ContractRegistry();
|
|
22
|
+
});
|
|
23
|
+
const TASKS_CONTRACT_ID = (0, utils_1.newid)();
|
|
24
|
+
const PROJECTS_CONTRACT_ID = (0, utils_1.newid)();
|
|
25
|
+
const FANCY_TASKS_CONTRACT_ID = (0, utils_1.newid)();
|
|
26
|
+
function defineTasksPackage() {
|
|
27
|
+
return (0, builder_1.definePackage)((pkg) => {
|
|
28
|
+
pkg.packageId = (0, utils_1.newid)();
|
|
29
|
+
const tasks = pkg.contract(TASKS_CONTRACT_ID, 1, "Tasks");
|
|
30
|
+
tasks.tables = [
|
|
31
|
+
{
|
|
32
|
+
metaData: {
|
|
33
|
+
name: "Tasks",
|
|
34
|
+
description: "Task management table",
|
|
35
|
+
primaryKeyName: "taskId",
|
|
36
|
+
fields: [],
|
|
37
|
+
},
|
|
38
|
+
schema: zod_1.z.object({
|
|
39
|
+
taskId: zod_1.z.string().describe("Primary key"),
|
|
40
|
+
title: zod_1.z.string().describe("Task title"),
|
|
41
|
+
status: zod_1.z.string().describe("Task status"),
|
|
42
|
+
sortOrder: zod_1.z.number().describe("Sort order"),
|
|
43
|
+
}),
|
|
44
|
+
},
|
|
45
|
+
];
|
|
46
|
+
tasks.tools = [
|
|
47
|
+
{
|
|
48
|
+
tool: {
|
|
49
|
+
name: "create-task",
|
|
50
|
+
usageDescription: "Creates a new task",
|
|
51
|
+
inputSchema: {
|
|
52
|
+
fields: [
|
|
53
|
+
{ name: "title", type: field_type_1.FieldType.string, description: "Task title" },
|
|
54
|
+
{
|
|
55
|
+
name: "status",
|
|
56
|
+
type: field_type_1.FieldType.string,
|
|
57
|
+
description: "Initial status",
|
|
58
|
+
optional: true,
|
|
59
|
+
},
|
|
60
|
+
],
|
|
61
|
+
},
|
|
62
|
+
outputSchema: {
|
|
63
|
+
fields: [{ name: "taskId", type: field_type_1.FieldType.id, description: "Created task ID" }],
|
|
64
|
+
},
|
|
65
|
+
},
|
|
66
|
+
},
|
|
67
|
+
];
|
|
68
|
+
tasks.observables = [
|
|
69
|
+
{
|
|
70
|
+
name: "taskCount",
|
|
71
|
+
description: "Total task count",
|
|
72
|
+
valueType: field_type_1.FieldType.number,
|
|
73
|
+
writable: false,
|
|
74
|
+
},
|
|
75
|
+
];
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
function defineProjectsPackage() {
|
|
79
|
+
return (0, builder_1.definePackage)((pkg) => {
|
|
80
|
+
pkg.packageId = (0, utils_1.newid)();
|
|
81
|
+
pkg.consumes(TASKS_CONTRACT_ID, 1);
|
|
82
|
+
const projects = pkg.contract(PROJECTS_CONTRACT_ID, 1, "Projects");
|
|
83
|
+
projects.tables = [
|
|
84
|
+
{
|
|
85
|
+
metaData: {
|
|
86
|
+
name: "Projects",
|
|
87
|
+
description: "Project management",
|
|
88
|
+
primaryKeyName: "projectId",
|
|
89
|
+
fields: [],
|
|
90
|
+
},
|
|
91
|
+
schema: zod_1.z.object({
|
|
92
|
+
projectId: zod_1.z.string().describe("PK"),
|
|
93
|
+
name: zod_1.z.string().describe("Project name"),
|
|
94
|
+
}),
|
|
95
|
+
},
|
|
96
|
+
];
|
|
97
|
+
});
|
|
98
|
+
}
|
|
99
|
+
function defineFancyTasksPackage() {
|
|
100
|
+
return (0, builder_1.definePackage)((pkg) => {
|
|
101
|
+
pkg.packageId = (0, utils_1.newid)();
|
|
102
|
+
const fancy = pkg.contract(FANCY_TASKS_CONTRACT_ID, 1, "Fancy Tasks");
|
|
103
|
+
fancy.alsoImplements(TASKS_CONTRACT_ID, 1);
|
|
104
|
+
fancy.tables = [
|
|
105
|
+
{
|
|
106
|
+
metaData: {
|
|
107
|
+
name: "Tasks",
|
|
108
|
+
description: "Enhanced task table",
|
|
109
|
+
primaryKeyName: "taskId",
|
|
110
|
+
fields: [],
|
|
111
|
+
},
|
|
112
|
+
schema: zod_1.z.object({
|
|
113
|
+
taskId: zod_1.z.string().describe("Primary key"),
|
|
114
|
+
title: zod_1.z.string().describe("Task title"),
|
|
115
|
+
status: zod_1.z.string().describe("Task status"),
|
|
116
|
+
sortOrder: zod_1.z.number().describe("Sort order"),
|
|
117
|
+
priority: zod_1.z.number().optional().describe("Priority level"),
|
|
118
|
+
tags: zod_1.z.string().array().optional().describe("Tags"),
|
|
119
|
+
}),
|
|
120
|
+
},
|
|
121
|
+
];
|
|
122
|
+
fancy.tools = [
|
|
123
|
+
{
|
|
124
|
+
tool: {
|
|
125
|
+
name: "create-task",
|
|
126
|
+
usageDescription: "Creates a fancy task",
|
|
127
|
+
inputSchema: {
|
|
128
|
+
fields: [
|
|
129
|
+
{ name: "title", type: field_type_1.FieldType.string, description: "Task title" },
|
|
130
|
+
{
|
|
131
|
+
name: "status",
|
|
132
|
+
type: field_type_1.FieldType.string,
|
|
133
|
+
description: "Initial status",
|
|
134
|
+
optional: true,
|
|
135
|
+
},
|
|
136
|
+
{
|
|
137
|
+
name: "priority",
|
|
138
|
+
type: field_type_1.FieldType.number,
|
|
139
|
+
description: "Priority",
|
|
140
|
+
optional: true,
|
|
141
|
+
},
|
|
142
|
+
],
|
|
143
|
+
},
|
|
144
|
+
outputSchema: {
|
|
145
|
+
fields: [{ name: "taskId", type: field_type_1.FieldType.id, description: "Created task ID" }],
|
|
146
|
+
},
|
|
147
|
+
},
|
|
148
|
+
},
|
|
149
|
+
];
|
|
150
|
+
fancy.observables = [
|
|
151
|
+
{
|
|
152
|
+
name: "taskCount",
|
|
153
|
+
description: "Total task count",
|
|
154
|
+
valueType: field_type_1.FieldType.number,
|
|
155
|
+
writable: false,
|
|
156
|
+
},
|
|
157
|
+
];
|
|
158
|
+
});
|
|
159
|
+
}
|
|
160
|
+
it("defines a Tasks package and registers it as a provider", () => {
|
|
161
|
+
const tasksPkg = defineTasksPackage();
|
|
162
|
+
expect(tasksPkg.contracts).toHaveLength(1);
|
|
163
|
+
const pkgId = (0, utils_1.newid)();
|
|
164
|
+
const result = registry.register(pkgId, tasksPkg.contracts[0]);
|
|
165
|
+
expect(result.valid).toBe(true);
|
|
166
|
+
const resolved = registry.resolve(TASKS_CONTRACT_ID, 1);
|
|
167
|
+
expect(resolved).toBeDefined();
|
|
168
|
+
expect(resolved?.tables[0].name).toBe("Tasks");
|
|
169
|
+
expect(resolved?.tools[0].name).toBe("create-task");
|
|
170
|
+
expect(resolved?.observables[0].name).toBe("taskCount");
|
|
171
|
+
});
|
|
172
|
+
it("defines a Projects package that consumes Tasks, and verifies dependencies", () => {
|
|
173
|
+
const tasksPkg = defineTasksPackage();
|
|
174
|
+
const tasksPkgId = (0, utils_1.newid)();
|
|
175
|
+
registry.register(tasksPkgId, tasksPkg.contracts[0]);
|
|
176
|
+
const projectsPkg = defineProjectsPackage();
|
|
177
|
+
const projectsPkgId = (0, utils_1.newid)();
|
|
178
|
+
registry.register(projectsPkgId, projectsPkg.contracts[0]);
|
|
179
|
+
// Projects consumes Tasks — check that the dependency is satisfied
|
|
180
|
+
const deps = registry.checkConsumerDependencies(projectsPkg.consumes);
|
|
181
|
+
expect(deps.satisfied).toHaveLength(1);
|
|
182
|
+
expect(deps.satisfied[0].contractId).toBe(TASKS_CONTRACT_ID);
|
|
183
|
+
expect(deps.missing).toHaveLength(0);
|
|
184
|
+
});
|
|
185
|
+
it("rejects Projects installation when Tasks provider is missing", () => {
|
|
186
|
+
const projectsPkg = defineProjectsPackage();
|
|
187
|
+
const projectsPkgId = (0, utils_1.newid)();
|
|
188
|
+
registry.register(projectsPkgId, projectsPkg.contracts[0]);
|
|
189
|
+
const deps = registry.checkConsumerDependencies(projectsPkg.consumes);
|
|
190
|
+
expect(deps.missing).toHaveLength(1);
|
|
191
|
+
expect(deps.missing[0].contractId).toBe(TASKS_CONTRACT_ID);
|
|
192
|
+
});
|
|
193
|
+
it("FancyTasks provides alsoImplements(tasks:v1) and passes validation", () => {
|
|
194
|
+
// Register the original tasks contract first (so alsoImplements can find it)
|
|
195
|
+
const tasksPkg = defineTasksPackage();
|
|
196
|
+
const tasksPkgId = (0, utils_1.newid)();
|
|
197
|
+
registry.register(tasksPkgId, tasksPkg.contracts[0]);
|
|
198
|
+
// Now register the fancy tasks package with alsoImplements
|
|
199
|
+
const fancyPkg = defineFancyTasksPackage();
|
|
200
|
+
const fancyContract = fancyPkg.contracts[0];
|
|
201
|
+
const key = (0, types_2.contractKey)(FANCY_TASKS_CONTRACT_ID, 1);
|
|
202
|
+
const alsoImpl = fancyPkg.alsoImplements.get(key) ?? [];
|
|
203
|
+
const fancyPkgId = (0, utils_1.newid)();
|
|
204
|
+
const result = registry.register(fancyPkgId, fancyContract, alsoImpl);
|
|
205
|
+
expect(result.valid).toBe(true);
|
|
206
|
+
});
|
|
207
|
+
it("swaps Tasks provider from original to FancyTasks", () => {
|
|
208
|
+
// Register both providers for the tasks contract
|
|
209
|
+
const tasksPkg = defineTasksPackage();
|
|
210
|
+
const tasksPkgId = (0, utils_1.newid)();
|
|
211
|
+
const fancyPkgId = (0, utils_1.newid)();
|
|
212
|
+
registry.register(tasksPkgId, tasksPkg.contracts[0]);
|
|
213
|
+
// Register FancyTasks as an additional provider for the TASKS contract
|
|
214
|
+
const fancyPkg = defineFancyTasksPackage();
|
|
215
|
+
const fancyContract = fancyPkg.contracts[0];
|
|
216
|
+
// Register the fancy tasks contract under its own ID
|
|
217
|
+
const key = (0, types_2.contractKey)(FANCY_TASKS_CONTRACT_ID, 1);
|
|
218
|
+
const alsoImpl = fancyPkg.alsoImplements.get(key) ?? [];
|
|
219
|
+
registry.register(fancyPkgId, fancyContract, alsoImpl);
|
|
220
|
+
// Also register it as a provider for the original tasks contract
|
|
221
|
+
registry.register(fancyPkgId, tasksPkg.contracts[0]);
|
|
222
|
+
// Verify original is still active
|
|
223
|
+
expect(registry.getProviderPackageId(TASKS_CONTRACT_ID, 1)).toBe(tasksPkgId);
|
|
224
|
+
// Swap to fancy
|
|
225
|
+
const switched = registry.setActiveProvider(TASKS_CONTRACT_ID, 1, fancyPkgId);
|
|
226
|
+
expect(switched).toBe(true);
|
|
227
|
+
expect(registry.getProviderPackageId(TASKS_CONTRACT_ID, 1)).toBe(fancyPkgId);
|
|
228
|
+
// Projects still resolves — it doesn't care which provider is active
|
|
229
|
+
const projectsPkg = defineProjectsPackage();
|
|
230
|
+
const deps = registry.checkConsumerDependencies(projectsPkg.consumes);
|
|
231
|
+
expect(deps.satisfied).toHaveLength(1);
|
|
232
|
+
expect(deps.missing).toHaveLength(0);
|
|
233
|
+
});
|
|
234
|
+
it("removing the active provider falls back to the alternative", () => {
|
|
235
|
+
const tasksPkg = defineTasksPackage();
|
|
236
|
+
const tasksPkgId = (0, utils_1.newid)();
|
|
237
|
+
const backupPkgId = (0, utils_1.newid)();
|
|
238
|
+
registry.register(tasksPkgId, tasksPkg.contracts[0]);
|
|
239
|
+
registry.register(backupPkgId, tasksPkg.contracts[0]);
|
|
240
|
+
expect(registry.getProviderPackageId(TASKS_CONTRACT_ID, 1)).toBe(tasksPkgId);
|
|
241
|
+
registry.unregister(tasksPkgId);
|
|
242
|
+
expect(registry.getProviderPackageId(TASKS_CONTRACT_ID, 1)).toBe(backupPkgId);
|
|
243
|
+
// Consumer dependency still satisfied
|
|
244
|
+
const projectsPkg = defineProjectsPackage();
|
|
245
|
+
const deps = registry.checkConsumerDependencies(projectsPkg.consumes);
|
|
246
|
+
expect(deps.missing).toHaveLength(0);
|
|
247
|
+
});
|
|
248
|
+
it("removing the only provider makes consumer dependencies unsatisfied", () => {
|
|
249
|
+
const tasksPkg = defineTasksPackage();
|
|
250
|
+
const tasksPkgId = (0, utils_1.newid)();
|
|
251
|
+
registry.register(tasksPkgId, tasksPkg.contracts[0]);
|
|
252
|
+
const projectsPkg = defineProjectsPackage();
|
|
253
|
+
// Satisfied before removal
|
|
254
|
+
let deps = registry.checkConsumerDependencies(projectsPkg.consumes);
|
|
255
|
+
expect(deps.missing).toHaveLength(0);
|
|
256
|
+
registry.unregister(tasksPkgId);
|
|
257
|
+
// Now unsatisfied
|
|
258
|
+
deps = registry.checkConsumerDependencies(projectsPkg.consumes);
|
|
259
|
+
expect(deps.missing).toHaveLength(1);
|
|
260
|
+
});
|
|
261
|
+
it("FancyTasks alsoImplements validation fails when it is missing an observable", () => {
|
|
262
|
+
// Register original tasks with an observable
|
|
263
|
+
const tasksPkg = defineTasksPackage();
|
|
264
|
+
const tasksPkgId = (0, utils_1.newid)();
|
|
265
|
+
registry.register(tasksPkgId, tasksPkg.contracts[0]);
|
|
266
|
+
// Define a broken fancy tasks that is missing the taskCount observable
|
|
267
|
+
const brokenFancy = (0, builder_1.definePackage)((pkg) => {
|
|
268
|
+
pkg.packageId = (0, utils_1.newid)();
|
|
269
|
+
const fancy = pkg.contract(FANCY_TASKS_CONTRACT_ID, 1, "Broken Fancy Tasks", "dev");
|
|
270
|
+
fancy.alsoImplements(TASKS_CONTRACT_ID, 1);
|
|
271
|
+
fancy.tables = [
|
|
272
|
+
{
|
|
273
|
+
metaData: {
|
|
274
|
+
name: "Tasks",
|
|
275
|
+
description: "d",
|
|
276
|
+
primaryKeyName: "taskId",
|
|
277
|
+
fields: [],
|
|
278
|
+
},
|
|
279
|
+
schema: zod_1.z.object({
|
|
280
|
+
taskId: zod_1.z.string().describe("PK"),
|
|
281
|
+
title: zod_1.z.string().describe("Title"),
|
|
282
|
+
status: zod_1.z.string().describe("Status"),
|
|
283
|
+
sortOrder: zod_1.z.number().describe("Sort"),
|
|
284
|
+
}),
|
|
285
|
+
},
|
|
286
|
+
];
|
|
287
|
+
fancy.tools = [
|
|
288
|
+
{
|
|
289
|
+
tool: {
|
|
290
|
+
name: "create-task",
|
|
291
|
+
usageDescription: "Creates a task",
|
|
292
|
+
inputSchema: {
|
|
293
|
+
fields: [
|
|
294
|
+
{ name: "title", type: field_type_1.FieldType.string, description: "Title" },
|
|
295
|
+
{ name: "status", type: field_type_1.FieldType.string, description: "Status", optional: true },
|
|
296
|
+
],
|
|
297
|
+
},
|
|
298
|
+
outputSchema: {
|
|
299
|
+
fields: [{ name: "taskId", type: field_type_1.FieldType.id, description: "ID" }],
|
|
300
|
+
},
|
|
301
|
+
},
|
|
302
|
+
},
|
|
303
|
+
];
|
|
304
|
+
// Deliberately omit observables
|
|
305
|
+
});
|
|
306
|
+
const brokenContract = brokenFancy.contracts[0];
|
|
307
|
+
const key = (0, types_2.contractKey)(FANCY_TASKS_CONTRACT_ID, 1);
|
|
308
|
+
const alsoImpl = brokenFancy.alsoImplements.get(key) ?? [];
|
|
309
|
+
const brokenPkgId = (0, utils_1.newid)();
|
|
310
|
+
const result = registry.register(brokenPkgId, brokenContract, alsoImpl);
|
|
311
|
+
expect(result.valid).toBe(false);
|
|
312
|
+
expect(result.errors.some((e) => e.message.includes("taskCount"))).toBe(true);
|
|
313
|
+
});
|
|
314
|
+
it("full lifecycle: define, register, consume, upgrade, swap, remove", () => {
|
|
315
|
+
const tasksPkgId = (0, utils_1.newid)();
|
|
316
|
+
const projectsPkgId = (0, utils_1.newid)();
|
|
317
|
+
const fancyPkgId = (0, utils_1.newid)();
|
|
318
|
+
// 1. Register Tasks v1
|
|
319
|
+
const tasksPkg = defineTasksPackage();
|
|
320
|
+
registry.register(tasksPkgId, tasksPkg.contracts[0]);
|
|
321
|
+
// 2. Register Projects (consumes tasks:v1)
|
|
322
|
+
const projectsPkg = defineProjectsPackage();
|
|
323
|
+
registry.register(projectsPkgId, projectsPkg.contracts[0]);
|
|
324
|
+
let deps = registry.checkConsumerDependencies(projectsPkg.consumes);
|
|
325
|
+
expect(deps.missing).toHaveLength(0);
|
|
326
|
+
// 3. Register FancyTasks (provides its own contract + alsoImplements tasks:v1)
|
|
327
|
+
const fancyPkg = defineFancyTasksPackage();
|
|
328
|
+
const fancyKey = (0, types_2.contractKey)(FANCY_TASKS_CONTRACT_ID, 1);
|
|
329
|
+
registry.register(fancyPkgId, fancyPkg.contracts[0], fancyPkg.alsoImplements.get(fancyKey) ?? []);
|
|
330
|
+
// 4. Also register FancyTasks as a provider of the original tasks contract
|
|
331
|
+
registry.register(fancyPkgId, tasksPkg.contracts[0]);
|
|
332
|
+
// 5. Swap tasks:v1 active provider to FancyTasks
|
|
333
|
+
registry.setActiveProvider(TASKS_CONTRACT_ID, 1, fancyPkgId);
|
|
334
|
+
expect(registry.getProviderPackageId(TASKS_CONTRACT_ID, 1)).toBe(fancyPkgId);
|
|
335
|
+
// 6. Projects' dependencies still satisfied
|
|
336
|
+
deps = registry.checkConsumerDependencies(projectsPkg.consumes);
|
|
337
|
+
expect(deps.missing).toHaveLength(0);
|
|
338
|
+
// 7. Remove original tasks package — fancy is still providing
|
|
339
|
+
registry.unregister(tasksPkgId);
|
|
340
|
+
deps = registry.checkConsumerDependencies(projectsPkg.consumes);
|
|
341
|
+
expect(deps.missing).toHaveLength(0);
|
|
342
|
+
expect(registry.getProviderPackageId(TASKS_CONTRACT_ID, 1)).toBe(fancyPkgId);
|
|
343
|
+
// 8. Remove fancy — now projects' dependency is unsatisfied
|
|
344
|
+
registry.unregister(fancyPkgId);
|
|
345
|
+
deps = registry.checkConsumerDependencies(projectsPkg.consumes);
|
|
346
|
+
expect(deps.missing).toHaveLength(1);
|
|
347
|
+
});
|
|
348
|
+
});
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,324 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const field_type_1 = require("../../types/field-type");
|
|
4
|
+
const utils_1 = require("../../utils");
|
|
5
|
+
const registry_1 = require("../registry");
|
|
6
|
+
const DEFAULT_CONTRACT_ID = (0, utils_1.newid)();
|
|
7
|
+
function makeContract(overrides = {}) {
|
|
8
|
+
return {
|
|
9
|
+
contractId: DEFAULT_CONTRACT_ID,
|
|
10
|
+
version: 1,
|
|
11
|
+
name: "Test Contract",
|
|
12
|
+
description: "A test contract",
|
|
13
|
+
tables: [],
|
|
14
|
+
tools: [],
|
|
15
|
+
observables: [],
|
|
16
|
+
...overrides,
|
|
17
|
+
};
|
|
18
|
+
}
|
|
19
|
+
describe("ContractRegistry", () => {
|
|
20
|
+
let registry;
|
|
21
|
+
beforeEach(() => {
|
|
22
|
+
registry = new registry_1.ContractRegistry();
|
|
23
|
+
});
|
|
24
|
+
describe("register and resolve", () => {
|
|
25
|
+
it("registers a provider and resolves it", () => {
|
|
26
|
+
const def = makeContract({
|
|
27
|
+
tables: [
|
|
28
|
+
{
|
|
29
|
+
name: "Tasks",
|
|
30
|
+
description: "d",
|
|
31
|
+
primaryKeyName: "taskId",
|
|
32
|
+
fields: [{ name: "taskId", type: field_type_1.FieldType.id }],
|
|
33
|
+
},
|
|
34
|
+
],
|
|
35
|
+
});
|
|
36
|
+
const pkgId = (0, utils_1.newid)();
|
|
37
|
+
const result = registry.register(pkgId, def);
|
|
38
|
+
expect(result.valid).toBe(true);
|
|
39
|
+
const resolved = registry.resolve(DEFAULT_CONTRACT_ID, 1);
|
|
40
|
+
expect(resolved).toBeDefined();
|
|
41
|
+
expect(resolved?.tables[0].name).toBe("Tasks");
|
|
42
|
+
});
|
|
43
|
+
it("returns the provider package ID", () => {
|
|
44
|
+
const def = makeContract();
|
|
45
|
+
const pkgId = (0, utils_1.newid)();
|
|
46
|
+
registry.register(pkgId, def);
|
|
47
|
+
expect(registry.getProviderPackageId(DEFAULT_CONTRACT_ID, 1)).toBe(pkgId);
|
|
48
|
+
});
|
|
49
|
+
it("returns undefined for unregistered contracts", () => {
|
|
50
|
+
const unknownId = (0, utils_1.newid)();
|
|
51
|
+
expect(registry.resolve(unknownId, 1)).toBeUndefined();
|
|
52
|
+
expect(registry.getProviderPackageId(unknownId, 1)).toBeUndefined();
|
|
53
|
+
});
|
|
54
|
+
it("auto-activates the first registered provider", () => {
|
|
55
|
+
const def = makeContract();
|
|
56
|
+
const pkgA = (0, utils_1.newid)();
|
|
57
|
+
registry.register(pkgA, def);
|
|
58
|
+
expect(registry.getProviderPackageId(DEFAULT_CONTRACT_ID, 1)).toBe(pkgA);
|
|
59
|
+
});
|
|
60
|
+
it("does not auto-switch when a second provider is registered", () => {
|
|
61
|
+
const def = makeContract();
|
|
62
|
+
const pkgA = (0, utils_1.newid)();
|
|
63
|
+
const pkgB = (0, utils_1.newid)();
|
|
64
|
+
registry.register(pkgA, def);
|
|
65
|
+
registry.register(pkgB, def);
|
|
66
|
+
expect(registry.getProviderPackageId(DEFAULT_CONTRACT_ID, 1)).toBe(pkgA);
|
|
67
|
+
});
|
|
68
|
+
});
|
|
69
|
+
describe("setActiveProvider", () => {
|
|
70
|
+
it("switches the active provider", () => {
|
|
71
|
+
const def = makeContract();
|
|
72
|
+
const pkgA = (0, utils_1.newid)();
|
|
73
|
+
const pkgB = (0, utils_1.newid)();
|
|
74
|
+
registry.register(pkgA, def);
|
|
75
|
+
registry.register(pkgB, def);
|
|
76
|
+
const switched = registry.setActiveProvider(DEFAULT_CONTRACT_ID, 1, pkgB);
|
|
77
|
+
expect(switched).toBe(true);
|
|
78
|
+
expect(registry.getProviderPackageId(DEFAULT_CONTRACT_ID, 1)).toBe(pkgB);
|
|
79
|
+
});
|
|
80
|
+
it("returns false if the target package is not registered", () => {
|
|
81
|
+
const def = makeContract();
|
|
82
|
+
const pkgA = (0, utils_1.newid)();
|
|
83
|
+
const unknownPkg = (0, utils_1.newid)();
|
|
84
|
+
registry.register(pkgA, def);
|
|
85
|
+
const switched = registry.setActiveProvider(DEFAULT_CONTRACT_ID, 1, unknownPkg);
|
|
86
|
+
expect(switched).toBe(false);
|
|
87
|
+
expect(registry.getProviderPackageId(DEFAULT_CONTRACT_ID, 1)).toBe(pkgA);
|
|
88
|
+
});
|
|
89
|
+
it("returns false for unknown contracts", () => {
|
|
90
|
+
const unknownId = (0, utils_1.newid)();
|
|
91
|
+
const pkgA = (0, utils_1.newid)();
|
|
92
|
+
expect(registry.setActiveProvider(unknownId, 1, pkgA)).toBe(false);
|
|
93
|
+
});
|
|
94
|
+
});
|
|
95
|
+
describe("unregister", () => {
|
|
96
|
+
it("removes a provider and its resolution", () => {
|
|
97
|
+
const def = makeContract();
|
|
98
|
+
const pkgA = (0, utils_1.newid)();
|
|
99
|
+
registry.register(pkgA, def);
|
|
100
|
+
registry.unregister(pkgA);
|
|
101
|
+
expect(registry.resolve(DEFAULT_CONTRACT_ID, 1)).toBeUndefined();
|
|
102
|
+
expect(registry.getProviderPackageId(DEFAULT_CONTRACT_ID, 1)).toBeUndefined();
|
|
103
|
+
});
|
|
104
|
+
it("falls back to next provider when active is removed", () => {
|
|
105
|
+
const def = makeContract();
|
|
106
|
+
const pkgA = (0, utils_1.newid)();
|
|
107
|
+
const pkgB = (0, utils_1.newid)();
|
|
108
|
+
registry.register(pkgA, def);
|
|
109
|
+
registry.register(pkgB, def);
|
|
110
|
+
registry.unregister(pkgA);
|
|
111
|
+
expect(registry.getProviderPackageId(DEFAULT_CONTRACT_ID, 1)).toBe(pkgB);
|
|
112
|
+
});
|
|
113
|
+
it("is a no-op for unknown packages", () => {
|
|
114
|
+
const unknownPkg = (0, utils_1.newid)();
|
|
115
|
+
expect(() => registry.unregister(unknownPkg)).not.toThrow();
|
|
116
|
+
});
|
|
117
|
+
it("handles a package that provides multiple contracts", () => {
|
|
118
|
+
const contractId1 = (0, utils_1.newid)();
|
|
119
|
+
const contractId2 = (0, utils_1.newid)();
|
|
120
|
+
const def1 = makeContract({ contractId: contractId1 });
|
|
121
|
+
const def2 = makeContract({ contractId: contractId2 });
|
|
122
|
+
const pkgA = (0, utils_1.newid)();
|
|
123
|
+
registry.register(pkgA, def1);
|
|
124
|
+
registry.register(pkgA, def2);
|
|
125
|
+
registry.unregister(pkgA);
|
|
126
|
+
expect(registry.resolve(contractId1, 1)).toBeUndefined();
|
|
127
|
+
expect(registry.resolve(contractId2, 1)).toBeUndefined();
|
|
128
|
+
});
|
|
129
|
+
});
|
|
130
|
+
describe("checkConsumerDependencies", () => {
|
|
131
|
+
it("reports all satisfied when providers exist", () => {
|
|
132
|
+
const tasksId = (0, utils_1.newid)();
|
|
133
|
+
const authId = (0, utils_1.newid)();
|
|
134
|
+
const pkgA = (0, utils_1.newid)();
|
|
135
|
+
const pkgB = (0, utils_1.newid)();
|
|
136
|
+
registry.register(pkgA, makeContract({ contractId: tasksId }));
|
|
137
|
+
registry.register(pkgB, makeContract({ contractId: authId }));
|
|
138
|
+
const result = registry.checkConsumerDependencies([
|
|
139
|
+
{ contractId: tasksId, version: 1 },
|
|
140
|
+
{ contractId: authId, version: 1 },
|
|
141
|
+
]);
|
|
142
|
+
expect(result.satisfied).toHaveLength(2);
|
|
143
|
+
expect(result.missing).toHaveLength(0);
|
|
144
|
+
});
|
|
145
|
+
it("reports missing when no provider exists", () => {
|
|
146
|
+
const tasksId = (0, utils_1.newid)();
|
|
147
|
+
const authId = (0, utils_1.newid)();
|
|
148
|
+
const pkgA = (0, utils_1.newid)();
|
|
149
|
+
registry.register(pkgA, makeContract({ contractId: tasksId }));
|
|
150
|
+
const result = registry.checkConsumerDependencies([
|
|
151
|
+
{ contractId: tasksId, version: 1 },
|
|
152
|
+
{ contractId: authId, version: 1 },
|
|
153
|
+
]);
|
|
154
|
+
expect(result.satisfied).toHaveLength(1);
|
|
155
|
+
expect(result.missing).toHaveLength(1);
|
|
156
|
+
expect(result.missing[0].contractId).toBe(authId);
|
|
157
|
+
});
|
|
158
|
+
it("reports all missing when registry is empty", () => {
|
|
159
|
+
const tasksId = (0, utils_1.newid)();
|
|
160
|
+
const result = registry.checkConsumerDependencies([{ contractId: tasksId, version: 1 }]);
|
|
161
|
+
expect(result.satisfied).toHaveLength(0);
|
|
162
|
+
expect(result.missing).toHaveLength(1);
|
|
163
|
+
});
|
|
164
|
+
it("returns empty lists for empty consumer list", () => {
|
|
165
|
+
const result = registry.checkConsumerDependencies([]);
|
|
166
|
+
expect(result.satisfied).toHaveLength(0);
|
|
167
|
+
expect(result.missing).toHaveLength(0);
|
|
168
|
+
});
|
|
169
|
+
});
|
|
170
|
+
describe("immutability enforcement", () => {
|
|
171
|
+
it("rejects re-registration of a frozen contract with a different shape", () => {
|
|
172
|
+
const def1 = makeContract({
|
|
173
|
+
tables: [
|
|
174
|
+
{
|
|
175
|
+
name: "T",
|
|
176
|
+
description: "d",
|
|
177
|
+
primaryKeyName: "id",
|
|
178
|
+
fields: [{ name: "id", type: field_type_1.FieldType.id }],
|
|
179
|
+
},
|
|
180
|
+
],
|
|
181
|
+
});
|
|
182
|
+
const pkgA = (0, utils_1.newid)();
|
|
183
|
+
registry.register(pkgA, def1);
|
|
184
|
+
const def2 = makeContract({
|
|
185
|
+
tables: [
|
|
186
|
+
{
|
|
187
|
+
name: "T",
|
|
188
|
+
description: "d",
|
|
189
|
+
primaryKeyName: "id",
|
|
190
|
+
fields: [
|
|
191
|
+
{ name: "id", type: field_type_1.FieldType.id },
|
|
192
|
+
{ name: "extra", type: field_type_1.FieldType.string },
|
|
193
|
+
],
|
|
194
|
+
},
|
|
195
|
+
],
|
|
196
|
+
});
|
|
197
|
+
const pkgB = (0, utils_1.newid)();
|
|
198
|
+
const result = registry.register(pkgB, def2);
|
|
199
|
+
expect(result.valid).toBe(false);
|
|
200
|
+
expect(result.errors[0].message).toContain("frozen");
|
|
201
|
+
});
|
|
202
|
+
it("allows re-registration of a dev contract with a different shape", () => {
|
|
203
|
+
const def1 = makeContract({
|
|
204
|
+
devTag: "dev",
|
|
205
|
+
tables: [
|
|
206
|
+
{
|
|
207
|
+
name: "T",
|
|
208
|
+
description: "d",
|
|
209
|
+
primaryKeyName: "id",
|
|
210
|
+
fields: [{ name: "id", type: field_type_1.FieldType.id }],
|
|
211
|
+
},
|
|
212
|
+
],
|
|
213
|
+
});
|
|
214
|
+
const pkgA = (0, utils_1.newid)();
|
|
215
|
+
registry.register(pkgA, def1);
|
|
216
|
+
const def2 = makeContract({
|
|
217
|
+
devTag: "dev",
|
|
218
|
+
tables: [
|
|
219
|
+
{
|
|
220
|
+
name: "T",
|
|
221
|
+
description: "d",
|
|
222
|
+
primaryKeyName: "id",
|
|
223
|
+
fields: [
|
|
224
|
+
{ name: "id", type: field_type_1.FieldType.id },
|
|
225
|
+
{ name: "extra", type: field_type_1.FieldType.string },
|
|
226
|
+
],
|
|
227
|
+
},
|
|
228
|
+
],
|
|
229
|
+
});
|
|
230
|
+
const pkgB = (0, utils_1.newid)();
|
|
231
|
+
const result = registry.register(pkgB, def2);
|
|
232
|
+
expect(result.valid).toBe(true);
|
|
233
|
+
});
|
|
234
|
+
it("rejects dev registration after a stable version exists", () => {
|
|
235
|
+
const stableDef = makeContract({
|
|
236
|
+
tables: [
|
|
237
|
+
{
|
|
238
|
+
name: "T",
|
|
239
|
+
description: "d",
|
|
240
|
+
primaryKeyName: "id",
|
|
241
|
+
fields: [{ name: "id", type: field_type_1.FieldType.id }],
|
|
242
|
+
},
|
|
243
|
+
],
|
|
244
|
+
});
|
|
245
|
+
const pkgA = (0, utils_1.newid)();
|
|
246
|
+
registry.register(pkgA, stableDef);
|
|
247
|
+
const devDef = makeContract({
|
|
248
|
+
devTag: "dev",
|
|
249
|
+
tables: [
|
|
250
|
+
{
|
|
251
|
+
name: "T",
|
|
252
|
+
description: "d",
|
|
253
|
+
primaryKeyName: "id",
|
|
254
|
+
fields: [{ name: "id", type: field_type_1.FieldType.id }],
|
|
255
|
+
},
|
|
256
|
+
],
|
|
257
|
+
});
|
|
258
|
+
const pkgB = (0, utils_1.newid)();
|
|
259
|
+
const result = registry.register(pkgB, devDef);
|
|
260
|
+
expect(result.valid).toBe(false);
|
|
261
|
+
expect(result.errors[0].message).toContain("frozen");
|
|
262
|
+
});
|
|
263
|
+
it("allows dev-to-stable transition, then rejects subsequent dev registration", () => {
|
|
264
|
+
const devDef = makeContract({
|
|
265
|
+
devTag: "dev",
|
|
266
|
+
tables: [
|
|
267
|
+
{
|
|
268
|
+
name: "T",
|
|
269
|
+
description: "d",
|
|
270
|
+
primaryKeyName: "id",
|
|
271
|
+
fields: [{ name: "id", type: field_type_1.FieldType.id }],
|
|
272
|
+
},
|
|
273
|
+
],
|
|
274
|
+
});
|
|
275
|
+
const pkgA = (0, utils_1.newid)();
|
|
276
|
+
registry.register(pkgA, devDef);
|
|
277
|
+
const stableDef = makeContract({
|
|
278
|
+
tables: [
|
|
279
|
+
{
|
|
280
|
+
name: "T",
|
|
281
|
+
description: "d",
|
|
282
|
+
primaryKeyName: "id",
|
|
283
|
+
fields: [
|
|
284
|
+
{ name: "id", type: field_type_1.FieldType.id },
|
|
285
|
+
{ name: "title", type: field_type_1.FieldType.string },
|
|
286
|
+
],
|
|
287
|
+
},
|
|
288
|
+
],
|
|
289
|
+
});
|
|
290
|
+
const pkgB = (0, utils_1.newid)();
|
|
291
|
+
const stableResult = registry.register(pkgB, stableDef);
|
|
292
|
+
expect(stableResult.valid).toBe(true);
|
|
293
|
+
const devDef2 = makeContract({
|
|
294
|
+
devTag: "dev",
|
|
295
|
+
tables: [
|
|
296
|
+
{
|
|
297
|
+
name: "T",
|
|
298
|
+
description: "d",
|
|
299
|
+
primaryKeyName: "id",
|
|
300
|
+
fields: [{ name: "id", type: field_type_1.FieldType.id }],
|
|
301
|
+
},
|
|
302
|
+
],
|
|
303
|
+
});
|
|
304
|
+
const pkgC = (0, utils_1.newid)();
|
|
305
|
+
const devResult = registry.register(pkgC, devDef2);
|
|
306
|
+
expect(devResult.valid).toBe(false);
|
|
307
|
+
expect(devResult.errors[0].message).toContain("frozen");
|
|
308
|
+
});
|
|
309
|
+
});
|
|
310
|
+
describe("getDefinition", () => {
|
|
311
|
+
it("returns the stored definition", () => {
|
|
312
|
+
const def = makeContract({ name: "My Contract" });
|
|
313
|
+
const pkgA = (0, utils_1.newid)();
|
|
314
|
+
registry.register(pkgA, def);
|
|
315
|
+
const stored = registry.getDefinition(DEFAULT_CONTRACT_ID, 1);
|
|
316
|
+
expect(stored).toBeDefined();
|
|
317
|
+
expect(stored?.name).toBe("My Contract");
|
|
318
|
+
});
|
|
319
|
+
it("returns undefined for unknown contracts", () => {
|
|
320
|
+
const unknownId = (0, utils_1.newid)();
|
|
321
|
+
expect(registry.getDefinition(unknownId, 1)).toBeUndefined();
|
|
322
|
+
});
|
|
323
|
+
});
|
|
324
|
+
});
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|