Files
prowler/ui/actions/scans/scans.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

271 lines
6.6 KiB
TypeScript

"use server";
import { redirect } from "next/navigation";
import { apiBaseUrl, getAuthHeaders, getErrorMessage } from "@/lib";
import { handleApiError, handleApiResponse } from "@/lib/server-actions-helper";
export const getScans = async ({
page = 1,
query = "",
sort = "",
filters = {},
pageSize = 10,
fields = {},
include = "",
}) => {
const headers = await getAuthHeaders({ contentType: false });
if (isNaN(Number(page)) || page < 1) redirect("/scans");
const url = new URL(`${apiBaseUrl}/scans`);
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);
if (include) url.searchParams.append("include", include);
// Handle fields parameters
Object.entries(fields).forEach(([key, value]) => {
url.searchParams.append(`fields[${key}]`, String(value));
});
// Add dynamic filters (e.g., "filter[state]", "fields[scans]")
Object.entries(filters).forEach(([key, value]) => {
url.searchParams.append(key, String(value));
});
try {
const response = await fetch(url.toString(), { headers });
return handleApiResponse(response);
} catch (error) {
console.error("Error fetching scans:", error);
return undefined;
}
};
export const getScansByState = async () => {
const headers = await getAuthHeaders({ contentType: false });
const url = new URL(`${apiBaseUrl}/scans`);
// Request only the necessary fields to optimize the response
url.searchParams.append("fields[scans]", "state");
try {
const response = await fetch(url.toString(), {
headers,
});
return handleApiResponse(response);
} catch (error) {
console.error("Error fetching scans by state:", error);
return undefined;
}
};
export const getScan = async (scanId: string) => {
const headers = await getAuthHeaders({ contentType: false });
const url = new URL(`${apiBaseUrl}/scans/${scanId}`);
try {
const response = await fetch(url.toString(), {
headers,
});
return handleApiResponse(response);
} catch (error) {
return handleApiError(error);
}
};
export const scanOnDemand = async (formData: FormData) => {
const headers = await getAuthHeaders({ contentType: true });
const providerId = formData.get("providerId");
const scanName = formData.get("scanName") || undefined;
if (!providerId) {
return { error: "Provider ID is required" };
}
const url = new URL(`${apiBaseUrl}/scans`);
try {
const requestBody = {
data: {
type: "scans",
attributes: scanName ? { name: scanName } : {},
relationships: {
provider: {
data: {
type: "providers",
id: providerId,
},
},
},
},
};
const response = await fetch(url.toString(), {
method: "POST",
headers: headers,
body: JSON.stringify(requestBody),
});
return handleApiResponse(response, "/scans");
} catch (error) {
return handleApiError(error);
}
};
export const scheduleDaily = async (formData: FormData) => {
const headers = await getAuthHeaders({ contentType: true });
const providerId = formData.get("providerId");
const url = new URL(`${apiBaseUrl}/schedules/daily`);
try {
const response = await fetch(url.toString(), {
method: "POST",
headers,
body: JSON.stringify({
data: {
type: "daily-schedules",
attributes: {
provider_id: providerId,
},
},
}),
});
return handleApiResponse(response, "/scans");
} catch (error) {
return handleApiError(error);
}
};
export const updateScan = async (formData: FormData) => {
const headers = await getAuthHeaders({ contentType: true });
const scanId = formData.get("scanId");
const scanName = formData.get("scanName");
const url = new URL(`${apiBaseUrl}/scans/${scanId}`);
try {
const response = await fetch(url.toString(), {
method: "PATCH",
headers,
body: JSON.stringify({
data: {
type: "scans",
id: scanId,
attributes: {
name: scanName,
},
},
}),
});
return handleApiResponse(response, "/scans");
} catch (error) {
return handleApiError(error);
}
};
export const getExportsZip = async (scanId: string) => {
const headers = await getAuthHeaders({ contentType: false });
const url = new URL(`${apiBaseUrl}/scans/${scanId}/report`);
try {
const response = await fetch(url.toString(), {
headers,
});
if (response.status === 202) {
const json = await response.json();
const taskId = json?.data?.id;
const state = json?.data?.attributes?.state;
return {
pending: true,
state,
taskId,
};
}
if (!response.ok) {
const errorData = await response.json();
throw new Error(
errorData?.errors?.detail ||
"Unable to fetch scan report. Contact support if the issue continues.",
);
}
// Get the blob data as an array buffer
const arrayBuffer = await response.arrayBuffer();
// Convert to base64
const base64 = Buffer.from(arrayBuffer).toString("base64");
return {
success: true,
data: base64,
filename: `scan-${scanId}-report.zip`,
};
} catch (error) {
return {
error: getErrorMessage(error),
};
}
};
export const getComplianceCsv = async (
scanId: string,
complianceId: string,
) => {
const headers = await getAuthHeaders({ contentType: false });
const url = new URL(
`${apiBaseUrl}/scans/${scanId}/compliance/${complianceId}`,
);
try {
const response = await fetch(url.toString(), { headers });
if (response.status === 202) {
const json = await response.json();
const taskId = json?.data?.id;
const state = json?.data?.attributes?.state;
return {
pending: true,
state,
taskId,
};
}
if (!response.ok) {
const errorData = await response.json();
throw new Error(
errorData?.errors?.detail ||
"Unable to retrieve compliance report. Contact support if the issue continues.",
);
}
const arrayBuffer = await response.arrayBuffer();
const base64 = Buffer.from(arrayBuffer).toString("base64");
return {
success: true,
data: base64,
filename: `scan-${scanId}-compliance-${complianceId}.csv`,
};
} catch (error) {
return {
error: getErrorMessage(error),
};
}
};