diff --git a/hadoop-hdfs-project/hadoop-hdfs-client/src/main/java/org/apache/hadoop/hdfs/DistributedFileSystem.java b/hadoop-hdfs-project/hadoop-hdfs-client/src/main/java/org/apache/hadoop/hdfs/DistributedFileSystem.java index 1d5c5af8c3..f4fb621b29 100644 --- a/hadoop-hdfs-project/hadoop-hdfs-client/src/main/java/org/apache/hadoop/hdfs/DistributedFileSystem.java +++ b/hadoop-hdfs-project/hadoop-hdfs-client/src/main/java/org/apache/hadoop/hdfs/DistributedFileSystem.java @@ -2068,6 +2068,12 @@ public void disallowSnapshot(final Path path) throws IOException { new FileSystemLinkResolver() { @Override public Void doCall(final Path p) throws IOException { + String ssTrashRoot = + new Path(p, FileSystem.TRASH_PREFIX).toUri().getPath(); + if (dfs.exists(ssTrashRoot)) { + throw new IOException("Found trash root under path " + p + ". " + + "Please remove or move the trash root and then try again."); + } dfs.disallowSnapshot(getPathName(p)); return null; } diff --git a/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/TestDistributedFileSystem.java b/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/TestDistributedFileSystem.java index 6824fe62f2..37b2350a2b 100644 --- a/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/TestDistributedFileSystem.java +++ b/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/TestDistributedFileSystem.java @@ -2269,6 +2269,7 @@ public void testGetTrashRoots() throws IOException { assertEquals(trashRootsAfter2.size() + 1, trashRootsAfter3.size()); // Cleanup + dfs.delete(new Path(testDir, FileSystem.TRASH_PREFIX), true); dfs.disallowSnapshot(testDir); dfs.delete(testDir, true); } finally { @@ -2323,6 +2324,7 @@ public void testGetTrashRootsOnSnapshottableDirWithEZ() assertEquals(trashRoots, trashRootsAfter); // Cleanup + dfs.delete(new Path(testDir, FileSystem.TRASH_PREFIX), true); dfs.disallowSnapshot(testDir); dfs.delete(testDir, true); } finally { @@ -2452,4 +2454,34 @@ public void testGetTrashRootOnEZInSnapshottableDir() } } } + + @Test + public void testDisallowSnapshotShouldThrowWhenTrashRootExists() + throws Exception { + Configuration conf = getTestConfiguration(); + MiniDFSCluster cluster = + new MiniDFSCluster.Builder(conf).numDataNodes(1).build(); + try { + DistributedFileSystem dfs = cluster.getFileSystem(); + Path testDir = new Path("/disallowss/test1/"); + Path file0path = new Path(testDir, "file-0"); + dfs.create(file0path); + dfs.allowSnapshot(testDir); + // Create trash root manually + Path testDirTrashRoot = new Path(testDir, FileSystem.TRASH_PREFIX); + dfs.mkdirs(testDirTrashRoot); + // Try disallowing snapshot, should throw + LambdaTestUtils.intercept(IOException.class, + () -> dfs.disallowSnapshot(testDir)); + // Remove the trash root and try again, should pass this time + dfs.delete(testDirTrashRoot, true); + dfs.disallowSnapshot(testDir); + // Cleanup + dfs.delete(testDir, true); + } finally { + if (cluster != null) { + cluster.shutdown(); + } + } + } }