HDDS-354. VolumeInfo.getScmUsed throws NPE. Contributed by Hanisha Koneru.

This commit is contained in:
Bharat Viswanadham 2018-10-04 21:39:29 -07:00
parent e6b77ad65f
commit 2a07617f85
6 changed files with 129 additions and 116 deletions

View File

@ -286,17 +286,6 @@ public final class OzoneConfigKeys {
public static final double public static final double
HDDS_DATANODE_STORAGE_UTILIZATION_CRITICAL_THRESHOLD_DEFAULT = 0.75; HDDS_DATANODE_STORAGE_UTILIZATION_CRITICAL_THRESHOLD_DEFAULT = 0.75;
public static final String
HDDS_WRITE_LOCK_REPORTING_THRESHOLD_MS_KEY =
"hdds.write.lock.reporting.threshold.ms";
public static final long
HDDS_WRITE_LOCK_REPORTING_THRESHOLD_MS_DEFAULT = 5000L;
public static final String
HDDS_LOCK_SUPPRESS_WARNING_INTERVAL_MS_KEY =
"hdds.lock.suppress.warning.interval.ms";
public static final long
HDDS_LOCK_SUPPRESS_WARNING_INTERVAL_MS_DEAFULT = 10000L;
public static final String OZONE_CONTAINER_COPY_WORKDIR = public static final String OZONE_CONTAINER_COPY_WORKDIR =
"hdds.datanode.replication.work.dir"; "hdds.datanode.replication.work.dir";

View File

@ -69,12 +69,16 @@ public EndpointStateMachine.EndPointStates call() throws Exception {
VersionResponse response = VersionResponse.getFromProtobuf( VersionResponse response = VersionResponse.getFromProtobuf(
versionResponse); versionResponse);
rpcEndPoint.setVersion(response); rpcEndPoint.setVersion(response);
VolumeSet volumeSet = ozoneContainer.getVolumeSet();
Map<String, HddsVolume> volumeMap = volumeSet.getVolumeMap();
String scmId = response.getValue(OzoneConsts.SCM_ID); String scmId = response.getValue(OzoneConsts.SCM_ID);
String clusterId = response.getValue(OzoneConsts.CLUSTER_ID); String clusterId = response.getValue(OzoneConsts.CLUSTER_ID);
// Check volumes
VolumeSet volumeSet = ozoneContainer.getVolumeSet();
volumeSet.readLock();
try {
Map<String, HddsVolume> volumeMap = volumeSet.getVolumeMap();
Preconditions.checkNotNull(scmId, "Reply from SCM: scmId cannot be " + Preconditions.checkNotNull(scmId, "Reply from SCM: scmId cannot be " +
"null"); "null");
Preconditions.checkNotNull(clusterId, "Reply from SCM: clusterId " + Preconditions.checkNotNull(clusterId, "Reply from SCM: clusterId " +
@ -94,6 +98,10 @@ public EndpointStateMachine.EndPointStates call() throws Exception {
throw new DiskOutOfSpaceException("All configured Volumes are in " + throw new DiskOutOfSpaceException("All configured Volumes are in " +
"Inconsistent State"); "Inconsistent State");
} }
} finally {
volumeSet.readUnlock();
}
ozoneContainer.getDispatcher().setScmId(scmId); ozoneContainer.getDispatcher().setScmId(scmId);
EndpointStateMachine.EndPointStates nextState = EndpointStateMachine.EndPointStates nextState =

View File

@ -164,7 +164,7 @@ private static String getProperty(Properties props, String propName, File
} }
/** /**
* Check Volume is consistent state or not. * Check Volume is in consistent state or not.
* @param hddsVolume * @param hddsVolume
* @param scmId * @param scmId
* @param clusterId * @param clusterId

View File

@ -33,15 +33,11 @@
.StorageContainerDatanodeProtocolProtos; .StorageContainerDatanodeProtocolProtos;
import org.apache.hadoop.hdds.protocol.proto import org.apache.hadoop.hdds.protocol.proto
.StorageContainerDatanodeProtocolProtos.NodeReportProto; .StorageContainerDatanodeProtocolProtos.NodeReportProto;
import org.apache.hadoop.ozone.OzoneConfigKeys;
import org.apache.hadoop.ozone.common.InconsistentStorageStateException; import org.apache.hadoop.ozone.common.InconsistentStorageStateException;
import org.apache.hadoop.ozone.container.common.impl.StorageLocationReport; import org.apache.hadoop.ozone.container.common.impl.StorageLocationReport;
import org.apache.hadoop.ozone.container.common.utils.HddsVolumeUtil; import org.apache.hadoop.ozone.container.common.utils.HddsVolumeUtil;
import org.apache.hadoop.ozone.container.common.volume.HddsVolume.VolumeState; import org.apache.hadoop.ozone.container.common.volume.HddsVolume.VolumeState;
import org.apache.hadoop.ozone.container.common.interfaces.VolumeChoosingPolicy;
import org.apache.hadoop.util.AutoCloseableLock;
import org.apache.hadoop.util.DiskChecker.DiskOutOfSpaceException; import org.apache.hadoop.util.DiskChecker.DiskOutOfSpaceException;
import org.apache.hadoop.util.InstrumentedLock;
import org.apache.hadoop.util.ShutdownHookManager; import org.apache.hadoop.util.ShutdownHookManager;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
@ -53,8 +49,7 @@
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.TimeUnit; import java.util.concurrent.locks.ReentrantReadWriteLock;
import java.util.concurrent.locks.ReentrantLock;
/** /**
* VolumeSet to manage volumes in a DataNode. * VolumeSet to manage volumes in a DataNode.
@ -84,11 +79,12 @@ public class VolumeSet {
private EnumMap<StorageType, List<HddsVolume>> volumeStateMap; private EnumMap<StorageType, List<HddsVolume>> volumeStateMap;
/** /**
* Lock to synchronize changes to the VolumeSet. Any update to * A Reentrant Read Write Lock to synchronize volume operations in VolumeSet.
* {@link VolumeSet#volumeMap}, {@link VolumeSet#failedVolumeMap}, or * Any update to {@link VolumeSet#volumeMap},
* {@link VolumeSet#volumeStateMap} should be done after acquiring this lock. * {@link VolumeSet#failedVolumeMap}, or {@link VolumeSet#volumeStateMap}
* should be done after acquiring the write lock.
*/ */
private final AutoCloseableLock volumeSetLock; private final ReentrantReadWriteLock volumeSetRWLock;
private final String datanodeUuid; private final String datanodeUuid;
private String clusterID; private String clusterID;
@ -105,17 +101,7 @@ public VolumeSet(String dnUuid, String clusterID, Configuration conf)
this.datanodeUuid = dnUuid; this.datanodeUuid = dnUuid;
this.clusterID = clusterID; this.clusterID = clusterID;
this.conf = conf; this.conf = conf;
this.volumeSetLock = new AutoCloseableLock( this.volumeSetRWLock = new ReentrantReadWriteLock();
new InstrumentedLock(getClass().getName(), LOG,
new ReentrantLock(true),
conf.getTimeDuration(
OzoneConfigKeys.HDDS_WRITE_LOCK_REPORTING_THRESHOLD_MS_KEY,
OzoneConfigKeys.HDDS_WRITE_LOCK_REPORTING_THRESHOLD_MS_DEFAULT,
TimeUnit.MILLISECONDS),
conf.getTimeDuration(
OzoneConfigKeys.HDDS_LOCK_SUPPRESS_WARNING_INTERVAL_MS_KEY,
OzoneConfigKeys.HDDS_LOCK_SUPPRESS_WARNING_INTERVAL_MS_DEAFULT,
TimeUnit.MILLISECONDS)));
initializeVolumeSet(); initializeVolumeSet();
} }
@ -198,14 +184,35 @@ private void checkAndSetClusterID(String idFromVersionFile)
} }
} }
public void acquireLock() { /**
volumeSetLock.acquire(); * Acquire Volume Set Read lock.
*/
public void readLock() {
volumeSetRWLock.readLock().lock();
} }
public void releaseLock() { /**
volumeSetLock.release(); * Release Volume Set Read lock.
*/
public void readUnlock() {
volumeSetRWLock.readLock().unlock();
} }
/**
* Acquire Volume Set Write lock.
*/
public void writeLock() {
volumeSetRWLock.writeLock().lock();
}
/**
* Release Volume Set Write lock.
*/
public void writeUnlock() {
volumeSetRWLock.writeLock().unlock();
}
private HddsVolume createVolume(String locationString, private HddsVolume createVolume(String locationString,
StorageType storageType) throws IOException { StorageType storageType) throws IOException {
HddsVolume.Builder volumeBuilder = new HddsVolume.Builder(locationString) HddsVolume.Builder volumeBuilder = new HddsVolume.Builder(locationString)
@ -227,7 +234,8 @@ public boolean addVolume(String volumeRoot, StorageType storageType) {
String hddsRoot = HddsVolumeUtil.getHddsRoot(volumeRoot); String hddsRoot = HddsVolumeUtil.getHddsRoot(volumeRoot);
boolean success; boolean success;
try (AutoCloseableLock lock = volumeSetLock.acquire()) { this.writeLock();
try {
if (volumeMap.containsKey(hddsRoot)) { if (volumeMap.containsKey(hddsRoot)) {
LOG.warn("Volume : {} already exists in VolumeMap", hddsRoot); LOG.warn("Volume : {} already exists in VolumeMap", hddsRoot);
success = false; success = false;
@ -247,6 +255,8 @@ public boolean addVolume(String volumeRoot, StorageType storageType) {
} catch (IOException ex) { } catch (IOException ex) {
LOG.error("Failed to add volume " + volumeRoot + " to VolumeSet", ex); LOG.error("Failed to add volume " + volumeRoot + " to VolumeSet", ex);
success = false; success = false;
} finally {
this.writeUnlock();
} }
return success; return success;
} }
@ -255,7 +265,8 @@ public boolean addVolume(String volumeRoot, StorageType storageType) {
public void failVolume(String dataDir) { public void failVolume(String dataDir) {
String hddsRoot = HddsVolumeUtil.getHddsRoot(dataDir); String hddsRoot = HddsVolumeUtil.getHddsRoot(dataDir);
try (AutoCloseableLock lock = volumeSetLock.acquire()) { this.writeLock();
try {
if (volumeMap.containsKey(hddsRoot)) { if (volumeMap.containsKey(hddsRoot)) {
HddsVolume hddsVolume = volumeMap.get(hddsRoot); HddsVolume hddsVolume = volumeMap.get(hddsRoot);
hddsVolume.failVolume(); hddsVolume.failVolume();
@ -270,6 +281,8 @@ public void failVolume(String dataDir) {
} else { } else {
LOG.warn("Volume : {} does not exist in VolumeSet", hddsRoot); LOG.warn("Volume : {} does not exist in VolumeSet", hddsRoot);
} }
} finally {
this.writeUnlock();
} }
} }
@ -277,7 +290,8 @@ public void failVolume(String dataDir) {
public void removeVolume(String dataDir) throws IOException { public void removeVolume(String dataDir) throws IOException {
String hddsRoot = HddsVolumeUtil.getHddsRoot(dataDir); String hddsRoot = HddsVolumeUtil.getHddsRoot(dataDir);
try (AutoCloseableLock lock = volumeSetLock.acquire()) { this.writeLock();
try {
if (volumeMap.containsKey(hddsRoot)) { if (volumeMap.containsKey(hddsRoot)) {
HddsVolume hddsVolume = volumeMap.get(hddsRoot); HddsVolume hddsVolume = volumeMap.get(hddsRoot);
hddsVolume.shutdown(); hddsVolume.shutdown();
@ -295,14 +309,11 @@ public void removeVolume(String dataDir) throws IOException {
} else { } else {
LOG.warn("Volume : {} does not exist in VolumeSet", hddsRoot); LOG.warn("Volume : {} does not exist in VolumeSet", hddsRoot);
} }
} finally {
this.writeUnlock();
} }
} }
public HddsVolume chooseVolume(long containerSize,
VolumeChoosingPolicy choosingPolicy) throws IOException {
return choosingPolicy.chooseVolume(getVolumesList(), containerSize);
}
/** /**
* This method, call shutdown on each volume to shutdown volume usage * This method, call shutdown on each volume to shutdown volume usage
* thread and write scmUsed on each volume. * thread and write scmUsed on each volume.
@ -352,6 +363,8 @@ public Map<StorageType, List<HddsVolume>> getVolumeStateMap() {
public StorageContainerDatanodeProtocolProtos.NodeReportProto getNodeReport() public StorageContainerDatanodeProtocolProtos.NodeReportProto getNodeReport()
throws IOException { throws IOException {
boolean failed; boolean failed;
this.readLock();
try {
StorageLocationReport[] reports = new StorageLocationReport[volumeMap StorageLocationReport[] reports = new StorageLocationReport[volumeMap
.size() + failedVolumeMap.size()]; .size() + failedVolumeMap.size()];
int counter = 0; int counter = 0;
@ -402,5 +415,8 @@ public StorageContainerDatanodeProtocolProtos.NodeReportProto getNodeReport()
nrb.addStorageReport(reports[i].getProtoBufMessage()); nrb.addStorageReport(reports[i].getProtoBufMessage());
} }
return nrb.build(); return nrb.build();
} finally {
this.readUnlock();
}
} }
} }

View File

@ -108,8 +108,8 @@ public void create(VolumeSet volumeSet, VolumeChoosingPolicy
Preconditions.checkNotNull(scmId, "scmId cannot be null"); Preconditions.checkNotNull(scmId, "scmId cannot be null");
File containerMetaDataPath = null; File containerMetaDataPath = null;
//acquiring volumeset lock and container lock //acquiring volumeset read lock
volumeSet.acquireLock(); volumeSet.readLock();
long maxSize = containerData.getMaxSize(); long maxSize = containerData.getMaxSize();
try { try {
HddsVolume containerVolume = volumeChoosingPolicy.chooseVolume(volumeSet HddsVolume containerVolume = volumeChoosingPolicy.chooseVolume(volumeSet
@ -166,7 +166,7 @@ public void create(VolumeSet volumeSet, VolumeChoosingPolicy
throw new StorageContainerException("Container creation failed.", ex, throw new StorageContainerException("Container creation failed.", ex,
CONTAINER_INTERNAL_ERROR); CONTAINER_INTERNAL_ERROR);
} finally { } finally {
volumeSet.releaseLock(); volumeSet.readUnlock();
} }
} }

View File

@ -271,14 +271,14 @@ ContainerCommandResponseProto handleCreateContainer(
public void populateContainerPathFields(KeyValueContainer container, public void populateContainerPathFields(KeyValueContainer container,
long maxSize) throws IOException { long maxSize) throws IOException {
volumeSet.acquireLock(); volumeSet.readLock();
try { try {
HddsVolume containerVolume = volumeChoosingPolicy.chooseVolume(volumeSet HddsVolume containerVolume = volumeChoosingPolicy.chooseVolume(volumeSet
.getVolumesList(), maxSize); .getVolumesList(), maxSize);
String hddsVolumeDir = containerVolume.getHddsRootDir().toString(); String hddsVolumeDir = containerVolume.getHddsRootDir().toString();
container.populatePathFields(scmID, containerVolume, hddsVolumeDir); container.populatePathFields(scmID, containerVolume, hddsVolumeDir);
} finally { } finally {
volumeSet.releaseLock(); volumeSet.readUnlock();
} }
} }