1579 lines
60 KiB
TypeScript
1579 lines
60 KiB
TypeScript
import React, { useState, useEffect, useMemo, useRef } from "react";
|
|
import { Button } from "@/components/ui/button.tsx";
|
|
import { Badge } from "@/components/ui/badge.tsx";
|
|
import { ScrollArea } from "@/components/ui/scroll-area.tsx";
|
|
import { Input } from "@/components/ui/input.tsx";
|
|
import {
|
|
Accordion,
|
|
AccordionContent,
|
|
AccordionItem,
|
|
AccordionTrigger,
|
|
} from "@/components/ui/accordion.tsx";
|
|
import {
|
|
Tooltip,
|
|
TooltipContent,
|
|
TooltipProvider,
|
|
TooltipTrigger,
|
|
} from "@/components/ui/tooltip.tsx";
|
|
import {
|
|
getSSHHosts,
|
|
deleteSSHHost,
|
|
bulkImportSSHHosts,
|
|
updateSSHHost,
|
|
renameFolder,
|
|
exportSSHHostWithCredentials,
|
|
getSSHFolders,
|
|
updateFolderMetadata,
|
|
deleteAllHostsInFolder,
|
|
getServerStatusById,
|
|
} from "@/ui/main-axios.ts";
|
|
import { toast } from "sonner";
|
|
import { useTranslation } from "react-i18next";
|
|
import { useConfirmation } from "@/hooks/use-confirmation.ts";
|
|
import { Status, StatusIndicator } from "@/components/ui/shadcn-io/status";
|
|
import {
|
|
Edit,
|
|
Trash2,
|
|
Server,
|
|
Folder,
|
|
Tag,
|
|
Pin,
|
|
Terminal,
|
|
Network,
|
|
FileEdit,
|
|
Search,
|
|
Upload,
|
|
X,
|
|
Check,
|
|
Pencil,
|
|
FolderMinus,
|
|
Copy,
|
|
Activity,
|
|
Clock,
|
|
Palette,
|
|
Trash,
|
|
Cloud,
|
|
Database,
|
|
Box,
|
|
Package,
|
|
Layers,
|
|
Archive,
|
|
HardDrive,
|
|
Globe,
|
|
FolderOpen,
|
|
} from "lucide-react";
|
|
import type {
|
|
SSHHost,
|
|
SSHFolder,
|
|
SSHManagerHostViewerProps,
|
|
} from "../../../../types/index.js";
|
|
import { DEFAULT_STATS_CONFIG } from "@/types/stats-widgets";
|
|
import { FolderEditDialog } from "./components/FolderEditDialog";
|
|
import { useTabs } from "@/ui/desktop/navigation/tabs/TabContext";
|
|
|
|
export function HostManagerViewer({ onEditHost }: SSHManagerHostViewerProps) {
|
|
const { t } = useTranslation();
|
|
const { confirmWithToast } = useConfirmation();
|
|
const { addTab } = useTabs();
|
|
const [hosts, setHosts] = useState<SSHHost[]>([]);
|
|
const [loading, setLoading] = useState(true);
|
|
const [error, setError] = useState<string | null>(null);
|
|
const [searchQuery, setSearchQuery] = useState("");
|
|
const [importing, setImporting] = useState(false);
|
|
const [draggedHost, setDraggedHost] = useState<SSHHost | null>(null);
|
|
const [dragOverFolder, setDragOverFolder] = useState<string | null>(null);
|
|
const [editingFolder, setEditingFolder] = useState<string | null>(null);
|
|
const [editingFolderName, setEditingFolderName] = useState("");
|
|
const [operationLoading, setOperationLoading] = useState(false);
|
|
const [folderMetadata, setFolderMetadata] = useState<Map<string, SSHFolder>>(
|
|
new Map(),
|
|
);
|
|
const [editingFolderAppearance, setEditingFolderAppearance] = useState<
|
|
string | null
|
|
>(null);
|
|
const [serverStatuses, setServerStatuses] = useState<
|
|
Map<number, "online" | "offline" | "degraded">
|
|
>(new Map());
|
|
const dragCounter = useRef(0);
|
|
|
|
useEffect(() => {
|
|
fetchHosts();
|
|
fetchFolderMetadata();
|
|
|
|
const handleHostsRefresh = () => {
|
|
fetchHosts();
|
|
fetchFolderMetadata();
|
|
};
|
|
|
|
const handleFoldersRefresh = () => {
|
|
fetchFolderMetadata();
|
|
};
|
|
|
|
window.addEventListener("hosts:refresh", handleHostsRefresh);
|
|
window.addEventListener("ssh-hosts:changed", handleHostsRefresh);
|
|
window.addEventListener("folders:changed", handleFoldersRefresh);
|
|
|
|
return () => {
|
|
window.removeEventListener("hosts:refresh", handleHostsRefresh);
|
|
window.removeEventListener("ssh-hosts:changed", handleHostsRefresh);
|
|
window.removeEventListener("folders:changed", handleFoldersRefresh);
|
|
};
|
|
}, []);
|
|
|
|
const fetchHosts = async () => {
|
|
try {
|
|
setLoading(true);
|
|
const data = await getSSHHosts();
|
|
|
|
const cleanedHosts = data.map((host) => {
|
|
const cleanedHost = { ...host };
|
|
if (cleanedHost.credentialId && cleanedHost.key) {
|
|
cleanedHost.key = undefined;
|
|
cleanedHost.keyPassword = undefined;
|
|
cleanedHost.keyType = undefined;
|
|
cleanedHost.authType = "credential";
|
|
} else if (cleanedHost.credentialId && cleanedHost.password) {
|
|
cleanedHost.password = undefined;
|
|
cleanedHost.authType = "credential";
|
|
} else if (cleanedHost.key && cleanedHost.password) {
|
|
cleanedHost.password = undefined;
|
|
cleanedHost.authType = "key";
|
|
}
|
|
return cleanedHost;
|
|
});
|
|
|
|
setHosts(cleanedHosts);
|
|
setError(null);
|
|
} catch {
|
|
setError(t("hosts.failedToLoadHosts"));
|
|
} finally {
|
|
setLoading(false);
|
|
}
|
|
};
|
|
|
|
const fetchFolderMetadata = async () => {
|
|
try {
|
|
const folders = await getSSHFolders();
|
|
const metadataMap = new Map<string, SSHFolder>();
|
|
folders.forEach((folder) => {
|
|
metadataMap.set(folder.name, folder);
|
|
});
|
|
setFolderMetadata(metadataMap);
|
|
} catch (error) {
|
|
console.error("Failed to fetch folder metadata:", error);
|
|
}
|
|
};
|
|
|
|
const handleSaveFolderAppearance = async (
|
|
folderName: string,
|
|
color: string,
|
|
icon: string,
|
|
) => {
|
|
try {
|
|
await updateFolderMetadata(folderName, color, icon);
|
|
toast.success(t("hosts.folderAppearanceUpdated"));
|
|
await fetchFolderMetadata();
|
|
window.dispatchEvent(new CustomEvent("folders:changed"));
|
|
} catch (error) {
|
|
console.error("Failed to update folder appearance:", error);
|
|
toast.error(t("hosts.failedToUpdateFolderAppearance"));
|
|
}
|
|
};
|
|
|
|
const handleDeleteAllHostsInFolder = async (folderName: string) => {
|
|
const hostsInFolder = hostsByFolder[folderName] || [];
|
|
confirmWithToast(
|
|
t("hosts.confirmDeleteAllHostsInFolder", {
|
|
folder: folderName,
|
|
count: hostsInFolder.length,
|
|
}),
|
|
async () => {
|
|
try {
|
|
const result = await deleteAllHostsInFolder(folderName);
|
|
toast.success(
|
|
t("hosts.allHostsInFolderDeleted", {
|
|
folder: folderName,
|
|
count: result.deletedCount,
|
|
}),
|
|
);
|
|
await fetchHosts();
|
|
await fetchFolderMetadata();
|
|
window.dispatchEvent(new CustomEvent("ssh-hosts:changed"));
|
|
} catch (error) {
|
|
console.error("Failed to delete hosts in folder:", error);
|
|
toast.error(t("hosts.failedToDeleteHostsInFolder"));
|
|
}
|
|
},
|
|
"destructive",
|
|
);
|
|
};
|
|
|
|
useEffect(() => {
|
|
if (hosts.length === 0) return;
|
|
|
|
const statusIntervals: NodeJS.Timeout[] = [];
|
|
const statusCancelled: boolean[] = [];
|
|
|
|
hosts.forEach((host, index) => {
|
|
const statsConfig = (() => {
|
|
try {
|
|
return host.statsConfig
|
|
? JSON.parse(host.statsConfig)
|
|
: DEFAULT_STATS_CONFIG;
|
|
} catch {
|
|
return DEFAULT_STATS_CONFIG;
|
|
}
|
|
})();
|
|
|
|
const shouldShowStatus = statsConfig.statusCheckEnabled !== false;
|
|
|
|
if (!shouldShowStatus) {
|
|
setServerStatuses((prev) => {
|
|
const next = new Map(prev);
|
|
next.set(host.id, "offline");
|
|
return next;
|
|
});
|
|
return;
|
|
}
|
|
|
|
const fetchStatus = async () => {
|
|
try {
|
|
const res = await getServerStatusById(host.id);
|
|
if (!statusCancelled[index]) {
|
|
setServerStatuses((prev) => {
|
|
const next = new Map(prev);
|
|
next.set(
|
|
host.id,
|
|
res?.status === "online" ? "online" : "offline",
|
|
);
|
|
return next;
|
|
});
|
|
}
|
|
} catch (error: unknown) {
|
|
if (!statusCancelled[index]) {
|
|
const err = error as { response?: { status?: number } };
|
|
let status: "online" | "offline" | "degraded" = "offline";
|
|
if (err?.response?.status === 504) {
|
|
status = "degraded";
|
|
}
|
|
setServerStatuses((prev) => {
|
|
const next = new Map(prev);
|
|
next.set(host.id, status);
|
|
return next;
|
|
});
|
|
}
|
|
}
|
|
};
|
|
|
|
fetchStatus();
|
|
const intervalId = setInterval(fetchStatus, 10000);
|
|
statusIntervals.push(intervalId);
|
|
});
|
|
|
|
return () => {
|
|
statusCancelled.fill(true);
|
|
statusIntervals.forEach((interval) => clearInterval(interval));
|
|
};
|
|
}, [hosts]);
|
|
|
|
const getFolderIcon = (folderName: string) => {
|
|
const metadata = folderMetadata.get(folderName);
|
|
if (!metadata?.icon) return Folder;
|
|
|
|
const iconMap: Record<string, React.ComponentType> = {
|
|
Folder,
|
|
Server,
|
|
Cloud,
|
|
Database,
|
|
Box,
|
|
Package,
|
|
Layers,
|
|
Archive,
|
|
HardDrive,
|
|
Globe,
|
|
};
|
|
|
|
return iconMap[metadata.icon] || Folder;
|
|
};
|
|
|
|
const getFolderColor = (folderName: string) => {
|
|
const metadata = folderMetadata.get(folderName);
|
|
return metadata?.color;
|
|
};
|
|
|
|
const handleDelete = async (hostId: number, hostName: string) => {
|
|
confirmWithToast(
|
|
t("hosts.confirmDelete", { name: hostName }),
|
|
async () => {
|
|
try {
|
|
await deleteSSHHost(hostId);
|
|
toast.success(t("hosts.hostDeletedSuccessfully", { name: hostName }));
|
|
await fetchHosts();
|
|
window.dispatchEvent(new CustomEvent("ssh-hosts:changed"));
|
|
|
|
const { refreshServerPolling } = await import("@/ui/main-axios.ts");
|
|
refreshServerPolling();
|
|
} catch {
|
|
toast.error(t("hosts.failedToDeleteHost"));
|
|
}
|
|
},
|
|
"destructive",
|
|
);
|
|
};
|
|
|
|
const handleExport = (host: SSHHost) => {
|
|
const actualAuthType = host.credentialId
|
|
? "credential"
|
|
: host.key
|
|
? "key"
|
|
: "password";
|
|
|
|
if (actualAuthType === "credential") {
|
|
const confirmMessage = t("hosts.exportCredentialWarning", {
|
|
name: host.name || `${host.username}@${host.ip}`,
|
|
});
|
|
|
|
confirmWithToast(confirmMessage, () => {
|
|
performExport(host);
|
|
});
|
|
return;
|
|
} else if (actualAuthType === "password" || actualAuthType === "key") {
|
|
const confirmMessage = t("hosts.exportSensitiveDataWarning", {
|
|
name: host.name || `${host.username}@${host.ip}`,
|
|
});
|
|
|
|
confirmWithToast(confirmMessage, () => {
|
|
performExport(host);
|
|
});
|
|
return;
|
|
}
|
|
|
|
performExport(host);
|
|
};
|
|
|
|
const performExport = async (host: SSHHost) => {
|
|
try {
|
|
const decryptedHost = await exportSSHHostWithCredentials(host.id);
|
|
|
|
const cleanExportData = Object.fromEntries(
|
|
Object.entries(decryptedHost).filter(
|
|
([, value]) => value !== undefined,
|
|
),
|
|
);
|
|
|
|
const blob = new Blob([JSON.stringify(cleanExportData, null, 2)], {
|
|
type: "application/json",
|
|
});
|
|
const url = URL.createObjectURL(blob);
|
|
const a = document.createElement("a");
|
|
a.href = url;
|
|
a.download = `${host.name || host.username + "@" + host.ip}-host-config.json`;
|
|
document.body.appendChild(a);
|
|
a.click();
|
|
document.body.removeChild(a);
|
|
URL.revokeObjectURL(url);
|
|
|
|
toast.success(
|
|
`Exported host configuration for ${host.name || host.username}@${host.ip}`,
|
|
);
|
|
} catch {
|
|
toast.error(t("hosts.failedToExportHost"));
|
|
}
|
|
};
|
|
|
|
const handleEdit = (host: SSHHost) => {
|
|
if (onEditHost) {
|
|
onEditHost(host);
|
|
}
|
|
};
|
|
|
|
const handleClone = (host: SSHHost) => {
|
|
if (onEditHost) {
|
|
const clonedHost = { ...host };
|
|
delete clonedHost.id;
|
|
onEditHost(clonedHost);
|
|
}
|
|
};
|
|
|
|
const handleRemoveFromFolder = async (host: SSHHost) => {
|
|
confirmWithToast(
|
|
t("hosts.confirmRemoveFromFolder", {
|
|
name: host.name || `${host.username}@${host.ip}`,
|
|
folder: host.folder,
|
|
}),
|
|
async () => {
|
|
try {
|
|
setOperationLoading(true);
|
|
const updatedHost = { ...host, folder: "" };
|
|
await updateSSHHost(host.id, updatedHost);
|
|
toast.success(
|
|
t("hosts.removedFromFolder", {
|
|
name: host.name || `${host.username}@${host.ip}`,
|
|
}),
|
|
);
|
|
await fetchHosts();
|
|
window.dispatchEvent(new CustomEvent("ssh-hosts:changed"));
|
|
} catch {
|
|
toast.error(t("hosts.failedToRemoveFromFolder"));
|
|
} finally {
|
|
setOperationLoading(false);
|
|
}
|
|
},
|
|
);
|
|
};
|
|
|
|
const handleFolderRename = async (oldName: string) => {
|
|
if (!editingFolderName.trim() || editingFolderName === oldName) {
|
|
setEditingFolder(null);
|
|
setEditingFolderName("");
|
|
return;
|
|
}
|
|
|
|
try {
|
|
setOperationLoading(true);
|
|
await renameFolder(oldName, editingFolderName.trim());
|
|
toast.success(
|
|
t("hosts.folderRenamed", {
|
|
oldName,
|
|
newName: editingFolderName.trim(),
|
|
}),
|
|
);
|
|
await fetchHosts();
|
|
window.dispatchEvent(new CustomEvent("ssh-hosts:changed"));
|
|
setEditingFolder(null);
|
|
setEditingFolderName("");
|
|
} catch {
|
|
toast.error(t("hosts.failedToRenameFolder"));
|
|
} finally {
|
|
setOperationLoading(false);
|
|
}
|
|
};
|
|
|
|
const startFolderEdit = (folderName: string) => {
|
|
setEditingFolder(folderName);
|
|
setEditingFolderName(folderName);
|
|
};
|
|
|
|
const cancelFolderEdit = () => {
|
|
setEditingFolder(null);
|
|
setEditingFolderName("");
|
|
};
|
|
|
|
const handleDragStart = (e: React.DragEvent, host: SSHHost) => {
|
|
setDraggedHost(host);
|
|
e.dataTransfer.effectAllowed = "move";
|
|
e.dataTransfer.setData("text/plain", "");
|
|
};
|
|
|
|
const handleDragEnd = () => {
|
|
setDraggedHost(null);
|
|
setDragOverFolder(null);
|
|
dragCounter.current = 0;
|
|
};
|
|
|
|
const handleDragOver = (e: React.DragEvent) => {
|
|
e.preventDefault();
|
|
e.dataTransfer.dropEffect = "move";
|
|
};
|
|
|
|
const handleDragEnter = (e: React.DragEvent, folderName: string) => {
|
|
e.preventDefault();
|
|
dragCounter.current++;
|
|
setDragOverFolder(folderName);
|
|
};
|
|
|
|
const handleDragLeave = () => {
|
|
dragCounter.current--;
|
|
if (dragCounter.current === 0) {
|
|
setDragOverFolder(null);
|
|
}
|
|
};
|
|
|
|
const handleDrop = async (e: React.DragEvent, targetFolder: string) => {
|
|
e.preventDefault();
|
|
dragCounter.current = 0;
|
|
setDragOverFolder(null);
|
|
|
|
if (!draggedHost) return;
|
|
|
|
const newFolder =
|
|
targetFolder === t("hosts.uncategorized") ? "" : targetFolder;
|
|
|
|
if (draggedHost.folder === newFolder) {
|
|
setDraggedHost(null);
|
|
return;
|
|
}
|
|
|
|
try {
|
|
setOperationLoading(true);
|
|
const updatedHost = { ...draggedHost, folder: newFolder };
|
|
await updateSSHHost(draggedHost.id, updatedHost);
|
|
toast.success(
|
|
t("hosts.movedToFolder", {
|
|
name: draggedHost.name || `${draggedHost.username}@${draggedHost.ip}`,
|
|
folder: targetFolder,
|
|
}),
|
|
);
|
|
await fetchHosts();
|
|
window.dispatchEvent(new CustomEvent("ssh-hosts:changed"));
|
|
} catch {
|
|
toast.error(t("hosts.failedToMoveToFolder"));
|
|
} finally {
|
|
setOperationLoading(false);
|
|
setDraggedHost(null);
|
|
}
|
|
};
|
|
|
|
const handleJsonImport = async (
|
|
event: React.ChangeEvent<HTMLInputElement>,
|
|
) => {
|
|
const file = event.target.files?.[0];
|
|
if (!file) return;
|
|
|
|
try {
|
|
setImporting(true);
|
|
const text = await file.text();
|
|
const data = JSON.parse(text);
|
|
|
|
if (!Array.isArray(data.hosts) && !Array.isArray(data)) {
|
|
throw new Error(t("hosts.jsonMustContainHosts"));
|
|
}
|
|
|
|
const hostsArray = Array.isArray(data.hosts) ? data.hosts : data;
|
|
|
|
if (hostsArray.length === 0) {
|
|
throw new Error(t("hosts.noHostsInJson"));
|
|
}
|
|
|
|
if (hostsArray.length > 100) {
|
|
throw new Error(t("hosts.maxHostsAllowed"));
|
|
}
|
|
|
|
const result = await bulkImportSSHHosts(hostsArray);
|
|
|
|
if (result.success > 0) {
|
|
toast.success(
|
|
t("hosts.importCompleted", {
|
|
success: result.success,
|
|
failed: result.failed,
|
|
}),
|
|
);
|
|
if (result.errors.length > 0) {
|
|
toast.error(`Import errors: ${result.errors.join(", ")}`);
|
|
}
|
|
await fetchHosts();
|
|
window.dispatchEvent(new CustomEvent("ssh-hosts:changed"));
|
|
} else {
|
|
toast.error(t("hosts.importFailed") + `: ${result.errors.join(", ")}`);
|
|
}
|
|
} catch (err) {
|
|
const errorMessage =
|
|
err instanceof Error ? err.message : t("hosts.failedToImportJson");
|
|
toast.error(t("hosts.importError") + `: ${errorMessage}`);
|
|
} finally {
|
|
setImporting(false);
|
|
event.target.value = "";
|
|
}
|
|
};
|
|
|
|
const getMonitoringStatus = (host: SSHHost) => {
|
|
try {
|
|
const statsConfig = host.statsConfig
|
|
? JSON.parse(host.statsConfig)
|
|
: DEFAULT_STATS_CONFIG;
|
|
|
|
const formatInterval = (seconds: number): string => {
|
|
if (seconds >= 60) {
|
|
const minutes = Math.round(seconds / 60);
|
|
return `${minutes}m`;
|
|
}
|
|
return `${seconds}s`;
|
|
};
|
|
|
|
const statusEnabled = statsConfig.statusCheckEnabled !== false;
|
|
const metricsEnabled = statsConfig.metricsEnabled !== false;
|
|
const statusInterval = statusEnabled
|
|
? formatInterval(statsConfig.statusCheckInterval || 30)
|
|
: null;
|
|
const metricsInterval = metricsEnabled
|
|
? formatInterval(statsConfig.metricsInterval || 30)
|
|
: null;
|
|
|
|
return {
|
|
statusEnabled,
|
|
metricsEnabled,
|
|
statusInterval,
|
|
metricsInterval,
|
|
bothDisabled: !statusEnabled && !metricsEnabled,
|
|
};
|
|
} catch {
|
|
return {
|
|
statusEnabled: true,
|
|
metricsEnabled: true,
|
|
statusInterval: "30s",
|
|
metricsInterval: "30s",
|
|
bothDisabled: false,
|
|
};
|
|
}
|
|
};
|
|
|
|
const filteredAndSortedHosts = useMemo(() => {
|
|
let filtered = hosts;
|
|
|
|
if (searchQuery.trim()) {
|
|
const query = searchQuery.toLowerCase();
|
|
filtered = hosts.filter((host) => {
|
|
const searchableText = [
|
|
host.name || "",
|
|
host.username,
|
|
host.ip,
|
|
host.folder || "",
|
|
...(host.tags || []),
|
|
host.authType,
|
|
host.defaultPath || "",
|
|
]
|
|
.join(" ")
|
|
.toLowerCase();
|
|
return searchableText.includes(query);
|
|
});
|
|
}
|
|
|
|
return filtered.sort((a, b) => {
|
|
if (a.pin && !b.pin) return -1;
|
|
if (!a.pin && b.pin) return 1;
|
|
|
|
const aName = a.name || a.username;
|
|
const bName = b.name || b.username;
|
|
return aName.localeCompare(bName);
|
|
});
|
|
}, [hosts, searchQuery]);
|
|
|
|
const hostsByFolder = useMemo(() => {
|
|
const grouped: { [key: string]: SSHHost[] } = {};
|
|
|
|
filteredAndSortedHosts.forEach((host) => {
|
|
const folder = host.folder || t("hosts.uncategorized");
|
|
if (!grouped[folder]) {
|
|
grouped[folder] = [];
|
|
}
|
|
grouped[folder].push(host);
|
|
});
|
|
|
|
const sortedFolders = Object.keys(grouped).sort((a, b) => {
|
|
if (a === t("hosts.uncategorized")) return -1;
|
|
if (b === t("hosts.uncategorized")) return 1;
|
|
return a.localeCompare(b);
|
|
});
|
|
|
|
const sortedGrouped: { [key: string]: SSHHost[] } = {};
|
|
sortedFolders.forEach((folder) => {
|
|
sortedGrouped[folder] = grouped[folder];
|
|
});
|
|
|
|
return sortedGrouped;
|
|
}, [filteredAndSortedHosts]);
|
|
|
|
if (loading) {
|
|
return (
|
|
<div className="flex items-center justify-center h-full">
|
|
<div className="text-center">
|
|
<div className="animate-spin rounded-full h-8 w-8 border-b-2 border-white mx-auto mb-2"></div>
|
|
<p className="text-muted-foreground">{t("hosts.loadingHosts")}</p>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
if (error) {
|
|
return (
|
|
<div className="flex items-center justify-center h-full">
|
|
<div className="text-center">
|
|
<p className="text-red-500 mb-4">{error}</p>
|
|
<Button onClick={fetchHosts} variant="outline">
|
|
{t("hosts.retry")}
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
if (hosts.length === 0) {
|
|
return (
|
|
<div className="flex flex-col h-full min-h-0">
|
|
<div className="flex items-center justify-between mb-6">
|
|
<div>
|
|
<h2 className="text-xl font-semibold">{t("hosts.sshHosts")}</h2>
|
|
<p className="text-muted-foreground">
|
|
{t("hosts.hostsCount", { count: 0 })}
|
|
</p>
|
|
</div>
|
|
<div className="flex items-center gap-2">
|
|
<TooltipProvider>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button
|
|
variant="outline"
|
|
size="sm"
|
|
className="relative"
|
|
onClick={() =>
|
|
document.getElementById("json-import-input")?.click()
|
|
}
|
|
disabled={importing}
|
|
>
|
|
{importing ? t("hosts.importing") : t("hosts.importJson")}
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent
|
|
side="bottom"
|
|
className="max-w-sm bg-popover text-popover-foreground border border-border shadow-lg"
|
|
>
|
|
<div className="space-y-2">
|
|
<p className="font-semibold text-sm">
|
|
{t("hosts.importJsonTitle")}
|
|
</p>
|
|
<p className="text-xs text-muted-foreground">
|
|
{t("hosts.importJsonDesc")}
|
|
</p>
|
|
</div>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
</TooltipProvider>
|
|
|
|
<Button
|
|
variant="outline"
|
|
size="sm"
|
|
onClick={() => {
|
|
const sampleData = {
|
|
hosts: [
|
|
{
|
|
name: t("interface.webServerProduction"),
|
|
ip: "192.168.1.100",
|
|
port: 22,
|
|
username: "admin",
|
|
authType: "password",
|
|
password: "your_secure_password_here",
|
|
folder: t("interface.productionFolder"),
|
|
tags: ["web", "production", "nginx"],
|
|
pin: true,
|
|
enableTerminal: true,
|
|
enableTunnel: false,
|
|
enableFileManager: true,
|
|
defaultPath: "/var/www",
|
|
},
|
|
{
|
|
name: t("interface.databaseServer"),
|
|
ip: "192.168.1.101",
|
|
port: 22,
|
|
username: "dbadmin",
|
|
authType: "key",
|
|
key: "-----BEGIN OPENSSH PRIVATE KEY-----\nYour SSH private key content here\n-----END OPENSSH PRIVATE KEY-----",
|
|
keyPassword: "optional_key_passphrase",
|
|
keyType: "ssh-ed25519",
|
|
folder: t("interface.productionFolder"),
|
|
tags: ["database", "production", "postgresql"],
|
|
pin: false,
|
|
enableTerminal: true,
|
|
enableTunnel: true,
|
|
enableFileManager: false,
|
|
tunnelConnections: [
|
|
{
|
|
sourcePort: 5432,
|
|
endpointPort: 5432,
|
|
endpointHost: t("interface.webServerProduction"),
|
|
maxRetries: 3,
|
|
retryInterval: 10,
|
|
autoStart: true,
|
|
},
|
|
],
|
|
},
|
|
{
|
|
name: t("interface.developmentServer"),
|
|
ip: "192.168.1.102",
|
|
port: 2222,
|
|
username: "developer",
|
|
authType: "credential",
|
|
credentialId: 1,
|
|
folder: t("interface.developmentFolder"),
|
|
tags: ["dev", "testing"],
|
|
pin: false,
|
|
enableTerminal: true,
|
|
enableTunnel: false,
|
|
enableFileManager: true,
|
|
defaultPath: "/home/developer",
|
|
},
|
|
],
|
|
};
|
|
|
|
const blob = new Blob([JSON.stringify(sampleData, null, 2)], {
|
|
type: "application/json",
|
|
});
|
|
const url = URL.createObjectURL(blob);
|
|
const a = document.createElement("a");
|
|
a.href = url;
|
|
a.download = "sample-ssh-hosts.json";
|
|
document.body.appendChild(a);
|
|
a.click();
|
|
document.body.removeChild(a);
|
|
URL.revokeObjectURL(url);
|
|
}}
|
|
>
|
|
{t("hosts.downloadSample")}
|
|
</Button>
|
|
|
|
<Button
|
|
variant="outline"
|
|
size="sm"
|
|
onClick={() => {
|
|
window.open("https://docs.termix.site/json-import", "_blank");
|
|
}}
|
|
>
|
|
{t("hosts.formatGuide")}
|
|
</Button>
|
|
|
|
<div className="w-px h-6 bg-border mx-2" />
|
|
|
|
<Button onClick={fetchHosts} variant="outline" size="sm">
|
|
{t("hosts.refresh")}
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
|
|
<input
|
|
id="json-import-input"
|
|
type="file"
|
|
accept=".json"
|
|
onChange={handleJsonImport}
|
|
className="hidden"
|
|
/>
|
|
|
|
<div className="flex items-center justify-center flex-1">
|
|
<div className="text-center">
|
|
<Server className="h-12 w-12 text-muted-foreground mx-auto mb-4" />
|
|
<h3 className="text-lg font-semibold mb-2">{t("hosts.noHosts")}</h3>
|
|
<p className="text-muted-foreground mb-4">
|
|
{t("hosts.noHostsMessage")}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<div className="flex flex-col h-full min-h-0">
|
|
<div className="flex items-center justify-between mb-2">
|
|
<div>
|
|
<h2 className="text-xl font-semibold">{t("hosts.sshHosts")}</h2>
|
|
<p className="text-muted-foreground">
|
|
{t("hosts.hostsCount", { count: filteredAndSortedHosts.length })}
|
|
</p>
|
|
</div>
|
|
<div className="flex items-center gap-2">
|
|
<TooltipProvider>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button
|
|
variant="outline"
|
|
size="sm"
|
|
className="relative"
|
|
onClick={() =>
|
|
document.getElementById("json-import-input")?.click()
|
|
}
|
|
disabled={importing}
|
|
>
|
|
{importing ? t("hosts.importing") : t("hosts.importJson")}
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent
|
|
side="bottom"
|
|
className="max-w-sm bg-popover text-popover-foreground border border-border shadow-lg"
|
|
>
|
|
<div className="space-y-2">
|
|
<p className="font-semibold text-sm">
|
|
{t("hosts.importJsonTitle")}
|
|
</p>
|
|
<p className="text-xs text-muted-foreground">
|
|
{t("hosts.importJsonDesc")}
|
|
</p>
|
|
</div>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
</TooltipProvider>
|
|
|
|
<Button
|
|
variant="outline"
|
|
size="sm"
|
|
onClick={() => {
|
|
const sampleData = {
|
|
hosts: [
|
|
{
|
|
name: t("interface.webServerProduction"),
|
|
ip: "192.168.1.100",
|
|
port: 22,
|
|
username: "admin",
|
|
authType: "password",
|
|
password: "your_secure_password_here",
|
|
folder: t("interface.productionFolder"),
|
|
tags: ["web", "production", "nginx"],
|
|
pin: true,
|
|
enableTerminal: true,
|
|
enableTunnel: false,
|
|
enableFileManager: true,
|
|
defaultPath: "/var/www",
|
|
},
|
|
{
|
|
name: t("interface.databaseServer"),
|
|
ip: "192.168.1.101",
|
|
port: 22,
|
|
username: "dbadmin",
|
|
authType: "key",
|
|
key: "-----BEGIN OPENSSH PRIVATE KEY-----\nYour SSH private key content here\n-----END OPENSSH PRIVATE KEY-----",
|
|
keyPassword: "optional_key_passphrase",
|
|
keyType: "ssh-ed25519",
|
|
folder: t("interface.productionFolder"),
|
|
tags: ["database", "production", "postgresql"],
|
|
pin: false,
|
|
enableTerminal: true,
|
|
enableTunnel: true,
|
|
enableFileManager: false,
|
|
tunnelConnections: [
|
|
{
|
|
sourcePort: 5432,
|
|
endpointPort: 5432,
|
|
endpointHost: t("interface.webServerProduction"),
|
|
maxRetries: 3,
|
|
retryInterval: 10,
|
|
autoStart: true,
|
|
},
|
|
],
|
|
},
|
|
{
|
|
name: t("interface.developmentServer"),
|
|
ip: "192.168.1.102",
|
|
port: 2222,
|
|
username: "developer",
|
|
authType: "credential",
|
|
credentialId: 1,
|
|
folder: t("interface.developmentFolder"),
|
|
tags: ["dev", "testing"],
|
|
pin: false,
|
|
enableTerminal: true,
|
|
enableTunnel: false,
|
|
enableFileManager: true,
|
|
defaultPath: "/home/developer",
|
|
},
|
|
],
|
|
};
|
|
|
|
const blob = new Blob([JSON.stringify(sampleData, null, 2)], {
|
|
type: "application/json",
|
|
});
|
|
const url = URL.createObjectURL(blob);
|
|
const a = document.createElement("a");
|
|
a.href = url;
|
|
a.download = "sample-ssh-hosts.json";
|
|
document.body.appendChild(a);
|
|
a.click();
|
|
document.body.removeChild(a);
|
|
URL.revokeObjectURL(url);
|
|
}}
|
|
>
|
|
{t("hosts.downloadSample")}
|
|
</Button>
|
|
|
|
<Button
|
|
variant="outline"
|
|
size="sm"
|
|
onClick={() => {
|
|
window.open("https://docs.termix.site/json-import", "_blank");
|
|
}}
|
|
>
|
|
{t("hosts.formatGuide")}
|
|
</Button>
|
|
|
|
<div className="w-px h-6 bg-border mx-2" />
|
|
|
|
<Button onClick={fetchHosts} variant="outline" size="sm">
|
|
{t("hosts.refresh")}
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
|
|
<input
|
|
id="json-import-input"
|
|
type="file"
|
|
accept=".json"
|
|
onChange={handleJsonImport}
|
|
className="hidden"
|
|
/>
|
|
|
|
<div className="relative mb-3">
|
|
<Search className="absolute left-3 top-1/2 transform -translate-y-1/2 h-4 w-4 text-muted-foreground" />
|
|
<Input
|
|
placeholder={t("placeholders.searchHosts")}
|
|
value={searchQuery}
|
|
onChange={(e) => setSearchQuery(e.target.value)}
|
|
className="pl-10"
|
|
/>
|
|
</div>
|
|
|
|
<ScrollArea className="flex-1 min-h-0">
|
|
<div className="space-y-2 pb-20">
|
|
{Object.entries(hostsByFolder).map(([folder, folderHosts]) => (
|
|
<div
|
|
key={folder}
|
|
className={`border rounded-md transition-all duration-200 ${
|
|
dragOverFolder === folder
|
|
? "border-blue-500 bg-blue-500/10"
|
|
: ""
|
|
}`}
|
|
onDragOver={handleDragOver}
|
|
onDragEnter={(e) => handleDragEnter(e, folder)}
|
|
onDragLeave={handleDragLeave}
|
|
onDrop={(e) => handleDrop(e, folder)}
|
|
>
|
|
<Accordion
|
|
type="multiple"
|
|
defaultValue={Object.keys(hostsByFolder)}
|
|
>
|
|
<AccordionItem value={folder} className="border-none">
|
|
<AccordionTrigger className="px-2 py-1 bg-muted/20 border-b hover:no-underline rounded-t-md">
|
|
<div className="flex items-center gap-2 flex-1">
|
|
{(() => {
|
|
const FolderIcon = getFolderIcon(folder);
|
|
const folderColor = getFolderColor(folder);
|
|
return (
|
|
<FolderIcon
|
|
className="h-4 w-4"
|
|
style={
|
|
folderColor ? { color: folderColor } : undefined
|
|
}
|
|
/>
|
|
);
|
|
})()}
|
|
{editingFolder === folder ? (
|
|
<div
|
|
className="flex items-center gap-2"
|
|
onClick={(e) => e.stopPropagation()}
|
|
>
|
|
<Input
|
|
value={editingFolderName}
|
|
onChange={(e) =>
|
|
setEditingFolderName(e.target.value)
|
|
}
|
|
onKeyDown={(e) => {
|
|
if (e.key === "Enter") handleFolderRename(folder);
|
|
if (e.key === "Escape") cancelFolderEdit();
|
|
}}
|
|
className="h-6 text-sm px-2 flex-1"
|
|
autoFocus
|
|
disabled={operationLoading}
|
|
/>
|
|
<Button
|
|
size="sm"
|
|
variant="ghost"
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
handleFolderRename(folder);
|
|
}}
|
|
className="h-6 w-6 p-0"
|
|
disabled={operationLoading}
|
|
>
|
|
<Check className="h-3 w-3" />
|
|
</Button>
|
|
<Button
|
|
size="sm"
|
|
variant="ghost"
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
cancelFolderEdit();
|
|
}}
|
|
className="h-6 w-6 p-0"
|
|
disabled={operationLoading}
|
|
>
|
|
<X className="h-3 w-3" />
|
|
</Button>
|
|
</div>
|
|
) : (
|
|
<>
|
|
<span
|
|
className="font-medium cursor-pointer hover:text-blue-400 transition-colors"
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
if (folder !== t("hosts.uncategorized")) {
|
|
startFolderEdit(folder);
|
|
}
|
|
}}
|
|
title={
|
|
folder !== t("hosts.uncategorized")
|
|
? "Click to rename folder"
|
|
: ""
|
|
}
|
|
>
|
|
{folder}
|
|
</span>
|
|
{folder !== t("hosts.uncategorized") && (
|
|
<Button
|
|
size="sm"
|
|
variant="ghost"
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
startFolderEdit(folder);
|
|
}}
|
|
className="h-4 w-4 p-0 opacity-50 hover:opacity-100 transition-opacity"
|
|
title="Rename folder"
|
|
>
|
|
<Pencil className="h-3 w-3" />
|
|
</Button>
|
|
)}
|
|
</>
|
|
)}
|
|
<Badge variant="secondary" className="text-xs">
|
|
{folderHosts.length}
|
|
</Badge>
|
|
{folder !== t("hosts.uncategorized") && (
|
|
<div className="flex items-center gap-1 ml-auto">
|
|
<TooltipProvider>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button
|
|
size="sm"
|
|
variant="ghost"
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
setEditingFolderAppearance(folder);
|
|
}}
|
|
className="h-6 w-6 p-0 opacity-50 hover:opacity-100 transition-opacity"
|
|
>
|
|
<Palette className="h-3 w-3" />
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
{t("hosts.editFolderAppearance")}
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
</TooltipProvider>
|
|
<TooltipProvider>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button
|
|
size="sm"
|
|
variant="ghost"
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
handleDeleteAllHostsInFolder(folder);
|
|
}}
|
|
className="h-6 w-6 p-0 opacity-50 hover:opacity-100 hover:text-red-400 transition-all"
|
|
>
|
|
<Trash className="h-3 w-3" />
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
{t("hosts.deleteAllHostsInFolder")}
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
</TooltipProvider>
|
|
</div>
|
|
)}
|
|
</div>
|
|
</AccordionTrigger>
|
|
<AccordionContent className="p-2">
|
|
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-2">
|
|
{folderHosts.map((host) => (
|
|
<TooltipProvider key={host.id}>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<div
|
|
draggable
|
|
onDragStart={(e) => handleDragStart(e, host)}
|
|
onDragEnd={handleDragEnd}
|
|
className={`bg-dark-bg-input border border-input rounded-lg cursor-pointer hover:shadow-lg hover:border-blue-400/50 hover:bg-dark-hover-alt transition-all duration-200 p-3 group relative ${
|
|
draggedHost?.id === host.id
|
|
? "opacity-50 scale-95"
|
|
: ""
|
|
}`}
|
|
onClick={() => handleEdit(host)}
|
|
>
|
|
<div className="flex items-start justify-between">
|
|
<div className="flex-1 min-w-0">
|
|
<div className="flex items-center gap-1">
|
|
{(() => {
|
|
const statsConfig = (() => {
|
|
try {
|
|
return host.statsConfig
|
|
? JSON.parse(host.statsConfig)
|
|
: DEFAULT_STATS_CONFIG;
|
|
} catch {
|
|
return DEFAULT_STATS_CONFIG;
|
|
}
|
|
})();
|
|
const shouldShowStatus =
|
|
statsConfig.statusCheckEnabled !==
|
|
false;
|
|
const serverStatus =
|
|
serverStatuses.get(host.id) ||
|
|
"degraded";
|
|
|
|
return shouldShowStatus ? (
|
|
<Status
|
|
status={serverStatus}
|
|
className="!bg-transparent !p-0.75 flex-shrink-0"
|
|
>
|
|
<StatusIndicator />
|
|
</Status>
|
|
) : null;
|
|
})()}
|
|
{host.pin && (
|
|
<Pin className="h-3 w-3 text-yellow-500 flex-shrink-0" />
|
|
)}
|
|
<h3 className="font-medium truncate text-sm">
|
|
{host.name ||
|
|
`${host.username}@${host.ip}`}
|
|
</h3>
|
|
</div>
|
|
<p className="text-xs text-muted-foreground truncate">
|
|
{host.ip}:{host.port}
|
|
</p>
|
|
<p className="text-xs text-muted-foreground truncate">
|
|
{host.username}
|
|
</p>
|
|
<p className="text-xs text-muted-foreground truncate">
|
|
ID: {host.id}
|
|
</p>
|
|
</div>
|
|
<div className="flex gap-1 flex-shrink-0 ml-1">
|
|
{host.folder && host.folder !== "" && (
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button
|
|
size="sm"
|
|
variant="ghost"
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
handleRemoveFromFolder(host);
|
|
}}
|
|
className="h-5 w-5 p-0 text-orange-500 hover:text-orange-700 hover:bg-orange-500/10"
|
|
disabled={operationLoading}
|
|
>
|
|
<FolderMinus className="h-3 w-3" />
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>
|
|
Remove from folder "{host.folder}"
|
|
</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
)}
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button
|
|
size="sm"
|
|
variant="ghost"
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
handleEdit(host);
|
|
}}
|
|
className="h-5 w-5 p-0 hover:bg-blue-500/10"
|
|
>
|
|
<Edit className="h-3 w-3" />
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>Edit host</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button
|
|
size="sm"
|
|
variant="ghost"
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
handleDelete(
|
|
host.id,
|
|
host.name ||
|
|
`${host.username}@${host.ip}`,
|
|
);
|
|
}}
|
|
className="h-5 w-5 p-0 text-red-500 hover:text-red-700 hover:bg-red-500/10"
|
|
>
|
|
<Trash2 className="h-3 w-3" />
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>Delete host</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button
|
|
size="sm"
|
|
variant="ghost"
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
handleExport(host);
|
|
}}
|
|
className="h-5 w-5 p-0 text-blue-500 hover:text-blue-700 hover:bg-blue-500/10"
|
|
>
|
|
<Upload className="h-3 w-3" />
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>Export host</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button
|
|
size="sm"
|
|
variant="ghost"
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
handleClone(host);
|
|
}}
|
|
className="h-5 w-5 p-0 text-emerald-500 hover:text-emerald-700 hover:bg-emerald-500/10"
|
|
>
|
|
<Copy className="h-3 w-3" />
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>Clone host</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="mt-2 space-y-1">
|
|
{host.tags && host.tags.length > 0 && (
|
|
<div className="flex flex-wrap gap-1">
|
|
{host.tags
|
|
.slice(0, 6)
|
|
.map((tag, index) => (
|
|
<Badge
|
|
key={index}
|
|
variant="outline"
|
|
className="text-xs px-1 py-0"
|
|
>
|
|
<Tag className="h-2 w-2 mr-0.5" />
|
|
{tag}
|
|
</Badge>
|
|
))}
|
|
{host.tags.length > 6 && (
|
|
<Badge
|
|
variant="outline"
|
|
className="text-xs px-1 py-0"
|
|
>
|
|
+{host.tags.length - 6}
|
|
</Badge>
|
|
)}
|
|
</div>
|
|
)}
|
|
|
|
<div className="flex flex-wrap gap-1">
|
|
{host.enableTerminal && (
|
|
<Badge
|
|
variant="outline"
|
|
className="text-xs px-1 py-0"
|
|
>
|
|
<Terminal className="h-2 w-2 mr-0.5" />
|
|
{t("hosts.terminalBadge")}
|
|
</Badge>
|
|
)}
|
|
{host.enableTunnel && (
|
|
<Badge
|
|
variant="outline"
|
|
className="text-xs px-1 py-0"
|
|
>
|
|
<Network className="h-2 w-2 mr-0.5" />
|
|
{t("hosts.tunnelBadge")}
|
|
{host.tunnelConnections &&
|
|
host.tunnelConnections.length > 0 && (
|
|
<span className="ml-0.5">
|
|
({host.tunnelConnections.length})
|
|
</span>
|
|
)}
|
|
</Badge>
|
|
)}
|
|
{host.enableFileManager && (
|
|
<Badge
|
|
variant="outline"
|
|
className="text-xs px-1 py-0"
|
|
>
|
|
<FileEdit className="h-2 w-2 mr-0.5" />
|
|
{t("hosts.fileManagerBadge")}
|
|
</Badge>
|
|
)}
|
|
|
|
{(() => {
|
|
const monitoringStatus =
|
|
getMonitoringStatus(host);
|
|
|
|
if (monitoringStatus.bothDisabled) {
|
|
return (
|
|
<Badge
|
|
variant="outline"
|
|
className="text-xs px-1 py-0 text-muted-foreground"
|
|
>
|
|
<Activity className="h-2 w-2 mr-0.5" />
|
|
{t("hosts.monitoringDisabledBadge")}
|
|
</Badge>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<>
|
|
{monitoringStatus.statusEnabled && (
|
|
<Badge
|
|
variant="outline"
|
|
className="text-xs px-1 py-0"
|
|
>
|
|
<Activity className="h-2 w-2 mr-0.5" />
|
|
{t("hosts.statusMonitoring")}:{" "}
|
|
{monitoringStatus.statusInterval}
|
|
</Badge>
|
|
)}
|
|
{monitoringStatus.metricsEnabled && (
|
|
<Badge
|
|
variant="outline"
|
|
className="text-xs px-1 py-0"
|
|
>
|
|
<Clock className="h-2 w-2 mr-0.5" />
|
|
{t("hosts.metricsMonitoring")}:{" "}
|
|
{monitoringStatus.metricsInterval}
|
|
</Badge>
|
|
)}
|
|
</>
|
|
);
|
|
})()}
|
|
</div>
|
|
</div>
|
|
|
|
<div className="mt-3 pt-3 border-t border-border/50 flex items-center justify-center gap-1">
|
|
{host.enableTerminal && (
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button
|
|
size="sm"
|
|
variant="outline"
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
const title = host.name?.trim()
|
|
? host.name
|
|
: `${host.username}@${host.ip}:${host.port}`;
|
|
addTab({
|
|
type: "terminal",
|
|
title,
|
|
hostConfig: host,
|
|
});
|
|
}}
|
|
className="h-7 px-2 hover:bg-blue-500/10 hover:border-blue-500/50 flex-1"
|
|
>
|
|
<Terminal className="h-3.5 w-3.5" />
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>Open Terminal</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
)}
|
|
{host.enableFileManager && (
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button
|
|
size="sm"
|
|
variant="outline"
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
const title = host.name?.trim()
|
|
? host.name
|
|
: `${host.username}@${host.ip}:${host.port}`;
|
|
addTab({
|
|
type: "file_manager",
|
|
title,
|
|
hostConfig: host,
|
|
});
|
|
}}
|
|
className="h-7 px-2 hover:bg-emerald-500/10 hover:border-emerald-500/50 flex-1"
|
|
>
|
|
<FolderOpen className="h-3.5 w-3.5" />
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>Open File Manager</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
)}
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button
|
|
size="sm"
|
|
variant="outline"
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
const title = host.name?.trim()
|
|
? host.name
|
|
: `${host.username}@${host.ip}:${host.port}`;
|
|
addTab({
|
|
type: "server",
|
|
title,
|
|
hostConfig: host,
|
|
});
|
|
}}
|
|
className="h-7 px-2 hover:bg-purple-500/10 hover:border-purple-500/50 flex-1"
|
|
>
|
|
<Server className="h-3.5 w-3.5" />
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>Open Server Details</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
</div>
|
|
</div>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<div className="text-center">
|
|
<p className="font-medium">
|
|
Click to edit host
|
|
</p>
|
|
<p className="text-xs text-muted-foreground">
|
|
Drag to move between folders
|
|
</p>
|
|
</div>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
</TooltipProvider>
|
|
))}
|
|
</div>
|
|
</AccordionContent>
|
|
</AccordionItem>
|
|
</Accordion>
|
|
</div>
|
|
))}
|
|
</div>
|
|
</ScrollArea>
|
|
|
|
{editingFolderAppearance && (
|
|
<FolderEditDialog
|
|
folderName={editingFolderAppearance}
|
|
currentColor={getFolderColor(editingFolderAppearance)}
|
|
currentIcon={folderMetadata.get(editingFolderAppearance)?.icon}
|
|
open={editingFolderAppearance !== null}
|
|
onOpenChange={(open) => {
|
|
if (!open) setEditingFolderAppearance(null);
|
|
}}
|
|
onSave={async (color, icon) => {
|
|
await handleSaveFolderAppearance(
|
|
editingFolderAppearance,
|
|
color,
|
|
icon,
|
|
);
|
|
setEditingFolderAppearance(null);
|
|
}}
|
|
/>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|