mirror of
https://github.com/Dokploy/dokploy.git
synced 2026-06-20 14:45:42 +02:00
- Introduced new test files for permission checks, including `check-permission.test.ts`, `enterprise-only-resources.test.ts`, `resolve-permissions.test.ts`, and `service-access.test.ts`. - Implemented permission checks in various components to ensure actions are gated by user permissions, including `ShowTraefikConfig`, `UpdateTraefikConfig`, `ShowVolumes`, `ShowDomains`, and others. - Enhanced the logic for displaying UI elements based on user permissions, ensuring that only authorized users can access or modify resources.
274 lines
8.3 KiB
TypeScript
274 lines
8.3 KiB
TypeScript
import * as TooltipPrimitive from "@radix-ui/react-tooltip";
|
|
import { Ban, CheckCircle2, RefreshCcw, Rocket, Terminal } from "lucide-react";
|
|
import { useState } from "react";
|
|
import { toast } from "sonner";
|
|
import { DialogAction } from "@/components/shared/dialog-action";
|
|
import { DrawerLogs } from "@/components/shared/drawer-logs";
|
|
import { Button } from "@/components/ui/button";
|
|
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card";
|
|
import {
|
|
Tooltip,
|
|
TooltipContent,
|
|
TooltipProvider,
|
|
TooltipTrigger,
|
|
} from "@/components/ui/tooltip";
|
|
import { api } from "@/utils/api";
|
|
import { type LogLine, parseLogs } from "../../docker/logs/utils";
|
|
import { DockerTerminalModal } from "../../settings/web-server/docker-terminal-modal";
|
|
|
|
interface Props {
|
|
postgresId: string;
|
|
}
|
|
|
|
export const ShowGeneralPostgres = ({ postgresId }: Props) => {
|
|
const { data: permissions } = api.user.getPermissions.useQuery();
|
|
const canDeploy = permissions?.deployment.create ?? false;
|
|
const { data, refetch } = api.postgres.one.useQuery(
|
|
{
|
|
postgresId: postgresId,
|
|
},
|
|
{ enabled: !!postgresId },
|
|
);
|
|
|
|
const { mutateAsync: reload, isPending: isReloading } =
|
|
api.postgres.reload.useMutation();
|
|
|
|
const { mutateAsync: stop, isPending: isStopping } =
|
|
api.postgres.stop.useMutation();
|
|
|
|
const { mutateAsync: start, isPending: isStarting } =
|
|
api.postgres.start.useMutation();
|
|
|
|
const [isDrawerOpen, setIsDrawerOpen] = useState(false);
|
|
const [filteredLogs, setFilteredLogs] = useState<LogLine[]>([]);
|
|
const [isDeploying, setIsDeploying] = useState(false);
|
|
api.postgres.deployWithLogs.useSubscription(
|
|
{
|
|
postgresId: postgresId,
|
|
},
|
|
{
|
|
enabled: isDeploying,
|
|
onData(log) {
|
|
if (!isDrawerOpen) {
|
|
setIsDrawerOpen(true);
|
|
}
|
|
|
|
if (log === "Deployment completed successfully!") {
|
|
setIsDeploying(false);
|
|
}
|
|
const parsedLogs = parseLogs(log);
|
|
setFilteredLogs((prev) => [...prev, ...parsedLogs]);
|
|
},
|
|
onError(error) {
|
|
console.error("Deployment logs error:", error);
|
|
setIsDeploying(false);
|
|
},
|
|
},
|
|
);
|
|
|
|
return (
|
|
<>
|
|
<div className="flex w-full flex-col gap-5 ">
|
|
<Card className="bg-background">
|
|
<CardHeader>
|
|
<CardTitle className="text-xl">Deploy Settings</CardTitle>
|
|
</CardHeader>
|
|
<CardContent className="flex flex-row gap-4 flex-wrap">
|
|
<TooltipProvider disableHoverableContent={false}>
|
|
{canDeploy && (
|
|
<DialogAction
|
|
title="Deploy PostgreSQL"
|
|
description="Are you sure you want to deploy this postgres?"
|
|
type="default"
|
|
onClick={async () => {
|
|
setIsDeploying(true);
|
|
await new Promise((resolve) => setTimeout(resolve, 1000));
|
|
refetch();
|
|
}}
|
|
>
|
|
<Button
|
|
variant="default"
|
|
isLoading={data?.applicationStatus === "running"}
|
|
className="flex items-center gap-1.5 focus-visible:ring-2 focus-visible:ring-offset-2"
|
|
>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<div className="flex items-center">
|
|
<Rocket className="size-4 mr-1" />
|
|
Deploy
|
|
</div>
|
|
</TooltipTrigger>
|
|
<TooltipPrimitive.Portal>
|
|
<TooltipContent sideOffset={5} className="z-[60]">
|
|
<p>Downloads and sets up the PostgreSQL database</p>
|
|
</TooltipContent>
|
|
</TooltipPrimitive.Portal>
|
|
</Tooltip>
|
|
</Button>
|
|
</DialogAction>
|
|
)}
|
|
{canDeploy && (
|
|
<DialogAction
|
|
title="Reload PostgreSQL"
|
|
description="Are you sure you want to reload this postgres?"
|
|
type="default"
|
|
onClick={async () => {
|
|
await reload({
|
|
postgresId: postgresId,
|
|
appName: data?.appName || "",
|
|
})
|
|
.then(() => {
|
|
toast.success("PostgreSQL reloaded successfully");
|
|
refetch();
|
|
})
|
|
.catch(() => {
|
|
toast.error("Error reloading PostgreSQL");
|
|
});
|
|
}}
|
|
>
|
|
<Button
|
|
variant="secondary"
|
|
isLoading={isReloading}
|
|
className="flex items-center gap-1.5 focus-visible:ring-2 focus-visible:ring-offset-2"
|
|
>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<div className="flex items-center">
|
|
<RefreshCcw className="size-4 mr-1" />
|
|
Reload
|
|
</div>
|
|
</TooltipTrigger>
|
|
<TooltipPrimitive.Portal>
|
|
<TooltipContent sideOffset={5} className="z-[60]">
|
|
<p>
|
|
Restart the PostgreSQL service without rebuilding
|
|
</p>
|
|
</TooltipContent>
|
|
</TooltipPrimitive.Portal>
|
|
</Tooltip>
|
|
</Button>
|
|
</DialogAction>
|
|
)}
|
|
{canDeploy &&
|
|
(data?.applicationStatus === "idle" ? (
|
|
<DialogAction
|
|
title="Start PostgreSQL"
|
|
description="Are you sure you want to start this postgres?"
|
|
type="default"
|
|
onClick={async () => {
|
|
await start({
|
|
postgresId: postgresId,
|
|
})
|
|
.then(() => {
|
|
toast.success("PostgreSQL started successfully");
|
|
refetch();
|
|
})
|
|
.catch(() => {
|
|
toast.error("Error starting PostgreSQL");
|
|
});
|
|
}}
|
|
>
|
|
<Button
|
|
variant="secondary"
|
|
isLoading={isStarting}
|
|
className="flex items-center gap-1.5 focus-visible:ring-2 focus-visible:ring-offset-2"
|
|
>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<div className="flex items-center">
|
|
<CheckCircle2 className="size-4 mr-1" />
|
|
Start
|
|
</div>
|
|
</TooltipTrigger>
|
|
<TooltipPrimitive.Portal>
|
|
<TooltipContent sideOffset={5} className="z-[60]">
|
|
<p>
|
|
Start the PostgreSQL database (requires a previous
|
|
successful setup)
|
|
</p>
|
|
</TooltipContent>
|
|
</TooltipPrimitive.Portal>
|
|
</Tooltip>
|
|
</Button>
|
|
</DialogAction>
|
|
) : (
|
|
<DialogAction
|
|
title="Stop PostgreSQL"
|
|
description="Are you sure you want to stop this postgres?"
|
|
onClick={async () => {
|
|
await stop({
|
|
postgresId: postgresId,
|
|
})
|
|
.then(() => {
|
|
toast.success("PostgreSQL stopped successfully");
|
|
refetch();
|
|
})
|
|
.catch(() => {
|
|
toast.error("Error stopping PostgreSQL");
|
|
});
|
|
}}
|
|
>
|
|
<Button
|
|
variant="destructive"
|
|
isLoading={isStopping}
|
|
className="flex items-center gap-1.5 focus-visible:ring-2 focus-visible:ring-offset-2"
|
|
>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<div className="flex items-center">
|
|
<Ban className="size-4 mr-1" />
|
|
Stop
|
|
</div>
|
|
</TooltipTrigger>
|
|
<TooltipPrimitive.Portal>
|
|
<TooltipContent sideOffset={5} className="z-[60]">
|
|
<p>
|
|
Stop the currently running PostgreSQL database
|
|
</p>
|
|
</TooltipContent>
|
|
</TooltipPrimitive.Portal>
|
|
</Tooltip>
|
|
</Button>
|
|
</DialogAction>
|
|
))}
|
|
</TooltipProvider>
|
|
<DockerTerminalModal
|
|
appName={data?.appName || ""}
|
|
serverId={data?.serverId || ""}
|
|
>
|
|
<Button
|
|
variant="outline"
|
|
className="flex items-center gap-1.5 focus-visible:ring-2 focus-visible:ring-offset-2"
|
|
>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<div className="flex items-center">
|
|
<Terminal className="size-4 mr-1" />
|
|
Open Terminal
|
|
</div>
|
|
</TooltipTrigger>
|
|
<TooltipPrimitive.Portal>
|
|
<TooltipContent sideOffset={5} className="z-[60]">
|
|
<p>Open a terminal to the PostgreSQL container</p>
|
|
</TooltipContent>
|
|
</TooltipPrimitive.Portal>
|
|
</Tooltip>
|
|
</Button>
|
|
</DockerTerminalModal>
|
|
</CardContent>
|
|
</Card>
|
|
<DrawerLogs
|
|
isOpen={isDrawerOpen}
|
|
onClose={() => {
|
|
setIsDrawerOpen(false);
|
|
setFilteredLogs([]);
|
|
setIsDeploying(false);
|
|
refetch();
|
|
}}
|
|
filteredLogs={filteredLogs}
|
|
/>
|
|
</div>
|
|
</>
|
|
);
|
|
};
|