oh-my-opencode/src/tools/background-task/full-session-format.ts
YeonGyu-Kim 119e18c810 refactor: wave 2 - split atlas, auto-update-checker, session-recovery, todo-enforcer, background-task hooks
- Extract atlas/ into 15 focused modules (hook, event handler, tool policies, types, etc.)
- Split auto-update-checker into checker/ and hook/ subdirectories with single-purpose files
- Decompose session-recovery into separate recovery strategy files per error type
- Extract todo-continuation-enforcer from monolith to directory with dedicated modules
- Split background-task/tools.ts into individual tool creator files
- Extract command-executor, tmux-utils into focused sub-modules
- Split config/schema.ts into domain-specific schema files
- Decompose cli/config-manager.ts into focused modules
- Rollback skill-mcp-manager, model-availability, index.ts splits that broke tests
- Fix all import path depths for moved files (../../ -> ../../../)
- Add explicit type annotations to resolve TS7006 implicit any errors

Typecheck: 0 errors
Tests: 2359 pass, 5 fail (all pre-existing)
2026-02-08 15:01:42 +09:00

149 lines
4.8 KiB
TypeScript

import type { BackgroundTask } from "../../features/background-agent"
import type { BackgroundOutputClient, BackgroundOutputMessagesResult, BackgroundOutputMessage } from "./clients"
import { extractMessages, getErrorMessage } from "./session-messages"
import { formatMessageTime } from "./time-format"
import { truncateText } from "./truncate-text"
import { formatTaskStatus } from "./task-status-format"
const MAX_MESSAGE_LIMIT = 100
const THINKING_MAX_CHARS = 2000
function extractToolResultText(part: NonNullable<BackgroundOutputMessage["parts"]>[number]): string[] {
if (typeof part.content === "string" && part.content.length > 0) {
return [part.content]
}
if (Array.isArray(part.content)) {
const blocks: string[] = []
for (const block of part.content) {
if ((block.type === "text" || block.type === "reasoning") && block.text) {
blocks.push(block.text)
}
}
if (blocks.length > 0) return blocks
}
if (part.output && part.output.length > 0) {
return [part.output]
}
return []
}
export async function formatFullSession(
task: BackgroundTask,
client: BackgroundOutputClient,
options: {
includeThinking: boolean
messageLimit?: number
sinceMessageId?: string
includeToolResults: boolean
thinkingMaxChars?: number
}
): Promise<string> {
if (!task.sessionID) {
return formatTaskStatus(task)
}
const messagesResult: BackgroundOutputMessagesResult = await client.session.messages({
path: { id: task.sessionID },
})
const errorMessage = getErrorMessage(messagesResult)
if (errorMessage) {
return `Error fetching messages: ${errorMessage}`
}
const rawMessages = extractMessages(messagesResult)
if (!Array.isArray(rawMessages)) {
return "Error fetching messages: invalid response"
}
const sortedMessages = [...rawMessages].sort((a, b) => {
const timeA = String(a.info?.time ?? "")
const timeB = String(b.info?.time ?? "")
return timeA.localeCompare(timeB)
})
let filteredMessages = sortedMessages
if (options.sinceMessageId) {
const index = filteredMessages.findIndex((message) => message.id === options.sinceMessageId)
if (index === -1) {
return `Error: since_message_id not found: ${options.sinceMessageId}`
}
filteredMessages = filteredMessages.slice(index + 1)
}
const includeThinking = options.includeThinking
const includeToolResults = options.includeToolResults
const thinkingMaxChars = options.thinkingMaxChars ?? THINKING_MAX_CHARS
const normalizedMessages: BackgroundOutputMessage[] = []
for (const message of filteredMessages) {
const parts = (message.parts ?? []).filter((part) => {
if (part.type === "thinking" || part.type === "reasoning") {
return includeThinking
}
if (part.type === "tool_result") {
return includeToolResults
}
return part.type === "text"
})
if (parts.length === 0) {
continue
}
normalizedMessages.push({ ...message, parts })
}
const limit = typeof options.messageLimit === "number" ? Math.min(options.messageLimit, MAX_MESSAGE_LIMIT) : undefined
const hasMore = limit !== undefined && normalizedMessages.length > limit
const visibleMessages = limit !== undefined ? normalizedMessages.slice(0, limit) : normalizedMessages
const lines: string[] = []
lines.push("# Full Session Output")
lines.push("")
lines.push(`Task ID: ${task.id}`)
lines.push(`Description: ${task.description}`)
lines.push(`Status: ${task.status}`)
lines.push(`Session ID: ${task.sessionID}`)
lines.push(`Total messages: ${normalizedMessages.length}`)
lines.push(`Returned: ${visibleMessages.length}`)
lines.push(`Has more: ${hasMore ? "true" : "false"}`)
lines.push("")
lines.push("## Messages")
if (visibleMessages.length === 0) {
lines.push("")
lines.push("(No messages found)")
return lines.join("\n")
}
for (const message of visibleMessages) {
const role = message.info?.role ?? "unknown"
const agent = message.info?.agent ? ` (${message.info.agent})` : ""
const time = formatMessageTime(message.info?.time)
const idLabel = message.id ? ` id=${message.id}` : ""
lines.push("")
lines.push(`[${role}${agent}] ${time}${idLabel}`)
for (const part of message.parts ?? []) {
if (part.type === "text" && part.text) {
lines.push(part.text.trim())
} else if (part.type === "thinking" && part.thinking) {
lines.push(`[thinking] ${truncateText(part.thinking, thinkingMaxChars)}`)
} else if (part.type === "reasoning" && part.text) {
lines.push(`[thinking] ${truncateText(part.text, thinkingMaxChars)}`)
} else if (part.type === "tool_result") {
const toolTexts = extractToolResultText(part)
for (const toolText of toolTexts) {
lines.push(`[tool result] ${toolText}`)
}
}
}
}
return lines.join("\n")
}