fix: prevent agent communication bounds from hiding CI regressions

Tighten the UDS auth, framing, and response-reader boundaries while keeping the AgentSummary lifecycle covered so Codecov and CI fail on real regressions instead of missing coverage. The poorMode settings mock mirrors unrelated real settings defaults to avoid Bun mock retention changing later permission tests.

Constraint: PR #369 must fix Codecov/CI precisely without warning suppression, fallback masking, or mock pollution

Rejected: Delete AgentSummary lifecycle coverage | would hide Codecov loss and stale-summary behavior

Rejected: Store inline UDS rejection in a hidden input sentinel | cloned observable inputs can drop it and bypass rejection

Rejected: Ignore malformed UDS frames until timeout | leaves client slots and SendMessage calls open to exhaustion

Confidence: high

Scope-risk: moderate

Directive: Keep empty #token= markers rejected; do not require a non-empty token value in hasInlineUdsToken

Tested: bun test packages/builtin-tools/src/tools/SendMessageTool/__tests__/udsRecipientSanitization.test.ts src/utils/__tests__/udsMessaging.test.ts src/utils/__tests__/udsResponseReader.test.ts src/utils/__tests__/ndjsonFramer.test.ts

Tested: bunx tsc --noEmit --pretty false

Tested: bun run lint

Tested: bun test --coverage --coverage-reporter lcov --coverage-dir coverage

Tested: bun run test:all

Tested: bun audit

Tested: bun run build

Tested: bun run build:vite

Not-tested: GitHub-hosted Codecov upload until pushed PR checks rerun
This commit is contained in:
unraid
2026-04-27 12:50:08 +08:00
parent ee0d788e58
commit 379928fa10
21 changed files with 1725 additions and 335 deletions

View File

@@ -0,0 +1,180 @@
import { describe, expect, test } from 'bun:test'
import type { Message } from 'src/types/message.js'
import { filterIncompleteToolCalls } from '../filterIncompleteToolCalls.js'
describe('filterIncompleteToolCalls', () => {
test('drops assistant tool uses that do not have matching results', () => {
const messages = [
{
type: 'assistant',
uuid: 'a1',
message: {
role: 'assistant',
content: [{ type: 'tool_use', id: 'missing', name: 'Read' }],
},
},
{
type: 'user',
uuid: 'u1',
message: { role: 'user', content: 'continue' },
},
] as unknown as Message[]
expect(
filterIncompleteToolCalls(messages).map(message => String(message.uuid)),
).toEqual(['u1'])
})
test('preserves assistant text when dropping orphan tool uses', () => {
const messages = [
{
type: 'assistant',
uuid: 'a1',
message: {
role: 'assistant',
content: [
{ type: 'text', text: 'I will read the file.' },
{ type: 'tool_use', id: 'missing', name: 'Read' },
],
},
},
] as unknown as Message[]
const filtered = filterIncompleteToolCalls(messages)
expect(filtered).toHaveLength(1)
const first = filtered[0]!
const content = first.message!.content
expect(
Array.isArray(content) ? content.map(block => block.type) : [],
).toEqual(['text'])
})
test('keeps completed parallel tool calls when dropping an orphan', () => {
const messages = [
{
type: 'assistant',
uuid: 'a1',
message: {
role: 'assistant',
content: [
{ type: 'tool_use', id: 'done', name: 'Read' },
{ type: 'tool_use', id: 'missing', name: 'Grep' },
],
},
},
{
type: 'user',
uuid: 'u1',
message: {
role: 'user',
content: [{ type: 'tool_result', tool_use_id: 'done', content: 'ok' }],
},
},
] as unknown as Message[]
const filtered = filterIncompleteToolCalls(messages)
expect(filtered.map(message => String(message.uuid))).toEqual(['a1', 'u1'])
const first = filtered[0]!
const content = first.message!.content
expect(
Array.isArray(content)
? content.map(block =>
block.type === 'tool_use' ? block.id : block.type,
)
: [],
).toEqual(['done'])
})
test('keeps assistant tool uses that have matching results', () => {
const messages = [
{
type: 'assistant',
uuid: 'a1',
message: {
role: 'assistant',
content: [{ type: 'tool_use', id: 'done', name: 'Read' }],
},
},
{
type: 'user',
uuid: 'u1',
message: {
role: 'user',
content: [{ type: 'tool_result', tool_use_id: 'done', content: 'ok' }],
},
},
] as unknown as Message[]
expect(
filterIncompleteToolCalls(messages).map(message => String(message.uuid)),
).toEqual(['a1', 'u1'])
})
test('drops orphan tool results when their tool use was removed', () => {
const messages = [
{
type: 'user',
uuid: 'u1',
message: {
role: 'user',
content: [
{ type: 'tool_result', tool_use_id: 'missing', content: 'late' },
],
},
},
] as unknown as Message[]
expect(filterIncompleteToolCalls(messages)).toEqual([])
})
test('keeps user text while dropping orphan tool results', () => {
const messages = [
{
type: 'assistant',
uuid: 'a1',
message: { role: 'assistant', content: 'done' },
},
{
type: 'user',
uuid: 'u1',
message: {
role: 'user',
content: [
{ type: 'text', text: 'keep this' },
{ type: 'tool_result', tool_use_id: 'missing', content: 'late' },
],
},
},
] as unknown as Message[]
const filtered = filterIncompleteToolCalls(messages)
expect(filtered.map(message => String(message.uuid))).toEqual(['a1', 'u1'])
const content = filtered[1]!.message!.content
expect(Array.isArray(content) ? content : []).toEqual([
{ type: 'text', text: 'keep this' },
])
})
test('drops malformed tool blocks without ids', () => {
const messages = [
{
type: 'assistant',
uuid: 'a1',
message: {
role: 'assistant',
content: [{ type: 'tool_use', name: 'Read' }],
},
},
{
type: 'user',
uuid: 'u1',
message: {
role: 'user',
content: [{ type: 'tool_result', content: 'late' }],
},
},
] as unknown as Message[]
expect(filterIncompleteToolCalls(messages)).toEqual([])
})
})

View File

@@ -0,0 +1,110 @@
import type {
AssistantMessage,
Message,
UserMessage,
} from 'src/types/message.js'
/**
* Removes invalid or orphaned tool_use/tool_result blocks while preserving
* completed tool-call pairs. This is intentionally block-level, not
* message-level, so completed parallel tool calls stay paired with results.
*/
export function filterIncompleteToolCalls(messages: Message[]): Message[] {
const toolUseIdsWithResults = new Set<string>()
for (const message of messages) {
if (message?.type === 'user') {
const userMessage = message as UserMessage
const content = userMessage.message.content
if (Array.isArray(content)) {
for (const block of content) {
if (block.type === 'tool_result' && block.tool_use_id) {
toolUseIdsWithResults.add(block.tool_use_id)
}
}
}
}
}
const retainedToolUseIds = new Set<string>()
const withoutOrphanToolUses: Message[] = []
for (const message of messages) {
if (message?.type === 'assistant') {
const assistantMessage = message as AssistantMessage
const content = assistantMessage.message.content
if (Array.isArray(content)) {
let changed = false
const filteredContent = content.filter(block => {
if (block.type !== 'tool_use') return true
if (!block.id) {
changed = true
return false
}
if (toolUseIdsWithResults.has(block.id)) {
retainedToolUseIds.add(block.id)
return true
}
changed = true
return false
})
if (!changed) {
withoutOrphanToolUses.push(message)
continue
}
if (filteredContent.length > 0) {
withoutOrphanToolUses.push({
...assistantMessage,
message: {
...assistantMessage.message,
content: filteredContent,
},
})
}
continue
}
}
withoutOrphanToolUses.push(message)
}
const filteredMessages: Message[] = []
for (const message of withoutOrphanToolUses) {
if (message?.type !== 'user') {
filteredMessages.push(message)
continue
}
const userMessage = message as UserMessage
const content = userMessage.message.content
if (!Array.isArray(content)) {
filteredMessages.push(message)
continue
}
let changed = false
const filteredContent = content.filter(block => {
if (block.type !== 'tool_result') return true
if (!block.tool_use_id) {
changed = true
return false
}
if (retainedToolUseIds.has(block.tool_use_id)) return true
changed = true
return false
})
if (!changed) {
filteredMessages.push(message)
continue
}
if (filteredContent.length > 0) {
filteredMessages.push({
...userMessage,
message: {
...userMessage.message,
content: filteredContent,
},
})
}
}
return filteredMessages
}

View File

@@ -86,8 +86,11 @@ import {
import type { ContentReplacementState } from 'src/utils/toolResultStorage.js' import type { ContentReplacementState } from 'src/utils/toolResultStorage.js'
import { createAgentId } from 'src/utils/uuid.js' import { createAgentId } from 'src/utils/uuid.js'
import { resolveAgentTools } from './agentToolUtils.js' import { resolveAgentTools } from './agentToolUtils.js'
import { filterIncompleteToolCalls } from './filterIncompleteToolCalls.js'
import { type AgentDefinition, isBuiltInAgent } from './loadAgentsDir.js' import { type AgentDefinition, isBuiltInAgent } from './loadAgentsDir.js'
export { filterIncompleteToolCalls } from './filterIncompleteToolCalls.js'
/** /**
* Initialize agent-specific MCP servers * Initialize agent-specific MCP servers
* Agents can define their own MCP servers in their frontmatter that are additive * Agents can define their own MCP servers in their frontmatter that are additive
@@ -886,50 +889,6 @@ export async function* runAgent({
} }
} }
/**
* Filters out assistant messages with incomplete tool calls (tool uses without results).
* This prevents API errors when sending messages with orphaned tool calls.
*/
export function filterIncompleteToolCalls(messages: Message[]): Message[] {
// Build a set of tool use IDs that have results
const toolUseIdsWithResults = new Set<string>()
for (const message of messages) {
if (message?.type === 'user') {
const userMessage = message as UserMessage
const content = userMessage.message.content
if (Array.isArray(content)) {
for (const block of content) {
if (block.type === 'tool_result' && block.tool_use_id) {
toolUseIdsWithResults.add(block.tool_use_id)
}
}
}
}
}
// Filter out assistant messages that contain tool calls without results
return messages.filter(message => {
if (message?.type === 'assistant') {
const assistantMessage = message as AssistantMessage
const content = assistantMessage.message.content
if (Array.isArray(content)) {
// Check if this assistant message has any tool uses without results
const hasIncompleteToolCall = content.some(
block =>
block.type === 'tool_use' &&
block.id &&
!toolUseIdsWithResults.has(block.id),
)
// Exclude messages with incomplete tool calls
return !hasIncompleteToolCall
}
}
// Keep all non-assistant messages and assistant messages without tool calls
return true
})
}
async function getAgentSystemPrompt( async function getAgentSystemPrompt(
agentDefinition: AgentDefinition, agentDefinition: AgentDefinition,
toolUseContext: Pick<ToolUseContext, 'options'>, toolUseContext: Pick<ToolUseContext, 'options'>,

View File

@@ -131,15 +131,16 @@ export type SendMessageToolOutput =
| ResponseOutput | ResponseOutput
const UDS_INLINE_TOKEN_MARKER = '#token=' const UDS_INLINE_TOKEN_MARKER = '#token='
const UDS_INLINE_TOKEN_REJECTED_KEY = '__udsInlineTokenRejected'
function stripInlineUdsToken(target: string): string { function stripInlineUdsToken(target: string): string {
const markerIndex = target.lastIndexOf(UDS_INLINE_TOKEN_MARKER) const markerIndex = target.indexOf(UDS_INLINE_TOKEN_MARKER)
return markerIndex === -1 ? target : target.slice(0, markerIndex) return markerIndex === -1 ? target : target.slice(0, markerIndex)
} }
function hasInlineUdsToken(to: string): boolean { function hasInlineUdsToken(to: string): boolean {
const addr = parseAddress(to) const addr = parseAddress(to)
// Empty-token markers are still inline-token attempts. Observable input
// redaction preserves "#token=" so cloned inputs remain rejected.
return ( return (
addr.scheme === 'uds' && addr.target.includes(UDS_INLINE_TOKEN_MARKER) addr.scheme === 'uds' && addr.target.includes(UDS_INLINE_TOKEN_MARKER)
) )
@@ -151,20 +152,17 @@ function recipientForDisplay(to: string): string {
return `uds:${stripInlineUdsToken(addr.target)}` return `uds:${stripInlineUdsToken(addr.target)}`
} }
function markAndRedactInlineUdsToken( function redactInlineUdsTokenForRejection(to: string): string {
input: { to: string } & Record<string, unknown>, const addr = parseAddress(to)
): void { if (addr.scheme !== 'uds') return to
if (!hasInlineUdsToken(input.to)) return const markerIndex = addr.target.indexOf(UDS_INLINE_TOKEN_MARKER)
input.to = recipientForDisplay(input.to) if (markerIndex === -1) return to
input[UDS_INLINE_TOKEN_REJECTED_KEY] = true return `uds:${addr.target.slice(0, markerIndex)}${UDS_INLINE_TOKEN_MARKER}`
} }
function wasInlineUdsTokenRejected(input: unknown): boolean { function redactObservableInlineUdsToken(input: { to: string }): void {
return ( if (!hasInlineUdsToken(input.to)) return
typeof input === 'object' && input.to = redactInlineUdsTokenForRejection(input.to)
input !== null &&
(input as Record<string, unknown>)[UDS_INLINE_TOKEN_REJECTED_KEY] === true
)
} }
function findTeammateColor( function findTeammateColor(
@@ -580,9 +578,7 @@ export const SendMessageTool: Tool<InputSchema, SendMessageToolOutput> =
backfillObservableInput(input) { backfillObservableInput(input) {
if (typeof input.to !== 'string') return if (typeof input.to !== 'string') return
markAndRedactInlineUdsToken( redactObservableInlineUdsToken(input as { to: string })
input as { to: string } & Record<string, unknown>,
)
if ('type' in input) return if ('type' in input) return
if (input.to === '*') { if (input.to === '*') {
@@ -620,7 +616,10 @@ export const SendMessageTool: Tool<InputSchema, SendMessageToolOutput> =
case 'shutdown_response': case 'shutdown_response':
return `shutdown_response ${input.message.approve ? 'approve' : 'reject'} ${input.message.request_id}` return `shutdown_response ${input.message.approve ? 'approve' : 'reject'} ${input.message.request_id}`
case 'plan_approval_response': case 'plan_approval_response':
return `plan_approval ${input.message.approve ? 'approve' : 'reject'} to ${recipient}` const planApprovalDecision = input.message.approve
? 'approve'
: 'reject'
return `plan_approval ${planApprovalDecision} to ${recipient}`
} }
}, },
@@ -674,7 +673,7 @@ export const SendMessageTool: Tool<InputSchema, SendMessageToolOutput> =
} }
if ( if (
addr.scheme === 'uds' && addr.scheme === 'uds' &&
(hasInlineUdsToken(input.to) || wasInlineUdsTokenRejected(input)) hasInlineUdsToken(input.to)
) { ) {
return { return {
result: false, result: false,
@@ -808,10 +807,7 @@ export const SendMessageTool: Tool<InputSchema, SendMessageToolOutput> =
async call(input, context, canUseTool, assistantMessage) { async call(input, context, canUseTool, assistantMessage) {
if (typeof input.message === 'string') { if (typeof input.message === 'string') {
const addr = parseAddress(input.to) const addr = parseAddress(input.to)
if ( if (addr.scheme === 'uds' && hasInlineUdsToken(input.to)) {
addr.scheme === 'uds' &&
(hasInlineUdsToken(input.to) || wasInlineUdsTokenRejected(input))
) {
return { return {
data: { data: {
success: false, success: false,
@@ -841,10 +837,10 @@ export const SendMessageTool: Tool<InputSchema, SendMessageToolOutput> =
const { postInterClaudeMessage } = const { postInterClaudeMessage } =
require('src/bridge/peerSessions.js') as typeof import('src/bridge/peerSessions.js') require('src/bridge/peerSessions.js') as typeof import('src/bridge/peerSessions.js')
/* eslint-enable @typescript-eslint/no-require-imports */ /* eslint-enable @typescript-eslint/no-require-imports */
const result = (await postInterClaudeMessage( const result = await postInterClaudeMessage(
addr.target, addr.target,
input.message, input.message,
)) as { ok: boolean; error?: string } ) as { ok: boolean; error?: string }
const preview = input.summary || truncate(input.message, 50) const preview = input.summary || truncate(input.message, 50)
return { return {
data: { data: {
@@ -856,16 +852,6 @@ export const SendMessageTool: Tool<InputSchema, SendMessageToolOutput> =
} }
} }
if (addr.scheme === 'uds') { if (addr.scheme === 'uds') {
const recipient = recipientForDisplay(input.to)
if (hasInlineUdsToken(input.to) || wasInlineUdsTokenRejected(input)) {
return {
data: {
success: false,
message:
'uds addresses must not include inline auth tokens; use the ListPeers address',
},
}
}
/* eslint-disable @typescript-eslint/no-require-imports */ /* eslint-disable @typescript-eslint/no-require-imports */
const { sendToUdsSocket } = const { sendToUdsSocket } =
require('src/utils/udsClient.js') as typeof import('src/utils/udsClient.js') require('src/utils/udsClient.js') as typeof import('src/utils/udsClient.js')
@@ -876,14 +862,14 @@ export const SendMessageTool: Tool<InputSchema, SendMessageToolOutput> =
return { return {
data: { data: {
success: true, success: true,
message: `${preview}” → ${recipient}`, message: `${preview}” → ${input.to}`,
}, },
} }
} catch (e) { } catch (e) {
return { return {
data: { data: {
success: false, success: false,
message: `Failed to send to ${recipient}: ${errorMessage(e)}`, message: `Failed to send to ${input.to}: ${errorMessage(e)}`,
}, },
} }
} }

View File

@@ -1,8 +1,8 @@
import { describe, expect, test } from 'bun:test' import { describe, expect, test } from 'bun:test'
import { SendMessageTool } from '../SendMessageTool.js'
describe('SendMessageTool UDS recipient handling', () => { describe('SendMessageTool UDS recipient handling', () => {
test('redacts inline UDS tokens before classifier and observable paths', async () => { test('redacts inline UDS tokens before classifier and observable paths', async () => {
const { SendMessageTool } = await import('../SendMessageTool.js')
const tokenAddress = 'uds:/tmp/peer.sock#token=secret-token' const tokenAddress = 'uds:/tmp/peer.sock#token=secret-token'
const observableInput = { const observableInput = {
@@ -12,6 +12,7 @@ describe('SendMessageTool UDS recipient handling', () => {
SendMessageTool.backfillObservableInput!(observableInput) SendMessageTool.backfillObservableInput!(observableInput)
expect(observableInput.recipient).toBe('uds:/tmp/peer.sock') expect(observableInput.recipient).toBe('uds:/tmp/peer.sock')
expect(observableInput.to).toBe('uds:/tmp/peer.sock#token=')
expect(JSON.stringify(observableInput)).not.toContain('secret-token') expect(JSON.stringify(observableInput)).not.toContain('secret-token')
expect( expect(
SendMessageTool.toAutoClassifierInput({ SendMessageTool.toAutoClassifierInput({
@@ -22,7 +23,6 @@ describe('SendMessageTool UDS recipient handling', () => {
}) })
test('keeps redacted UDS token rejection through observable backfill', async () => { test('keeps redacted UDS token rejection through observable backfill', async () => {
const { SendMessageTool } = await import('../SendMessageTool.js')
const observableInput = { const observableInput = {
to: 'uds:/tmp/peer.sock#token=secret-token', to: 'uds:/tmp/peer.sock#token=secret-token',
message: { message: {
@@ -35,7 +35,7 @@ describe('SendMessageTool UDS recipient handling', () => {
SendMessageTool.backfillObservableInput!(observableInput) SendMessageTool.backfillObservableInput!(observableInput)
expect(observableInput.to).toBe('uds:/tmp/peer.sock') expect(observableInput.to).toBe('uds:/tmp/peer.sock#token=')
expect(observableInput.recipient).toBe('uds:/tmp/peer.sock') expect(observableInput.recipient).toBe('uds:/tmp/peer.sock')
expect(observableInput.type).toBe('plan_approval_response') expect(observableInput.type).toBe('plan_approval_response')
expect(observableInput.request_id).toBe('req-1') expect(observableInput.request_id).toBe('req-1')
@@ -55,8 +55,37 @@ describe('SendMessageTool UDS recipient handling', () => {
expect(result.message).toContain('inline auth tokens') expect(result.message).toContain('inline auth tokens')
}) })
test('keeps inline-token rejection when observable input is cloned', async () => {
const observableInput = {
to: 'uds:/tmp/peer.sock#token=secret-token',
message: 'hello',
} as Record<string, unknown>
SendMessageTool.backfillObservableInput!(observableInput)
const clonedInput = {
to: observableInput.to,
message: observableInput.message,
summary: 'hello peer',
}
const validation = await SendMessageTool.validateInput!(
clonedInput as never,
{} as never,
)
const result = await SendMessageTool.call(
clonedInput as never,
{} as never,
undefined as never,
undefined as never,
)
expect(validation.result).toBe(false)
expect(result.data.success).toBe(false)
expect(JSON.stringify(clonedInput)).not.toContain('secret-token')
expect(JSON.stringify(result)).not.toContain('secret-token')
})
test('redacts UDS tokens in structured classifier text', async () => { test('redacts UDS tokens in structured classifier text', async () => {
const { SendMessageTool } = await import('../SendMessageTool.js')
const to = 'uds:/tmp/peer.sock#token=secret-token' const to = 'uds:/tmp/peer.sock#token=secret-token'
expect( expect(
@@ -75,10 +104,50 @@ describe('SendMessageTool UDS recipient handling', () => {
}, },
}), }),
).toBe('plan_approval approve to uds:/tmp/peer.sock') ).toBe('plan_approval approve to uds:/tmp/peer.sock')
expect(
SendMessageTool.toAutoClassifierInput({
to,
message: {
type: 'plan_approval_response',
request_id: 'req-2',
approve: false,
},
}),
).toBe('plan_approval reject to uds:/tmp/peer.sock')
expect(
SendMessageTool.toAutoClassifierInput({
to,
message: {
type: 'shutdown_response',
request_id: 'shutdown-1',
approve: false,
},
}),
).toBe('shutdown_response reject shutdown-1')
})
test('redacts from the first inline UDS token marker', async () => {
const tokenAddress = 'uds:/tmp/peer.sock#token=first#token=second'
const observableInput = {
to: tokenAddress,
message: 'hello',
} as Record<string, unknown>
SendMessageTool.backfillObservableInput!(observableInput)
expect(observableInput.to).toBe('uds:/tmp/peer.sock#token=')
expect(observableInput.recipient).toBe('uds:/tmp/peer.sock')
expect(JSON.stringify(observableInput)).not.toContain('first')
expect(JSON.stringify(observableInput)).not.toContain('second')
expect(
SendMessageTool.toAutoClassifierInput({
to: tokenAddress,
message: 'hello',
}),
).toBe('to uds:/tmp/peer.sock: hello')
}) })
test('rejects inline UDS tokens during validation', async () => { test('rejects inline UDS tokens during validation', async () => {
const { SendMessageTool } = await import('../SendMessageTool.js')
const result = await SendMessageTool.validateInput!( const result = await SendMessageTool.validateInput!(
{ {
to: 'uds:/tmp/peer.sock#token=secret-token', to: 'uds:/tmp/peer.sock#token=secret-token',
@@ -96,7 +165,6 @@ describe('SendMessageTool UDS recipient handling', () => {
}) })
test('rejects inline UDS tokens during execution without leaking them', async () => { test('rejects inline UDS tokens during execution without leaking them', async () => {
const { SendMessageTool } = await import('../SendMessageTool.js')
const result = await SendMessageTool.call( const result = await SendMessageTool.call(
{ {
to: 'uds:/tmp/peer.sock#token=secret-token', to: 'uds:/tmp/peer.sock#token=secret-token',

View File

@@ -5,7 +5,8 @@
* After the fix, it reads from / writes to settings.json via * After the fix, it reads from / writes to settings.json via
* getInitialSettings() and updateSettingsForSource(). * getInitialSettings() and updateSettingsForSource().
*/ */
import { describe, expect, test, beforeEach, mock } from 'bun:test' import { afterAll, describe, expect, test, beforeEach, mock } from 'bun:test'
import * as settingsModule from '../../../utils/settings/settings.js'
// ── Mocks must be declared before the module under test is imported ────────── // ── Mocks must be declared before the module under test is imported ──────────
@@ -13,24 +14,48 @@ let mockSettings: Record<string, unknown> = {}
let lastUpdate: { source: string; patch: Record<string, unknown> } | null = null let lastUpdate: { source: string; patch: Record<string, unknown> } | null = null
mock.module('src/utils/settings/settings.js', () => ({ mock.module('src/utils/settings/settings.js', () => ({
loadManagedFileSettings: () => ({ settings: null, errors: [] }),
getManagedFileSettingsPresence: () => ({
hasBase: false,
hasDropIns: false,
}),
parseSettingsFile: () => ({ settings: null, errors: [] }),
getSettingsRootPathForSource: () => '',
getSettingsFilePathForSource: () => undefined,
getRelativeSettingsFilePathForSource: () => '',
getInitialSettings: () => mockSettings, getInitialSettings: () => mockSettings,
getSettingsForSource: () => mockSettings,
getPolicySettingsOrigin: () => null,
getSettingsWithErrors: () => ({ settings: mockSettings, errors: [] }),
getSettingsWithSources: () => ({ effective: mockSettings, sources: [] }),
getSettings_DEPRECATED: () => mockSettings,
settingsMergeCustomizer: () => undefined,
getManagedSettingsKeysForLogging: () => [],
// Keep unrelated exports aligned with the real settings module so this
// full-surface mock cannot change later test files if Bun keeps it alive.
hasAutoModeOptIn: () => true,
hasSkipDangerousModePermissionPrompt: () => false,
getAutoModeConfig: () => undefined,
getUseAutoModeDuringPlan: () => true,
rawSettingsContainsKey: (key: string) => key in mockSettings,
updateSettingsForSource: (source: string, patch: Record<string, unknown>) => { updateSettingsForSource: (source: string, patch: Record<string, unknown>) => {
lastUpdate = { source, patch } lastUpdate = { source, patch }
mockSettings = { ...mockSettings, ...patch } mockSettings = { ...mockSettings, ...patch }
}, },
})) }))
// Import AFTER mocks are registered afterAll(() => {
const { isPoorModeActive, setPoorMode } = await import('../poorMode.js') mock.restore()
mock.module('src/utils/settings/settings.js', () => settingsModule)
})
// ── Helpers ────────────────────────────────────────────────────────────────── // Import AFTER mocks are registered. The query suffix gives this file its own
// module instance so cross-file poorMode.js mocks cannot replace the subject
/** Reset module-level singleton between tests by re-importing a fresh copy. */ // under test during Bun's shared coverage run.
async function freshModule() { const poorModeModulePath = '../poorMode.js?poorModeTest'
// Bun caches modules; we manipulate the exported functions directly since const { isPoorModeActive, setPoorMode } = (await import(
// the singleton `poorModeActive` is reset to null only on first import. poorModeModulePath
// Instead we test the observable behaviour through set/get pairs. )) as typeof import('../poorMode.js')
}
// ── Tests ──────────────────────────────────────────────────────────────────── // ── Tests ────────────────────────────────────────────────────────────────────

View File

@@ -1,16 +1,14 @@
import { import { beforeEach, describe, expect, test } from 'bun:test'
afterAll,
afterEach,
beforeEach,
describe,
expect,
mock,
test,
} from 'bun:test'
import { debugMock } from '../../../../tests/mocks/debug'
import { logMock } from '../../../../tests/mocks/log'
import { asAgentId } from '../../../types/ids.js' import { asAgentId } from '../../../types/ids.js'
import type { CacheSafeParams } from '../../../utils/forkedAgent.js' import type { Message } from '../../../types/message.js'
import type {
CacheSafeParams,
ForkedAgentResult,
} from '../../../utils/forkedAgent.js'
import {
type AgentSummaryDependencies,
startAgentSummarization,
} from '../agentSummary.js'
const transcriptMessages = [ const transcriptMessages = [
{ type: 'user', message: { content: 'start' }, uuid: 'u1' }, { type: 'user', message: { content: 'start' }, uuid: 'u1' },
@@ -20,114 +18,195 @@ const transcriptMessages = [
uuid: 'a1', uuid: 'a1',
}, },
{ type: 'user', message: { content: 'continue' }, uuid: 'u2' }, { type: 'user', message: { content: 'continue' }, uuid: 'u2' },
] ] as unknown as Message[]
let poorModeActive = false type ForkCall = {
let forkCalls = 0 cacheSafeParams: CacheSafeParams
let updateCalls: Array<{ taskId: string; summary: string }> = []
let transcript = { messages: transcriptMessages }
const sessionStorageSnapshot = {
...(require('../../../utils/sessionStorage.ts') as Record<string, unknown>),
} }
mock.module('src/commands/poor/poorMode.js', () => ({
isPoorModeActive: () => poorModeActive,
}))
mock.module('src/tasks/LocalAgentTask/LocalAgentTask.js', () => ({
updateAgentSummary: (taskId: string, summary: string) => {
updateCalls.push({ taskId, summary })
},
}))
mock.module(
'@claude-code-best/builtin-tools/tools/AgentTool/runAgent.js',
() => ({
filterIncompleteToolCalls: <T>(messages: T) => messages,
}),
)
mock.module('src/utils/debug.js', debugMock)
mock.module('src/utils/log.js', logMock)
mock.module('src/utils/forkedAgent.js', () => ({
runForkedAgent: async () => {
forkCalls += 1
return {
messages: [
{
type: 'assistant',
message: {
content: [{ type: 'text', text: 'Reading udsClient.ts' }],
},
},
],
}
},
}))
mock.module('src/utils/sessionStorage.js', () => ({
...sessionStorageSnapshot,
getAgentTranscript: async () => transcript,
}))
afterAll(() => {
mock.module('src/utils/sessionStorage.js', () =>
require('../../../utils/sessionStorage.ts'),
)
})
describe('startAgentSummarization', () => { describe('startAgentSummarization', () => {
const realSetTimeout = globalThis.setTimeout let scheduled: (() => void | Promise<void>) | undefined
const realClearTimeout = globalThis.clearTimeout let handle: { stop: () => void } | undefined
let scheduled: let forkCalls: ForkCall[]
| ((...args: Parameters<TimerHandler & ((...args: unknown[]) => void)>) => void) let updateCalls: Array<{ taskId: string; summary: string }>
| undefined let transcriptMessagesForTest: Message[]
let debugLogs: string[]
let loggedErrors: Error[]
let clearedHandles: unknown[]
beforeEach(() => { function startTestSummarization(
poorModeActive = false dependencies: AgentSummaryDependencies = {},
forkCalls = 0 ): { stop: () => void } {
updateCalls = [] return startAgentSummarization(
transcript = { messages: transcriptMessages }
scheduled = undefined
globalThis.setTimeout = ((callback: TimerHandler) => {
scheduled = callback as (...args: unknown[]) => void
return 1 as unknown as ReturnType<typeof setTimeout>
}) as unknown as typeof setTimeout
globalThis.clearTimeout = (() => undefined) as typeof clearTimeout
})
afterEach(() => {
globalThis.setTimeout = realSetTimeout
globalThis.clearTimeout = realClearTimeout
})
test('summarizes bounded transcript once and skips unchanged fingerprints', async () => {
const { startAgentSummarization } = await import('../agentSummary.js')
const handle = startAgentSummarization(
'task-1', 'task-1',
asAgentId('a0000000000000000'), asAgentId('a0000000000000000'),
{ {
forkContextMessages: [{ type: 'user', message: { content: 'old' } }], forkContextMessages: [
{ type: 'user', message: { content: 'stale' }, uuid: 'old' },
],
model: 'claude-test', model: 'claude-test',
} as unknown as CacheSafeParams, } as unknown as CacheSafeParams,
() => undefined, () => undefined,
{
clearTimeout: ((timeoutId: unknown) => {
clearedHandles.push(timeoutId)
}) as typeof clearTimeout,
getAgentTranscript: async () => ({
messages: transcriptMessagesForTest,
contentReplacements: [],
}),
isPoorModeActive: () => false,
logError: error => {
loggedErrors.push(
error instanceof Error ? error : new Error(String(error)),
)
},
logForDebugging: message => {
debugLogs.push(message)
},
runForkedAgent: async (args: ForkCall) => {
forkCalls.push(args)
return {
messages: [
{
type: 'assistant',
message: {
content: [{ type: 'text', text: 'Reading udsClient.ts' }],
},
},
],
} as unknown as ForkedAgentResult
},
setTimeout: ((callback: TimerHandler) => {
if (typeof callback !== 'function') {
throw new Error('Expected timer callback')
}
scheduled = callback as () => void | Promise<void>
return 1 as unknown as ReturnType<typeof setTimeout>
}) as unknown as typeof setTimeout,
updateAgentSummary: (taskId: string, summary: string) => {
updateCalls.push({ taskId, summary })
},
...dependencies,
},
) )
}
beforeEach(() => {
forkCalls = []
updateCalls = []
scheduled = undefined
handle = undefined
transcriptMessagesForTest = transcriptMessages
debugLogs = []
loggedErrors = []
clearedHandles = []
})
test('summarizes bounded transcript once and skips unchanged fingerprints', async () => {
handle = startTestSummarization()
expect(typeof scheduled).toBe('function') expect(typeof scheduled).toBe('function')
await scheduled!() await scheduled!()
expect(forkCalls).toBe(1) expect(forkCalls).toHaveLength(1)
expect(updateCalls).toEqual([ expect(updateCalls).toEqual([
{ taskId: 'task-1', summary: 'Reading udsClient.ts' }, { taskId: 'task-1', summary: 'Reading udsClient.ts' },
]) ])
const forkContext = forkCalls[0].cacheSafeParams.forkContextMessages ?? []
expect(forkContext.map(message => String(message.uuid))).toEqual([
'u1',
'a1',
'u2',
])
expect(forkContext.some(message => String(message.uuid) === 'old')).toBe(
false,
)
await scheduled!() await scheduled!()
expect(forkCalls).toBe(1) expect(forkCalls).toHaveLength(1)
expect(updateCalls).toHaveLength(1) expect(updateCalls).toHaveLength(1)
})
test('skips summarization when filtering leaves too little bounded context', async () => {
transcriptMessagesForTest = [
{ type: 'user', message: { content: 'start' }, uuid: 'u1' },
{
type: 'assistant',
uuid: 'a1',
message: {
content: [{ type: 'tool_use', id: 'missing', name: 'Read' }],
},
},
{ type: 'user', message: { content: 'continue' }, uuid: 'u2' },
] as unknown as Message[]
handle = startTestSummarization()
expect(typeof scheduled).toBe('function')
await scheduled!()
expect(forkCalls).toEqual([])
expect(updateCalls).toEqual([])
expect(debugLogs).toContain(
'[AgentSummary] Skipping summary for task-1: no bounded context available',
)
})
test('skips summarization before building context when transcript is too short', async () => {
transcriptMessagesForTest = transcriptMessages.slice(0, 2)
handle = startTestSummarization()
expect(typeof scheduled).toBe('function')
await scheduled!()
expect(forkCalls).toEqual([])
expect(updateCalls).toEqual([])
expect(debugLogs).toContain(
'[AgentSummary] Skipping summary for task-1: not enough messages (2)',
)
})
test('skips and reschedules while poor mode is active', async () => {
handle = startTestSummarization({
isPoorModeActive: () => true,
})
expect(typeof scheduled).toBe('function')
await scheduled!()
expect(forkCalls).toEqual([])
expect(updateCalls).toEqual([])
expect(debugLogs).toContain(
'[AgentSummary] Skipping summary — poor mode active',
)
})
test('logs summary errors and keeps the next timer owned by the summarizer', async () => {
const error = new Error('fork failed')
handle = startTestSummarization({
runForkedAgent: async () => {
throw error
},
})
expect(typeof scheduled).toBe('function')
await scheduled!()
expect(loggedErrors).toEqual([error])
expect(updateCalls).toEqual([])
})
test('stop clears the pending summary timer', () => {
handle = startTestSummarization()
handle.stop() handle.stop()
expect(debugLogs).toContain(
'[AgentSummary] Stopping summarization for task-1',
)
expect(clearedHandles).toEqual([1])
}) })
}) })

View File

@@ -1,6 +1,8 @@
import { describe, expect, test } from 'bun:test' import { describe, expect, test } from 'bun:test'
import type { Message } from '../../../types/message.js' import type { Message } from '../../../types/message.js'
import { import {
buildSummaryContext,
estimateMessageChars,
getSummaryContextFingerprint, getSummaryContextFingerprint,
MAX_SUMMARY_CONTEXT_CHARS, MAX_SUMMARY_CONTEXT_CHARS,
selectSummaryContextMessages, selectSummaryContextMessages,
@@ -75,6 +77,21 @@ describe('selectSummaryContextMessages', () => {
expect(selected).toEqual([]) expect(selected).toEqual([])
}) })
test('stops at an older oversized message after keeping the recent suffix', () => {
const messages = [
makeMessage('user', 'u1', 'x'.repeat(5_000)),
makeMessage('user', 'u2', 'small prompt'),
makeMessage('assistant', 'a2', 'small answer'),
]
const selected = selectSummaryContextMessages(messages, {
maxMessages: 10,
maxChars: 1_000,
})
expect(selected.map(message => String(message.uuid))).toEqual(['u2', 'a2'])
})
test('drops leading orphan tool results after bounding', () => { test('drops leading orphan tool results after bounding', () => {
const messages = [ const messages = [
makeMessage('assistant', 'a0', 'older assistant'), makeMessage('assistant', 'a0', 'older assistant'),
@@ -102,6 +119,35 @@ describe('selectSummaryContextMessages', () => {
}) })
describe('getSummaryContextFingerprint', () => { describe('getSummaryContextFingerprint', () => {
test('estimates circular messages as unbounded', () => {
const circular = makeMessage('assistant', 'a1', 'cycle') as Message & {
self?: unknown
}
circular.self = circular
expect(estimateMessageChars(circular)).toBe(Number.POSITIVE_INFINITY)
})
test('ignores non-json primitive fields in size estimates', () => {
const message = makeMessage('assistant', 'a1', 'metadata') as Message & {
skipUndefined?: undefined
skipFunction?: () => void
skipSymbol?: symbol
}
message.skipUndefined = undefined
message.skipFunction = () => undefined
message.skipSymbol = Symbol('ignored')
expect(estimateMessageChars(message)).toBeGreaterThan(0)
})
test('treats unsupported top-level primitives as zero-size estimates', () => {
expect(
estimateMessageChars((() => undefined) as unknown as Message),
).toBe(0)
expect(estimateMessageChars(1n as unknown as Message)).toBe(0)
})
test('returns null for an empty transcript', () => { test('returns null for an empty transcript', () => {
expect(getSummaryContextFingerprint([])).toBeNull() expect(getSummaryContextFingerprint([])).toBeNull()
}) })
@@ -146,4 +192,77 @@ describe('getSummaryContextFingerprint', () => {
expect(first).not.toBe(second) expect(first).not.toBe(second)
}) })
test('fingerprints circular message references without recursing forever', () => {
const circular = makeMessage('assistant', 'a1', 'cycle') as Message & {
self?: unknown
}
circular.self = circular
expect(getSummaryContextFingerprint([circular])).toContain(':a1:')
})
})
describe('buildSummaryContext', () => {
test('returns bounded messages and fingerprint for summarizable context', () => {
const messages = [
{ type: 'user', uuid: 'u1', message: { content: 'start' } },
{
type: 'assistant',
uuid: 'a1',
message: { content: [{ type: 'text', text: 'working' }] },
},
{ type: 'user', uuid: 'u2', message: { content: 'continue' } },
] as unknown as Message[]
const result = buildSummaryContext(messages, null)
expect(result.skipReason).toBeUndefined()
expect(result.messages.map(message => String(message.uuid))).toEqual([
'u1',
'a1',
'u2',
])
expect(result.fingerprint).toContain('3:u2:')
})
test('reports unchanged contexts by fingerprint', () => {
const messages = [
{ type: 'user', uuid: 'u1', message: { content: 'start' } },
{
type: 'assistant',
uuid: 'a1',
message: { content: [{ type: 'text', text: 'working' }] },
},
{ type: 'user', uuid: 'u2', message: { content: 'continue' } },
] as unknown as Message[]
const first = buildSummaryContext(messages, null)
const second = buildSummaryContext(messages, first.fingerprint)
expect(second.skipReason).toBe('unchanged')
expect(second.fingerprint).toBe(first.fingerprint)
})
test('filters incomplete tool calls before deciding context is too small', () => {
const messages = [
{ type: 'user', uuid: 'u1', message: { content: 'start' } },
{
type: 'assistant',
uuid: 'a1',
message: {
content: [{ type: 'tool_use', id: 'missing', name: 'Read' }],
},
},
{ type: 'user', uuid: 'u2', message: { content: 'continue' } },
] as unknown as Message[]
const result = buildSummaryContext(messages, null)
expect(result.skipReason).toBe('too_small')
expect(result.messages.map(message => String(message.uuid))).toEqual([
'u1',
'u2',
])
})
}) })

View File

@@ -0,0 +1,34 @@
import { describe, expect, test } from 'bun:test'
import {
buildSummaryPrompt,
createSummaryPromptMessage,
} from '../summaryPrompt.js'
describe('buildSummaryPrompt', () => {
test('builds the first summary prompt without previous-summary pressure', () => {
const prompt = buildSummaryPrompt(null)
expect(prompt).toContain('Describe your most recent action')
expect(prompt).toContain('Good: "Reading runAgent.ts"')
expect(prompt).not.toContain('Previous:')
})
test('asks for a new summary when a previous one exists', () => {
const prompt = buildSummaryPrompt('Reading udsMessaging.ts')
expect(prompt).toContain('Previous: "Reading udsMessaging.ts"')
expect(prompt).toContain('say something NEW')
})
})
describe('createSummaryPromptMessage', () => {
test('creates the minimal user message shape used by forked summaries', () => {
const message = createSummaryPromptMessage('Summarize progress')
expect(message.type).toBe('user')
expect(message.message.role).toBe('user')
expect(message.message.content).toBe('Summarize progress')
expect(message.uuid).toBeString()
expect(message.timestamp).toBeString()
})
})

View File

@@ -13,7 +13,6 @@
import type { TaskContext } from '../../Task.js' import type { TaskContext } from '../../Task.js'
import { isPoorModeActive } from '../../commands/poor/poorMode.js' import { isPoorModeActive } from '../../commands/poor/poorMode.js'
import { updateAgentSummary } from '../../tasks/LocalAgentTask/LocalAgentTask.js' import { updateAgentSummary } from '../../tasks/LocalAgentTask/LocalAgentTask.js'
import { filterIncompleteToolCalls } from '@claude-code-best/builtin-tools/tools/AgentTool/runAgent.js'
import type { AgentId } from '../../types/ids.js' import type { AgentId } from '../../types/ids.js'
import { logForDebugging } from '../../utils/debug.js' import { logForDebugging } from '../../utils/debug.js'
import { import {
@@ -21,38 +20,32 @@ import {
runForkedAgent, runForkedAgent,
} from '../../utils/forkedAgent.js' } from '../../utils/forkedAgent.js'
import { logError } from '../../utils/log.js' import { logError } from '../../utils/log.js'
import { createUserMessage } from '../../utils/messages.js'
import { getAgentTranscript } from '../../utils/sessionStorage.js' import { getAgentTranscript } from '../../utils/sessionStorage.js'
import { buildSummaryContext } from './summaryContext.js'
import { import {
getSummaryContextFingerprint, buildSummaryPrompt,
selectSummaryContextMessages, createSummaryPromptMessage,
} from './summaryContext.js' } from './summaryPrompt.js'
const SUMMARY_INTERVAL_MS = 30_000 const SUMMARY_INTERVAL_MS = 30_000
function buildSummaryPrompt(previousSummary: string | null): string { export type AgentSummaryDependencies = Partial<{
const prevLine = previousSummary clearTimeout: typeof clearTimeout
? `\nPrevious: "${previousSummary}" — say something NEW.\n` getAgentTranscript: typeof getAgentTranscript
: '' isPoorModeActive: typeof isPoorModeActive
logError: typeof logError
return `Describe your most recent action in 3-5 words using present tense (-ing). Name the file or function, not the branch. Do not use tools. logForDebugging: typeof logForDebugging
${prevLine} runForkedAgent: typeof runForkedAgent
Good: "Reading runAgent.ts" setTimeout: typeof setTimeout
Good: "Fixing null check in validate.ts" updateAgentSummary: typeof updateAgentSummary
Good: "Running auth module tests" }>
Good: "Adding retry logic to fetchUser"
Bad (past tense): "Analyzed the branch diff"
Bad (too vague): "Investigating the issue"
Bad (too long): "Reviewing full branch diff and AgentTool.tsx integration"
Bad (branch name): "Analyzed adam/background-summary branch diff"`
}
export function startAgentSummarization( export function startAgentSummarization(
taskId: string, taskId: string,
agentId: AgentId, agentId: AgentId,
cacheSafeParams: CacheSafeParams, cacheSafeParams: CacheSafeParams,
setAppState: TaskContext['setAppState'], setAppState: TaskContext['setAppState'],
dependencies: AgentSummaryDependencies = {},
): { stop: () => void } { ): { stop: () => void } {
// Drop forkContextMessages from the closure — runSummary rebuilds it each // Drop forkContextMessages from the closure — runSummary rebuilds it each
// tick from getAgentTranscript(). Without this, the original fork messages // tick from getAgentTranscript(). Without this, the original fork messages
@@ -63,46 +56,53 @@ export function startAgentSummarization(
let stopped = false let stopped = false
let previousSummary: string | null = null let previousSummary: string | null = null
let lastHandledTranscriptFingerprint: string | null = null let lastHandledTranscriptFingerprint: string | null = null
const clearTimeoutImpl = dependencies.clearTimeout ?? clearTimeout
const getAgentTranscriptImpl =
dependencies.getAgentTranscript ?? getAgentTranscript
const isPoorModeActiveImpl =
dependencies.isPoorModeActive ?? isPoorModeActive
const logErrorImpl = dependencies.logError ?? logError
const logForDebuggingImpl =
dependencies.logForDebugging ?? logForDebugging
const runForkedAgentImpl = dependencies.runForkedAgent ?? runForkedAgent
const setTimeoutImpl = dependencies.setTimeout ?? setTimeout
const updateAgentSummaryImpl =
dependencies.updateAgentSummary ?? updateAgentSummary
async function runSummary(): Promise<void> { async function runSummary(): Promise<void> {
if (stopped) return if (stopped) return
if (isPoorModeActive()) { if (isPoorModeActiveImpl()) {
logForDebugging('[AgentSummary] Skipping summary — poor mode active') logForDebuggingImpl('[AgentSummary] Skipping summary — poor mode active')
scheduleNext() scheduleNext()
return return
} }
logForDebugging(`[AgentSummary] Timer fired for agent ${agentId}`) logForDebuggingImpl(`[AgentSummary] Timer fired for agent ${agentId}`)
try { try {
// Read current messages from transcript // Read current messages from transcript
const transcript = await getAgentTranscript(agentId) const transcript = await getAgentTranscriptImpl(agentId)
if (!transcript || transcript.messages.length < 3) { if (!transcript || transcript.messages.length < 3) {
// Not enough context yet — finally block will schedule next attempt // Not enough context yet — finally block will schedule next attempt
logForDebugging( logForDebuggingImpl(
`[AgentSummary] Skipping summary for ${taskId}: not enough messages (${transcript?.messages.length ?? 0})`, `[AgentSummary] Skipping summary for ${taskId}: not enough messages (${transcript?.messages.length ?? 0})`,
) )
return return
} }
// Filter to clean message state const summaryContext = buildSummaryContext(
const cleanMessages = filterIncompleteToolCalls(transcript.messages) transcript.messages,
const summaryContext = filterIncompleteToolCalls( lastHandledTranscriptFingerprint,
selectSummaryContextMessages(cleanMessages),
) )
const transcriptFingerprint = getSummaryContextFingerprint(summaryContext) if (summaryContext.skipReason === 'unchanged') {
if ( logForDebuggingImpl(
transcriptFingerprint &&
transcriptFingerprint === lastHandledTranscriptFingerprint
) {
logForDebugging(
`[AgentSummary] Skipping summary for ${taskId}: transcript unchanged`, `[AgentSummary] Skipping summary for ${taskId}: transcript unchanged`,
) )
return return
} }
if (summaryContext.length < 3) { if (summaryContext.skipReason === 'too_small') {
logForDebugging( logForDebuggingImpl(
`[AgentSummary] Skipping summary for ${taskId}: no bounded context available`, `[AgentSummary] Skipping summary for ${taskId}: no bounded context available`,
) )
return return
@@ -111,11 +111,11 @@ export function startAgentSummarization(
// Build fork params with current messages // Build fork params with current messages
const forkParams: CacheSafeParams = { const forkParams: CacheSafeParams = {
...baseParams, ...baseParams,
forkContextMessages: summaryContext, forkContextMessages: summaryContext.messages,
} }
logForDebugging( logForDebuggingImpl(
`[AgentSummary] Forking for summary, ${summaryContext.length} messages in context`, `[AgentSummary] Forking for summary, ${summaryContext.messages.length} messages in context`,
) )
// Create abort controller for this summary // Create abort controller for this summary
@@ -137,9 +137,9 @@ export function startAgentSummarization(
// ContentReplacementState is cloned by default in createSubagentContext // ContentReplacementState is cloned by default in createSubagentContext
// from forkParams.toolUseContext (the subagent's LIVE state captured at // from forkParams.toolUseContext (the subagent's LIVE state captured at
// onCacheSafeParams time). No explicit override needed. // onCacheSafeParams time). No explicit override needed.
const result = await runForkedAgent({ const result = await runForkedAgentImpl({
promptMessages: [ promptMessages: [
createUserMessage({ content: buildSummaryPrompt(previousSummary) }), createSummaryPromptMessage(buildSummaryPrompt(previousSummary)),
], ],
cacheSafeParams: forkParams, cacheSafeParams: forkParams,
canUseTool, canUseTool,
@@ -167,18 +167,18 @@ export function startAgentSummarization(
const textBlock = contentArr.find(b => b.type === 'text') const textBlock = contentArr.find(b => b.type === 'text')
if (textBlock?.type === 'text' && textBlock.text.trim()) { if (textBlock?.type === 'text' && textBlock.text.trim()) {
const summaryText = textBlock.text.trim() const summaryText = textBlock.text.trim()
logForDebugging( logForDebuggingImpl(
`[AgentSummary] Summary result for ${taskId}: ${summaryText}`, `[AgentSummary] Summary result for ${taskId}: ${summaryText}`,
) )
lastHandledTranscriptFingerprint = transcriptFingerprint lastHandledTranscriptFingerprint = summaryContext.fingerprint
previousSummary = summaryText previousSummary = summaryText
updateAgentSummary(taskId, summaryText, setAppState) updateAgentSummaryImpl(taskId, summaryText, setAppState)
break break
} }
} }
} catch (e) { } catch (e) {
if (!stopped && e instanceof Error) { if (!stopped && e instanceof Error) {
logError(e) logErrorImpl(e)
} }
} finally { } finally {
summaryAbortController = null summaryAbortController = null
@@ -191,14 +191,14 @@ export function startAgentSummarization(
function scheduleNext(): void { function scheduleNext(): void {
if (stopped) return if (stopped) return
timeoutId = setTimeout(runSummary, SUMMARY_INTERVAL_MS) timeoutId = setTimeoutImpl(runSummary, SUMMARY_INTERVAL_MS)
} }
function stop(): void { function stop(): void {
logForDebugging(`[AgentSummary] Stopping summarization for ${taskId}`) logForDebuggingImpl(`[AgentSummary] Stopping summarization for ${taskId}`)
stopped = true stopped = true
if (timeoutId) { if (timeoutId) {
clearTimeout(timeoutId) clearTimeoutImpl(timeoutId)
timeoutId = null timeoutId = null
} }
if (summaryAbortController) { if (summaryAbortController) {

View File

@@ -1,4 +1,5 @@
import { createHash } from 'crypto' import { createHash } from 'node:crypto'
import { filterIncompleteToolCalls } from '@claude-code-best/builtin-tools/tools/AgentTool/filterIncompleteToolCalls.js'
import type { Message } from '../../types/message.js' import type { Message } from '../../types/message.js'
export const MAX_SUMMARY_CONTEXT_MESSAGES = 120 export const MAX_SUMMARY_CONTEXT_MESSAGES = 120
@@ -178,3 +179,41 @@ export function selectSummaryContextMessages(
return selected return selected
} }
export type SummaryContextBuildResult = {
messages: Message[]
fingerprint: string | null
skipReason?: 'too_small' | 'unchanged'
}
export function buildSummaryContext(
messages: Message[],
previousFingerprint: string | null,
): SummaryContextBuildResult {
const cleanMessages = filterIncompleteToolCalls(messages)
const boundedMessages = filterIncompleteToolCalls(
selectSummaryContextMessages(cleanMessages),
)
const fingerprint = getSummaryContextFingerprint(boundedMessages)
if (fingerprint && fingerprint === previousFingerprint) {
return {
messages: boundedMessages,
fingerprint,
skipReason: 'unchanged',
}
}
if (boundedMessages.length < 3) {
return {
messages: boundedMessages,
fingerprint,
skipReason: 'too_small',
}
}
return {
messages: boundedMessages,
fingerprint,
}
}

View File

@@ -0,0 +1,32 @@
import { randomUUID, type UUID } from 'node:crypto'
import type { UserMessage } from '../../types/message.js'
export function buildSummaryPrompt(previousSummary: string | null): string {
const prevLine = previousSummary
? `\nPrevious: "${previousSummary}" — say something NEW.\n`
: ''
return `Describe your most recent action in 3-5 words using present tense (-ing). Name the file or function, not the branch. Do not use tools.
${prevLine}
Good: "Reading runAgent.ts"
Good: "Fixing null check in validate.ts"
Good: "Running auth module tests"
Good: "Adding retry logic to fetchUser"
Bad (past tense): "Analyzed the branch diff"
Bad (too vague): "Investigating the issue"
Bad (too long): "Reviewing full branch diff and AgentTool.tsx integration"
Bad (branch name): "Analyzed adam/background-summary branch diff"`
}
export function createSummaryPromptMessage(content: string): UserMessage {
return {
type: 'user',
message: {
role: 'user',
content,
},
uuid: randomUUID() as UUID,
timestamp: new Date().toISOString(),
}
}

View File

@@ -88,4 +88,66 @@ describe('attachNdjsonFramer', () => {
expect(errors[0]?.message).toContain('NDJSON frame exceeded') expect(errors[0]?.message).toContain('NDJSON frame exceeded')
expect(socket.destroyed).toBe(true) expect(socket.destroyed).toBe(true)
}) })
test('lets callers own oversized-frame shutdown when configured', () => {
const socket = createTestSocket()
const errors: Error[] = []
attachNdjsonFramer(
socket,
() => undefined,
text => JSON.parse(text) as unknown,
{
maxFrameBytes: 8,
onFrameError: error => errors.push(error),
destroyOnFrameError: false,
},
)
socket.emitData(Buffer.from('{"long":true}\n'))
expect(errors[0]?.message).toContain('NDJSON frame exceeded')
expect(socket.destroyed).toBe(false)
})
test('reports malformed non-empty frames without changing default compatibility', () => {
const socket = createTestSocket()
const messages: unknown[] = []
const errors: Error[] = []
attachNdjsonFramer(
socket,
msg => messages.push(msg),
text => JSON.parse(text) as unknown,
{
onInvalidFrame: error => errors.push(error),
},
)
socket.emitData(Buffer.from('{not-json\n'))
expect(messages).toEqual([])
expect(errors).toHaveLength(1)
expect(socket.destroyed).toBe(false)
})
test('destroys malformed frames when configured by the caller', () => {
const socket = createTestSocket()
const errors: Error[] = []
attachNdjsonFramer(
socket,
() => undefined,
text => JSON.parse(text) as unknown,
{
destroyOnInvalidFrame: true,
onInvalidFrame: error => errors.push(error),
},
)
socket.emitData(Buffer.from('{not-json\n'))
expect(errors).toHaveLength(1)
expect(socket.destroyed).toBe(true)
})
}) })

View File

@@ -3,7 +3,7 @@ import { mkdir, readFile, rm, writeFile } from 'node:fs/promises'
import { mkdtempSync } from 'node:fs' import { mkdtempSync } from 'node:fs'
import { tmpdir } from 'node:os' import { tmpdir } from 'node:os'
import { dirname, join } from 'node:path' import { dirname, join } from 'node:path'
import type { Message } from '../../types/message.js' import type { Message } from 'src/types/message.js'
import { import {
compactMailboxMessages, compactMailboxMessages,
getLastPeerDmSummary, getLastPeerDmSummary,
@@ -13,13 +13,14 @@ import {
markMessagesAsRead, markMessagesAsRead,
markMessagesAsReadByPredicate, markMessagesAsReadByPredicate,
MAX_MAILBOX_MESSAGE_TEXT_BYTES, MAX_MAILBOX_MESSAGE_TEXT_BYTES,
MAX_MAILBOX_FILE_BYTES,
MAX_MAILBOX_MESSAGES, MAX_MAILBOX_MESSAGES,
MAX_READ_MAILBOX_MESSAGES, MAX_READ_MAILBOX_MESSAGES,
MAX_UNREAD_PROTOCOL_MAILBOX_MESSAGES, MAX_UNREAD_PROTOCOL_MAILBOX_MESSAGES,
readMailbox, readMailbox,
type TeammateMessage, type TeammateMessage,
writeToMailbox, writeToMailbox,
} from '../teammateMailbox.js' } from 'src/utils/teammateMailbox.js'
let tempHome = '' let tempHome = ''
let previousConfigDir: string | undefined let previousConfigDir: string | undefined
@@ -55,21 +56,6 @@ async function readRawMailbox(
return JSON.parse(content) as TeammateMessage[] return JSON.parse(content) as TeammateMessage[]
} }
beforeEach(() => {
previousConfigDir = process.env.CLAUDE_CONFIG_DIR
tempHome = mkdtempSync(join(tmpdir(), 'teammate-mailbox-'))
process.env.CLAUDE_CONFIG_DIR = tempHome
})
afterEach(async () => {
if (previousConfigDir === undefined) {
delete process.env.CLAUDE_CONFIG_DIR
} else {
process.env.CLAUDE_CONFIG_DIR = previousConfigDir
}
await rm(tempHome, { recursive: true, force: true })
})
describe('compactMailboxMessages', () => { describe('compactMailboxMessages', () => {
test('prioritizes unread messages and keeps only recent read history', () => { test('prioritizes unread messages and keeps only recent read history', () => {
const compacted = compactMailboxMessages( const compacted = compactMailboxMessages(
@@ -175,9 +161,46 @@ describe('compactMailboxMessages', () => {
expect(compacted.length).toBeLessThan(20) expect(compacted.length).toBeLessThan(20)
expect(compacted.at(-1)?.text).toContain('msg-19') expect(compacted.at(-1)?.text).toContain('msg-19')
}) })
test('returns an empty mailbox when even one message exceeds retained budget', () => {
const compacted = compactMailboxMessages([message('too-large', false)], {
maxMessages: 10,
maxReadMessages: 0,
maxRetainedBytes: 1,
})
expect(compacted).toEqual([])
})
test('returns an empty mailbox when all retention lanes are disabled', () => {
const compacted = compactMailboxMessages([message('unread', false)], {
maxMessages: 0,
maxReadMessages: 0,
maxUnreadProtocolMessages: 0,
maxRetainedBytes: 1_000,
})
expect(compacted).toEqual([])
})
}) })
describe('teammate mailbox retention', () => { describe('teammate mailbox retention', () => {
beforeEach(() => {
previousConfigDir = process.env.CLAUDE_CONFIG_DIR
tempHome = mkdtempSync(join(tmpdir(), 'teammate-mailbox-'))
process.env.CLAUDE_CONFIG_DIR = tempHome
})
afterEach(async () => {
if (previousConfigDir === undefined) {
delete process.env.CLAUDE_CONFIG_DIR
} else {
process.env.CLAUDE_CONFIG_DIR = previousConfigDir
}
await rm(tempHome, { recursive: true, force: true })
tempHome = ''
})
test('writeToMailbox compacts oversized unread inbox files', async () => { test('writeToMailbox compacts oversized unread inbox files', async () => {
const existing = Array.from( const existing = Array.from(
{ length: MAX_MAILBOX_MESSAGES + 20 }, { length: MAX_MAILBOX_MESSAGES + 20 },
@@ -319,6 +342,23 @@ describe('teammate mailbox retention', () => {
expect(await readFile(inboxPath, 'utf-8')).toBe('{not-json') expect(await readFile(inboxPath, 'utf-8')).toBe('{not-json')
}) })
test('writeToMailbox rejects when the inbox path is already a directory', async () => {
const inboxPath = getInboxPath('worker', 'alpha')
await mkdir(inboxPath, { recursive: true })
await expect(
writeToMailbox(
'worker',
{
from: 'team-lead',
text: 'new',
timestamp: new Date(5).toISOString(),
},
'alpha',
),
).rejects.toThrow()
})
test('readMailbox fails closed on corrupt mailbox content', async () => { test('readMailbox fails closed on corrupt mailbox content', async () => {
const inboxPath = getInboxPath('worker', 'alpha') const inboxPath = getInboxPath('worker', 'alpha')
await mkdir(dirname(inboxPath), { recursive: true }) await mkdir(dirname(inboxPath), { recursive: true })
@@ -326,6 +366,76 @@ describe('teammate mailbox retention', () => {
await expect(readMailbox('worker', 'alpha')).rejects.toThrow() await expect(readMailbox('worker', 'alpha')).rejects.toThrow()
}) })
test('readMailbox rejects non-array mailbox files', async () => {
const inboxPath = getInboxPath('worker', 'alpha')
await mkdir(dirname(inboxPath), { recursive: true })
await writeFile(inboxPath, JSON.stringify({ text: 'not an array' }), 'utf-8')
await expect(readMailbox('worker', 'alpha')).rejects.toThrow(
'expected message array',
)
})
test('readMailbox rejects malformed stored message shapes', async () => {
const inboxPath = getInboxPath('worker', 'alpha')
await mkdir(dirname(inboxPath), { recursive: true })
await writeFile(
inboxPath,
JSON.stringify([{ from: 'lead', text: 'missing timestamp' }]),
'utf-8',
)
await expect(readMailbox('worker', 'alpha')).rejects.toThrow(
'Invalid mailbox message shape',
)
})
test('readMailbox rejects non-object stored messages', async () => {
const inboxPath = getInboxPath('worker', 'alpha')
await mkdir(dirname(inboxPath), { recursive: true })
await writeFile(inboxPath, JSON.stringify(['not an object']), 'utf-8')
await expect(readMailbox('worker', 'alpha')).rejects.toThrow(
'expected object',
)
})
test('readMailbox rejects oversized mailbox files before parsing', async () => {
const inboxPath = getInboxPath('worker', 'alpha')
await mkdir(dirname(inboxPath), { recursive: true })
await writeFile(inboxPath, `[${' '.repeat(MAX_MAILBOX_FILE_BYTES)}]`, 'utf-8')
await expect(readMailbox('worker', 'alpha')).rejects.toThrow(
'Mailbox file exceeds',
)
})
test('markMessageAsReadByIdentity returns false for missing mailbox files', async () => {
await expect(
markMessageAsReadByIdentity('worker', 'alpha', message('absent', false)),
).resolves.toBe(false)
})
test('markMessageAsReadByIdentity returns false when the expected message moved out', async () => {
await seedMailbox('worker', 'alpha', [message('other', false)])
await expect(
markMessageAsReadByIdentity('worker', 'alpha', message('missing', false)),
).resolves.toBe(false)
expect((await readRawMailbox('worker', 'alpha'))[0]?.read).toBe(false)
})
test('markMessageAsReadByIdentity returns false on corrupt mailbox content', async () => {
const inboxPath = getInboxPath('worker', 'alpha')
await mkdir(dirname(inboxPath), { recursive: true })
await writeFile(inboxPath, '{not-json', 'utf-8')
await expect(
markMessageAsReadByIdentity('worker', 'alpha', message('missing', false)),
).resolves.toBe(false)
})
}) })
describe('getLastPeerDmSummary', () => { describe('getLastPeerDmSummary', () => {

View File

@@ -3,24 +3,31 @@ import {
chmod, chmod,
mkdir, mkdir,
mkdtemp, mkdtemp,
readdir,
rm, rm,
stat, stat,
symlink, symlink,
unlink, unlink,
writeFile,
} from 'node:fs/promises' } from 'node:fs/promises'
import { createHash } from 'node:crypto'
import { createConnection, createServer } from 'node:net' import { createConnection, createServer } from 'node:net'
import { dirname, join } from 'node:path' import { dirname, join } from 'node:path'
import { tmpdir } from 'node:os' import { tmpdir } from 'node:os'
import { import {
drainInbox, drainInbox,
getDefaultUdsSocketPath,
MAX_UDS_INBOX_ENTRIES, MAX_UDS_INBOX_ENTRIES,
MAX_UDS_INBOX_BYTES, MAX_UDS_INBOX_BYTES,
MAX_UDS_FRAME_BYTES, MAX_UDS_FRAME_BYTES,
MAX_UDS_CLIENTS,
formatUdsAddress,
parseUdsTarget, parseUdsTarget,
sendUdsMessage, sendUdsMessage,
setOnEnqueue, setOnEnqueue,
startUdsMessaging, startUdsMessaging,
stopUdsMessaging, stopUdsMessaging,
UDS_AUTH_TIMEOUT_MS,
} from '../udsMessaging.js' } from '../udsMessaging.js'
let previousConfigDir: string | undefined let previousConfigDir: string | undefined
@@ -192,7 +199,7 @@ describe('UDS inbox retention', () => {
try { try {
const { isPeerAlive } = await import('../udsClient.js') const { isPeerAlive } = await import('../udsClient.js')
expect(await isPeerAlive(path)).toBe(false) expect(await isPeerAlive(path, 3_000, 'test-token')).toBe(false)
} finally { } finally {
await closeServer(receiver) await closeServer(receiver)
if (process.platform !== 'win32') { if (process.platform !== 'win32') {
@@ -210,6 +217,29 @@ describe('UDS inbox retention', () => {
) )
}) })
test('udsClient send reports connection failures without leaking token state', async () => {
const path = socketPath('uds-client-connect-error')
const capabilityDir = join(tempConfigDir, 'messaging-capabilities')
const capabilityName = `${createHash('sha256').update(path).digest('hex')}.json`
await mkdir(capabilityDir, { recursive: true, mode: 0o700 })
await writeFile(
join(capabilityDir, capabilityName),
JSON.stringify({ socketPath: path, authToken: 'test-token' }),
'utf-8',
)
const { sendToUdsSocket } = await import('../udsClient.js')
await expect(sendToUdsSocket(path, 'hello')).rejects.toThrow(
'Failed to connect to peer',
)
})
test('sendUdsMessage fails closed before connecting without an auth token', async () => {
await expect(
sendUdsMessage(socketPath('no-auth-token'), { type: 'text', data: 'x' }),
).rejects.toThrow('without auth token')
})
test('drained entries never expose the UDS auth token', async () => { test('drained entries never expose the UDS auth token', async () => {
const path = socketPath('strip-token') const path = socketPath('strip-token')
await startUdsMessaging(path, { isExplicit: true }) await startUdsMessaging(path, { isExplicit: true })
@@ -232,6 +262,7 @@ describe('UDS inbox retention', () => {
await startUdsMessaging(path, { isExplicit: true }) await startUdsMessaging(path, { isExplicit: true })
const response = await new Promise<string>((resolve, reject) => { const response = await new Promise<string>((resolve, reject) => {
let responseText = ''
const conn = createConnection(path, () => { const conn = createConnection(path, () => {
conn.write(`${JSON.stringify({ type: 'text', data: 'bad' })}\n`) conn.write(`${JSON.stringify({ type: 'text', data: 'bad' })}\n`)
}) })
@@ -242,10 +273,10 @@ describe('UDS inbox retention', () => {
conn.on('data', chunk => { conn.on('data', chunk => {
const text = chunk.toString('utf-8') const text = chunk.toString('utf-8')
if (text.includes('\n')) { if (text.includes('\n')) {
conn.end() responseText = text
resolve(text)
} }
}) })
conn.on('close', () => resolve(responseText))
conn.on('error', reject) conn.on('error', reject)
}) })
@@ -253,6 +284,56 @@ describe('UDS inbox retention', () => {
expect(drainInbox()).toEqual([]) expect(drainInbox()).toEqual([])
}) })
test('disconnects malformed JSON clients without enqueueing inbox work', async () => {
const path = socketPath('malformed-client')
await startUdsMessaging(path, { isExplicit: true })
const response = await new Promise<string>((resolve, reject) => {
let responseText = ''
const conn = createConnection(path, () => {
conn.write('{not-json\n')
})
conn.setTimeout(5_000, () => {
conn.destroy()
reject(new Error('Timed out waiting for malformed frame close'))
})
conn.on('data', chunk => {
responseText += chunk.toString('utf-8')
})
conn.on('close', () => resolve(responseText))
conn.on('error', reject)
})
const parsed = JSON.parse(response)
expect(parsed.type).toBe('error')
expect(parsed.data).toBe('invalid frame')
expect(drainInbox()).toEqual([])
})
test('disconnects idle unauthenticated clients', async () => {
const path = socketPath('idle-client')
await startUdsMessaging(path, { isExplicit: true })
const response = await new Promise<string>((resolve, reject) => {
let responseText = ''
const conn = createConnection(path)
conn.setTimeout(UDS_AUTH_TIMEOUT_MS + 2_000, () => {
conn.destroy()
reject(new Error('Timed out waiting for auth timeout close'))
})
conn.on('data', chunk => {
responseText += chunk.toString('utf-8')
})
conn.on('close', () => resolve(responseText))
conn.on('error', reject)
})
const parsed = JSON.parse(response)
expect(parsed.type).toBe('error')
expect(parsed.data).toBe('authentication timeout')
expect(drainInbox()).toEqual([])
})
test('destroys oversized frames before enqueueing inbox work', async () => { test('destroys oversized frames before enqueueing inbox work', async () => {
const path = socketPath('oversized') const path = socketPath('oversized')
await startUdsMessaging(path, { isExplicit: true }) await startUdsMessaging(path, { isExplicit: true })
@@ -272,6 +353,14 @@ describe('UDS inbox retention', () => {
expect(drainInbox()).toEqual([]) expect(drainInbox()).toEqual([])
}) })
test('default socket path is regenerated after stop', async () => {
const firstPath = getDefaultUdsSocketPath()
await startUdsMessaging(firstPath)
await stopUdsMessaging()
expect(getDefaultUdsSocketPath()).not.toBe(firstPath)
})
test('rejects oversized receiver responses before retaining them', async () => { test('rejects oversized receiver responses before retaining them', async () => {
const path = socketPath('oversized-response') const path = socketPath('oversized-response')
if (process.platform !== 'win32') { if (process.platform !== 'win32') {
@@ -303,9 +392,71 @@ describe('UDS inbox retention', () => {
} }
}) })
test('rejects closed receiver responses without waiting for timeout', async () => {
const path = socketPath('closed-response')
if (process.platform !== 'win32') {
await mkdir(dirname(path), { recursive: true })
}
const receiver = createServer(socket => {
socket.end()
})
await new Promise<void>((resolve, reject) => {
receiver.on('error', reject)
receiver.listen(path, () => resolve())
})
try {
await expect(
sendUdsMessage(
path,
{ type: 'text', data: 'hello' },
{ authToken: 'test-token' },
),
).rejects.toThrow('before response')
} finally {
await closeServer(receiver)
if (process.platform !== 'win32') {
await unlink(path).catch(() => undefined)
}
}
})
test('rejects malformed receiver responses without waiting for timeout', async () => {
const path = socketPath('malformed-response')
if (process.platform !== 'win32') {
await mkdir(dirname(path), { recursive: true })
}
const receiver = createServer(socket => {
socket.on('data', () => {
socket.write('{not-json\n')
})
})
await new Promise<void>((resolve, reject) => {
receiver.on('error', reject)
receiver.listen(path, () => resolve())
})
try {
await expect(
sendUdsMessage(
path,
{ type: 'text', data: 'hello' },
{ authToken: 'test-token' },
),
).rejects.toThrow('Invalid UDS response frame')
} finally {
await closeServer(receiver)
if (process.platform !== 'win32') {
await unlink(path).catch(() => undefined)
}
}
})
test('rejects inline auth token UDS targets instead of parsing them', async () => { test('rejects inline auth token UDS targets instead of parsing them', async () => {
const path = socketPath('inline-token') const path = socketPath('inline-token')
expect(formatUdsAddress(path)).toBe(`uds:${path}`)
const targetWithToken = `${path}#token=secret` const targetWithToken = `${path}#token=secret`
expect(() => parseUdsTarget(targetWithToken)).toThrow('inline auth token') expect(() => parseUdsTarget(targetWithToken)).toThrow('inline auth token')
try { try {
@@ -320,6 +471,23 @@ describe('UDS inbox retention', () => {
) )
}) })
test('fails closed and cleans temp files when capability target is occupied', async () => {
const path = socketPath('capability-target-dir')
const capabilityDir = join(tempConfigDir, 'messaging-capabilities')
const capabilityName = `${createHash('sha256').update(path).digest('hex')}.json`
await mkdir(join(capabilityDir, capabilityName), {
recursive: true,
mode: 0o700,
})
await expect(
startUdsMessaging(path, { isExplicit: true }),
).rejects.toThrow()
expect(process.env.CLAUDE_CODE_MESSAGING_SOCKET).toBeUndefined()
expect(await readdir(capabilityDir)).toEqual([capabilityName])
})
if (process.platform !== 'win32') { if (process.platform !== 'win32') {
test('creates the listening socket with owner-only permissions', async () => { test('creates the listening socket with owner-only permissions', async () => {
const path = socketPath('socket-mode') const path = socketPath('socket-mode')
@@ -341,9 +509,11 @@ describe('UDS inbox retention', () => {
await chmod(capabilityDir, 0o755) await chmod(capabilityDir, 0o755)
try { try {
const path = socketPath('broad-capdir')
await expect( await expect(
startUdsMessaging(socketPath('broad-capdir'), { isExplicit: true }), startUdsMessaging(path, { isExplicit: true }),
).rejects.toThrow('permissions are too broad') ).rejects.toThrow('permissions are too broad')
await expect(stat(path)).rejects.toThrow()
} finally { } finally {
if (previousConfigDir === undefined) { if (previousConfigDir === undefined) {
delete process.env.CLAUDE_CONFIG_DIR delete process.env.CLAUDE_CONFIG_DIR
@@ -397,5 +567,65 @@ describe('UDS inbox retention', () => {
await rm(parent, { recursive: true, force: true }) await rm(parent, { recursive: true, force: true })
} }
}) })
test('fails closed when an explicit socket parent is a file', async () => {
const parentFile = join(
tmpdir(),
`uds-socket-parent-file-${process.pid}-${Date.now()}-${Math.random().toString(16).slice(2)}`,
)
await writeFile(parentFile, 'not a directory', 'utf-8')
try {
await expect(
startUdsMessaging(join(parentFile, 'messaging.sock'), {
isExplicit: true,
}),
).rejects.toThrow('socket parent is not a directory')
} finally {
await rm(parentFile, { force: true })
}
})
test('stop tolerates an already removed socket path', async () => {
const path = socketPath('already-removed')
await startUdsMessaging(path, { isExplicit: true })
await unlink(path)
await stopUdsMessaging()
expect(process.env.CLAUDE_CODE_MESSAGING_SOCKET).toBeUndefined()
})
test('rejects clients over the configured connection cap', async () => {
const path = socketPath('client-cap')
await startUdsMessaging(path, { isExplicit: true })
const sockets: ReturnType<typeof createConnection>[] = []
try {
for (let i = 0; i < MAX_UDS_CLIENTS; i++) {
const socket = await new Promise<ReturnType<typeof createConnection>>(
(resolve, reject) => {
const conn = createConnection(path, () => resolve(conn))
conn.on('error', reject)
},
)
sockets.push(socket)
}
await new Promise<void>((resolve, reject) => {
const extra = createConnection(path)
extra.on('close', () => resolve())
extra.on('error', reject)
extra.setTimeout(5_000, () => {
extra.destroy()
reject(new Error('Timed out waiting for client cap close'))
})
})
} finally {
for (const socket of sockets) {
socket.destroy()
}
}
})
} }
}) })

View File

@@ -0,0 +1,218 @@
import { describe, expect, test } from 'bun:test'
import { EventEmitter } from 'node:events'
import type { Socket } from 'node:net'
import { attachUdsResponseReader } from '../udsResponseReader.js'
class FakeSocket extends EventEmitter {
destroyed = false
ended = false
destroy(): this {
this.destroyed = true
this.emit('close', true)
return this
}
end(): this {
this.ended = true
this.emit('close', false)
return this
}
emitData(chunk: Buffer): void {
this.emit('data', chunk)
}
}
function asSocket(socket: FakeSocket): Socket {
return socket as unknown as Socket
}
describe('attachUdsResponseReader', () => {
test('tracks byte limits across split multibyte response chunks', () => {
const socket = new FakeSocket()
let settled = false
let settledError: Error | undefined
attachUdsResponseReader(asSocket(socket), {
maxFrameBytes: 128,
onSettled: error => {
settled = true
settledError = error
},
})
const multibyte = String.fromCodePoint(0x20ac)
const frame = Buffer.from(
JSON.stringify({ type: 'response', data: `ok ${multibyte}` }) + '\n',
'utf8',
)
const multibyteStart = frame.indexOf(Buffer.from(multibyte, 'utf8')[0])
socket.emitData(frame.subarray(0, multibyteStart + 1))
expect(settled).toBe(false)
socket.emitData(frame.subarray(multibyteStart + 1))
expect(settled).toBe(true)
expect(settledError).toBeUndefined()
expect(socket.ended).toBe(true)
})
test('rejects malformed response frames immediately', () => {
const socket = new FakeSocket()
let settledError: Error | undefined
attachUdsResponseReader(asSocket(socket), {
maxFrameBytes: 128,
onSettled: error => {
settledError = error
},
})
socket.emitData(Buffer.from('{bad-json}\n'))
expect(settledError?.message).toBe('Invalid UDS response frame')
expect(socket.destroyed).toBe(true)
})
test('skips blank frames before a valid response', () => {
const socket = new FakeSocket()
let settled = false
let settledError: Error | undefined
attachUdsResponseReader(asSocket(socket), {
maxFrameBytes: 128,
onSettled: error => {
settled = true
settledError = error
},
})
socket.emitData(Buffer.from('\n \n'))
expect(settled).toBe(false)
socket.emitData(Buffer.from(`${JSON.stringify({ type: 'response' })}\n`))
expect(settled).toBe(true)
expect(settledError).toBeUndefined()
expect(socket.ended).toBe(true)
})
test('continues scanning when blank and valid frames share one chunk', () => {
const socket = new FakeSocket()
let settled = false
let settledError: Error | undefined
attachUdsResponseReader(asSocket(socket), {
maxFrameBytes: 128,
onSettled: error => {
settled = true
settledError = error
},
})
socket.emitData(
Buffer.from(`\n${JSON.stringify({ type: 'response' })}\n`),
)
expect(settled).toBe(true)
expect(settledError).toBeUndefined()
expect(socket.ended).toBe(true)
})
test('rejects receiver error frames', () => {
const socket = new FakeSocket()
let settledError: Error | undefined
attachUdsResponseReader(asSocket(socket), {
maxFrameBytes: 128,
onSettled: error => {
settledError = error
},
})
socket.emitData(
Buffer.from(`${JSON.stringify({ type: 'error', data: 'denied' })}\n`),
)
expect(settledError?.message).toBe('denied')
expect(socket.destroyed).toBe(true)
})
test('ignores unrelated receiver frames until a terminal response arrives', () => {
const socket = new FakeSocket()
let settled = false
let settledError: Error | undefined
attachUdsResponseReader(asSocket(socket), {
maxFrameBytes: 128,
onSettled: error => {
settled = true
settledError = error
},
})
socket.emitData(
Buffer.from(
`${JSON.stringify({ type: 'notification', data: 'queued' })}\n`,
),
)
expect(settled).toBe(false)
socket.emitData(Buffer.from(`${JSON.stringify({ type: 'response' })}\n`))
expect(settled).toBe(true)
expect(settledError).toBeUndefined()
})
test('uses custom socket error formatting', () => {
const socket = new FakeSocket()
let settledError: Error | undefined
attachUdsResponseReader(asSocket(socket), {
maxFrameBytes: 128,
onSettled: error => {
settledError = error
},
formatSocketError: error =>
new Error(`wrapped:${(error as Error).message}`),
})
socket.emit('error', new Error('connect failed'))
expect(settledError?.message).toBe('wrapped:connect failed')
expect(socket.destroyed).toBe(true)
})
test('rejects socket end before response', () => {
const socket = new FakeSocket()
let settledError: Error | undefined
attachUdsResponseReader(asSocket(socket), {
maxFrameBytes: 128,
onSettled: error => {
settledError = error
},
})
socket.emit('end')
expect(settledError?.message).toBe('UDS socket ended before response')
expect(socket.destroyed).toBe(true)
})
test('rejects clean socket close before response', () => {
const socket = new FakeSocket()
let settledError: Error | undefined
attachUdsResponseReader(asSocket(socket), {
maxFrameBytes: 128,
onSettled: error => {
settledError = error
},
})
socket.emit('close', false)
expect(settledError?.message).toBe('UDS socket closed before response')
expect(socket.destroyed).toBe(true)
})
})

View File

@@ -87,10 +87,8 @@ export function buildSystemInitMessage(inputs: SystemInitInputs): SDKMessage {
// Hidden from public SDK types — ant-only UDS messaging socket path // Hidden from public SDK types — ant-only UDS messaging socket path
if (feature('UDS_INBOX')) { if (feature('UDS_INBOX')) {
/* eslint-disable @typescript-eslint/no-require-imports */ /* eslint-disable @typescript-eslint/no-require-imports */
const udsMessaging =
require('../udsMessaging.js') as typeof import('../udsMessaging.js')
;(initMessage as Record<string, unknown>).messaging_socket_path = ;(initMessage as Record<string, unknown>).messaging_socket_path =
udsMessaging.getUdsMessagingSocketPath() require('../udsMessaging.js').getUdsMessagingSocketPath()
/* eslint-enable @typescript-eslint/no-require-imports */ /* eslint-enable @typescript-eslint/no-require-imports */
} }
initMessage.fast_mode_state = getFastModeState(inputs.model, inputs.fastMode) initMessage.fast_mode_state = getFastModeState(inputs.model, inputs.fastMode)

View File

@@ -10,11 +10,15 @@ import type { Socket } from 'net'
export type NdjsonFramerOptions = { export type NdjsonFramerOptions = {
maxFrameBytes?: number maxFrameBytes?: number
onFrameError?: (error: Error) => void onFrameError?: (error: Error) => void
destroyOnFrameError?: boolean
onInvalidFrame?: (error: Error) => void
destroyOnInvalidFrame?: boolean
} }
/** /**
* Attach an NDJSON framer to a socket. Calls `onMessage` for each * Attach an NDJSON framer to a socket. Calls `onMessage` for each
* complete JSON line received. Malformed lines are silently skipped. * complete JSON line received. Malformed lines are skipped by default;
* callers may opt into error callbacks or socket destruction.
* *
* @param parse - Optional custom JSON parser (defaults to JSON.parse). * @param parse - Optional custom JSON parser (defaults to JSON.parse).
* Useful when the caller uses a wrapped parser like jsonParse * Useful when the caller uses a wrapped parser like jsonParse
@@ -35,15 +39,26 @@ export function attachNdjsonFramer<T = unknown>(
`NDJSON frame exceeded ${maxFrameBytes} bytes (${bytes})`, `NDJSON frame exceeded ${maxFrameBytes} bytes (${bytes})`,
) )
options.onFrameError?.(error) options.onFrameError?.(error)
socket.destroy(error) if (options.destroyOnFrameError ?? true) {
socket.destroy(error)
}
}
const rejectInvalidFrame = (error: unknown): void => {
const frameError =
error instanceof Error ? error : new Error('Invalid NDJSON frame')
options.onInvalidFrame?.(frameError)
if (options.destroyOnInvalidFrame ?? false) {
socket.destroy(frameError)
}
} }
const emitLine = (line: string): void => { const emitLine = (line: string): void => {
if (!line.trim()) return if (!line.trim()) return
try { try {
onMessage(parse(line)) onMessage(parse(line))
} catch { } catch (error) {
// Malformed JSON — skip rejectInvalidFrame(error)
} }
} }

View File

@@ -1246,13 +1246,8 @@ export async function runInProcessTeammate(
// Track in-progress tool use IDs for animation in transcript view // Track in-progress tool use IDs for animation in transcript view
let inProgressToolUseIDs = task.inProgressToolUseIDs let inProgressToolUseIDs = task.inProgressToolUseIDs
if (message.type === 'assistant') { if (message.type === 'assistant') {
for (const block of Array.isArray(message.message!.content) for (const block of (Array.isArray(message.message!.content) ? message.message!.content : [])) {
? message.message!.content if (typeof block !== 'string' && block.type === 'tool_use') {
: []) {
if (
typeof block !== 'string' &&
block.type === 'tool_use'
) {
inProgressToolUseIDs = new Set([ inProgressToolUseIDs = new Set([
...(inProgressToolUseIDs ?? []), ...(inProgressToolUseIDs ?? []),
block.id, block.id,
@@ -1323,10 +1318,7 @@ export async function runInProcessTeammate(
setAppState, setAppState,
) )
if (currentAutonomyRunId) { if (currentAutonomyRunId) {
await markAutonomyRunFailed( await markAutonomyRunFailed(currentAutonomyRunId, ERROR_MESSAGE_USER_ABORT)
currentAutonomyRunId,
ERROR_MESSAGE_USER_ABORT,
)
currentAutonomyRunId = undefined currentAutonomyRunId = undefined
} }
} else if (currentAutonomyRunId) { } else if (currentAutonomyRunId) {

View File

@@ -82,6 +82,8 @@ export const MAX_UDS_INBOX_ENTRIES = 1_000
export const MAX_UDS_FRAME_BYTES = 64 * 1024 export const MAX_UDS_FRAME_BYTES = 64 * 1024
export const MAX_UDS_INBOX_BYTES = 2 * 1024 * 1024 export const MAX_UDS_INBOX_BYTES = 2 * 1024 * 1024
export const MAX_UDS_CLIENTS = 128 export const MAX_UDS_CLIENTS = 128
export const UDS_AUTH_TIMEOUT_MS = 2_000
export const UDS_IDLE_TIMEOUT_MS = 30_000
// --------------------------------------------------------------------------- // ---------------------------------------------------------------------------
// Public API — socket path helpers // Public API — socket path helpers
@@ -339,6 +341,43 @@ function writeSocketMessage(socket: Socket, message: UdsMessage): void {
socket.write(jsonStringify(message) + '\n') socket.write(jsonStringify(message) + '\n')
} }
function writeSocketMessageAndDestroy(socket: Socket, message: UdsMessage): void {
if (socket.destroyed) return
socket.write(jsonStringify(message) + '\n', () => {
if (!socket.destroyed) socket.destroy()
})
}
function writeSocketErrorAndDestroy(socket: Socket, data: string): void {
writeSocketMessageAndDestroy(socket, {
type: 'error',
data,
ts: new Date().toISOString(),
})
}
function unrefTimer(timer: ReturnType<typeof setTimeout>): void {
const maybeUnref = (timer as { unref?: () => void }).unref
if (typeof maybeUnref === 'function') {
maybeUnref.call(timer)
}
}
async function closeServer(serverToClose: Server): Promise<void> {
await new Promise<void>(resolve => {
serverToClose.close(() => resolve())
})
}
async function removeSocketPath(path: string): Promise<void> {
if (process.platform === 'win32') return
try {
await unlink(path)
} catch {
// Already gone.
}
}
function stripAuthToken(message: UdsMessage): UdsMessage { function stripAuthToken(message: UdsMessage): UdsMessage {
const { authToken: _authToken, ...metaWithoutAuth } = message.meta ?? {} const { authToken: _authToken, ...metaWithoutAuth } = message.meta ?? {}
return { return {
@@ -391,10 +430,9 @@ export async function startUdsMessaging(
} }
const token = ensureAuthToken() const token = ensureAuthToken()
let startedServer: Server | null = null
let exportedSocketEnv = false
try { try {
await writeCapabilityFile(path, token)
socketPath = path
await new Promise<void>((resolve, reject) => { await new Promise<void>((resolve, reject) => {
const srv = createServer(socket => { const srv = createServer(socket => {
if (clients.size >= MAX_UDS_CLIENTS) { if (clients.size >= MAX_UDS_CLIENTS) {
@@ -408,6 +446,24 @@ export async function startUdsMessaging(
logForDebugging( logForDebugging(
`[udsMessaging] client connected (total: ${clients.size})`, `[udsMessaging] client connected (total: ${clients.size})`,
) )
let authenticated = false
let closing = false
const closeWithError = (data: string): void => {
if (closing || socket.destroyed) return
closing = true
socket.pause()
writeSocketErrorAndDestroy(socket, data)
}
const authTimer = setTimeout(() => {
if (authenticated || socket.destroyed) return
logForDebugging('[udsMessaging] closing unauthenticated idle client')
closeWithError('authentication timeout')
}, UDS_AUTH_TIMEOUT_MS)
unrefTimer(authTimer)
socket.setTimeout(UDS_IDLE_TIMEOUT_MS, () => {
logForDebugging('[udsMessaging] closing idle client')
closeWithError('idle timeout')
})
attachNdjsonFramer<UdsMessage>( attachNdjsonFramer<UdsMessage>(
socket, socket,
@@ -416,17 +472,13 @@ export async function startUdsMessaging(
logForDebugging( logForDebugging(
`[udsMessaging] rejected unauthenticated message type=${msg.type}`, `[udsMessaging] rejected unauthenticated message type=${msg.type}`,
) )
if (!socket.destroyed) { closeWithError('unauthorized')
socket.write(
jsonStringify({
type: 'error',
data: 'unauthorized',
ts: new Date().toISOString(),
} satisfies UdsMessage) + '\n',
)
}
return return
} }
if (!authenticated) {
authenticated = true
clearTimeout(authTimer)
}
// Handle ping with automatic pong // Handle ping with automatic pong
if (msg.type === 'ping') { if (msg.type === 'ping') {
@@ -447,11 +499,7 @@ export async function startUdsMessaging(
status: 'pending', status: 'pending',
} }
if (!enqueueInboxEntry(entry)) { if (!enqueueInboxEntry(entry)) {
writeSocketMessage(socket, { closeWithError('inbox full')
type: 'error',
data: 'inbox full',
ts: new Date().toISOString(),
})
return return
} }
logForDebugging( logForDebugging(
@@ -470,21 +518,40 @@ export async function startUdsMessaging(
maxFrameBytes: MAX_UDS_FRAME_BYTES, maxFrameBytes: MAX_UDS_FRAME_BYTES,
onFrameError: error => { onFrameError: error => {
logForDebugging(`[udsMessaging] ${error.message}`) logForDebugging(`[udsMessaging] ${error.message}`)
closeWithError(error.message)
}, },
onInvalidFrame: error => {
logForDebugging(
`[udsMessaging] invalid client frame: ${errorMessage(error)}`,
)
closeWithError('invalid frame')
},
destroyOnFrameError: false,
}, },
) )
socket.on('close', () => { socket.on('close', () => {
clearTimeout(authTimer)
clients.delete(socket) clients.delete(socket)
}) })
socket.on('error', err => { socket.on('error', err => {
clearTimeout(authTimer)
clients.delete(socket) clients.delete(socket)
logForDebugging(`[udsMessaging] client error: ${errorMessage(err)}`) logForDebugging(`[udsMessaging] client error: ${errorMessage(err)}`)
}) })
}) })
srv.on('error', reject) const rejectBeforeListen = (error: Error): void => {
reject(error)
}
const logRuntimeError = (error: Error): void => {
logForDebugging(
`[udsMessaging] server error on ${path}${opts?.isExplicit ? ' (explicit)' : ''}: ${errorMessage(error)}`,
)
}
srv.once('error', rejectBeforeListen)
srv.listen(path, () => { srv.listen(path, () => {
void (async () => { void (async () => {
@@ -492,19 +559,41 @@ export async function startUdsMessaging(
if (process.platform !== 'win32') { if (process.platform !== 'win32') {
await chmod(path, 0o600) await chmod(path, 0o600)
} }
srv.off('error', rejectBeforeListen)
srv.on('error', logRuntimeError)
server = srv server = srv
// Export so child processes can discover the socket startedServer = srv
process.env.CLAUDE_CODE_MESSAGING_SOCKET = path
logForDebugging(
`[udsMessaging] server listening on ${path}${opts?.isExplicit ? ' (explicit)' : ''}`,
)
resolve() resolve()
} catch (error) { } catch (error) {
srv.close(() => reject(error)) srv.off('error', rejectBeforeListen)
const closeError =
error instanceof Error ? error : new Error(errorMessage(error))
let rejected = false
const rejectOnce = (): void => {
if (rejected) return
rejected = true
reject(closeError)
}
const fallback = setTimeout(rejectOnce, 1_000)
unrefTimer(fallback)
srv.close(() => {
clearTimeout(fallback)
rejectOnce()
})
} }
})() })()
}) })
}) })
await writeCapabilityFile(path, token)
socketPath = path
// Export so child processes can discover the socket only after the
// capability file exists and the listener is ready.
process.env.CLAUDE_CODE_MESSAGING_SOCKET = path
exportedSocketEnv = true
logForDebugging(
`[udsMessaging] server listening on ${path}${opts?.isExplicit ? ' (explicit)' : ''}`,
)
} catch (error) { } catch (error) {
if (capabilityFilePath) { if (capabilityFilePath) {
try { try {
@@ -514,7 +603,18 @@ export async function startUdsMessaging(
} }
capabilityFilePath = null capabilityFilePath = null
} }
if (startedServer) {
await closeServer(startedServer)
}
if (server === startedServer) {
server = null
}
await removeSocketPath(path)
if (exportedSocketEnv) {
delete process.env.CLAUDE_CODE_MESSAGING_SOCKET
}
socketPath = null socketPath = null
defaultSocketPath = null
authToken = null authToken = null
throw error throw error
} }
@@ -529,6 +629,7 @@ export async function startUdsMessaging(
* Stop the UDS messaging server and clean up the socket file. * Stop the UDS messaging server and clean up the socket file.
*/ */
export async function stopUdsMessaging(): Promise<void> { export async function stopUdsMessaging(): Promise<void> {
defaultSocketPath = null
if (!server) return if (!server) return
// Close all connected clients // Close all connected clients
@@ -547,13 +648,7 @@ export async function stopUdsMessaging(): Promise<void> {
// Remove socket file (skip on Windows — pipe paths aren't files) // Remove socket file (skip on Windows — pipe paths aren't files)
if (socketPath) { if (socketPath) {
if (process.platform !== 'win32') { await removeSocketPath(socketPath)
try {
await unlink(socketPath)
} catch {
// Already gone
}
}
delete process.env.CLAUDE_CODE_MESSAGING_SOCKET delete process.env.CLAUDE_CODE_MESSAGING_SOCKET
logForDebugging( logForDebugging(
`[udsMessaging] server stopped, socket removed: ${socketPath}`, `[udsMessaging] server stopped, socket removed: ${socketPath}`,

View File

@@ -1,4 +1,5 @@
import type { Socket } from 'net' import type { Socket } from 'net'
import { StringDecoder } from 'node:string_decoder'
import { errorMessage } from './errors.js' import { errorMessage } from './errors.js'
import { jsonParse } from './slowOperations.js' import { jsonParse } from './slowOperations.js'
import type { UdsMessage } from './udsMessaging.js' import type { UdsMessage } from './udsMessaging.js'
@@ -16,11 +17,11 @@ export function getChunkBytes(chunk: string | Buffer): number {
: chunk.byteLength : chunk.byteLength
} }
function parseResponseLine(line: string): UdsMessage | null { function parseResponseLine(line: string): UdsMessage {
try { try {
return jsonParse(line) as UdsMessage return jsonParse(line) as UdsMessage
} catch { } catch {
return null throw new Error('Invalid UDS response frame')
} }
} }
@@ -29,35 +30,58 @@ export function attachUdsResponseReader(
options: UdsResponseReaderOptions, options: UdsResponseReaderOptions,
): void { ): void {
let buffer = '' let buffer = ''
let bufferBytes = 0
let settled = false let settled = false
const decoder = new StringDecoder('utf8')
const finish = (error?: Error): void => { function cleanupListeners(): void {
socket.off('data', onData)
socket.off('error', onError)
socket.off('end', onEnd)
socket.off('close', onClose)
}
function finish(error?: Error): void {
if (settled) return if (settled) return
settled = true settled = true
buffer = ''
bufferBytes = 0
cleanupListeners()
if (error) { if (error) {
socket.destroy(error) socket.destroy()
} else { } else {
socket.end() socket.end()
} }
options.onSettled(error) options.onSettled(error)
} }
socket.on('data', chunk => { function onData(chunk: Buffer): void {
if ( const decoded = decoder.write(chunk)
Buffer.byteLength(buffer, 'utf8') + getChunkBytes(chunk) > const decodedBytes = Buffer.byteLength(decoded, 'utf8')
options.maxFrameBytes if (bufferBytes + decodedBytes > options.maxFrameBytes) {
) {
finish(new Error('UDS response frame exceeded size limit')) finish(new Error('UDS response frame exceeded size limit'))
return return
} }
buffer += chunk.toString() buffer += decoded
const lines = buffer.split('\n') bufferBytes += decodedBytes
buffer = lines.pop() ?? '' let newlineIndex = buffer.indexOf('\n')
for (const line of lines) { while (newlineIndex !== -1) {
if (!line.trim()) continue const line = buffer.slice(0, newlineIndex)
const response = parseResponseLine(line) const consumed = buffer.slice(0, newlineIndex + 1)
if (!response) continue buffer = buffer.slice(newlineIndex + 1)
bufferBytes -= Buffer.byteLength(consumed, 'utf8')
if (!line.trim()) {
newlineIndex = buffer.indexOf('\n')
continue
}
let response: UdsMessage
try {
response = parseResponseLine(line)
} catch (error) {
finish(error instanceof Error ? error : new Error(errorMessage(error)))
return
}
if ( if (
response.type === 'response' || response.type === 'response' ||
(options.acceptPong === true && response.type === 'pong') (options.acceptPong === true && response.type === 'pong')
@@ -69,13 +93,28 @@ export function attachUdsResponseReader(
finish(new Error(response.data ?? 'UDS receiver rejected message')) finish(new Error(response.data ?? 'UDS receiver rejected message'))
return return
} }
newlineIndex = buffer.indexOf('\n')
} }
}) }
socket.on('error', error => { function onError(error: Error): void {
finish( finish(
options.formatSocketError?.(error) ?? options.formatSocketError?.(error) ??
(error instanceof Error ? error : new Error(errorMessage(error))), (error instanceof Error ? error : new Error(errorMessage(error))),
) )
}) }
function onEnd(): void {
finish(new Error('UDS socket ended before response'))
}
function onClose(hadError: boolean): void {
if (hadError) return
finish(new Error('UDS socket closed before response'))
}
socket.on('data', onData)
socket.on('error', onError)
socket.on('end', onEnd)
socket.on('close', onClose)
} }