@h-rig/rig-extension 0.0.6-alpha.91
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +1 -0
- package/dist/src/board-theme.js +150 -0
- package/dist/src/board-views.js +475 -0
- package/dist/src/extension.js +3743 -0
- package/dist/src/workflow/identity.js +429 -0
- package/dist/src/workflow/projections.js +389 -0
- package/package.json +36 -0
|
@@ -0,0 +1,389 @@
|
|
|
1
|
+
// @bun
|
|
2
|
+
// packages/rig-extension/src/workflow/projections.ts
|
|
3
|
+
var RIG_WORKFLOW_STARTED = "rig.workflow.started";
|
|
4
|
+
var RIG_WORKFLOW_TARGET_SELECTED = "rig.workflow.target.selected";
|
|
5
|
+
var RIG_WORKFLOW_TASK_SELECTED = "rig.workflow.task.selected";
|
|
6
|
+
var RIG_WORKFLOW_STATUS_CHANGED = "rig.workflow.status.changed";
|
|
7
|
+
var RIG_WORKFLOW_OPERATOR_NOTE = "rig.workflow.operator.note";
|
|
8
|
+
var RIG_WORKFLOW_INBOX_REQUESTED = "rig.workflow.inbox.requested";
|
|
9
|
+
var RIG_WORKFLOW_INBOX_RESOLVED = "rig.workflow.inbox.resolved";
|
|
10
|
+
var EMPTY_PROJECTION = {
|
|
11
|
+
started: null,
|
|
12
|
+
target: null,
|
|
13
|
+
task: null,
|
|
14
|
+
status: null,
|
|
15
|
+
notes: [],
|
|
16
|
+
inbox: [],
|
|
17
|
+
resolvedInbox: [],
|
|
18
|
+
updatedAt: null
|
|
19
|
+
};
|
|
20
|
+
function asRecord(value) {
|
|
21
|
+
return value && typeof value === "object" && !Array.isArray(value) ? value : null;
|
|
22
|
+
}
|
|
23
|
+
function optionalString(value) {
|
|
24
|
+
return typeof value === "string" && value.trim().length > 0 ? value.trim() : undefined;
|
|
25
|
+
}
|
|
26
|
+
function requiredString(value) {
|
|
27
|
+
const text = optionalString(value);
|
|
28
|
+
return text ?? null;
|
|
29
|
+
}
|
|
30
|
+
function timestamp(value) {
|
|
31
|
+
const text = requiredString(value);
|
|
32
|
+
if (!text)
|
|
33
|
+
return null;
|
|
34
|
+
const millis = Date.parse(text);
|
|
35
|
+
return Number.isFinite(millis) ? text : null;
|
|
36
|
+
}
|
|
37
|
+
function newer(left, right) {
|
|
38
|
+
if (!right)
|
|
39
|
+
return false;
|
|
40
|
+
if (!left)
|
|
41
|
+
return true;
|
|
42
|
+
return Date.parse(right) >= Date.parse(left);
|
|
43
|
+
}
|
|
44
|
+
function isWorkflowTarget(value) {
|
|
45
|
+
return value === "local" || value === "remote";
|
|
46
|
+
}
|
|
47
|
+
function isWorkflowStatus(value) {
|
|
48
|
+
return value === "starting" || value === "running" || value === "waiting-approval" || value === "waiting-input" || value === "completed" || value === "failed" || value === "stopped";
|
|
49
|
+
}
|
|
50
|
+
function isInboxKind(value) {
|
|
51
|
+
return value === "approval" || value === "input";
|
|
52
|
+
}
|
|
53
|
+
function isInboxDecision(value) {
|
|
54
|
+
return value === "approved" || value === "rejected" || value === "answered";
|
|
55
|
+
}
|
|
56
|
+
function parseOwner(value) {
|
|
57
|
+
const record = asRecord(value);
|
|
58
|
+
if (!record)
|
|
59
|
+
return null;
|
|
60
|
+
const githubUserId = requiredString(record.githubUserId);
|
|
61
|
+
const login = requiredString(record.login);
|
|
62
|
+
const namespaceKey = requiredString(record.namespaceKey);
|
|
63
|
+
return githubUserId && login && namespaceKey ? { githubUserId, login, namespaceKey } : null;
|
|
64
|
+
}
|
|
65
|
+
function parseStringArray(value) {
|
|
66
|
+
if (!Array.isArray(value))
|
|
67
|
+
return;
|
|
68
|
+
const options = value.map(optionalString).filter((option) => option !== undefined);
|
|
69
|
+
return options.length > 0 ? options : undefined;
|
|
70
|
+
}
|
|
71
|
+
function hasOwn(record, key) {
|
|
72
|
+
return Object.prototype.hasOwnProperty.call(record, key);
|
|
73
|
+
}
|
|
74
|
+
function entryUpdatedAt(entry) {
|
|
75
|
+
if ("createdAt" in entry)
|
|
76
|
+
return entry.createdAt;
|
|
77
|
+
if ("selectedAt" in entry)
|
|
78
|
+
return entry.selectedAt;
|
|
79
|
+
if ("changedAt" in entry)
|
|
80
|
+
return entry.changedAt;
|
|
81
|
+
if ("notedAt" in entry)
|
|
82
|
+
return entry.notedAt;
|
|
83
|
+
if ("requestedAt" in entry)
|
|
84
|
+
return entry.requestedAt;
|
|
85
|
+
return entry.resolvedAt;
|
|
86
|
+
}
|
|
87
|
+
function createWorkflowStarted(input) {
|
|
88
|
+
return {
|
|
89
|
+
schemaVersion: 1,
|
|
90
|
+
workflowId: input.workflowId,
|
|
91
|
+
target: input.target,
|
|
92
|
+
selectedRepo: input.selectedRepo,
|
|
93
|
+
owner: input.owner,
|
|
94
|
+
createdAt: input.createdAt ?? new Date().toISOString()
|
|
95
|
+
};
|
|
96
|
+
}
|
|
97
|
+
function createWorkflowTargetSelected(input) {
|
|
98
|
+
return {
|
|
99
|
+
schemaVersion: 1,
|
|
100
|
+
target: input.target,
|
|
101
|
+
...input.reason ? { reason: input.reason } : {},
|
|
102
|
+
selectedAt: input.selectedAt ?? new Date().toISOString()
|
|
103
|
+
};
|
|
104
|
+
}
|
|
105
|
+
function createWorkflowTaskSelected(input) {
|
|
106
|
+
return {
|
|
107
|
+
schemaVersion: 1,
|
|
108
|
+
taskId: input.taskId,
|
|
109
|
+
...input.title ? { title: input.title } : {},
|
|
110
|
+
selectedAt: input.selectedAt ?? new Date().toISOString()
|
|
111
|
+
};
|
|
112
|
+
}
|
|
113
|
+
function createWorkflowStatusChanged(input) {
|
|
114
|
+
return {
|
|
115
|
+
schemaVersion: 1,
|
|
116
|
+
status: input.status,
|
|
117
|
+
...input.detail ? { detail: input.detail } : {},
|
|
118
|
+
changedAt: input.changedAt ?? new Date().toISOString()
|
|
119
|
+
};
|
|
120
|
+
}
|
|
121
|
+
function createWorkflowOperatorNote(input) {
|
|
122
|
+
return {
|
|
123
|
+
schemaVersion: 1,
|
|
124
|
+
...input.noteId ? { noteId: input.noteId } : {},
|
|
125
|
+
note: input.note,
|
|
126
|
+
notedAt: input.notedAt ?? new Date().toISOString()
|
|
127
|
+
};
|
|
128
|
+
}
|
|
129
|
+
function createWorkflowInboxRequested(input) {
|
|
130
|
+
return {
|
|
131
|
+
schemaVersion: 1,
|
|
132
|
+
requestId: input.requestId,
|
|
133
|
+
kind: input.kind,
|
|
134
|
+
title: input.title,
|
|
135
|
+
...input.body ? { body: input.body } : {},
|
|
136
|
+
...input.options && input.options.length > 0 ? { options: input.options } : {},
|
|
137
|
+
requestedAt: input.requestedAt ?? new Date().toISOString()
|
|
138
|
+
};
|
|
139
|
+
}
|
|
140
|
+
function createWorkflowInboxResolved(input) {
|
|
141
|
+
const base = {
|
|
142
|
+
schemaVersion: 1,
|
|
143
|
+
requestId: input.requestId,
|
|
144
|
+
decision: input.decision,
|
|
145
|
+
resolvedAt: input.resolvedAt ?? new Date().toISOString()
|
|
146
|
+
};
|
|
147
|
+
return hasOwn(input, "answer") ? { ...base, answer: input.answer } : base;
|
|
148
|
+
}
|
|
149
|
+
function parseWorkflowStarted(data) {
|
|
150
|
+
const record = asRecord(data);
|
|
151
|
+
if (!record || record.schemaVersion !== 1)
|
|
152
|
+
return null;
|
|
153
|
+
const workflowId = requiredString(record.workflowId);
|
|
154
|
+
const target = optionalString(record.target);
|
|
155
|
+
const selectedRepo = requiredString(record.selectedRepo);
|
|
156
|
+
const owner = parseOwner(record.owner);
|
|
157
|
+
const createdAt = timestamp(record.createdAt);
|
|
158
|
+
if (!workflowId || !isWorkflowTarget(target) || !selectedRepo || !owner || !createdAt)
|
|
159
|
+
return null;
|
|
160
|
+
return {
|
|
161
|
+
schemaVersion: 1,
|
|
162
|
+
workflowId,
|
|
163
|
+
target,
|
|
164
|
+
selectedRepo,
|
|
165
|
+
owner,
|
|
166
|
+
createdAt
|
|
167
|
+
};
|
|
168
|
+
}
|
|
169
|
+
function parseWorkflowTargetSelected(data) {
|
|
170
|
+
const record = asRecord(data);
|
|
171
|
+
if (!record || record.schemaVersion !== 1)
|
|
172
|
+
return null;
|
|
173
|
+
const target = optionalString(record.target);
|
|
174
|
+
const selectedAt = timestamp(record.selectedAt);
|
|
175
|
+
if (!isWorkflowTarget(target) || !selectedAt)
|
|
176
|
+
return null;
|
|
177
|
+
const reason = optionalString(record.reason);
|
|
178
|
+
return {
|
|
179
|
+
schemaVersion: 1,
|
|
180
|
+
target,
|
|
181
|
+
...reason ? { reason } : {},
|
|
182
|
+
selectedAt
|
|
183
|
+
};
|
|
184
|
+
}
|
|
185
|
+
function parseWorkflowTaskSelected(data) {
|
|
186
|
+
const record = asRecord(data);
|
|
187
|
+
if (!record || record.schemaVersion !== 1)
|
|
188
|
+
return null;
|
|
189
|
+
const taskId = requiredString(record.taskId);
|
|
190
|
+
const selectedAt = timestamp(record.selectedAt);
|
|
191
|
+
if (!taskId || !selectedAt)
|
|
192
|
+
return null;
|
|
193
|
+
const title = optionalString(record.title);
|
|
194
|
+
return { schemaVersion: 1, taskId, ...title ? { title } : {}, selectedAt };
|
|
195
|
+
}
|
|
196
|
+
function parseWorkflowStatusChanged(data) {
|
|
197
|
+
const record = asRecord(data);
|
|
198
|
+
if (!record || record.schemaVersion !== 1)
|
|
199
|
+
return null;
|
|
200
|
+
const status = optionalString(record.status);
|
|
201
|
+
const changedAt = timestamp(record.changedAt);
|
|
202
|
+
if (!changedAt || !isWorkflowStatus(status))
|
|
203
|
+
return null;
|
|
204
|
+
const detail = optionalString(record.detail);
|
|
205
|
+
return { schemaVersion: 1, status, ...detail ? { detail } : {}, changedAt };
|
|
206
|
+
}
|
|
207
|
+
function parseWorkflowOperatorNote(data) {
|
|
208
|
+
const record = asRecord(data);
|
|
209
|
+
if (!record || record.schemaVersion !== 1)
|
|
210
|
+
return null;
|
|
211
|
+
const note = requiredString(record.note);
|
|
212
|
+
const notedAt = timestamp(record.notedAt);
|
|
213
|
+
if (!note || !notedAt)
|
|
214
|
+
return null;
|
|
215
|
+
const noteId = optionalString(record.noteId);
|
|
216
|
+
return { schemaVersion: 1, ...noteId ? { noteId } : {}, note, notedAt };
|
|
217
|
+
}
|
|
218
|
+
function parseWorkflowInboxRequested(data) {
|
|
219
|
+
const record = asRecord(data);
|
|
220
|
+
if (!record || record.schemaVersion !== 1)
|
|
221
|
+
return null;
|
|
222
|
+
const requestId = requiredString(record.requestId);
|
|
223
|
+
const kind = optionalString(record.kind);
|
|
224
|
+
const title = requiredString(record.title);
|
|
225
|
+
const requestedAt = timestamp(record.requestedAt);
|
|
226
|
+
if (!requestId || !isInboxKind(kind) || !title || !requestedAt)
|
|
227
|
+
return null;
|
|
228
|
+
const body = optionalString(record.body);
|
|
229
|
+
const options = parseStringArray(record.options);
|
|
230
|
+
return {
|
|
231
|
+
schemaVersion: 1,
|
|
232
|
+
requestId,
|
|
233
|
+
kind,
|
|
234
|
+
title,
|
|
235
|
+
...body ? { body } : {},
|
|
236
|
+
...options ? { options } : {},
|
|
237
|
+
requestedAt
|
|
238
|
+
};
|
|
239
|
+
}
|
|
240
|
+
function parseWorkflowInboxResolved(data) {
|
|
241
|
+
const record = asRecord(data);
|
|
242
|
+
if (!record || record.schemaVersion !== 1)
|
|
243
|
+
return null;
|
|
244
|
+
const requestId = requiredString(record.requestId);
|
|
245
|
+
const decision = optionalString(record.decision);
|
|
246
|
+
const resolvedAt = timestamp(record.resolvedAt);
|
|
247
|
+
if (!requestId || !isInboxDecision(decision) || !resolvedAt)
|
|
248
|
+
return null;
|
|
249
|
+
const base = { schemaVersion: 1, requestId, decision, resolvedAt };
|
|
250
|
+
return hasOwn(record, "answer") ? { ...base, answer: record.answer } : base;
|
|
251
|
+
}
|
|
252
|
+
function collectPendingInboxRequests(entries) {
|
|
253
|
+
const requested = new Map;
|
|
254
|
+
const resolved = new Map;
|
|
255
|
+
for (const entry of entries) {
|
|
256
|
+
if (entry.type !== "custom")
|
|
257
|
+
continue;
|
|
258
|
+
if (entry.customType === RIG_WORKFLOW_INBOX_REQUESTED) {
|
|
259
|
+
const request = parseWorkflowInboxRequested(entry.data);
|
|
260
|
+
if (request && newer(requested.get(request.requestId)?.requestedAt, request.requestedAt))
|
|
261
|
+
requested.set(request.requestId, request);
|
|
262
|
+
} else if (entry.customType === RIG_WORKFLOW_INBOX_RESOLVED) {
|
|
263
|
+
const resolution = parseWorkflowInboxResolved(entry.data);
|
|
264
|
+
if (resolution && newer(resolved.get(resolution.requestId), resolution.resolvedAt))
|
|
265
|
+
resolved.set(resolution.requestId, resolution.resolvedAt);
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
for (const [requestId, request] of requested) {
|
|
269
|
+
const resolvedAt = resolved.get(requestId);
|
|
270
|
+
if (resolvedAt && Date.parse(resolvedAt) >= Date.parse(request.requestedAt))
|
|
271
|
+
requested.delete(requestId);
|
|
272
|
+
}
|
|
273
|
+
return Array.from(requested.values()).sort((a, b) => Date.parse(b.requestedAt) - Date.parse(a.requestedAt));
|
|
274
|
+
}
|
|
275
|
+
function collectResolvedInboxRequests(entries) {
|
|
276
|
+
const requested = new Map;
|
|
277
|
+
const resolved = new Map;
|
|
278
|
+
for (const entry of entries) {
|
|
279
|
+
if (entry.type !== "custom")
|
|
280
|
+
continue;
|
|
281
|
+
if (entry.customType === RIG_WORKFLOW_INBOX_REQUESTED) {
|
|
282
|
+
const request = parseWorkflowInboxRequested(entry.data);
|
|
283
|
+
if (request && newer(requested.get(request.requestId), request.requestedAt))
|
|
284
|
+
requested.set(request.requestId, request.requestedAt);
|
|
285
|
+
} else if (entry.customType === RIG_WORKFLOW_INBOX_RESOLVED) {
|
|
286
|
+
const resolution = parseWorkflowInboxResolved(entry.data);
|
|
287
|
+
if (resolution && newer(resolved.get(resolution.requestId)?.resolvedAt, resolution.resolvedAt)) {
|
|
288
|
+
resolved.set(resolution.requestId, resolution);
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
for (const [requestId, resolution] of resolved) {
|
|
293
|
+
const requestedAt = requested.get(requestId);
|
|
294
|
+
if (requestedAt && Date.parse(requestedAt) > Date.parse(resolution.resolvedAt))
|
|
295
|
+
resolved.delete(requestId);
|
|
296
|
+
}
|
|
297
|
+
return Array.from(resolved.values()).sort((a, b) => Date.parse(b.resolvedAt) - Date.parse(a.resolvedAt));
|
|
298
|
+
}
|
|
299
|
+
function projectCollabWorkflowMarker(collab) {
|
|
300
|
+
const status = createWorkflowStatusChanged({
|
|
301
|
+
status: collab.stale ? "stopped" : "running",
|
|
302
|
+
detail: collab.stale ? "Stale OMP collab session discovered from the sanctioned collab registry." : "Live OMP collab session discovered from the sanctioned collab registry.",
|
|
303
|
+
changedAt: collab.updatedAt
|
|
304
|
+
});
|
|
305
|
+
return {
|
|
306
|
+
started: null,
|
|
307
|
+
target: null,
|
|
308
|
+
task: null,
|
|
309
|
+
status,
|
|
310
|
+
notes: [],
|
|
311
|
+
inbox: [],
|
|
312
|
+
resolvedInbox: [],
|
|
313
|
+
updatedAt: newer(collab.startedAt, collab.updatedAt) ? collab.updatedAt : collab.startedAt
|
|
314
|
+
};
|
|
315
|
+
}
|
|
316
|
+
function projectWorkflowEntries(entries) {
|
|
317
|
+
let started = null;
|
|
318
|
+
let target = null;
|
|
319
|
+
let task = null;
|
|
320
|
+
let status = null;
|
|
321
|
+
const notes = [];
|
|
322
|
+
let updatedAt = null;
|
|
323
|
+
for (const entry of entries) {
|
|
324
|
+
if (entry.type !== "custom")
|
|
325
|
+
continue;
|
|
326
|
+
if (entry.customType === RIG_WORKFLOW_STARTED) {
|
|
327
|
+
const parsed = parseWorkflowStarted(entry.data);
|
|
328
|
+
if (parsed && newer(started?.createdAt, parsed.createdAt))
|
|
329
|
+
started = parsed;
|
|
330
|
+
} else if (entry.customType === RIG_WORKFLOW_TARGET_SELECTED) {
|
|
331
|
+
const parsed = parseWorkflowTargetSelected(entry.data);
|
|
332
|
+
if (parsed && newer(target?.selectedAt, parsed.selectedAt))
|
|
333
|
+
target = parsed;
|
|
334
|
+
} else if (entry.customType === RIG_WORKFLOW_TASK_SELECTED) {
|
|
335
|
+
const parsed = parseWorkflowTaskSelected(entry.data);
|
|
336
|
+
if (parsed && newer(task?.selectedAt, parsed.selectedAt))
|
|
337
|
+
task = parsed;
|
|
338
|
+
} else if (entry.customType === RIG_WORKFLOW_STATUS_CHANGED) {
|
|
339
|
+
const parsed = parseWorkflowStatusChanged(entry.data);
|
|
340
|
+
if (parsed && newer(status?.changedAt, parsed.changedAt))
|
|
341
|
+
status = parsed;
|
|
342
|
+
} else if (entry.customType === RIG_WORKFLOW_OPERATOR_NOTE) {
|
|
343
|
+
const parsed = parseWorkflowOperatorNote(entry.data);
|
|
344
|
+
if (parsed)
|
|
345
|
+
notes.push(parsed);
|
|
346
|
+
}
|
|
347
|
+
}
|
|
348
|
+
notes.sort((a, b) => Date.parse(b.notedAt) - Date.parse(a.notedAt));
|
|
349
|
+
for (const value of [started, target, task, status, ...notes].map((entry) => entry ? entryUpdatedAt(entry) : null)) {
|
|
350
|
+
if (newer(updatedAt, value))
|
|
351
|
+
updatedAt = value;
|
|
352
|
+
}
|
|
353
|
+
const inbox = collectPendingInboxRequests(entries);
|
|
354
|
+
const resolvedInbox = collectResolvedInboxRequests(entries);
|
|
355
|
+
for (const entry of [...inbox, ...resolvedInbox]) {
|
|
356
|
+
if (newer(updatedAt, entryUpdatedAt(entry)))
|
|
357
|
+
updatedAt = entryUpdatedAt(entry);
|
|
358
|
+
}
|
|
359
|
+
if (!started && !target && !task && !status && notes.length === 0 && inbox.length === 0 && resolvedInbox.length === 0 && !updatedAt)
|
|
360
|
+
return EMPTY_PROJECTION;
|
|
361
|
+
return { started, target, task, status, notes, inbox, resolvedInbox, updatedAt };
|
|
362
|
+
}
|
|
363
|
+
export {
|
|
364
|
+
projectWorkflowEntries,
|
|
365
|
+
projectCollabWorkflowMarker,
|
|
366
|
+
parseWorkflowTaskSelected,
|
|
367
|
+
parseWorkflowTargetSelected,
|
|
368
|
+
parseWorkflowStatusChanged,
|
|
369
|
+
parseWorkflowStarted,
|
|
370
|
+
parseWorkflowOperatorNote,
|
|
371
|
+
parseWorkflowInboxResolved,
|
|
372
|
+
parseWorkflowInboxRequested,
|
|
373
|
+
createWorkflowTaskSelected,
|
|
374
|
+
createWorkflowTargetSelected,
|
|
375
|
+
createWorkflowStatusChanged,
|
|
376
|
+
createWorkflowStarted,
|
|
377
|
+
createWorkflowOperatorNote,
|
|
378
|
+
createWorkflowInboxResolved,
|
|
379
|
+
createWorkflowInboxRequested,
|
|
380
|
+
collectResolvedInboxRequests,
|
|
381
|
+
collectPendingInboxRequests,
|
|
382
|
+
RIG_WORKFLOW_TASK_SELECTED,
|
|
383
|
+
RIG_WORKFLOW_TARGET_SELECTED,
|
|
384
|
+
RIG_WORKFLOW_STATUS_CHANGED,
|
|
385
|
+
RIG_WORKFLOW_STARTED,
|
|
386
|
+
RIG_WORKFLOW_OPERATOR_NOTE,
|
|
387
|
+
RIG_WORKFLOW_INBOX_RESOLVED,
|
|
388
|
+
RIG_WORKFLOW_INBOX_REQUESTED
|
|
389
|
+
};
|
package/package.json
ADDED
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@h-rig/rig-extension",
|
|
3
|
+
"version": "0.0.6-alpha.91",
|
|
4
|
+
"type": "module",
|
|
5
|
+
"description": "Rig package",
|
|
6
|
+
"license": "UNLICENSED",
|
|
7
|
+
"files": [
|
|
8
|
+
"dist",
|
|
9
|
+
"README.md"
|
|
10
|
+
],
|
|
11
|
+
"exports": {
|
|
12
|
+
".": {
|
|
13
|
+
"types": "./dist/src/extension.d.ts",
|
|
14
|
+
"import": "./dist/src/extension.js"
|
|
15
|
+
},
|
|
16
|
+
"./package.json": "./package.json"
|
|
17
|
+
},
|
|
18
|
+
"engines": {
|
|
19
|
+
"bun": ">=1.3.11"
|
|
20
|
+
},
|
|
21
|
+
"main": "./dist/src/extension.js",
|
|
22
|
+
"omp": {
|
|
23
|
+
"extensions": [
|
|
24
|
+
"./dist/src/extension.js"
|
|
25
|
+
]
|
|
26
|
+
},
|
|
27
|
+
"dependencies": {
|
|
28
|
+
"@rig/contracts": "npm:@h-rig/contracts@0.0.6-alpha.91",
|
|
29
|
+
"@rig/core": "npm:@h-rig/core@0.0.6-alpha.91",
|
|
30
|
+
"@rig/relay-registry": "0.0.6-alpha.91",
|
|
31
|
+
"@rig/runtime": "npm:@h-rig/runtime@0.0.6-alpha.91",
|
|
32
|
+
"@oh-my-pi/pi-coding-agent": "16.0.4",
|
|
33
|
+
"@oh-my-pi/pi-tui": "16.0.4",
|
|
34
|
+
"effect": "4.0.0-beta.78"
|
|
35
|
+
}
|
|
36
|
+
}
|