@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
@@ -0,0 +1,144 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.upsertGroupContextParticipants = upsertGroupContextParticipants;
4
+ const node_crypto_1 = require("node:crypto");
5
+ const runtime_1 = require("../../nerves/runtime");
6
+ const CURRENT_SCHEMA_VERSION = 1;
7
+ function normalizeDisplayName(externalId, displayName) {
8
+ const trimmed = displayName?.trim();
9
+ return trimmed && trimmed.length > 0 ? trimmed : externalId;
10
+ }
11
+ function buildNameNotes(name, now) {
12
+ return name !== "Unknown"
13
+ ? { name: { value: name, savedAt: now } }
14
+ : {};
15
+ }
16
+ function dedupeParticipants(participants) {
17
+ const deduped = new Map();
18
+ for (const participant of participants) {
19
+ const externalId = participant.externalId.trim();
20
+ if (!externalId)
21
+ continue;
22
+ const key = `${participant.provider}:${externalId}`;
23
+ if (!deduped.has(key)) {
24
+ deduped.set(key, {
25
+ ...participant,
26
+ externalId,
27
+ displayName: participant.displayName?.trim() || undefined,
28
+ });
29
+ }
30
+ }
31
+ return Array.from(deduped.values());
32
+ }
33
+ function createGroupExternalId(provider, groupExternalId, linkedAt) {
34
+ return {
35
+ provider,
36
+ externalId: groupExternalId,
37
+ linkedAt,
38
+ };
39
+ }
40
+ function shouldPromoteToAcquaintance(friend) {
41
+ return (friend.trustLevel ?? "stranger") === "stranger";
42
+ }
43
+ function createAcquaintanceRecord(participant, groupExternalId, linkedAt) {
44
+ const name = normalizeDisplayName(participant.externalId, participant.displayName);
45
+ return {
46
+ id: (0, node_crypto_1.randomUUID)(),
47
+ name,
48
+ role: "acquaintance",
49
+ trustLevel: "acquaintance",
50
+ connections: [],
51
+ externalIds: [
52
+ {
53
+ provider: participant.provider,
54
+ externalId: participant.externalId,
55
+ linkedAt,
56
+ },
57
+ createGroupExternalId(participant.provider, groupExternalId, linkedAt),
58
+ ],
59
+ tenantMemberships: [],
60
+ toolPreferences: {},
61
+ notes: buildNameNotes(name, linkedAt),
62
+ totalTokens: 0,
63
+ createdAt: linkedAt,
64
+ updatedAt: linkedAt,
65
+ schemaVersion: CURRENT_SCHEMA_VERSION,
66
+ };
67
+ }
68
+ async function upsertGroupContextParticipants(input) {
69
+ (0, runtime_1.emitNervesEvent)({
70
+ component: "friends",
71
+ event: "friends.group_context_upsert_start",
72
+ message: "upserting shared-group participant context",
73
+ meta: {
74
+ participantCount: input.participants.length,
75
+ hasGroupExternalId: input.groupExternalId.trim().length > 0,
76
+ },
77
+ });
78
+ const groupExternalId = input.groupExternalId.trim();
79
+ if (!groupExternalId) {
80
+ return [];
81
+ }
82
+ const now = input.now ?? (() => new Date().toISOString());
83
+ const participants = dedupeParticipants(input.participants);
84
+ const results = [];
85
+ for (const participant of participants) {
86
+ const linkedAt = now();
87
+ const existing = await input.store.findByExternalId(participant.provider, participant.externalId);
88
+ if (!existing) {
89
+ const created = createAcquaintanceRecord(participant, groupExternalId, linkedAt);
90
+ await input.store.put(created.id, created);
91
+ results.push({
92
+ friendId: created.id,
93
+ name: created.name,
94
+ trustLevel: "acquaintance",
95
+ created: true,
96
+ updated: false,
97
+ addedGroupExternalId: true,
98
+ });
99
+ continue;
100
+ }
101
+ const hasGroupExternalId = existing.externalIds.some((externalId) => externalId.externalId === groupExternalId);
102
+ const promoteToAcquaintance = shouldPromoteToAcquaintance(existing);
103
+ const trustLevel = promoteToAcquaintance
104
+ ? "acquaintance"
105
+ : existing.trustLevel;
106
+ const role = promoteToAcquaintance
107
+ ? "acquaintance"
108
+ : existing.role;
109
+ const updatedExternalIds = hasGroupExternalId
110
+ ? existing.externalIds
111
+ : [...existing.externalIds, createGroupExternalId(participant.provider, groupExternalId, linkedAt)];
112
+ const updated = promoteToAcquaintance || !hasGroupExternalId;
113
+ const record = updated
114
+ ? {
115
+ ...existing,
116
+ role,
117
+ trustLevel,
118
+ externalIds: updatedExternalIds,
119
+ updatedAt: linkedAt,
120
+ }
121
+ : existing;
122
+ if (updated) {
123
+ await input.store.put(record.id, record);
124
+ }
125
+ results.push({
126
+ friendId: record.id,
127
+ name: record.name,
128
+ trustLevel,
129
+ created: false,
130
+ updated,
131
+ addedGroupExternalId: !hasGroupExternalId,
132
+ });
133
+ }
134
+ (0, runtime_1.emitNervesEvent)({
135
+ component: "friends",
136
+ event: "friends.group_context_upsert_end",
137
+ message: "upserted shared-group participant context",
138
+ meta: {
139
+ participantCount: participants.length,
140
+ updatedCount: results.filter((result) => result.created || result.updated).length,
141
+ },
142
+ });
143
+ return results;
144
+ }
@@ -100,6 +100,25 @@ class FileFriendStore {
100
100
  }
101
101
  return entries.some((entry) => entry.endsWith(".json"));
102
102
  }
103
+ async listAll() {
104
+ let entries;
105
+ try {
106
+ entries = await fsPromises.readdir(this.friendsPath);
107
+ }
108
+ catch {
109
+ return [];
110
+ }
111
+ const records = [];
112
+ for (const entry of entries) {
113
+ if (!entry.endsWith(".json"))
114
+ continue;
115
+ const raw = await this.readJson(path.join(this.friendsPath, entry));
116
+ if (!raw)
117
+ continue;
118
+ records.push(this.normalize(raw));
119
+ }
120
+ return records;
121
+ }
103
122
  normalize(raw) {
104
123
  const trustLevel = raw.trustLevel;
105
124
  const normalizedTrustLevel = trustLevel === "family" ||
@@ -0,0 +1,74 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.describeTrustContext = describeTrustContext;
4
+ const runtime_1 = require("../../nerves/runtime");
5
+ function findRelatedGroupId(friend) {
6
+ return friend.externalIds.find((externalId) => externalId.externalId.startsWith("group:"))?.externalId;
7
+ }
8
+ function resolveLevel(friend) {
9
+ return friend.trustLevel ?? "stranger";
10
+ }
11
+ function describeTrustContext(input) {
12
+ const level = resolveLevel(input.friend);
13
+ const relatedGroupId = findRelatedGroupId(input.friend);
14
+ const explanation = level === "family" || level === "friend"
15
+ ? {
16
+ level,
17
+ basis: "direct",
18
+ summary: level === "family"
19
+ ? "direct family trust"
20
+ : "direct trusted relationship",
21
+ why: "this relationship is directly trusted rather than inferred through a shared group or cold first contact.",
22
+ permits: [
23
+ "local operations when appropriate",
24
+ "proactive follow-through",
25
+ "full collaborative problem solving",
26
+ ],
27
+ constraints: [],
28
+ }
29
+ : level === "acquaintance"
30
+ ? {
31
+ level,
32
+ basis: "shared_group",
33
+ summary: relatedGroupId
34
+ ? "known through the shared project group"
35
+ : "known through a shared group context",
36
+ why: relatedGroupId
37
+ ? `this trust comes from the shared group context ${relatedGroupId}, not from direct endorsement.`
38
+ : "this trust comes from shared group context rather than direct endorsement.",
39
+ permits: [
40
+ "group-safe coordination",
41
+ "normal conversation inside the shared context",
42
+ ],
43
+ constraints: [
44
+ "guarded local actions",
45
+ "do not assume broad private authority",
46
+ ],
47
+ relatedGroupId,
48
+ }
49
+ : {
50
+ level,
51
+ basis: "unknown",
52
+ summary: "truly unknown first-contact context",
53
+ why: "this person is not known through direct trust or a shared group context.",
54
+ permits: [
55
+ "safe first-contact orientation only",
56
+ ],
57
+ constraints: [
58
+ "first contact does not reach the full model on open channels",
59
+ "no local or privileged actions",
60
+ ],
61
+ };
62
+ (0, runtime_1.emitNervesEvent)({
63
+ component: "friends",
64
+ event: "friends.trust_explained",
65
+ message: "built explicit trust explanation",
66
+ meta: {
67
+ channel: input.channel,
68
+ level: explanation.level,
69
+ basis: explanation.basis,
70
+ hasRelatedGroup: Boolean(explanation.relatedGroupId),
71
+ },
72
+ });
73
+ return explanation;
74
+ }
@@ -2,8 +2,10 @@
2
2
  // Context kernel type definitions.
3
3
  // FriendRecord (merged identity + memory), channel capabilities, and resolved context.
4
4
  Object.defineProperty(exports, "__esModule", { value: true });
5
+ exports.TRUSTED_LEVELS = void 0;
5
6
  exports.isIdentityProvider = isIdentityProvider;
6
7
  exports.isIntegration = isIntegration;
8
+ exports.isTrustedLevel = isTrustedLevel;
7
9
  const runtime_1 = require("../../nerves/runtime");
8
10
  const IDENTITY_PROVIDERS = new Set(["aad", "local", "teams-conversation", "imessage-handle"]);
9
11
  function isIdentityProvider(value) {
@@ -19,3 +21,9 @@ const INTEGRATIONS = new Set(["ado", "github", "graph"]);
19
21
  function isIntegration(value) {
20
22
  return typeof value === "string" && INTEGRATIONS.has(value);
21
23
  }
24
+ /** Trust levels that grant full tool access and proactive send capability. */
25
+ exports.TRUSTED_LEVELS = new Set(["family", "friend"]);
26
+ /** Whether a trust level grants full access (family or friend). Defaults to "friend" for legacy records. */
27
+ function isTrustedLevel(trustLevel) {
28
+ return exports.TRUSTED_LEVELS.has(trustLevel ?? "friend");
29
+ }
@@ -33,7 +33,6 @@ var __importStar = (this && this.__importStar) || (function () {
33
33
  };
34
34
  })();
35
35
  Object.defineProperty(exports, "__esModule", { value: true });
36
- exports.__memoryTestUtils = void 0;
37
36
  exports.ensureMemoryStorePaths = ensureMemoryStorePaths;
38
37
  exports.appendFactsWithDedup = appendFactsWithDedup;
39
38
  exports.readMemoryFacts = readMemoryFacts;
@@ -46,7 +45,9 @@ const crypto_1 = require("crypto");
46
45
  const config_1 = require("../heart/config");
47
46
  const identity_1 = require("../heart/identity");
48
47
  const runtime_1 = require("../nerves/runtime");
48
+ const associative_recall_1 = require("./associative-recall");
49
49
  const DEDUP_THRESHOLD = 0.6;
50
+ const SEMANTIC_DEDUP_THRESHOLD = 0.95;
50
51
  const ENTITY_TOKEN = /[a-z0-9]+/g;
51
52
  const DEFAULT_EMBEDDING_MODEL = "text-embedding-3-small";
52
53
  class OpenAIEmbeddingProvider {
@@ -120,9 +121,16 @@ function readExistingFacts(factsPath) {
120
121
  const raw = fs.readFileSync(factsPath, "utf8").trim();
121
122
  if (!raw)
122
123
  return [];
123
- return raw
124
- .split("\n")
125
- .map((line) => JSON.parse(line));
124
+ const facts = [];
125
+ for (const line of raw.split("\n")) {
126
+ try {
127
+ facts.push(JSON.parse(line));
128
+ }
129
+ catch {
130
+ // Skip corrupt lines (e.g. partial write from a crash).
131
+ }
132
+ }
133
+ return facts;
126
134
  }
127
135
  function readEntityIndex(entitiesPath) {
128
136
  if (!fs.existsSync(entitiesPath))
@@ -170,13 +178,24 @@ function appendDailyFact(dailyDir, fact) {
170
178
  const dayPath = path.join(dailyDir, `${day}.jsonl`);
171
179
  fs.appendFileSync(dayPath, `${JSON.stringify(fact)}\n`, "utf8");
172
180
  }
173
- function appendFactsWithDedup(stores, incoming) {
181
+ function appendFactsWithDedup(stores, incoming, options) {
174
182
  const existing = readExistingFacts(stores.factsPath);
175
183
  const all = [...existing];
176
184
  let added = 0;
177
185
  let skipped = 0;
186
+ const semanticThreshold = options?.semanticThreshold;
178
187
  for (const fact of incoming) {
179
- const duplicate = all.some((prior) => overlapScore(prior.text, fact.text) > DEDUP_THRESHOLD);
188
+ const duplicate = all.some((prior) => {
189
+ if (overlapScore(prior.text, fact.text) > DEDUP_THRESHOLD)
190
+ return true;
191
+ if (semanticThreshold !== undefined &&
192
+ Array.isArray(fact.embedding) && fact.embedding.length > 0 &&
193
+ Array.isArray(prior.embedding) && prior.embedding.length > 0 &&
194
+ fact.embedding.length === prior.embedding.length) {
195
+ return (0, associative_recall_1.cosineSimilarity)(fact.embedding, prior.embedding) > semanticThreshold;
196
+ }
197
+ return false;
198
+ });
180
199
  if (duplicate) {
181
200
  skipped++;
182
201
  continue;
@@ -195,24 +214,6 @@ function appendFactsWithDedup(stores, incoming) {
195
214
  });
196
215
  return { added, skipped };
197
216
  }
198
- function cosineSimilarity(left, right) {
199
- if (left.length === 0 || right.length === 0 || left.length !== right.length)
200
- return 0;
201
- let dot = 0;
202
- let leftNorm = 0;
203
- let rightNorm = 0;
204
- for (let i = 0; i < left.length; i += 1) {
205
- dot += left[i] * right[i];
206
- leftNorm += left[i] * left[i];
207
- rightNorm += right[i] * right[i];
208
- }
209
- if (leftNorm === 0 || rightNorm === 0)
210
- return 0;
211
- return dot / (Math.sqrt(leftNorm) * Math.sqrt(rightNorm));
212
- }
213
- exports.__memoryTestUtils = {
214
- cosineSimilarity,
215
- };
216
217
  function createDefaultEmbeddingProvider() {
217
218
  const apiKey = (0, config_1.getOpenAIEmbeddingsApiKey)().trim();
218
219
  if (!apiKey)
@@ -264,7 +265,7 @@ async function saveMemoryFact(options) {
264
265
  createdAt: (options.now ?? (() => new Date()))().toISOString(),
265
266
  embedding,
266
267
  };
267
- return appendFactsWithDedup(stores, [fact]);
268
+ return appendFactsWithDedup(stores, [fact], { semanticThreshold: SEMANTIC_DEDUP_THRESHOLD });
268
269
  }
269
270
  async function backfillEmbeddings(options) {
270
271
  const memoryRoot = options?.memoryRoot ?? path.join((0, identity_1.getAgentRoot)(), "psyche", "memory");
@@ -365,7 +366,7 @@ async function searchMemoryFacts(query, facts, embeddingProvider) {
365
366
  .filter((fact) => fact.embedding.length === queryEmbedding.length)
366
367
  .map((fact) => ({
367
368
  fact,
368
- score: cosineSimilarity(queryEmbedding, fact.embedding),
369
+ score: (0, associative_recall_1.cosineSimilarity)(queryEmbedding, fact.embedding),
369
370
  }))
370
371
  .filter((entry) => entry.score > 0)
371
372
  .sort((left, right) => right.score - left.score)
@@ -0,0 +1,221 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.findActivePersistentObligation = findActivePersistentObligation;
4
+ exports.findStatusObligation = findStatusObligation;
5
+ exports.renderActiveObligationSteering = renderActiveObligationSteering;
6
+ exports.renderConcreteStatusGuidance = renderConcreteStatusGuidance;
7
+ exports.renderLiveThreadStatusShape = renderLiveThreadStatusShape;
8
+ exports.buildExactStatusReply = buildExactStatusReply;
9
+ exports.renderExactStatusReplyContract = renderExactStatusReplyContract;
10
+ const config_1 = require("../heart/config");
11
+ const active_work_1 = require("../heart/active-work");
12
+ const runtime_1 = require("../nerves/runtime");
13
+ function findActivePersistentObligation(frame) {
14
+ if (!frame)
15
+ return null;
16
+ return (frame.pendingObligations ?? []).find((ob) => ob.status !== "pending" && ob.status !== "fulfilled") ?? null;
17
+ }
18
+ function obligationTimestampMs(obligation) {
19
+ return Date.parse(obligation.updatedAt ?? obligation.createdAt);
20
+ }
21
+ function newestObligationFirst(left, right) {
22
+ return obligationTimestampMs(right) - obligationTimestampMs(left);
23
+ }
24
+ function matchesCurrentSession(frame, obligation) {
25
+ return matchesSessionOrigin(frame, obligation.origin);
26
+ }
27
+ function matchesSessionOrigin(frame, origin) {
28
+ return Boolean(frame.currentSession
29
+ && origin.friendId === frame.currentSession.friendId
30
+ && origin.channel === frame.currentSession.channel
31
+ && (0, config_1.sanitizeKey)(origin.key) === (0, config_1.sanitizeKey)(frame.currentSession.key));
32
+ }
33
+ function findStatusObligation(frame) {
34
+ if (!frame)
35
+ return null;
36
+ const openObligations = [...(frame.pendingObligations ?? [])]
37
+ .filter((obligation) => obligation.status !== "fulfilled")
38
+ .sort(newestObligationFirst);
39
+ const sameSession = openObligations.find((obligation) => matchesCurrentSession(frame, obligation));
40
+ if (sameSession)
41
+ return sameSession;
42
+ return openObligations[0] ?? null;
43
+ }
44
+ function findCurrentSessionStatusObligation(frame) {
45
+ const openObligations = [...(frame.pendingObligations ?? [])]
46
+ .filter((obligation) => obligation.status !== "fulfilled")
47
+ .sort(newestObligationFirst);
48
+ return openObligations.find((obligation) => matchesCurrentSession(frame, obligation)) ?? null;
49
+ }
50
+ function renderActiveObligationSteering(obligation) {
51
+ (0, runtime_1.emitNervesEvent)({
52
+ component: "mind",
53
+ event: "mind.obligation_steering_rendered",
54
+ message: "rendered active obligation steering",
55
+ meta: {
56
+ hasObligation: Boolean(obligation),
57
+ hasSurface: Boolean(obligation?.currentSurface?.label),
58
+ },
59
+ });
60
+ if (!obligation)
61
+ return "";
62
+ const name = obligation.origin.friendId;
63
+ const surfaceLine = obligation.currentSurface?.label
64
+ ? `\nright now that work is happening in ${obligation.currentSurface.label}.`
65
+ : "";
66
+ return `## where my attention is
67
+ i'm already working on something i owe ${name}.${surfaceLine}
68
+
69
+ i should close that loop before i act like this is a fresh blank turn.`;
70
+ }
71
+ function mergeArtifactFallback(obligation) {
72
+ const trimmed = obligation.content.trim();
73
+ if (!trimmed)
74
+ return "the fix";
75
+ const stripped = trimmed.replace(/^merge(?:\s+|$)/i, "").trim();
76
+ return stripped || "the fix";
77
+ }
78
+ function formatMergeArtifact(obligation) {
79
+ const currentArtifact = obligation.currentArtifact?.trim();
80
+ if (currentArtifact)
81
+ return currentArtifact;
82
+ if (obligation.currentSurface?.kind === "merge") {
83
+ const surfaceLabel = obligation.currentSurface.label.trim();
84
+ if (surfaceLabel)
85
+ return surfaceLabel;
86
+ }
87
+ return mergeArtifactFallback(obligation);
88
+ }
89
+ function formatActiveLane(frame, obligation) {
90
+ const liveCodingSession = frame.codingSessions?.[0];
91
+ if (liveCodingSession) {
92
+ const sameThread = frame.currentSession
93
+ && liveCodingSession.originSession
94
+ && liveCodingSession.originSession.friendId === frame.currentSession.friendId
95
+ && liveCodingSession.originSession.channel === frame.currentSession.channel
96
+ && liveCodingSession.originSession.key === frame.currentSession.key;
97
+ return sameThread
98
+ ? `${liveCodingSession.runner} ${liveCodingSession.id} for this same thread`
99
+ : liveCodingSession.originSession
100
+ ? `${liveCodingSession.runner} ${liveCodingSession.id} for ${liveCodingSession.originSession.channel}/${liveCodingSession.originSession.key}`
101
+ : `${liveCodingSession.runner} ${liveCodingSession.id}`;
102
+ }
103
+ return obligation.currentSurface?.label
104
+ || (matchesCurrentSession(frame, obligation) ? "this same thread" : "this live loop");
105
+ }
106
+ function formatCurrentArtifact(frame, obligation) {
107
+ if (obligation?.currentArtifact)
108
+ return obligation.currentArtifact;
109
+ if (obligation?.currentSurface?.kind === "merge")
110
+ return obligation.currentSurface.label;
111
+ if ((frame.codingSessions ?? []).length > 0)
112
+ return "no PR or merge artifact yet";
113
+ return obligation ? "no artifact yet" : "";
114
+ }
115
+ function formatNextAction(frame, obligation) {
116
+ const obligationHasConcreteArtifact = Boolean(obligation?.currentArtifact?.trim())
117
+ || obligation?.currentSurface?.kind === "merge";
118
+ if (obligation?.status === "waiting_for_merge") {
119
+ return obligation.nextAction?.trim() || `wait for checks, merge ${formatMergeArtifact(obligation)}, then update runtime`;
120
+ }
121
+ if (obligation?.status === "updating_runtime") {
122
+ return obligation.nextAction?.trim() || "update runtime, verify version/changelog, then re-observe";
123
+ }
124
+ if (obligationHasConcreteArtifact && obligation?.nextAction?.trim()) {
125
+ return obligation.nextAction.trim();
126
+ }
127
+ const liveCodingSession = frame.codingSessions?.[0];
128
+ if (liveCodingSession?.status === "waiting_input") {
129
+ return `answer ${liveCodingSession.runner} ${liveCodingSession.id} and continue`;
130
+ }
131
+ if (liveCodingSession?.status === "stalled") {
132
+ return `unstick ${liveCodingSession.runner} ${liveCodingSession.id} and continue`;
133
+ }
134
+ if (liveCodingSession) {
135
+ return "finish the coding pass and bring the result back here";
136
+ }
137
+ if (obligation?.nextAction?.trim())
138
+ return obligation.nextAction.trim();
139
+ if (obligation?.content?.trim()) {
140
+ return `work on "${obligation.content.trim()}" and bring back a concrete artifact`;
141
+ }
142
+ return obligation ? "continue the active loop and bring the result back here" : "";
143
+ }
144
+ function renderConcreteStatusGuidance(frame, obligation) {
145
+ const activeLane = obligation ? formatActiveLane(frame, obligation) : "";
146
+ const currentArtifact = formatCurrentArtifact(frame, obligation);
147
+ const nextAction = formatNextAction(frame, obligation);
148
+ const liveConversation = frame.currentSession
149
+ ? `${frame.currentSession.channel}/${frame.currentSession.key}`
150
+ : "";
151
+ if (!activeLane && !currentArtifact && !nextAction)
152
+ return "";
153
+ return `if someone asks what i'm doing or for status mid-task, i answer from these live facts instead of copying a canned block.
154
+ the live conversation is ${liveConversation || "not in a live conversation"}.
155
+ the active lane is ${activeLane}.
156
+ the current artifact is ${currentArtifact}.
157
+ if i just finished or verified something concrete in this live lane, i name that as the latest checkpoint.
158
+ the next action is ${nextAction}.
159
+
160
+ i answer naturally from those facts instead of forcing a canned status block.`;
161
+ }
162
+ function renderLiveThreadStatusShape(frame) {
163
+ if (!frame.currentSession)
164
+ return "";
165
+ return `if someone asks what i'm doing or for status mid-task in this live thread, i answer in these exact lines, in order, with no intro paragraph:
166
+ live conversation: ${frame.currentSession.channel}/${frame.currentSession.key}
167
+ active lane: this same thread
168
+ current artifact: <actual artifact or "no artifact yet">
169
+ latest checkpoint: <freshest concrete thing i just finished or verified>
170
+ next action: <smallest concrete next step i'm taking now>
171
+
172
+ no recap paragraph before those lines.
173
+ no option list.
174
+ present tense only.
175
+ if a finished step matters, i label it "just finished" instead of presenting it as current work.`;
176
+ }
177
+ function buildExactStatusReply(frame, obligation, latestCheckpoint, statusCheckScope) {
178
+ const headerObligation = statusCheckScope === "all-sessions-family"
179
+ ? findCurrentSessionStatusObligation(frame)
180
+ : obligation;
181
+ const liveConversation = frame.currentSession
182
+ ? `${frame.currentSession.channel}/${frame.currentSession.key}`
183
+ : "not in a live conversation";
184
+ const activeLane = headerObligation
185
+ ? formatActiveLane(frame, headerObligation)
186
+ : (frame.currentSession ? "this same thread" : "this live loop");
187
+ const currentArtifact = formatCurrentArtifact(frame, headerObligation) || "no artifact yet";
188
+ const nextAction = formatNextAction(frame, headerObligation) || "continue the active loop and bring the result back here";
189
+ const latest = latestCheckpoint.trim() || "<freshest concrete thing i just finished or verified>";
190
+ const lines = [
191
+ `live conversation: ${liveConversation}`,
192
+ `active lane: ${activeLane}`,
193
+ `current artifact: ${currentArtifact}`,
194
+ `latest checkpoint: ${latest}`,
195
+ `next action: ${nextAction}`,
196
+ ];
197
+ if (statusCheckScope === "all-sessions-family") {
198
+ lines.push("other active sessions:");
199
+ const summaries = (0, active_work_1.formatOtherActiveSessionSummaries)(frame);
200
+ lines.push(...(summaries.length > 0 ? summaries : ["- none"]));
201
+ }
202
+ return lines.join("\n");
203
+ }
204
+ function renderExactStatusReplyContract(frame, obligation, statusCheckScope) {
205
+ const headerObligation = statusCheckScope === "all-sessions-family"
206
+ ? findCurrentSessionStatusObligation(frame)
207
+ : obligation;
208
+ if (statusCheckScope === "all-sessions-family") {
209
+ return `reply using exactly this status shape and nothing else:
210
+ live conversation: ${frame.currentSession ? `${frame.currentSession.channel}/${frame.currentSession.key}` : "not in a live conversation"}
211
+ active lane: ${headerObligation ? formatActiveLane(frame, headerObligation) : (frame.currentSession ? "this same thread" : "this live loop")}
212
+ current artifact: ${formatCurrentArtifact(frame, headerObligation) || "no artifact yet"}
213
+ latest checkpoint: <freshest concrete thing i just finished or verified>
214
+ next action: ${formatNextAction(frame, headerObligation) || "continue the active loop and bring the result back here"}
215
+ other active sessions:
216
+ - <session label>: <what i'm doing there right now>`;
217
+ }
218
+ return `reply using exactly these five lines and nothing else:
219
+ ${buildExactStatusReply(frame, obligation, "<freshest concrete thing i just finished or verified>")}
220
+ `;
221
+ }