summaryrefslogtreecommitdiffhomepage
path: root/packages/app/src/context/sync.tsx
diff options
context:
space:
mode:
Diffstat (limited to 'packages/app/src/context/sync.tsx')
-rw-r--r--packages/app/src/context/sync.tsx115
1 files changed, 78 insertions, 37 deletions
diff --git a/packages/app/src/context/sync.tsx b/packages/app/src/context/sync.tsx
index 5623a2c7c..db7b06388 100644
--- a/packages/app/src/context/sync.tsx
+++ b/packages/app/src/context/sync.tsx
@@ -3,6 +3,12 @@ import { createStore, produce, reconcile } from "solid-js/store"
import { Binary } from "@opencode-ai/util/binary"
import { retry } from "@opencode-ai/util/retry"
import { createSimpleContext } from "@opencode-ai/ui/context"
+import {
+ clearSessionPrefetch,
+ getSessionPrefetch,
+ getSessionPrefetchPromise,
+ setSessionPrefetch,
+} from "./global-sync/session-prefetch"
import { useGlobalSync } from "./global-sync"
import { useSDK } from "./sdk"
import type { Message, Part } from "@opencode-ai/sdk/v2/client"
@@ -160,6 +166,7 @@ export const { use: useSync, provider: SyncProvider } = createSimpleContext({
const evict = (directory: string, setStore: Setter, sessionIDs: string[]) => {
if (sessionIDs.length === 0) return
+ clearSessionPrefetch(directory, sessionIDs)
for (const sessionID of sessionIDs) {
globalSync.todo.set(sessionID, undefined)
}
@@ -217,6 +224,12 @@ export const { use: useSync, provider: SyncProvider } = createSimpleContext({
}
setMeta("limit", key, input.limit)
setMeta("complete", key, next.complete)
+ setSessionPrefetch({
+ directory: input.directory,
+ sessionID: input.sessionID,
+ limit: input.limit,
+ complete: next.complete,
+ })
})
})
.finally(() => {
@@ -280,54 +293,82 @@ export const { use: useSync, provider: SyncProvider } = createSimpleContext({
parts: input.parts,
})
},
- async sync(sessionID: string) {
+ async sync(sessionID: string, opts?: { force?: boolean }) {
const directory = sdk.directory
const client = sdk.client
const [store, setStore] = globalSync.child(directory)
const key = keyFor(directory, sessionID)
- const hasSession = Binary.search(store.session, sessionID, (s) => s.id).found
touch(directory, setStore, sessionID)
- if (store.message[sessionID] !== undefined && hasSession && meta.limit[key] !== undefined) return
-
- const limit = meta.limit[key] ?? messagePageSize
-
- const sessionReq = hasSession
- ? Promise.resolve()
- : retry(() => client.session.get({ sessionID })).then((session) => {
- if (!tracked(directory, sessionID)) return
- const data = session.data
- if (!data) return
- setStore(
- "session",
- produce((draft) => {
- const match = Binary.search(draft, sessionID, (s) => s.id)
- if (match.found) {
- draft[match.index] = data
- return
- }
- draft.splice(match.index, 0, data)
- }),
- )
- })
-
- const messagesReq = loadMessages({
- directory,
- client,
- setStore,
- sessionID,
- limit,
- })
+ const seeded = getSessionPrefetch(directory, sessionID)
+ if (seeded && store.message[sessionID] !== undefined && meta.limit[key] === undefined) {
+ batch(() => {
+ setMeta("limit", key, seeded.limit)
+ setMeta("complete", key, seeded.complete)
+ setMeta("loading", key, false)
+ })
+ }
- return runInflight(inflight, key, () => Promise.all([sessionReq, messagesReq]).then(() => {}))
+ return runInflight(inflight, key, async () => {
+ const pending = getSessionPrefetchPromise(directory, sessionID)
+ if (pending) {
+ await pending
+ const seeded = getSessionPrefetch(directory, sessionID)
+ if (seeded && store.message[sessionID] !== undefined && meta.limit[key] === undefined) {
+ batch(() => {
+ setMeta("limit", key, seeded.limit)
+ setMeta("complete", key, seeded.complete)
+ setMeta("loading", key, false)
+ })
+ }
+ }
+
+ const hasSession = Binary.search(store.session, sessionID, (s) => s.id).found
+ const cached = store.message[sessionID] !== undefined && meta.limit[key] !== undefined
+ if (cached && hasSession && !opts?.force) return
+
+ const limit = meta.limit[key] ?? messagePageSize
+ const sessionReq =
+ hasSession && !opts?.force
+ ? Promise.resolve()
+ : retry(() => client.session.get({ sessionID })).then((session) => {
+ if (!tracked(directory, sessionID)) return
+ const data = session.data
+ if (!data) return
+ setStore(
+ "session",
+ produce((draft) => {
+ const match = Binary.search(draft, sessionID, (s) => s.id)
+ if (match.found) {
+ draft[match.index] = data
+ return
+ }
+ draft.splice(match.index, 0, data)
+ }),
+ )
+ })
+
+ const messagesReq =
+ cached && !opts?.force
+ ? Promise.resolve()
+ : loadMessages({
+ directory,
+ client,
+ setStore,
+ sessionID,
+ limit,
+ })
+
+ await Promise.all([sessionReq, messagesReq])
+ })
},
- async diff(sessionID: string) {
+ async diff(sessionID: string, opts?: { force?: boolean }) {
const directory = sdk.directory
const client = sdk.client
const [store, setStore] = globalSync.child(directory)
touch(directory, setStore, sessionID)
- if (store.session_diff[sessionID] !== undefined) return
+ if (store.session_diff[sessionID] !== undefined && !opts?.force) return
const key = keyFor(directory, sessionID)
return runInflight(inflightDiff, key, () =>
@@ -337,7 +378,7 @@ export const { use: useSync, provider: SyncProvider } = createSimpleContext({
}),
)
},
- async todo(sessionID: string) {
+ async todo(sessionID: string, opts?: { force?: boolean }) {
const directory = sdk.directory
const client = sdk.client
const [store, setStore] = globalSync.child(directory)
@@ -348,7 +389,7 @@ export const { use: useSync, provider: SyncProvider } = createSimpleContext({
if (cached === undefined) {
globalSync.todo.set(sessionID, existing)
}
- return
+ if (!opts?.force) return
}
if (cached !== undefined) {