259 lines
11 KiB
TypeScript
259 lines
11 KiB
TypeScript
import { Component, Show, For, onMount, createSignal } from "solid-js"
|
|
import type { Instance } from "../types/instance"
|
|
|
|
interface InstanceInfoProps {
|
|
instance: Instance
|
|
compact?: boolean
|
|
}
|
|
|
|
function parseMcpStatus(status: unknown): Array<{ name: string; status: "running" | "stopped" | "error" }> {
|
|
if (!status || typeof status !== "object") return []
|
|
|
|
try {
|
|
const obj = status as Record<string, string>
|
|
return Object.entries(obj).map(([name, statusValue]) => {
|
|
let mappedStatus: "running" | "stopped" | "error"
|
|
|
|
if (statusValue === "connected") {
|
|
mappedStatus = "running"
|
|
} else if (statusValue === "disabled") {
|
|
mappedStatus = "stopped"
|
|
} else if (statusValue === "failed") {
|
|
mappedStatus = "error"
|
|
} else {
|
|
mappedStatus = "stopped"
|
|
}
|
|
|
|
return {
|
|
name,
|
|
status: mappedStatus,
|
|
}
|
|
})
|
|
} catch {
|
|
return []
|
|
}
|
|
}
|
|
|
|
const InstanceInfo: Component<InstanceInfoProps> = (props) => {
|
|
const [isLoadingMetadata, setIsLoadingMetadata] = createSignal(true)
|
|
|
|
const metadata = () => props.instance.metadata
|
|
const mcpServers = () => {
|
|
const status = metadata()?.mcpStatus
|
|
return parseMcpStatus(status)
|
|
}
|
|
|
|
onMount(async () => {
|
|
if (!props.instance.client) {
|
|
setIsLoadingMetadata(false)
|
|
return
|
|
}
|
|
|
|
setIsLoadingMetadata(true)
|
|
try {
|
|
const [projectResult, mcpResult] = await Promise.allSettled([
|
|
props.instance.client.project.current(),
|
|
props.instance.client.mcp.status(),
|
|
])
|
|
|
|
const project = projectResult.status === "fulfilled" ? projectResult.value.data : undefined
|
|
const mcpStatus = mcpResult.status === "fulfilled" ? mcpResult.value.data : undefined
|
|
|
|
const { updateInstance } = await import("../stores/instances")
|
|
updateInstance(props.instance.id, {
|
|
metadata: {
|
|
project,
|
|
mcpStatus,
|
|
version: "0.15.8",
|
|
},
|
|
})
|
|
} catch (error) {
|
|
console.error("Failed to load instance metadata:", error)
|
|
} finally {
|
|
setIsLoadingMetadata(false)
|
|
}
|
|
})
|
|
|
|
return (
|
|
<div class="bg-white dark:bg-gray-800 rounded-lg shadow-sm border border-gray-200 dark:border-gray-700">
|
|
<div class="px-4 py-3 border-b border-gray-200 dark:border-gray-700 bg-gray-50 dark:bg-gray-900">
|
|
<h2 class="text-base font-semibold text-gray-900 dark:text-gray-100">Instance Information</h2>
|
|
</div>
|
|
<div class="p-4 space-y-3">
|
|
<div>
|
|
<div class="text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wide mb-1">Folder</div>
|
|
<div class="text-xs text-gray-900 dark:text-gray-100 font-mono break-all bg-gray-50 dark:bg-gray-900 px-2 py-1.5 rounded border border-gray-200 dark:border-gray-700">
|
|
{props.instance.folder}
|
|
</div>
|
|
</div>
|
|
|
|
<Show when={!isLoadingMetadata() && metadata()?.project}>
|
|
{(project) => (
|
|
<>
|
|
<div>
|
|
<div class="text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wide mb-1">
|
|
Project
|
|
</div>
|
|
<div class="text-xs text-gray-900 dark:text-gray-100 font-mono bg-gray-50 dark:bg-gray-900 px-2 py-1.5 rounded border border-gray-200 dark:border-gray-700 truncate">
|
|
{project().id}
|
|
</div>
|
|
</div>
|
|
|
|
<Show when={project().vcs}>
|
|
<div>
|
|
<div class="text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wide mb-1">
|
|
Version Control
|
|
</div>
|
|
<div class="flex items-center gap-2 text-xs text-gray-900 dark:text-gray-100">
|
|
<svg
|
|
class="w-3.5 h-3.5 text-orange-600 dark:text-orange-500"
|
|
fill="currentColor"
|
|
viewBox="0 0 24 24"
|
|
>
|
|
<path d="M12 0C5.37 0 0 5.37 0 12c0 5.31 3.435 9.795 8.205 11.385.6.105.825-.255.825-.57 0-.285-.015-1.23-.015-2.235-3.015.555-3.795-.735-4.035-1.41-.135-.345-.72-1.41-1.23-1.695-.42-.225-1.02-.78-.015-.795.945-.015 1.62.87 1.845 1.23 1.08 1.815 2.805 1.305 3.495.99.105-.78.42-1.305.765-1.605-2.67-.3-5.46-1.335-5.46-5.925 0-1.305.465-2.385 1.23-3.225-.12-.3-.54-1.53.12-3.18 0 0 1.005-.315 3.3 1.23.96-.27 1.98-.405 3-.405s2.04.135 3 .405c2.295-1.56 3.3-1.23 3.3-1.23.66 1.65.24 2.88.12 3.18.765.84 1.23 1.905 1.23 3.225 0 4.605-2.805 5.625-5.475 5.925.435.375.81 1.095.81 2.22 0 1.605-.015 2.895-.015 3.3 0 .315.225.69.825.57A12.02 12.02 0 0024 12c0-6.63-5.37-12-12-12z" />
|
|
</svg>
|
|
<span class="capitalize">{project().vcs}</span>
|
|
</div>
|
|
</div>
|
|
</Show>
|
|
</>
|
|
)}
|
|
</Show>
|
|
|
|
<Show when={metadata()?.version}>
|
|
<div>
|
|
<div class="text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wide mb-1">
|
|
OpenCode Version
|
|
</div>
|
|
<div class="text-xs text-gray-900 dark:text-gray-100 bg-gray-50 dark:bg-gray-900 px-2 py-1.5 rounded border border-gray-200 dark:border-gray-700">
|
|
v{metadata()?.version}
|
|
</div>
|
|
</div>
|
|
</Show>
|
|
|
|
<Show when={props.instance.binaryPath}>
|
|
<div>
|
|
<div class="text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wide mb-1">
|
|
Binary Path
|
|
</div>
|
|
<div class="text-xs text-gray-900 dark:text-gray-100 font-mono break-all bg-gray-50 dark:bg-gray-900 px-2 py-1.5 rounded border border-gray-200 dark:border-gray-700">
|
|
{props.instance.binaryPath}
|
|
</div>
|
|
</div>
|
|
</Show>
|
|
|
|
<Show when={props.instance.environmentVariables && Object.keys(props.instance.environmentVariables).length > 0}>
|
|
<div>
|
|
<div class="text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wide mb-1.5">
|
|
Environment Variables ({Object.keys(props.instance.environmentVariables!).length})
|
|
</div>
|
|
<div class="space-y-1">
|
|
<For each={Object.entries(props.instance.environmentVariables!)}>
|
|
{([key, value]) => (
|
|
<div class="flex items-center gap-2 px-2 py-1.5 bg-gray-50 dark:bg-gray-900 rounded border border-gray-200 dark:border-gray-700">
|
|
<span class="text-xs text-gray-900 dark:text-gray-100 font-mono font-medium flex-1" title={key}>
|
|
{key}
|
|
</span>
|
|
<span class="text-xs text-gray-600 dark:text-gray-400 font-mono flex-1" title={value}>
|
|
{value}
|
|
</span>
|
|
</div>
|
|
)}
|
|
</For>
|
|
</div>
|
|
</div>
|
|
</Show>
|
|
|
|
<Show when={!isLoadingMetadata() && mcpServers().length > 0}>
|
|
<div>
|
|
<div class="text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wide mb-1.5">
|
|
MCP Servers
|
|
</div>
|
|
<div class="space-y-1.5">
|
|
<For each={mcpServers()}>
|
|
{(server) => (
|
|
<div class="flex items-center justify-between px-2 py-1.5 bg-gray-50 dark:bg-gray-900 rounded border border-gray-200 dark:border-gray-700">
|
|
<span class="text-xs text-gray-900 dark:text-gray-100 font-medium truncate">{server.name}</span>
|
|
<div class="flex items-center gap-1.5 flex-shrink-0">
|
|
<Show
|
|
when={server.status === "running"}
|
|
fallback={
|
|
<Show
|
|
when={server.status === "error"}
|
|
fallback={<div class="w-1.5 h-1.5 rounded-full bg-gray-400" />}
|
|
>
|
|
<div class="w-1.5 h-1.5 rounded-full bg-red-500" />
|
|
</Show>
|
|
}
|
|
>
|
|
<div class="w-1.5 h-1.5 rounded-full bg-green-500 animate-pulse" />
|
|
</Show>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</For>
|
|
</div>
|
|
</div>
|
|
</Show>
|
|
|
|
<Show when={isLoadingMetadata()}>
|
|
<div class="text-xs text-gray-500 dark:text-gray-400 py-1">
|
|
<div class="flex items-center gap-1.5">
|
|
<svg class="animate-spin h-3 w-3" fill="none" viewBox="0 0 24 24">
|
|
<circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4" />
|
|
<path
|
|
class="opacity-75"
|
|
fill="currentColor"
|
|
d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"
|
|
/>
|
|
</svg>
|
|
Loading...
|
|
</div>
|
|
</div>
|
|
</Show>
|
|
|
|
<div>
|
|
<div class="text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wide mb-1.5">Server</div>
|
|
<div class="space-y-1 text-xs">
|
|
<div class="flex justify-between items-center">
|
|
<span class="text-gray-600 dark:text-gray-400">Port:</span>
|
|
<span class="text-gray-900 dark:text-gray-100 font-mono">{props.instance.port}</span>
|
|
</div>
|
|
<div class="flex justify-between items-center">
|
|
<span class="text-gray-600 dark:text-gray-400">PID:</span>
|
|
<span class="text-gray-900 dark:text-gray-100 font-mono">{props.instance.pid}</span>
|
|
</div>
|
|
<div class="flex justify-between items-center">
|
|
<span class="text-gray-600 dark:text-gray-400">Status:</span>
|
|
<span
|
|
class="inline-flex items-center gap-1 px-1.5 py-0.5 rounded text-xs font-medium"
|
|
classList={{
|
|
"bg-green-100 text-green-800 dark:bg-green-900/30 dark:text-green-400":
|
|
props.instance.status === "ready",
|
|
"bg-yellow-100 text-yellow-800 dark:bg-yellow-900/30 dark:text-yellow-400":
|
|
props.instance.status === "starting",
|
|
"bg-red-100 text-red-800 dark:bg-red-900/30 dark:text-red-400": props.instance.status === "error",
|
|
"bg-gray-100 text-gray-800 dark:bg-gray-700 dark:text-gray-300": props.instance.status === "stopped",
|
|
}}
|
|
>
|
|
<div
|
|
class="w-1 h-1 rounded-full"
|
|
classList={{
|
|
"bg-green-600 animate-pulse": props.instance.status === "ready",
|
|
"bg-yellow-600 animate-pulse": props.instance.status === "starting",
|
|
"bg-red-600": props.instance.status === "error",
|
|
"bg-gray-600": props.instance.status === "stopped",
|
|
}}
|
|
/>
|
|
{props.instance.status}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
export default InstanceInfo
|