From 19fd8be920cf3f7ae5ac57c1d6b52e78f214ab35 Mon Sep 17 00:00:00 2001 From: amlannandy Date: Tue, 14 Jan 2025 19:08:43 +0530 Subject: [PATCH] feat: daemonsets implementation for K8s Infra Monitoring --- .../infraMonitoring/getK8sDaemonSetsList.ts | 70 +++ frontend/src/constants/reactQueryKeys.ts | 1 + .../DaemonSetDetails.interfaces.ts | 7 + .../DaemonSetDetails.styles.scss | 247 ++++++++ .../DaemonSetDetails/DaemonSetDetails.tsx | 575 ++++++++++++++++++ .../Events/DaemonSetEvents.styles.scss | 289 +++++++++ .../Events/DaemonSetEvents.tsx | 366 +++++++++++ .../Events/NoEventsContainer.tsx | 16 + .../DaemonSetDetails/Events/constants.ts | 65 ++ .../DaemonSetDetails/Events/index.ts | 3 + .../Logs/DaemonSetLogs.styles.scss | 133 ++++ .../DaemonSetDetails/Logs/DaemonSetLogs.tsx | 216 +++++++ .../Logs/DaemonSetLogsDetailedView.tsx | 99 +++ .../DaemonSetDetails/Logs/NoLogsContainer.tsx | 16 + .../DaemonSetDetails/Logs/constants.ts | 65 ++ .../DaemonSets/DaemonSetDetails/Logs/index.ts | 3 + .../Metrics/DaemonSetMetrics.styles.scss | 45 ++ .../Metrics/DaemonSetMetrics.tsx | 141 +++++ .../DaemonSetDetails/Metrics/constants.ts | 544 +++++++++++++++++ .../DaemonSetDetails/Metrics/index.ts | 3 + .../Traces/DaemonSetTraces.styles.scss | 193 ++++++ .../Traces/DaemonSetTraces.tsx | 199 ++++++ .../DaemonSetDetails/Traces/constants.ts | 200 ++++++ .../DaemonSetDetails/Traces/index.ts | 3 + .../DaemonSets/DaemonSetDetails/constants.ts | 6 + .../DaemonSets/DaemonSetDetails/index.ts | 3 + .../DaemonSets/K8sDaemonSetsList.styles.scss | 27 + .../DaemonSets/K8sDaemonSetsList.tsx | 517 ++++++++++++++++ .../InfraMonitoringK8s/DaemonSets/utils.tsx | 335 ++++++++++ .../InfraMonitoringK8s/InfraMonitoringK8s.tsx | 53 +- .../container/InfraMonitoringK8s/constants.ts | 35 +- .../useGetK8sDaemonSetsList.ts | 54 ++ 32 files changed, 4505 insertions(+), 24 deletions(-) create mode 100644 frontend/src/api/infraMonitoring/getK8sDaemonSetsList.ts create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/DaemonSetDetails.interfaces.ts create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/DaemonSetDetails.styles.scss create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/DaemonSetDetails.tsx create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Events/DaemonSetEvents.styles.scss create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Events/DaemonSetEvents.tsx create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Events/NoEventsContainer.tsx create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Events/constants.ts create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Events/index.ts create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Logs/DaemonSetLogs.styles.scss create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Logs/DaemonSetLogs.tsx create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Logs/DaemonSetLogsDetailedView.tsx create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Logs/NoLogsContainer.tsx create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Logs/constants.ts create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Logs/index.ts create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Metrics/DaemonSetMetrics.styles.scss create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Metrics/DaemonSetMetrics.tsx create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Metrics/constants.ts create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Metrics/index.ts create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Traces/DaemonSetTraces.styles.scss create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Traces/DaemonSetTraces.tsx create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Traces/constants.ts create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/Traces/index.ts create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/constants.ts create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/index.ts create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/K8sDaemonSetsList.styles.scss create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/K8sDaemonSetsList.tsx create mode 100644 frontend/src/container/InfraMonitoringK8s/DaemonSets/utils.tsx create mode 100644 frontend/src/hooks/infraMonitoring/useGetK8sDaemonSetsList.ts diff --git a/frontend/src/api/infraMonitoring/getK8sDaemonSetsList.ts b/frontend/src/api/infraMonitoring/getK8sDaemonSetsList.ts new file mode 100644 index 0000000000..f07651dccc --- /dev/null +++ b/frontend/src/api/infraMonitoring/getK8sDaemonSetsList.ts @@ -0,0 +1,70 @@ +import { ApiBaseInstance } from 'api'; +import { ErrorResponseHandler } from 'api/ErrorResponseHandler'; +import { AxiosError } from 'axios'; +import { ErrorResponse, SuccessResponse } from 'types/api'; +import { BaseAutocompleteData } from 'types/api/queryBuilder/queryAutocompleteResponse'; +import { TagFilter } from 'types/api/queryBuilder/queryBuilderData'; + +export interface K8sDaemonSetsListPayload { + filters: TagFilter; + groupBy?: BaseAutocompleteData[]; + offset?: number; + limit?: number; + orderBy?: { + columnName: string; + order: 'asc' | 'desc'; + }; +} + +export interface K8sDaemonSetsData { + daemonSetName: string; + cpuUsage: number; + memoryUsage: number; + cpuRequest: number; + memoryRequest: number; + cpuLimit: number; + memoryLimit: number; + restarts: number; + desiredNodes: number; + availableNodes: number; + meta: { + k8s_cluster_name: string; + k8s_daemonset_name: string; + k8s_namespace_name: string; + }; +} + +export interface K8sDaemonSetsListResponse { + status: string; + data: { + type: string; + records: K8sDaemonSetsData[]; + groups: null; + total: number; + sentAnyHostMetricsData: boolean; + isSendingK8SAgentMetrics: boolean; + }; +} + +export const getK8sDaemonSetsList = async ( + props: K8sDaemonSetsListPayload, + signal?: AbortSignal, + headers?: Record, +): Promise | ErrorResponse> => { + try { + const response = await ApiBaseInstance.post('/daemonsets/list', props, { + signal, + headers, + }); + + return { + statusCode: 200, + error: null, + message: 'Success', + payload: response.data, + params: props, + }; + } catch (error) { + return ErrorResponseHandler(error as AxiosError); + } +}; diff --git a/frontend/src/constants/reactQueryKeys.ts b/frontend/src/constants/reactQueryKeys.ts index 2311566db9..a45b5e5bcb 100644 --- a/frontend/src/constants/reactQueryKeys.ts +++ b/frontend/src/constants/reactQueryKeys.ts @@ -24,4 +24,5 @@ export const REACT_QUERY_KEY = { GET_POD_LIST: 'GET_POD_LIST', GET_STATEFULSET_LIST: 'GET_STATEFULSET_LIST', GET_JOB_LIST: 'GET_JOB_LIST', + GET_DAEMONSET_LIST: 'GET_DAEMONSET_LIST,', }; diff --git a/frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/DaemonSetDetails.interfaces.ts b/frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/DaemonSetDetails.interfaces.ts new file mode 100644 index 0000000000..6b21b2f77e --- /dev/null +++ b/frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/DaemonSetDetails.interfaces.ts @@ -0,0 +1,7 @@ +import { K8sDaemonSetsData } from 'api/infraMonitoring/getK8sDaemonSetsList'; + +export type DaemonSetDetailsProps = { + daemonSet: K8sDaemonSetsData | null; + isModalTimeSelection: boolean; + onClose: () => void; +}; diff --git a/frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/DaemonSetDetails.styles.scss b/frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/DaemonSetDetails.styles.scss new file mode 100644 index 0000000000..265bbf06f3 --- /dev/null +++ b/frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/DaemonSetDetails.styles.scss @@ -0,0 +1,247 @@ +.daemonSet-detail-drawer { + border-left: 1px solid var(--bg-slate-500); + background: var(--bg-ink-400); + box-shadow: -4px 10px 16px 2px rgba(0, 0, 0, 0.2); + + .ant-drawer-header { + padding: 8px 16px; + border-bottom: none; + + align-items: stretch; + + border-bottom: 1px solid var(--bg-slate-500); + background: var(--bg-ink-400); + } + + .ant-drawer-close { + margin-inline-end: 0px; + } + + .ant-drawer-body { + display: flex; + flex-direction: column; + padding: 16px; + } + + .title { + color: var(--text-vanilla-400); + font-family: 'Geist Mono'; + font-size: 14px; + font-style: normal; + font-weight: 500; + line-height: 20px; /* 142.857% */ + letter-spacing: -0.07px; + } + + .radio-button { + display: flex; + align-items: center; + justify-content: center; + padding-top: var(--padding-1); + border: 1px solid var(--bg-slate-400); + background: var(--bg-ink-300); + box-shadow: 0px 0px 8px 0px rgba(0, 0, 0, 0.1); + } + + .daemonSet-detail-drawer__daemonSet { + .daemonSet-details-grid { + .labels-row, + .values-row { + display: grid; + grid-template-columns: 1fr 1.5fr 1.5fr 1.5fr; + gap: 30px; + align-items: center; + } + + .labels-row { + margin-bottom: 8px; + } + + .daemonSet-details-metadata-label { + color: var(--text-vanilla-400); + font-family: Inter; + font-size: 11px; + font-style: normal; + font-weight: 500; + line-height: 18px; /* 163.636% */ + letter-spacing: 0.44px; + text-transform: uppercase; + } + + .daemonSet-details-metadata-value { + color: var(--text-vanilla-400); + font-family: 'Geist Mono'; + font-size: 12px; + font-style: normal; + font-weight: 500; + line-height: 20px; /* 142.857% */ + letter-spacing: -0.07px; + + width: 100%; + white-space: nowrap; + overflow: hidden; + text-overflow: ellipsis; + } + + .status-tag { + margin: 0; + + &.active { + color: var(--success-500); + background: var(--success-100); + border-color: var(--success-500); + } + + &.inactive { + color: var(--error-500); + background: var(--error-100); + border-color: var(--error-500); + } + } + + .progress-container { + width: 158px; + .ant-progress { + margin: 0; + + .ant-progress-text { + font-weight: 600; + } + } + } + + .ant-card { + &.ant-card-bordered { + border: 1px solid var(--bg-slate-500) !important; + } + } + } + } + + .tabs-and-search { + display: flex; + justify-content: space-between; + align-items: center; + margin: 16px 0; + + .action-btn { + border-radius: 2px; + border: 1px solid var(--bg-slate-400); + background: var(--bg-ink-300); + box-shadow: 0px 0px 8px 0px rgba(0, 0, 0, 0.1); + display: flex; + align-items: center; + justify-content: center; + } + } + + .views-tabs-container { + margin-top: 1.5rem; + display: flex; + justify-content: space-between; + align-items: center; + + .views-tabs { + color: var(--text-vanilla-400); + + .view-title { + display: flex; + gap: var(--margin-2); + align-items: center; + justify-content: center; + font-size: var(--font-size-xs); + font-style: normal; + font-weight: var(--font-weight-normal); + } + + .tab { + border: 1px solid var(--bg-slate-400); + width: 114px; + } + + .tab::before { + background: var(--bg-slate-400); + } + + .selected_view { + background: var(--bg-slate-300); + color: var(--text-vanilla-100); + border: 1px solid var(--bg-slate-400); + } + + .selected_view::before { + background: var(--bg-slate-400); + } + } + + .compass-button { + width: 30px; + height: 30px; + + border-radius: 2px; + border: 1px solid var(--bg-slate-400); + background: var(--bg-ink-300); + box-shadow: 0px 0px 8px 0px rgba(0, 0, 0, 0.1); + } + } + .ant-drawer-close { + padding: 0px; + } +} + +.lightMode { + .ant-drawer-header { + border-bottom: 1px solid var(--bg-vanilla-400); + background: var(--bg-vanilla-100); + } + + .daemonSet-detail-drawer { + .title { + color: var(--text-ink-300); + } + + .daemonSet-detail-drawer__daemonSet { + .ant-typography { + color: var(--text-ink-300); + background: transparent; + } + } + + .radio-button { + border: 1px solid var(--bg-vanilla-400); + background: var(--bg-vanilla-100); + color: var(--text-ink-300); + } + + .views-tabs { + .tab { + background: var(--bg-vanilla-100); + } + + .selected_view { + background: var(--bg-vanilla-300); + border: 1px solid var(--bg-slate-300); + color: var(--text-ink-400); + } + + .selected_view::before { + background: var(--bg-vanilla-300); + border-left: 1px solid var(--bg-slate-300); + } + } + + .compass-button { + border: 1px solid var(--bg-vanilla-300); + background: var(--bg-vanilla-100); + box-shadow: 0px 0px 8px 0px rgba(0, 0, 0, 0.1); + } + + .tabs-and-search { + .action-btn { + border: 1px solid var(--bg-vanilla-400); + background: var(--bg-vanilla-100); + color: var(--text-ink-300); + } + } + } +} diff --git a/frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/DaemonSetDetails.tsx b/frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/DaemonSetDetails.tsx new file mode 100644 index 0000000000..19cac2c122 --- /dev/null +++ b/frontend/src/container/InfraMonitoringK8s/DaemonSets/DaemonSetDetails/DaemonSetDetails.tsx @@ -0,0 +1,575 @@ +/* eslint-disable sonarjs/no-identical-functions */ +import './DaemonSetDetails.styles.scss'; + +import { Color, Spacing } from '@signozhq/design-tokens'; +import { Button, Divider, Drawer, Radio, Tooltip, Typography } from 'antd'; +import { RadioChangeEvent } from 'antd/lib'; +import logEvent from 'api/common/logEvent'; +import { VIEW_TYPES, VIEWS } from 'components/HostMetricsDetail/constants'; +import { QueryParams } from 'constants/query'; +import { + initialQueryBuilderFormValuesMap, + initialQueryState, +} from 'constants/queryBuilder'; +import ROUTES from 'constants/routes'; +import { + CustomTimeType, + Time, +} from 'container/TopNav/DateTimeSelectionV2/config'; +import { useIsDarkMode } from 'hooks/useDarkMode'; +import useUrlQuery from 'hooks/useUrlQuery'; +import GetMinMax from 'lib/getMinMax'; +import { + BarChart2, + ChevronsLeftRight, + Compass, + DraftingCompass, + ScrollText, + X, +} from 'lucide-react'; +import { useCallback, useEffect, useMemo, useState } from 'react'; +import { useSelector } from 'react-redux'; +import { AppState } from 'store/reducers'; +import { DataTypes } from 'types/api/queryBuilder/queryAutocompleteResponse'; +import { + IBuilderQuery, + TagFilterItem, +} from 'types/api/queryBuilder/queryBuilderData'; +import { + LogsAggregatorOperator, + TracesAggregatorOperator, +} from 'types/common/queryBuilder'; +import { GlobalReducer } from 'types/reducer/globalTime'; +import { v4 as uuidv4 } from 'uuid'; + +import { QUERY_KEYS } from './constants'; +import { DaemonSetDetailsProps } from './DaemonSetDetails.interfaces'; +import DaemonSetEvents from './Events'; +import DaemonSetLogs from './Logs'; +import DaemonSetMetrics from './Metrics'; +import DaemonSetTraces from './Traces'; + +function DaemonSetDetails({ + daemonSet, + onClose, + isModalTimeSelection, +}: DaemonSetDetailsProps): JSX.Element { + const { maxTime, minTime, selectedTime } = useSelector< + AppState, + GlobalReducer + >((state) => state.globalTime); + + const startMs = useMemo(() => Math.floor(Number(minTime) / 1000000000), [ + minTime, + ]); + const endMs = useMemo(() => Math.floor(Number(maxTime) / 1000000000), [ + maxTime, + ]); + + const urlQuery = useUrlQuery(); + + const [modalTimeRange, setModalTimeRange] = useState(() => ({ + startTime: startMs, + endTime: endMs, + })); + + const [selectedInterval, setSelectedInterval] = useState