From e38270156405f507a06147958144439d203c1093 Mon Sep 17 00:00:00 2001 From: Suresh Kumar Anaparti Date: Thu, 7 Nov 2024 19:09:01 +0530 Subject: [PATCH 1/7] Add & Remove PowerFlex/ScaleIO MDMs while preparing & unpreparing the storage SDC connections (instead of start & stop scini) --- .../api/UnprepareStorageClientCommand.java | 10 ++- .../vmsnapshot/ScaleIOVMSnapshotStrategy.java | 13 +++- ...tUnprepareStorageClientCommandWrapper.java | 2 +- .../kvm/storage/KVMStoragePoolManager.java | 4 +- .../kvm/storage/ScaleIOStorageAdaptor.java | 31 ++++++++-- .../kvm/storage/StorageAdaptor.java | 3 +- ...repareStorageClientCommandWrapperTest.java | 14 ++++- .../storage/ScaleIOStorageAdaptorTest.java | 58 ++++++------------ .../datastore/api/StorageConfiguration.java | 48 +++++++++++++++ .../storage/datastore/api/StoragePool.java | 11 ++++ .../client/ScaleIOGatewayClient.java | 3 + .../ScaleIOGatewayClientConnectionPool.java | 23 +++++-- .../client/ScaleIOGatewayClientImpl.java | 10 +++ .../driver/ScaleIOPrimaryDataStoreDriver.java | 6 +- .../ScaleIOPrimaryDataStoreLifeCycle.java | 4 ++ .../manager/ScaleIOSDCManagerImpl.java | 61 ++++++++++++++++--- .../provider/ScaleIOHostListener.java | 7 ++- .../storage/datastore/util/ScaleIOUtil.java | 38 ++++++++++++ 18 files changed, 279 insertions(+), 67 deletions(-) create mode 100644 plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/api/StorageConfiguration.java diff --git a/core/src/main/java/com/cloud/agent/api/UnprepareStorageClientCommand.java b/core/src/main/java/com/cloud/agent/api/UnprepareStorageClientCommand.java index bebd30ca519d..b98ce6fa345e 100644 --- a/core/src/main/java/com/cloud/agent/api/UnprepareStorageClientCommand.java +++ b/core/src/main/java/com/cloud/agent/api/UnprepareStorageClientCommand.java @@ -19,18 +19,22 @@ package com.cloud.agent.api; +import java.util.Map; + import com.cloud.storage.Storage.StoragePoolType; public class UnprepareStorageClientCommand extends Command { private StoragePoolType poolType; private String poolUuid; + private Map details; public UnprepareStorageClientCommand() { } - public UnprepareStorageClientCommand(StoragePoolType poolType, String poolUuid) { + public UnprepareStorageClientCommand(StoragePoolType poolType, String poolUuid, Map details) { this.poolType = poolType; this.poolUuid = poolUuid; + this.details = details; } @Override @@ -45,4 +49,8 @@ public StoragePoolType getPoolType() { public String getPoolUuid() { return poolUuid; } + + public Map getDetails() { + return details; + } } diff --git a/engine/storage/snapshot/src/main/java/org/apache/cloudstack/storage/vmsnapshot/ScaleIOVMSnapshotStrategy.java b/engine/storage/snapshot/src/main/java/org/apache/cloudstack/storage/vmsnapshot/ScaleIOVMSnapshotStrategy.java index 1ec6e20fc9e1..09280921039a 100644 --- a/engine/storage/snapshot/src/main/java/org/apache/cloudstack/storage/vmsnapshot/ScaleIOVMSnapshotStrategy.java +++ b/engine/storage/snapshot/src/main/java/org/apache/cloudstack/storage/vmsnapshot/ScaleIOVMSnapshotStrategy.java @@ -34,6 +34,7 @@ import org.apache.cloudstack.storage.datastore.client.ScaleIOGatewayClient; import org.apache.cloudstack.storage.datastore.client.ScaleIOGatewayClientConnectionPool; import org.apache.cloudstack.storage.datastore.db.PrimaryDataStoreDao; +import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailVO; import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailsDao; import org.apache.cloudstack.storage.datastore.db.StoragePoolVO; import org.apache.cloudstack.storage.datastore.util.ScaleIOUtil; @@ -301,7 +302,11 @@ public boolean revertVMSnapshot(VMSnapshot vmSnapshot) { srcSnapshotDestVolumeMap.put(srcSnapshotVolumeId, destVolumeId); } - String systemId = storagePoolDetailsDao.findDetail(storagePoolId, ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID).getValue(); + String systemId = null; + StoragePoolDetailVO systemIdDetail = storagePoolDetailsDao.findDetail(storagePoolId, ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID); + if (systemIdDetail != null) { + systemId = systemIdDetail.getValue(); + } if (systemId == null) { throw new CloudRuntimeException("Failed to get the system id for PowerFlex storage pool for reverting VM snapshot: " + vmSnapshot.getName()); } @@ -380,7 +385,11 @@ public boolean deleteVMSnapshot(VMSnapshot vmSnapshot) { try { List volumeTOs = vmSnapshotHelper.getVolumeTOList(vmSnapshot.getVmId()); StoragePoolVO storagePool = vmSnapshotHelper.getStoragePoolForVM(userVm); - String systemId = storagePoolDetailsDao.findDetail(storagePool.getId(), ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID).getValue(); + String systemId = null; + StoragePoolDetailVO systemIdDetail = storagePoolDetailsDao.findDetail(storagePool.getId(), ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID); + if (systemIdDetail != null) { + systemId = systemIdDetail.getValue(); + } if (systemId == null) { throw new CloudRuntimeException("Failed to get the system id for PowerFlex storage pool for deleting VM snapshot: " + vmSnapshot.getName()); } diff --git a/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/resource/wrapper/LibvirtUnprepareStorageClientCommandWrapper.java b/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/resource/wrapper/LibvirtUnprepareStorageClientCommandWrapper.java index 2f23a934003f..178f68367c07 100644 --- a/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/resource/wrapper/LibvirtUnprepareStorageClientCommandWrapper.java +++ b/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/resource/wrapper/LibvirtUnprepareStorageClientCommandWrapper.java @@ -34,7 +34,7 @@ public class LibvirtUnprepareStorageClientCommandWrapper extends CommandWrapper< @Override public Answer execute(UnprepareStorageClientCommand cmd, LibvirtComputingResource libvirtComputingResource) { final KVMStoragePoolManager storagePoolMgr = libvirtComputingResource.getStoragePoolMgr(); - Pair unprepareStorageClientResult = storagePoolMgr.unprepareStorageClient(cmd.getPoolType(), cmd.getPoolUuid()); + Pair unprepareStorageClientResult = storagePoolMgr.unprepareStorageClient(cmd.getPoolType(), cmd.getPoolUuid(), cmd.getDetails()); if (!unprepareStorageClientResult.first()) { String msg = unprepareStorageClientResult.second(); logger.debug("Couldn't unprepare storage client, due to: " + msg); diff --git a/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/KVMStoragePoolManager.java b/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/KVMStoragePoolManager.java index e27547acbb27..2cac056413d5 100644 --- a/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/KVMStoragePoolManager.java +++ b/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/KVMStoragePoolManager.java @@ -483,8 +483,8 @@ public Ternary, String> prepareStorageClient(Storag return adaptor.prepareStorageClient(type, uuid, details); } - public Pair unprepareStorageClient(StoragePoolType type, String uuid) { + public Pair unprepareStorageClient(StoragePoolType type, String uuid, Map details) { StorageAdaptor adaptor = getStorageAdaptor(type); - return adaptor.unprepareStorageClient(type, uuid); + return adaptor.unprepareStorageClient(type, uuid, details); } } diff --git a/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/ScaleIOStorageAdaptor.java b/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/ScaleIOStorageAdaptor.java index 335ea0d03d26..f77f86835617 100644 --- a/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/ScaleIOStorageAdaptor.java +++ b/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/ScaleIOStorageAdaptor.java @@ -584,14 +584,24 @@ public Ternary, String> prepareStorageClient(Storag if (!ScaleIOUtil.startSDCService()) { return new Ternary<>(false, null, "Couldn't start SDC service on host"); } - } else if (!ScaleIOUtil.restartSDCService()) { - return new Ternary<>(false, null, "Couldn't restart SDC service on host"); + } + + if (details != null && details.containsKey(ScaleIOGatewayClient.STORAGE_POOL_MDMS)) { + // Assuming SDC service is started, add mdms + String mdms = details.get(ScaleIOGatewayClient.STORAGE_POOL_MDMS); + String[] mdmAddresses = mdms.split(","); + if (mdmAddresses.length > 0) { + ScaleIOUtil.addMdms(Arrays.asList(mdmAddresses)); + if (!ScaleIOUtil.mdmAdded(mdmAddresses[0])) { + return new Ternary<>(false, null, "Failed to add MDMs"); + } + } } return new Ternary<>( true, getSDCDetails(details), "Prepared client successfully"); } - public Pair unprepareStorageClient(Storage.StoragePoolType type, String uuid) { + public Pair unprepareStorageClient(Storage.StoragePoolType type, String uuid, Map details) { if (!ScaleIOUtil.isSDCServiceInstalled()) { logger.debug("SDC service not installed on host, no need to unprepare the SDC client"); return new Pair<>(true, "SDC service not installed on host, no need to unprepare the SDC client"); @@ -602,8 +612,19 @@ public Pair unprepareStorageClient(Storage.StoragePoolType type return new Pair<>(true, "SDC service not enabled on host, no need to unprepare the SDC client"); } - if (!ScaleIOUtil.stopSDCService()) { - return new Pair<>(false, "Couldn't stop SDC service on host"); + if (details != null && details.containsKey(ScaleIOGatewayClient.STORAGE_POOL_MDMS)) { + String mdms = details.get(ScaleIOGatewayClient.STORAGE_POOL_MDMS); + String[] mdmAddresses = mdms.split(","); + if (mdmAddresses.length > 0) { + if (!ScaleIOUtil.mdmAdded(mdmAddresses[0])) { + return new Pair<>(true, "MDM not added, no need to unprepare the SDC client"); + } + + ScaleIOUtil.removeMdms(Arrays.asList(mdmAddresses)); + if (ScaleIOUtil.mdmAdded(mdmAddresses[0])) { + return new Pair<>(false, "Failed to remove MDMs, unable to unprepare the SDC client"); + } + } } return new Pair<>(true, "Unprepared SDC client successfully"); diff --git a/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/StorageAdaptor.java b/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/StorageAdaptor.java index 8439a50073e6..29a6cb769320 100644 --- a/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/StorageAdaptor.java +++ b/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/StorageAdaptor.java @@ -140,9 +140,10 @@ default Ternary, String> prepareStorageClient(Stora * Unprepares the storage client. * @param type type of the storage pool * @param uuid uuid of the storage pool + * @param details any details of the storage pool that are required for client unpreparation * @return status, & message in case failed */ - default Pair unprepareStorageClient(StoragePoolType type, String uuid) { + default Pair unprepareStorageClient(StoragePoolType type, String uuid, Map details) { return new Pair<>(true, ""); } } diff --git a/plugins/hypervisors/kvm/src/test/java/com/cloud/hypervisor/kvm/resource/wrapper/LibvirtUnprepareStorageClientCommandWrapperTest.java b/plugins/hypervisors/kvm/src/test/java/com/cloud/hypervisor/kvm/resource/wrapper/LibvirtUnprepareStorageClientCommandWrapperTest.java index 7409b286f32e..942a6eb18379 100644 --- a/plugins/hypervisors/kvm/src/test/java/com/cloud/hypervisor/kvm/resource/wrapper/LibvirtUnprepareStorageClientCommandWrapperTest.java +++ b/plugins/hypervisors/kvm/src/test/java/com/cloud/hypervisor/kvm/resource/wrapper/LibvirtUnprepareStorageClientCommandWrapperTest.java @@ -14,6 +14,10 @@ package com.cloud.hypervisor.kvm.resource.wrapper; +import java.util.HashMap; +import java.util.Map; + +import org.apache.cloudstack.storage.datastore.client.ScaleIOGatewayClient; import org.junit.Assert; import org.junit.Test; import org.junit.runner.RunWith; @@ -45,10 +49,13 @@ public void testUnprepareStorageClientSuccess() { UnprepareStorageClientCommand cmd = Mockito.mock(UnprepareStorageClientCommand.class); Mockito.when(cmd.getPoolType()).thenReturn(Storage.StoragePoolType.PowerFlex); Mockito.when(cmd.getPoolUuid()).thenReturn(poolUuid); + Map details = new HashMap<>(); + details.put(ScaleIOGatewayClient.STORAGE_POOL_MDMS, "1.1.1.1,2.2.2.2"); + Mockito.when(cmd.getDetails()).thenReturn(details); KVMStoragePoolManager storagePoolMgr = Mockito.mock(KVMStoragePoolManager.class); Mockito.when(libvirtComputingResourceMock.getStoragePoolMgr()).thenReturn(storagePoolMgr); - Mockito.when(storagePoolMgr.unprepareStorageClient(cmd.getPoolType(), cmd.getPoolUuid())).thenReturn(new Pair<>(true, "")); + Mockito.when(storagePoolMgr.unprepareStorageClient(cmd.getPoolType(), cmd.getPoolUuid(), cmd.getDetails())).thenReturn(new Pair<>(true, "")); UnprepareStorageClientAnswer result = (UnprepareStorageClientAnswer) libvirtUnprepareStorageClientCommandWrapperSpy.execute(cmd, libvirtComputingResourceMock); @@ -60,10 +67,13 @@ public void testUnprepareStorageClientFailure() { UnprepareStorageClientCommand cmd = Mockito.mock(UnprepareStorageClientCommand.class); Mockito.when(cmd.getPoolType()).thenReturn(Storage.StoragePoolType.PowerFlex); Mockito.when(cmd.getPoolUuid()).thenReturn(poolUuid); + Map details = new HashMap<>(); + details.put(ScaleIOGatewayClient.STORAGE_POOL_MDMS, "1.1.1.1,2.2.2.2"); + Mockito.when(cmd.getDetails()).thenReturn(details); KVMStoragePoolManager storagePoolMgr = Mockito.mock(KVMStoragePoolManager.class); Mockito.when(libvirtComputingResourceMock.getStoragePoolMgr()).thenReturn(storagePoolMgr); - Mockito.when(storagePoolMgr.unprepareStorageClient(cmd.getPoolType(), cmd.getPoolUuid())).thenReturn(new Pair<>(false, "Unprepare storage client failed")); + Mockito.when(storagePoolMgr.unprepareStorageClient(cmd.getPoolType(), cmd.getPoolUuid(), cmd.getDetails())).thenReturn(new Pair<>(false, "Unprepare storage client failed")); UnprepareStorageClientAnswer result = (UnprepareStorageClientAnswer) libvirtUnprepareStorageClientCommandWrapperSpy.execute(cmd, libvirtComputingResourceMock); diff --git a/plugins/hypervisors/kvm/src/test/java/com/cloud/hypervisor/kvm/storage/ScaleIOStorageAdaptorTest.java b/plugins/hypervisors/kvm/src/test/java/com/cloud/hypervisor/kvm/storage/ScaleIOStorageAdaptorTest.java index 07aea0cfbee5..072fd6fbd4be 100644 --- a/plugins/hypervisors/kvm/src/test/java/com/cloud/hypervisor/kvm/storage/ScaleIOStorageAdaptorTest.java +++ b/plugins/hypervisors/kvm/src/test/java/com/cloud/hypervisor/kvm/storage/ScaleIOStorageAdaptorTest.java @@ -93,34 +93,6 @@ public void testPrepareStorageClient_SDCServiceNotEnabled() { Assert.assertEquals("SDC service not enabled on host", result.third()); } - @Test - public void testPrepareStorageClient_SDCServiceNotRestarted() { - when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl status scini"))).thenReturn(3); - when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl is-enabled scini"))).thenReturn(0); - when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl is-active scini"))).thenReturn(0); - when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl restart scini"))).thenReturn(1); - - Ternary, String> result = scaleIOStorageAdaptor.prepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid, new HashMap<>()); - - Assert.assertFalse(result.first()); - Assert.assertNull(result.second()); - Assert.assertEquals("Couldn't restart SDC service on host", result.third()); - } - - @Test - public void testPrepareStorageClient_SDCServiceRestarted() { - when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl status scini"))).thenReturn(3); - when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl is-enabled scini"))).thenReturn(0); - when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl is-active scini"))).thenReturn(0); - when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl restart scini"))).thenReturn(0); - - Ternary, String> result = scaleIOStorageAdaptor.prepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid, new HashMap<>()); - - Assert.assertTrue(result.first()); - Assert.assertNotNull(result.second()); - Assert.assertTrue(result.second().isEmpty()); - } - @Test public void testPrepareStorageClient_SDCServiceNotStarted() { when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl status scini"))).thenReturn(3); @@ -181,9 +153,10 @@ public void testPrepareStorageClient_SDCServiceStartedReturnSDCGuid() { @Test public void testUnprepareStorageClient_SDCServiceNotInstalled() { + Map details = new HashMap<>(); when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl status scini"))).thenReturn(4); - Pair result = scaleIOStorageAdaptor.unprepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid); + Pair result = scaleIOStorageAdaptor.unprepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid, details); Assert.assertTrue(result.first()); Assert.assertEquals("SDC service not installed on host, no need to unprepare the SDC client", result.second()); @@ -191,35 +164,42 @@ public void testUnprepareStorageClient_SDCServiceNotInstalled() { @Test public void testUnprepareStorageClient_SDCServiceNotEnabled() { + Map details = new HashMap<>(); when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl status scini"))).thenReturn(3); when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl is-enabled scini"))).thenReturn(1); - Pair result = scaleIOStorageAdaptor.unprepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid); + Pair result = scaleIOStorageAdaptor.unprepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid, details); Assert.assertTrue(result.first()); Assert.assertEquals("SDC service not enabled on host, no need to unprepare the SDC client", result.second()); } @Test - public void testUnprepareStorageClient_SDCServiceNotStopped() { + public void testUnprepareStorageClient_MDMNotAdded() { + Map details = new HashMap<>(); + details.put(ScaleIOGatewayClient.STORAGE_POOL_MDMS, "1.1.1.1,2.2.2.2"); when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl status scini"))).thenReturn(3); when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl is-enabled scini"))).thenReturn(0); - when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl stop scini"))).thenReturn(1); + when(Script.runSimpleBashScript(Mockito.eq("/opt/emc/scaleio/sdc/bin/drv_cfg --query_mdms|grep 1.1.1.1"))).thenReturn("MDM-ID 71fd458f0775010f SDC ID 4421a91a00000000 INSTALLATION ID 204930df2cbcaf8e IPs [0]-3.3.3.3 [1]-4.4.4.4"); - Pair result = scaleIOStorageAdaptor.unprepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid); + Pair result = scaleIOStorageAdaptor.unprepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid, details); - Assert.assertFalse(result.first()); - Assert.assertEquals("Couldn't stop SDC service on host", result.second()); + Assert.assertTrue(result.first()); + Assert.assertEquals("MDM not added, no need to unprepare the SDC client", result.second()); } @Test - public void testUnprepareStorageClient_SDCServiceStopped() { + public void testUnprepareStorageClient_RemoveMDMFailed() { + Map details = new HashMap<>(); + details.put(ScaleIOGatewayClient.STORAGE_POOL_MDMS, "1.1.1.1,2.2.2.2"); when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl status scini"))).thenReturn(3); when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl is-enabled scini"))).thenReturn(0); - when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl stop scini"))).thenReturn(0); + when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl restart scini"))).thenReturn(0); + when(Script.runSimpleBashScript(Mockito.eq("/opt/emc/scaleio/sdc/bin/drv_cfg --query_mdms|grep 1.1.1.1"))).thenReturn("MDM-ID 71fd458f0775010f SDC ID 4421a91a00000000 INSTALLATION ID 204930df2cbcaf8e IPs [0]-1.1.1.1 [1]-2.2.2.2"); - Pair result = scaleIOStorageAdaptor.unprepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid); + Pair result = scaleIOStorageAdaptor.unprepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid, details); - Assert.assertTrue(result.first()); + Assert.assertFalse(result.first()); + Assert.assertEquals("Failed to remove MDMs, unable to unprepare the SDC client", result.second()); } } diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/api/StorageConfiguration.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/api/StorageConfiguration.java new file mode 100644 index 000000000000..a485d5dab937 --- /dev/null +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/api/StorageConfiguration.java @@ -0,0 +1,48 @@ +// Licensed to the Apache Software Foundation (ASF) under one +// or more contributor license agreements. See the NOTICE file +// distributed with this work for additional information +// regarding copyright ownership. The ASF licenses this file +// to you under the Apache License, Version 2.0 (the +// "License"); you may not use this file except in compliance +// with the License. You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, +// software distributed under the License is distributed on an +// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +// KIND, either express or implied. See the License for the +// specific language governing permissions and limitations +// under the License. + +package org.apache.cloudstack.storage.datastore.api; + +public class StorageConfiguration { + String systemId; + Long mdmPort; + String[] mdmAddresses; + + public String getSystemId() { + return systemId; + } + + public void setSystemId(String systemId) { + this.systemId = systemId; + } + + public Long getMdmPort() { + return mdmPort; + } + + public void setMdmPort(Long mdmPort) { + this.mdmPort = mdmPort; + } + + public String[] getMdmAddresses() { + return mdmAddresses; + } + + public void setMdmAddresses(String[] mdmAddresses) { + this.mdmAddresses = mdmAddresses; + } +} diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/api/StoragePool.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/api/StoragePool.java index df903bb67f7c..36375ac2f441 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/api/StoragePool.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/api/StoragePool.java @@ -17,12 +17,15 @@ package org.apache.cloudstack.storage.datastore.api; +import java.util.List; + public class StoragePool { String id; String name; String mediaType; String protectionDomainId; String systemId; + List mdmAddresses; StoragePoolStatistics statistics; public String getId() { @@ -65,6 +68,14 @@ public void setSystemId(String systemId) { this.systemId = systemId; } + public List getMdmAddresses() { + return mdmAddresses; + } + + public void setMdmAddresses(List mdmAddresses) { + this.mdmAddresses = mdmAddresses; + } + public StoragePoolStatistics getStatistics() { return statistics; } diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/client/ScaleIOGatewayClient.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/client/ScaleIOGatewayClient.java index fd2b93bc674b..2dc5acffcbc1 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/client/ScaleIOGatewayClient.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/client/ScaleIOGatewayClient.java @@ -38,6 +38,7 @@ public interface ScaleIOGatewayClient { String GATEWAY_API_PASSWORD = "powerflex.gw.password"; String STORAGE_POOL_NAME = "powerflex.storagepool.name"; String STORAGE_POOL_SYSTEM_ID = "powerflex.storagepool.system.id"; + String STORAGE_POOL_MDMS = "powerflex.storagepool.mdms"; String SDC_ID = "powerflex.sdc.id"; String SDC_GUID = "powerflex.sdc.guid"; @@ -91,4 +92,6 @@ Volume createVolume(final String name, final String storagePoolId, boolean haveConnectedSdcs(); boolean isSdcConnected(String sdcId); boolean isSdcConnectedByIP(String ipAddress); + + List getMdmAddresses(); } diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/client/ScaleIOGatewayClientConnectionPool.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/client/ScaleIOGatewayClientConnectionPool.java index e605b159c991..39bcc2055865 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/client/ScaleIOGatewayClientConnectionPool.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/client/ScaleIOGatewayClientConnectionPool.java @@ -24,6 +24,7 @@ import com.cloud.storage.StoragePool; import org.apache.cloudstack.engine.subsystem.api.storage.DataStore; +import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailVO; import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailsDao; import org.apache.logging.log4j.Logger; import org.apache.logging.log4j.LogManager; @@ -76,11 +77,23 @@ private ScaleIOGatewayClient getClient(Long storagePoolId, String storagePoolUui synchronized (gatewayClients) { client = gatewayClients.get(storagePoolId); if (client == null) { - final String url = storagePoolDetailsDao.findDetail(storagePoolId, ScaleIOGatewayClient.GATEWAY_API_ENDPOINT).getValue(); - final String encryptedUsername = storagePoolDetailsDao.findDetail(storagePoolId, ScaleIOGatewayClient.GATEWAY_API_USERNAME).getValue(); - final String username = DBEncryptionUtil.decrypt(encryptedUsername); - final String encryptedPassword = storagePoolDetailsDao.findDetail(storagePoolId, ScaleIOGatewayClient.GATEWAY_API_PASSWORD).getValue(); - final String password = DBEncryptionUtil.decrypt(encryptedPassword); + String url = null; + StoragePoolDetailVO urlDetail = storagePoolDetailsDao.findDetail(storagePoolId, ScaleIOGatewayClient.GATEWAY_API_ENDPOINT); + if (urlDetail != null) { + url = urlDetail.getValue(); + } + String username = null; + StoragePoolDetailVO encryptedUsernameDetail = storagePoolDetailsDao.findDetail(storagePoolId, ScaleIOGatewayClient.GATEWAY_API_USERNAME); + if (encryptedUsernameDetail != null) { + final String encryptedUsername = encryptedUsernameDetail.getValue(); + username = DBEncryptionUtil.decrypt(encryptedUsername); + } + String password = null; + StoragePoolDetailVO encryptedPasswordDetail = storagePoolDetailsDao.findDetail(storagePoolId, ScaleIOGatewayClient.GATEWAY_API_PASSWORD); + if (encryptedPasswordDetail != null) { + final String encryptedPassword = encryptedPasswordDetail.getValue(); + password = DBEncryptionUtil.decrypt(encryptedPassword); + } final int clientTimeout = StorageManager.STORAGE_POOL_CLIENT_TIMEOUT.valueIn(storagePoolId); final int clientMaxConnections = StorageManager.STORAGE_POOL_CLIENT_MAX_CONNECTIONS.valueIn(storagePoolId); diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/client/ScaleIOGatewayClientImpl.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/client/ScaleIOGatewayClientImpl.java index 62581ca066e4..55e30bf29711 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/client/ScaleIOGatewayClientImpl.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/client/ScaleIOGatewayClientImpl.java @@ -34,6 +34,7 @@ import javax.net.ssl.SSLContext; import javax.net.ssl.X509TrustManager; +import org.apache.cloudstack.storage.datastore.api.StorageConfiguration; import org.apache.commons.lang3.StringUtils; import org.apache.cloudstack.api.ApiErrorCode; import org.apache.cloudstack.api.ServerApiException; @@ -1126,6 +1127,15 @@ public boolean isSdcConnectedByIP(String ipAddress) { return false; } + @Override + public List getMdmAddresses() { + StorageConfiguration storageConfiguration = get("/Configuration", StorageConfiguration.class); + if (storageConfiguration != null && storageConfiguration.getMdmAddresses().length > 0) { + return Arrays.asList(storageConfiguration.getMdmAddresses()); + } + return new ArrayList<>(); + } + private String getConnectionManagerStats() { StringBuilder sb = new StringBuilder(); sb.append("\n").append("Client Connection Manager Stats => "); diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java index 192ae4636e9c..d77ad1bb6c16 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java @@ -1067,7 +1067,11 @@ private Map getVolumeDetails(VolumeInfo volumeInfo, DataStore da volumeDetails.put(DiskTO.CHAP_TARGET_SECRET, chapInfo.getTargetSecret()); } - String systemId = storagePoolDetailsDao.findDetail(storagePoolId, ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID).getValue(); + String systemId = null; + StoragePoolDetailVO systemIdDetail = storagePoolDetailsDao.findDetail(storagePoolId, ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID); + if (systemIdDetail != null) { + systemId = systemIdDetail.getValue(); + } volumeDetails.put(ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID, systemId); return volumeDetails; diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/lifecycle/ScaleIOPrimaryDataStoreLifeCycle.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/lifecycle/ScaleIOPrimaryDataStoreLifeCycle.java index 38f9dc20fbdc..1a2fc96e8ef0 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/lifecycle/ScaleIOPrimaryDataStoreLifeCycle.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/lifecycle/ScaleIOPrimaryDataStoreLifeCycle.java @@ -47,6 +47,7 @@ import org.apache.cloudstack.storage.datastore.db.PrimaryDataStoreDao; import org.apache.cloudstack.storage.datastore.db.StoragePoolVO; import org.apache.cloudstack.storage.volume.datastore.PrimaryDataStoreHelper; +import org.apache.commons.lang3.StringUtils; import com.cloud.agent.AgentManager; import com.cloud.agent.api.Answer; @@ -114,6 +115,8 @@ private org.apache.cloudstack.storage.datastore.api.StoragePool findStoragePool( String systemId = client.getSystemId(pool.getProtectionDomainId()); pool.setSystemId(systemId); + List mdmAddresses = client.getMdmAddresses(); + pool.setMdmAddresses(mdmAddresses); return pool; } } @@ -245,6 +248,7 @@ public DataStore initialize(Map dsInfos) { details.put(ScaleIOGatewayClient.GATEWAY_API_PASSWORD, DBEncryptionUtil.encrypt(gatewayPassword)); details.put(ScaleIOGatewayClient.STORAGE_POOL_NAME, storagePoolName); details.put(ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID, scaleIOPool.getSystemId()); + details.put(ScaleIOGatewayClient.STORAGE_POOL_MDMS, StringUtils.join(scaleIOPool.getMdmAddresses(), ",")); parameters.setDetails(details); return dataStoreHelper.createPrimaryDataStore(parameters); diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java index f1177acc7b4a..cf767b87190d 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java @@ -18,6 +18,7 @@ package org.apache.cloudstack.storage.datastore.manager; import java.util.HashMap; +import java.util.List; import java.util.Map; import javax.inject.Inject; @@ -30,6 +31,7 @@ import org.apache.cloudstack.storage.datastore.client.ScaleIOGatewayClient; import org.apache.cloudstack.storage.datastore.client.ScaleIOGatewayClientConnectionPool; import org.apache.cloudstack.storage.datastore.db.PrimaryDataStoreDao; +import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailVO; import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailsDao; import org.apache.cloudstack.storage.datastore.db.StoragePoolVO; import org.apache.commons.collections.MapUtils; @@ -121,7 +123,11 @@ public String prepareSDC(Host host, DataStore dataStore) { return getConnectedSdc(host, dataStore); } - String systemId = storagePoolDetailsDao.findDetail(dataStore.getId(), ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID).getValue(); + String systemId = null; + StoragePoolDetailVO systemIdDetail = storagePoolDetailsDao.findDetail(dataStore.getId(), ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID); + if (systemIdDetail != null) { + systemId = systemIdDetail.getValue(); + } if (systemId == null) { throw new CloudRuntimeException("Unable to prepare SDC, failed to get the system id for PowerFlex storage pool: " + dataStore.getName()); } @@ -168,7 +174,19 @@ public String prepareSDC(Host host, DataStore dataStore) { throw new CloudRuntimeException(errorMsg); } - sdcId = prepareSDCOnHost(host, dataStore, systemId); + String mdms = null; + StoragePoolDetailVO mdmsDetail = storagePoolDetailsDao.findDetail(dataStore.getId(), ScaleIOGatewayClient.STORAGE_POOL_MDMS); + if (mdmsDetail != null) { + mdms = mdmsDetail.getValue(); + } + if (StringUtils.isBlank(mdms)) { + List mdmAddresses = getMdms(dataStore.getId()); + mdms = StringUtils.join(mdmAddresses, ","); + StoragePoolDetailVO storagePoolDetailVO = new StoragePoolDetailVO(dataStore.getId(), ScaleIOGatewayClient.STORAGE_POOL_MDMS, mdms, false); + storagePoolDetailsDao.persist(storagePoolDetailVO); + } + + sdcId = prepareSDCOnHost(host, dataStore, systemId, mdms); StoragePoolHostVO storagePoolHost = storagePoolHostDao.findByPoolHost(poolId, hostId); if (StringUtils.isBlank(sdcId)) { @@ -202,10 +220,11 @@ public String prepareSDC(Host host, DataStore dataStore) { } } - private String prepareSDCOnHost(Host host, DataStore dataStore, String systemId) { + private String prepareSDCOnHost(Host host, DataStore dataStore, String systemId, String mdms) { logger.debug("Preparing SDC on the host {}", host); Map details = new HashMap<>(); details.put(ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID, systemId); + details.put(ScaleIOGatewayClient.STORAGE_POOL_MDMS, mdms); PrepareStorageClientCommand cmd = new PrepareStorageClientCommand(((PrimaryDataStore) dataStore).getPoolType(), dataStore.getUuid(), details); int timeoutSeconds = 60; cmd.setWait(timeoutSeconds); @@ -260,7 +279,11 @@ public boolean stopSDC(Host host, DataStore dataStore) { return true; } - String systemId = storagePoolDetailsDao.findDetail(dataStore.getId(), ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID).getValue(); + String systemId = null; + StoragePoolDetailVO systemIdDetail = storagePoolDetailsDao.findDetail(dataStore.getId(), ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID); + if (systemIdDetail != null) { + systemId = systemIdDetail.getValue(); + } if (systemId == null) { throw new CloudRuntimeException("Unable to unprepare SDC, failed to get the system id for PowerFlex storage pool: " + dataStore); } @@ -287,7 +310,19 @@ public boolean stopSDC(Host host, DataStore dataStore) { return true; } - return unprepareSDCOnHost(host, dataStore); + String mdms = null; + StoragePoolDetailVO mdmsDetail = storagePoolDetailsDao.findDetail(dataStore.getId(), ScaleIOGatewayClient.STORAGE_POOL_MDMS); + if (mdmsDetail != null) { + mdms = mdmsDetail.getValue(); + } + if (StringUtils.isBlank(mdms)) { + List mdmAddresses = getMdms(dataStore.getId()); + mdms = StringUtils.join(mdmAddresses, ","); + StoragePoolDetailVO storagePoolDetailVO = new StoragePoolDetailVO(dataStore.getId(), ScaleIOGatewayClient.STORAGE_POOL_MDMS, mdms, false); + storagePoolDetailsDao.persist(storagePoolDetailVO); + } + + return unprepareSDCOnHost(host, dataStore, mdms); } finally { if (lock != null) { lock.unlock(); @@ -296,9 +331,11 @@ public boolean stopSDC(Host host, DataStore dataStore) { } } - private boolean unprepareSDCOnHost(Host host, DataStore dataStore) { + private boolean unprepareSDCOnHost(Host host, DataStore dataStore, String mdms) { logger.debug(String.format("Unpreparing SDC on the host %s (%s)", host.getId(), host.getName())); - UnprepareStorageClientCommand cmd = new UnprepareStorageClientCommand(((PrimaryDataStore) dataStore).getPoolType(), dataStore.getUuid()); + Map details = new HashMap<>(); + details.put(ScaleIOGatewayClient.STORAGE_POOL_MDMS, mdms); + UnprepareStorageClientCommand cmd = new UnprepareStorageClientCommand(((PrimaryDataStore) dataStore).getPoolType(), dataStore.getUuid(), details); int timeoutSeconds = 60; cmd.setWait(timeoutSeconds); @@ -369,6 +406,16 @@ private boolean hostSdcConnected(String sdcId, DataStore dataStore, int waitTime return isHostSdcConnected(sdcId, poolId); } + private List getMdms(long poolId) { + try { + final ScaleIOGatewayClient client = getScaleIOClient(poolId); + return client.getMdmAddresses(); + } catch (Exception e) { + logger.error("Failed to get MDMs", e); + throw new CloudRuntimeException("Failed to fetch PowerFlex MDM details"); + } + } + private boolean isHostSdcConnected(String sdcId, long poolId) { try { final ScaleIOGatewayClient client = getScaleIOClient(poolId); diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/provider/ScaleIOHostListener.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/provider/ScaleIOHostListener.java index 5fc4868902eb..a4f838cd4861 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/provider/ScaleIOHostListener.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/provider/ScaleIOHostListener.java @@ -31,6 +31,7 @@ import org.apache.cloudstack.storage.datastore.client.ScaleIOGatewayClient; import org.apache.cloudstack.storage.datastore.client.ScaleIOGatewayClientConnectionPool; import org.apache.cloudstack.storage.datastore.db.PrimaryDataStoreDao; +import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailVO; import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailsDao; import org.apache.cloudstack.storage.datastore.db.StoragePoolVO; import org.apache.commons.collections.MapUtils; @@ -98,7 +99,11 @@ public boolean hostConnect(long hostId, long poolId) { private String getSdcIdOfHost(HostVO host, StoragePool storagePool) { long hostId = host.getId(); long poolId = storagePool.getId(); - String systemId = _storagePoolDetailsDao.findDetail(poolId, ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID).getValue(); + String systemId = null; + StoragePoolDetailVO systemIdDetail = _storagePoolDetailsDao.findDetail(poolId, ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID); + if (systemIdDetail != null) { + systemId = systemIdDetail.getValue(); + } if (systemId == null) { throw new CloudRuntimeException("Failed to get the system id for PowerFlex storage pool " + storagePool.getName()); } diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/util/ScaleIOUtil.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/util/ScaleIOUtil.java index 4bb8df9b60db..7837af4ea317 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/util/ScaleIOUtil.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/util/ScaleIOUtil.java @@ -17,6 +17,8 @@ package org.apache.cloudstack.storage.datastore.util; +import java.util.List; + import org.apache.logging.log4j.Logger; import org.apache.logging.log4j.LogManager; @@ -85,6 +87,42 @@ public class ScaleIOUtil { */ private static final String QUERY_MDMS_CMD = "drv_cfg --query_mdms"; + private static final String ADD_MDMS_CMD = "drv_cfg --add_mdm"; + private static final String DRV_CFG_FILE = "/etc/emc/scaleio/drv_cfg.txt"; + + public static void addMdms(List mdmAddresses) { + // Sample Cmd - /opt/emc/scaleio/sdc/bin/drv_cfg --add_mdm --ip x.x.x.x,x.x.x.x --file /etc/emc/scaleio/drv_cfg.txt + String addMdmsCmd = ScaleIOUtil.SDC_HOME_PATH + "/bin/" + ScaleIOUtil.ADD_MDMS_CMD; + addMdmsCmd += " --ip " + String.join(",", mdmAddresses); + addMdmsCmd += " --file " + DRV_CFG_FILE; + String result = Script.runSimpleBashScript(addMdmsCmd); + if (result == null) { + LOGGER.warn("Failed to add mdms"); + } + } + + public static void removeMdms(List mdmAddresses) { + // (i) Remove MDMs from config file (ii) Restart scini + // Sample Cmd - sed -i '/x.x.x.x\,/d' /etc/emc/scaleio/drv_cfg.txt + String removeMdmsCmdFormat = "sed -i '/%s\\,/d' %s"; + for (String mdmAddress : mdmAddresses) { + String removeMdmsCmd = String.format(removeMdmsCmdFormat, mdmAddress, DRV_CFG_FILE); + Script.runSimpleBashScript(removeMdmsCmd); + } + restartSDCService(); + } + + public static boolean mdmAdded(String mdmAddress) { + //query_mdms outputs "MDM-ID SDC ID INSTALLATION ID IPs [0]-x.x.x.x [1]-x.x.x.x" for a MDM with ID: + String queryMdmsCmd = ScaleIOUtil.SDC_HOME_PATH + "/bin/" + ScaleIOUtil.QUERY_MDMS_CMD; + queryMdmsCmd += "|grep " + mdmAddress; + String result = Script.runSimpleBashScript(queryMdmsCmd); + if (StringUtils.isNotBlank(result) && result.contains(mdmAddress)) { + return true; + } + return false; + } + public static String getSdcHomePath() { String sdcHomePath = DEFAULT_SDC_HOME_PATH; String sdcHomePropertyCmdFormat = "sed -n '/%s/p' '%s' 2>/dev/null | sed 's/%s=//g' 2>/dev/null"; From d161485b2ecfc94c9e2fe5b1cfd8124aaa18f040 Mon Sep 17 00:00:00 2001 From: Suresh Kumar Anaparti Date: Tue, 24 Dec 2024 10:19:05 +0530 Subject: [PATCH 2/7] Add/Remove MDM IP addresses during Host connection/disconnection to/from storage pool when powerflex.connect.on.demand is false --- .../agent/api/ModifyStoragePoolAnswer.java | 4 + .../api/storage/DataStoreDriver.java | 4 + .../datastore/db/PrimaryDataStoreDao.java | 4 + .../datastore/db/PrimaryDataStoreDaoImpl.java | 14 ++- .../datastore/PrimaryDataStoreHelper.java | 19 +++- ...ibvirtModifyStoragePoolCommandWrapper.java | 12 +- .../kvm/storage/KVMStoragePoolManager.java | 24 +++- .../kvm/storage/ScaleIOStorageAdaptor.java | 34 +++++- .../kvm/storage/StorageAdaptor.java | 10 +- .../storage/ScaleIOStorageAdaptorTest.java | 20 ++-- .../driver/ScaleIOPrimaryDataStoreDriver.java | 9 +- .../ScaleIOPrimaryDataStoreLifeCycle.java | 45 +++++++- .../datastore/manager/ScaleIOSDCManager.java | 49 +++++++- .../manager/ScaleIOSDCManagerImpl.java | 80 ++++++------- .../provider/ScaleIOHostListener.java | 105 +++++++++++++----- .../cloud/storage/StoragePoolAutomation.java | 10 +- .../storage/StoragePoolAutomationImpl.java | 23 +++- 17 files changed, 362 insertions(+), 104 deletions(-) diff --git a/core/src/main/java/com/cloud/agent/api/ModifyStoragePoolAnswer.java b/core/src/main/java/com/cloud/agent/api/ModifyStoragePoolAnswer.java index 552ffb85aaf6..a3401d6faed7 100644 --- a/core/src/main/java/com/cloud/agent/api/ModifyStoragePoolAnswer.java +++ b/core/src/main/java/com/cloud/agent/api/ModifyStoragePoolAnswer.java @@ -46,6 +46,10 @@ public ModifyStoragePoolAnswer(ModifyStoragePoolCommand cmd, long capacityBytes, templateInfo = tInfo; } + public ModifyStoragePoolAnswer(ModifyStoragePoolCommand cmd, boolean success, String details) { + super(cmd, success, details); + } + public void setPoolInfo(StoragePoolInfo poolInfo) { this.poolInfo = poolInfo; } diff --git a/engine/api/src/main/java/org/apache/cloudstack/engine/subsystem/api/storage/DataStoreDriver.java b/engine/api/src/main/java/org/apache/cloudstack/engine/subsystem/api/storage/DataStoreDriver.java index b197afad863a..af3b38b368c3 100644 --- a/engine/api/src/main/java/org/apache/cloudstack/engine/subsystem/api/storage/DataStoreDriver.java +++ b/engine/api/src/main/java/org/apache/cloudstack/engine/subsystem/api/storage/DataStoreDriver.java @@ -45,4 +45,8 @@ public interface DataStoreDriver { boolean canCopy(DataObject srcData, DataObject destData); void resize(DataObject data, AsyncCompletionCallback callback); + + default boolean canDisplayDetails() { + return true; + } } diff --git a/engine/schema/src/main/java/org/apache/cloudstack/storage/datastore/db/PrimaryDataStoreDao.java b/engine/schema/src/main/java/org/apache/cloudstack/storage/datastore/db/PrimaryDataStoreDao.java index f0c235e842c1..d205379cdb24 100644 --- a/engine/schema/src/main/java/org/apache/cloudstack/storage/datastore/db/PrimaryDataStoreDao.java +++ b/engine/schema/src/main/java/org/apache/cloudstack/storage/datastore/db/PrimaryDataStoreDao.java @@ -60,6 +60,8 @@ public interface PrimaryDataStoreDao extends GenericDao { StoragePoolVO persist(StoragePoolVO pool, Map details, List tags, Boolean isTagARule); + StoragePoolVO persist(StoragePoolVO pool, Map details, List tags, Boolean isTagARule, boolean displayDetails); + /** * Find pool by name. * @@ -103,6 +105,8 @@ public interface PrimaryDataStoreDao extends GenericDao { void updateDetails(long poolId, Map details); + void removeDetails(long poolId); + Map getDetails(long poolId); List searchForStoragePoolTags(long poolId); diff --git a/engine/schema/src/main/java/org/apache/cloudstack/storage/datastore/db/PrimaryDataStoreDaoImpl.java b/engine/schema/src/main/java/org/apache/cloudstack/storage/datastore/db/PrimaryDataStoreDaoImpl.java index 1658fe0a537e..f0769ae0b479 100644 --- a/engine/schema/src/main/java/org/apache/cloudstack/storage/datastore/db/PrimaryDataStoreDaoImpl.java +++ b/engine/schema/src/main/java/org/apache/cloudstack/storage/datastore/db/PrimaryDataStoreDaoImpl.java @@ -296,14 +296,19 @@ public StoragePoolVO listById(Integer id) { } @Override - @DB public StoragePoolVO persist(StoragePoolVO pool, Map details, List tags, Boolean isTagARule) { + return persist(pool, details, tags, isTagARule, true); + } + + @Override + @DB + public StoragePoolVO persist(StoragePoolVO pool, Map details, List tags, Boolean isTagARule, boolean displayDetails) { TransactionLegacy txn = TransactionLegacy.currentTxn(); txn.start(); pool = super.persist(pool); if (details != null) { for (Map.Entry detail : details.entrySet()) { - StoragePoolDetailVO vo = new StoragePoolDetailVO(pool.getId(), detail.getKey(), detail.getValue(), true); + StoragePoolDetailVO vo = new StoragePoolDetailVO(pool.getId(), detail.getKey(), detail.getValue(), displayDetails); _detailsDao.persist(vo); } } @@ -570,6 +575,11 @@ public void updateDetails(long poolId, Map details) { } } + @Override + public void removeDetails(long poolId) { + _detailsDao.removeDetails(poolId); + } + @Override public Map getDetails(long poolId) { return _detailsDao.listDetailsKeyPairs(poolId); diff --git a/engine/storage/src/main/java/org/apache/cloudstack/storage/volume/datastore/PrimaryDataStoreHelper.java b/engine/storage/src/main/java/org/apache/cloudstack/storage/volume/datastore/PrimaryDataStoreHelper.java index 7f28224a3168..66adce76172e 100644 --- a/engine/storage/src/main/java/org/apache/cloudstack/storage/volume/datastore/PrimaryDataStoreHelper.java +++ b/engine/storage/src/main/java/org/apache/cloudstack/storage/volume/datastore/PrimaryDataStoreHelper.java @@ -29,13 +29,16 @@ import com.cloud.dc.dao.ClusterDao; import org.apache.cloudstack.annotation.AnnotationService; import org.apache.cloudstack.annotation.dao.AnnotationDao; -import org.apache.logging.log4j.Logger; import org.apache.logging.log4j.LogManager; +import org.apache.logging.log4j.Logger; import org.springframework.stereotype.Component; import org.apache.cloudstack.engine.subsystem.api.storage.ClusterScope; import org.apache.cloudstack.engine.subsystem.api.storage.DataStore; +import org.apache.cloudstack.engine.subsystem.api.storage.DataStoreDriver; import org.apache.cloudstack.engine.subsystem.api.storage.DataStoreManager; +import org.apache.cloudstack.engine.subsystem.api.storage.DataStoreProvider; +import org.apache.cloudstack.engine.subsystem.api.storage.DataStoreProviderManager; import org.apache.cloudstack.engine.subsystem.api.storage.HostScope; import org.apache.cloudstack.engine.subsystem.api.storage.PrimaryDataStoreParameters; import org.apache.cloudstack.storage.datastore.db.PrimaryDataStoreDao; @@ -78,6 +81,8 @@ public class PrimaryDataStoreHelper { protected ClusterDao clusterDao; @Inject private AnnotationDao annotationDao; + @Inject + DataStoreProviderManager dataStoreProviderMgr; public DataStore createPrimaryDataStore(PrimaryDataStoreParameters params) { if(params == null) @@ -144,7 +149,17 @@ public DataStore createPrimaryDataStore(PrimaryDataStoreParameters params) { storageTags.add(tag); } } - dataStoreVO = dataStoreDao.persist(dataStoreVO, details, storageTags, params.isTagARule()); + + boolean displayDetails = true; + DataStoreProvider storeProvider = dataStoreProviderMgr.getDataStoreProvider(params.getProviderName()); + if (storeProvider != null) { + DataStoreDriver storeDriver = storeProvider.getDataStoreDriver(); + if (storeDriver != null) { + displayDetails = storeDriver.canDisplayDetails(); + } + } + + dataStoreVO = dataStoreDao.persist(dataStoreVO, details, storageTags, params.isTagARule(), displayDetails); return dataStoreMgr.getDataStore(dataStoreVO.getId(), DataStoreRole.Primary); } diff --git a/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/resource/wrapper/LibvirtModifyStoragePoolCommandWrapper.java b/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/resource/wrapper/LibvirtModifyStoragePoolCommandWrapper.java index 7dcedb12340d..06883c708d96 100644 --- a/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/resource/wrapper/LibvirtModifyStoragePoolCommandWrapper.java +++ b/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/resource/wrapper/LibvirtModifyStoragePoolCommandWrapper.java @@ -38,6 +38,17 @@ public final class LibvirtModifyStoragePoolCommandWrapper extends CommandWrapper @Override public Answer execute(final ModifyStoragePoolCommand command, final LibvirtComputingResource libvirtComputingResource) { final KVMStoragePoolManager storagePoolMgr = libvirtComputingResource.getStoragePoolMgr(); + if (!command.getAdd()) { + boolean status = storagePoolMgr.deleteStoragePool(command.getPool().getType(), command.getPool().getUuid(), command.getDetails()); + if (status) { + final ModifyStoragePoolAnswer answer = new ModifyStoragePoolAnswer(command, true, null); + return answer; + } + + final ModifyStoragePoolAnswer answer = new ModifyStoragePoolAnswer(command, false, "Failed to delete storage pool"); + return answer; + } + final KVMStoragePool storagepool = storagePoolMgr.createStoragePool(command.getPool().getUuid(), command.getPool().getHost(), command.getPool().getPort(), command.getPool().getPath(), command.getPool() .getUserInfo(), command.getPool().getType(), command.getDetails()); @@ -47,7 +58,6 @@ public Answer execute(final ModifyStoragePoolCommand command, final LibvirtCompu final Map tInfo = new HashMap(); final ModifyStoragePoolAnswer answer = new ModifyStoragePoolAnswer(command, storagepool.getCapacity(), storagepool.getAvailable(), tInfo, storagepool.getDetails()); - return answer; } } diff --git a/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/KVMStoragePoolManager.java b/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/KVMStoragePoolManager.java index 2cac056413d5..b3bbb55cee34 100644 --- a/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/KVMStoragePoolManager.java +++ b/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/KVMStoragePoolManager.java @@ -411,12 +411,26 @@ public boolean disconnectPhysicalDisk(StoragePoolType type, String poolUuid, Str public boolean deleteStoragePool(StoragePoolType type, String uuid) { StorageAdaptor adaptor = getStorageAdaptor(type); - _haMonitor.removeStoragePool(uuid); - adaptor.deleteStoragePool(uuid); + if (type == StoragePoolType.NetworkFilesystem) { + _haMonitor.removeStoragePool(uuid); + } + boolean deleteStatus = adaptor.deleteStoragePool(uuid);; + synchronized (_storagePools) { + _storagePools.remove(uuid); + } + return deleteStatus; + } + + public boolean deleteStoragePool(StoragePoolType type, String uuid, Map details) { + StorageAdaptor adaptor = getStorageAdaptor(type); + if (type == StoragePoolType.NetworkFilesystem) { + _haMonitor.removeStoragePool(uuid); + } + boolean deleteStatus = adaptor.deleteStoragePool(uuid, details); synchronized (_storagePools) { _storagePools.remove(uuid); } - return true; + return deleteStatus; } public KVMPhysicalDisk createDiskFromTemplate(KVMPhysicalDisk template, String name, Storage.ProvisioningType provisioningType, @@ -480,11 +494,11 @@ public KVMPhysicalDisk createPhysicalDiskFromDirectDownloadTemplate(String templ public Ternary, String> prepareStorageClient(StoragePoolType type, String uuid, Map details) { StorageAdaptor adaptor = getStorageAdaptor(type); - return adaptor.prepareStorageClient(type, uuid, details); + return adaptor.prepareStorageClient(uuid, details); } public Pair unprepareStorageClient(StoragePoolType type, String uuid, Map details) { StorageAdaptor adaptor = getStorageAdaptor(type); - return adaptor.unprepareStorageClient(type, uuid, details); + return adaptor.unprepareStorageClient(uuid, details); } } diff --git a/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/ScaleIOStorageAdaptor.java b/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/ScaleIOStorageAdaptor.java index f77f86835617..29c152f934f6 100644 --- a/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/ScaleIOStorageAdaptor.java +++ b/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/ScaleIOStorageAdaptor.java @@ -28,6 +28,7 @@ import java.util.UUID; import org.apache.cloudstack.storage.datastore.client.ScaleIOGatewayClient; +import org.apache.cloudstack.storage.datastore.manager.ScaleIOSDCManager; import org.apache.cloudstack.storage.datastore.util.ScaleIOUtil; import org.apache.cloudstack.utils.cryptsetup.CryptSetup; import org.apache.cloudstack.utils.cryptsetup.CryptSetupException; @@ -148,12 +149,37 @@ public KVMPhysicalDisk getPhysicalDisk(String volumePath, KVMStoragePool pool) { @Override public KVMStoragePool createStoragePool(String uuid, String host, int port, String path, String userInfo, Storage.StoragePoolType type, Map details, boolean isPrimaryStorage) { ScaleIOStoragePool storagePool = new ScaleIOStoragePool(uuid, host, port, path, type, details, this); + if (details != null && details.containsKey(ScaleIOSDCManager.ConnectOnDemand.key())) { + String connectOnDemand = details.get(ScaleIOSDCManager.ConnectOnDemand.key()); + if (connectOnDemand != null && !Boolean.parseBoolean(connectOnDemand)) { + Ternary, String> prepareStorageClientStatus = prepareStorageClient(uuid, details); + if (prepareStorageClientStatus.first()) { + details.putAll(prepareStorageClientStatus.second()); + } + } + } MapStorageUuidToStoragePool.put(uuid, storagePool); return storagePool; } @Override public boolean deleteStoragePool(String uuid) { + ScaleIOStoragePool storagePool = (ScaleIOStoragePool) MapStorageUuidToStoragePool.get(uuid); + if (storagePool != null) { + unprepareStorageClient(uuid, storagePool.getDetails()); + } + return MapStorageUuidToStoragePool.remove(uuid) != null; + } + + @Override + public boolean deleteStoragePool(String uuid, Map details) { + if (details != null && details.containsKey(ScaleIOSDCManager.ConnectOnDemand.key())) { + String connectOnDemand = details.get(ScaleIOSDCManager.ConnectOnDemand.key()); + if (connectOnDemand != null && !Boolean.parseBoolean(connectOnDemand)) { + Pair unprepareStorageClientStatus = unprepareStorageClient(uuid, details); + return MapStorageUuidToStoragePool.remove(uuid) != null && unprepareStorageClientStatus.first(); + } + } return MapStorageUuidToStoragePool.remove(uuid) != null; } @@ -567,7 +593,7 @@ public void resizeQcow2ToVolume(String volumePath, QemuImageOptions options, Lis qemu.resize(options, objects, usableSizeBytes); } - public Ternary, String> prepareStorageClient(Storage.StoragePoolType type, String uuid, Map details) { + public Ternary, String> prepareStorageClient(String uuid, Map details) { if (!ScaleIOUtil.isSDCServiceInstalled()) { logger.debug("SDC service not installed on host, preparing the SDC client not possible"); return new Ternary<>(false, null, "SDC service not installed on host"); @@ -591,6 +617,10 @@ public Ternary, String> prepareStorageClient(Storag String mdms = details.get(ScaleIOGatewayClient.STORAGE_POOL_MDMS); String[] mdmAddresses = mdms.split(","); if (mdmAddresses.length > 0) { + if (ScaleIOUtil.mdmAdded(mdmAddresses[0])) { + return new Ternary<>(true, getSDCDetails(details), "MDM added, no need to prepare the SDC client"); + } + ScaleIOUtil.addMdms(Arrays.asList(mdmAddresses)); if (!ScaleIOUtil.mdmAdded(mdmAddresses[0])) { return new Ternary<>(false, null, "Failed to add MDMs"); @@ -601,7 +631,7 @@ public Ternary, String> prepareStorageClient(Storag return new Ternary<>( true, getSDCDetails(details), "Prepared client successfully"); } - public Pair unprepareStorageClient(Storage.StoragePoolType type, String uuid, Map details) { + public Pair unprepareStorageClient(String uuid, Map details) { if (!ScaleIOUtil.isSDCServiceInstalled()) { logger.debug("SDC service not installed on host, no need to unprepare the SDC client"); return new Pair<>(true, "SDC service not installed on host, no need to unprepare the SDC client"); diff --git a/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/StorageAdaptor.java b/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/StorageAdaptor.java index 29a6cb769320..76b5a413e70f 100644 --- a/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/StorageAdaptor.java +++ b/plugins/hypervisors/kvm/src/main/java/com/cloud/hypervisor/kvm/storage/StorageAdaptor.java @@ -44,6 +44,10 @@ public interface StorageAdaptor { public boolean deleteStoragePool(String uuid); + public default boolean deleteStoragePool(String uuid, Map details) { + return true; + } + public default KVMPhysicalDisk createPhysicalDisk(String name, KVMStoragePool pool, PhysicalDiskFormat format, Storage.ProvisioningType provisioningType, long size, Long usableSize, byte[] passphrase) { return createPhysicalDisk(name, pool, format, provisioningType, size, passphrase); @@ -127,23 +131,21 @@ default boolean supportsPhysicalDiskCopy(StoragePoolType type) { /** * Prepares the storage client. - * @param type type of the storage pool * @param uuid uuid of the storage pool * @param details any details of the storage pool that are required for client preparation * @return status, client details, & message in case failed */ - default Ternary, String> prepareStorageClient(StoragePoolType type, String uuid, Map details) { + default Ternary, String> prepareStorageClient(String uuid, Map details) { return new Ternary<>(true, new HashMap<>(), ""); } /** * Unprepares the storage client. - * @param type type of the storage pool * @param uuid uuid of the storage pool * @param details any details of the storage pool that are required for client unpreparation * @return status, & message in case failed */ - default Pair unprepareStorageClient(StoragePoolType type, String uuid, Map details) { + default Pair unprepareStorageClient(String uuid, Map details) { return new Pair<>(true, ""); } } diff --git a/plugins/hypervisors/kvm/src/test/java/com/cloud/hypervisor/kvm/storage/ScaleIOStorageAdaptorTest.java b/plugins/hypervisors/kvm/src/test/java/com/cloud/hypervisor/kvm/storage/ScaleIOStorageAdaptorTest.java index 072fd6fbd4be..421fee096341 100644 --- a/plugins/hypervisors/kvm/src/test/java/com/cloud/hypervisor/kvm/storage/ScaleIOStorageAdaptorTest.java +++ b/plugins/hypervisors/kvm/src/test/java/com/cloud/hypervisor/kvm/storage/ScaleIOStorageAdaptorTest.java @@ -34,7 +34,6 @@ import org.mockito.Mockito; import org.mockito.junit.MockitoJUnitRunner; -import com.cloud.storage.Storage; import com.cloud.storage.StorageLayer; import com.cloud.utils.Pair; import com.cloud.utils.Ternary; @@ -73,7 +72,7 @@ public void getUsableBytesFromRawBytesTest() { public void testPrepareStorageClient_SDCServiceNotInstalled() { when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl status scini"))).thenReturn(4); - Ternary, String> result = scaleIOStorageAdaptor.prepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid, new HashMap<>()); + Ternary, String> result = scaleIOStorageAdaptor.prepareStorageClient(poolUuid, new HashMap<>()); Assert.assertFalse(result.first()); Assert.assertNull(result.second()); @@ -86,7 +85,7 @@ public void testPrepareStorageClient_SDCServiceNotEnabled() { when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl is-enabled scini"))).thenReturn(1); when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl enable scini"))).thenReturn(1); - Ternary, String> result = scaleIOStorageAdaptor.prepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid, new HashMap<>()); + Ternary, String> result = scaleIOStorageAdaptor.prepareStorageClient(poolUuid, new HashMap<>()); Assert.assertFalse(result.first()); Assert.assertNull(result.second()); @@ -100,7 +99,7 @@ public void testPrepareStorageClient_SDCServiceNotStarted() { when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl is-active scini"))).thenReturn(1); when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl start scini"))).thenReturn(1); - Ternary, String> result = scaleIOStorageAdaptor.prepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid, new HashMap<>()); + Ternary, String> result = scaleIOStorageAdaptor.prepareStorageClient(poolUuid, new HashMap<>()); Assert.assertFalse(result.first()); Assert.assertNull(result.second()); @@ -121,7 +120,7 @@ public void testPrepareStorageClient_SDCServiceStartedReturnSDCId() { String sdcId = "301b852c00000003"; when(ScaleIOUtil.getSdcId(systemId)).thenReturn(sdcId); - Ternary, String> result = scaleIOStorageAdaptor.prepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid, details); + Ternary, String> result = scaleIOStorageAdaptor.prepareStorageClient(poolUuid, details); Assert.assertTrue(result.first()); Assert.assertNotNull(result.second()); @@ -144,7 +143,8 @@ public void testPrepareStorageClient_SDCServiceStartedReturnSDCGuid() { when(ScaleIOUtil.getSdcId(systemId)).thenReturn(null); when(ScaleIOUtil.getSdcGuid()).thenReturn(sdcGuid); - Ternary, String> result = scaleIOStorageAdaptor.prepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid, details); + Ternary, String> result = scaleIOStorageAdaptor.prepareStorageClient(poolUuid, details); + Assert.assertTrue(result.first()); Assert.assertNotNull(result.second()); Assert.assertEquals(sdcGuid, result.second().get(ScaleIOGatewayClient.SDC_GUID)); @@ -156,7 +156,7 @@ public void testUnprepareStorageClient_SDCServiceNotInstalled() { Map details = new HashMap<>(); when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl status scini"))).thenReturn(4); - Pair result = scaleIOStorageAdaptor.unprepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid, details); + Pair result = scaleIOStorageAdaptor.unprepareStorageClient(poolUuid, details); Assert.assertTrue(result.first()); Assert.assertEquals("SDC service not installed on host, no need to unprepare the SDC client", result.second()); @@ -168,7 +168,7 @@ public void testUnprepareStorageClient_SDCServiceNotEnabled() { when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl status scini"))).thenReturn(3); when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl is-enabled scini"))).thenReturn(1); - Pair result = scaleIOStorageAdaptor.unprepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid, details); + Pair result = scaleIOStorageAdaptor.unprepareStorageClient(poolUuid, details); Assert.assertTrue(result.first()); Assert.assertEquals("SDC service not enabled on host, no need to unprepare the SDC client", result.second()); @@ -182,7 +182,7 @@ public void testUnprepareStorageClient_MDMNotAdded() { when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl is-enabled scini"))).thenReturn(0); when(Script.runSimpleBashScript(Mockito.eq("/opt/emc/scaleio/sdc/bin/drv_cfg --query_mdms|grep 1.1.1.1"))).thenReturn("MDM-ID 71fd458f0775010f SDC ID 4421a91a00000000 INSTALLATION ID 204930df2cbcaf8e IPs [0]-3.3.3.3 [1]-4.4.4.4"); - Pair result = scaleIOStorageAdaptor.unprepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid, details); + Pair result = scaleIOStorageAdaptor.unprepareStorageClient(poolUuid, details); Assert.assertTrue(result.first()); Assert.assertEquals("MDM not added, no need to unprepare the SDC client", result.second()); @@ -197,7 +197,7 @@ public void testUnprepareStorageClient_RemoveMDMFailed() { when(Script.runSimpleBashScriptForExitValue(Mockito.eq("systemctl restart scini"))).thenReturn(0); when(Script.runSimpleBashScript(Mockito.eq("/opt/emc/scaleio/sdc/bin/drv_cfg --query_mdms|grep 1.1.1.1"))).thenReturn("MDM-ID 71fd458f0775010f SDC ID 4421a91a00000000 INSTALLATION ID 204930df2cbcaf8e IPs [0]-1.1.1.1 [1]-2.2.2.2"); - Pair result = scaleIOStorageAdaptor.unprepareStorageClient(Storage.StoragePoolType.PowerFlex, poolUuid, details); + Pair result = scaleIOStorageAdaptor.unprepareStorageClient(poolUuid, details); Assert.assertFalse(result.first()); Assert.assertEquals("Failed to remove MDMs, unable to unprepare the SDC client", result.second()); diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java index d77ad1bb6c16..8441d222d4df 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java @@ -266,7 +266,7 @@ public void revokeAccess(DataObject dataObject, Host host, DataStore dataStore) } if (client.listVolumesMappedToSdc(sdcId).isEmpty()) { sdcManager = ComponentContext.inject(sdcManager); - sdcManager.stopSDC(host, dataStore); + sdcManager.unprepareSDC(host, dataStore); } } catch (Exception e) { logger.warn("Failed to revoke access due to: " + e.getMessage(), e); @@ -294,7 +294,7 @@ public void revokeVolumeAccess(String volumePath, Host host, DataStore dataStore client.unmapVolumeFromSdc(ScaleIOUtil.getVolumePath(volumePath), sdcId); if (client.listVolumesMappedToSdc(sdcId).isEmpty()) { sdcManager = ComponentContext.inject(sdcManager); - sdcManager.stopSDC(host, dataStore); + sdcManager.unprepareSDC(host, dataStore); } } catch (Exception e) { logger.warn("Failed to revoke access due to: " + e.getMessage(), e); @@ -1550,4 +1550,9 @@ public boolean volumesRequireGrantAccessWhenUsed() { public boolean zoneWideVolumesAvailableWithoutClusterMotion() { return true; } + + @Override + public boolean canDisplayDetails() { + return false; + } } diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/lifecycle/ScaleIOPrimaryDataStoreLifeCycle.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/lifecycle/ScaleIOPrimaryDataStoreLifeCycle.java index 1a2fc96e8ef0..f21abae2d319 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/lifecycle/ScaleIOPrimaryDataStoreLifeCycle.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/lifecycle/ScaleIOPrimaryDataStoreLifeCycle.java @@ -25,6 +25,7 @@ import java.security.KeyManagementException; import java.security.NoSuchAlgorithmException; import java.util.ArrayList; +import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.UUID; @@ -32,7 +33,10 @@ import javax.inject.Inject; import org.apache.cloudstack.storage.datastore.client.ScaleIOGatewayClientConnectionPool; +import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailVO; import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailsDao; +import org.apache.cloudstack.storage.datastore.manager.ScaleIOSDCManager; +import org.apache.cloudstack.storage.datastore.manager.ScaleIOSDCManagerImpl; import org.apache.cloudstack.storage.datastore.util.ScaleIOUtil; import org.apache.commons.collections.CollectionUtils; import org.apache.cloudstack.engine.subsystem.api.storage.ClusterScope; @@ -71,6 +75,7 @@ import com.cloud.storage.dao.StoragePoolHostDao; import com.cloud.template.TemplateManager; import com.cloud.utils.UriUtils; +import com.cloud.utils.component.ComponentContext; import com.cloud.utils.crypt.DBEncryptionUtil; import com.cloud.utils.exception.CloudRuntimeException; @@ -97,8 +102,10 @@ public class ScaleIOPrimaryDataStoreLifeCycle extends BasePrimaryDataStoreLifeCy private TemplateManager templateMgr; @Inject private AgentManager agentMgr; + private ScaleIOSDCManager sdcManager; public ScaleIOPrimaryDataStoreLifeCycle() { + sdcManager = new ScaleIOSDCManagerImpl(); } private org.apache.cloudstack.storage.datastore.api.StoragePool findStoragePool(String url, String username, String password, String storagePoolName) { @@ -322,15 +329,43 @@ public boolean attachZone(DataStore dataStore, ZoneScope scope, Hypervisor.Hyper @Override public boolean maintain(DataStore store) { - storagePoolAutomation.maintain(store); + Map details = new HashMap<>(); + StoragePoolDetailVO systemIdDetail = storagePoolDetailsDao.findDetail(store.getId(), ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID); + if (systemIdDetail != null) { + details.put(ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID, systemIdDetail.getValue()); + StoragePoolDetailVO mdmsDetail = storagePoolDetailsDao.findDetail(store.getId(), ScaleIOGatewayClient.STORAGE_POOL_MDMS); + if (mdmsDetail != null) { + details.put(ScaleIOGatewayClient.STORAGE_POOL_MDMS, mdmsDetail.getValue()); + } + details.put(ScaleIOSDCManager.ConnectOnDemand.key(), "false"); + } + + storagePoolAutomation.maintain(store, details); dataStoreHelper.maintain(store); return true; } @Override public boolean cancelMaintain(DataStore store) { + Map details = new HashMap<>(); + StoragePoolDetailVO systemIdDetail = storagePoolDetailsDao.findDetail(store.getId(), ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID); + if (systemIdDetail != null) { + details.put(ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID, systemIdDetail.getValue()); + sdcManager = ComponentContext.inject(sdcManager); + if (sdcManager.areSDCConnectionsWithinLimit(store.getId())) { + StoragePoolVO storagePoolVO = primaryDataStoreDao.findById(store.getId()); + if (storagePoolVO != null) { + details.put(ScaleIOSDCManager.ConnectOnDemand.key(), String.valueOf(ScaleIOSDCManager.ConnectOnDemand.valueIn(storagePoolVO.getDataCenterId()))); + StoragePoolDetailVO mdmsDetail = storagePoolDetailsDao.findDetail(store.getId(), ScaleIOGatewayClient.STORAGE_POOL_MDMS); + if (mdmsDetail != null) { + details.put(ScaleIOGatewayClient.STORAGE_POOL_MDMS, mdmsDetail.getValue()); + } + } + } + } + dataStoreHelper.cancelMaintain(store); - storagePoolAutomation.cancelMaintain(store); + storagePoolAutomation.cancelMaintain(store, details); return true; } @@ -376,7 +411,11 @@ public boolean deleteDataStore(DataStore dataStore) { ScaleIOGatewayClientConnectionPool.getInstance().removeClient(dataStore); - return dataStoreHelper.deletePrimaryDataStore(dataStore); + boolean isDeleted = dataStoreHelper.deletePrimaryDataStore(dataStore); + if (isDeleted) { + primaryDataStoreDao.removeDetails(dataStore.getId()); + } + return isDeleted; } @Override diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManager.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManager.java index 696643cb17a1..e4f0eeef66fb 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManager.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManager.java @@ -18,10 +18,20 @@ package org.apache.cloudstack.storage.datastore.manager; import org.apache.cloudstack.engine.subsystem.api.storage.DataStore; +import org.apache.cloudstack.framework.config.ConfigKey; import com.cloud.host.Host; public interface ScaleIOSDCManager { + ConfigKey ConnectOnDemand = new ConfigKey<>("Storage", + Boolean.class, + "powerflex.connect.on.demand", + Boolean.FALSE.toString(), + "Connect PowerFlex client on Host when first Volume is mapped to SDC and disconnect when last Volume is unmapped from SDC," + + " otherwise no action (that is connection remains in the same state whichever it is, connected or disconnected).", + Boolean.TRUE, + ConfigKey.Scope.Zone); + /** * Checks SDC connections limit. * @param storagePoolId the storage pool id @@ -30,18 +40,49 @@ public interface ScaleIOSDCManager { boolean areSDCConnectionsWithinLimit(Long storagePoolId); /** - * Prepares/starts the SDC on the host. + * Returns connected SDC Id. * @param host the host * @param dataStore the datastore * @return SDC Id of the host */ + String getConnectedSdc(Host host, DataStore dataStore); + + /** + * Prepares the SDC on the host (adds the MDM IPs to SDC, starts scini service if required). + * @param host the host + * @param dataStore the datastore + * @return SDC Id of the host if SDC is successfully prepared-ed on the hostgetSdcIdByGuid + */ String prepareSDC(Host host, DataStore dataStore); /** - * Stops the SDC on the host. + * Unprepares the SDC on the host (removes the MDM IPs from SDC, restarts scini service). * @param host the host * @param dataStore the datastore - * @return true if SDC stopped on the host + * @return true if SDC is successfully unprepared-ed on the host + */ + boolean unprepareSDC(Host host, DataStore dataStore); + + /** + * Returns the SDC Id of the host for the pool. + * @param sdcGuid the SDC GUID + * @param poolId the pool id + * @return SDC Id of the host for the pool + */ + String getHostSdcId(String sdcGuid, long poolId); + + /** + * Returns the connection status of host SDC of the pool. + * @param sdcId the SDC id + * @param poolId the pool id + * @return true if Host SDC is connected to the pool + */ + boolean isHostSdcConnected(String sdcId, long poolId, int waitTimeInSecs); + + /** + * Returns the comma-separated list of MDM IPs of the pool. + * @param poolId the pool id + * @return Comma-separated list of MDM IPs of the pool */ - boolean stopSDC(Host host, DataStore dataStore); + String getMdms(long poolId); } diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java index cf767b87190d..4d01032d040d 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java @@ -34,6 +34,7 @@ import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailVO; import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailsDao; import org.apache.cloudstack.storage.datastore.db.StoragePoolVO; +import org.apache.commons.collections.CollectionUtils; import org.apache.commons.collections.MapUtils; import org.apache.commons.lang3.StringUtils; import org.apache.logging.log4j.LogManager; @@ -60,15 +61,6 @@ public class ScaleIOSDCManagerImpl implements ScaleIOSDCManager, Configurable { private Logger logger = LogManager.getLogger(getClass()); - static ConfigKey ConnectOnDemand = new ConfigKey<>("Storage", - Boolean.class, - "powerflex.connect.on.demand", - Boolean.FALSE.toString(), - "Connect PowerFlex client on Host when first Volume is mapped to SDC and disconnect when last Volume is unmapped from SDC," + - " otherwise no action (that is connection remains in the same state whichever it is, connected or disconnected).", - Boolean.TRUE, - ConfigKey.Scope.Zone); - @Inject AgentManager agentManager; @Inject @@ -93,6 +85,7 @@ public boolean areSDCConnectionsWithinLimit(Long storagePoolId) { try { int connectedClientsLimit = StorageManager.STORAGE_POOL_CONNECTED_CLIENTS_LIMIT.valueIn(storagePoolId); if (connectedClientsLimit <= 0) { + logger.debug(String.format("SDC connections limit (unlimited) on PowerFlex Storage with pool id: %d", storagePoolId)); return true; } @@ -174,18 +167,7 @@ public String prepareSDC(Host host, DataStore dataStore) { throw new CloudRuntimeException(errorMsg); } - String mdms = null; - StoragePoolDetailVO mdmsDetail = storagePoolDetailsDao.findDetail(dataStore.getId(), ScaleIOGatewayClient.STORAGE_POOL_MDMS); - if (mdmsDetail != null) { - mdms = mdmsDetail.getValue(); - } - if (StringUtils.isBlank(mdms)) { - List mdmAddresses = getMdms(dataStore.getId()); - mdms = StringUtils.join(mdmAddresses, ","); - StoragePoolDetailVO storagePoolDetailVO = new StoragePoolDetailVO(dataStore.getId(), ScaleIOGatewayClient.STORAGE_POOL_MDMS, mdms, false); - storagePoolDetailsDao.persist(storagePoolDetailVO); - } - + String mdms = getMdms(dataStore.getId()); sdcId = prepareSDCOnHost(host, dataStore, systemId, mdms); StoragePoolHostVO storagePoolHost = storagePoolHostDao.findByPoolHost(poolId, hostId); @@ -204,7 +186,7 @@ public String prepareSDC(Host host, DataStore dataStore) { } int waitTimeInSecs = 15; // Wait for 15 secs (usual tests with SDC service start took 10-15 secs) - if (hostSdcConnected(sdcId, dataStore, waitTimeInSecs)) { + if (isHostSdcConnected(sdcId, dataStore, waitTimeInSecs)) { return sdcId; } return null; @@ -273,7 +255,7 @@ private String prepareSDCOnHost(Host host, DataStore dataStore, String systemId, } @Override - public boolean stopSDC(Host host, DataStore dataStore) { + public boolean unprepareSDC(Host host, DataStore dataStore) { if (Boolean.FALSE.equals(ConnectOnDemand.valueIn(host.getDataCenterId()))) { logger.debug(String.format("On-demand connect/disconnect config %s disabled in the zone %d, no need to unprepare SDC", ConnectOnDemand.key(), host.getDataCenterId())); return true; @@ -307,22 +289,23 @@ public boolean stopSDC(Host host, DataStore dataStore) { String sdcId = getConnectedSdc(host, dataStore); if (StringUtils.isBlank(sdcId)) { logger.debug("SDC not connected, no need to unprepare it"); + StoragePoolHostVO storagePoolHost = storagePoolHostDao.findByPoolHost(dataStore.getId(), host.getId()); + if (storagePoolHost != null) { + storagePoolHostDao.deleteStoragePoolHostDetails(host.getId(), dataStore.getId()); + } return true; } - String mdms = null; - StoragePoolDetailVO mdmsDetail = storagePoolDetailsDao.findDetail(dataStore.getId(), ScaleIOGatewayClient.STORAGE_POOL_MDMS); - if (mdmsDetail != null) { - mdms = mdmsDetail.getValue(); - } - if (StringUtils.isBlank(mdms)) { - List mdmAddresses = getMdms(dataStore.getId()); - mdms = StringUtils.join(mdmAddresses, ","); - StoragePoolDetailVO storagePoolDetailVO = new StoragePoolDetailVO(dataStore.getId(), ScaleIOGatewayClient.STORAGE_POOL_MDMS, mdms, false); - storagePoolDetailsDao.persist(storagePoolDetailVO); + String mdms = getMdms(dataStore.getId());; + boolean unprepareSDCStatus = unprepareSDCOnHost(host, dataStore, mdms); + if (unprepareSDCStatus) { + StoragePoolHostVO storagePoolHost = storagePoolHostDao.findByPoolHost(dataStore.getId(), host.getId()); + if (storagePoolHost != null) { + storagePoolHostDao.deleteStoragePoolHostDetails(host.getId(), dataStore.getId()); + } } - return unprepareSDCOnHost(host, dataStore, mdms); + return unprepareSDCStatus; } finally { if (lock != null) { lock.unlock(); @@ -354,7 +337,8 @@ private boolean unprepareSDCOnHost(Host host, DataStore dataStore, String mdms) return true; } - private String getHostSdcId(String sdcGuid, DataStore dataStore ) { + @Override + public String getHostSdcId(String sdcGuid, DataStore dataStore) { try { logger.debug("Try to get host SDC Id for pool: {}, with SDC guid {}", dataStore, sdcGuid); ScaleIOGatewayClient client = getScaleIOClient(dataStore.getId()); @@ -365,7 +349,8 @@ private String getHostSdcId(String sdcGuid, DataStore dataStore ) { } } - private String getConnectedSdc(Host host, DataStore dataStore) { + @Override + public String getConnectedSdc(Host host, DataStore dataStore) { long poolId = dataStore.getId(); long hostId = host.getId(); @@ -388,7 +373,8 @@ private String getConnectedSdc(Host host, DataStore dataStore) { return null; } - private boolean hostSdcConnected(String sdcId, DataStore dataStore, int waitTimeInSecs) { + @Override + public boolean isHostSdcConnected(String sdcId, DataStore dataStore, int waitTimeInSecs) { long poolId = dataStore.getId(); logger.debug(String.format("Waiting (for %d secs) for the SDC %s of the pool %s to connect", waitTimeInSecs, sdcId, dataStore)); @@ -406,10 +392,26 @@ private boolean hostSdcConnected(String sdcId, DataStore dataStore, int waitTime return isHostSdcConnected(sdcId, poolId); } - private List getMdms(long poolId) { + @Override + public String getMdms(long poolId) { + String mdms = null; + StoragePoolDetailVO mdmsDetail = storagePoolDetailsDao.findDetail(poolId, ScaleIOGatewayClient.STORAGE_POOL_MDMS); + if (mdmsDetail != null) { + mdms = mdmsDetail.getValue(); + } + if (StringUtils.isNotBlank(mdms)) { + return mdms; + } + try { final ScaleIOGatewayClient client = getScaleIOClient(poolId); - return client.getMdmAddresses(); + List mdmAddresses = client.getMdmAddresses(); + if (CollectionUtils.isNotEmpty(mdmAddresses)) { + mdms = StringUtils.join(mdmAddresses, ","); + StoragePoolDetailVO storagePoolDetailVO = new StoragePoolDetailVO(poolId, ScaleIOGatewayClient.STORAGE_POOL_MDMS, mdms, false); + storagePoolDetailsDao.persist(storagePoolDetailVO); + } + return mdms; } catch (Exception e) { logger.error("Failed to get MDMs", e); throw new CloudRuntimeException("Failed to fetch PowerFlex MDM details"); diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/provider/ScaleIOHostListener.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/provider/ScaleIOHostListener.java index a4f838cd4861..5c240fa6fce0 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/provider/ScaleIOHostListener.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/provider/ScaleIOHostListener.java @@ -26,14 +26,16 @@ import javax.inject.Inject; +import org.apache.cloudstack.engine.subsystem.api.storage.DataStore; import org.apache.cloudstack.engine.subsystem.api.storage.DataStoreManager; import org.apache.cloudstack.engine.subsystem.api.storage.HypervisorHostListener; import org.apache.cloudstack.storage.datastore.client.ScaleIOGatewayClient; -import org.apache.cloudstack.storage.datastore.client.ScaleIOGatewayClientConnectionPool; import org.apache.cloudstack.storage.datastore.db.PrimaryDataStoreDao; import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailVO; import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailsDao; import org.apache.cloudstack.storage.datastore.db.StoragePoolVO; +import org.apache.cloudstack.storage.datastore.manager.ScaleIOSDCManager; +import org.apache.cloudstack.storage.datastore.manager.ScaleIOSDCManagerImpl; import org.apache.commons.collections.MapUtils; import org.apache.commons.lang3.StringUtils; import org.apache.logging.log4j.Logger; @@ -50,6 +52,7 @@ import com.cloud.storage.StoragePool; import com.cloud.storage.StoragePoolHostVO; import com.cloud.storage.dao.StoragePoolHostDao; +import com.cloud.utils.component.ComponentContext; import com.cloud.utils.exception.CloudRuntimeException; public class ScaleIOHostListener implements HypervisorHostListener { @@ -62,6 +65,7 @@ public class ScaleIOHostListener implements HypervisorHostListener { @Inject private StoragePoolHostDao _storagePoolHostDao; @Inject private PrimaryDataStoreDao _primaryDataStoreDao; @Inject private StoragePoolDetailsDao _storagePoolDetailsDao; + private ScaleIOSDCManager _sdcManager = new ScaleIOSDCManagerImpl(); @Override public boolean hostAdded(long hostId) { @@ -76,7 +80,8 @@ public boolean hostConnect(long hostId, long poolId) { return false; } - StoragePool storagePool = (StoragePool)_dataStoreMgr.getDataStore(poolId, DataStoreRole.Primary); + DataStore dataStore = _dataStoreMgr.getDataStore(poolId, DataStoreRole.Primary); + StoragePool storagePool = (StoragePool) dataStore; StoragePoolHostVO storagePoolHost = _storagePoolHostDao.findByPoolHost(poolId, hostId); String sdcId = getSdcIdOfHost(host, storagePool); if (StringUtils.isBlank(sdcId)) { @@ -109,6 +114,12 @@ private String getSdcIdOfHost(HostVO host, StoragePool storagePool) { } Map details = new HashMap<>(); details.put(ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID, systemId); + _sdcManager = ComponentContext.inject(_sdcManager); + if (_sdcManager.areSDCConnectionsWithinLimit(poolId)) { + details.put(ScaleIOSDCManager.ConnectOnDemand.key(), String.valueOf(ScaleIOSDCManager.ConnectOnDemand.valueIn(host.getDataCenterId()))); + String mdms = _sdcManager.getMdms(poolId); + details.put(ScaleIOGatewayClient.STORAGE_POOL_MDMS, mdms); + } ModifyStoragePoolCommand cmd = new ModifyStoragePoolCommand(true, storagePool, storagePool.getPath(), details); ModifyStoragePoolAnswer answer = sendModifyStoragePoolCommand(cmd, storagePool, host); @@ -116,7 +127,7 @@ private String getSdcIdOfHost(HostVO host, StoragePool storagePool) { if (MapUtils.isEmpty(poolDetails)) { String msg = String.format("PowerFlex storage SDC details not found on the host: %s, (re)install SDC and restart agent", host); logger.warn(msg); - _alertMgr.sendAlert(AlertManager.AlertType.ALERT_TYPE_HOST, host.getDataCenterId(), host.getPodId(), "SDC not found on host: " + host.getUuid(), msg); + _alertMgr.sendAlert(AlertManager.AlertType.ALERT_TYPE_HOST, host.getDataCenterId(), host.getPodId(), "SDC details not found on host: " + host.getUuid(), msg); return null; } @@ -125,7 +136,7 @@ private String getSdcIdOfHost(HostVO host, StoragePool storagePool) { sdcId = poolDetails.get(ScaleIOGatewayClient.SDC_ID); } else if (poolDetails.containsKey(ScaleIOGatewayClient.SDC_GUID)) { String sdcGuid = poolDetails.get(ScaleIOGatewayClient.SDC_GUID); - sdcId = getHostSdcId(sdcGuid, poolId); + sdcId = _sdcManager.getHostSdcId(sdcGuid, poolId); } if (StringUtils.isBlank(sdcId)) { @@ -135,47 +146,83 @@ private String getSdcIdOfHost(HostVO host, StoragePool storagePool) { return null; } - return sdcId; - } - - private String getHostSdcId(String sdcGuid, long poolId) { - StoragePoolVO storagePool = _primaryDataStoreDao.findById(poolId); - try { - logger.debug(String.format("Try to get host SDC Id for pool: %s, with SDC guid %s", storagePool, sdcGuid)); - ScaleIOGatewayClient client = ScaleIOGatewayClientConnectionPool.getInstance().getClient(storagePool, _storagePoolDetailsDao); - return client.getSdcIdByGuid(sdcGuid); - } catch (NoSuchAlgorithmException | KeyManagementException | URISyntaxException e) { - logger.error(String.format("Failed to get host SDC Id for pool: %s", storagePool), e); - throw new CloudRuntimeException(String.format( - "Failed to establish connection with PowerFlex Gateway to get host SDC Id for pool: %s", - storagePool)); + if (details.containsKey(ScaleIOSDCManager.ConnectOnDemand.key())) { + String connectOnDemand = details.get(ScaleIOSDCManager.ConnectOnDemand.key()); + if (connectOnDemand != null && !Boolean.parseBoolean(connectOnDemand) && !_sdcManager.isHostSdcConnected(sdcId, poolId, 15)) { + logger.warn("SDC not connected on the host: " + hostId); + String msg = "SDC not connected on the host: " + hostId + ", reconnect the SDC to MDM and restart agent"; + _alertMgr.sendAlert(AlertManager.AlertType.ALERT_TYPE_HOST, host.getDataCenterId(), host.getPodId(), "SDC not connected on host: " + host.getUuid(), msg); + return null; + } } + + return sdcId; } private ModifyStoragePoolAnswer sendModifyStoragePoolCommand(ModifyStoragePoolCommand cmd, StoragePool storagePool, HostVO host) { Answer answer = _agentMgr.easySend(host.getId(), cmd); if (answer == null) { - throw new CloudRuntimeException("Unable to get an answer to the modify storage pool command (" + storagePool.getName() + ")"); + throw new CloudRuntimeException(String.format("Unable to get an answer to the modify storage pool command (add: %s) for PowerFlex storage pool %s, sent to host %d", + cmd.getAdd(), getStoragePoolDetails(storagePool), hostId)); } if (!answer.getResult()) { - String msg = "Unable to attach PowerFlex storage pool " + storagePool + " to host " + host.getUuid(); + if (cmd.getAdd()) { + String msg = "Unable to attach PowerFlex storage pool " + getStoragePoolDetails(storagePool) + " to the host " + hostId; + + _alertMgr.sendAlert(AlertManager.AlertType.ALERT_TYPE_HOST, storagePool.getDataCenterId(), storagePool.getPodId(), msg, msg); - _alertMgr.sendAlert(AlertManager.AlertType.ALERT_TYPE_HOST, storagePool.getDataCenterId(), storagePool.getPodId(), msg, msg); + throw new CloudRuntimeException("Unable to connect to PowerFlex storage pool " + getStoragePoolDetails(storagePool) + " due to " + answer.getDetails()); + } else { + String msg = "Unable to detach PowerFlex storage pool " + getStoragePoolDetails(storagePool) + " from the host " + hostId; - throw new CloudRuntimeException("Unable to establish a connection from agent to PowerFlex storage pool " + storagePool + " due to " + answer.getDetails() + - " (" + storagePool.getId() + ")"); + _alertMgr.sendAlert(AlertManager.AlertType.ALERT_TYPE_HOST, storagePool.getDataCenterId(), storagePool.getPodId(), msg, msg); + } } - assert (answer instanceof ModifyStoragePoolAnswer) : "ModifyStoragePoolAnswer expected ; PowerFlex Storage Pool = " + storagePool.getId() + " Host = " + host; + assert (answer instanceof ModifyStoragePoolAnswer) : "ModifyStoragePoolAnswer expected ; PowerFlex Storage Pool = " + storagePool + " Host = " + host; return (ModifyStoragePoolAnswer) answer; } @Override public boolean hostDisconnected(long hostId, long poolId) { - // SDC ID is getting updated upon host connect, no need to delete the storage_pool_host_ref entry + HostVO host = _hostDao.findById(hostId); + if (host == null) { + logger.error("Failed to disconnect host by HostListener as host was not found with id : " + hostId); + return false; + } + + DataStore dataStore = _dataStoreMgr.getDataStore(poolId, DataStoreRole.Primary); + StoragePool storagePool = (StoragePool) dataStore; + String systemId = null; + StoragePoolDetailVO systemIdDetail = _storagePoolDetailsDao.findDetail(poolId, ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID); + if (systemIdDetail != null) { + systemId = systemIdDetail.getValue(); + } + if (systemId == null) { + throw new CloudRuntimeException("Failed to get the system id for PowerFlex storage pool " + storagePool.getName()); + } + Map details = new HashMap<>(); + details.put(ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID, systemId); + details.put(ScaleIOSDCManager.ConnectOnDemand.key(), String.valueOf(ScaleIOSDCManager.ConnectOnDemand.valueIn(host.getDataCenterId()))); + _sdcManager = ComponentContext.inject(_sdcManager); + String mdms = _sdcManager.getMdms(poolId); + details.put(ScaleIOGatewayClient.STORAGE_POOL_MDMS, mdms); + + ModifyStoragePoolCommand cmd = new ModifyStoragePoolCommand(false, storagePool, storagePool.getPath(), details); + ModifyStoragePoolAnswer answer = sendModifyStoragePoolCommand(cmd, storagePool, hostId); + if (!answer.getResult()) { + logger.error("Failed to disconnect storage pool: " + storagePool + " and host: " + hostId); + return false; + } + + StoragePoolHostVO storagePoolHost = _storagePoolHostDao.findByPoolHost(poolId, hostId); + if (storagePoolHost != null) { + _storagePoolHostDao.deleteStoragePoolHostDetails(hostId, poolId); + } + logger.info("Connection removed between storage pool: " + storagePool + " and host: " + hostId); return true; } @@ -193,4 +240,12 @@ public boolean hostRemoved(long hostId, long clusterId) { public boolean hostEnabled(long hostId) { return true; } + + private String getStoragePoolDetails(StoragePool storagePool) { + String poolDetails = ""; + if (storagePool != null) { + poolDetails = String.format("%s (id: %d, uuid: %s)", storagePool.getName(), storagePool.getId(), storagePool.getUuid()); + } + return poolDetails; + } } diff --git a/server/src/main/java/com/cloud/storage/StoragePoolAutomation.java b/server/src/main/java/com/cloud/storage/StoragePoolAutomation.java index 1acd574c2cc6..ea5619182ccb 100644 --- a/server/src/main/java/com/cloud/storage/StoragePoolAutomation.java +++ b/server/src/main/java/com/cloud/storage/StoragePoolAutomation.java @@ -18,10 +18,16 @@ */ package com.cloud.storage; +import java.util.Map; + import org.apache.cloudstack.engine.subsystem.api.storage.DataStore; public interface StoragePoolAutomation { - public boolean maintain(DataStore store); + boolean maintain(DataStore store); + + boolean maintain(DataStore store, Map details); + + boolean cancelMaintain(DataStore store); - public boolean cancelMaintain(DataStore store); + boolean cancelMaintain(DataStore store, Map details); } diff --git a/server/src/main/java/com/cloud/storage/StoragePoolAutomationImpl.java b/server/src/main/java/com/cloud/storage/StoragePoolAutomationImpl.java index 60494dcb05c0..71a879eb2b16 100644 --- a/server/src/main/java/com/cloud/storage/StoragePoolAutomationImpl.java +++ b/server/src/main/java/com/cloud/storage/StoragePoolAutomationImpl.java @@ -31,14 +31,15 @@ import org.apache.cloudstack.storage.datastore.db.PrimaryDataStoreDao; import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailsDao; import org.apache.cloudstack.storage.datastore.db.StoragePoolVO; -import org.apache.logging.log4j.Logger; +import org.apache.commons.collections.MapUtils; import org.apache.logging.log4j.LogManager; +import org.apache.logging.log4j.Logger; import org.springframework.stereotype.Component; import com.cloud.agent.AgentManager; import com.cloud.agent.api.Answer; -import com.cloud.agent.api.ModifyStoragePoolCommand; import com.cloud.agent.api.ModifyStoragePoolAnswer; +import com.cloud.agent.api.ModifyStoragePoolCommand; import com.cloud.alert.AlertManager; import com.cloud.host.HostVO; import com.cloud.host.Status; @@ -112,6 +113,11 @@ public class StoragePoolAutomationImpl implements StoragePoolAutomation { @Override public boolean maintain(DataStore store) { + return maintain(store, null); + } + + @Override + public boolean maintain(DataStore store, Map details) { Long userId = CallContext.current().getCallingUserId(); User user = _userDao.findById(userId); Account account = CallContext.current().getCallingAccount(); @@ -161,6 +167,9 @@ public boolean maintain(DataStore store) { // remove heartbeat for (HostVO host : hosts) { ModifyStoragePoolCommand cmd = new ModifyStoragePoolCommand(false, storagePool); + if (MapUtils.isNotEmpty(details)) { + cmd.setDetails(details); + } final Answer answer = agentMgr.easySend(host.getId(), cmd); if (answer == null || !answer.getResult()) { if (logger.isDebugEnabled()) { @@ -299,6 +308,11 @@ public boolean maintain(DataStore store) { @Override public boolean cancelMaintain(DataStore store) { + return cancelMaintain(store, null); + } + + @Override + public boolean cancelMaintain(DataStore store, Map details) { // Change the storage state back to up Long userId = CallContext.current().getCallingUserId(); User user = _userDao.findById(userId); @@ -327,7 +341,10 @@ public boolean cancelMaintain(DataStore store) { Pair, Boolean> nfsMountOpts = storageManager.getStoragePoolNFSMountOpts(pool, null); // add heartbeat for (HostVO host : hosts) { - ModifyStoragePoolCommand msPoolCmd = new ModifyStoragePoolCommand(true, pool, nfsMountOpts.first()); + ModifyStoragePoolCommand msPoolCmd = new ModifyStoragePoolCommand(true, pool); + if (MapUtils.isNotEmpty(details)) { + msPoolCmd.setDetails(details); + } final Answer answer = agentMgr.easySend(host.getId(), msPoolCmd); if (answer == null || !answer.getResult()) { if (logger.isDebugEnabled()) { From 0c02fcee53f04747b4b510bf982eadde1e04b752 Mon Sep 17 00:00:00 2001 From: Suresh Kumar Anaparti Date: Thu, 26 Dec 2024 11:40:04 +0530 Subject: [PATCH 3/7] unit test fixes --- .../storage/datastore/manager/ScaleIOSDCManager.java | 2 +- .../lifecycle/ScaleIOPrimaryDataStoreLifeCycleTest.java | 5 +++-- 2 files changed, 4 insertions(+), 3 deletions(-) diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManager.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManager.java index e4f0eeef66fb..820b75722e9e 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManager.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManager.java @@ -51,7 +51,7 @@ public interface ScaleIOSDCManager { * Prepares the SDC on the host (adds the MDM IPs to SDC, starts scini service if required). * @param host the host * @param dataStore the datastore - * @return SDC Id of the host if SDC is successfully prepared-ed on the hostgetSdcIdByGuid + * @return SDC Id of the host if SDC is successfully prepared-ed on the host */ String prepareSDC(Host host, DataStore dataStore); diff --git a/plugins/storage/volume/scaleio/src/test/java/org/apache/cloudstack/storage/datastore/lifecycle/ScaleIOPrimaryDataStoreLifeCycleTest.java b/plugins/storage/volume/scaleio/src/test/java/org/apache/cloudstack/storage/datastore/lifecycle/ScaleIOPrimaryDataStoreLifeCycleTest.java index dbeba0e4bded..9932d815ae9c 100644 --- a/plugins/storage/volume/scaleio/src/test/java/org/apache/cloudstack/storage/datastore/lifecycle/ScaleIOPrimaryDataStoreLifeCycleTest.java +++ b/plugins/storage/volume/scaleio/src/test/java/org/apache/cloudstack/storage/datastore/lifecycle/ScaleIOPrimaryDataStoreLifeCycleTest.java @@ -22,6 +22,7 @@ import static com.google.common.truth.Truth.assertThat; import static org.mockito.ArgumentMatchers.any; import static org.mockito.ArgumentMatchers.anyLong; +import static org.mockito.ArgumentMatchers.anyMap; import static org.mockito.ArgumentMatchers.eq; import static org.mockito.Mockito.lenient; import static org.mockito.Mockito.mock; @@ -173,7 +174,7 @@ public void testAttachZone_UnsupportedHypervisor() throws Exception { @Test public void testMaintain() { final DataStore store = mock(DataStore.class); - when(storagePoolAutomation.maintain(any(DataStore.class))).thenReturn(true); + when(storagePoolAutomation.maintain(any(DataStore.class), anyMap())).thenReturn(true); when(dataStoreHelper.maintain(any(DataStore.class))).thenReturn(true); final boolean result = scaleIOPrimaryDataStoreLifeCycleTest.maintain(store); assertThat(result).isTrue(); @@ -183,7 +184,7 @@ public void testMaintain() { public void testCancelMaintain() { final DataStore store = mock(DataStore.class); when(dataStoreHelper.cancelMaintain(any(DataStore.class))).thenReturn(true); - when(storagePoolAutomation.cancelMaintain(any(DataStore.class))).thenReturn(true); + when(storagePoolAutomation.cancelMaintain(any(DataStore.class), anyMap())).thenReturn(true); final boolean result = scaleIOPrimaryDataStoreLifeCycleTest.cancelMaintain(store); assertThat(result).isTrue(); } From 820107cdb89a66b8e99fc6ada2fad09a86c4dab5 Mon Sep 17 00:00:00 2001 From: Suresh Kumar Anaparti Date: Fri, 27 Dec 2024 18:58:02 +0530 Subject: [PATCH 4/7] Don't remove MDM IPs from SDC when any volumes mapped to SDC --- .../api/storage/PrimaryDataStoreDriver.java | 8 +++++ .../com/cloud/storage/StorageManager.java | 2 ++ .../driver/ScaleIOPrimaryDataStoreDriver.java | 25 ++++++++++++++++ .../ScaleIOPrimaryDataStoreLifeCycle.java | 2 +- .../datastore/manager/ScaleIOSDCManager.java | 12 ++++++-- .../manager/ScaleIOSDCManagerImpl.java | 30 +++++++++++++++++++ .../provider/ScaleIOHostListener.java | 8 +++-- .../storage/datastore/util/ScaleIOUtil.java | 15 +++++++++- .../com/cloud/storage/StorageManagerImpl.java | 11 +++++++ .../storage/StoragePoolAutomationImpl.java | 2 +- 10 files changed, 107 insertions(+), 8 deletions(-) diff --git a/engine/api/src/main/java/org/apache/cloudstack/engine/subsystem/api/storage/PrimaryDataStoreDriver.java b/engine/api/src/main/java/org/apache/cloudstack/engine/subsystem/api/storage/PrimaryDataStoreDriver.java index c8d9015af902..d908ef6af508 100644 --- a/engine/api/src/main/java/org/apache/cloudstack/engine/subsystem/api/storage/PrimaryDataStoreDriver.java +++ b/engine/api/src/main/java/org/apache/cloudstack/engine/subsystem/api/storage/PrimaryDataStoreDriver.java @@ -145,6 +145,14 @@ default boolean canHostPrepareStoragePoolAccess(Host host, StoragePool pool) { return false; } + /** + * intended for managed storage + * returns true if the host can be disconnected from storage pool + */ + default boolean canDisconnectHostFromStoragePool(Host host, StoragePool pool) { + return true; + } + /** * Used by storage pools which want to keep VMs' information * @return true if additional VM info is needed (intended for storage pools). diff --git a/engine/components-api/src/main/java/com/cloud/storage/StorageManager.java b/engine/components-api/src/main/java/com/cloud/storage/StorageManager.java index 0b9f7bcb7db4..05822a279ad0 100644 --- a/engine/components-api/src/main/java/com/cloud/storage/StorageManager.java +++ b/engine/components-api/src/main/java/com/cloud/storage/StorageManager.java @@ -308,6 +308,8 @@ static Boolean getFullCloneConfiguration(Long storeId) { boolean canHostPrepareStoragePoolAccess(Host host, StoragePool pool); + boolean canDisconnectHostFromStoragePool(Host host, StoragePool pool); + Host getHost(long hostId); Host updateSecondaryStorage(long secStorageId, String newUrl); diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java index 8441d222d4df..d8a5f872d434 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java @@ -1481,6 +1481,31 @@ public boolean canHostPrepareStoragePoolAccess(Host host, StoragePool pool) { return sdcManager.areSDCConnectionsWithinLimit(pool.getId()); } + @Override + public boolean canDisconnectHostFromStoragePool(Host host, StoragePool pool) { + if (host == null || pool == null) { + return false; + } + + StoragePoolHostVO poolHostVO = storagePoolHostDao.findByPoolHost(pool.getId(), host.getId()); + if (poolHostVO == null) { + return false; + } + + final String sdcId = poolHostVO.getLocalPath(); + if (StringUtils.isBlank(sdcId)) { + return false; + } + + try { + final ScaleIOGatewayClient client = getScaleIOClient(pool.getId()); + return client.listVolumesMappedToSdc(sdcId).isEmpty(); + } catch (Exception e) { + logger.warn("Unable to check whether the host: " + host.getId() + " can be disconnected from storage pool: " + pool.getId() + ", due to " + e.getMessage(), e); + return false; + } + } + private void alertHostSdcDisconnection(Host host) { if (host == null) { return; diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/lifecycle/ScaleIOPrimaryDataStoreLifeCycle.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/lifecycle/ScaleIOPrimaryDataStoreLifeCycle.java index f21abae2d319..e0a7684f1638 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/lifecycle/ScaleIOPrimaryDataStoreLifeCycle.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/lifecycle/ScaleIOPrimaryDataStoreLifeCycle.java @@ -336,8 +336,8 @@ public boolean maintain(DataStore store) { StoragePoolDetailVO mdmsDetail = storagePoolDetailsDao.findDetail(store.getId(), ScaleIOGatewayClient.STORAGE_POOL_MDMS); if (mdmsDetail != null) { details.put(ScaleIOGatewayClient.STORAGE_POOL_MDMS, mdmsDetail.getValue()); + details.put(ScaleIOSDCManager.ConnectOnDemand.key(), "false"); } - details.put(ScaleIOSDCManager.ConnectOnDemand.key(), "false"); } storagePoolAutomation.maintain(store, details); diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManager.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManager.java index 820b75722e9e..e73fa1e79ff0 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManager.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManager.java @@ -27,8 +27,8 @@ public interface ScaleIOSDCManager { Boolean.class, "powerflex.connect.on.demand", Boolean.FALSE.toString(), - "Connect PowerFlex client on Host when first Volume is mapped to SDC and disconnect when last Volume is unmapped from SDC," + - " otherwise no action (that is connection remains in the same state whichever it is, connected or disconnected).", + "When true, connects PowerFlex client on Host when first Volume is mapped to SDC & client connections configured 'storage.pool.connected.clients.limit' are within the limit and disconnects when last Volume is unmapped from SDC; " + + "and When false, connects PowerFlex client on Host when host connects to storage pool & client connections configured 'storage.pool.connected.clients.limit' are within the limit and disconnects when host disconnects from storage pool & no volumes mapped to SDC.", Boolean.TRUE, ConfigKey.Scope.Zone); @@ -63,6 +63,14 @@ public interface ScaleIOSDCManager { */ boolean unprepareSDC(Host host, DataStore dataStore); + /** + * Checks if the SDC can be unprepared on the host (don't remove MDM IPs from SDC if any volumes mapped to SDC). + * @param host the host + * @param dataStore the datastore + * @return true if SDC can be unprepared on the host + */ + boolean canUnprepareSDC(Host host, DataStore dataStore); + /** * Returns the SDC Id of the host for the pool. * @param sdcGuid the SDC GUID diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java index 4d01032d040d..0939d96fb789 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java @@ -296,6 +296,11 @@ public boolean unprepareSDC(Host host, DataStore dataStore) { return true; } + if (!canUnprepareSDC(host, dataStore)) { + logger.debug("Cannot unprepare SDC, there might be some volumes mapped to the SDC that belongs to the storage pools of PowerFlex storage cluster"); + return false; + } + String mdms = getMdms(dataStore.getId());; boolean unprepareSDCStatus = unprepareSDCOnHost(host, dataStore, mdms); if (unprepareSDCStatus) { @@ -337,6 +342,31 @@ private boolean unprepareSDCOnHost(Host host, DataStore dataStore, String mdms) return true; } + @Override + public boolean canUnprepareSDC(Host host, DataStore dataStore) { + if (host == null || dataStore == null) { + return false; + } + + StoragePoolHostVO poolHostVO = storagePoolHostDao.findByPoolHost(dataStore.getId(), host.getId()); + if (poolHostVO == null) { + return false; + } + + final String sdcId = poolHostVO.getLocalPath(); + if (StringUtils.isBlank(sdcId)) { + return false; + } + + try { + final ScaleIOGatewayClient client = getScaleIOClient(dataStore.getId()); + return client.listVolumesMappedToSdc(sdcId).isEmpty(); + } catch (Exception e) { + logger.warn("Unable to check whether the SDC of the pool: " + dataStore.getId() + " can be unprepared on the host: " + host.getId() + ", due to " + e.getMessage(), e); + return false; + } + } + @Override public String getHostSdcId(String sdcGuid, DataStore dataStore) { try { diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/provider/ScaleIOHostListener.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/provider/ScaleIOHostListener.java index 5c240fa6fce0..80a98f58d1cd 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/provider/ScaleIOHostListener.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/provider/ScaleIOHostListener.java @@ -206,10 +206,12 @@ public boolean hostDisconnected(long hostId, long poolId) { } Map details = new HashMap<>(); details.put(ScaleIOGatewayClient.STORAGE_POOL_SYSTEM_ID, systemId); - details.put(ScaleIOSDCManager.ConnectOnDemand.key(), String.valueOf(ScaleIOSDCManager.ConnectOnDemand.valueIn(host.getDataCenterId()))); _sdcManager = ComponentContext.inject(_sdcManager); - String mdms = _sdcManager.getMdms(poolId); - details.put(ScaleIOGatewayClient.STORAGE_POOL_MDMS, mdms); + if (_sdcManager.canUnprepareSDC(host, dataStore)) { + details.put(ScaleIOSDCManager.ConnectOnDemand.key(), String.valueOf(ScaleIOSDCManager.ConnectOnDemand.valueIn(host.getDataCenterId()))); + String mdms = _sdcManager.getMdms(poolId); + details.put(ScaleIOGatewayClient.STORAGE_POOL_MDMS, mdms); + } ModifyStoragePoolCommand cmd = new ModifyStoragePoolCommand(false, storagePool, storagePool.getPath(), details); ModifyStoragePoolAnswer answer = sendModifyStoragePoolCommand(cmd, storagePool, hostId); diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/util/ScaleIOUtil.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/util/ScaleIOUtil.java index 7837af4ea317..a8c02a90bf9e 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/util/ScaleIOUtil.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/util/ScaleIOUtil.java @@ -19,6 +19,7 @@ import java.util.List; +import org.apache.commons.collections.CollectionUtils; import org.apache.logging.log4j.Logger; import org.apache.logging.log4j.LogManager; @@ -91,6 +92,9 @@ public class ScaleIOUtil { private static final String DRV_CFG_FILE = "/etc/emc/scaleio/drv_cfg.txt"; public static void addMdms(List mdmAddresses) { + if (CollectionUtils.isEmpty(mdmAddresses)) { + return; + } // Sample Cmd - /opt/emc/scaleio/sdc/bin/drv_cfg --add_mdm --ip x.x.x.x,x.x.x.x --file /etc/emc/scaleio/drv_cfg.txt String addMdmsCmd = ScaleIOUtil.SDC_HOME_PATH + "/bin/" + ScaleIOUtil.ADD_MDMS_CMD; addMdmsCmd += " --ip " + String.join(",", mdmAddresses); @@ -102,14 +106,23 @@ public static void addMdms(List mdmAddresses) { } public static void removeMdms(List mdmAddresses) { + if (CollectionUtils.isEmpty(mdmAddresses)) { + return; + } // (i) Remove MDMs from config file (ii) Restart scini // Sample Cmd - sed -i '/x.x.x.x\,/d' /etc/emc/scaleio/drv_cfg.txt + boolean restartSDC = false; String removeMdmsCmdFormat = "sed -i '/%s\\,/d' %s"; for (String mdmAddress : mdmAddresses) { + if (mdmAdded(mdmAddress)) { + restartSDC = true; + } String removeMdmsCmd = String.format(removeMdmsCmdFormat, mdmAddress, DRV_CFG_FILE); Script.runSimpleBashScript(removeMdmsCmd); } - restartSDCService(); + if (restartSDC) { + restartSDCService(); + } } public static boolean mdmAdded(String mdmAddress) { diff --git a/server/src/main/java/com/cloud/storage/StorageManagerImpl.java b/server/src/main/java/com/cloud/storage/StorageManagerImpl.java index 1c5da19fbbc1..dec137beccf5 100644 --- a/server/src/main/java/com/cloud/storage/StorageManagerImpl.java +++ b/server/src/main/java/com/cloud/storage/StorageManagerImpl.java @@ -2833,6 +2833,17 @@ public boolean canHostPrepareStoragePoolAccess(Host host, StoragePool pool) { return storeDriver instanceof PrimaryDataStoreDriver && ((PrimaryDataStoreDriver)storeDriver).canHostPrepareStoragePoolAccess(host, pool); } + @Override + public boolean canDisconnectHostFromStoragePool(Host host, StoragePool pool) { + if (pool == null || !pool.isManaged()) { + return true; + } + + DataStoreProvider storeProvider = _dataStoreProviderMgr.getDataStoreProvider(pool.getStorageProviderName()); + DataStoreDriver storeDriver = storeProvider.getDataStoreDriver(); + return storeDriver instanceof PrimaryDataStoreDriver && ((PrimaryDataStoreDriver)storeDriver).canDisconnectHostFromStoragePool(host, pool); + } + @Override @DB public Host getHost(long hostId) { diff --git a/server/src/main/java/com/cloud/storage/StoragePoolAutomationImpl.java b/server/src/main/java/com/cloud/storage/StoragePoolAutomationImpl.java index 71a879eb2b16..eea4da4879f4 100644 --- a/server/src/main/java/com/cloud/storage/StoragePoolAutomationImpl.java +++ b/server/src/main/java/com/cloud/storage/StoragePoolAutomationImpl.java @@ -167,7 +167,7 @@ public boolean maintain(DataStore store, Map details) { // remove heartbeat for (HostVO host : hosts) { ModifyStoragePoolCommand cmd = new ModifyStoragePoolCommand(false, storagePool); - if (MapUtils.isNotEmpty(details)) { + if (MapUtils.isNotEmpty(details) && storageManager.canDisconnectHostFromStoragePool(host, storagePool)) { cmd.setDetails(details); } final Answer answer = agentMgr.easySend(host.getId(), cmd); From 5c65d43365d70d987b43a1f2253c0a031aca38cf Mon Sep 17 00:00:00 2001 From: Suresh Kumar Anaparti Date: Wed, 8 Jan 2025 13:57:26 +0530 Subject: [PATCH 5/7] Don't remove MDM IPs when other pools of same ScaleIO/PowerFlex cluster are connected --- .../com/cloud/storage/dao/StoragePoolHostDao.java | 2 ++ .../com/cloud/storage/dao/StoragePoolHostDaoImpl.java | 11 +++++++++++ .../datastore/manager/ScaleIOSDCManagerImpl.java | 9 ++++++++- 3 files changed, 21 insertions(+), 1 deletion(-) diff --git a/engine/schema/src/main/java/com/cloud/storage/dao/StoragePoolHostDao.java b/engine/schema/src/main/java/com/cloud/storage/dao/StoragePoolHostDao.java index 62ef5b7570d1..1da1f636f635 100644 --- a/engine/schema/src/main/java/com/cloud/storage/dao/StoragePoolHostDao.java +++ b/engine/schema/src/main/java/com/cloud/storage/dao/StoragePoolHostDao.java @@ -30,6 +30,8 @@ public interface StoragePoolHostDao extends GenericDao public StoragePoolHostVO findByPoolHost(long poolId, long hostId); + List findByLocalPath(String path); + List listByHostStatus(long poolId, Status hostStatus); List findHostsConnectedToPools(List poolIds); diff --git a/engine/schema/src/main/java/com/cloud/storage/dao/StoragePoolHostDaoImpl.java b/engine/schema/src/main/java/com/cloud/storage/dao/StoragePoolHostDaoImpl.java index 987a42f410e7..9894b5c9b6fe 100644 --- a/engine/schema/src/main/java/com/cloud/storage/dao/StoragePoolHostDaoImpl.java +++ b/engine/schema/src/main/java/com/cloud/storage/dao/StoragePoolHostDaoImpl.java @@ -45,6 +45,7 @@ public class StoragePoolHostDaoImpl extends GenericDaoBase PoolSearch; protected final SearchBuilder HostSearch; protected final SearchBuilder PoolHostSearch; + protected final SearchBuilder LocalPathSearch; protected SearchBuilder poolNotInClusterSearch; @@ -77,6 +78,9 @@ public StoragePoolHostDaoImpl() { PoolHostSearch.and("host_id", PoolHostSearch.entity().getHostId(), SearchCriteria.Op.EQ); PoolHostSearch.done(); + LocalPathSearch = createSearchBuilder(); + LocalPathSearch.and("local_path", LocalPathSearch.entity().getLocalPath(), SearchCriteria.Op.EQ); + LocalPathSearch.done(); } @PostConstruct @@ -117,6 +121,13 @@ public StoragePoolHostVO findByPoolHost(long poolId, long hostId) { return findOneIncludingRemovedBy(sc); } + @Override + public List findByLocalPath(String path) { + SearchCriteria sc = LocalPathSearch.create(); + sc.setParameters("local_path", path); + return listBy(sc); + } + @Override public List listByHostStatus(long poolId, Status hostStatus) { TransactionLegacy txn = TransactionLegacy.currentTxn(); diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java index 0939d96fb789..7cc39bfd9b3c 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java @@ -297,7 +297,8 @@ public boolean unprepareSDC(Host host, DataStore dataStore) { } if (!canUnprepareSDC(host, dataStore)) { - logger.debug("Cannot unprepare SDC, there might be some volumes mapped to the SDC that belongs to the storage pools of PowerFlex storage cluster"); + logger.debug("Cannot unprepare SDC, there might be other connected pools of same PowerFlex storage cluster," + + "or some volumes mapped to the SDC that belongs to any of the storage pools of the PowerFlex storage cluster"); return false; } @@ -358,6 +359,12 @@ public boolean canUnprepareSDC(Host host, DataStore dataStore) { return false; } + List poolHostVOsBySdc = storagePoolHostDao.findByLocalPath(sdcId); + if (CollectionUtils.isNotEmpty(poolHostVOsBySdc) && poolHostVOsBySdc.size() > 1) { + LOGGER.debug(String.format("There are other connected pools with the same SDC of the host %s, shouldn't unprepare SDC", host)); + return false; + } + try { final ScaleIOGatewayClient client = getScaleIOClient(dataStore.getId()); return client.listVolumesMappedToSdc(sdcId).isEmpty(); From 9013e56d9f6f93225be633322225635e50409486 Mon Sep 17 00:00:00 2001 From: Suresh Kumar Anaparti Date: Fri, 10 Jan 2025 14:35:26 +0530 Subject: [PATCH 6/7] rebase fixes --- .../driver/ScaleIOPrimaryDataStoreDriver.java | 2 +- .../datastore/manager/ScaleIOSDCManager.java | 8 +++---- .../manager/ScaleIOSDCManagerImpl.java | 2 +- .../provider/ScaleIOHostListener.java | 23 ++++++++----------- 4 files changed, 16 insertions(+), 19 deletions(-) diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java index d8a5f872d434..237f9bd52e39 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java @@ -1498,7 +1498,7 @@ public boolean canDisconnectHostFromStoragePool(Host host, StoragePool pool) { } try { - final ScaleIOGatewayClient client = getScaleIOClient(pool.getId()); + final ScaleIOGatewayClient client = getScaleIOClient(pool); return client.listVolumesMappedToSdc(sdcId).isEmpty(); } catch (Exception e) { logger.warn("Unable to check whether the host: " + host.getId() + " can be disconnected from storage pool: " + pool.getId() + ", due to " + e.getMessage(), e); diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManager.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManager.java index e73fa1e79ff0..b02732051b5e 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManager.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManager.java @@ -74,18 +74,18 @@ public interface ScaleIOSDCManager { /** * Returns the SDC Id of the host for the pool. * @param sdcGuid the SDC GUID - * @param poolId the pool id + * @param dataStore the datastore * @return SDC Id of the host for the pool */ - String getHostSdcId(String sdcGuid, long poolId); + String getHostSdcId(String sdcGuid, DataStore dataStore); /** * Returns the connection status of host SDC of the pool. * @param sdcId the SDC id - * @param poolId the pool id + * @param dataStore the datastore * @return true if Host SDC is connected to the pool */ - boolean isHostSdcConnected(String sdcId, long poolId, int waitTimeInSecs); + boolean isHostSdcConnected(String sdcId, DataStore dataStore, int waitTimeInSecs); /** * Returns the comma-separated list of MDM IPs of the pool. diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java index 7cc39bfd9b3c..79f82a1e0df0 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/manager/ScaleIOSDCManagerImpl.java @@ -361,7 +361,7 @@ public boolean canUnprepareSDC(Host host, DataStore dataStore) { List poolHostVOsBySdc = storagePoolHostDao.findByLocalPath(sdcId); if (CollectionUtils.isNotEmpty(poolHostVOsBySdc) && poolHostVOsBySdc.size() > 1) { - LOGGER.debug(String.format("There are other connected pools with the same SDC of the host %s, shouldn't unprepare SDC", host)); + logger.debug(String.format("There are other connected pools with the same SDC of the host %s, shouldn't unprepare SDC", host)); return false; } diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/provider/ScaleIOHostListener.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/provider/ScaleIOHostListener.java index 80a98f58d1cd..0db329204e8c 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/provider/ScaleIOHostListener.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/provider/ScaleIOHostListener.java @@ -18,9 +18,6 @@ */ package org.apache.cloudstack.storage.datastore.provider; -import java.net.URISyntaxException; -import java.security.KeyManagementException; -import java.security.NoSuchAlgorithmException; import java.util.HashMap; import java.util.Map; @@ -33,7 +30,6 @@ import org.apache.cloudstack.storage.datastore.db.PrimaryDataStoreDao; import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailVO; import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailsDao; -import org.apache.cloudstack.storage.datastore.db.StoragePoolVO; import org.apache.cloudstack.storage.datastore.manager.ScaleIOSDCManager; import org.apache.cloudstack.storage.datastore.manager.ScaleIOSDCManagerImpl; import org.apache.commons.collections.MapUtils; @@ -83,7 +79,7 @@ public boolean hostConnect(long hostId, long poolId) { DataStore dataStore = _dataStoreMgr.getDataStore(poolId, DataStoreRole.Primary); StoragePool storagePool = (StoragePool) dataStore; StoragePoolHostVO storagePoolHost = _storagePoolHostDao.findByPoolHost(poolId, hostId); - String sdcId = getSdcIdOfHost(host, storagePool); + String sdcId = getSdcIdOfHost(host, dataStore); if (StringUtils.isBlank(sdcId)) { if (storagePoolHost != null) { _storagePoolHostDao.deleteStoragePoolHostDetails(hostId, poolId); @@ -101,7 +97,8 @@ public boolean hostConnect(long hostId, long poolId) { return true; } - private String getSdcIdOfHost(HostVO host, StoragePool storagePool) { + private String getSdcIdOfHost(HostVO host, DataStore dataStore) { + StoragePool storagePool = (StoragePool) dataStore; long hostId = host.getId(); long poolId = storagePool.getId(); String systemId = null; @@ -136,7 +133,7 @@ private String getSdcIdOfHost(HostVO host, StoragePool storagePool) { sdcId = poolDetails.get(ScaleIOGatewayClient.SDC_ID); } else if (poolDetails.containsKey(ScaleIOGatewayClient.SDC_GUID)) { String sdcGuid = poolDetails.get(ScaleIOGatewayClient.SDC_GUID); - sdcId = _sdcManager.getHostSdcId(sdcGuid, poolId); + sdcId = _sdcManager.getHostSdcId(sdcGuid, dataStore); } if (StringUtils.isBlank(sdcId)) { @@ -148,7 +145,7 @@ private String getSdcIdOfHost(HostVO host, StoragePool storagePool) { if (details.containsKey(ScaleIOSDCManager.ConnectOnDemand.key())) { String connectOnDemand = details.get(ScaleIOSDCManager.ConnectOnDemand.key()); - if (connectOnDemand != null && !Boolean.parseBoolean(connectOnDemand) && !_sdcManager.isHostSdcConnected(sdcId, poolId, 15)) { + if (connectOnDemand != null && !Boolean.parseBoolean(connectOnDemand) && !_sdcManager.isHostSdcConnected(sdcId, dataStore, 15)) { logger.warn("SDC not connected on the host: " + hostId); String msg = "SDC not connected on the host: " + hostId + ", reconnect the SDC to MDM and restart agent"; _alertMgr.sendAlert(AlertManager.AlertType.ALERT_TYPE_HOST, host.getDataCenterId(), host.getPodId(), "SDC not connected on host: " + host.getUuid(), msg); @@ -163,19 +160,19 @@ private ModifyStoragePoolAnswer sendModifyStoragePoolCommand(ModifyStoragePoolCo Answer answer = _agentMgr.easySend(host.getId(), cmd); if (answer == null) { - throw new CloudRuntimeException(String.format("Unable to get an answer to the modify storage pool command (add: %s) for PowerFlex storage pool %s, sent to host %d", - cmd.getAdd(), getStoragePoolDetails(storagePool), hostId)); + throw new CloudRuntimeException(String.format("Unable to get an answer to the modify storage pool command (add: %s) for PowerFlex storage pool %s, sent to host %s", + cmd.getAdd(), getStoragePoolDetails(storagePool), host)); } if (!answer.getResult()) { if (cmd.getAdd()) { - String msg = "Unable to attach PowerFlex storage pool " + getStoragePoolDetails(storagePool) + " to the host " + hostId; + String msg = "Unable to attach PowerFlex storage pool " + getStoragePoolDetails(storagePool) + " to the host " + host; _alertMgr.sendAlert(AlertManager.AlertType.ALERT_TYPE_HOST, storagePool.getDataCenterId(), storagePool.getPodId(), msg, msg); throw new CloudRuntimeException("Unable to connect to PowerFlex storage pool " + getStoragePoolDetails(storagePool) + " due to " + answer.getDetails()); } else { - String msg = "Unable to detach PowerFlex storage pool " + getStoragePoolDetails(storagePool) + " from the host " + hostId; + String msg = "Unable to detach PowerFlex storage pool " + getStoragePoolDetails(storagePool) + " from the host " + host; _alertMgr.sendAlert(AlertManager.AlertType.ALERT_TYPE_HOST, storagePool.getDataCenterId(), storagePool.getPodId(), msg, msg); } @@ -214,7 +211,7 @@ public boolean hostDisconnected(long hostId, long poolId) { } ModifyStoragePoolCommand cmd = new ModifyStoragePoolCommand(false, storagePool, storagePool.getPath(), details); - ModifyStoragePoolAnswer answer = sendModifyStoragePoolCommand(cmd, storagePool, hostId); + ModifyStoragePoolAnswer answer = sendModifyStoragePoolCommand(cmd, storagePool, host); if (!answer.getResult()) { logger.error("Failed to disconnect storage pool: " + storagePool + " and host: " + hostId); return false; From 376673efa5f2c7c56aebb6b4355786fbf6c9171e Mon Sep 17 00:00:00 2001 From: Suresh Kumar Anaparti Date: Fri, 10 Jan 2025 17:44:08 +0530 Subject: [PATCH 7/7] update changes, to not remove/disconnect MDMs on maintenance --- .../datastore/driver/ScaleIOPrimaryDataStoreDriver.java | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java index 237f9bd52e39..4e3671503efc 100644 --- a/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java +++ b/plugins/storage/volume/scaleio/src/main/java/org/apache/cloudstack/storage/datastore/driver/ScaleIOPrimaryDataStoreDriver.java @@ -1497,6 +1497,12 @@ public boolean canDisconnectHostFromStoragePool(Host host, StoragePool pool) { return false; } + List poolHostVOsBySdc = storagePoolHostDao.findByLocalPath(sdcId); + if (CollectionUtils.isNotEmpty(poolHostVOsBySdc) && poolHostVOsBySdc.size() > 1) { + logger.debug(String.format("There are other connected pools with the same SDC of the host %s, shouldn't disconnect SDC", host)); + return false; + } + try { final ScaleIOGatewayClient client = getScaleIOClient(pool); return client.listVolumesMappedToSdc(sdcId).isEmpty();