@tarquinen/opencode-dcp 3.2.4-beta0 → 3.2.6-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.
Files changed (46) hide show
  1. package/dcp.schema.json +329 -0
  2. package/dist/lib/config.js +2 -2
  3. package/dist/lib/config.js.map +1 -1
  4. package/index.ts +141 -0
  5. package/lib/auth.ts +37 -0
  6. package/lib/commands/compression-targets.ts +137 -0
  7. package/lib/commands/context.ts +132 -0
  8. package/lib/commands/decompress.ts +275 -0
  9. package/lib/commands/help.ts +76 -0
  10. package/lib/commands/index.ts +11 -0
  11. package/lib/commands/manual.ts +125 -0
  12. package/lib/commands/recompress.ts +224 -0
  13. package/lib/commands/stats.ts +148 -0
  14. package/lib/commands/sweep.ts +268 -0
  15. package/lib/compress-permission.ts +25 -0
  16. package/lib/config.ts +2 -2
  17. package/lib/hooks.ts +378 -0
  18. package/lib/host-permissions.ts +101 -0
  19. package/lib/messages/index.ts +8 -0
  20. package/lib/messages/inject/inject.ts +215 -0
  21. package/lib/messages/inject/subagent-results.ts +82 -0
  22. package/lib/messages/inject/utils.ts +374 -0
  23. package/lib/messages/priority.ts +102 -0
  24. package/lib/messages/prune.ts +238 -0
  25. package/lib/messages/reasoning-strip.ts +40 -0
  26. package/lib/messages/sync.ts +124 -0
  27. package/lib/messages/utils.ts +187 -0
  28. package/lib/prompts/compress-message.ts +42 -0
  29. package/lib/prompts/compress-range.ts +60 -0
  30. package/lib/prompts/context-limit-nudge.ts +18 -0
  31. package/lib/prompts/extensions/nudge.ts +43 -0
  32. package/lib/prompts/extensions/system.ts +32 -0
  33. package/lib/prompts/extensions/tool.ts +35 -0
  34. package/lib/prompts/index.ts +29 -0
  35. package/lib/prompts/iteration-nudge.ts +6 -0
  36. package/lib/prompts/store.ts +467 -0
  37. package/lib/prompts/system.ts +33 -0
  38. package/lib/prompts/turn-nudge.ts +10 -0
  39. package/lib/protected-patterns.ts +128 -0
  40. package/lib/strategies/deduplication.ts +127 -0
  41. package/lib/strategies/index.ts +2 -0
  42. package/lib/strategies/purge-errors.ts +88 -0
  43. package/lib/subagents/subagent-results.ts +74 -0
  44. package/lib/ui/notification.ts +346 -0
  45. package/lib/ui/utils.ts +287 -0
  46. package/package.json +14 -19
