-
Notifications
You must be signed in to change notification settings - Fork 790
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: Added dashboard for Alert Quality (#1977)
Signed-off-by: Rajesh Jonnalagadda <[email protected]> Co-authored-by: Matvey Kukuy <[email protected]> Co-authored-by: Tal <[email protected]> Co-authored-by: Rajesh Jonnalagadda <[email protected]> Co-authored-by: Rajesh Jonnalagadda <[email protected]>
- Loading branch information
1 parent
e1e7eff
commit 5a93f10
Showing
16 changed files
with
975 additions
and
145 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,361 @@ | ||
"use client"; // Add this line at the top to make this a Client Component | ||
|
||
import React, { | ||
useState, | ||
useEffect, | ||
Dispatch, | ||
SetStateAction, | ||
useMemo, | ||
} from "react"; | ||
import { GenericTable } from "@/components/table/GenericTable"; | ||
import { useAlertQualityMetrics } from "utils/hooks/useAlertQuality"; | ||
import { useProviders } from "utils/hooks/useProviders"; | ||
import { Provider, ProvidersResponse } from "app/providers/providers"; | ||
import { TabGroup, TabList, Tab } from "@tremor/react"; | ||
import { GenericFilters } from "@/components/filters/GenericFilters"; | ||
import { useSearchParams } from "next/navigation"; | ||
import { AlertKnownKeys } from "./models"; | ||
import { createColumnHelper, DisplayColumnDef } from "@tanstack/react-table"; | ||
|
||
const tabs = [ | ||
{ name: "All", value: "all" }, | ||
{ name: "Installed", value: "installed" }, | ||
{ name: "Linked", value: "linked" }, | ||
]; | ||
|
||
const ALERT_QUALITY_FILTERS = [ | ||
{ | ||
type: "date", | ||
key: "time_stamp", | ||
value: "", | ||
name: "Last received", | ||
}, | ||
]; | ||
|
||
export const FilterTabs = ({ | ||
tabs, | ||
setTab, | ||
tab, | ||
}: { | ||
tabs: { name: string; value: string }[]; | ||
setTab: Dispatch<SetStateAction<number>>; | ||
tab: number; | ||
}) => { | ||
return ( | ||
<div className="max-w-lg space-y-12 pt-6"> | ||
<TabGroup | ||
index={tab} | ||
onIndexChange={(index: number) => { | ||
setTab(index); | ||
}} | ||
> | ||
<TabList variant="solid" color="black" className="bg-gray-300"> | ||
{tabs.map((tabItem) => ( | ||
<Tab key={tabItem.value}>{tabItem.name}</Tab> | ||
))} | ||
</TabList> | ||
</TabGroup> | ||
</div> | ||
); | ||
}; | ||
|
||
interface AlertMetricQuality { | ||
alertsReceived: number; | ||
alertsCorrelatedToIncidentsPercentage: number; | ||
alertsWithSeverityPercentage: number; | ||
[key: string]: number; | ||
} | ||
|
||
type FinalAlertQuality = Provider & | ||
AlertMetricQuality & { provider_display_name: string }; | ||
interface Pagination { | ||
limit: number; | ||
offset: number; | ||
} | ||
|
||
const QualityTable = ({ | ||
providersMeta, | ||
alertsQualityMetrics, | ||
isDashBoard, | ||
setFields, | ||
fieldsValue, | ||
}: { | ||
providersMeta: ProvidersResponse | undefined; | ||
alertsQualityMetrics: Record<string, Record<string, any>> | undefined; | ||
isDashBoard?: boolean; | ||
setFields: (fields: string | string[] | Record<string, string>) => void; | ||
fieldsValue: string | string[] | Record<string, string>; | ||
}) => { | ||
const [pagination, setPagination] = useState<Pagination>({ | ||
limit: 10, | ||
offset: 0, | ||
}); | ||
const customFieldFilter = { | ||
type: "select", | ||
key: "fields", | ||
value: isDashBoard ? fieldsValue : "", | ||
name: "Field", | ||
options: AlertKnownKeys.map((key) => ({ value: key, label: key })), | ||
// only_one: true, | ||
searchParamsNotNeed: isDashBoard, | ||
can_select: 3, | ||
setFilter: setFields, | ||
}; | ||
const searchParams = useSearchParams(); | ||
const entries = searchParams ? Array.from(searchParams.entries()) : []; | ||
const columnHelper = createColumnHelper<FinalAlertQuality>(); | ||
|
||
const params = entries.reduce( | ||
(acc, [key, value]) => { | ||
if (key in acc) { | ||
if (Array.isArray(acc[key])) { | ||
acc[key] = [...acc[key], value]; | ||
return acc; | ||
} else { | ||
acc[key] = [acc[key] as string, value]; | ||
} | ||
return acc; | ||
} | ||
acc[key] = value; | ||
return acc; | ||
}, | ||
{} as Record<string, string | string[]> | ||
); | ||
function toArray(value: string | string[]) { | ||
if (!value) { | ||
return []; | ||
} | ||
|
||
if (!Array.isArray(value) && value) { | ||
return [value]; | ||
} | ||
|
||
return value; | ||
} | ||
const fields = toArray( | ||
params?.["fields"] || (fieldsValue as string | string[]) || [] | ||
) as string[]; | ||
const [tab, setTab] = useState(0); | ||
|
||
const handlePaginationChange = (newLimit: number, newOffset: number) => { | ||
setPagination({ limit: newLimit, offset: newOffset }); | ||
}; | ||
|
||
useEffect(() => { | ||
handlePaginationChange(10, 0); | ||
}, [tab, searchParams?.toString()]); | ||
|
||
// Construct columns based on the fields selected | ||
const columns = useMemo(() => { | ||
const baseColumns = [ | ||
columnHelper.display({ | ||
id: "provider_display_name", | ||
header: "Provider Name", | ||
cell: ({ row }) => { | ||
const displayName = row.original.provider_display_name; | ||
return ( | ||
<div className="flex flex-col gap-2"> | ||
<div>{displayName}</div> | ||
<div>id: {row.original.id}</div> | ||
<div>type: {row.original.type}</div> | ||
</div> | ||
); | ||
}, | ||
}), | ||
columnHelper.accessor("alertsReceived", { | ||
id: "alertsReceived", | ||
header: "Alerts Received", | ||
}), | ||
columnHelper.display({ | ||
id: "alertsCorrelatedToIncidentsPercentage", | ||
header: "% of Alerts Correlated to Incidents", | ||
cell: ({ row }) => { | ||
return `${row.original.alertsCorrelatedToIncidentsPercentage.toFixed(2)}%`; | ||
}, | ||
}), | ||
] as DisplayColumnDef<FinalAlertQuality>[]; | ||
|
||
// Add dynamic columns based on the fields | ||
const dynamicColumns = fields.map((field: string) => | ||
columnHelper.accessor( | ||
`alertsWith${field.charAt(0).toUpperCase() + field.slice(1)}Percentage`, | ||
{ | ||
id: `alertsWith${ | ||
field.charAt(0).toUpperCase() + field.slice(1) | ||
}Percentage`, | ||
header: `% of Alerts Having ${ | ||
field.charAt(0).toUpperCase() + field.slice(1) | ||
}`, | ||
cell: (info: any) => `${info.getValue().toFixed(2)}%`, | ||
} | ||
) | ||
) as DisplayColumnDef<FinalAlertQuality>[]; | ||
|
||
return [ | ||
...baseColumns, | ||
...dynamicColumns, | ||
] as DisplayColumnDef<FinalAlertQuality>[]; | ||
}, [fields]); | ||
|
||
// Process data and include dynamic fields | ||
const finalData = useMemo(() => { | ||
let providers: Provider[] | null = null; | ||
|
||
if (!providersMeta || !alertsQualityMetrics) { | ||
return null; | ||
} | ||
|
||
switch (tab) { | ||
case 0: | ||
providers = [ | ||
...providersMeta?.installed_providers, | ||
...providersMeta?.linked_providers, | ||
]; | ||
break; | ||
case 1: | ||
providers = providersMeta?.installed_providers || []; | ||
break; | ||
case 2: | ||
providers = providersMeta?.linked_providers || []; | ||
break; | ||
default: | ||
providers = [ | ||
...providersMeta?.installed_providers, | ||
...providersMeta?.linked_providers, | ||
]; | ||
break; | ||
} | ||
|
||
if (!providers) { | ||
return null; | ||
} | ||
|
||
function getProviderDisplayName(provider: Provider) { | ||
return ( | ||
(provider?.details?.name | ||
? `${provider.details.name} (${provider.display_name})` | ||
: provider.display_name) || provider.type | ||
); | ||
} | ||
|
||
const innerData: FinalAlertQuality[] = providers.map((provider) => { | ||
const providerId = provider.id; | ||
const providerType = provider.type; | ||
const key = `${providerId}_${providerType}`; | ||
const alertQuality = alertsQualityMetrics[key]; | ||
const totalAlertsReceived = alertQuality?.total_alerts ?? 0; | ||
const correlated_alerts = alertQuality?.correlated_alerts ?? 0; | ||
const correltedPert = | ||
totalAlertsReceived && correlated_alerts | ||
? (correlated_alerts / totalAlertsReceived) * 100 | ||
: 0; | ||
const severityPert = totalAlertsReceived | ||
? ((alertQuality?.severity_count ?? 0) / totalAlertsReceived) * 100 | ||
: 0; | ||
|
||
// Calculate percentages for dynamic fields | ||
const dynamicFieldPercentages = fields.reduce( | ||
(acc, field: string) => { | ||
acc[ | ||
`alertsWith${ | ||
field.charAt(0).toUpperCase() + field.slice(1) | ||
}Percentage` | ||
] = totalAlertsReceived | ||
? ((alertQuality?.[`${field}_count`] ?? 0) / totalAlertsReceived) * | ||
100 | ||
: 0; | ||
return acc; | ||
}, | ||
{} as Record<string, number> | ||
); | ||
|
||
return { | ||
...provider, | ||
alertsReceived: totalAlertsReceived, | ||
alertsCorrelatedToIncidentsPercentage: correltedPert, | ||
alertsWithSeverityPercentage: severityPert, | ||
...dynamicFieldPercentages, // Add dynamic field percentages here | ||
provider_display_name: getProviderDisplayName(provider), | ||
} as FinalAlertQuality; | ||
}); | ||
|
||
return innerData; | ||
}, [tab, providersMeta, alertsQualityMetrics, fields]); | ||
|
||
return ( | ||
<div | ||
className={`flex flex-col gap-2 p-2 px-4 ${isDashBoard ? "h-[90%]" : ""}`} | ||
> | ||
<div> | ||
{!isDashBoard && ( | ||
<h1 className="text-lg font-semibold text-gray-800 dark:text-gray-100 mb-4"> | ||
Alert Quality Dashboard | ||
</h1> | ||
)} | ||
<div className="flex items-end mb-4"> | ||
<div className="flex-1"> | ||
<FilterTabs tabs={tabs} setTab={setTab} tab={tab} /> | ||
</div> | ||
<GenericFilters | ||
filters={ | ||
isDashBoard | ||
? [customFieldFilter] | ||
: [...ALERT_QUALITY_FILTERS, customFieldFilter] | ||
} | ||
/> | ||
</div> | ||
</div> | ||
{finalData && ( | ||
<GenericTable<FinalAlertQuality> | ||
data={finalData} | ||
columns={columns} | ||
rowCount={finalData?.length} | ||
offset={pagination.offset} | ||
limit={pagination.limit} | ||
onPaginationChange={handlePaginationChange} | ||
dataFetchedAtOneGO={true} | ||
onRowClick={(row) => { | ||
console.log("Row clicked:", row); | ||
}} | ||
/> | ||
)} | ||
</div> | ||
); | ||
}; | ||
|
||
const AlertQuality = ({ | ||
isDashBoard, | ||
filters, | ||
setFilters, | ||
}: { | ||
isDashBoard?: boolean; | ||
filters: { | ||
[x: string]: string | string[]; | ||
}; | ||
setFilters: any; | ||
}) => { | ||
const fieldsValue = filters?.fields || ""; | ||
const { data: providersMeta } = useProviders(); | ||
const { data: alertsQualityMetrics, error } = useAlertQualityMetrics( | ||
isDashBoard ? (fieldsValue as string | string[]) : "" | ||
); | ||
|
||
return ( | ||
<QualityTable | ||
providersMeta={providersMeta} | ||
alertsQualityMetrics={alertsQualityMetrics} | ||
isDashBoard={isDashBoard} | ||
setFields={(field) => { | ||
setFilters((filters: any) => { | ||
return { | ||
...filters, | ||
fields: field, | ||
}; | ||
}); | ||
}} | ||
fieldsValue={fieldsValue} | ||
/> | ||
); | ||
}; | ||
|
||
export default AlertQuality; |
Oops, something went wrong.