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

org.factcast.factus.projection.parameter.DefaultHandlerParameterContributor Maven / Gradle / Ivy

The newest version!
/*
 * Copyright © 2017-2023 factcast.org
 *
 * 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 org.factcast.factus.projection.parameter;

import com.google.common.base.Predicates;
import java.lang.annotation.Annotation;
import java.lang.reflect.*;
import java.util.*;
import javax.annotation.Nullable;
import lombok.NonNull;
import org.factcast.core.*;
import org.factcast.factus.Meta;
import org.factcast.factus.event.*;
import org.factcast.factus.event.EventObject;
import org.factcast.factus.projection.Projection;

@SuppressWarnings({"OptionalIsPresent", "java:S1872"})
public class DefaultHandlerParameterContributor implements HandlerParameterContributor {

  @Nullable
  @Override
  public HandlerParameterProvider providerFor(
      @NonNull Class type, @Nullable Type genericType, @NonNull Set annotations) {
    if (EventObject.class.isAssignableFrom(type)) {
      return (s, f, p) -> s.deserialize((Class) type, f.jsonPayload());
    }

    if (Fact.class == type) {
      return new FactProvider();
    }

    if (FactHeader.class == type) {
      return new FactHeaderProvider();
    }

    if (UUID.class == type) {
      return new FactIdProvider();
    }

    if (FactStreamPosition.class == type) {
      return new FactStreamPositionProvider();
    }

    Optional metaAnnotation =
        annotations.stream()
            .filter(Predicates.instanceOf(Meta.class))
            .map(Meta.class::cast)
            .findFirst();
    if (metaAnnotation.isPresent())
      return new MetaProvider(metaAnnotation.get(), type, genericType, annotations);

    // fall through
    return null;
  }
}

class FactProvider implements HandlerParameterProvider {
  @Override
  public Object apply(
      @NonNull EventSerializer s, @NonNull Fact fact, @NonNull Projection projection) {
    return fact;
  }
}

class FactIdProvider implements HandlerParameterProvider {
  @Override
  public Object apply(
      @NonNull EventSerializer s, @NonNull Fact fact, @NonNull Projection projection) {
    return fact.id();
  }
}

class FactStreamPositionProvider implements HandlerParameterProvider {
  @Override
  public Object apply(
      @NonNull EventSerializer s, @NonNull Fact fact, @NonNull Projection projection) {
    return FactStreamPosition.from(fact);
  }
}

class FactHeaderProvider implements HandlerParameterProvider {
  @Override
  public Object apply(
      @NonNull EventSerializer s, @NonNull Fact fact, @NonNull Projection projection) {
    return fact.header();
  }
}

@SuppressWarnings({"java:S1872"})
class MetaProvider implements HandlerParameterProvider {
  private final String key;
  private final Class targetType;

  public MetaProvider(
      Meta meta, Class targetType, Type genericType, Set annotations) {
    this.targetType = targetType;

    key = meta.value();

    checkPreconditionsForAllowedTypes(targetType);
    checkPreconditionsForKey(key);
    checkPreconditionsForString(targetType, annotations);
    checkPreconditionsForOptional(targetType, genericType);
  }

  private void checkPreconditionsForKey(String key) {
    if (key == null || key.trim().isEmpty())
      throw new IllegalArgumentException("@Meta must specify a valid key");
  }

  private void checkPreconditionsForAllowedTypes(Class targetType) {
    if (!(targetType == Optional.class || targetType == String.class))
      throw new IllegalArgumentException(
          "Only String or Optional types for @Meta annotated Parameters are allowed");
  }

  private static void checkPreconditionsForString(
      Class targetType, Set annotations) {
    if (targetType == String.class
        && annotations.stream()
            .noneMatch(a -> a.annotationType().getSimpleName().equals("Nullable")))
      throw new IllegalArgumentException(
          "Parameters of type String declared with @Meta must also be annotated with @Nullable. You could also change it to Optional.");
  }

  private static void checkPreconditionsForOptional(Class targetType, Type genericType) {
    if (targetType == Optional.class) {
      if (!(genericType instanceof ParameterizedType))
        throw new IllegalArgumentException(
            "Unparametrized Optional detected. It should be Optional instead.");
      else {
        if (((ParameterizedType) genericType).getActualTypeArguments()[0] != String.class)
          throw new IllegalArgumentException(
              "Badly parametrized Optional detected. It should be Optional instead.");
      }
    }
  }

  @Override
  public Object apply(
      @NonNull EventSerializer s, @NonNull Fact fact, @NonNull Projection projection) {
    String value = fact.header().meta(key);
    if (targetType == Optional.class) return Optional.ofNullable(value);
    else return value;
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy