@ouro.bot/cli 0.1.0-alpha.13 → 0.1.0-alpha.130

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 +808 -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 +432 -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
@@ -0,0 +1,138 @@
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
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.registerUpdateHook = registerUpdateHook;
37
+ exports.getRegisteredHooks = getRegisteredHooks;
38
+ exports.clearRegisteredHooks = clearRegisteredHooks;
39
+ exports.applyPendingUpdates = applyPendingUpdates;
40
+ const fs = __importStar(require("fs"));
41
+ const path = __importStar(require("path"));
42
+ const semver = __importStar(require("semver"));
43
+ const runtime_1 = require("../../nerves/runtime");
44
+ const _hooks = [];
45
+ function registerUpdateHook(hook) {
46
+ _hooks.push(hook);
47
+ (0, runtime_1.emitNervesEvent)({
48
+ component: "daemon",
49
+ event: "daemon.update_hook_registered",
50
+ message: "registered update hook",
51
+ meta: {},
52
+ });
53
+ }
54
+ function getRegisteredHooks() {
55
+ return _hooks;
56
+ }
57
+ function clearRegisteredHooks() {
58
+ _hooks.length = 0;
59
+ }
60
+ async function applyPendingUpdates(bundlesRoot, currentVersion) {
61
+ const summary = { updated: [] };
62
+ (0, runtime_1.emitNervesEvent)({
63
+ component: "daemon",
64
+ event: "daemon.apply_pending_updates_start",
65
+ message: "applying pending updates",
66
+ meta: { bundlesRoot, currentVersion },
67
+ });
68
+ if (!fs.existsSync(bundlesRoot)) {
69
+ return summary;
70
+ }
71
+ let entries;
72
+ try {
73
+ entries = fs.readdirSync(bundlesRoot, { withFileTypes: true });
74
+ }
75
+ catch {
76
+ return summary;
77
+ }
78
+ for (const entry of entries) {
79
+ if (!entry.isDirectory() || !entry.name.endsWith(".ouro"))
80
+ continue;
81
+ const agentRoot = path.join(bundlesRoot, entry.name);
82
+ let previousVersion;
83
+ const metaPath = path.join(agentRoot, "bundle-meta.json");
84
+ try {
85
+ if (fs.existsSync(metaPath)) {
86
+ const raw = fs.readFileSync(metaPath, "utf-8");
87
+ const meta = JSON.parse(raw);
88
+ previousVersion = meta.runtimeVersion;
89
+ if (previousVersion === currentVersion) {
90
+ continue;
91
+ }
92
+ // Skip downgrades — only update forward
93
+ if (semver.valid(previousVersion) && semver.valid(currentVersion) && semver.gte(previousVersion, currentVersion)) {
94
+ (0, runtime_1.emitNervesEvent)({
95
+ component: "daemon",
96
+ event: "daemon.update_hook_skip_downgrade",
97
+ message: "skipping downgrade",
98
+ meta: { agentRoot, previousVersion, currentVersion },
99
+ });
100
+ continue;
101
+ }
102
+ }
103
+ }
104
+ catch {
105
+ // Malformed or unreadable bundle-meta.json -- treat as needing update
106
+ previousVersion = undefined;
107
+ }
108
+ const ctx = { agentRoot, currentVersion, previousVersion };
109
+ for (const hook of _hooks) {
110
+ try {
111
+ await hook(ctx);
112
+ }
113
+ catch (err) {
114
+ (0, runtime_1.emitNervesEvent)({
115
+ component: "daemon",
116
+ event: "daemon.update_hook_error",
117
+ message: "update hook threw",
118
+ meta: {
119
+ agentRoot,
120
+ error: err instanceof Error ? err.message : /* v8 ignore next -- defensive: non-Error catch branch @preserve */ String(err),
121
+ },
122
+ });
123
+ }
124
+ }
125
+ summary.updated.push({
126
+ agent: entry.name.replace(/\.ouro$/, ""),
127
+ from: previousVersion,
128
+ to: currentVersion,
129
+ });
130
+ }
131
+ (0, runtime_1.emitNervesEvent)({
132
+ component: "daemon",
133
+ event: "daemon.apply_pending_updates_end",
134
+ message: "pending updates applied",
135
+ meta: { bundlesRoot },
136
+ });
137
+ return summary;
138
+ }
@@ -0,0 +1,86 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.assessWrapperPublishSync = assessWrapperPublishSync;
4
+ const runtime_1 = require("../../nerves/runtime");
5
+ function wrapperPackageChanged(changedFiles) {
6
+ return changedFiles.some((file) => file.startsWith("packages/ouro.bot/"));
7
+ }
8
+ function assessWrapperPublishSync(input) {
9
+ let result;
10
+ if (input.localVersion !== input.cliVersion) {
11
+ result = {
12
+ ok: false,
13
+ message: `ouro.bot wrapper version ${input.localVersion} must match @ouro.bot/cli version ${input.cliVersion}`,
14
+ };
15
+ (0, runtime_1.emitNervesEvent)({
16
+ level: "warn",
17
+ component: "daemon",
18
+ event: "daemon.wrapper_publish_guard_checked",
19
+ message: "evaluated wrapper publish sync",
20
+ meta: {
21
+ changed: wrapperPackageChanged(input.changedFiles),
22
+ localVersion: input.localVersion,
23
+ cliVersion: input.cliVersion,
24
+ publishedVersion: input.publishedVersion,
25
+ ok: result.ok,
26
+ },
27
+ });
28
+ return result;
29
+ }
30
+ if (!wrapperPackageChanged(input.changedFiles)) {
31
+ result = {
32
+ ok: true,
33
+ message: "wrapper package unchanged",
34
+ };
35
+ (0, runtime_1.emitNervesEvent)({
36
+ component: "daemon",
37
+ event: "daemon.wrapper_publish_guard_checked",
38
+ message: "evaluated wrapper publish sync",
39
+ meta: {
40
+ changed: false,
41
+ localVersion: input.localVersion,
42
+ cliVersion: input.cliVersion,
43
+ publishedVersion: input.publishedVersion,
44
+ ok: result.ok,
45
+ },
46
+ });
47
+ return result;
48
+ }
49
+ if (input.publishedVersion === input.localVersion) {
50
+ result = {
51
+ ok: false,
52
+ message: `ouro.bot wrapper changed but ouro.bot@${input.localVersion} is already published; bump packages/ouro.bot/package.json before merging`,
53
+ };
54
+ (0, runtime_1.emitNervesEvent)({
55
+ level: "warn",
56
+ component: "daemon",
57
+ event: "daemon.wrapper_publish_guard_checked",
58
+ message: "evaluated wrapper publish sync",
59
+ meta: {
60
+ changed: true,
61
+ localVersion: input.localVersion,
62
+ cliVersion: input.cliVersion,
63
+ publishedVersion: input.publishedVersion,
64
+ ok: result.ok,
65
+ },
66
+ });
67
+ return result;
68
+ }
69
+ result = {
70
+ ok: true,
71
+ message: "wrapper package changed and local wrapper version is unpublished",
72
+ };
73
+ (0, runtime_1.emitNervesEvent)({
74
+ component: "daemon",
75
+ event: "daemon.wrapper_publish_guard_checked",
76
+ message: "evaluated wrapper publish sync",
77
+ meta: {
78
+ changed: true,
79
+ localVersion: input.localVersion,
80
+ cliVersion: input.cliVersion,
81
+ publishedVersion: input.publishedVersion,
82
+ ok: result.ok,
83
+ },
84
+ });
85
+ return result;
86
+ }
@@ -0,0 +1,62 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.decideDelegation = decideDelegation;
4
+ const runtime_1 = require("../nerves/runtime");
5
+ const CROSS_SESSION_TOOLS = new Set(["query_session", "send_message", "bridge_manage"]);
6
+ const FAST_PATH_TOOLS = new Set(["final_answer"]);
7
+ const REFLECTION_PATTERN = /\b(think|reflect|ponder|surface|surfaces|surfaced|sit with|metaboli[sz]e)\b/i;
8
+ const CROSS_SESSION_PATTERN = /\b(other chat|other session|across chats?|across sessions?|keep .* aligned|relay|carry .* across)\b/i;
9
+ function hasExplicitReflection(ingressTexts) {
10
+ return ingressTexts.some((text) => REFLECTION_PATTERN.test(text));
11
+ }
12
+ function hasCrossSessionPressure(ingressTexts, requestedToolNames) {
13
+ if (requestedToolNames.some((name) => CROSS_SESSION_TOOLS.has(name))) {
14
+ return true;
15
+ }
16
+ return ingressTexts.some((text) => CROSS_SESSION_PATTERN.test(text));
17
+ }
18
+ function hasNonFastPathToolRequest(requestedToolNames) {
19
+ return requestedToolNames.some((name) => !FAST_PATH_TOOLS.has(name));
20
+ }
21
+ function decideDelegation(input) {
22
+ const requestedToolNames = (input.requestedToolNames ?? [])
23
+ .map((name) => name.trim())
24
+ .filter((name) => name.length > 0);
25
+ const reasons = [];
26
+ if (hasExplicitReflection(input.ingressTexts)) {
27
+ reasons.push("explicit_reflection");
28
+ }
29
+ if (hasCrossSessionPressure(input.ingressTexts, requestedToolNames)) {
30
+ reasons.push("cross_session");
31
+ }
32
+ if (input.activeWork.centerOfGravity === "shared-work" || input.activeWork.bridges.some((bridge) => bridge.lifecycle === "active")) {
33
+ reasons.push("bridge_state");
34
+ }
35
+ if (input.activeWork.taskPressure.liveTaskNames.length > 0) {
36
+ reasons.push("task_state");
37
+ }
38
+ if (hasNonFastPathToolRequest(requestedToolNames)) {
39
+ reasons.push("non_fast_path_tool");
40
+ }
41
+ if (input.mustResolveBeforeHandoff || input.activeWork.mustResolveBeforeHandoff) {
42
+ reasons.push("unresolved_obligation");
43
+ }
44
+ const target = reasons.length === 0 ? "fast-path" : "delegate-inward";
45
+ const decision = {
46
+ target,
47
+ reasons,
48
+ outwardClosureRequired: target === "delegate-inward" && input.channel !== "inner",
49
+ };
50
+ (0, runtime_1.emitNervesEvent)({
51
+ component: "engine",
52
+ event: "engine.delegation_decide",
53
+ message: "computed delegation hint",
54
+ meta: {
55
+ channel: input.channel,
56
+ target: decision.target,
57
+ reasons: decision.reasons,
58
+ outwardClosureRequired: decision.outwardClosureRequired,
59
+ },
60
+ });
61
+ return decision;
62
+ }
@@ -33,16 +33,24 @@ var __importStar = (this && this.__importStar) || (function () {
33
33
  };
34
34
  })();
35
35
  Object.defineProperty(exports, "__esModule", { value: true });
36
- exports.DEFAULT_AGENT_SENSES = exports.DEFAULT_AGENT_PHRASES = exports.DEFAULT_AGENT_CONTEXT = void 0;
36
+ exports.HARNESS_CANONICAL_REPO_URL = exports.DEFAULT_AGENT_SENSES = exports.DEFAULT_AGENT_PHRASES = exports.DEFAULT_AGENT_CONTEXT = void 0;
37
37
  exports.buildDefaultAgentTemplate = buildDefaultAgentTemplate;
38
38
  exports.getAgentName = getAgentName;
39
39
  exports.getRepoRoot = getRepoRoot;
40
40
  exports.getAgentBundlesRoot = getAgentBundlesRoot;
41
41
  exports.getAgentRoot = getAgentRoot;
42
+ exports.getAgentStateRoot = getAgentStateRoot;
43
+ exports.getAgentRepoWorkspacesRoot = getAgentRepoWorkspacesRoot;
44
+ exports.getAgentDaemonStateRoot = getAgentDaemonStateRoot;
45
+ exports.getAgentDaemonLogsDir = getAgentDaemonLogsDir;
46
+ exports.getAgentDaemonLoggingConfigPath = getAgentDaemonLoggingConfigPath;
47
+ exports.getAgentMessagesRoot = getAgentMessagesRoot;
48
+ exports.getAgentToolsRoot = getAgentToolsRoot;
42
49
  exports.getAgentSecretsPath = getAgentSecretsPath;
