[feat] troubleshooters (#15)

* feat: add extension loading troubleshooter

* feat: add extension permission inspector

* feat: add extension window map troubleshooter (WIP)

* fix: unregister extension when window is closed
This commit is contained in:
Huakun Shen 2024-11-05 06:04:34 -05:00 committed by GitHub
parent f64e562034
commit 2c99f231f7
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
14 changed files with 456 additions and 48 deletions

View File

@ -30,7 +30,8 @@
"semver": "^7.6.3",
"svelte-radix": "^2.0.1",
"svelte-sonner": "^0.3.28",
"sveltekit-superforms": "^2.20.0"
"sveltekit-superforms": "^2.20.0",
"uuid": "^11.0.2"
},
"devDependencies": {
"@kksh/types": "workspace:*",

View File

@ -2,10 +2,12 @@ import { appConfig, appState } from "@/stores"
import { checkUpdateAndInstall } from "@/utils/updater"
import type { BuiltinCmd } from "@kksh/ui/types"
import { getVersion } from "@tauri-apps/api/app"
import { WebviewWindow } from "@tauri-apps/api/webviewWindow"
import { exit } from "@tauri-apps/plugin-process"
import { dev } from "$app/environment"
import { goto } from "$app/navigation"
import { toast } from "svelte-sonner"
import { v4 as uuidv4 } from "uuid"
export const builtinCmds: BuiltinCmd[] = [
{
@ -79,42 +81,39 @@ export const builtinCmds: BuiltinCmd[] = [
goto("/settings/set-dev-ext-path")
}
},
// {
// name: "Extension Window Troubleshooter",
// iconifyIcon: "material-symbols:window-outline",
// description: "",
// function: async () => {
// const appStateStore = useAppStateStore()
// appStateStore.setSearchTermSync("")
// // goto("/window-troubleshooter")
// const winLabel = `main:window-troubleshooter-${uuidv4()}`
// console.log(winLabel)
// new WebviewWindow(winLabel, {
// url: "/window-troubleshooter",
// title: "Window Troubleshooter"
// })
// }
// },
// {
// name: "Extension Permission Inspector",
// iconifyIcon: "hugeicons:inspect-code",
// description: "",
// function: async () => {
// const appStateStore = useAppStateStore()
// appStateStore.setSearchTermSync("")
// goto("/ext-permission-inspector")
// }
// },
// {
// name: "Extension Loading Troubleshooter",
// iconifyIcon: "material-symbols:troubleshoot",
// description: "",
// function: async () => {
// const appStateStore = useAppStateStore()
// appStateStore.setSearchTermSync("")
// goto("/extension-load-troubleshooter")
// }
// },
{
name: "Extension Window Troubleshooter",
iconifyIcon: "material-symbols:window-outline",
description: "",
function: async () => {
appState.clearSearchTerm()
// goto("/window-troubleshooter")
const winLabel = `main:extension-window-troubleshooter-${uuidv4()}`
console.log(winLabel)
new WebviewWindow(winLabel, {
url: "/troubleshooters/extension-window",
title: "Extension Window Troubleshooter"
})
}
},
{
name: "Extension Permission Inspector",
iconifyIcon: "hugeicons:inspect-code",
description: "",
function: async () => {
appState.clearSearchTerm()
goto("/extension/permission-inspector")
}
},
{
name: "Extension Loading Troubleshooter",
iconifyIcon: "material-symbols:troubleshoot",
description: "",
function: async () => {
appState.clearSearchTerm()
goto("/troubleshooters/extension-loading")
}
},
// {
// name: "Create Quicklink",
// iconifyIcon: "material-symbols:link",

View File

@ -1,3 +1,4 @@
import { appState } from "@/stores"
import { winExtMap } from "@/stores/winExtMap"
import { trimSlash } from "@/utils/url"
import { constructExtensionSupportDir } from "@kksh/api"
@ -48,6 +49,9 @@ export async function onCustomUiCmdSelect(
})
console.log("Launch new window, ", winLabel)
const window = launchNewExtWindow(winLabel, url2, cmd.window)
window.onCloseRequested(async (event) => {
await winExtMap.unregisterExtensionFromWindow(winLabel)
})
} else {
console.log("Launch main window")
return winExtMap
@ -58,4 +62,5 @@ export async function onCustomUiCmdSelect(
})
.then(() => goto(url2))
}
appState.clearSearchTerm()
}

View File

@ -63,8 +63,6 @@ passing everything through props will be very complicated and hard to maintain.
/>
<Command.List class="max-h-screen grow">
<Command.Empty data-tauri-drag-region>No results found.</Command.Empty>
<BuiltinCmds {builtinCmds} />
<SystemCmds {systemCommands} />
{#if $appConfig.extensionsInstallDir && $devStoreExts.length > 0}
<ExtCmdsGroup
extensions={$devStoreExts}
@ -83,6 +81,8 @@ passing everything through props will be very complicated and hard to maintain.
onExtCmdSelect={commandLaunchers.onExtCmdSelect}
/>
{/if}
<BuiltinCmds {builtinCmds} />
<SystemCmds {systemCommands} />
<Command.Separator />
</Command.List>
<GlobalCommandPaletteFooter />

View File

@ -58,11 +58,11 @@ function createWinExtMapStore(): Writable<WinExtMap> & API {
await killProcesses(winExtMap[windowLabel].pids)
delete winExtMap[windowLabel]
} else {
winExtMap[windowLabel] = {
windowLabel,
extPath,
pids: []
}
// winExtMap[windowLabel] = {
// windowLabel,
// extPath,
// pids: []
// }
}
}
const returnedWinLabel = await registerExtensionWindow({
@ -70,6 +70,11 @@ function createWinExtMapStore(): Writable<WinExtMap> & API {
windowLabel,
dist
})
winExtMap[returnedWinLabel] = {
windowLabel: returnedWinLabel,
extPath,
pids: []
}
store.set(winExtMap)
return returnedWinLabel
},

View File

@ -0,0 +1,57 @@
import { DEEP_LINK_PATH_REFRESH_DEV_EXTENSION } from "@kksh/api"
import {
emit,
emitTo,
listen,
TauriEvent,
type Event,
type EventCallback,
type UnlistenFn
} from "@tauri-apps/api/event"
export const FileDragDrop = "tauri://drag-drop"
export const FileDragEnter = "tauri://drag-enter"
export const FileDragLeave = "tauri://drag-leave"
export const FileDragOver = "tauri://drag-over"
export const NewClipboardItemAddedEvent = "new_clipboard_item_added"
export const RefreshConfigEvent = "kunkun://refresh-config"
export const RefreshExtEvent = "kunkun://refresh-extensions"
export function listenToFileDrop(cb: EventCallback<{ paths: string[] }>) {
return listen<{ paths: string[] }>(FileDragDrop, cb)
}
export function listenToWindowBlur(cb: EventCallback<null>) {
return listen(TauriEvent.WINDOW_BLUR, cb)
}
export function listenToWindowFocus(cb: EventCallback<null>) {
return listen(TauriEvent.WINDOW_FOCUS, cb)
}
export function listenToNewClipboardItem(cb: EventCallback<null>) {
return listen(NewClipboardItemAddedEvent, cb)
}
export function emitRefreshConfig() {
return emit(RefreshConfigEvent)
}
export function listenToRefreshConfig(cb: EventCallback<null>) {
return listen(RefreshConfigEvent, cb)
}
export function emitRefreshExt() {
return emitTo("main", RefreshExtEvent)
}
export function listenToRefreshExt(cb: EventCallback<null>) {
return listen(RefreshExtEvent, cb)
}
export function emitRefreshDevExt() {
return emit(DEEP_LINK_PATH_REFRESH_DEV_EXTENSION)
}
export function listenToRefreshDevExt(cb: EventCallback<null>) {
return listen(DEEP_LINK_PATH_REFRESH_DEV_EXTENSION, cb)
}

View File

