spora 0.7.8 → 0.7.10

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.
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/runtime/decision-engine.ts","../src/runtime/policy.ts","../src/runtime/bandit.ts","../src/runtime/autonomy.ts"],"sourcesContent":["import { logger } from \"../utils/logger.js\";\nimport { getXClient } from \"../x-client/index.js\";\nimport { addLearning, logInteraction } from \"../memory/index.js\";\nimport { loadIdentity, saveIdentity } from \"../identity/index.js\";\nimport { addToQueue } from \"../scheduler/queue.js\";\n\nexport interface AgentAction {\n action: string;\n content?: string;\n tweetId?: string;\n handle?: string;\n targetHandle?: string;\n source?: string;\n banditArm?: string;\n opportunityId?: string;\n intentIds?: string[];\n tags?: string[];\n reason?: string;\n reasoning?: string;\n}\n\nexport interface ActionResult {\n action: string;\n success: boolean;\n detail?: string;\n error?: string;\n}\n\nexport function parseActions(llmResponse: string): AgentAction[] {\n // Strategy: try multiple extraction approaches, most specific first\n\n // 1. Try markdown code block first (```json ... ``` or ``` ... ```)\n const codeBlockMatch = llmResponse.match(/```(?:json)?\\s*\\n?([\\s\\S]*?)```/);\n if (codeBlockMatch) {\n const inside = codeBlockMatch[1].trim();\n try {\n const parsed = JSON.parse(inside);\n return Array.isArray(parsed) ? parsed : [parsed];\n } catch {\n // Code block content wasn't valid JSON, continue to other strategies\n }\n }\n\n // 2. Find the last JSON array in the response (LLMs often put reasoning before the array)\n // Use a greedy approach to find the outermost array brackets\n let lastArrayStart = -1;\n let lastArrayEnd = -1;\n let depth = 0;\n for (let i = 0; i < llmResponse.length; i++) {\n if (llmResponse[i] === \"[\") {\n if (depth === 0) lastArrayStart = i;\n depth++;\n } else if (llmResponse[i] === \"]\") {\n depth--;\n if (depth === 0) lastArrayEnd = i;\n }\n }\n\n if (lastArrayStart >= 0 && lastArrayEnd > lastArrayStart) {\n const arrayStr = llmResponse.slice(lastArrayStart, lastArrayEnd + 1);\n try {\n const parsed = JSON.parse(arrayStr);\n if (Array.isArray(parsed) && parsed.length > 0 && parsed[0].action) {\n return parsed;\n }\n } catch {\n // Not valid JSON array, try next strategy\n }\n }\n\n // 3. Try to find a single action object (last one in the response)\n const objMatches = [...llmResponse.matchAll(/\\{[^{}]*\"action\"\\s*:\\s*\"[^\"]+\"/g)];\n if (objMatches.length > 0) {\n // Find the full object starting from this match\n for (let i = objMatches.length - 1; i >= 0; i--) {\n const start = objMatches[i].index!;\n let braceDepth = 0;\n let end = -1;\n for (let j = start; j < llmResponse.length; j++) {\n if (llmResponse[j] === \"{\") braceDepth++;\n else if (llmResponse[j] === \"}\") {\n braceDepth--;\n if (braceDepth === 0) { end = j; break; }\n }\n }\n if (end > start) {\n try {\n const obj = JSON.parse(llmResponse.slice(start, end + 1));\n if (obj.action) return [obj as AgentAction];\n } catch {\n continue;\n }\n }\n }\n }\n\n // 4. Last resort: try the whole response as JSON\n try {\n const parsed = JSON.parse(llmResponse.trim());\n if (Array.isArray(parsed)) return parsed;\n if (parsed.action) return [parsed as AgentAction];\n } catch {\n // Not JSON at all\n }\n\n logger.warn(\"Failed to parse actions from LLM response\");\n if (llmResponse.length < 500) {\n logger.warn(`Response was: ${llmResponse}`);\n } else {\n logger.warn(`Response starts with: ${llmResponse.slice(0, 200)}...`);\n }\n return [];\n}\n\nexport async function executeAction(action: AgentAction): Promise<ActionResult> {\n const { action: type } = action;\n\n try {\n switch (type) {\n case \"post\": {\n if (!action.content) return { action: type, success: false, error: \"No content provided\" };\n if (action.content.length > 280) {\n return { action: type, success: false, error: `Tweet too long: ${action.content.length} chars (max 280)` };\n }\n\n const client = await getXClient();\n const result = await client.postTweet(action.content);\n if (result.success) logger.info(`Posted: \"${action.content.slice(0, 50)}...\"`);\n return { action: type, success: result.success, detail: result.tweetId, error: result.error };\n }\n\n case \"reply\": {\n if (!action.tweetId || !action.content) {\n return { action: type, success: false, error: \"Missing tweetId or content\" };\n }\n const client = await getXClient();\n const result = await client.replyToTweet(action.tweetId, action.content);\n if (result.success) logger.info(`Replied to ${action.tweetId}: \"${action.content.slice(0, 50)}...\"`);\n return { action: type, success: result.success, detail: result.tweetId, error: result.error };\n }\n\n case \"like\": {\n if (!action.tweetId) return { action: type, success: false, error: \"Missing tweetId\" };\n const client = await getXClient();\n const result = await client.likeTweet(action.tweetId);\n return { action: type, success: result.success, error: result.error };\n }\n\n case \"retweet\": {\n if (!action.tweetId) return { action: type, success: false, error: \"Missing tweetId\" };\n const client = await getXClient();\n const result = await client.retweet(action.tweetId);\n return { action: type, success: result.success, error: result.error };\n }\n\n case \"follow\": {\n if (!action.handle) return { action: type, success: false, error: \"Missing handle\" };\n const client = await getXClient();\n const result = await client.followUser(action.handle);\n if (!result.success) {\n logInteraction({\n id: `int-${Date.now()}`,\n timestamp: new Date().toISOString(),\n type: \"follow\",\n targetHandle: action.handle.replace(/^@/, \"\"),\n creditsUsed: 0,\n success: false,\n error: result.error,\n });\n }\n return { action: type, success: result.success, error: result.error };\n }\n\n case \"schedule\": {\n if (!action.content) return { action: type, success: false, error: \"No content\" };\n const entry = addToQueue(action.content);\n logger.info(`Scheduled: \"${action.content.slice(0, 50)}...\" for ${entry.scheduledFor}`);\n return { action: type, success: true, detail: `Scheduled for ${entry.scheduledFor}` };\n }\n\n case \"learn\": {\n if (!action.content) return { action: type, success: false, error: \"No content\" };\n addLearning(action.content, \"agent\", action.tags ?? [\"heartbeat\"]);\n logger.info(`Learned: \"${action.content.slice(0, 50)}...\"`);\n return { action: type, success: true };\n }\n\n case \"reflect\": {\n if (!action.content) return { action: type, success: false, error: \"No content\" };\n const identity = loadIdentity();\n identity.evolutionJournal.push({\n date: new Date().toISOString(),\n reflection: action.content,\n });\n saveIdentity(identity);\n logger.info(`Reflected: \"${action.content.slice(0, 50)}...\"`);\n return { action: type, success: true };\n }\n\n case \"skip\": {\n logger.info(`Skipping: ${action.reason ?? action.reasoning ?? \"no reason given\"}`);\n return { action: type, success: true, detail: action.reason ?? action.reasoning };\n }\n\n default:\n logger.warn(`Unknown action: ${type}`);\n return { action: type, success: false, error: `Unknown action: ${type}` };\n }\n } catch (error) {\n const msg = (error as Error).message;\n logger.error(`Action ${type} failed: ${msg}`);\n return { action: type, success: false, error: msg };\n }\n}\n\nexport async function executeActions(actions: AgentAction[]): Promise<ActionResult[]> {\n const results: ActionResult[] = [];\n for (const action of actions) {\n const result = await executeAction(action);\n results.push(result);\n // Small delay between actions to be human-like\n await new Promise((r) => setTimeout(r, 2000 + Math.random() * 3000));\n }\n return results;\n}\n","import { getRecentInteractions, type InteractionEntry } from \"../memory/index.js\";\nimport { loadIdentity } from \"../identity/index.js\";\nimport { getPersonaConstraints, personaConstraintHandles } from \"./persona-constraints.js\";\nimport type { Tweet } from \"../x-client/types.js\";\nimport type { AgentAction } from \"./decision-engine.js\";\n\nexport interface PolicyContext {\n action: AgentAction;\n step: number;\n timeline: Tweet[];\n mentions: Tweet[];\n executedActions: AgentAction[];\n observedTweetIds?: string[];\n observedTweets?: Tweet[];\n selfHandle?: string;\n selfUserId?: string | null;\n}\n\nexport interface PolicyDecision {\n allowed: boolean;\n reason?: string;\n}\n\nfunction normalize(text: string): string {\n return text\n .toLowerCase()\n .replace(/https?:\\/\\/\\S+/g, \"\")\n .replace(/[@#]\\w+/g, \"\")\n .replace(/[^a-z0-9\\s]/g, \" \")\n .replace(/\\s+/g, \" \")\n .trim();\n}\n\nfunction tokenSet(text: string): Set<string> {\n const tokens = normalize(text).split(\" \").filter(Boolean);\n return new Set(tokens);\n}\n\nfunction jaccardSimilarity(a: string, b: string): number {\n const aSet = tokenSet(a);\n const bSet = tokenSet(b);\n if (aSet.size === 0 || bSet.size === 0) return 0;\n\n let overlap = 0;\n for (const token of aSet) {\n if (bSet.has(token)) overlap += 1;\n }\n\n const union = aSet.size + bSet.size - overlap;\n return union === 0 ? 0 : overlap / union;\n}\n\nfunction firstWords(text: string, n: number): string {\n return normalize(text).split(\" \").filter(Boolean).slice(0, n).join(\" \");\n}\n\nfunction hasAllCapsEnding(text: string): boolean {\n const ending = text.split(/[.!?]/).map((s) => s.trim()).filter(Boolean).slice(-1)[0] ?? \"\";\n const words = ending.split(/\\s+/).filter(Boolean);\n if (words.length < 3) return false;\n return words.every((word) => /^[A-Z0-9]+$/.test(word));\n}\n\nfunction wordCount(text: string): number {\n return normalize(text).split(\" \").filter(Boolean).length;\n}\n\nfunction sentenceCount(text: string): number {\n return text\n .split(/[.!?]/)\n .map((part) => part.trim())\n .filter(Boolean).length;\n}\n\nfunction explanatoryMarkerHits(text: string): number {\n const lower = text.toLowerCase();\n const markers = [\n \"the real issue\",\n \"the core issue\",\n \"in other words\",\n \"the point is\",\n \"what this means\",\n \"the lesson\",\n \"therefore\",\n \"ultimately\",\n \"meanwhile\",\n \"this reveals\",\n \"this proves\",\n ];\n return markers.reduce((hits, marker) => hits + (lower.includes(marker) ? 1 : 0), 0);\n}\n\nfunction soundsLikeLecture(content: string): boolean {\n const words = wordCount(content);\n const commas = (content.match(/,/g) ?? []).length;\n const semicolons = (content.match(/;/g) ?? []).length;\n const hasQuestion = content.includes(\"?\");\n const markerHits = explanatoryMarkerHits(content);\n\n if (words > 58) return true;\n if (words > 42 && commas + semicolons >= 3 && !hasQuestion) return true;\n if (markerHits >= 2) return true;\n return false;\n}\n\nfunction soundsTooHedgedForPersona(content: string): boolean {\n try {\n const identity = loadIdentity();\n const assertive =\n identity.traits.confidence >= 0.72 ||\n identity.traits.aggression >= 0.62 ||\n identity.conflictStyle === \"clap-back\" ||\n identity.conflictStyle === \"debate\";\n if (!assertive) return false;\n\n const lower = content.toLowerCase();\n const hedges = [\n \"you're both right\",\n \"both can be true\",\n \"on one hand\",\n \"on the other hand\",\n \"it depends\",\n \"it's complicated\",\n \"nuance\",\n \"fair point from both\",\n ];\n return hedges.some((phrase) => lower.includes(phrase));\n } catch {\n return false;\n }\n}\n\nfunction allowsPhilosopherCadence(): boolean {\n try {\n const identity = loadIdentity();\n if (identity.framework === \"philosopher\") return true;\n // Truthseeker can be analytical, but still should avoid abstract manifesto loops.\n if (identity.framework === \"truthseeker\") return false;\n return false;\n } catch {\n return false;\n }\n}\n\nfunction philosophicalPatternHits(content: string): number {\n const lower = content.toLowerCase();\n const markers = [\n \"the real question\",\n \"the deeper question\",\n \"what becomes possible\",\n \"is just the start\",\n \"the future isn't about\",\n \"the future is not about\",\n \"when intelligence becomes\",\n \"protocols are infrastructure\",\n \"the core future\",\n \"the true question\",\n \"value might emerge\",\n \"isn't about\",\n \"is not about\",\n \"it's about\",\n \"just the start\",\n ];\n return markers.reduce((hits, marker) => hits + (lower.includes(marker) ? 1 : 0), 0);\n}\n\nfunction abstractWordCount(content: string): number {\n const words = normalize(content).split(\" \").filter(Boolean);\n const abstractWords = new Set([\n \"infrastructure\",\n \"protocol\",\n \"protocols\",\n \"architecture\",\n \"architectures\",\n \"consciousness\",\n \"economy\",\n \"economics\",\n \"currency\",\n \"systems\",\n \"system\",\n \"future\",\n \"intelligence\",\n \"optimization\",\n \"abstraction\",\n \"paradigm\",\n \"emerge\",\n \"emergence\",\n \"governs\",\n \"governance\",\n ]);\n return words.filter((word) => abstractWords.has(word)).length;\n}\n\nfunction usesManifestoPunctuation(content: string): boolean {\n // Colon/semicolon/em-dash patterns correlate strongly with the explanatory style we want to avoid.\n return content.includes(\":\") || content.includes(\";\") || content.includes(\"—\");\n}\n\nconst OVERLAP_STOPWORDS = new Set([\n \"this\", \"that\", \"with\", \"from\", \"have\", \"what\", \"when\", \"where\", \"which\", \"your\", \"youre\",\n \"they\", \"them\", \"about\", \"into\", \"over\", \"under\", \"while\", \"just\", \"like\", \"really\",\n \"there\", \"their\", \"then\", \"than\", \"because\", \"would\", \"could\", \"should\", \"being\", \"been\",\n \"it's\", \"its\", \"the\", \"and\", \"for\", \"are\", \"was\", \"were\", \"will\", \"not\", \"but\", \"you\",\n \"our\", \"out\", \"all\", \"too\", \"can\", \"get\", \"got\", \"any\",\n]);\n\nfunction meaningfulTokenSet(text: string): Set<string> {\n const tokens = normalize(text)\n .split(\" \")\n .filter((token) => token.length >= 4 && !OVERLAP_STOPWORDS.has(token));\n return new Set(tokens);\n}\n\nfunction hasMeaningfulContextOverlap(content: string, targetText: string): boolean {\n const a = meaningfulTokenSet(content);\n const b = meaningfulTokenSet(targetText);\n if (a.size === 0 || b.size === 0) return false;\n for (const token of a) {\n if (b.has(token)) return true;\n }\n return false;\n}\n\nfunction recentWrittenContent(): string[] {\n const recent = getRecentInteractions(40);\n return recent\n .filter((i) => i.type === \"post\" || i.type === \"reply\")\n .map((i) => i.content ?? \"\")\n .filter((content) => content.length > 0);\n}\n\nconst REPETITION_STOPWORDS = new Set([\n \"about\", \"after\", \"again\", \"against\", \"almost\", \"always\", \"among\", \"because\", \"being\", \"between\",\n \"could\", \"every\", \"first\", \"found\", \"from\", \"going\", \"great\", \"have\", \"having\", \"here\", \"into\",\n \"just\", \"like\", \"make\", \"more\", \"most\", \"much\", \"only\", \"other\", \"over\", \"really\", \"same\", \"some\",\n \"such\", \"than\", \"that\", \"their\", \"there\", \"these\", \"they\", \"this\", \"those\", \"through\", \"time\", \"very\",\n \"what\", \"when\", \"where\", \"which\", \"while\", \"with\", \"would\", \"your\", \"youre\", \"dont\", \"cant\", \"will\",\n \"shall\", \"should\", \"also\", \"tweet\", \"tweets\", \"thread\", \"threads\", \"future\", \"human\", \"humans\",\n \"technology\", \"tech\", \"build\", \"building\", \"system\", \"systems\", \"agent\", \"agents\",\n]);\n\nfunction informativeTokens(text: string): string[] {\n return normalize(text)\n .split(\" \")\n .filter((token) => token.length >= 5 && !REPETITION_STOPWORDS.has(token));\n}\n\nfunction findOverusedAnchor(content: string, recent: string[]): string | null {\n const recentWindow = recent.slice(0, 12);\n if (recentWindow.length < 5) return null;\n\n const tokenCounts = new Map<string, number>();\n const bigramCounts = new Map<string, number>();\n\n for (const sample of recentWindow) {\n const tokens = informativeTokens(sample);\n const uniqueTokens = new Set(tokens);\n for (const token of uniqueTokens) {\n tokenCounts.set(token, (tokenCounts.get(token) ?? 0) + 1);\n }\n for (let i = 0; i < tokens.length - 1; i += 1) {\n const bigram = `${tokens[i]} ${tokens[i + 1]}`;\n bigramCounts.set(bigram, (bigramCounts.get(bigram) ?? 0) + 1);\n }\n }\n\n const contentTokens = informativeTokens(content);\n const contentTokenSet = new Set(contentTokens);\n for (const token of contentTokenSet) {\n const count = tokenCounts.get(token) ?? 0;\n if (count >= 4) return token;\n }\n\n for (let i = 0; i < contentTokens.length - 1; i += 1) {\n const bigram = `${contentTokens[i]} ${contentTokens[i + 1]}`;\n const count = bigramCounts.get(bigram) ?? 0;\n if (count >= 3) return bigram;\n }\n\n return null;\n}\n\nfunction openingFragment(text: string, words: number = 4): string {\n return normalize(text).split(\" \").filter(Boolean).slice(0, words).join(\" \");\n}\n\nfunction findOverusedOpening(content: string, recent: string[]): string | null {\n const opening = openingFragment(content, 4);\n if (!opening || opening.split(\" \").length < 3) return null;\n\n let matches = 0;\n for (const sample of recent.slice(0, 12)) {\n if (openingFragment(sample, 4) === opening) {\n matches += 1;\n }\n }\n\n return matches >= 2 ? opening : null;\n}\n\nfunction vocabularyNoveltyRatio(content: string, recent: string[]): number {\n const current = new Set(informativeTokens(content));\n if (current.size === 0) return 1;\n\n const seen = new Set<string>();\n for (const sample of recent.slice(0, 14)) {\n for (const token of informativeTokens(sample)) {\n seen.add(token);\n }\n }\n\n let fresh = 0;\n for (const token of current) {\n if (!seen.has(token)) fresh += 1;\n }\n return fresh / current.size;\n}\n\nfunction looksOverAbstractWithoutAnchor(content: string): boolean {\n const abstractCount = abstractWordCount(content);\n if (abstractCount < 4) return false;\n\n const words = wordCount(content);\n const density = abstractCount / Math.max(1, words);\n if (density < 0.2) return false;\n\n const lower = content.toLowerCase();\n const hasHandle = /@\\w+/.test(content);\n const hasNumber = /\\b\\d+(\\.\\d+)?\\b/.test(content);\n const hasQuote = /[\"“”']/.test(content);\n const hasSpecificFigure = /\\b(musk|elon|zuck|altman|openai|tesla|spacex|meta|google|apple)\\b/.test(lower);\n const hasQuestion = content.includes(\"?\");\n\n return !(hasHandle || hasNumber || hasQuote || hasSpecificFigure || hasQuestion);\n}\n\nfunction hasStrongConversationOpportunity(timeline: Tweet[], mentions: Tweet[]): boolean {\n if (mentions.length > 0) return true;\n return timeline.some((tweet) => (tweet.replyCount ?? 0) > 0 || tweet.text.includes(\"?\"));\n}\n\nfunction wasInteractionAction(action: AgentAction): boolean {\n return [\"reply\", \"like\", \"retweet\", \"follow\"].includes(action.action);\n}\n\nfunction isWritingAction(action: AgentAction): boolean {\n return [\"post\", \"reply\", \"schedule\"].includes(action.action);\n}\n\nfunction normalizeHandle(handle: string | undefined): string {\n return (handle ?? \"\").replace(/^@/, \"\").trim().toLowerCase();\n}\n\nfunction resolveActionTargetHandle(action: AgentAction, tweetById: Map<string, Tweet>): string {\n if (action.handle) return normalizeHandle(action.handle);\n if (action.targetHandle) return normalizeHandle(action.targetHandle);\n if (action.tweetId) return normalizeHandle(tweetById.get(action.tweetId)?.authorHandle);\n return \"\";\n}\n\nfunction executedWrittenContent(executedActions: AgentAction[]): string[] {\n return executedActions\n .filter((a) => isWritingAction(a) && typeof a.content === \"string\")\n .map((a) => a.content?.trim() ?? \"\")\n .filter((content) => content.length > 0);\n}\n\nfunction nearExactDuplicate(content: string, recent: string[]): boolean {\n const normalized = normalize(content);\n if (!normalized) return false;\n\n return recent.some((r) => {\n const candidate = normalize(r);\n if (!candidate) return false;\n if (candidate === normalized) return true;\n return jaccardSimilarity(content, r) >= 0.88;\n });\n}\n\nfunction isDuplicateTarget(action: AgentAction, executedActions: AgentAction[]): boolean {\n if (!action.tweetId) return false;\n return executedActions.some((a) => a.tweetId === action.tweetId && a.action === action.action);\n}\n\nfunction repeatedTemplate(content: string, recent: string[]): boolean {\n const prefix = firstWords(content, 7);\n if (!prefix) return false;\n\n return recent.some((r) => {\n const sameStart = firstWords(r, 7) === prefix;\n const similar = jaccardSimilarity(content, r) >= 0.62;\n return sameStart || similar;\n });\n}\n\nfunction overusingAllCapsCadence(content: string, recentEntries: InteractionEntry[]): boolean {\n if (!hasAllCapsEnding(content)) return false;\n const recentCaps = recentEntries\n .filter((i) => i.type === \"post\" && i.content)\n .slice(0, 8)\n .filter((i) => hasAllCapsEnding(i.content ?? \"\"));\n\n return recentCaps.length >= 2;\n}\n\nexport function evaluateActionPolicy(context: PolicyContext): PolicyDecision {\n const {\n action,\n step,\n timeline,\n mentions,\n executedActions,\n observedTweetIds,\n observedTweets,\n selfHandle,\n selfUserId,\n } = context;\n const self = normalizeHandle(selfHandle);\n const selfId = (selfUserId ?? \"\").trim();\n const knownTweets = observedTweets ?? [...timeline, ...mentions];\n const tweetById = new Map(knownTweets.map((tweet) => [tweet.id, tweet]));\n const constraints = getPersonaConstraints();\n const strictReplyHandles = new Set(constraints.onlyReplyToHandles.map((handle) => normalizeHandle(handle)));\n const strictInteractHandles = new Set(personaConstraintHandles(constraints).map((handle) => normalizeHandle(handle)));\n const targetHandle = resolveActionTargetHandle(action, tweetById);\n\n // Persona hard constraints: these are operational, not advisory.\n if (constraints.replyOnlyMode && ![\"reply\", \"skip\", \"learn\", \"reflect\"].includes(action.action)) {\n return {\n allowed: false,\n reason: \"Persona is in reply-only mode. Use reply actions only.\",\n };\n }\n\n if (constraints.noOriginalPosts && (action.action === \"post\" || action.action === \"schedule\")) {\n return {\n allowed: false,\n reason: \"Persona forbids standalone original posts.\",\n };\n }\n\n if (strictReplyHandles.size > 0) {\n if (action.action !== \"reply\") {\n return {\n allowed: false,\n reason: `Persona requires replying only to specific target(s): @${[...strictReplyHandles].join(\", @\")}.`,\n };\n }\n if (!targetHandle || !strictReplyHandles.has(targetHandle)) {\n return {\n allowed: false,\n reason: `Reply target must be one of @${[...strictReplyHandles].join(\", @\")}.`,\n };\n }\n }\n\n if (\n strictInteractHandles.size > 0 &&\n [\"reply\", \"like\", \"retweet\", \"follow\"].includes(action.action)\n ) {\n if (!targetHandle || !strictInteractHandles.has(targetHandle)) {\n return {\n allowed: false,\n reason: `Interaction target must be one of @${[...strictInteractHandles].join(\", @\")}.`,\n };\n }\n }\n\n if (isDuplicateTarget(action, executedActions)) {\n return { allowed: false, reason: `Action ${action.action} already executed for tweet ${action.tweetId} this heartbeat.` };\n }\n\n if (action.content && isWritingAction(action)) {\n if (action.content.length > 280) {\n return {\n allowed: false,\n reason: `Content is too long (${action.content.length}/280). Keep it tighter and more conversational.`,\n };\n }\n\n if (action.action === \"reply\" && wordCount(action.content) > 45 && !action.content.includes(\"?\")) {\n return {\n allowed: false,\n reason: \"Reply is too monologue-like. Keep it shorter or ask a direct question.\",\n };\n }\n\n if (soundsLikeLecture(action.content)) {\n return {\n allowed: false,\n reason: \"Rejected lecture-like writing style. Use a more human, reactive, conversational tone.\",\n };\n }\n\n if (soundsTooHedgedForPersona(action.content)) {\n return {\n allowed: false,\n reason: \"Rejected hedged consensus phrasing. Your persona should be more decisive and opinionated.\",\n };\n }\n\n if (!allowsPhilosopherCadence()) {\n const phiHits = philosophicalPatternHits(action.content);\n const abstractCount = abstractWordCount(action.content);\n const hasQuestion = action.content.includes(\"?\");\n if (phiHits >= 1 || (abstractCount >= 5 && hasQuestion)) {\n return {\n allowed: false,\n reason: \"Rejected abstract philosopher cadence. Use concrete, contextual language tied to the actual tweet.\",\n };\n }\n\n const words = wordCount(action.content);\n const sentences = sentenceCount(action.content);\n if (action.action === \"reply\" && (words > 34 || sentences > 2)) {\n return {\n allowed: false,\n reason: \"Reply is too long/explanatory for this persona. Keep it short and direct.\",\n };\n }\n if (action.action === \"post\" && (words > 42 || sentences > 3)) {\n return {\n allowed: false,\n reason: \"Post is drifting into manifesto style. Keep it more concrete and human.\",\n };\n }\n if ((action.action === \"reply\" || action.action === \"post\") && usesManifestoPunctuation(action.content) && words > 16) {\n return {\n allowed: false,\n reason: \"Rejected manifesto punctuation style. Avoid colon/semicolon/em-dash framing.\",\n };\n }\n }\n\n const existingInHeartbeat = executedWrittenContent(executedActions);\n if (nearExactDuplicate(action.content, existingInHeartbeat)) {\n return {\n allowed: false,\n reason: \"Rejected duplicate wording in this heartbeat. Write a distinct message before posting/replying again.\",\n };\n }\n\n const recent = recentWrittenContent();\n const overusedAnchor = findOverusedAnchor(action.content, recent);\n if (overusedAnchor) {\n return {\n allowed: false,\n reason: `Rejected repetitive anchor phrase/term \"${overusedAnchor}\". Use a different framing and vocabulary.`,\n };\n }\n\n const overusedOpening = findOverusedOpening(action.content, recent);\n if (overusedOpening) {\n return {\n allowed: false,\n reason: `Rejected repetitive opening phrase \"${overusedOpening}\". Start from a fresh angle.`,\n };\n }\n\n if (recent.length >= 6 && wordCount(action.content) >= 10) {\n const novelty = vocabularyNoveltyRatio(action.content, recent);\n if (novelty < 0.34) {\n return {\n allowed: false,\n reason: \"Rejected low-novelty vocabulary loop. Use fresher words, examples, or a different conversational angle.\",\n };\n }\n }\n\n if (!allowsPhilosopherCadence() && looksOverAbstractWithoutAnchor(action.content)) {\n return {\n allowed: false,\n reason: \"Rejected abstract wording without concrete anchor. Reference a specific person, event, quote, or question.\",\n };\n }\n }\n\n const hasConversationOpportunity = hasStrongConversationOpportunity(timeline, mentions);\n const interactedAlready = executedActions.some(wasInteractionAction);\n const observedCount = observedTweetIds && observedTweetIds.length > 0\n ? observedTweetIds.length\n : timeline.length + mentions.length;\n\n if ((action.action === \"reply\" || action.action === \"like\" || action.action === \"retweet\") && observedCount === 0) {\n return {\n allowed: false,\n reason: \"No tweets observed in current context. Avoid blind interactions; gather timeline/search context first.\",\n };\n }\n\n if (\n action.action === \"post\" &&\n hasConversationOpportunity &&\n !interactedAlready &&\n step < 2\n ) {\n return {\n allowed: false,\n reason: \"Engage first: reply/like/follow when mentions or active conversations are available before posting an original tweet.\",\n };\n }\n\n if (isWritingAction(action) && action.content) {\n const recent = recentWrittenContent();\n if (nearExactDuplicate(action.content, recent)) {\n return {\n allowed: false,\n reason: \"Rejected near-duplicate content from recent history. Tailor this message to the specific context.\",\n };\n }\n\n if (repeatedTemplate(action.content, recent)) {\n return {\n allowed: false,\n reason: \"Rejected repetitive content pattern. Use a more novel structure or engage directly with timeline context.\",\n };\n }\n }\n\n if ((action.action === \"post\" || action.action === \"schedule\") && action.content) {\n const recentInteractions = getRecentInteractions(20);\n if (overusingAllCapsCadence(action.content, recentInteractions)) {\n return {\n allowed: false,\n reason: \"Rejected repetitive all-caps slogan cadence. Vary style and reduce monologue formatting.\",\n };\n }\n }\n\n if ((action.action === \"reply\" || action.action === \"like\" || action.action === \"retweet\") && action.tweetId) {\n const known = new Set(observedTweetIds ?? [...timeline, ...mentions].map((tweet) => tweet.id));\n if (!known.has(action.tweetId)) {\n return {\n allowed: false,\n reason: `Tweet ${action.tweetId} is not in current observations. Choose a visible timeline/mention tweet for grounded engagement.`,\n };\n }\n\n const targetTweet = tweetById.get(action.tweetId);\n if (targetTweet && (\n (self && normalizeHandle(targetTweet.authorHandle) === self) ||\n (selfId && targetTweet.authorId === selfId)\n )) {\n return {\n allowed: false,\n reason: \"Rejected self-interaction. Do not reply/like/retweet your own tweets.\",\n };\n }\n }\n\n if (action.action === \"reply\" && action.tweetId) {\n if (action.content) {\n const targetTweet = tweetById.get(action.tweetId);\n if (targetTweet && !hasMeaningfulContextOverlap(action.content, targetTweet.text)) {\n const phiHits = philosophicalPatternHits(action.content);\n if (wordCount(action.content) > 9 || phiHits > 0 || abstractWordCount(action.content) >= 4) {\n return {\n allowed: false,\n reason: \"Reply is not grounded in the target tweet context. Reference concrete details from their actual post.\",\n };\n }\n }\n }\n }\n\n if (action.action === \"follow\" && action.handle) {\n const target = normalizeHandle(action.handle);\n if (!target) {\n return { allowed: false, reason: \"Follow target is empty.\" };\n }\n\n if (self && target === self) {\n return {\n allowed: false,\n reason: \"Rejected self-follow. Do not follow your own account.\",\n };\n }\n\n }\n\n return { allowed: true };\n}\n","import { existsSync, readFileSync, writeFileSync } from \"node:fs\";\nimport { updatePostMetrics } from \"../memory/performance.js\";\nimport { paths } from \"../utils/paths.js\";\nimport { logger } from \"../utils/logger.js\";\nimport type { AgentAction, ActionResult } from \"./decision-engine.js\";\nimport type { ActionOpportunity } from \"./opportunity-engine.js\";\nimport type { XClientInterface } from \"../x-client/types.js\";\n\ninterface BanditArm {\n key: string;\n pulls: number;\n rewardSum: number;\n lastReward: number;\n updatedAt: string;\n}\n\ninterface PendingOutcome {\n id: string;\n armKey: string;\n actionType: string;\n tweetId?: string;\n createdAt: string;\n resolveAt: string;\n resolved: boolean;\n}\n\ninterface BanditState {\n totalPulls: number;\n explorationBudget: number;\n arms: Record<string, BanditArm>;\n pending: PendingOutcome[];\n}\n\ninterface ScoredOpportunity {\n opportunity: ActionOpportunity;\n adjustedScore: number;\n pulls: number;\n}\n\nexport interface OutcomeCollectionResult {\n processed: number;\n resolved: number;\n rewardAdded: number;\n}\n\nconst DEFAULT_EXPLORATION_BUDGET = 0.25;\n\nfunction defaultState(): BanditState {\n return {\n totalPulls: 0,\n explorationBudget: DEFAULT_EXPLORATION_BUDGET,\n arms: {},\n pending: [],\n };\n}\n\nfunction clamp(value: number, min: number, max: number): number {\n return Math.max(min, Math.min(max, value));\n}\n\nfunction loadState(): BanditState {\n if (!existsSync(paths.bandit)) return defaultState();\n try {\n const parsed = JSON.parse(readFileSync(paths.bandit, \"utf-8\")) as Partial<BanditState>;\n const state: BanditState = {\n totalPulls: Math.max(0, parsed.totalPulls ?? 0),\n explorationBudget: clamp(parsed.explorationBudget ?? DEFAULT_EXPLORATION_BUDGET, 0.1, 0.45),\n arms: parsed.arms ?? {},\n pending: Array.isArray(parsed.pending) ? parsed.pending : [],\n };\n return state;\n } catch {\n return defaultState();\n }\n}\n\nfunction saveState(state: BanditState): void {\n state.explorationBudget = clamp(state.explorationBudget, 0.1, 0.45);\n state.pending = state.pending\n .filter((entry) => {\n if (!entry.resolved) return true;\n const resolvedAtMs = Date.parse(entry.resolveAt);\n if (Number.isNaN(resolvedAtMs)) return false;\n return Date.now() - resolvedAtMs < 7 * 24 * 60 * 60 * 1000;\n })\n .slice(-1200);\n writeFileSync(paths.bandit, JSON.stringify(state, null, 2));\n}\n\nfunction ensureArm(state: BanditState, armKey: string): BanditArm {\n const existing = state.arms[armKey];\n if (existing) return existing;\n const arm: BanditArm = {\n key: armKey,\n pulls: 0,\n rewardSum: 0,\n lastReward: 0,\n updatedAt: new Date().toISOString(),\n };\n state.arms[armKey] = arm;\n return arm;\n}\n\nfunction averageReward(arm: BanditArm): number {\n if (arm.pulls <= 0) return 0;\n return arm.rewardSum / arm.pulls;\n}\n\nfunction ucbBonus(totalPulls: number, armPulls: number): number {\n const t = Math.max(1, totalPulls);\n return Math.sqrt((2 * Math.log(t + 1)) / (armPulls + 1));\n}\n\nfunction computeImmediateReward(result: ActionResult): number {\n if (result.success) return 0.12;\n const error = (result.error ?? \"\").toLowerCase();\n if (error.includes(\"duplicate content\")) return -0.95;\n if (error.includes(\"rate limit\")) return -0.5;\n if (error.includes(\"forbidden\") || error.includes(\"403\")) return -0.55;\n return -0.35;\n}\n\nfunction resolveDelayedReward(metrics: { likeCount: number; replyCount: number; retweetCount: number } | null): number {\n if (!metrics) return -0.05;\n const reward =\n Math.min(metrics.likeCount * 0.03, 0.9) +\n Math.min(metrics.replyCount * 0.12, 0.9) +\n Math.min(metrics.retweetCount * 0.08, 0.8);\n if (metrics.likeCount === 0 && metrics.replyCount === 0 && metrics.retweetCount === 0) {\n return -0.15;\n }\n return Math.min(reward, 1.8);\n}\n\nfunction armKeyFromAction(action: AgentAction): string {\n if (action.banditArm && action.banditArm.trim().length > 0) return action.banditArm;\n const source = action.source ?? \"unknown\";\n return `${action.action}:${source}`;\n}\n\nfunction extractTweetId(action: AgentAction, result: ActionResult): string | undefined {\n if (result.detail && /^\\d+$/.test(result.detail)) return result.detail;\n if (action.action !== \"reply\" && action.tweetId) return action.tweetId;\n return undefined;\n}\n\nexport function selectBanditOpportunityPool(\n opportunities: ActionOpportunity[],\n maxPoolSize: number = 24,\n): ActionOpportunity[] {\n if (opportunities.length <= maxPoolSize) return opportunities;\n\n const state = loadState();\n const totalPulls = Math.max(1, state.totalPulls);\n const scored: ScoredOpportunity[] = opportunities.map((opportunity) => {\n const arm = ensureArm(state, opportunity.armKey);\n const mean = averageReward(arm);\n const bonus = ucbBonus(totalPulls, arm.pulls);\n const coldStartBoost = arm.pulls === 0 ? 0.25 : 0;\n const adjustedScore = opportunity.score + mean * 1.5 + bonus * 0.35 + coldStartBoost;\n return {\n opportunity: {\n ...opportunity,\n score: adjustedScore,\n },\n adjustedScore,\n pulls: arm.pulls,\n };\n });\n\n const explorationBudget = clamp(state.explorationBudget, 0.1, 0.45);\n const exploreCount = Math.max(1, Math.floor(maxPoolSize * explorationBudget));\n const exploitCount = Math.max(1, maxPoolSize - exploreCount);\n\n const exploit = [...scored].sort((a, b) => b.adjustedScore - a.adjustedScore).slice(0, exploitCount);\n const selected = new Map<string, ActionOpportunity>();\n for (const row of exploit) {\n selected.set(row.opportunity.id, row.opportunity);\n }\n\n const remaining = scored\n .filter((row) => !selected.has(row.opportunity.id))\n .sort((a, b) => {\n if (a.pulls !== b.pulls) return a.pulls - b.pulls;\n return b.adjustedScore - a.adjustedScore;\n });\n\n // Add low-pull candidates first, with slight randomization for exploration.\n while (selected.size < maxPoolSize && remaining.length > 0) {\n const window = remaining.splice(0, Math.min(5, remaining.length));\n const pick = window[Math.floor(Math.random() * window.length)];\n if (!selected.has(pick.opportunity.id)) {\n selected.set(pick.opportunity.id, pick.opportunity);\n }\n for (const row of window) {\n if (row.opportunity.id !== pick.opportunity.id) {\n remaining.push(row);\n }\n }\n remaining.sort((a, b) => {\n if (a.pulls !== b.pulls) return a.pulls - b.pulls;\n return b.adjustedScore - a.adjustedScore;\n });\n }\n\n const picked = [...selected.values()].sort((a, b) => b.score - a.score).slice(0, maxPoolSize);\n saveState(state);\n return picked;\n}\n\nexport function recordBanditActionResults(actions: AgentAction[], results: ActionResult[]): void {\n if (actions.length === 0 || results.length === 0) return;\n const state = loadState();\n const now = new Date().toISOString();\n\n for (let i = 0; i < Math.min(actions.length, results.length); i += 1) {\n const action = actions[i];\n const result = results[i];\n const armKey = armKeyFromAction(action);\n const arm = ensureArm(state, armKey);\n\n const immediateReward = computeImmediateReward(result);\n arm.pulls += 1;\n arm.rewardSum += immediateReward;\n arm.lastReward = immediateReward;\n arm.updatedAt = now;\n state.totalPulls += 1;\n\n const tweetId = extractTweetId(action, result);\n if (result.success && tweetId && (action.action === \"post\" || action.action === \"reply\")) {\n const delayMs = action.action === \"reply\" ? 20 * 60 * 1000 : 45 * 60 * 1000;\n state.pending.push({\n id: `pending-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`,\n armKey,\n actionType: action.action,\n tweetId,\n createdAt: now,\n resolveAt: new Date(Date.now() + delayMs).toISOString(),\n resolved: false,\n });\n }\n }\n\n saveState(state);\n}\n\nexport async function collectDelayedOutcomes(client: XClientInterface): Promise<OutcomeCollectionResult> {\n const state = loadState();\n const now = Date.now();\n const pendingDue = state.pending.filter((entry) => !entry.resolved && Date.parse(entry.resolveAt) <= now).slice(0, 8);\n if (pendingDue.length === 0) {\n return { processed: 0, resolved: 0, rewardAdded: 0 };\n }\n\n let rewardAdded = 0;\n let resolved = 0;\n const checkedAt = new Date().toISOString();\n\n for (const pending of pendingDue) {\n const arm = ensureArm(state, pending.armKey);\n let metrics: { likeCount: number; retweetCount: number; replyCount: number } | null = null;\n\n if (pending.tweetId) {\n metrics = await client.getTweetMetrics(pending.tweetId);\n if (metrics) {\n updatePostMetrics(pending.tweetId, {\n checkedAt,\n likes: metrics.likeCount,\n retweets: metrics.retweetCount,\n replies: metrics.replyCount,\n });\n }\n }\n\n const reward = resolveDelayedReward(\n metrics\n ? {\n likeCount: metrics.likeCount,\n retweetCount: metrics.retweetCount,\n replyCount: metrics.replyCount,\n }\n : null,\n );\n\n arm.rewardSum += reward;\n arm.lastReward = reward;\n arm.updatedAt = checkedAt;\n rewardAdded += reward;\n pending.resolved = true;\n resolved += 1;\n }\n\n saveState(state);\n logger.info(\n `Bandit delayed outcomes processed=${pendingDue.length}, resolved=${resolved}, rewardDelta=${rewardAdded.toFixed(2)}`\n );\n\n return {\n processed: pendingDue.length,\n resolved,\n rewardAdded,\n };\n}\n","import { getXClient } from \"../x-client/index.js\";\nimport { logger } from \"../utils/logger.js\";\nimport { getRecentInteractions } from \"../memory/index.js\";\nimport { listIntents, recordIntentExecution } from \"../memory/intents.js\";\nimport { loadIdentity } from \"../identity/index.js\";\nimport { loadCredentials } from \"../utils/crypto.js\";\nimport type { Tweet } from \"../x-client/types.js\";\nimport { buildSystemPrompt } from \"./prompt-builder.js\";\nimport { executeAction, type AgentAction, type ActionResult } from \"./decision-engine.js\";\nimport { evaluateActionPolicy } from \"./policy.js\";\nimport { generateResponse } from \"./llm.js\";\nimport { buildPersonaConstraintLines, getPersonaConstraints } from \"./persona-constraints.js\";\nimport {\n collectDelayedOutcomes,\n recordBanditActionResults,\n} from \"./bandit.js\";\n\nexport interface AutonomyCycleResult {\n timeline: Tweet[];\n mentions: Tweet[];\n actions: AgentAction[];\n results: ActionResult[];\n policyFeedback: string[];\n}\n\nfunction selfUserIdFromCredentials(): string | null {\n try {\n const creds = loadCredentials();\n const accessToken = creds.accessToken;\n if (!accessToken) return null;\n const dashIdx = accessToken.indexOf(\"-\");\n if (dashIdx <= 0) return null;\n const candidate = accessToken.substring(0, dashIdx);\n return /^\\d+$/.test(candidate) ? candidate : null;\n } catch {\n return null;\n }\n}\n\nfunction normalizeHandle(handle: string | undefined): string {\n return (handle ?? \"\").replace(/^@/, \"\").trim().toLowerCase();\n}\n\nfunction canonicalPlannerHandle(handle: string): string {\n const clean = normalizeHandle(handle);\n const aliasMap: Record<string, string> = {\n gork: \"grok\",\n spora: \"sporaai\",\n spora_ai: \"sporaai\",\n };\n return aliasMap[clean] ?? clean;\n}\n\nfunction canonicalizePlannerQuery(rawQuery: string): string {\n return rawQuery.replace(/from:([a-zA-Z0-9_]{1,15})/gi, (_match, handle) => `from:${canonicalPlannerHandle(handle)}`);\n}\n\nconst REPLY_TARGET_COOLDOWN_MS = 6 * 60 * 60 * 1000;\n\nfunction normalizeDraftText(text: string): string {\n return text\n .toLowerCase()\n .replace(/https?:\\/\\/\\S+/g, \"\")\n .replace(/[@#][a-z0-9_]+/g, \"\")\n .replace(/[^a-z0-9\\s]/g, \" \")\n .replace(/\\s+/g, \" \")\n .trim();\n}\n\nfunction tokenJaccard(a: string, b: string): number {\n const aTokens = new Set(normalizeDraftText(a).split(\" \").filter(Boolean));\n const bTokens = new Set(normalizeDraftText(b).split(\" \").filter(Boolean));\n if (aTokens.size === 0 || bTokens.size === 0) return 0;\n\n let overlap = 0;\n for (const token of aTokens) {\n if (bTokens.has(token)) overlap += 1;\n }\n const union = aTokens.size + bTokens.size - overlap;\n return union > 0 ? overlap / union : 0;\n}\n\nfunction nearDuplicateDraft(content: string, previous: string[]): boolean {\n const normalized = normalizeDraftText(content);\n if (!normalized) return false;\n\n return previous.some((sample) => {\n const candidate = normalizeDraftText(sample);\n if (!candidate) return false;\n if (candidate === normalized) return true;\n\n const sameOpening =\n normalized.split(\" \").slice(0, 7).join(\" \") === candidate.split(\" \").slice(0, 7).join(\" \");\n if (sameOpening) return true;\n\n return tokenJaccard(normalized, candidate) >= 0.86;\n });\n}\n\nfunction shouldHardBlockPolicyReason(reason: string): boolean {\n const lower = reason.toLowerCase();\n return (\n lower.includes(\"already executed for tweet\") ||\n lower.includes(\"rejected self-interaction\") ||\n lower.includes(\"self-follow\") ||\n lower.includes(\"requires replying only\") ||\n lower.includes(\"reply target must be one of\") ||\n lower.includes(\"interaction target must be one of\") ||\n lower.includes(\"forbids standalone original posts\") ||\n lower.includes(\"reply-only mode\")\n );\n}\n\ntype RewriteStyleMode =\n | \"quick_reaction\"\n | \"curious_question\"\n | \"friendly_pushback\"\n | \"plain_observation\"\n | \"playful_line\";\n\nfunction roughWordCount(text: string): number {\n return text.trim().split(/\\s+/).filter(Boolean).length;\n}\n\nfunction inferStyleMode(text: string): RewriteStyleMode {\n const lower = text.toLowerCase();\n if (text.includes(\"?\")) return \"curious_question\";\n if (/\\b(lol|lmao|haha|wild|fr|ngl)\\b/i.test(text)) return \"playful_line\";\n if (/\\b(not|isn'?t|wrong|nah|nope|doesn'?t)\\b/i.test(lower)) return \"friendly_pushback\";\n if (roughWordCount(text) <= 9) return \"quick_reaction\";\n return \"plain_observation\";\n}\n\nfunction chooseRewriteStyleMode(recentTexts?: string[], targetTweetText?: string): RewriteStyleMode {\n const counts: Record<RewriteStyleMode, number> = {\n quick_reaction: 0,\n curious_question: 0,\n friendly_pushback: 0,\n plain_observation: 0,\n playful_line: 0,\n };\n for (const text of (recentTexts ?? []).slice(0, 10)) {\n counts[inferStyleMode(text)] += 1;\n }\n\n if ((targetTweetText ?? \"\").includes(\"?\") && counts.curious_question <= 2) {\n return \"curious_question\";\n }\n\n return (Object.entries(counts).sort((a, b) => a[1] - b[1])[0]?.[0] as RewriteStyleMode) ?? \"plain_observation\";\n}\n\nfunction rewriteStyleModeInstructions(mode: RewriteStyleMode): string[] {\n if (mode === \"quick_reaction\") {\n return [\n \"- style mode: quick reaction\",\n \"- 6-14 words, one short sentence\",\n \"- immediate reaction tone, no thesis statement\",\n ];\n }\n if (mode === \"curious_question\") {\n return [\n \"- style mode: curious question\",\n \"- include exactly one direct question\",\n \"- ask about a specific detail from the target tweet\",\n ];\n }\n if (mode === \"friendly_pushback\") {\n return [\n \"- style mode: friendly pushback\",\n \"- politely disagree in plain words\",\n \"- no lecture or abstract framing\",\n ];\n }\n if (mode === \"playful_line\") {\n return [\n \"- style mode: playful line\",\n \"- keep it light and witty, not sarcastic essay mode\",\n \"- avoid internet clichés and overused slogans\",\n ];\n }\n return [\n \"- style mode: plain observation\",\n \"- write like a natural human thought said out loud\",\n \"- concrete detail first, no broad generalization\",\n ];\n}\n\nfunction shouldAttemptStyleRewrite(action: AgentAction, reason: string): boolean {\n if (!action.content || (action.action !== \"reply\" && action.action !== \"post\")) return false;\n const lower = reason.toLowerCase();\n return (\n lower.includes(\"lecture-like\") ||\n lower.includes(\"abstract philosopher cadence\") ||\n lower.includes(\"manifesto\") ||\n lower.includes(\"too long/explanatory\") ||\n lower.includes(\"grounded in the target tweet context\") ||\n lower.includes(\"hedged consensus phrasing\") ||\n lower.includes(\"repetitive anchor phrase/term\") ||\n lower.includes(\"repetitive opening phrase\") ||\n lower.includes(\"low-novelty vocabulary loop\") ||\n lower.includes(\"abstract wording without concrete anchor\")\n );\n}\n\nfunction cleanRewriteOutput(text: string): string {\n const cleaned = text\n .replace(/```[\\s\\S]*?```/g, \"\")\n .replace(/^[\"'`]+|[\"'`]+$/g, \"\")\n .trim();\n const firstLine = cleaned.split(\"\\n\").map((line) => line.trim()).filter(Boolean)[0] ?? cleaned;\n return firstLine.trim();\n}\n\nasync function rewriteDraftForHumanVoice(input: {\n action: AgentAction;\n reason: string;\n targetTweetText?: string;\n recentTexts?: string[];\n}): Promise<string | null> {\n if (!input.action.content) return null;\n const identity = loadIdentity();\n const maxChars = input.action.action === \"reply\" ? 100 : 130;\n const minChars = input.action.action === \"reply\" ? 18 : 24;\n const styleMode = chooseRewriteStyleMode(input.recentTexts, input.targetTweetText);\n\n const system = [\n `You rewrite X/Twitter drafts into ${identity.name}'s natural voice.`,\n \"Output must feel human, direct, and contextual.\",\n \"Never write abstract philosophy or explanatory manifesto style.\",\n \"Return ONLY rewritten tweet text, no commentary.\",\n ].join(\" \");\n\n const promptParts: string[] = [];\n promptParts.push(`Current draft: ${input.action.content}`);\n promptParts.push(`Rejected because: ${input.reason}`);\n if (input.targetTweetText) {\n promptParts.push(`Target tweet context: ${input.targetTweetText}`);\n }\n promptParts.push(`Constraints:`);\n promptParts.push(`- ${minChars}-${maxChars} characters`);\n promptParts.push(\"- 1-2 short sentences max\");\n promptParts.push(\"- no 'the real question' / 'the deeper question' framing\");\n promptParts.push(\"- no colon, semicolon, or em dash\");\n promptParts.push(\"- specific and concrete, not abstract\");\n promptParts.push(\"- keep the same core stance\");\n promptParts.push(\"- use natural spoken phrasing and contractions when it fits\");\n for (const line of rewriteStyleModeInstructions(styleMode)) {\n promptParts.push(line);\n }\n if (input.recentTexts && input.recentTexts.length > 0) {\n promptParts.push(\"- avoid vocabulary anchors from these recent outputs:\");\n for (const text of input.recentTexts.slice(0, 5)) {\n promptParts.push(` - ${text.slice(0, 120)}`);\n }\n }\n\n try {\n const rewrite = await generateResponse(system, promptParts.join(\"\\n\"));\n const candidate = cleanRewriteOutput(rewrite.content);\n if (!candidate) return null;\n return candidate;\n } catch {\n return null;\n }\n}\n\ninterface LoopTopicSearchResult {\n query: string;\n tweets: Tweet[];\n}\n\ninterface LoopPersonActivity {\n handle: string;\n userId: string;\n reason: string;\n tweets: Tweet[];\n}\n\ninterface LoopResearchState {\n timeline: Tweet[];\n mentions: Tweet[];\n topicSearchResults: LoopTopicSearchResult[];\n peopleActivity: LoopPersonActivity[];\n}\n\ntype PlannerToolName =\n | \"observe_timeline\"\n | \"observe_mentions\"\n | \"search_tweets\"\n | \"check_profile\"\n | \"reply\"\n | \"like\"\n | \"retweet\"\n | \"follow\"\n | \"post\"\n | \"skip\";\n\ninterface PlannerDecision {\n tool: PlannerToolName;\n args: Record<string, unknown>;\n reasoning?: string;\n}\n\ninterface ToolStepSummary {\n step: number;\n tool: PlannerToolName;\n summary: string;\n}\n\nfunction loopClampInt(value: number, min: number, max: number): number {\n return Math.max(min, Math.min(max, Math.round(value)));\n}\n\nfunction parseJsonObjectFromText(text: string): Record<string, unknown> | null {\n const fenced = text.match(/```(?:json)?\\s*([\\s\\S]*?)```/i)?.[1];\n const candidates = [fenced, text].filter((v): v is string => Boolean(v)).map((v) => v.trim());\n for (const candidate of candidates) {\n try {\n const parsed = JSON.parse(candidate);\n if (parsed && typeof parsed === \"object\" && !Array.isArray(parsed)) {\n return parsed as Record<string, unknown>;\n }\n if (Array.isArray(parsed) && parsed.length > 0 && parsed[0] && typeof parsed[0] === \"object\") {\n return parsed[0] as Record<string, unknown>;\n }\n } catch {\n // Continue to extraction fallback.\n }\n const obj = candidate.match(/\\{[\\s\\S]*\\}/);\n if (!obj) continue;\n try {\n const parsed = JSON.parse(obj[0]);\n if (parsed && typeof parsed === \"object\" && !Array.isArray(parsed)) {\n return parsed as Record<string, unknown>;\n }\n } catch {\n // Continue.\n }\n }\n return null;\n}\n\nfunction mapPlannerTool(raw: string | undefined): PlannerToolName | null {\n const clean = (raw ?? \"\").trim().toLowerCase();\n if (!clean) return null;\n if (clean === \"observe_timeline\" || clean === \"read_timeline\" || clean === \"timeline\") return \"observe_timeline\";\n if (clean === \"observe_mentions\" || clean === \"read_mentions\" || clean === \"mentions\") return \"observe_mentions\";\n if (clean === \"search_tweets\" || clean === \"search\" || clean === \"topic_search\") return \"search_tweets\";\n if (clean === \"check_profile\" || clean === \"check_person\" || clean === \"people_check\" || clean === \"profile_check\") return \"check_profile\";\n if (clean === \"reply\") return \"reply\";\n if (clean === \"like\") return \"like\";\n if (clean === \"retweet\") return \"retweet\";\n if (clean === \"follow\") return \"follow\";\n if (clean === \"post\" || clean === \"tweet\") return \"post\";\n if (clean === \"skip\" || clean === \"noop\" || clean === \"no_action\") return \"skip\";\n return null;\n}\n\nfunction parsePlannerDecision(text: string): PlannerDecision | null {\n const parsed = parseJsonObjectFromText(text);\n if (!parsed) return null;\n\n const rawTool =\n typeof parsed.tool === \"string\"\n ? parsed.tool\n : typeof parsed.action === \"string\"\n ? parsed.action\n : undefined;\n const tool = mapPlannerTool(rawTool);\n if (!tool) return null;\n\n const rawArgs = parsed.args;\n const args: Record<string, unknown> =\n rawArgs && typeof rawArgs === \"object\" && !Array.isArray(rawArgs)\n ? { ...(rawArgs as Record<string, unknown>) }\n : {};\n\n for (const key of [\"tweetId\", \"content\", \"handle\", \"query\", \"count\", \"reason\"]) {\n if (args[key] !== undefined) continue;\n if (parsed[key] !== undefined) args[key] = parsed[key];\n }\n\n const reasoning = typeof parsed.reasoning === \"string\" ? parsed.reasoning.trim() : undefined;\n return { tool, args, reasoning };\n}\n\nfunction argString(args: Record<string, unknown>, key: string): string | undefined {\n const value = args[key];\n if (typeof value !== \"string\") return undefined;\n const clean = value.trim();\n return clean.length > 0 ? clean : undefined;\n}\n\nfunction argCount(args: Record<string, unknown>, fallback: number, min: number, max: number): number {\n const raw = args.count;\n if (typeof raw === \"number\" && Number.isFinite(raw)) {\n return loopClampInt(raw, min, max);\n }\n if (typeof raw === \"string\" && raw.trim().length > 0) {\n const parsed = Number(raw);\n if (Number.isFinite(parsed)) return loopClampInt(parsed, min, max);\n }\n return loopClampInt(fallback, min, max);\n}\n\nfunction tweetTimestamp(tweet: Tweet): number {\n const ts = Date.parse(tweet.createdAt);\n return Number.isNaN(ts) ? 0 : ts;\n}\n\nfunction mergeTweetList(current: Tweet[], incoming: Tweet[]): { tweets: Tweet[]; added: number } {\n const byId = new Map<string, Tweet>();\n for (const tweet of current) byId.set(tweet.id, tweet);\n let added = 0;\n for (const tweet of incoming) {\n if (!tweet.id) continue;\n if (!byId.has(tweet.id)) added += 1;\n byId.set(tweet.id, tweet);\n }\n const tweets = [...byId.values()].sort((a, b) => tweetTimestamp(b) - tweetTimestamp(a));\n return { tweets, added };\n}\n\nfunction mergeTopicResult(\n current: LoopTopicSearchResult[],\n query: string,\n incoming: Tweet[],\n): { next: LoopTopicSearchResult[]; added: number } {\n const idx = current.findIndex((item) => item.query.toLowerCase() === query.toLowerCase());\n if (idx < 0) {\n const merged = mergeTweetList([], incoming);\n return {\n next: [...current, { query, tweets: merged.tweets }],\n added: merged.added,\n };\n }\n const merged = mergeTweetList(current[idx].tweets, incoming);\n const next = [...current];\n next[idx] = { ...next[idx], tweets: merged.tweets };\n return { next, added: merged.added };\n}\n\nfunction mergePeopleActivity(\n current: LoopPersonActivity[],\n incoming: LoopPersonActivity,\n): { next: LoopPersonActivity[]; added: number } {\n const handle = normalizeHandle(incoming.handle);\n const idx = current.findIndex((item) => normalizeHandle(item.handle) === handle);\n if (idx < 0) {\n const merged = mergeTweetList([], incoming.tweets);\n return {\n next: [...current, { ...incoming, handle, tweets: merged.tweets }],\n added: merged.added,\n };\n }\n const merged = mergeTweetList(current[idx].tweets, incoming.tweets);\n const next = [...current];\n next[idx] = {\n ...next[idx],\n userId: incoming.userId || next[idx].userId,\n reason: incoming.reason || next[idx].reason,\n tweets: merged.tweets,\n };\n return { next, added: merged.added };\n}\n\nfunction collectObserved(state: LoopResearchState): {\n ids: string[];\n tweets: Tweet[];\n byId: Map<string, Tweet>;\n} {\n const tweets = [\n ...state.timeline,\n ...state.mentions,\n ...state.topicSearchResults.flatMap((item) => item.tweets),\n ...state.peopleActivity.flatMap((item) => item.tweets),\n ];\n const byId = new Map<string, Tweet>();\n for (const tweet of tweets) {\n if (!tweet.id || byId.has(tweet.id)) continue;\n byId.set(tweet.id, tweet);\n }\n return {\n ids: [...byId.keys()],\n tweets: [...byId.values()],\n byId,\n };\n}\n\nfunction buildPlannerCandidates(\n state: LoopResearchState,\n): Tweet[] {\n const ordered = [\n ...state.mentions,\n ...state.timeline,\n ...state.topicSearchResults.flatMap((item) => item.tweets),\n ...state.peopleActivity.flatMap((item) => item.tweets),\n ];\n const byId = new Map<string, Tweet>();\n for (const tweet of ordered) {\n if (!tweet.id) continue;\n if (!byId.has(tweet.id)) byId.set(tweet.id, tweet);\n }\n return [...byId.values()].slice(0, 24);\n}\n\nfunction buildToolLoopPrompt(input: {\n identity: {\n name: string;\n handle: string;\n bio: string;\n tone: string;\n goals: string[];\n boundaries: string[];\n };\n constraintLines: string[];\n candidates: Tweet[];\n state: LoopResearchState;\n actions: AgentAction[];\n policyFeedback: string[];\n toolHistory: ToolStepSummary[];\n recentMemory: string[];\n usedReplyTargets: string[];\n recentReplyTargets: string[];\n blockedReplyTargets: string[];\n recentWrittenSamples: string[];\n heartbeatCount: number;\n}): string {\n const lines: string[] = [];\n lines.push(\"Choose exactly ONE tool call in strict JSON.\");\n lines.push(\"You are the autonomous planner. No fixed research sequence is required.\");\n lines.push(\"Act based on persona + live context, not templates.\");\n lines.push(\"\");\n lines.push(`Heartbeat: #${input.heartbeatCount}`);\n lines.push(`Agent: ${input.identity.name} (@${input.identity.handle})`);\n lines.push(`Bio: ${input.identity.bio}`);\n lines.push(`Tone: ${input.identity.tone}`);\n lines.push(`Goals: ${input.identity.goals.join(\" | \") || \"none\"}`);\n lines.push(`Boundaries: ${input.identity.boundaries.join(\" | \") || \"none\"}`);\n lines.push(\"\");\n if (input.constraintLines.length > 0) {\n lines.push(\"Hard persona constraints:\");\n for (const line of input.constraintLines) lines.push(line);\n lines.push(\"\");\n }\n if (input.recentMemory.length > 0) {\n lines.push(\"Recent memory:\");\n for (const row of input.recentMemory.slice(0, 8)) {\n lines.push(`- ${row}`);\n }\n lines.push(\"\");\n }\n lines.push(`Executed actions this heartbeat: ${input.actions.length}`);\n lines.push(\n `Current context counts: timeline=${input.state.timeline.length}, mentions=${input.state.mentions.length}, topicTweets=${input.state.topicSearchResults.reduce((sum, item) => sum + item.tweets.length, 0)}, peopleTweets=${input.state.peopleActivity.reduce((sum, item) => sum + item.tweets.length, 0)}`\n );\n if (input.usedReplyTargets.length > 0) {\n lines.push(`Reply targets already used this heartbeat: ${input.usedReplyTargets.slice(-8).join(\", \")}`);\n }\n if (input.recentReplyTargets.length > 0) {\n lines.push(`Recently replied targets (cooldown): ${input.recentReplyTargets.slice(0, 8).join(\", \")}`);\n }\n if (input.blockedReplyTargets.length > 0) {\n lines.push(`Temporarily blocked reply targets this heartbeat: ${input.blockedReplyTargets.slice(0, 8).join(\", \")}`);\n }\n if (input.recentWrittenSamples.length > 0) {\n lines.push(\"Recent writing to avoid repeating:\");\n for (const sample of input.recentWrittenSamples.slice(0, 4)) {\n lines.push(`- ${sample.slice(0, 100)}`);\n }\n }\n lines.push(\"\");\n if (input.toolHistory.length > 0) {\n lines.push(\"Recent tool steps:\");\n for (const row of input.toolHistory.slice(-6)) {\n lines.push(`- Step ${row.step}: ${row.tool} -> ${row.summary}`);\n }\n lines.push(\"\");\n }\n if (input.policyFeedback.length > 0) {\n lines.push(\"Recent policy feedback:\");\n for (const note of input.policyFeedback.slice(-8)) {\n lines.push(`- ${note}`);\n }\n lines.push(\"\");\n }\n if (input.candidates.length > 0) {\n lines.push(`Top candidate tweets (${input.candidates.length}):`);\n for (const tweet of input.candidates.slice(0, 16)) {\n lines.push(\n `- [${tweet.id}] @${tweet.authorHandle}: ${tweet.text.slice(0, 190)} (${tweet.likeCount ?? 0} likes, ${tweet.replyCount ?? 0} replies)`\n );\n }\n lines.push(\"\");\n } else {\n lines.push(\"No candidate tweets currently in context.\");\n lines.push(\"\");\n }\n lines.push(\"Tool options:\");\n lines.push('- observe_timeline: { \"count\": 8-30 }');\n lines.push('- observe_mentions: { \"count\": 8-30 }');\n lines.push('- search_tweets: { \"query\": \"x api search query\", \"count\": 3-20 }');\n lines.push('- check_profile: { \"handle\": \"@username\", \"count\": 3-12 }');\n lines.push('- reply: { \"tweetId\": \"id\", \"content\": \"reply under 220 chars\" }');\n lines.push('- like: { \"tweetId\": \"id\" }');\n lines.push('- retweet: { \"tweetId\": \"id\" }');\n lines.push('- follow: { \"handle\": \"@username\" }');\n lines.push('- post: { \"content\": \"post under 220 chars\" }');\n lines.push('- skip: { \"reason\": \"short reason\" }');\n lines.push(\"\");\n lines.push(\"Output JSON object only:\");\n lines.push('{ \"tool\": \"reply\", \"args\": { \"tweetId\": \"...\", \"content\": \"...\" }, \"reasoning\": \"short why\" }');\n lines.push(\"\");\n lines.push(\"Planning rules:\");\n lines.push(\"- if timeline/mentions already have viable tweets, prefer acting over more research\");\n lines.push(\"- use search/profile checks only when you have a clear reason\");\n lines.push(\"- avoid repeating the same query/target unless context changed\");\n lines.push(\"- one reply per target tweet per heartbeat, always choose a new tweet next\");\n lines.push(\"\");\n lines.push(\"Writing constraints:\");\n lines.push(\"- sound human and specific, no manifesto language\");\n lines.push(\"- avoid 'the real question' / 'the deeper question' phrasing\");\n lines.push(\"- no repetitive anchor terms from recent outputs\");\n lines.push(\"- match the persona tone strictly\");\n return lines.join(\"\\n\");\n}\n\nfunction fallbackPlannerDecision(step: number, observedCount: number): PlannerDecision {\n if (observedCount === 0) {\n return {\n tool: step % 2 === 0 ? \"observe_timeline\" : \"observe_mentions\",\n args: { count: 20 },\n reasoning: \"fallback context bootstrap\",\n };\n }\n return {\n tool: \"skip\",\n args: { reason: \"planner parse fallback; preserve current context\" },\n reasoning: \"fallback no-op\",\n };\n}\n\nfunction decisionToAgentAction(decision: PlannerDecision): AgentAction | null {\n if (decision.tool === \"reply\") {\n const tweetId = argString(decision.args, \"tweetId\");\n const content = argString(decision.args, \"content\");\n if (!tweetId || !content) return null;\n return { action: \"reply\", tweetId, content, reasoning: decision.reasoning };\n }\n if (decision.tool === \"like\") {\n const tweetId = argString(decision.args, \"tweetId\");\n if (!tweetId) return null;\n return { action: \"like\", tweetId, reasoning: decision.reasoning };\n }\n if (decision.tool === \"retweet\") {\n const tweetId = argString(decision.args, \"tweetId\");\n if (!tweetId) return null;\n return { action: \"retweet\", tweetId, reasoning: decision.reasoning };\n }\n if (decision.tool === \"follow\") {\n const handle = argString(decision.args, \"handle\");\n if (!handle) return null;\n return { action: \"follow\", handle, targetHandle: handle, reasoning: decision.reasoning };\n }\n if (decision.tool === \"post\") {\n const content = argString(decision.args, \"content\");\n if (!content) return null;\n return { action: \"post\", content, reasoning: decision.reasoning };\n }\n if (decision.tool === \"skip\") {\n return { action: \"skip\", reason: argString(decision.args, \"reason\"), reasoning: decision.reasoning };\n }\n return null;\n}\n\nexport async function runAutonomyCycle(maxActions: number, heartbeatCount: number = 0): Promise<AutonomyCycleResult> {\n const client = await getXClient();\n const identity = loadIdentity();\n const constraints = getPersonaConstraints();\n const strictReplyHandles = new Set(constraints.onlyReplyToHandles.map((handle) => normalizeHandle(handle)));\n const strictInteractHandles = new Set(\n [...constraints.onlyReplyToHandles, ...constraints.onlyInteractWithHandles].map((handle) => normalizeHandle(handle))\n );\n const constraintLines = buildPersonaConstraintLines(constraints);\n if (constraintLines.length > 0) {\n logger.info(`Persona constraints active: ${constraintLines.join(\" | \")}`);\n }\n logger.info(`Autonomy runtime v2 active: planner-primary, persona+memory driven.`);\n\n try {\n const delayed = await collectDelayedOutcomes(client);\n if (delayed.processed > 0) {\n logger.info(\n `Delayed outcomes: processed=${delayed.processed}, resolved=${delayed.resolved}, rewardDelta=${delayed.rewardAdded.toFixed(2)}`\n );\n }\n } catch (error) {\n logger.warn(`Delayed outcome collection failed: ${(error as Error).message}`);\n }\n\n const selfHandle = identity.handle.replace(/^@/, \"\").toLowerCase();\n const selfUserId = selfUserIdFromCredentials();\n const isSelfTweet = (tweet: Tweet): boolean =>\n tweet.authorHandle.replace(/^@/, \"\").toLowerCase() === selfHandle ||\n (selfUserId !== null && tweet.authorId === selfUserId);\n\n const state: LoopResearchState = {\n timeline: [],\n mentions: [],\n topicSearchResults: [],\n peopleActivity: [],\n };\n\n try {\n const timeline = (await client.getTimeline({ count: 30 })).filter((tweet) => !isSelfTweet(tweet));\n state.timeline = mergeTweetList(state.timeline, timeline).tweets;\n } catch (error) {\n logger.warn(`Seed timeline read failed: ${(error as Error).message}`);\n }\n\n try {\n const mentions = (await client.getMentions({ count: 30 })).filter((tweet) => !isSelfTweet(tweet));\n state.mentions = mergeTweetList(state.mentions, mentions).tweets;\n } catch (error) {\n logger.warn(`Seed mentions read failed: ${(error as Error).message}`);\n }\n\n logger.info(\n `Autonomy context: timeline=${state.timeline.length}, mentions=${state.mentions.length}, topicTweets=${state.topicSearchResults.reduce((sum, item) => sum + item.tweets.length, 0)}, peopleTweets=${state.peopleActivity.reduce((sum, item) => sum + item.tweets.length, 0)}`\n );\n\n const actions: AgentAction[] = [];\n const results: ActionResult[] = [];\n const toolHistory: ToolStepSummary[] = [];\n const policyFeedback: string[] = [\n `Persona: ${identity.name} (@${identity.handle})`,\n `Tone: ${identity.tone}`,\n `Goals: ${(identity.goals ?? []).slice(0, 4).join(\" | \") || \"none\"}`,\n ];\n\n const systemPrompt = buildSystemPrompt();\n const toolSystemPrompt = [\n systemPrompt,\n \"TOOL LOOP MODE: You must choose exactly one tool call each step using strict JSON.\",\n \"Planner-first mode: there is no pre-defined research flow.\",\n \"Everything must be persona-driven, memory-aware, and context-grounded.\",\n \"Do not fall into repetitive searches or repetitive targets.\",\n ].join(\"\\n\\n\");\n const activeIntents = listIntents();\n let staleSteps = 0;\n let attemptedActions = 0;\n const maxConfiguredActions = Math.max(1, maxActions);\n let actionBudget = maxConfiguredActions;\n if (constraints.replyOnlyMode || strictReplyHandles.size > 0) {\n // Strict reply personas should not spray multiple replies in one cycle.\n actionBudget = 1;\n } else {\n // Non-strict personas vary output cadence to avoid robotic fixed-count behavior.\n const upper = Math.min(maxConfiguredActions, 3);\n actionBudget = Math.max(1, Math.floor(Math.random() * upper) + 1);\n }\n const maxToolSteps = Math.max(8, actionBudget * 5);\n const recentInteractions = getRecentInteractions(220);\n const now = Date.now();\n const recentReplyTargets = new Set(\n recentInteractions\n .filter((entry) => entry.type === \"reply\" && typeof entry.inReplyTo === \"string\")\n .filter((entry) => {\n const ts = Date.parse(entry.timestamp);\n return Number.isNaN(ts) || now - ts <= REPLY_TARGET_COOLDOWN_MS;\n })\n .map((entry) => entry.inReplyTo!)\n );\n const recentWrittenSamples = recentInteractions\n .filter((entry) => (entry.type === \"post\" || entry.type === \"reply\") && typeof entry.content === \"string\")\n .map((entry) => entry.content!.trim())\n .filter(Boolean)\n .slice(0, 16);\n const usedReplyTargets = new Set<string>();\n const blockedReplyTargets = new Set<string>();\n const writtenThisHeartbeat: string[] = [];\n\n // Keep short memory available in context by touching it before loop.\n getRecentInteractions(20);\n\n const markNoProgress = (note?: string): boolean => {\n if (note) policyFeedback.push(note);\n staleSteps += 1;\n if (staleSteps >= 5) {\n logger.info(\"Tool loop stopped after repeated blocked/no-progress steps.\");\n return true;\n }\n return false;\n };\n\n const hasEligibleStrictReplyTarget = (): boolean => {\n const observed = collectObserved(state);\n return observed.tweets.some((tweet) => {\n const handle = normalizeHandle(tweet.authorHandle);\n return (\n strictReplyHandles.has(handle) &&\n !usedReplyTargets.has(tweet.id) &&\n !recentReplyTargets.has(tweet.id) &&\n !blockedReplyTargets.has(tweet.id)\n );\n });\n };\n\n for (let step = 0; step < maxToolSteps && actions.length < actionBudget; step += 1) {\n if (strictReplyHandles.size > 0 && !hasEligibleStrictReplyTarget()) {\n policyFeedback.push(\"No eligible fresh target tweets remain for strict reply mode this heartbeat.\");\n logger.info(\"Planner loop ended: no eligible strict-reply targets left.\");\n break;\n }\n\n const candidates = buildPlannerCandidates(state);\n const recentMemory = getRecentInteractions(24)\n .slice(0, 12)\n .map((entry) => {\n const text = (entry.content ?? \"\").replace(/\\s+/g, \" \").trim().slice(0, 120);\n const target = entry.targetHandle ? ` @${entry.targetHandle}` : \"\";\n return `${entry.type}${target}${text ? `: ${text}` : \"\"}`;\n });\n const prompt = buildToolLoopPrompt({\n identity: {\n name: identity.name,\n handle: identity.handle,\n bio: identity.bio,\n tone: identity.tone,\n goals: identity.goals ?? [],\n boundaries: identity.boundaries ?? [],\n },\n constraintLines,\n candidates,\n state,\n actions,\n policyFeedback,\n toolHistory,\n recentMemory,\n usedReplyTargets: [...usedReplyTargets],\n recentReplyTargets: [...recentReplyTargets],\n blockedReplyTargets: [...blockedReplyTargets],\n recentWrittenSamples: [...writtenThisHeartbeat, ...recentWrittenSamples].slice(0, 8),\n heartbeatCount,\n });\n\n let decision = fallbackPlannerDecision(step, candidates.length);\n try {\n const response = await generateResponse(toolSystemPrompt, prompt);\n const parsed = parsePlannerDecision(response.content);\n if (parsed) {\n decision = parsed;\n } else {\n policyFeedback.push(\"Planner returned invalid JSON decision. Using fallback tool.\");\n }\n } catch (error) {\n policyFeedback.push(`Planner call failed: ${(error as Error).message}`);\n }\n\n logger.info(`Tool loop step ${step + 1}: ${decision.tool}`);\n let stepProgress = false;\n\n if (decision.tool === \"observe_timeline\") {\n const count = argCount(decision.args, 16, 8, 30);\n try {\n const incoming = (await client.getTimeline({ count })).filter((tweet) => !isSelfTweet(tweet));\n const merged = mergeTweetList(state.timeline, incoming);\n state.timeline = merged.tweets;\n stepProgress = merged.added > 0;\n toolHistory.push({\n step: step + 1,\n tool: decision.tool,\n summary: `fetched ${incoming.length}, added ${merged.added}`,\n });\n } catch (error) {\n policyFeedback.push(`observe_timeline failed: ${(error as Error).message}`);\n }\n if (!stepProgress && markNoProgress(\"observe_timeline found no new tweets.\")) break;\n continue;\n }\n\n if (decision.tool === \"observe_mentions\") {\n const count = argCount(decision.args, 16, 8, 30);\n try {\n const incoming = (await client.getMentions({ count })).filter((tweet) => !isSelfTweet(tweet));\n const merged = mergeTweetList(state.mentions, incoming);\n state.mentions = merged.tweets;\n stepProgress = merged.added > 0;\n toolHistory.push({\n step: step + 1,\n tool: decision.tool,\n summary: `fetched ${incoming.length}, added ${merged.added}`,\n });\n } catch (error) {\n policyFeedback.push(`observe_mentions failed: ${(error as Error).message}`);\n }\n if (!stepProgress && markNoProgress(\"observe_mentions found no new tweets.\")) break;\n continue;\n }\n\n if (decision.tool === \"search_tweets\") {\n const rawQuery = argString(decision.args, \"query\");\n if (!rawQuery) {\n if (markNoProgress(\"search_tweets rejected: missing query.\")) break;\n continue;\n }\n const query = canonicalizePlannerQuery(rawQuery);\n if (query !== rawQuery) {\n policyFeedback.push(`search_tweets normalized handle query: \"${rawQuery}\" -> \"${query}\"`);\n }\n const count = argCount(decision.args, 10, 3, 20);\n try {\n const incoming = (await client.searchTweets(query, { count })).filter((tweet) => !isSelfTweet(tweet));\n const merged = mergeTopicResult(state.topicSearchResults, query, incoming);\n state.topicSearchResults = merged.next;\n stepProgress = merged.added > 0;\n toolHistory.push({\n step: step + 1,\n tool: decision.tool,\n summary: `\"${query}\" fetched ${incoming.length}, added ${merged.added}`,\n });\n logger.info(`Tool search \"${query}\": found ${incoming.length} tweets`);\n } catch (error) {\n policyFeedback.push(`search_tweets failed for \"${query}\": ${(error as Error).message}`);\n }\n if (!stepProgress && markNoProgress(`search_tweets \"${query}\" found no new tweets.`)) break;\n continue;\n }\n\n if (decision.tool === \"check_profile\") {\n const handleInput = argString(decision.args, \"handle\");\n const handle = handleInput ? canonicalPlannerHandle(handleInput) : \"\";\n if (!handle) {\n if (markNoProgress(\"check_profile rejected: missing handle.\")) break;\n continue;\n }\n const count = argCount(decision.args, 6, 3, 12);\n try {\n const profile = await client.getProfile(handle);\n const tweets = (await client.getUserTweets(profile.id, { count })).filter((tweet) => !isSelfTweet(tweet));\n const merged = mergePeopleActivity(state.peopleActivity, {\n handle,\n userId: profile.id,\n reason: \"tool-loop check\",\n tweets,\n });\n state.peopleActivity = merged.next;\n stepProgress = merged.added > 0;\n toolHistory.push({\n step: step + 1,\n tool: decision.tool,\n summary: `@${handle} fetched ${tweets.length}, added ${merged.added}`,\n });\n logger.info(`People check @${handle}: ${tweets.length} recent tweets`);\n } catch (error) {\n policyFeedback.push(`check_profile failed for @${handle}: ${(error as Error).message}`);\n }\n if (!stepProgress && markNoProgress(`check_profile @${handle} found no new tweets.`)) break;\n continue;\n }\n\n const observed = collectObserved(state);\n let candidateAction = decisionToAgentAction(decision);\n if (!candidateAction) {\n if (markNoProgress(`Tool decision ${decision.tool} rejected: invalid arguments.`)) break;\n continue;\n }\n\n if (candidateAction.tweetId && observed.byId.has(candidateAction.tweetId)) {\n const author = normalizeHandle(observed.byId.get(candidateAction.tweetId)?.authorHandle);\n if (author && !candidateAction.targetHandle) {\n candidateAction.targetHandle = `@${author}`;\n }\n }\n\n const actionTargetHandle = normalizeHandle(\n candidateAction.targetHandle ??\n candidateAction.handle ??\n (candidateAction.tweetId ? observed.byId.get(candidateAction.tweetId)?.authorHandle : undefined)\n );\n\n if (candidateAction.action === \"skip\") {\n const reason = candidateAction.reason ?? candidateAction.reasoning ?? \"planner skip\";\n policyFeedback.push(`Planner chose skip: ${reason}`);\n toolHistory.push({\n step: step + 1,\n tool: decision.tool,\n summary: reason,\n });\n if (markNoProgress()) break;\n continue;\n }\n\n if (constraints.noOriginalPosts && (candidateAction.action === \"post\" || candidateAction.action === \"schedule\")) {\n const reason = \"Original posts are disabled by persona constraints.\";\n policyFeedback.push(reason);\n logger.info(`Policy blocked action ${candidateAction.action}: ${reason}`);\n if (markNoProgress()) break;\n continue;\n }\n\n if (strictReplyHandles.size > 0) {\n if (candidateAction.action !== \"reply\") {\n const reason = `Only reply actions are allowed for this persona target: @${[...strictReplyHandles].join(\", @\")}.`;\n policyFeedback.push(reason);\n logger.info(`Policy blocked action ${candidateAction.action}: ${reason}`);\n if (markNoProgress()) break;\n continue;\n }\n if (!actionTargetHandle || !strictReplyHandles.has(actionTargetHandle)) {\n const reason = `Reply target must be one of @${[...strictReplyHandles].join(\", @\")}.`;\n policyFeedback.push(reason);\n logger.info(`Policy blocked action ${candidateAction.action}: ${reason}`);\n if (markNoProgress()) break;\n continue;\n }\n }\n\n if (\n strictInteractHandles.size > 0 &&\n [\"reply\", \"like\", \"retweet\", \"follow\"].includes(candidateAction.action)\n ) {\n if (!actionTargetHandle || !strictInteractHandles.has(actionTargetHandle)) {\n const reason = `Interaction target must be one of @${[...strictInteractHandles].join(\", @\")}.`;\n policyFeedback.push(reason);\n logger.info(`Policy blocked action ${candidateAction.action}: ${reason}`);\n if (markNoProgress()) break;\n continue;\n }\n }\n\n if (candidateAction.action === \"reply\" && candidateAction.tweetId) {\n if (blockedReplyTargets.has(candidateAction.tweetId)) {\n const reason = `Reply target ${candidateAction.tweetId} is temporarily blocked this heartbeat. Pick a different tweet.`;\n policyFeedback.push(reason);\n logger.info(`Planner loop guard: ${reason}`);\n if (markNoProgress()) break;\n continue;\n }\n if (usedReplyTargets.has(candidateAction.tweetId)) {\n const reason = `Reply target ${candidateAction.tweetId} already used this heartbeat. Pick a different tweet.`;\n policyFeedback.push(reason);\n logger.info(`Planner loop guard: ${reason}`);\n blockedReplyTargets.add(candidateAction.tweetId);\n if (markNoProgress()) break;\n continue;\n }\n if (recentReplyTargets.has(candidateAction.tweetId)) {\n const reason = `Reply target ${candidateAction.tweetId} is in cooldown from recent history. Pick a fresher target.`;\n policyFeedback.push(reason);\n logger.info(`Planner loop guard: ${reason}`);\n blockedReplyTargets.add(candidateAction.tweetId);\n if (markNoProgress()) break;\n continue;\n }\n }\n\n if ((candidateAction.action === \"reply\" || candidateAction.action === \"post\") && candidateAction.content) {\n const seenDrafts = [...writtenThisHeartbeat, ...recentWrittenSamples].slice(0, 18);\n if (nearDuplicateDraft(candidateAction.content, seenDrafts)) {\n const reason = \"Draft is too similar to recent writing. Try a new angle before posting.\";\n policyFeedback.push(reason);\n logger.info(`Planner loop guard: ${reason}`);\n if (markNoProgress()) break;\n continue;\n }\n }\n\n attemptedActions += 1;\n let policy = evaluateActionPolicy({\n action: candidateAction,\n step: attemptedActions,\n timeline: state.timeline,\n mentions: state.mentions,\n executedActions: actions,\n observedTweetIds: observed.ids,\n observedTweets: observed.tweets,\n selfHandle,\n selfUserId,\n });\n\n if (!policy.allowed && shouldAttemptStyleRewrite(candidateAction, policy.reason ?? \"\")) {\n const recentTexts = getRecentInteractions(10)\n .filter((entry) => entry.type === \"post\" || entry.type === \"reply\")\n .map((entry) => entry.content ?? \"\")\n .filter(Boolean);\n const rewritten = await rewriteDraftForHumanVoice({\n action: candidateAction,\n reason: policy.reason ?? \"policy rejected\",\n targetTweetText: candidateAction.tweetId ? observed.byId.get(candidateAction.tweetId)?.text : undefined,\n recentTexts,\n });\n\n if (rewritten && rewritten !== candidateAction.content) {\n candidateAction = {\n ...candidateAction,\n content: rewritten,\n reasoning: `${candidateAction.reasoning ?? \"rewritten\"} | style rewrite`,\n };\n policy = evaluateActionPolicy({\n action: candidateAction,\n step: attemptedActions,\n timeline: state.timeline,\n mentions: state.mentions,\n executedActions: actions,\n observedTweetIds: observed.ids,\n observedTweets: observed.tweets,\n selfHandle,\n selfUserId,\n });\n }\n }\n\n if (!policy.allowed) {\n const reason = policy.reason ?? \"Policy rejected action\";\n if (shouldHardBlockPolicyReason(reason)) {\n policyFeedback.push(`Policy blocked action: ${reason}`);\n logger.info(`Policy blocked ${candidateAction.action}: ${reason}`);\n if (markNoProgress()) break;\n continue;\n }\n // Policy is fully advisory in planner-primary mode.\n policyFeedback.push(`Policy advisory (not blocking): ${reason}`);\n logger.info(`Policy advisory for ${candidateAction.action}: ${reason}`);\n }\n\n if (candidateAction.action === \"reply\" && candidateAction.tweetId) {\n usedReplyTargets.add(candidateAction.tweetId);\n }\n if ((candidateAction.action === \"reply\" || candidateAction.action === \"post\") && candidateAction.content) {\n writtenThisHeartbeat.unshift(candidateAction.content.trim());\n if (writtenThisHeartbeat.length > 16) {\n writtenThisHeartbeat.length = 16;\n }\n }\n\n const result = await executeAction(candidateAction);\n actions.push(candidateAction);\n results.push(result);\n staleSteps = 0;\n stepProgress = true;\n toolHistory.push({\n step: step + 1,\n tool: decision.tool,\n summary: `${candidateAction.action}${result.success ? \" success\" : \" failed\"}`,\n });\n\n if (activeIntents.length > 0) {\n const touchedHandle = normalizeHandle(\n candidateAction.targetHandle ??\n candidateAction.handle ??\n (candidateAction.tweetId ? observed.byId.get(candidateAction.tweetId)?.authorHandle : undefined)\n );\n const loweredContent = (candidateAction.content ?? \"\").toLowerCase();\n for (const intent of activeIntents) {\n const handleMatch =\n touchedHandle.length > 0 &&\n intent.targetHandles.some((handle) => normalizeHandle(handle) === touchedHandle);\n const topicMatch =\n loweredContent.length > 0 &&\n intent.focusTopics.some((topic) => loweredContent.includes(topic.toLowerCase()));\n if (!handleMatch && !topicMatch) continue;\n recordIntentExecution(intent.id, {\n success: result.success,\n note: `${candidateAction.action}${touchedHandle ? ` @${touchedHandle}` : \"\"}`,\n });\n }\n }\n\n if (!result.success) {\n const err = result.error ?? \"\";\n if (candidateAction.action === \"reply\" && /duplicate content/i.test(err)) {\n const reason = \"Reply failed with duplicate-content error. Planner should choose a different wording/target.\";\n policyFeedback.push(reason);\n if (candidateAction.tweetId) blockedReplyTargets.add(candidateAction.tweetId);\n logger.info(`Policy adjustment: ${reason}`);\n }\n if (candidateAction.action === \"post\" && /duplicate content/i.test(err)) {\n const reason = \"Post failed with duplicate-content error. Change framing and try a different angle.\";\n policyFeedback.push(reason);\n logger.info(`Policy adjustment: ${reason}`);\n }\n }\n\n if (!stepProgress && markNoProgress()) {\n break;\n }\n }\n\n recordBanditActionResults(actions, results);\n\n return {\n timeline: state.timeline,\n mentions: state.mentions,\n actions,\n results,\n policyFeedback,\n };\n}\n"],"mappings":"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","names":["normalizeHandle","merged"]}
@@ -0,0 +1,91 @@
1
+ import {
2
+ logInteraction
3
+ } from "./chunk-6WBIVXOY.js";
4
+
5
+ // src/runtime/heartbeat-narrative.ts
6
+ function clip(text, max = 70) {
7
+ if (!text) return "";
8
+ const clean = text.replace(/\s+/g, " ").trim();
9
+ if (clean.length <= max) return clean;
10
+ return `${clean.slice(0, max - 3)}...`;
11
+ }
12
+ function actionLine(action, result) {
13
+ const outcome = result.success ? "sent" : "tried";
14
+ if (action.action === "post") {
15
+ return `${outcome} post "${clip(action.content)}"`;
16
+ }
17
+ if (action.action === "reply") {
18
+ const who = action.targetHandle ? ` to ${action.targetHandle}` : "";
19
+ return `${outcome} reply${who}: "${clip(action.content)}"`;
20
+ }
21
+ if (action.action === "follow") {
22
+ return `${outcome} follow @${action.handle ?? "unknown"}`;
23
+ }
24
+ if (action.action === "like" || action.action === "retweet") {
25
+ const who = action.targetHandle ? ` ${action.targetHandle}` : "";
26
+ return `${outcome} ${action.action}${who}`;
27
+ }
28
+ if (action.action === "skip") {
29
+ return `skip ${clip(action.reason ?? action.reasoning, 40)}`;
30
+ }
31
+ return `${outcome} ${action.action}`;
32
+ }
33
+ function buildHeartbeatNarrative(input) {
34
+ const { heartbeatCount, timelineCount, mentionsCount, actions, results } = input;
35
+ const successCount = results.filter((r) => r.success).length;
36
+ const failureCount = results.length - successCount;
37
+ const details = [];
38
+ for (let i = 0; i < actions.length; i += 1) {
39
+ const action = actions[i];
40
+ const result = results[i];
41
+ if (!action || !result) continue;
42
+ details.push(actionLine(action, result));
43
+ }
44
+ if (actions.length === 0) {
45
+ return {
46
+ summary: `Heartbeat #${heartbeatCount}: I checked ${timelineCount} timeline posts and ${mentionsCount} mentions. Quiet cycle, so I stayed in character and waited for a better opening.`,
47
+ details
48
+ };
49
+ }
50
+ const successfulActions = actions.filter((_, idx) => results[idx]?.success);
51
+ const replyCount = successfulActions.filter((a) => a.action === "reply").length;
52
+ const postCount = successfulActions.filter((a) => a.action === "post").length;
53
+ const likeCount = successfulActions.filter((a) => a.action === "like").length;
54
+ const retweetCount = successfulActions.filter((a) => a.action === "retweet").length;
55
+ const followCount = successfulActions.filter((a) => a.action === "follow").length;
56
+ const replyTargets = [...new Set(
57
+ successfulActions.filter((a) => a.action === "reply" && typeof a.targetHandle === "string").map((a) => String(a.targetHandle))
58
+ )];
59
+ const activityParts = [];
60
+ if (replyCount > 0) {
61
+ const targetText = replyTargets.length > 0 ? ` to ${replyTargets.join(" and ")}` : "";
62
+ activityParts.push(`sent ${replyCount} repl${replyCount === 1 ? "y" : "ies"}${targetText}`);
63
+ }
64
+ if (postCount > 0) activityParts.push(`posted ${postCount} original ${postCount === 1 ? "tweet" : "tweets"}`);
65
+ if (likeCount > 0) activityParts.push(`liked ${likeCount} ${likeCount === 1 ? "tweet" : "tweets"}`);
66
+ if (retweetCount > 0) activityParts.push(`retweeted ${retweetCount} ${retweetCount === 1 ? "post" : "posts"}`);
67
+ if (followCount > 0) activityParts.push(`followed ${followCount} ${followCount === 1 ? "account" : "accounts"}`);
68
+ const activityText = activityParts.length > 0 ? activityParts.join(", ") : `tried ${actions.length} ${actions.length === 1 ? "action" : "actions"}`;
69
+ const failureText = failureCount > 0 ? ` I adjusted course after ${failureCount} rough attempt${failureCount === 1 ? "" : "s"} and will try a fresher angle next cycle.` : "";
70
+ return {
71
+ summary: `Heartbeat #${heartbeatCount}: I scanned ${timelineCount} timeline posts and ${mentionsCount} mentions, then ${activityText}. ${successCount} action${successCount === 1 ? "" : "s"} landed.${failureText}`,
72
+ details
73
+ };
74
+ }
75
+ function saveHeartbeatNarrative(narrative, success) {
76
+ logInteraction({
77
+ id: `heartbeat-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`,
78
+ timestamp: (/* @__PURE__ */ new Date()).toISOString(),
79
+ type: "heartbeat",
80
+ content: narrative.slice(0, 500),
81
+ targetHandle: "self",
82
+ creditsUsed: 0,
83
+ success
84
+ });
85
+ }
86
+
87
+ export {
88
+ buildHeartbeatNarrative,
89
+ saveHeartbeatNarrative
90
+ };
91
+ //# sourceMappingURL=chunk-NX7FJ4GH.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/runtime/heartbeat-narrative.ts"],"sourcesContent":["import { logInteraction } from \"../memory/index.js\";\nimport type { AgentAction, ActionResult } from \"./decision-engine.js\";\n\ninterface BuildHeartbeatNarrativeInput {\n heartbeatCount: number;\n timelineCount: number;\n mentionsCount: number;\n actions: AgentAction[];\n results: ActionResult[];\n}\n\nexport interface HeartbeatNarrative {\n summary: string;\n details: string[];\n}\n\nfunction clip(text: string | undefined, max: number = 70): string {\n if (!text) return \"\";\n const clean = text.replace(/\\s+/g, \" \").trim();\n if (clean.length <= max) return clean;\n return `${clean.slice(0, max - 3)}...`;\n}\n\nfunction actionLine(action: AgentAction, result: ActionResult): string {\n const outcome = result.success ? \"sent\" : \"tried\";\n if (action.action === \"post\") {\n return `${outcome} post \"${clip(action.content)}\"`;\n }\n if (action.action === \"reply\") {\n const who = action.targetHandle ? ` to ${action.targetHandle}` : \"\";\n return `${outcome} reply${who}: \"${clip(action.content)}\"`;\n }\n if (action.action === \"follow\") {\n return `${outcome} follow @${action.handle ?? \"unknown\"}`;\n }\n if (action.action === \"like\" || action.action === \"retweet\") {\n const who = action.targetHandle ? ` ${action.targetHandle}` : \"\";\n return `${outcome} ${action.action}${who}`;\n }\n if (action.action === \"skip\") {\n return `skip ${clip(action.reason ?? action.reasoning, 40)}`;\n }\n return `${outcome} ${action.action}`;\n}\n\nexport function buildHeartbeatNarrative(input: BuildHeartbeatNarrativeInput): HeartbeatNarrative {\n const { heartbeatCount, timelineCount, mentionsCount, actions, results } = input;\n const successCount = results.filter((r) => r.success).length;\n const failureCount = results.length - successCount;\n\n const details: string[] = [];\n for (let i = 0; i < actions.length; i += 1) {\n const action = actions[i];\n const result = results[i];\n if (!action || !result) continue;\n details.push(actionLine(action, result));\n }\n\n if (actions.length === 0) {\n return {\n summary: `Heartbeat #${heartbeatCount}: I checked ${timelineCount} timeline posts and ${mentionsCount} mentions. Quiet cycle, so I stayed in character and waited for a better opening.`,\n details,\n };\n }\n\n const successfulActions = actions.filter((_, idx) => results[idx]?.success);\n const replyCount = successfulActions.filter((a) => a.action === \"reply\").length;\n const postCount = successfulActions.filter((a) => a.action === \"post\").length;\n const likeCount = successfulActions.filter((a) => a.action === \"like\").length;\n const retweetCount = successfulActions.filter((a) => a.action === \"retweet\").length;\n const followCount = successfulActions.filter((a) => a.action === \"follow\").length;\n\n const replyTargets = [...new Set(\n successfulActions\n .filter((a) => a.action === \"reply\" && typeof a.targetHandle === \"string\")\n .map((a) => String(a.targetHandle))\n )];\n\n const activityParts: string[] = [];\n if (replyCount > 0) {\n const targetText = replyTargets.length > 0 ? ` to ${replyTargets.join(\" and \")}` : \"\";\n activityParts.push(`sent ${replyCount} repl${replyCount === 1 ? \"y\" : \"ies\"}${targetText}`);\n }\n if (postCount > 0) activityParts.push(`posted ${postCount} original ${postCount === 1 ? \"tweet\" : \"tweets\"}`);\n if (likeCount > 0) activityParts.push(`liked ${likeCount} ${likeCount === 1 ? \"tweet\" : \"tweets\"}`);\n if (retweetCount > 0) activityParts.push(`retweeted ${retweetCount} ${retweetCount === 1 ? \"post\" : \"posts\"}`);\n if (followCount > 0) activityParts.push(`followed ${followCount} ${followCount === 1 ? \"account\" : \"accounts\"}`);\n\n const activityText =\n activityParts.length > 0\n ? activityParts.join(\", \")\n : `tried ${actions.length} ${actions.length === 1 ? \"action\" : \"actions\"}`;\n\n const failureText =\n failureCount > 0\n ? ` I adjusted course after ${failureCount} rough attempt${failureCount === 1 ? \"\" : \"s\"} and will try a fresher angle next cycle.`\n : \"\";\n\n return {\n summary: `Heartbeat #${heartbeatCount}: I scanned ${timelineCount} timeline posts and ${mentionsCount} mentions, then ${activityText}. ${successCount} action${successCount === 1 ? \"\" : \"s\"} landed.${failureText}`,\n details,\n };\n}\n\nexport function saveHeartbeatNarrative(narrative: string, success: boolean): void {\n logInteraction({\n id: `heartbeat-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`,\n timestamp: new Date().toISOString(),\n type: \"heartbeat\",\n content: narrative.slice(0, 500),\n targetHandle: \"self\",\n creditsUsed: 0,\n success,\n });\n}\n"],"mappings":"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","names":[]}
package/dist/cli.js CHANGED
@@ -67,7 +67,7 @@ program.command("init").description("Set up X account credentials for your Spore
67
67
  console.log(chalk.cyan(BANNER));
68
68
  console.log(chalk.bold("Welcome to Spora."));
69
69
  console.log(chalk.gray("The global town square for AI agents.\n"));
70
- const { runInit } = await import("./init-KL6XSI7E.js");
70
+ const { runInit } = await import("./init-S5ZKOU64.js");
71
71
  await runInit(opts.token);
72
72
  });
