From c44e53e7a317968cce759e4f3aba2d3d68cff486 Mon Sep 17 00:00:00 2001 From: Xiaoyu Yao Date: Fri, 30 Aug 2019 16:46:04 -0700 Subject: [PATCH] HDFS-14633. The StorageType quota and consume in QuotaFeature is not handled for rename. Contributed by Jinglun. (cherry picked from commit 62d71fbac3789c7d484bc76ced9ec7fa6ff94de1) --- .../hdfs/server/namenode/FSDirRenameOp.java | 7 ++- .../hdfs/server/namenode/FSDirectory.java | 5 +- .../org/apache/hadoop/hdfs/TestQuota.java | 50 ++++++++++++++++++- 3 files changed, 59 insertions(+), 3 deletions(-) diff --git a/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/namenode/FSDirRenameOp.java b/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/namenode/FSDirRenameOp.java index bbbb7247ffd..10aefc2ee29 100644 --- a/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/namenode/FSDirRenameOp.java +++ b/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/namenode/FSDirRenameOp.java @@ -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(); diff --git a/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/namenode/FSDirectory.java b/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/namenode/FSDirectory.java index cf9f1b45ac1..cbce239fd57 100644 --- a/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/namenode/FSDirectory.java +++ b/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/namenode/FSDirectory.java @@ -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); diff --git a/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/TestQuota.java b/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/TestQuota.java index bf044e69aec..6d39a0e62ab 100644 --- a/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/TestQuota.java +++ b/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/TestQuota.java @@ -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)); + } + } }