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

org.swisspush.gateleen.routing.Forwarder Maven / Gradle / Ivy

There is a newer version: 2.1.13
Show newest version
package org.swisspush.gateleen.routing;

import io.netty.channel.ConnectTimeoutException;
import io.netty.handler.codec.http.HttpResponseStatus;
import io.vertx.core.*;
import io.vertx.core.buffer.Buffer;
import io.vertx.core.http.*;
import io.vertx.core.json.JsonObject;
import io.vertx.core.streams.Pump;
import io.vertx.core.streams.WriteStream;
import io.vertx.ext.web.RoutingContext;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.swisspush.gateleen.core.http.HeaderFunctions;
import org.swisspush.gateleen.core.http.RequestLoggerFactory;
import org.swisspush.gateleen.core.storage.ResourceStorage;
import org.swisspush.gateleen.core.util.HttpHeaderUtil;
import org.swisspush.gateleen.core.util.ResponseStatusCodeLogUtil;
import org.swisspush.gateleen.core.util.StatusCode;
import org.swisspush.gateleen.core.util.StringUtils;
import org.swisspush.gateleen.logging.LogAppenderRepository;
import org.swisspush.gateleen.logging.LoggingHandler;
import org.swisspush.gateleen.logging.LoggingResourceManager;
import org.swisspush.gateleen.logging.LoggingWriteStream;
import org.swisspush.gateleen.monitoring.MonitoringHandler;
import org.swisspush.gateleen.routing.auth.AuthHeader;
import org.swisspush.gateleen.routing.auth.AuthStrategy;

import javax.annotation.Nullable;
import java.util.HashMap;
import java.util.Map;
import java.util.Optional;
import java.util.concurrent.TimeoutException;
import java.util.regex.Pattern;

/**
 * Forwards requests to the backend.
 *
 * @author https://github.com/lbovet [Laurent Bovet]
 */
public class Forwarder extends AbstractForwarder {

    private final String userProfilePath;
    private final HttpClient client;
    private final Pattern urlPattern;
    private String target;
    private int port;
    private final Rule rule;
    private final LoggingResourceManager loggingResourceManager;
    private final LogAppenderRepository logAppenderRepository;
    private final MonitoringHandler monitoringHandler;
    private final ResourceStorage storage;
    @Nullable
    private final AuthStrategy authStrategy;
    private final Vertx vertx;

    private static final String ON_BEHALF_OF_HEADER = "x-on-behalf-of";
    private static final String USER_HEADER = "x-rp-usr";
    private static final String USER_HEADER_PREFIX = "x-user-";
    private static final String ETAG_HEADER = "Etag";
    private static final String IF_NONE_MATCH_HEADER = "if-none-match";
    private static final String SELF_REQUEST_HEADER = "x-self-request";
    private static final String HOST_HEADER = "Host";
    private static final int STATUS_CODE_2XX = 2;

    private static final Logger LOG = LoggerFactory.getLogger(Forwarder.class);

    public Forwarder(Vertx vertx, HttpClient client, Rule rule, final ResourceStorage storage,
                     LoggingResourceManager loggingResourceManager, LogAppenderRepository logAppenderRepository, MonitoringHandler monitoringHandler,
                     String userProfilePath, @Nullable AuthStrategy authStrategy) {
        super(rule, loggingResourceManager, logAppenderRepository, monitoringHandler);
        this.vertx = vertx;
        this.client = client;
        this.rule = rule;
        this.loggingResourceManager = loggingResourceManager;
        this.logAppenderRepository = logAppenderRepository;
        this.monitoringHandler = monitoringHandler;
        this.storage = storage;
        this.urlPattern = Pattern.compile(rule.getUrlPattern());
        this.target = rule.getHost() + ":" + rule.getPort();
        this.userProfilePath = userProfilePath;
        this.authStrategy = authStrategy;
    }

    private Map createProfileHeaderValues(JsonObject profile, Logger log) {
        Map profileValues = new HashMap<>();
        if (rule.getProfile() != null) {
            String[] ruleProfile = rule.getProfile();
            for (String headerKey : ruleProfile) {
                String headerValue = profile.getString(headerKey);
                if (headerKey != null && headerValue != null) {
                    profileValues.put(USER_HEADER_PREFIX + headerKey, headerValue);
                    log.debug("Sending header-information for key {}, value = {}", headerKey, headerValue);
                } else {
                    if (headerKey != null) {
                        log.debug("We should send profile information '{}' but this information was not found in profile.", headerKey);
                    } else {
                        log.debug("We should send profile information but header key is null.");
                    }
                }
            }
        } else {
            log.debug("rule.profile is null, this rule will not send profile information.");
        }
        return profileValues;
    }

    @Override
    public void handle(final RoutingContext ctx) {
        handle(ctx, null, null);
    }

    /**
     * Allows to handle a request which was already consumed.
     * If the parameter bodyData is not null, the
     * request was consumed, and you can't read the body of the
     * request again.
     *
     * @param ctx      - the original request context
     * @param bodyData - a buffer with the body data, null if the request
     *                 was not yet consumed
     */
    public void handle(final RoutingContext ctx, final Buffer bodyData, @Nullable final Handler afterHandler) {
        HttpServerRequest req = ctx.request();
        final Logger log = RequestLoggerFactory.getLogger(Forwarder.class, req);

        if (rule.hasHeadersFilterPattern() && !doHeadersFilterMatch(ctx.request())) {
            ctx.next();
            return;
        }

        if (rule.hasPortWildcard()) {
            String dynamicPortStr = null;
            try {
                dynamicPortStr = urlPattern.matcher(req.uri()).replaceFirst(rule.getPortWildcard());
                log.debug("Dynamic port for wildcard {} is {}", rule.getPortWildcard(), dynamicPortStr);
                port = Integer.parseInt(dynamicPortStr);
            } catch (NumberFormatException ex) {
                log.error("Could not extract a numeric value from wildcard {}. Got {}", rule.getPortWildcard(), dynamicPortStr);
                respondError(req, StatusCode.INTERNAL_SERVER_ERROR);
                return;
            } catch (IndexOutOfBoundsException ex) {
                log.error("No group could be found for wildcard {}", rule.getPortWildcard());
                respondError(req, StatusCode.INTERNAL_SERVER_ERROR);
                return;
            }
        } else {
            port = rule.getPort();
        }
        target = rule.getHost() + ":" + port;
        monitoringHandler.updateRequestsMeter(target, req.uri());
        monitoringHandler.updateRequestPerRuleMonitoring(req, rule.getMetricName());
        final String targetUri = urlPattern.matcher(req.uri()).replaceFirst(rule.getPath()).replaceAll("\\/\\/", "/");
        log.debug("Forwarding request: {} to {}://{} with rule {}", req.uri(), rule.getScheme(), target + targetUri, rule.getRuleIdentifier());
        final String userId = extractUserId(req, log);
        req.pause(); // pause the request to avoid problems with starting another async request (storage)

        maybeAuthenticate(rule).onComplete(event -> {
            if(event.failed()) {
                req.resume();
                log.error("Failed to authenticate request. Cause: {}", event.cause().getMessage());
                respondError(req, StatusCode.UNAUTHORIZED);
                return;
            }
            Optional authHeader = event.result();
            if (userId != null && rule.getProfile() != null && userProfilePath != null) {
                log.debug("Get profile information for user '{}' to append to headers", userId);
                String userProfileKey = String.format(userProfilePath, userId);
                storage.get(userProfileKey, buffer -> {
                    Map profileHeaderMap = new HashMap<>();
                    if (buffer != null) {
                        JsonObject profile = new JsonObject(buffer.toString());
                        profileHeaderMap = createProfileHeaderValues(profile, log);
                        log.debug("Got profile information of user '{}'", userId);
                        log.debug("Going to send parts of the profile in header: {}", profileHeaderMap);
                    } else {
                        log.debug("No profile information found in local storage for user '{}'", userId);
                    }
                    handleRequest(req, bodyData, targetUri, log, profileHeaderMap, authHeader, afterHandler);
                });
            } else {
                handleRequest(req, bodyData, targetUri, log, null, authHeader, afterHandler);
            }
        });
    }

    /**
     * Returns the userId defined in the on-behalf-of-header if provided, the userId from user-header otherwise.
     *
     * @param request request
     * @param log     log
     */
    private String extractUserId(HttpServerRequest request, Logger log) {
        String onBehalfOf = StringUtils.getStringOrEmpty(request.headers().get(ON_BEHALF_OF_HEADER));
        if (StringUtils.isNotEmpty(onBehalfOf)) {
            log.debug("Using values from x-on-behalf-of header instead of taking them from x-rp-usr header");
            return onBehalfOf;
        } else {
            return request.headers().get(USER_HEADER);
        }
    }

    /**
     * Execute the HeaderFunctions chain and apply the configured rules headers therefore
     *
     * @param log     The logger to be used
     * @param headers The headers which must be updated according the current forwarders rule
     * @return null if everything is properly done or a error message if something went wrong.
     */
    String applyHeaderFunctions(final Logger log, MultiMap headers) {
        final String hostHeaderBefore = HttpHeaderUtil.getHeaderValue(headers, HOST_HEADER);
        final HeaderFunctions.EvalScope evalScope = rule.getHeaderFunction().apply(headers);
        final String hostHeaderAfter = HttpHeaderUtil.getHeaderValue(headers, HOST_HEADER);
        // see https://github.com/swisspush/gateleen/issues/394
        if (hostHeaderAfter == null || hostHeaderAfter.equals(hostHeaderBefore)) {
            // there was no host header before or the host header was not updated by the rule given,
            // therefore it will be forced overwritten independent of the incoming value if necessary.
            final String newHost = target.split("/")[0];
            if (newHost != null && !newHost.isEmpty() && !newHost.equals(hostHeaderAfter)) {
                headers.set(HOST_HEADER, newHost);
                log.debug("Host header {} replaced by default target value: {}",
                        hostHeaderBefore,
                        newHost);
            }
        } else {
            // the host header was changed by the configured routing and therefore
            // it is not updated. This allows us to configure for certain routings to external
            // url a dedicated Host header which will not be overwritten.
            log.debug("Host header {} replaced by rule value: {}",
                    hostHeaderBefore,
                    hostHeaderAfter);
        }
        return evalScope.getErrorMessage();
    }

    private void handleRequest(final HttpServerRequest req, final Buffer bodyData, final String targetUri,
                               final Logger log, final Map profileHeaderMap,
                               Optional authHeader, @Nullable final Handler afterHandler) {
        final LoggingHandler loggingHandler = new LoggingHandler(loggingResourceManager, logAppenderRepository, req, vertx.eventBus());

        final String uniqueId = req.headers().get("x-rp-unique_id");
        final String timeout = req.headers().get("x-timeout");
        final long startTime = monitoringHandler.startRequestMetricTracking(rule.getMetricName(), req.uri());

        client.request(req.method(), port, rule.getHost(), targetUri, new Handler<>() {
            @Override
            public void handle(AsyncResult event) {
                req.resume();

                if (event.failed()) {
                    log.warn("Problem to request {}: {}", targetUri, event.cause());
                    final HttpServerResponse response = req.response();
                    response.setStatusCode(StatusCode.SERVICE_UNAVAILABLE.getStatusCode());
                    response.setStatusMessage(StatusCode.SERVICE_UNAVAILABLE.getStatusMessage());
                    response.end();
                    return;
                }
                HttpClientRequest cReq = event.result();
                final Handler> cResHandler = getAsyncHttpClientResponseHandler(req, targetUri, log, profileHeaderMap, loggingHandler, startTime, afterHandler);
                cReq.response(cResHandler);

                if (timeout != null) {
                    cReq.idleTimeout(Long.parseLong(timeout));
                } else {
                    cReq.idleTimeout(rule.getTimeout());
                }

                // per https://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.10
                MultiMap headersToForward = req.headers();
                headersToForward = HttpHeaderUtil.removeNonForwardHeaders(headersToForward);
                HttpHeaderUtil.mergeHeaders(cReq.headers(), headersToForward, targetUri);
                if (!ResponseStatusCodeLogUtil.isRequestToExternalTarget(target)) {
                    cReq.headers().set(SELF_REQUEST_HEADER, "true");
                }

                if (uniqueId != null) {
                    cReq.headers().set("x-rp-unique-id", uniqueId);
                }
                setProfileHeaders(log, profileHeaderMap, cReq);

                authHeader.ifPresent(authHeaderValue -> cReq.headers().set(authHeaderValue.key(), authHeaderValue.value()));

                final String errorMessage = applyHeaderFunctions(log, cReq.headers());
                if (errorMessage != null) {
                    log.warn("Problem invoking Header functions: {}", errorMessage);
                    final HttpServerResponse response = req.response();
                    response.setStatusCode(StatusCode.BAD_REQUEST.getStatusCode());
                    response.setStatusMessage(StatusCode.BAD_REQUEST.getStatusMessage());
                    response.end(errorMessage);
                    return;
                }

                installExceptionHandler(req, targetUri, startTime, cReq);

                /*
                 * If no bodyData is available
                 * this means, that the request body isn't
                 * consumed yet. So we can use the regular
                 * request for the data.
                 * If the body is already consumed, we use
                 * the buffer bodyData.
                 */
                if (bodyData == null) {

                    // Gateleen internal requests (e.g. from scedulers or delegates) often have neither "Content-Length" nor "Transfer-Encoding: chunked"
                    // header - so we must wait for a body buffer to know: Is there a body or not? Only looking on the headers and/or the http-method is not
                    // sustainable to know "has body or not"
                    // But: if there is a body, then we need to either setChunked or a Content-Length header (otherwise Vertx complains with an Exception)
                    //
                    // Setting 'chunked' always has the downside that we use it also for GET, HEAD, OPTIONS etc... Those request methods normally have no body at all
                    // But still it's allowed - so they 'could' have one. So using http-method to decide "chunked or not" is also not a sustainable solution.
                    //
                    // --> we need to wrap the client-Request to catch up the first (body)-buffer and "setChucked(true)" in advance and just-in-time.
                    WriteStream cReqWrapped = new WriteStream<>() {
                        private boolean firstBuffer = true;

                        @Override
                        public WriteStream exceptionHandler(Handler handler) {
                            cReq.exceptionHandler(handler);
                            return this;
                        }

                        @Override
                        public Future write(Buffer data) {
                            // only now we know for sure that there IS a body.
                            if (firstBuffer) {
                                // avoid multiple calls due to a 'syncronized' block in HttpClient's implementation
                                firstBuffer = false;
                                cReq.setChunked(true);
                            }
                            return cReq.write(data);
                        }

                        @Override
                        public void write(Buffer data, Handler> handler) {
                            write(data).onComplete(handler);
                        }

                        @Override
                        public Future end() {
                            Promise promise = Promise.promise();
                            cReq.send(asyncResult -> {
                                        cResHandler.handle(asyncResult);
                                        promise.complete();
                                    }
                            );
                            return promise.future();
                        }

                        @Override
                        public void end(Handler> handler) {
                            this.end().onComplete(handler);
                        }

                        @Override
                        public WriteStream setWriteQueueMaxSize(int maxSize) {
                            cReq.setWriteQueueMaxSize(maxSize);
                            return this;
                        }

                        @Override
                        public boolean writeQueueFull() {
                            return cReq.writeQueueFull();
                        }

                        @Override
                        public WriteStream drainHandler(@Nullable Handler handler) {
                            cReq.drainHandler(handler);
                            return this;
                        }
                    };

                    req.exceptionHandler(t -> {
                        log.info("Exception during forwarding - closing (forwarding) client connection", t);
                        HttpConnection connection = cReq.connection();
                        if (connection != null) {
                            connection.close();
                        } else {
                            log.warn("There's no connection we could close in {}, gateleen wishes your request a happy timeout ({})",
                                    cReq.getClass(), req.uri());
                        }
                    });

                    final LoggingWriteStream loggingWriteStream = new LoggingWriteStream(cReqWrapped, loggingHandler, true);
                    final Pump pump = Pump.pump(req, loggingWriteStream);
                    if (req.isEnded()) {
                        // since Vert.x 3.6.0 it can happen that requests without body (e.g. a GET) are ended even while in paused-State
                        // Setting the endHandler would then lead to an Exception
                        // see also https://github.com/eclipse-vertx/vert.x/issues/2763
                        // so we now check if the request already is ended before installing an endHandler
                        cReq.send(cResHandler);
                    } else {
                        req.endHandler(v -> cReq.send(cResHandler));
                        pump.start();
                    }
                } else {
                    loggingHandler.appendRequestPayload(bodyData);
                    // we already have the body complete in-memory - so we can use Content-Length header and avoid chunked transfer
                    cReq.putHeader(HttpHeaders.CONTENT_LENGTH, Integer.toString(bodyData.length()));
                    cReq.send(bodyData, cResHandler);
                }

                loggingHandler.request(cReq.headers());
            }
        });
    }

