"use client"; import { Button } from "@app/components/ui/button"; import { toast } from "@app/hooks/useToast"; import { useState, useRef, useEffect, useTransition } from "react"; import { createApiClient } from "@app/lib/api"; import { useEnvContext } from "@app/hooks/useEnvContext"; import { useParams, useRouter, useSearchParams } from "next/navigation"; import { useTranslations } from "next-intl"; import { LogDataTable } from "@app/components/LogDataTable"; import { ColumnDef } from "@tanstack/react-table"; import { DateTimeValue } from "@app/components/DateTimePicker"; import { ArrowUpRight, Key, User } from "lucide-react"; import Link from "next/link"; import { ColumnFilter } from "@app/components/ColumnFilter"; import SettingsSectionTitle from "@app/components/SettingsSectionTitle"; import { build } from "@server/build"; import { getSevenDaysAgo } from "@app/lib/getSevenDaysAgo"; import axios from "axios"; import { useStoredPageSize } from "@app/hooks/useStoredPageSize"; import { PaidFeaturesAlert } from "@app/components/PaidFeaturesAlert"; import { usePaidStatus } from "@app/hooks/usePaidStatus"; import { tierMatrix } from "@server/lib/billing/tierMatrix"; export default function GeneralPage() { const router = useRouter(); const searchParams = useSearchParams(); const api = createApiClient(useEnvContext()); const t = useTranslations(); const { orgId } = useParams(); const { isPaidUser } = usePaidStatus(); const [rows, setRows] = useState([]); const [isRefreshing, setIsRefreshing] = useState(false); const [isExporting, startTransition] = useTransition(); const [filterAttributes, setFilterAttributes] = useState<{ actors: string[]; resources: { id: number; name: string | null; }[]; locations: string[]; }>({ actors: [], resources: [], locations: [] }); // Filter states - unified object for all filters const [filters, setFilters] = useState<{ action?: string; type?: string; resourceId?: string; location?: string; actor?: string; }>({ action: searchParams.get("action") || undefined, type: searchParams.get("type") || undefined, resourceId: searchParams.get("resourceId") || undefined, location: searchParams.get("location") || undefined, actor: searchParams.get("actor") || undefined }); // Pagination state const [totalCount, setTotalCount] = useState(0); const [currentPage, setCurrentPage] = useState(0); const [isLoading, setIsLoading] = useState(false); // Initialize page size from storage or default const [pageSize, setPageSize] = useStoredPageSize("access-audit-logs", 20); // Set default date range to last 24 hours const getDefaultDateRange = () => { // if the time is in the url params, use that instead const startParam = searchParams.get("start"); const endParam = searchParams.get("end"); if (startParam && endParam) { return { startDate: { date: new Date(startParam) }, endDate: { date: new Date(endParam) } }; } const now = new Date(); const lastWeek = getSevenDaysAgo(); return { startDate: { date: lastWeek }, endDate: { date: now } }; }; const [dateRange, setDateRange] = useState<{ startDate: DateTimeValue; endDate: DateTimeValue; }>(getDefaultDateRange()); // Trigger search with default values on component mount useEffect(() => { const defaultRange = getDefaultDateRange(); queryDateTime( defaultRange.startDate, defaultRange.endDate, 0, pageSize ); }, [orgId]); // Re-run if orgId changes const handleDateRangeChange = ( startDate: DateTimeValue, endDate: DateTimeValue ) => { setDateRange({ startDate, endDate }); setCurrentPage(0); // Reset to first page when filtering // put the search params in the url for the time updateUrlParamsForAllFilters({ start: startDate.date?.toISOString() || "", end: endDate.date?.toISOString() || "" }); queryDateTime(startDate, endDate, 0, pageSize); }; // Handle page changes const handlePageChange = (newPage: number) => { setCurrentPage(newPage); queryDateTime( dateRange.startDate, dateRange.endDate, newPage, pageSize ); }; // Handle page size changes const handlePageSizeChange = (newPageSize: number) => { setPageSize(newPageSize); setCurrentPage(0); // Reset to first page when changing page size queryDateTime(dateRange.startDate, dateRange.endDate, 0, newPageSize); }; // Handle filter changes generically const handleFilterChange = ( filterType: keyof typeof filters, value: string | undefined ) => { // Create new filters object with updated value const newFilters = { ...filters, [filterType]: value }; setFilters(newFilters); setCurrentPage(0); // Reset to first page when filtering // Update URL params updateUrlParamsForAllFilters(newFilters); // Trigger new query with updated filters (pass directly to avoid async state issues) queryDateTime( dateRange.startDate, dateRange.endDate, 0, pageSize, newFilters ); }; const updateUrlParamsForAllFilters = ( newFilters: | typeof filters | { start: string; end: string; } ) => { const params = new URLSearchParams(searchParams); Object.entries(newFilters).forEach(([key, value]) => { if (value) { params.set(key, value); } else { params.delete(key); } }); router.replace(`?${params.toString()}`, { scroll: false }); }; const queryDateTime = async ( startDate: DateTimeValue, endDate: DateTimeValue, page: number = currentPage, size: number = pageSize, filtersParam?: { action?: string; type?: string; resourceId?: string; location?: string; actor?: string; } ) => { console.log("Date range changed:", { startDate, endDate, page, size }); if (!isPaidUser(tierMatrix.accessLogs) || build === "oss") { console.log( "Access denied: subscription inactive or license locked" ); return; } setIsLoading(true); try { // Use the provided filters or fall back to current state const activeFilters = filtersParam || filters; // Convert the date/time values to API parameters const params: any = { limit: size, offset: page * size, ...activeFilters }; if (startDate?.date) { const startDateTime = new Date(startDate.date); if (startDate.time) { const [hours, minutes, seconds] = startDate.time .split(":") .map(Number); startDateTime.setHours(hours, minutes, seconds || 0); } params.timeStart = startDateTime.toISOString(); } if (endDate?.date) { const endDateTime = new Date(endDate.date); if (endDate.time) { const [hours, minutes, seconds] = endDate.time .split(":") .map(Number); endDateTime.setHours(hours, minutes, seconds || 0); } else { // If no time is specified, set to NOW const now = new Date(); endDateTime.setHours( now.getHours(), now.getMinutes(), now.getSeconds(), now.getMilliseconds() ); } params.timeEnd = endDateTime.toISOString(); } const res = await api.get(`/org/${orgId}/logs/access`, { params }); if (res.status === 200) { setRows(res.data.data.log || []); setTotalCount(res.data.data.pagination?.total || 0); setFilterAttributes(res.data.data.filterAttributes); console.log("Fetched logs:", res.data); } } catch (error) { toast({ title: t("error"), description: t("Failed to filter logs"), variant: "destructive" }); } finally { setIsLoading(false); } }; const refreshData = async () => { console.log("Data refreshed"); setIsRefreshing(true); try { // Refresh data with current date range and pagination await queryDateTime( dateRange.startDate, dateRange.endDate, currentPage, pageSize ); } catch (error) { toast({ title: t("error"), description: t("refreshError"), variant: "destructive" }); } finally { setIsRefreshing(false); } }; const exportData = async () => { try { // Prepare query params for export const params: any = { timeStart: dateRange.startDate?.date ? new Date(dateRange.startDate.date).toISOString() : undefined, timeEnd: dateRange.endDate?.date ? new Date(dateRange.endDate.date).toISOString() : undefined, ...filters }; const response = await api.get(`/org/${orgId}/logs/access/export`, { responseType: "blob", params }); // Create a URL for the blob and trigger a download const url = window.URL.createObjectURL(new Blob([response.data])); const link = document.createElement("a"); link.href = url; const epoch = Math.floor(Date.now() / 1000); link.setAttribute( "download", `access-audit-logs-${orgId}-${epoch}.csv` ); document.body.appendChild(link); link.click(); link.parentNode?.removeChild(link); } catch (error) { let apiErrorMessage: string | null = null; if (axios.isAxiosError(error) && error.response) { const data = error.response.data; if (data instanceof Blob && data.type === "application/json") { // Parse the Blob as JSON const text = await data.text(); const errorData = JSON.parse(text); apiErrorMessage = errorData.message; } } toast({ title: t("error"), description: apiErrorMessage ?? t("exportError"), variant: "destructive" }); } }; const columns: ColumnDef[] = [ { accessorKey: "timestamp", header: ({ column }) => { return t("timestamp"); }, cell: ({ row }) => { return (
{new Date( row.original.timestamp * 1000 ).toLocaleString()}
); } }, { accessorKey: "action", header: ({ column }) => { return (
{t("action")} handleFilterChange("action", value) } // placeholder="" searchPlaceholder="Search..." emptyMessage="None found" />
); }, cell: ({ row }) => { return ( {row.original.action ? <>Allowed : <>Denied} ); } }, { accessorKey: "ip", header: ({ column }) => { return t("ip"); } }, { accessorKey: "location", header: ({ column }) => { return (
{t("location")} ({ value: location, label: location }) )} selectedValue={filters.location} onValueChange={(value) => handleFilterChange("location", value) } // placeholder="" searchPlaceholder="Search..." emptyMessage="None found" />
); }, cell: ({ row }) => { return ( {row.original.location ? ( {row.original.location} ) : ( - )} ); } }, { accessorKey: "resourceName", header: ({ column }) => { return (
{t("resource")} ({ value: res.id.toString(), label: res.name || "Unnamed Resource" }))} selectedValue={filters.resourceId} onValueChange={(value) => handleFilterChange("resourceId", value) } // placeholder="" searchPlaceholder="Search..." emptyMessage="None found" />
); }, cell: ({ row }) => { return ( ); } }, { accessorKey: "type", header: ({ column }) => { return (
{t("type")} handleFilterChange("type", value) } // placeholder="" searchPlaceholder="Search..." emptyMessage="None found" />
); }, cell: ({ row }) => { // should be capitalized first letter return ( {row.original.type.charAt(0).toUpperCase() + row.original.type.slice(1) || "-"} ); } }, { accessorKey: "actor", header: ({ column }) => { return (
{t("actor")} ({ value: actor, label: actor }))} selectedValue={filters.actor} onValueChange={(value) => handleFilterChange("actor", value) } // placeholder="" searchPlaceholder="Search..." emptyMessage="None found" />
); }, cell: ({ row }) => { return ( {row.original.actor ? ( <> {row.original.actorType == "user" ? ( ) : ( )} {row.original.actor} ) : ( <>- )} ); } }, { accessorKey: "actorId", header: ({ column }) => { return t("actorId"); }, cell: ({ row }) => { return ( {row.original.actorId || "-"} ); } } ]; const renderExpandedRow = (row: any) => { return (
{row.userAgent != "node" && (
User Agent:

{row.userAgent || "N/A"}

)}
Metadata:
                            {row.metadata
                                ? JSON.stringify(
                                      JSON.parse(row.metadata),
                                      null,
                                      2
                                  )
                                : "N/A"}
                        
); }; return ( <> startTransition(exportData)} isExporting={isExporting} // isExportDisabled={ // not disabling this because the user should be able to click the button and get the feedback about needing to upgrade the plan // !isPaidUser(tierMatrix.accessLogs) || build === "oss" // } onDateRangeChange={handleDateRangeChange} dateRange={{ start: dateRange.startDate, end: dateRange.endDate }} defaultSort={{ id: "timestamp", desc: true }} // Server-side pagination props totalCount={totalCount} currentPage={currentPage} pageSize={pageSize} onPageChange={handlePageChange} onPageSizeChange={handlePageSizeChange} isLoading={isLoading} // Row expansion props expandable={true} renderExpandedRow={renderExpandedRow} disabled={!isPaidUser(tierMatrix.accessLogs) || build === "oss"} /> ); }