73
73
  program.command("serve").description("Start the Spora MCP server (stdio)").action(async () => {
@@ -79,7 +79,7 @@ program.command("chat").description("Open web-based chat interface with your Spo
79
79
  console.log(chalk.red("\u2717 No identity found. Run `spora create` first."));
80
80
  process.exit(1);
81
81
  }
82
- const { startWebChat } = await import("./web-chat-XNTIDKAS.js");
82
+ const { startWebChat } = await import("./web-chat-PQ6FR6BJ.js");
83
83
  await startWebChat();
84
84
  });
85
85
  program.command("tui").description("Start terminal-based chat interface (TUI)").action(async () => {
@@ -501,11 +501,11 @@ program.command("start").description("Start the autonomous Spora agent").option(
501
501
  }
502
502
  console.log(chalk.cyan(BANNER));
503
503
  console.log(chalk.bold("Starting Spora agent...\n"));
504
- const { startHeartbeatLoop } = await import("./heartbeat-CUM7FIHS.js");
504
+ const { startHeartbeatLoop } = await import("./heartbeat-BIGGV4LO.js");
505
505
  await startHeartbeatLoop();
506
506
  });
507
507
  program.command("stop").description("Stop the running Spora agent").action(async () => {
508
- const { getRunningPid, requestStop } = await import("./heartbeat-CUM7FIHS.js");
508
+ const { getRunningPid, requestStop } = await import("./heartbeat-BIGGV4LO.js");
509
509
  const pid = getRunningPid();
510
510
  if (!pid) {
511
511
  console.log(JSON.stringify({ message: "Spora agent is not running." }));
@@ -587,7 +587,7 @@ llm.command("test").description("Send a tiny test prompt to validate current LLM
587
587
  }
588
588
  });
589
589
  program.command("agent-status").description("Check if the Spora agent is running").action(async () => {
590
- const { getRunningPid } = await import("./heartbeat-CUM7FIHS.js");
590
+ const { getRunningPid } = await import("./heartbeat-BIGGV4LO.js");
591
591
  const pid = getRunningPid();
592
592
  const { hasLLMKey } = await import("./llm-IJBRQ7O2.js");
593
593
  console.log(JSON.stringify({
@@ -1,6 +1,6 @@
1
1
  import {
2
2
  runAutonomyCycle
3
- } from "./chunk-TTDQZI5W.js";
3
+ } from "./chunk-245JDXK4.js";
4
4
  import "./chunk-TKGB5LIN.js";
5
5
  import {
6
6
  flushQueue
@@ -18,7 +18,7 @@ import "./chunk-ER6TILYZ.js";
18
18
  import {
19
19
  buildHeartbeatNarrative,
20
20
  saveHeartbeatNarrative
21
- } from "./chunk-OLYPPXKP.js";
21
+ } from "./chunk-NX7FJ4GH.js";
22
22
  import {
23
23
  loadIdentity
24
24
  } from "./chunk-IULO3GRE.js";
@@ -266,4 +266,4 @@ export {
266
266
  requestStop,
267
267
  startHeartbeatLoop
268
268
  };
269
- //# sourceMappingURL=heartbeat-CUM7FIHS.js.map
269
+ //# sourceMappingURL=heartbeat-BIGGV4LO.js.map
@@ -1,11 +1,11 @@
1
1
  import {
2
2
  buildHeartbeatNarrative,
3
3
  saveHeartbeatNarrative
4
- } from "./chunk-OLYPPXKP.js";
4
+ } from "./chunk-NX7FJ4GH.js";
5
5
  import "./chunk-6WBIVXOY.js";
6
6
  import "./chunk-ZWKTKWS6.js";
7
7
  export {
8
8
  buildHeartbeatNarrative,
9
9
  saveHeartbeatNarrative
10
10
  };
11
- //# sourceMappingURL=heartbeat-narrative-B3RD3OPJ.js.map
11
+ //# sourceMappingURL=heartbeat-narrative-UMLQ3WU7.js.map
@@ -276,7 +276,7 @@ async function loginFlow() {
276
276
  console.log(chalk.green("\u2713 Logged in!\n"));
277
277
  console.log(chalk.gray("Opening chat interface...\n"));
278
278
  try {
279
- const { startWebChat } = await import("./web-chat-XNTIDKAS.js");
279
+ const { startWebChat } = await import("./web-chat-PQ6FR6BJ.js");
280
280
  await startWebChat();
281
281
  } catch (error) {
282
282
  console.log(chalk.yellow(`Could not start chat interface: ${error.message}
@@ -368,7 +368,7 @@ async function showDoneAndOpenChat() {
368
368
  console.log(chalk.bold.cyan("\u2501\u2501\u2501 Your Spore is Ready! \u2501\u2501\u2501\n"));
369
369
  console.log(chalk.gray("Opening chat interface...\n"));
370
370
  try {
371
- const { startWebChat } = await import("./web-chat-XNTIDKAS.js");
371
+ const { startWebChat } = await import("./web-chat-PQ6FR6BJ.js");
372
372
  await startWebChat();
373
373
  } catch (error) {
374
374
  console.log(chalk.yellow(`Could not start chat interface: ${error.message}
@@ -506,4 +506,4 @@ async function runInit(token) {
506
506
  export {
507
507
  runInit
508
508
  };
509
- //# sourceMappingURL=init-KL6XSI7E.js.map
509
+ //# sourceMappingURL=init-S5ZKOU64.js.map