Files
prowler/ui/actions/roles/roles.ts
Alan Buscaglia 4d5676f00e feat: upgrade to React 19, Next.js 15, React Compiler, HeroUI and Tailwind 4 (#8748)
Co-authored-by: Alan Buscaglia <alanbuscaglia@MacBook-Pro.local>
Co-authored-by: alejandrobailo <alejandrobailo94@gmail.com>
Co-authored-by: César Arroba <cesar@prowler.com>
Co-authored-by: Alejandro Bailo <59607668+alejandrobailo@users.noreply.github.com>
2025-09-30 09:59:51 +02:00

227 lines
6.1 KiB
TypeScript

"use server";
import { revalidatePath } from "next/cache";
import { redirect } from "next/navigation";
import { apiBaseUrl, getAuthHeaders } from "@/lib";
import { handleApiError, handleApiResponse } from "@/lib/server-actions-helper";
export const getRoles = async ({
page = 1,
query = "",
sort = "",
filters = {},
pageSize = 10,
}) => {
const headers = await getAuthHeaders({ contentType: false });
if (isNaN(Number(page)) || page < 1) redirect("/roles");
const url = new URL(`${apiBaseUrl}/roles`);
if (page) url.searchParams.append("page[number]", page.toString());
if (pageSize) url.searchParams.append("page[size]", pageSize.toString());
if (query) url.searchParams.append("filter[search]", query);
if (sort) url.searchParams.append("sort", sort);
// Handle multiple filters
Object.entries(filters).forEach(([key, value]) => {
if (key !== "filter[search]") {
url.searchParams.append(key, String(value));
}
});
try {
const response = await fetch(url.toString(), {
headers,
});
return handleApiResponse(response);
} catch (error) {
console.error("Error fetching roles:", error);
return undefined;
}
};
export const getRoleInfoById = async (roleId: string) => {
const headers = await getAuthHeaders({ contentType: false });
const url = new URL(`${apiBaseUrl}/roles/${roleId}`);
try {
const response = await fetch(url.toString(), {
method: "GET",
headers,
});
return handleApiResponse(response);
} catch (error) {
handleApiError(error);
}
};
export const getRolesByIds = async (roleIds: string[]) => {
if (!roleIds || roleIds.length === 0) {
return { data: [] };
}
const headers = await getAuthHeaders({ contentType: false });
const url = new URL(`${apiBaseUrl}/roles`);
// Add filter for role IDs
url.searchParams.append("filter[id__in]", roleIds.join(","));
// Request all results on a single page with reasonable size
url.searchParams.append("page[size]", "100");
try {
const response = await fetch(url.toString(), {
method: "GET",
headers,
});
return handleApiResponse(response);
} catch (error) {
console.error("Error fetching roles by IDs:", error);
return { data: [] };
}
};
export const addRole = async (formData: FormData) => {
const headers = await getAuthHeaders({ contentType: true });
const name = formData.get("name") as string;
const groups = formData.getAll("groups[]") as string[];
const payload: any = {
data: {
type: "roles",
attributes: {
name,
manage_users: formData.get("manage_users") === "true",
manage_providers: formData.get("manage_providers") === "true",
manage_scans: formData.get("manage_scans") === "true",
manage_account: formData.get("manage_account") === "true",
manage_integrations: formData.get("manage_integrations") === "true",
unlimited_visibility: formData.get("unlimited_visibility") === "true",
},
relationships: {},
},
};
// Conditionally include manage_billing for cloud environment
if (process.env.NEXT_PUBLIC_IS_CLOUD_ENV === "true") {
payload.data.attributes.manage_billing =
formData.get("manage_billing") === "true";
}
// Add provider groups relationships only if there are items
if (groups.length > 0) {
payload.data.relationships.provider_groups = {
data: groups.map((groupId: string) => ({
type: "provider-groups",
id: groupId,
})),
};
}
const body = JSON.stringify(payload);
try {
const url = new URL(`${apiBaseUrl}/roles`);
const response = await fetch(url.toString(), {
method: "POST",
headers,
body,
});
return handleApiResponse(response, "/roles", false);
} catch (error) {
handleApiError(error);
}
};
export const updateRole = async (formData: FormData, roleId: string) => {
const headers = await getAuthHeaders({ contentType: true });
const name = formData.get("name") as string;
const groups = formData.getAll("groups[]") as string[];
const payload: any = {
data: {
type: "roles",
id: roleId,
attributes: {
...(name && { name }), // Include name only if provided
manage_users: formData.get("manage_users") === "true",
manage_providers: formData.get("manage_providers") === "true",
manage_account: formData.get("manage_account") === "true",
manage_scans: formData.get("manage_scans") === "true",
manage_integrations: formData.get("manage_integrations") === "true",
unlimited_visibility: formData.get("unlimited_visibility") === "true",
},
relationships: {},
},
};
// Conditionally include manage_billing for cloud environments
if (process.env.NEXT_PUBLIC_IS_CLOUD_ENV === "true") {
payload.data.attributes.manage_billing =
formData.get("manage_billing") === "true";
}
// Add provider groups relationships only if there are items
if (groups.length > 0) {
payload.data.relationships.provider_groups = {
data: groups.map((groupId: string) => ({
type: "provider-groups",
id: groupId,
})),
};
}
const body = JSON.stringify(payload);
try {
const url = new URL(`${apiBaseUrl}/roles/${roleId}`);
const response = await fetch(url.toString(), {
method: "PATCH",
headers,
body,
});
return handleApiResponse(response, "/roles", false);
} catch (error) {
handleApiError(error);
}
};
export const deleteRole = async (roleId: string) => {
const headers = await getAuthHeaders({ contentType: false });
const url = new URL(`${apiBaseUrl}/roles/${roleId}`);
try {
const response = await fetch(url.toString(), {
method: "DELETE",
headers,
});
if (!response.ok) {
try {
const errorData = await response.json();
throw new Error(errorData?.message || "Failed to delete the role");
} catch {
throw new Error("Failed to delete the role");
}
}
let data = null;
if (response.status !== 204) {
data = await response.json();
}
revalidatePath("/roles");
return data || { success: true };
} catch (error) {
handleApiError(error);
}
};