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:
parent
d42d0860cb
commit
d9e2514d21
@ -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)
|
||||||
|
@ -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
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -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++;
|
||||||
}
|
}
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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
|
||||||
|
@ -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());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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++) {
|
||||||
|
@ -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) {
|
||||||
|
@ -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
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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(
|
||||||
|
@ -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")));
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
Loading…
x
Reference in New Issue
Block a user