HDFS-14633. The StorageType quota and consume in QuotaFeature is not handled for rename. Contributed by Jinglun.
(cherry picked from commit 62d71fbac3789c7d484bc76ced9ec7fa6ff94de1)
This commit is contained in:
parent
1c9255aebf
commit
c44e53e7a3
|
@ -78,7 +78,12 @@ class FSDirRenameOp {
|
|||
while(src.getINode(i) == dst.getINode(i)) { i++; }
|
||||
// src[i - 1] is the last common ancestor.
|
||||
BlockStoragePolicySuite bsps = fsd.getBlockStoragePolicySuite();
|
||||
final QuotaCounts delta = src.getLastINode().computeQuotaUsage(bsps);
|
||||
// Assume dstParent existence check done by callers.
|
||||
INode dstParent = dst.getINode(-2);
|
||||
// Use the destination parent's storage policy for quota delta verify.
|
||||
final QuotaCounts delta = src.getLastINode()
|
||||
.computeQuotaUsage(bsps, dstParent.getStoragePolicyID(), false,
|
||||
Snapshot.CURRENT_STATE_ID);
|
||||
|
||||
// Reduce the required quota by dst that is being removed
|
||||
final INode dstINode = dst.getLastINode();
|
||||
|
|
|
@ -17,6 +17,7 @@
|
|||
*/
|
||||
package org.apache.hadoop.hdfs.server.namenode;
|
||||
|
||||
import org.apache.hadoop.hdfs.server.namenode.snapshot.Snapshot;
|
||||
import org.apache.hadoop.util.StringUtils;
|
||||
|
||||
import com.google.common.annotations.VisibleForTesting;
|
||||
|
@ -1194,7 +1195,9 @@ public class FSDirectory implements Closeable {
|
|||
// always verify inode name
|
||||
verifyINodeName(inode.getLocalNameBytes());
|
||||
|
||||
final QuotaCounts counts = inode.computeQuotaUsage(getBlockStoragePolicySuite());
|
||||
final QuotaCounts counts = inode
|
||||
.computeQuotaUsage(getBlockStoragePolicySuite(),
|
||||
parent.getStoragePolicyID(), false, Snapshot.CURRENT_STATE_ID);
|
||||
updateCount(existing, pos, counts, checkQuota);
|
||||
|
||||
boolean isRename = (inode.getParent() != null);
|
||||
|
|
|
@ -43,7 +43,6 @@ import org.apache.hadoop.fs.FileSystem;
|
|||
import org.apache.hadoop.fs.Path;
|
||||
import org.apache.hadoop.fs.QuotaUsage;
|
||||
import org.apache.hadoop.fs.StorageType;
|
||||
import org.apache.hadoop.hdfs.client.HdfsClientConfigKeys;
|
||||
import org.apache.hadoop.hdfs.client.impl.LeaseRenewer;
|
||||
import org.apache.hadoop.hdfs.protocol.DSQuotaExceededException;
|
||||
import org.apache.hadoop.hdfs.protocol.HdfsConstants;
|
||||
|
@ -1540,6 +1539,44 @@ public class TestQuota {
|
|||
assertEquals(0, cluster.getNamesystem().getNumFilesUnderConstruction());
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testRename() throws Exception {
|
||||
int fileLen = 1024;
|
||||
short replication = 3;
|
||||
|
||||
final Path parent = new Path(PathUtils.getTestDir(getClass()).getPath(),
|
||||
GenericTestUtils.getMethodName());
|
||||
assertTrue(dfs.mkdirs(parent));
|
||||
|
||||
final Path srcDir = new Path(parent, "src-dir");
|
||||
Path file = new Path(srcDir, "file1");
|
||||
DFSTestUtil.createFile(dfs, file, fileLen, replication, 0);
|
||||
dfs.setStoragePolicy(srcDir, HdfsConstants.HOT_STORAGE_POLICY_NAME);
|
||||
|
||||
final Path dstDir = new Path(parent, "dst-dir");
|
||||
assertTrue(dfs.mkdirs(dstDir));
|
||||
dfs.setStoragePolicy(dstDir, HdfsConstants.ALLSSD_STORAGE_POLICY_NAME);
|
||||
|
||||
dfs.setQuota(srcDir, 100000, 100000);
|
||||
dfs.setQuota(dstDir, 100000, 100000);
|
||||
|
||||
Path dstFile = new Path(dstDir, "file1");
|
||||
// Test quota check of rename. Expect a QuotaExceedException.
|
||||
dfs.setQuotaByStorageType(dstDir, StorageType.SSD, 10);
|
||||
try {
|
||||
dfs.rename(file, dstFile);
|
||||
fail("Expect QuotaExceedException.");
|
||||
} catch (QuotaExceededException qe) {
|
||||
}
|
||||
|
||||
// Set enough quota, expect a successful rename.
|
||||
dfs.setQuotaByStorageType(dstDir, StorageType.SSD, fileLen * replication);
|
||||
dfs.rename(file, dstFile);
|
||||
// Verify the storage type usage is properly updated on source and dst.
|
||||
checkQuotaAndCount(dfs, srcDir);
|
||||
checkQuotaAndCount(dfs, dstDir);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testSpaceQuotaExceptionOnAppend() throws Exception {
|
||||
GenericTestUtils.setLogLevel(DFSOutputStream.LOG, Level.TRACE);
|
||||
|
@ -1610,4 +1647,15 @@ public class TestQuota {
|
|||
}
|
||||
scanner.close();
|
||||
}
|
||||
|
||||
// quota and count should match.
|
||||
private void checkQuotaAndCount(DistributedFileSystem fs, Path path)
|
||||
throws IOException {
|
||||
QuotaUsage qu = fs.getQuotaUsage(path);
|
||||
ContentSummary cs = fs.getContentSummary(path);
|
||||
for (StorageType st : StorageType.values()) {
|
||||
// it will fail here, because the quota and consume is not handled right.
|
||||
assertEquals(qu.getTypeConsumed(st), cs.getTypeConsumed(st));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue