mirror of
https://github.com/kunkunsh/kunkun.git
synced 2025-07-09 17:41:31 +00:00

* chore: add vendor submodules * feat: add packages for db,ci,schema,api,jarvis cmds * feat: add tauri-jarvis-plugin * feat: implement extension commands list * fix(desktop): import path errors after packages refactor * chore: add self signed cert * fix: prevent prerender for desktop * fix(desktop): desktop sveltekit static build, use csr for dynamic route * feat: add error handling page and components * refactor: component lib * refactor: move more types, functions and components out of desktop * refactor(ui): more refactor * refactor(ui): move store components to @kksh/ui * ci: add CI for build & test * refactor: rename @kksh/extensions to @kksh/extension * ci: add 2 more ci * ci: fix * fix: CI env var * chore: add changeset * feat: implement extension store item detail view * feat: implement extension store install, uninstall, upgrade * format * revert: upgradable logic, the new one doesn't work yet * refactor: make @kksh/ui dependent only on @kksh/api Reason: @kksh/ui may be published later for building website, all its dependency packages must be also published. To avoid trouble it should be standalone, depend only on packages already published * refactor: cleanup * fixed: some typescript error * chore: got typedoc working on @kksh/api * ci: disable manifest schema upload CI on push
224 lines
5.3 KiB
TypeScript
224 lines
5.3 KiB
TypeScript
import { invoke } from "@tauri-apps/api/core"
|
|
import { listen, type UnlistenFn } from "@tauri-apps/api/event"
|
|
import { generateJarvisPluginCommand } from "./common"
|
|
|
|
const storageCmdPrefix = `ext_store_wrapper_`
|
|
|
|
function computeCommandName(command: string): string {
|
|
return generateJarvisPluginCommand(`${storageCmdPrefix}${command}`)
|
|
}
|
|
|
|
interface ChangePayload<T> {
|
|
path: string
|
|
key: string
|
|
value: T | null
|
|
}
|
|
|
|
/**
|
|
* This Store is actually a wrapper over the tauri-plugin-store. Customized to be used with Jarvis Extensions, the APIs are exactly the same.
|
|
* A key-value store for Jarvis Extensions. Create a store in UI Extensions to store any data.
|
|
* filename is optional for the constructor if you only need one store file.
|
|
* If you plan to have multiple stores, e.g. one for settings, one for data, you can specify different filenames.
|
|
* @example
|
|
* ```ts
|
|
* const store = new JarvisStore("settings.bin");
|
|
* await store.set("theme", "dark");
|
|
* const theme = await store.get("theme");
|
|
* console.log(theme); // dark
|
|
* ```
|
|
*/
|
|
export class JarvisStore {
|
|
path: string
|
|
/**
|
|
* filename is optional if you only need one store file.
|
|
* If you plan to have multiple stores, e.g. one for settings, one for data, you can specify different filenames.
|
|
* @example
|
|
* ```ts
|
|
* const store = new JarvisStore("settings.bin");
|
|
* await store.set("theme", "dark");
|
|
* const theme = await store.get("theme");
|
|
* console.log(theme); // dark
|
|
* ```
|
|
* @param filename filename for the store. Defaults to `default.bin`.
|
|
*/
|
|
constructor(filename: string = "default.bin") {
|
|
this.path = filename
|
|
}
|
|
|
|
/**
|
|
* Inserts a key-value pair into the store.
|
|
*
|
|
* @param key
|
|
* @param value
|
|
* @returns
|
|
*/
|
|
async set(key: string, value: unknown): Promise<void> {
|
|
await invoke(computeCommandName("set"), {
|
|
path: this.path,
|
|
key,
|
|
value
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Returns the value for the given `key` or `null` the key does not exist.
|
|
*
|
|
* @param key
|
|
* @returns
|
|
*/
|
|
async get<T>(key: string): Promise<T | null> {
|
|
return await invoke(computeCommandName("get"), {
|
|
path: this.path,
|
|
key
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Returns `true` if the given `key` exists in the store.
|
|
*
|
|
* @param key
|
|
* @returns
|
|
*/
|
|
async has(key: string): Promise<boolean> {
|
|
return await invoke(computeCommandName("has"), {
|
|
path: this.path,
|
|
key
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Removes a key-value pair from the store.
|
|
*
|
|
* @param key
|
|
* @returns
|
|
*/
|
|
async delete(key: string): Promise<boolean> {
|
|
return await invoke(computeCommandName("delete"), {
|
|
path: this.path,
|
|
key
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Clears the store, removing all key-value pairs.
|
|
*
|
|
* Note: To clear the storage and reset it to it's `default` value, use `reset` instead.
|
|
* @returns
|
|
*/
|
|
async clear(): Promise<void> {
|
|
await invoke(computeCommandName("clear"), {
|
|
path: this.path
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Resets the store to it's `default` value.
|
|
*
|
|
* If no default value has been set, this method behaves identical to `clear`.
|
|
* @returns
|
|
*/
|
|
async reset(): Promise<void> {
|
|
await invoke(computeCommandName("reset"), {
|
|
path: this.path
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Returns a list of all key in the store.
|
|
*
|
|
* @returns
|
|
*/
|
|
async keys(): Promise<string[]> {
|
|
return await invoke(computeCommandName("keys"), {
|
|
path: this.path
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Returns a list of all values in the store.
|
|
*
|
|
* @returns
|
|
*/
|
|
async values<T>(): Promise<T[]> {
|
|
return await invoke(computeCommandName("values"), {
|
|
path: this.path
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Returns a list of all entries in the store.
|
|
*
|
|
* @returns
|
|
*/
|
|
async entries<T>(): Promise<Array<[key: string, value: T]>> {
|
|
return await invoke(computeCommandName("entries"), {
|
|
path: this.path
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Returns the number of key-value pairs in the store.
|
|
*
|
|
* @returns
|
|
*/
|
|
async length(): Promise<number> {
|
|
return await invoke(computeCommandName("length"), {
|
|
path: this.path
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Attempts to load the on-disk state at the stores `path` into memory.
|
|
*
|
|
* This method is useful if the on-disk state was edited by the user and you want to synchronize the changes.
|
|
*
|
|
* Note: This method does not emit change events.
|
|
* @returns
|
|
*/
|
|
async load(): Promise<void> {
|
|
await invoke(computeCommandName("load"), {
|
|
path: this.path
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Saves the store to disk at the stores `path`.
|
|
*
|
|
* As the store is only persisted to disk before the apps exit, changes might be lost in a crash.
|
|
* This method lets you persist the store to disk whenever you deem necessary.
|
|
* @returns
|
|
*/
|
|
async save(): Promise<void> {
|
|
await invoke(computeCommandName("save"), {
|
|
path: this.path
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Listen to changes on a store key.
|
|
* @param key
|
|
* @param cb
|
|
* @returns A promise resolving to a function to unlisten to the event.
|
|
*/
|
|
async onKeyChange<T>(key: string, cb: (value: T | null) => void): Promise<UnlistenFn> {
|
|
return await listen<ChangePayload<T>>("store://change", (event) => {
|
|
if (event.payload.path === this.path && event.payload.key === key) {
|
|
cb(event.payload.value)
|
|
}
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Listen to changes on the store.
|
|
* @param cb
|
|
* @returns A promise resolving to a function to unlisten to the event.
|
|
*/
|
|
async onChange<T>(cb: (key: string, value: T | null) => void): Promise<UnlistenFn> {
|
|
return await listen<ChangePayload<T>>("store://change", (event) => {
|
|
if (event.payload.path === this.path) {
|
|
cb(event.payload.key, event.payload.value)
|
|
}
|
|
})
|
|
}
|
|
}
|