HDFS-4149. Implement the disallowSnapshot(..) in FSNamesystem and add resetSnapshottable(..) to SnapshotManager.
git-svn-id: https://svn.apache.org/repos/asf/hadoop/common/branches/HDFS-2802@1405683 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
deaf979d41
commit
f84000900a
|
@ -2222,9 +2222,9 @@ public abstract class FileSystem extends Configured implements Closeable {
|
||||||
/**
|
/**
|
||||||
* Create a snapshot
|
* Create a snapshot
|
||||||
* @param snapshotName The name of the snapshot
|
* @param snapshotName The name of the snapshot
|
||||||
* @param snapshotRoot The directory where the snapshot will be taken
|
* @param path The directory where snapshots will be taken.
|
||||||
*/
|
*/
|
||||||
public void createSnapshot(String snapshotName, String snapshotRoot)
|
public void createSnapshot(String snapshotName, String path)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
throw new UnsupportedOperationException(getClass().getSimpleName()
|
throw new UnsupportedOperationException(getClass().getSimpleName()
|
||||||
+ " doesn't support createSnapshot");
|
+ " doesn't support createSnapshot");
|
||||||
|
|
|
@ -50,3 +50,6 @@ Branch-2802 Snapshot (Unreleased)
|
||||||
|
|
||||||
HDFS-4146. Use getter and setter in INodeFileWithLink to access blocks and
|
HDFS-4146. Use getter and setter in INodeFileWithLink to access blocks and
|
||||||
initialize root directory as snapshottable. (szetszwo)
|
initialize root directory as snapshottable. (szetszwo)
|
||||||
|
|
||||||
|
HDFS-4149. Implement the disallowSnapshot(..) in FSNamesystem and add
|
||||||
|
resetSnapshottable(..) to SnapshotManager. (szetszwo)
|
||||||
|
|
|
@ -872,28 +872,26 @@ public class DistributedFileSystem extends FileSystem {
|
||||||
/**
|
/**
|
||||||
* Allow snapshot on a directory.
|
* Allow snapshot on a directory.
|
||||||
*
|
*
|
||||||
* @param snapshotRoot the directory to be snapped
|
* @param path the directory to be taken snapshots
|
||||||
* @throws IOException
|
* @throws IOException
|
||||||
*/
|
*/
|
||||||
public void allowSnapshot(String snapshotRoot)
|
public void allowSnapshot(String path) throws IOException {
|
||||||
throws IOException {
|
dfs.allowSnapshot(path);
|
||||||
dfs.allowSnapshot(snapshotRoot);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Disallow snapshot on a directory.
|
* Disallow snapshot on a directory.
|
||||||
*
|
*
|
||||||
* @param snapshotRoot the directory to be snapped
|
* @param path the snapshottable directory.
|
||||||
* @throws IOException
|
* @throws IOException
|
||||||
*/
|
*/
|
||||||
public void disallowSnapshot(String snapshotRoot)
|
public void disallowSnapshot(String path) throws IOException {
|
||||||
throws IOException {
|
dfs.disallowSnapshot(path);
|
||||||
dfs.disallowSnapshot(snapshotRoot);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void createSnapshot(String snapshotName, String snapshotRoot)
|
public void createSnapshot(String snapshotName, String path)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
dfs.createSnapshot(snapshotName, snapshotRoot);
|
dfs.createSnapshot(snapshotName, path);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5543,22 +5543,38 @@ public class FSNamesystem implements Namesystem, FSClusterStats,
|
||||||
}
|
}
|
||||||
getEditLog().logSync();
|
getEditLog().logSync();
|
||||||
|
|
||||||
|
//TODO: need to update metrics in corresponding SnapshotManager method
|
||||||
|
|
||||||
if (auditLog.isInfoEnabled() && isExternalInvocation()) {
|
if (auditLog.isInfoEnabled() && isExternalInvocation()) {
|
||||||
logAuditEvent(UserGroupInformation.getCurrentUser(), getRemoteIp(),
|
logAuditEvent(UserGroupInformation.getCurrentUser(), getRemoteIp(),
|
||||||
"allowSnapshot", path, null, null);
|
"allowSnapshot", path, null, null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Disallow snapshot on a directory.
|
/** Disallow snapshot on a directory. */
|
||||||
@VisibleForTesting
|
public void disallowSnapshot(String path)
|
||||||
public void disallowSnapshot(String snapshotRoot)
|
|
||||||
throws SafeModeException, IOException {
|
throws SafeModeException, IOException {
|
||||||
// TODO: implement, also need to update metrics in corresponding
|
writeLock();
|
||||||
// SnapshotManager method
|
try {
|
||||||
|
checkOperation(OperationCategory.WRITE);
|
||||||
|
if (isInSafeMode()) {
|
||||||
|
throw new SafeModeException("Cannot disallow snapshot for " + path,
|
||||||
|
safeMode);
|
||||||
|
}
|
||||||
|
checkOwner(path);
|
||||||
|
|
||||||
|
snapshotManager.resetSnapshottable(path);
|
||||||
|
getEditLog().logDisallowSnapshot(path);
|
||||||
|
} finally {
|
||||||
|
writeUnlock();
|
||||||
|
}
|
||||||
|
getEditLog().logSync();
|
||||||
|
|
||||||
|
//TODO: need to update metrics in corresponding SnapshotManager method
|
||||||
|
|
||||||
if (auditLog.isInfoEnabled() && isExternalInvocation()) {
|
if (auditLog.isInfoEnabled() && isExternalInvocation()) {
|
||||||
logAuditEvent(UserGroupInformation.getCurrentUser(), getRemoteIp(),
|
logAuditEvent(UserGroupInformation.getCurrentUser(), getRemoteIp(),
|
||||||
"disallowSnapshot", snapshotRoot, null, null);
|
"disallowSnapshot", path, null, null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -22,6 +22,7 @@ import java.io.IOException;
|
||||||
import java.io.PrintWriter;
|
import java.io.PrintWriter;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
|
import java.util.Iterator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import org.apache.hadoop.fs.UnresolvedLinkException;
|
import org.apache.hadoop.fs.UnresolvedLinkException;
|
||||||
|
@ -30,8 +31,8 @@ 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.HdfsConstants;
|
import org.apache.hadoop.hdfs.protocol.HdfsConstants;
|
||||||
import org.apache.hadoop.hdfs.protocol.UnresolvedPathException;
|
import org.apache.hadoop.hdfs.protocol.UnresolvedPathException;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.INodeDirectoryWithSnapshot;
|
|
||||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.INodeDirectorySnapshottable;
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.INodeDirectorySnapshottable;
|
||||||
|
import org.apache.hadoop.hdfs.server.namenode.snapshot.INodeDirectoryWithSnapshot;
|
||||||
|
|
||||||
import com.google.common.annotations.VisibleForTesting;
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
|
|
||||||
|
@ -248,7 +249,7 @@ public class INodeDirectory extends INode {
|
||||||
}
|
}
|
||||||
// Resolve snapshot root
|
// Resolve snapshot root
|
||||||
curNode = ((INodeDirectorySnapshottable) parentDir)
|
curNode = ((INodeDirectorySnapshottable) parentDir)
|
||||||
.getSnapshotINode(components[count + 1]);
|
.getSnapshotRoot(components[count + 1]);
|
||||||
if (index >= -1) {
|
if (index >= -1) {
|
||||||
existing.snapshotRootIndex = existing.size;
|
existing.snapshotRootIndex = existing.size;
|
||||||
}
|
}
|
||||||
|
@ -601,20 +602,14 @@ public class INodeDirectory extends INode {
|
||||||
*/
|
*/
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
protected static void dumpTreeRecursively(PrintWriter out,
|
protected static void dumpTreeRecursively(PrintWriter out,
|
||||||
StringBuilder prefix, List<? extends INode> subs) {
|
StringBuilder prefix, Iterable<? extends INode> subs) {
|
||||||
prefix.append(DUMPTREE_EXCEPT_LAST_ITEM);
|
if (subs != null) {
|
||||||
if (subs != null && subs.size() != 0) {
|
for(final Iterator<? extends INode> i = subs.iterator(); i.hasNext();) {
|
||||||
int i = 0;
|
final INode inode = i.next();
|
||||||
for(; i < subs.size() - 1; i++) {
|
prefix.append(i.hasNext()? DUMPTREE_EXCEPT_LAST_ITEM: DUMPTREE_LAST_ITEM);
|
||||||
subs.get(i).dumpTreeRecursively(out, prefix);
|
inode.dumpTreeRecursively(out, prefix);
|
||||||
prefix.setLength(prefix.length() - 2);
|
|
||||||
prefix.append(DUMPTREE_EXCEPT_LAST_ITEM);
|
|
||||||
}
|
|
||||||
|
|
||||||
prefix.setLength(prefix.length() - 2);
|
|
||||||
prefix.append(DUMPTREE_LAST_ITEM);
|
|
||||||
subs.get(i).dumpTreeRecursively(out, prefix);
|
|
||||||
}
|
|
||||||
prefix.setLength(prefix.length() - 2);
|
prefix.setLength(prefix.length() - 2);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -21,6 +21,7 @@ import java.io.IOException;
|
||||||
import java.io.PrintWriter;
|
import java.io.PrintWriter;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
|
import java.util.Iterator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import org.apache.hadoop.HadoopIllegalArgumentException;
|
import org.apache.hadoop.HadoopIllegalArgumentException;
|
||||||
|
@ -30,7 +31,12 @@ import org.apache.hadoop.hdfs.server.namenode.INodeDirectory;
|
||||||
import org.apache.hadoop.hdfs.server.namenode.INodeDirectoryWithQuota;
|
import org.apache.hadoop.hdfs.server.namenode.INodeDirectoryWithQuota;
|
||||||
import org.apache.hadoop.util.Time;
|
import org.apache.hadoop.util.Time;
|
||||||
|
|
||||||
/** Directories where taking snapshots is allowed. */
|
/**
|
||||||
|
* Directories where taking snapshots is allowed.
|
||||||
|
*
|
||||||
|
* Like other {@link INode} subclasses, this class is synchronized externally
|
||||||
|
* by the namesystem and FSDirectory locks.
|
||||||
|
*/
|
||||||
@InterfaceAudience.Private
|
@InterfaceAudience.Private
|
||||||
public class INodeDirectorySnapshottable extends INodeDirectoryWithQuota {
|
public class INodeDirectorySnapshottable extends INodeDirectoryWithQuota {
|
||||||
static public INodeDirectorySnapshottable newInstance(
|
static public INodeDirectorySnapshottable newInstance(
|
||||||
|
@ -51,27 +57,16 @@ public class INodeDirectorySnapshottable extends INodeDirectoryWithQuota {
|
||||||
INode inode, String src) throws IOException {
|
INode inode, String src) throws IOException {
|
||||||
final INodeDirectory dir = INodeDirectory.valueOf(inode, src);
|
final INodeDirectory dir = INodeDirectory.valueOf(inode, src);
|
||||||
if (!dir.isSnapshottable()) {
|
if (!dir.isSnapshottable()) {
|
||||||
throw new SnapshotException(src + " is not a snapshottable directory.");
|
throw new SnapshotException(
|
||||||
|
"Directory is not a snapshottable directory: " + src);
|
||||||
}
|
}
|
||||||
return (INodeDirectorySnapshottable)dir;
|
return (INodeDirectorySnapshottable)dir;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** A list of snapshots of this directory. */
|
/** Snapshots of this directory in ascending order of snapshot id. */
|
||||||
private final List<INodeDirectoryWithSnapshot> snapshots
|
private final List<Snapshot> snapshots = new ArrayList<Snapshot>();
|
||||||
= new ArrayList<INodeDirectoryWithSnapshot>();
|
|
||||||
|
|
||||||
public INode getSnapshotINode(byte[] name) {
|
/** Number of snapshots allowed. */
|
||||||
if (snapshots == null || snapshots.size() == 0) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
int low = Collections.binarySearch(snapshots, name);
|
|
||||||
if (low >= 0) {
|
|
||||||
return snapshots.get(low);
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Number of snapshots is allowed. */
|
|
||||||
private int snapshotQuota;
|
private int snapshotQuota;
|
||||||
|
|
||||||
private INodeDirectorySnapshottable(long nsQuota, long dsQuota,
|
private INodeDirectorySnapshottable(long nsQuota, long dsQuota,
|
||||||
|
@ -80,14 +75,30 @@ public class INodeDirectorySnapshottable extends INodeDirectoryWithQuota {
|
||||||
setSnapshotQuota(snapshotQuota);
|
setSnapshotQuota(snapshotQuota);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int getNumSnapshots() {
|
||||||
|
return snapshots.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return the root directory of a snapshot. */
|
||||||
|
public INodeDirectory getSnapshotRoot(byte[] snapshotName) {
|
||||||
|
if (snapshots == null || snapshots.size() == 0) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
int low = Collections.binarySearch(snapshots, snapshotName);
|
||||||
|
if (low >= 0) {
|
||||||
|
return snapshots.get(low).getRoot();
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
public int getSnapshotQuota() {
|
public int getSnapshotQuota() {
|
||||||
return snapshotQuota;
|
return snapshotQuota;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setSnapshotQuota(int snapshotQuota) {
|
public void setSnapshotQuota(int snapshotQuota) {
|
||||||
if (snapshotQuota <= 0) {
|
if (snapshotQuota < 0) {
|
||||||
throw new HadoopIllegalArgumentException(
|
throw new HadoopIllegalArgumentException(
|
||||||
"Cannot set snapshot quota to " + snapshotQuota + " <= 0");
|
"Cannot set snapshot quota to " + snapshotQuota + " < 0");
|
||||||
}
|
}
|
||||||
this.snapshotQuota = snapshotQuota;
|
this.snapshotQuota = snapshotQuota;
|
||||||
}
|
}
|
||||||
|
@ -98,8 +109,7 @@ public class INodeDirectorySnapshottable extends INodeDirectoryWithQuota {
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Add a snapshot root under this directory. */
|
/** Add a snapshot root under this directory. */
|
||||||
INodeDirectoryWithSnapshot addSnapshotRoot(final String name
|
void addSnapshot(final Snapshot s) throws SnapshotException {
|
||||||
) throws SnapshotException {
|
|
||||||
//check snapshot quota
|
//check snapshot quota
|
||||||
if (snapshots.size() + 1 > snapshotQuota) {
|
if (snapshots.size() + 1 > snapshotQuota) {
|
||||||
throw new SnapshotException("Failed to add snapshot: there are already "
|
throw new SnapshotException("Failed to add snapshot: there are already "
|
||||||
|
@ -107,14 +117,12 @@ public class INodeDirectorySnapshottable extends INodeDirectoryWithQuota {
|
||||||
+ snapshotQuota);
|
+ snapshotQuota);
|
||||||
}
|
}
|
||||||
|
|
||||||
final INodeDirectoryWithSnapshot r = new INodeDirectoryWithSnapshot(name, this);
|
snapshots.add(s);
|
||||||
snapshots.add(r);
|
|
||||||
|
|
||||||
//set modification time
|
//set modification time
|
||||||
final long timestamp = Time.now();
|
final long timestamp = Time.now();
|
||||||
r.setModificationTime(timestamp);
|
s.getRoot().setModificationTime(timestamp);
|
||||||
setModificationTime(timestamp);
|
setModificationTime(timestamp);
|
||||||
return r;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -126,6 +134,28 @@ public class INodeDirectorySnapshottable extends INodeDirectoryWithQuota {
|
||||||
out.print(snapshots.size() <= 1 ? " snapshot of " : " snapshots of ");
|
out.print(snapshots.size() <= 1 ? " snapshot of " : " snapshots of ");
|
||||||
out.println(getLocalName());
|
out.println(getLocalName());
|
||||||
|
|
||||||
dumpTreeRecursively(out, prefix, snapshots);
|
dumpTreeRecursively(out, prefix, new Iterable<INodeDirectoryWithSnapshot>() {
|
||||||
|
@Override
|
||||||
|
public Iterator<INodeDirectoryWithSnapshot> iterator() {
|
||||||
|
return new Iterator<INodeDirectoryWithSnapshot>() {
|
||||||
|
final Iterator<Snapshot> i = snapshots.iterator();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean hasNext() {
|
||||||
|
return i.hasNext();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public INodeDirectoryWithSnapshot next() {
|
||||||
|
return i.next().getRoot();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void remove() {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,43 @@
|
||||||
|
/**
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. The ASF licenses this file
|
||||||
|
* to you under the Apache License, Version 2.0 (the
|
||||||
|
* "License"); you may not use this file except in compliance
|
||||||
|
* with the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.apache.hadoop.hdfs.server.namenode.snapshot;
|
||||||
|
|
||||||
|
import org.apache.hadoop.classification.InterfaceAudience;
|
||||||
|
|
||||||
|
/** Snapshot of a sub-tree in the namesystem. */
|
||||||
|
@InterfaceAudience.Private
|
||||||
|
public class Snapshot implements Comparable<byte[]> {
|
||||||
|
/** Snapshot ID. */
|
||||||
|
private final int id;
|
||||||
|
/** The root directory of the snapshot. */
|
||||||
|
private final INodeDirectoryWithSnapshot root;
|
||||||
|
|
||||||
|
Snapshot(int id, String name, INodeDirectorySnapshottable dir) {
|
||||||
|
this.id = id;
|
||||||
|
this.root = new INodeDirectoryWithSnapshot(name, dir);
|
||||||
|
}
|
||||||
|
|
||||||
|
INodeDirectoryWithSnapshot getRoot() {
|
||||||
|
return root;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int compareTo(byte[] bytes) {
|
||||||
|
return root.compareTo(bytes);
|
||||||
|
}
|
||||||
|
}
|
|
@ -20,7 +20,7 @@ package org.apache.hadoop.hdfs.server.namenode.snapshot;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.concurrent.atomic.AtomicLong;
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
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;
|
||||||
|
@ -34,8 +34,11 @@ import org.apache.hadoop.hdfs.server.namenode.INodeSymlink;
|
||||||
public class SnapshotManager implements SnapshotStats {
|
public class SnapshotManager implements SnapshotStats {
|
||||||
private final FSNamesystem namesystem;
|
private final FSNamesystem namesystem;
|
||||||
private final FSDirectory fsdir;
|
private final FSDirectory fsdir;
|
||||||
private AtomicLong numSnapshottableDirs = new AtomicLong();
|
|
||||||
private AtomicLong numSnapshots = new AtomicLong();
|
private final AtomicInteger numSnapshottableDirs = new AtomicInteger();
|
||||||
|
private final AtomicInteger numSnapshots = new AtomicInteger();
|
||||||
|
|
||||||
|
private int snapshotID = 0;
|
||||||
|
|
||||||
/** All snapshottable directories in the namesystem. */
|
/** All snapshottable directories in the namesystem. */
|
||||||
private final List<INodeDirectorySnapshottable> snapshottables
|
private final List<INodeDirectorySnapshottable> snapshottables
|
||||||
|
@ -49,17 +52,14 @@ public class SnapshotManager implements SnapshotStats {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set the given directory as a snapshottable directory.
|
* Set the given directory as a snapshottable directory.
|
||||||
* If the path is already a snapshottable directory, this is a no-op.
|
* If the path is already a snapshottable directory, update the quota.
|
||||||
* Otherwise, the {@link INodeDirectory} of the path is replaced by an
|
|
||||||
* {@link INodeDirectorySnapshottable}.
|
|
||||||
*/
|
*/
|
||||||
public void setSnapshottable(final String path, final int snapshotQuota
|
public void setSnapshottable(final String path, final int snapshotQuota
|
||||||
) throws IOException {
|
) throws IOException {
|
||||||
namesystem.writeLock();
|
|
||||||
try {
|
|
||||||
final INodeDirectory d = INodeDirectory.valueOf(fsdir.getINode(path), path);
|
final INodeDirectory d = INodeDirectory.valueOf(fsdir.getINode(path), path);
|
||||||
if (d.isSnapshottable()) {
|
if (d.isSnapshottable()) {
|
||||||
//The directory is already a snapshottable directory.
|
//The directory is already a snapshottable directory.
|
||||||
|
((INodeDirectorySnapshottable)d).setSnapshotQuota(snapshotQuota);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -67,12 +67,31 @@ public class SnapshotManager implements SnapshotStats {
|
||||||
= INodeDirectorySnapshottable.newInstance(d, snapshotQuota);
|
= INodeDirectorySnapshottable.newInstance(d, snapshotQuota);
|
||||||
fsdir.replaceINodeDirectory(path, d, s);
|
fsdir.replaceINodeDirectory(path, d, s);
|
||||||
snapshottables.add(s);
|
snapshottables.add(s);
|
||||||
} finally {
|
|
||||||
namesystem.writeUnlock();
|
|
||||||
}
|
|
||||||
numSnapshottableDirs.getAndIncrement();
|
numSnapshottableDirs.getAndIncrement();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the given snapshottable directory to non-snapshottable.
|
||||||
|
*
|
||||||
|
* @throws SnapshotException if there are snapshots in the directory.
|
||||||
|
*/
|
||||||
|
public void resetSnapshottable(final String path
|
||||||
|
) throws IOException {
|
||||||
|
final INodeDirectorySnapshottable s = INodeDirectorySnapshottable.valueOf(
|
||||||
|
fsdir.getINode(path), path);
|
||||||
|
if (s.getNumSnapshots() > 0) {
|
||||||
|
throw new SnapshotException("The directory " + path + " has snapshot(s). "
|
||||||
|
+ "Please redo the operation after removing all the snapshots.");
|
||||||
|
}
|
||||||
|
|
||||||
|
final INodeDirectory d = new INodeDirectory(s);
|
||||||
|
fsdir.replaceINodeDirectory(path, s, d);
|
||||||
|
snapshottables.remove(s);
|
||||||
|
|
||||||
|
numSnapshottableDirs.getAndDecrement();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a snapshot of the given path.
|
* Create a snapshot of the given path.
|
||||||
*
|
*
|
||||||
|
@ -81,7 +100,18 @@ public class SnapshotManager implements SnapshotStats {
|
||||||
*/
|
*/
|
||||||
public void createSnapshot(final String snapshotName, final String path
|
public void createSnapshot(final String snapshotName, final String path
|
||||||
) throws IOException {
|
) throws IOException {
|
||||||
new SnapshotCreation(path).run(snapshotName);
|
// Find the source root directory path where the snapshot is taken.
|
||||||
|
final INodeDirectorySnapshottable srcRoot
|
||||||
|
= INodeDirectorySnapshottable.valueOf(fsdir.getINode(path), path);
|
||||||
|
|
||||||
|
synchronized(this) {
|
||||||
|
final Snapshot s = new Snapshot(snapshotID, snapshotName, srcRoot);
|
||||||
|
srcRoot.addSnapshot(s);
|
||||||
|
new SnapshotCreation().processRecursively(srcRoot, s.getRoot());
|
||||||
|
|
||||||
|
//create success, update id
|
||||||
|
snapshotID++;
|
||||||
|
}
|
||||||
numSnapshots.getAndIncrement();
|
numSnapshots.getAndIncrement();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -92,22 +122,6 @@ public class SnapshotManager implements SnapshotStats {
|
||||||
* where N = # files + # directories + # symlinks.
|
* where N = # files + # directories + # symlinks.
|
||||||
*/
|
*/
|
||||||
class SnapshotCreation {
|
class SnapshotCreation {
|
||||||
/** The source root directory path where the snapshot is taken. */
|
|
||||||
final INodeDirectorySnapshottable srcRoot;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Constructor.
|
|
||||||
* @param path The path must be a snapshottable directory.
|
|
||||||
*/
|
|
||||||
private SnapshotCreation(final String path) throws IOException {
|
|
||||||
srcRoot = INodeDirectorySnapshottable.valueOf(fsdir.getINode(path), path);
|
|
||||||
}
|
|
||||||
|
|
||||||
void run(final String name) throws IOException {
|
|
||||||
final INodeDirectoryWithSnapshot root = srcRoot.addSnapshotRoot(name);
|
|
||||||
processRecursively(srcRoot, root);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Process snapshot creation recursively. */
|
/** Process snapshot creation recursively. */
|
||||||
private void processRecursively(final INodeDirectory srcDir,
|
private void processRecursively(final INodeDirectory srcDir,
|
||||||
final INodeDirectory dstDir) throws IOException {
|
final INodeDirectory dstDir) throws IOException {
|
||||||
|
|
|
@ -107,10 +107,12 @@ public class TestFSDirectory {
|
||||||
|
|
||||||
for(; (line = in.readLine()) != null; ) {
|
for(; (line = in.readLine()) != null; ) {
|
||||||
line = line.trim();
|
line = line.trim();
|
||||||
|
if (!line.contains("snapshot")) {
|
||||||
Assert.assertTrue(line.startsWith(INodeDirectory.DUMPTREE_LAST_ITEM)
|
Assert.assertTrue(line.startsWith(INodeDirectory.DUMPTREE_LAST_ITEM)
|
||||||
|| line.startsWith(INodeDirectory.DUMPTREE_EXCEPT_LAST_ITEM));
|
|| line.startsWith(INodeDirectory.DUMPTREE_EXCEPT_LAST_ITEM));
|
||||||
checkClassName(line);
|
checkClassName(line);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
LOG.info("Create a new file " + file4);
|
LOG.info("Create a new file " + file4);
|
||||||
DFSTestUtil.createFile(hdfs, file4, 1024, REPLICATION, seed);
|
DFSTestUtil.createFile(hdfs, file4, 1024, REPLICATION, seed);
|
||||||
|
@ -134,8 +136,7 @@ public class TestFSDirectory {
|
||||||
int i = line.lastIndexOf('(');
|
int i = line.lastIndexOf('(');
|
||||||
int j = line.lastIndexOf('@');
|
int j = line.lastIndexOf('@');
|
||||||
final String classname = line.substring(i+1, j);
|
final String classname = line.substring(i+1, j);
|
||||||
Assert.assertTrue(classname.equals(INodeFile.class.getSimpleName())
|
Assert.assertTrue(classname.startsWith(INodeFile.class.getSimpleName())
|
||||||
|| classname.equals(INodeDirectory.class.getSimpleName())
|
|| classname.startsWith(INodeDirectory.class.getSimpleName()));
|
||||||
|| classname.equals(INodeDirectoryWithQuota.class.getSimpleName()));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -89,10 +89,19 @@ public class TestSnapshotPathINodes {
|
||||||
|
|
||||||
// After a directory is snapshottable
|
// After a directory is snapshottable
|
||||||
hdfs.allowSnapshot(path);
|
hdfs.allowSnapshot(path);
|
||||||
|
{
|
||||||
final INode after = fsdir.getINode(path);
|
final INode after = fsdir.getINode(path);
|
||||||
Assert.assertTrue(after instanceof INodeDirectorySnapshottable);
|
Assert.assertTrue(after instanceof INodeDirectorySnapshottable);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
hdfs.disallowSnapshot(path);
|
||||||
|
{
|
||||||
|
final INode after = fsdir.getINode(path);
|
||||||
|
Assert.assertTrue(after instanceof INodeDirectory);
|
||||||
|
Assert.assertFalse(after instanceof INodeDirectorySnapshottable);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test {@link INodeDirectory#getExistingPathINodes(byte[][], int, boolean)}
|
* Test {@link INodeDirectory#getExistingPathINodes(byte[][], int, boolean)}
|
||||||
* for normal (non-snapshot) file.
|
* for normal (non-snapshot) file.
|
||||||
|
|
Loading…
Reference in New Issue