THRIFT-5563: fix deprecation and enable xlint for java library (#2575)
diff --git a/lib/java/gradle/sourceConfiguration.gradle b/lib/java/gradle/sourceConfiguration.gradle
index c510a40..b45fdc8 100644
--- a/lib/java/gradle/sourceConfiguration.gradle
+++ b/lib/java/gradle/sourceConfiguration.gradle
@@ -60,7 +60,16 @@
if (JavaVersion.current() > JavaVersion.VERSION_1_8) {
options.compilerArgs.addAll(['--release', '8'])
}
- // options.compilerArgs.addAll('-Xlint:unchecked')
+ options.compilerArgs.addAll([
+ '-Werror',
+ '-Xlint:deprecation',
+ '-Xlint:cast',
+ '-Xlint:empty',
+ '-Xlint:fallthrough',
+ '-Xlint:finally',
+ '-Xlint:overrides',
+ // we can't enable -Xlint:unchecked just yet
+ ])
}
// ----------------------------------------------------------------------------
diff --git a/lib/java/src/org/apache/thrift/partial/TFieldData.java b/lib/java/src/org/apache/thrift/partial/TFieldData.java
index 9ba1a17..d77302e 100644
--- a/lib/java/src/org/apache/thrift/partial/TFieldData.java
+++ b/lib/java/src/org/apache/thrift/partial/TFieldData.java
@@ -27,7 +27,7 @@
*/
public class TFieldData {
public static int encode(byte type) {
- return (int) (type & 0xff);
+ return type & 0xff;
}
public static int encode(byte type, short id) {
diff --git a/lib/java/src/org/apache/thrift/partial/ThriftMetadata.java b/lib/java/src/org/apache/thrift/partial/ThriftMetadata.java
index 1146720..46a37f2 100644
--- a/lib/java/src/org/apache/thrift/partial/ThriftMetadata.java
+++ b/lib/java/src/org/apache/thrift/partial/ThriftMetadata.java
@@ -25,23 +25,20 @@
import org.apache.thrift.TFieldRequirementType;
import org.apache.thrift.TUnion;
import org.apache.thrift.meta_data.FieldMetaData;
-import org.apache.thrift.meta_data.FieldValueMetaData;
import org.apache.thrift.meta_data.ListMetaData;
import org.apache.thrift.meta_data.MapMetaData;
import org.apache.thrift.meta_data.SetMetaData;
import org.apache.thrift.meta_data.StructMetaData;
-import org.apache.thrift.partial.Validate;
import org.apache.thrift.protocol.TType;
import java.io.Serializable;
-import java.lang.reflect.Field;
+import java.lang.reflect.Constructor;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
-import java.util.Set;
/**
* Container for Thrift metadata classes such as {@link ThriftPrimitive},
@@ -59,8 +56,8 @@
MAP_VALUE((short) 4, "mapValue"),
SET_ELEMENT((short) 5, "setElement");
- private short id;
- private String name;
+ private final short id;
+ private final String name;
FieldTypeEnum(short id, String name) {
this.id = id;
@@ -415,9 +412,9 @@
Iterable<ThriftField> fieldsData) {
if (isUnion(data)) {
- return new ThriftUnion(parent, fieldId, data, fieldsData);
+ return new ThriftUnion<>(parent, fieldId, data, fieldsData);
} else {
- return new ThriftStruct(parent, fieldId, data, fieldsData);
+ return new ThriftStruct<>(parent, fieldId, data, fieldsData);
}
}
}
@@ -463,18 +460,13 @@
}
public <T extends TBase> T createNewStruct() {
- T instance = null;
-
try {
Class<T> structClass = getStructClass(this.data);
- instance = (T) structClass.newInstance();
- } catch (InstantiationException e) {
- throw new RuntimeException(e);
- } catch (IllegalAccessException e) {
+ Constructor<T> declaredConstructor = structClass.getDeclaredConstructor();
+ return declaredConstructor.newInstance();
+ } catch (ReflectiveOperationException e) {
throw new RuntimeException(e);
}
-
- return instance;
}
public static <T extends TBase> ThriftStruct of(Class<T> clasz) {
@@ -494,7 +486,7 @@
Validate.checkNotNull(clasz, "clasz");
Validate.checkNotNull(fields, "fields");
- return new ThriftStruct(
+ return new ThriftStruct<>(
null,
FieldTypeEnum.ROOT,
new FieldMetaData(
@@ -519,7 +511,7 @@
if (this.fields.size() == 0) {
this.append(sb, "%s*;", indent2);
} else {
- List<Integer> ids = new ArrayList(this.fields.keySet());
+ List<Integer> ids = new ArrayList<>(this.fields.keySet());
Collections.sort(ids);
for (Integer id : ids) {
this.fields.get(id).toPrettyString(sb, level + 1);
@@ -535,7 +527,7 @@
Map<? extends TFieldIdEnum, FieldMetaData> fieldsMetaData =
FieldMetaData.getStructMetaDataMap(clasz);
- Map<Integer, ThriftObject> fields = new HashMap();
+ Map<Integer, ThriftObject> fields = new HashMap<>();
boolean getAllFields = !fieldsData.iterator().hasNext();
if (getAllFields) {
diff --git a/lib/java/src/org/apache/thrift/protocol/TCompactProtocol.java b/lib/java/src/org/apache/thrift/protocol/TCompactProtocol.java
index 832e197..3bace8e 100644
--- a/lib/java/src/org/apache/thrift/protocol/TCompactProtocol.java
+++ b/lib/java/src/org/apache/thrift/protocol/TCompactProtocol.java
@@ -117,7 +117,7 @@
* Used to keep track of the last field for the current and previous structs,
* so we can do the delta stuff.
*/
- private ShortStack lastField_ = new ShortStack(15);
+ private final ShortStack lastField_ = new ShortStack(15);
private short lastFieldId_ = 0;
@@ -617,7 +617,7 @@
*/
public boolean readBool() throws TException {
if (boolValue_ != null) {
- boolean result = boolValue_.booleanValue();
+ boolean result = boolValue_;
boolValue_ = null;
return result;
}
@@ -750,10 +750,15 @@
// These methods are here for the struct to call, but don't have any wire
// encoding.
//
+ @Override
public void readMessageEnd() throws TException {}
+ @Override
public void readFieldEnd() throws TException {}
+ @Override
public void readMapEnd() throws TException {}
+ @Override
public void readListEnd() throws TException {}
+ @Override
public void readSetEnd() throws TException {}
//
@@ -773,7 +778,7 @@
int off = 0;
while (true) {
byte b = buf[pos+off];
- result |= (int) (b & 0x7f) << shift;
+ result |= (b & 0x7f) << shift;
if ((b & 0x80) != 0x80) break;
shift += 7;
off++;
@@ -782,7 +787,7 @@
} else {
while (true) {
byte b = readByte();
- result |= (int) (b & 0x7f) << shift;
+ result |= (b & 0x7f) << shift;
if ((b & 0x80) != 0x80) break;
shift += 7;
}
diff --git a/lib/java/src/org/apache/thrift/transport/THttpClient.java b/lib/java/src/org/apache/thrift/transport/THttpClient.java
index 7d61b5c..5746822 100644
--- a/lib/java/src/org/apache/thrift/transport/THttpClient.java
+++ b/lib/java/src/org/apache/thrift/transport/THttpClient.java
@@ -21,11 +21,11 @@
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
-import java.io.InputStream;
import java.io.IOException;
-
-import java.net.URL;
+import java.io.InputStream;
import java.net.HttpURLConnection;
+import java.net.URL;
+import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
@@ -34,9 +34,9 @@
import org.apache.http.HttpResponse;
import org.apache.http.HttpStatus;
import org.apache.http.client.HttpClient;
+import org.apache.http.client.config.RequestConfig;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.entity.ByteArrayEntity;
-import org.apache.http.params.CoreConnectionPNames;
import org.apache.thrift.TConfiguration;
/**
@@ -68,7 +68,7 @@
public class THttpClient extends TEndpointTransport {
- private URL url_ = null;
+ private final URL url_;
private final ByteArrayOutputStream requestBuffer_ = new ByteArrayOutputStream();
@@ -84,6 +84,8 @@
private final HttpClient client;
+ private static final Map<String, String> DEFAULT_HEADERS = Collections.unmodifiableMap(getDefaultHeaders());
+
public static class Factory extends TTransportFactory {
private final String url;
@@ -159,35 +161,27 @@
public void setConnectTimeout(int timeout) {
connectTimeout_ = timeout;
- if (null != this.client) {
- // WARNING, this modifies the HttpClient params, this might have an impact elsewhere if the
- // same HttpClient is used for something else.
- client.getParams().setParameter(CoreConnectionPNames.CONNECTION_TIMEOUT, connectTimeout_);
- }
}
public void setReadTimeout(int timeout) {
readTimeout_ = timeout;
- if (null != this.client) {
- // WARNING, this modifies the HttpClient params, this might have an impact elsewhere if the
- // same HttpClient is used for something else.
- client.getParams().setParameter(CoreConnectionPNames.SO_TIMEOUT, readTimeout_);
- }
}
public void setCustomHeaders(Map<String,String> headers) {
- customHeaders_ = headers;
+ customHeaders_ = new HashMap<>(headers);
}
public void setCustomHeader(String key, String value) {
if (customHeaders_ == null) {
- customHeaders_ = new HashMap<String, String>();
+ customHeaders_ = new HashMap<>();
}
customHeaders_.put(key, value);
}
+ @Override
public void open() {}
+ @Override
public void close() {
if (null != inputStream_) {
try {
@@ -198,10 +192,12 @@
}
}
+ @Override
public boolean isOpen() {
return true;
}
+ @Override
public int read(byte[] buf, int off, int len) throws TTransportException {
if (inputStream_ == null) {
throw new TTransportException("Response buffer is empty, no request.");
@@ -222,10 +218,30 @@
}
}
+ @Override
public void write(byte[] buf, int off, int len) {
requestBuffer_.write(buf, off, len);
}
+ private RequestConfig getRequestConfig() {
+ RequestConfig requestConfig = RequestConfig.DEFAULT;
+ if (connectTimeout_ > 0) {
+ requestConfig = RequestConfig.copy(requestConfig).setConnectionRequestTimeout(connectTimeout_).build();
+ }
+ if (readTimeout_ > 0) {
+ requestConfig = RequestConfig.copy(requestConfig).setSocketTimeout(readTimeout_).build();
+ }
+ return requestConfig;
+ }
+
+ private static Map<String, String> getDefaultHeaders() {
+ Map<String, String> headers = new HashMap<>();
+ headers.put("Content-Type", "application/x-thrift");
+ headers.put("Accept", "application/x-thrift");
+ headers.put("User-Agent", "Java/THttpClient/HC");
+ return headers;
+ }
+
/**
* copy from org.apache.http.util.EntityUtils#consume. Android has it's own httpcore
* that doesn't have a consume.
@@ -243,7 +259,6 @@
}
private void flushUsingHttpClient() throws TTransportException {
-
if (null == this.client) {
throw new TTransportException("Null HttpClient, aborting.");
}
@@ -252,63 +267,36 @@
byte[] data = requestBuffer_.toByteArray();
requestBuffer_.reset();
- HttpPost post = null;
-
- InputStream is = null;
-
+ HttpPost post = new HttpPost(this.url_.getFile());
try {
// Set request to path + query string
- post = new HttpPost(this.url_.getFile());
-
- //
- // Headers are added to the HttpPost instance, not
- // to HttpClient.
- //
-
- post.setHeader("Content-Type", "application/x-thrift");
- post.setHeader("Accept", "application/x-thrift");
- post.setHeader("User-Agent", "Java/THttpClient/HC");
-
+ post.setConfig(getRequestConfig());
+ DEFAULT_HEADERS.forEach(post::addHeader);
if (null != customHeaders_) {
- for (Map.Entry<String, String> header : customHeaders_.entrySet()) {
- post.setHeader(header.getKey(), header.getValue());
- }
+ customHeaders_.forEach(post::addHeader);
}
-
post.setEntity(new ByteArrayEntity(data));
-
HttpResponse response = this.client.execute(this.host, post);
+ handleResponse(response);
+ } catch (IOException ioe) {
+ // Abort method so the connection gets released back to the connection manager
+ post.abort();
+ throw new TTransportException(ioe);
+ } finally {
+ resetConsumedMessageSize(-1);
+ post.releaseConnection();
+ }
+ }
+
+ private void handleResponse(HttpResponse response) throws TTransportException {
+ // Retrieve the InputStream BEFORE checking the status code so
+ // resources get freed in the with clause.
+ try (InputStream is = response.getEntity().getContent()) {
int responseCode = response.getStatusLine().getStatusCode();
-
- //
- // Retrieve the inputstream BEFORE checking the status code so
- // resources get freed in the finally clause.
- //
-
- is = response.getEntity().getContent();
-
if (responseCode != HttpStatus.SC_OK) {
throw new TTransportException("HTTP Response code: " + responseCode);
}
-
- // Read the responses into a byte array so we can release the connection
- // early. This implies that the whole content will have to be read in
- // memory, and that momentarily we might use up twice the memory (while the
- // thrift struct is being read up the chain).
- // Proceeding differently might lead to exhaustion of connections and thus
- // to app failure.
-
- byte[] buf = new byte[1024];
- ByteArrayOutputStream baos = new ByteArrayOutputStream();
-
- int len = 0;
- do {
- len = is.read(buf);
- if (len > 0) {
- baos.write(buf, 0, len);
- }
- } while (-1 != len);
-
+ byte[] readByteArray = readIntoByteArray(is);
try {
// Indicate we're done with the content.
consume(response.getEntity());
@@ -316,30 +304,37 @@
// We ignore this exception, it might only mean the server has no
// keep-alive capability.
}
-
- inputStream_ = new ByteArrayInputStream(baos.toByteArray());
+ inputStream_ = new ByteArrayInputStream(readByteArray);
} catch (IOException ioe) {
- // Abort method so the connection gets released back to the connection manager
- if (null != post) {
- post.abort();
- }
throw new TTransportException(ioe);
- } finally {
- resetConsumedMessageSize(-1);
- if (null != is) {
- // Close the entity's input stream, this will release the underlying connection
- try {
- is.close();
- } catch (IOException ioe) {
- throw new TTransportException(ioe);
- }
- }
- if (post != null) {
- post.releaseConnection();
- }
}
}
+ /**
+ * Read the responses into a byte array so we can release the connection
+ * early. This implies that the whole content will have to be read in
+ * memory, and that momentarily we might use up twice the memory (while the
+ * thrift struct is being read up the chain).
+ * Proceeding differently might lead to exhaustion of connections and thus
+ * to app failure.
+ *
+ * @param is input stream
+ * @return read bytes
+ * @throws IOException when exception during read
+ */
+ private static byte[] readIntoByteArray(InputStream is) throws IOException {
+ ByteArrayOutputStream baos = new ByteArrayOutputStream();
+ byte[] buf = new byte[1024];
+ int len;
+ do {
+ len = is.read(buf);
+ if (len > 0) {
+ baos.write(buf, 0, len);
+ }
+ } while (-1 != len);
+ return baos.toByteArray();
+ }
+
public void flush() throws TTransportException {
if (null != this.client) {
diff --git a/lib/java/test/org/apache/thrift/protocol/ProtocolTestBase.java b/lib/java/test/org/apache/thrift/protocol/ProtocolTestBase.java
index 52b1074..a873271 100644
--- a/lib/java/test/org/apache/thrift/protocol/ProtocolTestBase.java
+++ b/lib/java/test/org/apache/thrift/protocol/ProtocolTestBase.java
@@ -167,7 +167,7 @@
}
public void readMethod(TProtocol proto) throws TException {
- assertEquals((byte)b, proto.readByte());
+ assertEquals(b, proto.readByte());
}
});
}
diff --git a/lib/java/test/org/apache/thrift/test/JavaBeansTest.java b/lib/java/test/org/apache/thrift/test/JavaBeansTest.java
index 6a2a0ed..0bfcefe 100644
--- a/lib/java/test/org/apache/thrift/test/JavaBeansTest.java
+++ b/lib/java/test/org/apache/thrift/test/JavaBeansTest.java
@@ -61,10 +61,10 @@
// Everything is set
ooe.set_a_bite((byte) 1);
ooe.set_base64(ByteBuffer.wrap("bytes".getBytes()));
- ooe.set_byte_list(new LinkedList<Byte>());
+ ooe.set_byte_list(new LinkedList<>());
ooe.set_double_precision(1);
- ooe.set_i16_list(new LinkedList<Short>());
- ooe.set_i64_list(new LinkedList<Long>());
+ ooe.set_i16_list(new LinkedList<>());
+ ooe.set_i64_list(new LinkedList<>());
ooe.set_boolean_field(true);
ooe.set_integer16((short) 1);
ooe.set_integer32(1);
@@ -102,7 +102,7 @@
// Should throw exception when field doesn't exist
boolean exceptionThrown = false;
try{
- if (ooe.isSet(ooe.fieldForId(100)));
+ ooe.isSet(ooe.fieldForId(100));
} catch (IllegalArgumentException e){
exceptionThrown = true;
}