Files
prowler/ui/components/roles/workflow/forms/edit-role-form.tsx
T
2026-05-08 10:36:43 +02:00

279 lines
8.6 KiB
TypeScript

"use client";
import { Checkbox } from "@heroui/checkbox";
import { Divider } from "@heroui/divider";
import { Tooltip } from "@heroui/tooltip";
import { zodResolver } from "@hookform/resolvers/zod";
import { clsx } from "clsx";
import { InfoIcon } from "lucide-react";
import { useRouter } from "next/navigation";
import { useEffect } from "react";
import { Controller, useForm } from "react-hook-form";
import { z } from "zod";
import { updateRole } from "@/actions/roles/roles";
import { EnhancedMultiSelect } from "@/components/shadcn/select/enhanced-multi-select";
import { useToast } from "@/components/ui";
import { CustomInput } from "@/components/ui/custom";
import { Form, FormButtons } from "@/components/ui/form";
import { getErrorMessage, permissionFormFields } from "@/lib";
import { ApiError, editRoleFormSchema } from "@/types";
type FormValues = z.input<typeof editRoleFormSchema>;
export const EditRoleForm = ({
roleId,
roleData,
groups,
}: {
roleId: string;
roleData: {
data: {
attributes: FormValues;
relationships?: {
provider_groups?: {
data: Array<{ id: string; type: string }>;
};
};
};
};
groups: { id: string; name: string }[];
}) => {
const { toast } = useToast();
const router = useRouter();
const isCloudEnvironment = process.env.NEXT_PUBLIC_IS_CLOUD_ENV === "true";
const visiblePermissionFormFields = permissionFormFields.filter(
(permission) =>
!["manage_billing", "manage_alerts"].includes(permission.field) ||
isCloudEnvironment,
);
const form = useForm<FormValues>({
resolver: zodResolver(editRoleFormSchema),
defaultValues: {
...roleData.data.attributes,
groups:
roleData.data.relationships?.provider_groups?.data.map((g) => g.id) ||
[],
},
});
const { watch, setValue } = form;
const manageProviders = watch("manage_providers");
const unlimitedVisibility = watch("unlimited_visibility");
useEffect(() => {
if (manageProviders && !unlimitedVisibility) {
setValue("unlimited_visibility", true, {
shouldValidate: true,
shouldDirty: true,
shouldTouch: true,
});
}
}, [manageProviders, unlimitedVisibility, setValue]);
const isLoading = form.formState.isSubmitting;
const onSelectAllChange = (checked: boolean) => {
visiblePermissionFormFields.forEach(({ field }) => {
form.setValue(field as keyof FormValues, checked, {
shouldValidate: true,
shouldDirty: true,
shouldTouch: true,
});
});
};
const onSubmitClient = async (values: FormValues) => {
try {
const updatedFields: Partial<FormValues> = {};
if (values.name !== roleData.data.attributes.name) {
updatedFields.name = values.name;
}
updatedFields.manage_users = values.manage_users;
updatedFields.manage_providers = values.manage_providers;
updatedFields.manage_account = values.manage_account;
updatedFields.manage_integrations = values.manage_integrations;
updatedFields.manage_scans = values.manage_scans;
updatedFields.unlimited_visibility = values.unlimited_visibility;
if (isCloudEnvironment) {
updatedFields.manage_billing = values.manage_billing;
updatedFields.manage_alerts = values.manage_alerts;
}
if (
JSON.stringify(values.groups) !==
JSON.stringify(
roleData.data.relationships?.provider_groups?.data.map((g) => g.id),
)
) {
updatedFields.groups = values.groups;
}
const formData = new FormData();
Object.entries(updatedFields).forEach(([key, value]) => {
if (key === "groups" && Array.isArray(value)) {
value.forEach((group) => {
formData.append("groups[]", group);
});
} else {
formData.append(key, String(value));
}
});
const data = await updateRole(formData, roleId);
if (data?.errors && data.errors.length > 0) {
data.errors.forEach((error: ApiError) => {
const errorMessage = error.detail;
const pointer = error.source?.pointer;
switch (pointer) {
case "/data/attributes/name":
form.setError("name", {
type: "server",
message: errorMessage,
});
break;
default:
toast({
variant: "destructive",
title: "Error",
description: errorMessage,
});
}
});
} else {
toast({
title: "Role Updated",
description: "The role was updated successfully.",
});
router.push("/roles");
}
} catch (error) {
toast({
variant: "destructive",
title: "Error",
description: getErrorMessage(error),
});
}
};
return (
<Form {...form}>
<form
onSubmit={form.handleSubmit(onSubmitClient)}
className="flex flex-col gap-6"
>
<CustomInput
control={form.control}
name="name"
type="text"
label="Role Name"
labelPlacement="inside"
placeholder="Enter role name"
variant="bordered"
isRequired
/>
<div className="flex flex-col gap-4">
<span className="text-lg font-semibold">Admin Permissions</span>
{/* Select All Checkbox */}
<Checkbox
isSelected={visiblePermissionFormFields.every((perm) =>
form.watch(perm.field as keyof FormValues),
)}
onChange={(e) => onSelectAllChange(e.target.checked)}
classNames={{
label: "text-small",
wrapper: "checkbox-update",
}}
color="default"
>
Grant all admin permissions
</Checkbox>
{/* Permissions Grid */}
<div className="grid grid-cols-2 gap-4">
{visiblePermissionFormFields.map(
({ field, label, description }) => (
<div key={field} className="flex items-center gap-2">
<Checkbox
{...form.register(field as keyof FormValues)}
isSelected={!!form.watch(field as keyof FormValues)}
classNames={{
label: "text-small",
wrapper: "checkbox-update",
}}
color="default"
>
{label}
</Checkbox>
<Tooltip content={description} placement="right">
<div className="flex w-fit items-center justify-center">
<InfoIcon
className={clsx(
"text-default-400 group-data-[selected=true]:text-foreground cursor-pointer",
)}
aria-hidden={"true"}
width={16}
/>
</div>
</Tooltip>
</div>
),
)}
</div>
</div>
<Divider className="my-4" />
{!unlimitedVisibility && (
<div className="flex flex-col gap-4">
<span className="text-lg font-semibold">Groups visibility</span>
<p className="text-small text-default-700 font-medium">
Select the groups this role will have access to. If no groups are
selected and unlimited visibility is not enabled, the role will
not have access to any accounts.
</p>
<Controller
name="groups"
control={form.control}
render={({ field }) => (
<div className="flex flex-col gap-2">
<EnhancedMultiSelect
options={groups.map((group) => ({
label: group.name,
value: group.id,
}))}
onValueChange={field.onChange}
defaultValue={field.value || []}
placeholder="Select groups"
searchable={true}
hideSelectAll={true}
emptyIndicator="No results found"
resetOnDefaultValueChange={true}
/>
</div>
)}
/>
{form.formState.errors.groups && (
<p className="mt-2 text-sm text-red-600">
{form.formState.errors.groups.message}
</p>
)}
</div>
)}
<FormButtons submitText="Update Role" isDisabled={isLoading} />
</form>
</Form>
);
};