diff --git a/frontend/src/api/infraMonitoring/getK8sNamespacesList.ts b/frontend/src/api/infraMonitoring/getK8sNamespacesList.ts new file mode 100644 index 0000000000..5a6fab10b5 --- /dev/null +++ b/frontend/src/api/infraMonitoring/getK8sNamespacesList.ts @@ -0,0 +1,62 @@ +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 K8sNamespacesListPayload { + filters: TagFilter; + groupBy?: BaseAutocompleteData[]; + offset?: number; + limit?: number; + orderBy?: { + columnName: string; + order: 'asc' | 'desc'; + }; +} + +export interface K8sNamespacesData { + namespaceName: string; + cpuUsage: number; + memoryUsage: number; + meta: { + k8s_cluster_name: string; + k8s_namespace_uid: string; + }; +} + +export interface K8sNamespacesListResponse { + status: string; + data: { + type: string; + records: K8sNamespacesData[]; + groups: null; + total: number; + sentAnyHostMetricsData: boolean; + isSendingK8SAgentMetrics: boolean; + }; +} + +export const getK8sNamespacesList = async ( + props: K8sNamespacesListPayload, + signal?: AbortSignal, + headers?: Record, +): Promise | ErrorResponse> => { + try { + const response = await ApiBaseInstance.post('/namespaces/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/container/InfraMonitoringK8s/Namespaces/K8sNamespacesList.tsx b/frontend/src/container/InfraMonitoringK8s/Namespaces/K8sNamespacesList.tsx new file mode 100644 index 0000000000..ef1bf43e5e --- /dev/null +++ b/frontend/src/container/InfraMonitoringK8s/Namespaces/K8sNamespacesList.tsx @@ -0,0 +1,248 @@ +/* eslint-disable @typescript-eslint/explicit-function-return-type */ +import '../InfraMonitoringK8s.styles.scss'; + +import { LoadingOutlined } from '@ant-design/icons'; +import { + Skeleton, + Spin, + Table, + TablePaginationConfig, + TableProps, + Typography, +} from 'antd'; +import { SorterResult } from 'antd/es/table/interface'; +import logEvent from 'api/common/logEvent'; +import { K8sNamespacesListPayload } from 'api/infraMonitoring/getK8sNamespacesList'; +import { useGetK8sNamespacesList } from 'hooks/infraMonitoring/useGetK8sNamespacesList'; +import { useCallback, useEffect, useMemo, useState } from 'react'; +import { useSelector } from 'react-redux'; +import { AppState } from 'store/reducers'; +import { IBuilderQuery } from 'types/api/queryBuilder/queryBuilderData'; +import { GlobalReducer } from 'types/reducer/globalTime'; + +import K8sHeader from '../K8sHeader'; +import { + defaultAddedColumns, + formatDataForTable, + getK8sNamespacesListColumns, + getK8sNamespacesListQuery, + K8sNamespacesRowData, +} from './utils'; + +function K8sNamespacesList({ + isFiltersVisible, + handleFilterVisibilityChange, +}: { + isFiltersVisible: boolean; + handleFilterVisibilityChange: () => void; +}): JSX.Element { + const { maxTime, minTime } = useSelector( + (state) => state.globalTime, + ); + + const [currentPage, setCurrentPage] = useState(1); + + const [filters, setFilters] = useState({ + items: [], + op: 'and', + }); + + const [orderBy, setOrderBy] = useState<{ + columnName: string; + order: 'asc' | 'desc'; + } | null>(null); + + // const [selectedNamespaceUID, setselectedNamespaceUID] = useState(null); + + const pageSize = 10; + + const query = useMemo(() => { + const baseQuery = getK8sNamespacesListQuery(); + return { + ...baseQuery, + limit: pageSize, + offset: (currentPage - 1) * pageSize, + filters, + start: Math.floor(minTime / 1000000), + end: Math.floor(maxTime / 1000000), + orderBy, + }; + }, [currentPage, filters, minTime, maxTime, orderBy]); + + const { data, isFetching, isLoading, isError } = useGetK8sNamespacesList( + query as K8sNamespacesListPayload, + { + queryKey: ['hostList', query], + enabled: !!query, + }, + ); + + const NamespacesData = useMemo(() => data?.payload?.data?.records || [], [ + data, + ]); + const totalCount = data?.payload?.data?.total || 0; + + const formattedNamespacesData = useMemo( + () => formatDataForTable(NamespacesData), + [NamespacesData], + ); + + const columns = useMemo(() => getK8sNamespacesListColumns(), []); + + const handleTableChange: TableProps['onChange'] = useCallback( + ( + pagination: TablePaginationConfig, + _filters: Record, + sorter: + | SorterResult + | SorterResult[], + ): void => { + if (pagination.current) { + setCurrentPage(pagination.current); + } + + if ('field' in sorter && sorter.order) { + setOrderBy({ + columnName: sorter.field as string, + order: sorter.order === 'ascend' ? 'asc' : 'desc', + }); + } else { + setOrderBy(null); + } + }, + [], + ); + + const handleFiltersChange = useCallback( + (value: IBuilderQuery['filters']): void => { + const isNewFilterAdded = value.items.length !== filters.items.length; + if (isNewFilterAdded) { + setFilters(value); + setCurrentPage(1); + + logEvent('Infra Monitoring: K8s list filters applied', { + filters: value, + }); + } + }, + [filters], + ); + + useEffect(() => { + logEvent('Infra Monitoring: K8s list page visited', {}); + }, []); + + // const selectedNamespaceData = useMemo(() => { + // if (!selectedNamespaceUID) return null; + // return NamespacesData.find((Namespace) => Namespace.NamespaceUID === selectedNamespaceUID) || null; + // }, [selectedNamespaceUID, NamespacesData]); + + const handleRowClick = (record: K8sNamespacesRowData): void => { + // setselectedNamespaceUID(record.NamespaceUID); + + logEvent('Infra Monitoring: K8s Namespace list item clicked', { + namespaceName: record.namespaceName, + }); + }; + + // const handleCloseNamespaceDetail = (): void => { + // setselectedNamespaceUID(null); + // }; + + const showsNamespacesTable = + !isError && + !isLoading && + !isFetching && + !(formattedNamespacesData.length === 0 && filters.items.length > 0); + + const showNoFilteredNamespacesMessage = + !isFetching && + !isLoading && + formattedNamespacesData.length === 0 && + filters.items.length > 0; + + return ( +
+ {}} + onRemoveColumn={() => {}} + /> + {isError && {data?.error || 'Something went wrong'}} + + {showNoFilteredNamespacesMessage && ( +
+
+ thinking-emoji + + + This query had no results. Edit your query and try again! + +
+
+ )} + + {(isFetching || isLoading) && ( +
+ + + +
+ )} + + {showsNamespacesTable && ( + } />, + }} + tableLayout="fixed" + rowKey={(record): string => record.namespaceName} + onChange={handleTableChange} + onRow={(record): { onClick: () => void; className: string } => ({ + onClick: (): void => handleRowClick(record), + className: 'clickable-row', + })} + /> + )} + {/* TODO - Handle Namespace Details flow */} + + ); +} + +export default K8sNamespacesList; diff --git a/frontend/src/container/InfraMonitoringK8s/Namespaces/utils.tsx b/frontend/src/container/InfraMonitoringK8s/Namespaces/utils.tsx new file mode 100644 index 0000000000..3e00ce7a26 --- /dev/null +++ b/frontend/src/container/InfraMonitoringK8s/Namespaces/utils.tsx @@ -0,0 +1,118 @@ +import { ColumnType } from 'antd/es/table'; +import { + K8sNamespacesData, + K8sNamespacesListPayload, +} from 'api/infraMonitoring/getK8sNamespacesList'; + +import { IEntityColumn } from '../utils'; + +export const defaultAddedColumns: IEntityColumn[] = [ + { + label: 'Namespace Status', + value: 'NamespaceStatus', + id: 'NamespaceStatus', + canRemove: false, + }, + { + label: 'CPU Utilization (cores)', + value: 'cpuUsage', + id: 'cpuUsage', + canRemove: false, + }, + { + label: 'CPU Allocatable (cores)', + value: 'cpuAllocatable', + id: 'cpuAllocatable', + canRemove: false, + }, + { + label: 'Memory Allocatable (bytes)', + value: 'memoryAllocatable', + id: 'memoryAllocatable', + canRemove: false, + }, + { + label: 'Pods count by phase', + value: 'podsCount', + id: 'podsCount', + canRemove: false, + }, +]; + +export interface K8sNamespacesRowData { + key: string; + namespaceUID: string; + namespaceName: string; + cpuUsage: React.ReactNode; + memoryUsage: React.ReactNode; + podsCount: number; + containerRestarts: React.ReactNode; +} + +export const getK8sNamespacesListQuery = (): K8sNamespacesListPayload => ({ + filters: { + items: [], + op: 'and', + }, + orderBy: { columnName: 'cpu', order: 'desc' }, +}); + +const columnsConfig = [ + { + title:
Namespace
, + dataIndex: 'namespaceName', + key: 'namespaceName', + ellipsis: true, + width: 150, + sorter: true, + align: 'left', + }, + { + title:
CPU Utilization (cores)
, + dataIndex: 'cpuUsage', + key: 'cpuUsage', + width: 100, + sorter: true, + align: 'left', + }, + { + title:
Memory Utilization (bytes)
, + dataIndex: 'memoryUsage', + key: 'memoryUsage', + width: 80, + sorter: true, + align: 'left', + }, + { + title:
Container Restarts
, + dataIndex: 'containerRestarts', + key: 'containerRestarts', + width: 80, + sorter: true, + align: 'left', + }, + { + title:
Pods count by phase
, + dataIndex: 'podsCounts', + key: 'podsCount', + width: 50, + sorter: true, + align: 'left', + }, +]; + +export const getK8sNamespacesListColumns = (): ColumnType[] => + columnsConfig as ColumnType[]; + +export const formatDataForTable = ( + data: K8sNamespacesData[], +): K8sNamespacesRowData[] => + data.map((namespace, index) => ({ + key: `${namespace.namespaceName}-${index}`, + namespaceUID: namespace.meta.k8s_namespace_uid, + namespaceName: namespace.namespaceName, + cpuUsage: namespace.cpuUsage, + memoryUsage: namespace.memoryUsage, + podsCount: namespace.cpuUsage, + containerRestarts: namespace.cpuUsage, + })); diff --git a/frontend/src/hooks/infraMonitoring/useGetK8sNamespacesList.ts b/frontend/src/hooks/infraMonitoring/useGetK8sNamespacesList.ts new file mode 100644 index 0000000000..323e5c1bc4 --- /dev/null +++ b/frontend/src/hooks/infraMonitoring/useGetK8sNamespacesList.ts @@ -0,0 +1,48 @@ +import { + getK8sNamespacesList, + K8sNamespacesListPayload, + K8sNamespacesListResponse, +} from 'api/infraMonitoring/getK8sNamespacesList'; +import { REACT_QUERY_KEY } from 'constants/reactQueryKeys'; +import { useMemo } from 'react'; +import { useQuery, UseQueryOptions, UseQueryResult } from 'react-query'; +import { ErrorResponse, SuccessResponse } from 'types/api'; + +type UseGetK8sNamespacesList = ( + requestData: K8sNamespacesListPayload, + options?: UseQueryOptions< + SuccessResponse | ErrorResponse, + Error + >, + headers?: Record, +) => UseQueryResult< + SuccessResponse | ErrorResponse, + Error +>; + +export const useGetK8sNamespacesList: UseGetK8sNamespacesList = ( + requestData, + options, + headers, +) => { + const queryKey = useMemo(() => { + if (options?.queryKey && Array.isArray(options.queryKey)) { + return [...options.queryKey]; + } + + if (options?.queryKey && typeof options.queryKey === 'string') { + return options.queryKey; + } + + return [REACT_QUERY_KEY.GET_HOST_LIST, requestData]; + }, [options?.queryKey, requestData]); + + return useQuery< + SuccessResponse | ErrorResponse, + Error + >({ + queryFn: ({ signal }) => getK8sNamespacesList(requestData, signal, headers), + ...options, + queryKey, + }); +};