Testing JSON Element Parsing and Reading Implementation in google/gson
This test suite validates the JsonTreeReader functionality in Google’s GSON library, focusing on parsing and reading JSON elements. It thoroughly tests JSON data type handling, strict/lenient parsing modes, and error conditions.
Test Coverage Overview
Implementation Analysis
Technical Details
Best Practices Demonstrated
google/gson
gson/src/test/java/com/google/gson/internal/bind/JsonElementReaderTest.java
/*
* Copyright (C) 2011 Google Inc.
*
* 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.internal.bind;
import static com.google.common.truth.Truth.assertThat;
import static org.junit.Assert.assertThrows;
import com.google.gson.JsonElement;
import com.google.gson.JsonParser;
import com.google.gson.JsonPrimitive;
import com.google.gson.Strictness;
import com.google.gson.stream.JsonToken;
import com.google.gson.stream.MalformedJsonException;
import java.io.IOException;
import org.junit.Test;
@SuppressWarnings("resource")
public final class JsonElementReaderTest {
@Test
public void testNumbers() throws IOException {
JsonElement element = JsonParser.parseString("[1, 2, 3]");
JsonTreeReader reader = new JsonTreeReader(element);
reader.beginArray();
assertThat(reader.nextInt()).isEqualTo(1);
assertThat(reader.nextLong()).isEqualTo(2L);
assertThat(reader.nextDouble()).isEqualTo(3.0);
reader.endArray();
}
@Test
public void testLenientNansAndInfinities() throws IOException {
JsonElement element = JsonParser.parseString("[NaN, -Infinity, Infinity]");
JsonTreeReader reader = new JsonTreeReader(element);
reader.setStrictness(Strictness.LENIENT);
reader.beginArray();
assertThat(reader.nextDouble()).isNaN();
assertThat(reader.nextDouble()).isEqualTo(Double.NEGATIVE_INFINITY);
assertThat(reader.nextDouble()).isEqualTo(Double.POSITIVE_INFINITY);
reader.endArray();
}
@Test
public void testStrictNansAndInfinities() throws IOException {
JsonElement element = JsonParser.parseString("[NaN, -Infinity, Infinity]");
JsonTreeReader reader = new JsonTreeReader(element);
reader.setStrictness(Strictness.LEGACY_STRICT);
reader.beginArray();
var e = assertThrows(MalformedJsonException.class, () -> reader.nextDouble());
assertThat(e).hasMessageThat().isEqualTo("JSON forbids NaN and infinities: NaN");
assertThat(reader.nextString()).isEqualTo("NaN");
e = assertThrows(MalformedJsonException.class, () -> reader.nextDouble());
assertThat(e).hasMessageThat().isEqualTo("JSON forbids NaN and infinities: -Infinity");
assertThat(reader.nextString()).isEqualTo("-Infinity");
e = assertThrows(MalformedJsonException.class, () -> reader.nextDouble());
assertThat(e).hasMessageThat().isEqualTo("JSON forbids NaN and infinities: Infinity");
assertThat(reader.nextString()).isEqualTo("Infinity");
reader.endArray();
}
@Test
public void testNumbersFromStrings() throws IOException {
JsonElement element = JsonParser.parseString("[\"1\", \"2\", \"3\"]");
JsonTreeReader reader = new JsonTreeReader(element);
reader.beginArray();
assertThat(reader.nextInt()).isEqualTo(1);
assertThat(reader.nextLong()).isEqualTo(2L);
assertThat(reader.nextDouble()).isEqualTo(3.0);
reader.endArray();
}
@Test
public void testStringsFromNumbers() throws IOException {
JsonElement element = JsonParser.parseString("[1]");
JsonTreeReader reader = new JsonTreeReader(element);
reader.beginArray();
assertThat(reader.nextString()).isEqualTo("1");
reader.endArray();
}
@Test
public void testBooleans() throws IOException {
JsonElement element = JsonParser.parseString("[true, false]");
JsonTreeReader reader = new JsonTreeReader(element);
reader.beginArray();
assertThat(reader.nextBoolean()).isEqualTo(true);
assertThat(reader.nextBoolean()).isEqualTo(false);
reader.endArray();
}
@Test
public void testNulls() throws IOException {
JsonElement element = JsonParser.parseString("[null,null]");
JsonTreeReader reader = new JsonTreeReader(element);
reader.beginArray();
reader.nextNull();
reader.nextNull();
reader.endArray();
}
@Test
public void testStrings() throws IOException {
JsonElement element = JsonParser.parseString("[\"A\",\"B\"]");
JsonTreeReader reader = new JsonTreeReader(element);
reader.beginArray();
assertThat(reader.nextString()).isEqualTo("A");
assertThat(reader.nextString()).isEqualTo("B");
reader.endArray();
}
@Test
public void testArray() throws IOException {
JsonElement element = JsonParser.parseString("[1, 2, 3]");
JsonTreeReader reader = new JsonTreeReader(element);
assertThat(reader.peek()).isEqualTo(JsonToken.BEGIN_ARRAY);
reader.beginArray();
assertThat(reader.peek()).isEqualTo(JsonToken.NUMBER);
assertThat(reader.nextInt()).isEqualTo(1);
assertThat(reader.peek()).isEqualTo(JsonToken.NUMBER);
assertThat(reader.nextInt()).isEqualTo(2);
assertThat(reader.peek()).isEqualTo(JsonToken.NUMBER);
assertThat(reader.nextInt()).isEqualTo(3);
assertThat(reader.peek()).isEqualTo(JsonToken.END_ARRAY);
reader.endArray();
assertThat(reader.peek()).isEqualTo(JsonToken.END_DOCUMENT);
}
@Test
public void testObject() throws IOException {
JsonElement element = JsonParser.parseString("{\"A\": 1, \"B\": 2}");
JsonTreeReader reader = new JsonTreeReader(element);
assertThat(reader.peek()).isEqualTo(JsonToken.BEGIN_OBJECT);
reader.beginObject();
assertThat(reader.peek()).isEqualTo(JsonToken.NAME);
assertThat(reader.nextName()).isEqualTo("A");
assertThat(reader.peek()).isEqualTo(JsonToken.NUMBER);
assertThat(reader.nextInt()).isEqualTo(1);
assertThat(reader.peek()).isEqualTo(JsonToken.NAME);
assertThat(reader.nextName()).isEqualTo("B");
assertThat(reader.peek()).isEqualTo(JsonToken.NUMBER);
assertThat(reader.nextInt()).isEqualTo(2);
assertThat(reader.peek()).isEqualTo(JsonToken.END_OBJECT);
reader.endObject();
assertThat(reader.peek()).isEqualTo(JsonToken.END_DOCUMENT);
}
@Test
public void testEmptyArray() throws IOException {
JsonElement element = JsonParser.parseString("[]");
JsonTreeReader reader = new JsonTreeReader(element);
reader.beginArray();
reader.endArray();
}
@Test
public void testNestedArrays() throws IOException {
JsonElement element = JsonParser.parseString("[[],[[]]]");
JsonTreeReader reader = new JsonTreeReader(element);
reader.beginArray();
reader.beginArray();
reader.endArray();
reader.beginArray();
reader.beginArray();
reader.endArray();
reader.endArray();
reader.endArray();
}
@Test
public void testNestedObjects() throws IOException {
JsonElement element = JsonParser.parseString("{\"A\":{},\"B\":{\"C\":{}}}");
JsonTreeReader reader = new JsonTreeReader(element);
reader.beginObject();
assertThat(reader.nextName()).isEqualTo("A");
reader.beginObject();
reader.endObject();
assertThat(reader.nextName()).isEqualTo("B");
reader.beginObject();
assertThat(reader.nextName()).isEqualTo("C");
reader.beginObject();
reader.endObject();
reader.endObject();
reader.endObject();
}
@Test
public void testEmptyObject() throws IOException {
JsonElement element = JsonParser.parseString("{}");
JsonTreeReader reader = new JsonTreeReader(element);
reader.beginObject();
reader.endObject();
}
@Test
public void testSkipValue() throws IOException {
JsonElement element = JsonParser.parseString("[\"A\",{\"B\":[[]]},\"C\",[[]],\"D\",null]");
JsonTreeReader reader = new JsonTreeReader(element);
reader.beginArray();
assertThat(reader.nextString()).isEqualTo("A");
reader.skipValue();
assertThat(reader.nextString()).isEqualTo("C");
reader.skipValue();
assertThat(reader.nextString()).isEqualTo("D");
reader.skipValue();
reader.endArray();
}
@Test
public void testWrongType() throws IOException {
JsonElement element = JsonParser.parseString("[[],\"A\"]");
JsonTreeReader reader = new JsonTreeReader(element);
reader.beginArray();
assertThrows(IllegalStateException.class, () -> reader.nextBoolean());
assertThrows(IllegalStateException.class, () -> reader.nextNull());
assertThrows(IllegalStateException.class, () -> reader.nextString());
assertThrows(IllegalStateException.class, () -> reader.nextInt());
assertThrows(IllegalStateException.class, () -> reader.nextLong());
assertThrows(IllegalStateException.class, () -> reader.nextDouble());
assertThrows(IllegalStateException.class, () -> reader.nextName());
assertThrows(IllegalStateException.class, () -> reader.beginObject());
assertThrows(IllegalStateException.class, () -> reader.endArray());
assertThrows(IllegalStateException.class, () -> reader.endObject());
reader.beginArray();
reader.endArray();
assertThrows(IllegalStateException.class, () -> reader.nextBoolean());
assertThrows(IllegalStateException.class, () -> reader.nextNull());
assertThrows(NumberFormatException.class, () -> reader.nextInt());
assertThrows(NumberFormatException.class, () -> reader.nextLong());
assertThrows(NumberFormatException.class, () -> reader.nextDouble());
assertThrows(IllegalStateException.class, () -> reader.nextName());
assertThat(reader.nextString()).isEqualTo("A");
reader.endArray();
}
@Test
public void testNextJsonElement() throws IOException {
JsonElement element = JsonParser.parseString("{\"A\": 1, \"B\" : {}, \"C\" : []}");
JsonTreeReader reader = new JsonTreeReader(element);
reader.beginObject();
var e = assertThrows(IllegalStateException.class, () -> reader.nextJsonElement());
assertThat(e).hasMessageThat().isEqualTo("Unexpected NAME when reading a JsonElement.");
assertThat(reader.nextName()).isEqualTo("A");
assertThat(reader.nextJsonElement()).isEqualTo(new JsonPrimitive(1));
assertThat(reader.nextName()).isEqualTo("B");
reader.beginObject();
assertThrows(IllegalStateException.class, () -> reader.nextJsonElement());
reader.endObject();
assertThat(reader.nextName()).isEqualTo("C");
reader.beginArray();
assertThrows(IllegalStateException.class, () -> reader.nextJsonElement());
reader.endArray();
reader.endObject();
assertThrows(IllegalStateException.class, () -> reader.nextJsonElement());
}
@Test
public void testEarlyClose() throws IOException {
JsonElement element = JsonParser.parseString("[1, 2, 3]");
JsonTreeReader reader = new JsonTreeReader(element);
reader.beginArray();
reader.close();
var e = assertThrows(IllegalStateException.class, () -> reader.peek());
assertThat(e).hasMessageThat().isEqualTo("JsonReader is closed");
}
}