diff --git a/packages/mco/components/create-dr-policy/select-cluster-list.tsx b/packages/mco/components/create-dr-policy/select-cluster-list.tsx index 3c1a10d37..f95610479 100644 --- a/packages/mco/components/create-dr-policy/select-cluster-list.tsx +++ b/packages/mco/components/create-dr-policy/select-cluster-list.tsx @@ -1,16 +1,21 @@ import * as React from 'react'; import { getManagedClusterResourceObj } from '@odf/mco/hooks'; +import { ODFInfoYamlObject } from '@odf/mco/types'; import { getMajorVersion, ValidateManagedClusterCondition, getValueFromClusterClaim, isMinimumSupportedODFVersion, + getManagedClusterViewName, } from '@odf/mco/utils'; import { StatusBox } from '@odf/shared/generic/status-box'; import { getName, getNamespace } from '@odf/shared/selectors'; +import { ConfigMapKind } from '@odf/shared/types'; import { useCustomTranslation } from '@odf/shared/useCustomTranslationHook'; +import { referenceForModel } from '@odf/shared/utils'; import { useK8sWatchResource } from '@openshift-console/dynamic-plugin-sdk'; import { Select, SelectOption } from '@patternfly/react-core/deprecated'; +import { safeLoad } from 'js-yaml'; import { DataList, DataListItem, @@ -33,9 +38,9 @@ import { MANAGED_CLUSTER_REGION_CLAIM, MANAGED_CLUSTER_JOINED, MANAGED_CLUSTER_CONDITION_AVAILABLE, - ClusterClaimTypes, } from '../../constants'; -import { ACMManagedClusterKind } from '../../types'; +import { ACMManagedClusterViewModel } from '../../models'; +import { ACMManagedClusterKind, ACMManagedClusterViewKind } from '../../types'; import { DRPolicyAction, DRPolicyActionType, @@ -59,48 +64,65 @@ const getFilteredClusters = ( }; const getODFInfo = ( - managedCluster: ACMManagedClusterKind, - requiredODFVersion: string + requiredODFVersion: string, + odfInfoConfigData: { [key: string]: string } ): ODFConfigInfoType => { - const clusterClaims = managedCluster?.status?.clusterClaims; - const odfVersion = getValueFromClusterClaim( - clusterClaims, - ClusterClaimTypes.ODF_VERSION - ); - const storageClusterNamespacedName = getValueFromClusterClaim( - clusterClaims, - ClusterClaimTypes.STORAGE_CLUSTER_NAME - ); - const storageSystemNamespacedName = getValueFromClusterClaim( - clusterClaims, - ClusterClaimTypes.STORAGE_SYSTEM_NAME - ); - const cephFsid = getValueFromClusterClaim( - clusterClaims, - ClusterClaimTypes.CEPH_FSID - ); - const storageClusterCount = getValueFromClusterClaim( - clusterClaims, - ClusterClaimTypes.STORAGE_CLUSTER_COUNT - ); - const isDROptimized = getValueFromClusterClaim( - clusterClaims, - ClusterClaimTypes.DR_OPTIMIZED - ); - return { - odfVersion: odfVersion, - isValidODFVersion: isMinimumSupportedODFVersion( - getMajorVersion(odfVersion), - requiredODFVersion - ), - storageClusterCount: Number(storageClusterCount || '0'), - storageClusterInfo: { - storageClusterNamespacedName: storageClusterNamespacedName, - storageSystemNamespacedName: storageSystemNamespacedName, - cephFSID: cephFsid, - isDROptimized: isDROptimized === 'true', - }, - }; + try { + // Managed cluster with multiple StorageSystems is not currently supported for DR + // ToDo: Update this once we add support for multiple clusters + const odfInfoKey = Object.keys(odfInfoConfigData)[0]; + const odfInfoYaml = odfInfoConfigData[odfInfoKey]; + const odfInfo: ODFInfoYamlObject = safeLoad(odfInfoYaml); + + const storageClusterName = odfInfo?.storageCluster?.namespacedName?.name; + const storageClusterNamespace = + odfInfo?.storageCluster?.namespacedName?.namespace; + const storageSystemNamespace = odfInfo?.storageSystemName; + + const odfVersion = odfInfo?.version; + const storageClusterCount = Object.keys(odfInfoConfigData).length; + const storageClusterNamespacedName = + !!storageClusterName && !!storageClusterNamespace + ? storageClusterName + '/' + storageClusterNamespace + : ''; + const storageSystemNamespacedName = + !!storageSystemNamespace && !!storageClusterNamespace + ? storageSystemNamespace + '/' + storageClusterNamespace + : ''; + const cephFSID = odfInfo?.storageCluster?.cephClusterFSID; + + return { + odfVersion, + isValidODFVersion: isMinimumSupportedODFVersion( + getMajorVersion(odfVersion), + requiredODFVersion + ), + storageClusterCount, + storageClusterInfo: { + storageClusterNamespacedName, + storageSystemNamespacedName, + cephFSID, + // REMOVE THIS + isDROptimized: true, + }, + }; + } catch (err) { + // eslint-disable-next-line no-console + console.error(err); + + return { + odfVersion: '', + isValidODFVersion: false, + storageClusterCount: 0, + storageClusterInfo: { + storageClusterNamespacedName: '', + storageSystemNamespacedName: '', + cephFSID: '', + // REMOVE THIS + isDROptimized: false, + }, + }; + } }; const filterRegions = (filteredClusters: ManagedClusterInfoType[]) => @@ -113,7 +135,8 @@ const filterRegions = (filteredClusters: ManagedClusterInfoType[]) => const getManagedClusterInfo = ( cluster: ACMManagedClusterKind, - requiredODFVersion: string + requiredODFVersion: string, + odfInfoConfigData: { [key: string]: string } ): ManagedClusterInfoType => ({ name: getName(cluster), namesapce: getNamespace(cluster), @@ -125,7 +148,7 @@ const getManagedClusterInfo = ( cluster, MANAGED_CLUSTER_CONDITION_AVAILABLE ), - odfInfo: getODFInfo(cluster, requiredODFVersion), + odfInfo: getODFInfo(requiredODFVersion, odfInfoConfigData), }); const isChecked = (clusters: ManagedClusterInfoType[], clusterName: string) => @@ -154,18 +177,47 @@ export const SelectClusterList: React.FC = ({ ACMManagedClusterKind[] >(getManagedClusterResourceObj()); + const [mcvs, mcvsLoaded, mcvsLoadError] = useK8sWatchResource< + ACMManagedClusterViewKind[] + >({ + kind: referenceForModel(ACMManagedClusterViewModel), + isList: true, + }); + + const allLoaded = loaded && mcvsLoaded; + const anyError = loadError || mcvsLoadError; + const clusters: ManagedClusterInfoType[] = React.useMemo(() => { - if (!!requiredODFVersion && loaded && !loadError) { - return managedClusters?.reduce( - (acc, cluster) => - ValidateManagedClusterCondition(cluster, MANAGED_CLUSTER_JOINED) - ? [...acc, getManagedClusterInfo(cluster, requiredODFVersion)] - : acc, - [] - ); + if (!!requiredODFVersion && allLoaded && !anyError) { + return managedClusters?.reduce((acc, cluster) => { + if (ValidateManagedClusterCondition(cluster, MANAGED_CLUSTER_JOINED)) { + // OCS creates a ConfigMap on the managed clusters, with details about StorageClusters, Clients. + // MCO creates ManagedClusterView on the hub cluster, referencing that ConfigMap. + const managedClusterName = getName(cluster); + const mcv = + mcvs.find( + (obj: ACMManagedClusterViewKind) => + getName(obj) === + getManagedClusterViewName(managedClusterName) && + getNamespace(obj) === managedClusterName + ) || {}; + const odfInfoConfigData = + (mcv.status?.result as ConfigMapKind)?.data || {}; + return [ + ...acc, + getManagedClusterInfo( + cluster, + requiredODFVersion, + odfInfoConfigData + ), + ]; + } + + return acc; + }, []); } return []; - }, [requiredODFVersion, managedClusters, loaded, loadError]); + }, [requiredODFVersion, managedClusters, mcvs, allLoaded, anyError]); const filteredClusters: ManagedClusterInfoType[] = React.useMemo( () => getFilteredClusters(clusters, region, nameSearch), @@ -250,8 +302,8 @@ export const SelectClusterList: React.FC = ({ ({ [EnrollDiscoveredApplicationSteps.Replication]: t('Replication'), [EnrollDiscoveredApplicationSteps.Review]: t('Review'), }); + +export const MCV_NAME_TEMPLATE = 'odf-multicluster-mcv-'; diff --git a/packages/mco/types/odf-mco.ts b/packages/mco/types/odf-mco.ts index 92036b7b2..3993dc479 100644 --- a/packages/mco/types/odf-mco.ts +++ b/packages/mco/types/odf-mco.ts @@ -13,3 +13,25 @@ export type MirrorPeerKind = K8sResourceCommon & { type: string; }; }; + +export type ConnectedClient = { + name: string; + clusterId: string; +}; + +export type InfoStorageCluster = { + namespacedName: { + name: string; + namespace: string; + }; + storageProviderEndpoint: string; + cephClusterFSID: string; +}; + +export type ODFInfoYamlObject = { + version: string; + deploymentType: string; + clients: ConnectedClient[]; + storageCluster: InfoStorageCluster; + storageSystemName: string; +}; diff --git a/packages/mco/utils/disaster-recovery.tsx b/packages/mco/utils/disaster-recovery.tsx index 3f47e948a..6cc82c53f 100644 --- a/packages/mco/utils/disaster-recovery.tsx +++ b/packages/mco/utils/disaster-recovery.tsx @@ -34,6 +34,7 @@ import { LABELS_SPLIT_CHAR, DR_BLOCK_LISTED_LABELS, PLACEMENT_RULE_REF_LABEL, + MCV_NAME_TEMPLATE, } from '../constants'; import { DRPC_NAMESPACE_ANNOTATION, @@ -643,3 +644,6 @@ export const getLabelsFromSearchResult = ( return acc; }, {}); }; + +export const getManagedClusterViewName = (managedClusterName: string): string => + MCV_NAME_TEMPLATE + managedClusterName;