mirror of
https://github.com/prowler-cloud/prowler.git
synced 2026-07-04 19:21:51 +00:00
279 lines
8.6 KiB
TypeScript
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>
|
|
);
|
|
};
|