mirror of
https://github.com/kunkunsh/kunkun.git
synced 2025-04-03 22:26:43 +00:00
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:
parent
66135624b9
commit
a92c266d32
@ -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
6
Cargo.lock
generated
@ -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",
|
||||
|
@ -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"
|
||||
|
@ -1,5 +1,12 @@
|
||||
# kksh
|
||||
|
||||
## 0.1.2
|
||||
|
||||
### Patch Changes
|
||||
|
||||
- Updated dependencies
|
||||
- @kksh/api@0.1.4
|
||||
|
||||
## 0.1.1
|
||||
|
||||
### Patch Changes
|
||||
|
@ -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",
|
||||
|
@ -1,5 +1,12 @@
|
||||
# create-kunkun
|
||||
|
||||
## 0.1.48
|
||||
|
||||
### Patch Changes
|
||||
|
||||
- Updated dependencies
|
||||
- @kksh/api@0.1.4
|
||||
|
||||
## 0.1.45
|
||||
|
||||
### Patch Changes
|
||||
|
@ -1,7 +1,7 @@
|
||||
{
|
||||
"name": "create-kunkun",
|
||||
"type": "module",
|
||||
"version": "0.1.47",
|
||||
"version": "0.1.48",
|
||||
"bin": {
|
||||
"create-kunkun": "dist/index.mjs"
|
||||
},
|
||||
|
@ -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"
|
||||
},
|
||||
|
@ -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"
|
||||
|
@ -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",
|
||||
|
@ -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,
|
||||
|
@ -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
|
||||
|
@ -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)
|
||||
})
|
||||
)
|
||||
|
@ -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}
|
||||
|
@ -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(() => {
|
||||
|
@ -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
121
deno.lock
generated
@ -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
4
deno.ts
Normal file
@ -0,0 +1,4 @@
|
||||
let idx = 0
|
||||
setInterval(() => {
|
||||
console.log(idx++)
|
||||
}, 500)
|
@ -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"
|
||||
|
@ -1,5 +1,11 @@
|
||||
# @kksh/api
|
||||
|
||||
## 0.1.4
|
||||
|
||||
### Patch Changes
|
||||
|
||||
- Add killPid extension API
|
||||
|
||||
## 0.1.2
|
||||
|
||||
### Patch Changes
|
||||
|
@ -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"
|
||||
},
|
||||
|
@ -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(),
|
||||
|
@ -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,
|
||||
|
@ -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,
|
||||
|
@ -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.",
|
||||
|
@ -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"],
|
||||
|
@ -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")
|
||||
])
|
||||
|
@ -1,5 +1,12 @@
|
||||
# demo-template-extension
|
||||
|
||||
## 0.0.10
|
||||
|
||||
### Patch Changes
|
||||
|
||||
- Updated dependencies
|
||||
- @kksh/api@0.1.4
|
||||
|
||||
## 0.0.9
|
||||
|
||||
### Patch Changes
|
||||
|
@ -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": [
|
||||
|
@ -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()
|
||||
}
|
||||
}
|
||||
|
@ -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: [
|
||||
|
@ -1,5 +1,12 @@
|
||||
# template-ext-sveltekit
|
||||
|
||||
## 0.0.10
|
||||
|
||||
### Patch Changes
|
||||
|
||||
- Updated dependencies
|
||||
- @kksh/api@0.1.4
|
||||
|
||||
## 0.0.9
|
||||
|
||||
### Patch Changes
|
||||
|
@ -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",
|
||||
|
@ -1,5 +1,12 @@
|
||||
# template-ext-worker
|
||||
|
||||
## 0.0.9
|
||||
|
||||
### Patch Changes
|
||||
|
||||
- Updated dependencies
|
||||
- @kksh/api@0.1.4
|
||||
|
||||
## 0.0.8
|
||||
|
||||
### Patch Changes
|
||||
|
@ -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": {
|
||||
|
@ -1,5 +1,12 @@
|
||||
# template-ext-next
|
||||
|
||||
## 0.1.8
|
||||
|
||||
### Patch Changes
|
||||
|
||||
- Updated dependencies
|
||||
- @kksh/api@0.1.4
|
||||
|
||||
## 0.1.7
|
||||
|
||||
### Patch Changes
|
||||
|
@ -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",
|
||||
|
@ -1,5 +1,12 @@
|
||||
# template-ext-nuxt
|
||||
|
||||
## 0.0.10
|
||||
|
||||
### Patch Changes
|
||||
|
||||
- Updated dependencies
|
||||
- @kksh/api@0.1.4
|
||||
|
||||
## 0.0.9
|
||||
|
||||
### Patch Changes
|
||||
|
@ -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": {
|
||||
|
@ -1,5 +1,12 @@
|
||||
# template-ext-react
|
||||
|
||||
## 0.0.9
|
||||
|
||||
### Patch Changes
|
||||
|
||||
- Updated dependencies
|
||||
- @kksh/api@0.1.4
|
||||
|
||||
## 0.0.8
|
||||
|
||||
### Patch Changes
|
||||
|
@ -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",
|
||||
|
@ -1,5 +1,12 @@
|
||||
# template-ext-svelte
|
||||
|
||||
## 0.0.9
|
||||
|
||||
### Patch Changes
|
||||
|
||||
- Updated dependencies
|
||||
- @kksh/api@0.1.4
|
||||
|
||||
## 0.0.8
|
||||
|
||||
### Patch Changes
|
||||
|
@ -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",
|
||||
|
@ -1,5 +1,12 @@
|
||||
# template-ext-sveltekit
|
||||
|
||||
## 0.0.10
|
||||
|
||||
### Patch Changes
|
||||
|
||||
- Updated dependencies
|
||||
- @kksh/api@0.1.4
|
||||
|
||||
## 0.0.9
|
||||
|
||||
### Patch Changes
|
||||
|
@ -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": [
|
||||
|
@ -1,5 +1,12 @@
|
||||
# template-ext-vue
|
||||
|
||||
## 0.0.7
|
||||
|
||||
### Patch Changes
|
||||
|
||||
- Updated dependencies
|
||||
- @kksh/api@0.1.4
|
||||
|
||||
## 0.0.6
|
||||
|
||||
### Patch Changes
|
||||
|
@ -1,7 +1,7 @@
|
||||
{
|
||||
"name": "template-ext-vue",
|
||||
"license": "MIT",
|
||||
"version": "0.0.6",
|
||||
"version": "0.0.7",
|
||||
"type": "module",
|
||||
"scripts": {
|
||||
"dev": "vite",
|
||||
|
@ -1,5 +1,12 @@
|
||||
# template-ext-worker
|
||||
|
||||
## 0.0.9
|
||||
|
||||
### Patch Changes
|
||||
|
||||
- Updated dependencies
|
||||
- @kksh/api@0.1.4
|
||||
|
||||
## 0.0.8
|
||||
|
||||
### Patch Changes
|
||||
|
@ -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": {
|
||||
|
@ -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
543
pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
@ -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"
|
||||
|
Loading…
x
Reference in New Issue
Block a user