All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.google.api.server.spi.config.model.Serializers Maven / Gradle / Ivy

There is a newer version: 2.2.2
Show newest version
/*
 * Copyright 2016 Google Inc. All Rights Reserved.
 *
 * 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.api.server.spi.config.model;

import com.google.api.server.spi.config.ApiTransformer;
import com.google.api.server.spi.config.Transformer;
import com.google.common.base.Joiner;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Lists;
import com.google.common.reflect.TypeToken;

import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;

import javax.annotation.Nullable;

/**
 * Common utility functions for {@link Transformer}s.
 */
public final class Serializers {

  private enum SerializerConstructor {
    TYPE {
      @Override
      > S construct(Class serializerClass, Type sourceType)
          throws Exception {
        return serializerClass.getDeclaredConstructor(Type.class).newInstance(sourceType);
      }

      @Override
      ImmutableList getArgs() {
        return ImmutableList.of(Type.class);
      }
    },
    CLASS {
      @Override
      > S construct(Class serializerClass, Type sourceType)
          throws Exception {
        Class sourceClass = TypeToken.of(sourceType).getRawType();
        return serializerClass.getDeclaredConstructor(Class.class).newInstance(sourceClass);
      }

      @Override
      ImmutableList getArgs() {
        return ImmutableList.of(Class.class);
      }
    },
    DEFAULT {
      @Override
      > S construct(Class serializerClass, Type sourceType)
          throws Exception {
        return serializerClass.getDeclaredConstructor().newInstance();
      }

      @Override
      ImmutableList getArgs() {
        return ImmutableList.of();
      }
    };

    abstract > S construct(Class serializerClass, Type sourceType)
          throws Exception;

    abstract ImmutableList getArgs();

    @Override
    public String toString() {
      return String.format("(%s)", Joiner.on(", ").join(getArgs()));
    }
  }

  private Serializers() { }

  /**
   * Instantiates a new serializer instance.
   *
   * @param serializerClass a serializer class
   * @param sourceType the type being serialized
   * @return an instance of the serializer
   * @throws IllegalStateException if instantiation failed, or the class does not implement
   *     Serializer
   */
  @SuppressWarnings({"unchecked"})
  public static > S instantiate(final Class serializerClass,
      Type sourceType) {
    if (!TypeToken.of(getSourceType(serializerClass)).isSupertypeOf(sourceType)) {
      throw new IllegalArgumentException(String.format(
          "Can not instantiate %s, the serializer source %s is not assignable from %s",
          serializerClass, getSourceType(serializerClass), sourceType));
    }
    Exception parentException = null;
    for (SerializerConstructor constructor : SerializerConstructor.values()) {
      try {
        return constructor.construct(serializerClass, sourceType);
      } catch (NoSuchMethodException e) {
        continue;
      } catch (Exception e) {
        String errorMessage = String.format(
            "Failed to instantiate custom serializer constructor %s%s with source type: %s",
            serializerClass.getName(),
            constructor,
            sourceType);
        throw new IllegalStateException(errorMessage, e);
      }
    }
    String message = String.format(
        "Failed to instantiate custom serializer %s, constructors not found: %s",
        serializerClass.getName(),
        Arrays.toString(SerializerConstructor.values()));
    throw new IllegalStateException(message, parentException);
  }

  /**
   * Gets the {@link Transformer} class for a particular data type. This first checks to see if the
   * type or any of its parents is annotated with {@link ApiTransformer}. If that fails, a lookup
   * is made in the serialization config.
   *
   * @param type a type to find a serializer for
   * @param config a serialization config
   * @return All matching serializers. Could be > 1 if multiple interfaces provide serializers.
   */
  public static List>> getSerializerClasses(
      @Nullable Type type, @Nullable final ApiSerializationConfig config) {
    if (type == null) {
      return Collections.emptyList();
    }

    List>> allParentSerializers = Lists.newArrayList();
    List> serializedTypes = Lists.newArrayList();
    for (TypeToken typeToken : TypeToken.of(type).getTypes()) {
      ApiTransformer apiSerialization = typeToken.getRawType().getAnnotation(ApiTransformer.class);
      if (isSupertypeOf(typeToken, serializedTypes)) {
        continue;
      }
      if (apiSerialization != null) {
        allParentSerializers.add(apiSerialization.value());
        serializedTypes.add(typeToken);
      } else if (config != null) {
        ApiSerializationConfig.SerializerConfig serializerConfig =
            config.getSerializerConfig(typeToken.getType());
        if (serializerConfig != null) {
          allParentSerializers.add(serializerConfig.getSerializer());
          serializedTypes.add(typeToken);
        }
      }
    }

    return allParentSerializers;
  }

  /**
   * Gets the {@code Serializer} source type for a class. This resolves placeholders in generics.
   *
   * @param clazz a class, possibly implementing {@code Serializer}
   * @return the resolved source type, null if clazz is not a serializer
   */
  @Nullable
  public static Type getSourceType(@Nullable Class> clazz) {
    ParameterizedType types = Serializers.getResolvedType(clazz);
    return types == null ? null : types.getActualTypeArguments()[0];
  }

  /**
   * Gets the {@code Serializer} target type for a class. This resolves placeholders in generics.
   *
   * @param clazz a class, possibly implementing {@code Serializer}
   * @return the resolved target type, null if clazz is not a serializer
   */
  @Nullable
  public static Type getTargetType(@Nullable Class> clazz) {
    ParameterizedType types = Serializers.getResolvedType(clazz);
    return types == null ? null : types.getActualTypeArguments()[1];
  }

  @Nullable
  private static ParameterizedType getResolvedType(
      @Nullable Class> clazz) {
    if (clazz == null || !Transformer.class.isAssignableFrom(clazz)) {
        return null;
    }
    for (TypeToken token : TypeToken.of(clazz).getTypes().interfaces()) {
      if (token.getRawType().equals(Transformer.class)) {
        Type tokenType = token.getType();
        return tokenType instanceof ParameterizedType ? (ParameterizedType) tokenType : null;
      }
    }
    return null;
  }

  private static boolean isSupertypeOf(TypeToken typeToken, List> subtypes) {
    for (TypeToken subType : subtypes) {
      if (typeToken.isSupertypeOf(subType)) {
        return true;
      }
    }
    return false;
  }
}