Feature: fine grain kill API for extension (#201)

* upgrade tauri-plugin-shellx

* feat(shell): add killPid method to shell API with new permission

* fix: extension new window

* feat(shell): enhance process management and logging in extensions

- Add debug logging for extension process events
- Implement process tracking in UI worker
- Update shell API to support custom process recording
- Modify extension template to demonstrate process spawning
- Refactor shell command handling with improved error handling

* Add killPid extension API to @kksh/api

* chore(deps): update tauri-plugin-shellx-api to version 2.0.15

* pnpm lock

* chore(deps): update dependencies and lock file

- Upgrade ESLint to version 9.21.0
- Update @types/bun to version 1.2.3
- Bump various development dependencies
- Reorganize package.json dependencies in ui package

* chore(deps): update SvelteKit template dependencies

- Add @eslint/js version 9.21.0
- Update package.json dependency order
- Sync pnpm-lock.yaml with package.json changes

* chore: add eslint to desktop

* chore(deps): remove local tauri-plugin-shellx and use published version 2.0.15

* bump desktop to 0.1.28
This commit is contained in:
Huakun 2025-02-26 02:06:06 -05:00 committed by GitHub
parent 66135624b9
commit a92c266d32
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
53 changed files with 792 additions and 310 deletions

View File

@ -34,7 +34,7 @@ If you are interested in contributing to the project, please read the following
```bash
git clone https://github.com/kunkunsh/kunkun.git --recursive
pnpm install
pnpm prepare
pnpm build # build submodules
```
### Run Desktop App

6
Cargo.lock generated
View File

@ -4753,7 +4753,7 @@ version = "0.7.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "af1844ef2428cc3e1cb900be36181049ef3d3193c63e43026cfe202983b27a56"
dependencies = [
"proc-macro-crate 1.3.1",
"proc-macro-crate 2.0.0",
"proc-macro2",
"quote",
"syn 2.0.87",
@ -7871,9 +7871,9 @@ dependencies = [
[[package]]
name = "tauri-plugin-shellx"
version = "2.0.12"
version = "2.0.15"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "32409a09fd9351a3d3e25f7a2e2ba501b114c6869afb7b46d06a5473055c995f"
checksum = "88a5b6b883070de00f8fd5025aa5b81e3b012a5a3d9f38875cdb45809c71232c"
dependencies = [
"encoding_rs",
"open",

View File

@ -20,7 +20,8 @@ tokio-util = "0.7.12"
mdns-sd = "0.11.1"
tauri-plugin-network = { path = "./vendors/tauri-plugin-network" }
tauri-plugin-keyring = { path = "./vendors/tauri-plugin-keyring" }
tauri-plugin-clipboard = "2.1.8"
tauri-plugin-shellx = { version = "2.0.15" }
tauri-plugin-clipboard = "2.1.11"
mac-security-rs = { path = "./packages/mac-security-rs" }
log = "0.4.22"
strum = "0.26"

View File

@ -1,5 +1,12 @@
# kksh
## 0.1.2
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.4
## 0.1.1
### Patch Changes

View File

@ -1,7 +1,7 @@
{
"name": "kksh",
"module": "dist/cli.js",
"version": "0.1.1",
"version": "0.1.2",
"type": "module",
"bin": {
"kksh": "./dist/cli.js",

View File

@ -1,5 +1,12 @@
# create-kunkun
## 0.1.48
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.4
## 0.1.45
### Patch Changes

View File

@ -1,7 +1,7 @@
{
"name": "create-kunkun",
"type": "module",
"version": "0.1.47",
"version": "0.1.48",
"bin": {
"create-kunkun": "dist/index.mjs"
},

View File

@ -1,6 +1,6 @@
{
"name": "@kksh/desktop",
"version": "0.1.27",
"version": "0.1.28",
"description": "",
"type": "module",
"scripts": {
@ -30,6 +30,7 @@
"@tauri-apps/plugin-shell": "^2.2.0",
"@tauri-apps/plugin-stronghold": "^2.2.0",
"dompurify": "^3.2.3",
"eslint": "^9.21.0",
"fuse.js": "^7.1.0",
"gsap": "^3.12.5",
"kkrpc": "^0.1.1",
@ -39,6 +40,7 @@
"svelte-sonner": "^0.3.28",
"sveltekit-superforms": "^2.22.1",
"tauri-plugin-clipboard-api": "^2.1.11",
"tauri-plugin-shellx-api": "^2.0.15",
"tauri-plugin-user-input-api": "workspace:*",
"uuid": "^11.0.3"
},

View File

@ -34,7 +34,7 @@ chrono = { workspace = true }
log = { workspace = true }
urlencoding = "2.1.3"
tauri-plugin-process = "2.2.0"
tauri-plugin-shellx = "2.0.12"
tauri-plugin-shellx = { workspace = true }
tauri-plugin-fs = { version = "2.2.0", features = ["watch"] }
tauri-plugin-dialog = "2.2.0"
tauri-plugin-notification = "2.2.1"

View File

@ -60,6 +60,7 @@
"shellx:allow-execute",
"shellx:allow-open",
"shellx:allow-kill",
"shellx:allow-kill-pid",
"shellx:allow-spawn",
"shellx:allow-stdin-write",
"shellx:allow-fix-path-env",

View File

@ -20,10 +20,12 @@ import * as v from "valibot"
export const KunkunIframeExtParams = v.object({
url: v.string(),
cmdName: v.optional(v.string()),
extPath: v.string()
})
export type KunkunIframeExtParams = v.InferOutput<typeof KunkunIframeExtParams>
export const KunkunTemplateExtParams = v.object({
url: v.optional(v.string()),
extPath: v.string(),
cmdName: v.string()
})
@ -36,10 +38,10 @@ export async function createExtSupportDir(extPath: string) {
}
}
function setTemplateExtParams(extPath: string, cmdName: string) {
function setTemplateExtParams(extPath: string, cmdName: string, url?: string) {
localStorage.setItem(
"kunkun-template-ext-params",
JSON.stringify({ extPath, cmdName } satisfies KunkunTemplateExtParams)
JSON.stringify({ extPath, cmdName, url } satisfies KunkunTemplateExtParams)
)
}
@ -50,13 +52,15 @@ export async function onTemplateUiCmdSelect(
) {
await createExtSupportDir(ext.extPath)
const url = `/app/extension/ui-worker?extPath=${encodeURIComponent(ext.extPath)}&cmdName=${encodeURIComponent(cmd.name)}`
setTemplateExtParams(ext.extPath, cmd.name)
setTemplateExtParams(ext.extPath, cmd.name, url)
if (cmd.window) {
const winLabel = await winExtMap.registerExtensionWithWindow({ extPath: ext.extPath })
localStorage.setItem(
"kunkun-template-ext-params",
JSON.stringify({ url, extPath: ext.extPath } satisfies KunkunIframeExtParams)
)
const paramsStr = JSON.stringify({
url,
extPath: ext.extPath,
cmdName: cmd.name
} satisfies KunkunIframeExtParams)
localStorage.setItem("kunkun-template-ext-params", paramsStr)
const window = launchNewExtWindow(winLabel, url, cmd.window)
window.onCloseRequested(async (event) => {
await winExtMap.unregisterExtensionFromWindow(winLabel)
@ -89,7 +93,16 @@ export async function onHeadlessCmdSelect(
}
const serverAPI: IKunkunFullServerAPI = constructJarvisServerAPIWithPermissions(
loadedExt.kunkun.permissions,
loadedExt.extPath
loadedExt.extPath,
{
recordSpawnedProcess: async (pid: number) => {
console.log("recordSpawnedProcess pid", pid)
},
getSpawnedProcesses: async () => {
console.log("getSpawnedProcesses")
return []
}
}
)
const serverAPI2 = {
...serverAPI,

View File

@ -104,6 +104,7 @@ function createWinExtMapStore(): Writable<WinExtMap> & API {
}
},
registerProcess: async (windowLabel: string, pid: number) => {
console.log("registerProcess", windowLabel, pid)
const winExtMap = get(store)
await registerExtensionSpawnedProcess(windowLabel, pid)
if (!winExtMap[windowLabel]) {
@ -116,6 +117,7 @@ function createWinExtMapStore(): Writable<WinExtMap> & API {
const winExtMap = get(store)
const found = Object.entries(winExtMap).find(([windowLabel, ext]) => ext.pids.includes(pid))
if (!found) {
warn(`Process ${pid} does not have an extension registered, thus will not be killed`)
return
}
const [windowLabel, ext] = found

View File

@ -14,7 +14,7 @@
import { Constants, ViewTransition } from "@kksh/ui"
import type { UnlistenFn } from "@tauri-apps/api/event"
import { getCurrentWebviewWindow } from "@tauri-apps/api/webviewWindow"
import { attachConsole, error, info } from "@tauri-apps/plugin-log"
import { attachConsole, debug, error, info } from "@tauri-apps/plugin-log"
import { afterNavigate, beforeNavigate } from "$app/navigation"
import { gsap } from "gsap"
import { Flip } from "gsap/Flip"
@ -89,13 +89,13 @@
extensions.init()
unlisteners.push(
await listenToRecordExtensionProcessEvent(async (event) => {
console.log("record extension process event", event)
debug(`record extension process event ${event.payload.pid}`)
winExtMap.registerProcess(event.payload.windowLabel, event.payload.pid)
})
)
unlisteners.push(
await listenToKillProcessEvent((event) => {
console.log("kill process event", event)
debug(`kill process event ${event.payload.pid}`)
winExtMap.unregisterProcess(event.payload.pid)
})
)

View File

@ -35,6 +35,7 @@
SystemCmds
} from "@kksh/ui/main"
import { cn } from "@kksh/ui/utils"
import { Channel, invoke } from "@tauri-apps/api/core"
import { getCurrentWebviewWindow } from "@tauri-apps/api/webviewWindow"
import { getCurrentWindow, Window } from "@tauri-apps/api/window"
import { platform } from "@tauri-apps/plugin-os"
@ -42,6 +43,7 @@
import { goto } from "$app/navigation"
import { ArrowBigUpIcon, CircleXIcon, EllipsisVerticalIcon, RefreshCcwIcon } from "lucide-svelte"
import { onMount } from "svelte"
import * as shell from "tauri-plugin-shellx-api"
const win = getCurrentWindow()
let inputEle: HTMLInputElement | null = $state(null)
@ -82,6 +84,43 @@
}
})
})
async function spawn() {
const cmd = shell.Command.create("deno", ["run", "/Users/hk/Dev/kunkun/deno.ts"])
cmd.stdout.on("data", (data) => {
console.log("stdout", data)
})
const child = await cmd.spawn()
console.log("child", child)
setTimeout(() => {
child
.kill()
.then(() => {
console.log("child killed")
})
.catch((err) => {
console.error("child kill error", err)
})
}, 5000)
// invoke<number>("plugin:shellx|spawn", {
// program: "deno",
// args: ["run", "/Users/hk/Dev/kunkun/deno.ts"],
// options: {},
// onEvent: new Channel<CommandEvent<string>>()
// }).then((pid) => {
// console.log("spawned process (shell server) pid:", pid)
// setTimeout(() => {
// console.log("killing process (shell server) pid:", pid)
// killPid(pid)
// .then(() => {
// console.log("killed process (shell server) pid:", pid)
// })
// .catch((err) => {
// console.error("kill process (shell server) pid:", pid, err)
// })
// }, 3000)
// })
}
</script>
<svelte:window
@ -198,6 +237,7 @@
</DropdownMenu.Root>
{/snippet}
</CustomCommandInput>
<Button onclick={spawn}>Spawn</Button>
<Command.List class="max-h-screen grow">
<Command.Empty data-tauri-drag-region>No results found.</Command.Empty>
{#if $appConfig.extensionsInstallDir && $devStoreExtsFiltered.length > 0}

View File

@ -27,12 +27,19 @@
type IComponent,
type TemplateUiCommand
} from "@kksh/api/ui/template"
import { Button } from "@kksh/svelte5"
import { LoadingBar } from "@kksh/ui"
import { Templates } from "@kksh/ui/extension"
import { GlobalCommandPaletteFooter } from "@kksh/ui/main"
import type { IKunkunFullServerAPI } from "@kunkunapi/src/api/server"
import type { UnlistenFn } from "@tauri-apps/api/event"
import {
RECORD_EXTENSION_PROCESS_EVENT,
type IRecordExtensionProcessEvent
} from "@kunkunapi/src/events.js"
import { Channel, invoke } from "@tauri-apps/api/core"
import { emitTo, type UnlistenFn } from "@tauri-apps/api/event"
import { getCurrentWebviewWindow } from "@tauri-apps/api/webviewWindow"
import { getCurrentWindow } from "@tauri-apps/api/window"
import * as fs from "@tauri-apps/plugin-fs"
import { readTextFile } from "@tauri-apps/plugin-fs"
import { debug } from "@tauri-apps/plugin-log"
@ -40,6 +47,7 @@
import { goto } from "$app/navigation"
import { RPCChannel, WorkerParentIO } from "kkrpc/browser"
import { onDestroy, onMount, tick } from "svelte"
import { type CommandEvent } from "tauri-plugin-shellx-api"
import * as v from "valibot"
const { data } = $props()
@ -58,6 +66,7 @@
let loading = $state(false)
let searchTerm = $state("")
let searchBarPlaceholder = $state("")
let extSpawnedProcesses = $state<number[]>([])
const appWin = getCurrentWebviewWindow()
const loadingBar = $derived($appState.loadingBar || extensionLoadingBar)
let loaded = $state(false)
@ -199,7 +208,6 @@
searchTerm = term
},
async setSearchBarPlaceholder(placeholder: string) {
console.log("setSearchBarPlaceholder", placeholder)
searchBarPlaceholder = placeholder
},
async goBack() {
@ -221,7 +229,20 @@
worker = new Worker(blobURL)
const serverAPI: IKunkunFullServerAPI = constructJarvisServerAPIWithPermissions(
loadedExt.kunkun.permissions,
loadedExt.extPath
loadedExt.extPath,
{
recordSpawnedProcess: async (pid: number) => {
extSpawnedProcesses = [...extSpawnedProcesses, pid]
// winExtMap.registerProcess(appWin.label, pid)
const curWin = await getCurrentWindow()
await emitTo("main", RECORD_EXTENSION_PROCESS_EVENT, {
windowLabel: curWin.label,
pid
} satisfies IRecordExtensionProcessEvent)
// TODO: record process in a store
},
getSpawnedProcesses: () => Promise.resolve(extSpawnedProcesses)
}
)
const serverAPI2 = {
...serverAPI,
@ -234,7 +255,6 @@
language: () => Promise.resolve("en")
} satisfies IApp
}
const io = new WorkerParentIO(worker)
const rpc = new RPCChannel<typeof serverAPI2, TemplateUiCommand>(io, {
expose: serverAPI2
@ -250,20 +270,20 @@
}
})
function onPkgJsonChange(evt: fs.WatchEvent) {
const parsed = v.safeParse(WatchEvent, evt)
if (parsed.success) {
if (
parsed.output.type.modify.kind === "data" &&
parsed.output.type.modify.mode === "content" &&
parsed.output.paths.includes(data.pkgJsonPath)
) {
console.log("pkgJson changed", parsed.output.paths)
// emit event to reload extension commands
emitReloadOneExtension(loadedExt.extPath)
}
}
}
// function onPkgJsonChange(evt: fs.WatchEvent) {
// const parsed = v.safeParse(WatchEvent, evt)
// if (parsed.success) {
// if (
// parsed.output.type.modify.kind === "data" &&
// parsed.output.type.modify.mode === "content" &&
// parsed.output.paths.includes(data.pkgJsonPath)
// ) {
// console.log("pkgJson changed", parsed.output.paths)
// // emit event to reload extension commands
// emitReloadOneExtension(loadedExt.extPath)
// }
// }
// }
onMount(async () => {
setTimeout(() => {
@ -284,10 +304,9 @@
appState.setLoadingBar(false)
loaded = true
}, 500)
console.log("watching", data.pkgJsonPath)
fs.watch(data.pkgJsonPath, onPkgJsonChange).then((unlisten) => {
unlistenPkgJsonWatch = unlisten
})
// fs.watch(data.pkgJsonPath, onPkgJsonChange).then((unlisten) => {
// unlistenPkgJsonWatch = unlisten
// })
})
onDestroy(() => {

View File

@ -5,6 +5,7 @@ import type { Ext as ExtInfoInDB, ExtPackageJsonExtra } from "@kksh/api/models"
import { loadExtensionManifestFromDisk } from "@kksh/extension"
import { error as sbError, error as svError } from "@sveltejs/kit"
import { join } from "@tauri-apps/api/path"
import { getCurrentWindow } from "@tauri-apps/api/window"
import { exists, readTextFile } from "@tauri-apps/plugin-fs"
import { error } from "@tauri-apps/plugin-log"
import { goto } from "$app/navigation"
@ -14,15 +15,16 @@ import type { PageLoad } from "./$types"
export const load: PageLoad = async ({ url }) => {
// both query parameter must exist
const rawKunkunTemplateExtParams = localStorage.getItem("kunkun-template-ext-params")
if (!rawKunkunTemplateExtParams) {
toast.error("Invalid extension path or url")
return svError(404, "Invalid extension path or url")
}
const parsed = v.safeParse(KunkunTemplateExtParams, JSON.parse(rawKunkunTemplateExtParams))
const json = JSON.parse(rawKunkunTemplateExtParams)
const parsed = v.safeParse(KunkunTemplateExtParams, json)
if (!parsed.success) {
getCurrentWindow().show()
console.error(v.flatten<typeof KunkunTemplateExtParams>(parsed.issues))
toast.error("Fail to parse extension params from local storage", {
description: `${v.flatten<typeof KunkunTemplateExtParams>(parsed.issues)}`
})

121
deno.lock generated
View File

@ -5,6 +5,7 @@
"npm:@changesets/cli@^2.27.11": "2.27.12",
"npm:@eslint/js@^9.18.0": "9.19.0",
"npm:@eslint/js@^9.19.0": "9.19.0",
"npm:@eslint/js@^9.21.0": "9.21.0",
"npm:@formkit/auto-animate@~0.8.2": "0.8.2",
"npm:@grpc/grpc-js@^1.12.2": "1.12.5",
"npm:@grpc/proto-loader@~0.7.13": "0.7.13",
@ -117,6 +118,7 @@
"npm:eslint@8": "8.57.1",
"npm:eslint@^8.57.0": "8.57.1",
"npm:eslint@^9.17.0": "9.19.0",
"npm:eslint@^9.21.0": "9.21.0",
"npm:formsnap@2.0.0-next.1": "2.0.0-next.1_svelte@5.19.6__acorn@8.14.0_sveltekit-superforms@2.23.1__@sveltejs+kit@2.16.1___@sveltejs+vite-plugin-svelte@5.0.3____svelte@5.19.6_____acorn@8.14.0____vite@6.0.11_____@types+node@20.17.16_____jiti@2.4.2____@types+node@20.17.16___svelte@5.19.6____acorn@8.14.0___vite@5.4.14____@types+node@20.17.16___vite@6.0.11____@types+node@20.17.16____jiti@2.4.2___@types+node@20.17.16__svelte@5.19.6___acorn@8.14.0__valibot@1.0.0-beta.12___typescript@5.6.3__zod@3.24.1__@sveltejs+vite-plugin-svelte@5.0.3___svelte@5.19.6____acorn@8.14.0___vite@6.0.11____@types+node@20.17.16____jiti@2.4.2___@types+node@20.17.16__vite@5.4.14___@types+node@20.17.16__typescript@5.6.3__vite@6.0.11___@types+node@20.17.16___jiti@2.4.2__@types+node@20.17.16_@sveltejs+kit@2.16.1__@sveltejs+vite-plugin-svelte@5.0.3___svelte@5.19.6____acorn@8.14.0___vite@6.0.11____@types+node@20.17.16____jiti@2.4.2___@types+node@20.17.16__svelte@5.19.6___acorn@8.14.0__vite@5.4.14___@types+node@20.17.16__vite@6.0.11___@types+node@20.17.16___jiti@2.4.2__@types+node@20.17.16_valibot@1.0.0-beta.12__typescript@5.6.3_zod@3.24.1_@sveltejs+vite-plugin-svelte@5.0.3__svelte@5.19.6___acorn@8.14.0__vite@6.0.11___@types+node@20.17.16___jiti@2.4.2__@types+node@20.17.16_vite@5.4.14__@types+node@20.17.16_typescript@5.6.3_vite@6.0.11__@types+node@20.17.16__jiti@2.4.2_@types+node@20.17.16",
"npm:fs-extra@^11.2.0": "11.3.0",
"npm:get-folder-size@5": "5.0.0",
@ -178,8 +180,9 @@
"npm:tailwindcss@^3.4.6": "3.4.17_postcss@8.5.1",
"npm:tailwindcss@^3.4.7": "3.4.17_postcss@8.5.1",
"npm:tar@^7.4.3": "7.4.3",
"npm:tauri-api-adapter@~0.3.20": "0.3.20_typescript@5.6.3",
"npm:tauri-api-adapter@~0.3.23": "0.3.23_typescript@5.6.3",
"npm:tauri-plugin-clipboard-api@^2.1.11": "2.1.11_typescript@5.6.3",
"npm:tauri-plugin-shellx-api@2.0.15": "2.0.15",
"npm:tauri-plugin-shellx-api@^2.0.14": "2.0.14",
"npm:tauri-plugin-system-info-api@2.0.8": "2.0.8_typescript@5.6.3",
"npm:ts-proto@^2.3.0": "2.6.1",
@ -1442,6 +1445,13 @@
"eslint-visitor-keys@3.4.3"
]
},
"@eslint-community/eslint-utils@4.4.1_eslint@9.21.0": {
"integrity": "sha512-s3O3waFUrMV8P/XaF/+ZTp1X9XBZW1a4B97ZnjQF2KYWaFD2A8KyFBsrsfSjEmjn3RGWAIuvlneuZm3CUK3jbA==",
"dependencies": [
"eslint@9.21.0",
"eslint-visitor-keys@3.4.3"
]
},
"@eslint-community/regexpp@4.12.1": {
"integrity": "sha512-CCZCDJuduB9OUkFkY2IgppNZMi2lBQgD2qzwXkEia16cge2pijY/aXi96CJMquDMn3nJdlPV1A5KrJEXwfLNzQ=="
},
@ -1459,6 +1469,12 @@
"@types/json-schema"
]
},
"@eslint/core@0.12.0": {
"integrity": "sha512-cmrR6pytBuSMTaBweKoGMwu3EiHiEC+DoyupPmlZ0HxBJBtIxwe+j/E4XPIKNx+Q74c8lXKPwYawBf5glsTkHg==",
"dependencies": [
"@types/json-schema"
]
},
"@eslint/eslintrc@2.1.4": {
"integrity": "sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ==",
"dependencies": [
@ -1487,19 +1503,43 @@
"strip-json-comments@3.1.1"
]
},
"@eslint/eslintrc@3.3.0": {
"integrity": "sha512-yaVPAiNAalnCZedKLdR21GOGILMLKPyqSLWaAjQFvYA2i/ciDi8ArYVr69Anohb6cH2Ukhqti4aFnYyPm8wdwQ==",
"dependencies": [
"ajv@6.12.6",
"debug@4.4.0",
"espree@10.3.0_acorn@8.14.0",
"globals@14.0.0",
"ignore@5.3.2",
"import-fresh",
"js-yaml@4.1.0",
"minimatch@3.1.2",
"strip-json-comments@3.1.1"
]
},
"@eslint/js@8.57.1": {
"integrity": "sha512-d9zaMRSTIKDLhctzH12MtXvJKSSUhaHcjV+2Z+GK+EEY7XKpP5yR4x+N3TAcHTcu963nIr+TMcCb4DBCYX1z6Q=="
},
"@eslint/js@9.19.0": {
"integrity": "sha512-rbq9/g38qjfqFLOVPvwjIvFFdNziEC5S65jmjPw5r6A//QH+W91akh9irMwjDN8zKUTak6W9EsAv4m/7Wnw0UQ=="
},
"@eslint/js@9.21.0": {
"integrity": "sha512-BqStZ3HX8Yz6LvsF5ByXYrtigrV5AXADWLAGc7PH/1SxOb7/FIYYMszZZWiUou/GB9P2lXWk2SV4d+Z8h0nknw=="
},
"@eslint/object-schema@2.1.6": {
"integrity": "sha512-RBMg5FRL0I0gs51M/guSAj5/e14VQ4tpZnQNWwuDT66P14I43ItmPfIZRhO9fUVIPOAQXU47atlywZ/czoqFPA=="
},
"@eslint/plugin-kit@0.2.5": {
"integrity": "sha512-lB05FkqEdUg2AA0xEbUz0SnkXT1LcCTa438W4IWTUh4hdOnVbQyOJ81OrDXsJk/LSiJHubgGEFoR5EHq1NsH1A==",
"dependencies": [
"@eslint/core",
"@eslint/core@0.10.0",
"levn"
]
},
"@eslint/plugin-kit@0.2.7": {
"integrity": "sha512-JubJ5B2pJ4k4yGxaNLdbjrnk9d/iDz6/q8wOilpIowd6PJPgaxCuHBnBszq7Ce2TyMrywm5r4PnKm6V3iiZF+g==",
"dependencies": [
"@eslint/core@0.12.0",
"levn"
]
},
@ -1625,6 +1665,9 @@
"@humanwhocodes/retry@0.4.1": {
"integrity": "sha512-c7hNEllBlenFTHBky65mhq8WD2kbN9Q6gk0bTk8lSBvc554jpXSkST1iePudpt7+A/AQvuHs9EMqjHDXMY1lrA=="
},
"@humanwhocodes/retry@0.4.2": {
"integrity": "sha512-xeO57FpIu4p1Ri3Jq/EXq4ClRm86dVF2z/+kvFnyqVYRavTZmaFaUBbWCOuuTh0o/g7DSsk6kc2vrS4Vl5oPOQ=="
},
"@ianvs/prettier-plugin-sort-imports@4.4.1_prettier@3.4.2": {
"integrity": "sha512-F0/Hrcfpy8WuxlQyAWJTEren/uxKhYonOGY4OyWmwRdeTvkh9mMSCxowZLjNkhwi/2ipqCgtXwwOk7tW0mWXkA==",
"dependencies": [
@ -2201,7 +2244,7 @@
"minimatch@10.0.1",
"semver@7.7.0",
"tauri-api-adapter@0.3.8_typescript@5.6.3_rollup@4.34.0_tslib@2.8.1",
"tauri-plugin-shellx-api",
"tauri-plugin-shellx-api@2.0.14",
"valibot@0.40.0_typescript@5.6.3",
"vue-sonner"
]
@ -8590,10 +8633,10 @@
"@eslint-community/eslint-utils@4.4.1_eslint@9.19.0",
"@eslint-community/regexpp",
"@eslint/config-array",
"@eslint/core",
"@eslint/core@0.10.0",
"@eslint/eslintrc@3.2.0",
"@eslint/js@9.19.0",
"@eslint/plugin-kit",
"@eslint/plugin-kit@0.2.5",
"@humanfs/node",
"@humanwhocodes/module-importer",
"@humanwhocodes/retry@0.4.1",
@ -8623,6 +8666,45 @@
"optionator"
]
},
"eslint@9.21.0": {
"integrity": "sha512-KjeihdFqTPhOMXTt7StsDxriV4n66ueuF/jfPNC3j/lduHwr/ijDwJMsF+wyMJethgiKi5wniIE243vi07d3pg==",
"dependencies": [
"@eslint-community/eslint-utils@4.4.1_eslint@9.21.0",
"@eslint-community/regexpp",
"@eslint/config-array",
"@eslint/core@0.12.0",
"@eslint/eslintrc@3.3.0",
"@eslint/js@9.21.0",
"@eslint/plugin-kit@0.2.7",
"@humanfs/node",
"@humanwhocodes/module-importer",
"@humanwhocodes/retry@0.4.2",
"@types/estree",
"@types/json-schema",
"ajv@6.12.6",
"chalk@4.1.2",
"cross-spawn",
"debug@4.4.0",
"escape-string-regexp@4.0.0",
"eslint-scope@8.2.0",
"eslint-visitor-keys@4.2.0",
"espree@10.3.0_acorn@8.14.0",
"esquery",
"esutils",
"fast-deep-equal",
"file-entry-cache@8.0.0",
"find-up@5.0.0",
"glob-parent@6.0.2",
"ignore@5.3.2",
"imurmurhash",
"is-glob",
"json-stable-stringify-without-jsonify",
"lodash.merge",
"minimatch@3.1.2",
"natural-compare",
"optionator"
]
},
"esm-env@1.2.2": {
"integrity": "sha512-Epxrv+Nr/CaL4ZcFGPJIYLWFom+YeV1DqMLHJoEd9SYRxNbaFruBwfEX/kkHUJf55j2+TUbmDcmuilbP1TmXHA=="
},
@ -9155,7 +9237,7 @@
"integrity": "sha512-zrQDm8XPnYEKawJScsnM0QzobJxlT/kHOOlRTio8IH/GrmxRE5fjllkzdaHclIuNjUQTJYH2xHNIGfdpJkDJUw==",
"dependencies": [
"foreground-child",
"jackspeak@4.0.2",
"jackspeak@4.1.0",
"minimatch@10.0.1",
"minipass@7.1.2",
"package-json-from-dist",
@ -9906,8 +9988,8 @@
"@pkgjs/parseargs"
]
},
"jackspeak@4.0.2": {
"integrity": "sha512-bZsjR/iRjl1Nk1UkjGpAzLNfQtzuijhn2g+pbZb98HQ1Gk8vM9hfbxeMBP+M2/UUdwj0RqGG3mlvk2MsAqwvEw==",
"jackspeak@4.1.0": {
"integrity": "sha512-9DDdhb5j6cpeitCbvLO7n7J4IxnbM6hoF6O1g4HQ5TfhvvKN8ywDM7668ZhMHRqVmxqhps/F6syWK2KcPxYlkw==",
"dependencies": [
"@isaacs/cliui"
]
@ -13664,8 +13746,8 @@
"yallist@5.0.0"
]
},
"tauri-api-adapter@0.3.20_typescript@5.6.3": {
"integrity": "sha512-tRK25c1d34ZRd5CJAXfrBeTr4eGh0UPshLac1DBm4TBF+EZ1TFCwLpvUj9DQ3VS2gHKA9fNc4J5pNpHh3OD4Og==",
"tauri-api-adapter@0.3.23_typescript@5.6.3": {
"integrity": "sha512-WT/NVCdZZSn63Yu/lhZqeUxMGy/VoBZik7yuY2ot+kCynsI1zJQ9ZB6ORmPL8+VBBRbSpEn1yOUbSTSucRFLOQ==",
"dependencies": [
"@tauri-apps/api@2.2.0",
"@tauri-apps/plugin-dialog@2.2.0",
@ -13681,7 +13763,7 @@
"shx",
"tauri-plugin-clipboard-api",
"tauri-plugin-network-api",
"tauri-plugin-shellx-api",
"tauri-plugin-shellx-api@2.0.15",
"tauri-plugin-system-info-api",
"tsc-alias",
"typescript@5.6.3",
@ -13708,7 +13790,7 @@
"shx",
"tauri-plugin-clipboard-api",
"tauri-plugin-network-api",
"tauri-plugin-shellx-api",
"tauri-plugin-shellx-api@2.0.14",
"tauri-plugin-system-info-api",
"tsc-alias",
"typescript@5.6.3",
@ -13735,6 +13817,12 @@
"@tauri-apps/api@2.2.0"
]
},
"tauri-plugin-shellx-api@2.0.15": {
"integrity": "sha512-MrgArTe90o/zLHbIv/x0kTtLm1tiQiEIPnvUcIwuz1pBCGn45+E0ghOgjjGMlGaA2ln6KRV213e52/Az4MqMPQ==",
"dependencies": [
"@tauri-apps/api@2.2.0"
]
},
"tauri-plugin-system-info-api@2.0.8_typescript@5.6.3": {
"integrity": "sha512-EFdLXNGp6Zu9SNsZCkU+55A8027OnrVw/TQrd0oJHgfZzs4qvm1iMmSvyid4MLftt33iZDhjCzxYijaaOxeKSg==",
"dependencies": [
@ -15012,7 +15100,6 @@
"npm:supabase@^2.2.1",
"npm:svelte-check@^4.1.1",
"npm:svelte@^5.16.6",
"npm:tauri-plugin-shellx-api@^2.0.14",
"npm:turbo@^2.3.4",
"npm:typescript@5.7.2",
"npm:valibot@^1.0.0-beta.11",
@ -15089,6 +15176,7 @@
"npm:dompurify@^3.2.3",
"npm:eslint-config-prettier@^9.1.0",
"npm:eslint-plugin-svelte@^2.46.1",
"npm:eslint@^9.21.0",
"npm:globals@^15.14.0",
"npm:gsap@^3.12.5",
"npm:kkrpc@~0.1.1",
@ -15146,9 +15234,9 @@
"npm:node-fetch@^3.3.2",
"npm:semver@^7.6.3",
"npm:svelte-sonner@~0.3.28",
"npm:tauri-api-adapter@~0.3.20",
"npm:tauri-api-adapter@~0.3.23",
"npm:tauri-plugin-network-api@2.0.5",
"npm:tauri-plugin-shellx-api@^2.0.14",
"npm:tauri-plugin-shellx-api@2.0.15",
"npm:tauri-plugin-system-info-api@2.0.8",
"npm:typedoc@~0.27.6",
"npm:typescript@5",
@ -15380,6 +15468,7 @@
"packages/templates/template-ext-sveltekit": {
"packageJson": {
"dependencies": [
"npm:@eslint/js@^9.21.0",
"npm:@kksh/svelte5@0.1.15",
"npm:@sveltejs/adapter-auto@^3.3.1",
"npm:@sveltejs/adapter-static@^3.0.8",
@ -15466,6 +15555,7 @@
"npm:dompurify@^3.2.3",
"npm:eslint-config-prettier@^10.0.1",
"npm:eslint-plugin-svelte@^2.46.1",
"npm:eslint@^9.21.0",
"npm:formsnap@2.0.0-next.1",
"npm:globals@^15.14.0",
"npm:gsap@^3.12.7",
@ -15484,7 +15574,6 @@
"npm:tailwind-variants@0.3",
"npm:tailwindcss-animate@^1.0.7",
"npm:tailwindcss@^3.4.17",
"npm:tauri-plugin-shellx-api@^2.0.14",
"npm:typescript-eslint@^8.20.0",
"npm:valibot@1.0.0-beta.12",
"npm:zod@^3.24.1"

4
deno.ts Normal file
View File

@ -0,0 +1,4 @@
let idx = 0
setInterval(() => {
console.log(idx++)
}, 500)

View File

@ -22,7 +22,7 @@
"typescript": "5.7.2",
"verify-package-export": "^0.0.3"
},
"packageManager": "pnpm@9.15.4",
"packageManager": "pnpm@10.4.1",
"engines": {
"node": ">=22"
},
@ -47,7 +47,6 @@
"supabase": "^2.2.1",
"tauri-plugin-keyring-api": "workspace:*",
"tauri-plugin-network-api": "workspace:*",
"tauri-plugin-shellx-api": "^2.0.14",
"tauri-plugin-system-info-api": "workspace:*",
"valibot": "^1.0.0-beta.11",
"zod": "^3.24.1"

View File

@ -1,5 +1,11 @@
# @kksh/api
## 0.1.4
### Patch Changes
- Add killPid extension API
## 0.1.2
### Patch Changes

View File

@ -1,6 +1,6 @@
{
"name": "@kksh/api",
"version": "0.1.3",
"version": "0.1.4",
"type": "module",
"repository": {
"type": "git",
@ -71,9 +71,9 @@
"node-fetch": "^3.3.2",
"semver": "^7.6.3",
"svelte-sonner": "^0.3.28",
"tauri-api-adapter": "^0.3.20",
"tauri-api-adapter": "^0.3.23",
"tauri-plugin-network-api": "2.0.5",
"tauri-plugin-shellx-api": "^2.0.14",
"tauri-plugin-shellx-api": "^2.0.15",
"tauri-plugin-system-info-api": "2.0.8",
"valibot": "^1.0.0-beta.10"
},

View File

@ -126,7 +126,11 @@ export type IKunkunFullServerAPI = {
*/
export function constructJarvisServerAPIWithPermissions(
permissions: AllPermissions[],
extPath: string
extPath: string,
customFunctions: {
recordSpawnedProcess: (pid: number) => Promise<void>
getSpawnedProcesses: () => Promise<number[]>
}
): IKunkunFullServerAPI {
return {
clipboard: constructClipboardApi(
@ -193,7 +197,9 @@ export function constructJarvisServerAPIWithPermissions(
p.permission.startsWith("shell:")
)
],
extPath
extPath,
customFunctions.recordSpawnedProcess,
customFunctions.getSpawnedProcesses
),
iframeUi: constructIframeUiApi(),
utils: constructUtilsApi(),

View File

@ -2,6 +2,7 @@ import { emitKillProcessEvent } from "@kksh/api/events"
import { Channel, invoke } from "@tauri-apps/api/core"
import { emitTo } from "@tauri-apps/api/event"
import { getCurrentWindow } from "@tauri-apps/api/window"
import { toast } from "svelte-sonner"
import {
hasCommand,
whereIsCommand,
@ -10,6 +11,7 @@ import {
type InternalSpawnOptions,
type IOPayload
} from "tauri-plugin-shellx-api"
import * as shell from "tauri-plugin-shellx-api"
import type { DenoRunConfig } from "../../api/client"
import type { IShellServer } from "../../api/server-types"
import { RECORD_EXTENSION_PROCESS_EVENT, type IRecordExtensionProcessEvent } from "../../events"
@ -73,13 +75,14 @@ async function verifyShellCmdPermission(
*/
export function constructShellApi(
permissions: (ShellPermissionScoped | ShellPermission)[],
extPath: string
extPath: string,
recordSpawnedProcess: (pid: number) => Promise<void>,
getSpawnedProcesses: () => Promise<number[]>
): IShellServer {
const stringPermissiongs = permissions.filter((p) => typeof p === "string") as ShellPermission[]
const objectPermissions = permissions.filter(
(p) => typeof p !== "string"
) as ShellPermissionScoped[]
async function execute(
program: string,
args: string[],
@ -99,14 +102,19 @@ export function constructShellApi(
options: options
})
}
function kill(pid: number) {
if (!stringPermissiongs.some((p) => ShellPermissionMap.kill.includes(p)))
async function kill(pid: number) {
if (!stringPermissiongs.some((p) => ShellPermissionMap.kill.includes(p))) {
return Promise.reject(
new Error(`Permission denied. Requires one of ${ShellPermissionMap.kill}`)
)
}
const pids = await getSpawnedProcesses()
if (!pids.includes(pid)) {
return Promise.reject(new Error(`Process ${pid} not spawned by this extension`))
}
return invoke<void>("plugin:shellx|kill", {
cmd: "killChild",
pid: pid
pid
}).then(() => {
emitKillProcessEvent(pid)
})
@ -146,13 +154,24 @@ export function constructShellApi(
options: InternalSpawnOptions,
cb: (evt: CommandEvent<O>) => void
) {
await verifyShellCmdPermission(ShellPermissionMap.rawSpawn, objectPermissions, program, args)
await verifyShellCmdPermission(
ShellPermissionMap.rawSpawn,
objectPermissions,
program,
args
).catch((err) => {
toast.error("Permission denied", {
description: err.message
})
console.error("rawSpawn permission denied", err)
throw err
})
const onEvent = new Channel<CommandEvent<O>>()
onEvent.onmessage = cb
return invoke<number>("plugin:shellx|spawn", {
program: program,
args: args,
options: options,
program: "deno",
args: ["run", "/Users/hk/Dev/kunkun/deno.ts"],
options,
onEvent
})
}
@ -210,6 +229,14 @@ export function constructShellApi(
return likelyOnWindows()
}
function killPid(pid: number) {
if (!stringPermissiongs.some((p) => ShellPermissionMap.killPid.includes(p)))
return Promise.reject(
new Error(`Permission denied. Requires one of ${ShellPermissionMap.killPid}`)
)
return shell.killPid(pid)
}
return {
whereIsCommand(command: string): Promise<string | null> {
const cleanedCommand = command.trim().split(" ")[0]
@ -218,17 +245,7 @@ export function constructShellApi(
}
return whereIsCommand(cleanedCommand).then((res) => (res === "" ? null : res))
},
async recordSpawnedProcess(pid: number): Promise<void> {
// get window label
const curWin = await getCurrentWindow()
console.log("recordSpawnedProcess", pid, curWin.label)
await emitTo("main", RECORD_EXTENSION_PROCESS_EVENT, {
windowLabel: curWin.label,
pid
} satisfies IRecordExtensionProcessEvent)
// TODO: record process in a store
return Promise.resolve()
},
recordSpawnedProcess,
async denoExecute(
scriptPath: string,
config: DenoRunConfig,
@ -247,7 +264,6 @@ export function constructShellApi(
args1,
extPath
)
console.log("denoExecute", program, args, options)
return invoke<ChildProcess<IOPayload>>("plugin:shellx|execute", {
program,
args,
@ -286,6 +302,7 @@ export function constructShellApi(
},
execute,
kill,
killPid,
stdinWrite,
open,
rawSpawn,

View File

@ -59,8 +59,8 @@ export class Child {
*
* @since 2.0.0
*/
async kill(): Promise<void> {
this.api.kill(this.pid)
kill(): Promise<void> {
return this.api.kill(this.pid)
// await invoke("plugin:shellx|kill", {
// cmd: "killChild",
// pid: this.pid
@ -184,6 +184,7 @@ export class DenoCommand<O extends IOPayload> extends BaseShellCommand<O> {
}
})
.then(async (pid) => {
console.log("spawned deno process", pid)
await this.api.recordSpawnedProcess(pid)
return new Child(pid, this.api)
})
@ -232,6 +233,7 @@ export type IShell = {
}>
RPCChannel: typeof RPCChannel
whereIsCommand: (command: string) => Promise<string | null>
killPid: (pid: number) => Promise<void>
}
export class TauriShellStdio implements IoInterface {
@ -347,6 +349,7 @@ export function constructShellAPI(api: IShellServer): IShell {
return {
open: api.open,
killPid: api.killPid,
makeBashScript,
makePowershellScript,
makeAppleScript,

View File

@ -24,6 +24,7 @@ export const permissionDescriptions: PermissionDescriptions = {
"shell:execute": "Allows executing shell commands",
"shell:spawn": "Allow spawning a new process and listen to the streaming of its output",
"shell:kill": "Allows killing processes by pid. Need this to kill the process you started.",
"shell:kill-any": "Allows killing any process by pid",
"shell:all":
"Grant all shell related permissions. Path scope and args regex validation is still required.",
"shell:stdin-write": "Allows writing to a command created.",

View File

@ -98,6 +98,7 @@ export const EventPermissionMap: Record<keyof IEvent, EventPermission[]> = {
export const ShellPermissionMap: Record<keyof IShellServer, ShellPermission[]> = {
execute: ["shell:all", "shell:execute"],
kill: ["shell:all", "shell:kill"],
killPid: ["shell:all", "shell:kill-any"],
stdinWrite: ["shell:all", "shell:stdin-write", "shell:execute"],
open: ["shell:all", "shell:open"],
rawSpawn: ["shell:all", "shell:spawn"],

View File

@ -133,6 +133,7 @@ export const ShellPermissionSchema = v.union([
v.literal("shell:deno:spawn"),
v.literal("shell:open"),
v.literal("shell:kill"),
v.literal("shell:kill-any"),
v.literal("shell:all"),
v.literal("shell:stdin-write")
])

View File

@ -1,5 +1,12 @@
# demo-template-extension
## 0.0.10
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.4
## 0.0.9
### Patch Changes

View File

@ -1,7 +1,7 @@
{
"$schema": "../../schema/manifest-json-schema.json",
"name": "demo-template-extension",
"version": "0.0.9",
"version": "0.0.10",
"type": "module",
"license": "MIT",
"kunkun": {
@ -11,14 +11,27 @@
"identifier": "demo-worker-template-ext",
"permissions": [
"fetch:all",
"shell:kill",
"security:mac:all",
"clipboard:read-all",
{
"permission": "shell:spawn",
"allow": [
{
"cmd": {
"program": "deno",
"args": [
"run",
"/Users/hk/Dev/kunkun/deno.ts"
]
}
}
]
},
{
"permission": "shell:deno:spawn",
"allow": [
{
"path": "$EXTENSION/deno-src/rpc.ts",
"path": "/Users/hk/Dev/kunkun/deno.ts",
"env": "*",
"ffi": "*",
"read": "*",
@ -27,6 +40,8 @@
}
]
},
"shell:stdin-write",
"shell:kill",
{
"permission": "open:file",
"allow": [
@ -35,7 +50,6 @@
}
]
},
"shell:stdin-write",
{
"permission": "shell:execute",
"allow": [

View File

@ -1,9 +1,8 @@
import { expose, HeadlessCommand, toast } from "@kksh/api/headless"
import { expose, HeadlessCommand, shell, toast } from "@kksh/api/headless"
class DemoHeadlessExt extends HeadlessCommand {
load(): Promise<void> {
console.log("Demo Headless Extension Loaded")
toast.info("Demo Headless Extension Loaded")
shell.killPid(84812)
return Promise.resolve()
}
}

View File

@ -57,7 +57,23 @@ class ExtensionTemplate extends TemplateUiCommand {
ui.showLoadingBar(false)
}, 2000)
const extPath = await path.extensionDir()
// console.log("Extension path:", extPath)
const cmd = shell.createCommand("deno", ["run", "/Users/hk/Dev/kunkun/deno.ts"])
cmd.stdout.on("data", (data) => {
console.log("within ext stdout", data)
})
const child = await cmd.spawn()
console.log("in ext child", child)
setTimeout(() => {
child
.kill()
.then(() => {
console.log("child killed")
})
.catch((err) => {
console.error("child kill error", err)
})
}, 5000)
const tagList = new List.ItemDetailMetadataTagList({
title: "Tag List Title",
tags: [

View File

@ -1,5 +1,12 @@
# template-ext-sveltekit
## 0.0.10
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.4
## 0.0.9
### Patch Changes

View File

@ -1,7 +1,7 @@
{
"$schema": "https://schema.kunkun.sh",
"name": "ext-sveltekit-exp",
"version": "0.0.9",
"version": "0.0.10",
"license": "MIT",
"kunkun": {
"name": "TODO: Change Display Name",

View File

@ -1,5 +1,12 @@
# template-ext-worker
## 0.0.9
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.4
## 0.0.8
### Patch Changes

View File

@ -1,7 +1,7 @@
{
"$schema": "./node_modules/@kksh/api/dist/schema.json",
"name": "template-ext-headless",
"version": "0.0.8",
"version": "0.0.9",
"license": "MIT",
"type": "module",
"kunkun": {

View File

@ -1,5 +1,12 @@
# template-ext-next
## 0.1.8
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.4
## 0.1.7
### Patch Changes

View File

@ -1,7 +1,7 @@
{
"$schema": "./node_modules/@kksh/api/dist/schema.json",
"name": "template-ext-next",
"version": "0.1.7",
"version": "0.1.8",
"license": "MIT",
"kunkun": {
"name": "TODO: Change Display Name",

View File

@ -1,5 +1,12 @@
# template-ext-nuxt
## 0.0.10
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.4
## 0.0.9
### Patch Changes

View File

@ -1,7 +1,7 @@
{
"$schema": "./node_modules/@kksh/api/dist/schema.json",
"name": "template-ext-nuxt",
"version": "0.0.9",
"version": "0.0.10",
"type": "module",
"license": "MIT",
"kunkun": {

View File

@ -1,5 +1,12 @@
# template-ext-react
## 0.0.9
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.4
## 0.0.8
### Patch Changes

View File

@ -2,7 +2,7 @@
"$schema": "./node_modules/@kksh/api/dist/schema.json",
"name": "template-ext-react",
"license": "MIT",
"version": "0.0.8",
"version": "0.0.9",
"type": "module",
"kunkun": {
"name": "TODO: Change Display Name",

View File

@ -1,5 +1,12 @@
# template-ext-svelte
## 0.0.9
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.4
## 0.0.8
### Patch Changes

View File

@ -2,7 +2,7 @@
"$schema": "./node_modules/@kksh/api/dist/schema.json",
"name": "template-ext-svelte",
"license": "MIT",
"version": "0.0.8",
"version": "0.0.9",
"type": "module",
"kunkun": {
"name": "TODO: Change Display Name",

View File

@ -1,5 +1,12 @@
# template-ext-sveltekit
## 0.0.10
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.4
## 0.0.9
### Patch Changes

View File

@ -1,7 +1,7 @@
{
"$schema": "./node_modules/@kksh/api/dist/schema.json",
"name": "template-ext-sveltekit",
"version": "0.0.9",
"version": "0.0.10",
"license": "MIT",
"kunkun": {
"name": "TODO: Change Display Name",
@ -54,14 +54,11 @@
"tailwind-variants": "^0.3.0"
},
"devDependencies": {
"@eslint/js": "^9.21.0",
"@sveltejs/adapter-auto": "^3.3.1",
"@sveltejs/adapter-static": "^3.0.8",
"@sveltejs/kit": "^2.15.2",
"@sveltejs/vite-plugin-svelte": "^5.0.3",
"svelte": "^5.16.6",
"svelte-check": "^4.1.1",
"typescript": "^5.7.2",
"vite": "^6.0.7",
"@sveltejs/adapter-static": "^3.0.8",
"@tailwindcss/typography": "^0.5.16",
"@types/eslint": "^9.6.1",
"autoprefixer": "^10.4.20",
@ -73,8 +70,12 @@
"prettier": "^3.4.2",
"prettier-plugin-svelte": "^3.3.2",
"prettier-plugin-tailwindcss": "^0.6.9",
"svelte": "^5.16.6",
"svelte-check": "^4.1.1",
"tailwindcss": "^3.4.17",
"typescript-eslint": "^8.19.1"
"typescript": "^5.7.2",
"typescript-eslint": "^8.19.1",
"vite": "^6.0.7"
},
"type": "module",
"files": [

View File

@ -1,5 +1,12 @@
# template-ext-vue
## 0.0.7
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.4
## 0.0.6
### Patch Changes

View File

@ -1,7 +1,7 @@
{
"name": "template-ext-vue",
"license": "MIT",
"version": "0.0.6",
"version": "0.0.7",
"type": "module",
"scripts": {
"dev": "vite",

View File

@ -1,5 +1,12 @@
# template-ext-worker
## 0.0.9
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.4
## 0.0.8
### Patch Changes

View File

@ -1,7 +1,7 @@
{
"$schema": "./node_modules/@kksh/api/dist/schema.json",
"name": "template-ext-worker",
"version": "0.0.8",
"version": "0.0.9",
"license": "MIT",
"type": "module",
"kunkun": {

View File

@ -46,6 +46,7 @@
"@typescript-eslint/parser": "^8.20.0",
"bits-ui": "1.0.0-next.77",
"clsx": "^2.1.1",
"eslint": "^9.21.0",
"eslint-config-prettier": "^10.0.1",
"eslint-plugin-svelte": "^2.46.1",
"formsnap": "2.0.0-next.1",
@ -61,17 +62,17 @@
"tailwind-variants": "^0.3.0",
"tailwindcss": "^3.4.17",
"tailwindcss-animate": "^1.0.7",
"tauri-plugin-shellx-api": "^2.0.14",
"tauri-plugin-shellx-api": "^2.0.15",
"typescript-eslint": "^8.20.0",
"zod": "^3.24.1"
},
"dependencies": {
"@formkit/auto-animate": "^0.8.2",
"@shikijs/langs": "^2.3.2",
"@shikijs/themes": "^2.3.2",
"@inlang/paraglide-sveltekit": "^0.15.5",
"@internationalized/date": "^3.7.0",
"@kksh/supabase": "workspace:*",
"@shikijs/langs": "^2.3.2",
"@shikijs/themes": "^2.3.2",
"@std/semver": "npm:@jsr/std__semver@^1.0.3",
"dompurify": "^3.2.3",
"gsap": "^3.12.7",

543
pnpm-lock.yaml generated

File diff suppressed because it is too large Load Diff

View File

@ -5,6 +5,7 @@ packages:
- "packages/templates/*"
- "packages/tauri-plugins/*"
- "vendors/tauri-plugin-network"
- "vendors/tauri-plugin-shellx"
- "vendors/tauri-plugin-keyring"
- "vendors/tauri-plugin-system-info"
- "vendors/tauri-plugin-user-input"