    private Future> maybeAuthenticate(Rule rule) {
        if (authStrategy == null) {
            return Future.succeededFuture(Optional.empty());
        }
        return authStrategy.authenticate(rule).compose(authHeader -> Future.succeededFuture(Optional.of(authHeader)));
    }

    private void setProfileHeaders(Logger log, Map profileHeaderMap, HttpClientRequest cReq) {
        if (profileHeaderMap != null && !profileHeaderMap.isEmpty()) {
            log.debug("Putting partial profile to header for the backend request (profileHeaderMap).");
            for (Map.Entry entry : profileHeaderMap.entrySet()) {
                cReq.headers().set(entry.getKey(), entry.getValue());
            }
        }
    }

    private void installExceptionHandler(final HttpServerRequest req, final String targetUri, final long startTime, HttpClientRequest cReq) {
        cReq.exceptionHandler(exception -> {
            monitoringHandler.stopRequestMetricTracking(rule.getMetricName(), startTime, req.uri());
            if (exception instanceof TimeoutException) {
                error("Timeout", req, targetUri);
                respondError(req, StatusCode.TIMEOUT);
            } else {
                if (exception instanceof ConnectTimeoutException) {
                    // Don't log stacktrace in case connection timeout
                    LOG.warn("Failed to '{} {}'", req.method(), targetUri);
                } else {
                    LOG.warn("Failed to '{} {}'", req.method(), targetUri, exception);
                }
                error(exception.getMessage(), req, targetUri);
                if (req.response().ended() || req.response().headWritten()) {
                    LOG.info("{}: Response already written. Not sure about the state. Closing server connection for stability reason", targetUri);
                    req.response().close();
                    return;
                }
                respondError(req, StatusCode.SERVICE_UNAVAILABLE);
            }
        });
    }

