robotrock 0.1.0 → 0.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +190 -33
- package/dist/ai/index.d.ts +24 -0
- package/dist/ai/index.js +46 -0
- package/dist/ai/index.js.map +1 -0
- package/dist/ai/trigger.d.ts +5 -0
- package/dist/ai/trigger.js +24 -0
- package/dist/ai/trigger.js.map +1 -0
- package/dist/ai/workflow.d.ts +5 -0
- package/dist/ai/workflow.js +24 -0
- package/dist/ai/workflow.js.map +1 -0
- package/dist/chunk-D2FBSEZK.js +67 -0
- package/dist/chunk-D2FBSEZK.js.map +1 -0
- package/dist/chunk-DSZ3GMT4.js +518 -0
- package/dist/chunk-DSZ3GMT4.js.map +1 -0
- package/dist/chunk-KOXJCIST.js +332 -0
- package/dist/chunk-KOXJCIST.js.map +1 -0
- package/dist/chunk-LXM7VS4Q.js +129 -0
- package/dist/chunk-LXM7VS4Q.js.map +1 -0
- package/dist/client-Dhk9qxhL.d.ts +104 -0
- package/dist/handler-webhook-BqEi6Bk-.d.ts +16 -0
- package/dist/index.d.ts +84 -33
- package/dist/index.js +117 -38
- package/dist/index.js.map +1 -1
- package/dist/schemas/index.d.ts +643 -0
- package/dist/schemas/index.js +11 -0
- package/dist/schemas/index.js.map +1 -0
- package/dist/trigger/index.d.ts +34 -37
- package/dist/trigger/index.js +44 -72
- package/dist/trigger/index.js.map +1 -1
- package/dist/workflow/index.d.ts +35 -0
- package/dist/workflow/index.js +99 -0
- package/dist/workflow/index.js.map +1 -0
- package/dist/workflow-BYeIZgD0.d.ts +309 -0
- package/package.json +38 -8
- package/dist/chunk-TUQXDKV6.js +0 -209
- package/dist/chunk-TUQXDKV6.js.map +0 -1
- package/dist/client-BQ-j7q68.d.ts +0 -37
|
@@ -0,0 +1,518 @@
|
|
|
1
|
+
import {
|
|
2
|
+
assignToSchema
|
|
3
|
+
} from "./chunk-LXM7VS4Q.js";
|
|
4
|
+
|
|
5
|
+
// src/ai/context.ts
|
|
6
|
+
var APPROVE_BY_HUMAN_ACTIONS = [
|
|
7
|
+
{ id: "approve", title: "Approve" },
|
|
8
|
+
{ id: "decline", title: "Decline" }
|
|
9
|
+
];
|
|
10
|
+
function isRobotRockClient(value) {
|
|
11
|
+
return typeof value === "object" && value !== null && "sendToHuman" in value && typeof value.sendToHuman === "function";
|
|
12
|
+
}
|
|
13
|
+
function normalizeRobotRockAiContext(clientOrContext) {
|
|
14
|
+
if (isRobotRockClient(clientOrContext)) {
|
|
15
|
+
return { mode: "polling", client: clientOrContext };
|
|
16
|
+
}
|
|
17
|
+
if (clientOrContext.mode === "trigger" || clientOrContext.mode === "workflow") {
|
|
18
|
+
return clientOrContext;
|
|
19
|
+
}
|
|
20
|
+
if (clientOrContext.mode === "polling" || clientOrContext.mode === void 0) {
|
|
21
|
+
if (!("client" in clientOrContext) || !clientOrContext.client) {
|
|
22
|
+
throw new Error("RobotRock AI polling mode requires `client` on the context object.");
|
|
23
|
+
}
|
|
24
|
+
return { mode: "polling", client: clientOrContext.client };
|
|
25
|
+
}
|
|
26
|
+
throw new Error(`Unknown RobotRock AI mode: ${String(clientOrContext.mode)}`);
|
|
27
|
+
}
|
|
28
|
+
async function sendToHumanForAi(context, payload) {
|
|
29
|
+
if (context.mode === "trigger") {
|
|
30
|
+
const { sendToHumanTask } = await import("./trigger/index.js");
|
|
31
|
+
const waitResult = await sendToHumanTask.triggerAndWait({
|
|
32
|
+
...payload,
|
|
33
|
+
...context.app ? { app: context.app } : {}
|
|
34
|
+
});
|
|
35
|
+
if (!waitResult.ok) {
|
|
36
|
+
throw waitResult.error;
|
|
37
|
+
}
|
|
38
|
+
return waitResult.output;
|
|
39
|
+
}
|
|
40
|
+
if (context.mode === "workflow") {
|
|
41
|
+
const { sendToHumanInWorkflow } = await import("./workflow/index.js");
|
|
42
|
+
const result2 = await sendToHumanInWorkflow({
|
|
43
|
+
...payload,
|
|
44
|
+
...context.app ? { app: context.app } : {}
|
|
45
|
+
});
|
|
46
|
+
return result2;
|
|
47
|
+
}
|
|
48
|
+
const result = await context.client.sendToHuman(payload);
|
|
49
|
+
if (result.mode !== "handled") {
|
|
50
|
+
throw new Error(
|
|
51
|
+
"RobotRock task was created but not handled. Configure client polling or webhook, or use mode: 'trigger' / 'workflow' for durable waits."
|
|
52
|
+
);
|
|
53
|
+
}
|
|
54
|
+
return {
|
|
55
|
+
actionId: result.actionId,
|
|
56
|
+
data: result.data,
|
|
57
|
+
handledBy: result.handledBy,
|
|
58
|
+
handledAt: result.handledAt,
|
|
59
|
+
taskId: result.taskId
|
|
60
|
+
};
|
|
61
|
+
}
|
|
62
|
+
async function approveByHumanForAi(context, payload) {
|
|
63
|
+
if (context.mode === "trigger") {
|
|
64
|
+
const { approveByHumanTask } = await import("./trigger/index.js");
|
|
65
|
+
const waitResult = await approveByHumanTask.triggerAndWait({
|
|
66
|
+
...payload,
|
|
67
|
+
...context.app ? { app: context.app } : {}
|
|
68
|
+
});
|
|
69
|
+
if (!waitResult.ok) {
|
|
70
|
+
throw waitResult.error;
|
|
71
|
+
}
|
|
72
|
+
return waitResult.output;
|
|
73
|
+
}
|
|
74
|
+
if (context.mode === "workflow") {
|
|
75
|
+
const { approveByHumanInWorkflow } = await import("./workflow/index.js");
|
|
76
|
+
return await approveByHumanInWorkflow({
|
|
77
|
+
...payload,
|
|
78
|
+
...context.app ? { app: context.app } : {}
|
|
79
|
+
});
|
|
80
|
+
}
|
|
81
|
+
const result = await context.client.sendToHuman({
|
|
82
|
+
...payload,
|
|
83
|
+
actions: APPROVE_BY_HUMAN_ACTIONS
|
|
84
|
+
});
|
|
85
|
+
if (result.mode !== "handled") {
|
|
86
|
+
throw new Error(
|
|
87
|
+
"RobotRock approval was not handled. Configure client polling or use mode: 'trigger' / 'workflow' for durable waits."
|
|
88
|
+
);
|
|
89
|
+
}
|
|
90
|
+
return {
|
|
91
|
+
actionId: result.actionId,
|
|
92
|
+
data: result.data,
|
|
93
|
+
handledBy: result.handledBy,
|
|
94
|
+
handledAt: result.handledAt,
|
|
95
|
+
taskId: result.taskId
|
|
96
|
+
};
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
// src/ai/human-tool-result.ts
|
|
100
|
+
var APPROVE_IDS = /* @__PURE__ */ new Set(["approve", "approved"]);
|
|
101
|
+
var DECLINE_IDS = /* @__PURE__ */ new Set(["decline", "reject", "deny", "denied"]);
|
|
102
|
+
function toHumanToolResult(result) {
|
|
103
|
+
const payload = {
|
|
104
|
+
taskId: result.taskId,
|
|
105
|
+
actionId: result.actionId,
|
|
106
|
+
data: result.data,
|
|
107
|
+
handledBy: result.handledBy,
|
|
108
|
+
handledAt: result.handledAt.toISOString()
|
|
109
|
+
};
|
|
110
|
+
if (APPROVE_IDS.has(result.actionId)) {
|
|
111
|
+
payload.approved = true;
|
|
112
|
+
} else if (DECLINE_IDS.has(result.actionId)) {
|
|
113
|
+
payload.approved = false;
|
|
114
|
+
}
|
|
115
|
+
return payload;
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
// src/ai/approve-by-human-tool.ts
|
|
119
|
+
import { tool } from "ai";
|
|
120
|
+
import { z } from "zod";
|
|
121
|
+
var APPROVE_BY_HUMAN_ACTIONS2 = [
|
|
122
|
+
{ id: "approve", title: "Approve" },
|
|
123
|
+
{ id: "decline", title: "Decline" }
|
|
124
|
+
];
|
|
125
|
+
var approveByHumanInputSchema = z.object({
|
|
126
|
+
type: z.string().optional().describe("Task type slug; defaults to ai-approval"),
|
|
127
|
+
name: z.string().describe("Short title for the approval request"),
|
|
128
|
+
description: z.string().describe("What needs approval and the consequences of approving or declining"),
|
|
129
|
+
contextSummary: z.string().optional().describe("Optional markdown summary shown to the reviewer")
|
|
130
|
+
});
|
|
131
|
+
function approveByHumanTool(clientOrContext, maybeOptions = {}) {
|
|
132
|
+
const isDurable = typeof clientOrContext === "object" && clientOrContext !== null && "mode" in clientOrContext && (clientOrContext.mode === "trigger" || clientOrContext.mode === "workflow");
|
|
133
|
+
const aiContext = normalizeRobotRockAiContext(
|
|
134
|
+
isDurable ? {
|
|
135
|
+
mode: clientOrContext.mode,
|
|
136
|
+
app: clientOrContext.app
|
|
137
|
+
} : clientOrContext
|
|
138
|
+
);
|
|
139
|
+
const options = isDurable ? clientOrContext : maybeOptions;
|
|
140
|
+
const description = options.description ?? "Request explicit human approval before a sensitive or irreversible step. Returns whether the human approved or declined.";
|
|
141
|
+
return tool({
|
|
142
|
+
description,
|
|
143
|
+
inputSchema: approveByHumanInputSchema,
|
|
144
|
+
execute: async (input) => {
|
|
145
|
+
const taskContext = input.contextSummary !== void 0 ? {
|
|
146
|
+
data: { summary: input.contextSummary },
|
|
147
|
+
ui: {
|
|
148
|
+
summary: { "ui:widget": "textarea", "ui:options": { rows: 6 } }
|
|
149
|
+
}
|
|
150
|
+
} : void 0;
|
|
151
|
+
const result = await approveByHumanForAi(aiContext, {
|
|
152
|
+
type: input.type ?? options.defaultType ?? "ai-approval",
|
|
153
|
+
name: input.name,
|
|
154
|
+
description: input.description,
|
|
155
|
+
context: taskContext
|
|
156
|
+
});
|
|
157
|
+
return toHumanToolResult(result);
|
|
158
|
+
}
|
|
159
|
+
});
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
// src/ai/create-send-to-human-tool.ts
|
|
163
|
+
import { tool as tool2 } from "ai";
|
|
164
|
+
import { z as z2 } from "zod";
|
|
165
|
+
var contextInputSchema = z2.object({
|
|
166
|
+
data: z2.record(z2.unknown()).optional(),
|
|
167
|
+
ui: z2.record(z2.unknown()).optional()
|
|
168
|
+
}).optional();
|
|
169
|
+
var sendToHumanToolInputSchema = z2.object({
|
|
170
|
+
type: z2.string().describe("Task type slug shown in the RobotRock inbox"),
|
|
171
|
+
name: z2.string().describe("Short title for the human reviewer"),
|
|
172
|
+
description: z2.string().optional().describe("What you need from the human and why you cannot proceed alone"),
|
|
173
|
+
context: contextInputSchema.describe("Optional structured context for the inbox UI"),
|
|
174
|
+
validUntil: z2.string().datetime().optional().describe("Optional ISO deadline for the task"),
|
|
175
|
+
assignTo: assignToSchema.optional().describe(
|
|
176
|
+
"Assign to tenant member emails and/or group slugs; narrows who sees the task in the inbox"
|
|
177
|
+
)
|
|
178
|
+
});
|
|
179
|
+
function createSendToHumanTool(clientOrOptions, maybeOptions) {
|
|
180
|
+
const isDurable = typeof clientOrOptions === "object" && clientOrOptions !== null && "mode" in clientOrOptions && (clientOrOptions.mode === "trigger" || clientOrOptions.mode === "workflow");
|
|
181
|
+
const aiContext = normalizeRobotRockAiContext(
|
|
182
|
+
isDurable ? {
|
|
183
|
+
mode: clientOrOptions.mode,
|
|
184
|
+
app: clientOrOptions.app
|
|
185
|
+
} : clientOrOptions
|
|
186
|
+
);
|
|
187
|
+
const options = isDurable ? clientOrOptions : maybeOptions;
|
|
188
|
+
const description = options.description ?? "Request structured input or a decision from a human in the RobotRock inbox. Use when you lack required information, need policy approval, or must confirm an irreversible step.";
|
|
189
|
+
return tool2({
|
|
190
|
+
description,
|
|
191
|
+
inputSchema: sendToHumanToolInputSchema,
|
|
192
|
+
execute: async (input) => {
|
|
193
|
+
const taskContext = input.context ? {
|
|
194
|
+
data: input.context.data ?? {},
|
|
195
|
+
ui: input.context.ui
|
|
196
|
+
} : void 0;
|
|
197
|
+
const payload = {
|
|
198
|
+
type: input.type ?? options.defaultType ?? "ai-human-input",
|
|
199
|
+
name: input.name,
|
|
200
|
+
description: input.description,
|
|
201
|
+
context: taskContext,
|
|
202
|
+
validUntil: input.validUntil,
|
|
203
|
+
assignTo: input.assignTo,
|
|
204
|
+
actions: options.actions
|
|
205
|
+
};
|
|
206
|
+
const result = await sendToHumanForAi(aiContext, payload);
|
|
207
|
+
return toHumanToolResult(result);
|
|
208
|
+
}
|
|
209
|
+
});
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
// src/ai/create-ai-tools.ts
|
|
213
|
+
function createRobotRockAiTools(options) {
|
|
214
|
+
const mode = options.mode ?? (options.client ? "polling" : "trigger");
|
|
215
|
+
if (mode === "polling" && !options.client) {
|
|
216
|
+
throw new Error("createRobotRockAiTools: polling mode requires `client`.");
|
|
217
|
+
}
|
|
218
|
+
const context = mode === "trigger" ? { mode: "trigger", app: options.app } : mode === "workflow" ? { mode: "workflow", app: options.app } : { mode: "polling", client: options.client };
|
|
219
|
+
const durableContext = context.mode === "trigger" || context.mode === "workflow" ? context : null;
|
|
220
|
+
const pollingClient = context.mode === "polling" ? context.client : void 0;
|
|
221
|
+
return {
|
|
222
|
+
context,
|
|
223
|
+
approveByHuman: (toolOptions) => durableContext ? approveByHumanTool({ ...durableContext, ...toolOptions }) : approveByHumanTool(pollingClient, toolOptions),
|
|
224
|
+
sendToHuman: (toolOptions) => durableContext ? createSendToHumanTool({ ...durableContext, ...toolOptions }) : createSendToHumanTool(pollingClient, toolOptions)
|
|
225
|
+
};
|
|
226
|
+
}
|
|
227
|
+
function createRobotRockAiTriggerContext(options = {}) {
|
|
228
|
+
return { mode: "trigger", ...options };
|
|
229
|
+
}
|
|
230
|
+
function createRobotRockAiWorkflowContext(options = {}) {
|
|
231
|
+
return { mode: "workflow", ...options };
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
// src/ai/format-tool-approval-task.ts
|
|
235
|
+
var DEFAULT_APPROVE_ACTIONS = [
|
|
236
|
+
{ id: "approve", title: "Approve execution" },
|
|
237
|
+
{ id: "deny", title: "Deny" }
|
|
238
|
+
];
|
|
239
|
+
function defaultFormatToolApprovalTask(toolCall, options = {}) {
|
|
240
|
+
const approveId = options.approveActionId ?? "approve";
|
|
241
|
+
const denyId = options.denyActionId ?? "deny";
|
|
242
|
+
let inputPreview;
|
|
243
|
+
try {
|
|
244
|
+
inputPreview = JSON.stringify(toolCall.input, null, 2);
|
|
245
|
+
} catch {
|
|
246
|
+
inputPreview = String(toolCall.input);
|
|
247
|
+
}
|
|
248
|
+
return {
|
|
249
|
+
type: options.type ?? "ai-tool-approval",
|
|
250
|
+
name: `Approve: ${toolCall.toolName}`,
|
|
251
|
+
description: `Review the proposed \`${toolCall.toolName}\` tool call before it runs.`,
|
|
252
|
+
context: {
|
|
253
|
+
data: {
|
|
254
|
+
toolName: toolCall.toolName,
|
|
255
|
+
toolCallId: toolCall.toolCallId,
|
|
256
|
+
input: toolCall.input
|
|
257
|
+
},
|
|
258
|
+
ui: {
|
|
259
|
+
toolName: { "ui:widget": "text" },
|
|
260
|
+
toolCallId: { "ui:widget": "text" },
|
|
261
|
+
input: { "ui:widget": "textarea" }
|
|
262
|
+
}
|
|
263
|
+
},
|
|
264
|
+
actions: [
|
|
265
|
+
{ id: approveId, title: "Approve execution" },
|
|
266
|
+
{ id: denyId, title: "Deny" }
|
|
267
|
+
]
|
|
268
|
+
};
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
// src/ai/tool-approval-bridge.ts
|
|
272
|
+
function buildToolMatcher(config) {
|
|
273
|
+
const names = config.tools ? new Set(config.tools) : null;
|
|
274
|
+
return (toolCall) => {
|
|
275
|
+
if (config.when?.(toolCall)) {
|
|
276
|
+
return true;
|
|
277
|
+
}
|
|
278
|
+
if (names && names.has(toolCall.toolName)) {
|
|
279
|
+
return true;
|
|
280
|
+
}
|
|
281
|
+
return false;
|
|
282
|
+
};
|
|
283
|
+
}
|
|
284
|
+
function createRobotRockToolApproval(config) {
|
|
285
|
+
const matches = buildToolMatcher(config);
|
|
286
|
+
const toolApproval = async (options) => {
|
|
287
|
+
return matches(options.toolCall) ? "user-approval" : void 0;
|
|
288
|
+
};
|
|
289
|
+
return toolApproval;
|
|
290
|
+
}
|
|
291
|
+
function createRobotRockNeedsApproval(config) {
|
|
292
|
+
const matches = buildToolMatcher(config);
|
|
293
|
+
return async (_input, options) => {
|
|
294
|
+
const toolCall = findToolCallInMessages(options.messages, options.toolCallId);
|
|
295
|
+
if (!toolCall) {
|
|
296
|
+
return false;
|
|
297
|
+
}
|
|
298
|
+
return matches(toolCall);
|
|
299
|
+
};
|
|
300
|
+
}
|
|
301
|
+
function applyRobotRockToolApprovalToTools(tools, config) {
|
|
302
|
+
const names = config.tools ? new Set(config.tools) : null;
|
|
303
|
+
const needsApproval = createRobotRockNeedsApproval(config);
|
|
304
|
+
const next = { ...tools };
|
|
305
|
+
for (const key of Object.keys(tools)) {
|
|
306
|
+
const name = String(key);
|
|
307
|
+
const shouldApply = names && names.has(name) || names === null && config.when !== void 0;
|
|
308
|
+
if (!shouldApply) {
|
|
309
|
+
continue;
|
|
310
|
+
}
|
|
311
|
+
const existing = tools[key];
|
|
312
|
+
next[key] = {
|
|
313
|
+
...existing,
|
|
314
|
+
needsApproval
|
|
315
|
+
};
|
|
316
|
+
}
|
|
317
|
+
return next;
|
|
318
|
+
}
|
|
319
|
+
function findToolCallInMessages(messages, toolCallId) {
|
|
320
|
+
if (!messages) {
|
|
321
|
+
return void 0;
|
|
322
|
+
}
|
|
323
|
+
for (const message of messages) {
|
|
324
|
+
if (typeof message !== "object" || message === null) {
|
|
325
|
+
continue;
|
|
326
|
+
}
|
|
327
|
+
const role = message.role;
|
|
328
|
+
if (role !== "assistant") {
|
|
329
|
+
continue;
|
|
330
|
+
}
|
|
331
|
+
const content = message.content;
|
|
332
|
+
if (!Array.isArray(content)) {
|
|
333
|
+
continue;
|
|
334
|
+
}
|
|
335
|
+
for (const part of content) {
|
|
336
|
+
if (typeof part !== "object" || part === null) {
|
|
337
|
+
continue;
|
|
338
|
+
}
|
|
339
|
+
const p = part;
|
|
340
|
+
if (p.type === "tool-call" && p.toolCallId === toolCallId) {
|
|
341
|
+
return {
|
|
342
|
+
toolName: String(p.toolName ?? ""),
|
|
343
|
+
toolCallId,
|
|
344
|
+
input: p.input
|
|
345
|
+
};
|
|
346
|
+
}
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
return void 0;
|
|
350
|
+
}
|
|
351
|
+
function collectApprovalRequests(source) {
|
|
352
|
+
const requests = [];
|
|
353
|
+
const visit = (value) => {
|
|
354
|
+
if (!value) {
|
|
355
|
+
return;
|
|
356
|
+
}
|
|
357
|
+
if (Array.isArray(value)) {
|
|
358
|
+
for (const item of value) {
|
|
359
|
+
visit(item);
|
|
360
|
+
}
|
|
361
|
+
return;
|
|
362
|
+
}
|
|
363
|
+
if (typeof value !== "object") {
|
|
364
|
+
return;
|
|
365
|
+
}
|
|
366
|
+
const obj = value;
|
|
367
|
+
if (obj.type === "tool-approval-request" && typeof obj.approvalId === "string") {
|
|
368
|
+
if (obj.isAutomatic === true) {
|
|
369
|
+
return;
|
|
370
|
+
}
|
|
371
|
+
const embedded = obj.toolCall;
|
|
372
|
+
const toolCallId = typeof obj.toolCallId === "string" ? obj.toolCallId : typeof embedded?.toolCallId === "string" ? embedded.toolCallId : void 0;
|
|
373
|
+
let toolCall;
|
|
374
|
+
if (embedded && typeof embedded.toolName === "string") {
|
|
375
|
+
toolCall = {
|
|
376
|
+
toolName: embedded.toolName,
|
|
377
|
+
toolCallId: String(embedded.toolCallId ?? toolCallId ?? ""),
|
|
378
|
+
input: embedded.input
|
|
379
|
+
};
|
|
380
|
+
} else if (toolCallId) {
|
|
381
|
+
toolCall = findToolCallInMessages(
|
|
382
|
+
source.messages,
|
|
383
|
+
toolCallId
|
|
384
|
+
);
|
|
385
|
+
}
|
|
386
|
+
requests.push({
|
|
387
|
+
type: "tool-approval-request",
|
|
388
|
+
approvalId: obj.approvalId,
|
|
389
|
+
toolCallId,
|
|
390
|
+
toolCall,
|
|
391
|
+
isAutomatic: obj.isAutomatic === true
|
|
392
|
+
});
|
|
393
|
+
return;
|
|
394
|
+
}
|
|
395
|
+
if (Array.isArray(obj.content)) {
|
|
396
|
+
visit(obj.content);
|
|
397
|
+
}
|
|
398
|
+
if (Array.isArray(obj.steps)) {
|
|
399
|
+
visit(obj.steps);
|
|
400
|
+
}
|
|
401
|
+
if (obj.response && typeof obj.response === "object") {
|
|
402
|
+
visit(obj.response.messages);
|
|
403
|
+
}
|
|
404
|
+
};
|
|
405
|
+
visit(source);
|
|
406
|
+
if (typeof source === "object" && source !== null && Array.isArray(source.content)) {
|
|
407
|
+
visit(source.content);
|
|
408
|
+
}
|
|
409
|
+
const seen = /* @__PURE__ */ new Set();
|
|
410
|
+
return requests.filter((r) => {
|
|
411
|
+
if (seen.has(r.approvalId)) {
|
|
412
|
+
return false;
|
|
413
|
+
}
|
|
414
|
+
seen.add(r.approvalId);
|
|
415
|
+
return true;
|
|
416
|
+
});
|
|
417
|
+
}
|
|
418
|
+
function resolveToolCallForRequest(request, source, messages) {
|
|
419
|
+
if (request.toolCall) {
|
|
420
|
+
return request.toolCall;
|
|
421
|
+
}
|
|
422
|
+
const toolCallId = request.toolCallId;
|
|
423
|
+
if (toolCallId) {
|
|
424
|
+
const fromMessages = findToolCallInMessages(messages, toolCallId);
|
|
425
|
+
if (fromMessages) {
|
|
426
|
+
return fromMessages;
|
|
427
|
+
}
|
|
428
|
+
}
|
|
429
|
+
if (typeof source === "object" && source !== null && Array.isArray(source.toolCalls)) {
|
|
430
|
+
for (const call of source.toolCalls) {
|
|
431
|
+
if (call.toolCallId === toolCallId) {
|
|
432
|
+
return {
|
|
433
|
+
toolName: String(call.toolName ?? ""),
|
|
434
|
+
toolCallId: String(call.toolCallId ?? ""),
|
|
435
|
+
input: call.input
|
|
436
|
+
};
|
|
437
|
+
}
|
|
438
|
+
}
|
|
439
|
+
}
|
|
440
|
+
throw new Error(
|
|
441
|
+
`Could not resolve tool call for approval ${request.approvalId}. Pass messages that include the assistant tool-call.`
|
|
442
|
+
);
|
|
443
|
+
}
|
|
444
|
+
async function resolveToolApprovalsViaRobotRock(clientOrContext, source, options = {}) {
|
|
445
|
+
const context = normalizeRobotRockAiContext(clientOrContext);
|
|
446
|
+
const approveId = options.approveActionId ?? "approve";
|
|
447
|
+
const denyId = options.denyActionId ?? "deny";
|
|
448
|
+
const formatTask = options.formatTask ?? defaultFormatToolApprovalTask;
|
|
449
|
+
const baseMessages = typeof source === "object" && source !== null && Array.isArray(source.messages) ? [...source.messages] : [];
|
|
450
|
+
const requests = collectApprovalRequests(source);
|
|
451
|
+
const responses = [];
|
|
452
|
+
for (const request of requests) {
|
|
453
|
+
const toolCall = resolveToolCallForRequest(request, source, baseMessages);
|
|
454
|
+
const taskInput = formatTask(toolCall, {
|
|
455
|
+
approveActionId: approveId,
|
|
456
|
+
denyActionId: denyId
|
|
457
|
+
});
|
|
458
|
+
const result = await sendToHumanForAi(context, taskInput);
|
|
459
|
+
const approved = result.actionId === approveId;
|
|
460
|
+
const reason = typeof result.data === "object" && result.data !== null && "reason" in result.data && typeof result.data.reason === "string" ? result.data.reason : approved ? "Approved in RobotRock inbox" : "Denied in RobotRock inbox";
|
|
461
|
+
responses.push({
|
|
462
|
+
type: "tool-approval-response",
|
|
463
|
+
approvalId: request.approvalId,
|
|
464
|
+
approved,
|
|
465
|
+
reason
|
|
466
|
+
});
|
|
467
|
+
}
|
|
468
|
+
const messages = [...baseMessages];
|
|
469
|
+
if (responses.length > 0) {
|
|
470
|
+
messages.push({ role: "tool", content: responses });
|
|
471
|
+
}
|
|
472
|
+
return { responses, messages };
|
|
473
|
+
}
|
|
474
|
+
async function runWithRobotRockApprovals(options) {
|
|
475
|
+
const clientOrContext = options.context ?? (options.client ? options.client : (() => {
|
|
476
|
+
throw new Error("runWithRobotRockApprovals requires `client` or `context`.");
|
|
477
|
+
})());
|
|
478
|
+
const maxRounds = options.maxRounds ?? 20;
|
|
479
|
+
let messages = options.messages ? [...options.messages] : [];
|
|
480
|
+
let lastResult;
|
|
481
|
+
for (let round = 0; round < maxRounds; round++) {
|
|
482
|
+
lastResult = await options.generate(messages);
|
|
483
|
+
const { responses, messages: nextMessages } = await resolveToolApprovalsViaRobotRock(
|
|
484
|
+
clientOrContext,
|
|
485
|
+
{ ...lastResult, messages },
|
|
486
|
+
options.resolveOptions
|
|
487
|
+
);
|
|
488
|
+
if (responses.length === 0) {
|
|
489
|
+
return lastResult;
|
|
490
|
+
}
|
|
491
|
+
messages = nextMessages;
|
|
492
|
+
}
|
|
493
|
+
throw new Error(`RobotRock approval loop exceeded maxRounds (${maxRounds})`);
|
|
494
|
+
}
|
|
495
|
+
|
|
496
|
+
export {
|
|
497
|
+
normalizeRobotRockAiContext,
|
|
498
|
+
sendToHumanForAi,
|
|
499
|
+
approveByHumanForAi,
|
|
500
|
+
toHumanToolResult,
|
|
501
|
+
APPROVE_BY_HUMAN_ACTIONS2 as APPROVE_BY_HUMAN_ACTIONS,
|
|
502
|
+
approveByHumanInputSchema,
|
|
503
|
+
approveByHumanTool,
|
|
504
|
+
sendToHumanToolInputSchema,
|
|
505
|
+
createSendToHumanTool,
|
|
506
|
+
createRobotRockAiTools,
|
|
507
|
+
createRobotRockAiTriggerContext,
|
|
508
|
+
createRobotRockAiWorkflowContext,
|
|
509
|
+
DEFAULT_APPROVE_ACTIONS,
|
|
510
|
+
defaultFormatToolApprovalTask,
|
|
511
|
+
createRobotRockToolApproval,
|
|
512
|
+
createRobotRockNeedsApproval,
|
|
513
|
+
applyRobotRockToolApprovalToTools,
|
|
514
|
+
collectApprovalRequests,
|
|
515
|
+
resolveToolApprovalsViaRobotRock,
|
|
516
|
+
runWithRobotRockApprovals
|
|
517
|
+
};
|
|
518
|
+
//# sourceMappingURL=chunk-DSZ3GMT4.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/ai/context.ts","../src/ai/human-tool-result.ts","../src/ai/approve-by-human-tool.ts","../src/ai/create-send-to-human-tool.ts","../src/ai/create-ai-tools.ts","../src/ai/format-tool-approval-task.ts","../src/ai/tool-approval-bridge.ts"],"sourcesContent":["import type { RobotRock, SendToHumanActionInput, SendToHumanInput } from \"../client.js\";\nimport type { DiscriminatedApprovalResult } from \"../schemas/index.js\";\nconst APPROVE_BY_HUMAN_ACTIONS = [\n { id: \"approve\", title: \"Approve\" },\n { id: \"decline\", title: \"Decline\" },\n] as const;\n\n/** How RobotRock waits for a human inside AI SDK tool `execute`. */\nexport type RobotRockAiMode = \"polling\" | \"trigger\" | \"workflow\";\n\n/**\n * Polling: `client.sendToHuman()` blocks until handled (scripts, API routes with long timeout).\n * Trigger: `sendToHumanTask.triggerAndWait()` uses wait tokens (Trigger.dev workers).\n * Workflow: `sendToHumanInWorkflow()` uses workflow webhooks (Vercel Workflow).\n */\nexport type RobotRockAiPollingContext = {\n mode?: \"polling\";\n client: RobotRock;\n};\n\nexport type RobotRockAiTriggerContext = {\n mode: \"trigger\";\n /** Inbox app bucket; falls back to `ROBOTROCK_APP` in the Trigger worker. */\n app?: string;\n};\n\nexport type RobotRockAiWorkflowContext = {\n mode: \"workflow\";\n /** Inbox app bucket; falls back to `ROBOTROCK_APP` in the workflow run. */\n app?: string;\n};\n\nexport type RobotRockAiContext =\n | RobotRockAiPollingContext\n | RobotRockAiTriggerContext\n | RobotRockAiWorkflowContext;\n\nexport function isRobotRockClient(value: unknown): value is RobotRock {\n return (\n typeof value === \"object\" &&\n value !== null &&\n \"sendToHuman\" in value &&\n typeof (value as RobotRock).sendToHuman === \"function\"\n );\n}\n\nexport function normalizeRobotRockAiContext(\n clientOrContext: RobotRock | RobotRockAiContext\n): RobotRockAiContext {\n if (isRobotRockClient(clientOrContext)) {\n return { mode: \"polling\", client: clientOrContext };\n }\n\n if (clientOrContext.mode === \"trigger\" || clientOrContext.mode === \"workflow\") {\n return clientOrContext;\n }\n\n if (clientOrContext.mode === \"polling\" || clientOrContext.mode === undefined) {\n if (!(\"client\" in clientOrContext) || !clientOrContext.client) {\n throw new Error('RobotRock AI polling mode requires `client` on the context object.');\n }\n return { mode: \"polling\", client: clientOrContext.client };\n }\n\n throw new Error(`Unknown RobotRock AI mode: ${String((clientOrContext as { mode?: string }).mode)}`);\n}\n\nexport async function sendToHumanForAi<\n A extends readonly SendToHumanActionInput[] = readonly SendToHumanActionInput[],\n>(\n context: RobotRockAiContext,\n payload: SendToHumanInput<A>\n): Promise<DiscriminatedApprovalResult<A>> {\n if (context.mode === \"trigger\") {\n const { sendToHumanTask } = await import(\"../trigger/index.js\");\n const waitResult = await sendToHumanTask.triggerAndWait({\n ...payload,\n ...(context.app ? { app: context.app } : {}),\n });\n\n if (!waitResult.ok) {\n throw waitResult.error;\n }\n\n return waitResult.output as DiscriminatedApprovalResult<A>;\n }\n\n if (context.mode === \"workflow\") {\n const { sendToHumanInWorkflow } = await import(\"../workflow/index.js\");\n const result = await sendToHumanInWorkflow({\n ...payload,\n ...(context.app ? { app: context.app } : {}),\n });\n return result as unknown as DiscriminatedApprovalResult<A>;\n }\n\n const result = await context.client.sendToHuman(payload);\n\n if (result.mode !== \"handled\") {\n throw new Error(\n \"RobotRock task was created but not handled. Configure client polling or webhook, or use mode: 'trigger' / 'workflow' for durable waits.\"\n );\n }\n\n return {\n actionId: result.actionId,\n data: result.data,\n handledBy: result.handledBy,\n handledAt: result.handledAt,\n taskId: result.taskId,\n } as DiscriminatedApprovalResult<A>;\n}\n\nexport async function approveByHumanForAi(\n context: RobotRockAiContext,\n payload: Omit<SendToHumanInput, \"actions\"> & { app?: string }\n): Promise<DiscriminatedApprovalResult<typeof APPROVE_BY_HUMAN_ACTIONS>> {\n if (context.mode === \"trigger\") {\n const { approveByHumanTask } = await import(\"../trigger/index.js\");\n const waitResult = await approveByHumanTask.triggerAndWait({\n ...payload,\n ...(context.app ? { app: context.app } : {}),\n });\n\n if (!waitResult.ok) {\n throw waitResult.error;\n }\n\n return waitResult.output;\n }\n\n if (context.mode === \"workflow\") {\n const { approveByHumanInWorkflow } = await import(\"../workflow/index.js\");\n return await approveByHumanInWorkflow({\n ...payload,\n ...(context.app ? { app: context.app } : {}),\n });\n }\n\n const result = await context.client.sendToHuman({\n ...payload,\n actions: APPROVE_BY_HUMAN_ACTIONS,\n });\n\n if (result.mode !== \"handled\") {\n throw new Error(\n \"RobotRock approval was not handled. Configure client polling or use mode: 'trigger' / 'workflow' for durable waits.\"\n );\n }\n\n return {\n actionId: result.actionId,\n data: result.data,\n handledBy: result.handledBy,\n handledAt: result.handledAt,\n taskId: result.taskId,\n } as DiscriminatedApprovalResult<typeof APPROVE_BY_HUMAN_ACTIONS>;\n}\n","import type { DiscriminatedApprovalResult } from \"../schemas/index.js\";\nimport type { HumanToolResult } from \"./types.js\";\n\nconst APPROVE_IDS = new Set([\"approve\", \"approved\"]);\nconst DECLINE_IDS = new Set([\"decline\", \"reject\", \"deny\", \"denied\"]);\n\nexport function toHumanToolResult(\n result: DiscriminatedApprovalResult<readonly { id: string }[]>\n): HumanToolResult {\n const payload: HumanToolResult = {\n taskId: result.taskId,\n actionId: result.actionId,\n data: result.data,\n handledBy: result.handledBy,\n handledAt: result.handledAt.toISOString(),\n };\n\n if (APPROVE_IDS.has(result.actionId)) {\n payload.approved = true;\n } else if (DECLINE_IDS.has(result.actionId)) {\n payload.approved = false;\n }\n\n return payload;\n}\n","import { tool } from \"ai\";\nimport { z } from \"zod\";\nimport type { RobotRock } from \"../client.js\";\nimport { approveByHumanForAi, normalizeRobotRockAiContext, type RobotRockAiContext } from \"./context.js\";\nimport { toHumanToolResult } from \"./human-tool-result.js\";\nimport type { HumanToolResult } from \"./types.js\";\n\nconst APPROVE_BY_HUMAN_ACTIONS = [\n { id: \"approve\", title: \"Approve\" },\n { id: \"decline\", title: \"Decline\" },\n] as const;\n\nconst approveByHumanInputSchema = z.object({\n type: z\n .string()\n .optional()\n .describe(\"Task type slug; defaults to ai-approval\"),\n name: z.string().describe(\"Short title for the approval request\"),\n description: z\n .string()\n .describe(\"What needs approval and the consequences of approving or declining\"),\n contextSummary: z\n .string()\n .optional()\n .describe(\"Optional markdown summary shown to the reviewer\"),\n});\n\nexport type ApproveByHumanToolOptions = {\n defaultType?: string;\n description?: string;\n toolName?: string;\n};\n\nexport type ApproveByHumanToolDurableOptions = ApproveByHumanToolOptions &\n RobotRockAiContext & { mode: \"trigger\" | \"workflow\" };\n\n/**\n * AI SDK tool with fixed approve/decline actions; blocks until a human decides.\n *\n * - `approveByHumanTool(robotrock)` — polling via `client.sendToHuman()` (default).\n * - `approveByHumanTool({ mode: \"trigger\", app?: \"my-agent\" })` — durable wait via Trigger.dev.\n * - `approveByHumanTool({ mode: \"workflow\", app?: \"my-agent\" })` — durable wait via Vercel Workflow.\n */\nexport function approveByHumanTool(\n clientOrContext: RobotRock | ApproveByHumanToolDurableOptions,\n maybeOptions: ApproveByHumanToolOptions = {}\n) {\n const isDurable =\n typeof clientOrContext === \"object\" &&\n clientOrContext !== null &&\n \"mode\" in clientOrContext &&\n (clientOrContext.mode === \"trigger\" || clientOrContext.mode === \"workflow\");\n\n const aiContext = normalizeRobotRockAiContext(\n isDurable\n ? {\n mode: (clientOrContext as ApproveByHumanToolDurableOptions).mode,\n app: (clientOrContext as ApproveByHumanToolDurableOptions).app,\n }\n : (clientOrContext as RobotRock)\n );\n const options = isDurable\n ? (clientOrContext as ApproveByHumanToolDurableOptions)\n : maybeOptions;\n const description =\n options.description ??\n \"Request explicit human approval before a sensitive or irreversible step. Returns whether the human approved or declined.\";\n\n return tool({\n description,\n inputSchema: approveByHumanInputSchema,\n execute: async (input: z.infer<typeof approveByHumanInputSchema>): Promise<HumanToolResult> => {\n const taskContext =\n input.contextSummary !== undefined\n ? {\n data: { summary: input.contextSummary },\n ui: {\n summary: { \"ui:widget\": \"textarea\", \"ui:options\": { rows: 6 } },\n },\n }\n : undefined;\n\n const result = await approveByHumanForAi(aiContext, {\n type: input.type ?? options.defaultType ?? \"ai-approval\",\n name: input.name,\n description: input.description,\n context: taskContext,\n });\n\n return toHumanToolResult(result);\n },\n });\n}\n\nexport { APPROVE_BY_HUMAN_ACTIONS, approveByHumanInputSchema };\n","import { tool } from \"ai\";\nimport { z } from \"zod\";\nimport { assignToSchema } from \"../schemas/index.js\";\nimport type { RobotRock, SendToHumanActionInput, SendToHumanInput } from \"../client.js\";\nimport type { TaskContextInput } from \"../schemas/index.js\";\nimport {\n normalizeRobotRockAiContext,\n sendToHumanForAi,\n type RobotRockAiContext,\n} from \"./context.js\";\nimport { toHumanToolResult } from \"./human-tool-result.js\";\nimport type { HumanToolResult } from \"./types.js\";\n\nconst contextInputSchema = z\n .object({\n data: z.record(z.unknown()).optional(),\n ui: z.record(z.unknown()).optional(),\n })\n .optional();\n\nconst sendToHumanToolInputSchema = z.object({\n type: z.string().describe(\"Task type slug shown in the RobotRock inbox\"),\n name: z.string().describe(\"Short title for the human reviewer\"),\n description: z\n .string()\n .optional()\n .describe(\"What you need from the human and why you cannot proceed alone\"),\n context: contextInputSchema.describe(\"Optional structured context for the inbox UI\"),\n validUntil: z\n .string()\n .datetime()\n .optional()\n .describe(\"Optional ISO deadline for the task\"),\n assignTo: assignToSchema\n .optional()\n .describe(\n \"Assign to tenant member emails and/or group slugs; narrows who sees the task in the inbox\"\n ),\n});\n\nexport type CreateSendToHumanToolOptions<\n A extends readonly SendToHumanActionInput[] = readonly SendToHumanActionInput[],\n> = {\n actions: A;\n defaultType?: string;\n description?: string;\n toolName?: string;\n};\n\nexport type CreateSendToHumanToolDurableOptions<\n A extends readonly SendToHumanActionInput[] = readonly SendToHumanActionInput[],\n> = CreateSendToHumanToolOptions<A> & RobotRockAiContext & { mode: \"trigger\" | \"workflow\" };\n\n/**\n * AI SDK tool that blocks until a human handles a RobotRock task with developer-defined actions.\n *\n * - `createSendToHumanTool(robotrock, options)` — polling (default).\n * - `createSendToHumanTool({ mode: \"trigger\", actions: [...], app?: \"my-agent\" })` — Trigger.dev wait tokens.\n * - `createSendToHumanTool({ mode: \"workflow\", actions: [...], app?: \"my-agent\" })` — Vercel Workflow webhooks.\n */\nexport function createSendToHumanTool<\n A extends readonly SendToHumanActionInput[] = readonly SendToHumanActionInput[],\n>(\n clientOrOptions: RobotRock | CreateSendToHumanToolDurableOptions<A>,\n maybeOptions?: CreateSendToHumanToolOptions<A>\n) {\n const isDurable =\n typeof clientOrOptions === \"object\" &&\n clientOrOptions !== null &&\n \"mode\" in clientOrOptions &&\n (clientOrOptions.mode === \"trigger\" || clientOrOptions.mode === \"workflow\");\n\n const aiContext = normalizeRobotRockAiContext(\n isDurable\n ? {\n mode: (clientOrOptions as CreateSendToHumanToolDurableOptions<A>).mode,\n app: (clientOrOptions as CreateSendToHumanToolDurableOptions<A>).app,\n }\n : (clientOrOptions as RobotRock)\n );\n const options = (isDurable ? clientOrOptions : maybeOptions!) as CreateSendToHumanToolOptions<A>;\n const description =\n options.description ??\n \"Request structured input or a decision from a human in the RobotRock inbox. Use when you lack required information, need policy approval, or must confirm an irreversible step.\";\n\n return tool({\n description,\n inputSchema: sendToHumanToolInputSchema,\n execute: async (\n input: z.infer<typeof sendToHumanToolInputSchema>\n ): Promise<HumanToolResult> => {\n const taskContext: TaskContextInput[\"context\"] | undefined = input.context\n ? ({\n data: input.context.data ?? {},\n ui: input.context.ui,\n } as TaskContextInput[\"context\"])\n : undefined;\n\n const payload: SendToHumanInput<A> = {\n type: input.type ?? options.defaultType ?? \"ai-human-input\",\n name: input.name,\n description: input.description,\n context: taskContext,\n validUntil: input.validUntil,\n assignTo: input.assignTo,\n actions: options.actions,\n };\n\n const result = await sendToHumanForAi(aiContext, payload);\n\n return toHumanToolResult(result);\n },\n });\n}\n\nexport { sendToHumanToolInputSchema };\n","import type { RobotRock, SendToHumanActionInput } from \"../client.js\";\nimport type {\n RobotRockAiContext,\n RobotRockAiTriggerContext,\n RobotRockAiWorkflowContext,\n} from \"./context.js\";\nimport { normalizeRobotRockAiContext } from \"./context.js\";\nimport { approveByHumanTool as buildApproveByHumanTool } from \"./approve-by-human-tool.js\";\nimport {\n createSendToHumanTool as buildSendToHumanTool,\n type CreateSendToHumanToolOptions,\n} from \"./create-send-to-human-tool.js\";\nimport type { ApproveByHumanToolOptions } from \"./approve-by-human-tool.js\";\n\nexport type CreateRobotRockAiToolsOptions = {\n /** @default \"polling\" when `client` is passed; `\"trigger\"` or `\"workflow\"` for durable waits. */\n mode?: \"polling\" | \"trigger\" | \"workflow\";\n client?: RobotRock;\n app?: string;\n};\n\n/**\n * Build AI SDK tools for a given RobotRock execution mode.\n *\n * @example Polling (Next.js route, scripts)\n * ```ts\n * const tools = createRobotRockAiTools({ client: robotrock });\n * ```\n *\n * @example Trigger.dev worker\n * ```ts\n * const tools = createRobotRockAiTools({ mode: \"trigger\", app: \"my-agent\" });\n * ```\n *\n * @example Vercel Workflow + DurableAgent\n * ```ts\n * const tools = createRobotRockAiTools({ mode: \"workflow\", app: \"my-agent\" });\n * ```\n */\nexport function createRobotRockAiTools(options: CreateRobotRockAiToolsOptions) {\n const mode = options.mode ?? (options.client ? \"polling\" : \"trigger\");\n\n if (mode === \"polling\" && !options.client) {\n throw new Error('createRobotRockAiTools: polling mode requires `client`.');\n }\n\n const context: RobotRockAiContext =\n mode === \"trigger\"\n ? { mode: \"trigger\", app: options.app }\n : mode === \"workflow\"\n ? { mode: \"workflow\", app: options.app }\n : { mode: \"polling\", client: options.client! };\n\n const durableContext =\n context.mode === \"trigger\" || context.mode === \"workflow\" ? context : null;\n const pollingClient = context.mode === \"polling\" ? context.client : undefined;\n\n return {\n context,\n approveByHuman: (toolOptions?: ApproveByHumanToolOptions) =>\n durableContext\n ? buildApproveByHumanTool({ ...durableContext, ...toolOptions })\n : buildApproveByHumanTool(pollingClient!, toolOptions),\n sendToHuman: <A extends readonly SendToHumanActionInput[]>(\n toolOptions: CreateSendToHumanToolOptions<A>\n ) =>\n durableContext\n ? buildSendToHumanTool({ ...durableContext, ...toolOptions })\n : buildSendToHumanTool(pollingClient!, toolOptions),\n };\n}\n\n/** Shorthand for `{ mode: \"trigger\", app }`. */\nexport function createRobotRockAiTriggerContext(\n options: Omit<RobotRockAiTriggerContext, \"mode\"> = {}\n): RobotRockAiContext {\n return { mode: \"trigger\", ...options };\n}\n\n/** Shorthand for `{ mode: \"workflow\", app }`. */\nexport function createRobotRockAiWorkflowContext(\n options: Omit<RobotRockAiWorkflowContext, \"mode\"> = {}\n): RobotRockAiContext {\n return { mode: \"workflow\", ...options };\n}\n\nexport { normalizeRobotRockAiContext };\n","import type { SendToHumanInput } from \"../client.js\";\nimport type { FormatToolApprovalTaskOptions, RobotRockToolCallInfo } from \"./types.js\";\n\nconst DEFAULT_APPROVE_ACTIONS = [\n { id: \"approve\", title: \"Approve execution\" },\n { id: \"deny\", title: \"Deny\" },\n] as const;\n\n/**\n * Build a RobotRock task payload for an AI SDK tool approval request.\n */\nexport function defaultFormatToolApprovalTask(\n toolCall: RobotRockToolCallInfo,\n options: FormatToolApprovalTaskOptions = {}\n): SendToHumanInput {\n const approveId = options.approveActionId ?? \"approve\";\n const denyId = options.denyActionId ?? \"deny\";\n\n let inputPreview: string;\n try {\n inputPreview = JSON.stringify(toolCall.input, null, 2);\n } catch {\n inputPreview = String(toolCall.input);\n }\n\n return {\n type: options.type ?? \"ai-tool-approval\",\n name: `Approve: ${toolCall.toolName}`,\n description: `Review the proposed \\`${toolCall.toolName}\\` tool call before it runs.`,\n context: {\n data: {\n toolName: toolCall.toolName,\n toolCallId: toolCall.toolCallId,\n input: toolCall.input,\n },\n ui: {\n toolName: { \"ui:widget\": \"text\" },\n toolCallId: { \"ui:widget\": \"text\" },\n input: { \"ui:widget\": \"textarea\" },\n },\n },\n actions: [\n { id: approveId, title: \"Approve execution\" },\n { id: denyId, title: \"Deny\" },\n ],\n };\n}\n\nexport { DEFAULT_APPROVE_ACTIONS };\n","import type { ToolApprovalResponse } from \"ai\";\nimport type { RobotRock } from \"../client.js\";\nimport {\n normalizeRobotRockAiContext,\n sendToHumanForAi,\n type RobotRockAiContext,\n} from \"./context.js\";\nimport { defaultFormatToolApprovalTask } from \"./format-tool-approval-task.js\";\nimport type {\n ResolveToolApprovalsOptions,\n RobotRockToolApprovalConfig,\n RobotRockToolCallInfo,\n RunWithRobotRockApprovalsOptions,\n ToolApprovalRequestPart,\n} from \"./types.js\";\n\nexport type RobotRockToolApprovalDecision = \"user-approval\" | undefined;\n\nfunction buildToolMatcher(config: RobotRockToolApprovalConfig) {\n const names = config.tools ? new Set(config.tools) : null;\n\n return (toolCall: RobotRockToolCallInfo): boolean => {\n if (config.when?.(toolCall)) {\n return true;\n }\n if (names && names.has(toolCall.toolName)) {\n return true;\n }\n return false;\n };\n}\n\n/**\n * AI SDK 7+: pass the return value to `toolApproval` on `generateText`, `streamText`, or `ToolLoopAgent`.\n * Returns `'user-approval'` for configured tools so execution pauses until a human responds via RobotRock.\n */\nexport function createRobotRockToolApproval(config: RobotRockToolApprovalConfig) {\n const matches = buildToolMatcher(config);\n\n const toolApproval = async (options: {\n toolCall: RobotRockToolCallInfo;\n }): Promise<RobotRockToolApprovalDecision> => {\n return matches(options.toolCall) ? \"user-approval\" : undefined;\n };\n\n return toolApproval;\n}\n\n/**\n * AI SDK 5–6: use as `needsApproval` on tool definitions (or via {@link applyRobotRockToolApprovalToTools}).\n */\nexport function createRobotRockNeedsApproval(config: RobotRockToolApprovalConfig) {\n const matches = buildToolMatcher(config);\n\n return async (\n _input: unknown,\n options: { toolCallId: string; messages?: unknown[] }\n ): Promise<boolean> => {\n const toolCall = findToolCallInMessages(options.messages, options.toolCallId);\n if (!toolCall) {\n return false;\n }\n return matches(toolCall);\n };\n}\n\n/**\n * AI SDK 5–6: merge `needsApproval` into each matching tool in a tools object.\n */\nexport function applyRobotRockToolApprovalToTools<T extends Record<string, object>>(\n tools: T,\n config: RobotRockToolApprovalConfig\n): T {\n const names = config.tools ? new Set(config.tools) : null;\n const needsApproval = createRobotRockNeedsApproval(config);\n\n const next = { ...tools } as T;\n\n for (const key of Object.keys(tools) as Array<keyof T>) {\n const name = String(key);\n const shouldApply =\n (names && names.has(name)) || (names === null && config.when !== undefined);\n\n if (!shouldApply) {\n continue;\n }\n\n const existing = tools[key];\n next[key] = {\n ...existing,\n needsApproval,\n } as T[keyof T];\n }\n\n return next;\n}\n\nfunction findToolCallInMessages(\n messages: unknown[] | undefined,\n toolCallId: string\n): RobotRockToolCallInfo | undefined {\n if (!messages) {\n return undefined;\n }\n\n for (const message of messages) {\n if (typeof message !== \"object\" || message === null) {\n continue;\n }\n const role = (message as { role?: string }).role;\n if (role !== \"assistant\") {\n continue;\n }\n const content = (message as { content?: unknown }).content;\n if (!Array.isArray(content)) {\n continue;\n }\n\n for (const part of content) {\n if (typeof part !== \"object\" || part === null) {\n continue;\n }\n const p = part as Record<string, unknown>;\n if (p.type === \"tool-call\" && p.toolCallId === toolCallId) {\n return {\n toolName: String(p.toolName ?? \"\"),\n toolCallId,\n input: p.input,\n };\n }\n }\n }\n\n return undefined;\n}\n\nfunction collectApprovalRequests(source: unknown): ToolApprovalRequestPart[] {\n const requests: ToolApprovalRequestPart[] = [];\n\n const visit = (value: unknown) => {\n if (!value) {\n return;\n }\n\n if (Array.isArray(value)) {\n for (const item of value) {\n visit(item);\n }\n return;\n }\n\n if (typeof value !== \"object\") {\n return;\n }\n\n const obj = value as Record<string, unknown>;\n\n if (obj.type === \"tool-approval-request\" && typeof obj.approvalId === \"string\") {\n if (obj.isAutomatic === true) {\n return;\n }\n\n const embedded = obj.toolCall as Record<string, unknown> | undefined;\n const toolCallId =\n typeof obj.toolCallId === \"string\"\n ? obj.toolCallId\n : typeof embedded?.toolCallId === \"string\"\n ? embedded.toolCallId\n : undefined;\n\n let toolCall: RobotRockToolCallInfo | undefined;\n if (embedded && typeof embedded.toolName === \"string\") {\n toolCall = {\n toolName: embedded.toolName,\n toolCallId: String(embedded.toolCallId ?? toolCallId ?? \"\"),\n input: embedded.input,\n };\n } else if (toolCallId) {\n toolCall = findToolCallInMessages(\n (source as { messages?: unknown[] }).messages,\n toolCallId\n );\n }\n\n requests.push({\n type: \"tool-approval-request\",\n approvalId: obj.approvalId,\n toolCallId,\n toolCall,\n isAutomatic: obj.isAutomatic === true,\n });\n return;\n }\n\n if (Array.isArray(obj.content)) {\n visit(obj.content);\n }\n if (Array.isArray(obj.steps)) {\n visit(obj.steps);\n }\n if (obj.response && typeof obj.response === \"object\") {\n visit((obj.response as { messages?: unknown[] }).messages);\n }\n };\n\n visit(source);\n\n if (typeof source === \"object\" && source !== null && Array.isArray((source as { content?: unknown[] }).content)) {\n visit((source as { content: unknown[] }).content);\n }\n\n const seen = new Set<string>();\n return requests.filter((r) => {\n if (seen.has(r.approvalId)) {\n return false;\n }\n seen.add(r.approvalId);\n return true;\n });\n}\n\nfunction resolveToolCallForRequest(\n request: ToolApprovalRequestPart,\n source: unknown,\n messages: unknown[]\n): RobotRockToolCallInfo {\n if (request.toolCall) {\n return request.toolCall;\n }\n\n const toolCallId = request.toolCallId;\n if (toolCallId) {\n const fromMessages = findToolCallInMessages(messages, toolCallId);\n if (fromMessages) {\n return fromMessages;\n }\n }\n\n if (typeof source === \"object\" && source !== null && Array.isArray((source as { toolCalls?: unknown[] }).toolCalls)) {\n for (const call of (source as { toolCalls: Array<Record<string, unknown>> }).toolCalls) {\n if (call.toolCallId === toolCallId) {\n return {\n toolName: String(call.toolName ?? \"\"),\n toolCallId: String(call.toolCallId ?? \"\"),\n input: call.input,\n };\n }\n }\n }\n\n throw new Error(\n `Could not resolve tool call for approval ${request.approvalId}. Pass messages that include the assistant tool-call.`\n );\n}\n\n/**\n * Create RobotRock tasks for pending AI SDK tool approvals and return `tool-approval-response` parts.\n */\nexport async function resolveToolApprovalsViaRobotRock(\n clientOrContext: RobotRock | RobotRockAiContext,\n source: unknown,\n options: ResolveToolApprovalsOptions = {}\n): Promise<{\n responses: ToolApprovalResponse[];\n messages: unknown[];\n}> {\n const context = normalizeRobotRockAiContext(clientOrContext);\n const approveId = options.approveActionId ?? \"approve\";\n const denyId = options.denyActionId ?? \"deny\";\n const formatTask = options.formatTask ?? defaultFormatToolApprovalTask;\n\n const baseMessages =\n typeof source === \"object\" && source !== null && Array.isArray((source as { messages?: unknown[] }).messages)\n ? [...(source as { messages: unknown[] }).messages]\n : [];\n\n const requests = collectApprovalRequests(source);\n const responses: ToolApprovalResponse[] = [];\n\n for (const request of requests) {\n const toolCall = resolveToolCallForRequest(request, source, baseMessages);\n const taskInput = formatTask(toolCall, {\n approveActionId: approveId,\n denyActionId: denyId,\n });\n\n const result = await sendToHumanForAi(context, taskInput);\n\n const approved = result.actionId === approveId;\n const reason =\n typeof result.data === \"object\" &&\n result.data !== null &&\n \"reason\" in result.data &&\n typeof (result.data as { reason?: unknown }).reason === \"string\"\n ? (result.data as { reason: string }).reason\n : approved\n ? \"Approved in RobotRock inbox\"\n : \"Denied in RobotRock inbox\";\n\n responses.push({\n type: \"tool-approval-response\",\n approvalId: request.approvalId,\n approved,\n reason,\n });\n }\n\n const messages = [...baseMessages];\n if (responses.length > 0) {\n messages.push({ role: \"tool\", content: responses });\n }\n\n return { responses, messages };\n}\n\n/**\n * Run `generate` in a loop until manual tool approvals are resolved via RobotRock or `maxRounds` is reached.\n */\nexport async function runWithRobotRockApprovals<T>(\n options: RunWithRobotRockApprovalsOptions<T>\n): Promise<T> {\n const clientOrContext =\n options.context ??\n (options.client\n ? options.client\n : (() => {\n throw new Error(\"runWithRobotRockApprovals requires `client` or `context`.\");\n })());\n\n const maxRounds = options.maxRounds ?? 20;\n let messages = options.messages ? [...options.messages] : [];\n let lastResult: T | undefined;\n\n for (let round = 0; round < maxRounds; round++) {\n lastResult = await options.generate(messages);\n const { responses, messages: nextMessages } = await resolveToolApprovalsViaRobotRock(\n clientOrContext,\n { ...(lastResult as object), messages },\n options.resolveOptions\n );\n\n if (responses.length === 0) {\n return lastResult;\n }\n\n messages = nextMessages;\n }\n\n throw new Error(`RobotRock approval loop exceeded maxRounds (${maxRounds})`);\n}\n\nexport { collectApprovalRequests };\n"],"mappings":"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","names":["result","APPROVE_BY_HUMAN_ACTIONS","tool","z","z","tool"]}
|