import { swrFetcher } from "@/api/api" import { deleteCron, runCron } from "@/api/cron" import { ActionButtonGroup } from "@/components/action-button-group" import { CopyButton } from "@/components/copy-button" import { CronCard } from "@/components/cron" import { HeaderButtonGroup } from "@/components/header-button-group" import { Checkbox } from "@/components/ui/checkbox" import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow, } from "@/components/ui/table" import { IconButton } from "@/components/xui/icon-button" import { ModelCron } from "@/types" import { cronTypes } from "@/types" import { ColumnDef, flexRender, getCoreRowModel, useReactTable } from "@tanstack/react-table" import { useEffect, useMemo } from "react" import { useTranslation } from "react-i18next" import { toast } from "sonner" import useSWR from "swr" export default function CronPage() { const { t } = useTranslation() const { data, mutate, error, isLoading } = useSWR("/api/v1/cron", swrFetcher) useEffect(() => { if (error) toast(t("Error"), { description: t("Results.ErrorFetchingResource", { error: error.message, }), }) // eslint-disable-next-line react-hooks/exhaustive-deps }, [error]) const columns: ColumnDef[] = [ { id: "select", header: ({ table }) => ( table.toggleAllPageRowsSelected(!!value)} aria-label="Select all" /> ), cell: ({ row }) => ( row.toggleSelected(!!value)} aria-label="Select row" /> ), enableSorting: false, enableHiding: false, }, { header: "ID", accessorKey: "id", accessorFn: (row) => row.id, }, { header: t("Name"), accessorKey: "name", cell: ({ row }) => { const s = row.original return
{s.name}
}, }, { header: t("Type"), accessorKey: "taskType", accessorFn: (row) => cronTypes[row.task_type] || "", }, { header: t("CronExpression"), accessorKey: "scheduler", accessorFn: (row) => row.scheduler, }, { header: t("Command"), accessorKey: "command", cell: ({ row }) => { const s = row.original return }, }, { header: t("NotifierGroup"), accessorKey: "ngroup", accessorFn: (row) => row.notification_group_id, }, { header: t("SendSuccessNotification"), accessorKey: "pushSuccessful", accessorFn: (row) => row.push_successful ?? false, }, { header: t("Coverage"), accessorKey: "cover", accessorFn: (row) => row.cover, cell: ({ row }) => { const s = row.original return (
{(() => { switch (s.cover) { case 0: { return {t("IgnoreAll")} } case 1: { return {t("CoverAll")} } case 2: { return {t("OnAlert")} } } })()}
) }, }, { header: t("SpecificServers"), accessorKey: "servers", accessorFn: (row) => row.servers, cell: ({ row }) => { const s = row.original return (
{(s.servers || []).join(",")}
) }, }, { header: t("LastExecution"), accessorKey: "lastExecution", accessorFn: (row) => row.last_executed_at, cell: ({ row }) => { const s = row.original return (
{s.last_executed_at}
) }, }, { header: t("Result"), accessorKey: "lastResult", accessorFn: (row) => row.last_result ?? false, }, { id: "actions", header: t("Actions"), cell: ({ row }) => { const s = row.original return ( <> { try { await runCron(s.id) } catch (e) { console.error(e) toast(t("Error"), { description: t("Results.UnExpectedError"), }) await mutate() return } toast(t("Success"), { description: t("Results.TaskTriggeredSuccessfully"), }) await mutate() }} /> ) }, }, ] const dataCache = useMemo(() => { return data ?? [] }, [data]) const table = useReactTable({ data: dataCache, columns, getCoreRowModel: getCoreRowModel(), }) const selectedRows = table.getSelectedRowModel().rows return (

{t("Task")}

r.original.id), mutate: mutate, }} >
{table.getHeaderGroups().map((headerGroup) => ( {headerGroup.headers.map((header) => { return ( {header.isPlaceholder ? null : flexRender( header.column.columnDef.header, header.getContext(), )} ) })} ))} {isLoading ? ( {t("Loading")}... ) : table.getRowModel().rows?.length ? ( table.getRowModel().rows.map((row) => ( {row.getVisibleCells().map((cell) => ( {flexRender(cell.column.columnDef.cell, cell.getContext())} ))} )) ) : ( {t("NoResults")} )}
) }