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

se.fortnox.reactivewizard.jaxrs.JaxRsResource Maven / Gradle / Ivy

There is a newer version: 24.6.0
Show newest version
package se.fortnox.reactivewizard.jaxrs;

import io.netty.handler.codec.http.HttpMethod;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import reactor.netty.http.server.HttpServerRequest;
import reactor.netty.http.server.HttpServerResponse;
import se.fortnox.reactivewizard.jaxrs.params.ParamResolver;
import se.fortnox.reactivewizard.jaxrs.params.ParamResolverFactories;
import se.fortnox.reactivewizard.jaxrs.response.JaxRsResult;
import se.fortnox.reactivewizard.jaxrs.response.JaxRsResultFactory;
import se.fortnox.reactivewizard.jaxrs.response.JaxRsResultFactoryFactory;
import se.fortnox.reactivewizard.util.FluxRxConverter;
import se.fortnox.reactivewizard.util.ReflectionUtil;

import javax.ws.rs.Consumes;
import javax.ws.rs.core.MediaType;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.List;
import java.util.function.Function;
import java.util.regex.Pattern;

import static java.lang.String.format;
import static java.util.Arrays.asList;
import static se.fortnox.reactivewizard.util.FluxRxConverter.isFlux;
import static se.fortnox.reactivewizard.util.FluxRxConverter.isMono;

/**
 * Represents a JaxRs resource. Maps to a method of a resource class. Use the call method with an incoming request to
 * invoke the method of the resource. Returns null if the request does not match.
 */
public class JaxRsResource implements Comparable {

    private static final Logger LOG = LoggerFactory.getLogger(JaxRsResource.class);
    private static final Object EMPTY_ARG = new Object();
    private final Pattern                           pathPattern;
    private final Method                            method;
    private final Method                            instanceMethod;
    private final Integer                           paramCount;
    private final List               argumentExtractors;
    private final JaxRsResultFactory             resultFactory;
    private final JaxRsMeta                         meta;
    private final RequestLogger                     requestLogger;
    private final Function> methodCaller;

    public JaxRsResource(Method method,
                         Object resourceInstance,
                         ParamResolverFactories paramResolverFactories,
                         JaxRsResultFactoryFactory jaxRsResultFactoryFactory,
                         JaxRsMeta meta,
                         RequestLogger requestLogger) {
        this.method = method;
        this.meta = meta;
        this.pathPattern = createPathPattern(meta.getFullPath());
        this.paramCount = method.getParameterCount();
        this.requestLogger = requestLogger;

        instanceMethod = ReflectionUtil.getInstanceMethod(method, resourceInstance);

        this.argumentExtractors = paramResolverFactories.createParamResolvers(instanceMethod, getConsumes());
        this.resultFactory = jaxRsResultFactoryFactory.createResultFactory(this);
        this.methodCaller = createMethodCaller(method, resourceInstance);
    }

    private static Pattern createPathPattern(String path) {
        // Vars with custom regex, like this: {myvar:myregex}
        path = path.replaceAll("\\{([^}]+):([^}]+)\\}", "(?<$1>$2)");
        // Vars without custom regex, like this: {myvar}
        path = path.replaceAll("\\{([^}]+)\\}", "(?<$1>[^/]+)");
        // Allow trailing slash
        path = "^" + path + "[/\\s]*$";

        return Pattern.compile(path);
    }

    /**
     * Check if this resource can handle a request.
     * @param request the request to check for
     * @return whether the resource can be handled
     */
    public boolean canHandleRequest(JaxRsRequest request) {
        if (!request.hasMethod(meta.getHttpMethod())) {
            return false;
        }
        if (!request.matchesPath(pathPattern)) {
            return false;
        }
        return true;
    }

    protected Mono> call(JaxRsRequest request) {
        return request.loadBody()
            .flatMap(this::resolveArgs)
            .map(this::call);
    }

    private JaxRsResult call(Object[] args) {
        Flux output = methodCaller.apply(args);
        return resultFactory.create(output, args);
    }

    @SuppressWarnings("unchecked")
    private Function> createMethodCaller(Method method, Object resourceInstance) {
        Class returnType = method.getReturnType();
        Function> fluxConverter = FluxRxConverter.converterToFlux(returnType);

        if (!isFlux(returnType) && !isMono(returnType)) {
            throw new IllegalArgumentException(format(
                "Can only serve methods that are of type Flux or Mono. %s had unsupported return type %s",
                method, returnType));
        }

        if (fluxConverter == null) {
            throw new IllegalArgumentException(format(
                    "Can only serve methods that are reactive. %s had unsupported return type %s",
                    method, returnType));
        }

        return args -> {
            try {
                Object result = method.invoke(resourceInstance, args);
                return fluxConverter.apply(result);
            } catch (InvocationTargetException e) {
                return Flux.error(e.getTargetException());
            } catch (Throwable e) {
                return Flux.error(e);
            }
        };
    }

    @SuppressWarnings("unchecked")
    private Mono resolveArgs(JaxRsRequest request) {
        if (argumentExtractors.isEmpty()) {
            return Mono.just(new Object[0]);
        }

        Mono[] obsArgs = new Mono[argumentExtractors.size()];
        for (int i = 0; i < obsArgs.length; i++) {
            obsArgs[i] = argumentExtractors.get(i).resolve(request).defaultIfEmpty(EMPTY_ARG);
        }

        return Mono.zip(asList(obsArgs), array -> {
            for (int i = 0; i < array.length; i++) {
                if (array[i] == EMPTY_ARG) {
                    array[i] = null;
                }
            }
            return array;
        });
    }

    @Override
    public int compareTo(JaxRsResource jaxRsResource) {
        int pathCompare = this.meta.getFullPath().compareTo(jaxRsResource.meta.getFullPath());
        if (pathCompare == 0) {
            return jaxRsResource.paramCount.compareTo(this.paramCount);
        }
        return pathCompare;
    }

    @Override
    public String toString() {
        return format("%1$s\t%2$s (%3$s.%4$s)",
            meta.getHttpMethod(),
            meta.getFullPath(),
            method.getDeclaringClass().getName(),
            method.getName());
    }

    public Method getResourceMethod() {
        return method;
    }

    public Method getInstanceMethod() {
        return instanceMethod;
    }

    public HttpMethod getHttpMethod() {
        return meta.getHttpMethod();
    }

    public String getProduces() {
        return meta.getProduces();
    }

    /**
     * Only supports a single consumes media type for now, because we construct the param extractor at init.
     *
     * @return Annotation value
     */
    private String[] getConsumes() {
        Consumes consumesAnnotation = meta.getConsumes();
        if (consumesAnnotation != null) {
            String[] mediaTypes = consumesAnnotation.value();
            if (mediaTypes.length != 0) {
                return mediaTypes;
            }
        }
        return new String[]{MediaType.APPLICATION_JSON};
    }

    public void log(HttpServerRequest request, HttpServerResponse response, long requestStartTime) {
        requestLogger.log(LOG, request, response, requestStartTime);
    }

    public String getPath() {
        return meta.getFullPath();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy