summaryrefslogtreecommitdiffhomepage
path: root/packages/app/src/utils/persist.ts
diff options
context:
space:
mode:
authorAdam <[email protected]>2026-01-06 21:38:08 -0600
committerAdam <[email protected]>2026-01-06 21:42:03 -0600
commit761863ae355b3ca1e606ea5eb2106772fa763c19 (patch)
treef8fef26687875e75f4279b6695e35a33977b728c /packages/app/src/utils/persist.ts
parentdadc08ddc7f3c9b1a34e6f401a99406b6714b965 (diff)
downloadopencode-761863ae355b3ca1e606ea5eb2106772fa763c19.tar.gz
opencode-761863ae355b3ca1e606ea5eb2106772fa763c19.zip
chore(app): rework storage approach
Diffstat (limited to 'packages/app/src/utils/persist.ts')
-rw-r--r--packages/app/src/utils/persist.ts228
1 files changed, 223 insertions, 5 deletions
diff --git a/packages/app/src/utils/persist.ts b/packages/app/src/utils/persist.ts
index 12b334f9f..0c20ee31c 100644
--- a/packages/app/src/utils/persist.ts
+++ b/packages/app/src/utils/persist.ts
@@ -1,17 +1,235 @@
import { usePlatform } from "@/context/platform"
-import { makePersisted } from "@solid-primitives/storage"
+import { makePersisted, type AsyncStorage, type SyncStorage } from "@solid-primitives/storage"
+import { checksum } from "@opencode-ai/util/encode"
import { createResource, type Accessor } from "solid-js"
import type { SetStoreFunction, Store } from "solid-js/store"
type InitType = Promise<string> | string | null
type PersistedWithReady<T> = [Store<T>, SetStoreFunction<T>, InitType, Accessor<boolean>]
-export function persisted<T>(key: string, store: [Store<T>, SetStoreFunction<T>]): PersistedWithReady<T> {
+type PersistTarget = {
+ storage?: string
+ key: string
+ legacy?: string[]
+ migrate?: (value: unknown) => unknown
+}
+
+const LEGACY_STORAGE = "default.dat"
+const GLOBAL_STORAGE = "opencode.global.dat"
+
+function snapshot(value: unknown) {
+ return JSON.parse(JSON.stringify(value)) as unknown
+}
+
+function isRecord(value: unknown): value is Record<string, unknown> {
+ return typeof value === "object" && value !== null && !Array.isArray(value)
+}
+
+function merge(defaults: unknown, value: unknown): unknown {
+ if (value === undefined) return defaults
+ if (value === null) return value
+
+ if (Array.isArray(defaults)) {
+ if (Array.isArray(value)) return value
+ return defaults
+ }
+
+ if (isRecord(defaults)) {
+ if (!isRecord(value)) return defaults
+
+ const result: Record<string, unknown> = { ...defaults }
+ for (const key of Object.keys(value)) {
+ if (key in defaults) {
+ result[key] = merge((defaults as Record<string, unknown>)[key], (value as Record<string, unknown>)[key])
+ } else {
+ result[key] = (value as Record<string, unknown>)[key]
+ }
+ }
+ return result
+ }
+
+ return value
+}
+
+function parse(value: string) {
+ try {
+ return JSON.parse(value) as unknown
+ } catch {
+ return undefined
+ }
+}
+
+function workspaceStorage(dir: string) {
+ const head = dir.slice(0, 12) || "workspace"
+ const sum = checksum(dir) ?? "0"
+ return `opencode.workspace.${head}.${sum}.dat`
+}
+
+function localStorageWithPrefix(prefix: string): SyncStorage {
+ const base = `${prefix}:`
+ return {
+ getItem: (key) => localStorage.getItem(base + key),
+ setItem: (key, value) => localStorage.setItem(base + key, value),
+ removeItem: (key) => localStorage.removeItem(base + key),
+ }
+}
+
+export const Persist = {
+ global(key: string, legacy?: string[]): PersistTarget {
+ return { storage: GLOBAL_STORAGE, key, legacy }
+ },
+ workspace(dir: string, key: string, legacy?: string[]): PersistTarget {
+ return { storage: workspaceStorage(dir), key: `workspace:${key}`, legacy }
+ },
+ session(dir: string, session: string, key: string, legacy?: string[]): PersistTarget {
+ return { storage: workspaceStorage(dir), key: `session:${session}:${key}`, legacy }
+ },
+ scoped(dir: string, session: string | undefined, key: string, legacy?: string[]): PersistTarget {
+ if (session) return Persist.session(dir, session, key, legacy)
+ return Persist.workspace(dir, key, legacy)
+ },
+}
+
+export function removePersisted(target: { storage?: string; key: string }) {
const platform = usePlatform()
- const [state, setState, init] = makePersisted(store, { name: key, storage: platform.storage?.() ?? localStorage })
+ const isDesktop = platform.platform === "desktop" && !!platform.storage
+
+ if (isDesktop) {
+ return platform.storage?.(target.storage)?.removeItem(target.key)
+ }
+
+ if (!target.storage) {
+ localStorage.removeItem(target.key)
+ return
+ }
+
+ localStorageWithPrefix(target.storage).removeItem(target.key)
+}
+
+export function persisted<T>(
+ target: string | PersistTarget,
+ store: [Store<T>, SetStoreFunction<T>],
+): PersistedWithReady<T> {
+ const platform = usePlatform()
+ const config: PersistTarget = typeof target === "string" ? { key: target } : target
+
+ const defaults = snapshot(store[0])
+ const legacy = config.legacy ?? []
+
+ const isDesktop = platform.platform === "desktop" && !!platform.storage
+
+ const currentStorage = (() => {
+ if (isDesktop) return platform.storage?.(config.storage)
+ if (!config.storage) return localStorage
+ return localStorageWithPrefix(config.storage)
+ })()
+
+ const legacyStorage = (() => {
+ if (!isDesktop) return localStorage
+ if (!config.storage) return platform.storage?.()
+ return platform.storage?.(LEGACY_STORAGE)
+ })()
+
+ const storage = (() => {
+ if (!isDesktop) {
+ const current = currentStorage as SyncStorage
+ const legacyStore = legacyStorage as SyncStorage
+
+ const api: SyncStorage = {
+ getItem: (key) => {
+ const raw = current.getItem(key)
+ if (raw !== null) {
+ const parsed = parse(raw)
+ if (parsed === undefined) return raw
+
+ const migrated = config.migrate ? config.migrate(parsed) : parsed
+ const merged = merge(defaults, migrated)
+ const next = JSON.stringify(merged)
+ if (raw !== next) current.setItem(key, next)
+ return next
+ }
+
+ for (const legacyKey of legacy) {
+ const legacyRaw = legacyStore.getItem(legacyKey)
+ if (legacyRaw === null) continue
+
+ current.setItem(key, legacyRaw)
+ legacyStore.removeItem(legacyKey)
+
+ const parsed = parse(legacyRaw)
+ if (parsed === undefined) return legacyRaw
+
+ const migrated = config.migrate ? config.migrate(parsed) : parsed
+ const merged = merge(defaults, migrated)
+ const next = JSON.stringify(merged)
+ if (legacyRaw !== next) current.setItem(key, next)
+ return next
+ }
+
+ return null
+ },
+ setItem: (key, value) => {
+ current.setItem(key, value)
+ },
+ removeItem: (key) => {
+ current.removeItem(key)
+ },
+ }
+
+ return api
+ }
+
+ const current = currentStorage as AsyncStorage
+ const legacyStore = legacyStorage as AsyncStorage | undefined
+
+ const api: AsyncStorage = {
+ getItem: async (key) => {
+ const raw = await current.getItem(key)
+ if (raw !== null) {
+ const parsed = parse(raw)
+ if (parsed === undefined) return raw
+
+ const migrated = config.migrate ? config.migrate(parsed) : parsed
+ const merged = merge(defaults, migrated)
+ const next = JSON.stringify(merged)
+ if (raw !== next) await current.setItem(key, next)
+ return next
+ }
+
+ if (!legacyStore) return null
+
+ for (const legacyKey of legacy) {
+ const legacyRaw = await legacyStore.getItem(legacyKey)
+ if (legacyRaw === null) continue
+
+ await current.setItem(key, legacyRaw)
+ await legacyStore.removeItem(legacyKey)
+
+ const parsed = parse(legacyRaw)
+ if (parsed === undefined) return legacyRaw
+
+ const migrated = config.migrate ? config.migrate(parsed) : parsed
+ const merged = merge(defaults, migrated)
+ const next = JSON.stringify(merged)
+ if (legacyRaw !== next) await current.setItem(key, next)
+ return next
+ }
+
+ return null
+ },
+ setItem: async (key, value) => {
+ await current.setItem(key, value)
+ },
+ removeItem: async (key) => {
+ await current.removeItem(key)
+ },
+ }
+
+ return api
+ })()
+
+ const [state, setState, init] = makePersisted(store, { name: config.key, storage })
- // Create a resource that resolves when the store is initialized
- // This integrates with Suspense and provides a ready signal
const isAsync = init instanceof Promise
const [ready] = createResource(
() => init,