HDFS-16531. Avoid setReplication writing an edit record if old replication equals the new value (#4148). Contributed by Stephen O'Donnell.
(cherry picked from commit dbeeee0363
)
This commit is contained in:
parent
c913dc3072
commit
8ae033d1a3
@ -49,6 +49,11 @@
|
|||||||
import static org.apache.hadoop.hdfs.DFSConfigKeys.DFS_QUOTA_BY_STORAGETYPE_ENABLED_KEY;
|
import static org.apache.hadoop.hdfs.DFSConfigKeys.DFS_QUOTA_BY_STORAGETYPE_ENABLED_KEY;
|
||||||
|
|
||||||
public class FSDirAttrOp {
|
public class FSDirAttrOp {
|
||||||
|
|
||||||
|
protected enum SetRepStatus {
|
||||||
|
UNCHANGED, INVALID, SUCCESS
|
||||||
|
}
|
||||||
|
|
||||||
static FileStatus setPermission(
|
static FileStatus setPermission(
|
||||||
FSDirectory fsd, FSPermissionChecker pc, final String src,
|
FSDirectory fsd, FSPermissionChecker pc, final String src,
|
||||||
FsPermission permission) throws IOException {
|
FsPermission permission) throws IOException {
|
||||||
@ -129,11 +134,11 @@ static FileStatus setTimes(
|
|||||||
return fsd.getAuditFileInfo(iip);
|
return fsd.getAuditFileInfo(iip);
|
||||||
}
|
}
|
||||||
|
|
||||||
static boolean setReplication(
|
static SetRepStatus setReplication(
|
||||||
FSDirectory fsd, FSPermissionChecker pc, BlockManager bm, String src,
|
FSDirectory fsd, FSPermissionChecker pc, BlockManager bm, String src,
|
||||||
final short replication) throws IOException {
|
final short replication) throws IOException {
|
||||||
bm.verifyReplication(src, replication, null);
|
bm.verifyReplication(src, replication, null);
|
||||||
final boolean isFile;
|
final SetRepStatus status;
|
||||||
fsd.writeLock();
|
fsd.writeLock();
|
||||||
try {
|
try {
|
||||||
final INodesInPath iip = fsd.resolvePath(pc, src, DirOp.WRITE);
|
final INodesInPath iip = fsd.resolvePath(pc, src, DirOp.WRITE);
|
||||||
@ -141,16 +146,14 @@ static boolean setReplication(
|
|||||||
fsd.checkPathAccess(pc, iip, FsAction.WRITE);
|
fsd.checkPathAccess(pc, iip, FsAction.WRITE);
|
||||||
}
|
}
|
||||||
|
|
||||||
final BlockInfo[] blocks = unprotectedSetReplication(fsd, iip,
|
status = unprotectedSetReplication(fsd, iip, replication);
|
||||||
replication);
|
if (status == SetRepStatus.SUCCESS) {
|
||||||
isFile = blocks != null;
|
|
||||||
if (isFile) {
|
|
||||||
fsd.getEditLog().logSetReplication(iip.getPath(), replication);
|
fsd.getEditLog().logSetReplication(iip.getPath(), replication);
|
||||||
}
|
}
|
||||||
} finally {
|
} finally {
|
||||||
fsd.writeUnlock();
|
fsd.writeUnlock();
|
||||||
}
|
}
|
||||||
return isFile;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
static FileStatus unsetStoragePolicy(FSDirectory fsd, FSPermissionChecker pc,
|
static FileStatus unsetStoragePolicy(FSDirectory fsd, FSPermissionChecker pc,
|
||||||
@ -374,7 +377,7 @@ static INodeDirectory unprotectedSetQuota(
|
|||||||
return dirNode;
|
return dirNode;
|
||||||
}
|
}
|
||||||
|
|
||||||
static BlockInfo[] unprotectedSetReplication(
|
static SetRepStatus unprotectedSetReplication(
|
||||||
FSDirectory fsd, INodesInPath iip, short replication)
|
FSDirectory fsd, INodesInPath iip, short replication)
|
||||||
throws QuotaExceededException, UnresolvedLinkException,
|
throws QuotaExceededException, UnresolvedLinkException,
|
||||||
SnapshotAccessControlException, UnsupportedActionException {
|
SnapshotAccessControlException, UnsupportedActionException {
|
||||||
@ -384,12 +387,20 @@ static BlockInfo[] unprotectedSetReplication(
|
|||||||
final INode inode = iip.getLastINode();
|
final INode inode = iip.getLastINode();
|
||||||
if (inode == null || !inode.isFile() || inode.asFile().isStriped()) {
|
if (inode == null || !inode.isFile() || inode.asFile().isStriped()) {
|
||||||
// TODO we do not support replication on stripe layout files yet
|
// TODO we do not support replication on stripe layout files yet
|
||||||
return null;
|
// We return invalid here, so we skip writing an edit, but also write an
|
||||||
|
// unsuccessful audit message.
|
||||||
|
return SetRepStatus.INVALID;
|
||||||
}
|
}
|
||||||
|
|
||||||
INodeFile file = inode.asFile();
|
INodeFile file = inode.asFile();
|
||||||
// Make sure the directory has sufficient quotas
|
// Make sure the directory has sufficient quotas
|
||||||
short oldBR = file.getPreferredBlockReplication();
|
short oldBR = file.getPreferredBlockReplication();
|
||||||
|
if (oldBR == replication) {
|
||||||
|
// No need to do anything as the requested rep factor is the same as
|
||||||
|
// existing. Returning UNCHANGED to we can skip writing edits, but still
|
||||||
|
// log a successful audit message.
|
||||||
|
return SetRepStatus.UNCHANGED;
|
||||||
|
}
|
||||||
|
|
||||||
long size = file.computeFileSize(true, true);
|
long size = file.computeFileSize(true, true);
|
||||||
// Ensure the quota does not exceed
|
// Ensure the quota does not exceed
|
||||||
@ -420,7 +431,7 @@ static BlockInfo[] unprotectedSetReplication(
|
|||||||
oldBR, iip.getPath());
|
oldBR, iip.getPath());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return file.getBlocks();
|
return SetRepStatus.SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void unprotectedSetStoragePolicy(FSDirectory fsd, BlockManager bm,
|
static void unprotectedSetStoragePolicy(FSDirectory fsd, BlockManager bm,
|
||||||
|
@ -2370,7 +2370,7 @@ void createSymlink(String target, String link,
|
|||||||
boolean setReplication(final String src, final short replication)
|
boolean setReplication(final String src, final short replication)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
final String operationName = "setReplication";
|
final String operationName = "setReplication";
|
||||||
boolean success = false;
|
FSDirAttrOp.SetRepStatus status;
|
||||||
checkOperation(OperationCategory.WRITE);
|
checkOperation(OperationCategory.WRITE);
|
||||||
final FSPermissionChecker pc = getPermissionChecker();
|
final FSPermissionChecker pc = getPermissionChecker();
|
||||||
FSPermissionChecker.setOperationType(operationName);
|
FSPermissionChecker.setOperationType(operationName);
|
||||||
@ -2379,7 +2379,7 @@ boolean setReplication(final String src, final short replication)
|
|||||||
try {
|
try {
|
||||||
checkOperation(OperationCategory.WRITE);
|
checkOperation(OperationCategory.WRITE);
|
||||||
checkNameNodeSafeMode("Cannot set replication for " + src);
|
checkNameNodeSafeMode("Cannot set replication for " + src);
|
||||||
success = FSDirAttrOp.setReplication(dir, pc, blockManager, src,
|
status = FSDirAttrOp.setReplication(dir, pc, blockManager, src,
|
||||||
replication);
|
replication);
|
||||||
} finally {
|
} finally {
|
||||||
writeUnlock(operationName);
|
writeUnlock(operationName);
|
||||||
@ -2388,11 +2388,12 @@ boolean setReplication(final String src, final short replication)
|
|||||||
logAuditEvent(false, operationName, src);
|
logAuditEvent(false, operationName, src);
|
||||||
throw e;
|
throw e;
|
||||||
}
|
}
|
||||||
if (success) {
|
if (status == FSDirAttrOp.SetRepStatus.SUCCESS) {
|
||||||
getEditLog().logSync();
|
getEditLog().logSync();
|
||||||
logAuditEvent(true, operationName, src);
|
|
||||||
}
|
}
|
||||||
return success;
|
logAuditEvent(status != FSDirAttrOp.SetRepStatus.INVALID,
|
||||||
|
operationName, src);
|
||||||
|
return status != FSDirAttrOp.SetRepStatus.INVALID;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -82,6 +82,12 @@ static void setrep(int fromREP, int toREP, boolean simulatedStorage) throws IOEx
|
|||||||
public void testSetrepIncreasing() throws IOException {
|
public void testSetrepIncreasing() throws IOException {
|
||||||
setrep(3, 7, false);
|
setrep(3, 7, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test(timeout=120000)
|
||||||
|
public void testSetrepSameRepValue() throws IOException {
|
||||||
|
setrep(3, 3, false);
|
||||||
|
}
|
||||||
|
|
||||||
@Test(timeout=120000)
|
@Test(timeout=120000)
|
||||||
public void testSetrepIncreasingSimulatedStorage() throws IOException {
|
public void testSetrepIncreasingSimulatedStorage() throws IOException {
|
||||||
setrep(3, 7, true);
|
setrep(3, 7, true);
|
||||||
|
Loading…
Reference in New Issue
Block a user