gson-comments/gson/src/test/java/com/google/gson/GsonTest.java

662 lines
23 KiB
Java

/*
* Copyright (C) 2016 The Gson Authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.google.gson;
import static com.google.common.truth.Truth.assertThat;
import static org.junit.Assert.assertThrows;
import com.google.gson.Gson.FutureTypeAdapter;
import com.google.gson.internal.Excluder;
import com.google.gson.reflect.TypeToken;
import com.google.gson.stream.JsonReader;
import com.google.gson.stream.JsonWriter;
import com.google.gson.stream.MalformedJsonException;
import java.io.IOException;
import java.io.StringReader;
import java.io.StringWriter;
import java.lang.reflect.Field;
import java.lang.reflect.Type;
import java.text.DateFormat;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicReference;
import org.junit.Test;
/**
* Unit tests for {@link Gson}.
*
* @author Ryan Harter
*/
public final class GsonTest {
private static final Excluder CUSTOM_EXCLUDER =
Excluder.DEFAULT.excludeFieldsWithoutExposeAnnotation().disableInnerClassSerialization();
private static final FieldNamingStrategy CUSTOM_FIELD_NAMING_STRATEGY =
new FieldNamingStrategy() {
@Override
public String translateName(Field f) {
return "foo";
}
};
private static final ToNumberStrategy CUSTOM_OBJECT_TO_NUMBER_STRATEGY = ToNumberPolicy.DOUBLE;
private static final ToNumberStrategy CUSTOM_NUMBER_TO_NUMBER_STRATEGY =
ToNumberPolicy.LAZILY_PARSED_NUMBER;
@Test
public void testStrictnessDefault() {
assertThat(new Gson().strictness).isEqualTo(Strictness.LENIENT);
}
@Test
public void testOverridesDefaultExcluder() {
Gson gson =
new Gson(
CUSTOM_EXCLUDER,
CUSTOM_FIELD_NAMING_STRATEGY,
new HashMap<Type, InstanceCreator<?>>(),
true,
false,
true,
false,
false,
FormattingStyle.PRETTY,
Strictness.LENIENT,
false,
false,
true,
LongSerializationPolicy.DEFAULT,
null,
DateFormat.DEFAULT,
DateFormat.DEFAULT,
new ArrayList<TypeAdapterFactory>(),
new ArrayList<TypeAdapterFactory>(),
new ArrayList<TypeAdapterFactory>(),
CUSTOM_OBJECT_TO_NUMBER_STRATEGY,
CUSTOM_NUMBER_TO_NUMBER_STRATEGY,
Collections.<ReflectionAccessFilter>emptyList());
assertThat(gson.excluder).isEqualTo(CUSTOM_EXCLUDER);
assertThat(gson.fieldNamingStrategy()).isEqualTo(CUSTOM_FIELD_NAMING_STRATEGY);
assertThat(gson.serializeNulls()).isTrue();
assertThat(gson.htmlSafe()).isFalse();
}
@Test
public void testClonedTypeAdapterFactoryListsAreIndependent() {
Gson original =
new Gson(
CUSTOM_EXCLUDER,
CUSTOM_FIELD_NAMING_STRATEGY,
new HashMap<Type, InstanceCreator<?>>(),
true,
false,
true,
false,
false,
FormattingStyle.PRETTY,
Strictness.LENIENT,
false,
false,
true,
LongSerializationPolicy.DEFAULT,
null,
DateFormat.DEFAULT,
DateFormat.DEFAULT,
new ArrayList<TypeAdapterFactory>(),
new ArrayList<TypeAdapterFactory>(),
new ArrayList<TypeAdapterFactory>(),
CUSTOM_OBJECT_TO_NUMBER_STRATEGY,
CUSTOM_NUMBER_TO_NUMBER_STRATEGY,
Collections.<ReflectionAccessFilter>emptyList());
Gson clone =
original.newBuilder().registerTypeAdapter(int.class, new TestTypeAdapter()).create();
assertThat(clone.factories).hasSize(original.factories.size() + 1);
}
private static final class TestTypeAdapter extends TypeAdapter<Object> {
@Override
public void write(JsonWriter out, Object value) {
// Test stub.
}
@Override
public Object read(JsonReader in) {
return null;
}
}
@Test
public void testGetAdapter_Null() {
Gson gson = new Gson();
NullPointerException e =
assertThrows(NullPointerException.class, () -> gson.getAdapter((TypeToken<?>) null));
assertThat(e).hasMessageThat().isEqualTo("type must not be null");
}
@Test
public void testGetAdapter_Concurrency() {
class DummyAdapter<T> extends TypeAdapter<T> {
@Override
public void write(JsonWriter out, T value) throws IOException {
throw new AssertionError("not needed for this test");
}
@Override
public T read(JsonReader in) throws IOException {
throw new AssertionError("not needed for this test");
}
}
final AtomicInteger adapterInstancesCreated = new AtomicInteger(0);
final AtomicReference<TypeAdapter<?>> threadAdapter = new AtomicReference<>();
final Class<?> requestedType = Number.class;
Gson gson =
new GsonBuilder()
.registerTypeAdapterFactory(
new TypeAdapterFactory() {
private volatile boolean isFirstCall = true;
@Override
public <T> TypeAdapter<T> create(final Gson gson, TypeToken<T> type) {
if (isFirstCall) {
isFirstCall = false;
// Create a separate thread which requests an adapter for the same type
// This will cause this factory to return a different adapter instance than
// the one it is currently creating
Thread thread =
new Thread() {
@Override
public void run() {
threadAdapter.set(gson.getAdapter(requestedType));
}
};
thread.start();
try {
thread.join();
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
}
// Create a new dummy adapter instance
adapterInstancesCreated.incrementAndGet();
return new DummyAdapter<>();
}
})
.create();
TypeAdapter<?> adapter = gson.getAdapter(requestedType);
assertThat(adapterInstancesCreated.get()).isEqualTo(2);
assertThat(adapter).isInstanceOf(DummyAdapter.class);
assertThat(threadAdapter.get()).isInstanceOf(DummyAdapter.class);
}
/**
* Verifies that two threads calling {@link Gson#getAdapter(TypeToken)} do not see the same
* unresolved {@link FutureTypeAdapter} instance, since that would not be thread-safe.
*
* <p>This test constructs the cyclic dependency {@literal CustomClass1 -> CustomClass2 ->
* CustomClass1} and lets one thread wait after the adapter for CustomClass2 has been obtained
* (which still refers to the nested unresolved FutureTypeAdapter for CustomClass1).
*/
@Test
public void testGetAdapter_FutureAdapterConcurrency() throws Exception {
/**
* Adapter which wraps another adapter. Can be imagined as a simplified version of the {@code
* ReflectiveTypeAdapterFactory$Adapter}.
*/
class WrappingAdapter<T> extends TypeAdapter<T> {
final TypeAdapter<?> wrapped;
boolean isFirstCall = true;
WrappingAdapter(TypeAdapter<?> wrapped) {
this.wrapped = wrapped;
}
@Override
public void write(JsonWriter out, T value) throws IOException {
// Due to how this test is set up there is infinite recursion, therefore
// need to track how deeply nested this call is
if (isFirstCall) {
isFirstCall = false;
out.beginArray();
wrapped.write(out, null);
out.endArray();
isFirstCall = true;
} else {
out.value("wrapped-nested");
}
}
@Override
public T read(JsonReader in) throws IOException {
throw new AssertionError("not needed for this test");
}
}
final CountDownLatch isThreadWaiting = new CountDownLatch(1);
final CountDownLatch canThreadProceed = new CountDownLatch(1);
final Gson gson =
new GsonBuilder()
.registerTypeAdapterFactory(
new TypeAdapterFactory() {
// volatile instead of AtomicBoolean is safe here because CountDownLatch prevents
// "true" concurrency
volatile boolean isFirstCaller = true;
@Override
public <T> TypeAdapter<T> create(Gson gson, TypeToken<T> type) {
Class<?> raw = type.getRawType();
if (raw == CustomClass1.class) {
// Retrieves a WrappingAdapter containing a nested FutureAdapter for
// CustomClass1
TypeAdapter<?> adapter = gson.getAdapter(CustomClass2.class);
// Let thread wait so the FutureAdapter for CustomClass1 nested in the adapter
// for CustomClass2 is not resolved yet
if (isFirstCaller) {
isFirstCaller = false;
isThreadWaiting.countDown();
try {
canThreadProceed.await();
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
}
return new WrappingAdapter<>(adapter);
} else if (raw == CustomClass2.class) {
TypeAdapter<?> adapter = gson.getAdapter(CustomClass1.class);
assertThat(adapter).isInstanceOf(FutureTypeAdapter.class);
return new WrappingAdapter<>(adapter);
} else {
throw new AssertionError("Adapter for unexpected type requested: " + raw);
}
}
})
.create();
final AtomicReference<TypeAdapter<?>> otherThreadAdapter = new AtomicReference<>();
Thread thread =
new Thread() {
@Override
public void run() {
otherThreadAdapter.set(gson.getAdapter(CustomClass1.class));
}
};
thread.start();
// Wait until other thread has obtained FutureAdapter
isThreadWaiting.await();
TypeAdapter<?> adapter = gson.getAdapter(CustomClass1.class);
// Should not fail due to referring to unresolved FutureTypeAdapter
assertThat(adapter.toJson(null)).isEqualTo("[[\"wrapped-nested\"]]");
// Let other thread proceed and have it resolve its FutureTypeAdapter
canThreadProceed.countDown();
thread.join();
assertThat(otherThreadAdapter.get().toJson(null)).isEqualTo("[[\"wrapped-nested\"]]");
}
@Test
public void testGetDelegateAdapter() {
class DummyAdapter extends TypeAdapter<Number> {
private final int number;
DummyAdapter(int number) {
this.number = number;
}
@Override
public Number read(JsonReader in) throws IOException {
throw new AssertionError("not needed for test");
}
@Override
public void write(JsonWriter out, Number value) throws IOException {
throw new AssertionError("not needed for test");
}
// Override toString() for better assertion error messages
@Override
public String toString() {
return "adapter-" + number;
}
}
class DummyFactory implements TypeAdapterFactory {
private final DummyAdapter adapter;
DummyFactory(DummyAdapter adapter) {
this.adapter = adapter;
}
@SuppressWarnings("unchecked")
@Override
public <T> TypeAdapter<T> create(Gson gson, TypeToken<T> type) {
return (TypeAdapter<T>) adapter;
}
// Override equals to verify that reference equality check is performed by Gson,
// and this method is ignored
@Override
public boolean equals(Object obj) {
return obj instanceof DummyFactory && ((DummyFactory) obj).adapter.equals(adapter);
}
@Override
public int hashCode() {
return adapter.hashCode();
}
}
DummyAdapter adapter1 = new DummyAdapter(1);
DummyFactory factory1 = new DummyFactory(adapter1);
DummyAdapter adapter2 = new DummyAdapter(2);
DummyFactory factory2 = new DummyFactory(adapter2);
Gson gson =
new GsonBuilder()
// Note: This is 'last in, first out' order; Gson will first use factory2, then factory1
.registerTypeAdapterFactory(factory1)
.registerTypeAdapterFactory(factory2)
.create();
TypeToken<?> type = TypeToken.get(Number.class);
assertThrows(NullPointerException.class, () -> gson.getDelegateAdapter(null, type));
assertThrows(NullPointerException.class, () -> gson.getDelegateAdapter(factory1, null));
// For unknown factory the first adapter for that type should be returned
assertThat(gson.getDelegateAdapter(new DummyFactory(new DummyAdapter(0)), type))
.isEqualTo(adapter2);
assertThat(gson.getDelegateAdapter(factory2, type)).isEqualTo(adapter1);
// Default Gson adapter should be returned
assertThat(gson.getDelegateAdapter(factory1, type)).isNotInstanceOf(DummyAdapter.class);
DummyFactory factory1Eq = new DummyFactory(adapter1);
// Verify that test setup is correct
assertThat(factory1.equals(factory1Eq)).isTrue();
// Should only consider reference equality and ignore that custom `equals` method considers
// factories to be equal, therefore returning `adapter2` which came from `factory2` instead
// of skipping past `factory1`
assertThat(gson.getDelegateAdapter(factory1Eq, type)).isEqualTo(adapter2);
}
@Test
public void testNewJsonWriter_Default() throws IOException {
StringWriter writer = new StringWriter();
JsonWriter jsonWriter = new GsonBuilder().create().newJsonWriter(writer);
jsonWriter.beginObject();
jsonWriter.name("test");
jsonWriter.nullValue();
jsonWriter.name("<test2");
jsonWriter.value(true);
jsonWriter.endObject();
// Additional top-level value
IllegalStateException e = assertThrows(IllegalStateException.class, () -> jsonWriter.value(1));
assertThat(e).hasMessageThat().isEqualTo("JSON must have only one top-level value.");
jsonWriter.close();
assertThat(writer.toString()).isEqualTo("{\"\\u003ctest2\":true}");
}
@SuppressWarnings({"deprecation", "InlineMeInliner"}) // for GsonBuilder.setLenient
@Test
public void testNewJsonWriter_Custom() throws IOException {
StringWriter writer = new StringWriter();
JsonWriter jsonWriter =
new GsonBuilder()
.disableHtmlEscaping()
.generateNonExecutableJson()
.setPrettyPrinting()
.serializeNulls()
.setLenient()
.create()
.newJsonWriter(writer);
jsonWriter.beginObject();
jsonWriter.name("test");
jsonWriter.nullValue();
jsonWriter.name("<test2");
jsonWriter.value(true);
jsonWriter.endObject();
// Additional top-level value
jsonWriter.value(1);
jsonWriter.close();
assertThat(writer.toString()).isEqualTo(")]}'\n{\n \"test\": null,\n \"<test2\": true\n}1");
}
@Test
public void testNewJsonReader_Default() throws IOException {
String json = "test"; // String without quotes
JsonReader jsonReader = new GsonBuilder().create().newJsonReader(new StringReader(json));
assertThrows(MalformedJsonException.class, jsonReader::nextString);
jsonReader.close();
}
@SuppressWarnings({"deprecation", "InlineMeInliner"}) // for GsonBuilder.setLenient
@Test
public void testNewJsonReader_Custom() throws IOException {
String json = "test"; // String without quotes
JsonReader jsonReader =
new GsonBuilder().setLenient().create().newJsonReader(new StringReader(json));
assertThat(jsonReader.nextString()).isEqualTo("test");
jsonReader.close();
}
/**
* Modifying a GsonBuilder obtained from {@link Gson#newBuilder()} of a {@code new Gson()} should
* not affect the Gson instance it came from.
*/
@Test
public void testDefaultGsonNewBuilderModification() {
Gson gson = new Gson();
GsonBuilder gsonBuilder = gson.newBuilder();
// Modifications of `gsonBuilder` should not affect `gson` object
gsonBuilder.registerTypeAdapter(
CustomClass1.class,
new TypeAdapter<CustomClass1>() {
@Override
public CustomClass1 read(JsonReader in) throws IOException {
throw new UnsupportedOperationException();
}
@Override
public void write(JsonWriter out, CustomClass1 value) throws IOException {
out.value("custom-adapter");
}
});
gsonBuilder.registerTypeHierarchyAdapter(
CustomClass2.class,
new JsonSerializer<CustomClass2>() {
@Override
public JsonElement serialize(
CustomClass2 src, Type typeOfSrc, JsonSerializationContext context) {
return new JsonPrimitive("custom-hierarchy-adapter");
}
});
gsonBuilder.registerTypeAdapter(
CustomClass3.class,
new InstanceCreator<CustomClass3>() {
@Override
public CustomClass3 createInstance(Type type) {
return new CustomClass3("custom-instance");
}
});
assertDefaultGson(gson);
// New GsonBuilder created from `gson` should not have been affected by changes either
assertDefaultGson(gson.newBuilder().create());
// But new Gson instance from `gsonBuilder` should use custom adapters
assertCustomGson(gsonBuilder.create());
}
private static void assertDefaultGson(Gson gson) {
// Should use default reflective adapter
String json1 = gson.toJson(new CustomClass1());
assertThat(json1).isEqualTo("{}");
// Should use default reflective adapter
String json2 = gson.toJson(new CustomClass2());
assertThat(json2).isEqualTo("{}");
// Should use default instance creator
CustomClass3 customClass3 = gson.fromJson("{}", CustomClass3.class);
assertThat(customClass3.s).isEqualTo(CustomClass3.NO_ARG_CONSTRUCTOR_VALUE);
}
/**
* Modifying a GsonBuilder obtained from {@link Gson#newBuilder()} of a custom Gson instance
* (created using a GsonBuilder) should not affect the Gson instance it came from.
*/
@Test
public void testNewBuilderModification() {
Gson gson =
new GsonBuilder()
.registerTypeAdapter(
CustomClass1.class,
new TypeAdapter<CustomClass1>() {
@Override
public CustomClass1 read(JsonReader in) throws IOException {
throw new UnsupportedOperationException();
}
@Override
public void write(JsonWriter out, CustomClass1 value) throws IOException {
out.value("custom-adapter");
}
})
.registerTypeHierarchyAdapter(
CustomClass2.class,
new JsonSerializer<CustomClass2>() {
@Override
public JsonElement serialize(
CustomClass2 src, Type typeOfSrc, JsonSerializationContext context) {
return new JsonPrimitive("custom-hierarchy-adapter");
}
})
.registerTypeAdapter(
CustomClass3.class,
new InstanceCreator<CustomClass3>() {
@Override
public CustomClass3 createInstance(Type type) {
return new CustomClass3("custom-instance");
}
})
.create();
assertCustomGson(gson);
// Modify `gson.newBuilder()`
GsonBuilder gsonBuilder = gson.newBuilder();
gsonBuilder.registerTypeAdapter(
CustomClass1.class,
new TypeAdapter<CustomClass1>() {
@Override
public CustomClass1 read(JsonReader in) throws IOException {
throw new UnsupportedOperationException();
}
@Override
public void write(JsonWriter out, CustomClass1 value) throws IOException {
out.value("overwritten custom-adapter");
}
});
gsonBuilder.registerTypeHierarchyAdapter(
CustomClass2.class,
new JsonSerializer<CustomClass2>() {
@Override
public JsonElement serialize(
CustomClass2 src, Type typeOfSrc, JsonSerializationContext context) {
return new JsonPrimitive("overwritten custom-hierarchy-adapter");
}
});
gsonBuilder.registerTypeAdapter(
CustomClass3.class,
new InstanceCreator<CustomClass3>() {
@Override
public CustomClass3 createInstance(Type type) {
return new CustomClass3("overwritten custom-instance");
}
});
// `gson` object should not have been affected by changes to new GsonBuilder
assertCustomGson(gson);
// New GsonBuilder based on `gson` should not have been affected either
assertCustomGson(gson.newBuilder().create());
// But new Gson instance from `gsonBuilder` should be affected by changes
Gson otherGson = gsonBuilder.create();
String json1 = otherGson.toJson(new CustomClass1());
assertThat(json1).isEqualTo("\"overwritten custom-adapter\"");
String json2 = otherGson.toJson(new CustomClass2());
assertThat(json2).isEqualTo("\"overwritten custom-hierarchy-adapter\"");
CustomClass3 customClass3 = otherGson.fromJson("{}", CustomClass3.class);
assertThat(customClass3.s).isEqualTo("overwritten custom-instance");
}
private static void assertCustomGson(Gson gson) {
String json1 = gson.toJson(new CustomClass1());
assertThat(json1).isEqualTo("\"custom-adapter\"");
String json2 = gson.toJson(new CustomClass2());
assertThat(json2).isEqualTo("\"custom-hierarchy-adapter\"");
CustomClass3 customClass3 = gson.fromJson("{}", CustomClass3.class);
assertThat(customClass3.s).isEqualTo("custom-instance");
}
private static class CustomClass1 {}
private static class CustomClass2 {}
private static class CustomClass3 {
static final String NO_ARG_CONSTRUCTOR_VALUE = "default instance";
final String s;
public CustomClass3(String s) {
this.s = s;
}
@SuppressWarnings("unused") // called by Gson
public CustomClass3() {
this(NO_ARG_CONSTRUCTOR_VALUE);
}
}
}