@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.
Files changed (126) hide show
  1. package/AdoptionSpecialist.ouro/psyche/SOUL.md +2 -2
  2. package/AdoptionSpecialist.ouro/psyche/identities/monty.md +2 -2
  3. package/README.md +147 -205
  4. package/changelog.json +814 -0
  5. package/dist/heart/active-work.js +622 -0
  6. package/dist/heart/bridges/manager.js +358 -0
  7. package/dist/heart/bridges/state-machine.js +135 -0
  8. package/dist/heart/bridges/store.js +123 -0
  9. package/dist/heart/commitments.js +105 -0
  10. package/dist/heart/config.js +66 -21
  11. package/dist/heart/core.js +518 -100
  12. package/dist/heart/cross-chat-delivery.js +146 -0
  13. package/dist/heart/daemon/agent-discovery.js +81 -0
  14. package/dist/heart/daemon/auth-flow.js +457 -0
  15. package/dist/heart/daemon/daemon-cli.js +1516 -195
  16. package/dist/heart/daemon/daemon-entry.js +43 -2
  17. package/dist/heart/daemon/daemon-runtime-sync.js +212 -0
  18. package/dist/heart/daemon/daemon.js +261 -1
  19. package/dist/heart/daemon/hatch-animation.js +10 -3
  20. package/dist/heart/daemon/hatch-flow.js +7 -72
  21. package/dist/heart/daemon/hooks/bundle-meta.js +92 -0
  22. package/dist/heart/daemon/launchd.js +159 -0
  23. package/dist/heart/daemon/log-tailer.js +4 -3
  24. package/dist/heart/daemon/message-router.js +17 -8
  25. package/dist/heart/daemon/ouro-bot-global-installer.js +128 -0
  26. package/dist/heart/daemon/ouro-path-installer.js +57 -29
  27. package/dist/heart/daemon/ouro-version-manager.js +171 -0
  28. package/dist/heart/daemon/process-manager.js +13 -0
  29. package/dist/heart/daemon/run-hooks.js +37 -0
  30. package/dist/heart/daemon/runtime-logging.js +58 -15
  31. package/dist/heart/daemon/runtime-metadata.js +219 -0
  32. package/dist/heart/daemon/runtime-mode.js +67 -0
  33. package/dist/heart/daemon/sense-manager.js +50 -2
  34. package/dist/heart/daemon/skill-management-installer.js +94 -0
  35. package/dist/heart/daemon/socket-client.js +202 -0
  36. package/dist/heart/daemon/specialist-orchestrator.js +2 -2
  37. package/dist/heart/daemon/specialist-prompt.js +7 -4
  38. package/dist/heart/daemon/specialist-tools.js +52 -3
  39. package/dist/heart/daemon/staged-restart.js +114 -0
  40. package/dist/heart/daemon/thoughts.js +507 -0
  41. package/dist/heart/daemon/update-checker.js +111 -0
  42. package/dist/heart/daemon/update-hooks.js +138 -0
  43. package/dist/heart/daemon/wrapper-publish-guard.js +86 -0
  44. package/dist/heart/delegation.js +62 -0
  45. package/dist/heart/identity.js +64 -21
  46. package/dist/heart/kicks.js +1 -19
  47. package/dist/heart/model-capabilities.js +48 -0
  48. package/dist/heart/obligations.js +197 -0
  49. package/dist/heart/progress-story.js +42 -0
  50. package/dist/heart/provider-failover.js +88 -0
  51. package/dist/heart/provider-ping.js +159 -0
  52. package/dist/heart/providers/anthropic-token.js +163 -0
  53. package/dist/heart/providers/anthropic.js +195 -34
  54. package/dist/heart/providers/azure.js +115 -9
  55. package/dist/heart/providers/github-copilot.js +157 -0
  56. package/dist/heart/providers/minimax.js +33 -3
  57. package/dist/heart/providers/openai-codex.js +49 -14
  58. package/dist/heart/safe-workspace.js +381 -0
  59. package/dist/heart/session-activity.js +173 -0
  60. package/dist/heart/session-recall.js +216 -0
  61. package/dist/heart/streaming.js +108 -24
  62. package/dist/heart/target-resolution.js +123 -0
  63. package/dist/heart/tool-loop.js +194 -0
  64. package/dist/heart/turn-coordinator.js +28 -0
  65. package/dist/mind/associative-recall.js +14 -2
  66. package/dist/mind/bundle-manifest.js +12 -0
  67. package/dist/mind/context.js +60 -14
  68. package/dist/mind/first-impressions.js +16 -2
  69. package/dist/mind/friends/channel.js +35 -0
  70. package/dist/mind/friends/group-context.js +144 -0
  71. package/dist/mind/friends/store-file.js +19 -0
  72. package/dist/mind/friends/trust-explanation.js +74 -0
  73. package/dist/mind/friends/types.js +8 -0
  74. package/dist/mind/memory.js +27 -26
  75. package/dist/mind/obligation-steering.js +221 -0
  76. package/dist/mind/pending.js +76 -9
  77. package/dist/mind/phrases.js +1 -0
  78. package/dist/mind/prompt.js +456 -77
  79. package/dist/mind/token-estimate.js +8 -12
  80. package/dist/nerves/cli-logging.js +15 -2
  81. package/dist/nerves/coverage/run-artifacts.js +1 -1
  82. package/dist/nerves/index.js +12 -0
  83. package/dist/nerves/runtime.js +5 -1
  84. package/dist/repertoire/ado-client.js +4 -2
  85. package/dist/repertoire/coding/context-pack.js +254 -0
  86. package/dist/repertoire/coding/feedback.js +301 -0
  87. package/dist/repertoire/coding/index.js +4 -1
  88. package/dist/repertoire/coding/manager.js +210 -4
  89. package/dist/repertoire/coding/spawner.js +39 -9
  90. package/dist/repertoire/coding/tools.js +171 -4
  91. package/dist/repertoire/data/ado-endpoints.json +188 -0
  92. package/dist/repertoire/guardrails.js +290 -0
  93. package/dist/repertoire/mcp-client.js +254 -0
  94. package/dist/repertoire/mcp-manager.js +198 -0
  95. package/dist/repertoire/skills.js +3 -26
  96. package/dist/repertoire/tasks/board.js +12 -0
  97. package/dist/repertoire/tasks/index.js +23 -9
  98. package/dist/repertoire/tasks/transitions.js +1 -2
  99. package/dist/repertoire/tools-base.js +925 -250
  100. package/dist/repertoire/tools-bluebubbles.js +93 -0
  101. package/dist/repertoire/tools-teams.js +58 -25
  102. package/dist/repertoire/tools.js +106 -53
  103. package/dist/senses/bluebubbles-client.js +210 -5
  104. package/dist/senses/bluebubbles-entry.js +2 -0
  105. package/dist/senses/bluebubbles-inbound-log.js +109 -0
  106. package/dist/senses/bluebubbles-media.js +339 -0
  107. package/dist/senses/bluebubbles-model.js +12 -4
  108. package/dist/senses/bluebubbles-mutation-log.js +45 -5
  109. package/dist/senses/bluebubbles-runtime-state.js +109 -0
  110. package/dist/senses/bluebubbles-session-cleanup.js +72 -0
  111. package/dist/senses/bluebubbles.js +915 -45
  112. package/dist/senses/cli-layout.js +187 -0
  113. package/dist/senses/cli.js +374 -131
  114. package/dist/senses/continuity.js +94 -0
  115. package/dist/senses/debug-activity.js +154 -0
  116. package/dist/senses/inner-dialog-worker.js +47 -18
  117. package/dist/senses/inner-dialog.js +388 -83
  118. package/dist/senses/pipeline.js +444 -0
  119. package/dist/senses/teams.js +607 -129
  120. package/dist/senses/trust-gate.js +112 -2
  121. package/package.json +9 -3
  122. package/subagents/README.md +4 -70
  123. package/dist/heart/daemon/subagent-installer.js +0 -134
  124. package/subagents/work-doer.md +0 -233
  125. package/subagents/work-merger.md +0 -624
  126. 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 \`npm run auth:claude-setup-token -- --agent ${agentName}\``,
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
- function withAnthropicAuthGuidance(error) {
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: 4096,
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
- if (system)
204
- params.system = system;
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
- params.tool_choice = { type: "any" };
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 withAnthropicAuthGuidance(error);
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
- if (block?.type === "tool_use") {
229
- const index = Number(event.index);
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: String(block.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
- request.callbacks.onReasoningChunk(String(delta?.thinking ?? ""));
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
- existing.arguments = mergeAnthropicToolArguments(existing.arguments, String(delta?.partial_json ?? ""));
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 withAnthropicAuthGuidance(error);
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 client = new sdk_1.default({
311
- authToken: credential.token,
312
- timeout: 30000,
313
- maxRetries: 0,
314
- defaultHeaders: {
315
- "anthropic-beta": ANTHROPIC_OAUTH_BETA_HEADER,
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
- return streamAnthropicMessages(client, anthropicConfig.model, request);
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
- function createAzureProviderRuntime() {
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
- const azureConfig = (0, config_1.getAzureConfig)();
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 client = new openai_1.AzureOpenAI({
20
- apiKey: azureConfig.apiKey,
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
  }