HDFS-4487. Fix snapshot diff report for HDFS-4446. Contributed by Jing Zhao

git-svn-id: https://svn.apache.org/repos/asf/hadoop/common/branches/HDFS-2802@1446385 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Tsz-wo Sze 2013-02-14 23:07:49 +00:00
parent d42d0860cb
commit d9e2514d21
18 changed files with 373 additions and 174 deletions

View File

@ -157,3 +157,5 @@ Branch-2802 Snapshot (Unreleased)
HDFS-4481. Change fsimage to support snapshot file diffs. (szetszwo) HDFS-4481. Change fsimage to support snapshot file diffs. (szetszwo)
HDFS-4500. Refactor snapshot INode methods. (szetszwo) HDFS-4500. Refactor snapshot INode methods. (szetszwo)
HDFS-4487. Fix snapshot diff report for HDFS-4446. (Jing Zhao via szetszwo)

View File

@ -91,6 +91,8 @@ import com.google.protobuf.BlockingService;
public class DFSUtil { public class DFSUtil {
public static final Log LOG = LogFactory.getLog(DFSUtil.class.getName()); public static final Log LOG = LogFactory.getLog(DFSUtil.class.getName());
public static final byte[] EMPTY_BYTES = {};
private DFSUtil() { /* Hidden constructor */ } private DFSUtil() { /* Hidden constructor */ }
private static final ThreadLocal<Random> RANDOM = new ThreadLocal<Random>() { private static final ThreadLocal<Random> RANDOM = new ThreadLocal<Random>() {
@Override @Override
@ -259,6 +261,37 @@ public class DFSUtil {
return result.toString(); return result.toString();
} }
/**
* Given a list of path components returns a byte array
*/
public static byte[] byteArray2bytes(byte[][] pathComponents) {
if (pathComponents.length == 0) {
return EMPTY_BYTES;
} else if (pathComponents.length == 1
&& (pathComponents[0] == null || pathComponents[0].length == 0)) {
return new byte[]{(byte) Path.SEPARATOR_CHAR};
}
int length = 0;
for (int i = 0; i < pathComponents.length; i++) {
length += pathComponents[i].length;
if (i < pathComponents.length - 1) {
length++; // for SEPARATOR
}
}
byte[] path = new byte[length];
int index = 0;
for (int i = 0; i < pathComponents.length; i++) {
System.arraycopy(pathComponents[i], 0, path, index,
pathComponents[i].length);
index += pathComponents[i].length;
if (i < pathComponents.length - 1) {
path[index] = (byte) Path.SEPARATOR_CHAR;
index++;
}
}
return path;
}
/** Convert an object representing a path to a string. */ /** Convert an object representing a path to a string. */
public static String path2String(final Object path) { public static String path2String(final Object path) {
return path == null? null return path == null? null

View File

@ -17,9 +17,12 @@
*/ */
package org.apache.hadoop.hdfs.protocol; package org.apache.hadoop.hdfs.protocol;
import java.util.Arrays;
import java.util.Collections; import java.util.Collections;
import java.util.List; import java.util.List;
import org.apache.hadoop.fs.Path;
import org.apache.hadoop.hdfs.DFSUtil;
import org.apache.hadoop.hdfs.server.namenode.snapshot.INodeDirectorySnapshottable.SnapshotDiffInfo; import org.apache.hadoop.hdfs.server.namenode.snapshot.INodeDirectorySnapshottable.SnapshotDiffInfo;
/** /**
@ -75,25 +78,42 @@ public class SnapshotDiffReport {
public static class DiffReportEntry { public static class DiffReportEntry {
/** The type of the difference. */ /** The type of the difference. */
private final DiffType type; private final DiffType type;
/** The full path of the file/directory where changes have happened */ /**
private final String fullPath; * The relative path (related to the snapshot root) of the file/directory
* where changes have happened
*/
private final byte[] relativePath;
public DiffReportEntry(DiffType type, String fullPath) { public DiffReportEntry(DiffType type, byte[] path) {
this.type = type; this.type = type;
this.fullPath = fullPath; this.relativePath = path;
}
public DiffReportEntry(DiffType type, byte[][] pathComponents) {
this.type = type;
this.relativePath = DFSUtil.byteArray2bytes(pathComponents);
} }
@Override @Override
public String toString() { public String toString() {
return type.getLabel() + "\t" + fullPath; return type.getLabel() + "\t" + getRelativePathString();
} }
public DiffType getType() { public DiffType getType() {
return type; return type;
} }
public String getFullPath() { public String getRelativePathString() {
return fullPath; String path = DFSUtil.bytes2String(relativePath);
if (path.isEmpty()) {
return ".";
} else {
return "." + Path.SEPARATOR + path;
}
}
public byte[] getRelativePath() {
return relativePath;
} }
@Override @Override
@ -104,14 +124,14 @@ public class SnapshotDiffReport {
if (other != null && other instanceof DiffReportEntry) { if (other != null && other instanceof DiffReportEntry) {
DiffReportEntry entry = (DiffReportEntry) other; DiffReportEntry entry = (DiffReportEntry) other;
return type.equals(entry.getType()) return type.equals(entry.getType())
&& fullPath.equals(entry.getFullPath()); && Arrays.equals(relativePath, entry.getRelativePath());
} }
return false; return false;
} }
@Override @Override
public int hashCode() { public int hashCode() {
return fullPath.hashCode(); return Arrays.hashCode(relativePath);
} }
} }

View File

@ -30,6 +30,7 @@ import org.apache.hadoop.fs.CreateFlag;
import org.apache.hadoop.fs.FsServerDefaults; import org.apache.hadoop.fs.FsServerDefaults;
import org.apache.hadoop.fs.permission.FsPermission; import org.apache.hadoop.fs.permission.FsPermission;
import org.apache.hadoop.ha.HAServiceProtocol.HAServiceState; import org.apache.hadoop.ha.HAServiceProtocol.HAServiceState;
import org.apache.hadoop.hdfs.DFSUtil;
import org.apache.hadoop.hdfs.protocol.Block; import org.apache.hadoop.hdfs.protocol.Block;
import org.apache.hadoop.hdfs.protocol.ClientProtocol; import org.apache.hadoop.hdfs.protocol.ClientProtocol;
import org.apache.hadoop.hdfs.protocol.CorruptFileBlocks; import org.apache.hadoop.hdfs.protocol.CorruptFileBlocks;
@ -299,8 +300,8 @@ public class PBHelper {
public static BlockKeyProto convert(BlockKey key) { public static BlockKeyProto convert(BlockKey key) {
byte[] encodedKey = key.getEncodedKey(); byte[] encodedKey = key.getEncodedKey();
ByteString keyBytes = ByteString.copyFrom(encodedKey == null ? new byte[0] ByteString keyBytes = ByteString.copyFrom(encodedKey == null ?
: encodedKey); DFSUtil.EMPTY_BYTES : encodedKey);
return BlockKeyProto.newBuilder().setKeyId(key.getKeyId()) return BlockKeyProto.newBuilder().setKeyId(key.getKeyId())
.setKeyBytes(keyBytes).setExpiryDate(key.getExpiryDate()).build(); .setKeyBytes(keyBytes).setExpiryDate(key.getExpiryDate()).build();
} }
@ -1119,8 +1120,8 @@ public class PBHelper {
int snapshotNumber = status.getSnapshotNumber(); int snapshotNumber = status.getSnapshotNumber();
int snapshotQuota = status.getSnapshotQuota(); int snapshotQuota = status.getSnapshotQuota();
byte[] parentFullPath = status.getParentFullPath(); byte[] parentFullPath = status.getParentFullPath();
ByteString parentFullPathBytes = ByteString ByteString parentFullPathBytes = ByteString.copyFrom(
.copyFrom(parentFullPath == null ? new byte[0] : parentFullPath); parentFullPath == null ? DFSUtil.EMPTY_BYTES : parentFullPath);
HdfsFileStatusProto fs = convert(status.getDirStatus()); HdfsFileStatusProto fs = convert(status.getDirStatus());
SnapshottableDirectoryStatusProto.Builder builder = SnapshottableDirectoryStatusProto.Builder builder =
SnapshottableDirectoryStatusProto SnapshottableDirectoryStatusProto
@ -1411,20 +1412,23 @@ public class PBHelper {
} }
DiffType type = DiffType.getTypeFromLabel(entry DiffType type = DiffType.getTypeFromLabel(entry
.getModificationLabel()); .getModificationLabel());
return type != null ? new DiffReportEntry(type, entry.getFullpath()) return type == null ? null :
: null; new DiffReportEntry(type, entry.getFullpath().toByteArray());
} }
public static SnapshotDiffReportEntryProto convert(DiffReportEntry entry) { public static SnapshotDiffReportEntryProto convert(DiffReportEntry entry) {
if (entry == null) { if (entry == null) {
return null; return null;
} }
String fullPath = entry.getFullPath(); byte[] fullPath = entry.getRelativePath();
ByteString fullPathString = ByteString
.copyFrom(fullPath == null ? DFSUtil.EMPTY_BYTES : fullPath);
String modification = entry.getType().getLabel(); String modification = entry.getType().getLabel();
SnapshotDiffReportEntryProto entryProto = SnapshotDiffReportEntryProto SnapshotDiffReportEntryProto entryProto = SnapshotDiffReportEntryProto
.newBuilder().setFullpath(fullPath).setModificationLabel(modification) .newBuilder().setFullpath(fullPathString)
.build(); .setModificationLabel(modification).build();
return entryProto; return entryProto;
} }

View File

@ -1496,21 +1496,49 @@ public class FSDirectory implements Closeable {
return fullPathName.toString(); return fullPathName.toString();
} }
/** Return the full path name of the specified inode */ /**
static String getFullPathName(INode inode) { * @return the relative path of an inode from one of its ancestors,
// calculate the depth of this inode from root * represented by an array of inodes.
*/
private static INode[] getRelativePathINodes(INode inode, INode ancestor) {
// calculate the depth of this inode from the ancestor
int depth = 0; int depth = 0;
for (INode i = inode; i != null; i = i.parent) { for (INode i = inode; i != null && !i.equals(ancestor); i = i.parent) {
depth++; depth++;
} }
INode[] inodes = new INode[depth]; INode[] inodes = new INode[depth];
// fill up the inodes in the path from this inode to root // fill up the inodes in the path from this inode to root
for (int i = 0; i < depth; i++) { for (int i = 0; i < depth; i++) {
inodes[depth-i-1] = inode; inodes[depth - i - 1] = inode;
inode = inode.parent; inode = inode.parent;
} }
return getFullPathName(inodes, depth-1); return inodes;
}
private static INode[] getFullPathINodes(INode inode) {
return getRelativePathINodes(inode, null);
}
/** Return the full path name of the specified inode */
static String getFullPathName(INode inode) {
INode[] inodes = getFullPathINodes(inode);
return getFullPathName(inodes, inodes.length - 1);
}
/**
* For a given inode, get its relative path from its ancestor.
* @param inode The given inode.
* @param ancestor An ancestor inode of the given inode.
* @return The relative path name represented in an array of byte array.
*/
static byte[][] getRelativePathNameBytes(INode inode, INode ancestor) {
INode[] inodes = getRelativePathINodes(inode, ancestor);
byte[][] path = new byte[inodes.length][];
for (int i = 0; i < inodes.length; i++) {
path[i] = inodes[i].getLocalNameBytes();
}
return path;
} }
/** /**

View File

@ -788,7 +788,7 @@ public class FSImageFormat {
byte[] byteStore = new byte[4*HdfsConstants.MAX_PATH_LENGTH]; byte[] byteStore = new byte[4*HdfsConstants.MAX_PATH_LENGTH];
ByteBuffer strbuf = ByteBuffer.wrap(byteStore); ByteBuffer strbuf = ByteBuffer.wrap(byteStore);
// save the root // save the root
FSImageSerialization.saveINode2Image(fsDir.rootDir, out); FSImageSerialization.saveINode2Image(fsDir.rootDir, out, false);
// save the rest of the nodes // save the rest of the nodes
saveImage(strbuf, fsDir.rootDir, out, null); saveImage(strbuf, fsDir.rootDir, out, null);
// save files under construction // save files under construction
@ -826,7 +826,7 @@ public class FSImageFormat {
int i = 0; int i = 0;
for(INode child : children) { for(INode child : children) {
// print all children first // print all children first
FSImageSerialization.saveINode2Image(child, out); FSImageSerialization.saveINode2Image(child, out, false);
if (child.isDirectory()) { if (child.isDirectory()) {
dirNum++; dirNum++;
} }

View File

@ -236,14 +236,15 @@ public class FSImageSerialization {
/** /**
* Save one inode's attributes to the image. * Save one inode's attributes to the image.
*/ */
public static void saveINode2Image(INode node, DataOutputStream out) public static void saveINode2Image(INode node, DataOutputStream out,
boolean writeUnderConstruction)
throws IOException { throws IOException {
if (node.isDirectory()) { if (node.isDirectory()) {
writeINodeDirectory((INodeDirectory) node, out); writeINodeDirectory((INodeDirectory) node, out);
} else if (node.isSymlink()) { } else if (node.isSymlink()) {
writeINodeSymlink((INodeSymlink) node, out); writeINodeSymlink((INodeSymlink) node, out);
} else { } else {
writeINodeFile((INodeFile) node, out, false); writeINodeFile((INodeFile) node, out, writeUnderConstruction);
} }
} }