@@ -0,0 +1,125 @@
1
+ /**
2
+ * DCP Manual mode command handler.
3
+ * Handles toggling manual mode and triggering individual tool executions.
4
+ *
5
+ * Usage:
6
+ * /dcp manual [on|off] - Toggle manual mode or set explicit state
7
+ * /dcp compress [focus] - Trigger manual compress execution
8
+ */
9
+
10
+ import type { Logger } from "../logger"
11
+ import type { SessionState, WithParts } from "../state"
12
+ import type { PluginConfig } from "../config"
13
+ import { sendIgnoredMessage } from "../ui/notification"
14
+ import { getCurrentParams } from "../token-utils"
15
+ import { buildCompressedBlockGuidance } from "../prompts/extensions/nudge"
16
+ import { isIgnoredUserMessage } from "../messages/query"
17
+
18
+ const MANUAL_MODE_ON = "Manual mode is now ON. Use /dcp compress to trigger context tools manually."
19
+
20
+ const MANUAL_MODE_OFF = "Manual mode is now OFF."
21
+
22
+ const COMPRESS_TRIGGER_PROMPT = [
23
+ "<compress triggered manually>",
24
+ "Manual mode trigger received. You must now use the compress tool.",
25
+ "Find the most significant completed conversation content that can be compressed into a high-fidelity technical summary.",
26
+ "Follow the active compress mode, preserve all critical implementation details, and choose safe targets.",
27
+ "Return after compress with a brief explanation of what content was compressed.",
28
+ ].join("\n\n")
29
+
30
+ function getTriggerPrompt(
31
+ tool: "compress",
32
+ state: SessionState,
33
+ config: PluginConfig,
34
+ userFocus?: string,
35
+ ): string {
36
+ const base = COMPRESS_TRIGGER_PROMPT
37
+ const compressedBlockGuidance =
38
+ config.compress.mode === "message" ? "" : buildCompressedBlockGuidance(state)
39
+
40
+ const sections = [base, compressedBlockGuidance]
41
+ if (userFocus && userFocus.trim().length > 0) {
42
+ sections.push(`Additional user focus:\n${userFocus.trim()}`)
43
+ }
44
+
45
+ return sections.join("\n\n")
46
+ }
47
+
48
+ export interface ManualCommandContext {
49
+ client: any
50
+ state: SessionState
51
+ config: PluginConfig
52
+ logger: Logger
53
+ sessionId: string
54
+ messages: WithParts[]
55
+ }
56
+
57
+ export async function handleManualToggleCommand(
58
+ ctx: ManualCommandContext,
59
+ modeArg?: string,
60
+ ): Promise<void> {
61
+ const { client, state, logger, sessionId, messages } = ctx
62
+
63
+ if (modeArg === "on") {
64
+ state.manualMode = "active"
65
+ } else if (modeArg === "off") {
66
+ state.manualMode = false
67
+ } else {
68
+ state.manualMode = state.manualMode ? false : "active"
69
+ }
70
+
71
+ const params = getCurrentParams(state, messages, logger)
72
+ await sendIgnoredMessage(
73
+ client,
74
+ sessionId,
75
+ state.manualMode ? MANUAL_MODE_ON : MANUAL_MODE_OFF,
76
+ params,
77
+ logger,
78
+ )
79
+
80
+ logger.info("Manual mode toggled", { manualMode: state.manualMode })
81
+ }
82
+
83
+ export async function handleManualTriggerCommand(
84
+ ctx: ManualCommandContext,
85
+ tool: "compress",
86
+ userFocus?: string,
87
+ ): Promise<string | null> {
88
+ return getTriggerPrompt(tool, ctx.state, ctx.config, userFocus)
89
+ }
90
+
91
+ export function applyPendingManualTrigger(
92
+ state: SessionState,
93
+ messages: WithParts[],
94
+ logger: Logger,
95
+ ): void {
96
+ const pending = state.pendingManualTrigger
97
+ if (!pending) {
98
+ return
99
+ }
100
+
101
+ if (!state.sessionId || pending.sessionId !== state.sessionId) {
102
+ state.pendingManualTrigger = null
103
+ return
104
+ }
105
+
106
+ for (let i = messages.length - 1; i >= 0; i--) {
107
+ const msg = messages[i]
108
+ if (msg.info.role !== "user" || isIgnoredUserMessage(msg)) {
109
+ continue
110
+ }
111
+
112
+ for (const part of msg.parts) {
113
+ if (part.type !== "text" || part.ignored || part.synthetic) {
114
+ continue
115
+ }
116
+
117
+ part.text = pending.prompt
118
+ state.pendingManualTrigger = null
119
+ logger.debug("Applied manual prompt", { sessionId: pending.sessionId })
120
+ return
121
+ }
122
+ }
123
+
124
+ state.pendingManualTrigger = null
125
+ }
@@ -0,0 +1,224 @@
1
+ import type { Logger } from "../logger"
2
+ import type { PruneMessagesState, SessionState, WithParts } from "../state"
3
+ import { syncCompressionBlocks } from "../messages"
4
+ import { parseBlockRef } from "../message-ids"
5
+ import { getCurrentParams } from "../token-utils"
6
+ import { saveSessionState } from "../state/persistence"
7
+ import { sendIgnoredMessage } from "../ui/notification"
8
+ import { formatTokenCount } from "../ui/utils"
9
+ import {
10
+ getRecompressibleCompressionTargets,
11
+ resolveCompressionTarget,
12
+ type CompressionTarget,
13
+ } from "./compression-targets"
14
+
15
+ export interface RecompressCommandContext {
16
+ client: any
17
+ state: SessionState
18
+ logger: Logger
19
+ sessionId: string
20
+ messages: WithParts[]
21
+ args: string[]
22
+ }
23
+
24
+ function parseBlockIdArg(arg: string): number | null {
25
+ const normalized = arg.trim().toLowerCase()
26
+ const blockRef = parseBlockRef(normalized)
27
+ if (blockRef !== null) {
28
+ return blockRef
29
+ }
30
+
31
+ if (!/^[1-9]\d*$/.test(normalized)) {
32
+ return null
33
+ }
34
+
35
+ const parsed = Number.parseInt(normalized, 10)
36
+ return Number.isInteger(parsed) && parsed > 0 ? parsed : null
37
+ }
38
+
39
+ function snapshotActiveMessages(messagesState: PruneMessagesState): Set<string> {
40
+ const activeMessages = new Set<string>()
41
+ for (const [messageId, entry] of messagesState.byMessageId) {
42
+ if (entry.activeBlockIds.length > 0) {
43
+ activeMessages.add(messageId)
44
+ }
45
+ }
46
+ return activeMessages
47
+ }
48
+
49
+ function formatRecompressMessage(
50
+ target: CompressionTarget,
51
+ recompressedMessageCount: number,
52
+ recompressedTokens: number,
53
+ deactivatedBlockIds: number[],
54
+ ): string {
55
+ const lines: string[] = []
56
+
57
+ lines.push(`Re-applied compression ${target.displayId}.`)
58
+ if (target.runId !== target.displayId || target.grouped) {
59
+ lines.push(`Tool call label: Compression #${target.runId}.`)
60
+ }
61
+ if (deactivatedBlockIds.length > 0) {
62
+ const refs = deactivatedBlockIds.map((id) => String(id)).join(", ")
63
+ lines.push(`Also re-compressed nested compression(s): ${refs}.`)
64
+ }
65
+
66
+ if (recompressedMessageCount > 0) {
67
+ lines.push(
68
+ `Re-compressed ${recompressedMessageCount} message(s) (~${formatTokenCount(recompressedTokens)}).`,
69
+ )
70
+ } else {
71
+ lines.push("No messages were re-compressed.")
72
+ }
73
+
74
+ return lines.join("\n")
75
+ }
76
+
77
+ function formatAvailableBlocksMessage(availableTargets: CompressionTarget[]): string {
78
+ const lines: string[] = []
79
+
80
+ lines.push("Usage: /dcp recompress <n>")
81
+ lines.push("")
82
+
83
+ if (availableTargets.length === 0) {
84
+ lines.push("No user-decompressed blocks are available to re-compress.")
85
+ return lines.join("\n")
86
+ }
87
+
88
+ lines.push("Available user-decompressed compressions:")
89
+ const entries = availableTargets.map((target) => {
90
+ const topic = target.topic.replace(/\s+/g, " ").trim() || "(no topic)"
91
+ const label = `${target.displayId} (${formatTokenCount(target.compressedTokens)})`
92
+ const details = target.grouped
93
+ ? `Compression #${target.runId} - ${target.blocks.length} messages`
94
+ : `Compression #${target.runId}`
95
+ return { label, topic: `${details} - ${topic}` }
96
+ })
97
+
98
+ const labelWidth = Math.max(...entries.map((entry) => entry.label.length)) + 4
99
+ for (const entry of entries) {
100
+ lines.push(` ${entry.label.padEnd(labelWidth)}${entry.topic}`)
101
+ }
102
+
103
+ return lines.join("\n")
104
+ }
105
+
106
+ export async function handleRecompressCommand(ctx: RecompressCommandContext): Promise<void> {
107
+ const { client, state, logger, sessionId, messages, args } = ctx
108
+
109
+ const params = getCurrentParams(state, messages, logger)
110
+ const targetArg = args[0]
111
+
112
+ if (args.length > 1) {
113
+ await sendIgnoredMessage(
114
+ client,
115
+ sessionId,
116
+ "Invalid arguments. Usage: /dcp recompress <n>",
117
+ params,
118
+ logger,
119
+ )
120
+ return
121
+ }
122
+
123
+ syncCompressionBlocks(state, logger, messages)
124
+ const messagesState = state.prune.messages
125
+ const availableMessageIds = new Set(messages.map((msg) => msg.info.id))
126
+
127
+ if (!targetArg) {
128
+ const availableTargets = getRecompressibleCompressionTargets(
129
+ messagesState,
130
+ availableMessageIds,
131
+ )
132
+ const message = formatAvailableBlocksMessage(availableTargets)
133
+ await sendIgnoredMessage(client, sessionId, message, params, logger)
134
+ return
135
+ }
136
+
137
+ const targetBlockId = parseBlockIdArg(targetArg)
138
+ if (targetBlockId === null) {
139
+ await sendIgnoredMessage(
140
+ client,
141
+ sessionId,
142
+ `Please enter a compression number. Example: /dcp recompress 2`,
143
+ params,
144
+ logger,
145
+ )
146
+ return
147
+ }
148
+
149
+ const target = resolveCompressionTarget(messagesState, targetBlockId)
150
+ if (!target) {
151
+ await sendIgnoredMessage(
152
+ client,
153
+ sessionId,
154
+ `Compression ${targetBlockId} does not exist.`,
155
+ params,
156
+ logger,
157
+ )
158
+ return
159
+ }
160
+
161
+ if (target.blocks.some((block) => !availableMessageIds.has(block.compressMessageId))) {
162
+ await sendIgnoredMessage(
163
+ client,
164
+ sessionId,
165
+ `Compression ${target.displayId} can no longer be re-applied because its origin message is no longer in this session.`,
166
+ params,
167
+ logger,
168
+ )
169
+ return
170
+ }
171
+
172
+ if (!target.blocks.some((block) => block.deactivatedByUser)) {
173
+ const message = target.blocks.some((block) => block.active)
174
+ ? `Compression ${target.displayId} is already active.`
175
+ : `Compression ${target.displayId} is not user-decompressed.`
176
+ await sendIgnoredMessage(client, sessionId, message, params, logger)
177
+ return
178
+ }
179
+
180
+ const activeMessagesBefore = snapshotActiveMessages(messagesState)
181
+ const activeBlockIdsBefore = new Set(messagesState.activeBlockIds)
182
+
183
+ for (const block of target.blocks) {
184
+ block.deactivatedByUser = false
185
+ block.deactivatedAt = undefined
186
+ block.deactivatedByBlockId = undefined
187
+ }
188
+
189
+ syncCompressionBlocks(state, logger, messages)
190
+
191
+ let recompressedMessageCount = 0
192
+ let recompressedTokens = 0
193
+ for (const [messageId, entry] of messagesState.byMessageId) {
194
+ const isActiveNow = entry.activeBlockIds.length > 0
195
+ if (isActiveNow && !activeMessagesBefore.has(messageId)) {
196
+ recompressedMessageCount++
197
+ recompressedTokens += entry.tokenCount
198
+ }
199
+ }
200
+
201
+ state.stats.totalPruneTokens += recompressedTokens
202
+
203
+ const deactivatedBlockIds = Array.from(activeBlockIdsBefore)
204
+ .filter((blockId) => !messagesState.activeBlockIds.has(blockId))
205
+ .sort((a, b) => a - b)
206
+
207
+ await saveSessionState(state, logger)
208
+
209
+ const message = formatRecompressMessage(
210
+ target,
211
+ recompressedMessageCount,
212
+ recompressedTokens,
213
+ deactivatedBlockIds,
214
+ )
215
+ await sendIgnoredMessage(client, sessionId, message, params, logger)
216
+
217
+ logger.info("Recompress command completed", {
218
+ targetBlockId: target.displayId,
219
+ targetRunId: target.runId,
220
+ recompressedMessageCount,
221
+ recompressedTokens,
222
+ deactivatedBlockIds,
223
+ })
224
+ }
@@ -0,0 +1,148 @@
1
+ /**
2
+ * DCP Stats command handler.
3
+ * Shows pruning statistics for the current session and all-time totals.
4
+ */
5
+
6
+ import type { Logger } from "../logger"
7
+ import type { SessionState, WithParts } from "../state"
8
+ import { sendIgnoredMessage } from "../ui/notification"
9
+ import { formatTokenCount } from "../ui/utils"
10
+ import { loadAllSessionStats, type AggregatedStats } from "../state/persistence"
11
+ import { getCurrentParams } from "../token-utils"
12
+ import { getActiveCompressionTargets } from "./compression-targets"
13
+
14
+ export interface StatsCommandContext {
15
+ client: any
16
+ state: SessionState
17
+ logger: Logger
18
+ sessionId: string
19
+ messages: WithParts[]
20
+ }
21
+
22
+ function formatStatsMessage(
23
+ sessionTokens: number,
24
+ sessionSummaryTokens: number,
25
+ sessionTools: number,
26
+ sessionMessages: number,
27
+ sessionDurationMs: number,
28
+ allTime: AggregatedStats,
29
+ ): string {
30
+ const lines: string[] = []
31
+
32
+ lines.push("╭───────────────────────────────────────────────────────────╮")
33
+ lines.push("│ DCP Statistics │")
34
+ lines.push("╰───────────────────────────────────────────────────────────╯")
35
+ lines.push("")
36
+ lines.push("Compression:")
37
+ lines.push("─".repeat(60))
38
+ lines.push(
39
+ ` Tokens in|out: ~${formatTokenCount(sessionTokens)} | ~${formatTokenCount(sessionSummaryTokens)}`,
40
+ )
41
+ lines.push(` Ratio: ${formatCompressionRatio(sessionTokens, sessionSummaryTokens)}`)
42
+ lines.push(` Time: ${formatCompressionTime(sessionDurationMs)}`)
43
+ lines.push(` Messages: ${sessionMessages}`)
44
+ lines.push(` Tools: ${sessionTools}`)
45
+ lines.push("")
46
+ lines.push("All-time:")
47
+ lines.push("─".repeat(60))
48
+ lines.push(` Tokens saved: ~${formatTokenCount(allTime.totalTokens)}`)
49
+ lines.push(` Tools pruned: ${allTime.totalTools}`)
50
+ lines.push(` Messages pruned: ${allTime.totalMessages}`)
51
+ lines.push(` Sessions: ${allTime.sessionCount}`)
52
+
53
+ return lines.join("\n")
54
+ }
55
+
56
+ function formatCompressionRatio(inputTokens: number, outputTokens: number): string {
57
+ if (inputTokens <= 0) {
58
+ return "0:1"
59
+ }
60
+
61
+ if (outputTokens <= 0) {
62
+ return "∞:1"
63
+ }
64
+
65
+ const ratio = Math.max(1, Math.round(inputTokens / outputTokens))
66
+ return `${ratio}:1`
67
+ }
68
+
69
+ function formatCompressionTime(ms: number): string {
70
+ const safeMs = Math.max(0, Math.round(ms))
71
+ if (safeMs < 1000) {
72
+ return `${safeMs} ms`
73
+ }
74
+
75
+ const totalSeconds = safeMs / 1000
76
+ if (totalSeconds < 60) {
77
+ return `${totalSeconds.toFixed(1)} s`
78
+ }
79
+
80
+ const wholeSeconds = Math.floor(totalSeconds)
81
+ const hours = Math.floor(wholeSeconds / 3600)
82
+ const minutes = Math.floor((wholeSeconds % 3600) / 60)
83
+ const seconds = wholeSeconds % 60
84
+
85
+ if (hours > 0) {
86
+ return `${hours}h ${minutes}m ${seconds}s`
87
+ }
88
+
89
+ return `${minutes}m ${seconds}s`
90
+ }
91
+
92
+ export async function handleStatsCommand(ctx: StatsCommandContext): Promise<void> {
93
+ const { client, state, logger, sessionId, messages } = ctx
94
+
95
+ // Session stats from in-memory state
96
+ const sessionTokens = state.stats.totalPruneTokens
97
+ const sessionSummaryTokens = Array.from(state.prune.messages.blocksById.values()).reduce(
98
+ (total, block) => (block.active ? total + block.summaryTokens : total),
99
+ 0,
100
+ )
101
+ const sessionDurationMs = getActiveCompressionTargets(state.prune.messages).reduce(
102
+ (total, target) => total + target.durationMs,
103
+ 0,
104
+ )
105
+
106
+ const prunedToolIds = new Set<string>(state.prune.tools.keys())
107
+ for (const block of state.prune.messages.blocksById.values()) {
108
+ if (block.active) {
109
+ for (const toolId of block.effectiveToolIds) {
110
+ prunedToolIds.add(toolId)
111
+ }
112
+ }
113
+ }
114
+ const sessionTools = prunedToolIds.size
115
+
116
+ let sessionMessages = 0
117
+ for (const entry of state.prune.messages.byMessageId.values()) {
118
+ if (entry.activeBlockIds.length > 0) {
119
+ sessionMessages++
120
+ }
121
+ }
122
+
123
+ // All-time stats from storage files
124
+ const allTime = await loadAllSessionStats(logger)
125
+
126
+ const message = formatStatsMessage(
127
+ sessionTokens,
128
+ sessionSummaryTokens,
129
+ sessionTools,
130
+ sessionMessages,
131
+ sessionDurationMs,
132
+ allTime,
133
+ )
134
+
135
+ const params = getCurrentParams(state, messages, logger)
136
+ await sendIgnoredMessage(client, sessionId, message, params, logger)
137
+
138
+ logger.info("Stats command executed", {
139
+ sessionTokens,
140
+ sessionSummaryTokens,
141
+ sessionTools,
142
+ sessionMessages,
143
+ sessionDurationMs,
144
+ allTimeTokens: allTime.totalTokens,
145
+ allTimeTools: allTime.totalTools,
146
+ allTimeMessages: allTime.totalMessages,
147
+ })
148
+ }