| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943 |
- import {
- type Message,
- type Agent,
- type Session,
- type Part,
- type Config,
- type Path,
- type Project,
- type FileDiff,
- type Todo,
- type SessionStatus,
- type ProviderListResponse,
- type ProviderAuthResponse,
- type Command,
- type McpStatus,
- type LspStatus,
- type VcsInfo,
- type PermissionRequest,
- type QuestionRequest,
- createOpencodeClient,
- } from "@opencode-ai/sdk/v2/client"
- import { createStore, produce, reconcile, type SetStoreFunction, type Store } from "solid-js/store"
- import { Binary } from "@opencode-ai/util/binary"
- import { retry } from "@opencode-ai/util/retry"
- import { useGlobalSDK } from "./global-sdk"
- import { ErrorPage, type InitError } from "../pages/error"
- import {
- batch,
- createContext,
- createEffect,
- untrack,
- getOwner,
- runWithOwner,
- useContext,
- onCleanup,
- onMount,
- type Accessor,
- type ParentProps,
- Switch,
- Match,
- } from "solid-js"
- import { showToast } from "@opencode-ai/ui/toast"
- import { getFilename } from "@opencode-ai/util/path"
- import { usePlatform } from "./platform"
- import { useLanguage } from "@/context/language"
- import { Persist, persisted } from "@/utils/persist"
- type ProjectMeta = {
- name?: string
- icon?: {
- override?: string
- color?: string
- }
- commands?: {
- start?: string
- }
- }
- type State = {
- status: "loading" | "partial" | "complete"
- agent: Agent[]
- command: Command[]
- project: string
- projectMeta: ProjectMeta | undefined
- icon: string | undefined
- provider: ProviderListResponse
- config: Config
- path: Path
- session: Session[]
- sessionTotal: number
- session_status: {
- [sessionID: string]: SessionStatus
- }
- session_diff: {
- [sessionID: string]: FileDiff[]
- }
- todo: {
- [sessionID: string]: Todo[]
- }
- permission: {
- [sessionID: string]: PermissionRequest[]
- }
- question: {
- [sessionID: string]: QuestionRequest[]
- }
- mcp: {
- [name: string]: McpStatus
- }
- lsp: LspStatus[]
- vcs: VcsInfo | undefined
- limit: number
- message: {
- [sessionID: string]: Message[]
- }
- part: {
- [messageID: string]: Part[]
- }
- }
- type VcsCache = {
- store: Store<{ value: VcsInfo | undefined }>
- setStore: SetStoreFunction<{ value: VcsInfo | undefined }>
- ready: Accessor<boolean>
- }
- type MetaCache = {
- store: Store<{ value: ProjectMeta | undefined }>
- setStore: SetStoreFunction<{ value: ProjectMeta | undefined }>
- ready: Accessor<boolean>
- }
- type IconCache = {
- store: Store<{ value: string | undefined }>
- setStore: SetStoreFunction<{ value: string | undefined }>
- ready: Accessor<boolean>
- }
- type ChildOptions = {
- bootstrap?: boolean
- }
- function normalizeProviderList(input: ProviderListResponse): ProviderListResponse {
- return {
- ...input,
- all: input.all.map((provider) => ({
- ...provider,
- models: Object.fromEntries(Object.entries(provider.models).filter(([, info]) => info.status !== "deprecated")),
- })),
- }
- }
- function createGlobalSync() {
- const globalSDK = useGlobalSDK()
- const platform = usePlatform()
- const language = useLanguage()
- const owner = getOwner()
- if (!owner) throw new Error("GlobalSync must be created within owner")
- const vcsCache = new Map<string, VcsCache>()
- const metaCache = new Map<string, MetaCache>()
- const iconCache = new Map<string, IconCache>()
- const sdkCache = new Map<string, ReturnType<typeof createOpencodeClient>>()
- const sdkFor = (directory: string) => {
- const cached = sdkCache.get(directory)
- if (cached) return cached
- const sdk = createOpencodeClient({
- baseUrl: globalSDK.url,
- fetch: platform.fetch,
- directory,
- throwOnError: true,
- })
- sdkCache.set(directory, sdk)
- return sdk
- }
- const [projectCache, setProjectCache, , projectCacheReady] = persisted(
- Persist.global("globalSync.project", ["globalSync.project.v1"]),
- createStore({ value: [] as Project[] }),
- )
- const sanitizeProject = (project: Project) => {
- if (!project.icon?.url && !project.icon?.override) return project
- return {
- ...project,
- icon: {
- ...project.icon,
- url: undefined,
- override: undefined,
- },
- }
- }
- const [globalStore, setGlobalStore] = createStore<{
- ready: boolean
- error?: InitError
- path: Path
- project: Project[]
- provider: ProviderListResponse
- provider_auth: ProviderAuthResponse
- config: Config
- reload: undefined | "pending" | "complete"
- }>({
- ready: false,
- path: { state: "", config: "", worktree: "", directory: "", home: "" },
- project: projectCache.value,
- provider: { all: [], connected: [], default: {} },
- provider_auth: {},
- config: {},
- reload: undefined,
- })
- let bootstrapQueue: string[] = []
- createEffect(() => {
- if (!projectCacheReady()) return
- if (globalStore.project.length !== 0) return
- const cached = projectCache.value
- if (cached.length === 0) return
- setGlobalStore("project", cached)
- })
- createEffect(() => {
- if (!projectCacheReady()) return
- const projects = globalStore.project
- if (projects.length === 0) {
- const cachedLength = untrack(() => projectCache.value.length)
- if (cachedLength !== 0) return
- }
- setProjectCache("value", projects.map(sanitizeProject))
- })
- createEffect(() => {
- if (globalStore.reload !== "complete") return
- if (bootstrapQueue.length) {
- for (const directory of bootstrapQueue) {
- bootstrapInstance(directory)
- }
- bootstrap()
- }
- bootstrapQueue = []
- setGlobalStore("reload", undefined)
- })
- const children: Record<string, [Store<State>, SetStoreFunction<State>]> = {}
- const booting = new Map<string, Promise<void>>()
- const sessionLoads = new Map<string, Promise<void>>()
- const sessionMeta = new Map<string, { limit: number }>()
- const sessionRecentWindow = 4 * 60 * 60 * 1000
- const sessionRecentLimit = 50
- function sessionUpdatedAt(session: Session) {
- return session.time.updated ?? session.time.created
- }
- function compareSessionRecent(a: Session, b: Session) {
- const aUpdated = sessionUpdatedAt(a)
- const bUpdated = sessionUpdatedAt(b)
- if (aUpdated !== bUpdated) return bUpdated - aUpdated
- return a.id.localeCompare(b.id)
- }
- function takeRecentSessions(sessions: Session[], limit: number, cutoff: number) {
- if (limit <= 0) return [] as Session[]
- const selected: Session[] = []
- const seen = new Set<string>()
- for (const session of sessions) {
- if (!session?.id) continue
- if (seen.has(session.id)) continue
- seen.add(session.id)
- if (sessionUpdatedAt(session) <= cutoff) continue
- const index = selected.findIndex((x) => compareSessionRecent(session, x) < 0)
- if (index === -1) selected.push(session)
- if (index !== -1) selected.splice(index, 0, session)
- if (selected.length > limit) selected.pop()
- }
- return selected
- }
- function trimSessions(input: Session[], options: { limit: number; permission: Record<string, PermissionRequest[]> }) {
- const limit = Math.max(0, options.limit)
- const cutoff = Date.now() - sessionRecentWindow
- const all = input
- .filter((s) => !!s?.id)
- .filter((s) => !s.time?.archived)
- .sort((a, b) => a.id.localeCompare(b.id))
- const roots = all.filter((s) => !s.parentID)
- const children = all.filter((s) => !!s.parentID)
- const base = roots.slice(0, limit)
- const recent = takeRecentSessions(roots.slice(limit), sessionRecentLimit, cutoff)
- const keepRoots = [...base, ...recent]
- const keepRootIds = new Set(keepRoots.map((s) => s.id))
- const keepChildren = children.filter((s) => {
- if (s.parentID && keepRootIds.has(s.parentID)) return true
- const perms = options.permission[s.id] ?? []
- if (perms.length > 0) return true
- return sessionUpdatedAt(s) > cutoff
- })
- return [...keepRoots, ...keepChildren].sort((a, b) => a.id.localeCompare(b.id))
- }
- function ensureChild(directory: string) {
- if (!directory) console.error("No directory provided")
- if (!children[directory]) {
- const vcs = runWithOwner(owner, () =>
- persisted(
- Persist.workspace(directory, "vcs", ["vcs.v1"]),
- createStore({ value: undefined as VcsInfo | undefined }),
- ),
- )
- if (!vcs) throw new Error("Failed to create persisted cache")
- const vcsStore = vcs[0]
- const vcsReady = vcs[3]
- vcsCache.set(directory, { store: vcsStore, setStore: vcs[1], ready: vcsReady })
- const meta = runWithOwner(owner, () =>
- persisted(
- Persist.workspace(directory, "project", ["project.v1"]),
- createStore({ value: undefined as ProjectMeta | undefined }),
- ),
- )
- if (!meta) throw new Error("Failed to create persisted project metadata")
- metaCache.set(directory, { store: meta[0], setStore: meta[1], ready: meta[3] })
- const icon = runWithOwner(owner, () =>
- persisted(
- Persist.workspace(directory, "icon", ["icon.v1"]),
- createStore({ value: undefined as string | undefined }),
- ),
- )
- if (!icon) throw new Error("Failed to create persisted project icon")
- iconCache.set(directory, { store: icon[0], setStore: icon[1], ready: icon[3] })
- const init = () => {
- const child = createStore<State>({
- project: "",
- projectMeta: meta[0].value,
- icon: icon[0].value,
- provider: { all: [], connected: [], default: {} },
- config: {},
- path: { state: "", config: "", worktree: "", directory: "", home: "" },
- status: "loading" as const,
- agent: [],
- command: [],
- session: [],
- sessionTotal: 0,
- session_status: {},
- session_diff: {},
- todo: {},
- permission: {},
- question: {},
- mcp: {},
- lsp: [],
- vcs: vcsStore.value,
- limit: 5,
- message: {},
- part: {},
- })
- children[directory] = child
- createEffect(() => {
- if (!vcsReady()) return
- const cached = vcsStore.value
- if (!cached?.branch) return
- child[1]("vcs", (value) => value ?? cached)
- })
- createEffect(() => {
- child[1]("projectMeta", meta[0].value)
- })
- createEffect(() => {
- child[1]("icon", icon[0].value)
- })
- }
- runWithOwner(owner, init)
- }
- const childStore = children[directory]
- if (!childStore) throw new Error("Failed to create store")
- return childStore
- }
- function child(directory: string, options: ChildOptions = {}) {
- const childStore = ensureChild(directory)
- const shouldBootstrap = options.bootstrap ?? true
- if (shouldBootstrap && childStore[0].status === "loading") {
- void bootstrapInstance(directory)
- }
- return childStore
- }
- async function loadSessions(directory: string) {
- const pending = sessionLoads.get(directory)
- if (pending) return pending
- const [store, setStore] = child(directory, { bootstrap: false })
- const meta = sessionMeta.get(directory)
- if (meta && meta.limit >= store.limit) {
- const next = trimSessions(store.session, { limit: store.limit, permission: store.permission })
- if (next.length !== store.session.length) {
- setStore("session", reconcile(next, { key: "id" }))
- }
- return
- }
- const promise = globalSDK.client.session
- .list({ directory, roots: true })
- .then((x) => {
- const nonArchived = (x.data ?? [])
- .filter((s) => !!s?.id)
- .filter((s) => !s.time?.archived)
- .sort((a, b) => a.id.localeCompare(b.id))
- // Read the current limit at resolve-time so callers that bump the limit while
- // a request is in-flight still get the expanded result.
- const limit = store.limit
- const children = store.session.filter((s) => !!s.parentID)
- const sessions = trimSessions([...nonArchived, ...children], { limit, permission: store.permission })
- // Store total session count (used for "load more" pagination)
- setStore("sessionTotal", nonArchived.length)
- setStore("session", reconcile(sessions, { key: "id" }))
- sessionMeta.set(directory, { limit })
- })
- .catch((err) => {
- console.error("Failed to load sessions", err)
- const project = getFilename(directory)
- showToast({ title: language.t("toast.session.listFailed.title", { project }), description: err.message })
- })
- sessionLoads.set(directory, promise)
- promise.finally(() => {
- sessionLoads.delete(directory)
- })
- return promise
- }
- async function bootstrapInstance(directory: string) {
- if (!directory) return
- const pending = booting.get(directory)
- if (pending) return pending
- const promise = (async () => {
- const [store, setStore] = ensureChild(directory)
- const cache = vcsCache.get(directory)
- if (!cache) return
- const meta = metaCache.get(directory)
- if (!meta) return
- const sdk = sdkFor(directory)
- setStore("status", "loading")
- // projectMeta is synced from persisted storage in ensureChild.
- // vcs is seeded from persisted storage in ensureChild.
- const blockingRequests = {
- project: () => sdk.project.current().then((x) => setStore("project", x.data!.id)),
- provider: () =>
- sdk.provider.list().then((x) => {
- setStore("provider", normalizeProviderList(x.data!))
- }),
- agent: () => sdk.app.agents().then((x) => setStore("agent", x.data ?? [])),
- config: () => sdk.config.get().then((x) => setStore("config", x.data!)),
- }
- try {
- await Promise.all(Object.values(blockingRequests).map((p) => retry(p)))
- } catch (err) {
- console.error("Failed to bootstrap instance", err)
- const project = getFilename(directory)
- const message = err instanceof Error ? err.message : String(err)
- showToast({ title: `Failed to reload ${project}`, description: message })
- setStore("status", "partial")
- return
- }
- if (store.status !== "complete") setStore("status", "partial")
- Promise.all([
- sdk.path.get().then((x) => setStore("path", x.data!)),
- sdk.command.list().then((x) => setStore("command", x.data ?? [])),
- sdk.session.status().then((x) => setStore("session_status", x.data!)),
- loadSessions(directory),
- sdk.mcp.status().then((x) => setStore("mcp", x.data!)),
- sdk.lsp.status().then((x) => setStore("lsp", x.data!)),
- sdk.vcs.get().then((x) => {
- const next = x.data ?? store.vcs
- setStore("vcs", next)
- if (next?.branch) cache.setStore("value", next)
- }),
- sdk.permission.list().then((x) => {
- const grouped: Record<string, PermissionRequest[]> = {}
- for (const perm of x.data ?? []) {
- if (!perm?.id || !perm.sessionID) continue
- const existing = grouped[perm.sessionID]
- if (existing) {
- existing.push(perm)
- continue
- }
- grouped[perm.sessionID] = [perm]
- }
- batch(() => {
- for (const sessionID of Object.keys(store.permission)) {
- if (grouped[sessionID]) continue
- setStore("permission", sessionID, [])
- }
- for (const [sessionID, permissions] of Object.entries(grouped)) {
- setStore(
- "permission",
- sessionID,
- reconcile(
- permissions.filter((p) => !!p?.id).sort((a, b) => a.id.localeCompare(b.id)),
- { key: "id" },
- ),
- )
- }
- })
- }),
- sdk.question.list().then((x) => {
- const grouped: Record<string, QuestionRequest[]> = {}
- for (const question of x.data ?? []) {
- if (!question?.id || !question.sessionID) continue
- const existing = grouped[question.sessionID]
- if (existing) {
- existing.push(question)
- continue
- }
- grouped[question.sessionID] = [question]
- }
- batch(() => {
- for (const sessionID of Object.keys(store.question)) {
- if (grouped[sessionID]) continue
- setStore("question", sessionID, [])
- }
- for (const [sessionID, questions] of Object.entries(grouped)) {
- setStore(
- "question",
- sessionID,
- reconcile(
- questions.filter((q) => !!q?.id).sort((a, b) => a.id.localeCompare(b.id)),
- { key: "id" },
- ),
- )
- }
- })
- }),
- ]).then(() => {
- setStore("status", "complete")
- })
- })()
- booting.set(directory, promise)
- promise.finally(() => {
- booting.delete(directory)
- })
- return promise
- }
- const unsub = globalSDK.event.listen((e) => {
- const directory = e.name
- const event = e.details
- if (directory === "global") {
- switch (event?.type) {
- case "global.disposed": {
- if (globalStore.reload) return
- bootstrap()
- break
- }
- case "project.updated": {
- const result = Binary.search(globalStore.project, event.properties.id, (s) => s.id)
- if (result.found) {
- setGlobalStore("project", result.index, reconcile(event.properties))
- return
- }
- setGlobalStore(
- "project",
- produce((draft) => {
- draft.splice(result.index, 0, event.properties)
- }),
- )
- break
- }
- }
- return
- }
- const existing = children[directory]
- if (!existing) return
- const [store, setStore] = existing
- switch (event.type) {
- case "server.instance.disposed": {
- if (globalStore.reload) {
- bootstrapQueue.push(directory)
- return
- }
- bootstrapInstance(directory)
- break
- }
- case "session.created": {
- const info = event.properties.info
- const result = Binary.search(store.session, info.id, (s) => s.id)
- if (result.found) {
- setStore("session", result.index, reconcile(info))
- break
- }
- const next = store.session.slice()
- next.splice(result.index, 0, info)
- const trimmed = trimSessions(next, { limit: store.limit, permission: store.permission })
- setStore("session", reconcile(trimmed, { key: "id" }))
- if (!info.parentID) {
- setStore("sessionTotal", (value) => value + 1)
- }
- break
- }
- case "session.updated": {
- const info = event.properties.info
- const result = Binary.search(store.session, info.id, (s) => s.id)
- if (info.time.archived) {
- if (result.found) {
- setStore(
- "session",
- produce((draft) => {
- draft.splice(result.index, 1)
- }),
- )
- }
- if (info.parentID) break
- setStore("sessionTotal", (value) => Math.max(0, value - 1))
- break
- }
- if (result.found) {
- setStore("session", result.index, reconcile(info))
- break
- }
- const next = store.session.slice()
- next.splice(result.index, 0, info)
- const trimmed = trimSessions(next, { limit: store.limit, permission: store.permission })
- setStore("session", reconcile(trimmed, { key: "id" }))
- 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)
- }),
- )
- }
- if (event.properties.info.parentID) break
- setStore("sessionTotal", (value) => Math.max(0, value - 1))
- break
- }
- case "session.diff":
- setStore("session_diff", event.properties.sessionID, reconcile(event.properties.diff, { key: "file" }))
- break
- case "todo.updated":
- setStore("todo", event.properties.sessionID, reconcile(event.properties.todos, { key: "id" }))
- break
- case "session.status": {
- setStore("session_status", event.properties.sessionID, reconcile(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)
- }),
- )
- break
- }
- case "message.removed": {
- const messages = store.message[event.properties.sessionID]
- if (!messages) break
- 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 part = event.properties.part
- const parts = store.part[part.messageID]
- if (!parts) {
- setStore("part", part.messageID, [part])
- break
- }
- const result = Binary.search(parts, part.id, (p) => p.id)
- if (result.found) {
- setStore("part", part.messageID, result.index, reconcile(part))
- break
- }
- setStore(
- "part",
- part.messageID,
- produce((draft) => {
- draft.splice(result.index, 0, part)
- }),
- )
- break
- }
- case "message.part.removed": {
- const parts = store.part[event.properties.messageID]
- if (!parts) break
- 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 "vcs.branch.updated": {
- const next = { branch: event.properties.branch }
- setStore("vcs", next)
- const cache = vcsCache.get(directory)
- if (cache) cache.setStore("value", next)
- break
- }
- case "permission.asked": {
- const sessionID = event.properties.sessionID
- const permissions = store.permission[sessionID]
- if (!permissions) {
- setStore("permission", sessionID, [event.properties])
- break
- }
- const result = Binary.search(permissions, event.properties.id, (p) => p.id)
- if (result.found) {
- setStore("permission", sessionID, result.index, reconcile(event.properties))
- break
- }
- setStore(
- "permission",
- sessionID,
- produce((draft) => {
- draft.splice(result.index, 0, event.properties)
- }),
- )
- break
- }
- case "permission.replied": {
- const permissions = store.permission[event.properties.sessionID]
- if (!permissions) break
- const result = Binary.search(permissions, event.properties.requestID, (p) => p.id)
- if (!result.found) break
- setStore(
- "permission",
- event.properties.sessionID,
- produce((draft) => {
- draft.splice(result.index, 1)
- }),
- )
- break
- }
- case "question.asked": {
- const sessionID = event.properties.sessionID
- const questions = store.question[sessionID]
- if (!questions) {
- setStore("question", sessionID, [event.properties])
- break
- }
- const result = Binary.search(questions, event.properties.id, (q) => q.id)
- if (result.found) {
- setStore("question", sessionID, result.index, reconcile(event.properties))
- break
- }
- setStore(
- "question",
- sessionID,
- produce((draft) => {
- draft.splice(result.index, 0, event.properties)
- }),
- )
- break
- }
- case "question.replied":
- case "question.rejected": {
- const questions = store.question[event.properties.sessionID]
- if (!questions) break
- const result = Binary.search(questions, event.properties.requestID, (q) => q.id)
- if (!result.found) break
- setStore(
- "question",
- event.properties.sessionID,
- produce((draft) => {
- draft.splice(result.index, 1)
- }),
- )
- break
- }
- case "lsp.updated": {
- sdkFor(directory)
- .lsp.status()
- .then((x) => setStore("lsp", x.data ?? []))
- break
- }
- }
- })
- onCleanup(unsub)
- async function bootstrap() {
- const health = await globalSDK.client.global
- .health()
- .then((x) => x.data)
- .catch(() => undefined)
- if (!health?.healthy) {
- setGlobalStore("error", new Error(language.t("error.globalSync.connectFailed", { url: globalSDK.url })))
- return
- }
- return Promise.all([
- retry(() =>
- globalSDK.client.path.get().then((x) => {
- setGlobalStore("path", x.data!)
- }),
- ),
- retry(() =>
- globalSDK.client.config.get().then((x) => {
- setGlobalStore("config", x.data!)
- }),
- ),
- retry(() =>
- globalSDK.client.project.list().then(async (x) => {
- const projects = (x.data ?? [])
- .filter((p) => !!p?.id)
- .filter((p) => !!p.worktree && !p.worktree.includes("opencode-test"))
- .slice()
- .sort((a, b) => a.id.localeCompare(b.id))
- setGlobalStore("project", projects)
- }),
- ),
- retry(() =>
- globalSDK.client.provider.list().then((x) => {
- setGlobalStore("provider", normalizeProviderList(x.data!))
- }),
- ),
- retry(() =>
- globalSDK.client.provider.auth().then((x) => {
- setGlobalStore("provider_auth", x.data ?? {})
- }),
- ),
- ])
- .then(() => setGlobalStore("ready", true))
- .catch((e) => setGlobalStore("error", e))
- }
- onMount(() => {
- bootstrap()
- })
- function projectMeta(directory: string, patch: ProjectMeta) {
- const [store, setStore] = ensureChild(directory)
- const cached = metaCache.get(directory)
- if (!cached) return
- const previous = store.projectMeta ?? {}
- const icon = patch.icon ? { ...(previous.icon ?? {}), ...patch.icon } : previous.icon
- const commands = patch.commands ? { ...(previous.commands ?? {}), ...patch.commands } : previous.commands
- const next = {
- ...previous,
- ...patch,
- icon,
- commands,
- }
- cached.setStore("value", next)
- setStore("projectMeta", next)
- }
- function projectIcon(directory: string, value: string | undefined) {
- const [store, setStore] = ensureChild(directory)
- const cached = iconCache.get(directory)
- if (!cached) return
- if (store.icon === value) return
- cached.setStore("value", value)
- setStore("icon", value)
- }
- return {
- data: globalStore,
- set: setGlobalStore,
- get ready() {
- return globalStore.ready
- },
- get error() {
- return globalStore.error
- },
- child,
- bootstrap,
- updateConfig: async (config: Config) => {
- setGlobalStore("reload", "pending")
- const response = await globalSDK.client.config.update({ config })
- setTimeout(() => {
- setGlobalStore("reload", "complete")
- }, 1000)
- return response
- },
- project: {
- loadSessions,
- meta: projectMeta,
- icon: projectIcon,
- },
- }
- }
- const GlobalSyncContext = createContext<ReturnType<typeof createGlobalSync>>()
- export function GlobalSyncProvider(props: ParentProps) {
- const value = createGlobalSync()
- return (
- <Switch>
- <Match when={value.error}>
- <ErrorPage error={value.error} />
- </Match>
- <Match when={value.ready}>
- <GlobalSyncContext.Provider value={value}>{props.children}</GlobalSyncContext.Provider>
- </Match>
- </Switch>
- )
- }
- export function useGlobalSync() {
- const context = useContext(GlobalSyncContext)
- if (!context) throw new Error("useGlobalSync must be used within GlobalSyncProvider")
- return context
- }
|