View File

@ -390,6 +390,13 @@ public abstract class INode implements Diff.Element<byte[]> {
return FSDirectory.getFullPathName(this); return FSDirectory.getFullPathName(this);
} }
/**
* @return The full path name represented in a list of byte array
*/
public byte[][] getRelativePathNameBytes(INode ancestor) {
return FSDirectory.getRelativePathNameBytes(this, ancestor);
}
@Override @Override
public String toString() { public String toString() {
return getLocalName(); return getLocalName();
@ -565,12 +572,10 @@ public abstract class INode implements Diff.Element<byte[]> {
return buf.toString(); return buf.toString();
} }
public static final byte[] EMPTY_BYTES = {};
@Override @Override
public final int compareTo(byte[] bytes) { public final int compareTo(byte[] bytes) {
final byte[] left = name == null? EMPTY_BYTES: name; final byte[] left = name == null? DFSUtil.EMPTY_BYTES: name;
final byte[] right = bytes == null? EMPTY_BYTES: bytes; final byte[] right = bytes == null? DFSUtil.EMPTY_BYTES: bytes;
return SignedBytes.lexicographicalComparator().compare(left, right); return SignedBytes.lexicographicalComparator().compare(left, right);
} }

View File

@ -675,7 +675,8 @@ public class INodeDirectory extends INode {
} }
private void updateLatestSnapshot(Snapshot s) { private void updateLatestSnapshot(Snapshot s) {
if (Snapshot.ID_COMPARATOR.compare(snapshot, s) < 0) { if (snapshot == null
|| (s != null && Snapshot.ID_COMPARATOR.compare(snapshot, s) < 0)) {
snapshot = s; snapshot = s;
} }
} }