43
50
  exports.loadAgentConfig = loadAgentConfig;
44
51
  exports.setAgentName = setAgentName;
45
52
  exports.setAgentConfigOverride = setAgentConfigOverride;
53
+ exports.resetAgentConfigCache = resetAgentConfigCache;
46
54
  exports.resetIdentity = resetIdentity;
47
55
  const fs = __importStar(require("fs"));
48
56
  const os = __importStar(require("os"));
@@ -132,7 +140,6 @@ function buildDefaultAgentTemplate(_agentName) {
132
140
  };
133
141
  }
134
142
  let _cachedAgentName = null;
135
- let _cachedAgentConfig = null;
136
143
  let _agentConfigOverride = null;
137
144
  /**
138
145
  * Parse `--agent <name>` from process.argv.
@@ -174,13 +181,49 @@ function getRepoRoot() {
174
181
  * Returns the shared bundle root directory: `~/AgentBundles/`
175
182
  */
176
183
  function getAgentBundlesRoot() {
177
- return path.join(os.homedir(), "AgentBundles");
184
+ const homeBase = process.env.WEBSITE_SITE_NAME ? "/home" : os.homedir();
185
+ return path.join(homeBase, "AgentBundles");
178
186
  }
179
187
  /**
180
188
  * Returns the agent-specific bundle directory: `~/AgentBundles/<agentName>.ouro/`
181
189
  */
182
- function getAgentRoot() {
183
- return path.join(getAgentBundlesRoot(), `${getAgentName()}.ouro`);
190
+ function getAgentRoot(agentName = getAgentName()) {
191
+ return path.join(getAgentBundlesRoot(), `${agentName}.ouro`);
192
+ }
193
+ function resolveOptionalAgentName(agentName) {
194
+ if (agentName && agentName.trim().length > 0)
195
+ return agentName.trim();
196
+ try {
197
+ return getAgentName();
198
+ }
199
+ catch {
200
+ return "slugger";
201
+ }
202
+ }
203
+ /**
204
+ * Returns the bundle-local runtime state directory: `~/AgentBundles/<agentName>.ouro/state/`
205
+ */
206
+ function getAgentStateRoot(agentName) {
207
+ return path.join(getAgentRoot(resolveOptionalAgentName(agentName)), "state");
208
+ }
209
+ exports.HARNESS_CANONICAL_REPO_URL = "https://github.com/ouroborosbot/ouroboros.git";
210
+ function getAgentRepoWorkspacesRoot(agentName) {
211
+ return path.join(getAgentStateRoot(resolveOptionalAgentName(agentName)), "workspaces");
212
+ }
213
+ function getAgentDaemonStateRoot(agentName) {
214
+ return path.join(getAgentStateRoot(resolveOptionalAgentName(agentName)), "daemon");
215
+ }
216
+ function getAgentDaemonLogsDir(agentName) {
217
+ return path.join(getAgentDaemonStateRoot(resolveOptionalAgentName(agentName)), "logs");
218
+ }
219
+ function getAgentDaemonLoggingConfigPath(agentName) {
220
+ return path.join(getAgentDaemonStateRoot(resolveOptionalAgentName(agentName)), "logging.json");
221
+ }
222
+ function getAgentMessagesRoot(agentName) {
223
+ return path.join(getAgentStateRoot(resolveOptionalAgentName(agentName)), "messages");
224
+ }
225
+ function getAgentToolsRoot(agentName) {
226
+ return path.join(getAgentStateRoot(resolveOptionalAgentName(agentName)), "tools");
184
227
  }
185
228
  /**
186
229
  * Returns the conventional secrets path: `~/.agentsecrets/<agentName>/secrets.json`
@@ -190,22 +233,13 @@ function getAgentSecretsPath(agentName = getAgentName()) {
190
233
  }
191
234
  /**
192
235
  * Load and parse `<agentRoot>/agent.json`.
193
- * Caches the result after first load.
236
+ * Reads the file fresh on each call unless an override is set.
194
237
  * Throws descriptive error if file is missing or contains invalid JSON.
195
238
  */
196
239
  function loadAgentConfig() {
197
240
  if (_agentConfigOverride) {
198
241
  return _agentConfigOverride;
199
242
  }
200
- if (_cachedAgentConfig) {
201
- (0, runtime_1.emitNervesEvent)({
202
- event: "identity.resolve",
203
- component: "config/identity",
204
- message: "loaded agent config from cache",
205
- meta: { source: "cache" },
206
- });
207
- return _cachedAgentConfig;
208
- }
209
243
  const agentRoot = getAgentRoot();
210
244
  const configFile = path.join(agentRoot, "agent.json");
211
245
  let raw;
@@ -267,7 +301,8 @@ function loadAgentConfig() {
267
301
  if (rawProvider !== "azure" &&
268
302
  rawProvider !== "minimax" &&
269
303
  rawProvider !== "anthropic" &&
270
- rawProvider !== "openai-codex") {
304
+ rawProvider !== "openai-codex" &&
305
+ rawProvider !== "github-copilot") {
271
306
  (0, runtime_1.emitNervesEvent)({
272
307
  level: "error",
273
308
  event: "config_identity.error",
@@ -278,8 +313,9 @@ function loadAgentConfig() {
278
313
  provider: rawProvider,
279
314
  },
280
315
  });
281
- throw new Error(`agent.json at ${configFile} must include provider: "azure", "minimax", "anthropic", or "openai-codex".`);
316
+ throw new Error(`agent.json at ${configFile} must include provider: "azure", "minimax", "anthropic", "openai-codex", or "github-copilot".`);
282
317
  }
318
+ const provider = rawProvider;
283
319
  const rawVersion = parsed.version;
284
320
  const version = rawVersion === undefined ? 1 : rawVersion;
285
321
  if (typeof version !== "number" ||
@@ -312,13 +348,14 @@ function loadAgentConfig() {
312
348
  });
313
349
  throw new Error(`agent.json at ${configFile} must include enabled as boolean.`);
314
350
  }
315
- _cachedAgentConfig = {
351
+ const config = {
316
352
  version,
317
353
  enabled,
318
- provider: rawProvider,
354
+ provider,
319
355
  context: parsed.context,
320
356
  logging: parsed.logging,
321
357
  senses: normalizeSenses(parsed.senses, configFile),
358
+ mcpServers: parsed.mcpServers,
322
359
  phrases: parsed.phrases,
323
360
  };
324
361
  (0, runtime_1.emitNervesEvent)({
@@ -327,7 +364,7 @@ function loadAgentConfig() {
327
364
  message: "loaded agent config from disk",
328
365
  meta: { source: "disk" },
329
366
  });
330
- return _cachedAgentConfig;
367
+ return config;
331
368
  }
332
369
  /**
333
370
  * Prime the agent name cache explicitly.
@@ -346,12 +383,18 @@ function setAgentName(name) {
346
383
  function setAgentConfigOverride(config) {
347
384
  _agentConfigOverride = config;
348
385
  }
386
+ /**
387
+ * Preserve the compatibility hook for callers that previously cleared cached
388
+ * disk-backed agent config. Agent config is now read fresh on every call.
389
+ */
390
+ function resetAgentConfigCache() {
391
+ // No-op: disk-backed agent config is no longer memoized in-process.
392
+ }
349
393
  /**
350
394
  * Clear all cached identity state.
351
395
  * Used in tests and when switching agent context.
352
396
  */
353
397
  function resetIdentity() {
354
398
  _cachedAgentName = null;
355
- _cachedAgentConfig = null;
356
399
  _agentConfigOverride = null;
357
400
  }
@@ -1,25 +1,8 @@
1
1
  "use strict";
2
- // TODO: Kicks enforce "any action" but not "meaningful action". After a narration
3
- // kick, the model can satisfy the constraint by calling a no-op tool like
4
- // get_current_time({}). We need to detect trivial compliance and either re-kick
5
- // or discount the tool call. Ideally, the kick message would suggest a specific
6
- // tool call based on conversation context (what the user asked, what tools are
7
- // relevant) rather than just saying "call a tool". That's a bigger piece of work —
8
- // it requires the kick system to be context-aware.
9
- // See ouroboros' observation: "i'm not chickening out. i'm satisfying a crude
10
- // constraint. poorly."
11
- //
12
- // A kick is a self-correction. When the harness detects a malformed response,
13
- // it injects an assistant-role message as if the model caught its own mistake.
14
- //
15
- // Kicks are:
16
- // - assistant role (self-correction, not external rebuke)
17
- // - first person ("I" not "you")
18
- // - forward-looking (what I'm doing next, not what I did wrong)
19
- // - short (one sentence)
20
2
  Object.defineProperty(exports, "__esModule", { value: true });
21
3
  exports.hasToolIntent = hasToolIntent;
22
4
  exports.detectKick = detectKick;
5
+ const runtime_1 = require("../nerves/runtime");
23
6
  const KICK_MESSAGES = {
24
7
  empty: "I sent an empty message by accident — let me try again.",
25
8
  narration: "I narrated instead of acting. Using the tool now -- if done, calling final_answer.",
@@ -141,4 +124,3 @@ function detectKick(content, options) {
141
124
  }
142
125
  return null;
143
126
  }
144
- const runtime_1 = require("../nerves/runtime");
@@ -0,0 +1,48 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.MODEL_CAPABILITIES = void 0;
4
+ exports.getModelCapabilities = getModelCapabilities;
5
+ const runtime_1 = require("../nerves/runtime");
6
+ exports.MODEL_CAPABILITIES = {
7
+ "claude-opus-4-6": {
8
+ reasoningEffort: ["low", "medium", "high", "max"],
9
+ thinkingFormat: "anthropic",
10
+ maxOutputTokens: 128000,
11
+ },
12
+ "claude-sonnet-4-6": {
13
+ reasoningEffort: ["low", "medium", "high"],
14
+ thinkingFormat: "anthropic",
15
+ maxOutputTokens: 64000,
16
+ },
17
+ "claude-opus-4.6": {
18
+ reasoningEffort: ["low", "medium", "high", "max"],
19
+ maxOutputTokens: 128000,
20
+ },
21
+ "claude-sonnet-4.6": {
22
+ reasoningEffort: ["low", "medium", "high"],
23
+ maxOutputTokens: 64000,
24
+ },
25
+ "gpt-5.4": {
26
+ reasoningEffort: ["low", "medium", "high"],
27
+ phase: true,
28
+ maxOutputTokens: 100000,
29
+ },
30
+ "gpt-5.3-codex": {
31
+ reasoningEffort: ["low", "medium", "high"],
32
+ phase: true,
33
+ maxOutputTokens: 100000,
34
+ },
35
+ };
36
+ const EMPTY_CAPABILITIES = Object.freeze({});
37
+ function getModelCapabilities(modelId) {
38
+ (0, runtime_1.emitNervesEvent)({
39
+ component: "engine",
40
+ event: "engine.model_capabilities_lookup",
41
+ message: `model capabilities lookup: ${modelId}`,
42
+ meta: { modelId, found: modelId in exports.MODEL_CAPABILITIES },
43
+ });
44
+ const entry = exports.MODEL_CAPABILITIES[modelId];
45
+ if (entry)
46
+ return entry;
47
+ return { ...EMPTY_CAPABILITIES };
48
+ }