Files
prowler/ui/actions/scans/scans.ts
2025-11-27 12:37:20 +01:00

337 lines
8.4 KiB
TypeScript

"use server";
import { redirect } from "next/navigation";
import { apiBaseUrl, getAuthHeaders, getErrorMessage } from "@/lib";
import {
COMPLIANCE_REPORT_DISPLAY_NAMES,
type ComplianceReportType,
} from "@/lib/compliance/compliance-report-types";
import { addScanOperation } from "@/lib/sentry-breadcrumbs";
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" };
}
addScanOperation("create", undefined, {
provider_id: String(providerId),
scan_name: scanName ? String(scanName) : undefined,
});
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),
});
const result = await handleApiResponse(response, "/scans");
if (result?.data?.id) {
addScanOperation("start", result.data.id);
}
return result;
} catch (error) {
addScanOperation("create");
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),
};
}
};
/**
* Generic function to get a compliance PDF report (ThreatScore, ENS, etc.)
* @param scanId - The scan ID
* @param reportType - Type of report (from COMPLIANCE_REPORT_TYPES)
* @returns Promise with the PDF data or error
*/
export const getCompliancePdfReport = async (
scanId: string,
reportType: ComplianceReportType,
) => {
const headers = await getAuthHeaders({ contentType: false });
const url = new URL(`${apiBaseUrl}/scans/${scanId}/${reportType}`);
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();
const reportName = COMPLIANCE_REPORT_DISPLAY_NAMES[reportType];
throw new Error(
errorData?.errors?.detail ||
`Unable to retrieve ${reportName} PDF 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}-${reportType}.pdf`,
};
} catch (error) {
return {
error: getErrorMessage(error),
};
}
};