diff --git a/hadoop-hdfs-project/hadoop-hdfs-httpfs/src/main/java/org/apache/hadoop/fs/http/server/FSOperations.java b/hadoop-hdfs-project/hadoop-hdfs-httpfs/src/main/java/org/apache/hadoop/fs/http/server/FSOperations.java index 7a8634eb92f..9471c0f0604 100644 --- a/hadoop-hdfs-project/hadoop-hdfs-httpfs/src/main/java/org/apache/hadoop/fs/http/server/FSOperations.java +++ b/hadoop-hdfs-project/hadoop-hdfs-httpfs/src/main/java/org/apache/hadoop/fs/http/server/FSOperations.java @@ -75,15 +75,17 @@ public class FSOperations { /** * @param fileStatuses list of FileStatus objects + * @param isFile is the fileStatuses from a file path * @return JSON map suitable for wire transport */ @SuppressWarnings({"unchecked"}) - private static Map toJson(FileStatus[] fileStatuses) { + private static Map toJson(FileStatus[] fileStatuses, + boolean isFile) { Map json = new LinkedHashMap<>(); Map inner = new LinkedHashMap<>(); JSONArray statuses = new JSONArray(); for (FileStatus f : fileStatuses) { - statuses.add(toJsonInner(f, false)); + statuses.add(toJsonInner(f, isFile)); } inner.put(HttpFSFileSystem.FILE_STATUS_JSON, statuses); json.put(HttpFSFileSystem.FILE_STATUSES_JSON, inner); @@ -126,13 +128,14 @@ public class FSOperations { * These two classes are slightly different, due to the impedance * mismatches between the WebHDFS and FileSystem APIs. * @param entries + * @param isFile is the entries from a file path * @return json */ private static Map toJson(FileSystem.DirectoryEntries - entries) { + entries, boolean isFile) { Map json = new LinkedHashMap<>(); Map inner = new LinkedHashMap<>(); - Map fileStatuses = toJson(entries.getEntries()); + Map fileStatuses = toJson(entries.getEntries(), isFile); inner.put(HttpFSFileSystem.PARTIAL_LISTING_JSON, fileStatuses); inner.put(HttpFSFileSystem.REMAINING_ENTRIES_JSON, entries.hasMore() ? 1 : 0); @@ -687,7 +690,7 @@ public class FSOperations { @Override public Map execute(FileSystem fs) throws IOException { FileStatus[] fileStatuses = fs.listStatus(path, filter); - return toJson(fileStatuses); + return toJson(fileStatuses, fs.getFileStatus(path).isFile()); } @Override @@ -732,7 +735,7 @@ public class FSOperations { WrappedFileSystem wrappedFS = new WrappedFileSystem(fs); FileSystem.DirectoryEntries entries = wrappedFS.listStatusBatch(path, token); - return toJson(entries); + return toJson(entries, wrappedFS.getFileStatus(path).isFile()); } } diff --git a/hadoop-hdfs-project/hadoop-hdfs-httpfs/src/test/java/org/apache/hadoop/fs/http/client/BaseTestHttpFSWith.java b/hadoop-hdfs-project/hadoop-hdfs-httpfs/src/test/java/org/apache/hadoop/fs/http/client/BaseTestHttpFSWith.java index 8dd0116c88f..d794143c6c3 100644 --- a/hadoop-hdfs-project/hadoop-hdfs-httpfs/src/test/java/org/apache/hadoop/fs/http/client/BaseTestHttpFSWith.java +++ b/hadoop-hdfs-project/hadoop-hdfs-httpfs/src/test/java/org/apache/hadoop/fs/http/client/BaseTestHttpFSWith.java @@ -363,8 +363,15 @@ public abstract class BaseTestHttpFSWith extends HFSTestCase { assertEquals(status2.getLen(), status1.getLen()); FileStatus[] stati = fs.listStatus(path.getParent()); - assertEquals(stati.length, 1); + assertEquals(1, stati.length); assertEquals(stati[0].getPath().getName(), path.getName()); + + // The full path should be the path to the file. See HDFS-12139 + FileStatus[] statl = fs.listStatus(path); + Assert.assertEquals(1, statl.length); + Assert.assertEquals(status2.getPath(), statl[0].getPath()); + Assert.assertEquals(statl[0].getPath().getName(), path.getName()); + Assert.assertEquals(stati[0].getPath(), statl[0].getPath()); } private static void assertSameListing(FileSystem expected, FileSystem @@ -410,6 +417,23 @@ public abstract class BaseTestHttpFSWith extends HFSTestCase { proxyFs.create(new Path(dir, "file" + i)).close(); assertSameListing(proxyFs, httpFs, dir); } + + // Test for HDFS-12139 + Path dir1 = new Path(getProxiedFSTestDir(), "dir1"); + proxyFs.mkdirs(dir1); + Path file1 = new Path(dir1, "file1"); + proxyFs.create(file1).close(); + + RemoteIterator si = proxyFs.listStatusIterator(dir1); + FileStatus statusl = si.next(); + FileStatus status = proxyFs.getFileStatus(file1); + Assert.assertEquals(file1.getName(), statusl.getPath().getName()); + Assert.assertEquals(status.getPath(), statusl.getPath()); + + si = proxyFs.listStatusIterator(file1); + statusl = si.next(); + Assert.assertEquals(file1.getName(), statusl.getPath().getName()); + Assert.assertEquals(status.getPath(), statusl.getPath()); } private void testWorkingdirectory() throws Exception {