From 156d9e7787ffb6808a6c12832c73b8ff12bcc3f2 Mon Sep 17 00:00:00 2001 From: Piyush Kumar Date: Wed, 28 Aug 2024 15:59:16 +0530 Subject: [PATCH 1/6] cluster offline status in progress --- gql-queries-generator/doc/queries.graphql | 2 + src/apps/console/components/icons.tsx | 2 + src/apps/console/hooks/use-cluster-status.tsx | 7 +- .../$environment+/apps/apps-resources-v2.tsx | 56 +++-- .../managed-resources-resource-v2.tsx | 47 ++-- .../environments/environment-resources-v2.tsx | 212 ++++++++++++------ .../backend-services-resources-V2.tsx | 48 ++-- .../server/gql/queries/environment-queries.ts | 2 + src/generated/gql/sdl.graphql | 2 + src/generated/gql/server.ts | 3 + 10 files changed, 250 insertions(+), 131 deletions(-) diff --git a/gql-queries-generator/doc/queries.graphql b/gql-queries-generator/doc/queries.graphql index 88f55ca2a..83340bfd3 100644 --- a/gql-queries-generator/doc/queries.graphql +++ b/gql-queries-generator/doc/queries.graphql @@ -739,6 +739,7 @@ query consoleGetEnvironment($name: String!) { privateIngressClass publicIngressClass } + suspend targetNamespace } status { @@ -814,6 +815,7 @@ query consoleListEnvironments($search: SearchEnvironments, $pq: CursorPagination privateIngressClass publicIngressClass } + suspend targetNamespace } status { diff --git a/src/apps/console/components/icons.tsx b/src/apps/console/components/icons.tsx index 425f63616..9cec1d0a5 100644 --- a/src/apps/console/components/icons.tsx +++ b/src/apps/console/components/icons.tsx @@ -135,4 +135,6 @@ export { Sliders, FileLock, StackSimple, + Play, + Pause, } from '@jengaicons/react'; diff --git a/src/apps/console/hooks/use-cluster-status.tsx b/src/apps/console/hooks/use-cluster-status.tsx index 8c7a92f5d..ff28c1449 100644 --- a/src/apps/console/hooks/use-cluster-status.tsx +++ b/src/apps/console/hooks/use-cluster-status.tsx @@ -13,12 +13,7 @@ const findClusterStatus = (item?: { lastOnlineAt?: string }): boolean => { const timeDifference = (currentTime.getTime() - lastTime.getTime()) / (1000 * 60); - switch (true) { - case timeDifference <= 2: - return true; - default: - return false; - } + return timeDifference <= 2; }; const useClusterStatus = () => { diff --git a/src/apps/console/routes/_main+/$account+/env+/$environment+/apps/apps-resources-v2.tsx b/src/apps/console/routes/_main+/$account+/env+/$environment+/apps/apps-resources-v2.tsx index fccfcb945..e169367d5 100644 --- a/src/apps/console/routes/_main+/$account+/env+/$environment+/apps/apps-resources-v2.tsx +++ b/src/apps/console/routes/_main+/$account+/env+/$environment+/apps/apps-resources-v2.tsx @@ -1,11 +1,10 @@ -import { - GearSix, - LinkBreak, - Link as LinkIcon, - Repeat, -} from '~/console/components/icons'; import { Link, useOutletContext, useParams } from '@remix-run/react'; +import { useEffect, useState } from 'react'; +import { Badge } from '~/components/atoms/badge'; +import TooltipV2 from '~/components/atoms/tooltipV2'; +import { toast } from '~/components/molecule/toast'; import { generateKey, titleCase } from '~/components/utils'; +import { CopyContentToClipboard } from '~/console/components/common-console-components'; import { ListItem, ListItemV2, @@ -14,33 +13,34 @@ import { listClass, } from '~/console/components/console-list-components'; import Grid from '~/console/components/grid'; +import { + GearSix, + LinkBreak, + Link as LinkIcon, + Repeat, +} from '~/console/components/icons'; import ListGridView from '~/console/components/list-grid-view'; +import ListV2 from '~/console/components/listV2'; import ResourceExtraAction, { IResourceExtraItem, } from '~/console/components/resource-extra-action'; +import { SyncStatusV2 } from '~/console/components/sync-status'; +import useClusterStatus from '~/console/hooks/use-cluster-status'; import { useConsoleApi } from '~/console/server/gql/api-provider'; import { IApps } from '~/console/server/gql/queries/app-queries'; import { ExtractNodeType, parseName, - parseName as pn, parseUpdateOrCreatedBy, parseUpdateOrCreatedOn, + parseName as pn, } from '~/console/server/r-utils/common'; -import { handleError } from '~/lib/utils/common'; -import { toast } from '~/components/molecule/toast'; import { useReload } from '~/lib/client/helpers/reloader'; -import { SyncStatusV2 } from '~/console/components/sync-status'; import { useWatchReload } from '~/lib/client/helpers/socket/useWatch'; -import ListV2 from '~/console/components/listV2'; -import { useState } from 'react'; -import { Badge } from '~/components/atoms/badge'; -import { CopyContentToClipboard } from '~/console/components/common-console-components'; +import { handleError } from '~/lib/utils/common'; import { NN } from '~/root/lib/types/common'; -import TooltipV2 from '~/components/atoms/tooltipV2'; -import useClusterStatus from '~/console/hooks/use-cluster-status'; -import HandleIntercept from './handle-intercept'; import { IEnvironmentContext } from '../_layout'; +import HandleIntercept from './handle-intercept'; const RESOURCE_NAME = 'app'; type BaseType = ExtractNodeType; @@ -242,6 +242,17 @@ const ListView = ({ items = [], onAction }: IResource) => { useOutletContext(); const { findClusterStatus, clusters, loading } = useClusterStatus(); + const [clusterOnlineStatus, setClusterOnlineStatus] = useState< + Record + >({}); + useEffect(() => { + const states: Record = {}; + clusters.forEach((c) => { + states[c.metadata.name] = findClusterStatus(c); + }); + setClusterOnlineStatus(states); + }, [clusters]); + return ( { }, ], rows: items.map((i) => { - const isClusterOnline = findClusterStatus( - clusters.length > 0 - ? clusters.find((c) => parseName(c) === parseName(cluster)) - : cluster - ); + const isClusterOnline = clusterOnlineStatus[parseName(cluster)]; const { name, id, updateInfo } = parseItem(i); return { @@ -325,6 +332,11 @@ const ListView = ({ items = [], onAction }: IResource) => { if (loading) { return null; } + + if (environment.spec?.suspend) { + return null; + } + if (!isClusterOnline) { return Cluster Offline; } diff --git a/src/apps/console/routes/_main+/$account+/env+/$environment+/managed-resources/managed-resources-resource-v2.tsx b/src/apps/console/routes/_main+/$account+/env+/$environment+/managed-resources/managed-resources-resource-v2.tsx index 3710a0dab..54a3f506e 100644 --- a/src/apps/console/routes/_main+/$account+/env+/$environment+/managed-resources/managed-resources-resource-v2.tsx +++ b/src/apps/console/routes/_main+/$account+/env+/$environment+/managed-resources/managed-resources-resource-v2.tsx @@ -1,4 +1,7 @@ -import { LockSimple, Trash } from '~/console/components/icons'; +import { useOutletContext, useParams } from '@remix-run/react'; +import { useEffect, useState } from 'react'; +import { Badge } from '~/components/atoms/badge'; +import { toast } from '~/components/molecule/toast'; import { generateKey, titleCase } from '~/components/utils'; import { ListItem, @@ -7,31 +10,28 @@ import { ListTitleV2, listClass, } from '~/console/components/console-list-components'; +import DeleteDialog from '~/console/components/delete-dialog'; import Grid from '~/console/components/grid'; +import { LockSimple, Trash } from '~/console/components/icons'; import ListGridView from '~/console/components/list-grid-view'; +import ListV2 from '~/console/components/listV2'; +import ResourceExtraAction from '~/console/components/resource-extra-action'; +import useClusterStatus from '~/console/hooks/use-cluster-status'; +import { useConsoleApi } from '~/console/server/gql/api-provider'; +import { IImportedManagedResources } from '~/console/server/gql/queries/imported-managed-resource-queries'; +import { IMSvTemplates } from '~/console/server/gql/queries/managed-templates-queries'; import { ExtractNodeType, parseName, parseUpdateOrCreatedBy, parseUpdateOrCreatedOn, } from '~/console/server/r-utils/common'; -import DeleteDialog from '~/console/components/delete-dialog'; -import ResourceExtraAction from '~/console/components/resource-extra-action'; -import { useConsoleApi } from '~/console/server/gql/api-provider'; +import { getManagedTemplateLogo } from '~/console/utils/commons'; import { useReload } from '~/lib/client/helpers/reloader'; -import { useState } from 'react'; -import { handleError } from '~/lib/utils/common'; -import { toast } from '~/components/molecule/toast'; -import { useOutletContext, useParams } from '@remix-run/react'; import { useWatchReload } from '~/lib/client/helpers/socket/useWatch'; -import ListV2 from '~/console/components/listV2'; -import { IMSvTemplates } from '~/console/server/gql/queries/managed-templates-queries'; -import { getManagedTemplateLogo } from '~/console/utils/commons'; -import { IImportedManagedResources } from '~/console/server/gql/queries/imported-managed-resource-queries'; -import { Badge } from '~/components/atoms/badge'; -import useClusterStatus from '~/console/hooks/use-cluster-status'; -import { ViewSecret } from './handle-managed-resource-v2'; +import { handleError } from '~/lib/utils/common'; import { IEnvironmentContext } from '../_layout'; +import { ViewSecret } from './handle-managed-resource-v2'; const RESOURCE_NAME = 'integrated resource'; type BaseType = ExtractNodeType; @@ -143,6 +143,17 @@ const ListView = ({ items = [], onAction, templates }: IResource) => { const { cluster } = useOutletContext(); const { findClusterStatus, clusters, loading } = useClusterStatus(); + const [clusterOnlineStatus, setClusterOnlineStatus] = useState< + Record + >({}); + useEffect(() => { + const states: Record = {}; + clusters.forEach((c) => { + states[c.metadata.name] = findClusterStatus(c); + }); + setClusterOnlineStatus(states); + }, [clusters]); + return ( { ], rows: items.map((i) => { const { name, id, logo, updateInfo } = parseItem(i, templates); - const isClusterOnline = findClusterStatus( - clusters.length > 0 - ? clusters.find((c) => parseName(c) === parseName(cluster)) - : cluster - ); + const isClusterOnline = clusterOnlineStatus[parseName(cluster)]; return { columns: { diff --git a/src/apps/console/routes/_main+/$account+/environments/environment-resources-v2.tsx b/src/apps/console/routes/_main+/$account+/environments/environment-resources-v2.tsx index 276a0a3dc..5f713aa71 100644 --- a/src/apps/console/routes/_main+/$account+/environments/environment-resources-v2.tsx +++ b/src/apps/console/routes/_main+/$account+/environments/environment-resources-v2.tsx @@ -1,6 +1,7 @@ -import { Copy, GearSix, Trash } from '~/console/components/icons'; import { Link, useOutletContext, useParams } from '@remix-run/react'; -import { useState } from 'react'; +import { useEffect, useState } from 'react'; +import { Badge } from '~/components/atoms/badge'; +import { toast } from '~/components/molecule/toast'; import { generateKey, titleCase } from '~/components/utils'; import ConsoleAvatar from '~/console/components/console-avatar'; import { @@ -10,9 +11,18 @@ import { ListTitleV2, listClass, } from '~/console/components/console-list-components'; +import DeleteDialog from '~/console/components/delete-dialog'; import Grid from '~/console/components/grid'; +import { Copy, GearSix, Pause, Play, Trash } from '~/console/components/icons'; import ListGridView from '~/console/components/list-grid-view'; -import ResourceExtraAction from '~/console/components/resource-extra-action'; +import ListV2 from '~/console/components/listV2'; +import ResourceExtraAction, { + IResourceExtraItem, +} from '~/console/components/resource-extra-action'; +import { SyncStatusV2 } from '~/console/components/sync-status'; +import useClusterStatus from '~/console/hooks/use-cluster-status'; +import { IAccountContext } from '~/console/routes/_main+/$account+/_layout'; +import { useConsoleApi } from '~/console/server/gql/api-provider'; import { IEnvironments } from '~/console/server/gql/queries/environment-queries'; import { ExtractNodeType, @@ -20,17 +30,9 @@ import { parseUpdateOrCreatedBy, parseUpdateOrCreatedOn, } from '~/console/server/r-utils/common'; -import { SyncStatusV2 } from '~/console/components/sync-status'; -import { IAccountContext } from '~/console/routes/_main+/$account+/_layout'; import { useWatchReload } from '~/lib/client/helpers/socket/useWatch'; -import ListV2 from '~/console/components/listV2'; -import DeleteDialog from '~/console/components/delete-dialog'; -import { useConsoleApi } from '~/console/server/gql/api-provider'; import { useReload } from '~/root/lib/client/helpers/reloader'; -import { toast } from '~/components/molecule/toast'; import { handleError } from '~/root/lib/utils/common'; -import { Badge } from '~/components/atoms/badge'; -import useClusterStatus from '~/console/hooks/use-cluster-status'; import CloneEnvironment from './clone-environment'; const RESOURCE_NAME = 'environment'; @@ -51,7 +53,7 @@ type OnAction = ({ action, item, }: { - action: 'clone' | 'delete'; + action: 'clone' | 'delete' | 'suspend' | 'resumed'; item: BaseType; }) => void; @@ -60,46 +62,73 @@ type IExtraButton = { item: BaseType; }; -const ExtraButton = ({ item, onAction }: IExtraButton) => { +const ExtraButton = ({ + item, + onAction, + isClusterOnline, +}: IExtraButton & { isClusterOnline?: boolean }) => { const { account } = useParams(); - return item.isArchived ? ( - , - type: 'item', - key: 'clone', - onClick: () => onAction({ action: 'clone', item }), - }, - { - label: 'Delete', - icon: , - type: 'item', - onClick: () => onAction({ action: 'delete', item }), - key: 'delete', - className: '!text-text-critical', - }, - ]} - /> - ) : ( - , - type: 'item', - key: 'clone', - onClick: () => onAction({ action: 'clone', item }), - }, - { - label: 'Delete', - icon: , - type: 'item', - onClick: () => onAction({ action: 'delete', item }), - key: 'delete', - className: '!text-text-critical', - }, + const iconSize = 16; + let options: IResourceExtraItem[] = [ + { + label: 'Clone', + icon: , + type: 'item', + key: 'clone', + onClick: () => onAction({ action: 'clone', item }), + }, + { + label: 'Delete', + icon: , + type: 'item', + onClick: () => onAction({ action: 'delete', item }), + key: 'delete', + className: '!text-text-critical', + }, + ]; + + if (!item.isArchived) { + if (isClusterOnline) { + if (item.spec?.suspend) { + options = [ + ...options, + { + label: 'Resumed', + icon: , + type: 'item', + key: 'resumed', + onClick: () => onAction({ action: 'resumed', item }), + }, + { + label: 'Settings', + icon: , + type: 'item', + to: `/${account}/env/${parseName(item)}/settings/general`, + key: 'settings', + }, + ]; + } else { + options = [ + ...options, + { + label: 'Suspend', + icon: , + type: 'item', + key: 'suspend', + onClick: () => onAction({ action: 'suspend', item }), + }, + { + label: 'Settings', + icon: , + type: 'item', + to: `/${account}/env/${parseName(item)}/settings/general`, + key: 'settings', + }, + ]; + } + } else { + options = [ + ...options, { label: 'Settings', icon: , @@ -107,9 +136,11 @@ const ExtraButton = ({ item, onAction }: IExtraButton) => { to: `/${account}/env/${parseName(item)}/settings/general`, key: 'settings', }, - ]} - /> - ); + ]; + } + } + + return ; }; interface IResource { @@ -162,7 +193,17 @@ const GridView = ({ items = [], onAction }: IResource) => { const ListView = ({ items, onAction }: IResource) => { const { account } = useParams(); const { findClusterStatus, clusters, loading } = useClusterStatus(); - const { clustersMap } = useOutletContext(); + + const [clusterOnlineStatus, setClusterOnlineStatus] = useState< + Record + >({}); + useEffect(() => { + const states: Record = {}; + clusters.forEach((c) => { + states[c.metadata.name] = findClusterStatus(c); + }); + setClusterOnlineStatus(states); + }, [clusters]); return ( { ], rows: items.map((i) => { const { name, id, updateInfo } = parseItem(i); - const isClusterOnline = findClusterStatus( - clusters.length > 0 - ? clusters.find((c) => parseName(c) === i.clusterName) - : clustersMap[i.clusterName] - ); + const isClusterOnline = clusterOnlineStatus[i.clusterName]; + return { columns: { name: { @@ -231,10 +269,15 @@ const ListView = ({ items, onAction }: IResource) => { if (loading) { return null; } + if (!isClusterOnline) { return Cluster Offline; } + if (i.spec?.suspend) { + return Suspended; + } + return ; }, }, @@ -247,7 +290,13 @@ const ListView = ({ items, onAction }: IResource) => { ), }, action: { - render: () => , + render: () => ( + + ), }, }, ...(i.isArchived ? {} : { to: `/${account}/env/${id}` }), @@ -260,18 +309,49 @@ const ListView = ({ items, onAction }: IResource) => { const EnvironmentResourcesV2 = ({ items = [] }: { items: BaseType[] }) => { const { account } = useOutletContext(); + const api = useConsoleApi(); + const reloadPage = useReload(); useWatchReload( items.map((i) => { return `account:${parseName(account)}.environment:${parseName(i)}`; }) ); + const suspendEnvironment = async (item: BaseType, suspend: boolean) => { + try { + const { errors } = await api.updateEnvironment({ + env: { + displayName: item.displayName, + clusterName: item.clusterName, + metadata: { + name: parseName(item), + }, + spec: { + suspend, + }, + }, + }); + + if (errors) { + throw errors[0]; + } + toast.success( + `${ + suspend + ? 'Environment suspended successfully' + : 'Environment resumed successfully' + }` + ); + reloadPage(); + } catch (err) { + handleError(err); + } + }; + const [showDeleteDialog, setShowDeleteDialog] = useState( null ); const [visible, setVisible] = useState(null); - const api = useConsoleApi(); - const reloadPage = useReload(); const props: IResource = { // @ts-ignore @@ -281,6 +361,12 @@ const EnvironmentResourcesV2 = ({ items = [] }: { items: BaseType[] }) => { case 'clone': setVisible(item); break; + case 'suspend': + suspendEnvironment(item, true); + break; + case 'resumed': + suspendEnvironment(item, false); + break; case 'delete': setShowDeleteDialog(item); break; diff --git a/src/apps/console/routes/_main+/$account+/managed-services/backend-services-resources-V2.tsx b/src/apps/console/routes/_main+/$account+/managed-services/backend-services-resources-V2.tsx index 61dbb7fb6..10162c936 100644 --- a/src/apps/console/routes/_main+/$account+/managed-services/backend-services-resources-V2.tsx +++ b/src/apps/console/routes/_main+/$account+/managed-services/backend-services-resources-V2.tsx @@ -1,4 +1,7 @@ -import { GearSix, Trash } from '~/console/components/icons'; +import { Link, useOutletContext, useParams } from '@remix-run/react'; +import { useEffect, useState } from 'react'; +import { Badge } from '~/components/atoms/badge'; +import { toast } from '~/components/molecule/toast'; import { generateKey, titleCase } from '~/components/utils'; import { ListItem, @@ -7,30 +10,27 @@ import { ListTitleV2, listClass, } from '~/console/components/console-list-components'; +import DeleteDialog from '~/console/components/delete-dialog'; import Grid from '~/console/components/grid'; +import { GearSix, Trash } from '~/console/components/icons'; import ListGridView from '~/console/components/list-grid-view'; +import ListV2 from '~/console/components/listV2'; +import ResourceExtraAction from '~/console/components/resource-extra-action'; +import { SyncStatusV2 } from '~/console/components/sync-status'; +import useClusterStatus from '~/console/hooks/use-cluster-status'; +import { useConsoleApi } from '~/console/server/gql/api-provider'; +import { IClusterMSvs } from '~/console/server/gql/queries/cluster-managed-services-queries'; +import { IMSvTemplates } from '~/console/server/gql/queries/managed-templates-queries'; import { ExtractNodeType, parseName, parseUpdateOrCreatedBy, parseUpdateOrCreatedOn, } from '~/console/server/r-utils/common'; -import { IMSvTemplates } from '~/console/server/gql/queries/managed-templates-queries'; import { getManagedTemplate } from '~/console/utils/commons'; -import ResourceExtraAction from '~/console/components/resource-extra-action'; -import { Link, useOutletContext, useParams } from '@remix-run/react'; -import { SyncStatusV2 } from '~/console/components/sync-status'; -import ListV2 from '~/console/components/listV2'; -import { IClusterMSvs } from '~/console/server/gql/queries/cluster-managed-services-queries'; -import { useWatchReload } from '~/root/lib/client/helpers/socket/useWatch'; -import { useState } from 'react'; -import DeleteDialog from '~/console/components/delete-dialog'; -import { useConsoleApi } from '~/console/server/gql/api-provider'; import { useReload } from '~/root/lib/client/helpers/reloader'; -import { toast } from '~/components/molecule/toast'; +import { useWatchReload } from '~/root/lib/client/helpers/socket/useWatch'; import { handleError } from '~/root/lib/utils/common'; -import { Badge } from '~/components/atoms/badge'; -import useClusterStatus from '~/console/hooks/use-cluster-status'; import { IAccountContext } from '../_layout'; import { IClusterContext } from '../infra+/$cluster+/_layout'; import CloneManagedService from './clone-managed-service'; @@ -161,8 +161,20 @@ const GridView = ({ items, templates, onAction }: IResource) => { }; const ListView = ({ items, templates, onAction }: IResource) => { - const { account, clustersMap } = useOutletContext(); + const { account } = useOutletContext(); const { findClusterStatus, clusters, loading } = useClusterStatus(); + + const [clusterOnlineStatus, setClusterOnlineStatus] = useState< + Record + >({}); + useEffect(() => { + const states: Record = {}; + clusters.forEach((c) => { + states[c.metadata.name] = findClusterStatus(c); + }); + setClusterOnlineStatus(states); + }, [clusters]); + return ( { }, ], rows: items.map((i) => { - const isClusterOnline = findClusterStatus( - clusters.length > 0 - ? clusters.find((c) => parseName(c) === i.clusterName) - : clustersMap[i.clusterName] - ); + const isClusterOnline = clusterOnlineStatus[i.clusterName]; const { name, id, logo, updateInfo } = parseItem(i, templates); return { columns: { diff --git a/src/apps/console/server/gql/queries/environment-queries.ts b/src/apps/console/server/gql/queries/environment-queries.ts index 481aeb903..4d0d0a31d 100644 --- a/src/apps/console/server/gql/queries/environment-queries.ts +++ b/src/apps/console/server/gql/queries/environment-queries.ts @@ -60,6 +60,7 @@ export const environmentQueries = (executor: IExecutor) => ({ privateIngressClass publicIngressClass } + suspend targetNamespace } status { @@ -172,6 +173,7 @@ export const environmentQueries = (executor: IExecutor) => ({ privateIngressClass publicIngressClass } + suspend targetNamespace } status { diff --git a/src/generated/gql/sdl.graphql b/src/generated/gql/sdl.graphql index a59a83361..0c4045f64 100644 --- a/src/generated/gql/sdl.graphql +++ b/src/generated/gql/sdl.graphql @@ -1540,11 +1540,13 @@ enum Github__com___kloudlite___operator___apis___crds___v1__EnvironmentRoutingMo type Github__com___kloudlite___operator___apis___crds___v1__EnvironmentSpec { routing: Github__com___kloudlite___operator___apis___crds___v1__EnvironmentRouting + suspend: Boolean targetNamespace: String } input Github__com___kloudlite___operator___apis___crds___v1__EnvironmentSpecIn { routing: Github__com___kloudlite___operator___apis___crds___v1__EnvironmentRoutingIn + suspend: Boolean targetNamespace: String } diff --git a/src/generated/gql/server.ts b/src/generated/gql/server.ts index dd36c5a82..66be5860d 100644 --- a/src/generated/gql/server.ts +++ b/src/generated/gql/server.ts @@ -716,6 +716,7 @@ export type EnvironmentIn = { export type Github__Com___Kloudlite___Operator___Apis___Crds___V1__EnvironmentSpecIn = { routing?: InputMaybe; + suspend?: InputMaybe; targetNamespace?: InputMaybe; }; @@ -2470,6 +2471,7 @@ export type ConsoleGetEnvironmentQuery = { namespace?: string; }; spec?: { + suspend?: boolean; targetNamespace?: string; routing?: { mode?: Github__Com___Kloudlite___Operator___Apis___Crds___V1__EnvironmentRoutingMode; @@ -2545,6 +2547,7 @@ export type ConsoleListEnvironmentsQuery = { lastUpdatedBy: { userEmail: string; userId: string; userName: string }; metadata?: { generation: number; name: string; namespace?: string }; spec?: { + suspend?: boolean; targetNamespace?: string; routing?: { mode?: Github__Com___Kloudlite___Operator___Apis___Crds___V1__EnvironmentRoutingMode; From 37aa2ebee97bf805a8a8ab26ef77069e4522891b Mon Sep 17 00:00:00 2001 From: Bikash Date: Wed, 28 Aug 2024 18:46:59 +0545 Subject: [PATCH 2/6] updated cluster status hook --- .../console/hooks/use-cluster-status-v2.tsx | 82 +++++++++++++++++++ src/apps/console/hooks/use-cluster-status.tsx | 4 +- .../routes/_main+/$account+/_layout.tsx | 32 +++++--- .../environments/environment-resources-v2.tsx | 43 +++++----- .../infra+/clusters/cluster-resources-v2.tsx | 10 ++- .../console/routes/_main+/_layout/_layout.tsx | 33 ++++---- 6 files changed, 156 insertions(+), 48 deletions(-) create mode 100644 src/apps/console/hooks/use-cluster-status-v2.tsx diff --git a/src/apps/console/hooks/use-cluster-status-v2.tsx b/src/apps/console/hooks/use-cluster-status-v2.tsx new file mode 100644 index 000000000..598267437 --- /dev/null +++ b/src/apps/console/hooks/use-cluster-status-v2.tsx @@ -0,0 +1,82 @@ +import { + Dispatch, + ReactNode, + SetStateAction, + createContext, + useCallback, + useContext, + useMemo, + useState, +} from 'react'; +import { ExtractNodeType, parseNodes } from '../server/r-utils/common'; +import { IByocClusters } from '../server/gql/queries/byok-cluster-queries'; +import { useParams } from '@remix-run/react'; +import { useSocketWatch } from '~/root/lib/client/helpers/socket/useWatch'; +import { useConsoleApi } from '../server/gql/api-provider'; +import useDebounce from '~/root/lib/client/hooks/use-debounce'; + +type IClusterMap = { [key: string]: ExtractNodeType }; + +const ClusterStatusContext = createContext<{ + clusters: IClusterMap; + setClusters: Dispatch>; +}>({ clusters: {}, setClusters: () => {} }); + +const ClusterStatusProvider = ({ children }: { children: ReactNode }) => { + const [clusters, setClusters] = useState({}); + const api = useConsoleApi(); + const [update, setUpdate] = useState(false); + + const { account } = useParams(); + + const topic = useCallback(() => { + return Object.keys(clusters).map((c) => `account:${account}.cluster:${c}`); + }, [clusters])(); + + const listCluster = useCallback(async () => { + try { + const cl = await api.listAllClusters(); + const parsed = parseNodes(cl.data).reduce( + (acc, c) => { + acc[c.metadata.name] = c; + return acc; + }, + {} as { [key: string]: ExtractNodeType }, + ); + setClusters(parsed); + return clusters; + } catch (err) { + console.error(err); + return false; + } + }, []); + + useDebounce( + () => { + listCluster(); + }, + 3000, + [update], + ); + + useSocketWatch(() => { + setUpdate((p) => !p); + }, topic); + + return ( + ({ clusters, setClusters }), + [clusters, setClusters], + )} + > + {children} + + ); +}; + +export default ClusterStatusProvider; + +export const useClusterStatusV2 = () => { + return useContext(ClusterStatusContext); +}; diff --git a/src/apps/console/hooks/use-cluster-status.tsx b/src/apps/console/hooks/use-cluster-status.tsx index ff28c1449..a825a5066 100644 --- a/src/apps/console/hooks/use-cluster-status.tsx +++ b/src/apps/console/hooks/use-cluster-status.tsx @@ -2,7 +2,9 @@ import { useCallback, useEffect, useState } from 'react'; import { useConsoleApi } from '../server/gql/api-provider'; import { parseNodes } from '../server/r-utils/common'; -const findClusterStatus = (item?: { lastOnlineAt?: string }): boolean => { +export const findClusterStatus = (item?: { + lastOnlineAt?: string; +}): boolean => { if (!item || !item.lastOnlineAt) { return false; } diff --git a/src/apps/console/routes/_main+/$account+/_layout.tsx b/src/apps/console/routes/_main+/$account+/_layout.tsx index db421d465..800590cb0 100644 --- a/src/apps/console/routes/_main+/$account+/_layout.tsx +++ b/src/apps/console/routes/_main+/$account+/_layout.tsx @@ -51,6 +51,7 @@ import { useSearch } from '~/root/lib/client/helpers/search-filter'; import { IMSvTemplates } from '~/console/server/gql/queries/managed-templates-queries'; import { IByocClusters } from '~/console/server/gql/queries/byok-cluster-queries'; import { IConsoleRootContext } from '../_layout/_layout'; +import { useClusterStatusV2 } from '~/console/hooks/use-cluster-status-v2'; export const loader = async (ctx: IRemixCtx) => { const { account } = ctx.params; @@ -66,14 +67,14 @@ export const loader = async (ctx: IRemixCtx) => { } const { data: msvTemplates, errors: msvError } = await GQLServerHandler( - ctx.request + ctx.request, ).listMSvTemplates({}); if (msvError) { throw msvError[0]; } const { data: clusterList, errors: clusterError } = await GQLServerHandler( - ctx.request + ctx.request, ).listByokClusters({ pagination: { first: 100, @@ -84,10 +85,13 @@ export const loader = async (ctx: IRemixCtx) => { throw clusterError[0]; } - const cMaps = parseNodes(clusterList).reduce((acc, c) => { - acc[c.metadata.name] = c; - return acc; - }, {} as { [key: string]: ExtractNodeType }); + const cMaps = parseNodes(clusterList).reduce( + (acc, c) => { + acc[c.metadata.name] = c; + return acc; + }, + {} as { [key: string]: ExtractNodeType }, + ); acccountData = data; return { @@ -187,6 +191,14 @@ const Account = () => { useEffect(() => { ensureAccountClientSide(params); }, []); + + const { setClusters } = useClusterStatusV2(); + + useEffect(() => { + // @ts-ignore + setClusters(clustersMap); + }, [clustersMap]); + return ( <> { const { data: accounts } = useCustomSwr( () => '/accounts', - async () => api.listAccounts({}) + async () => api.listAccounts({}), ); const [searchText, setSearchText] = useState(''); @@ -387,7 +399,7 @@ const CurrentBreadcrum = ({ account }: { account: IAccount }) => { searchText, keys: ['searchField'], }, - [searchText, accounts] + [searchText, accounts], ); const [open, setOpen] = useState(false); @@ -422,7 +434,7 @@ const CurrentBreadcrum = ({ account }: { account: IAccount }) => { aria-label="accounts" className={cn( 'outline-none rounded py-lg px-md mx-md bg-surface-basic-hovered', - open || isMouseOver ? 'bg-surface-basic-pressed' : '' + open || isMouseOver ? 'bg-surface-basic-pressed' : '', )} onMouseOver={() => { setIsMouseOver(true); @@ -470,7 +482,7 @@ const CurrentBreadcrum = ({ account }: { account: IAccount }) => { 'flex flex-row items-center justify-between', parseName(item) === parseName(account) ? 'bg-surface-basic-pressed hover:!bg-surface-basic-pressed' - : '' + : '', )} > {item.displayName} diff --git a/src/apps/console/routes/_main+/$account+/environments/environment-resources-v2.tsx b/src/apps/console/routes/_main+/$account+/environments/environment-resources-v2.tsx index 5f713aa71..c9a325f64 100644 --- a/src/apps/console/routes/_main+/$account+/environments/environment-resources-v2.tsx +++ b/src/apps/console/routes/_main+/$account+/environments/environment-resources-v2.tsx @@ -20,7 +20,9 @@ import ResourceExtraAction, { IResourceExtraItem, } from '~/console/components/resource-extra-action'; import { SyncStatusV2 } from '~/console/components/sync-status'; -import useClusterStatus from '~/console/hooks/use-cluster-status'; +import useClusterStatus, { + findClusterStatus, +} from '~/console/hooks/use-cluster-status'; import { IAccountContext } from '~/console/routes/_main+/$account+/_layout'; import { useConsoleApi } from '~/console/server/gql/api-provider'; import { IEnvironments } from '~/console/server/gql/queries/environment-queries'; @@ -34,6 +36,7 @@ import { useWatchReload } from '~/lib/client/helpers/socket/useWatch'; import { useReload } from '~/root/lib/client/helpers/reloader'; import { handleError } from '~/root/lib/utils/common'; import CloneEnvironment from './clone-environment'; +import { useClusterStatusV2 } from '~/console/hooks/use-cluster-status-v2'; const RESOURCE_NAME = 'environment'; type BaseType = ExtractNodeType; @@ -192,18 +195,19 @@ const GridView = ({ items = [], onAction }: IResource) => { const ListView = ({ items, onAction }: IResource) => { const { account } = useParams(); - const { findClusterStatus, clusters, loading } = useClusterStatus(); + // const { findClusterStatus } = useClusterStatus(); + const { clusters } = useClusterStatusV2(); - const [clusterOnlineStatus, setClusterOnlineStatus] = useState< - Record - >({}); - useEffect(() => { - const states: Record = {}; - clusters.forEach((c) => { - states[c.metadata.name] = findClusterStatus(c); - }); - setClusterOnlineStatus(states); - }, [clusters]); + // const [clusterOnlineStatus, setClusterOnlineStatus] = useState< + // Record + // >({}); + // useEffect(() => { + // const states: Record = {}; + // clusters.forEach((c) => { + // states[c.metadata.name] = findClusterStatus(c); + // }); + // setClusterOnlineStatus(states); + // }, [clusters]); return ( { ], rows: items.map((i) => { const { name, id, updateInfo } = parseItem(i); - const isClusterOnline = clusterOnlineStatus[i.clusterName]; + // const isClusterOnline = clusterOnlineStatus[i.clusterName];= + const isClusterOnline = findClusterStatus(clusters[i.clusterName]); return { columns: { @@ -266,9 +271,9 @@ const ListView = ({ items, onAction }: IResource) => { if (i.isArchived) { return Archived; } - if (loading) { - return null; - } + // if (loading) { + // return null; + // } if (!isClusterOnline) { return Cluster Offline; @@ -314,7 +319,7 @@ const EnvironmentResourcesV2 = ({ items = [] }: { items: BaseType[] }) => { useWatchReload( items.map((i) => { return `account:${parseName(account)}.environment:${parseName(i)}`; - }) + }), ); const suspendEnvironment = async (item: BaseType, suspend: boolean) => { @@ -340,7 +345,7 @@ const EnvironmentResourcesV2 = ({ items = [] }: { items: BaseType[] }) => { suspend ? 'Environment suspended successfully' : 'Environment resumed successfully' - }` + }`, ); reloadPage(); } catch (err) { @@ -349,7 +354,7 @@ const EnvironmentResourcesV2 = ({ items = [] }: { items: BaseType[] }) => { }; const [showDeleteDialog, setShowDeleteDialog] = useState( - null + null, ); const [visible, setVisible] = useState(null); diff --git a/src/apps/console/routes/_main+/$account+/infra+/clusters/cluster-resources-v2.tsx b/src/apps/console/routes/_main+/$account+/infra+/clusters/cluster-resources-v2.tsx index cfcf149cb..d572bf860 100644 --- a/src/apps/console/routes/_main+/$account+/infra+/clusters/cluster-resources-v2.tsx +++ b/src/apps/console/routes/_main+/$account+/infra+/clusters/cluster-resources-v2.tsx @@ -51,6 +51,7 @@ import { ViewClusterLogs } from '~/console/components/cluster-logs-popop'; import { ensureAccountClientSide } from '~/console/server/utils/auth-utils'; import TooltipV2 from '~/components/atoms/tooltipV2'; import HandleByokCluster from '../byok-cluster/handle-byok-cluster'; +import { useClusterStatusV2 } from '~/console/hooks/use-cluster-status-v2'; type BaseType = ExtractNodeType & { type: 'normal' }; type ByokBaseType = ExtractNodeType & { type: 'byok' }; @@ -120,7 +121,7 @@ const ByokInstructionsPopup = ({ return api.getBYOKClusterInstructions({ name: item.metadata.name, }); - } + }, ); return ( @@ -415,6 +416,7 @@ const GridView = ({ items = [], onEdit, onDelete, onShowLogs }: IResource) => { }; const ListView = ({ items = [], onEdit, onDelete, onShowLogs }: IResource) => { const { account } = useParams(); + const { clusters } = useClusterStatusV2(); return ( { ), }, status: { - render: () => , + render: () => ( + + ), }, updated: { render: () => ( @@ -530,7 +534,7 @@ const ClusterResourcesV2 = ({ useWatchReload( bItems.map((i) => { return `account:${parseName(account)}.cluster:${parseName(i)}`; - }) + }), ); const [showDeleteDialog, setShowDeleteDialog] = diff --git a/src/apps/console/routes/_main+/_layout/_layout.tsx b/src/apps/console/routes/_main+/_layout/_layout.tsx index 1ef207f45..a7119a5e3 100644 --- a/src/apps/console/routes/_main+/_layout/_layout.tsx +++ b/src/apps/console/routes/_main+/_layout/_layout.tsx @@ -49,6 +49,7 @@ import { ExtractNodeType, parseNodes } from '~/console/server/r-utils/common'; import { ICommsNotifications } from '~/console/server/gql/queries/comms-queries'; import { LoadingPlaceHolder } from '~/console/components/loading'; import logger from '~/root/lib/client/helpers/log'; +import ClusterStatusProvider from '~/console/hooks/use-cluster-status-v2'; const restActions = (ctx: IExtRemixCtx) => { return withContext(ctx, {}); @@ -304,7 +305,7 @@ const NotificationMenu = () => { first: 100, }, }), - true + true, ); const notifications = parseNodes(notificationsData); @@ -477,7 +478,7 @@ const Console = () => { {breadcrum.map((bc: any, index) => cloneElement(bc.handle.breadcrum(bc), { key: generateKey(index), - }) + }), )} ) @@ -494,19 +495,21 @@ const Console = () => { } /> - - - - - - - - - + + + + + + + + + + + ); }; From 88bd72f4f8e9b1502b805926538a2fdbaa54e0a6 Mon Sep 17 00:00:00 2001 From: Piyush Kumar Date: Thu, 29 Aug 2024 19:00:12 +0530 Subject: [PATCH 3/6] minor ui changes --- gql-queries-generator/doc/queries.graphql | 45 +++ .../console/hooks/use-cluster-status-v2.tsx | 23 +- .../$account+/env+/$environment+/_layout.tsx | 25 +- .../$environment+/apps/apps-resources-v2.tsx | 13 +- .../handle-managed-resource-v2.tsx | 58 ++- .../managed-resources-resource-v2.tsx | 22 +- .../$environment+/managed-resources/route.tsx | 30 +- .../environments/environment-resources-v2.tsx | 39 +- .../infra+/clusters/cluster-resources-v2.tsx | 40 +-- .../backend-services-resources-V2.tsx | 13 +- .../handle-backend-service.tsx | 26 +- .../$account+/managed-services/route.tsx | 26 +- .../_main+/$account+/msvc+/$msv+/_layout.tsx | 24 +- .../msvc+/$msv+/managed-resources/route.tsx | 26 +- .../$msv+/new-managed-resource/_index.tsx | 54 ++- .../$account+/new-managed-service/_index.tsx | 14 +- .../image-pull-secrets-resource-v2.tsx | 22 +- .../ips+/$imagepullsecret+/_layout.tsx | 169 +++++++++ .../images/images-resources.tsx | 340 ++++++++++++++++++ .../ips+/$imagepullsecret+/images/route.tsx | 92 +++++ .../ips+/$imagepullsecret+/images/tools.tsx | 29 ++ .../ips+/$imagepullsecret+/index.tsx | 5 + .../console/routes/_main+/_layout/_layout.tsx | 42 +-- .../gql/queries/byok-cluster-queries.ts | 8 +- .../gql/queries/image-pull-secrets-queries.ts | 66 +++- .../components/atoms/progress-bar.tsx | 2 +- src/generated/gql/server.ts | 41 +++ 27 files changed, 1020 insertions(+), 274 deletions(-) create mode 100644 src/apps/console/routes/_main+/$account+/settings+/ips+/$imagepullsecret+/_layout.tsx create mode 100644 src/apps/console/routes/_main+/$account+/settings+/ips+/$imagepullsecret+/images/images-resources.tsx create mode 100644 src/apps/console/routes/_main+/$account+/settings+/ips+/$imagepullsecret+/images/route.tsx create mode 100644 src/apps/console/routes/_main+/$account+/settings+/ips+/$imagepullsecret+/images/tools.tsx create mode 100644 src/apps/console/routes/_main+/$account+/settings+/ips+/$imagepullsecret+/index.tsx diff --git a/gql-queries-generator/doc/queries.graphql b/gql-queries-generator/doc/queries.graphql index 83340bfd3..566e9f6b6 100644 --- a/gql-queries-generator/doc/queries.graphql +++ b/gql-queries-generator/doc/queries.graphql @@ -3992,6 +3992,51 @@ mutation consoleDeleteImagePullSecrets($name: String!) { core_deleteImagePullSecret(name: $name) } +query consoleGetImagePullSecret($name: String!) { + core_getImagePullSecret(name: $name) { + accountName + createdBy { + userEmail + userId + userName + } + creationTime + displayName + dockerConfigJson + environments + format + id + lastUpdatedBy { + userEmail + userId + userName + } + markedForDeletion + metadata { + annotations + creationTimestamp + deletionTimestamp + generation + labels + name + namespace + } + recordVersion + registryPassword + registryURL + registryUsername + syncStatus { + action + error + lastSyncedAt + recordVersion + state + syncScheduledAt + } + updateTime + } +} + query consoleListImagePullSecrets($search: SearchImagePullSecrets, $pq: CursorPaginationIn) { core_listImagePullSecrets(search: $search, pq: $pq) { edges { diff --git a/src/apps/console/hooks/use-cluster-status-v2.tsx b/src/apps/console/hooks/use-cluster-status-v2.tsx index 598267437..c2e795abc 100644 --- a/src/apps/console/hooks/use-cluster-status-v2.tsx +++ b/src/apps/console/hooks/use-cluster-status-v2.tsx @@ -1,3 +1,4 @@ +import { useParams } from '@remix-run/react'; import { Dispatch, ReactNode, @@ -8,12 +9,11 @@ import { useMemo, useState, } from 'react'; -import { ExtractNodeType, parseNodes } from '../server/r-utils/common'; -import { IByocClusters } from '../server/gql/queries/byok-cluster-queries'; -import { useParams } from '@remix-run/react'; import { useSocketWatch } from '~/root/lib/client/helpers/socket/useWatch'; -import { useConsoleApi } from '../server/gql/api-provider'; import useDebounce from '~/root/lib/client/hooks/use-debounce'; +import { useConsoleApi } from '../server/gql/api-provider'; +import { IByocClusters } from '../server/gql/queries/byok-cluster-queries'; +import { ExtractNodeType, parseNodes } from '../server/r-utils/common'; type IClusterMap = { [key: string]: ExtractNodeType }; @@ -36,13 +36,10 @@ const ClusterStatusProvider = ({ children }: { children: ReactNode }) => { const listCluster = useCallback(async () => { try { const cl = await api.listAllClusters(); - const parsed = parseNodes(cl.data).reduce( - (acc, c) => { - acc[c.metadata.name] = c; - return acc; - }, - {} as { [key: string]: ExtractNodeType }, - ); + const parsed = parseNodes(cl.data).reduce((acc, c) => { + acc[c.metadata.name] = c; + return acc; + }, {} as { [key: string]: ExtractNodeType }); setClusters(parsed); return clusters; } catch (err) { @@ -56,7 +53,7 @@ const ClusterStatusProvider = ({ children }: { children: ReactNode }) => { listCluster(); }, 3000, - [update], + [update] ); useSocketWatch(() => { @@ -67,7 +64,7 @@ const ClusterStatusProvider = ({ children }: { children: ReactNode }) => { ({ clusters, setClusters }), - [clusters, setClusters], + [clusters, setClusters] )} > {children} diff --git a/src/apps/console/routes/_main+/$account+/env+/$environment+/_layout.tsx b/src/apps/console/routes/_main+/$account+/env+/$environment+/_layout.tsx index 7f1ec3995..f1f694714 100644 --- a/src/apps/console/routes/_main+/$account+/env+/$environment+/_layout.tsx +++ b/src/apps/console/routes/_main+/$account+/env+/$environment+/_layout.tsx @@ -1,10 +1,3 @@ -import { - BackingServices, - CirclesFour, - GearSix, - File, - // TreeStructure, -} from '~/console/components/icons'; import { Link, Outlet, @@ -13,20 +6,26 @@ import { useParams, } from '@remix-run/react'; import { useState } from 'react'; +import Breadcrum from '~/console/components/breadcrum'; import { CommonTabs } from '~/console/components/common-navbar-tabs'; +import { + BackingServices, + CirclesFour, + File, + GearSix, +} from '~/console/components/icons'; import HandleScope from '~/console/page-components/handle-environment'; +import { ICluster } from '~/console/server/gql/queries/cluster-queries'; +import { IEnvironment } from '~/console/server/gql/queries/environment-queries'; +import { ILoginUrls, ILogins } from '~/console/server/gql/queries/git-queries'; import { GQLServerHandler } from '~/console/server/gql/saved-queries'; import { parseName } from '~/console/server/r-utils/common'; import { ensureAccountSet } from '~/console/server/utils/auth-utils'; +import { BreadcrumSlash, tabIconSize } from '~/console/utils/commons'; import { SubNavDataProvider } from '~/lib/client/hooks/use-create-subnav-action'; import { IRemixCtx, LoaderResult } from '~/lib/types/common'; -import { BreadcrumSlash, tabIconSize } from '~/console/utils/commons'; -import { IEnvironment } from '~/console/server/gql/queries/environment-queries'; -import { ILoginUrls, ILogins } from '~/console/server/gql/queries/git-queries'; import logger from '~/root/lib/client/helpers/log'; -import Breadcrum from '~/console/components/breadcrum'; import { handleError } from '~/root/lib/utils/common'; -import { ICluster } from '~/console/server/gql/queries/cluster-queries'; import { IAccountContext } from '../../_layout'; const Environment = () => { @@ -65,7 +64,7 @@ const tabs = [ label: ( - Integrated Resources + Imported Managed Resources ), to: '/managed-resources', diff --git a/src/apps/console/routes/_main+/$account+/env+/$environment+/apps/apps-resources-v2.tsx b/src/apps/console/routes/_main+/$account+/env+/$environment+/apps/apps-resources-v2.tsx index e169367d5..e75a9c675 100644 --- a/src/apps/console/routes/_main+/$account+/env+/$environment+/apps/apps-resources-v2.tsx +++ b/src/apps/console/routes/_main+/$account+/env+/$environment+/apps/apps-resources-v2.tsx @@ -25,7 +25,8 @@ import ResourceExtraAction, { IResourceExtraItem, } from '~/console/components/resource-extra-action'; import { SyncStatusV2 } from '~/console/components/sync-status'; -import useClusterStatus from '~/console/hooks/use-cluster-status'; +import { findClusterStatus } from '~/console/hooks/use-cluster-status'; +import { useClusterStatusV2 } from '~/console/hooks/use-cluster-status-v2'; import { useConsoleApi } from '~/console/server/gql/api-provider'; import { IApps } from '~/console/server/gql/queries/app-queries'; import { @@ -240,15 +241,15 @@ const GridView = ({ items = [], onAction: _ }: IResource) => { const ListView = ({ items = [], onAction }: IResource) => { const { environment, account, cluster } = useOutletContext(); - const { findClusterStatus, clusters, loading } = useClusterStatus(); + const { clusters } = useClusterStatusV2(); const [clusterOnlineStatus, setClusterOnlineStatus] = useState< Record >({}); useEffect(() => { const states: Record = {}; - clusters.forEach((c) => { - states[c.metadata.name] = findClusterStatus(c); + Object.entries(clusters).forEach(([key, value]) => { + states[key] = findClusterStatus(value); }); setClusterOnlineStatus(states); }, [clusters]); @@ -329,10 +330,6 @@ const ListView = ({ items = [], onAction }: IResource) => { }, status: { render: () => { - if (loading) { - return null; - } - if (environment.spec?.suspend) { return null; } diff --git a/src/apps/console/routes/_main+/$account+/env+/$environment+/managed-resources/handle-managed-resource-v2.tsx b/src/apps/console/routes/_main+/$account+/env+/$environment+/managed-resources/handle-managed-resource-v2.tsx index 8366869e6..a52e428e2 100644 --- a/src/apps/console/routes/_main+/$account+/env+/$environment+/managed-resources/handle-managed-resource-v2.tsx +++ b/src/apps/console/routes/_main+/$account+/env+/$environment+/managed-resources/handle-managed-resource-v2.tsx @@ -1,30 +1,30 @@ /* eslint-disable react/destructuring-assignment */ +import { useOutletContext, useParams } from '@remix-run/react'; +import { useCallback, useEffect, useState } from 'react'; +import Select from '~/components/atoms/select'; import Popup from '~/components/molecule/popup'; -import { useReload } from '~/root/lib/client/helpers/reloader'; -import useForm, { dummyEvent } from '~/root/lib/client/hooks/use-form'; -import Yup from '~/root/lib/server/helpers/yup'; -import { handleError } from '~/root/lib/utils/common'; +import { toast } from '~/components/molecule/toast'; +import { useMapper } from '~/components/utils'; +import { CopyContentToClipboard } from '~/console/components/common-console-components'; +import { ListItem } from '~/console/components/console-list-components'; +import ListV2 from '~/console/components/listV2'; +import { LoadingPlaceHolder } from '~/console/components/loading'; +import MultiStep, { useMultiStep } from '~/console/components/multi-step'; +import { NameIdView } from '~/console/components/name-id-view'; import { IDialogBase } from '~/console/components/types.d'; +import { useConsoleApi } from '~/console/server/gql/api-provider'; +import { IImportedManagedResources } from '~/console/server/gql/queries/imported-managed-resource-queries'; import { ExtractNodeType, parseName, parseNodes, } from '~/console/server/r-utils/common'; -import Select from '~/components/atoms/select'; -import { useConsoleApi } from '~/console/server/gql/api-provider'; -import { useOutletContext, useParams } from '@remix-run/react'; -import useCustomSwr from '~/root/lib/client/hooks/use-custom-swr'; -import { useMapper } from '~/components/utils'; -import MultiStep, { useMultiStep } from '~/console/components/multi-step'; -import { LoadingPlaceHolder } from '~/console/components/loading'; -import ListV2 from '~/console/components/listV2'; -import { ListItem } from '~/console/components/console-list-components'; -import { CopyContentToClipboard } from '~/console/components/common-console-components'; -import { useCallback, useEffect, useState } from 'react'; -import { toast } from '~/components/molecule/toast'; import { ensureAccountClientSide } from '~/console/server/utils/auth-utils'; -import { NameIdView } from '~/console/components/name-id-view'; -import { IImportedManagedResources } from '~/console/server/gql/queries/imported-managed-resource-queries'; +import { useReload } from '~/root/lib/client/helpers/reloader'; +import useCustomSwr from '~/root/lib/client/hooks/use-custom-swr'; +import useForm, { dummyEvent } from '~/root/lib/client/hooks/use-form'; +import Yup from '~/root/lib/server/helpers/yup'; +import { handleError } from '~/root/lib/utils/common'; import { IEnvironmentContext } from '../_layout'; type BaseType = ExtractNodeType; @@ -90,11 +90,9 @@ const Root = (props: IDialog) => { managedResourceName: '', }, validationSchema: Yup.object({ - managedServiceName: Yup.string().required( - 'integrated service is required' - ), + managedServiceName: Yup.string().required('Managed service is required'), managedResourceName: Yup.string().required( - 'integrated resource name is required' + 'Managed resource name is required' ), }), onSubmit: async (val) => { @@ -113,9 +111,7 @@ const Root = (props: IDialog) => { reloadPage(); resetValues(); toast.success( - `integrated resource ${ - isUpdate ? 'updated' : 'imported' - } successfully` + `Managed resource ${isUpdate ? 'updated' : 'imported'} successfully` ); setVisible(false); } catch (err) { @@ -171,7 +167,7 @@ const Root = (props: IDialog) => {
{ /> [ ...((mresList && mresList.filter((mres) => { @@ -243,7 +239,7 @@ const HandleManagedResourceV2 = (props: IDialog) => { return ( setVisible(v)}> - {isUpdate ? 'Edit External Name' : 'Import Integrated Resource'} + {isUpdate ? 'Edit External Name' : 'Import Managed Resource'} {(!isUpdate || (isUpdate && props.data)) && } diff --git a/src/apps/console/routes/_main+/$account+/env+/$environment+/managed-resources/managed-resources-resource-v2.tsx b/src/apps/console/routes/_main+/$account+/env+/$environment+/managed-resources/managed-resources-resource-v2.tsx index 54a3f506e..4c2bc2ed9 100644 --- a/src/apps/console/routes/_main+/$account+/env+/$environment+/managed-resources/managed-resources-resource-v2.tsx +++ b/src/apps/console/routes/_main+/$account+/env+/$environment+/managed-resources/managed-resources-resource-v2.tsx @@ -16,7 +16,8 @@ import { LockSimple, Trash } from '~/console/components/icons'; import ListGridView from '~/console/components/list-grid-view'; import ListV2 from '~/console/components/listV2'; import ResourceExtraAction from '~/console/components/resource-extra-action'; -import useClusterStatus from '~/console/hooks/use-cluster-status'; +import { findClusterStatus } from '~/console/hooks/use-cluster-status'; +import { useClusterStatusV2 } from '~/console/hooks/use-cluster-status-v2'; import { useConsoleApi } from '~/console/server/gql/api-provider'; import { IImportedManagedResources } from '~/console/server/gql/queries/imported-managed-resource-queries'; import { IMSvTemplates } from '~/console/server/gql/queries/managed-templates-queries'; @@ -33,7 +34,7 @@ import { handleError } from '~/lib/utils/common'; import { IEnvironmentContext } from '../_layout'; import { ViewSecret } from './handle-managed-resource-v2'; -const RESOURCE_NAME = 'integrated resource'; +const RESOURCE_NAME = 'managed resource'; type BaseType = ExtractNodeType; const parseItem = (item: BaseType, templates: IMSvTemplates) => { @@ -69,13 +70,6 @@ const ExtraButton = ({ onAction, item }: IExtraButton) => { return ( , - // type: 'item', - // onClick: () => onAction({ action: 'edit', item }), - // key: 'edit', - // }, { label: 'View Secret', icon: , @@ -141,15 +135,15 @@ const GridView = ({ items = [], onAction, templates }: IResource) => { const ListView = ({ items = [], onAction, templates }: IResource) => { const { cluster } = useOutletContext(); - const { findClusterStatus, clusters, loading } = useClusterStatus(); + const { clusters } = useClusterStatusV2(); const [clusterOnlineStatus, setClusterOnlineStatus] = useState< Record >({}); useEffect(() => { const states: Record = {}; - clusters.forEach((c) => { - states[c.metadata.name] = findClusterStatus(c); + Object.entries(clusters).forEach(([key, value]) => { + states[key] = findClusterStatus(value); }); setClusterOnlineStatus(states); }, [clusters]); @@ -236,10 +230,6 @@ const ListView = ({ items = [], onAction, templates }: IResource) => { }, status: { render: () => { - if (loading) { - return null; - } - if (!isClusterOnline) { return Cluster Offline; } diff --git a/src/apps/console/routes/_main+/$account+/env+/$environment+/managed-resources/route.tsx b/src/apps/console/routes/_main+/$account+/env+/$environment+/managed-resources/route.tsx index 5b2c9115f..2a92143c4 100644 --- a/src/apps/console/routes/_main+/$account+/env+/$environment+/managed-resources/route.tsx +++ b/src/apps/console/routes/_main+/$account+/env+/$environment+/managed-resources/route.tsx @@ -1,21 +1,21 @@ -import { Plus } from '~/console/components/icons'; import { defer } from '@remix-run/node'; import { Link, useLoaderData, useOutletContext } from '@remix-run/react'; +import { useState } from 'react'; +import { Button } from '~/components/atoms/button'; +import { EmptyManagedResourceImage } from '~/console/components/empty-resource-images'; +import { Plus } from '~/console/components/icons'; import { LoadingComp, pWrapper } from '~/console/components/loading-component'; import Wrapper from '~/console/components/wrapper'; +import { IAccountContext } from '~/console/routes/_main+/$account+/_layout'; import { GQLServerHandler } from '~/console/server/gql/saved-queries'; import { parseNodes } from '~/console/server/r-utils/common'; +import { ensureAccountSet } from '~/console/server/utils/auth-utils'; +import { getPagination, getSearch } from '~/console/server/utils/common'; import { IRemixCtx } from '~/lib/types/common'; import fake from '~/root/fake-data-generator/fake'; -import { Button } from '~/components/atoms/button'; -import { useState } from 'react'; -import { IAccountContext } from '~/console/routes/_main+/$account+/_layout'; -import { EmptyManagedResourceImage } from '~/console/components/empty-resource-images'; -import { getPagination, getSearch } from '~/console/server/utils/common'; -import { ensureAccountSet } from '~/console/server/utils/auth-utils'; -import Tools from './tools'; -import ManagedResourceResourcesV2 from './managed-resources-resource-v2'; import HandleManagedResourceV2 from './handle-managed-resource-v2'; +import ManagedResourceResourcesV2 from './managed-resources-resource-v2'; +import Tools from './tools'; export const loader = (ctx: IRemixCtx) => { const { environment } = ctx.params; @@ -59,11 +59,11 @@ const KlOperatorServices = () => { return ( 0 && (
+ ); +}; + +export const handle = ({ + promise: { imagePullSecret, error }, +}: { + promise: any; +}) => { + if (error) { + return {}; + } + + return { + navbar: , + breadcrum: () => , + }; +}; + +export interface IImagePullSecretContext extends IAccountContext { + imagepullsecret: IImagePullSecret; +} + +const IPSOutlet = ({ + imagePullSecret: OImagePullSecret, +}: { + imagePullSecret: IImagePullSecret; +}) => { + const rootContext = useOutletContext(); + + return ( + + ); +}; + +export const loader = async (ctx: IRemixCtx) => { + const promise = pWrapper(async () => { + ensureAccountSet(ctx); + const { imagepullsecret } = ctx.params; + try { + const { data, errors } = await GQLServerHandler( + ctx.request + ).getImagePullSecret({ + name: imagepullsecret, + }); + if (errors) { + throw errors[0]; + } + + return { + imagePullSecret: data, + }; + } catch (err) { + logger.log(err); + + return { + imagePullSecret: {} as IImagePullSecret, + redirect: `../image-pull-secrets`, + }; + } + }); + return defer({ promise: await promise }); +}; + +const ImagePullSecret = () => { + const { promise } = useLoaderData(); + return ( + + {({ imagePullSecret }) => { + return ; + }} + + ); +}; + +export default ImagePullSecret; diff --git a/src/apps/console/routes/_main+/$account+/settings+/ips+/$imagepullsecret+/images/images-resources.tsx b/src/apps/console/routes/_main+/$account+/settings+/ips+/$imagepullsecret+/images/images-resources.tsx new file mode 100644 index 000000000..c84643f12 --- /dev/null +++ b/src/apps/console/routes/_main+/$account+/settings+/ips+/$imagepullsecret+/images/images-resources.tsx @@ -0,0 +1,340 @@ +import { Link, useOutletContext, useParams } from '@remix-run/react'; +import { useState } from 'react'; +import { toast } from '~/components/molecule/toast'; +import { generateKey, titleCase } from '~/components/utils'; +import ConsoleAvatar from '~/console/components/console-avatar'; +import { + ListItem, + ListItemV2, + ListTitle, + ListTitleV2, + listClass, +} from '~/console/components/console-list-components'; +import DeleteDialog from '~/console/components/delete-dialog'; +import Grid from '~/console/components/grid'; +import { Copy, Trash } from '~/console/components/icons'; +import ListGridView from '~/console/components/list-grid-view'; +import ListV2 from '~/console/components/listV2'; +import ResourceExtraAction, { + IResourceExtraItem, +} from '~/console/components/resource-extra-action'; +import { IAccountContext } from '~/console/routes/_main+/$account+/_layout'; +import { useConsoleApi } from '~/console/server/gql/api-provider'; +import { IEnvironments } from '~/console/server/gql/queries/environment-queries'; +import { + ExtractNodeType, + parseName, + parseUpdateOrCreatedBy, + parseUpdateOrCreatedOn, +} from '~/console/server/r-utils/common'; +import { useWatchReload } from '~/lib/client/helpers/socket/useWatch'; +import { useReload } from '~/root/lib/client/helpers/reloader'; +import { handleError } from '~/root/lib/utils/common'; + +const RESOURCE_NAME = 'environment'; +type BaseType = ExtractNodeType; + +const parseItem = (item: BaseType) => { + return { + name: item.displayName, + id: parseName(item), + updateInfo: { + author: `Updated by ${titleCase(parseUpdateOrCreatedBy(item))}`, + time: parseUpdateOrCreatedOn(item), + }, + }; +}; + +type OnAction = ({ + action, + item, +}: { + action: 'clone' | 'delete' | 'suspend' | 'resumed'; + item: BaseType; +}) => void; + +type IExtraButton = { + onAction: OnAction; + item: BaseType; +}; + +const ExtraButton = ({ item, onAction }: IExtraButton) => { + const iconSize = 16; + const options: IResourceExtraItem[] = [ + { + label: 'Clone', + icon: , + type: 'item', + key: 'clone', + onClick: () => onAction({ action: 'clone', item }), + }, + { + label: 'Delete', + icon: , + type: 'item', + onClick: () => onAction({ action: 'delete', item }), + key: 'delete', + className: '!text-text-critical', + }, + ]; + + return ; +}; + +interface IResource { + items: (BaseType & { isClusterOnline: boolean })[]; + onAction: OnAction; +} + +const GridView = ({ items = [], onAction }: IResource) => { + const { account } = useParams(); + return ( + + {items.map((item, index) => { + const { name, id, updateInfo } = parseItem(item); + const keyPrefix = `${RESOURCE_NAME}-${id}-${index}`; + return ( + ( + } + avatar={} + /> + ), + }, + { + key: generateKey(keyPrefix, updateInfo.author), + className: listClass.author, + render: () => ( + + ), + }, + ]} + /> + ); + })} + + ); +}; + +const ListView = ({ items, onAction }: IResource) => { + const { account } = useParams(); + // const { clusters } = useClusterStatusV2(); + + // const [clusterOnlineStatus, setClusterOnlineStatus] = useState< + // Record + // >({}); + // useEffect(() => { + // const states: Record = {}; + // Object.entries(clusters).forEach(([key, value]) => { + // states[key] = findClusterStatus(value); + // }); + // setClusterOnlineStatus(states); + // }, [clusters]); + + return ( + 'Resource Name', + name: 'name', + className: listClass.title, + }, + { + render: () => 'Cluster', + name: 'cluster', + className: listClass.item, + }, + { + render: () => '', + name: 'flex-post', + className: listClass.flex, + }, + // { + // render: () => 'Status', + // name: 'status', + // className: listClass.status, + // }, + { + render: () => 'Updated', + name: 'updated', + className: listClass.updated, + }, + { + render: () => '', + name: 'action', + className: listClass.action, + }, + ], + rows: items.map((i) => { + const { name, id, updateInfo } = parseItem(i); + // const isClusterOnline = clusterOnlineStatus[i.clusterName]; + + return { + columns: { + name: { + render: () => ( + } + /> + ), + }, + cluster: { + render: () => ( + + ), + }, + // status: { + // render: () => { + // if (i.isArchived) { + // return Archived; + // } + + // if (!isClusterOnline) { + // return Cluster Offline; + // } + + // if (i.spec?.suspend) { + // return Suspended; + // } + + // return ; + // }, + // }, + updated: { + render: () => ( + + ), + }, + action: { + render: () => , + }, + }, + ...(i.isArchived ? {} : { to: `/${account}/env/${id}` }), + }; + }), + }} + /> + ); +}; + +const ImagesResource = ({ items = [] }: { items: BaseType[] }) => { + const { account } = useOutletContext(); + const api = useConsoleApi(); + const reloadPage = useReload(); + useWatchReload( + items.map((i) => { + return `account:${parseName(account)}.environment:${parseName(i)}`; + }) + ); + + const suspendEnvironment = async (item: BaseType, suspend: boolean) => { + try { + const { errors } = await api.updateEnvironment({ + env: { + displayName: item.displayName, + clusterName: item.clusterName, + metadata: { + name: parseName(item), + }, + spec: { + suspend, + }, + }, + }); + + if (errors) { + throw errors[0]; + } + toast.success( + `${ + suspend + ? 'Environment suspended successfully' + : 'Environment resumed successfully' + }` + ); + reloadPage(); + } catch (err) { + handleError(err); + } + }; + + const [showDeleteDialog, setShowDeleteDialog] = useState( + null + ); + const [visible, setVisible] = useState(null); + + const props: IResource = { + // @ts-ignore + items, + onAction: ({ action, item }) => { + switch (action) { + case 'clone': + setVisible(item); + break; + case 'suspend': + suspendEnvironment(item, true); + break; + case 'resumed': + suspendEnvironment(item, false); + break; + case 'delete': + setShowDeleteDialog(item); + break; + default: + break; + } + }, + }; + + return ( + <> + } + gridView={} + /> + { + try { + const { errors } = await api.deleteEnvironment({ + envName: parseName(showDeleteDialog), + }); + + if (errors) { + throw errors[0]; + } + reloadPage(); + toast.success(`Environment deleted successfully`); + setShowDeleteDialog(null); + } catch (err) { + handleError(err); + } + }} + /> + + ); +}; + +export default ImagesResource; diff --git a/src/apps/console/routes/_main+/$account+/settings+/ips+/$imagepullsecret+/images/route.tsx b/src/apps/console/routes/_main+/$account+/settings+/ips+/$imagepullsecret+/images/route.tsx new file mode 100644 index 000000000..48c8803ee --- /dev/null +++ b/src/apps/console/routes/_main+/$account+/settings+/ips+/$imagepullsecret+/images/route.tsx @@ -0,0 +1,92 @@ +import { defer } from '@remix-run/node'; +import { Link, useLoaderData } from '@remix-run/react'; +import { Button } from '~/components/atoms/button'; +import { EmptyManagedResourceImage } from '~/console/components/empty-resource-images'; +import { Plus } from '~/console/components/icons'; +import { LoadingComp, pWrapper } from '~/console/components/loading-component'; +import Wrapper from '~/console/components/wrapper'; +import { GQLServerHandler } from '~/console/server/gql/saved-queries'; +import { parseNodes } from '~/console/server/r-utils/common'; +import { ensureAccountSet } from '~/console/server/utils/auth-utils'; +import { getPagination, getSearch } from '~/console/server/utils/common'; +import { IRemixCtx } from '~/lib/types/common'; +import ImagesResource from './images-resources'; +import Tools from './tools'; + +export const loader = (ctx: IRemixCtx) => { + const { imagepullsecret } = ctx.params; + console.log('====>>>>', imagepullsecret); + const promise = pWrapper(async () => { + ensureAccountSet(ctx); + + const { data, errors } = await GQLServerHandler( + ctx.request + ).listEnvironments({ + pq: getPagination(ctx), + search: getSearch(ctx), + }); + + if (errors) { + throw errors[0]; + } + return { imagesData: data }; + }); + return defer({ promise }); +}; + +const Images = () => { + const { promise } = useLoaderData(); + return ( + + {({ imagesData }) => { + const images = parseNodes(imagesData); + + return ( + 0 && ( +