View File

@ -145,6 +145,34 @@ abstract class AbstractINodeDiffList<N extends INode,
} }
} }
/**
* Check if changes have happened between two snapshots.
* @param earlierSnapshot The snapshot taken earlier
* @param laterSnapshot The snapshot taken later
* @return Whether or not modifications (including diretory/file metadata
* change, file creation/deletion under the directory) have happened
* between snapshots.
*/
final boolean changedBetweenSnapshots(Snapshot earlierSnapshot,
Snapshot laterSnapshot) {
final int size = diffs.size();
int earlierDiffIndex = Collections.binarySearch(diffs, earlierSnapshot);
if (-earlierDiffIndex - 1 == size) {
// if the earlierSnapshot is after the latest SnapshotDiff stored in
// diffs, no modification happened after the earlierSnapshot
return false;
}
if (laterSnapshot != null) {
int laterDiffIndex = Collections.binarySearch(diffs, laterSnapshot);
if (laterDiffIndex == -1 || laterDiffIndex == 0) {
// if the laterSnapshot is the earliest SnapshotDiff stored in diffs, or
// before it, no modification happened before the laterSnapshot
return false;
}
}
return true;
}
/** /**
* @return the inode corresponding to the given snapshot. * @return the inode corresponding to the given snapshot.
* Note that the current inode is returned if there is no change * Note that the current inode is returned if there is no change

View File

@ -22,11 +22,10 @@ import java.io.PrintWriter;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collections; import java.util.Collections;
import java.util.Comparator; import java.util.Comparator;
import java.util.HashMap;
import java.util.Iterator; import java.util.Iterator;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.SortedMap;
import java.util.TreeMap;
import org.apache.hadoop.HadoopIllegalArgumentException; import org.apache.hadoop.HadoopIllegalArgumentException;
import org.apache.hadoop.classification.InterfaceAudience; import org.apache.hadoop.classification.InterfaceAudience;
@ -36,11 +35,12 @@ import org.apache.hadoop.hdfs.protocol.SnapshotDiffReport.DiffReportEntry;
import org.apache.hadoop.hdfs.protocol.SnapshotDiffReport.DiffType; import org.apache.hadoop.hdfs.protocol.SnapshotDiffReport.DiffType;
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.INodeDirectory;
import org.apache.hadoop.hdfs.server.namenode.snapshot.diff.Diff; import org.apache.hadoop.hdfs.server.namenode.INodeFile;
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.base.Preconditions; import com.google.common.base.Preconditions;
import com.google.common.primitives.SignedBytes;
/** /**
* Directories where taking snapshots is allowed. * Directories where taking snapshots is allowed.
@ -69,6 +69,7 @@ public class INodeDirectorySnapshottable extends INodeDirectoryWithSnapshot {
* directory. * directory.
*/ */
public static class SnapshotDiffInfo { public static class SnapshotDiffInfo {
/** Compare two inodes based on their full names */
public static final Comparator<INode> INODE_COMPARATOR = public static final Comparator<INode> INODE_COMPARATOR =
new Comparator<INode>() { new Comparator<INode>() {
@Override @Override
@ -76,7 +77,13 @@ public class INodeDirectorySnapshottable extends INodeDirectoryWithSnapshot {
if (left == null) { if (left == null) {
return right == null ? 0 : -1; return right == null ? 0 : -1;
} else { } else {
return right == null ? 1 : left.compareTo(right.getLocalNameBytes()); if (right == null) {
return 1;
} else {
int cmp = compare(left.getParent(), right.getParent());
return cmp == 0 ? SignedBytes.lexicographicalComparator().compare(
left.getLocalNameBytes(), right.getLocalNameBytes()) : cmp;
}
} }
} }
}; };
@ -88,40 +95,46 @@ public class INodeDirectorySnapshottable extends INodeDirectoryWithSnapshot {
/** The end point of the difference */ /** The end point of the difference */
private final Snapshot to; private final Snapshot to;
/** /**
* A map capturing the detailed difference. Each key indicates a directory * The list recording modified INodeFile and INodeDirectory. Sorted based on
* whose metadata or children have been changed between the two snapshots, * their names.
* while its associated value is a {@link Diff} storing the changes happened
* to the children (files).
*/ */
private final SortedMap<INodeDirectoryWithSnapshot, ChildrenDiff> diffMap; private final List<INode> diffList = new ArrayList<INode>();;
/**
* A map capturing the detailed difference about file creation/deletion.
* Each key indicates a directory whose children have been changed between
* the two snapshots, while its associated value is a {@link ChildrenDiff}
* storing the changes (creation/deletion) happened to the children (files).
*/
private final Map<INodeDirectoryWithSnapshot, ChildrenDiff> diffMap =
new HashMap<INodeDirectoryWithSnapshot, ChildrenDiff>();
public SnapshotDiffInfo(INodeDirectorySnapshottable snapshotRoot, SnapshotDiffInfo(INodeDirectorySnapshottable snapshotRoot, Snapshot start,
Snapshot start, Snapshot end) { Snapshot end) {
this.snapshotRoot = snapshotRoot; this.snapshotRoot = snapshotRoot;
this.from = start; this.from = start;
this.to = end; this.to = end;
this.diffMap = new TreeMap<INodeDirectoryWithSnapshot, ChildrenDiff>(
INODE_COMPARATOR);
} }
/** Add a dir-diff pair into {@link #diffMap} */ /** Add a dir-diff pair */
public void addDiff(INodeDirectoryWithSnapshot dir, ChildrenDiff diff) { private void addDirDiff(INodeDirectoryWithSnapshot dir, ChildrenDiff diff) {
diffMap.put(dir, diff); diffMap.put(dir, diff);
int i = Collections.binarySearch(diffList, dir, INODE_COMPARATOR);
if (i < 0) {
diffList.add(-i - 1, dir);
}
} }
/** /** Add a modified file */
* Get the name of the given snapshot. private void addFileDiff(INodeFile file) {
* @param s The given snapshot. int i = Collections.binarySearch(diffList, file, INODE_COMPARATOR);
* @return The name of the snapshot, or an empty string if {@code s} is null if (i < 0) {
*/ diffList.add(-i - 1, file);
private static String getSnapshotName(Snapshot s) { }
return s != null ? s.getRoot().getLocalName() : "";
} }
/** @return True if {@link #from} is earlier than {@link #to} */ /** @return True if {@link #from} is earlier than {@link #to} */
private boolean isFromEarlier() { private boolean isFromEarlier() {
return to == null return Snapshot.ID_COMPARATOR.compare(from, to) < 0;
|| (from != null && Snapshot.ID_COMPARATOR.compare(from, to) < 0);
} }
/** /**
@ -129,17 +142,20 @@ public class INodeDirectorySnapshottable extends INodeDirectoryWithSnapshot {
* @return A {@link SnapshotDiffReport} describing the difference * @return A {@link SnapshotDiffReport} describing the difference
*/ */
public SnapshotDiffReport generateReport() { public SnapshotDiffReport generateReport() {
List<DiffReportEntry> diffList = new ArrayList<DiffReportEntry>(); List<DiffReportEntry> diffReportList = new ArrayList<DiffReportEntry>();
for (Map.Entry<INodeDirectoryWithSnapshot, ChildrenDiff> entry : diffMap for (INode node : diffList) {
.entrySet()) { diffReportList.add(new DiffReportEntry(DiffType.MODIFY, node
diffList.add(new DiffReportEntry(DiffType.MODIFY, entry.getKey() .getRelativePathNameBytes(snapshotRoot)));
.getFullPathName())); if (node.isDirectory()) {
List<DiffReportEntry> subList = entry.getValue().generateReport( ChildrenDiff dirDiff = diffMap.get(node);
entry.getKey(), isFromEarlier()); List<DiffReportEntry> subList = dirDiff.generateReport(snapshotRoot,
diffList.addAll(subList); (INodeDirectoryWithSnapshot) node, isFromEarlier());
diffReportList.addAll(subList);
}
} }
return new SnapshotDiffReport(snapshotRoot.getFullPathName(), return new SnapshotDiffReport(snapshotRoot.getFullPathName(),
getSnapshotName(from), getSnapshotName(to), diffList); Snapshot.getSnapshotName(from), Snapshot.getSnapshotName(to),
diffReportList);
} }
} }
@ -315,7 +331,7 @@ public class INodeDirectorySnapshottable extends INodeDirectoryWithSnapshot {
Snapshot toSnapshot = getSnapshotByName(to); Snapshot toSnapshot = getSnapshotByName(to);
SnapshotDiffInfo diffs = new SnapshotDiffInfo(this, fromSnapshot, SnapshotDiffInfo diffs = new SnapshotDiffInfo(this, fromSnapshot,
toSnapshot); toSnapshot);
computeDiffInDir(this, diffs); computeDiffRecursively(this, diffs);
return diffs; return diffs;
} }
@ -343,30 +359,41 @@ public class INodeDirectorySnapshottable extends INodeDirectoryWithSnapshot {
/** /**
* Recursively compute the difference between snapshots under a given * Recursively compute the difference between snapshots under a given
* directory. * directory/file.
* @param dir The directory under which the diff is computed. * @param node The directory/file under which the diff is computed.
* @param diffReport data structure used to store the diff. * @param diffReport data structure used to store the diff.
*/ */
private void computeDiffInDir(INodeDirectory dir, private void computeDiffRecursively(INode node,
SnapshotDiffInfo diffReport) { SnapshotDiffInfo diffReport) {
ChildrenDiff diff = new ChildrenDiff(); ChildrenDiff diff = new ChildrenDiff();
if (node instanceof INodeDirectory) {
INodeDirectory dir = (INodeDirectory) node;
if (dir instanceof INodeDirectoryWithSnapshot) { if (dir instanceof INodeDirectoryWithSnapshot) {
boolean change = ((INodeDirectoryWithSnapshot) dir) INodeDirectoryWithSnapshot sdir = (INodeDirectoryWithSnapshot) dir;
.computeDiffBetweenSnapshots(diffReport.from, boolean change = sdir.computeDiffBetweenSnapshots(
diffReport.to, diff); diffReport.from, diffReport.to, diff);
if (change) { if (change) {
diffReport.addDiff((INodeDirectoryWithSnapshot) dir, diffReport.addDirDiff(sdir, diff);
diff);
} }
} }
ReadOnlyList<INode> children = dir.getChildrenList(null); ReadOnlyList<INode> children = dir.getChildrenList(diffReport
.isFromEarlier() ? diffReport.to : diffReport.from);
for (INode child : children) { for (INode child : children) {
if (child instanceof INodeDirectory if (diff.searchCreated(child.getLocalNameBytes()) == null
&& diff.searchCreated(child.getLocalNameBytes()) == null) { && diff.searchDeleted(child.getLocalNameBytes()) == null) {
// Compute diff recursively for children that are directories. We do not computeDiffRecursively(child, diffReport);
// need to compute diff for those contained in the created list since }
// directory contained in the created list must be new created. }
computeDiffInDir((INodeDirectory) child, diffReport); } else if (node instanceof FileWithSnapshot) {
FileWithSnapshot file = (FileWithSnapshot) node;
Snapshot earlierSnapshot = diffReport.isFromEarlier() ? diffReport.from
: diffReport.to;
Snapshot laterSnapshot = diffReport.isFromEarlier() ? diffReport.to
: diffReport.from;
boolean change = file.getDiffs().changedBetweenSnapshots(earlierSnapshot,
laterSnapshot);
if (change) {
diffReport.addFileDiff(file.asINodeFile());
} }
} }
} }

View File

@ -25,7 +25,6 @@ import java.util.Iterator;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import org.apache.hadoop.fs.Path;
import org.apache.hadoop.hdfs.protocol.SnapshotDiffReport.DiffReportEntry; import org.apache.hadoop.hdfs.protocol.SnapshotDiffReport.DiffReportEntry;
import org.apache.hadoop.hdfs.protocol.SnapshotDiffReport.DiffType; import org.apache.hadoop.hdfs.protocol.SnapshotDiffReport.DiffType;
import org.apache.hadoop.hdfs.server.namenode.FSImageSerialization; import org.apache.hadoop.hdfs.server.namenode.FSImageSerialization;
@ -77,7 +76,7 @@ public class INodeDirectoryWithSnapshot extends INodeDirectoryWithQuota {
final List<INode> deleted = getDeletedList(); final List<INode> deleted = getDeletedList();
out.writeInt(deleted.size()); out.writeInt(deleted.size());
for (INode node : deleted) { for (INode node : deleted) {
FSImageSerialization.saveINode2Image(node, out); FSImageSerialization.saveINode2Image(node, out, true);
} }
} }
@ -100,52 +99,62 @@ public class INodeDirectoryWithSnapshot extends INodeDirectoryWithQuota {
/** /**
* Interpret the diff and generate a list of {@link DiffReportEntry}. * Interpret the diff and generate a list of {@link DiffReportEntry}.
* @root The snapshot root of the diff report.
* @param parent The directory that the diff belongs to. * @param parent The directory that the diff belongs to.
* @param fromEarlier True indicates {@code diff=later-earlier}, * @param fromEarlier True indicates {@code diff=later-earlier},
* False indicates {@code diff=earlier-later} * False indicates {@code diff=earlier-later}
* @return A list of {@link DiffReportEntry} as the diff report. * @return A list of {@link DiffReportEntry} as the diff report.
*/ */
public List<DiffReportEntry> generateReport( public List<DiffReportEntry> generateReport(
INodeDirectoryWithSnapshot parent, boolean fromEarlier) { INodeDirectorySnapshottable root, INodeDirectoryWithSnapshot parent,
List<DiffReportEntry> mList = new ArrayList<DiffReportEntry>(); boolean fromEarlier) {
List<DiffReportEntry> cList = new ArrayList<DiffReportEntry>(); List<DiffReportEntry> cList = new ArrayList<DiffReportEntry>();
List<DiffReportEntry> dList = new ArrayList<DiffReportEntry>(); List<DiffReportEntry> dList = new ArrayList<DiffReportEntry>();
int c = 0, d = 0; int c = 0, d = 0;
List<INode> created = getCreatedList(); List<INode> created = getCreatedList();
List<INode> deleted = getDeletedList(); List<INode> deleted = getDeletedList();
byte[][] parentPath = parent.getRelativePathNameBytes(root);
byte[][] fullPath = new byte[parentPath.length + 1][];
System.arraycopy(parentPath, 0, fullPath, 0, parentPath.length);
for (; c < created.size() && d < deleted.size(); ) { for (; c < created.size() && d < deleted.size(); ) {
INode cnode = created.get(c); INode cnode = created.get(c);
INode dnode = deleted.get(d); INode dnode = deleted.get(d);
if (cnode.equals(dnode)) { if (cnode.equals(dnode)) {
mList.add(new DiffReportEntry(DiffType.MODIFY, parent fullPath[fullPath.length - 1] = cnode.getLocalNameBytes();
.getFullPathName() + Path.SEPARATOR + cnode.getLocalName())); if (cnode.isSymlink() && dnode.isSymlink()) {
dList.add(new DiffReportEntry(DiffType.MODIFY, fullPath));
} else {
// must be the case: delete first and then create an inode with the
// same name
cList.add(new DiffReportEntry(DiffType.CREATE, fullPath));
dList.add(new DiffReportEntry(DiffType.DELETE, fullPath));
}
c++; c++;
d++; d++;
} else if (cnode.compareTo(dnode.getLocalNameBytes()) < 0) { } else if (cnode.compareTo(dnode.getLocalNameBytes()) < 0) {
fullPath[fullPath.length - 1] = cnode.getLocalNameBytes();
cList.add(new DiffReportEntry(fromEarlier ? DiffType.CREATE cList.add(new DiffReportEntry(fromEarlier ? DiffType.CREATE
: DiffType.DELETE, parent.getFullPathName() + Path.SEPARATOR : DiffType.DELETE, fullPath));
+ cnode.getLocalName()));
c++; c++;
} else { } else {
fullPath[fullPath.length - 1] = dnode.getLocalNameBytes();
dList.add(new DiffReportEntry(fromEarlier ? DiffType.DELETE dList.add(new DiffReportEntry(fromEarlier ? DiffType.DELETE
: DiffType.CREATE, parent.getFullPathName() + Path.SEPARATOR : DiffType.CREATE, fullPath));
+ dnode.getLocalName()));
d++; d++;
} }
} }
for (; d < deleted.size(); d++) { for (; d < deleted.size(); d++) {
fullPath[fullPath.length - 1] = deleted.get(d).getLocalNameBytes();
dList.add(new DiffReportEntry(fromEarlier ? DiffType.DELETE dList.add(new DiffReportEntry(fromEarlier ? DiffType.DELETE
: DiffType.CREATE, parent.getFullPathName() + Path.SEPARATOR : DiffType.CREATE, fullPath));
+ deleted.get(d).getLocalName()));
} }
for (; c < created.size(); c++) { for (; c < created.size(); c++) {
fullPath[fullPath.length - 1] = created.get(c).getLocalNameBytes();
cList.add(new DiffReportEntry(fromEarlier ? DiffType.CREATE cList.add(new DiffReportEntry(fromEarlier ? DiffType.CREATE
: DiffType.DELETE, parent.getFullPathName() + Path.SEPARATOR : DiffType.DELETE, fullPath));
+ created.get(c).getLocalName()));
} }
cList.addAll(dList); dList.addAll(cList);
cList.addAll(mList); return dList;
return cList;
} }
} }
@ -329,35 +338,27 @@ public class INodeDirectoryWithSnapshot extends INodeDirectoryWithQuota {
Snapshot toSnapshot, ChildrenDiff diff) { Snapshot toSnapshot, ChildrenDiff diff) {
Snapshot earlierSnapshot = fromSnapshot; Snapshot earlierSnapshot = fromSnapshot;
Snapshot laterSnapshot = toSnapshot; Snapshot laterSnapshot = toSnapshot;
if (fromSnapshot == null if (Snapshot.ID_COMPARATOR.compare(fromSnapshot, toSnapshot) > 0) {
|| (toSnapshot != null && Snapshot.ID_COMPARATOR.compare(fromSnapshot,
toSnapshot) > 0)) {
earlierSnapshot = toSnapshot; earlierSnapshot = toSnapshot;
laterSnapshot = fromSnapshot; laterSnapshot = fromSnapshot;
} }
boolean modified = diffs.changedBetweenSnapshots(earlierSnapshot,
laterSnapshot);
if (!modified) {
return false;
}
final List<DirectoryDiff> difflist = diffs.asList(); final List<DirectoryDiff> difflist = diffs.asList();
final int size = difflist.size(); final int size = difflist.size();
int earlierDiffIndex = Collections.binarySearch(difflist, earlierSnapshot); int earlierDiffIndex = Collections.binarySearch(difflist, earlierSnapshot);
if (earlierDiffIndex < 0 && (-earlierDiffIndex - 1) == size) { int laterDiffIndex = laterSnapshot == null ? size : Collections
// if the earlierSnapshot is after the latest SnapshotDiff stored in diffs, .binarySearch(difflist, laterSnapshot);
// no modification happened after the earlierSnapshot
return false;
}
int laterDiffIndex = size;
if (laterSnapshot != null) {
laterDiffIndex = Collections.binarySearch(difflist, laterSnapshot);
if (laterDiffIndex == -1 || laterDiffIndex == 0) {
// if the endSnapshot is the earliest SnapshotDiff stored in
// diffs, or before it, no modification happened before the endSnapshot
return false;
}
}
earlierDiffIndex = earlierDiffIndex < 0 ? (-earlierDiffIndex - 1) earlierDiffIndex = earlierDiffIndex < 0 ? (-earlierDiffIndex - 1)
: earlierDiffIndex; : earlierDiffIndex;
laterDiffIndex = laterDiffIndex < 0 ? (-laterDiffIndex - 1) laterDiffIndex = laterDiffIndex < 0 ? (-laterDiffIndex - 1)
: laterDiffIndex; : laterDiffIndex;
boolean dirMetadataChanged = false; boolean dirMetadataChanged = false;
INodeDirectory dirCopy = null; INodeDirectory dirCopy = null;
for (int i = earlierDiffIndex; i < laterDiffIndex; i++) { for (int i = earlierDiffIndex; i < laterDiffIndex; i++) {

View File

@ -33,15 +33,19 @@ import org.apache.hadoop.hdfs.util.ReadOnlyList;
/** Snapshot of a sub-tree in the namesystem. */ /** Snapshot of a sub-tree in the namesystem. */
@InterfaceAudience.Private @InterfaceAudience.Private
public class Snapshot implements Comparable<byte[]> { public class Snapshot implements Comparable<byte[]> {
/** Compare snapshot IDs with null <= s for any snapshot s. */ /**
* Compare snapshot IDs. Null indicates the current status thus is greater
* than non-null snapshots.
*/
public static final Comparator<Snapshot> ID_COMPARATOR public static final Comparator<Snapshot> ID_COMPARATOR
= new Comparator<Snapshot>() { = new Comparator<Snapshot>() {
@Override @Override
public int compare(Snapshot left, Snapshot right) { public int compare(Snapshot left, Snapshot right) {
// null means the current state, thus should be the largest
if (left == null) { if (left == null) {
return right == null? 0: -1; return right == null? 0: 1;
} else { } else {
return right == null? 1: left.id - right.id; return right == null? -1: left.id - right.id;
} }
} }
}; };
@ -52,7 +56,8 @@ public class Snapshot implements Comparable<byte[]> {
for(; inode != null; inode = inode.getParent()) { for(; inode != null; inode = inode.getParent()) {
if (inode instanceof INodeDirectorySnapshottable) { if (inode instanceof INodeDirectorySnapshottable) {
final Snapshot s = ((INodeDirectorySnapshottable)inode).getLastSnapshot(); final Snapshot s = ((INodeDirectorySnapshottable)inode).getLastSnapshot();
if (ID_COMPARATOR.compare(latest, s) < 0) { if (latest == null
|| (s != null && ID_COMPARATOR.compare(latest, s) < 0)) {
latest = s; latest = s;
} }
} }
@ -60,6 +65,15 @@ public class Snapshot implements Comparable<byte[]> {
return latest; return latest;
} }
/**
* Get the name of the given snapshot.
* @param s The given snapshot.
* @return The name of the snapshot, or an empty string if {@code s} is null
*/
public static String getSnapshotName(Snapshot s) {
return s != null ? s.getRoot().getLocalName() : "";
}
/** The root directory of the snapshot. */ /** The root directory of the snapshot. */
public class Root extends INodeDirectory { public class Root extends INodeDirectory {
Root(INodeDirectory other) { Root(INodeDirectory other) {

View File

@ -193,7 +193,7 @@ public class SnapshotFSImageFormat {
int deletedSize = in.readInt(); int deletedSize = in.readInt();
List<INode> deletedList = new ArrayList<INode>(deletedSize); List<INode> deletedList = new ArrayList<INode>(deletedSize);
for (int i = 0; i < deletedSize; i++) { for (int i = 0; i < deletedSize; i++) {
final INode deleted = loader.loadINodeWithLocalName(false, in); final INode deleted = loader.loadINodeWithLocalName(true, in);
deletedList.add(deleted); deletedList.add(deleted);
// set parent: the parent field of an INode in the deleted list is not // set parent: the parent field of an INode in the deleted list is not
// useful, but set the parent here to be consistent with the original // useful, but set the parent here to be consistent with the original

View File

@ -28,7 +28,6 @@ import org.apache.hadoop.hdfs.DFSUtil;
import org.apache.hadoop.hdfs.protocol.SnapshottableDirectoryStatus; import org.apache.hadoop.hdfs.protocol.SnapshottableDirectoryStatus;
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.BlocksMapUpdateInfo; 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.INodesInPath; import org.apache.hadoop.hdfs.server.namenode.INodeDirectory.INodesInPath;
@ -225,8 +224,9 @@ public class SnapshotManager implements SnapshotStats {
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.getNumSnapshots(), dir.getLocalNameBytes(), dir.getId(), dir.getNumSnapshots(),
dir.getSnapshotQuota(), dir.getParent() == null ? INode.EMPTY_BYTES dir.getSnapshotQuota(), dir.getParent() == null ?
: DFSUtil.string2Bytes(dir.getParent().getFullPathName())); DFSUtil.EMPTY_BYTES :
DFSUtil.string2Bytes(dir.getParent().getFullPathName()));
statusList.add(status); statusList.add(status);
} }
} }

View File

@ -234,7 +234,7 @@ message SnapshottableDirectoryListingProto {
* Snapshot diff report entry * Snapshot diff report entry
*/ */
message SnapshotDiffReportEntryProto { message SnapshotDiffReportEntryProto {
required string fullpath = 1; required bytes fullpath = 1;
required string modificationLabel = 2; required string modificationLabel = 2;
} }

View File

@ -185,8 +185,8 @@ public class TestNestedSnapshots {
Assert.assertEquals(0, Snapshot.ID_COMPARATOR.compare(null, null)); Assert.assertEquals(0, Snapshot.ID_COMPARATOR.compare(null, null));
for(Snapshot s : snapshots) { for(Snapshot s : snapshots) {
Assert.assertTrue(Snapshot.ID_COMPARATOR.compare(null, s) < 0); Assert.assertTrue(Snapshot.ID_COMPARATOR.compare(null, s) > 0);
Assert.assertTrue(Snapshot.ID_COMPARATOR.compare(s, null) > 0); Assert.assertTrue(Snapshot.ID_COMPARATOR.compare(s, null) < 0);
for(Snapshot t : snapshots) { for(Snapshot t : snapshots) {
final int expected = s.getRoot().getLocalName().compareTo( final int expected = s.getRoot().getLocalName().compareTo(

View File

@ -27,6 +27,7 @@ import java.util.HashMap;
import org.apache.hadoop.conf.Configuration; import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.fs.Path; import org.apache.hadoop.fs.Path;
import org.apache.hadoop.hdfs.DFSTestUtil; import org.apache.hadoop.hdfs.DFSTestUtil;
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.protocol.SnapshotDiffReport; import org.apache.hadoop.hdfs.protocol.SnapshotDiffReport;
@ -43,6 +44,7 @@ import org.junit.Test;
public class TestSnapshotDiffReport { public class TestSnapshotDiffReport {
protected static final long seed = 0; protected static final long seed = 0;
protected static final short REPLICATION = 3; protected static final short REPLICATION = 3;
protected static final short REPLICATION_1 = 2;
protected static final long BLOCKSIZE = 1024; protected static final long BLOCKSIZE = 1024;
public static final int SNAPSHOTNUMBER = 10; public static final int SNAPSHOTNUMBER = 10;
@ -92,14 +94,10 @@ public class TestSnapshotDiffReport {
Path file13 = new Path(modifyDir, "file13"); Path file13 = new Path(modifyDir, "file13");
Path file14 = new Path(modifyDir, "file14"); Path file14 = new Path(modifyDir, "file14");
Path file15 = new Path(modifyDir, "file15"); Path file15 = new Path(modifyDir, "file15");
DFSTestUtil.createFile(hdfs, file10, BLOCKSIZE, (short) (REPLICATION - 1), DFSTestUtil.createFile(hdfs, file10, BLOCKSIZE, REPLICATION_1, seed);
seed); DFSTestUtil.createFile(hdfs, file11, BLOCKSIZE, REPLICATION_1, seed);
DFSTestUtil.createFile(hdfs, file11, BLOCKSIZE, (short) (REPLICATION - 1), DFSTestUtil.createFile(hdfs, file12, BLOCKSIZE, REPLICATION_1, seed);
seed); DFSTestUtil.createFile(hdfs, file13, BLOCKSIZE, REPLICATION_1, seed);
DFSTestUtil.createFile(hdfs, file12, BLOCKSIZE, (short) (REPLICATION - 1),
seed);
DFSTestUtil.createFile(hdfs, file13, BLOCKSIZE, (short) (REPLICATION - 1),
seed);
// create snapshot // create snapshot
for (Path snapshotDir : snapshotDirs) { for (Path snapshotDir : snapshotDirs) {
hdfs.allowSnapshot(snapshotDir.toString()); hdfs.allowSnapshot(snapshotDir.toString());
@ -161,18 +159,17 @@ public class TestSnapshotDiffReport {
} else if (entry.getType() == DiffType.DELETE) { } else if (entry.getType() == DiffType.DELETE) {
assertTrue(report.getDiffList().contains(entry)); assertTrue(report.getDiffList().contains(entry));
assertTrue(inverseReport.getDiffList().contains( assertTrue(inverseReport.getDiffList().contains(
new DiffReportEntry(DiffType.CREATE, entry.getFullPath()))); new DiffReportEntry(DiffType.CREATE, entry.getRelativePath())));
} else if (entry.getType() == DiffType.CREATE) { } else if (entry.getType() == DiffType.CREATE) {
assertTrue(report.getDiffList().contains(entry)); assertTrue(report.getDiffList().contains(entry));
assertTrue(inverseReport.getDiffList().contains( assertTrue(inverseReport.getDiffList().contains(
new DiffReportEntry(DiffType.DELETE, entry.getFullPath()))); new DiffReportEntry(DiffType.DELETE, entry.getRelativePath())));
} }
} }
} }
/** Test the computation and representation of diff between snapshots */ /** Test the computation and representation of diff between snapshots */
// TODO: fix diff report @Test
// @Test
public void testDiffReport() throws Exception { public void testDiffReport() throws Exception {
Path subsub1 = new Path(sub1, "subsub1"); Path subsub1 = new Path(sub1, "subsub1");
Path subsubsub1 = new Path(subsub1, "subsubsub1"); Path subsubsub1 = new Path(subsub1, "subsubsub1");
@ -203,56 +200,94 @@ public class TestSnapshotDiffReport {
assertEquals(0, report.getDiffList().size()); assertEquals(0, report.getDiffList().size());
verifyDiffReport(sub1, "s0", "s2", verifyDiffReport(sub1, "s0", "s2",
new DiffReportEntry(DiffType.MODIFY, "/TestSnapshot/sub1"), new DiffReportEntry(DiffType.MODIFY, DFSUtil.string2Bytes("")),
new DiffReportEntry(DiffType.CREATE, "/TestSnapshot/sub1/file15"), new DiffReportEntry(DiffType.CREATE, DFSUtil.string2Bytes("file15")),
new DiffReportEntry(DiffType.DELETE, "/TestSnapshot/sub1/file12"), new DiffReportEntry(DiffType.DELETE, DFSUtil.string2Bytes("file12")),
new DiffReportEntry(DiffType.MODIFY, "/TestSnapshot/sub1/file11"), new DiffReportEntry(DiffType.DELETE, DFSUtil.string2Bytes("file11")),
new DiffReportEntry(DiffType.MODIFY, "/TestSnapshot/sub1/file13")); new DiffReportEntry(DiffType.CREATE, DFSUtil.string2Bytes("file11")),
new DiffReportEntry(DiffType.MODIFY, DFSUtil.string2Bytes("file13")));
verifyDiffReport(sub1, "s0", "s5", verifyDiffReport(sub1, "s0", "s5",
new DiffReportEntry(DiffType.MODIFY, "/TestSnapshot/sub1"), new DiffReportEntry(DiffType.MODIFY, DFSUtil.string2Bytes("")),
new DiffReportEntry(DiffType.CREATE, "/TestSnapshot/sub1/file15"), new DiffReportEntry(DiffType.CREATE, DFSUtil.string2Bytes("file15")),
new DiffReportEntry(DiffType.DELETE, "/TestSnapshot/sub1/file12"), new DiffReportEntry(DiffType.DELETE, DFSUtil.string2Bytes("file12")),
new DiffReportEntry(DiffType.MODIFY, "/TestSnapshot/sub1/file10"), new DiffReportEntry(DiffType.MODIFY, DFSUtil.string2Bytes("file10")),
new DiffReportEntry(DiffType.MODIFY, "/TestSnapshot/sub1/file11"), new DiffReportEntry(DiffType.DELETE, DFSUtil.string2Bytes("file11")),
new DiffReportEntry(DiffType.MODIFY, "/TestSnapshot/sub1/file13"), new DiffReportEntry(DiffType.CREATE, DFSUtil.string2Bytes("file11")),
new DiffReportEntry(DiffType.MODIFY, DFSUtil.string2Bytes("file13")),
new DiffReportEntry(DiffType.MODIFY, new DiffReportEntry(DiffType.MODIFY,
"/TestSnapshot/sub1/subsub1/subsubsub1"), DFSUtil.string2Bytes("subsub1/subsubsub1")),
new DiffReportEntry(DiffType.CREATE, new DiffReportEntry(DiffType.CREATE,
"/TestSnapshot/sub1/subsub1/subsubsub1/file10"), DFSUtil.string2Bytes("subsub1/subsubsub1/file10")),
new DiffReportEntry(DiffType.CREATE, new DiffReportEntry(DiffType.CREATE,
"/TestSnapshot/sub1/subsub1/subsubsub1/file11"), DFSUtil.string2Bytes("subsub1/subsubsub1/file11")),
new DiffReportEntry(DiffType.CREATE, new DiffReportEntry(DiffType.CREATE,
"/TestSnapshot/sub1/subsub1/subsubsub1/file13"), DFSUtil.string2Bytes("subsub1/subsubsub1/file13")),
new DiffReportEntry(DiffType.CREATE, new DiffReportEntry(DiffType.CREATE,
"/TestSnapshot/sub1/subsub1/subsubsub1/file15")); DFSUtil.string2Bytes("subsub1/subsubsub1/file15")));
verifyDiffReport(sub1, "s2", "s5", verifyDiffReport(sub1, "s2", "s5",
new DiffReportEntry(DiffType.MODIFY, "/TestSnapshot/sub1"), new DiffReportEntry(DiffType.MODIFY, DFSUtil.string2Bytes("file10")),
new DiffReportEntry(DiffType.MODIFY, "/TestSnapshot/sub1/file10"),
new DiffReportEntry(DiffType.MODIFY, new DiffReportEntry(DiffType.MODIFY,
"/TestSnapshot/sub1/subsub1/subsubsub1"), DFSUtil.string2Bytes("subsub1/subsubsub1")),
new DiffReportEntry(DiffType.CREATE, new DiffReportEntry(DiffType.CREATE,
"/TestSnapshot/sub1/subsub1/subsubsub1/file10"), DFSUtil.string2Bytes("subsub1/subsubsub1/file10")),
new DiffReportEntry(DiffType.CREATE, new DiffReportEntry(DiffType.CREATE,
"/TestSnapshot/sub1/subsub1/subsubsub1/file11"), DFSUtil.string2Bytes("subsub1/subsubsub1/file11")),
new DiffReportEntry(DiffType.CREATE, new DiffReportEntry(DiffType.CREATE,
"/TestSnapshot/sub1/subsub1/subsubsub1/file13"), DFSUtil.string2Bytes("subsub1/subsubsub1/file13")),
new DiffReportEntry(DiffType.CREATE, new DiffReportEntry(DiffType.CREATE,
"/TestSnapshot/sub1/subsub1/subsubsub1/file15")); DFSUtil.string2Bytes("subsub1/subsubsub1/file15")));
verifyDiffReport(sub1, "s3", "", verifyDiffReport(sub1, "s3", "",
new DiffReportEntry(DiffType.MODIFY, new DiffReportEntry(DiffType.MODIFY,
"/TestSnapshot/sub1/subsub1/subsubsub1"), DFSUtil.string2Bytes("subsub1/subsubsub1")),
new DiffReportEntry(DiffType.CREATE, new DiffReportEntry(DiffType.CREATE,
"/TestSnapshot/sub1/subsub1/subsubsub1/file15"), DFSUtil.string2Bytes("subsub1/subsubsub1/file15")),
new DiffReportEntry(DiffType.DELETE, new DiffReportEntry(DiffType.DELETE,
"/TestSnapshot/sub1/subsub1/subsubsub1/file12"), DFSUtil.string2Bytes("subsub1/subsubsub1/file12")),
new DiffReportEntry(DiffType.MODIFY, new DiffReportEntry(DiffType.MODIFY,
"/TestSnapshot/sub1/subsub1/subsubsub1/file10"), DFSUtil.string2Bytes("subsub1/subsubsub1/file10")),
new DiffReportEntry(DiffType.DELETE,
DFSUtil.string2Bytes("subsub1/subsubsub1/file11")),
new DiffReportEntry(DiffType.CREATE,
DFSUtil.string2Bytes("subsub1/subsubsub1/file11")),
new DiffReportEntry(DiffType.MODIFY, new DiffReportEntry(DiffType.MODIFY,
"/TestSnapshot/sub1/subsub1/subsubsub1/file11"), DFSUtil.string2Bytes("subsub1/subsubsub1/file13")));
new DiffReportEntry(DiffType.MODIFY,
"/TestSnapshot/sub1/subsub1/subsubsub1/file13"));
} }
/**
* Make changes under a sub-directory, then delete the sub-directory. Make
* sure the diff report computation correctly retrieve the diff from the
* deleted sub-directory.
*/
@Test
public void testDiffReport2() throws Exception {
Path subsub1 = new Path(sub1, "subsub1");
Path subsubsub1 = new Path(subsub1, "subsubsub1");
hdfs.mkdirs(subsubsub1);
modifyAndCreateSnapshot(subsubsub1, new Path[]{sub1});
// delete subsub1
hdfs.delete(subsub1, true);
// check diff report between s0 and s2
verifyDiffReport(sub1, "s0", "s2",
new DiffReportEntry(DiffType.MODIFY,
DFSUtil.string2Bytes("subsub1/subsubsub1")),
new DiffReportEntry(DiffType.CREATE,
DFSUtil.string2Bytes("subsub1/subsubsub1/file15")),
new DiffReportEntry(DiffType.DELETE,
DFSUtil.string2Bytes("subsub1/subsubsub1/file12")),
new DiffReportEntry(DiffType.DELETE,
DFSUtil.string2Bytes("subsub1/subsubsub1/file11")),
new DiffReportEntry(DiffType.CREATE,
DFSUtil.string2Bytes("subsub1/subsubsub1/file11")),
new DiffReportEntry(DiffType.MODIFY,
DFSUtil.string2Bytes("subsub1/subsubsub1/file13")));
// check diff report between s0 and the current status
verifyDiffReport(sub1, "s0", "",
new DiffReportEntry(DiffType.MODIFY, DFSUtil.string2Bytes("")),
new DiffReportEntry(DiffType.DELETE, DFSUtil.string2Bytes("subsub1")));
}
} }