Back to Repositories

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

The test suite provides comprehensive coverage of JSON parsing and reading operations.

  • Data type handling: numbers, strings, booleans, nulls
  • Array and object structure validation
  • Strict and lenient parsing modes for special values
  • Error handling and edge cases
  • Nested structure handling

Implementation Analysis

The testing approach uses JUnit to systematically verify JsonTreeReader behavior.

  • Structured test methods for each JSON data type
  • Assertion-based validation using Truth framework
  • Exception testing for invalid operations
  • Comprehensive token parsing verification

Technical Details

  • Testing Framework: JUnit
  • Assertion Library: Google Truth
  • Test Subject: JsonTreeReader class
  • Key Dependencies: GSON library components
  • Configuration: Includes both strict and lenient parsing modes

Best Practices Demonstrated

The test suite exemplifies high-quality testing practices in Java.

  • Isolated test cases for specific functionality
  • Thorough error condition testing
  • Clear test method naming conventions
  • Comprehensive validation of API behaviors
  • Resource cleanup handling

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");
  }
}