HDFS-9715. Check storage ID uniqueness on datanode startup (Contributed by Lei (Eddy) Xu)
(cherry picked from commit04375756a5
) (cherry picked from commit8577c0b6a2
)
This commit is contained in:
parent
0ad3c51dfb
commit
bf2936fd38
|
@ -959,6 +959,9 @@ Release 2.8.0 - UNRELEASED
|
||||||
HDFS-8999. Allow a file to be closed with COMMITTED but not yet COMPLETE
|
HDFS-8999. Allow a file to be closed with COMMITTED but not yet COMPLETE
|
||||||
blocks. (szetszwo)
|
blocks. (szetszwo)
|
||||||
|
|
||||||
|
HDFS-9715. Check storage ID uniqueness on datanode startup
|
||||||
|
(Lei (Eddy) Xu via vinayakumarb)
|
||||||
|
|
||||||
BUG FIXES
|
BUG FIXES
|
||||||
|
|
||||||
HDFS-8091: ACLStatus and XAttributes should be presented to
|
HDFS-8091: ACLStatus and XAttributes should be presented to
|
||||||
|
|
|
@ -377,6 +377,31 @@ class FsDatasetImpl implements FsDatasetSpi<FsVolumeImpl> {
|
||||||
return volumeFailureInfos;
|
return volumeFailureInfos;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Activate a volume to serve requests.
|
||||||
|
* @throws IOException if the storage UUID already exists.
|
||||||
|
*/
|
||||||
|
private synchronized void activateVolume(
|
||||||
|
ReplicaMap replicaMap,
|
||||||
|
Storage.StorageDirectory sd, StorageType storageType,
|
||||||
|
FsVolumeReference ref) throws IOException {
|
||||||
|
DatanodeStorage dnStorage = storageMap.get(sd.getStorageUuid());
|
||||||
|
if (dnStorage != null) {
|
||||||
|
final String errorMsg = String.format(
|
||||||
|
"Found duplicated storage UUID: %s in %s.",
|
||||||
|
sd.getStorageUuid(), sd.getVersionFile());
|
||||||
|
LOG.error(errorMsg);
|
||||||
|
throw new IOException(errorMsg);
|
||||||
|
}
|
||||||
|
volumeMap.addAll(replicaMap);
|
||||||
|
storageMap.put(sd.getStorageUuid(),
|
||||||
|
new DatanodeStorage(sd.getStorageUuid(),
|
||||||
|
DatanodeStorage.State.NORMAL,
|
||||||
|
storageType));
|
||||||
|
asyncDiskService.addVolume(sd.getCurrentDir());
|
||||||
|
volumes.addVolume(ref);
|
||||||
|
}
|
||||||
|
|
||||||
private void addVolume(Collection<StorageLocation> dataLocations,
|
private void addVolume(Collection<StorageLocation> dataLocations,
|
||||||
Storage.StorageDirectory sd) throws IOException {
|
Storage.StorageDirectory sd) throws IOException {
|
||||||
final File dir = sd.getCurrentDir();
|
final File dir = sd.getCurrentDir();
|
||||||
|
@ -392,16 +417,7 @@ class FsDatasetImpl implements FsDatasetSpi<FsVolumeImpl> {
|
||||||
ReplicaMap tempVolumeMap = new ReplicaMap(this);
|
ReplicaMap tempVolumeMap = new ReplicaMap(this);
|
||||||
fsVolume.getVolumeMap(tempVolumeMap, ramDiskReplicaTracker);
|
fsVolume.getVolumeMap(tempVolumeMap, ramDiskReplicaTracker);
|
||||||
|
|
||||||
synchronized (this) {
|
activateVolume(tempVolumeMap, sd, storageType, ref);
|
||||||
volumeMap.addAll(tempVolumeMap);
|
|
||||||
storageMap.put(sd.getStorageUuid(),
|
|
||||||
new DatanodeStorage(sd.getStorageUuid(),
|
|
||||||
DatanodeStorage.State.NORMAL,
|
|
||||||
storageType));
|
|
||||||
asyncDiskService.addVolume(sd.getCurrentDir());
|
|
||||||
volumes.addVolume(ref);
|
|
||||||
}
|
|
||||||
|
|
||||||
LOG.info("Added volume - " + dir + ", StorageType: " + storageType);
|
LOG.info("Added volume - " + dir + ", StorageType: " + storageType);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -459,15 +475,7 @@ class FsDatasetImpl implements FsDatasetSpi<FsVolumeImpl> {
|
||||||
setupAsyncLazyPersistThread(fsVolume);
|
setupAsyncLazyPersistThread(fsVolume);
|
||||||
|
|
||||||
builder.build();
|
builder.build();
|
||||||
synchronized (this) {
|
activateVolume(tempVolumeMap, sd, storageType, ref);
|
||||||
volumeMap.addAll(tempVolumeMap);
|
|
||||||
storageMap.put(sd.getStorageUuid(),
|
|
||||||
new DatanodeStorage(sd.getStorageUuid(),
|
|
||||||
DatanodeStorage.State.NORMAL,
|
|
||||||
storageType));
|
|
||||||
asyncDiskService.addVolume(sd.getCurrentDir());
|
|
||||||
volumes.addVolume(ref);
|
|
||||||
}
|
|
||||||
LOG.info("Added volume - " + dir + ", StorageType: " + storageType);
|
LOG.info("Added volume - " + dir + ", StorageType: " + storageType);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -30,6 +30,7 @@ import java.io.OutputStream;
|
||||||
import java.net.URI;
|
import java.net.URI;
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
|
import java.util.Properties;
|
||||||
import java.util.zip.CRC32;
|
import java.util.zip.CRC32;
|
||||||
|
|
||||||
import org.apache.hadoop.conf.Configuration;
|
import org.apache.hadoop.conf.Configuration;
|
||||||
|
@ -49,6 +50,7 @@ import org.apache.hadoop.hdfs.server.datanode.BlockPoolSliceStorage;
|
||||||
import org.apache.hadoop.hdfs.server.datanode.DataNodeLayoutVersion;
|
import org.apache.hadoop.hdfs.server.datanode.DataNodeLayoutVersion;
|
||||||
import org.apache.hadoop.hdfs.server.datanode.DataStorage;
|
import org.apache.hadoop.hdfs.server.datanode.DataStorage;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.NNStorage;
|
import org.apache.hadoop.hdfs.server.namenode.NNStorage;
|
||||||
|
import org.apache.hadoop.hdfs.server.protocol.DatanodeStorage;
|
||||||
import org.apache.hadoop.hdfs.server.protocol.NamenodeProtocols;
|
import org.apache.hadoop.hdfs.server.protocol.NamenodeProtocols;
|
||||||
|
|
||||||
import com.google.common.base.Preconditions;
|
import com.google.common.base.Preconditions;
|
||||||
|
@ -379,6 +381,14 @@ public class UpgradeUtilities {
|
||||||
localFS.copyToLocalFile(new Path(datanodeStorage.toString(), "current"),
|
localFS.copyToLocalFile(new Path(datanodeStorage.toString(), "current"),
|
||||||
new Path(newDir.toString()),
|
new Path(newDir.toString()),
|
||||||
false);
|
false);
|
||||||
|
// Change the storage UUID to avoid conflicts when DN starts up.
|
||||||
|
StorageDirectory sd = new StorageDirectory(
|
||||||
|
new File(datanodeStorage.toString()));
|
||||||
|
sd.setStorageUuid(DatanodeStorage.generateUuid());
|
||||||
|
Properties properties = Storage.readPropertiesFile(sd.getVersionFile());
|
||||||
|
properties.setProperty("storageID", sd.getStorageUuid());
|
||||||
|
Storage.writeProperties(sd.getVersionFile(), properties);
|
||||||
|
|
||||||
retVal[i] = newDir;
|
retVal[i] = newDir;
|
||||||
}
|
}
|
||||||
return retVal;
|
return retVal;
|
||||||
|
|
|
@ -207,6 +207,32 @@ public class TestFsDatasetImpl {
|
||||||
assertTrue(actualVolumes.containsAll(expectedVolumes));
|
assertTrue(actualVolumes.containsAll(expectedVolumes));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAddVolumeWithSameStorageUuid() throws IOException {
|
||||||
|
HdfsConfiguration conf = new HdfsConfiguration();
|
||||||
|
MiniDFSCluster cluster = new MiniDFSCluster.Builder(conf)
|
||||||
|
.numDataNodes(1).build();
|
||||||
|
try {
|
||||||
|
cluster.waitActive();
|
||||||
|
assertTrue(cluster.getDataNodes().get(0).isConnectedToNN(
|
||||||
|
cluster.getNameNode().getServiceRpcAddress()));
|
||||||
|
|
||||||
|
MiniDFSCluster.DataNodeProperties dn = cluster.stopDataNode(0);
|
||||||
|
File vol0 = cluster.getStorageDir(0, 0);
|
||||||
|
File vol1 = cluster.getStorageDir(0, 1);
|
||||||
|
Storage.StorageDirectory sd0 = new Storage.StorageDirectory(vol0);
|
||||||
|
Storage.StorageDirectory sd1 = new Storage.StorageDirectory(vol1);
|
||||||
|
FileUtils.copyFile(sd0.getVersionFile(), sd1.getVersionFile());
|
||||||
|
|
||||||
|
cluster.restartDataNode(dn, true);
|
||||||
|
cluster.waitActive();
|
||||||
|
assertFalse(cluster.getDataNodes().get(0).isConnectedToNN(
|
||||||
|
cluster.getNameNode().getServiceRpcAddress()));
|
||||||
|
} finally {
|
||||||
|
cluster.shutdown();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Test(timeout = 30000)
|
@Test(timeout = 30000)
|
||||||
public void testRemoveVolumes() throws IOException {
|
public void testRemoveVolumes() throws IOException {
|
||||||
// Feed FsDataset with block metadata.
|
// Feed FsDataset with block metadata.
|
||||||
|
|
|
@ -34,6 +34,8 @@ import java.util.Arrays;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.EnumMap;
|
import java.util.EnumMap;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Iterator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Map.Entry;
|
import java.util.Map.Entry;
|
||||||
|
@ -313,21 +315,24 @@ public abstract class FSImageTestUtil {
|
||||||
fileList.add(f);
|
fileList.add(f);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Set<String> ignoredProperties = new HashSet<>();
|
||||||
|
ignoredProperties.add("storageID");
|
||||||
for (List<File> sameNameList : groupedByName.values()) {
|
for (List<File> sameNameList : groupedByName.values()) {
|
||||||
if (sameNameList.get(0).isDirectory()) {
|
if (sameNameList.get(0).isDirectory()) {
|
||||||
// recurse
|
// recurse
|
||||||
assertParallelFilesAreIdentical(sameNameList, ignoredFileNames);
|
assertParallelFilesAreIdentical(sameNameList, ignoredFileNames);
|
||||||
} else {
|
} else {
|
||||||
if ("VERSION".equals(sameNameList.get(0).getName())) {
|
if ("VERSION".equals(sameNameList.get(0).getName())) {
|
||||||
assertPropertiesFilesSame(sameNameList.toArray(new File[0]));
|
assertPropertiesFilesSame(sameNameList.toArray(new File[0]),
|
||||||
|
ignoredProperties);
|
||||||
} else {
|
} else {
|
||||||
assertFileContentsSame(sameNameList.toArray(new File[0]));
|
assertFileContentsSame(sameNameList.toArray(new File[0]));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Assert that a set of properties files all contain the same data.
|
* Assert that a set of properties files all contain the same data.
|
||||||
* We cannot simply check the md5sums here, since Properties files
|
* We cannot simply check the md5sums here, since Properties files
|
||||||
|
@ -339,6 +344,20 @@ public abstract class FSImageTestUtil {
|
||||||
*/
|
*/
|
||||||
public static void assertPropertiesFilesSame(File[] propFiles)
|
public static void assertPropertiesFilesSame(File[] propFiles)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
|
assertPropertiesFilesSame(propFiles, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Assert that a set of properties files all contain the same data.
|
||||||
|
*
|
||||||
|
* @param propFiles the files to compare.
|
||||||
|
* @param ignoredProperties the property names to be ignored during
|
||||||
|
* comparison.
|
||||||
|
* @throws IOException if the files cannot be opened or read
|
||||||
|
* @throws AssertionError if the files differ
|
||||||
|
*/
|
||||||
|
public static void assertPropertiesFilesSame(
|
||||||
|
File[] propFiles, Set<String> ignoredProperties) throws IOException {
|
||||||
Set<Map.Entry<Object, Object>> prevProps = null;
|
Set<Map.Entry<Object, Object>> prevProps = null;
|
||||||
|
|
||||||
for (File f : propFiles) {
|
for (File f : propFiles) {
|
||||||
|
@ -355,7 +374,13 @@ public abstract class FSImageTestUtil {
|
||||||
} else {
|
} else {
|
||||||
Set<Entry<Object,Object>> diff =
|
Set<Entry<Object,Object>> diff =
|
||||||
Sets.symmetricDifference(prevProps, props.entrySet());
|
Sets.symmetricDifference(prevProps, props.entrySet());
|
||||||
if (!diff.isEmpty()) {
|
Iterator<Entry<Object, Object>> it = diff.iterator();
|
||||||
|
while (it.hasNext()) {
|
||||||
|
Entry<Object, Object> entry = it.next();
|
||||||
|
if (ignoredProperties != null &&
|
||||||
|
ignoredProperties.contains(entry.getKey())) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
fail("Properties file " + f + " differs from " + propFiles[0]);
|
fail("Properties file " + f + " differs from " + propFiles[0]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue