Back to Repositories

Testing @Expose Annotation Exclusion Strategy in Google GSON

This test suite validates the functionality of GSON’s @Expose annotation and its exclusion strategy, ensuring proper field serialization and deserialization behavior in JSON processing. The tests verify how fields with different @Expose configurations are handled during JSON operations.

Test Coverage Overview

The test suite provides comprehensive coverage of @Expose annotation behavior in GSON.

Key areas tested include:
  • Class-level exclusion verification
  • Non-annotated field handling
  • Explicitly exposed and hidden fields
  • Different serialize/deserialize mode combinations
  • Field-level annotation processing

Implementation Analysis

The testing approach uses JUnit framework with systematic verification of field exclusion patterns. The implementation leverages GSON’s Excluder class to test various @Expose annotation configurations, with separate checks for serialization and deserialization scenarios.

The test structure employs helper methods for consistent assertion patterns and uses a mock object class to represent different field annotation scenarios.

Technical Details

Testing tools and configuration:
  • JUnit test framework
  • Google Truth assertion library
  • GSON’s Excluder class for exclusion strategy
  • Mock object class with annotated fields
  • Helper methods for field attribute creation

Best Practices Demonstrated

The test suite exemplifies several testing best practices:

  • Clear test method naming conventions
  • Isolated test cases for each scenario
  • Proper setup of test infrastructure
  • Consistent assertion patterns
  • Well-structured mock objects
  • Comprehensive coverage of edge cases

google/gson

gson/src/test/java/com/google/gson/ExposeAnnotationExclusionStrategyTest.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;

import static com.google.common.truth.Truth.assertThat;

import com.google.gson.annotations.Expose;
import com.google.gson.internal.Excluder;
import java.lang.reflect.Field;
import org.junit.Test;

/**
 * Unit tests for GsonBuilder.REQUIRE_EXPOSE_DESERIALIZE.
 *
 * @author Joel Leitch
 */
public class ExposeAnnotationExclusionStrategyTest {
  private Excluder excluder = Excluder.DEFAULT.excludeFieldsWithoutExposeAnnotation();

  private void assertIncludesClass(Class<?> c) {
    assertThat(excluder.excludeClass(c, true)).isFalse();
    assertThat(excluder.excludeClass(c, false)).isFalse();
  }

  private void assertIncludesField(Field f) {
    assertThat(excluder.excludeField(f, true)).isFalse();
    assertThat(excluder.excludeField(f, false)).isFalse();
  }

  private void assertExcludesField(Field f) {
    assertThat(excluder.excludeField(f, true)).isTrue();
    assertThat(excluder.excludeField(f, false)).isTrue();
  }

  @Test
  public void testNeverSkipClasses() {
    assertIncludesClass(MockObject.class);
  }

  @Test
  public void testSkipNonAnnotatedFields() throws Exception {
    Field f = createFieldAttributes("hiddenField");
    assertExcludesField(f);
  }

  @Test
  public void testSkipExplicitlySkippedFields() throws Exception {
    Field f = createFieldAttributes("explicitlyHiddenField");
    assertExcludesField(f);
  }

  @Test
  public void testNeverSkipExposedAnnotatedFields() throws Exception {
    Field f = createFieldAttributes("exposedField");
    assertIncludesField(f);
  }

  @Test
  public void testNeverSkipExplicitlyExposedAnnotatedFields() throws Exception {
    Field f = createFieldAttributes("explicitlyExposedField");
    assertIncludesField(f);
  }

  @Test
  public void testDifferentSerializeAndDeserializeField() throws Exception {
    Field f = createFieldAttributes("explicitlyDifferentModeField");
    assertThat(excluder.excludeField(f, true)).isFalse();
    assertThat(excluder.excludeField(f, false)).isTrue();
  }

  private static Field createFieldAttributes(String fieldName) throws Exception {
    return MockObject.class.getField(fieldName);
  }

  @SuppressWarnings("unused")
  private static class MockObject {
    @Expose public final int exposedField = 0;

    @Expose(serialize = true, deserialize = true)
    public final int explicitlyExposedField = 0;

    @Expose(serialize = false, deserialize = false)
    public final int explicitlyHiddenField = 0;

    @Expose(serialize = true, deserialize = false)
    public final int explicitlyDifferentModeField = 0;

    public final int hiddenField = 0;
  }
}