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

com.google.cloud.firestore.encoding.PojoBeanMapper Maven / Gradle / Ivy

There is a newer version: 3.29.1
Show newest version
/*
 * Copyright 2024 Google LLC
 *
 * 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.cloud.firestore.encoding;

import com.google.cloud.firestore.annotation.DocumentId;
import com.google.cloud.firestore.annotation.Exclude;
import com.google.cloud.firestore.annotation.ServerTimestamp;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.Type;
import java.lang.reflect.TypeVariable;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Locale;
import java.util.Map;
import java.util.logging.Logger;

// Helper class to convert from maps to custom objects (Beans), and vice versa.
class PojoBeanMapper extends BeanMapper {
  private static final Logger LOGGER = Logger.getLogger(PojoBeanMapper.class.getName());

  private final Constructor constructor;

  // Case insensitive mapping of properties to their case sensitive versions
  private final Map properties;

  // Below are maps to find getter/setter/field from a given property name.
  // A property name is the name annotated by @PropertyName, if exists; or their property name
  // following the Java Bean convention: field name is kept as-is while getters/setters will have
  // their prefixes removed. See method propertyName for details.
  private final Map getters;
  private final Map setters;
  private final Map fields;

  PojoBeanMapper(Class clazz) {
    super(clazz);
    properties = new HashMap<>();

    setters = new HashMap<>();
    getters = new HashMap<>();
    fields = new HashMap<>();

    Constructor constructor;
    try {
      constructor = clazz.getDeclaredConstructor();
      constructor.setAccessible(true);
    } catch (NoSuchMethodException e) {
      // We will only fail at deserialization time if no constructor is present
      constructor = null;
    }
    this.constructor = constructor;
    // Add any public getters to properties (including isXyz())
    for (Method method : clazz.getMethods()) {
      if (shouldIncludeGetter(method)) {
        String propertyName = propertyName(method);
        addProperty(propertyName);
        method.setAccessible(true);
        if (getters.containsKey(propertyName)) {
          throw new RuntimeException(
              "Found conflicting getters for name "
                  + method.getName()
                  + " on class "
                  + clazz.getName());
        }
        getters.put(propertyName, method);
        applyGetterAnnotations(method);
      }
    }

    // Add any public fields to properties
    for (Field field : clazz.getFields()) {
      if (shouldIncludeField(field)) {
        String propertyName = propertyName(field);
        addProperty(propertyName);
        applyFieldAnnotations(field);
      }
    }

    // We can use private setters and fields for known (public) properties/getters. Since
    // getMethods/getFields only returns public methods/fields we need to traverse the
    // class hierarchy to find the appropriate setter or field.
    Class currentClass = clazz;
    do {
      // Add any setters
      for (Method method : currentClass.getDeclaredMethods()) {
        if (shouldIncludeSetter(method)) {
          String propertyName = propertyName(method);
          String existingPropertyName = properties.get(propertyName.toLowerCase(Locale.US));
          if (existingPropertyName != null) {
            if (!existingPropertyName.equals(propertyName)) {
              throw new RuntimeException(
                  "Found setter on "
                      + currentClass.getName()
                      + " with invalid case-sensitive name: "
                      + method.getName());
            } else {
              Method existingSetter = setters.get(propertyName);
              if (existingSetter == null) {
                method.setAccessible(true);
                setters.put(propertyName, method);
                applySetterAnnotations(method);
              } else if (!isSetterOverride(method, existingSetter)) {
                // We require that setters with conflicting property names are
                // overrides from a base class
                if (currentClass == clazz) {
                  // TODO: Should we support overloads?
                  throw new RuntimeException(
                      "Class "
                          + clazz.getName()
                          + " has multiple setter overloads with name "
                          + method.getName());
                } else {
                  throw new RuntimeException(
                      "Found conflicting setters "
                          + "with name: "
                          + method.getName()
                          + " (conflicts with "
                          + existingSetter.getName()
                          + " defined on "
                          + existingSetter.getDeclaringClass().getName()
                          + ")");
                }
              }
            }
          }
        }
      }

      for (Field field : currentClass.getDeclaredFields()) {
        String propertyName = propertyName(field);

        // Case sensitivity is checked at deserialization time
        // Fields are only added if they don't exist on a subclass
        if (properties.containsKey(propertyName.toLowerCase(Locale.US))
            && !fields.containsKey(propertyName)) {
          field.setAccessible(true);
          fields.put(propertyName, field);
          applyFieldAnnotations(field);
        }
      }

      // Traverse class hierarchy until we reach java.lang.Object which contains a bunch
      // of fields/getters we don't want to serialize
      currentClass = currentClass.getSuperclass();
    } while (currentClass != null && !currentClass.equals(Object.class));

    if (properties.isEmpty()) {
      throw new RuntimeException("No properties to serialize found on class " + clazz.getName());
    }

    // Make sure we can write to @DocumentId annotated properties before proceeding.
    for (String docIdProperty : documentIdPropertyNames) {
      if (!setters.containsKey(docIdProperty) && !fields.containsKey(docIdProperty)) {
        throw new RuntimeException(
            "@DocumentId is annotated on property "
                + docIdProperty
                + " of class "
                + clazz.getName()
                + " but no field or public setter was found");
      }
    }
  }

  @Override
  Map serialize(T object, DeserializeContext.ErrorPath path) {
    verifyValidType(object);
    Map result = new HashMap<>();
    for (String property : properties.values()) {
      // Skip @DocumentId annotated properties;
      if (documentIdPropertyNames.contains(property)) {
        continue;
      }

      Object propertyValue;
      if (getters.containsKey(property)) {
        Method getter = getters.get(property);
        try {
          propertyValue = getter.invoke(object);
        } catch (IllegalAccessException | InvocationTargetException e) {
          throw new RuntimeException(e);
        }
      } else {
        // Must be a field
        Field field = fields.get(property);
        if (field == null) {
          throw new IllegalStateException("Bean property without field or getter: " + property);
        }
        try {
          propertyValue = field.get(object);
        } catch (IllegalAccessException e) {
          throw new RuntimeException(e);
        }
      }

      Object serializedValue = getSerializedValue(property, propertyValue, path);

      result.put(property, serializedValue);
    }
    return result;
  }

  @Override
  T deserialize(
      Map values,
      Map>, Type> types,
      DeserializeContext context) {
    if (constructor == null) {
      throw context.errorPath.deserializeError(
          "Class "
              + getClazz().getName()
              + " does not define a no-argument constructor. If you are using ProGuard, make "
              + "sure these constructors are not stripped");
    }

    T instance;
    try {
      instance = constructor.newInstance();
    } catch (InstantiationException | IllegalAccessException | InvocationTargetException e) {
      throw new RuntimeException(e);
    }
    HashSet deserializedProperties = new HashSet<>();
    for (Map.Entry entry : values.entrySet()) {
      String propertyName = entry.getKey();
      DeserializeContext.ErrorPath childPath = context.errorPath.child(propertyName);
      if (setters.containsKey(propertyName)) {
        Method setter = setters.get(propertyName);
        Type[] params = setter.getGenericParameterTypes();
        if (params.length != 1) {
          throw childPath.deserializeError("Setter does not have exactly one parameter");
        }
        Type resolvedType = resolveType(params[0], types);
        Object value =
            CustomClassMapper.deserializeToType(
                entry.getValue(), resolvedType, context.newInstanceWithErrorPath(childPath));
        try {
          setter.invoke(instance, value);
        } catch (IllegalAccessException | InvocationTargetException e) {
          throw new RuntimeException(e);
        }
        deserializedProperties.add(propertyName);
      } else if (fields.containsKey(propertyName)) {
        Field field = fields.get(propertyName);
        Type resolvedType = resolveType(field.getGenericType(), types);
        Object value =
            CustomClassMapper.deserializeToType(
                entry.getValue(), resolvedType, context.newInstanceWithErrorPath(childPath));
        try {
          field.set(instance, value);
        } catch (IllegalAccessException e) {
          throw new RuntimeException(e);
        }
        deserializedProperties.add(propertyName);
      } else {
        String message =
            "No setter/field for " + propertyName + " found on class " + getClazz().getName();
        if (properties.containsKey(propertyName.toLowerCase(Locale.US))) {
          message += " (fields/setters are case sensitive!)";
        }
        if (isThrowOnUnknownProperties()) {
          throw new RuntimeException(message);
        } else if (isWarnOnUnknownProperties()) {
          LOGGER.warning(message);
        }
      }
    }
    populateDocumentIdProperties(types, context, instance, deserializedProperties);

    return instance;
  }

  private void addProperty(String property) {
    String oldValue = properties.put(property.toLowerCase(Locale.US), property);
    if (oldValue != null && !property.equals(oldValue)) {
      throw new RuntimeException(
          "Found two getters or fields with conflicting case "
              + "sensitivity for property: "
              + property.toLowerCase(Locale.US));
    }
  }

  // Populate @DocumentId annotated fields. If there is a conflict (@DocumentId annotation is
  // applied to a property that is already deserialized from the firestore document)
  // a runtime exception will be thrown.
  private void populateDocumentIdProperties(
      Map>, Type> types,
      DeserializeContext context,
      T instance,
      HashSet deserializedProperties) {
    for (String docIdPropertyName : documentIdPropertyNames) {
      checkForDocIdConflict(docIdPropertyName, deserializedProperties, context);
      DeserializeContext.ErrorPath childPath = context.errorPath.child(docIdPropertyName);
      if (setters.containsKey(docIdPropertyName)) {
        Method setter = setters.get(docIdPropertyName);
        Type[] params = setter.getGenericParameterTypes();
        if (params.length != 1) {
          throw childPath.deserializeError("Setter does not have exactly one parameter");
        }
        Type resolvedType = resolveType(params[0], types);
        try {
          if (resolvedType == String.class) {
            setter.invoke(instance, context.documentRef.getId());
          } else {
            setter.invoke(instance, context.documentRef);
          }
        } catch (IllegalAccessException | InvocationTargetException e) {
          throw new RuntimeException(e);
        }
      } else {
        Field docIdField = fields.get(docIdPropertyName);
        try {
          if (docIdField.getType() == String.class) {
            docIdField.set(instance, context.documentRef.getId());
          } else {
            docIdField.set(instance, context.documentRef);
          }
        } catch (IllegalAccessException e) {
          throw new RuntimeException(e);
        }
      }
    }
  }

  private void applyGetterAnnotations(Method method) {
    Class returnType = method.getReturnType();
    if (method.isAnnotationPresent(ServerTimestamp.class)) {
      validateServerTimestampType("Method", "returns", returnType);
      serverTimestamps.add(propertyName(method));
    }
    // Even though the value will be skipped, we still check for type matching for consistency.
    if (method.isAnnotationPresent(DocumentId.class)) {
      validateDocumentIdType("Method", "returns", returnType);
      documentIdPropertyNames.add(propertyName(method));
    }
  }

  private void applySetterAnnotations(Method method) {
    if (method.isAnnotationPresent(ServerTimestamp.class)) {
      throw new IllegalArgumentException(
          "Method "
              + method.getName()
              + " is annotated with @ServerTimestamp but should not be. @ServerTimestamp can"
              + " only be applied to fields and getters, not setters.");
    }
    if (method.isAnnotationPresent(DocumentId.class)) {
      Class paramType = method.getParameterTypes()[0];
      validateDocumentIdType("Method", "accepts", paramType);
      documentIdPropertyNames.add(propertyName(method));
    }
  }

  private boolean shouldIncludeGetter(Method method) {
    if (!method.getName().startsWith("get") && !method.getName().startsWith("is")) {
      return false;
    }
    // Exclude methods from Object.class
    if (method.getDeclaringClass().equals(Object.class)) {
      return false;
    }
    // Non-public methods
    if (!Modifier.isPublic(method.getModifiers())) {
      return false;
    }
    // Static methods
    if (Modifier.isStatic(method.getModifiers())) {
      return false;
    }
    // No return type
    if (method.getReturnType().equals(Void.TYPE)) {
      return false;
    }
    // Non-zero parameters
    if (method.getParameterTypes().length != 0) {
      return false;
    }
    // Excluded methods
    if (method.isAnnotationPresent(Exclude.class)) {
      return false;
    }
    return true;
  }

  private boolean shouldIncludeSetter(Method method) {
    if (!method.getName().startsWith("set")) {
      return false;
    }
    // Exclude methods from Object.class
    if (method.getDeclaringClass().equals(Object.class)) {
      return false;
    }
    // Static methods
    if (Modifier.isStatic(method.getModifiers())) {
      return false;
    }
    // Has a return type
    if (!method.getReturnType().equals(Void.TYPE)) {
      return false;
    }
    // Methods without exactly one parameters
    if (method.getParameterTypes().length != 1) {
      return false;
    }
    // Excluded methods
    if (method.isAnnotationPresent(Exclude.class)) {
      return false;
    }
    return true;
  }

  private boolean shouldIncludeField(Field field) {
    // Exclude methods from Object.class
    if (field.getDeclaringClass().equals(Object.class)) {
      return false;
    }
    // Non-public fields
    if (!Modifier.isPublic(field.getModifiers())) {
      return false;
    }
    // Static fields
    if (Modifier.isStatic(field.getModifiers())) {
      return false;
    }
    // Transient fields
    if (Modifier.isTransient(field.getModifiers())) {
      return false;
    }
    // Excluded fields
    if (field.isAnnotationPresent(Exclude.class)) {
      return false;
    }
    return true;
  }

  private boolean isSetterOverride(Method base, Method override) {
    // We expect an overridden setter here
    hardAssert(
        base.getDeclaringClass().isAssignableFrom(override.getDeclaringClass()),
        "Expected override from a base class");
    hardAssert(base.getReturnType().equals(Void.TYPE), "Expected void return type");
    hardAssert(override.getReturnType().equals(Void.TYPE), "Expected void return type");

    Type[] baseParameterTypes = base.getParameterTypes();
    Type[] overrideParameterTypes = override.getParameterTypes();
    hardAssert(baseParameterTypes.length == 1, "Expected exactly one parameter");
    hardAssert(overrideParameterTypes.length == 1, "Expected exactly one parameter");

    return base.getName().equals(override.getName())
        && baseParameterTypes[0].equals(overrideParameterTypes[0]);
  }

  private String propertyName(Method method) {
    String annotatedName = annotatedName(method);
    return annotatedName != null ? annotatedName : serializedName(method.getName());
  }

  private String serializedName(String methodName) {
    String[] prefixes = new String[] {"get", "set", "is"};
    String methodPrefix = null;
    for (String prefix : prefixes) {
      if (methodName.startsWith(prefix)) {
        methodPrefix = prefix;
      }
    }
    if (methodPrefix == null) {
      throw new IllegalArgumentException("Unknown Bean prefix for method: " + methodName);
    }
    String strippedName = methodName.substring(methodPrefix.length());

    // Make sure the first word or upper-case prefix is converted to lower-case
    char[] chars = strippedName.toCharArray();
    int pos = 0;
    while (pos < chars.length && Character.isUpperCase(chars[pos])) {
      chars[pos] = Character.toLowerCase(chars[pos]);
      pos++;
    }
    return new String(chars);
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy