463 lines
16 KiB
TypeScript
463 lines
16 KiB
TypeScript
"use client";
|
|
|
|
import {
|
|
Dispatch,
|
|
SetStateAction,
|
|
useEffect,
|
|
useState,
|
|
useTransition,
|
|
} from "react";
|
|
import Link from "next/link";
|
|
import { zodResolver } from "@hookform/resolvers/zod";
|
|
import { User } from "@prisma/client";
|
|
import { useForm } from "react-hook-form";
|
|
import { toast } from "sonner";
|
|
|
|
import { getZoneDetail } from "@/lib/cloudflare";
|
|
import { DomainFormData } from "@/lib/dto/domains";
|
|
import { cn } from "@/lib/utils";
|
|
import { createDomainSchema } from "@/lib/validations/domain";
|
|
import { Button } from "@/components/ui/button";
|
|
import { Input } from "@/components/ui/input";
|
|
import { Label } from "@/components/ui/label";
|
|
import { Icons } from "@/components/shared/icons";
|
|
|
|
import { FormSectionColumns } from "../dashboard/form-section-columns";
|
|
import { Badge } from "../ui/badge";
|
|
import {
|
|
Collapsible,
|
|
CollapsibleContent,
|
|
CollapsibleTrigger,
|
|
} from "../ui/collapsible";
|
|
import { Switch } from "../ui/switch";
|
|
|
|
export type FormData = DomainFormData;
|
|
|
|
export type FormType = "add" | "edit";
|
|
|
|
export interface DomainFormProps {
|
|
user: Pick<User, "id" | "name">;
|
|
isShowForm: boolean;
|
|
setShowForm: Dispatch<SetStateAction<boolean>>;
|
|
type: FormType;
|
|
initData?: DomainFormData | null;
|
|
action: string;
|
|
onRefresh: () => void;
|
|
}
|
|
|
|
export function DomainForm({
|
|
setShowForm,
|
|
type,
|
|
initData,
|
|
action,
|
|
onRefresh,
|
|
}: DomainFormProps) {
|
|
const [isPending, startTransition] = useTransition();
|
|
const [isDeleting, startDeleteTransition] = useTransition();
|
|
const [isChecking, startCheckTransition] = useTransition();
|
|
const [currentRecordStatus, setCurrentRecordStatus] = useState(
|
|
initData?.enable_dns || false,
|
|
);
|
|
const [isChecked, setIsChecked] = useState(false);
|
|
|
|
const {
|
|
handleSubmit,
|
|
register,
|
|
formState: { errors },
|
|
getValues,
|
|
setValue,
|
|
} = useForm<FormData>({
|
|
resolver: zodResolver(createDomainSchema),
|
|
defaultValues: {
|
|
id: initData?.id || "",
|
|
domain_name: initData?.domain_name || "",
|
|
enable_short_link: initData?.enable_short_link || false,
|
|
enable_email: initData?.enable_email || false,
|
|
enable_dns: initData?.enable_dns || false,
|
|
cf_zone_id: initData?.cf_zone_id || "",
|
|
cf_api_key: initData?.cf_api_key || "",
|
|
cf_email: initData?.cf_email || "",
|
|
cf_api_key_encrypted: initData?.cf_api_key_encrypted || false,
|
|
max_short_links: initData?.max_short_links || 0,
|
|
max_email_forwards: initData?.max_email_forwards || 0,
|
|
max_dns_records: initData?.max_dns_records || 0,
|
|
active: initData?.active || true,
|
|
},
|
|
});
|
|
|
|
const onSubmit = handleSubmit((data) => {
|
|
if (type === "add") {
|
|
handleCreateDomain(data);
|
|
} else if (type === "edit") {
|
|
handleUpdateDomain(data);
|
|
}
|
|
});
|
|
|
|
const handleCreateDomain = async (data: DomainFormData) => {
|
|
startTransition(async () => {
|
|
const response = await fetch(`${action}`, {
|
|
method: "POST",
|
|
body: JSON.stringify({
|
|
data,
|
|
}),
|
|
});
|
|
if (!response.ok || response.status !== 200) {
|
|
toast.error("Created Failed!", {
|
|
description: await response.text(),
|
|
});
|
|
} else {
|
|
// const res = await response.json();
|
|
toast.success(`Created successfully!`);
|
|
setShowForm(false);
|
|
onRefresh();
|
|
}
|
|
});
|
|
};
|
|
|
|
const handleUpdateDomain = async (data: DomainFormData) => {
|
|
startTransition(async () => {
|
|
if (type === "edit") {
|
|
const response = await fetch(`${action}`, {
|
|
method: "PUT",
|
|
body: JSON.stringify(data),
|
|
});
|
|
if (!response.ok || response.status !== 200) {
|
|
toast.error("Update Failed", {
|
|
description: await response.text(),
|
|
});
|
|
} else {
|
|
await response.json();
|
|
toast.success(`Update successfully!`);
|
|
setShowForm(false);
|
|
onRefresh();
|
|
}
|
|
}
|
|
});
|
|
};
|
|
|
|
const handleDeleteDomain = async () => {
|
|
if (type === "edit") {
|
|
startDeleteTransition(async () => {
|
|
const response = await fetch(`${action}`, {
|
|
method: "DELETE",
|
|
body: JSON.stringify({
|
|
domain_name: initData?.domain_name,
|
|
}),
|
|
});
|
|
if (!response.ok || response.status !== 200) {
|
|
toast.error("Delete Failed", {
|
|
description: await response.text(),
|
|
});
|
|
} else {
|
|
await response.json();
|
|
toast.success(`Success`);
|
|
setShowForm(false);
|
|
onRefresh();
|
|
}
|
|
});
|
|
}
|
|
};
|
|
|
|
const handleCheckAccess = async (event) => {
|
|
event?.stopPropagation();
|
|
if (!currentRecordStatus) return;
|
|
|
|
if (isChecked) {
|
|
setIsChecked(false);
|
|
}
|
|
|
|
startCheckTransition(async () => {
|
|
const values = getValues(["cf_zone_id", "cf_api_key", "cf_email"]);
|
|
const res = await fetch(
|
|
`/api/domain/access-check?zone_id=${values[0]}&api_key=${values[1]}&email=${values[2]}`,
|
|
);
|
|
if (res.ok) {
|
|
const data = await res.json();
|
|
if (data === 200) {
|
|
setIsChecked(true);
|
|
return;
|
|
}
|
|
}
|
|
setIsChecked(false);
|
|
toast.error("Access Failed", {
|
|
description: "Please check your Cloudflare settings and try again.",
|
|
});
|
|
});
|
|
};
|
|
|
|
const ReadyBadge = (
|
|
<Badge
|
|
className={cn(
|
|
"ml-auto text-xs font-semibold",
|
|
!currentRecordStatus && "text-muted-foreground",
|
|
)}
|
|
variant={
|
|
currentRecordStatus ? (isChecked ? "green" : "default") : "outline"
|
|
}
|
|
onClick={(event) => handleCheckAccess(event)}
|
|
>
|
|
{isChecking && <Icons.spinner className="mr-1 size-3 animate-spin" />}
|
|
{isChecked && !isChecking && <Icons.check className="mr-1 size-3" />}
|
|
{isChecked ? "Ready" : "Access Check"}
|
|
</Badge>
|
|
);
|
|
|
|
return (
|
|
<div>
|
|
<div className="rounded-t-lg bg-muted px-4 py-2 text-lg font-semibold">
|
|
{type === "add" ? "Create" : "Edit"} Domain
|
|
</div>
|
|
<form className="p-4" onSubmit={onSubmit}>
|
|
<div className="relative flex flex-col items-center justify-start gap-0 rounded-md bg-neutral-100 p-4 dark:bg-neutral-800">
|
|
<h2 className="absolute left-2 top-2 text-xs font-semibold text-neutral-400">
|
|
Base
|
|
</h2>
|
|
<FormSectionColumns title="">
|
|
<div className="flex w-full items-start justify-between gap-2">
|
|
<Label className="mt-2.5 text-nowrap" htmlFor="domain_name">
|
|
Domain Name:
|
|
</Label>
|
|
<div className="w-full sm:w-3/5">
|
|
<Input
|
|
id="target"
|
|
className="flex-1 bg-neutral-50 shadow-inner"
|
|
size={32}
|
|
{...register("domain_name")}
|
|
/>
|
|
<div className="flex flex-col justify-between p-1">
|
|
{errors?.domain_name ? (
|
|
<p className="pb-0.5 text-[13px] text-red-600">
|
|
{errors.domain_name.message}
|
|
</p>
|
|
) : (
|
|
<p className="pb-0.5 text-[13px] text-muted-foreground">
|
|
Required. eg: example.com
|
|
</p>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</FormSectionColumns>
|
|
|
|
<div className="flex w-full items-center justify-between gap-2">
|
|
<Label className="" htmlFor="active">
|
|
Active:
|
|
</Label>
|
|
<Switch
|
|
id="active"
|
|
{...register("active")}
|
|
defaultChecked={initData?.active ?? true}
|
|
onCheckedChange={(value) => setValue("active", value)}
|
|
disabled
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="relative mt-2 flex flex-col items-center justify-start gap-4 rounded-md bg-neutral-100 p-4 pt-10 dark:bg-neutral-800">
|
|
<h2 className="absolute left-2 top-2 text-xs font-semibold text-neutral-400">
|
|
Services(Optional)
|
|
</h2>
|
|
|
|
<div className="flex w-full items-center justify-between gap-2">
|
|
<Label className="" htmlFor="short_url_service">
|
|
Short URL Service:
|
|
</Label>
|
|
<Switch
|
|
id="short_url_service"
|
|
{...register("enable_short_link")}
|
|
defaultChecked={initData?.enable_short_link ?? false}
|
|
onCheckedChange={(value) => setValue("enable_short_link", value)}
|
|
/>
|
|
</div>
|
|
|
|
<div className="flex w-full items-center justify-between gap-2">
|
|
<Label className="" htmlFor="email_service">
|
|
Email Service:
|
|
</Label>
|
|
<Switch
|
|
id="email_service"
|
|
{...register("enable_email")}
|
|
defaultChecked={initData?.enable_email ?? false}
|
|
onCheckedChange={(value) => setValue("enable_email", value)}
|
|
/>
|
|
</div>
|
|
|
|
<div className="flex w-full items-center justify-between gap-2">
|
|
<Label className="cursor-pointer" htmlFor="dns_record_service">
|
|
DNS Record Service:
|
|
</Label>
|
|
<Switch
|
|
id="dns_record_service"
|
|
{...register("enable_dns")}
|
|
defaultChecked={initData?.enable_dns ?? false}
|
|
onCheckedChange={(value) => {
|
|
setValue("enable_dns", value);
|
|
setCurrentRecordStatus(value);
|
|
}}
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<Collapsible className="relative mt-2 rounded-md bg-neutral-100 p-4 dark:bg-neutral-800">
|
|
<CollapsibleTrigger className="flex w-full items-center justify-between">
|
|
<h2 className="absolute left-2 top-5 text-xs font-semibold text-neutral-400">
|
|
Cloudflare Configs(Optional)
|
|
</h2>
|
|
{ReadyBadge}
|
|
<Icons.chevronDown className="ml-2 size-4" />
|
|
</CollapsibleTrigger>
|
|
<CollapsibleContent>
|
|
{!currentRecordStatus && (
|
|
<div className="mt-3 flex items-center gap-1 rounded bg-neutral-200 p-2 text-xs dark:bg-neutral-700">
|
|
<Icons.help className="size-3" /> Associate with "DNS Record
|
|
Service" status
|
|
</div>
|
|
)}
|
|
<FormSectionColumns title="">
|
|
<div className="flex w-full items-start justify-between gap-2">
|
|
<Label className="mt-2.5 text-nowrap" htmlFor="zone_id">
|
|
Zone ID:
|
|
</Label>
|
|
<div className="w-full sm:w-3/5">
|
|
<Input
|
|
id="target"
|
|
className="flex-1 bg-neutral-50 shadow-inner"
|
|
size={32}
|
|
{...register("cf_zone_id")}
|
|
disabled={!currentRecordStatus}
|
|
/>
|
|
<div className="flex flex-col justify-between p-1">
|
|
{errors?.cf_zone_id ? (
|
|
<p className="pb-0.5 text-[13px] text-red-600">
|
|
{errors.cf_zone_id.message}
|
|
</p>
|
|
) : (
|
|
<p className="pb-0.5 text-[13px] text-muted-foreground">
|
|
Optional.{" "}
|
|
<Link
|
|
className="text-blue-500"
|
|
href="/docs/developer/cloudflare"
|
|
target="_blank"
|
|
>
|
|
How to get zone id?
|
|
</Link>
|
|
</p>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</FormSectionColumns>
|
|
<FormSectionColumns title="">
|
|
<div className="flex w-full items-start justify-between gap-2">
|
|
<Label className="mt-2.5 text-nowrap" htmlFor="api-key">
|
|
API Token:
|
|
</Label>
|
|
<div className="w-full sm:w-3/5">
|
|
<Input
|
|
id="target"
|
|
className="flex-1 bg-neutral-50 shadow-inner"
|
|
size={32}
|
|
{...register("cf_api_key")}
|
|
disabled={!currentRecordStatus}
|
|
/>
|
|
<div className="flex flex-col justify-between p-1">
|
|
{errors?.cf_api_key ? (
|
|
<p className="pb-0.5 text-[13px] text-red-600">
|
|
{errors.cf_api_key.message}
|
|
</p>
|
|
) : (
|
|
<p className="pb-0.5 text-[13px] text-muted-foreground">
|
|
Optional.{" "}
|
|
<Link
|
|
className="text-blue-500"
|
|
href="/docs/developer/cloudflare"
|
|
target="_blank"
|
|
>
|
|
How to get api token?
|
|
</Link>
|
|
</p>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</FormSectionColumns>
|
|
<FormSectionColumns title="">
|
|
<div className="flex w-full items-start justify-between gap-2">
|
|
<Label className="mt-2.5 text-nowrap" htmlFor="email">
|
|
Account Email:
|
|
</Label>
|
|
<div className="w-full sm:w-3/5">
|
|
<Input
|
|
id="target"
|
|
className="flex-1 bg-neutral-50 shadow-inner"
|
|
size={32}
|
|
{...register("cf_email")}
|
|
disabled={!currentRecordStatus}
|
|
/>
|
|
<div className="flex flex-col justify-between p-1">
|
|
{errors?.cf_email ? (
|
|
<p className="pb-0.5 text-[13px] text-red-600">
|
|
{errors.cf_email.message}
|
|
</p>
|
|
) : (
|
|
<p className="pb-0.5 text-[13px] text-muted-foreground">
|
|
Optional.{" "}
|
|
<Link
|
|
className="text-blue-500"
|
|
href="/docs/developer/cloudflare"
|
|
target="_blank"
|
|
>
|
|
How to get cloudflare account email?
|
|
</Link>
|
|
</p>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</FormSectionColumns>
|
|
</CollapsibleContent>
|
|
</Collapsible>
|
|
|
|
{/* Action buttons */}
|
|
<div className="mt-3 flex justify-end gap-3">
|
|
{type === "edit" && (
|
|
<Button
|
|
type="button"
|
|
variant="destructive"
|
|
className="mr-auto w-[80px] px-0"
|
|
onClick={() => handleDeleteDomain()}
|
|
disabled={isDeleting}
|
|
>
|
|
{isDeleting ? (
|
|
<Icons.spinner className="size-4 animate-spin" />
|
|
) : (
|
|
<p>Delete</p>
|
|
)}
|
|
</Button>
|
|
)}
|
|
<Button
|
|
type="reset"
|
|
variant="outline"
|
|
className="w-[80px] px-0"
|
|
onClick={() => setShowForm(false)}
|
|
>
|
|
Cancle
|
|
</Button>
|
|
<Button
|
|
type="submit"
|
|
variant="blue"
|
|
disabled={isPending}
|
|
className="w-[80px] shrink-0 px-0"
|
|
>
|
|
{isPending ? (
|
|
<Icons.spinner className="size-4 animate-spin" />
|
|
) : (
|
|
<p>{type === "edit" ? "Update" : "Save"}</p>
|
|
)}
|
|
</Button>
|
|
</div>
|
|
</form>
|
|
</div>
|
|
);
|
|
}
|