@brianli/kimaki 0.4.72-brianli.1
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.
- package/bin.js +2 -0
- package/dist/ai-tool-to-genai.js +233 -0
- package/dist/ai-tool-to-genai.test.js +267 -0
- package/dist/ai-tool.js +6 -0
- package/dist/bin.js +87 -0
- package/dist/bot-token.js +121 -0
- package/dist/bot-token.test.js +134 -0
- package/dist/channel-management.js +101 -0
- package/dist/cli-parsing.test.js +89 -0
- package/dist/cli.js +2529 -0
- package/dist/commands/abort.js +82 -0
- package/dist/commands/action-buttons.js +257 -0
- package/dist/commands/add-project.js +114 -0
- package/dist/commands/agent.js +291 -0
- package/dist/commands/ask-question.js +223 -0
- package/dist/commands/compact.js +120 -0
- package/dist/commands/context-usage.js +140 -0
- package/dist/commands/create-new-project.js +118 -0
- package/dist/commands/diff.js +128 -0
- package/dist/commands/file-upload.js +275 -0
- package/dist/commands/fork.js +217 -0
- package/dist/commands/gemini-apikey.js +70 -0
- package/dist/commands/login.js +490 -0
- package/dist/commands/mention-mode.js +51 -0
- package/dist/commands/merge-worktree.js +124 -0
- package/dist/commands/model.js +694 -0
- package/dist/commands/permissions.js +163 -0
- package/dist/commands/queue.js +217 -0
- package/dist/commands/remove-project.js +115 -0
- package/dist/commands/restart-opencode-server.js +116 -0
- package/dist/commands/resume.js +159 -0
- package/dist/commands/run-command.js +79 -0
- package/dist/commands/session-id.js +78 -0
- package/dist/commands/session.js +192 -0
- package/dist/commands/share.js +80 -0
- package/dist/commands/types.js +2 -0
- package/dist/commands/undo-redo.js +159 -0
- package/dist/commands/unset-model.js +152 -0
- package/dist/commands/upgrade.js +42 -0
- package/dist/commands/user-command.js +148 -0
- package/dist/commands/verbosity.js +60 -0
- package/dist/commands/worktree-settings.js +50 -0
- package/dist/commands/worktree.js +299 -0
- package/dist/condense-memory.js +33 -0
- package/dist/config.js +110 -0
- package/dist/database.js +1050 -0
- package/dist/db.js +159 -0
- package/dist/db.test.js +49 -0
- package/dist/discord-api.js +28 -0
- package/dist/discord-auth.js +231 -0
- package/dist/discord-auth.test.js +80 -0
- package/dist/discord-bot.js +997 -0
- package/dist/discord-utils.js +560 -0
- package/dist/discord-utils.test.js +115 -0
- package/dist/errors.js +167 -0
- package/dist/escape-backticks.test.js +429 -0
- package/dist/format-tables.js +122 -0
- package/dist/format-tables.test.js +199 -0
- package/dist/forum-sync/config.js +79 -0
- package/dist/forum-sync/discord-operations.js +154 -0
- package/dist/forum-sync/index.js +5 -0
- package/dist/forum-sync/markdown.js +117 -0
- package/dist/forum-sync/sync-to-discord.js +417 -0
- package/dist/forum-sync/sync-to-files.js +190 -0
- package/dist/forum-sync/types.js +53 -0
- package/dist/forum-sync/watchers.js +307 -0
- package/dist/gateway-consumer.js +232 -0
- package/dist/gateway-consumer.test.js +18 -0
- package/dist/genai-worker-wrapper.js +111 -0
- package/dist/genai-worker.js +311 -0
- package/dist/genai.js +232 -0
- package/dist/generated/browser.js +17 -0
- package/dist/generated/client.js +35 -0
- package/dist/generated/commonInputTypes.js +10 -0
- package/dist/generated/enums.js +30 -0
- package/dist/generated/internal/class.js +41 -0
- package/dist/generated/internal/prismaNamespace.js +239 -0
- package/dist/generated/internal/prismaNamespaceBrowser.js +209 -0
- package/dist/generated/models/bot_api_keys.js +1 -0
- package/dist/generated/models/bot_tokens.js +1 -0
- package/dist/generated/models/channel_agents.js +1 -0
- package/dist/generated/models/channel_directories.js +1 -0
- package/dist/generated/models/channel_mention_mode.js +1 -0
- package/dist/generated/models/channel_models.js +1 -0
- package/dist/generated/models/channel_verbosity.js +1 -0
- package/dist/generated/models/channel_worktrees.js +1 -0
- package/dist/generated/models/forum_sync_configs.js +1 -0
- package/dist/generated/models/global_models.js +1 -0
- package/dist/generated/models/ipc_requests.js +1 -0
- package/dist/generated/models/part_messages.js +1 -0
- package/dist/generated/models/scheduled_tasks.js +1 -0
- package/dist/generated/models/session_agents.js +1 -0
- package/dist/generated/models/session_models.js +1 -0
- package/dist/generated/models/session_start_sources.js +1 -0
- package/dist/generated/models/thread_sessions.js +1 -0
- package/dist/generated/models/thread_worktrees.js +1 -0
- package/dist/generated/models.js +1 -0
- package/dist/heap-monitor.js +95 -0
- package/dist/hrana-server.js +416 -0
- package/dist/hrana-server.test.js +368 -0
- package/dist/image-utils.js +112 -0
- package/dist/interaction-handler.js +327 -0
- package/dist/ipc-polling.js +251 -0
- package/dist/kimaki-digital-twin.e2e.test.js +165 -0
- package/dist/limit-heading-depth.js +25 -0
- package/dist/limit-heading-depth.test.js +105 -0
- package/dist/logger.js +160 -0
- package/dist/markdown.js +342 -0
- package/dist/markdown.test.js +253 -0
- package/dist/message-formatting.js +433 -0
- package/dist/message-formatting.test.js +73 -0
- package/dist/openai-realtime.js +228 -0
- package/dist/opencode-plugin-loading.e2e.test.js +91 -0
- package/dist/opencode-plugin.js +536 -0
- package/dist/opencode-plugin.test.js +98 -0
- package/dist/opencode.js +409 -0
- package/dist/privacy-sanitizer.js +105 -0
- package/dist/runtime-mode.js +51 -0
- package/dist/runtime-mode.test.js +115 -0
- package/dist/sentry.js +127 -0
- package/dist/session-handler/state.js +151 -0
- package/dist/session-handler.js +1874 -0
- package/dist/session-search.js +100 -0
- package/dist/session-search.test.js +40 -0
- package/dist/startup-service.js +153 -0
- package/dist/system-message.js +499 -0
- package/dist/task-runner.js +282 -0
- package/dist/task-schedule.js +191 -0
- package/dist/task-schedule.test.js +71 -0
- package/dist/thinking-utils.js +35 -0
- package/dist/thread-message-queue.e2e.test.js +781 -0
- package/dist/tools.js +359 -0
- package/dist/unnest-code-blocks.js +136 -0
- package/dist/unnest-code-blocks.test.js +641 -0
- package/dist/upgrade.js +114 -0
- package/dist/utils.js +109 -0
- package/dist/voice-handler.js +606 -0
- package/dist/voice.js +304 -0
- package/dist/voice.test.js +187 -0
- package/dist/wait-session.js +94 -0
- package/dist/worker-types.js +4 -0
- package/dist/worktree-utils.js +727 -0
- package/dist/xml.js +92 -0
- package/dist/xml.test.js +32 -0
- package/package.json +82 -0
- package/schema.prisma +246 -0
- package/skills/batch/SKILL.md +87 -0
- package/skills/critique/SKILL.md +129 -0
- package/skills/errore/SKILL.md +589 -0
- package/skills/goke/.prettierrc +5 -0
- package/skills/goke/CHANGELOG.md +40 -0
- package/skills/goke/LICENSE +21 -0
- package/skills/goke/README.md +666 -0
- package/skills/goke/SKILL.md +458 -0
- package/skills/goke/package.json +43 -0
- package/skills/goke/src/__test__/coerce.test.ts +411 -0
- package/skills/goke/src/__test__/index.test.ts +1798 -0
- package/skills/goke/src/__test__/types.test-d.ts +111 -0
- package/skills/goke/src/coerce.ts +547 -0
- package/skills/goke/src/goke.ts +1362 -0
- package/skills/goke/src/index.ts +16 -0
- package/skills/goke/src/mri.ts +164 -0
- package/skills/goke/tsconfig.json +15 -0
- package/skills/jitter/EDITOR.md +219 -0
- package/skills/jitter/EXPORT-INTERNALS.md +309 -0
- package/skills/jitter/SKILL.md +158 -0
- package/skills/jitter/jitter-clipboard.json +1042 -0
- package/skills/jitter/package.json +14 -0
- package/skills/jitter/tsconfig.json +15 -0
- package/skills/jitter/utils/actions.ts +212 -0
- package/skills/jitter/utils/export.ts +114 -0
- package/skills/jitter/utils/index.ts +141 -0
- package/skills/jitter/utils/snapshot.ts +154 -0
- package/skills/jitter/utils/traverse.ts +246 -0
- package/skills/jitter/utils/types.ts +279 -0
- package/skills/jitter/utils/wait.ts +133 -0
- package/skills/playwriter/SKILL.md +31 -0
- package/skills/security-review/SKILL.md +208 -0
- package/skills/simplify/SKILL.md +58 -0
- package/skills/termcast/SKILL.md +945 -0
- package/skills/tuistory/SKILL.md +250 -0
- package/skills/zustand-centralized-state/SKILL.md +582 -0
- package/src/__snapshots__/compact-session-context-no-system.md +35 -0
- package/src/__snapshots__/compact-session-context.md +41 -0
- package/src/__snapshots__/first-session-no-info.md +17 -0
- package/src/__snapshots__/first-session-with-info.md +23 -0
- package/src/__snapshots__/session-1.md +17 -0
- package/src/__snapshots__/session-2.md +5871 -0
- package/src/__snapshots__/session-3.md +17 -0
- package/src/__snapshots__/session-with-tools.md +5871 -0
- package/src/ai-tool-to-genai.test.ts +296 -0
- package/src/ai-tool-to-genai.ts +282 -0
- package/src/ai-tool.ts +39 -0
- package/src/bin.ts +108 -0
- package/src/bot-token.test.ts +171 -0
- package/src/bot-token.ts +159 -0
- package/src/channel-management.ts +172 -0
- package/src/cli-parsing.test.ts +132 -0
- package/src/cli.ts +3605 -0
- package/src/commands/abort.ts +112 -0
- package/src/commands/action-buttons.ts +376 -0
- package/src/commands/add-project.ts +152 -0
- package/src/commands/agent.ts +404 -0
- package/src/commands/ask-question.ts +330 -0
- package/src/commands/compact.ts +157 -0
- package/src/commands/context-usage.ts +199 -0
- package/src/commands/create-new-project.ts +179 -0
- package/src/commands/diff.ts +165 -0
- package/src/commands/file-upload.ts +389 -0
- package/src/commands/fork.ts +320 -0
- package/src/commands/gemini-apikey.ts +104 -0
- package/src/commands/login.ts +634 -0
- package/src/commands/mention-mode.ts +77 -0
- package/src/commands/merge-worktree.ts +177 -0
- package/src/commands/model.ts +961 -0
- package/src/commands/permissions.ts +261 -0
- package/src/commands/queue.ts +296 -0
- package/src/commands/remove-project.ts +155 -0
- package/src/commands/restart-opencode-server.ts +162 -0
- package/src/commands/resume.ts +242 -0
- package/src/commands/run-command.ts +123 -0
- package/src/commands/session-id.ts +109 -0
- package/src/commands/session.ts +250 -0
- package/src/commands/share.ts +106 -0
- package/src/commands/types.ts +25 -0
- package/src/commands/undo-redo.ts +221 -0
- package/src/commands/unset-model.ts +189 -0
- package/src/commands/upgrade.ts +52 -0
- package/src/commands/user-command.ts +193 -0
- package/src/commands/verbosity.ts +88 -0
- package/src/commands/worktree-settings.ts +79 -0
- package/src/commands/worktree.ts +431 -0
- package/src/condense-memory.ts +36 -0
- package/src/config.ts +148 -0
- package/src/database.ts +1530 -0
- package/src/db.test.ts +60 -0
- package/src/db.ts +190 -0
- package/src/discord-api.ts +35 -0
- package/src/discord-bot.ts +1316 -0
- package/src/discord-utils.test.ts +132 -0
- package/src/discord-utils.ts +767 -0
- package/src/errors.ts +213 -0
- package/src/escape-backticks.test.ts +469 -0
- package/src/format-tables.test.ts +223 -0
- package/src/format-tables.ts +145 -0
- package/src/forum-sync/config.ts +92 -0
- package/src/forum-sync/discord-operations.ts +241 -0
- package/src/forum-sync/index.ts +9 -0
- package/src/forum-sync/markdown.ts +176 -0
- package/src/forum-sync/sync-to-discord.ts +595 -0
- package/src/forum-sync/sync-to-files.ts +294 -0
- package/src/forum-sync/types.ts +175 -0
- package/src/forum-sync/watchers.ts +454 -0
- package/src/genai-worker-wrapper.ts +164 -0
- package/src/genai-worker.ts +386 -0
- package/src/genai.ts +321 -0
- package/src/generated/browser.ts +109 -0
- package/src/generated/client.ts +131 -0
- package/src/generated/commonInputTypes.ts +512 -0
- package/src/generated/enums.ts +46 -0
- package/src/generated/internal/class.ts +362 -0
- package/src/generated/internal/prismaNamespace.ts +2251 -0
- package/src/generated/internal/prismaNamespaceBrowser.ts +308 -0
- package/src/generated/models/bot_api_keys.ts +1288 -0
- package/src/generated/models/bot_tokens.ts +1577 -0
- package/src/generated/models/channel_agents.ts +1256 -0
- package/src/generated/models/channel_directories.ts +2104 -0
- package/src/generated/models/channel_mention_mode.ts +1300 -0
- package/src/generated/models/channel_models.ts +1288 -0
- package/src/generated/models/channel_verbosity.ts +1224 -0
- package/src/generated/models/channel_worktrees.ts +1308 -0
- package/src/generated/models/forum_sync_configs.ts +1452 -0
- package/src/generated/models/global_models.ts +1288 -0
- package/src/generated/models/ipc_requests.ts +1485 -0
- package/src/generated/models/part_messages.ts +1302 -0
- package/src/generated/models/scheduled_tasks.ts +2320 -0
- package/src/generated/models/session_agents.ts +1086 -0
- package/src/generated/models/session_models.ts +1114 -0
- package/src/generated/models/session_start_sources.ts +1408 -0
- package/src/generated/models/thread_sessions.ts +1599 -0
- package/src/generated/models/thread_worktrees.ts +1352 -0
- package/src/generated/models.ts +29 -0
- package/src/heap-monitor.ts +121 -0
- package/src/hrana-server.test.ts +428 -0
- package/src/hrana-server.ts +547 -0
- package/src/image-utils.ts +149 -0
- package/src/interaction-handler.ts +461 -0
- package/src/ipc-polling.ts +325 -0
- package/src/kimaki-digital-twin.e2e.test.ts +201 -0
- package/src/limit-heading-depth.test.ts +116 -0
- package/src/limit-heading-depth.ts +26 -0
- package/src/logger.ts +203 -0
- package/src/markdown.test.ts +360 -0
- package/src/markdown.ts +410 -0
- package/src/message-formatting.test.ts +81 -0
- package/src/message-formatting.ts +549 -0
- package/src/openai-realtime.ts +362 -0
- package/src/opencode-plugin-loading.e2e.test.ts +112 -0
- package/src/opencode-plugin.test.ts +108 -0
- package/src/opencode-plugin.ts +652 -0
- package/src/opencode.ts +554 -0
- package/src/privacy-sanitizer.ts +142 -0
- package/src/schema.sql +158 -0
- package/src/sentry.ts +137 -0
- package/src/session-handler/state.ts +232 -0
- package/src/session-handler.ts +2668 -0
- package/src/session-search.test.ts +50 -0
- package/src/session-search.ts +148 -0
- package/src/startup-service.ts +200 -0
- package/src/system-message.ts +568 -0
- package/src/task-runner.ts +425 -0
- package/src/task-schedule.test.ts +84 -0
- package/src/task-schedule.ts +287 -0
- package/src/thinking-utils.ts +61 -0
- package/src/thread-message-queue.e2e.test.ts +997 -0
- package/src/tools.ts +432 -0
- package/src/unnest-code-blocks.test.ts +679 -0
- package/src/unnest-code-blocks.ts +168 -0
- package/src/upgrade.ts +127 -0
- package/src/utils.ts +145 -0
- package/src/voice-handler.ts +852 -0
- package/src/voice.test.ts +219 -0
- package/src/voice.ts +444 -0
- package/src/wait-session.ts +147 -0
- package/src/worker-types.ts +64 -0
- package/src/worktree-utils.ts +988 -0
- package/src/xml.test.ts +38 -0
- package/src/xml.ts +121 -0
|
@@ -0,0 +1,582 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: zustand-centralized-state
|
|
3
|
+
description: >
|
|
4
|
+
Centralized state management pattern using Zustand vanilla stores. One immutable
|
|
5
|
+
state atom, functional transitions via setState(), and a single subscribe() for
|
|
6
|
+
all reactive side effects. Based on Rich Hickey's "Simple Made Easy" principles:
|
|
7
|
+
prefer values over mutable state, derive instead of cache, centralize transitions,
|
|
8
|
+
and push side effects to the edges. Resource co-location in the same store is
|
|
9
|
+
also valid when lifecycle management is safer that way. Use this skill when
|
|
10
|
+
building any stateful TypeScript application (servers, extensions, CLIs, relays)
|
|
11
|
+
to keep state simple, testable, and easy to reason about. ALWAYS read this skill
|
|
12
|
+
when a project uses zustand/vanilla for state management outside of React.
|
|
13
|
+
version: 0.2.1
|
|
14
|
+
---
|
|
15
|
+
|
|
16
|
+
# Centralized State Management
|
|
17
|
+
|
|
18
|
+
A pattern for managing application state that keeps programs simple, testable, and
|
|
19
|
+
easy to reason about. Uses Zustand vanilla stores as the mechanism, but the
|
|
20
|
+
principles apply to any state management approach.
|
|
21
|
+
|
|
22
|
+
## Background
|
|
23
|
+
|
|
24
|
+
Rich Hickey's talk **"Simple Made Easy"** (2011) argues that most program complexity
|
|
25
|
+
comes from **complecting** (interleaving) things that should be independent. Mutable
|
|
26
|
+
state is one of the worst offenders: it interleaves *identity* (what thing are we
|
|
27
|
+
talking about), *state* (what is its current value), and *time* (when did it change).
|
|
28
|
+
|
|
29
|
+
When you mutate a Map in place, you lose the previous value, every reader is coupled
|
|
30
|
+
to every writer, and you can't reason about what the state was at any point in time.
|
|
31
|
+
State scattered across multiple mutable variables in different scopes makes it
|
|
32
|
+
impossible to answer "what does the program look like right now?"
|
|
33
|
+
|
|
34
|
+
The solution is not "never have state" -- that's impossible for real programs. The
|
|
35
|
+
solution is to **manage state explicitly**: one place it lives, controlled transitions,
|
|
36
|
+
immutable values, and side effects derived from state rather than scattered across
|
|
37
|
+
handlers.
|
|
38
|
+
|
|
39
|
+
This makes programs:
|
|
40
|
+
- **Simpler to reason about** -- one place to look for all state
|
|
41
|
+
- **Easier to test** -- pure state transitions, no I/O needed
|
|
42
|
+
- **Less buggy** -- impossible to have half-updated inconsistent state
|
|
43
|
+
- **Easier to debug** -- you can log/snapshot state at any transition
|
|
44
|
+
|
|
45
|
+
## Core Principles
|
|
46
|
+
|
|
47
|
+
### 1. Prefer values over mutable state
|
|
48
|
+
|
|
49
|
+
Use immutable data. When state changes, produce a new value instead of mutating in
|
|
50
|
+
place. In TypeScript with Zustand, this means `setState()` with functional updates
|
|
51
|
+
that return new objects/Maps rather than mutating existing ones.
|
|
52
|
+
|
|
53
|
+
```ts
|
|
54
|
+
// BAD: mutation scattered in handler
|
|
55
|
+
connectedTabs.set(tabId, { ...info, state: 'connected' })
|
|
56
|
+
connectionState = 'connected'
|
|
57
|
+
|
|
58
|
+
// GOOD: single atomic transition producing new values
|
|
59
|
+
store.setState((state) => {
|
|
60
|
+
const newTabs = new Map(state.tabs)
|
|
61
|
+
newTabs.set(tabId, { ...info, state: 'connected' })
|
|
62
|
+
return { tabs: newTabs, connectionState: 'connected' }
|
|
63
|
+
})
|
|
64
|
+
```
|
|
65
|
+
|
|
66
|
+
The second version is atomic -- both `tabs` and `connectionState` update together
|
|
67
|
+
or not at all. There's no intermediate state where tabs shows connected but
|
|
68
|
+
connectionState is still idle.
|
|
69
|
+
|
|
70
|
+
### 2. Derive instead of cache
|
|
71
|
+
|
|
72
|
+
If a value can be computed from existing state, compute it on demand instead of
|
|
73
|
+
maintaining a separate cache that must stay in sync.
|
|
74
|
+
|
|
75
|
+
```ts
|
|
76
|
+
// BAD: separate index that can get out of sync
|
|
77
|
+
const extensionKeyIndex = new Map<string, string>() // stableKey -> connectionId
|
|
78
|
+
|
|
79
|
+
// must remember to update on every add/remove:
|
|
80
|
+
extensionKeyIndex.set(ext.stableKey, ext.id)
|
|
81
|
+
// forgot to delete on disconnect? now you have a stale entry
|
|
82
|
+
|
|
83
|
+
// GOOD: derive it when needed
|
|
84
|
+
function findExtensionByKey(state: RelayState, key: string) {
|
|
85
|
+
for (const ext of state.extensions.values()) {
|
|
86
|
+
if (ext.stableKey === key) return ext
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
```
|
|
90
|
+
|
|
91
|
+
At small scales (dozens of entries, not millions), the linear scan is free and you've
|
|
92
|
+
eliminated an entire class of consistency bugs.
|
|
93
|
+
|
|
94
|
+
**Anti-pattern: parallel maps for the same entity.** A common mistake is splitting
|
|
95
|
+
one entity across two maps to "separate state from I/O" — e.g. a `clients` map for
|
|
96
|
+
domain fields and a `clientIO` map for WebSocket handles, keyed by the same ID.
|
|
97
|
+
This forces every add/remove to touch both maps and inevitably one gets forgotten
|
|
98
|
+
(leaking stale handles or leaving orphaned state). Instead, co-locate I/O handles
|
|
99
|
+
on the entity type itself:
|
|
100
|
+
|
|
101
|
+
```ts
|
|
102
|
+
// BAD: two maps that must stay in sync
|
|
103
|
+
type ClientState = { id: string; extensionId: string }
|
|
104
|
+
type ClientIO = { id: string; ws: WSContext }
|
|
105
|
+
type State = {
|
|
106
|
+
clients: Map<string, ClientState>
|
|
107
|
+
clientIO: Map<string, ClientIO> // same keys, always
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
// GOOD: one map, one entity, one add/remove
|
|
111
|
+
type Client = { id: string; extensionId: string; ws: WSContext }
|
|
112
|
+
type State = {
|
|
113
|
+
clients: Map<string, Client>
|
|
114
|
+
}
|
|
115
|
+
```
|
|
116
|
+
|
|
117
|
+
"Separate state from I/O" means keep `setState()` callbacks pure (no side effects) —
|
|
118
|
+
it does NOT mean store I/O handles in a separate map. Co-locating handles with their
|
|
119
|
+
entity prevents consistency bugs and makes cleanup trivial.
|
|
120
|
+
|
|
121
|
+
### 3. Centralize all state in one store
|
|
122
|
+
|
|
123
|
+
All application state lives in a single Zustand store. There should be one place to
|
|
124
|
+
look to understand the full state of the program.
|
|
125
|
+
|
|
126
|
+
```ts
|
|
127
|
+
import { createStore } from 'zustand/vanilla'
|
|
128
|
+
|
|
129
|
+
type AppState = {
|
|
130
|
+
connections: Map<string, Connection>
|
|
131
|
+
clients: Map<string, Client>
|
|
132
|
+
connectionState: 'idle' | 'connected' | 'error'
|
|
133
|
+
errorText: string | undefined
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
const store = createStore<AppState>(() => ({
|
|
137
|
+
connections: new Map(),
|
|
138
|
+
clients: new Map(),
|
|
139
|
+
connectionState: 'idle',
|
|
140
|
+
errorText: undefined,
|
|
141
|
+
}))
|
|
142
|
+
```
|
|
143
|
+
|
|
144
|
+
This is the single source of truth. No separate variables, no state scattered across
|
|
145
|
+
closures, no Maps defined in different scopes.
|
|
146
|
+
|
|
147
|
+
### 4. State transitions use only current state and event data
|
|
148
|
+
|
|
149
|
+
Every `setState()` call should be a pure function of the current state and the
|
|
150
|
+
incoming event data. No reading from external variables, no side effects inside
|
|
151
|
+
`setState()`.
|
|
152
|
+
|
|
153
|
+
```ts
|
|
154
|
+
// the transition only uses `state` (current) and `event` (incoming data)
|
|
155
|
+
store.setState((state) => {
|
|
156
|
+
const newTabs = new Map(state.tabs)
|
|
157
|
+
newTabs.set(event.tabId, {
|
|
158
|
+
sessionId: event.sessionId,
|
|
159
|
+
state: 'connected',
|
|
160
|
+
})
|
|
161
|
+
return { tabs: newTabs }
|
|
162
|
+
})
|
|
163
|
+
```
|
|
164
|
+
|
|
165
|
+
This makes every transition testable: given this state and this event, the new state
|
|
166
|
+
should be X. No mocks needed, no I/O setup, just data in and data out.
|
|
167
|
+
|
|
168
|
+
### 5. Resource co-location is allowed when it improves lifecycle safety
|
|
169
|
+
|
|
170
|
+
Putting runtime resources in Zustand is valid when keeping them outside the store
|
|
171
|
+
would create split-brain lifecycle management (state in one place, resources in
|
|
172
|
+
another) and increase leak risk.
|
|
173
|
+
|
|
174
|
+
Examples of colocated resources:
|
|
175
|
+
- WebSocket handles
|
|
176
|
+
- timers/interval handles
|
|
177
|
+
- pending request callback maps
|
|
178
|
+
- abort controllers
|
|
179
|
+
|
|
180
|
+
If resources live in the store:
|
|
181
|
+
- transitions still must be deterministic and side-effect free
|
|
182
|
+
- store references, don't execute effects inside transitions
|
|
183
|
+
- cleanup effects (close sockets, clear intervals) still run in handlers/subscribe
|
|
184
|
+
based on state transitions
|
|
185
|
+
|
|
186
|
+
Rule of thumb:
|
|
187
|
+
- Prefer plain-data state for maximal testability
|
|
188
|
+
- Co-locate resources when one centralized store materially improves cleanup and
|
|
189
|
+
ownership tracking
|
|
190
|
+
|
|
191
|
+
### 6. Mutable resources are state too
|
|
192
|
+
|
|
193
|
+
If a runtime resource has mutable lifecycle state, treat it as state and keep it in
|
|
194
|
+
the centralized store alongside the data it controls.
|
|
195
|
+
|
|
196
|
+
`AbortController` is the clearest example:
|
|
197
|
+
- it has mutable lifecycle (`signal.aborted` flips from `false` to `true`)
|
|
198
|
+
- that lifecycle controls behavior (whether work should continue)
|
|
199
|
+
- ownership and cleanup matter (who creates, replaces, aborts, and clears it)
|
|
200
|
+
|
|
201
|
+
In practice, an abort controller is often equivalent to a state bit with a handle.
|
|
202
|
+
Keeping it in a local variable while related domain state lives in Zustand creates
|
|
203
|
+
split-brain state and leak risk.
|
|
204
|
+
|
|
205
|
+
```ts
|
|
206
|
+
// BAD: split state (store + local mutable resource)
|
|
207
|
+
let requestController: AbortController | undefined
|
|
208
|
+
|
|
209
|
+
requestController = new AbortController()
|
|
210
|
+
|
|
211
|
+
// GOOD: one source of truth
|
|
212
|
+
type State = {
|
|
213
|
+
requestController: AbortController | undefined
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
store.setState((state) => {
|
|
217
|
+
return {
|
|
218
|
+
...state,
|
|
219
|
+
requestController: new AbortController(),
|
|
220
|
+
}
|
|
221
|
+
})
|
|
222
|
+
```
|
|
223
|
+
|
|
224
|
+
This keeps lifecycle ownership explicit: transitions decide when controller
|
|
225
|
+
references appear/disappear; handlers/subscribe perform side effects like
|
|
226
|
+
`controller.abort()` based on state transitions.
|
|
227
|
+
|
|
228
|
+
### 7. Centralize side effects in subscribe
|
|
229
|
+
|
|
230
|
+
Side effects (I/O, UI updates, cleanup, logging) go in a single `subscribe()`
|
|
231
|
+
callback that reacts to state changes. Side effects are **derived from state**, not
|
|
232
|
+
scattered across handlers.
|
|
233
|
+
|
|
234
|
+
```ts
|
|
235
|
+
store.subscribe((state, prevState) => {
|
|
236
|
+
// logging
|
|
237
|
+
logger.log('state changed:', state)
|
|
238
|
+
|
|
239
|
+
// UI update derived purely from current state
|
|
240
|
+
updateIcon(state.connectionState, state.tabs)
|
|
241
|
+
|
|
242
|
+
// cleanup: if a connection was removed, close its resources
|
|
243
|
+
for (const [id, conn] of prevState.connections) {
|
|
244
|
+
if (!state.connections.has(id)) {
|
|
245
|
+
conn.socket.close()
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
})
|
|
249
|
+
```
|
|
250
|
+
|
|
251
|
+
## The Pattern
|
|
252
|
+
|
|
253
|
+
The architecture has three layers:
|
|
254
|
+
|
|
255
|
+
```
|
|
256
|
+
Event handlers State store Subscribe
|
|
257
|
+
(imperative shell) (centralized atom) (reactive side effects)
|
|
258
|
+
~~~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~~~~~~~
|
|
259
|
+
|
|
260
|
+
onMessage(data) ------> store.setState( store.subscribe(
|
|
261
|
+
onConnect(ws) (state) => { (state, prev) => {
|
|
262
|
+
onDisconnect(id) // pure // side effects
|
|
263
|
+
onTimer() // transition // derived from
|
|
264
|
+
// no I/O // state shape
|
|
265
|
+
} }
|
|
266
|
+
) )
|
|
267
|
+
```
|
|
268
|
+
|
|
269
|
+
**Event handlers** parse incoming events and call `setState()`.
|
|
270
|
+
They may also do direct I/O that needs event data (like forwarding a message).
|
|
271
|
+
|
|
272
|
+
**State store** holds the single immutable state atom. Transitions are pure functions.
|
|
273
|
+
|
|
274
|
+
**Subscribe** reacts to state changes and performs side effects that are purely
|
|
275
|
+
derived from the current state shape (not from specific events).
|
|
276
|
+
|
|
277
|
+
## Rules
|
|
278
|
+
|
|
279
|
+
1. Use `zustand/vanilla` for non-React applications (servers, extensions, CLIs) --
|
|
280
|
+
it has no React dependency and works in any JS runtime
|
|
281
|
+
2. Define all state in a single `createStore()` call with a typed state interface
|
|
282
|
+
3. Never mutate state directly -- always use `store.setState()` with functional
|
|
283
|
+
updates that return new objects
|
|
284
|
+
4. Keep `setState()` callbacks deterministic -- no external effects, only compute
|
|
285
|
+
new state from current state + event data
|
|
286
|
+
5. Use a single `subscribe()` for all reactive side effects -- not multiple
|
|
287
|
+
subscribes scattered across the codebase
|
|
288
|
+
6. Side effects in subscribe should be derived from state shape, not from specific
|
|
289
|
+
events -- ask "given this state, what should the world look like?" not "what
|
|
290
|
+
event just happened?"
|
|
291
|
+
7. Derive computed values instead of caching them in separate state -- if it can be
|
|
292
|
+
computed from existing state, compute it
|
|
293
|
+
8. Use `(state, prevState)` diffing in subscribe when you need to react to specific
|
|
294
|
+
changes (e.g. "a connection was removed")
|
|
295
|
+
9. Keep the state interface minimal -- only store what you can't derive
|
|
296
|
+
10. For state transitions that are complex or reused, extract them as pure
|
|
297
|
+
functions that take state + event data and return new state
|
|
298
|
+
11. Resource co-location is acceptable: storing sockets/timers/callback maps in
|
|
299
|
+
Zustand is fine when it prevents lifecycle drift. Keep side effects out of
|
|
300
|
+
transitions.
|
|
301
|
+
12. Treat mutable runtime resources as state (e.g. `AbortController`) -- if a
|
|
302
|
+
resource has lifecycle state that drives behavior, keep its reference in the
|
|
303
|
+
same centralized store as related domain state.
|
|
304
|
+
|
|
305
|
+
## When subscribe does NOT fit
|
|
306
|
+
|
|
307
|
+
Not all side effects belong in subscribe. The subscribe callback gets
|
|
308
|
+
`(newState, prevState)` but doesn't know **what event caused the change**. This
|
|
309
|
+
matters for message routing:
|
|
310
|
+
|
|
311
|
+
```ts
|
|
312
|
+
// this does NOT fit subscribe -- you need the actual message, not just state diff
|
|
313
|
+
function onCdpEvent(extensionId: string, message: CdpMessage) {
|
|
314
|
+
// 1. state transition -> subscribe
|
|
315
|
+
store.setState((s) => addTarget(s, extensionId, message.params))
|
|
316
|
+
// 2. forward the exact message -> stays in handler (needs event data)
|
|
317
|
+
forwardToPlaywright(extensionId, message)
|
|
318
|
+
}
|
|
319
|
+
```
|
|
320
|
+
|
|
321
|
+
Rule of thumb:
|
|
322
|
+
- **Subscribe**: side effects derived from state shape ("icon should show green
|
|
323
|
+
because connectionState is 'connected'")
|
|
324
|
+
- **Handler**: side effects that need event data ("forward this specific CDP
|
|
325
|
+
message to the playwright client")
|
|
326
|
+
|
|
327
|
+
## Real-World Example: Chrome Extension State
|
|
328
|
+
|
|
329
|
+
A Chrome extension that manages browser tab connections. Before: mutable variables
|
|
330
|
+
scattered across the background script. After: one Zustand store, one subscribe.
|
|
331
|
+
|
|
332
|
+
### State definition
|
|
333
|
+
|
|
334
|
+
```ts
|
|
335
|
+
import { createStore } from 'zustand/vanilla'
|
|
336
|
+
|
|
337
|
+
type ConnectionState = 'idle' | 'connected' | 'extension-replaced'
|
|
338
|
+
type TabState = 'connecting' | 'connected' | 'error'
|
|
339
|
+
|
|
340
|
+
interface TabInfo {
|
|
341
|
+
sessionId?: string
|
|
342
|
+
targetId?: string
|
|
343
|
+
state: TabState
|
|
344
|
+
errorText?: string
|
|
345
|
+
pinnedCount?: number
|
|
346
|
+
attachOrder?: number
|
|
347
|
+
isRecording?: boolean
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
interface ExtensionState {
|
|
351
|
+
tabs: Map<number, TabInfo>
|
|
352
|
+
connectionState: ConnectionState
|
|
353
|
+
currentTabId: number | undefined
|
|
354
|
+
errorText: string | undefined
|
|
355
|
+
}
|
|
356
|
+
|
|
357
|
+
const store = createStore<ExtensionState>(() => ({
|
|
358
|
+
tabs: new Map(),
|
|
359
|
+
connectionState: 'idle',
|
|
360
|
+
currentTabId: undefined,
|
|
361
|
+
errorText: undefined,
|
|
362
|
+
}))
|
|
363
|
+
```
|
|
364
|
+
|
|
365
|
+
### State transitions in event handlers
|
|
366
|
+
|
|
367
|
+
```ts
|
|
368
|
+
// tab successfully attached
|
|
369
|
+
store.setState((state) => {
|
|
370
|
+
const newTabs = new Map(state.tabs)
|
|
371
|
+
newTabs.set(tabId, {
|
|
372
|
+
sessionId,
|
|
373
|
+
targetId,
|
|
374
|
+
state: 'connected',
|
|
375
|
+
attachOrder: newTabs.size,
|
|
376
|
+
})
|
|
377
|
+
return { tabs: newTabs, connectionState: 'connected' }
|
|
378
|
+
})
|
|
379
|
+
|
|
380
|
+
// tab detached
|
|
381
|
+
store.setState((state) => {
|
|
382
|
+
const newTabs = new Map(state.tabs)
|
|
383
|
+
newTabs.delete(tabId)
|
|
384
|
+
return { tabs: newTabs }
|
|
385
|
+
})
|
|
386
|
+
|
|
387
|
+
// WebSocket disconnected
|
|
388
|
+
store.setState((state) => {
|
|
389
|
+
const newTabs = new Map(state.tabs)
|
|
390
|
+
for (const [id, tab] of newTabs) {
|
|
391
|
+
newTabs.set(id, { ...tab, state: 'connecting' })
|
|
392
|
+
}
|
|
393
|
+
return { tabs: newTabs, connectionState: 'idle' }
|
|
394
|
+
})
|
|
395
|
+
|
|
396
|
+
// extension replaced (kicked by another instance)
|
|
397
|
+
store.setState({
|
|
398
|
+
tabs: new Map(),
|
|
399
|
+
connectionState: 'extension-replaced',
|
|
400
|
+
errorText: 'Another instance took over this connection',
|
|
401
|
+
})
|
|
402
|
+
```
|
|
403
|
+
|
|
404
|
+
### All side effects in one subscribe
|
|
405
|
+
|
|
406
|
+
```ts
|
|
407
|
+
store.subscribe((state, prevState) => {
|
|
408
|
+
// 1. log every state change
|
|
409
|
+
logger.log(state)
|
|
410
|
+
|
|
411
|
+
// 2. update extension icon based on current state
|
|
412
|
+
// purely derived from state -- doesn't care what event caused the change
|
|
413
|
+
void updateIcons(state)
|
|
414
|
+
|
|
415
|
+
// 3. show/hide context menu based on whether current tab is connected
|
|
416
|
+
updateContextMenuVisibility(state)
|
|
417
|
+
|
|
418
|
+
// 4. sync Chrome tab groups when tab list changes
|
|
419
|
+
if (serializeTabs(state.tabs) !== serializeTabs(prevState.tabs)) {
|
|
420
|
+
syncTabGroup(state.tabs)
|
|
421
|
+
}
|
|
422
|
+
})
|
|
423
|
+
```
|
|
424
|
+
|
|
425
|
+
The `updateIcons` function reads `connectionState`, `tabs`, and `errorText` to decide
|
|
426
|
+
which icon to show. It doesn't know or care whether the state changed because a tab
|
|
427
|
+
was attached, a WebSocket reconnected, or an error happened. It just asks: **given
|
|
428
|
+
this state, what should the icon look like?**
|
|
429
|
+
|
|
430
|
+
This is the key insight: side effects are a **projection of current state**, not a
|
|
431
|
+
reaction to specific events.
|
|
432
|
+
|
|
433
|
+
### Why this is better
|
|
434
|
+
|
|
435
|
+
**Before** (scattered side effects):
|
|
436
|
+
```
|
|
437
|
+
onTabAttached() -> update tabs Map, update icon, update badge, update tab group
|
|
438
|
+
onTabDetached() -> update tabs Map, update icon, update badge, update tab group
|
|
439
|
+
onWsConnected() -> update connectionState, update icon
|
|
440
|
+
onWsDisconnected() -> update tabs Map, update connectionState, update icon, clear badge
|
|
441
|
+
onError() -> update errorText, update icon, update badge
|
|
442
|
+
```
|
|
443
|
+
|
|
444
|
+
Every handler has to remember to update every side effect. Add a new side effect
|
|
445
|
+
(e.g. "update status bar")? You must find and update every handler.
|
|
446
|
+
|
|
447
|
+
**After** (centralized):
|
|
448
|
+
```
|
|
449
|
+
onTabAttached() -> store.setState(...)
|
|
450
|
+
onTabDetached() -> store.setState(...)
|
|
451
|
+
onWsConnected() -> store.setState(...)
|
|
452
|
+
onWsDisconnected() -> store.setState(...)
|
|
453
|
+
onError() -> store.setState(...)
|
|
454
|
+
|
|
455
|
+
subscribe() -> update icon, update badge, update tab group, update status bar
|
|
456
|
+
```
|
|
457
|
+
|
|
458
|
+
Handlers only update state. Subscribe handles all side effects. Add a new side
|
|
459
|
+
effect? Add one line in subscribe. Impossible to forget a handler.
|
|
460
|
+
|
|
461
|
+
## Testing
|
|
462
|
+
|
|
463
|
+
State transitions are pure functions, so testing requires no mocks, no WebSockets,
|
|
464
|
+
no I/O setup:
|
|
465
|
+
|
|
466
|
+
```ts
|
|
467
|
+
import { test, expect } from 'vitest'
|
|
468
|
+
|
|
469
|
+
test('attaching a tab updates state correctly', () => {
|
|
470
|
+
const before: ExtensionState = {
|
|
471
|
+
tabs: new Map(),
|
|
472
|
+
connectionState: 'idle',
|
|
473
|
+
currentTabId: undefined,
|
|
474
|
+
errorText: undefined,
|
|
475
|
+
}
|
|
476
|
+
|
|
477
|
+
const after = attachTab(before, {
|
|
478
|
+
tabId: 42,
|
|
479
|
+
sessionId: 'session-1',
|
|
480
|
+
targetId: 'target-1',
|
|
481
|
+
})
|
|
482
|
+
|
|
483
|
+
expect(after.tabs.size).toBe(1)
|
|
484
|
+
expect(after.tabs.get(42)?.state).toBe('connected')
|
|
485
|
+
expect(after.connectionState).toBe('connected')
|
|
486
|
+
// previous state is unchanged (immutable)
|
|
487
|
+
expect(before.tabs.size).toBe(0)
|
|
488
|
+
expect(before.connectionState).toBe('idle')
|
|
489
|
+
})
|
|
490
|
+
|
|
491
|
+
test('disconnecting resets all tabs to connecting', () => {
|
|
492
|
+
const before: ExtensionState = {
|
|
493
|
+
tabs: new Map([
|
|
494
|
+
[1, { state: 'connected', sessionId: 's1' }],
|
|
495
|
+
[2, { state: 'connected', sessionId: 's2' }],
|
|
496
|
+
]),
|
|
497
|
+
connectionState: 'connected',
|
|
498
|
+
currentTabId: 1,
|
|
499
|
+
errorText: undefined,
|
|
500
|
+
}
|
|
501
|
+
|
|
502
|
+
const after = onDisconnect(before)
|
|
503
|
+
|
|
504
|
+
expect(after.connectionState).toBe('idle')
|
|
505
|
+
for (const tab of after.tabs.values()) {
|
|
506
|
+
expect(tab.state).toBe('connecting')
|
|
507
|
+
}
|
|
508
|
+
// original unchanged
|
|
509
|
+
for (const tab of before.tabs.values()) {
|
|
510
|
+
expect(tab.state).toBe('connected')
|
|
511
|
+
}
|
|
512
|
+
})
|
|
513
|
+
```
|
|
514
|
+
|
|
515
|
+
No WebSocket mocks. No Chrome API stubs. No timers. Just data in, data out.
|
|
516
|
+
|
|
517
|
+
## Extracting reusable transition functions
|
|
518
|
+
|
|
519
|
+
When transitions are complex or reused across handlers, extract them as pure
|
|
520
|
+
functions:
|
|
521
|
+
|
|
522
|
+
```ts
|
|
523
|
+
// pure transition function -- takes state + event, returns new state
|
|
524
|
+
function attachTab(state: ExtensionState, event: {
|
|
525
|
+
tabId: number
|
|
526
|
+
sessionId: string
|
|
527
|
+
targetId: string
|
|
528
|
+
}): ExtensionState {
|
|
529
|
+
const newTabs = new Map(state.tabs)
|
|
530
|
+
newTabs.set(event.tabId, {
|
|
531
|
+
sessionId: event.sessionId,
|
|
532
|
+
targetId: event.targetId,
|
|
533
|
+
state: 'connected',
|
|
534
|
+
attachOrder: newTabs.size,
|
|
535
|
+
})
|
|
536
|
+
return { ...state, tabs: newTabs, connectionState: 'connected' }
|
|
537
|
+
}
|
|
538
|
+
|
|
539
|
+
// used in handler
|
|
540
|
+
store.setState((state) => attachTab(state, { tabId, sessionId, targetId }))
|
|
541
|
+
```
|
|
542
|
+
|
|
543
|
+
This keeps handlers minimal and transitions testable.
|
|
544
|
+
|
|
545
|
+
## Zustand vanilla API reference
|
|
546
|
+
|
|
547
|
+
```ts
|
|
548
|
+
import { createStore } from 'zustand/vanilla'
|
|
549
|
+
|
|
550
|
+
// create store with initial state
|
|
551
|
+
const store = createStore<MyState>(() => initialState)
|
|
552
|
+
|
|
553
|
+
// read current state (snapshot, safe to hold)
|
|
554
|
+
const snapshot = store.getState()
|
|
555
|
+
|
|
556
|
+
// functional update (preferred -- derives from current state)
|
|
557
|
+
store.setState((state) => ({ ...state, count: state.count + 1 }))
|
|
558
|
+
|
|
559
|
+
// direct merge (for simple top-level updates)
|
|
560
|
+
store.setState({ connectionState: 'connected' })
|
|
561
|
+
|
|
562
|
+
// subscribe to all changes (returns unsubscribe function)
|
|
563
|
+
const unsub = store.subscribe((state, prevState) => { ... })
|
|
564
|
+
|
|
565
|
+
// subscribe with selector (fires only when selected value changes)
|
|
566
|
+
const unsub = store.subscribe(
|
|
567
|
+
(state) => state.connectionState,
|
|
568
|
+
(connectionState, prevConnectionState) => { ... },
|
|
569
|
+
)
|
|
570
|
+
```
|
|
571
|
+
|
|
572
|
+
## Summary
|
|
573
|
+
|
|
574
|
+
| Principle | Practice |
|
|
575
|
+
|---|---|
|
|
576
|
+
| Values over state | `setState()` returns new objects, never mutate in place |
|
|
577
|
+
| Derive over cache | Compute indexes and aggregates on demand |
|
|
578
|
+
| Centralize state | One `createStore()`, one state type, one source of truth |
|
|
579
|
+
| Pure transitions | `setState((state) => newState)` with no side effects |
|
|
580
|
+
| Centralize side effects | One `subscribe()` for all reactive effects |
|
|
581
|
+
| State vs I/O boundary | Prefer separation, but co-location is valid for safer cleanup |
|
|
582
|
+
| Test with data | State in -> state out, no mocks needed |
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
[Tool bash]: command=grep -A100 "^export type Config = {" node_modules/@opencode-ai/.ignored_sdk/dist/gen/types.gen.d.ts, description=find Config type, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord
|
|
2
|
+
[Tool bash]: command=grep -B5 -A30 "^export type AgentConfig" node_modules/@opencode-ai/.ignored_sdk/dist/gen/types.gen.d, description=find AgentConfig type, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord
|
|
3
|
+
[Assistant]: Found it! `AgentConfig` has a `prompt` field. Let me check if `config.get()` returns this:
|
|
4
|
+
|
|
5
|
+
[Tool bash]: command=grep -A20 "ConfigGetResponses" node_modules/@opencode-ai/.ignored_sdk/dist/gen/types.gen.d.ts, description=check ConfigGet response, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord
|
|
6
|
+
[Tool bash]: command=grep -A20 "AppAgentsResponses" node_modules/@opencode-ai/.ignored_sdk/dist/gen/types.gen.d.ts, description=check agents response, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord
|
|
7
|
+
[Tool bash]: command=grep -B2 -A30 "^export type Agent = {" node_modules/@opencode-ai/.ignored_sdk/dist/gen/types.gen.d.t, description=check Agent type, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord
|
|
8
|
+
[Assistant]: Found `Agent.prompt` and `AgentConfig.prompt` fields. Let me test if `app.agents()` returns useful prompt info:
|
|
9
|
+
|
|
10
|
+
[Tool bash]: command=cat > /tmp/test-agents.ts << 'EOF' import { OpencodeClient } from '@opencode-ai/sdk' async function, description=test agents and config api, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord, timeout=30000
|
|
11
|
+
[Tool bash]: command=cd discord && cat > /tmp/test-agents.ts << 'EOF' import { OpencodeClient } from '@opencode-ai/sdk', description=test agents api from discord dir, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord, timeout=30000
|
|
12
|
+
[Tool bash]: command=cat > discord/src/test-agents-api.ts << 'EOF' import { OpencodeClient } from '@opencode-ai/sdk' asy, description=test agents api, timeout=30000
|
|
13
|
+
[Tool bash]: command=rm discord/src/test-agents-api.ts, description=cleanup test file
|
|
14
|
+
[Assistant]: found `app.agents()` returns agent-specific prompts:
|
|
15
|
+
|
|
16
|
+
```
|
|
17
|
+
plan: 2719 chars - "We are currently in the PLANNING phase..."
|
|
18
|
+
explore: 871 chars - "You are a file search specialist..."
|
|
19
|
+
title: 1508 chars - "You are a title generator..."
|
|
20
|
+
```
|
|
21
|
+
|
|
22
|
+
**but** these are just agent prompts from config, not the full system prompt. the full system prompt opencode builds includes:
|
|
23
|
+
- agents.md content
|
|
24
|
+
- environment info
|
|
25
|
+
- file tree
|
|
26
|
+
- tool descriptions
|
|
27
|
+
|
|
28
|
+
this full prompt is built dynamically and **not exposed via api**. we can only get the agent-specific custom prompts.
|
|
29
|
+
|
|
30
|
+
**options:**
|
|
31
|
+
1. use `app.agents()` to get agent prompts as partial context
|
|
32
|
+
2. read the AGENTS.md file directly from the project directory (we have it)
|
|
33
|
+
3. request opencode to expose the full system prompt in sdk
|
|
34
|
+
|
|
35
|
+
want me to add AGENTS.md file reading as fallback context?
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
[User]: search if there are ways to get it in other sdk methods
|
|
2
|
+
|
|
3
|
+
[Tool bash]: command=cat node_modules/@opencode-ai/.ignored_sdk/dist/gen/sdk.gen.d.ts, description=check sdk methods, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord
|
|
4
|
+
[Tool bash]: command=cat node_modules/@opencode-ai/.ignored_sdk/dist/gen/core/types.gen.d.ts | grep -A30 "SessionGetRespo, description=check response types, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord
|
|
5
|
+
[Tool bash]: command=cat node_modules/@opencode-ai/.ignored_sdk/dist/gen/types.gen.d.ts 2>/dev/null | head -200, description=check types file, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord
|
|
6
|
+
[Tool bash]: command=grep -A50 "^export type Session = {" node_modules/@opencode-ai/.ignored_sdk/dist/gen/types.gen.d.ts, description=find Session type, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord
|
|
7
|
+
[Tool bash]: command=grep -A100 "^export type Config = {" node_modules/@opencode-ai/.ignored_sdk/dist/gen/types.gen.d.ts, description=find Config type, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord
|
|
8
|
+
[Tool bash]: command=grep -B5 -A30 "^export type AgentConfig" node_modules/@opencode-ai/.ignored_sdk/dist/gen/types.gen.d, description=find AgentConfig type, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord
|
|
9
|
+
[Assistant]: Found it! `AgentConfig` has a `prompt` field. Let me check if `config.get()` returns this:
|
|
10
|
+
|
|
11
|
+
[Tool bash]: command=grep -A20 "ConfigGetResponses" node_modules/@opencode-ai/.ignored_sdk/dist/gen/types.gen.d.ts, description=check ConfigGet response, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord
|
|
12
|
+
[Tool bash]: command=grep -A20 "AppAgentsResponses" node_modules/@opencode-ai/.ignored_sdk/dist/gen/types.gen.d.ts, description=check agents response, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord
|
|
13
|
+
[Tool bash]: command=grep -B2 -A30 "^export type Agent = {" node_modules/@opencode-ai/.ignored_sdk/dist/gen/types.gen.d.t, description=check Agent type, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord
|
|
14
|
+
[Assistant]: Found `Agent.prompt` and `AgentConfig.prompt` fields. Let me test if `app.agents()` returns useful prompt info:
|
|
15
|
+
|
|
16
|
+
[Tool bash]: command=cat > /tmp/test-agents.ts << 'EOF' import { OpencodeClient } from '@opencode-ai/sdk' async function, description=test agents and config api, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord, timeout=30000
|
|
17
|
+
[Tool bash]: command=cd discord && cat > /tmp/test-agents.ts << 'EOF' import { OpencodeClient } from '@opencode-ai/sdk', description=test agents api from discord dir, workdir=/Users/morse/Documents/GitHub/kimakivoice/discord, timeout=30000
|
|
18
|
+
[Tool bash]: command=cat > discord/src/test-agents-api.ts << 'EOF' import { OpencodeClient } from '@opencode-ai/sdk' asy, description=test agents api, timeout=30000
|
|
19
|
+
[Tool bash]: command=rm discord/src/test-agents-api.ts, description=cleanup test file
|
|
20
|
+
[Assistant]: found `app.agents()` returns agent-specific prompts:
|
|
21
|
+
|
|
22
|
+
```
|
|
23
|
+
plan: 2719 chars - "We are currently in the PLANNING phase..."
|
|
24
|
+
explore: 871 chars - "You are a file search specialist..."
|
|
25
|
+
title: 1508 chars - "You are a title generator..."
|
|
26
|
+
```
|
|
27
|
+
|
|
28
|
+
**but** these are just agent prompts from config, not the full system prompt. the full system prompt opencode builds includes:
|
|
29
|
+
- agents.md content
|
|
30
|
+
- environment info
|
|
31
|
+
- file tree
|
|
32
|
+
- tool descriptions
|
|
33
|
+
|
|
34
|
+
this full prompt is built dynamically and **not exposed via api**. we can only get the agent-specific custom prompts.
|
|
35
|
+
|
|
36
|
+
**options:**
|
|
37
|
+
1. use `app.agents()` to get agent prompts as partial context
|
|
38
|
+
2. read the AGENTS.md file directly from the project directory (we have it)
|
|
39
|
+
3. request opencode to expose the full system prompt in sdk
|
|
40
|
+
|
|
41
|
+
want me to add AGENTS.md file reading as fallback context?
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
# Reply with exactly: kimaki digital twin ok
|
|
2
|
+
|
|
3
|
+
## Conversation
|
|
4
|
+
|
|
5
|
+
### 👤 User
|
|
6
|
+
|
|
7
|
+
[Current branch: main]
|
|
8
|
+
|
|
9
|
+
[Current branch: main]
|
|
10
|
+
|
|
11
|
+
Reply with exactly: kimaki digital twin ok
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
### 🤖 Assistant (gemini-2.5-flash)
|
|
15
|
+
|
|
16
|
+
|
|
17
|
+
*Completed in 397ms*
|