import { Button } from "@/components/ui/button"; import { Dialog, DialogContent, DialogDescription, DialogFooter, DialogHeader, DialogTitle, DialogTrigger, } from "@/components/ui/dialog"; import { Form, FormControl, FormDescription, FormField, FormItem, FormLabel, FormMessage, } from "@/components/ui/form"; import { api } from "@/utils/api"; import { AlertBlock } from "@/components/shared/alert-block"; import { zodResolver } from "@hookform/resolvers/zod"; import { useEffect } from "react"; import { useForm } from "react-hook-form"; import { toast } from "sonner"; import { z } from "zod"; import { HelpCircle, Settings } from "lucide-react"; import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger, } from "@/components/ui/tooltip"; import { CodeEditor } from "@/components/shared/code-editor"; const HealthCheckSwarmSchema = z .object({ Test: z.array(z.string()).optional(), Interval: z.number().optional(), Timeout: z.number().optional(), StartPeriod: z.number().optional(), Retries: z.number().optional(), }) .strict(); const RestartPolicySwarmSchema = z .object({ Condition: z.string().optional(), Delay: z.number().optional(), MaxAttempts: z.number().optional(), Window: z.number().optional(), }) .strict(); const PreferenceSchema = z .object({ Spread: z.object({ SpreadDescriptor: z.string(), }), }) .strict(); const PlatformSchema = z .object({ Architecture: z.string(), OS: z.string(), }) .strict(); const PlacementSwarmSchema = z .object({ Constraints: z.array(z.string()).optional(), Preferences: z.array(PreferenceSchema).optional(), MaxReplicas: z.number().optional(), Platforms: z.array(PlatformSchema).optional(), }) .strict(); const UpdateConfigSwarmSchema = z .object({ Parallelism: z.number(), Delay: z.number().optional(), FailureAction: z.string().optional(), Monitor: z.number().optional(), MaxFailureRatio: z.number().optional(), Order: z.string(), }) .strict(); const ReplicatedSchema = z .object({ Replicas: z.number().optional(), }) .strict(); const ReplicatedJobSchema = z .object({ MaxConcurrent: z.number().optional(), TotalCompletions: z.number().optional(), }) .strict(); const ServiceModeSwarmSchema = z .object({ Replicated: ReplicatedSchema.optional(), Global: z.object({}).optional(), ReplicatedJob: ReplicatedJobSchema.optional(), GlobalJob: z.object({}).optional(), }) .strict(); const NetworkSwarmSchema = z.array( z .object({ Target: z.string().optional(), Aliases: z.array(z.string()).optional(), DriverOpts: z.object({}).optional(), }) .strict(), ); const LabelsSwarmSchema = z.record(z.string()); const createStringToJSONSchema = (schema: z.ZodTypeAny) => { return z .string() .transform((str, ctx) => { if (str === null || str === "") { return null; } try { return JSON.parse(str); } catch (e) { ctx.addIssue({ code: "custom", message: "Invalid JSON format" }); return z.NEVER; } }) .superRefine((data, ctx) => { if (data === null) { return; } if (Object.keys(data).length === 0) { ctx.addIssue({ code: z.ZodIssueCode.custom, message: "Object cannot be empty", }); return; } const parseResult = schema.safeParse(data); if (!parseResult.success) { for (const error of parseResult.error.issues) { const path = error.path.join("."); ctx.addIssue({ code: z.ZodIssueCode.custom, message: `${path} ${error.message}`, }); } } }); }; const addSwarmSettings = z.object({ healthCheckSwarm: createStringToJSONSchema(HealthCheckSwarmSchema).nullable(), restartPolicySwarm: createStringToJSONSchema( RestartPolicySwarmSchema, ).nullable(), placementSwarm: createStringToJSONSchema(PlacementSwarmSchema).nullable(), updateConfigSwarm: createStringToJSONSchema( UpdateConfigSwarmSchema, ).nullable(), rollbackConfigSwarm: createStringToJSONSchema( UpdateConfigSwarmSchema, ).nullable(), modeSwarm: createStringToJSONSchema(ServiceModeSwarmSchema).nullable(), labelsSwarm: createStringToJSONSchema(LabelsSwarmSchema).nullable(), networkSwarm: createStringToJSONSchema(NetworkSwarmSchema).nullable(), }); type AddSwarmSettings = z.infer; interface Props { applicationId: string; } export const AddSwarmSettings = ({ applicationId }: Props) => { const { data, refetch } = api.application.one.useQuery( { applicationId, }, { enabled: !!applicationId, }, ); const { mutateAsync, isError, error, isLoading } = api.application.update.useMutation(); const form = useForm({ defaultValues: { healthCheckSwarm: null, restartPolicySwarm: null, placementSwarm: null, updateConfigSwarm: null, rollbackConfigSwarm: null, modeSwarm: null, labelsSwarm: null, networkSwarm: null, }, resolver: zodResolver(addSwarmSettings), }); useEffect(() => { if (data) { form.reset({ healthCheckSwarm: data.healthCheckSwarm ? JSON.stringify(data.healthCheckSwarm, null, 2) : null, restartPolicySwarm: data.restartPolicySwarm ? JSON.stringify(data.restartPolicySwarm, null, 2) : null, placementSwarm: data.placementSwarm ? JSON.stringify(data.placementSwarm, null, 2) : null, updateConfigSwarm: data.updateConfigSwarm ? JSON.stringify(data.updateConfigSwarm, null, 2) : null, rollbackConfigSwarm: data.rollbackConfigSwarm ? JSON.stringify(data.rollbackConfigSwarm, null, 2) : null, modeSwarm: data.modeSwarm ? JSON.stringify(data.modeSwarm, null, 2) : null, labelsSwarm: data.labelsSwarm ? JSON.stringify(data.labelsSwarm, null, 2) : null, networkSwarm: data.networkSwarm ? JSON.stringify(data.networkSwarm, null, 2) : null, }); } }, [form, form.reset, data]); const onSubmit = async (data: AddSwarmSettings) => { await mutateAsync({ applicationId, healthCheckSwarm: data.healthCheckSwarm, restartPolicySwarm: data.restartPolicySwarm, placementSwarm: data.placementSwarm, updateConfigSwarm: data.updateConfigSwarm, rollbackConfigSwarm: data.rollbackConfigSwarm, modeSwarm: data.modeSwarm, labelsSwarm: data.labelsSwarm, networkSwarm: data.networkSwarm, }) .then(async () => { toast.success("Swarm settings updated"); refetch(); }) .catch(() => { toast.error("Error to update the swarm settings"); }); }; return ( Swarm Settings Update certain settings using a json object. {isError && {error?.message}}
( Health Check Check the interface
														{`{
	Test?: string[] | undefined;
	Interval?: number | undefined;
	Timeout?: number | undefined;
	StartPeriod?: number | undefined;
	Retries?: number | undefined;
}`}
													
										
									
)} /> ( Restart Policy Check the interface
														{`{
	Condition?: string | undefined;
	Delay?: number | undefined;
	MaxAttempts?: number | undefined;
	Window?: number | undefined;
}`}
													
										
									
)} /> ( Placement Check the interface
														{`{
	Constraints?: string[] | undefined;
	Preferences?: Array<{ Spread: { SpreadDescriptor: string } }> | undefined;
	MaxReplicas?: number | undefined;
	Platforms?:
		| Array<{
				Architecture: string;
				OS: string;
		  }>
		| undefined;
}`}
													
										
									
)} /> ( Update Config Check the interface
														{`{
	Parallelism?: number;
	Delay?: number | undefined;
	FailureAction?: string | undefined;
	Monitor?: number | undefined;
	MaxFailureRatio?: number | undefined;
	Order: string;
}`}
													
										
									
)} /> ( Rollback Config Check the interface
														{`{
	Parallelism?: number;
	Delay?: number | undefined;
	FailureAction?: string | undefined;
	Monitor?: number | undefined;
	MaxFailureRatio?: number | undefined;
	Order: string;
}`}
													
										
									
)} /> ( Mode Check the interface
														{`{
	Replicated?: { Replicas?: number | undefined } | undefined;
	Global?: {} | undefined;
	ReplicatedJob?:
		| {
				MaxConcurrent?: number | undefined;
				TotalCompletions?: number | undefined;
		  }
		| undefined;
	GlobalJob?: {} | undefined;
}`}
													
										
									
)} /> ( Network Check the interface
														{`[
  {
	"Target" : string | undefined;
	"Aliases" : string[] | undefined;
	"DriverOpts" : { [key: string]: string } | undefined;
  }
]`}
													
										
									
)} /> ( Labels Check the interface
														{`{
	[name: string]: string;
}`}
													
										
									
)} />
); };