mirror of
https://github.com/dokploy/dokploy.git
synced 2026-06-13 19:09:49 +00:00
30b3e1fe48
* fix(migrate-auth-secret): exit cleanly when there are no 2FA records The empty-records branch of `main()` returned without calling `process.exit(0)`, leaving the Drizzle Postgres connection pool holding the event loop open. The `migrate-auth-secret` process then hangs indefinitely after printing "No 2FA records found, nothing to migrate." causing the upstream `0.29.3.sh` security migration script (which calls this via `docker exec`) to never reach its final `docker service update` step that mounts the new Docker Secret. Operators end up with the new secret created but the dokploy service still configured with the hardcoded `BETTER_AUTH_SECRET`, while believing the migration completed. Match the success branch a few lines below which already does `process.exit(0)`, and the pattern used in sibling scripts `reset-password.ts` and `reset-2fa.ts`. Closes #4392 * feat(compose): add import from base64 in create service dropdown Adds an "Import" option to the Create Service dropdown that lets users paste a base64-encoded compose export, preview the template (compose YAML, domains, envs, mounts) before confirming, and create the service only on confirm. Adds a `previewTemplate` tRPC procedure that processes the base64 without touching the DB, with server access validation via session. * [autofix.ci] apply automated fixes * Enhance version synchronization workflow to include SDK repository - Updated the GitHub Actions workflow to sync versioning across MCP, CLI, and SDK repositories. - Added steps to bump the version in the SDK repository and regenerate tools from the latest OpenAPI spec. - Improved commit message formatting to include source and release information for all repositories. - Ensured successful synchronization messages for each repository after the version update. * feat(deployment): add readLogs procedure to fetch deployment logs - Introduced a new `readLogs` procedure that allows users to retrieve logs for a specific deployment by providing the deployment ID and an optional tail parameter. - Implemented permission checks to ensure users have access to the requested logs. - Enhanced log retrieval for both cloud and non-cloud environments, utilizing appropriate commands based on the server context. Resolve https://github.com/Dokploy/mcp/issues/14 * feat(deployment): add server access validation for deployment actions - Implemented server access validation in deployment procedures to ensure users can only access deployments associated with their active organization. - Added checks to throw an UNAUTHORIZED error if a user attempts to access a deployment linked to a server outside their organization. This enhancement improves security and access control within the deployment management system. * feat(organization): prevent inviting users with owner role - Added validation to prevent users from being invited with the owner role in the organization and user routers. - Implemented TRPCError responses to ensure proper error handling when attempting to assign the owner role. This change enhances role management and security within the organization structure. https://github.com/Dokploy/dokploy/security/advisories/GHSA-fm9p-wmpw-gxjh * feat(user): implement session cleanup on user update - Added functionality to delete old sessions when a user updates their password, ensuring that only the current session remains active. - This change enhances security by preventing unauthorized access from previous sessions after a password change. Close here https://github.com/Dokploy/dokploy/security/advisories/GHSA-rr9m-w87g-46f3 * feat(settings): add copy button to server IP in web server settings (#4397) * fix: copy Dokploy server IP when clicking server badge (#4390) * fix: copy Dokploy server IP when clicking server badge When a service runs on the local Dokploy server (no remote server), clicking the server badge did nothing because `data.server` is null. Now falls back to the server IP from settings so the badge always copies an IP address. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * feat(copy-ip): implement IP address copying functionality across database service components - Added the ability to copy the server IP address to the clipboard when clicking the server badge in various database service components (Libsql, MariaDB, MongoDB, MySQL, PostgreSQL, Redis). - Integrated the `copy-to-clipboard` library and `sonner` for user feedback upon successful copy action. - Ensured fallback to the server IP from settings when the service data is not available, enhancing user experience and functionality. --------- Co-authored-by: Claude Opus 4.6 (1M context) <noreply@anthropic.com> Co-authored-by: Mauricio Siu <siumauricio@icloud.com> * fix: responsive layout (#4391) Signed-off-by: Nahidujjaman Hridoy <hridoyboss12@gmail.com> * fix: automatically converting username to lowercase both in creation of register, and build for extra. (#4382) * fix: allow square brackets in zip path validation for Next.js dynamic routes (#4468) * fix: allow square brackets in zip drop path validation for Next.js dynamic routes ZIP uploads containing Next.js dynamic route files (e.g. app/api/[id]/route.ts, pages/[slug].tsx) were rejected by readValidDirectory because the path regex did not include square bracket characters. * [autofix.ci] apply automated fixes --------- Co-authored-by: autofix-ci[bot] <114827586+autofix-ci[bot]@users.noreply.github.com> * fix: prevent webhook deploy crash when commit data lacks modified files (#4470) shouldDeploy passed undefined/null entries from commit.modified straight into micromatch, which throws "Expected input to be a string" and fails every webhook deployment when watch paths are configured. Filter out non-string values before matching. * fix: add type="button" to TooltipTrigger in form components to prevent accidental submission (#4422) Co-authored-by: Maks Pikov <mixelburg@users.noreply.github.com> * fix: enable comment toggle shortcut in env variable editor (#4402) (#4473) * fix: add tls=true label for domains when certificateType is none (#4018) (#4474) * fix: add tls=true label for compose domains when certificateType is none (#4018) * test: cover tls=true label for certificateType none, require https * fix: scope tls fix to compose labels, leave traefik file config unchanged (#4018) * chore: update version to v0.29.5 in package.json * chore(deps): upgrade next to 16.2.6 (#4477) Upgraded next dependency in apps/dokploy to 16.2.6 exactly. Verified typescript typecheck passes successfully. * feat: add self-hosted enterprise restrictions (remote-servers-only, enforce-sso) (#4511) * feat: add self-hosted enterprise restrictions (remote-servers-only, enforce-sso) - Add `remoteServersOnly` field to webServerSettings: prevents creating services on the local Dokploy VM, forcing all deployments to remote servers. Validated in all 8 service routers (application, compose, postgres, mysql, mongo, redis, mariadb, libsql). - Add `enforceSSO` field to webServerSettings: hides the email/password login form and shows only the SSO button on the login page. - Both settings are enterprise-only (enterpriseProcedure) and self-hosted-only (blocked at the API level when IS_CLOUD=true). - UI toggles added to the SSO settings page under a new "Self-hosted Restrictions" card (hidden in cloud). Login page reads enforceSSO from getServerSideProps to avoid client-side flash. - Migrations: 0167_fresh_goliath.sql, 0168_long_justice.sql * fix: add missing final newlines to migration files * refactor: improve code formatting for better readability in multiple components - Adjusted formatting in `add-application.tsx`, `add-compose.tsx`, and `add-database.tsx` to enhance readability by adding line breaks and consistent indentation. - Updated `toggle-enforce-sso.tsx` to simplify the Switch component's props. - Reformatted imports in `index.tsx` and `sso.tsx` for consistency. - Cleaned up conditional statements in various router files for improved clarity. * fix: add enforceSSO to test mock * fix: grant create and delete SSH key permissions when canAccessToSSHKeys is enabled for members (#4512) * fix: use create permission for basic auth delete instead of delete (#4513) * fix: wrap long server names and keep actions menu visible (#4434) On settings/servers, a long server name in the card title (h3) did not wrap and overflowed its container, overlapping nearby content and squeezing the three-dots actions menu until it disappeared. Allow the title block to shrink and wrap (min-w-0 + break-words), keep the server icon and the actions trigger from being crushed (shrink-0), and add gap between the title and the actions button. * chore: update version to v0.29.6 in package.json * fix: preserve HOME in compose deploy so --with-registry-auth can read docker config (#4485) The compose/stack deploy command runs under `env -i PATH="$PATH"`, which clears the environment except for PATH. That strips HOME, so when the generated command is `docker stack deploy --prune --with-registry-auth` the docker CLI cannot resolve `~/.docker/config.json` (e.g. `/root/.docker/config.json`) and ships no registry credentials to the swarm. Private-registry images then fail to pull on the nodes: image registry.example.com/... could not be accessed on a registry to record its digest. Each node will access ... independently while the deploy still logs "Docker Compose Deployed: ✅". Keep PATH isolation but preserve HOME so docker can read its config for both `stack deploy --with-registry-auth` and `compose up -d --build`. Add a regression test asserting the generated command preserves `HOME="$HOME"` for both stack and docker-compose deploys. Fixes #4401 Co-authored-by: Claude Opus 4.7 (1M context) <noreply@anthropic.com> --------- Signed-off-by: Nahidujjaman Hridoy <hridoyboss12@gmail.com> Co-authored-by: ngenohkevin <ngenohkevin19@gmail.com> Co-authored-by: Mauricio Siu <47042324+Siumauricio@users.noreply.github.com> Co-authored-by: Mauricio Siu <siumauricio@icloud.com> Co-authored-by: autofix-ci[bot] <114827586+autofix-ci[bot]@users.noreply.github.com> Co-authored-by: Volodymyr Kravchuk <volodymyr.kravch@gmail.com> Co-authored-by: Claude Opus 4.6 (1M context) <noreply@anthropic.com> Co-authored-by: Nahidujjaman Hridoy <75487507+nhridoy@users.noreply.github.com> Co-authored-by: Francis <9560564+Baker@users.noreply.github.com> Co-authored-by: mixelburg <52622705+mixelburg@users.noreply.github.com> Co-authored-by: Maks Pikov <mixelburg@users.noreply.github.com> Co-authored-by: Jasael <67719321+jasael@users.noreply.github.com> Co-authored-by: Philippe Parage <69145356+pparage@users.noreply.github.com> Co-authored-by: youcef zr <93142224+youcefzemmar@users.noreply.github.com>
803 lines
23 KiB
TypeScript
803 lines
23 KiB
TypeScript
import { standardSchemaResolver as zodResolver } from "@hookform/resolvers/standard-schema";
|
|
import { AlertTriangle, Database, HelpCircle } from "lucide-react";
|
|
import { useState } from "react";
|
|
import { useForm } from "react-hook-form";
|
|
import { toast } from "sonner";
|
|
import { z } from "zod";
|
|
import {
|
|
LibsqlIcon,
|
|
MariadbIcon,
|
|
MongodbIcon,
|
|
MysqlIcon,
|
|
PostgresqlIcon,
|
|
RedisIcon,
|
|
} from "@/components/icons/data-tools-icons";
|
|
import { Button } from "@/components/ui/button";
|
|
import {
|
|
Dialog,
|
|
DialogContent,
|
|
DialogFooter,
|
|
DialogHeader,
|
|
DialogTitle,
|
|
DialogTrigger,
|
|
} from "@/components/ui/dialog";
|
|
import { DropdownMenuItem } from "@/components/ui/dropdown-menu";
|
|
import {
|
|
Form,
|
|
FormControl,
|
|
FormField,
|
|
FormItem,
|
|
FormLabel,
|
|
FormMessage,
|
|
} from "@/components/ui/form";
|
|
import { Input } from "@/components/ui/input";
|
|
import { Label } from "@/components/ui/label";
|
|
import { RadioGroup, RadioGroupItem } from "@/components/ui/radio-group";
|
|
import {
|
|
Select,
|
|
SelectContent,
|
|
SelectGroup,
|
|
SelectItem,
|
|
SelectLabel,
|
|
SelectTrigger,
|
|
SelectValue,
|
|
} from "@/components/ui/select";
|
|
import { Switch } from "@/components/ui/switch";
|
|
import { Textarea } from "@/components/ui/textarea";
|
|
import {
|
|
Tooltip,
|
|
TooltipContent,
|
|
TooltipProvider,
|
|
TooltipTrigger,
|
|
} from "@/components/ui/tooltip";
|
|
import { slugify } from "@/lib/slug";
|
|
import { api } from "@/utils/api";
|
|
import { APP_NAME_MESSAGE, APP_NAME_REGEX } from "@/utils/schema";
|
|
|
|
type DbType = z.infer<typeof mySchema>["type"];
|
|
|
|
const dockerImageDefaultPlaceholder: Record<DbType, string> = {
|
|
mongo: "mongo:8",
|
|
libsql: "ghcr.io/tursodatabase/libsql-server:v0.24.32",
|
|
mariadb: "mariadb:11",
|
|
mysql: "mysql:8",
|
|
postgres: "postgres:18",
|
|
redis: "redis:7",
|
|
};
|
|
|
|
const databasesUserDefaultPlaceholder: Record<
|
|
Exclude<DbType, "redis">,
|
|
string
|
|
> = {
|
|
libsql: "libsql",
|
|
mariadb: "mariadb",
|
|
mongo: "mongo",
|
|
mysql: "mysql",
|
|
postgres: "postgres",
|
|
};
|
|
|
|
const baseDatabaseSchema = z.object({
|
|
name: z.string().min(1, "Name required"),
|
|
appName: z
|
|
.string()
|
|
.min(1, {
|
|
message: "App name is required",
|
|
})
|
|
.regex(APP_NAME_REGEX, {
|
|
message: APP_NAME_MESSAGE,
|
|
}),
|
|
databasePassword: z
|
|
.string()
|
|
.regex(/^[a-zA-Z0-9@#%^&*()_+\-=[\]{}|;:,.<>?~`]*$/, {
|
|
message:
|
|
"Password contains invalid characters. Please avoid: $ ! ' \" \\ / and space characters for database compatibility",
|
|
}),
|
|
dockerImage: z.string(),
|
|
description: z.string().nullable(),
|
|
serverId: z.string().nullable(),
|
|
});
|
|
|
|
const mySchema = z
|
|
.discriminatedUnion("type", [
|
|
z
|
|
.object({
|
|
type: z.literal("libsql"),
|
|
dockerImage: z
|
|
.string()
|
|
.default("ghcr.io/tursodatabase/libsql-server:v0.24.32"),
|
|
databaseUser: z.string().default("libsql"),
|
|
sqldNode: z.enum(["primary", "replica"]).default("primary"),
|
|
sqldPrimaryUrl: z.string().optional(),
|
|
enableNamespaces: z.boolean().default(false),
|
|
})
|
|
.merge(baseDatabaseSchema),
|
|
z
|
|
.object({
|
|
type: z.literal("mariadb"),
|
|
dockerImage: z.string().default("mariadb:4"),
|
|
databaseRootPassword: z
|
|
.string()
|
|
.regex(/^[a-zA-Z0-9@#%^&*()_+\-=[\]{}|;:,.<>?~`]*$/, {
|
|
message:
|
|
"Password contains invalid characters. Please avoid: $ ! ' \" \\ / and space characters for database compatibility",
|
|
})
|
|
.optional(),
|
|
databaseUser: z.string().default("mariadb"),
|
|
databaseName: z.string().default("mariadb"),
|
|
})
|
|
.merge(baseDatabaseSchema),
|
|
z
|
|
.object({
|
|
type: z.literal("mongo"),
|
|
databaseUser: z.string().default("mongo"),
|
|
replicaSets: z.boolean().default(false),
|
|
})
|
|
.merge(baseDatabaseSchema),
|
|
z
|
|
.object({
|
|
type: z.literal("mysql"),
|
|
databaseRootPassword: z
|
|
.string()
|
|
.regex(/^[a-zA-Z0-9@#%^&*()_+\-=[\]{}|;:,.<>?~`]*$/, {
|
|
message:
|
|
"Password contains invalid characters. Please avoid: $ ! ' \" \\ / and space characters for database compatibility",
|
|
})
|
|
.optional(),
|
|
databaseUser: z.string().default("mysql"),
|
|
databaseName: z.string().default("mysql"),
|
|
})
|
|
.merge(baseDatabaseSchema),
|
|
z
|
|
.object({
|
|
type: z.literal("postgres"),
|
|
databaseName: z.string().default("postgres"),
|
|
databaseUser: z.string().default("postgres"),
|
|
})
|
|
.merge(baseDatabaseSchema),
|
|
z
|
|
.object({
|
|
type: z.literal("redis"),
|
|
})
|
|
.merge(baseDatabaseSchema),
|
|
])
|
|
.superRefine((data, ctx) => {
|
|
if (data.type === "libsql") {
|
|
if (data.sqldNode === "replica" && !data.sqldPrimaryUrl) {
|
|
ctx.addIssue({
|
|
code: z.ZodIssueCode.custom,
|
|
path: ["sqldPrimaryUrl"],
|
|
message: "sqldPrimaryUrl is required when sqldNode is 'replica'.",
|
|
});
|
|
}
|
|
if (data.sqldNode !== "replica" && data.sqldPrimaryUrl) {
|
|
ctx.addIssue({
|
|
code: z.ZodIssueCode.custom,
|
|
path: ["sqldPrimaryUrl"],
|
|
message:
|
|
"sqldPrimaryUrl should not be provided when sqldNode is not 'replica'.",
|
|
});
|
|
}
|
|
}
|
|
});
|
|
|
|
const databasesMap = {
|
|
postgres: {
|
|
icon: <PostgresqlIcon />,
|
|
label: "PostgreSQL",
|
|
},
|
|
mongo: {
|
|
icon: <MongodbIcon />,
|
|
label: "MongoDB",
|
|
},
|
|
mariadb: {
|
|
icon: <MariadbIcon />,
|
|
label: "MariaDB",
|
|
},
|
|
mysql: {
|
|
icon: <MysqlIcon />,
|
|
label: "MySQL",
|
|
},
|
|
redis: {
|
|
icon: <RedisIcon />,
|
|
label: "Redis",
|
|
},
|
|
libsql: {
|
|
icon: <LibsqlIcon className="size-10" />,
|
|
label: "libSQL",
|
|
},
|
|
};
|
|
|
|
type AddDatabase = z.infer<typeof mySchema>;
|
|
|
|
interface Props {
|
|
environmentId: string;
|
|
projectName?: string;
|
|
}
|
|
|
|
export const AddDatabase = ({ environmentId, projectName }: Props) => {
|
|
const utils = api.useUtils();
|
|
const [visible, setVisible] = useState(false);
|
|
const slug = slugify(projectName);
|
|
const { data: isCloud } = api.settings.isCloud.useQuery();
|
|
const { data: webServerSettings } =
|
|
api.settings.getWebServerSettings.useQuery();
|
|
const showLocalOption = !isCloud && !webServerSettings?.remoteServersOnly;
|
|
const { data: servers } = api.server.withSSHKey.useQuery();
|
|
const libsqlMutation = api.libsql.create.useMutation();
|
|
const mariadbMutation = api.mariadb.create.useMutation();
|
|
const mongoMutation = api.mongo.create.useMutation();
|
|
const mysqlMutation = api.mysql.create.useMutation();
|
|
const postgresMutation = api.postgres.create.useMutation();
|
|
const redisMutation = api.redis.create.useMutation();
|
|
|
|
// Get environment data to extract projectId
|
|
const { data: environment } = api.environment.one.useQuery({ environmentId });
|
|
|
|
const hasServers = servers && servers.length > 0;
|
|
// Show dropdown logic based on cloud environment
|
|
// Cloud: show only if there are remote servers (no Dokploy option)
|
|
// Self-hosted: show only if there are remote servers (Dokploy is default, hide if no remote servers)
|
|
const shouldShowServerDropdown = hasServers;
|
|
|
|
const form = useForm({
|
|
defaultValues: {
|
|
type: "postgres",
|
|
dockerImage: "",
|
|
name: "",
|
|
appName: `${slug}-`,
|
|
databasePassword: "",
|
|
description: "",
|
|
databaseName: "",
|
|
databaseUser: "",
|
|
serverId: null,
|
|
},
|
|
resolver: zodResolver(mySchema),
|
|
});
|
|
const sqldNode = form.watch("sqldNode");
|
|
const type = form.watch("type");
|
|
const activeMutation = {
|
|
libsql: libsqlMutation,
|
|
mariadb: mariadbMutation,
|
|
mongo: mongoMutation,
|
|
mysql: mysqlMutation,
|
|
postgres: postgresMutation,
|
|
redis: redisMutation,
|
|
};
|
|
|
|
const onSubmit = async (data: AddDatabase) => {
|
|
const defaultDockerImage =
|
|
data.dockerImage || dockerImageDefaultPlaceholder[data.type];
|
|
|
|
let promise: Promise<unknown> | null = null;
|
|
const commonParams = {
|
|
name: data.name,
|
|
appName: data.appName,
|
|
dockerImage: defaultDockerImage,
|
|
serverId: data.serverId === "dokploy" ? undefined : data.serverId,
|
|
environmentId,
|
|
description: data.description,
|
|
};
|
|
|
|
if (data.type === "libsql") {
|
|
promise = libsqlMutation.mutateAsync({
|
|
...commonParams,
|
|
sqldNode: data.sqldNode,
|
|
sqldPrimaryUrl: data.sqldPrimaryUrl ?? null,
|
|
enableNamespaces: data.enableNamespaces,
|
|
databasePassword: data.databasePassword,
|
|
databaseUser:
|
|
data.databaseUser || databasesUserDefaultPlaceholder[data.type],
|
|
serverId: data.serverId === "dokploy" ? null : data.serverId,
|
|
});
|
|
} else if (data.type === "mariadb") {
|
|
promise = mariadbMutation.mutateAsync({
|
|
...commonParams,
|
|
databasePassword: data.databasePassword,
|
|
databaseRootPassword: data.databaseRootPassword || "",
|
|
databaseName: data.databaseName || "mariadb",
|
|
databaseUser:
|
|
data.databaseUser || databasesUserDefaultPlaceholder[data.type],
|
|
serverId: data.serverId === "dokploy" ? null : data.serverId,
|
|
});
|
|
} else if (data.type === "mongo") {
|
|
promise = mongoMutation.mutateAsync({
|
|
...commonParams,
|
|
databasePassword: data.databasePassword,
|
|
databaseUser:
|
|
data.databaseUser || databasesUserDefaultPlaceholder[data.type],
|
|
serverId: data.serverId === "dokploy" ? null : data.serverId,
|
|
replicaSets: data.replicaSets,
|
|
});
|
|
} else if (data.type === "mysql") {
|
|
promise = mysqlMutation.mutateAsync({
|
|
...commonParams,
|
|
databasePassword: data.databasePassword,
|
|
databaseName: data.databaseName || "mysql",
|
|
databaseUser:
|
|
data.databaseUser || databasesUserDefaultPlaceholder[data.type],
|
|
serverId: data.serverId === "dokploy" ? null : data.serverId,
|
|
databaseRootPassword: data.databaseRootPassword || "",
|
|
});
|
|
} else if (data.type === "postgres") {
|
|
promise = postgresMutation.mutateAsync({
|
|
...commonParams,
|
|
databasePassword: data.databasePassword,
|
|
databaseName: data.databaseName || "postgres",
|
|
databaseUser:
|
|
data.databaseUser || databasesUserDefaultPlaceholder[data.type],
|
|
serverId: data.serverId === "dokploy" ? null : data.serverId,
|
|
});
|
|
} else if (data.type === "redis") {
|
|
promise = redisMutation.mutateAsync({
|
|
...commonParams,
|
|
databasePassword: data.databasePassword,
|
|
serverId: data.serverId === "dokploy" ? null : data.serverId,
|
|
});
|
|
}
|
|
|
|
if (promise) {
|
|
await promise
|
|
.then(async () => {
|
|
toast.success("Database Created");
|
|
form.reset({
|
|
type: "postgres",
|
|
dockerImage: "",
|
|
name: "",
|
|
appName: `${projectName}-`,
|
|
databasePassword: "",
|
|
description: "",
|
|
databaseName: "",
|
|
databaseUser: "",
|
|
});
|
|
setVisible(false);
|
|
// Invalidate the project query to refresh the environment data
|
|
await utils.environment.one.invalidate({
|
|
environmentId,
|
|
});
|
|
})
|
|
.catch(() => {
|
|
toast.error("Error creating a database");
|
|
});
|
|
}
|
|
};
|
|
|
|
return (
|
|
<Dialog open={visible} onOpenChange={setVisible}>
|
|
<DialogTrigger className="w-full">
|
|
<DropdownMenuItem
|
|
className="w-full cursor-pointer space-x-3"
|
|
onSelect={(e) => e.preventDefault()}
|
|
>
|
|
<Database className="size-4 text-muted-foreground" />
|
|
<span>Database</span>
|
|
</DropdownMenuItem>
|
|
</DialogTrigger>
|
|
<DialogContent className="md:max-h-[90vh] sm:max-w-2xl">
|
|
<DialogHeader>
|
|
<DialogTitle>Databases</DialogTitle>
|
|
</DialogHeader>
|
|
|
|
<Form {...form}>
|
|
<form
|
|
id="hook-form"
|
|
onSubmit={form.handleSubmit(onSubmit)}
|
|
className="grid w-full gap-8 "
|
|
>
|
|
<FormField
|
|
control={form.control}
|
|
defaultValue={form.control._defaultValues.type}
|
|
name="type"
|
|
render={({ field }) => (
|
|
<FormItem className="space-y-3">
|
|
<FormLabel className="text-muted-foreground">
|
|
Select a database
|
|
</FormLabel>
|
|
<FormControl>
|
|
<RadioGroup
|
|
onValueChange={field.onChange}
|
|
defaultValue={field.value}
|
|
className="grid w-full grid-cols-1 sm:grid-cols-2 md:grid-cols-3 gap-4"
|
|
>
|
|
{Object.entries(databasesMap).map(([key, value]) => (
|
|
<FormItem
|
|
key={key}
|
|
className="flex w-full items-center space-x-3 space-y-0"
|
|
>
|
|
<FormControl className="w-full">
|
|
<div>
|
|
<RadioGroupItem
|
|
value={key}
|
|
id={key}
|
|
className="peer sr-only"
|
|
/>
|
|
<Label
|
|
htmlFor={key}
|
|
className="flex flex-col items-center justify-between rounded-md border-2 border-muted bg-popover p-4 hover:bg-accent hover:text-accent-foreground peer-data-[state=checked]:border-primary [&:has([data-state=checked])]:border-primary cursor-pointer"
|
|
>
|
|
{value.icon}
|
|
{value.label}
|
|
</Label>
|
|
</div>
|
|
</FormControl>
|
|
</FormItem>
|
|
))}
|
|
</RadioGroup>
|
|
</FormControl>
|
|
<FormMessage />
|
|
{activeMutation[field.value].isError && (
|
|
<div className="flex flex-row gap-4 rounded-lg bg-red-50 p-2 dark:bg-red-950">
|
|
<AlertTriangle className="text-red-600 dark:text-red-400" />
|
|
<span className="text-sm text-red-600 dark:text-red-400">
|
|
{activeMutation[field.value].error?.message}
|
|
</span>
|
|
</div>
|
|
)}
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
<div className="flex flex-col gap-4">
|
|
<FormLabel className="text-lg font-semibold leading-none tracking-tight">
|
|
Fill the next fields.
|
|
</FormLabel>
|
|
<div className="flex flex-col gap-2">
|
|
<FormField
|
|
control={form.control}
|
|
name="name"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Name</FormLabel>
|
|
<FormControl>
|
|
<Input
|
|
placeholder="Name"
|
|
{...field}
|
|
onChange={(e) => {
|
|
const val = e.target.value || "";
|
|
const serviceName = slugify(val.trim());
|
|
form.setValue("appName", `${slug}-${serviceName}`);
|
|
field.onChange(val);
|
|
}}
|
|
/>
|
|
</FormControl>
|
|
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
{shouldShowServerDropdown && (
|
|
<FormField
|
|
control={form.control}
|
|
name="serverId"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Select a Server</FormLabel>
|
|
<Select
|
|
onValueChange={field.onChange}
|
|
defaultValue={
|
|
field.value ||
|
|
(showLocalOption ? "dokploy" : undefined)
|
|
}
|
|
>
|
|
<SelectTrigger>
|
|
<SelectValue
|
|
placeholder={
|
|
showLocalOption ? "Dokploy" : "Select a Server"
|
|
}
|
|
/>
|
|
</SelectTrigger>
|
|
<SelectContent>
|
|
<SelectGroup>
|
|
{showLocalOption && (
|
|
<SelectItem value="dokploy">
|
|
<span className="flex items-center gap-2 justify-between w-full">
|
|
<span>Dokploy</span>
|
|
<span className="text-muted-foreground text-xs self-center">
|
|
Default
|
|
</span>
|
|
</span>
|
|
</SelectItem>
|
|
)}
|
|
{servers?.map((server) => (
|
|
<SelectItem
|
|
key={server.serverId}
|
|
value={server.serverId}
|
|
>
|
|
{server.name}
|
|
</SelectItem>
|
|
))}
|
|
<SelectLabel>
|
|
Servers (
|
|
{servers?.length + (showLocalOption ? 1 : 0)})
|
|
</SelectLabel>
|
|
</SelectGroup>
|
|
</SelectContent>
|
|
</Select>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
)}
|
|
<FormField
|
|
control={form.control}
|
|
name="appName"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel className="flex items-center gap-2">
|
|
App Name
|
|
<TooltipProvider delayDuration={0}>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<HelpCircle className="size-4 text-muted-foreground" />
|
|
</TooltipTrigger>
|
|
<TooltipContent side="right">
|
|
<p>
|
|
This will be the name of the Docker Swarm
|
|
service
|
|
</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
</TooltipProvider>
|
|
</FormLabel>
|
|
<FormControl>
|
|
<Input placeholder="my-app" {...field} />
|
|
</FormControl>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
|
|
<FormField
|
|
control={form.control}
|
|
name="description"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Description</FormLabel>
|
|
<FormControl>
|
|
<Textarea
|
|
className="h-24"
|
|
placeholder="Description"
|
|
{...field}
|
|
value={field.value || ""}
|
|
/>
|
|
</FormControl>
|
|
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
{(type === "mariadb" ||
|
|
type === "mysql" ||
|
|
type === "postgres") && (
|
|
<FormField
|
|
control={form.control}
|
|
name="databaseName"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Database Name</FormLabel>
|
|
<FormControl>
|
|
<Input placeholder="Database Name" {...field} />
|
|
</FormControl>
|
|
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
)}
|
|
|
|
{type === "libsql" && (
|
|
<FormField
|
|
control={form.control}
|
|
name="sqldNode"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Sqld Node</FormLabel>
|
|
<Select
|
|
onValueChange={field.onChange}
|
|
defaultValue={field.value || "primary"}
|
|
>
|
|
<SelectTrigger>
|
|
<SelectValue placeholder={"primary"} />
|
|
</SelectTrigger>
|
|
<SelectContent>
|
|
<SelectGroup>
|
|
{["primary", "replica"].map((node) => (
|
|
<SelectItem key={node} value={node}>
|
|
{node.charAt(0).toUpperCase() + node.slice(1)}
|
|
</SelectItem>
|
|
))}
|
|
</SelectGroup>
|
|
</SelectContent>
|
|
</Select>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
)}
|
|
{type === "libsql" && sqldNode === "replica" && (
|
|
<FormField
|
|
control={form.control}
|
|
name="sqldPrimaryUrl"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Sqld Primary URL</FormLabel>
|
|
<FormControl>
|
|
<Input
|
|
placeholder={"https://<host>:<port>"}
|
|
autoComplete="off"
|
|
{...field}
|
|
/>
|
|
</FormControl>
|
|
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
)}
|
|
{type === "libsql" && (
|
|
<FormField
|
|
control={form.control}
|
|
name="enableNamespaces"
|
|
render={({ field }) => {
|
|
return (
|
|
<FormItem>
|
|
<FormLabel>Enable Namespaces</FormLabel>
|
|
<FormControl>
|
|
<Select
|
|
onValueChange={(value) =>
|
|
field.onChange(Boolean(value))
|
|
}
|
|
defaultValue={
|
|
field.value ? String(field.value) : "false"
|
|
}
|
|
>
|
|
<SelectTrigger>
|
|
<SelectValue placeholder={"false"} />
|
|
</SelectTrigger>
|
|
<SelectContent>
|
|
<SelectGroup>
|
|
{["false", "true"].map((node) => (
|
|
<SelectItem key={node} value={node}>
|
|
{node.charAt(0).toUpperCase() +
|
|
node.slice(1)}
|
|
</SelectItem>
|
|
))}
|
|
</SelectGroup>
|
|
</SelectContent>
|
|
</Select>
|
|
</FormControl>
|
|
|
|
<FormMessage />
|
|
</FormItem>
|
|
);
|
|
}}
|
|
/>
|
|
)}
|
|
{(type === "libsql" ||
|
|
type === "mariadb" ||
|
|
type === "mongo" ||
|
|
type === "mysql" ||
|
|
type === "postgres") && (
|
|
<FormField
|
|
control={form.control}
|
|
name="databaseUser"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Database User</FormLabel>
|
|
<FormControl>
|
|
<Input
|
|
placeholder={`Default ${databasesUserDefaultPlaceholder[type]}`}
|
|
autoComplete="off"
|
|
{...field}
|
|
/>
|
|
</FormControl>
|
|
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
)}
|
|
|
|
<FormField
|
|
control={form.control}
|
|
name="databasePassword"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Database Password</FormLabel>
|
|
<FormControl>
|
|
<Input
|
|
type="password"
|
|
placeholder="******************"
|
|
autoComplete="one-time-code"
|
|
enablePasswordGenerator={true}
|
|
{...field}
|
|
/>
|
|
</FormControl>
|
|
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
{(type === "mariadb" || type === "mysql") && (
|
|
<FormField
|
|
control={form.control}
|
|
name="databaseRootPassword"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Database Root password</FormLabel>
|
|
<FormControl>
|
|
<Input
|
|
type="password"
|
|
placeholder="******************"
|
|
enablePasswordGenerator={true}
|
|
{...field}
|
|
/>
|
|
</FormControl>
|
|
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
)}
|
|
|
|
<FormField
|
|
control={form.control}
|
|
name="dockerImage"
|
|
defaultValue={form.formState.defaultValues?.dockerImage}
|
|
render={({ field }) => {
|
|
return (
|
|
<FormItem>
|
|
<FormLabel>Docker image</FormLabel>
|
|
<FormControl>
|
|
<Input
|
|
placeholder={`Default ${dockerImageDefaultPlaceholder[type]}`}
|
|
{...field}
|
|
/>
|
|
</FormControl>
|
|
|
|
<FormMessage />
|
|
</FormItem>
|
|
);
|
|
}}
|
|
/>
|
|
|
|
{type === "mongo" && (
|
|
<FormField
|
|
control={form.control}
|
|
name="replicaSets"
|
|
render={({ field }) => {
|
|
return (
|
|
<FormItem className="flex flex-row items-center justify-between p-3 mt-4 border rounded-lg shadow-sm">
|
|
<div className="space-y-0.5">
|
|
<FormLabel>Use Replica Sets</FormLabel>
|
|
</div>
|
|
<FormControl>
|
|
<Switch
|
|
checked={field.value}
|
|
onCheckedChange={field.onChange}
|
|
/>
|
|
</FormControl>
|
|
|
|
<FormMessage />
|
|
</FormItem>
|
|
);
|
|
}}
|
|
/>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</form>
|
|
|
|
<DialogFooter>
|
|
<Button
|
|
isLoading={form.formState.isSubmitting}
|
|
form="hook-form"
|
|
type="submit"
|
|
>
|
|
Create
|
|
</Button>
|
|
</DialogFooter>
|
|
</Form>
|
|
</DialogContent>
|
|
</Dialog>
|
|
);
|
|
};
|