From e0994b088a527c073d57993730496bf30361a066 Mon Sep 17 00:00:00 2001 From: David Leek Date: Fri, 22 Mar 2024 11:54:33 +0100 Subject: [PATCH] feat: traffic visibility UI and store (#6659) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Provides store method for retrieving traffic usage data based on period parameter, and UI + ui hook with the new chart for displaying traffic usage data spread out over selectable month. ![Skjermbilde 2024-03-21 kl 12 40 38](https://github.com/Unleash/unleash/assets/707867/539c6c98-b6f6-488a-97fb-baf4fccec687) In this PR we copied and adapted a plugin written by DX for highlighting a column in the chart: ![image](https://github.com/Unleash/unleash/assets/707867/70532b22-44ed-44c0-a9b4-75f65ed6a63d) There are some minor improvements planned which will come in a separate PR, reversing the order in legend and tooltip so the colors go from light to dark, and adding a month -sum below the legend ## Discussion points - Should any of this be extracted as a separate reusable component? --------- Co-authored-by: Nuno Góis --- .../src/component/admin/network/Network.tsx | 11 + .../NetworkTrafficUsage.tsx | 392 ++++++++++++++++++ .../useInstanceTrafficMetrics.ts | 40 ++ frontend/src/interfaces/uiConfig.ts | 1 + frontend/src/openapi/models/index.ts | 4 + .../models/trafficUsageApiDataSchema.ts | 16 + .../trafficUsageApiDataSchemaDaysItem.ts | 13 + ...geApiDataSchemaDaysItemTrafficTypesItem.ts | 12 + .../models/trafficUsageDataSegmentedSchema.ts | 16 + .../fake-traffic-data-usage-store.ts | 3 + .../traffic-data-usage-store-type.ts | 3 +- .../traffic-data-usage-store.test.ts | 67 ++- .../traffic-data-usage-store.ts | 12 +- 13 files changed, 576 insertions(+), 14 deletions(-) create mode 100644 frontend/src/component/admin/network/NetworkTrafficUsage/NetworkTrafficUsage.tsx create mode 100644 frontend/src/hooks/api/getters/useInstanceTrafficMetrics/useInstanceTrafficMetrics.ts create mode 100644 frontend/src/openapi/models/trafficUsageApiDataSchema.ts create mode 100644 frontend/src/openapi/models/trafficUsageApiDataSchemaDaysItem.ts create mode 100644 frontend/src/openapi/models/trafficUsageApiDataSchemaDaysItemTrafficTypesItem.ts create mode 100644 frontend/src/openapi/models/trafficUsageDataSegmentedSchema.ts diff --git a/frontend/src/component/admin/network/Network.tsx b/frontend/src/component/admin/network/Network.tsx index bc91d48426..6a7ae1ccb0 100644 --- a/frontend/src/component/admin/network/Network.tsx +++ b/frontend/src/component/admin/network/Network.tsx @@ -7,6 +7,9 @@ import { PageContent } from 'component/common/PageContent/PageContent'; const NetworkOverview = lazy(() => import('./NetworkOverview/NetworkOverview')); const NetworkTraffic = lazy(() => import('./NetworkTraffic/NetworkTraffic')); +const NetworkTrafficUsage = lazy( + () => import('./NetworkTrafficUsage/NetworkTrafficUsage'), +); const tabs = [ { @@ -17,6 +20,10 @@ const tabs = [ label: 'Traffic', path: '/admin/network/traffic', }, + { + label: 'Data Usage', + path: '/admin/network/data-usage', + }, ]; export const Network = () => { @@ -52,6 +59,10 @@ export const Network = () => { } /> } /> + } + /> diff --git a/frontend/src/component/admin/network/NetworkTrafficUsage/NetworkTrafficUsage.tsx b/frontend/src/component/admin/network/NetworkTrafficUsage/NetworkTrafficUsage.tsx new file mode 100644 index 0000000000..b7172e819b --- /dev/null +++ b/frontend/src/component/admin/network/NetworkTrafficUsage/NetworkTrafficUsage.tsx @@ -0,0 +1,392 @@ +import { useMemo, type VFC, useState, useEffect } from 'react'; +import useTheme from '@mui/material/styles/useTheme'; +import styled from '@mui/material/styles/styled'; +import { usePageTitle } from 'hooks/usePageTitle'; +import Select from 'component/common/select'; +import Box from '@mui/system/Box'; +import Alert from '@mui/material/Alert'; +import { ConditionallyRender } from 'component/common/ConditionallyRender/ConditionallyRender'; +import useUiConfig from 'hooks/api/getters/useUiConfig/useUiConfig'; +import { + Chart as ChartJS, + type ChartOptions, + CategoryScale, + LinearScale, + BarElement, + type ChartDataset, + Title, + Tooltip, + Legend, + type Chart, + type Tick, +} from 'chart.js'; + +import { Bar } from 'react-chartjs-2'; +import { + type IInstanceTrafficMetricsResponse, + useInstanceTrafficMetrics, +} from 'hooks/api/getters/useInstanceTrafficMetrics/useInstanceTrafficMetrics'; +import type { Theme } from '@mui/material/styles/createTheme'; +import Grid from '@mui/material/Grid'; +import { useUiFlag } from 'hooks/useUiFlag'; + +type ChartDatasetType = ChartDataset<'bar'>; + +type SelectablePeriod = { + key: string; + dayCount: number; + label: string; + year: number; + month: number; +}; + +type EndpointInfo = { + label: string; + color: string; + order: number; +}; + +const StyledHeader = styled('h3')(({ theme }) => ({ + display: 'flex', + gap: theme.spacing(1), + alignItems: 'center', + fontSize: theme.fontSizes.bodySize, + margin: 0, + marginTop: theme.spacing(1), + fontWeight: theme.fontWeight.bold, +})); + +const padMonth = (month: number): string => + month < 10 ? `0${month}` : `${month}`; + +const toSelectablePeriod = (date: Date, label?: string): SelectablePeriod => { + const year = date.getFullYear(); + const month = date.getMonth(); + const period = `${year}-${padMonth(month + 1)}`; + const dayCount = new Date(year, month + 1, 0).getDate(); + return { + key: period, + year, + month, + dayCount, + label: + label || + date.toLocaleString('en-US', { month: 'long', year: 'numeric' }), + }; +}; + +const getSelectablePeriods = (): SelectablePeriod[] => { + const current = new Date(Date.now()); + const selectablePeriods = [toSelectablePeriod(current, 'Current month')]; + for ( + let subtractMonthCount = 1; + subtractMonthCount < 13; + subtractMonthCount++ + ) { + // JavaScript wraps around the year, so we don't need to handle that. + const date = new Date( + current.getFullYear(), + current.getMonth() - subtractMonthCount, + 1, + ); + selectablePeriods.push(toSelectablePeriod(date)); + } + return selectablePeriods; +}; + +const toPeriodsRecord = ( + periods: SelectablePeriod[], +): Record => { + return periods.reduce( + (acc, period) => { + acc[period.key] = period; + return acc; + }, + {} as Record, + ); +}; + +const getDayLabels = (dayCount: number): number[] => { + return [...Array(dayCount).keys()].map((i) => i + 1); +}; + +const toChartData = ( + days: number[], + traffic: IInstanceTrafficMetricsResponse, + endpointsInfo: Record, +): ChartDatasetType[] => { + if (!traffic || !traffic.usage || !traffic.usage.apiData) { + return []; + } + + const data = traffic.usage.apiData + .filter((item) => !!endpointsInfo[item.apiPath]) + .sort( + (item1: any, item2: any) => + endpointsInfo[item1.apiPath].order - + endpointsInfo[item2.apiPath].order, + ) + .map((item: any) => { + const daysRec = days.reduce( + (acc, day: number) => { + acc[`d${day}`] = 0; + return acc; + }, + {} as Record, + ); + + for (const dayKey in item.days) { + const day = item.days[dayKey]; + const dayNum = new Date(Date.parse(day.day)).getDate(); + daysRec[`d${dayNum}`] = day.trafficTypes[0].count; + } + const epInfo = endpointsInfo[item.apiPath]; + + return { + label: epInfo.label, + data: Object.values(daysRec), + backgroundColor: epInfo.color, + hoverBackgroundColor: epInfo.color, + }; + }); + + return data; +}; + +const customHighlightPlugin = { + id: 'customLine', + beforeDraw: (chart: Chart) => { + const width = 36; + if (chart.tooltip?.opacity && chart.tooltip.x) { + const x = chart.tooltip.caretX; + const yAxis = chart.scales.y; + const ctx = chart.ctx; + ctx.save(); + const gradient = ctx.createLinearGradient( + x, + yAxis.top, + x, + yAxis.bottom + 34, + ); + gradient.addColorStop(0, 'rgba(129, 122, 254, 0)'); + gradient.addColorStop(1, 'rgba(129, 122, 254, 0.12)'); + ctx.fillStyle = gradient; + ctx.roundRect( + x - width / 2, + yAxis.top, + width, + yAxis.bottom - yAxis.top + 34, + 5, + ); + ctx.fill(); + ctx.restore(); + } + }, +}; + +const createBarChartOptions = ( + theme: Theme, + tooltipTitleCallback: (tooltipItems: any) => string, +): ChartOptions<'bar'> => ({ + plugins: { + legend: { + position: 'bottom', + labels: { + color: theme.palette.text.primary, + pointStyle: 'circle', + usePointStyle: true, + boxHeight: 6, + padding: 15, + boxPadding: 5, + }, + }, + tooltip: { + backgroundColor: theme.palette.background.paper, + titleColor: theme.palette.text.primary, + bodyColor: theme.palette.text.primary, + bodySpacing: 6, + padding: { + top: 20, + bottom: 20, + left: 30, + right: 30, + }, + borderColor: 'rgba(0, 0, 0, 0.05)', + borderWidth: 3, + usePointStyle: true, + caretSize: 0, + boxPadding: 10, + callbacks: { + title: tooltipTitleCallback, + }, + }, + }, + responsive: true, + scales: { + x: { + stacked: true, + ticks: { + color: theme.palette.text.secondary, + }, + grid: { + display: false, + }, + }, + y: { + stacked: true, + ticks: { + color: theme.palette.text.secondary, + maxTicksLimit: 5, + callback: ( + tickValue: string | number, + index: number, + ticks: Tick[], + ) => { + if (typeof tickValue === 'string') { + return tickValue; + } + const value = Number.parseInt(tickValue.toString()); + if (value > 999999) { + return `${value / 1000000}M`; + } + return value > 999 ? `${value / 1000}k` : value; + }, + }, + grid: { + drawBorder: false, + }, + }, + }, + elements: { + bar: { + borderRadius: 5, + }, + }, + interaction: { + mode: 'index', + intersect: false, + }, +}); + +export const NetworkTrafficUsage: VFC = () => { + usePageTitle('Network - Data Usage'); + const theme = useTheme(); + + const endpointsInfo: Record = { + '/api/admin': { + label: 'Admin', + color: '#6D66D9', + order: 1, + }, + '/api/frontend': { + label: 'Frontend', + color: '#A39EFF', + order: 2, + }, + '/api/client': { + label: 'Server', + color: '#D8D6FF', + order: 3, + }, + }; + + const selectablePeriods = getSelectablePeriods(); + const record = toPeriodsRecord(selectablePeriods); + const [period, setPeriod] = useState(selectablePeriods[0].key); + + const options = useMemo(() => { + return createBarChartOptions(theme, (tooltipItems: any) => { + const periodItem = record[period]; + const tooltipDate = new Date( + periodItem.year, + periodItem.month, + Number.parseInt(tooltipItems[0].label), + ); + return tooltipDate.toLocaleDateString('en-US', { + month: 'long', + day: 'numeric', + year: 'numeric', + }); + }); + }, [theme, period]); + + const traffic = useInstanceTrafficMetrics(period); + + const [labels, setLabels] = useState([]); + + const [datasets, setDatasets] = useState([]); + + const data = { + labels, + datasets, + }; + + const { isOss } = useUiConfig(); + const flagEnabled = useUiFlag('collectTrafficDataUsage'); + + useEffect(() => { + setDatasets(toChartData(labels, traffic, endpointsInfo)); + }, [labels, traffic]); + + useEffect(() => { + if (record && period) { + const periodData = record[period]; + setLabels(getDayLabels(periodData.dayCount)); + } + }, [period]); + + return ( + No data available.} + elseShow={ + <> + + + + Number of requests to Unleash + + + +