@ -0,0 +1,100 @@
<script lang="ts">
import { goBackOnEscape } from "@/utils/key"
import { goBack } from "@/utils/route"
import { listenToFileDrop } from "@/utils/tauri-events"
import type { ExtPackageJsonExtra } from "@kksh/api/models"
import { loadExtensionManifestFromDisk } from "@kksh/extension"
import { Button, Card } from "@kksh/svelte5"
import { PermissionInspector } from "@kksh/ui/extension"
import type { UnlistenFn } from "@tauri-apps/api/event"
import { join } from "@tauri-apps/api/path"
import { getCurrentWebview } from "@tauri-apps/api/webview"
import { open as openDialog } from "@tauri-apps/plugin-dialog"
import { exists } from "@tauri-apps/plugin-fs"
import { ArrowLeftIcon } from "lucide-svelte"
import { onDestroy, onMount } from "svelte"
import { toast } from "svelte-sonner"
let pkgJsons = $state<ExtPackageJsonExtra[]>([])
let unlistenDropEvt: UnlistenFn
onMount(async () => {
unlistenDropEvt = await getCurrentWebview().onDragDropEvent((event) => {
if (event.payload.type === "drop") {
inspectPaths(event.payload.paths)
}
})
})
onDestroy(() => {
unlistenDropEvt?.()
})
async function inspectPaths(paths: string[]) {
for (const path of paths) {
if (!(await exists(path))) {
toast.error("Selected path does not exist", { description: path })
continue
}
const manifestPath = await join(path, "package.json")
if (!(await exists(manifestPath))) {
toast.error("Selected path is not an extension", { description: path })
continue
}
try {
pkgJsons.push(await loadExtensionManifestFromDisk(manifestPath))
toast.success("Extension manifest loaded", { description: path })
} catch (err) {
toast.error(`Failed to load extension manifest: ${err}`, { description: path })
}
}
}
async function onPick() {
const paths = await openDialog({
directory: true,
multiple: true
})
if (!paths) {
return toast.error("No folder selected")
}
inspectPaths(paths)
}
</script>
<svelte:window on:keydown={goBackOnEscape} />
<main class="container w-screen pt-10">
<Button variant="outline" size="icon" class="absolute left-2 top-2 z-50" onclick={goBack}>
<ArrowLeftIcon class="h-4 w-4" />
</Button>
<h1 class="text-2xl font-bold">Extension Permission Inspector</h1>
<Button class="my-5" onclick={onPick}>Pick Extension Folder to Inspect</Button>
<div class="mb-5 flex flex-col gap-4">
{#each pkgJsons as pkgJson}
<Card.Root>
<Card.Header>
<Card.Title>{pkgJson.kunkun.name}</Card.Title>
<Card.Description>{pkgJson.kunkun.shortDescription}</Card.Description>
</Card.Header>
<Card.Content>
<PermissionInspector manifest={pkgJson.kunkun} />
</Card.Content>
<Card.Footer class="block">
<p class="text-sm">
<strong>Identifier:</strong> <code>{pkgJson.kunkun.identifier}</code>
</p>
<p class="text-sm">
<strong>Extension Path:</strong> <code>{pkgJson.extPath}</code>
</p>
</Card.Footer>
</Card.Root>
{/each}
</div>
</main>
<style>
:global(body) {
overflow-x: hidden;
}
</style>

View File

@ -6,7 +6,7 @@
import { ArrowLeftIcon } from "lucide-svelte"
</script>
<svelte:window on:keydown|preventDefault={goBackOnEscape} />
<svelte:window on:keydown={goBackOnEscape} />
<Button variant="outline" size="icon" class="absolute left-2 top-2 z-50" onclick={goBack}>
<ArrowLeftIcon class="h-4 w-4" />
</Button>

View File

@ -0,0 +1,129 @@
<script lang="ts">
import { goBackOnEscape } from "@/utils/key"
import { goBack } from "@/utils/route"
import { db } from "@kksh/api/commands"
import { loadExtensionManifestFromDisk } from "@kksh/extension"
import { Button, Dialog, ScrollArea, Table } from "@kksh/svelte5"
import { join } from "@tauri-apps/api/path"
import { exists } from "@tauri-apps/plugin-fs"
import { ArrowLeftIcon } from "lucide-svelte"
import { onMount } from "svelte"
import { toast } from "svelte-sonner"
import { open } from "tauri-plugin-shellx-api"
type Result = {
identifier: string
path: string
error?: string
}
let results = $state<Result[]>([])
let isDialogOpen = $state(false)
let errorMsg = $state<string | undefined>()
const sortedResults = $derived.by(() =>
results.slice().sort((a, b) => {
return a.error ? -1 : 1
})
)
async function check() {
results = []
const tmpResults = []
const extensions = await db.getAllExtensions()
for (const ext of extensions) {
if (!ext.path) continue
const _exists = await exists(ext.path)
let error: string | undefined = undefined
if (!_exists) {
error = `Extension path (${ext.path}) does not exist`
}
const pkgJsonPath = await join(ext.path, "package.json")
const _pkgJsonExists = await exists(pkgJsonPath)
if (!_pkgJsonExists) {
error = `Extension package.json (${pkgJsonPath}) does not exist`
}
try {
const manifest = await loadExtensionManifestFromDisk(pkgJsonPath)
} catch (err: any) {
error = `Failed to load manifest from ${pkgJsonPath}: ${err.message}`
}
tmpResults.push({
identifier: ext.identifier,
path: ext.path,
error
})
}
results = tmpResults
const numErrors = results.filter((r) => r.error).length
const toastFn = numErrors > 0 ? toast.error : toast.info
toastFn(`${numErrors} errors found`, {
description: numErrors > 0 ? "Click on an error to see more details" : undefined
})
}
function onErrorClick(errMsg?: string) {
if (errMsg) {
isDialogOpen = true
errorMsg = errMsg
} else {
toast.info("No error message")
}
}
onMount(() => {
check()
})
</script>
<svelte:window on:keydown={goBackOnEscape} />
<Button variant="outline" size="icon" class="absolute left-2 top-2 z-50" onclick={goBack}>
<ArrowLeftIcon class="h-4 w-4" />
</Button>
<div class="absolute left-0 top-0 h-10 w-screen" data-tauri-drag-region></div>
<div class="container pt-10">
<h1 class="text-2xl font-bold">Extension Loading Troubleshooter</h1>
<Button class="my-2" onclick={check}>Check</Button>
<Dialog.Root bind:open={isDialogOpen}>
<Dialog.Content class="sm:max-w-[425px]">
<Dialog.Header>
<Dialog.Title>Error Details</Dialog.Title>
</Dialog.Header>
{errorMsg}
</Dialog.Content>
</Dialog.Root>
<Table.Root>
<Table.Caption>A list of your extensions.</Table.Caption>
<Table.Header>
<Table.Row>
<Table.Head class="">Identifier</Table.Head>
<Table.Head>Path</Table.Head>
<Table.Head>Error</Table.Head>
</Table.Row>
</Table.Header>
<Table.Body>
{#each sortedResults as row}
<Table.Row>
<Table.Cell class="font-medium"><pre>{row.identifier}</pre></Table.Cell>
<Table.Cell class="">
<button onclick={() => open(row.path)} class="text-left">
<pre class="cursor-pointer text-wrap">{row.path}</pre>
</button>
</Table.Cell>
<Table.Cell class="text-right">
<button onclick={() => onErrorClick(row.error)}>
{row.error ? "⚠️" : "✅"}
</button>
</Table.Cell>
</Table.Row>
{/each}
</Table.Body>
</Table.Root>
</div>
<style>
:global(body) {
overflow-x: hidden;
}
</style>

View File

@ -0,0 +1,111 @@
<script lang="ts">
import { winExtMap } from "@/stores"
import { goBackOnEscape, goBackOnEscapeClearSearchTerm } from "@/utils/key"
import { goBack, goHome } from "@/utils/route"
import { getExtLabelMap, unregisterExtensionWindow } from "@kksh/api/commands"
import type { ExtensionLabelMap } from "@kksh/api/models"
import { Button, Checkbox, ScrollArea } from "@kksh/svelte5"
import { getCurrentWebviewWindow } from "@tauri-apps/api/webviewWindow"
import { ArrowLeftIcon, TrashIcon } from "lucide-svelte"
import { onMount } from "svelte"
import { toast } from "svelte-sonner"
const appWin = getCurrentWebviewWindow()
let winLabelMap = $state<ExtensionLabelMap>({})
let refreshEverySecond = $state(true)
let refreshCount = $state(0)
async function refresh() {
const extLabelMap = await getExtLabelMap()
winLabelMap = extLabelMap
refreshCount++
}
function refreshWinLabelMapRecursively() {
setTimeout(async () => {
await refresh()
if (refreshEverySecond) {
refreshWinLabelMapRecursively()
}
}, 1000)
}
onMount(async () => {
const extLabelMap = await getExtLabelMap()
winLabelMap = extLabelMap
refreshCount = 1
})
$effect(() => {
if (refreshEverySecond) {
refreshWinLabelMapRecursively()
}
})
function unregisterWindow(label: string) {
// winExtMap
// .unregisterExtensionFromWindow(label)
unregisterExtensionWindow(label)
.then(() => {
toast.success("Unregistered window")
})
.catch((err) => {
toast.error("Failed to unregister window", { description: err.message })
})
}
function onKeyDown(e: KeyboardEvent) {
if (e.key === "Escape") {
if (appWin.label === "main") {
goHome()
} else {
appWin.close()
}
}
}
</script>
<svelte:window on:keydown={onKeyDown} />
<Button variant="outline" size="icon" class="absolute left-2 top-2 z-50" onclick={goBack}>
<ArrowLeftIcon class="h-4 w-4" />
</Button>
<main class="container h-screen w-screen pt-10">
<div class="flex items-center justify-between space-x-2">
<div class="flex items-center space-x-2">
<Checkbox id="refreshEverySecond" bind:checked={refreshEverySecond} />
<label
for="refreshEverySecond"
class="text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70"
>
Refresh Every Second
</label>
</div>
<span class="flex items-center space-x-2">
<Button size="sm" onclick={refresh}>Refresh</Button>
<span>Refreshed {refreshCount} times</span>
</span>
</div>
<ScrollArea class="py-5" orientation="both">
{#each Object.entries(winLabelMap) as [label, content]}
<li>
<span class="flex gap-2">
<strong>Label:</strong>
<pre class="text-lime">{label}</pre>
</span>
<ul class="pl-5">
{#each Object.entries(content) as [key, value]}
<li>
<span class="flex gap-2">
<strong>{key}:</strong>
<pre class="text-lime">{value}</pre>
</span>
</li>
{/each}
</ul>
<Button variant="destructive" size="icon" onclick={() => unregisterWindow(label)}>
<TrashIcon />
</Button>
</li>
{/each}
</ScrollArea>
</main>

View File

@ -26,7 +26,6 @@ export function registerExtensionWindow(options: {
}
export function unregisterExtensionWindow(label: string): Promise<void> {
console.log("unregisterExtensionWindow", label)
return invoke(generateJarvisPluginCommand("unregister_extension_window"), {
label
})

View File

@ -20,7 +20,6 @@
<HoverCard.Root>
<HoverCard.Trigger class="flex items-center">
<IconMultiplexer
class="border"
icon={{
type: IconEnum.Iconify,
value: "material-symbols:info-outline"

View File

@ -126,7 +126,7 @@
/>
</span>
<div class="w-full">
<span class="flex items-center w-full" use:autoAnimate>
<span class="flex w-full items-center" use:autoAnimate>
<strong class="ext-name text-xl">{manifest?.name}</strong>
{#if isInstalled}
<CircleCheckBigIcon class="ml-2 inline text-green-400" />

3
pnpm-lock.yaml generated
View File

@ -165,6 +165,9 @@ importers:
sveltekit-superforms:
specifier: ^2.20.0
version: 2.20.0(@sveltejs/kit@2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(@types/json-schema@7.0.15)(svelte@5.1.9)(typescript@5.6.3)
uuid:
specifier: ^11.0.2
version: 11.0.2
devDependencies:
'@kksh/types':
specifier: workspace:*