* fix select edit host but not update view (#438) * fix: Checksum issue with chocolatey * fix: Remove homebrew old stuff * Add Korean translation (#439) Co-authored-by: 송준우 <2484@coreit.co.kr> * feat: Automate flatpak * fix: Add imagemagik to electron builder to resolve build error * fix: Build error with runtime repo flag * fix: Flatpak runtime error and install freedesktop ver warning * fix: Flatpak runtime error and install freedesktop ver warning * feat: Re-add homebrew cask and move scripts to backend * fix: No sandbox flag issue * fix: Change name for electron macos cask output * fix: Sandbox error with Linux * fix: Remove comming soon for app stores in readme * Adding Comment at the end of the public_key on the host on deploy (#440) * Add termix.rb Cask file * Update Termix to version 1.9.0 with new checksum * Update README to remove 'coming soon' notes * -Add New Interface for Credential DB -Add Credential Name as a comment into the server authorized_key file --------- Co-authored-by: Luke Gustafson <88517757+LukeGus@users.noreply.github.com> * Sudo auto fill password (#441) * Add termix.rb Cask file * Update Termix to version 1.9.0 with new checksum * Update README to remove 'coming soon' notes * Feature Sudo password auto-fill; * Fix locale json shema; --------- Co-authored-by: Luke Gustafson <88517757+LukeGus@users.noreply.github.com> * Added Italian Language; (#445) * Add termix.rb Cask file * Update Termix to version 1.9.0 with new checksum * Update README to remove 'coming soon' notes * Added Italian Language; --------- Co-authored-by: Luke Gustafson <88517757+LukeGus@users.noreply.github.com> * Auto collapse snippet folders (#448) * Add termix.rb Cask file * Update Termix to version 1.9.0 with new checksum * Update README to remove 'coming soon' notes * feat: Add collapsable snippets (customizable in user profile) * Translations (#447) * Add termix.rb Cask file * Update Termix to version 1.9.0 with new checksum * Update README to remove 'coming soon' notes * Added Italian Language; * Fix translations; Removed duplicate keys, synchronised other languages using English as the source, translated added keys, fixed inaccurate translations. --------- Co-authored-by: Luke Gustafson <88517757+LukeGus@users.noreply.github.com> * Remove PTY-level keepalive (#449) * Add termix.rb Cask file * Update Termix to version 1.9.0 with new checksum * Update README to remove 'coming soon' notes * Remove PTY-level keepalive to prevent unwanted terminal output; use SSH-level keepalive instead --------- Co-authored-by: Luke Gustafson <88517757+LukeGus@users.noreply.github.com> * feat: Seperate server stats and tunnel management (improved both UI's) then started initial docker implementation * fix: finalize adding docker to db * feat: Add docker management support (local squash) * Fix RBAC role system bugs and improve UX (#446) * Fix RBAC role system bugs and improve UX - Fix user list dropdown selection in host sharing - Fix role sharing permissions to include role-based access - Fix translation template interpolation for success messages - Standardize system roles to admin and user only - Auto-assign user role to new registrations - Remove blocking confirmation dialogs in modal contexts - Add missing i18n keys for common actions - Fix button type to prevent unintended form submissions * Enhance RBAC system with UI improvements and security fixes - Move role assignment to Users tab with per-user role management - Protect system roles (admin/user) from editing and manual assignment - Simplify permission system: remove Use level, keep View and Manage - Hide Update button and Sharing tab for view-only/shared hosts - Prevent users from sharing hosts with themselves - Unify table and modal styling across admin panels - Auto-assign system roles on user registration - Add permission metadata to host interface * Add empty state message for role assignment - Display helpful message when no custom roles available - Clarify that system roles are auto-assigned - Add noCustomRolesToAssign translation in English and Chinese * fix: Prevent credential sharing errors for shared hosts - Skip credential resolution for shared hosts with credential authentication to prevent decryption errors (credentials are encrypted per-user) - Add warning alert in sharing tab when host uses credential authentication - Inform users that shared users cannot connect to credential-based hosts - Add translations for credential sharing warning (EN/ZH) This prevents authentication failures when sharing hosts configured with credential authentication while maintaining security by keeping credentials isolated per user. * feat: Improve rbac UI and fixes some bugs --------- Co-authored-by: Luke Gustafson <88517757+LukeGus@users.noreply.github.com> Co-authored-by: LukeGus <bugattiguy527@gmail.com> * SOCKS5 support (#452) * Add termix.rb Cask file * Update Termix to version 1.9.0 with new checksum * Update README to remove 'coming soon' notes * SOCKS5 support Adding single and chain socks5 proxy support * fix: cleanup files --------- Co-authored-by: Luke Gustafson <88517757+LukeGus@users.noreply.github.com> Co-authored-by: LukeGus <bugattiguy527@gmail.com> * Notes and Expiry fields add (#453) * Add termix.rb Cask file * Update Termix to version 1.9.0 with new checksum * Update README to remove 'coming soon' notes * Notes and Expiry add * fix: cleanup files --------- Co-authored-by: Luke Gustafson <88517757+LukeGus@users.noreply.github.com> Co-authored-by: LukeGus <bugattiguy527@gmail.com> * fix: ssh host types * fix: sudo incorrect styling and remove expiration date * feat: add sudo password and add diagonal bg's * fix: snippet running on enter key * fix: base64 decoding * fix: improve server stats / rbac * fix: wrap ssh host json export in hosts array * feat: auto trim host inputs, fix file manager jump hosts, dashboard prevent duplicates, file manager terminal not size updating, improve left sidebar sorting, hide/show tags, add apperance user profile tab, add new host manager tabs. * feat: improve terminal connection speed * fix: sqlite constriant errors and support non-root user (nginx perm issue) * feat: add beta syntax highlighing to terminal * feat: update imports and improve admin settings user management * chore: update translations * chore: update translations * feat: Complete light mode implementation with semantic theme system (#450) - Add comprehensive light/dark mode CSS variables with semantic naming - Implement theme-aware scrollbars using CSS variables - Add light mode backgrounds: --bg-base, --bg-elevated, --bg-surface, etc. - Add theme-aware borders: --border-base, --border-panel, --border-subtle - Add semantic text colors: --foreground-secondary, --foreground-subtle - Convert oklch colors to hex for better compatibility - Add theme awareness to CodeMirror editors - Update dark mode colors for consistency (background, sidebar, card, muted, input) - Add Tailwind color mappings for semantic classes Co-authored-by: Luke Gustafson <88517757+LukeGus@users.noreply.github.com> * fix: syntax errors * chore: updating/match themes and split admin settings * feat: add translation workflow and remove old translation.json * fix: translation workflow error * fix: translation workflow error * feat: improve translation system and update workflow * fix: wrong path for translations * fix: change translation to flat files * fix: gh rule error * chore: auto-translate to multiple languages (#458) * chore: improve organization and made a few styling changes in host manager * feat: improve terminal stability and split out the host manager * fix: add unnversiioned files * chore: migrate all to use the new theme system * fix: wrong animation line colors * fix: rbac implementation general issues (local squash) * fix: remove unneeded files * feat: add 10 new langs * chore: update gitnore * chore: auto-translate to multiple languages (#459) * fix: improve tunnel system * fix: properly split tabs, still need to fix up the host manager * chore: cleanup files (possible RC) * feat: add norwegian * chore: auto-translate to multiple languages (#461) * fix: small qol fixes and began readme update * fix: run cleanup script * feat: add docker docs button * feat: general bug fixes and readme updates * fix: translations * chore: auto-translate to multiple languages (#462) * fix: cleanup files * fix: test new translation issue and add better server-stats support * fix: fix translate error * chore: auto-translate to multiple languages (#463) * fix: fix translate mismatching text * chore: auto-translate to multiple languages (#465) * fix: fix translate mismatching text * fix: fix translate mismatching text * chore: auto-translate to multiple languages (#466) * fix: fix translate mismatching text * fix: fix translate mismatching text * fix: fix translate mismatching text * chore: auto-translate to multiple languages (#467) * fix: fix translate mismatching text * chore: auto-translate to multiple languages (#468) * feat: add to readme, a few qol changes, and improve server stats in general * chore: auto-translate to multiple languages (#469) * feat: turned disk uage into graph and fixed issue with termina console * fix: electron build error and hide icons when shared * chore: run clean * fix: general server stats issues, file manager decoding, ui qol * fix: add dashboard line breaks * fix: docker console error * fix: docker console not loading and mismatched stripped background for electron * fix: docker console not loading * chore: docker console not loading in docker * chore: translate readme to chinese * chore: match package lock to package json * chore: nginx config issue for dokcer console * chore: auto-translate to multiple languages (#470) --------- Co-authored-by: Tran Trung Kien <kientt13.7@gmail.com> Co-authored-by: junu <bigdwarf_@naver.com> Co-authored-by: 송준우 <2484@coreit.co.kr> Co-authored-by: SlimGary <trash.slim@gmail.com> Co-authored-by: Nunzio Marfè <nunzio.marfe@protonmail.com> Co-authored-by: Wesley Reid <starhound@lostsouls.org> Co-authored-by: ZacharyZcR <zacharyzcr1984@gmail.com> Co-authored-by: Denis <38875137+Medvedinca@users.noreply.github.com> Co-authored-by: Peet McKinney <68706879+PeetMcK@users.noreply.github.com>
285 lines
9.3 KiB
TypeScript
285 lines
9.3 KiB
TypeScript
import React, { useState } from "react";
|
|
import {
|
|
Card,
|
|
CardContent,
|
|
CardDescription,
|
|
CardHeader,
|
|
CardTitle,
|
|
} from "@/components/ui/card";
|
|
import { Button } from "@/components/ui/button";
|
|
import { Input } from "@/components/ui/input";
|
|
import { PasswordInput } from "@/components/ui/password-input";
|
|
import { Label } from "@/components/ui/label";
|
|
import { Alert, AlertDescription, AlertTitle } from "@/components/ui/alert";
|
|
import { Tabs, TabsContent, TabsList, TabsTrigger } from "@/components/ui/tabs";
|
|
import { Shield, AlertCircle, Upload } from "lucide-react";
|
|
import { useTranslation } from "react-i18next";
|
|
import CodeMirror from "@uiw/react-codemirror";
|
|
import { oneDark } from "@codemirror/theme-one-dark";
|
|
import { EditorView } from "@codemirror/view";
|
|
|
|
interface SSHAuthDialogProps {
|
|
isOpen: boolean;
|
|
reason: "no_keyboard" | "auth_failed" | "timeout";
|
|
onSubmit: (credentials: {
|
|
password?: string;
|
|
sshKey?: string;
|
|
keyPassword?: string;
|
|
}) => void;
|
|
onCancel: () => void;
|
|
hostInfo: {
|
|
ip: string;
|
|
port: number;
|
|
username: string;
|
|
name?: string;
|
|
};
|
|
backgroundColor?: string;
|
|
}
|
|
|
|
export function SSHAuthDialog({
|
|
isOpen,
|
|
reason,
|
|
onSubmit,
|
|
onCancel,
|
|
hostInfo,
|
|
backgroundColor = "var(--bg-base)",
|
|
}: SSHAuthDialogProps) {
|
|
const { t } = useTranslation();
|
|
const [authTab, setAuthTab] = useState<"password" | "key">("password");
|
|
const [password, setPassword] = useState("");
|
|
const [sshKey, setSshKey] = useState("");
|
|
const [keyPassword, setKeyPassword] = useState("");
|
|
const [loading, setLoading] = useState(false);
|
|
|
|
if (!isOpen) return null;
|
|
|
|
const getReasonMessage = () => {
|
|
switch (reason) {
|
|
case "no_keyboard":
|
|
return t("auth.sshNoKeyboardInteractive");
|
|
case "auth_failed":
|
|
return t("auth.sshAuthenticationFailed");
|
|
case "timeout":
|
|
return t("auth.sshAuthenticationTimeout");
|
|
default:
|
|
return t("auth.sshAuthenticationRequired");
|
|
}
|
|
};
|
|
|
|
const getReasonDescription = () => {
|
|
switch (reason) {
|
|
case "no_keyboard":
|
|
return t("auth.sshNoKeyboardInteractiveDescription");
|
|
case "auth_failed":
|
|
return t("auth.sshAuthFailedDescription");
|
|
case "timeout":
|
|
return t("auth.sshTimeoutDescription");
|
|
default:
|
|
return t("auth.sshProvideCredentialsDescription");
|
|
}
|
|
};
|
|
|
|
const handleSubmit = async (e: React.FormEvent) => {
|
|
e.preventDefault();
|
|
setLoading(true);
|
|
|
|
try {
|
|
const credentials: {
|
|
password?: string;
|
|
sshKey?: string;
|
|
keyPassword?: string;
|
|
} = {};
|
|
|
|
if (authTab === "password") {
|
|
if (password.trim()) {
|
|
credentials.password = password;
|
|
}
|
|
} else {
|
|
if (sshKey.trim()) {
|
|
credentials.sshKey = sshKey;
|
|
if (keyPassword.trim()) {
|
|
credentials.keyPassword = keyPassword;
|
|
}
|
|
}
|
|
}
|
|
|
|
onSubmit(credentials);
|
|
} finally {
|
|
setLoading(false);
|
|
}
|
|
};
|
|
|
|
const handleKeyFileUpload = async (
|
|
e: React.ChangeEvent<HTMLInputElement>,
|
|
) => {
|
|
const file = e.target.files?.[0];
|
|
if (file) {
|
|
try {
|
|
const fileContent = await file.text();
|
|
setSshKey(fileContent);
|
|
} catch (error) {
|
|
console.error("Failed to read SSH key file:", error);
|
|
}
|
|
}
|
|
};
|
|
|
|
const canSubmit = () => {
|
|
if (authTab === "password") {
|
|
return password.trim() !== "";
|
|
} else {
|
|
return sshKey.trim() !== "";
|
|
}
|
|
};
|
|
|
|
const hostDisplay = hostInfo.name
|
|
? `${hostInfo.name} (${hostInfo.username}@${hostInfo.ip}:${hostInfo.port})`
|
|
: `${hostInfo.username}@${hostInfo.ip}:${hostInfo.port}`;
|
|
|
|
return (
|
|
<div
|
|
className="absolute inset-0 z-9999 flex items-center justify-center bg-canvas animate-in fade-in duration-200"
|
|
style={{ backgroundColor }}
|
|
>
|
|
<Card className="w-full max-w-2xl mx-4 border-2 animate-in fade-in zoom-in-95 duration-200">
|
|
<CardHeader>
|
|
<CardTitle className="flex items-center gap-2">
|
|
<Shield className="w-5 h-5" />
|
|
{t("auth.sshAuthenticationRequired")}
|
|
</CardTitle>
|
|
<CardDescription>{hostDisplay}</CardDescription>
|
|
</CardHeader>
|
|
<CardContent className="space-y-4">
|
|
<Alert variant={reason === "auth_failed" ? "destructive" : "default"}>
|
|
<AlertCircle className="h-4 w-4" />
|
|
<AlertTitle>{getReasonMessage()}</AlertTitle>
|
|
<AlertDescription>{getReasonDescription()}</AlertDescription>
|
|
</Alert>
|
|
|
|
<form onSubmit={handleSubmit} className="space-y-4">
|
|
<Tabs
|
|
value={authTab}
|
|
onValueChange={(v) => setAuthTab(v as "password" | "key")}
|
|
>
|
|
<TabsList className="grid w-full grid-cols-2">
|
|
<TabsTrigger value="password">
|
|
{t("credentials.password")}
|
|
</TabsTrigger>
|
|
<TabsTrigger value="key">{t("credentials.sshKey")}</TabsTrigger>
|
|
</TabsList>
|
|
|
|
<TabsContent value="password" className="space-y-4 mt-4">
|
|
<div className="space-y-2">
|
|
<Label htmlFor="ssh-password">
|
|
{t("credentials.password")}
|
|
</Label>
|
|
<PasswordInput
|
|
id="ssh-password"
|
|
placeholder={t("placeholders.enterPassword")}
|
|
value={password}
|
|
onChange={(e) => setPassword(e.target.value)}
|
|
autoFocus
|
|
/>
|
|
<p className="text-sm text-muted-foreground">
|
|
{t("auth.sshPasswordDescription")}
|
|
</p>
|
|
</div>
|
|
</TabsContent>
|
|
|
|
<TabsContent value="key" className="space-y-4 mt-4">
|
|
<div className="space-y-2">
|
|
<Label htmlFor="ssh-key">
|
|
{t("credentials.sshPrivateKey")}
|
|
</Label>
|
|
<div className="mb-2">
|
|
<div className="relative inline-block w-full">
|
|
<input
|
|
id="key-upload"
|
|
type="file"
|
|
accept="*,.pem,.key,.txt,.ppk"
|
|
onChange={handleKeyFileUpload}
|
|
className="absolute inset-0 w-full h-full opacity-0 cursor-pointer"
|
|
/>
|
|
<Button
|
|
type="button"
|
|
variant="outline"
|
|
className="w-full justify-start text-left"
|
|
>
|
|
<Upload className="w-4 h-4 mr-2" />
|
|
<span className="truncate">
|
|
{t("credentials.uploadPrivateKeyFile")}
|
|
</span>
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
<CodeMirror
|
|
value={sshKey}
|
|
onChange={(value) => setSshKey(value)}
|
|
placeholder={t("placeholders.pastePrivateKey")}
|
|
theme={oneDark}
|
|
className="border border-input rounded-md"
|
|
minHeight="200px"
|
|
maxHeight="300px"
|
|
basicSetup={{
|
|
lineNumbers: true,
|
|
foldGutter: false,
|
|
dropCursor: false,
|
|
allowMultipleSelections: false,
|
|
highlightSelectionMatches: false,
|
|
searchKeymap: false,
|
|
scrollPastEnd: false,
|
|
}}
|
|
extensions={[
|
|
EditorView.theme({
|
|
".cm-scroller": {
|
|
overflow: "auto",
|
|
scrollbarWidth: "thin",
|
|
scrollbarColor:
|
|
"var(--scrollbar-thumb) var(--scrollbar-track)",
|
|
},
|
|
}),
|
|
]}
|
|
/>
|
|
</div>
|
|
|
|
<div className="space-y-2">
|
|
<Label htmlFor="ssh-key-password">
|
|
{t("credentials.keyPassword")} ({t("common.optional")})
|
|
</Label>
|
|
<PasswordInput
|
|
id="ssh-key-password"
|
|
placeholder={t("placeholders.keyPassword")}
|
|
value={keyPassword}
|
|
onChange={(e) => setKeyPassword(e.target.value)}
|
|
/>
|
|
<p className="text-sm text-muted-foreground">
|
|
{t("auth.sshKeyPasswordDescription")}
|
|
</p>
|
|
</div>
|
|
</TabsContent>
|
|
</Tabs>
|
|
|
|
<div className="flex gap-3 pt-4">
|
|
<Button
|
|
type="button"
|
|
variant="outline"
|
|
onClick={onCancel}
|
|
disabled={loading}
|
|
className="flex-1"
|
|
>
|
|
{t("common.cancel")}
|
|
</Button>
|
|
<Button
|
|
type="submit"
|
|
disabled={!canSubmit() || loading}
|
|
className="flex-1"
|
|
>
|
|
{loading ? t("common.connecting") : t("common.connect")}
|
|
</Button>
|
|
</div>
|
|
</form>
|
|
</CardContent>
|
|
</Card>
|
|
</div>
|
|
);
|
|
}
|