From 7f3ed5f61ed3640c50df52f699b992a889b8b28b Mon Sep 17 00:00:00 2001 From: Daryn Sharp Date: Tue, 26 Sep 2017 15:21:37 -0500 Subject: [PATCH] [HDFS-12386] Add fsserver defaults call to WebhdfsFileSystem. (Rushabh Shah via daryn) --- .../hadoop/hdfs/web/JsonUtilClient.java | 39 +++++ .../hadoop/hdfs/web/WebHdfsFileSystem.java | 17 +++ .../hadoop/hdfs/web/resources/GetOpParam.java | 3 +- .../hdfs/server/namenode/FSNamesystem.java | 3 +- .../web/resources/NamenodeWebHdfsMethods.java | 21 +++ .../org/apache/hadoop/hdfs/web/JsonUtil.java | 19 +++ .../apache/hadoop/hdfs/web/TestWebHDFS.java | 134 ++++++++++++++++++ 7 files changed, 234 insertions(+), 2 deletions(-) diff --git a/hadoop-hdfs-project/hadoop-hdfs-client/src/main/java/org/apache/hadoop/hdfs/web/JsonUtilClient.java b/hadoop-hdfs-project/hadoop-hdfs-client/src/main/java/org/apache/hadoop/hdfs/web/JsonUtilClient.java index 43bb17f733d..0320614af97 100644 --- a/hadoop-hdfs-project/hadoop-hdfs-client/src/main/java/org/apache/hadoop/hdfs/web/JsonUtilClient.java +++ b/hadoop-hdfs-project/hadoop-hdfs-client/src/main/java/org/apache/hadoop/hdfs/web/JsonUtilClient.java @@ -25,6 +25,7 @@ import org.apache.hadoop.fs.ContentSummary; import org.apache.hadoop.fs.ContentSummary.Builder; import org.apache.hadoop.fs.FileChecksum; import org.apache.hadoop.fs.FileStatus; +import org.apache.hadoop.fs.FsServerDefaults; import org.apache.hadoop.fs.MD5MD5CRC32CastagnoliFileChecksum; import org.apache.hadoop.fs.MD5MD5CRC32FileChecksum; import org.apache.hadoop.fs.MD5MD5CRC32GzipFileChecksum; @@ -66,6 +67,8 @@ import java.util.Map; class JsonUtilClient { static final DatanodeInfo[] EMPTY_DATANODE_INFO_ARRAY = {}; + static final String UNSUPPPORTED_EXCEPTION_STR = + UnsupportedOperationException.class.getName(); /** Convert a Json map to a RemoteException. */ static RemoteException toRemoteException(final Map json) { @@ -73,6 +76,9 @@ class JsonUtilClient { RemoteException.class.getSimpleName()); final String message = (String)m.get("message"); final String javaClassName = (String)m.get("javaClassName"); + if (UNSUPPPORTED_EXCEPTION_STR.equals(javaClassName)) { + throw new UnsupportedOperationException(message); + } return new RemoteException(javaClassName, message); } @@ -692,4 +698,37 @@ class JsonUtilClient { return array; } } + + /* + * The parameters which have default value -1 are required fields according + * to hdfs.proto. + * The default values for optional fields are taken from + * hdfs.proto#FsServerDefaultsProto. + */ + public static FsServerDefaults toFsServerDefaults(final Map json) { + if (json == null) { + return null; + } + Map m = + (Map) json.get(FsServerDefaults.class.getSimpleName()); + long blockSize = getLong(m, "blockSize", -1); + int bytesPerChecksum = getInt(m, "bytesPerChecksum", -1); + int writePacketSize = getInt(m, "writePacketSize", -1); + short replication = (short) getInt(m, "replication", -1); + int fileBufferSize = getInt(m, "fileBufferSize", -1); + boolean encryptDataTransfer = m.containsKey("encryptDataTransfer") + ? (Boolean) m.get("encryptDataTransfer") + : false; + long trashInterval = getLong(m, "trashInterval", 0); + DataChecksum.Type type = + DataChecksum.Type.valueOf(getInt(m, "checksumType", 1)); + String keyProviderUri = (String) m.get("keyProviderUri"); + byte storagepolicyId = m.containsKey("defaultStoragePolicyId") + ? ((Number) m.get("defaultStoragePolicyId")).byteValue() + : HdfsConstants.BLOCK_STORAGE_POLICY_ID_UNSPECIFIED; + return new FsServerDefaults(blockSize, bytesPerChecksum, + writePacketSize, replication, fileBufferSize, + encryptDataTransfer, trashInterval, type, keyProviderUri, + storagepolicyId); + } } diff --git a/hadoop-hdfs-project/hadoop-hdfs-client/src/main/java/org/apache/hadoop/hdfs/web/WebHdfsFileSystem.java b/hadoop-hdfs-project/hadoop-hdfs-client/src/main/java/org/apache/hadoop/hdfs/web/WebHdfsFileSystem.java index 98777c42d56..d595ccf9971 100644 --- a/hadoop-hdfs-project/hadoop-hdfs-client/src/main/java/org/apache/hadoop/hdfs/web/WebHdfsFileSystem.java +++ b/hadoop-hdfs-project/hadoop-hdfs-client/src/main/java/org/apache/hadoop/hdfs/web/WebHdfsFileSystem.java @@ -63,6 +63,7 @@ import org.apache.hadoop.fs.FSDataOutputStream; import org.apache.hadoop.fs.FSInputStream; import org.apache.hadoop.fs.FileStatus; import org.apache.hadoop.fs.FileSystem; +import org.apache.hadoop.fs.FsServerDefaults; import org.apache.hadoop.fs.GlobalStorageStatistics; import org.apache.hadoop.fs.GlobalStorageStatistics.StorageStatisticsProvider; import org.apache.hadoop.fs.StorageStatistics; @@ -1825,6 +1826,22 @@ public class WebHdfsFileSystem extends FileSystem new FsPathRunner(op, src).run(); } + /* + * Caller of this method should handle UnsupportedOperationException in case + * when new client is talking to old namenode that don't support + * FsServerDefaults call. + */ + @Override + public FsServerDefaults getServerDefaults() throws IOException { + final HttpOpParam.Op op = GetOpParam.Op.GETSERVERDEFAULTS; + return new FsPathResponseRunner(op, null) { + @Override + FsServerDefaults decodeResponse(Map json) throws IOException { + return JsonUtilClient.toFsServerDefaults(json); + } + }.run(); + } + @VisibleForTesting InetSocketAddress[] getResolvedNNAddr() { return nnAddrs; diff --git a/hadoop-hdfs-project/hadoop-hdfs-client/src/main/java/org/apache/hadoop/hdfs/web/resources/GetOpParam.java b/hadoop-hdfs-project/hadoop-hdfs-client/src/main/java/org/apache/hadoop/hdfs/web/resources/GetOpParam.java index d32af330ae7..f35c16bf06c 100644 --- a/hadoop-hdfs-project/hadoop-hdfs-client/src/main/java/org/apache/hadoop/hdfs/web/resources/GetOpParam.java +++ b/hadoop-hdfs-project/hadoop-hdfs-client/src/main/java/org/apache/hadoop/hdfs/web/resources/GetOpParam.java @@ -56,7 +56,8 @@ public class GetOpParam extends HttpOpParam { NULL(false, HttpURLConnection.HTTP_NOT_IMPLEMENTED), CHECKACCESS(false, HttpURLConnection.HTTP_OK), - LISTSTATUS_BATCH(false, HttpURLConnection.HTTP_OK); + LISTSTATUS_BATCH(false, HttpURLConnection.HTTP_OK), + GETSERVERDEFAULTS(false, HttpURLConnection.HTTP_OK); final boolean redirect; final int expectedHttpResponseCode; diff --git a/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/namenode/FSNamesystem.java b/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/namenode/FSNamesystem.java index 5b55d8269a8..e21da7f77f6 100644 --- a/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/namenode/FSNamesystem.java +++ b/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/namenode/FSNamesystem.java @@ -1747,7 +1747,8 @@ public class FSNamesystem implements Namesystem, FSNamesystemMBean, return sw.toString(); } - FsServerDefaults getServerDefaults() throws StandbyException { + @VisibleForTesting + public FsServerDefaults getServerDefaults() throws StandbyException { checkOperation(OperationCategory.READ); return serverDefaults; } diff --git a/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/namenode/web/resources/NamenodeWebHdfsMethods.java b/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/namenode/web/resources/NamenodeWebHdfsMethods.java index 8fe16192fdd..885eaffe8e3 100644 --- a/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/namenode/web/resources/NamenodeWebHdfsMethods.java +++ b/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/namenode/web/resources/NamenodeWebHdfsMethods.java @@ -58,6 +58,7 @@ import org.apache.hadoop.fs.BlockLocation; import org.apache.hadoop.fs.ContentSummary; import org.apache.hadoop.fs.FileStatus; import org.apache.hadoop.fs.FileSystem; +import org.apache.hadoop.fs.FsServerDefaults; import org.apache.hadoop.fs.Options; import org.apache.hadoop.fs.XAttr; import org.apache.hadoop.fs.permission.AclStatus; @@ -113,6 +114,7 @@ public class NamenodeWebHdfsMethods { private Principal userPrincipal; private String remoteAddr; + private static volatile String serverDefaultsResponse = null; private @Context ServletContext context; private @Context HttpServletResponse response; @@ -1091,11 +1093,30 @@ public class NamenodeWebHdfsMethods { final String js = JsonUtil.toJsonString(storagePolicy); return Response.ok(js).type(MediaType.APPLICATION_JSON).build(); } + case GETSERVERDEFAULTS: { + // Since none of the server defaults values are hot reloaded, we can + // cache the output of serverDefaults. + if (serverDefaultsResponse == null) { + FsServerDefaults serverDefaults = np.getServerDefaults(); + serverDefaultsResponse = JsonUtil.toJsonString(serverDefaults); + } + return Response.ok(serverDefaultsResponse) + .type(MediaType.APPLICATION_JSON).build(); + } default: throw new UnsupportedOperationException(op + " is not supported"); } } + /* + * This is used only and only for testing. + * Please don't use it otherwise. + */ + @VisibleForTesting + public static void resetServerDefaultsResponse() { + serverDefaultsResponse = null; + } + private static String getTrashRoot(String fullPath, Configuration conf) throws IOException { FileSystem fs = FileSystem.get(conf != null ? conf : new Configuration()); diff --git a/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/web/JsonUtil.java b/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/web/JsonUtil.java index a96e541f1cd..ae1291429e1 100644 --- a/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/web/JsonUtil.java +++ b/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/web/JsonUtil.java @@ -499,4 +499,23 @@ public class JsonUtil { m.put("topologyPaths", blockLocation.getTopologyPaths()); return m; } + + public static String toJsonString(FsServerDefaults serverDefaults) { + return toJsonString(FsServerDefaults.class, toJsonMap(serverDefaults)); + } + + private static Object toJsonMap(FsServerDefaults serverDefaults) { + final Map m = new HashMap(); + m.put("blockSize", serverDefaults.getBlockSize()); + m.put("bytesPerChecksum", serverDefaults.getBytesPerChecksum()); + m.put("writePacketSize", serverDefaults.getWritePacketSize()); + m.put("replication", serverDefaults.getReplication()); + m.put("fileBufferSize", serverDefaults.getFileBufferSize()); + m.put("encryptDataTransfer", serverDefaults.getEncryptDataTransfer()); + m.put("trashInterval", serverDefaults.getTrashInterval()); + m.put("checksumType", serverDefaults.getChecksumType().id); + m.put("keyProviderUri", serverDefaults.getKeyProviderUri()); + m.put("defaultStoragePolicyId", serverDefaults.getDefaultStoragePolicyId()); + return m; + } } diff --git a/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/web/TestWebHDFS.java b/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/web/TestWebHDFS.java index 62b3e98912d..588fd0b00e2 100644 --- a/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/web/TestWebHDFS.java +++ b/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/web/TestWebHDFS.java @@ -18,6 +18,14 @@ package org.apache.hadoop.hdfs.web; +import static org.apache.hadoop.fs.CommonConfigurationKeysPublic.FS_TRASH_INTERVAL_KEY; +import static org.apache.hadoop.fs.CommonConfigurationKeysPublic.IO_FILE_BUFFER_SIZE_KEY; +import static org.apache.hadoop.hdfs.DFSConfigKeys.DFS_BLOCK_SIZE_KEY; +import static org.apache.hadoop.hdfs.DFSConfigKeys.DFS_CHECKSUM_TYPE_KEY; +import static org.apache.hadoop.hdfs.DFSConfigKeys.DFS_ENCRYPT_DATA_TRANSFER_KEY; +import static org.apache.hadoop.hdfs.DFSConfigKeys.DFS_REPLICATION_KEY; +import static org.apache.hadoop.hdfs.client.HdfsClientConfigKeys.DFS_BYTES_PER_CHECKSUM_KEY; +import static org.apache.hadoop.hdfs.client.HdfsClientConfigKeys.DFS_CLIENT_WRITE_PACKET_SIZE_KEY; import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertFalse; import static org.junit.Assert.assertNotEquals; @@ -62,6 +70,7 @@ import org.apache.hadoop.fs.FSDataInputStream; import org.apache.hadoop.fs.FSDataOutputStream; import org.apache.hadoop.fs.FileStatus; import org.apache.hadoop.fs.FileSystem; +import org.apache.hadoop.fs.FsServerDefaults; import org.apache.hadoop.fs.Path; import org.apache.hadoop.fs.RemoteIterator; import org.apache.hadoop.fs.StorageType; @@ -85,7 +94,9 @@ import org.apache.hadoop.hdfs.protocol.HdfsConstants; import org.apache.hadoop.hdfs.protocol.LocatedBlock; import org.apache.hadoop.hdfs.protocol.LocatedBlocks; import org.apache.hadoop.hdfs.protocol.SnapshottableDirectoryStatus; +import org.apache.hadoop.hdfs.server.namenode.FSNamesystem; import org.apache.hadoop.hdfs.server.namenode.NameNode; +import org.apache.hadoop.hdfs.server.namenode.NameNodeAdapter; import org.apache.hadoop.hdfs.server.namenode.snapshot.SnapshotTestHelper; import org.apache.hadoop.hdfs.server.namenode.web.resources.NamenodeWebHdfsMethods; import org.apache.hadoop.hdfs.server.protocol.NamenodeProtocols; @@ -105,11 +116,13 @@ import org.apache.hadoop.security.AccessControlException; import org.apache.hadoop.security.token.SecretManager.InvalidToken; import org.apache.hadoop.security.UserGroupInformation; import org.apache.hadoop.test.GenericTestUtils; +import org.apache.hadoop.util.DataChecksum; import org.apache.log4j.Level; import org.codehaus.jettison.json.JSONException; import org.codehaus.jettison.json.JSONObject; import org.junit.Assert; import org.junit.Test; +import org.mockito.Mockito; import org.mockito.internal.util.reflection.Whitebox; import com.fasterxml.jackson.databind.ObjectMapper; @@ -1518,4 +1531,125 @@ public class TestWebHDFS { } } } + + /** + * Test fsserver defaults response from {@link DistributedFileSystem} and + * {@link WebHdfsFileSystem} are the same. + * @throws Exception + */ + @Test + public void testFsserverDefaults() throws Exception { + MiniDFSCluster cluster = null; + final Configuration conf = WebHdfsTestUtil.createConf(); + // Here we override all the default values so that we can verify that it + // doesn't pick up the default value. + long blockSize = 256*1024*1024; + int bytesPerChecksum = 256; + int writePacketSize = 128*1024; + int replicationFactor = 0; + int bufferSize = 1024; + boolean encryptDataTransfer = true; + long trashInterval = 1; + String checksumType = "CRC32"; + // Setting policy to a special value 7 because BlockManager will + // create defaultSuite with policy id 7. + byte policyId = (byte) 7; + + conf.setLong(DFS_BLOCK_SIZE_KEY, blockSize); + conf.setInt(DFS_BYTES_PER_CHECKSUM_KEY, bytesPerChecksum); + conf.setInt(DFS_CLIENT_WRITE_PACKET_SIZE_KEY, writePacketSize); + conf.setInt(DFS_REPLICATION_KEY, replicationFactor); + conf.setInt(IO_FILE_BUFFER_SIZE_KEY, bufferSize); + conf.setBoolean(DFS_ENCRYPT_DATA_TRANSFER_KEY, encryptDataTransfer); + conf.setLong(FS_TRASH_INTERVAL_KEY, trashInterval); + conf.set(DFS_CHECKSUM_TYPE_KEY, checksumType); + FsServerDefaults originalServerDefaults = new FsServerDefaults(blockSize, + bytesPerChecksum, writePacketSize, (short)replicationFactor, + bufferSize, encryptDataTransfer, trashInterval, + DataChecksum.Type.valueOf(checksumType), "", policyId); + try { + cluster = new MiniDFSCluster.Builder(conf).numDataNodes(0).build(); + final DistributedFileSystem dfs = cluster.getFileSystem(); + final WebHdfsFileSystem webfs = WebHdfsTestUtil.getWebHdfsFileSystem( + conf, WebHdfsConstants.WEBHDFS_SCHEME); + FsServerDefaults dfsServerDefaults = dfs.getServerDefaults(); + FsServerDefaults webfsServerDefaults = webfs.getServerDefaults(); + // Verify whether server defaults value that we override is equal to + // dfsServerDefaults. + compareFsServerDefaults(originalServerDefaults, dfsServerDefaults); + // Verify whether dfs serverdefaults is equal to + // webhdfsServerDefaults. + compareFsServerDefaults(dfsServerDefaults, webfsServerDefaults); + webfs.getServerDefaults(); + } finally { + if (cluster != null) { + cluster.shutdown(); + } + } + } + + private void compareFsServerDefaults(FsServerDefaults serverDefaults1, + FsServerDefaults serverDefaults2) throws Exception { + Assert.assertEquals("Block size is different", + serverDefaults1.getBlockSize(), + serverDefaults2.getBlockSize()); + Assert.assertEquals("Bytes per checksum are different", + serverDefaults1.getBytesPerChecksum(), + serverDefaults2.getBytesPerChecksum()); + Assert.assertEquals("Write packet size is different", + serverDefaults1.getWritePacketSize(), + serverDefaults2.getWritePacketSize()); + Assert.assertEquals("Default replication is different", + serverDefaults1.getReplication(), + serverDefaults2.getReplication()); + Assert.assertEquals("File buffer size are different", + serverDefaults1.getFileBufferSize(), + serverDefaults2.getFileBufferSize()); + Assert.assertEquals("Encrypt data transfer key is different", + serverDefaults1.getEncryptDataTransfer(), + serverDefaults2.getEncryptDataTransfer()); + Assert.assertEquals("Trash interval is different", + serverDefaults1.getTrashInterval(), + serverDefaults2.getTrashInterval()); + Assert.assertEquals("Checksum type is different", + serverDefaults1.getChecksumType(), + serverDefaults2.getChecksumType()); + Assert.assertEquals("Key provider uri is different", + serverDefaults1.getKeyProviderUri(), + serverDefaults2.getKeyProviderUri()); + Assert.assertEquals("Default storage policy is different", + serverDefaults1.getDefaultStoragePolicyId(), + serverDefaults2.getDefaultStoragePolicyId()); + } + + /** + * Tests the case when client is upgraded to return {@link FsServerDefaults} + * but then namenode is not upgraded. + * @throws Exception + */ + @Test + public void testFsserverDefaultsBackwardsCompatible() throws Exception { + MiniDFSCluster cluster = null; + final Configuration conf = WebHdfsTestUtil.createConf(); + try { + cluster = new MiniDFSCluster.Builder(conf).numDataNodes(0).build(); + final WebHdfsFileSystem webfs = WebHdfsTestUtil.getWebHdfsFileSystem( + conf, WebHdfsConstants.WEBHDFS_SCHEME); + NamenodeWebHdfsMethods.resetServerDefaultsResponse(); + FSNamesystem fsnSpy = + NameNodeAdapter.spyOnNamesystem(cluster.getNameNode()); + Mockito.when(fsnSpy.getServerDefaults()). + thenThrow(new UnsupportedOperationException()); + try { + webfs.getServerDefaults(); + Assert.fail("should have thrown UnSupportedOperationException."); + } catch (UnsupportedOperationException uoe) { + //Expected exception. + } + } finally { + if (cluster != null) { + cluster.shutdown(); + } + } + } }