@tarquinen/opencode-dcp 3.2.3-beta0 → 3.2.5-beta0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,190 +0,0 @@
1
- import type { SessionState, ToolParameterEntry, WithParts } from "./types"
2
- import type { Logger } from "../logger"
3
- import { applyPendingCompressionDurations } from "../compress/timing"
4
- import { loadSessionState, saveSessionState } from "./persistence"
5
- import {
6
- isSubAgentSession,
7
- findLastCompactionTimestamp,
8
- countTurns,
9
- resetOnCompaction,
10
- createPruneMessagesState,
11
- loadPruneMessagesState,
12
- loadPruneMap,
13
- collectTurnNudgeAnchors,
14
- } from "./utils"
15
- import { getLastUserMessage } from "../messages/query"
16
-
17
- export const checkSession = async (
18
- client: any,
19
- state: SessionState,
20
- logger: Logger,
21
- messages: WithParts[],
22
- manualModeDefault: boolean,
23
- ): Promise<void> => {
24
- const lastUserMessage = getLastUserMessage(messages)
25
- if (!lastUserMessage) {
26
- return
27
- }
28
-
29
- const lastSessionId = lastUserMessage.info.sessionID
30
-
31
- if (state.sessionId === null || state.sessionId !== lastSessionId) {
32
- logger.info(`Session changed: ${state.sessionId} -> ${lastSessionId}`)
33
- try {
34
- await ensureSessionInitialized(
35
- client,
36
- state,
37
- lastSessionId,
38
- logger,
39
- messages,
40
- manualModeDefault,
41
- )
42
- } catch (err: any) {
43
- logger.error("Failed to initialize session state", { error: err.message })
44
- }
45
- }
46
-
47
- const lastCompactionTimestamp = findLastCompactionTimestamp(messages)
48
- if (lastCompactionTimestamp > state.lastCompaction) {
49
- state.lastCompaction = lastCompactionTimestamp
50
- resetOnCompaction(state)
51
- logger.info("Detected compaction - reset stale state", {
52
- timestamp: lastCompactionTimestamp,
53
- })
54
-
55
- saveSessionState(state, logger).catch((error) => {
56
- logger.warn("Failed to persist state reset after compaction", {
57
- error: error instanceof Error ? error.message : String(error),
58
- })
59
- })
60
- }
61
-
62
- state.currentTurn = countTurns(state, messages)
63
- }
64
-
65
- export function createSessionState(): SessionState {
66
- return {
67
- sessionId: null,
68
- isSubAgent: false,
69
- manualMode: false,
70
- compressPermission: undefined,
71
- pendingManualTrigger: null,
72
- prune: {
73
- tools: new Map<string, number>(),
74
- messages: createPruneMessagesState(),
75
- },
76
- nudges: {
77
- contextLimitAnchors: new Set<string>(),
78
- turnNudgeAnchors: new Set<string>(),
79
- iterationNudgeAnchors: new Set<string>(),
80
- },
81
- stats: {
82
- pruneTokenCounter: 0,
83
- totalPruneTokens: 0,
84
- },
85
- compressionTiming: {
86
- startsByCallId: new Map<string, number>(),
87
- pendingByCallId: new Map(),
88
- },
89
- toolParameters: new Map<string, ToolParameterEntry>(),
90
- subAgentResultCache: new Map<string, string>(),
91
- toolIdList: [],
92
- messageIds: {
93
- byRawId: new Map<string, string>(),
94
- byRef: new Map<string, string>(),
95
- nextRef: 1,
96
- },
97
- lastCompaction: 0,
98
- currentTurn: 0,
99
- variant: undefined,
100
- modelContextLimit: undefined,
101
- systemPromptTokens: undefined,
102
- }
103
- }
104
-
105
- export function resetSessionState(state: SessionState): void {
106
- state.sessionId = null
107
- state.isSubAgent = false
108
- state.manualMode = false
109
- state.compressPermission = undefined
110
- state.pendingManualTrigger = null
111
- state.prune = {
112
- tools: new Map<string, number>(),
113
- messages: createPruneMessagesState(),
114
- }
115
- state.nudges = {
116
- contextLimitAnchors: new Set<string>(),
117
- turnNudgeAnchors: new Set<string>(),
118
- iterationNudgeAnchors: new Set<string>(),
119
- }
120
- state.stats = {
121
- pruneTokenCounter: 0,
122
- totalPruneTokens: 0,
123
- }
124
- state.toolParameters.clear()
125
- state.subAgentResultCache.clear()
126
- state.toolIdList = []
127
- state.messageIds = {
128
- byRawId: new Map<string, string>(),
129
- byRef: new Map<string, string>(),
130
- nextRef: 1,
131
- }
132
- state.lastCompaction = 0
133
- state.currentTurn = 0
134
- state.variant = undefined
135
- state.modelContextLimit = undefined
136
- state.systemPromptTokens = undefined
137
- }
138
-
139
- export async function ensureSessionInitialized(
140
- client: any,
141
- state: SessionState,
142
- sessionId: string,
143
- logger: Logger,
144
- messages: WithParts[],
145
- manualModeEnabled: boolean,
146
- ): Promise<void> {
147
- if (state.sessionId === sessionId) {
148
- return
149
- }
150
-
151
- // logger.info("session ID = " + sessionId)
152
- // logger.info("Initializing session state", { sessionId: sessionId })
153
-
154
- resetSessionState(state)
155
- state.manualMode = manualModeEnabled ? "active" : false
156
- state.sessionId = sessionId
157
-
158
- const isSubAgent = await isSubAgentSession(client, sessionId)
159
- state.isSubAgent = isSubAgent
160
- // logger.info("isSubAgent = " + isSubAgent)
161
-
162
- state.lastCompaction = findLastCompactionTimestamp(messages)
163
- state.currentTurn = countTurns(state, messages)
164
- state.nudges.turnNudgeAnchors = collectTurnNudgeAnchors(messages)
165
-
166
- const persisted = await loadSessionState(sessionId, logger)
167
- if (persisted === null) {
168
- return
169
- }
170
-
171
- state.prune.tools = loadPruneMap(persisted.prune.tools)
172
- state.prune.messages = loadPruneMessagesState(persisted.prune.messages)
173
- state.nudges.contextLimitAnchors = new Set<string>(persisted.nudges.contextLimitAnchors || [])
174
- state.nudges.turnNudgeAnchors = new Set<string>([
175
- ...state.nudges.turnNudgeAnchors,
176
- ...(persisted.nudges.turnNudgeAnchors || []),
177
- ])
178
- state.nudges.iterationNudgeAnchors = new Set<string>(
179
- persisted.nudges.iterationNudgeAnchors || [],
180
- )
181
- state.stats = {
182
- pruneTokenCounter: persisted.stats?.pruneTokenCounter || 0,
183
- totalPruneTokens: persisted.stats?.totalPruneTokens || 0,
184
- }
185
-
186
- const applied = applyPendingCompressionDurations(state)
187
- if (applied > 0) {
188
- await saveSessionState(state, logger)
189
- }
190
- }
@@ -1,98 +0,0 @@
1
- import type { SessionState, ToolStatus, WithParts } from "./index"
2
- import type { Logger } from "../logger"
3
- import { PluginConfig } from "../config"
4
- import { isMessageCompacted } from "./utils"
5
- import { countToolTokens } from "../token-utils"
6
-
7
- const MAX_TOOL_CACHE_SIZE = 1000
8
-
9
- /**
10
- * Sync tool parameters from session messages.
11
- */
12
- export function syncToolCache(
13
- state: SessionState,
14
- config: PluginConfig,
15
- logger: Logger,
16
- messages: WithParts[],
17
- ): void {
18
- try {
19
- logger.info("Syncing tool parameters from OpenCode messages")
20
-
21
- let turnCounter = 0
22
-
23
- for (const msg of messages) {
24
- if (isMessageCompacted(state, msg)) {
25
- continue
26
- }
27
-
28
- const parts = Array.isArray(msg.parts) ? msg.parts : []
29
- for (const part of parts) {
30
- if (part.type === "step-start") {
31
- turnCounter++
32
- continue
33
- }
34
-
35
- if (part.type !== "tool" || !part.callID) {
36
- continue
37
- }
38
-
39
- const turnProtectionEnabled = config.turnProtection.enabled
40
- const turnProtectionTurns = config.turnProtection.turns
41
- const isProtectedByTurn =
42
- turnProtectionEnabled &&
43
- turnProtectionTurns > 0 &&
44
- state.currentTurn - turnCounter < turnProtectionTurns
45
-
46
- if (state.toolParameters.has(part.callID)) {
47
- continue
48
- }
49
-
50
- if (isProtectedByTurn) {
51
- continue
52
- }
53
-
54
- const tokenCount = countToolTokens(part)
55
-
56
- state.toolParameters.set(part.callID, {
57
- tool: part.tool,
58
- parameters: part.state?.input ?? {},
59
- status: part.state.status as ToolStatus | undefined,
60
- error: part.state.status === "error" ? part.state.error : undefined,
61
- turn: turnCounter,
62
- tokenCount,
63
- })
64
- logger.info(
65
- `Cached tool id: ${part.callID} (turn ${turnCounter}${tokenCount !== undefined ? `, ${tokenCount} tokens` : ""})`,
66
- )
67
- }
68
- }
69
-
70
- logger.info(
71
- `Synced cache - size: ${state.toolParameters.size}, currentTurn: ${state.currentTurn}`,
72
- )
73
- trimToolParametersCache(state)
74
- } catch (error) {
75
- logger.warn("Failed to sync tool parameters from OpenCode", {
76
- error: error instanceof Error ? error.message : String(error),
77
- })
78
- }
79
- }
80
-
81
- /**
82
- * Trim the tool parameters cache to prevent unbounded memory growth.
83
- * Uses FIFO eviction - removes oldest entries first.
84
- */
85
- export function trimToolParametersCache(state: SessionState): void {
86
- if (state.toolParameters.size <= MAX_TOOL_CACHE_SIZE) {
87
- return
88
- }
89
-
90
- const keysToRemove = Array.from(state.toolParameters.keys()).slice(
91
- 0,
92
- state.toolParameters.size - MAX_TOOL_CACHE_SIZE,
93
- )
94
-
95
- for (const key of keysToRemove) {
96
- state.toolParameters.delete(key)
97
- }
98
- }
@@ -1,112 +0,0 @@
1
- import type { CompressionTimingState } from "../compress/timing"
2
- import { Message, Part } from "@opencode-ai/sdk/v2"
3
-
4
- export interface WithParts {
5
- info: Message
6
- parts: Part[]
7
- }
8
-
9
- export type ToolStatus = "pending" | "running" | "completed" | "error"
10
-
11
- export interface ToolParameterEntry {
12
- tool: string
13
- parameters: any
14
- status?: ToolStatus
15
- error?: string
16
- turn: number
17
- tokenCount?: number
18
- }
19
-
20
- export interface SessionStats {
21
- pruneTokenCounter: number
22
- totalPruneTokens: number
23
- }
24
-
25
- export interface PrunedMessageEntry {
26
- tokenCount: number
27
- allBlockIds: number[]
28
- activeBlockIds: number[]
29
- }
30
-
31
- export type CompressionMode = "range" | "message"
32
-
33
- export interface CompressionBlock {
34
- blockId: number
35
- runId: number
36
- active: boolean
37
- deactivatedByUser: boolean
38
- compressedTokens: number
39
- summaryTokens: number
40
- durationMs: number
41
- mode?: CompressionMode
42
- topic: string
43
- batchTopic?: string
44
- startId: string
45
- endId: string
46
- anchorMessageId: string
47
- compressMessageId: string
48
- compressCallId?: string
49
- includedBlockIds: number[]
50
- consumedBlockIds: number[]
51
- parentBlockIds: number[]
52
- directMessageIds: string[]
53
- directToolIds: string[]
54
- effectiveMessageIds: string[]
55
- effectiveToolIds: string[]
56
- createdAt: number
57
- deactivatedAt?: number
58
- deactivatedByBlockId?: number
59
- summary: string
60
- }
61
-
62
- export interface PruneMessagesState {
63
- byMessageId: Map<string, PrunedMessageEntry>
64
- blocksById: Map<number, CompressionBlock>
65
- activeBlockIds: Set<number>
66
- activeByAnchorMessageId: Map<string, number>
67
- nextBlockId: number
68
- nextRunId: number
69
- }
70
-
71
- export interface Prune {
72
- tools: Map<string, number>
73
- messages: PruneMessagesState
74
- }
75
-
76
- export interface PendingManualTrigger {
77
- sessionId: string
78
- prompt: string
79
- }
80
-
81
- export interface MessageIdState {
82
- byRawId: Map<string, string>
83
- byRef: Map<string, string>
84
- nextRef: number
85
- }
86
-
87
- export interface Nudges {
88
- contextLimitAnchors: Set<string>
89
- turnNudgeAnchors: Set<string>
90
- iterationNudgeAnchors: Set<string>
91
- }
92
-
93
- export interface SessionState {
94
- sessionId: string | null
95
- isSubAgent: boolean
96
- manualMode: false | "active" | "compress-pending"
97
- compressPermission: "ask" | "allow" | "deny" | undefined
98
- pendingManualTrigger: PendingManualTrigger | null
99
- prune: Prune
100
- nudges: Nudges
101
- stats: SessionStats
102
- compressionTiming: CompressionTimingState
103
- toolParameters: Map<string, ToolParameterEntry>
104
- subAgentResultCache: Map<string, string>
105
- toolIdList: string[]
106
- messageIds: MessageIdState
107
- lastCompaction: number
108
- currentTurn: number
109
- variant: string | undefined
110
- modelContextLimit: number | undefined
111
- systemPromptTokens: number | undefined
112
- }