@ouro.bot/cli 0.1.0-alpha.13 → 0.1.0-alpha.131
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/AdoptionSpecialist.ouro/psyche/SOUL.md +2 -2
- package/AdoptionSpecialist.ouro/psyche/identities/monty.md +2 -2
- package/README.md +147 -205
- package/changelog.json +814 -0
- package/dist/heart/active-work.js +622 -0
- package/dist/heart/bridges/manager.js +358 -0
- package/dist/heart/bridges/state-machine.js +135 -0
- package/dist/heart/bridges/store.js +123 -0
- package/dist/heart/commitments.js +105 -0
- package/dist/heart/config.js +66 -21
- package/dist/heart/core.js +518 -100
- package/dist/heart/cross-chat-delivery.js +146 -0
- package/dist/heart/daemon/agent-discovery.js +81 -0
- package/dist/heart/daemon/auth-flow.js +457 -0
- package/dist/heart/daemon/daemon-cli.js +1516 -195
- package/dist/heart/daemon/daemon-entry.js +43 -2
- package/dist/heart/daemon/daemon-runtime-sync.js +212 -0
- package/dist/heart/daemon/daemon.js +261 -1
- package/dist/heart/daemon/hatch-animation.js +10 -3
- package/dist/heart/daemon/hatch-flow.js +7 -72
- package/dist/heart/daemon/hooks/bundle-meta.js +92 -0
- package/dist/heart/daemon/launchd.js +159 -0
- package/dist/heart/daemon/log-tailer.js +4 -3
- package/dist/heart/daemon/message-router.js +17 -8
- package/dist/heart/daemon/ouro-bot-global-installer.js +128 -0
- package/dist/heart/daemon/ouro-path-installer.js +57 -29
- package/dist/heart/daemon/ouro-version-manager.js +171 -0
- package/dist/heart/daemon/process-manager.js +13 -0
- package/dist/heart/daemon/run-hooks.js +37 -0
- package/dist/heart/daemon/runtime-logging.js +58 -15
- package/dist/heart/daemon/runtime-metadata.js +219 -0
- package/dist/heart/daemon/runtime-mode.js +67 -0
- package/dist/heart/daemon/sense-manager.js +50 -2
- package/dist/heart/daemon/skill-management-installer.js +94 -0
- package/dist/heart/daemon/socket-client.js +202 -0
- package/dist/heart/daemon/specialist-orchestrator.js +2 -2
- package/dist/heart/daemon/specialist-prompt.js +7 -4
- package/dist/heart/daemon/specialist-tools.js +52 -3
- package/dist/heart/daemon/staged-restart.js +114 -0
- package/dist/heart/daemon/thoughts.js +507 -0
- package/dist/heart/daemon/update-checker.js +111 -0
- package/dist/heart/daemon/update-hooks.js +138 -0
- package/dist/heart/daemon/wrapper-publish-guard.js +86 -0
- package/dist/heart/delegation.js +62 -0
- package/dist/heart/identity.js +64 -21
- package/dist/heart/kicks.js +1 -19
- package/dist/heart/model-capabilities.js +48 -0
- package/dist/heart/obligations.js +197 -0
- package/dist/heart/progress-story.js +42 -0
- package/dist/heart/provider-failover.js +88 -0
- package/dist/heart/provider-ping.js +159 -0
- package/dist/heart/providers/anthropic-token.js +163 -0
- package/dist/heart/providers/anthropic.js +195 -34
- package/dist/heart/providers/azure.js +115 -9
- package/dist/heart/providers/github-copilot.js +157 -0
- package/dist/heart/providers/minimax.js +33 -3
- package/dist/heart/providers/openai-codex.js +49 -14
- package/dist/heart/safe-workspace.js +381 -0
- package/dist/heart/session-activity.js +173 -0
- package/dist/heart/session-recall.js +216 -0
- package/dist/heart/streaming.js +108 -24
- package/dist/heart/target-resolution.js +123 -0
- package/dist/heart/tool-loop.js +194 -0
- package/dist/heart/turn-coordinator.js +28 -0
- package/dist/mind/associative-recall.js +14 -2
- package/dist/mind/bundle-manifest.js +12 -0
- package/dist/mind/context.js +60 -14
- package/dist/mind/first-impressions.js +16 -2
- package/dist/mind/friends/channel.js +35 -0
- package/dist/mind/friends/group-context.js +144 -0
- package/dist/mind/friends/store-file.js +19 -0
- package/dist/mind/friends/trust-explanation.js +74 -0
- package/dist/mind/friends/types.js +8 -0
- package/dist/mind/memory.js +27 -26
- package/dist/mind/obligation-steering.js +221 -0
- package/dist/mind/pending.js +76 -9
- package/dist/mind/phrases.js +1 -0
- package/dist/mind/prompt.js +456 -77
- package/dist/mind/token-estimate.js +8 -12
- package/dist/nerves/cli-logging.js +15 -2
- package/dist/nerves/coverage/run-artifacts.js +1 -1
- package/dist/nerves/index.js +12 -0
- package/dist/nerves/runtime.js +5 -1
- package/dist/repertoire/ado-client.js +4 -2
- package/dist/repertoire/coding/context-pack.js +254 -0
- package/dist/repertoire/coding/feedback.js +301 -0
- package/dist/repertoire/coding/index.js +4 -1
- package/dist/repertoire/coding/manager.js +210 -4
- package/dist/repertoire/coding/spawner.js +39 -9
- package/dist/repertoire/coding/tools.js +171 -4
- package/dist/repertoire/data/ado-endpoints.json +188 -0
- package/dist/repertoire/guardrails.js +290 -0
- package/dist/repertoire/mcp-client.js +254 -0
- package/dist/repertoire/mcp-manager.js +198 -0
- package/dist/repertoire/skills.js +3 -26
- package/dist/repertoire/tasks/board.js +12 -0
- package/dist/repertoire/tasks/index.js +23 -9
- package/dist/repertoire/tasks/transitions.js +1 -2
- package/dist/repertoire/tools-base.js +925 -250
- package/dist/repertoire/tools-bluebubbles.js +93 -0
- package/dist/repertoire/tools-teams.js +58 -25
- package/dist/repertoire/tools.js +106 -53
- package/dist/senses/bluebubbles-client.js +210 -5
- package/dist/senses/bluebubbles-entry.js +2 -0
- package/dist/senses/bluebubbles-inbound-log.js +109 -0
- package/dist/senses/bluebubbles-media.js +339 -0
- package/dist/senses/bluebubbles-model.js +12 -4
- package/dist/senses/bluebubbles-mutation-log.js +45 -5
- package/dist/senses/bluebubbles-runtime-state.js +109 -0
- package/dist/senses/bluebubbles-session-cleanup.js +72 -0
- package/dist/senses/bluebubbles.js +915 -45
- package/dist/senses/cli-layout.js +187 -0
- package/dist/senses/cli.js +374 -131
- package/dist/senses/continuity.js +94 -0
- package/dist/senses/debug-activity.js +154 -0
- package/dist/senses/inner-dialog-worker.js +47 -18
- package/dist/senses/inner-dialog.js +388 -83
- package/dist/senses/pipeline.js +444 -0
- package/dist/senses/teams.js +607 -129
- package/dist/senses/trust-gate.js +112 -2
- package/package.json +9 -3
- package/subagents/README.md +4 -70
- package/dist/heart/daemon/subagent-installer.js +0 -134
- package/subagents/work-doer.md +0 -233
- package/subagents/work-merger.md +0 -624
- package/subagents/work-planner.md +0 -373
|
@@ -1,13 +1,50 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
2
35
|
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
36
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
37
|
};
|
|
5
38
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
exports.toAnthropicMessages = toAnthropicMessages;
|
|
40
|
+
exports.classifyAnthropicError = classifyAnthropicError;
|
|
6
41
|
exports.createAnthropicProviderRuntime = createAnthropicProviderRuntime;
|
|
7
42
|
const sdk_1 = __importDefault(require("@anthropic-ai/sdk"));
|
|
8
43
|
const config_1 = require("../config");
|
|
9
44
|
const identity_1 = require("../identity");
|
|
10
45
|
const runtime_1 = require("../../nerves/runtime");
|
|
46
|
+
const streaming_1 = require("../streaming");
|
|
47
|
+
const model_capabilities_1 = require("../model-capabilities");
|
|
11
48
|
const ANTHROPIC_SETUP_TOKEN_PREFIX = "sk-ant-oat01-";
|
|
12
49
|
const ANTHROPIC_SETUP_TOKEN_MIN_LENGTH = 80;
|
|
13
50
|
const ANTHROPIC_OAUTH_BETA_HEADER = "claude-code-20250219,oauth-2025-04-20,fine-grained-tool-streaming-2025-05-14,interleaved-thinking-2025-05-14";
|
|
@@ -21,10 +58,10 @@ function getAnthropicSetupTokenInstructions() {
|
|
|
21
58
|
const agentName = getAnthropicAgentNameForGuidance();
|
|
22
59
|
return [
|
|
23
60
|
"Fix:",
|
|
24
|
-
` 1. Run \`
|
|
25
|
-
" (or run `claude setup-token` and paste the token manually)",
|
|
61
|
+
` 1. Run \`ouro auth --agent ${agentName}\``,
|
|
26
62
|
` 2. Open ${getAnthropicSecretsPathForGuidance()}`,
|
|
27
63
|
" 3. Confirm providers.anthropic.setupToken is set",
|
|
64
|
+
" 4. After reauth, retry the failed ouro command or reconnect this session.",
|
|
28
65
|
].join("\n");
|
|
29
66
|
}
|
|
30
67
|
function getAnthropicReauthGuidance(reason) {
|
|
@@ -92,6 +129,18 @@ function toAnthropicMessages(messages) {
|
|
|
92
129
|
if (msg.role === "assistant") {
|
|
93
130
|
const assistant = msg;
|
|
94
131
|
const blocks = [];
|
|
132
|
+
// Restore thinking blocks before text/tool_use blocks
|
|
133
|
+
const thinkingBlocks = assistant._thinking_blocks;
|
|
134
|
+
if (thinkingBlocks) {
|
|
135
|
+
for (const tb of thinkingBlocks) {
|
|
136
|
+
if (tb.type === "thinking") {
|
|
137
|
+
blocks.push({ type: "thinking", thinking: tb.thinking, signature: tb.signature });
|
|
138
|
+
}
|
|
139
|
+
else {
|
|
140
|
+
blocks.push({ type: "redacted_thinking", data: tb.data });
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
}
|
|
95
144
|
const text = toAnthropicTextContent(assistant.content);
|
|
96
145
|
if (text) {
|
|
97
146
|
blocks.push({ type: "text", text });
|
|
@@ -172,6 +221,29 @@ function mergeAnthropicToolArguments(current, partial) {
|
|
|
172
221
|
}
|
|
173
222
|
return current + partial;
|
|
174
223
|
}
|
|
224
|
+
/* v8 ignore start -- shared network error utility, tested via classification tests @preserve */
|
|
225
|
+
function isNetworkError(error) {
|
|
226
|
+
const code = error.code || "";
|
|
227
|
+
if (["ECONNRESET", "ECONNREFUSED", "ENOTFOUND", "ETIMEDOUT", "EPIPE",
|
|
228
|
+
"EAI_AGAIN", "EHOSTUNREACH", "ENETUNREACH", "ECONNABORTED"].includes(code))
|
|
229
|
+
return true;
|
|
230
|
+
const msg = error.message || "";
|
|
231
|
+
return msg.includes("fetch failed") || msg.includes("socket hang up") || msg.includes("getaddrinfo");
|
|
232
|
+
}
|
|
233
|
+
/* v8 ignore stop */
|
|
234
|
+
function classifyAnthropicError(error) {
|
|
235
|
+
const status = error.status;
|
|
236
|
+
if (status === 401 || status === 403 || isAnthropicAuthFailure(error))
|
|
237
|
+
return "auth-failure";
|
|
238
|
+
if (status === 429)
|
|
239
|
+
return "rate-limit";
|
|
240
|
+
if (status === 529 || (status && status >= 500))
|
|
241
|
+
return "server-error";
|
|
242
|
+
if (isNetworkError(error))
|
|
243
|
+
return "network-error";
|
|
244
|
+
return "unknown";
|
|
245
|
+
}
|
|
246
|
+
/* v8 ignore start -- auth detection: only called from classifyAnthropicError which always passes Error @preserve */
|
|
175
247
|
function isAnthropicAuthFailure(error) {
|
|
176
248
|
if (!(error instanceof Error))
|
|
177
249
|
return false;
|
|
@@ -184,40 +256,52 @@ function isAnthropicAuthFailure(error) {
|
|
|
184
256
|
lower.includes("unauthorized") ||
|
|
185
257
|
lower.includes("invalid api key"));
|
|
186
258
|
}
|
|
187
|
-
|
|
188
|
-
const base = error instanceof Error ? error.message : String(error);
|
|
189
|
-
if (isAnthropicAuthFailure(error)) {
|
|
190
|
-
return new Error(getAnthropicReauthGuidance(`Anthropic authentication failed (${base}).`));
|
|
191
|
-
}
|
|
192
|
-
return error instanceof Error ? error : new Error(String(error));
|
|
193
|
-
}
|
|
259
|
+
/* v8 ignore stop */
|
|
194
260
|
async function streamAnthropicMessages(client, model, request) {
|
|
195
261
|
const { system, messages } = toAnthropicMessages(request.messages);
|
|
196
262
|
const anthropicTools = toAnthropicTools(request.activeTools);
|
|
263
|
+
const modelCaps = (0, model_capabilities_1.getModelCapabilities)(model);
|
|
264
|
+
const maxTokens = modelCaps.maxOutputTokens ?? 16384;
|
|
197
265
|
const params = {
|
|
198
266
|
model,
|
|
199
|
-
max_tokens:
|
|
267
|
+
max_tokens: maxTokens,
|
|
200
268
|
messages,
|
|
201
269
|
stream: true,
|
|
270
|
+
thinking: { type: "adaptive" },
|
|
271
|
+
output_config: { effort: request.reasoningEffort ?? "medium" },
|
|
202
272
|
};
|
|
203
|
-
|
|
204
|
-
|
|
273
|
+
// The Anthropic API requires a Claude Code identification block in the system
|
|
274
|
+
// prompt when using OAuth setup tokens (sk-ant-oat01). Without it, Opus/Sonnet
|
|
275
|
+
// 4.6 requests are rejected with 400. This is the API's validation that the
|
|
276
|
+
// token is being used by a Claude Code client.
|
|
277
|
+
const claudeCodePreamble = { type: "text", text: "You are Claude Code, Anthropic's official CLI for Claude." };
|
|
278
|
+
if (system) {
|
|
279
|
+
params.system = [claudeCodePreamble, { type: "text", text: system }];
|
|
280
|
+
}
|
|
281
|
+
else {
|
|
282
|
+
params.system = [claudeCodePreamble];
|
|
283
|
+
}
|
|
205
284
|
if (anthropicTools.length > 0)
|
|
206
285
|
params.tools = anthropicTools;
|
|
207
286
|
if (request.toolChoiceRequired && anthropicTools.length > 0) {
|
|
208
|
-
|
|
287
|
+
// Thinking (adaptive or enabled) only supports tool_choice "auto" or "none".
|
|
288
|
+
// "any" forces tool use which is incompatible with extended thinking.
|
|
289
|
+
params.tool_choice = params.thinking ? { type: "auto" } : /* v8 ignore next -- no-thinking path: thinking always set for 4.6 models @preserve */ { type: "any" };
|
|
209
290
|
}
|
|
210
291
|
let response;
|
|
211
292
|
try {
|
|
212
293
|
response = await client.messages.create(params, request.signal ? { signal: request.signal } : {});
|
|
213
294
|
}
|
|
214
295
|
catch (error) {
|
|
215
|
-
throw
|
|
296
|
+
throw error instanceof Error ? error : new Error(String(error));
|
|
216
297
|
}
|
|
217
298
|
let content = "";
|
|
218
299
|
let streamStarted = false;
|
|
219
300
|
let usage;
|
|
220
301
|
const toolCalls = new Map();
|
|
302
|
+
const thinkingBlocks = new Map();
|
|
303
|
+
const redactedBlocks = new Map();
|
|
304
|
+
const answerStreamer = new streaming_1.FinalAnswerStreamer(request.callbacks, request.eagerFinalAnswerStreaming);
|
|
221
305
|
try {
|
|
222
306
|
for await (const event of response) {
|
|
223
307
|
if (request.signal?.aborted)
|
|
@@ -225,17 +309,29 @@ async function streamAnthropicMessages(client, model, request) {
|
|
|
225
309
|
const eventType = String(event.type ?? "");
|
|
226
310
|
if (eventType === "content_block_start") {
|
|
227
311
|
const block = event.content_block;
|
|
228
|
-
|
|
229
|
-
|
|
312
|
+
const index = Number(event.index);
|
|
313
|
+
if (block?.type === "thinking") {
|
|
314
|
+
thinkingBlocks.set(index, { type: "thinking", thinking: "", signature: "" });
|
|
315
|
+
}
|
|
316
|
+
else if (block?.type === "redacted_thinking") {
|
|
317
|
+
redactedBlocks.set(index, { type: "redacted_thinking", data: String(block.data ?? "") });
|
|
318
|
+
}
|
|
319
|
+
else if (block?.type === "tool_use") {
|
|
230
320
|
const rawInput = block.input;
|
|
231
321
|
const input = rawInput && typeof rawInput === "object"
|
|
232
322
|
? JSON.stringify(rawInput)
|
|
233
323
|
: "";
|
|
324
|
+
const name = String(block.name ?? "");
|
|
234
325
|
toolCalls.set(index, {
|
|
235
326
|
id: String(block.id ?? ""),
|
|
236
|
-
name
|
|
327
|
+
name,
|
|
237
328
|
arguments: input,
|
|
238
329
|
});
|
|
330
|
+
// Activate eager streaming for sole final_answer tool call
|
|
331
|
+
/* v8 ignore next -- final_answer streaming activation, tested via FinalAnswerStreamer unit tests @preserve */
|
|
332
|
+
if (name === "final_answer" && toolCalls.size === 1) {
|
|
333
|
+
answerStreamer.activate();
|
|
334
|
+
}
|
|
239
335
|
}
|
|
240
336
|
continue;
|
|
241
337
|
}
|
|
@@ -257,14 +353,31 @@ async function streamAnthropicMessages(client, model, request) {
|
|
|
257
353
|
request.callbacks.onModelStreamStart();
|
|
258
354
|
streamStarted = true;
|
|
259
355
|
}
|
|
260
|
-
|
|
356
|
+
const thinkingText = String(delta?.thinking ?? "");
|
|
357
|
+
request.callbacks.onReasoningChunk(thinkingText);
|
|
358
|
+
const thinkingIndex = Number(event.index);
|
|
359
|
+
const thinkingBlock = thinkingBlocks.get(thinkingIndex);
|
|
360
|
+
if (thinkingBlock)
|
|
361
|
+
thinkingBlock.thinking += thinkingText;
|
|
362
|
+
continue;
|
|
363
|
+
}
|
|
364
|
+
if (deltaType === "signature_delta") {
|
|
365
|
+
const sigIndex = Number(event.index);
|
|
366
|
+
const sigBlock = thinkingBlocks.get(sigIndex);
|
|
367
|
+
if (sigBlock)
|
|
368
|
+
sigBlock.signature += String(delta?.signature ?? "");
|
|
261
369
|
continue;
|
|
262
370
|
}
|
|
263
371
|
if (deltaType === "input_json_delta") {
|
|
264
372
|
const index = Number(event.index);
|
|
265
373
|
const existing = toolCalls.get(index);
|
|
266
374
|
if (existing) {
|
|
267
|
-
|
|
375
|
+
const partialJson = String(delta?.partial_json ?? "");
|
|
376
|
+
existing.arguments = mergeAnthropicToolArguments(existing.arguments, partialJson);
|
|
377
|
+
/* v8 ignore next -- final_answer delta streaming, tested via FinalAnswerStreamer unit tests @preserve */
|
|
378
|
+
if (existing.name === "final_answer" && toolCalls.size === 1) {
|
|
379
|
+
answerStreamer.processDelta(partialJson);
|
|
380
|
+
}
|
|
268
381
|
}
|
|
269
382
|
continue;
|
|
270
383
|
}
|
|
@@ -286,47 +399,95 @@ async function streamAnthropicMessages(client, model, request) {
|
|
|
286
399
|
}
|
|
287
400
|
}
|
|
288
401
|
catch (error) {
|
|
289
|
-
throw
|
|
402
|
+
throw error instanceof Error ? error : /* v8 ignore next -- defensive: stream errors are always Error @preserve */ new Error(String(error));
|
|
290
403
|
}
|
|
404
|
+
// Collect all thinking blocks (regular + redacted) sorted by index to preserve ordering
|
|
405
|
+
const allThinkingIndices = [...thinkingBlocks.keys(), ...redactedBlocks.keys()].sort((a, b) => a - b);
|
|
406
|
+
const outputItems = allThinkingIndices.map((idx) => {
|
|
407
|
+
const tb = thinkingBlocks.get(idx);
|
|
408
|
+
if (tb)
|
|
409
|
+
return tb;
|
|
410
|
+
return redactedBlocks.get(idx);
|
|
411
|
+
});
|
|
291
412
|
return {
|
|
292
413
|
content,
|
|
293
414
|
toolCalls: [...toolCalls.values()],
|
|
294
|
-
outputItems
|
|
415
|
+
outputItems,
|
|
295
416
|
usage,
|
|
417
|
+
finalAnswerStreamed: answerStreamer.streamed,
|
|
296
418
|
};
|
|
297
419
|
}
|
|
298
|
-
function createAnthropicProviderRuntime() {
|
|
420
|
+
function createAnthropicProviderRuntime(config) {
|
|
299
421
|
(0, runtime_1.emitNervesEvent)({
|
|
300
422
|
component: "engine",
|
|
301
423
|
event: "engine.provider_init",
|
|
302
424
|
message: "anthropic provider init",
|
|
303
425
|
meta: { provider: "anthropic" },
|
|
304
426
|
});
|
|
305
|
-
const anthropicConfig = (0, config_1.getAnthropicConfig)();
|
|
427
|
+
const anthropicConfig = config ?? (0, config_1.getAnthropicConfig)();
|
|
306
428
|
if (!(anthropicConfig.model && anthropicConfig.setupToken)) {
|
|
307
429
|
throw new Error(getAnthropicReauthGuidance("provider 'anthropic' is selected in agent.json but providers.anthropic.model/setupToken is incomplete in secrets.json."));
|
|
308
430
|
}
|
|
431
|
+
const modelCaps = (0, model_capabilities_1.getModelCapabilities)(anthropicConfig.model);
|
|
432
|
+
const capabilities = new Set();
|
|
433
|
+
if (modelCaps.reasoningEffort)
|
|
434
|
+
capabilities.add("reasoning-effort");
|
|
309
435
|
const credential = resolveAnthropicSetupTokenCredential();
|
|
310
|
-
const
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
436
|
+
const fullConfig = config ?? (0, config_1.getAnthropicConfig)();
|
|
437
|
+
const refreshToken = fullConfig.refreshToken;
|
|
438
|
+
const expiresAt = fullConfig.expiresAt;
|
|
439
|
+
function createClient(token) {
|
|
440
|
+
return new sdk_1.default({
|
|
441
|
+
authToken: token,
|
|
442
|
+
timeout: 30000,
|
|
443
|
+
maxRetries: 0,
|
|
444
|
+
defaultHeaders: {
|
|
445
|
+
"anthropic-beta": ANTHROPIC_OAUTH_BETA_HEADER,
|
|
446
|
+
"anthropic-dangerous-direct-browser-access": "true",
|
|
447
|
+
"user-agent": "claude-cli/2.1.2 (external, cli)",
|
|
448
|
+
"x-app": "cli",
|
|
449
|
+
},
|
|
450
|
+
});
|
|
451
|
+
}
|
|
452
|
+
let currentToken = credential.token;
|
|
453
|
+
let client = createClient(currentToken);
|
|
454
|
+
/* v8 ignore start -- token refresh: dynamic import + ensureFreshToken, tested via integration @preserve */
|
|
455
|
+
async function ensureClient() {
|
|
456
|
+
try {
|
|
457
|
+
const { ensureFreshToken } = await Promise.resolve().then(() => __importStar(require("./anthropic-token")));
|
|
458
|
+
const { getAgentName } = await Promise.resolve().then(() => __importStar(require("../identity")));
|
|
459
|
+
const freshToken = await ensureFreshToken(currentToken, refreshToken, expiresAt, getAgentName());
|
|
460
|
+
if (freshToken !== currentToken) {
|
|
461
|
+
currentToken = freshToken;
|
|
462
|
+
client = createClient(freshToken);
|
|
463
|
+
}
|
|
464
|
+
}
|
|
465
|
+
catch {
|
|
466
|
+
// refresh failed — use existing client
|
|
467
|
+
}
|
|
468
|
+
return client;
|
|
469
|
+
}
|
|
470
|
+
/* v8 ignore stop */
|
|
318
471
|
return {
|
|
319
472
|
id: "anthropic",
|
|
320
473
|
model: anthropicConfig.model,
|
|
321
|
-
client
|
|
474
|
+
/* v8 ignore next -- getter: returns mutable client ref @preserve */
|
|
475
|
+
get client() { return client; },
|
|
476
|
+
capabilities,
|
|
477
|
+
supportedReasoningEfforts: modelCaps.reasoningEffort,
|
|
322
478
|
resetTurnState(_messages) {
|
|
323
479
|
// Anthropic request payload is derived from canonical messages each turn.
|
|
324
480
|
},
|
|
325
481
|
appendToolOutput(_callId, _output) {
|
|
326
482
|
// Anthropic uses canonical messages for tool_result tracking.
|
|
327
483
|
},
|
|
328
|
-
streamTurn(request) {
|
|
329
|
-
|
|
484
|
+
async streamTurn(request) {
|
|
485
|
+
const freshClient = await ensureClient();
|
|
486
|
+
return streamAnthropicMessages(freshClient, anthropicConfig.model, request);
|
|
487
|
+
},
|
|
488
|
+
/* v8 ignore next 3 -- delegation: classification logic tested via classifyAnthropicError @preserve */
|
|
489
|
+
classifyError(error) {
|
|
490
|
+
return classifyAnthropicError(error);
|
|
330
491
|
},
|
|
331
492
|
};
|
|
332
493
|
}
|
|
@@ -1,35 +1,137 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
2
35
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
|
+
exports.classifyAzureError = classifyAzureError;
|
|
37
|
+
exports.createAzureTokenProvider = createAzureTokenProvider;
|
|
3
38
|
exports.createAzureProviderRuntime = createAzureProviderRuntime;
|
|
4
39
|
const openai_1 = require("openai");
|
|
5
40
|
const config_1 = require("../config");
|
|
6
41
|
const runtime_1 = require("../../nerves/runtime");
|
|
7
42
|
const streaming_1 = require("../streaming");
|
|
8
|
-
|
|
43
|
+
const model_capabilities_1 = require("../model-capabilities");
|
|
44
|
+
const COGNITIVE_SERVICES_SCOPE = "https://cognitiveservices.azure.com/.default";
|
|
45
|
+
/* v8 ignore start -- shared network error utility, tested via classification tests @preserve */
|
|
46
|
+
function isNetworkError(error) {
|
|
47
|
+
const code = error.code || "";
|
|
48
|
+
if (["ECONNRESET", "ECONNREFUSED", "ENOTFOUND", "ETIMEDOUT", "EPIPE",
|
|
49
|
+
"EAI_AGAIN", "EHOSTUNREACH", "ENETUNREACH", "ECONNABORTED"].includes(code))
|
|
50
|
+
return true;
|
|
51
|
+
const msg = error.message || "";
|
|
52
|
+
return msg.includes("fetch failed") || msg.includes("socket hang up") || msg.includes("getaddrinfo");
|
|
53
|
+
}
|
|
54
|
+
/* v8 ignore stop */
|
|
55
|
+
function classifyAzureError(error) {
|
|
56
|
+
const status = error.status;
|
|
57
|
+
if (status === 401 || status === 403)
|
|
58
|
+
return "auth-failure";
|
|
59
|
+
if (status === 429)
|
|
60
|
+
return "rate-limit";
|
|
61
|
+
if (status && status >= 500)
|
|
62
|
+
return "server-error";
|
|
63
|
+
if (isNetworkError(error))
|
|
64
|
+
return "network-error";
|
|
65
|
+
return "unknown";
|
|
66
|
+
}
|
|
67
|
+
// @azure/identity is imported dynamically (below) rather than at the top level
|
|
68
|
+
// because it's a heavy package (~30+ transitive deps) and we only need it when
|
|
69
|
+
// using the managed-identity auth path. API-key users and other providers
|
|
70
|
+
// shouldn't pay the cold-start cost.
|
|
71
|
+
function createAzureTokenProvider(managedIdentityClientId) {
|
|
72
|
+
let credential = null;
|
|
73
|
+
return async () => {
|
|
74
|
+
try {
|
|
75
|
+
if (!credential) {
|
|
76
|
+
const { DefaultAzureCredential } = await Promise.resolve().then(() => __importStar(require("@azure/identity")));
|
|
77
|
+
const credentialOptions = managedIdentityClientId
|
|
78
|
+
? { managedIdentityClientId }
|
|
79
|
+
: undefined;
|
|
80
|
+
credential = new DefaultAzureCredential(credentialOptions);
|
|
81
|
+
}
|
|
82
|
+
const tokenResponse = await credential.getToken(COGNITIVE_SERVICES_SCOPE);
|
|
83
|
+
return tokenResponse.token;
|
|
84
|
+
}
|
|
85
|
+
catch (err) {
|
|
86
|
+
const detail = err instanceof Error ? err.message : String(err);
|
|
87
|
+
throw new Error(`Azure OpenAI authentication failed: ${detail}\n` +
|
|
88
|
+
"To fix this, either:\n" +
|
|
89
|
+
" 1. Set providers.azure.apiKey in secrets.json, or\n" +
|
|
90
|
+
" 2. Run 'az login' to authenticate with your Azure account (for local dev), or\n" +
|
|
91
|
+
" 3. Attach a managed identity to your App Service and set providers.azure.managedIdentityClientId in secrets.json (for deployed environments)");
|
|
92
|
+
}
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
function createAzureProviderRuntime(config) {
|
|
96
|
+
const azureConfig = config ?? (0, config_1.getAzureConfig)();
|
|
97
|
+
const useApiKey = !!azureConfig.apiKey;
|
|
98
|
+
const authMethod = useApiKey ? "key" : "managed-identity";
|
|
9
99
|
(0, runtime_1.emitNervesEvent)({
|
|
10
100
|
component: "engine",
|
|
11
101
|
event: "engine.provider_init",
|
|
12
102
|
message: "azure provider init",
|
|
13
|
-
meta: { provider: "azure" },
|
|
103
|
+
meta: { provider: "azure", authMethod },
|
|
14
104
|
});
|
|
15
|
-
|
|
16
|
-
if (!(azureConfig.apiKey && azureConfig.endpoint && azureConfig.deployment && azureConfig.modelName)) {
|
|
105
|
+
if (!(azureConfig.endpoint && azureConfig.deployment && azureConfig.modelName)) {
|
|
17
106
|
throw new Error("provider 'azure' is selected in agent.json but providers.azure is incomplete in secrets.json.");
|
|
18
107
|
}
|
|
19
|
-
const
|
|
20
|
-
|
|
108
|
+
const modelCaps = (0, model_capabilities_1.getModelCapabilities)(azureConfig.modelName);
|
|
109
|
+
const capabilities = new Set();
|
|
110
|
+
if (modelCaps.reasoningEffort)
|
|
111
|
+
capabilities.add("reasoning-effort");
|
|
112
|
+
const clientOptions = {
|
|
21
113
|
endpoint: azureConfig.endpoint.replace(/\/openai.*$/, ""),
|
|
22
114
|
deployment: azureConfig.deployment,
|
|
23
115
|
apiVersion: azureConfig.apiVersion,
|
|
24
116
|
timeout: 30000,
|
|
25
117
|
maxRetries: 0,
|
|
26
|
-
}
|
|
118
|
+
};
|
|
119
|
+
if (useApiKey) {
|
|
120
|
+
clientOptions.apiKey = azureConfig.apiKey;
|
|
121
|
+
}
|
|
122
|
+
else {
|
|
123
|
+
const managedIdentityClientId = azureConfig.managedIdentityClientId || undefined;
|
|
124
|
+
clientOptions.azureADTokenProvider = createAzureTokenProvider(managedIdentityClientId);
|
|
125
|
+
}
|
|
126
|
+
const client = new openai_1.AzureOpenAI(clientOptions);
|
|
27
127
|
let nativeInput = null;
|
|
28
128
|
let nativeInstructions = "";
|
|
29
129
|
return {
|
|
30
130
|
id: "azure",
|
|
31
131
|
model: azureConfig.modelName,
|
|
32
132
|
client,
|
|
133
|
+
capabilities,
|
|
134
|
+
supportedReasoningEfforts: modelCaps.reasoningEffort,
|
|
33
135
|
resetTurnState(messages) {
|
|
34
136
|
const { instructions, input } = (0, streaming_1.toResponsesInput)(messages);
|
|
35
137
|
nativeInput = input;
|
|
@@ -48,7 +150,7 @@ function createAzureProviderRuntime() {
|
|
|
48
150
|
input: nativeInput,
|
|
49
151
|
instructions: nativeInstructions,
|
|
50
152
|
tools: (0, streaming_1.toResponsesTools)(request.activeTools),
|
|
51
|
-
reasoning: { effort: "medium", summary: "detailed" },
|
|
153
|
+
reasoning: { effort: request.reasoningEffort ?? "medium", summary: "detailed" },
|
|
52
154
|
stream: true,
|
|
53
155
|
store: false,
|
|
54
156
|
include: ["reasoning.encrypted_content"],
|
|
@@ -57,10 +159,14 @@ function createAzureProviderRuntime() {
|
|
|
57
159
|
params.metadata = { trace_id: request.traceId };
|
|
58
160
|
if (request.toolChoiceRequired)
|
|
59
161
|
params.tool_choice = "required";
|
|
60
|
-
const result = await (0, streaming_1.streamResponsesApi)(this.client, params, request.callbacks, request.signal);
|
|
162
|
+
const result = await (0, streaming_1.streamResponsesApi)(this.client, params, request.callbacks, request.signal, request.eagerFinalAnswerStreaming);
|
|
61
163
|
for (const item of result.outputItems)
|
|
62
164
|
nativeInput.push(item);
|
|
63
165
|
return result;
|
|
64
166
|
},
|
|
167
|
+
/* v8 ignore next 3 -- delegation: classification logic tested via classifyAzureError @preserve */
|
|
168
|
+
classifyError(error) {
|
|
169
|
+
return classifyAzureError(error);
|
|
170
|
+
},
|
|
65
171
|
};
|
|
66
172
|
}
|