    private Handler> getAsyncHttpClientResponseHandler(final HttpServerRequest req, final String targetUri, final Logger log, final Map profileHeaderMap, final LoggingHandler loggingHandler, final long startTime, @Nullable final Handler afterHandler) {
        return asyncResult -> {
            HttpClientResponse cRes = asyncResult.result();
            if (asyncResult.failed()) {
                error(asyncResult.cause().getMessage(), req, targetUri);
                HttpServerResponse rsp = req.response();
                int rspCode = HttpResponseStatus.INTERNAL_SERVER_ERROR.code();
                String shortMsg = asyncResult.cause().getMessage();
                if (rsp.headWritten()) {
                    log.warn("Already responded. Cannot send anymore: HTTP {} {}", rspCode, shortMsg);
                } else {
                    rsp.setStatusCode(rspCode);
                    rsp.setStatusMessage(shortMsg);
                    rsp.end();
                }
                return;
            }
            monitoringHandler.stopRequestMetricTracking(rule.getMetricName(), startTime, req.uri());
            loggingHandler.setResponse(cRes);
            req.response().setStatusCode(cRes.statusCode());
            req.response().setStatusMessage(cRes.statusMessage());

            int statusCode = cRes.statusCode();

            // translate with header info
            int translatedStatus = Translator.translateStatusCode(statusCode, req.headers());

            // nothing changed?
            if (statusCode == translatedStatus) {
                translatedStatus = Translator.translateStatusCode(statusCode, rule, log);
            }

            boolean translated = statusCode != translatedStatus;

            // set the statusCode (if nothing hapend, it will remain the same)
            statusCode = translatedStatus;

            req.response().setStatusCode(statusCode);

            if (translated) {
                req.response().setStatusMessage(HttpResponseStatus.valueOf(statusCode).reasonPhrase());
            }

            // Add received headers to original request but remove headers that should not get forwarded.
            MultiMap headersToForward = cRes.headers();
            headersToForward = HttpHeaderUtil.removeNonForwardHeaders(headersToForward);
            HttpHeaderUtil.mergeHeaders(req.response().headers(), headersToForward, targetUri);
            if (profileHeaderMap != null && !profileHeaderMap.isEmpty()) {
                HttpHeaderUtil.mergeHeaders(req.response().headers(), MultiMap.caseInsensitiveMultiMap().addAll(profileHeaderMap), targetUri);
            }
            // if we receive a chunked transfer then we also use chunked
            // otherwise, upstream must have sent a Content-Length - or no body at all (e.g. for "304 not modified" responses)
            if (req.response().headers().contains(HttpHeaders.TRANSFER_ENCODING, "chunked", true)) {
                req.response().setChunked(true);
            }

            final LoggingWriteStream loggingWriteStream = new LoggingWriteStream(req.response(), loggingHandler, false);
            final Pump pump = Pump.pump(cRes, loggingWriteStream);
            Handler cResEndHandler = v -> {
                try {
                    req.response().end();

                    // if everything is fine, we call the after handler
                    if (afterHandler != null && (cRes.statusCode() / 100) == STATUS_CODE_2XX) {
                        afterHandler.handle(null);
                    }
                    ResponseStatusCodeLogUtil.debug(req, StatusCode.fromCode(req.response().getStatusCode()), Forwarder.class);
                } catch (IllegalStateException e) {
                    // ignore because maybe already closed
                }
                vertx.runOnContext(event -> loggingHandler.log());
            };
            try {
                cRes.endHandler(cResEndHandler);
            } catch (IllegalStateException ex) {
                log.warn("cRes.endHandler() failed", ex);
                respondError(req, StatusCode.INTERNAL_SERVER_ERROR);
                return;
            }
            pump.start();

            Runnable unpump = () -> {
                // disconnect the clientResponse from the Pump and resume this (probably paused-by-pump) stream to keep it alive
                pump.stop();
//                cRes.handler(buf -> {
//                    // drain to nothing
//                });
                cRes.resume(); // resume the (probably paused) stream
            };

            cRes.exceptionHandler(exception -> {
                LOG.warn("Failed to read upstream response for '{} {}'", req.method(), targetUri, exception);
                unpump.run();
                error("Problem with backend: " + exception.getMessage(), req, targetUri);
                respondError(req, StatusCode.INTERNAL_SERVER_ERROR);
            });

            HttpConnection connection = req.connection();
            if (connection != null) {
                connection.closeHandler((Void v) -> unpump.run());
            } else {
                log.warn("TODO No way to call 'unpump.run()' in the right moment. As there seems"
                        + " to be no event we could register a handler for. Gateleen wishes you"
                        + " some happy timeouts ({})", req.uri());
            }
        };
    }

    private void error(String message, HttpServerRequest request, String uri) {
        RequestLoggerFactory.getLogger(Forwarder.class, request).error(rule.getScheme() + "://" + target + uri + " " + message);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy