HDFS-6609. Use DirectorySnapshottableFeature to represent a snapshottable directory. Contributed by Jing Zhao.
git-svn-id: https://svn.apache.org/repos/asf/hadoop/common/trunk@1608631 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
7b287d5ba6
commit
b68818c4f0
|
@ -179,6 +179,9 @@ Release 2.6.0 - UNRELEASED
|
||||||
HDFS-6959. Make the HDFS home directory location customizable. (yzhang via
|
HDFS-6959. Make the HDFS home directory location customizable. (yzhang via
|
||||||
cmccabe)
|
cmccabe)
|
||||||
|
|
||||||
|
HDFS-6609. Use DirectorySnapshottableFeature to represent a snapshottable
|
||||||
|
directory. (Jing Zhao via wheat9)
|
||||||
|
|
||||||
OPTIMIZATIONS
|
OPTIMIZATIONS
|
||||||
|
|
||||||
HDFS-6690. Deduplicate xattr names in memory. (wang)
|
HDFS-6690. Deduplicate xattr names in memory. (wang)
|
||||||
|
|
|
@ -80,7 +80,7 @@ import org.apache.hadoop.hdfs.server.blockmanagement.DatanodeStorageInfo;
|
||||||
import org.apache.hadoop.hdfs.server.common.HdfsServerConstants.BlockUCState;
|
import org.apache.hadoop.hdfs.server.common.HdfsServerConstants.BlockUCState;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.INode.BlocksMapUpdateInfo;
|
import org.apache.hadoop.hdfs.server.namenode.INode.BlocksMapUpdateInfo;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.INodeReference.WithCount;
|
import org.apache.hadoop.hdfs.server.namenode.INodeReference.WithCount;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.INodeDirectorySnapshottable;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectorySnapshottableFeature;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot.Root;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot.Root;
|
||||||
import org.apache.hadoop.hdfs.util.ByteArray;
|
import org.apache.hadoop.hdfs.util.ByteArray;
|
||||||
|
@ -100,7 +100,7 @@ import com.google.common.collect.Lists;
|
||||||
**/
|
**/
|
||||||
@InterfaceAudience.Private
|
@InterfaceAudience.Private
|
||||||
public class FSDirectory implements Closeable {
|
public class FSDirectory implements Closeable {
|
||||||
private static INodeDirectorySnapshottable createRoot(FSNamesystem namesystem) {
|
private static INodeDirectory createRoot(FSNamesystem namesystem) {
|
||||||
final INodeDirectory r = new INodeDirectory(
|
final INodeDirectory r = new INodeDirectory(
|
||||||
INodeId.ROOT_INODE_ID,
|
INodeId.ROOT_INODE_ID,
|
||||||
INodeDirectory.ROOT_NAME,
|
INodeDirectory.ROOT_NAME,
|
||||||
|
@ -109,9 +109,9 @@ public class FSDirectory implements Closeable {
|
||||||
r.addDirectoryWithQuotaFeature(
|
r.addDirectoryWithQuotaFeature(
|
||||||
DirectoryWithQuotaFeature.DEFAULT_NAMESPACE_QUOTA,
|
DirectoryWithQuotaFeature.DEFAULT_NAMESPACE_QUOTA,
|
||||||
DirectoryWithQuotaFeature.DEFAULT_DISKSPACE_QUOTA);
|
DirectoryWithQuotaFeature.DEFAULT_DISKSPACE_QUOTA);
|
||||||
final INodeDirectorySnapshottable s = new INodeDirectorySnapshottable(r);
|
r.addSnapshottableFeature();
|
||||||
s.setSnapshotQuota(0);
|
r.setSnapshotQuota(0);
|
||||||
return s;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
|
@ -633,8 +633,7 @@ public class FSDirectory implements Closeable {
|
||||||
|
|
||||||
ezManager.checkMoveValidity(srcIIP, dstIIP, src);
|
ezManager.checkMoveValidity(srcIIP, dstIIP, src);
|
||||||
final INode dstInode = dstIIP.getLastINode();
|
final INode dstInode = dstIIP.getLastINode();
|
||||||
List<INodeDirectorySnapshottable> snapshottableDirs =
|
List<INodeDirectory> snapshottableDirs = new ArrayList<INodeDirectory>();
|
||||||
new ArrayList<INodeDirectorySnapshottable>();
|
|
||||||
if (dstInode != null) { // Destination exists
|
if (dstInode != null) { // Destination exists
|
||||||
validateRenameOverwrite(src, dst, overwrite, srcInode, dstInode);
|
validateRenameOverwrite(src, dst, overwrite, srcInode, dstInode);
|
||||||
checkSnapshot(dstInode, snapshottableDirs);
|
checkSnapshot(dstInode, snapshottableDirs);
|
||||||
|
@ -1158,8 +1157,7 @@ public class FSDirectory implements Closeable {
|
||||||
if (!deleteAllowed(inodesInPath, src) ) {
|
if (!deleteAllowed(inodesInPath, src) ) {
|
||||||
filesRemoved = -1;
|
filesRemoved = -1;
|
||||||
} else {
|
} else {
|
||||||
List<INodeDirectorySnapshottable> snapshottableDirs =
|
List<INodeDirectory> snapshottableDirs = new ArrayList<INodeDirectory>();
|
||||||
new ArrayList<INodeDirectorySnapshottable>();
|
|
||||||
checkSnapshot(inodesInPath.getLastINode(), snapshottableDirs);
|
checkSnapshot(inodesInPath.getLastINode(), snapshottableDirs);
|
||||||
filesRemoved = unprotectedDelete(inodesInPath, collectedBlocks,
|
filesRemoved = unprotectedDelete(inodesInPath, collectedBlocks,
|
||||||
removedINodes, mtime);
|
removedINodes, mtime);
|
||||||
|
@ -1229,8 +1227,7 @@ public class FSDirectory implements Closeable {
|
||||||
normalizePath(src), false);
|
normalizePath(src), false);
|
||||||
long filesRemoved = -1;
|
long filesRemoved = -1;
|
||||||
if (deleteAllowed(inodesInPath, src)) {
|
if (deleteAllowed(inodesInPath, src)) {
|
||||||
List<INodeDirectorySnapshottable> snapshottableDirs =
|
List<INodeDirectory> snapshottableDirs = new ArrayList<INodeDirectory>();
|
||||||
new ArrayList<INodeDirectorySnapshottable>();
|
|
||||||
checkSnapshot(inodesInPath.getLastINode(), snapshottableDirs);
|
checkSnapshot(inodesInPath.getLastINode(), snapshottableDirs);
|
||||||
filesRemoved = unprotectedDelete(inodesInPath, collectedBlocks,
|
filesRemoved = unprotectedDelete(inodesInPath, collectedBlocks,
|
||||||
removedINodes, mtime);
|
removedINodes, mtime);
|
||||||
|
@ -1305,19 +1302,20 @@ public class FSDirectory implements Closeable {
|
||||||
* but do not have snapshots yet
|
* but do not have snapshots yet
|
||||||
*/
|
*/
|
||||||
private static void checkSnapshot(INode target,
|
private static void checkSnapshot(INode target,
|
||||||
List<INodeDirectorySnapshottable> snapshottableDirs) throws SnapshotException {
|
List<INodeDirectory> snapshottableDirs) throws SnapshotException {
|
||||||
if (target.isDirectory()) {
|
if (target.isDirectory()) {
|
||||||
INodeDirectory targetDir = target.asDirectory();
|
INodeDirectory targetDir = target.asDirectory();
|
||||||
if (targetDir.isSnapshottable()) {
|
DirectorySnapshottableFeature sf = targetDir
|
||||||
INodeDirectorySnapshottable ssTargetDir =
|
.getDirectorySnapshottableFeature();
|
||||||
(INodeDirectorySnapshottable) targetDir;
|
if (sf != null) {
|
||||||
if (ssTargetDir.getNumSnapshots() > 0) {
|
if (sf.getNumSnapshots() > 0) {
|
||||||
throw new SnapshotException("The directory " + ssTargetDir.getFullPathName()
|
String fullPath = targetDir.getFullPathName();
|
||||||
+ " cannot be deleted since " + ssTargetDir.getFullPathName()
|
throw new SnapshotException("The directory " + fullPath
|
||||||
|
+ " cannot be deleted since " + fullPath
|
||||||
+ " is snapshottable and already has snapshots");
|
+ " is snapshottable and already has snapshots");
|
||||||
} else {
|
} else {
|
||||||
if (snapshottableDirs != null) {
|
if (snapshottableDirs != null) {
|
||||||
snapshottableDirs.add(ssTargetDir);
|
snapshottableDirs.add(targetDir);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1410,9 +1408,13 @@ public class FSDirectory implements Closeable {
|
||||||
src.length() - HdfsConstants.DOT_SNAPSHOT_DIR.length()));
|
src.length() - HdfsConstants.DOT_SNAPSHOT_DIR.length()));
|
||||||
|
|
||||||
final INode node = this.getINode(dirPath);
|
final INode node = this.getINode(dirPath);
|
||||||
final INodeDirectorySnapshottable dirNode = INodeDirectorySnapshottable
|
final INodeDirectory dirNode = INodeDirectory.valueOf(node, dirPath);
|
||||||
.valueOf(node, dirPath);
|
final DirectorySnapshottableFeature sf = dirNode.getDirectorySnapshottableFeature();
|
||||||
final ReadOnlyList<Snapshot> snapshots = dirNode.getSnapshotList();
|
if (sf == null) {
|
||||||
|
throw new SnapshotException(
|
||||||
|
"Directory is not a snapshottable directory: " + dirPath);
|
||||||
|
}
|
||||||
|
final ReadOnlyList<Snapshot> snapshots = sf.getSnapshotList();
|
||||||
int skipSize = ReadOnlyList.Util.binarySearch(snapshots, startAfter);
|
int skipSize = ReadOnlyList.Util.binarySearch(snapshots, startAfter);
|
||||||
skipSize = skipSize < 0 ? -skipSize - 1 : skipSize + 1;
|
skipSize = skipSize < 0 ? -skipSize - 1 : skipSize + 1;
|
||||||
int numOfListing = Math.min(snapshots.size() - skipSize, this.lsLimit);
|
int numOfListing = Math.min(snapshots.size() - skipSize, this.lsLimit);
|
||||||
|
@ -1476,9 +1478,8 @@ public class FSDirectory implements Closeable {
|
||||||
src.length() - HdfsConstants.DOT_SNAPSHOT_DIR.length()));
|
src.length() - HdfsConstants.DOT_SNAPSHOT_DIR.length()));
|
||||||
|
|
||||||
final INode node = this.getINode(dirPath);
|
final INode node = this.getINode(dirPath);
|
||||||
if (node != null
|
if (node != null && node.isDirectory()
|
||||||
&& node.isDirectory()
|
&& node.asDirectory().isSnapshottable()) {
|
||||||
&& node.asDirectory() instanceof INodeDirectorySnapshottable) {
|
|
||||||
return node;
|
return node;
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
|
|
|
@ -59,7 +59,6 @@ import org.apache.hadoop.hdfs.server.common.HdfsServerConstants.StartupOption;
|
||||||
import org.apache.hadoop.hdfs.server.common.InconsistentFSStateException;
|
import org.apache.hadoop.hdfs.server.common.InconsistentFSStateException;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectoryWithSnapshotFeature;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectoryWithSnapshotFeature;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.FileDiffList;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.FileDiffList;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.INodeDirectorySnapshottable;
|
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.SnapshotFSImageFormat;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.SnapshotFSImageFormat;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.SnapshotFSImageFormat.ReferenceMap;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.SnapshotFSImageFormat.ReferenceMap;
|
||||||
|
@ -74,8 +73,8 @@ import org.apache.hadoop.io.MD5Hash;
|
||||||
import org.apache.hadoop.io.Text;
|
import org.apache.hadoop.io.Text;
|
||||||
import org.apache.hadoop.util.StringUtils;
|
import org.apache.hadoop.util.StringUtils;
|
||||||
|
|
||||||
import com.google.common.base.Preconditions;
|
|
||||||
import com.google.common.annotations.VisibleForTesting;
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
|
import com.google.common.base.Preconditions;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Contains inner classes for reading or writing the on-disk format for
|
* Contains inner classes for reading or writing the on-disk format for
|
||||||
|
@ -559,17 +558,13 @@ public class FSImageFormat {
|
||||||
// Step 2. Load snapshots if parent is snapshottable
|
// Step 2. Load snapshots if parent is snapshottable
|
||||||
int numSnapshots = in.readInt();
|
int numSnapshots = in.readInt();
|
||||||
if (numSnapshots >= 0) {
|
if (numSnapshots >= 0) {
|
||||||
final INodeDirectorySnapshottable snapshottableParent
|
|
||||||
= INodeDirectorySnapshottable.valueOf(parent, parent.getLocalName());
|
|
||||||
// load snapshots and snapshotQuota
|
// load snapshots and snapshotQuota
|
||||||
SnapshotFSImageFormat.loadSnapshotList(snapshottableParent,
|
SnapshotFSImageFormat.loadSnapshotList(parent, numSnapshots, in, this);
|
||||||
numSnapshots, in, this);
|
if (parent.getDirectorySnapshottableFeature().getSnapshotQuota() > 0) {
|
||||||
if (snapshottableParent.getSnapshotQuota() > 0) {
|
|
||||||
// add the directory to the snapshottable directory list in
|
// add the directory to the snapshottable directory list in
|
||||||
// SnapshotManager. Note that we only add root when its snapshot quota
|
// SnapshotManager. Note that we only add root when its snapshot quota
|
||||||
// is positive.
|
// is positive.
|
||||||
this.namesystem.getSnapshotManager().addSnapshottable(
|
this.namesystem.getSnapshotManager().addSnapshottable(parent);
|
||||||
snapshottableParent);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -832,7 +827,10 @@ public class FSImageFormat {
|
||||||
if (withSnapshot) {
|
if (withSnapshot) {
|
||||||
dir.addSnapshotFeature(null);
|
dir.addSnapshotFeature(null);
|
||||||
}
|
}
|
||||||
return snapshottable ? new INodeDirectorySnapshottable(dir) : dir;
|
if (snapshottable) {
|
||||||
|
dir.addSnapshottableFeature();
|
||||||
|
}
|
||||||
|
return dir;
|
||||||
} else if (numBlocks == -2) {
|
} else if (numBlocks == -2) {
|
||||||
//symlink
|
//symlink
|
||||||
if (!FileSystem.areSymlinksEnabled()) {
|
if (!FileSystem.areSymlinksEnabled()) {
|
||||||
|
@ -1383,10 +1381,8 @@ public class FSImageFormat {
|
||||||
|
|
||||||
// 2. Write INodeDirectorySnapshottable#snapshotsByNames to record all
|
// 2. Write INodeDirectorySnapshottable#snapshotsByNames to record all
|
||||||
// Snapshots
|
// Snapshots
|
||||||
if (current instanceof INodeDirectorySnapshottable) {
|
if (current.isDirectory() && current.asDirectory().isSnapshottable()) {
|
||||||
INodeDirectorySnapshottable snapshottableNode =
|
SnapshotFSImageFormat.saveSnapshots(current.asDirectory(), out);
|
||||||
(INodeDirectorySnapshottable) current;
|
|
||||||
SnapshotFSImageFormat.saveSnapshots(snapshottableNode, out);
|
|
||||||
} else {
|
} else {
|
||||||
out.writeInt(-1); // # of snapshots
|
out.writeInt(-1); // # of snapshots
|
||||||
}
|
}
|
||||||
|
|
|
@ -36,7 +36,6 @@ import org.apache.hadoop.hdfs.protocol.LayoutVersion;
|
||||||
import org.apache.hadoop.hdfs.server.blockmanagement.BlockInfo;
|
import org.apache.hadoop.hdfs.server.blockmanagement.BlockInfo;
|
||||||
import org.apache.hadoop.hdfs.server.blockmanagement.BlockInfoUnderConstruction;
|
import org.apache.hadoop.hdfs.server.blockmanagement.BlockInfoUnderConstruction;
|
||||||
import org.apache.hadoop.hdfs.server.common.HdfsServerConstants.BlockUCState;
|
import org.apache.hadoop.hdfs.server.common.HdfsServerConstants.BlockUCState;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.INodeDirectorySnapshottable;
|
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.SnapshotFSImageFormat;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.SnapshotFSImageFormat;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.SnapshotFSImageFormat.ReferenceMap;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.SnapshotFSImageFormat.ReferenceMap;
|
||||||
import org.apache.hadoop.hdfs.util.XMLUtils;
|
import org.apache.hadoop.hdfs.util.XMLUtils;
|
||||||
|
@ -241,7 +240,7 @@ public class FSImageSerialization {
|
||||||
|
|
||||||
writeQuota(node.getQuotaCounts(), out);
|
writeQuota(node.getQuotaCounts(), out);
|
||||||
|
|
||||||
if (node instanceof INodeDirectorySnapshottable) {
|
if (node.isSnapshottable()) {
|
||||||
out.writeBoolean(true);
|
out.writeBoolean(true);
|
||||||
} else {
|
} else {
|
||||||
out.writeBoolean(false);
|
out.writeBoolean(false);
|
||||||
|
|
|
@ -239,7 +239,6 @@ import org.apache.hadoop.hdfs.server.namenode.ha.HAContext;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.ha.StandbyCheckpointer;
|
import org.apache.hadoop.hdfs.server.namenode.ha.StandbyCheckpointer;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.metrics.FSNamesystemMBean;
|
import org.apache.hadoop.hdfs.server.namenode.metrics.FSNamesystemMBean;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.metrics.NameNodeMetrics;
|
import org.apache.hadoop.hdfs.server.namenode.metrics.NameNodeMetrics;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.INodeDirectorySnapshottable;
|
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.SnapshotManager;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.SnapshotManager;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.startupprogress.Phase;
|
import org.apache.hadoop.hdfs.server.namenode.startupprogress.Phase;
|
||||||
|
@ -7966,7 +7965,7 @@ public class FSNamesystem implements Namesystem, FSClusterStats,
|
||||||
* Remove a list of INodeDirectorySnapshottable from the SnapshotManager
|
* Remove a list of INodeDirectorySnapshottable from the SnapshotManager
|
||||||
* @param toRemove the list of INodeDirectorySnapshottable to be removed
|
* @param toRemove the list of INodeDirectorySnapshottable to be removed
|
||||||
*/
|
*/
|
||||||
void removeSnapshottableDirs(List<INodeDirectorySnapshottable> toRemove) {
|
void removeSnapshottableDirs(List<INodeDirectory> toRemove) {
|
||||||
if (snapshotManager != null) {
|
if (snapshotManager != null) {
|
||||||
snapshotManager.removeSnapshottable(toRemove);
|
snapshotManager.removeSnapshottable(toRemove);
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,10 +29,11 @@ import org.apache.hadoop.fs.PathIsNotDirectoryException;
|
||||||
import org.apache.hadoop.fs.permission.PermissionStatus;
|
import org.apache.hadoop.fs.permission.PermissionStatus;
|
||||||
import org.apache.hadoop.hdfs.DFSUtil;
|
import org.apache.hadoop.hdfs.DFSUtil;
|
||||||
import org.apache.hadoop.hdfs.protocol.QuotaExceededException;
|
import org.apache.hadoop.hdfs.protocol.QuotaExceededException;
|
||||||
|
import org.apache.hadoop.hdfs.protocol.SnapshotException;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.INodeReference.WithCount;
|
import org.apache.hadoop.hdfs.server.namenode.INodeReference.WithCount;
|
||||||
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectorySnapshottableFeature;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectoryWithSnapshotFeature;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectoryWithSnapshotFeature;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectoryWithSnapshotFeature.DirectoryDiffList;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectoryWithSnapshotFeature.DirectoryDiffList;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.INodeDirectorySnapshottable;
|
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot;
|
||||||
import org.apache.hadoop.hdfs.util.Diff.ListType;
|
import org.apache.hadoop.hdfs.util.Diff.ListType;
|
||||||
import org.apache.hadoop.hdfs.util.ReadOnlyList;
|
import org.apache.hadoop.hdfs.util.ReadOnlyList;
|
||||||
|
@ -102,11 +103,6 @@ public class INodeDirectory extends INodeWithAdditionalFields
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Is this a snapshottable directory? */
|
|
||||||
public boolean isSnapshottable() {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setQuota(long nsQuota, long dsQuota) {
|
void setQuota(long nsQuota, long dsQuota) {
|
||||||
DirectoryWithQuotaFeature quota = getDirectoryWithQuotaFeature();
|
DirectoryWithQuotaFeature quota = getDirectoryWithQuotaFeature();
|
||||||
if (quota != null) {
|
if (quota != null) {
|
||||||
|
@ -204,50 +200,71 @@ public class INodeDirectory extends INodeWithAdditionalFields
|
||||||
return super.toDetailString() + (sf == null ? "" : ", " + sf.getDiffs());
|
return super.toDetailString() + (sf == null ? "" : ", " + sf.getDiffs());
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Replace itself with an {@link INodeDirectorySnapshottable}. */
|
public DirectorySnapshottableFeature getDirectorySnapshottableFeature() {
|
||||||
public INodeDirectorySnapshottable replaceSelf4INodeDirectorySnapshottable(
|
return getFeature(DirectorySnapshottableFeature.class);
|
||||||
int latestSnapshotId, final INodeMap inodeMap)
|
|
||||||
throws QuotaExceededException {
|
|
||||||
Preconditions.checkState(!(this instanceof INodeDirectorySnapshottable),
|
|
||||||
"this is already an INodeDirectorySnapshottable, this=%s", this);
|
|
||||||
final INodeDirectorySnapshottable s = new INodeDirectorySnapshottable(this);
|
|
||||||
replaceSelf(s, inodeMap).getDirectoryWithSnapshotFeature().getDiffs()
|
|
||||||
.saveSelf2Snapshot(latestSnapshotId, s, this);
|
|
||||||
return s;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Replace itself with {@link INodeDirectory}. */
|
public boolean isSnapshottable() {
|
||||||
public INodeDirectory replaceSelf4INodeDirectory(final INodeMap inodeMap) {
|
return getDirectorySnapshottableFeature() != null;
|
||||||
Preconditions.checkState(getClass() != INodeDirectory.class,
|
|
||||||
"the class is already INodeDirectory, this=%s", this);
|
|
||||||
return replaceSelf(new INodeDirectory(this, true, this.getFeatures()),
|
|
||||||
inodeMap);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Replace itself with the given directory. */
|
public Snapshot getSnapshot(byte[] snapshotName) {
|
||||||
private final <N extends INodeDirectory> N replaceSelf(final N newDir,
|
return getDirectorySnapshottableFeature().getSnapshot(snapshotName);
|
||||||
final INodeMap inodeMap) {
|
|
||||||
final INodeReference ref = getParentReference();
|
|
||||||
if (ref != null) {
|
|
||||||
ref.setReferredINode(newDir);
|
|
||||||
if (inodeMap != null) {
|
|
||||||
inodeMap.put(newDir);
|
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
final INodeDirectory parent = getParent();
|
public void setSnapshotQuota(int snapshotQuota) {
|
||||||
Preconditions.checkArgument(parent != null, "parent is null, this=%s", this);
|
getDirectorySnapshottableFeature().setSnapshotQuota(snapshotQuota);
|
||||||
parent.replaceChild(this, newDir, inodeMap);
|
}
|
||||||
|
|
||||||
|
public Snapshot addSnapshot(int id, String name) throws SnapshotException,
|
||||||
|
QuotaExceededException {
|
||||||
|
return getDirectorySnapshottableFeature().addSnapshot(this, id, name);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Snapshot removeSnapshot(String snapshotName,
|
||||||
|
BlocksMapUpdateInfo collectedBlocks, final List<INode> removedINodes)
|
||||||
|
throws SnapshotException {
|
||||||
|
return getDirectorySnapshottableFeature().removeSnapshot(this,
|
||||||
|
snapshotName, collectedBlocks, removedINodes);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void renameSnapshot(String path, String oldName, String newName)
|
||||||
|
throws SnapshotException {
|
||||||
|
getDirectorySnapshottableFeature().renameSnapshot(path, oldName, newName);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** add DirectorySnapshottableFeature */
|
||||||
|
public void addSnapshottableFeature() {
|
||||||
|
Preconditions.checkState(!isSnapshottable(),
|
||||||
|
"this is already snapshottable, this=%s", this);
|
||||||
|
DirectoryWithSnapshotFeature s = this.getDirectoryWithSnapshotFeature();
|
||||||
|
final DirectorySnapshottableFeature snapshottable =
|
||||||
|
new DirectorySnapshottableFeature(s);
|
||||||
|
if (s != null) {
|
||||||
|
this.removeFeature(s);
|
||||||
|
}
|
||||||
|
this.addFeature(snapshottable);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** remove DirectorySnapshottableFeature */
|
||||||
|
public void removeSnapshottableFeature() {
|
||||||
|
DirectorySnapshottableFeature s = getDirectorySnapshottableFeature();
|
||||||
|
Preconditions.checkState(s != null,
|
||||||
|
"The dir does not have snapshottable feature: this=%s", this);
|
||||||
|
this.removeFeature(s);
|
||||||
|
if (s.getDiffs().asList().size() > 0) {
|
||||||
|
// add a DirectoryWithSnapshotFeature back
|
||||||
|
DirectoryWithSnapshotFeature sf = new DirectoryWithSnapshotFeature(
|
||||||
|
s.getDiffs());
|
||||||
|
addFeature(sf);
|
||||||
}
|
}
|
||||||
clear();
|
|
||||||
return newDir;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Replace the given child with a new child. Note that we no longer need to
|
* Replace the given child with a new child. Note that we no longer need to
|
||||||
* replace an normal INodeDirectory or INodeFile into an
|
* replace an normal INodeDirectory or INodeFile into an
|
||||||
* INodeDirectoryWithSnapshot or INodeFileUnderConstruction. The only cases
|
* INodeDirectoryWithSnapshot or INodeFileUnderConstruction. The only cases
|
||||||
* for child replacement is for {@link INodeDirectorySnapshottable} and
|
* for child replacement is for reference nodes.
|
||||||
* reference nodes.
|
|
||||||
*/
|
*/
|
||||||
public void replaceChild(INode oldChild, final INode newChild,
|
public void replaceChild(INode oldChild, final INode newChild,
|
||||||
final INodeMap inodeMap) {
|
final INodeMap inodeMap) {
|
||||||
|
@ -821,6 +838,11 @@ public class INodeDirectory extends INodeWithAdditionalFields
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
final DirectorySnapshottableFeature s = getDirectorySnapshottableFeature();
|
||||||
|
if (s != null) {
|
||||||
|
s.dumpTreeRecursively(this, out, prefix, snapshot);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -829,7 +851,7 @@ public class INodeDirectory extends INodeWithAdditionalFields
|
||||||
* @param subs The subtrees.
|
* @param subs The subtrees.
|
||||||
*/
|
*/
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
protected static void dumpTreeRecursively(PrintWriter out,
|
public static void dumpTreeRecursively(PrintWriter out,
|
||||||
StringBuilder prefix, Iterable<SnapshotAndINode> subs) {
|
StringBuilder prefix, Iterable<SnapshotAndINode> subs) {
|
||||||
if (subs != null) {
|
if (subs != null) {
|
||||||
for(final Iterator<SnapshotAndINode> i = subs.iterator(); i.hasNext();) {
|
for(final Iterator<SnapshotAndINode> i = subs.iterator(); i.hasNext();) {
|
||||||
|
@ -842,7 +864,7 @@ public class INodeDirectory extends INodeWithAdditionalFields
|
||||||
}
|
}
|
||||||
|
|
||||||
/** A pair of Snapshot and INode objects. */
|
/** A pair of Snapshot and INode objects. */
|
||||||
protected static class SnapshotAndINode {
|
public static class SnapshotAndINode {
|
||||||
public final int snapshotId;
|
public final int snapshotId;
|
||||||
public final INode inode;
|
public final INode inode;
|
||||||
|
|
||||||
|
|
|
@ -315,8 +315,9 @@ public abstract class INodeWithAdditionalFields extends INode
|
||||||
}
|
}
|
||||||
|
|
||||||
protected <T extends Feature> T getFeature(Class<? extends Feature> clazz) {
|
protected <T extends Feature> T getFeature(Class<? extends Feature> clazz) {
|
||||||
|
Preconditions.checkArgument(clazz != null);
|
||||||
for (Feature f : features) {
|
for (Feature f : features) {
|
||||||
if (f.getClass() == clazz) {
|
if (clazz.isAssignableFrom(f.getClass())) {
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
T ret = (T) f;
|
T ret = (T) f;
|
||||||
return ret;
|
return ret;
|
||||||
|
|
|
@ -27,7 +27,6 @@ import org.apache.hadoop.hdfs.DFSUtil;
|
||||||
import org.apache.hadoop.hdfs.protocol.HdfsConstants;
|
import org.apache.hadoop.hdfs.protocol.HdfsConstants;
|
||||||
import org.apache.hadoop.hdfs.protocol.UnresolvedPathException;
|
import org.apache.hadoop.hdfs.protocol.UnresolvedPathException;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectoryWithSnapshotFeature;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectoryWithSnapshotFeature;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.INodeDirectorySnapshottable;
|
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot;
|
||||||
|
|
||||||
import com.google.common.base.Preconditions;
|
import com.google.common.base.Preconditions;
|
||||||
|
@ -208,8 +207,7 @@ public class INodesInPath {
|
||||||
final byte[] childName = components[count + 1];
|
final byte[] childName = components[count + 1];
|
||||||
|
|
||||||
// check if the next byte[] in components is for ".snapshot"
|
// check if the next byte[] in components is for ".snapshot"
|
||||||
if (isDotSnapshotDir(childName)
|
if (isDotSnapshotDir(childName) && isDir && dir.isSnapshottable()) {
|
||||||
&& isDir && dir instanceof INodeDirectorySnapshottable) {
|
|
||||||
// skip the ".snapshot" in components
|
// skip the ".snapshot" in components
|
||||||
count++;
|
count++;
|
||||||
index++;
|
index++;
|
||||||
|
@ -222,8 +220,7 @@ public class INodesInPath {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
// Resolve snapshot root
|
// Resolve snapshot root
|
||||||
final Snapshot s = ((INodeDirectorySnapshottable)dir).getSnapshot(
|
final Snapshot s = dir.getSnapshot(components[count + 1]);
|
||||||
components[count + 1]);
|
|
||||||
if (s == null) {
|
if (s == null) {
|
||||||
//snapshot not found
|
//snapshot not found
|
||||||
curNode = null;
|
curNode = null;
|
||||||
|
|
|
@ -17,7 +17,6 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.hadoop.hdfs.server.namenode.snapshot;
|
package org.apache.hadoop.hdfs.server.namenode.snapshot;
|
||||||
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.io.PrintWriter;
|
import java.io.PrintWriter;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
|
@ -33,67 +32,43 @@ import org.apache.hadoop.hdfs.protocol.SnapshotException;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.Content;
|
import org.apache.hadoop.hdfs.server.namenode.Content;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.ContentSummaryComputationContext;
|
import org.apache.hadoop.hdfs.server.namenode.ContentSummaryComputationContext;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.INode;
|
import org.apache.hadoop.hdfs.server.namenode.INode;
|
||||||
|
import org.apache.hadoop.hdfs.server.namenode.INode.BlocksMapUpdateInfo;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.INodeDirectory;
|
import org.apache.hadoop.hdfs.server.namenode.INodeDirectory;
|
||||||
|
import org.apache.hadoop.hdfs.server.namenode.INodeDirectory.SnapshotAndINode;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.INodeFile;
|
import org.apache.hadoop.hdfs.server.namenode.INodeFile;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.INodeMap;
|
|
||||||
import org.apache.hadoop.hdfs.server.namenode.INodeReference;
|
import org.apache.hadoop.hdfs.server.namenode.INodeReference;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.INodeReference.WithCount;
|
import org.apache.hadoop.hdfs.server.namenode.INodeReference.WithCount;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.INodeReference.WithName;
|
import org.apache.hadoop.hdfs.server.namenode.INodeReference.WithName;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.Quota;
|
import org.apache.hadoop.hdfs.server.namenode.Quota;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectoryWithSnapshotFeature.ChildrenDiff;
|
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectoryWithSnapshotFeature.DirectoryDiff;
|
|
||||||
import org.apache.hadoop.hdfs.util.Diff.ListType;
|
import org.apache.hadoop.hdfs.util.Diff.ListType;
|
||||||
import org.apache.hadoop.hdfs.util.ReadOnlyList;
|
import org.apache.hadoop.hdfs.util.ReadOnlyList;
|
||||||
import org.apache.hadoop.util.Time;
|
import org.apache.hadoop.util.Time;
|
||||||
|
|
||||||
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
import com.google.common.base.Preconditions;
|
import com.google.common.base.Preconditions;
|
||||||
import com.google.common.collect.Lists;
|
import com.google.common.collect.Lists;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Directories where taking snapshots is allowed.
|
* A directory with this feature is a snapshottable directory, where snapshots
|
||||||
*
|
* can be taken. This feature extends {@link DirectoryWithSnapshotFeature}, and
|
||||||
* Like other {@link INode} subclasses, this class is synchronized externally
|
* maintains extra information about all the snapshots taken on this directory.
|
||||||
* by the namesystem and FSDirectory locks.
|
|
||||||
*/
|
*/
|
||||||
@InterfaceAudience.Private
|
@InterfaceAudience.Private
|
||||||
public class INodeDirectorySnapshottable extends INodeDirectory {
|
public class DirectorySnapshottableFeature extends DirectoryWithSnapshotFeature {
|
||||||
/** Limit the number of snapshot per snapshottable directory. */
|
/** Limit the number of snapshot per snapshottable directory. */
|
||||||
static final int SNAPSHOT_LIMIT = 1 << 16;
|
static final int SNAPSHOT_LIMIT = 1 << 16;
|
||||||
|
|
||||||
/** Cast INode to INodeDirectorySnapshottable. */
|
|
||||||
static public INodeDirectorySnapshottable valueOf(
|
|
||||||
INode inode, String src) throws IOException {
|
|
||||||
final INodeDirectory dir = INodeDirectory.valueOf(inode, src);
|
|
||||||
if (!dir.isSnapshottable()) {
|
|
||||||
throw new SnapshotException(
|
|
||||||
"Directory is not a snapshottable directory: " + src);
|
|
||||||
}
|
|
||||||
return (INodeDirectorySnapshottable)dir;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Snapshots of this directory in ascending order of snapshot names.
|
* Snapshots of this directory in ascending order of snapshot names.
|
||||||
* Note that snapshots in ascending order of snapshot id are stored in
|
* Note that snapshots in ascending order of snapshot id are stored in
|
||||||
* {@link INodeDirectoryWithSnapshot}.diffs (a private field).
|
* {@link INodeDirectoryWithSnapshot}.diffs (a private field).
|
||||||
*/
|
*/
|
||||||
private final List<Snapshot> snapshotsByNames = new ArrayList<Snapshot>();
|
private final List<Snapshot> snapshotsByNames = new ArrayList<Snapshot>();
|
||||||
|
|
||||||
/**
|
|
||||||
* @return {@link #snapshotsByNames}
|
|
||||||
*/
|
|
||||||
ReadOnlyList<Snapshot> getSnapshotsByNames() {
|
|
||||||
return ReadOnlyList.Util.asReadOnlyList(this.snapshotsByNames);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Number of snapshots allowed. */
|
/** Number of snapshots allowed. */
|
||||||
private int snapshotQuota = SNAPSHOT_LIMIT;
|
private int snapshotQuota = SNAPSHOT_LIMIT;
|
||||||
|
|
||||||
public INodeDirectorySnapshottable(INodeDirectory dir) {
|
public DirectorySnapshottableFeature(DirectoryWithSnapshotFeature feature) {
|
||||||
super(dir, true, dir.getFeatures());
|
super(feature == null ? null : feature.getDiffs());
|
||||||
// add snapshot feature if the original directory does not have it
|
|
||||||
if (!isWithSnapshot()) {
|
|
||||||
addSnapshotFeature(null);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/** @return the number of existing snapshots. */
|
/** @return the number of existing snapshots. */
|
||||||
|
@ -139,7 +114,7 @@ public class INodeDirectorySnapshottable extends INodeDirectory {
|
||||||
* name does not exist or a snapshot with the new name already
|
* name does not exist or a snapshot with the new name already
|
||||||
* exists
|
* exists
|
||||||
*/
|
*/
|
||||||
void renameSnapshot(String path, String oldName, String newName)
|
public void renameSnapshot(String path, String oldName, String newName)
|
||||||
throws SnapshotException {
|
throws SnapshotException {
|
||||||
if (newName.equals(oldName)) {
|
if (newName.equals(oldName)) {
|
||||||
return;
|
return;
|
||||||
|
@ -180,22 +155,17 @@ public class INodeDirectorySnapshottable extends INodeDirectory {
|
||||||
this.snapshotQuota = snapshotQuota;
|
this.snapshotQuota = snapshotQuota;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean isSnapshottable() {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Simply add a snapshot into the {@link #snapshotsByNames}. Used by FSImage
|
* Simply add a snapshot into the {@link #snapshotsByNames}. Used when loading
|
||||||
* loading.
|
* fsimage.
|
||||||
*/
|
*/
|
||||||
void addSnapshot(Snapshot snapshot) {
|
void addSnapshot(Snapshot snapshot) {
|
||||||
this.snapshotsByNames.add(snapshot);
|
this.snapshotsByNames.add(snapshot);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Add a snapshot. */
|
/** Add a snapshot. */
|
||||||
Snapshot addSnapshot(int id, String name) throws SnapshotException,
|
public Snapshot addSnapshot(INodeDirectory snapshotRoot, int id, String name)
|
||||||
QuotaExceededException {
|
throws SnapshotException, QuotaExceededException {
|
||||||
//check snapshot quota
|
//check snapshot quota
|
||||||
final int n = getNumSnapshots();
|
final int n = getNumSnapshots();
|
||||||
if (n + 1 > snapshotQuota) {
|
if (n + 1 > snapshotQuota) {
|
||||||
|
@ -203,7 +173,7 @@ public class INodeDirectorySnapshottable extends INodeDirectory {
|
||||||
+ n + " snapshot(s) and the snapshot quota is "
|
+ n + " snapshot(s) and the snapshot quota is "
|
||||||
+ snapshotQuota);
|
+ snapshotQuota);
|
||||||
}
|
}
|
||||||
final Snapshot s = new Snapshot(id, name, this);
|
final Snapshot s = new Snapshot(id, name, snapshotRoot);
|
||||||
final byte[] nameBytes = s.getRoot().getLocalNameBytes();
|
final byte[] nameBytes = s.getRoot().getLocalNameBytes();
|
||||||
final int i = searchSnapshot(nameBytes);
|
final int i = searchSnapshot(nameBytes);
|
||||||
if (i >= 0) {
|
if (i >= 0) {
|
||||||
|
@ -211,14 +181,14 @@ public class INodeDirectorySnapshottable extends INodeDirectory {
|
||||||
+ "snapshot with the same name \"" + Snapshot.getSnapshotName(s) + "\".");
|
+ "snapshot with the same name \"" + Snapshot.getSnapshotName(s) + "\".");
|
||||||
}
|
}
|
||||||
|
|
||||||
final DirectoryDiff d = getDiffs().addDiff(id, this);
|
final DirectoryDiff d = getDiffs().addDiff(id, snapshotRoot);
|
||||||
d.setSnapshotRoot(s.getRoot());
|
d.setSnapshotRoot(s.getRoot());
|
||||||
snapshotsByNames.add(-i - 1, s);
|
snapshotsByNames.add(-i - 1, s);
|
||||||
|
|
||||||
//set modification time
|
// set modification time
|
||||||
updateModificationTime(Time.now(), Snapshot.CURRENT_STATE_ID);
|
final long now = Time.now();
|
||||||
s.getRoot().setModificationTime(getModificationTime(),
|
snapshotRoot.updateModificationTime(now, Snapshot.CURRENT_STATE_ID);
|
||||||
Snapshot.CURRENT_STATE_ID);
|
s.getRoot().setModificationTime(now, Snapshot.CURRENT_STATE_ID);
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -226,26 +196,27 @@ public class INodeDirectorySnapshottable extends INodeDirectory {
|
||||||
* Remove the snapshot with the given name from {@link #snapshotsByNames},
|
* Remove the snapshot with the given name from {@link #snapshotsByNames},
|
||||||
* and delete all the corresponding DirectoryDiff.
|
* and delete all the corresponding DirectoryDiff.
|
||||||
*
|
*
|
||||||
|
* @param snapshotRoot The directory where we take snapshots
|
||||||
* @param snapshotName The name of the snapshot to be removed
|
* @param snapshotName The name of the snapshot to be removed
|
||||||
* @param collectedBlocks Used to collect information to update blocksMap
|
* @param collectedBlocks Used to collect information to update blocksMap
|
||||||
* @return The removed snapshot. Null if no snapshot with the given name
|
* @return The removed snapshot. Null if no snapshot with the given name
|
||||||
* exists.
|
* exists.
|
||||||
*/
|
*/
|
||||||
Snapshot removeSnapshot(String snapshotName,
|
public Snapshot removeSnapshot(INodeDirectory snapshotRoot,
|
||||||
BlocksMapUpdateInfo collectedBlocks, final List<INode> removedINodes)
|
String snapshotName, BlocksMapUpdateInfo collectedBlocks,
|
||||||
throws SnapshotException {
|
final List<INode> removedINodes) throws SnapshotException {
|
||||||
final int i = searchSnapshot(DFSUtil.string2Bytes(snapshotName));
|
final int i = searchSnapshot(DFSUtil.string2Bytes(snapshotName));
|
||||||
if (i < 0) {
|
if (i < 0) {
|
||||||
throw new SnapshotException("Cannot delete snapshot " + snapshotName
|
throw new SnapshotException("Cannot delete snapshot " + snapshotName
|
||||||
+ " from path " + this.getFullPathName()
|
+ " from path " + snapshotRoot.getFullPathName()
|
||||||
+ ": the snapshot does not exist.");
|
+ ": the snapshot does not exist.");
|
||||||
} else {
|
} else {
|
||||||
final Snapshot snapshot = snapshotsByNames.get(i);
|
final Snapshot snapshot = snapshotsByNames.get(i);
|
||||||
int prior = Snapshot.findLatestSnapshot(this, snapshot.getId());
|
int prior = Snapshot.findLatestSnapshot(snapshotRoot, snapshot.getId());
|
||||||
try {
|
try {
|
||||||
Quota.Counts counts = cleanSubtree(snapshot.getId(), prior,
|
Quota.Counts counts = snapshotRoot.cleanSubtree(snapshot.getId(),
|
||||||
collectedBlocks, removedINodes, true);
|
prior, collectedBlocks, removedINodes, true);
|
||||||
INodeDirectory parent = getParent();
|
INodeDirectory parent = snapshotRoot.getParent();
|
||||||
if (parent != null) {
|
if (parent != null) {
|
||||||
// there will not be any WithName node corresponding to the deleted
|
// there will not be any WithName node corresponding to the deleted
|
||||||
// snapshot, thus only update the quota usage in the current tree
|
// snapshot, thus only update the quota usage in the current tree
|
||||||
|
@ -253,7 +224,7 @@ public class INodeDirectorySnapshottable extends INodeDirectory {
|
||||||
-counts.get(Quota.DISKSPACE), true);
|
-counts.get(Quota.DISKSPACE), true);
|
||||||
}
|
}
|
||||||
} catch(QuotaExceededException e) {
|
} catch(QuotaExceededException e) {
|
||||||
LOG.error("BUG: removeSnapshot increases namespace usage.", e);
|
INode.LOG.error("BUG: removeSnapshot increases namespace usage.", e);
|
||||||
}
|
}
|
||||||
// remove from snapshotsByNames after successfully cleaning the subtree
|
// remove from snapshotsByNames after successfully cleaning the subtree
|
||||||
snapshotsByNames.remove(i);
|
snapshotsByNames.remove(i);
|
||||||
|
@ -261,10 +232,10 @@ public class INodeDirectorySnapshottable extends INodeDirectory {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public ContentSummaryComputationContext computeContentSummary(
|
public ContentSummaryComputationContext computeContentSummary(
|
||||||
|
final INodeDirectory snapshotRoot,
|
||||||
final ContentSummaryComputationContext summary) {
|
final ContentSummaryComputationContext summary) {
|
||||||
super.computeContentSummary(summary);
|
snapshotRoot.computeContentSummary(summary);
|
||||||
summary.getCounts().add(Content.SNAPSHOT, snapshotsByNames.size());
|
summary.getCounts().add(Content.SNAPSHOT, snapshotsByNames.size());
|
||||||
summary.getCounts().add(Content.SNAPSHOTTABLE_DIRECTORY, 1);
|
summary.getCounts().add(Content.SNAPSHOTTABLE_DIRECTORY, 1);
|
||||||
return summary;
|
return summary;
|
||||||
|
@ -282,36 +253,38 @@ public class INodeDirectorySnapshottable extends INodeDirectory {
|
||||||
* point, or if endSnapshotName is not null but cannot be identified
|
* point, or if endSnapshotName is not null but cannot be identified
|
||||||
* as a previous snapshot.
|
* as a previous snapshot.
|
||||||
*/
|
*/
|
||||||
SnapshotDiffInfo computeDiff(final String from, final String to)
|
SnapshotDiffInfo computeDiff(final INodeDirectory snapshotRoot,
|
||||||
throws SnapshotException {
|
final String from, final String to) throws SnapshotException {
|
||||||
Snapshot fromSnapshot = getSnapshotByName(from);
|
Snapshot fromSnapshot = getSnapshotByName(snapshotRoot, from);
|
||||||
Snapshot toSnapshot = getSnapshotByName(to);
|
Snapshot toSnapshot = getSnapshotByName(snapshotRoot, to);
|
||||||
// if the start point is equal to the end point, return null
|
// if the start point is equal to the end point, return null
|
||||||
if (from.equals(to)) {
|
if (from.equals(to)) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
SnapshotDiffInfo diffs = new SnapshotDiffInfo(this, fromSnapshot,
|
SnapshotDiffInfo diffs = new SnapshotDiffInfo(snapshotRoot, fromSnapshot,
|
||||||
toSnapshot);
|
toSnapshot);
|
||||||
computeDiffRecursively(this, new ArrayList<byte[]>(), diffs);
|
computeDiffRecursively(snapshotRoot, snapshotRoot, new ArrayList<byte[]>(),
|
||||||
|
diffs);
|
||||||
return diffs;
|
return diffs;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Find the snapshot matching the given name.
|
* Find the snapshot matching the given name.
|
||||||
*
|
*
|
||||||
|
* @param snapshotRoot The directory where snapshots were taken.
|
||||||
* @param snapshotName The name of the snapshot.
|
* @param snapshotName The name of the snapshot.
|
||||||
* @return The corresponding snapshot. Null if snapshotName is null or empty.
|
* @return The corresponding snapshot. Null if snapshotName is null or empty.
|
||||||
* @throws SnapshotException If snapshotName is not null or empty, but there
|
* @throws SnapshotException If snapshotName is not null or empty, but there
|
||||||
* is no snapshot matching the name.
|
* is no snapshot matching the name.
|
||||||
*/
|
*/
|
||||||
private Snapshot getSnapshotByName(String snapshotName)
|
private Snapshot getSnapshotByName(INodeDirectory snapshotRoot,
|
||||||
throws SnapshotException {
|
String snapshotName) throws SnapshotException {
|
||||||
Snapshot s = null;
|
Snapshot s = null;
|
||||||
if (snapshotName != null && !snapshotName.isEmpty()) {
|
if (snapshotName != null && !snapshotName.isEmpty()) {
|
||||||
final int index = searchSnapshot(DFSUtil.string2Bytes(snapshotName));
|
final int index = searchSnapshot(DFSUtil.string2Bytes(snapshotName));
|
||||||
if (index < 0) {
|
if (index < 0) {
|
||||||
throw new SnapshotException("Cannot find the snapshot of directory "
|
throw new SnapshotException("Cannot find the snapshot of directory "
|
||||||
+ this.getFullPathName() + " with name " + snapshotName);
|
+ snapshotRoot.getFullPathName() + " with name " + snapshotName);
|
||||||
}
|
}
|
||||||
s = snapshotsByNames.get(index);
|
s = snapshotsByNames.get(index);
|
||||||
}
|
}
|
||||||
|
@ -321,13 +294,14 @@ public class INodeDirectorySnapshottable extends INodeDirectory {
|
||||||
/**
|
/**
|
||||||
* Recursively compute the difference between snapshots under a given
|
* Recursively compute the difference between snapshots under a given
|
||||||
* directory/file.
|
* directory/file.
|
||||||
|
* @param snapshotRoot The directory where snapshots were taken.
|
||||||
* @param node The directory/file under which the diff is computed.
|
* @param node The directory/file under which the diff is computed.
|
||||||
* @param parentPath Relative path (corresponding to the snapshot root) of
|
* @param parentPath Relative path (corresponding to the snapshot root) of
|
||||||
* the node's parent.
|
* the node's parent.
|
||||||
* @param diffReport data structure used to store the diff.
|
* @param diffReport data structure used to store the diff.
|
||||||
*/
|
*/
|
||||||
private void computeDiffRecursively(INode node, List<byte[]> parentPath,
|
private void computeDiffRecursively(final INodeDirectory snapshotRoot,
|
||||||
SnapshotDiffInfo diffReport) {
|
INode node, List<byte[]> parentPath, SnapshotDiffInfo diffReport) {
|
||||||
final Snapshot earlierSnapshot = diffReport.isFromEarlier() ?
|
final Snapshot earlierSnapshot = diffReport.isFromEarlier() ?
|
||||||
diffReport.getFrom() : diffReport.getTo();
|
diffReport.getFrom() : diffReport.getTo();
|
||||||
final Snapshot laterSnapshot = diffReport.isFromEarlier() ?
|
final Snapshot laterSnapshot = diffReport.isFromEarlier() ?
|
||||||
|
@ -350,7 +324,8 @@ public class INodeDirectorySnapshottable extends INodeDirectory {
|
||||||
final byte[] name = child.getLocalNameBytes();
|
final byte[] name = child.getLocalNameBytes();
|
||||||
boolean toProcess = diff.searchIndex(ListType.DELETED, name) < 0;
|
boolean toProcess = diff.searchIndex(ListType.DELETED, name) < 0;
|
||||||
if (!toProcess && child instanceof INodeReference.WithName) {
|
if (!toProcess && child instanceof INodeReference.WithName) {
|
||||||
byte[][] renameTargetPath = findRenameTargetPath((WithName) child,
|
byte[][] renameTargetPath = findRenameTargetPath(
|
||||||
|
snapshotRoot, (WithName) child,
|
||||||
laterSnapshot == null ? Snapshot.CURRENT_STATE_ID :
|
laterSnapshot == null ? Snapshot.CURRENT_STATE_ID :
|
||||||
laterSnapshot.getId());
|
laterSnapshot.getId());
|
||||||
if (renameTargetPath != null) {
|
if (renameTargetPath != null) {
|
||||||
|
@ -360,7 +335,7 @@ public class INodeDirectorySnapshottable extends INodeDirectory {
|
||||||
}
|
}
|
||||||
if (toProcess) {
|
if (toProcess) {
|
||||||
parentPath.add(name);
|
parentPath.add(name);
|
||||||
computeDiffRecursively(child, parentPath, diffReport);
|
computeDiffRecursively(snapshotRoot, child, parentPath, diffReport);
|
||||||
parentPath.remove(parentPath.size() - 1);
|
parentPath.remove(parentPath.size() - 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -379,12 +354,12 @@ public class INodeDirectorySnapshottable extends INodeDirectory {
|
||||||
* However, we should include it in our snapshot diff report as rename only
|
* However, we should include it in our snapshot diff report as rename only
|
||||||
* if the rename target is also under the same snapshottable directory.
|
* if the rename target is also under the same snapshottable directory.
|
||||||
*/
|
*/
|
||||||
private byte[][] findRenameTargetPath(INodeReference.WithName wn,
|
private byte[][] findRenameTargetPath(final INodeDirectory snapshotRoot,
|
||||||
final int snapshotId) {
|
INodeReference.WithName wn, final int snapshotId) {
|
||||||
INode inode = wn.getReferredINode();
|
INode inode = wn.getReferredINode();
|
||||||
final LinkedList<byte[]> ancestors = Lists.newLinkedList();
|
final LinkedList<byte[]> ancestors = Lists.newLinkedList();
|
||||||
while (inode != null) {
|
while (inode != null) {
|
||||||
if (inode == this) {
|
if (inode == snapshotRoot) {
|
||||||
return ancestors.toArray(new byte[ancestors.size()][]);
|
return ancestors.toArray(new byte[ancestors.size()][]);
|
||||||
}
|
}
|
||||||
if (inode instanceof INodeReference.WithCount) {
|
if (inode instanceof INodeReference.WithCount) {
|
||||||
|
@ -407,39 +382,20 @@ public class INodeDirectorySnapshottable extends INodeDirectory {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Replace itself with {@link INodeDirectoryWithSnapshot} or
|
|
||||||
* {@link INodeDirectory} depending on the latest snapshot.
|
|
||||||
*/
|
|
||||||
INodeDirectory replaceSelf(final int latestSnapshotId, final INodeMap inodeMap)
|
|
||||||
throws QuotaExceededException {
|
|
||||||
if (latestSnapshotId == Snapshot.CURRENT_STATE_ID) {
|
|
||||||
Preconditions.checkState(getDirectoryWithSnapshotFeature()
|
|
||||||
.getLastSnapshotId() == Snapshot.CURRENT_STATE_ID, "this=%s", this);
|
|
||||||
}
|
|
||||||
INodeDirectory dir = replaceSelf4INodeDirectory(inodeMap);
|
|
||||||
if (latestSnapshotId != Snapshot.CURRENT_STATE_ID) {
|
|
||||||
dir.recordModification(latestSnapshotId);
|
|
||||||
}
|
|
||||||
return dir;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String toDetailString() {
|
public String toString() {
|
||||||
return super.toDetailString() + ", snapshotsByNames=" + snapshotsByNames;
|
return "snapshotsByNames=" + snapshotsByNames;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@VisibleForTesting
|
||||||
public void dumpTreeRecursively(PrintWriter out, StringBuilder prefix,
|
public void dumpTreeRecursively(INodeDirectory snapshotRoot, PrintWriter out,
|
||||||
int snapshot) {
|
StringBuilder prefix, int snapshot) {
|
||||||
super.dumpTreeRecursively(out, prefix, snapshot);
|
|
||||||
|
|
||||||
if (snapshot == Snapshot.CURRENT_STATE_ID) {
|
if (snapshot == Snapshot.CURRENT_STATE_ID) {
|
||||||
out.println();
|
out.println();
|
||||||
out.print(prefix);
|
out.print(prefix);
|
||||||
|
|
||||||
out.print("Snapshot of ");
|
out.print("Snapshot of ");
|
||||||
final String name = getLocalName();
|
final String name = snapshotRoot.getLocalName();
|
||||||
out.print(name.isEmpty()? "/": name);
|
out.print(name.isEmpty()? "/": name);
|
||||||
out.print(": quota=");
|
out.print(": quota=");
|
||||||
out.print(getSnapshotQuota());
|
out.print(getSnapshotQuota());
|
||||||
|
@ -455,7 +411,8 @@ public class INodeDirectorySnapshottable extends INodeDirectory {
|
||||||
out.print(", #snapshot=");
|
out.print(", #snapshot=");
|
||||||
out.println(n);
|
out.println(n);
|
||||||
|
|
||||||
dumpTreeRecursively(out, prefix, new Iterable<SnapshotAndINode>() {
|
INodeDirectory.dumpTreeRecursively(out, prefix,
|
||||||
|
new Iterable<SnapshotAndINode>() {
|
||||||
@Override
|
@Override
|
||||||
public Iterator<SnapshotAndINode> iterator() {
|
public Iterator<SnapshotAndINode> iterator() {
|
||||||
return new Iterator<SnapshotAndINode>() {
|
return new Iterator<SnapshotAndINode>() {
|
|
@ -48,7 +48,9 @@ import org.apache.hadoop.hdfs.util.ReadOnlyList;
|
||||||
import com.google.common.base.Preconditions;
|
import com.google.common.base.Preconditions;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Feature for directory with snapshot-related information.
|
* Feature used to store and process the snapshot diff information for a
|
||||||
|
* directory. In particular, it contains a directory diff list recording changes
|
||||||
|
* made to the directory and its children for each snapshot.
|
||||||
*/
|
*/
|
||||||
@InterfaceAudience.Private
|
@InterfaceAudience.Private
|
||||||
public class DirectoryWithSnapshotFeature implements INode.Feature {
|
public class DirectoryWithSnapshotFeature implements INode.Feature {
|
||||||
|
|
|
@ -127,9 +127,8 @@ public class FSImageFormatPBSnapshot {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Load the snapshots section from fsimage. Also convert snapshottable
|
* Load the snapshots section from fsimage. Also add snapshottable feature
|
||||||
* directories into {@link INodeDirectorySnapshottable}.
|
* to snapshottable directories.
|
||||||
*
|
|
||||||
*/
|
*/
|
||||||
public void loadSnapshotSection(InputStream in) throws IOException {
|
public void loadSnapshotSection(InputStream in) throws IOException {
|
||||||
SnapshotManager sm = fsn.getSnapshotManager();
|
SnapshotManager sm = fsn.getSnapshotManager();
|
||||||
|
@ -139,16 +138,13 @@ public class FSImageFormatPBSnapshot {
|
||||||
sm.setSnapshotCounter(section.getSnapshotCounter());
|
sm.setSnapshotCounter(section.getSnapshotCounter());
|
||||||
for (long sdirId : section.getSnapshottableDirList()) {
|
for (long sdirId : section.getSnapshottableDirList()) {
|
||||||
INodeDirectory dir = fsDir.getInode(sdirId).asDirectory();
|
INodeDirectory dir = fsDir.getInode(sdirId).asDirectory();
|
||||||
final INodeDirectorySnapshottable sdir;
|
|
||||||
if (!dir.isSnapshottable()) {
|
if (!dir.isSnapshottable()) {
|
||||||
sdir = new INodeDirectorySnapshottable(dir);
|
dir.addSnapshottableFeature();
|
||||||
fsDir.addToInodeMap(sdir);
|
|
||||||
} else {
|
} else {
|
||||||
// dir is root, and admin set root to snapshottable before
|
// dir is root, and admin set root to snapshottable before
|
||||||
sdir = (INodeDirectorySnapshottable) dir;
|
dir.setSnapshotQuota(DirectorySnapshottableFeature.SNAPSHOT_LIMIT);
|
||||||
sdir.setSnapshotQuota(INodeDirectorySnapshottable.SNAPSHOT_LIMIT);
|
|
||||||
}
|
}
|
||||||
sm.addSnapshottable(sdir);
|
sm.addSnapshottable(dir);
|
||||||
}
|
}
|
||||||
loadSnapshots(in, snum);
|
loadSnapshots(in, snum);
|
||||||
}
|
}
|
||||||
|
@ -160,12 +156,11 @@ public class FSImageFormatPBSnapshot {
|
||||||
INodeDirectory root = loadINodeDirectory(pbs.getRoot(),
|
INodeDirectory root = loadINodeDirectory(pbs.getRoot(),
|
||||||
parent.getLoaderContext());
|
parent.getLoaderContext());
|
||||||
int sid = pbs.getSnapshotId();
|
int sid = pbs.getSnapshotId();
|
||||||
INodeDirectorySnapshottable parent = (INodeDirectorySnapshottable) fsDir
|
INodeDirectory parent = fsDir.getInode(root.getId()).asDirectory();
|
||||||
.getInode(root.getId()).asDirectory();
|
|
||||||
Snapshot snapshot = new Snapshot(sid, root, parent);
|
Snapshot snapshot = new Snapshot(sid, root, parent);
|
||||||
// add the snapshot to parent, since we follow the sequence of
|
// add the snapshot to parent, since we follow the sequence of
|
||||||
// snapshotsByNames when saving, we do not need to sort when loading
|
// snapshotsByNames when saving, we do not need to sort when loading
|
||||||
parent.addSnapshot(snapshot);
|
parent.getDirectorySnapshottableFeature().addSnapshot(snapshot);
|
||||||
snapshotMap.put(sid, snapshot);
|
snapshotMap.put(sid, snapshot);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -373,14 +368,15 @@ public class FSImageFormatPBSnapshot {
|
||||||
.setSnapshotCounter(sm.getSnapshotCounter())
|
.setSnapshotCounter(sm.getSnapshotCounter())
|
||||||
.setNumSnapshots(sm.getNumSnapshots());
|
.setNumSnapshots(sm.getNumSnapshots());
|
||||||
|
|
||||||
INodeDirectorySnapshottable[] snapshottables = sm.getSnapshottableDirs();
|
INodeDirectory[] snapshottables = sm.getSnapshottableDirs();
|
||||||
for (INodeDirectorySnapshottable sdir : snapshottables) {
|
for (INodeDirectory sdir : snapshottables) {
|
||||||
b.addSnapshottableDir(sdir.getId());
|
b.addSnapshottableDir(sdir.getId());
|
||||||
}
|
}
|
||||||
b.build().writeDelimitedTo(out);
|
b.build().writeDelimitedTo(out);
|
||||||
int i = 0;
|
int i = 0;
|
||||||
for(INodeDirectorySnapshottable sdir : snapshottables) {
|
for(INodeDirectory sdir : snapshottables) {
|
||||||
for(Snapshot s : sdir.getSnapshotsByNames()) {
|
for (Snapshot s : sdir.getDirectorySnapshottableFeature()
|
||||||
|
.getSnapshotList()) {
|
||||||
Root sroot = s.getRoot();
|
Root sroot = s.getRoot();
|
||||||
SnapshotSection.Snapshot.Builder sb = SnapshotSection.Snapshot
|
SnapshotSection.Snapshot.Builder sb = SnapshotSection.Snapshot
|
||||||
.newBuilder().setSnapshotId(s.getId());
|
.newBuilder().setSnapshotId(s.getId());
|
||||||
|
|
|
@ -184,15 +184,14 @@ public class Snapshot implements Comparable<byte[]> {
|
||||||
/** The root directory of the snapshot. */
|
/** The root directory of the snapshot. */
|
||||||
private final Root root;
|
private final Root root;
|
||||||
|
|
||||||
Snapshot(int id, String name, INodeDirectorySnapshottable dir) {
|
Snapshot(int id, String name, INodeDirectory dir) {
|
||||||
this(id, dir, dir);
|
this(id, dir, dir);
|
||||||
this.root.setLocalName(DFSUtil.string2Bytes(name));
|
this.root.setLocalName(DFSUtil.string2Bytes(name));
|
||||||
}
|
}
|
||||||
|
|
||||||
Snapshot(int id, INodeDirectory dir, INodeDirectorySnapshottable parent) {
|
Snapshot(int id, INodeDirectory dir, INodeDirectory parent) {
|
||||||
this.id = id;
|
this.id = id;
|
||||||
this.root = new Root(dir);
|
this.root = new Root(dir);
|
||||||
|
|
||||||
this.root.setParent(parent);
|
this.root.setParent(parent);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -99,7 +99,7 @@ class SnapshotDiffInfo {
|
||||||
}
|
}
|
||||||
|
|
||||||
/** The root directory of the snapshots */
|
/** The root directory of the snapshots */
|
||||||
private final INodeDirectorySnapshottable snapshotRoot;
|
private final INodeDirectory snapshotRoot;
|
||||||
/** The starting point of the difference */
|
/** The starting point of the difference */
|
||||||
private final Snapshot from;
|
private final Snapshot from;
|
||||||
/** The end point of the difference */
|
/** The end point of the difference */
|
||||||
|
@ -122,8 +122,8 @@ class SnapshotDiffInfo {
|
||||||
private final Map<Long, RenameEntry> renameMap =
|
private final Map<Long, RenameEntry> renameMap =
|
||||||
new HashMap<Long, RenameEntry>();
|
new HashMap<Long, RenameEntry>();
|
||||||
|
|
||||||
SnapshotDiffInfo(INodeDirectorySnapshottable snapshotRoot, Snapshot start,
|
SnapshotDiffInfo(INodeDirectory snapshotRoot, Snapshot start, Snapshot end) {
|
||||||
Snapshot end) {
|
Preconditions.checkArgument(snapshotRoot.isSnapshottable());
|
||||||
this.snapshotRoot = snapshotRoot;
|
this.snapshotRoot = snapshotRoot;
|
||||||
this.from = start;
|
this.from = start;
|
||||||
this.to = end;
|
this.to = end;
|
||||||
|
|
|
@ -41,6 +41,8 @@ import org.apache.hadoop.hdfs.tools.snapshot.SnapshotDiff;
|
||||||
import org.apache.hadoop.hdfs.util.Diff.ListType;
|
import org.apache.hadoop.hdfs.util.Diff.ListType;
|
||||||
import org.apache.hadoop.hdfs.util.ReadOnlyList;
|
import org.apache.hadoop.hdfs.util.ReadOnlyList;
|
||||||
|
|
||||||
|
import com.google.common.base.Preconditions;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A helper class defining static methods for reading/writing snapshot related
|
* A helper class defining static methods for reading/writing snapshot related
|
||||||
* information from/to FSImage.
|
* information from/to FSImage.
|
||||||
|
@ -52,17 +54,19 @@ public class SnapshotFSImageFormat {
|
||||||
* @param out The {@link DataOutput} to write.
|
* @param out The {@link DataOutput} to write.
|
||||||
* @throws IOException
|
* @throws IOException
|
||||||
*/
|
*/
|
||||||
public static void saveSnapshots(INodeDirectorySnapshottable current,
|
public static void saveSnapshots(INodeDirectory current, DataOutput out)
|
||||||
DataOutput out) throws IOException {
|
throws IOException {
|
||||||
|
DirectorySnapshottableFeature sf = current.getDirectorySnapshottableFeature();
|
||||||
|
Preconditions.checkArgument(sf != null);
|
||||||
// list of snapshots in snapshotsByNames
|
// list of snapshots in snapshotsByNames
|
||||||
ReadOnlyList<Snapshot> snapshots = current.getSnapshotsByNames();
|
ReadOnlyList<Snapshot> snapshots = sf.getSnapshotList();
|
||||||
out.writeInt(snapshots.size());
|
out.writeInt(snapshots.size());
|
||||||
for (Snapshot s : snapshots) {
|
for (Snapshot s : snapshots) {
|
||||||
// write the snapshot id
|
// write the snapshot id
|
||||||
out.writeInt(s.getId());
|
out.writeInt(s.getId());
|
||||||
}
|
}
|
||||||
// snapshot quota
|
// snapshot quota
|
||||||
out.writeInt(current.getSnapshotQuota());
|
out.writeInt(sf.getSnapshotQuota());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -216,14 +220,17 @@ public class SnapshotFSImageFormat {
|
||||||
* @param loader
|
* @param loader
|
||||||
* The loader
|
* The loader
|
||||||
*/
|
*/
|
||||||
public static void loadSnapshotList(
|
public static void loadSnapshotList(INodeDirectory snapshottableParent,
|
||||||
INodeDirectorySnapshottable snapshottableParent, int numSnapshots,
|
int numSnapshots, DataInput in, FSImageFormat.Loader loader)
|
||||||
DataInput in, FSImageFormat.Loader loader) throws IOException {
|
throws IOException {
|
||||||
|
DirectorySnapshottableFeature sf = snapshottableParent
|
||||||
|
.getDirectorySnapshottableFeature();
|
||||||
|
Preconditions.checkArgument(sf != null);
|
||||||
for (int i = 0; i < numSnapshots; i++) {
|
for (int i = 0; i < numSnapshots; i++) {
|
||||||
// read snapshots
|
// read snapshots
|
||||||
final Snapshot s = loader.getSnapshot(in);
|
final Snapshot s = loader.getSnapshot(in);
|
||||||
s.getRoot().setParent(snapshottableParent);
|
s.getRoot().setParent(snapshottableParent);
|
||||||
snapshottableParent.addSnapshot(s);
|
sf.addSnapshot(s);
|
||||||
}
|
}
|
||||||
int snapshotQuota = in.readInt();
|
int snapshotQuota = in.readInt();
|
||||||
snapshottableParent.setSnapshotQuota(snapshotQuota);
|
snapshottableParent.setSnapshotQuota(snapshotQuota);
|
||||||
|
|
|
@ -44,6 +44,8 @@ import org.apache.hadoop.hdfs.server.namenode.INodeDirectory;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.INodesInPath;
|
import org.apache.hadoop.hdfs.server.namenode.INodesInPath;
|
||||||
import org.apache.hadoop.metrics2.util.MBeans;
|
import org.apache.hadoop.metrics2.util.MBeans;
|
||||||
|
|
||||||
|
import com.google.common.base.Preconditions;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Manage snapshottable directories and their snapshots.
|
* Manage snapshottable directories and their snapshots.
|
||||||
*
|
*
|
||||||
|
@ -66,8 +68,8 @@ public class SnapshotManager implements SnapshotStatsMXBean {
|
||||||
private int snapshotCounter = 0;
|
private int snapshotCounter = 0;
|
||||||
|
|
||||||
/** All snapshottable directories in the namesystem. */
|
/** All snapshottable directories in the namesystem. */
|
||||||
private final Map<Long, INodeDirectorySnapshottable> snapshottables
|
private final Map<Long, INodeDirectory> snapshottables =
|
||||||
= new HashMap<Long, INodeDirectorySnapshottable>();
|
new HashMap<Long, INodeDirectory>();
|
||||||
|
|
||||||
public SnapshotManager(final FSDirectory fsdir) {
|
public SnapshotManager(final FSDirectory fsdir) {
|
||||||
this.fsdir = fsdir;
|
this.fsdir = fsdir;
|
||||||
|
@ -84,7 +86,7 @@ public class SnapshotManager implements SnapshotStatsMXBean {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
for(INodeDirectorySnapshottable s : snapshottables.values()) {
|
for(INodeDirectory s : snapshottables.values()) {
|
||||||
if (s.isAncestorDirectory(dir)) {
|
if (s.isAncestorDirectory(dir)) {
|
||||||
throw new SnapshotException(
|
throw new SnapshotException(
|
||||||
"Nested snapshottable directories not allowed: path=" + path
|
"Nested snapshottable directories not allowed: path=" + path
|
||||||
|
@ -112,33 +114,30 @@ public class SnapshotManager implements SnapshotStatsMXBean {
|
||||||
checkNestedSnapshottable(d, path);
|
checkNestedSnapshottable(d, path);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
final INodeDirectorySnapshottable s;
|
|
||||||
if (d.isSnapshottable()) {
|
if (d.isSnapshottable()) {
|
||||||
//The directory is already a snapshottable directory.
|
//The directory is already a snapshottable directory.
|
||||||
s = (INodeDirectorySnapshottable)d;
|
d.setSnapshotQuota(DirectorySnapshottableFeature.SNAPSHOT_LIMIT);
|
||||||
s.setSnapshotQuota(INodeDirectorySnapshottable.SNAPSHOT_LIMIT);
|
|
||||||
} else {
|
} else {
|
||||||
s = d.replaceSelf4INodeDirectorySnapshottable(iip.getLatestSnapshotId(),
|
d.addSnapshottableFeature();
|
||||||
fsdir.getINodeMap());
|
|
||||||
}
|
}
|
||||||
addSnapshottable(s);
|
addSnapshottable(d);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Add the given snapshottable directory to {@link #snapshottables}. */
|
/** Add the given snapshottable directory to {@link #snapshottables}. */
|
||||||
public void addSnapshottable(INodeDirectorySnapshottable dir) {
|
public void addSnapshottable(INodeDirectory dir) {
|
||||||
|
Preconditions.checkArgument(dir.isSnapshottable());
|
||||||
snapshottables.put(dir.getId(), dir);
|
snapshottables.put(dir.getId(), dir);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Remove the given snapshottable directory from {@link #snapshottables}. */
|
/** Remove the given snapshottable directory from {@link #snapshottables}. */
|
||||||
private void removeSnapshottable(INodeDirectorySnapshottable s) {
|
private void removeSnapshottable(INodeDirectory s) {
|
||||||
snapshottables.remove(s.getId());
|
snapshottables.remove(s.getId());
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Remove snapshottable directories from {@link #snapshottables} */
|
/** Remove snapshottable directories from {@link #snapshottables} */
|
||||||
public void removeSnapshottable(List<INodeDirectorySnapshottable> toRemove) {
|
public void removeSnapshottable(List<INodeDirectory> toRemove) {
|
||||||
if (toRemove != null) {
|
if (toRemove != null) {
|
||||||
for (INodeDirectorySnapshottable s : toRemove) {
|
for (INodeDirectory s : toRemove) {
|
||||||
removeSnapshottable(s);
|
removeSnapshottable(s);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -152,22 +151,22 @@ public class SnapshotManager implements SnapshotStatsMXBean {
|
||||||
public void resetSnapshottable(final String path) throws IOException {
|
public void resetSnapshottable(final String path) throws IOException {
|
||||||
final INodesInPath iip = fsdir.getINodesInPath4Write(path);
|
final INodesInPath iip = fsdir.getINodesInPath4Write(path);
|
||||||
final INodeDirectory d = INodeDirectory.valueOf(iip.getLastINode(), path);
|
final INodeDirectory d = INodeDirectory.valueOf(iip.getLastINode(), path);
|
||||||
if (!d.isSnapshottable()) {
|
DirectorySnapshottableFeature sf = d.getDirectorySnapshottableFeature();
|
||||||
|
if (sf == null) {
|
||||||
// the directory is already non-snapshottable
|
// the directory is already non-snapshottable
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
final INodeDirectorySnapshottable s = (INodeDirectorySnapshottable) d;
|
if (sf.getNumSnapshots() > 0) {
|
||||||
if (s.getNumSnapshots() > 0) {
|
|
||||||
throw new SnapshotException("The directory " + path + " has snapshot(s). "
|
throw new SnapshotException("The directory " + path + " has snapshot(s). "
|
||||||
+ "Please redo the operation after removing all the snapshots.");
|
+ "Please redo the operation after removing all the snapshots.");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (s == fsdir.getRoot()) {
|
if (d == fsdir.getRoot()) {
|
||||||
s.setSnapshotQuota(0);
|
d.setSnapshotQuota(0);
|
||||||
} else {
|
} else {
|
||||||
s.replaceSelf(iip.getLatestSnapshotId(), fsdir.getINodeMap());
|
d.removeSnapshottableFeature();
|
||||||
}
|
}
|
||||||
removeSnapshottable(s);
|
removeSnapshottable(d);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -180,10 +179,15 @@ public class SnapshotManager implements SnapshotStatsMXBean {
|
||||||
* Throw IOException when the given path does not lead to an
|
* Throw IOException when the given path does not lead to an
|
||||||
* existing snapshottable directory.
|
* existing snapshottable directory.
|
||||||
*/
|
*/
|
||||||
public INodeDirectorySnapshottable getSnapshottableRoot(final String path
|
public INodeDirectory getSnapshottableRoot(final String path)
|
||||||
) throws IOException {
|
throws IOException {
|
||||||
final INodesInPath i = fsdir.getINodesInPath4Write(path);
|
final INodeDirectory dir = INodeDirectory.valueOf(fsdir
|
||||||
return INodeDirectorySnapshottable.valueOf(i.getLastINode(), path);
|
.getINodesInPath4Write(path).getLastINode(), path);
|
||||||
|
if (!dir.isSnapshottable()) {
|
||||||
|
throw new SnapshotException(
|
||||||
|
"Directory is not a snapshottable directory: " + path);
|
||||||
|
}
|
||||||
|
return dir;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -202,7 +206,7 @@ public class SnapshotManager implements SnapshotStatsMXBean {
|
||||||
*/
|
*/
|
||||||
public String createSnapshot(final String path, String snapshotName
|
public String createSnapshot(final String path, String snapshotName
|
||||||
) throws IOException {
|
) throws IOException {
|
||||||
INodeDirectorySnapshottable srcRoot = getSnapshottableRoot(path);
|
INodeDirectory srcRoot = getSnapshottableRoot(path);
|
||||||
|
|
||||||
if (snapshotCounter == getMaxSnapshotID()) {
|
if (snapshotCounter == getMaxSnapshotID()) {
|
||||||
// We have reached the maximum allowable snapshot ID and since we don't
|
// We have reached the maximum allowable snapshot ID and since we don't
|
||||||
|
@ -235,7 +239,7 @@ public class SnapshotManager implements SnapshotStatsMXBean {
|
||||||
// parse the path, and check if the path is a snapshot path
|
// parse the path, and check if the path is a snapshot path
|
||||||
// the INodeDirectorySnapshottable#valueOf method will throw Exception
|
// the INodeDirectorySnapshottable#valueOf method will throw Exception
|
||||||
// if the path is not for a snapshottable directory
|
// if the path is not for a snapshottable directory
|
||||||
INodeDirectorySnapshottable srcRoot = getSnapshottableRoot(path);
|
INodeDirectory srcRoot = getSnapshottableRoot(path);
|
||||||
srcRoot.removeSnapshot(snapshotName, collectedBlocks, removedINodes);
|
srcRoot.removeSnapshot(snapshotName, collectedBlocks, removedINodes);
|
||||||
numSnapshots.getAndDecrement();
|
numSnapshots.getAndDecrement();
|
||||||
}
|
}
|
||||||
|
@ -258,8 +262,7 @@ public class SnapshotManager implements SnapshotStatsMXBean {
|
||||||
final String newSnapshotName) throws IOException {
|
final String newSnapshotName) throws IOException {
|
||||||
// Find the source root directory path where the snapshot was taken.
|
// Find the source root directory path where the snapshot was taken.
|
||||||
// All the check for path has been included in the valueOf method.
|
// All the check for path has been included in the valueOf method.
|
||||||
final INodeDirectorySnapshottable srcRoot
|
final INodeDirectory srcRoot = getSnapshottableRoot(path);
|
||||||
= INodeDirectorySnapshottable.valueOf(fsdir.getINode(path), path);
|
|
||||||
// Note that renameSnapshot and createSnapshot are synchronized externally
|
// Note that renameSnapshot and createSnapshot are synchronized externally
|
||||||
// through FSNamesystem's write lock
|
// through FSNamesystem's write lock
|
||||||
srcRoot.renameSnapshot(path, oldSnapshotName, newSnapshotName);
|
srcRoot.renameSnapshot(path, oldSnapshotName, newSnapshotName);
|
||||||
|
@ -285,9 +288,9 @@ public class SnapshotManager implements SnapshotStatsMXBean {
|
||||||
snapshotCounter = counter;
|
snapshotCounter = counter;
|
||||||
}
|
}
|
||||||
|
|
||||||
INodeDirectorySnapshottable[] getSnapshottableDirs() {
|
INodeDirectory[] getSnapshottableDirs() {
|
||||||
return snapshottables.values().toArray(
|
return snapshottables.values().toArray(
|
||||||
new INodeDirectorySnapshottable[snapshottables.size()]);
|
new INodeDirectory[snapshottables.size()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -299,8 +302,9 @@ public class SnapshotManager implements SnapshotStatsMXBean {
|
||||||
out.writeInt(numSnapshots.get());
|
out.writeInt(numSnapshots.get());
|
||||||
|
|
||||||
// write all snapshots.
|
// write all snapshots.
|
||||||
for(INodeDirectorySnapshottable snapshottableDir : snapshottables.values()) {
|
for(INodeDirectory snapshottableDir : snapshottables.values()) {
|
||||||
for(Snapshot s : snapshottableDir.getSnapshotsByNames()) {
|
for (Snapshot s : snapshottableDir.getDirectorySnapshottableFeature()
|
||||||
|
.getSnapshotList()) {
|
||||||
s.write(out);
|
s.write(out);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -339,16 +343,16 @@ public class SnapshotManager implements SnapshotStatsMXBean {
|
||||||
|
|
||||||
List<SnapshottableDirectoryStatus> statusList =
|
List<SnapshottableDirectoryStatus> statusList =
|
||||||
new ArrayList<SnapshottableDirectoryStatus>();
|
new ArrayList<SnapshottableDirectoryStatus>();
|
||||||
for (INodeDirectorySnapshottable dir : snapshottables.values()) {
|
for (INodeDirectory dir : snapshottables.values()) {
|
||||||
if (userName == null || userName.equals(dir.getUserName())) {
|
if (userName == null || userName.equals(dir.getUserName())) {
|
||||||
SnapshottableDirectoryStatus status = new SnapshottableDirectoryStatus(
|
SnapshottableDirectoryStatus status = new SnapshottableDirectoryStatus(
|
||||||
dir.getModificationTime(), dir.getAccessTime(),
|
dir.getModificationTime(), dir.getAccessTime(),
|
||||||
dir.getFsPermission(), dir.getUserName(), dir.getGroupName(),
|
dir.getFsPermission(), dir.getUserName(), dir.getGroupName(),
|
||||||
dir.getLocalNameBytes(), dir.getId(),
|
dir.getLocalNameBytes(), dir.getId(),
|
||||||
dir.getChildrenNum(Snapshot.CURRENT_STATE_ID),
|
dir.getChildrenNum(Snapshot.CURRENT_STATE_ID),
|
||||||
dir.getNumSnapshots(),
|
dir.getDirectorySnapshottableFeature().getNumSnapshots(),
|
||||||
dir.getSnapshotQuota(), dir.getParent() == null ?
|
dir.getDirectorySnapshottableFeature().getSnapshotQuota(),
|
||||||
DFSUtil.EMPTY_BYTES :
|
dir.getParent() == null ? DFSUtil.EMPTY_BYTES :
|
||||||
DFSUtil.string2Bytes(dir.getParent().getFullPathName()));
|
DFSUtil.string2Bytes(dir.getParent().getFullPathName()));
|
||||||
statusList.add(status);
|
statusList.add(status);
|
||||||
}
|
}
|
||||||
|
@ -364,20 +368,18 @@ public class SnapshotManager implements SnapshotStatsMXBean {
|
||||||
*/
|
*/
|
||||||
public SnapshotDiffReport diff(final String path, final String from,
|
public SnapshotDiffReport diff(final String path, final String from,
|
||||||
final String to) throws IOException {
|
final String to) throws IOException {
|
||||||
|
// Find the source root directory path where the snapshots were taken.
|
||||||
|
// All the check for path has been included in the valueOf method.
|
||||||
|
final INodeDirectory snapshotRoot = getSnapshottableRoot(path);
|
||||||
|
|
||||||
if ((from == null || from.isEmpty())
|
if ((from == null || from.isEmpty())
|
||||||
&& (to == null || to.isEmpty())) {
|
&& (to == null || to.isEmpty())) {
|
||||||
// both fromSnapshot and toSnapshot indicate the current tree
|
// both fromSnapshot and toSnapshot indicate the current tree
|
||||||
return new SnapshotDiffReport(path, from, to,
|
return new SnapshotDiffReport(path, from, to,
|
||||||
Collections.<DiffReportEntry> emptyList());
|
Collections.<DiffReportEntry> emptyList());
|
||||||
}
|
}
|
||||||
|
final SnapshotDiffInfo diffs = snapshotRoot
|
||||||
// Find the source root directory path where the snapshots were taken.
|
.getDirectorySnapshottableFeature().computeDiff(snapshotRoot, from, to);
|
||||||
// All the check for path has been included in the valueOf method.
|
|
||||||
INodesInPath inodesInPath = fsdir.getINodesInPath4Write(path.toString());
|
|
||||||
final INodeDirectorySnapshottable snapshotRoot = INodeDirectorySnapshottable
|
|
||||||
.valueOf(inodesInPath.getLastINode(), path);
|
|
||||||
|
|
||||||
final SnapshotDiffInfo diffs = snapshotRoot.computeDiff(from, to);
|
|
||||||
return diffs != null ? diffs.generateReport() : new SnapshotDiffReport(
|
return diffs != null ? diffs.generateReport() : new SnapshotDiffReport(
|
||||||
path, from, to, Collections.<DiffReportEntry> emptyList());
|
path, from, to, Collections.<DiffReportEntry> emptyList());
|
||||||
}
|
}
|
||||||
|
@ -412,7 +414,7 @@ public class SnapshotManager implements SnapshotStatsMXBean {
|
||||||
getSnapshottableDirectories() {
|
getSnapshottableDirectories() {
|
||||||
List<SnapshottableDirectoryStatus.Bean> beans =
|
List<SnapshottableDirectoryStatus.Bean> beans =
|
||||||
new ArrayList<SnapshottableDirectoryStatus.Bean>();
|
new ArrayList<SnapshottableDirectoryStatus.Bean>();
|
||||||
for (INodeDirectorySnapshottable d : getSnapshottableDirs()) {
|
for (INodeDirectory d : getSnapshottableDirs()) {
|
||||||
beans.add(toBean(d));
|
beans.add(toBean(d));
|
||||||
}
|
}
|
||||||
return beans.toArray(new SnapshottableDirectoryStatus.Bean[beans.size()]);
|
return beans.toArray(new SnapshottableDirectoryStatus.Bean[beans.size()]);
|
||||||
|
@ -421,20 +423,19 @@ public class SnapshotManager implements SnapshotStatsMXBean {
|
||||||
@Override // SnapshotStatsMXBean
|
@Override // SnapshotStatsMXBean
|
||||||
public SnapshotInfo.Bean[] getSnapshots() {
|
public SnapshotInfo.Bean[] getSnapshots() {
|
||||||
List<SnapshotInfo.Bean> beans = new ArrayList<SnapshotInfo.Bean>();
|
List<SnapshotInfo.Bean> beans = new ArrayList<SnapshotInfo.Bean>();
|
||||||
for (INodeDirectorySnapshottable d : getSnapshottableDirs()) {
|
for (INodeDirectory d : getSnapshottableDirs()) {
|
||||||
for (Snapshot s : d.getSnapshotList()) {
|
for (Snapshot s : d.getDirectorySnapshottableFeature().getSnapshotList()) {
|
||||||
beans.add(toBean(s));
|
beans.add(toBean(s));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return beans.toArray(new SnapshotInfo.Bean[beans.size()]);
|
return beans.toArray(new SnapshotInfo.Bean[beans.size()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static SnapshottableDirectoryStatus.Bean toBean(
|
public static SnapshottableDirectoryStatus.Bean toBean(INodeDirectory d) {
|
||||||
INodeDirectorySnapshottable d) {
|
|
||||||
return new SnapshottableDirectoryStatus.Bean(
|
return new SnapshottableDirectoryStatus.Bean(
|
||||||
d.getFullPathName(),
|
d.getFullPathName(),
|
||||||
d.getNumSnapshots(),
|
d.getDirectorySnapshottableFeature().getNumSnapshots(),
|
||||||
d.getSnapshotQuota(),
|
d.getDirectorySnapshottableFeature().getSnapshotQuota(),
|
||||||
d.getModificationTime(),
|
d.getModificationTime(),
|
||||||
Short.valueOf(Integer.toOctalString(
|
Short.valueOf(Integer.toOctalString(
|
||||||
d.getFsPermissionShort())),
|
d.getFsPermissionShort())),
|
||||||
|
|
|
@ -43,7 +43,6 @@ import org.apache.hadoop.hdfs.protocol.HdfsConstants.SafeModeAction;
|
||||||
import org.apache.hadoop.hdfs.protocol.SnapshottableDirectoryStatus;
|
import org.apache.hadoop.hdfs.protocol.SnapshottableDirectoryStatus;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.NNStorage.NameNodeFile;
|
import org.apache.hadoop.hdfs.server.namenode.NNStorage.NameNodeFile;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectoryWithSnapshotFeature.DirectoryDiff;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectoryWithSnapshotFeature.DirectoryDiff;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.INodeDirectorySnapshottable;
|
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.SnapshotTestHelper;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.SnapshotTestHelper;
|
||||||
import org.apache.hadoop.hdfs.util.Canceler;
|
import org.apache.hadoop.hdfs.util.Canceler;
|
||||||
|
@ -194,8 +193,8 @@ public class TestFSImageWithSnapshot {
|
||||||
fsn = cluster.getNamesystem();
|
fsn = cluster.getNamesystem();
|
||||||
hdfs = cluster.getFileSystem();
|
hdfs = cluster.getFileSystem();
|
||||||
|
|
||||||
INodeDirectorySnapshottable rootNode =
|
INodeDirectory rootNode = fsn.dir.getINode4Write(root.toString())
|
||||||
(INodeDirectorySnapshottable) fsn.dir.getINode4Write(root.toString());
|
.asDirectory();
|
||||||
assertTrue("The children list of root should be empty",
|
assertTrue("The children list of root should be empty",
|
||||||
rootNode.getChildrenList(Snapshot.CURRENT_STATE_ID).isEmpty());
|
rootNode.getChildrenList(Snapshot.CURRENT_STATE_ID).isEmpty());
|
||||||
// one snapshot on root: s1
|
// one snapshot on root: s1
|
||||||
|
|
|
@ -30,7 +30,6 @@ import org.apache.hadoop.hdfs.DFSTestUtil;
|
||||||
import org.apache.hadoop.hdfs.DFSUtil;
|
import org.apache.hadoop.hdfs.DFSUtil;
|
||||||
import org.apache.hadoop.hdfs.DistributedFileSystem;
|
import org.apache.hadoop.hdfs.DistributedFileSystem;
|
||||||
import org.apache.hadoop.hdfs.MiniDFSCluster;
|
import org.apache.hadoop.hdfs.MiniDFSCluster;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.INodeDirectorySnapshottable;
|
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot;
|
||||||
import org.junit.AfterClass;
|
import org.junit.AfterClass;
|
||||||
import org.junit.Assert;
|
import org.junit.Assert;
|
||||||
|
@ -90,22 +89,20 @@ public class TestSnapshotPathINodes {
|
||||||
final INode before = fsdir.getINode(pathStr);
|
final INode before = fsdir.getINode(pathStr);
|
||||||
|
|
||||||
// Before a directory is snapshottable
|
// Before a directory is snapshottable
|
||||||
Assert.assertTrue(before instanceof INodeDirectory);
|
Assert.assertFalse(before.asDirectory().isSnapshottable());
|
||||||
Assert.assertFalse(before instanceof INodeDirectorySnapshottable);
|
|
||||||
|
|
||||||
// After a directory is snapshottable
|
// After a directory is snapshottable
|
||||||
final Path path = new Path(pathStr);
|
final Path path = new Path(pathStr);
|
||||||
hdfs.allowSnapshot(path);
|
hdfs.allowSnapshot(path);
|
||||||
{
|
{
|
||||||
final INode after = fsdir.getINode(pathStr);
|
final INode after = fsdir.getINode(pathStr);
|
||||||
Assert.assertTrue(after instanceof INodeDirectorySnapshottable);
|
Assert.assertTrue(after.asDirectory().isSnapshottable());
|
||||||
}
|
}
|
||||||
|
|
||||||
hdfs.disallowSnapshot(path);
|
hdfs.disallowSnapshot(path);
|
||||||
{
|
{
|
||||||
final INode after = fsdir.getINode(pathStr);
|
final INode after = fsdir.getINode(pathStr);
|
||||||
Assert.assertTrue(after instanceof INodeDirectory);
|
Assert.assertFalse(after.asDirectory().isSnapshottable());
|
||||||
Assert.assertFalse(after instanceof INodeDirectorySnapshottable);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -115,8 +112,7 @@ public class TestSnapshotPathINodes {
|
||||||
}
|
}
|
||||||
final int i = inodesInPath.getSnapshotRootIndex() - 1;
|
final int i = inodesInPath.getSnapshotRootIndex() - 1;
|
||||||
final INode inode = inodesInPath.getINodes()[i];
|
final INode inode = inodesInPath.getINodes()[i];
|
||||||
return ((INodeDirectorySnapshottable)inode).getSnapshot(
|
return inode.asDirectory().getSnapshot(DFSUtil.string2Bytes(name));
|
||||||
DFSUtil.string2Bytes(name));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void assertSnapshot(INodesInPath inodesInPath, boolean isSnapshot,
|
static void assertSnapshot(INodesInPath inodesInPath, boolean isSnapshot,
|
||||||
|
|
|
@ -43,6 +43,7 @@ import org.apache.hadoop.hdfs.protocol.LocatedBlocks;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.FSDirectory;
|
import org.apache.hadoop.hdfs.server.namenode.FSDirectory;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.FSNamesystem;
|
import org.apache.hadoop.hdfs.server.namenode.FSNamesystem;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.INode;
|
import org.apache.hadoop.hdfs.server.namenode.INode;
|
||||||
|
import org.apache.hadoop.hdfs.server.namenode.INodeDirectory;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.INodeFile;
|
import org.apache.hadoop.hdfs.server.namenode.INodeFile;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectoryWithSnapshotFeature.DirectoryDiff;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectoryWithSnapshotFeature.DirectoryDiff;
|
||||||
import org.apache.log4j.Level;
|
import org.apache.log4j.Level;
|
||||||
|
@ -153,8 +154,7 @@ public class TestINodeFileUnderConstructionWithSnapshot {
|
||||||
// deleted list, with size BLOCKSIZE*2
|
// deleted list, with size BLOCKSIZE*2
|
||||||
INodeFile fileNode = (INodeFile) fsdir.getINode(file.toString());
|
INodeFile fileNode = (INodeFile) fsdir.getINode(file.toString());
|
||||||
assertEquals(BLOCKSIZE * 2, fileNode.computeFileSize());
|
assertEquals(BLOCKSIZE * 2, fileNode.computeFileSize());
|
||||||
INodeDirectorySnapshottable dirNode = (INodeDirectorySnapshottable) fsdir
|
INodeDirectory dirNode = fsdir.getINode(dir.toString()).asDirectory();
|
||||||
.getINode(dir.toString());
|
|
||||||
DirectoryDiff last = dirNode.getDiffs().getLast();
|
DirectoryDiff last = dirNode.getDiffs().getLast();
|
||||||
|
|
||||||
// 2. append without closing stream
|
// 2. append without closing stream
|
||||||
|
@ -162,7 +162,7 @@ public class TestINodeFileUnderConstructionWithSnapshot {
|
||||||
out.hsync(EnumSet.of(SyncFlag.UPDATE_LENGTH));
|
out.hsync(EnumSet.of(SyncFlag.UPDATE_LENGTH));
|
||||||
|
|
||||||
// re-check nodeInDeleted_S0
|
// re-check nodeInDeleted_S0
|
||||||
dirNode = (INodeDirectorySnapshottable) fsdir.getINode(dir.toString());
|
dirNode = fsdir.getINode(dir.toString()).asDirectory();
|
||||||
assertEquals(BLOCKSIZE * 2, fileNode.computeFileSize(last.getSnapshotId()));
|
assertEquals(BLOCKSIZE * 2, fileNode.computeFileSize(last.getSnapshotId()));
|
||||||
|
|
||||||
// 3. take snapshot --> close stream
|
// 3. take snapshot --> close stream
|
||||||
|
@ -172,7 +172,7 @@ public class TestINodeFileUnderConstructionWithSnapshot {
|
||||||
// check: an INodeFileUnderConstructionWithSnapshot with size BLOCKSIZE*3 should
|
// check: an INodeFileUnderConstructionWithSnapshot with size BLOCKSIZE*3 should
|
||||||
// have been stored in s1's deleted list
|
// have been stored in s1's deleted list
|
||||||
fileNode = (INodeFile) fsdir.getINode(file.toString());
|
fileNode = (INodeFile) fsdir.getINode(file.toString());
|
||||||
dirNode = (INodeDirectorySnapshottable) fsdir.getINode(dir.toString());
|
dirNode = fsdir.getINode(dir.toString()).asDirectory();
|
||||||
last = dirNode.getDiffs().getLast();
|
last = dirNode.getDiffs().getLast();
|
||||||
assertTrue(fileNode.isWithSnapshot());
|
assertTrue(fileNode.isWithSnapshot());
|
||||||
assertEquals(BLOCKSIZE * 3, fileNode.computeFileSize(last.getSnapshotId()));
|
assertEquals(BLOCKSIZE * 3, fileNode.computeFileSize(last.getSnapshotId()));
|
||||||
|
|
|
@ -17,7 +17,7 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.hadoop.hdfs.server.namenode.snapshot;
|
package org.apache.hadoop.hdfs.server.namenode.snapshot;
|
||||||
|
|
||||||
import static org.apache.hadoop.hdfs.server.namenode.snapshot.INodeDirectorySnapshottable.SNAPSHOT_LIMIT;
|
import static org.apache.hadoop.hdfs.server.namenode.snapshot.DirectorySnapshottableFeature.SNAPSHOT_LIMIT;
|
||||||
import static org.junit.Assert.assertTrue;
|
import static org.junit.Assert.assertTrue;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
@ -312,10 +312,9 @@ public class TestNestedSnapshots {
|
||||||
public void testIdCmp() {
|
public void testIdCmp() {
|
||||||
final PermissionStatus perm = PermissionStatus.createImmutable(
|
final PermissionStatus perm = PermissionStatus.createImmutable(
|
||||||
"user", "group", FsPermission.createImmutable((short)0));
|
"user", "group", FsPermission.createImmutable((short)0));
|
||||||
final INodeDirectory dir = new INodeDirectory(0,
|
final INodeDirectory snapshottable = new INodeDirectory(0,
|
||||||
DFSUtil.string2Bytes("foo"), perm, 0L);
|
DFSUtil.string2Bytes("foo"), perm, 0L);
|
||||||
final INodeDirectorySnapshottable snapshottable
|
snapshottable.addSnapshottableFeature();
|
||||||
= new INodeDirectorySnapshottable(dir);
|
|
||||||
final Snapshot[] snapshots = {
|
final Snapshot[] snapshots = {
|
||||||
new Snapshot(1, "s1", snapshottable),
|
new Snapshot(1, "s1", snapshottable),
|
||||||
new Snapshot(1, "s1", snapshottable),
|
new Snapshot(1, "s1", snapshottable),
|
||||||
|
@ -362,7 +361,7 @@ public class TestNestedSnapshots {
|
||||||
|
|
||||||
hdfs.allowSnapshot(sub);
|
hdfs.allowSnapshot(sub);
|
||||||
subNode = fsdir.getINode(sub.toString());
|
subNode = fsdir.getINode(sub.toString());
|
||||||
assertTrue(subNode instanceof INodeDirectorySnapshottable);
|
assertTrue(subNode.isDirectory() && subNode.asDirectory().isSnapshottable());
|
||||||
|
|
||||||
hdfs.disallowSnapshot(sub);
|
hdfs.disallowSnapshot(sub);
|
||||||
subNode = fsdir.getINode(sub.toString());
|
subNode = fsdir.getINode(sub.toString());
|
||||||
|
|
|
@ -402,8 +402,7 @@ public class TestRenameWithSnapshots {
|
||||||
final Path foo_s3 = SnapshotTestHelper.getSnapshotPath(sdir1, "s3",
|
final Path foo_s3 = SnapshotTestHelper.getSnapshotPath(sdir1, "s3",
|
||||||
"foo");
|
"foo");
|
||||||
assertFalse(hdfs.exists(foo_s3));
|
assertFalse(hdfs.exists(foo_s3));
|
||||||
INodeDirectorySnapshottable sdir2Node =
|
INodeDirectory sdir2Node = fsdir.getINode(sdir2.toString()).asDirectory();
|
||||||
(INodeDirectorySnapshottable) fsdir.getINode(sdir2.toString());
|
|
||||||
Snapshot s2 = sdir2Node.getSnapshot(DFSUtil.string2Bytes("s2"));
|
Snapshot s2 = sdir2Node.getSnapshot(DFSUtil.string2Bytes("s2"));
|
||||||
INodeFile sfoo = fsdir.getINode(newfoo.toString()).asFile();
|
INodeFile sfoo = fsdir.getINode(newfoo.toString()).asFile();
|
||||||
assertEquals(s2.getId(), sfoo.getDiffs().getLastSnapshotId());
|
assertEquals(s2.getId(), sfoo.getDiffs().getLastSnapshotId());
|
||||||
|
@ -606,8 +605,7 @@ public class TestRenameWithSnapshots {
|
||||||
|
|
||||||
INodeFile snode = fsdir.getINode(newfoo.toString()).asFile();
|
INodeFile snode = fsdir.getINode(newfoo.toString()).asFile();
|
||||||
assertEquals(1, snode.getDiffs().asList().size());
|
assertEquals(1, snode.getDiffs().asList().size());
|
||||||
INodeDirectorySnapshottable sdir2Node =
|
INodeDirectory sdir2Node = fsdir.getINode(sdir2.toString()).asDirectory();
|
||||||
(INodeDirectorySnapshottable) fsdir.getINode(sdir2.toString());
|
|
||||||
Snapshot s2 = sdir2Node.getSnapshot(DFSUtil.string2Bytes("s2"));
|
Snapshot s2 = sdir2Node.getSnapshot(DFSUtil.string2Bytes("s2"));
|
||||||
assertEquals(s2.getId(), snode.getDiffs().getLastSnapshotId());
|
assertEquals(s2.getId(), snode.getDiffs().getLastSnapshotId());
|
||||||
|
|
||||||
|
@ -762,8 +760,7 @@ public class TestRenameWithSnapshots {
|
||||||
assertEquals(2, fooWithCount.getReferenceCount());
|
assertEquals(2, fooWithCount.getReferenceCount());
|
||||||
INodeDirectory foo = fooWithCount.asDirectory();
|
INodeDirectory foo = fooWithCount.asDirectory();
|
||||||
assertEquals(1, foo.getDiffs().asList().size());
|
assertEquals(1, foo.getDiffs().asList().size());
|
||||||
INodeDirectorySnapshottable sdir1Node =
|
INodeDirectory sdir1Node = fsdir.getINode(sdir1.toString()).asDirectory();
|
||||||
(INodeDirectorySnapshottable) fsdir.getINode(sdir1.toString());
|
|
||||||
Snapshot s1 = sdir1Node.getSnapshot(DFSUtil.string2Bytes("s1"));
|
Snapshot s1 = sdir1Node.getSnapshot(DFSUtil.string2Bytes("s1"));
|
||||||
assertEquals(s1.getId(), foo.getDirectoryWithSnapshotFeature()
|
assertEquals(s1.getId(), foo.getDirectoryWithSnapshotFeature()
|
||||||
.getLastSnapshotId());
|
.getLastSnapshotId());
|
||||||
|
@ -972,12 +969,9 @@ public class TestRenameWithSnapshots {
|
||||||
hdfs.rename(bar_dir2, bar_dir1);
|
hdfs.rename(bar_dir2, bar_dir1);
|
||||||
|
|
||||||
// check the internal details
|
// check the internal details
|
||||||
INodeDirectorySnapshottable sdir1Node =
|
INodeDirectory sdir1Node = fsdir.getINode(sdir1.toString()).asDirectory();
|
||||||
(INodeDirectorySnapshottable) fsdir.getINode(sdir1.toString());
|
INodeDirectory sdir2Node = fsdir.getINode(sdir2.toString()).asDirectory();
|
||||||
INodeDirectorySnapshottable sdir2Node =
|
INodeDirectory sdir3Node = fsdir.getINode(sdir3.toString()).asDirectory();
|
||||||
(INodeDirectorySnapshottable) fsdir.getINode(sdir2.toString());
|
|
||||||
INodeDirectorySnapshottable sdir3Node =
|
|
||||||
(INodeDirectorySnapshottable) fsdir.getINode(sdir3.toString());
|
|
||||||
|
|
||||||
INodeReference fooRef = fsdir.getINode4Write(foo_dir1.toString())
|
INodeReference fooRef = fsdir.getINode4Write(foo_dir1.toString())
|
||||||
.asReference();
|
.asReference();
|
||||||
|
@ -1182,8 +1176,7 @@ public class TestRenameWithSnapshots {
|
||||||
assertTrue(hdfs.exists(bar_s2));
|
assertTrue(hdfs.exists(bar_s2));
|
||||||
|
|
||||||
// check internal details
|
// check internal details
|
||||||
INodeDirectorySnapshottable sdir2Node =
|
INodeDirectory sdir2Node = fsdir.getINode(sdir2.toString()).asDirectory();
|
||||||
(INodeDirectorySnapshottable) fsdir.getINode(sdir2.toString());
|
|
||||||
Snapshot s2 = sdir2Node.getSnapshot(DFSUtil.string2Bytes("s2"));
|
Snapshot s2 = sdir2Node.getSnapshot(DFSUtil.string2Bytes("s2"));
|
||||||
final Path foo_s2 = SnapshotTestHelper.getSnapshotPath(sdir2, "s2", "foo");
|
final Path foo_s2 = SnapshotTestHelper.getSnapshotPath(sdir2, "s2", "foo");
|
||||||
INodeReference fooRef = fsdir.getINode(foo_s2.toString()).asReference();
|
INodeReference fooRef = fsdir.getINode(foo_s2.toString()).asReference();
|
||||||
|
@ -1290,8 +1283,8 @@ public class TestRenameWithSnapshots {
|
||||||
assertFalse(result);
|
assertFalse(result);
|
||||||
|
|
||||||
// check the current internal details
|
// check the current internal details
|
||||||
INodeDirectorySnapshottable dir1Node = (INodeDirectorySnapshottable) fsdir
|
INodeDirectory dir1Node = fsdir.getINode4Write(sdir1.toString())
|
||||||
.getINode4Write(sdir1.toString());
|
.asDirectory();
|
||||||
Snapshot s1 = dir1Node.getSnapshot(DFSUtil.string2Bytes("s1"));
|
Snapshot s1 = dir1Node.getSnapshot(DFSUtil.string2Bytes("s1"));
|
||||||
ReadOnlyList<INode> dir1Children = dir1Node
|
ReadOnlyList<INode> dir1Children = dir1Node
|
||||||
.getChildrenList(Snapshot.CURRENT_STATE_ID);
|
.getChildrenList(Snapshot.CURRENT_STATE_ID);
|
||||||
|
@ -1360,8 +1353,8 @@ public class TestRenameWithSnapshots {
|
||||||
assertFalse(result);
|
assertFalse(result);
|
||||||
|
|
||||||
// check the current internal details
|
// check the current internal details
|
||||||
INodeDirectorySnapshottable dir1Node = (INodeDirectorySnapshottable) fsdir
|
INodeDirectory dir1Node = fsdir.getINode4Write(sdir1.toString())
|
||||||
.getINode4Write(sdir1.toString());
|
.asDirectory();
|
||||||
Snapshot s1 = dir1Node.getSnapshot(DFSUtil.string2Bytes("s1"));
|
Snapshot s1 = dir1Node.getSnapshot(DFSUtil.string2Bytes("s1"));
|
||||||
ReadOnlyList<INode> dir1Children = dir1Node
|
ReadOnlyList<INode> dir1Children = dir1Node
|
||||||
.getChildrenList(Snapshot.CURRENT_STATE_ID);
|
.getChildrenList(Snapshot.CURRENT_STATE_ID);
|
||||||
|
@ -1427,11 +1420,11 @@ public class TestRenameWithSnapshots {
|
||||||
assertFalse(result);
|
assertFalse(result);
|
||||||
|
|
||||||
// check the current internal details
|
// check the current internal details
|
||||||
INodeDirectorySnapshottable dir1Node = (INodeDirectorySnapshottable) fsdir
|
INodeDirectory dir1Node = fsdir.getINode4Write(sdir1.toString())
|
||||||
.getINode4Write(sdir1.toString());
|
.asDirectory();
|
||||||
Snapshot s1 = dir1Node.getSnapshot(DFSUtil.string2Bytes("s1"));
|
Snapshot s1 = dir1Node.getSnapshot(DFSUtil.string2Bytes("s1"));
|
||||||
INodeDirectorySnapshottable dir2Node = (INodeDirectorySnapshottable) fsdir
|
INodeDirectory dir2Node = fsdir.getINode4Write(sdir2.toString())
|
||||||
.getINode4Write(sdir2.toString());
|
.asDirectory();
|
||||||
Snapshot s2 = dir2Node.getSnapshot(DFSUtil.string2Bytes("s2"));
|
Snapshot s2 = dir2Node.getSnapshot(DFSUtil.string2Bytes("s2"));
|
||||||
ReadOnlyList<INode> dir2Children = dir2Node
|
ReadOnlyList<INode> dir2Children = dir2Node
|
||||||
.getChildrenList(Snapshot.CURRENT_STATE_ID);
|
.getChildrenList(Snapshot.CURRENT_STATE_ID);
|
||||||
|
@ -1458,8 +1451,7 @@ public class TestRenameWithSnapshots {
|
||||||
assertFalse(result);
|
assertFalse(result);
|
||||||
|
|
||||||
// check internal details again
|
// check internal details again
|
||||||
dir2Node = (INodeDirectorySnapshottable) fsdir.getINode4Write(sdir2
|
dir2Node = fsdir.getINode4Write(sdir2.toString()).asDirectory();
|
||||||
.toString());
|
|
||||||
Snapshot s3 = dir2Node.getSnapshot(DFSUtil.string2Bytes("s3"));
|
Snapshot s3 = dir2Node.getSnapshot(DFSUtil.string2Bytes("s3"));
|
||||||
fooNode = fsdir.getINode4Write(foo_dir2.toString());
|
fooNode = fsdir.getINode4Write(foo_dir2.toString());
|
||||||
dir2Children = dir2Node.getChildrenList(Snapshot.CURRENT_STATE_ID);
|
dir2Children = dir2Node.getChildrenList(Snapshot.CURRENT_STATE_ID);
|
||||||
|
@ -1599,8 +1591,8 @@ public class TestRenameWithSnapshots {
|
||||||
assertTrue(diff.getChildrenDiff().getList(ListType.DELETED).isEmpty());
|
assertTrue(diff.getChildrenDiff().getList(ListType.DELETED).isEmpty());
|
||||||
|
|
||||||
// check dir2
|
// check dir2
|
||||||
INode dir2Node = fsdir.getINode4Write(dir2.toString());
|
INodeDirectory dir2Node = fsdir.getINode4Write(dir2.toString()).asDirectory();
|
||||||
assertTrue(dir2Node.getClass() == INodeDirectorySnapshottable.class);
|
assertTrue(dir2Node.isSnapshottable());
|
||||||
Quota.Counts counts = dir2Node.computeQuotaUsage();
|
Quota.Counts counts = dir2Node.computeQuotaUsage();
|
||||||
assertEquals(3, counts.get(Quota.NAMESPACE));
|
assertEquals(3, counts.get(Quota.NAMESPACE));
|
||||||
assertEquals(0, counts.get(Quota.DISKSPACE));
|
assertEquals(0, counts.get(Quota.DISKSPACE));
|
||||||
|
@ -1610,8 +1602,7 @@ public class TestRenameWithSnapshots {
|
||||||
INode subdir2Node = childrenList.get(0);
|
INode subdir2Node = childrenList.get(0);
|
||||||
assertSame(dir2Node, subdir2Node.getParent());
|
assertSame(dir2Node, subdir2Node.getParent());
|
||||||
assertSame(subdir2Node, fsdir.getINode4Write(subdir2.toString()));
|
assertSame(subdir2Node, fsdir.getINode4Write(subdir2.toString()));
|
||||||
diffList = ((INodeDirectorySnapshottable) dir2Node)
|
diffList = dir2Node.getDiffs().asList();
|
||||||
.getDiffs().asList();
|
|
||||||
assertEquals(1, diffList.size());
|
assertEquals(1, diffList.size());
|
||||||
diff = diffList.get(0);
|
diff = diffList.get(0);
|
||||||
assertTrue(diff.getChildrenDiff().getList(ListType.CREATED).isEmpty());
|
assertTrue(diff.getChildrenDiff().getList(ListType.CREATED).isEmpty());
|
||||||
|
@ -1673,8 +1664,8 @@ public class TestRenameWithSnapshots {
|
||||||
assertTrue(diff.getChildrenDiff().getList(ListType.DELETED).isEmpty());
|
assertTrue(diff.getChildrenDiff().getList(ListType.DELETED).isEmpty());
|
||||||
|
|
||||||
// check dir2
|
// check dir2
|
||||||
INode dir2Node = fsdir.getINode4Write(dir2.toString());
|
INodeDirectory dir2Node = fsdir.getINode4Write(dir2.toString()).asDirectory();
|
||||||
assertTrue(dir2Node.getClass() == INodeDirectorySnapshottable.class);
|
assertTrue(dir2Node.isSnapshottable());
|
||||||
Quota.Counts counts = dir2Node.computeQuotaUsage();
|
Quota.Counts counts = dir2Node.computeQuotaUsage();
|
||||||
assertEquals(4, counts.get(Quota.NAMESPACE));
|
assertEquals(4, counts.get(Quota.NAMESPACE));
|
||||||
assertEquals(0, counts.get(Quota.DISKSPACE));
|
assertEquals(0, counts.get(Quota.DISKSPACE));
|
||||||
|
@ -1689,7 +1680,7 @@ public class TestRenameWithSnapshots {
|
||||||
assertTrue(subsubdir2Node.getClass() == INodeDirectory.class);
|
assertTrue(subsubdir2Node.getClass() == INodeDirectory.class);
|
||||||
assertSame(subdir2Node, subsubdir2Node.getParent());
|
assertSame(subdir2Node, subsubdir2Node.getParent());
|
||||||
|
|
||||||
diffList = ((INodeDirectorySnapshottable) dir2Node).getDiffs().asList();
|
diffList = ( dir2Node).getDiffs().asList();
|
||||||
assertEquals(1, diffList.size());
|
assertEquals(1, diffList.size());
|
||||||
diff = diffList.get(0);
|
diff = diffList.get(0);
|
||||||
assertTrue(diff.getChildrenDiff().getList(ListType.CREATED).isEmpty());
|
assertTrue(diff.getChildrenDiff().getList(ListType.CREATED).isEmpty());
|
||||||
|
@ -1723,8 +1714,8 @@ public class TestRenameWithSnapshots {
|
||||||
}
|
}
|
||||||
|
|
||||||
// check
|
// check
|
||||||
INodeDirectorySnapshottable rootNode = (INodeDirectorySnapshottable) fsdir
|
INodeDirectory rootNode = fsdir.getINode4Write(root.toString())
|
||||||
.getINode4Write(root.toString());
|
.asDirectory();
|
||||||
INodeDirectory fooNode = fsdir.getINode4Write(foo.toString()).asDirectory();
|
INodeDirectory fooNode = fsdir.getINode4Write(foo.toString()).asDirectory();
|
||||||
ReadOnlyList<INode> children = fooNode
|
ReadOnlyList<INode> children = fooNode
|
||||||
.getChildrenList(Snapshot.CURRENT_STATE_ID);
|
.getChildrenList(Snapshot.CURRENT_STATE_ID);
|
||||||
|
@ -1794,7 +1785,7 @@ public class TestRenameWithSnapshots {
|
||||||
|
|
||||||
// check dir2
|
// check dir2
|
||||||
INode dir2Node = fsdir.getINode4Write(dir2.toString());
|
INode dir2Node = fsdir.getINode4Write(dir2.toString());
|
||||||
assertTrue(dir2Node.getClass() == INodeDirectorySnapshottable.class);
|
assertTrue(dir2Node.asDirectory().isSnapshottable());
|
||||||
Quota.Counts counts = dir2Node.computeQuotaUsage();
|
Quota.Counts counts = dir2Node.computeQuotaUsage();
|
||||||
assertEquals(7, counts.get(Quota.NAMESPACE));
|
assertEquals(7, counts.get(Quota.NAMESPACE));
|
||||||
assertEquals(BLOCKSIZE * REPL * 2, counts.get(Quota.DISKSPACE));
|
assertEquals(BLOCKSIZE * REPL * 2, counts.get(Quota.DISKSPACE));
|
||||||
|
@ -1961,12 +1952,12 @@ public class TestRenameWithSnapshots {
|
||||||
hdfs.deleteSnapshot(sdir2, "s3");
|
hdfs.deleteSnapshot(sdir2, "s3");
|
||||||
|
|
||||||
// check
|
// check
|
||||||
final INodeDirectorySnapshottable dir1Node =
|
final INodeDirectory dir1Node = fsdir.getINode4Write(sdir1.toString())
|
||||||
(INodeDirectorySnapshottable) fsdir.getINode4Write(sdir1.toString());
|
.asDirectory();
|
||||||
Quota.Counts q1 = dir1Node.getDirectoryWithQuotaFeature().getSpaceConsumed();
|
Quota.Counts q1 = dir1Node.getDirectoryWithQuotaFeature().getSpaceConsumed();
|
||||||
assertEquals(4, q1.get(Quota.NAMESPACE));
|
assertEquals(4, q1.get(Quota.NAMESPACE));
|
||||||
final INodeDirectorySnapshottable dir2Node =
|
final INodeDirectory dir2Node = fsdir.getINode4Write(sdir2.toString())
|
||||||
(INodeDirectorySnapshottable) fsdir.getINode4Write(sdir2.toString());
|
.asDirectory();
|
||||||
Quota.Counts q2 = dir2Node.getDirectoryWithQuotaFeature().getSpaceConsumed();
|
Quota.Counts q2 = dir2Node.getDirectoryWithQuotaFeature().getSpaceConsumed();
|
||||||
assertEquals(2, q2.get(Quota.NAMESPACE));
|
assertEquals(2, q2.get(Quota.NAMESPACE));
|
||||||
|
|
||||||
|
@ -2030,13 +2021,13 @@ public class TestRenameWithSnapshots {
|
||||||
hdfs.deleteSnapshot(sdir2, "s3");
|
hdfs.deleteSnapshot(sdir2, "s3");
|
||||||
|
|
||||||
// check
|
// check
|
||||||
final INodeDirectorySnapshottable dir1Node =
|
final INodeDirectory dir1Node = fsdir.getINode4Write(sdir1.toString())
|
||||||
(INodeDirectorySnapshottable) fsdir.getINode4Write(sdir1.toString());
|
.asDirectory();
|
||||||
// sdir1 + s1 + foo_s1 (foo) + foo (foo + s1 + bar~bar3)
|
// sdir1 + s1 + foo_s1 (foo) + foo (foo + s1 + bar~bar3)
|
||||||
Quota.Counts q1 = dir1Node.getDirectoryWithQuotaFeature().getSpaceConsumed();
|
Quota.Counts q1 = dir1Node.getDirectoryWithQuotaFeature().getSpaceConsumed();
|
||||||
assertEquals(9, q1.get(Quota.NAMESPACE));
|
assertEquals(9, q1.get(Quota.NAMESPACE));
|
||||||
final INodeDirectorySnapshottable dir2Node =
|
final INodeDirectory dir2Node = fsdir.getINode4Write(sdir2.toString())
|
||||||
(INodeDirectorySnapshottable) fsdir.getINode4Write(sdir2.toString());
|
.asDirectory();
|
||||||
Quota.Counts q2 = dir2Node.getDirectoryWithQuotaFeature().getSpaceConsumed();
|
Quota.Counts q2 = dir2Node.getDirectoryWithQuotaFeature().getSpaceConsumed();
|
||||||
assertEquals(2, q2.get(Quota.NAMESPACE));
|
assertEquals(2, q2.get(Quota.NAMESPACE));
|
||||||
|
|
||||||
|
@ -2252,8 +2243,8 @@ public class TestRenameWithSnapshots {
|
||||||
List<DirectoryDiff> barDiffList = barNode.getDiffs().asList();
|
List<DirectoryDiff> barDiffList = barNode.getDiffs().asList();
|
||||||
assertEquals(1, barDiffList.size());
|
assertEquals(1, barDiffList.size());
|
||||||
DirectoryDiff diff = barDiffList.get(0);
|
DirectoryDiff diff = barDiffList.get(0);
|
||||||
INodeDirectorySnapshottable testNode =
|
INodeDirectory testNode = fsdir.getINode4Write(test.toString())
|
||||||
(INodeDirectorySnapshottable) fsdir.getINode4Write(test.toString());
|
.asDirectory();
|
||||||
Snapshot s0 = testNode.getSnapshot(DFSUtil.string2Bytes("s0"));
|
Snapshot s0 = testNode.getSnapshot(DFSUtil.string2Bytes("s0"));
|
||||||
assertEquals(s0.getId(), diff.getSnapshotId());
|
assertEquals(s0.getId(), diff.getSnapshotId());
|
||||||
// and file should be stored in the deleted list of this snapshot diff
|
// and file should be stored in the deleted list of this snapshot diff
|
||||||
|
@ -2265,14 +2256,10 @@ public class TestRenameWithSnapshots {
|
||||||
INodeDirectory dir2Node = fsdir.getINode4Write(dir2.toString())
|
INodeDirectory dir2Node = fsdir.getINode4Write(dir2.toString())
|
||||||
.asDirectory();
|
.asDirectory();
|
||||||
List<DirectoryDiff> dir2DiffList = dir2Node.getDiffs().asList();
|
List<DirectoryDiff> dir2DiffList = dir2Node.getDiffs().asList();
|
||||||
// dir2Node should contain 2 snapshot diffs, one for s2, and the other was
|
// dir2Node should contain 1 snapshot diffs for s2
|
||||||
// originally s1 (created when dir2 was transformed to a snapshottable dir),
|
assertEquals(1, dir2DiffList.size());
|
||||||
// and currently is s0
|
dList = dir2DiffList.get(0).getChildrenDiff().getList(ListType.DELETED);
|
||||||
assertEquals(2, dir2DiffList.size());
|
|
||||||
dList = dir2DiffList.get(1).getChildrenDiff().getList(ListType.DELETED);
|
|
||||||
assertEquals(1, dList.size());
|
assertEquals(1, dList.size());
|
||||||
cList = dir2DiffList.get(0).getChildrenDiff().getList(ListType.CREATED);
|
|
||||||
assertTrue(cList.isEmpty());
|
|
||||||
final Path foo_s2 = SnapshotTestHelper.getSnapshotPath(dir2, "s2",
|
final Path foo_s2 = SnapshotTestHelper.getSnapshotPath(dir2, "s2",
|
||||||
foo.getName());
|
foo.getName());
|
||||||
INodeReference.WithName fooNode_s2 =
|
INodeReference.WithName fooNode_s2 =
|
||||||
|
|
|
@ -112,23 +112,20 @@ public class TestSetQuotaWithSnapshot {
|
||||||
hdfs.allowSnapshot(dir);
|
hdfs.allowSnapshot(dir);
|
||||||
hdfs.setQuota(dir, HdfsConstants.QUOTA_DONT_SET,
|
hdfs.setQuota(dir, HdfsConstants.QUOTA_DONT_SET,
|
||||||
HdfsConstants.QUOTA_DONT_SET);
|
HdfsConstants.QUOTA_DONT_SET);
|
||||||
INode dirNode = fsdir.getINode4Write(dir.toString());
|
INodeDirectory dirNode = fsdir.getINode4Write(dir.toString()).asDirectory();
|
||||||
assertTrue(dirNode instanceof INodeDirectorySnapshottable);
|
assertTrue(dirNode.isSnapshottable());
|
||||||
assertEquals(0, ((INodeDirectorySnapshottable) dirNode).getDiffs().asList()
|
assertEquals(0, dirNode.getDiffs().asList().size());
|
||||||
.size());
|
|
||||||
|
|
||||||
hdfs.setQuota(dir, HdfsConstants.QUOTA_DONT_SET - 1,
|
hdfs.setQuota(dir, HdfsConstants.QUOTA_DONT_SET - 1,
|
||||||
HdfsConstants.QUOTA_DONT_SET - 1);
|
HdfsConstants.QUOTA_DONT_SET - 1);
|
||||||
dirNode = fsdir.getINode4Write(dir.toString());
|
dirNode = fsdir.getINode4Write(dir.toString()).asDirectory();
|
||||||
assertTrue(dirNode instanceof INodeDirectorySnapshottable);
|
assertTrue(dirNode.isSnapshottable());
|
||||||
assertEquals(0, ((INodeDirectorySnapshottable) dirNode).getDiffs().asList()
|
assertEquals(0, dirNode.getDiffs().asList().size());
|
||||||
.size());
|
|
||||||
|
|
||||||
hdfs.setQuota(dir, HdfsConstants.QUOTA_RESET, HdfsConstants.QUOTA_RESET);
|
hdfs.setQuota(dir, HdfsConstants.QUOTA_RESET, HdfsConstants.QUOTA_RESET);
|
||||||
dirNode = fsdir.getINode4Write(dir.toString());
|
dirNode = fsdir.getINode4Write(dir.toString()).asDirectory();
|
||||||
assertTrue(dirNode instanceof INodeDirectorySnapshottable);
|
assertTrue(dirNode.isSnapshottable());
|
||||||
assertEquals(0, ((INodeDirectorySnapshottable) dirNode).getDiffs().asList()
|
assertEquals(0, dirNode.getDiffs().asList().size());
|
||||||
.size());
|
|
||||||
|
|
||||||
// allow snapshot on dir and create snapshot s1
|
// allow snapshot on dir and create snapshot s1
|
||||||
SnapshotTestHelper.createSnapshot(hdfs, dir, "s1");
|
SnapshotTestHelper.createSnapshot(hdfs, dir, "s1");
|
||||||
|
@ -136,10 +133,9 @@ public class TestSetQuotaWithSnapshot {
|
||||||
// clear quota of dir
|
// clear quota of dir
|
||||||
hdfs.setQuota(dir, HdfsConstants.QUOTA_RESET, HdfsConstants.QUOTA_RESET);
|
hdfs.setQuota(dir, HdfsConstants.QUOTA_RESET, HdfsConstants.QUOTA_RESET);
|
||||||
// dir should still be a snapshottable directory
|
// dir should still be a snapshottable directory
|
||||||
dirNode = fsdir.getINode4Write(dir.toString());
|
dirNode = fsdir.getINode4Write(dir.toString()).asDirectory();
|
||||||
assertTrue(dirNode instanceof INodeDirectorySnapshottable);
|
assertTrue(dirNode.isSnapshottable());
|
||||||
assertEquals(1, ((INodeDirectorySnapshottable) dirNode).getDiffs().asList()
|
assertEquals(1, dirNode.getDiffs().asList().size());
|
||||||
.size());
|
|
||||||
SnapshottableDirectoryStatus[] status = hdfs.getSnapshottableDirListing();
|
SnapshottableDirectoryStatus[] status = hdfs.getSnapshottableDirListing();
|
||||||
assertEquals(1, status.length);
|
assertEquals(1, status.length);
|
||||||
assertEquals(dir, status[0].getFullPath());
|
assertEquals(dir, status[0].getFullPath());
|
||||||
|
@ -154,8 +150,7 @@ public class TestSetQuotaWithSnapshot {
|
||||||
assertTrue(subNode.asDirectory().isWithSnapshot());
|
assertTrue(subNode.asDirectory().isWithSnapshot());
|
||||||
List<DirectoryDiff> diffList = subNode.asDirectory().getDiffs().asList();
|
List<DirectoryDiff> diffList = subNode.asDirectory().getDiffs().asList();
|
||||||
assertEquals(1, diffList.size());
|
assertEquals(1, diffList.size());
|
||||||
Snapshot s2 = ((INodeDirectorySnapshottable) dirNode).getSnapshot(DFSUtil
|
Snapshot s2 = dirNode.getSnapshot(DFSUtil.string2Bytes("s2"));
|
||||||
.string2Bytes("s2"));
|
|
||||||
assertEquals(s2.getId(), diffList.get(0).getSnapshotId());
|
assertEquals(s2.getId(), diffList.get(0).getSnapshotId());
|
||||||
List<INode> createdList = diffList.get(0).getChildrenDiff().getList(ListType.CREATED);
|
List<INode> createdList = diffList.get(0).getChildrenDiff().getList(ListType.CREATED);
|
||||||
assertEquals(1, createdList.size());
|
assertEquals(1, createdList.size());
|
||||||
|
|
|
@ -430,30 +430,31 @@ public class TestSnapshot {
|
||||||
.asDirectory();
|
.asDirectory();
|
||||||
assertTrue(rootNode.isSnapshottable());
|
assertTrue(rootNode.isSnapshottable());
|
||||||
// root is snapshottable dir, but with 0 snapshot quota
|
// root is snapshottable dir, but with 0 snapshot quota
|
||||||
assertEquals(0, ((INodeDirectorySnapshottable) rootNode).getSnapshotQuota());
|
assertEquals(0, rootNode.getDirectorySnapshottableFeature()
|
||||||
|
.getSnapshotQuota());
|
||||||
|
|
||||||
hdfs.allowSnapshot(root);
|
hdfs.allowSnapshot(root);
|
||||||
rootNode = fsdir.getINode4Write(root.toString()).asDirectory();
|
rootNode = fsdir.getINode4Write(root.toString()).asDirectory();
|
||||||
assertTrue(rootNode.isSnapshottable());
|
assertTrue(rootNode.isSnapshottable());
|
||||||
assertEquals(INodeDirectorySnapshottable.SNAPSHOT_LIMIT,
|
assertEquals(DirectorySnapshottableFeature.SNAPSHOT_LIMIT,
|
||||||
((INodeDirectorySnapshottable) rootNode).getSnapshotQuota());
|
rootNode.getDirectorySnapshottableFeature().getSnapshotQuota());
|
||||||
// call allowSnapshot again
|
// call allowSnapshot again
|
||||||
hdfs.allowSnapshot(root);
|
hdfs.allowSnapshot(root);
|
||||||
rootNode = fsdir.getINode4Write(root.toString()).asDirectory();
|
rootNode = fsdir.getINode4Write(root.toString()).asDirectory();
|
||||||
assertTrue(rootNode.isSnapshottable());
|
assertTrue(rootNode.isSnapshottable());
|
||||||
assertEquals(INodeDirectorySnapshottable.SNAPSHOT_LIMIT,
|
assertEquals(DirectorySnapshottableFeature.SNAPSHOT_LIMIT,
|
||||||
((INodeDirectorySnapshottable) rootNode).getSnapshotQuota());
|
rootNode.getDirectorySnapshottableFeature().getSnapshotQuota());
|
||||||
|
|
||||||
// disallowSnapshot on dir
|
// disallowSnapshot on dir
|
||||||
hdfs.disallowSnapshot(root);
|
hdfs.disallowSnapshot(root);
|
||||||
rootNode = fsdir.getINode4Write(root.toString()).asDirectory();
|
rootNode = fsdir.getINode4Write(root.toString()).asDirectory();
|
||||||
assertTrue(rootNode.isSnapshottable());
|
assertTrue(rootNode.isSnapshottable());
|
||||||
assertEquals(0, ((INodeDirectorySnapshottable) rootNode).getSnapshotQuota());
|
assertEquals(0, rootNode.getDirectorySnapshottableFeature().getSnapshotQuota());
|
||||||
// do it again
|
// do it again
|
||||||
hdfs.disallowSnapshot(root);
|
hdfs.disallowSnapshot(root);
|
||||||
rootNode = fsdir.getINode4Write(root.toString()).asDirectory();
|
rootNode = fsdir.getINode4Write(root.toString()).asDirectory();
|
||||||
assertTrue(rootNode.isSnapshottable());
|
assertTrue(rootNode.isSnapshottable());
|
||||||
assertEquals(0, ((INodeDirectorySnapshottable) rootNode).getSnapshotQuota());
|
assertEquals(0, rootNode.getDirectorySnapshottableFeature().getSnapshotQuota());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -282,10 +282,10 @@ public class TestSnapshotDeletion {
|
||||||
checkQuotaUsageComputation(dir, 14L, BLOCKSIZE * REPLICATION * 4);
|
checkQuotaUsageComputation(dir, 14L, BLOCKSIZE * REPLICATION * 4);
|
||||||
|
|
||||||
// get two snapshots for later use
|
// get two snapshots for later use
|
||||||
Snapshot snapshot0 = ((INodeDirectorySnapshottable) fsdir.getINode(dir
|
Snapshot snapshot0 = fsdir.getINode(dir.toString()).asDirectory()
|
||||||
.toString())).getSnapshot(DFSUtil.string2Bytes("s0"));
|
.getSnapshot(DFSUtil.string2Bytes("s0"));
|
||||||
Snapshot snapshot1 = ((INodeDirectorySnapshottable) fsdir.getINode(dir
|
Snapshot snapshot1 = fsdir.getINode(dir.toString()).asDirectory()
|
||||||
.toString())).getSnapshot(DFSUtil.string2Bytes("s1"));
|
.getSnapshot(DFSUtil.string2Bytes("s1"));
|
||||||
|
|
||||||
// Case 2 + Case 3: delete noChangeDirParent, noChangeFile, and
|
// Case 2 + Case 3: delete noChangeDirParent, noChangeFile, and
|
||||||
// metaChangeFile2. Note that when we directly delete a directory, the
|
// metaChangeFile2. Note that when we directly delete a directory, the
|
||||||
|
@ -510,8 +510,7 @@ public class TestSnapshotDeletion {
|
||||||
}
|
}
|
||||||
|
|
||||||
// check 1. there is no snapshot s0
|
// check 1. there is no snapshot s0
|
||||||
final INodeDirectorySnapshottable dirNode =
|
final INodeDirectory dirNode = fsdir.getINode(dir.toString()).asDirectory();
|
||||||
(INodeDirectorySnapshottable) fsdir.getINode(dir.toString());
|
|
||||||
Snapshot snapshot0 = dirNode.getSnapshot(DFSUtil.string2Bytes("s0"));
|
Snapshot snapshot0 = dirNode.getSnapshot(DFSUtil.string2Bytes("s0"));
|
||||||
assertNull(snapshot0);
|
assertNull(snapshot0);
|
||||||
Snapshot snapshot1 = dirNode.getSnapshot(DFSUtil.string2Bytes("s1"));
|
Snapshot snapshot1 = dirNode.getSnapshot(DFSUtil.string2Bytes("s1"));
|
||||||
|
|
|
@ -18,13 +18,19 @@
|
||||||
|
|
||||||
package org.apache.hadoop.hdfs.server.namenode.snapshot;
|
package org.apache.hadoop.hdfs.server.namenode.snapshot;
|
||||||
|
|
||||||
|
import static org.mockito.Matchers.anyString;
|
||||||
|
import static org.mockito.Mockito.doReturn;
|
||||||
|
import static org.mockito.Mockito.mock;
|
||||||
|
import static org.mockito.Mockito.spy;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
|
||||||
import org.apache.hadoop.hdfs.protocol.SnapshotException;
|
import org.apache.hadoop.hdfs.protocol.SnapshotException;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.FSDirectory;
|
import org.apache.hadoop.hdfs.server.namenode.FSDirectory;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.INode;
|
import org.apache.hadoop.hdfs.server.namenode.INode;
|
||||||
import org.junit.*;
|
import org.apache.hadoop.hdfs.server.namenode.INodeDirectory;
|
||||||
import static org.mockito.Mockito.*;
|
import org.junit.Assert;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -40,7 +46,7 @@ public class TestSnapshotManager {
|
||||||
public void testSnapshotLimits() throws Exception {
|
public void testSnapshotLimits() throws Exception {
|
||||||
// Setup mock objects for SnapshotManager.createSnapshot.
|
// Setup mock objects for SnapshotManager.createSnapshot.
|
||||||
//
|
//
|
||||||
INodeDirectorySnapshottable ids = mock(INodeDirectorySnapshottable.class);
|
INodeDirectory ids = mock(INodeDirectory.class);
|
||||||
FSDirectory fsdir = mock(FSDirectory.class);
|
FSDirectory fsdir = mock(FSDirectory.class);
|
||||||
|
|
||||||
SnapshotManager sm = spy(new SnapshotManager(fsdir));
|
SnapshotManager sm = spy(new SnapshotManager(fsdir));
|
||||||
|
|
|
@ -37,6 +37,7 @@ import org.apache.hadoop.hdfs.protocol.HdfsConstants;
|
||||||
import org.apache.hadoop.hdfs.protocol.SnapshotException;
|
import org.apache.hadoop.hdfs.protocol.SnapshotException;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.FSDirectory;
|
import org.apache.hadoop.hdfs.server.namenode.FSDirectory;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.FSNamesystem;
|
import org.apache.hadoop.hdfs.server.namenode.FSNamesystem;
|
||||||
|
import org.apache.hadoop.hdfs.server.namenode.INodeDirectory;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectoryWithSnapshotFeature.DirectoryDiff;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.DirectoryWithSnapshotFeature.DirectoryDiff;
|
||||||
import org.apache.hadoop.hdfs.util.ReadOnlyList;
|
import org.apache.hadoop.hdfs.util.ReadOnlyList;
|
||||||
import org.apache.hadoop.ipc.RemoteException;
|
import org.apache.hadoop.ipc.RemoteException;
|
||||||
|
@ -88,12 +89,13 @@ public class TestSnapshotRename {
|
||||||
public ExpectedException exception = ExpectedException.none();
|
public ExpectedException exception = ExpectedException.none();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check the correctness of snapshot list within
|
* Check the correctness of snapshot list within snapshottable dir
|
||||||
* {@link INodeDirectorySnapshottable}
|
|
||||||
*/
|
*/
|
||||||
private void checkSnapshotList(INodeDirectorySnapshottable srcRoot,
|
private void checkSnapshotList(INodeDirectory srcRoot,
|
||||||
String[] sortedNames, String[] names) {
|
String[] sortedNames, String[] names) {
|
||||||
ReadOnlyList<Snapshot> listByName = srcRoot.getSnapshotsByNames();
|
assertTrue(srcRoot.isSnapshottable());
|
||||||
|
ReadOnlyList<Snapshot> listByName = srcRoot
|
||||||
|
.getDirectorySnapshottableFeature().getSnapshotList();
|
||||||
assertEquals(sortedNames.length, listByName.size());
|
assertEquals(sortedNames.length, listByName.size());
|
||||||
for (int i = 0; i < listByName.size(); i++) {
|
for (int i = 0; i < listByName.size(); i++) {
|
||||||
assertEquals(sortedNames[i], listByName.get(i).getRoot().getLocalName());
|
assertEquals(sortedNames[i], listByName.get(i).getRoot().getLocalName());
|
||||||
|
@ -101,7 +103,8 @@ public class TestSnapshotRename {
|
||||||
List<DirectoryDiff> listByTime = srcRoot.getDiffs().asList();
|
List<DirectoryDiff> listByTime = srcRoot.getDiffs().asList();
|
||||||
assertEquals(names.length, listByTime.size());
|
assertEquals(names.length, listByTime.size());
|
||||||
for (int i = 0; i < listByTime.size(); i++) {
|
for (int i = 0; i < listByTime.size(); i++) {
|
||||||
Snapshot s = srcRoot.getSnapshotById(listByTime.get(i).getSnapshotId());
|
Snapshot s = srcRoot.getDirectorySnapshottableFeature().getSnapshotById(
|
||||||
|
listByTime.get(i).getSnapshotId());
|
||||||
assertEquals(names[i], s.getRoot().getLocalName());
|
assertEquals(names[i], s.getRoot().getLocalName());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -121,8 +124,7 @@ public class TestSnapshotRename {
|
||||||
// Rename s3 to s22
|
// Rename s3 to s22
|
||||||
hdfs.renameSnapshot(sub1, "s3", "s22");
|
hdfs.renameSnapshot(sub1, "s3", "s22");
|
||||||
// Check the snapshots list
|
// Check the snapshots list
|
||||||
INodeDirectorySnapshottable srcRoot = INodeDirectorySnapshottable.valueOf(
|
INodeDirectory srcRoot = fsdir.getINode(sub1.toString()).asDirectory();
|
||||||
fsdir.getINode(sub1.toString()), sub1.toString());
|
|
||||||
checkSnapshotList(srcRoot, new String[] { "s1", "s2", "s22" },
|
checkSnapshotList(srcRoot, new String[] { "s1", "s2", "s22" },
|
||||||
new String[] { "s1", "s2", "s22" });
|
new String[] { "s1", "s2", "s22" });
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue