mirror of
https://github.com/anomalyco/opencode.git
synced 2026-02-01 22:48:16 +00:00
475 lines
15 KiB
TypeScript
475 lines
15 KiB
TypeScript
import type {
|
|
Message,
|
|
Agent,
|
|
Provider,
|
|
Session,
|
|
Part,
|
|
Config,
|
|
Todo,
|
|
Command,
|
|
PermissionRequest,
|
|
QuestionRequest,
|
|
LspStatus,
|
|
McpStatus,
|
|
McpResource,
|
|
FormatterStatus,
|
|
SessionStatus,
|
|
ProviderListResponse,
|
|
ProviderAuthMethod,
|
|
VcsInfo,
|
|
AppSkillsResponse,
|
|
} from "@opencode-ai/sdk/v2"
|
|
import { createStore, produce, reconcile } from "solid-js/store"
|
|
import { useSDK } from "@tui/context/sdk"
|
|
import { Binary } from "@opencode-ai/util/binary"
|
|
import { createSimpleContext } from "./helper"
|
|
import type { Snapshot } from "@/snapshot"
|
|
import { useExit } from "./exit"
|
|
import { useArgs } from "./args"
|
|
import { batch, onMount } from "solid-js"
|
|
import { Log } from "@/util/log"
|
|
import type { Path } from "@opencode-ai/sdk"
|
|
|
|
export const { use: useSync, provider: SyncProvider } = createSimpleContext({
|
|
name: "Sync",
|
|
init: () => {
|
|
const [store, setStore] = createStore<{
|
|
status: "loading" | "partial" | "complete"
|
|
provider: Provider[]
|
|
provider_default: Record<string, string>
|
|
provider_next: ProviderListResponse
|
|
provider_auth: Record<string, ProviderAuthMethod[]>
|
|
agent: Agent[]
|
|
command: Command[]
|
|
skill: AppSkillsResponse
|
|
permission: {
|
|
[sessionID: string]: PermissionRequest[]
|
|
}
|
|
question: {
|
|
[sessionID: string]: QuestionRequest[]
|
|
}
|
|
config: Config
|
|
session: Session[]
|
|
session_status: {
|
|
[sessionID: string]: SessionStatus
|
|
}
|
|
session_diff: {
|
|
[sessionID: string]: Snapshot.FileDiff[]
|
|
}
|
|
todo: {
|
|
[sessionID: string]: Todo[]
|
|
}
|
|
message: {
|
|
[sessionID: string]: Message[]
|
|
}
|
|
part: {
|
|
[messageID: string]: Part[]
|
|
}
|
|
lsp: LspStatus[]
|
|
mcp: {
|
|
[key: string]: McpStatus
|
|
}
|
|
mcp_resource: {
|
|
[key: string]: McpResource
|
|
}
|
|
formatter: FormatterStatus[]
|
|
vcs: VcsInfo | undefined
|
|
path: Path
|
|
}>({
|
|
provider_next: {
|
|
all: [],
|
|
default: {},
|
|
connected: [],
|
|
},
|
|
provider_auth: {},
|
|
config: {},
|
|
status: "loading",
|
|
agent: [],
|
|
permission: {},
|
|
question: {},
|
|
command: [],
|
|
skill: [],
|
|
provider: [],
|
|
provider_default: {},
|
|
session: [],
|
|
session_status: {},
|
|
session_diff: {},
|
|
todo: {},
|
|
message: {},
|
|
part: {},
|
|
lsp: [],
|
|
mcp: {},
|
|
mcp_resource: {},
|
|
formatter: [],
|
|
vcs: undefined,
|
|
path: { state: "", config: "", worktree: "", directory: "" },
|
|
})
|
|
|
|
const sdk = useSDK()
|
|
|
|
sdk.event.listen((e) => {
|
|
const event = e.details
|
|
switch (event.type) {
|
|
case "server.instance.disposed":
|
|
bootstrap()
|
|
break
|
|
case "permission.replied": {
|
|
const requests = store.permission[event.properties.sessionID]
|
|
if (!requests) break
|
|
const match = Binary.search(requests, event.properties.requestID, (r) => r.id)
|
|
if (!match.found) break
|
|
setStore(
|
|
"permission",
|
|
event.properties.sessionID,
|
|
produce((draft) => {
|
|
draft.splice(match.index, 1)
|
|
}),
|
|
)
|
|
break
|
|
}
|
|
|
|
case "permission.asked": {
|
|
const request = event.properties
|
|
const requests = store.permission[request.sessionID]
|
|
if (!requests) {
|
|
setStore("permission", request.sessionID, [request])
|
|
break
|
|
}
|
|
const match = Binary.search(requests, request.id, (r) => r.id)
|
|
if (match.found) {
|
|
setStore("permission", request.sessionID, match.index, reconcile(request))
|
|
break
|
|
}
|
|
setStore(
|
|
"permission",
|
|
request.sessionID,
|
|
produce((draft) => {
|
|
draft.splice(match.index, 0, request)
|
|
}),
|
|
)
|
|
break
|
|
}
|
|
|
|
case "question.replied":
|
|
case "question.rejected": {
|
|
const requests = store.question[event.properties.sessionID]
|
|
if (!requests) break
|
|
const match = Binary.search(requests, event.properties.requestID, (r) => r.id)
|
|
if (!match.found) break
|
|
setStore(
|
|
"question",
|
|
event.properties.sessionID,
|
|
produce((draft) => {
|
|
draft.splice(match.index, 1)
|
|
}),
|
|
)
|
|
break
|
|
}
|
|
|
|
case "question.asked": {
|
|
const request = event.properties
|
|
const requests = store.question[request.sessionID]
|
|
if (!requests) {
|
|
setStore("question", request.sessionID, [request])
|
|
break
|
|
}
|
|
const match = Binary.search(requests, request.id, (r) => r.id)
|
|
if (match.found) {
|
|
setStore("question", request.sessionID, match.index, reconcile(request))
|
|
break
|
|
}
|
|
setStore(
|
|
"question",
|
|
request.sessionID,
|
|
produce((draft) => {
|
|
draft.splice(match.index, 0, request)
|
|
}),
|
|
)
|
|
break
|
|
}
|
|
|
|
case "todo.updated":
|
|
setStore("todo", event.properties.sessionID, event.properties.todos)
|
|
break
|
|
|
|
case "session.diff":
|
|
setStore("session_diff", event.properties.sessionID, event.properties.diff)
|
|
break
|
|
|
|
case "session.deleted": {
|
|
const result = Binary.search(store.session, event.properties.info.id, (s) => s.id)
|
|
if (result.found) {
|
|
setStore(
|
|
"session",
|
|
produce((draft) => {
|
|
draft.splice(result.index, 1)
|
|
}),
|
|
)
|
|
}
|
|
break
|
|
}
|
|
case "session.updated": {
|
|
const result = Binary.search(store.session, event.properties.info.id, (s) => s.id)
|
|
if (result.found) {
|
|
setStore("session", result.index, reconcile(event.properties.info))
|
|
break
|
|
}
|
|
setStore(
|
|
"session",
|
|
produce((draft) => {
|
|
draft.splice(result.index, 0, event.properties.info)
|
|
}),
|
|
)
|
|
break
|
|
}
|
|
|
|
case "session.status": {
|
|
setStore("session_status", event.properties.sessionID, event.properties.status)
|
|
break
|
|
}
|
|
|
|
case "message.updated": {
|
|
const messages = store.message[event.properties.info.sessionID]
|
|
if (!messages) {
|
|
setStore("message", event.properties.info.sessionID, [event.properties.info])
|
|
break
|
|
}
|
|
const result = Binary.search(messages, event.properties.info.id, (m) => m.id)
|
|
if (result.found) {
|
|
setStore("message", event.properties.info.sessionID, result.index, reconcile(event.properties.info))
|
|
break
|
|
}
|
|
setStore(
|
|
"message",
|
|
event.properties.info.sessionID,
|
|
produce((draft) => {
|
|
draft.splice(result.index, 0, event.properties.info)
|
|
}),
|
|
)
|
|
const updated = store.message[event.properties.info.sessionID]
|
|
if (updated.length > 100) {
|
|
const oldest = updated[0]
|
|
batch(() => {
|
|
setStore(
|
|
"message",
|
|
event.properties.info.sessionID,
|
|
produce((draft) => {
|
|
draft.shift()
|
|
}),
|
|
)
|
|
setStore(
|
|
"part",
|
|
produce((draft) => {
|
|
delete draft[oldest.id]
|
|
}),
|
|
)
|
|
})
|
|
}
|
|
break
|
|
}
|
|
case "message.removed": {
|
|
const messages = store.message[event.properties.sessionID]
|
|
const result = Binary.search(messages, event.properties.messageID, (m) => m.id)
|
|
if (result.found) {
|
|
setStore(
|
|
"message",
|
|
event.properties.sessionID,
|
|
produce((draft) => {
|
|
draft.splice(result.index, 1)
|
|
}),
|
|
)
|
|
}
|
|
break
|
|
}
|
|
case "message.part.updated": {
|
|
const parts = store.part[event.properties.part.messageID]
|
|
if (!parts) {
|
|
setStore("part", event.properties.part.messageID, [event.properties.part])
|
|
break
|
|
}
|
|
const result = Binary.search(parts, event.properties.part.id, (p) => p.id)
|
|
if (result.found) {
|
|
setStore("part", event.properties.part.messageID, result.index, reconcile(event.properties.part))
|
|
break
|
|
}
|
|
setStore(
|
|
"part",
|
|
event.properties.part.messageID,
|
|
produce((draft) => {
|
|
draft.splice(result.index, 0, event.properties.part)
|
|
}),
|
|
)
|
|
break
|
|
}
|
|
|
|
case "message.part.removed": {
|
|
const parts = store.part[event.properties.messageID]
|
|
const result = Binary.search(parts, event.properties.partID, (p) => p.id)
|
|
if (result.found)
|
|
setStore(
|
|
"part",
|
|
event.properties.messageID,
|
|
produce((draft) => {
|
|
draft.splice(result.index, 1)
|
|
}),
|
|
)
|
|
break
|
|
}
|
|
|
|
case "lsp.updated": {
|
|
sdk.client.lsp.status().then((x) => setStore("lsp", x.data!))
|
|
break
|
|
}
|
|
|
|
case "vcs.branch.updated": {
|
|
setStore("vcs", { branch: event.properties.branch })
|
|
break
|
|
}
|
|
}
|
|
})
|
|
|
|
const exit = useExit()
|
|
const args = useArgs()
|
|
|
|
async function bootstrap() {
|
|
console.log("bootstrapping")
|
|
const start = Date.now() - 30 * 24 * 60 * 60 * 1000
|
|
const sessionListPromise = sdk.client.session
|
|
.list({ start: start })
|
|
.then((x) => (x.data ?? []).toSorted((a, b) => a.id.localeCompare(b.id)))
|
|
|
|
// blocking - include session.list when continuing a session
|
|
const providersPromise = sdk.client.config.providers({}, { throwOnError: true })
|
|
const providerListPromise = sdk.client.provider.list({}, { throwOnError: true })
|
|
const agentsPromise = sdk.client.app.agents({}, { throwOnError: true })
|
|
const configPromise = sdk.client.config.get({}, { throwOnError: true })
|
|
const blockingRequests: Promise<unknown>[] = [
|
|
providersPromise,
|
|
providerListPromise,
|
|
agentsPromise,
|
|
configPromise,
|
|
...(args.continue ? [sessionListPromise] : []),
|
|
]
|
|
|
|
await Promise.all(blockingRequests)
|
|
.then(() => {
|
|
const providersResponse = providersPromise.then((x) => x.data!)
|
|
const providerListResponse = providerListPromise.then((x) => x.data!)
|
|
const agentsResponse = agentsPromise.then((x) => x.data ?? [])
|
|
const configResponse = configPromise.then((x) => x.data!)
|
|
const sessionListResponse = args.continue ? sessionListPromise : undefined
|
|
|
|
return Promise.all([
|
|
providersResponse,
|
|
providerListResponse,
|
|
agentsResponse,
|
|
configResponse,
|
|
...(sessionListResponse ? [sessionListResponse] : []),
|
|
]).then((responses) => {
|
|
const providers = responses[0]
|
|
const providerList = responses[1]
|
|
const agents = responses[2]
|
|
const config = responses[3]
|
|
const sessions = responses[4]
|
|
|
|
batch(() => {
|
|
setStore("provider", reconcile(providers.providers))
|
|
setStore("provider_default", reconcile(providers.default))
|
|
setStore("provider_next", reconcile(providerList))
|
|
setStore("agent", reconcile(agents))
|
|
setStore("config", reconcile(config))
|
|
if (sessions !== undefined) setStore("session", reconcile(sessions))
|
|
})
|
|
})
|
|
})
|
|
.then(() => {
|
|
if (store.status !== "complete") setStore("status", "partial")
|
|
// non-blocking
|
|
Promise.all([
|
|
...(args.continue ? [] : [sessionListPromise.then((sessions) => setStore("session", reconcile(sessions)))]),
|
|
sdk.client.command.list().then((x) => setStore("command", reconcile(x.data ?? []))),
|
|
sdk.client.app.skills().then((x) => setStore("skill", reconcile(x.data ?? []))),
|
|
sdk.client.lsp.status().then((x) => setStore("lsp", reconcile(x.data!))),
|
|
sdk.client.mcp.status().then((x) => setStore("mcp", reconcile(x.data!))),
|
|
sdk.client.experimental.resource.list().then((x) => setStore("mcp_resource", reconcile(x.data ?? {}))),
|
|
sdk.client.formatter.status().then((x) => setStore("formatter", reconcile(x.data!))),
|
|
sdk.client.session.status().then((x) => {
|
|
setStore("session_status", reconcile(x.data!))
|
|
}),
|
|
sdk.client.provider.auth().then((x) => setStore("provider_auth", reconcile(x.data ?? {}))),
|
|
sdk.client.vcs.get().then((x) => setStore("vcs", reconcile(x.data))),
|
|
sdk.client.path.get().then((x) => setStore("path", reconcile(x.data!))),
|
|
]).then(() => {
|
|
setStore("status", "complete")
|
|
})
|
|
})
|
|
.catch(async (e) => {
|
|
Log.Default.error("tui bootstrap failed", {
|
|
error: e instanceof Error ? e.message : String(e),
|
|
name: e instanceof Error ? e.name : undefined,
|
|
stack: e instanceof Error ? e.stack : undefined,
|
|
})
|
|
await exit(e)
|
|
})
|
|
}
|
|
|
|
onMount(() => {
|
|
bootstrap()
|
|
})
|
|
|
|
const fullSyncedSessions = new Set<string>()
|
|
const result = {
|
|
data: store,
|
|
set: setStore,
|
|
get status() {
|
|
return store.status
|
|
},
|
|
get ready() {
|
|
return store.status !== "loading"
|
|
},
|
|
session: {
|
|
get(sessionID: string) {
|
|
const match = Binary.search(store.session, sessionID, (s) => s.id)
|
|
if (match.found) return store.session[match.index]
|
|
return undefined
|
|
},
|
|
status(sessionID: string) {
|
|
const session = result.session.get(sessionID)
|
|
if (!session) return "idle"
|
|
if (session.time.compacting) return "compacting"
|
|
const messages = store.message[sessionID] ?? []
|
|
const last = messages.at(-1)
|
|
if (!last) return "idle"
|
|
if (last.role === "user") return "working"
|
|
return last.time.completed ? "idle" : "working"
|
|
},
|
|
async sync(sessionID: string) {
|
|
if (fullSyncedSessions.has(sessionID)) return
|
|
const [session, messages, todo, diff] = await Promise.all([
|
|
sdk.client.session.get({ sessionID }, { throwOnError: true }),
|
|
sdk.client.session.messages({ sessionID, limit: 100 }),
|
|
sdk.client.session.todo({ sessionID }),
|
|
sdk.client.session.diff({ sessionID }),
|
|
])
|
|
setStore(
|
|
produce((draft) => {
|
|
const match = Binary.search(draft.session, sessionID, (s) => s.id)
|
|
if (match.found) draft.session[match.index] = session.data!
|
|
if (!match.found) draft.session.splice(match.index, 0, session.data!)
|
|
draft.todo[sessionID] = todo.data ?? []
|
|
draft.message[sessionID] = messages.data!.map((x) => x.info)
|
|
for (const message of messages.data!) {
|
|
draft.part[message.info.id] = message.parts
|
|
}
|
|
draft.session_diff[sessionID] = diff.data ?? []
|
|
}),
|
|
)
|
|
fullSyncedSessions.add(sessionID)
|
|
},
|
|
},
|
|
bootstrap,
|
|
}
|
|
return result
|
|
},
|
|
})
|