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

com.github.mcollovati.vertx.http.HttpReverseProxy Maven / Gradle / Ivy

/*
 * The MIT License
 * Copyright © 2016-2020 Marco Collovati ([email protected])
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 * THE SOFTWARE.
 */
package com.github.mcollovati.vertx.http;

import java.util.concurrent.CompletableFuture;
import java.util.function.UnaryOperator;
import java.util.stream.Stream;

import com.vaadin.base.devserver.ViteHandler;
import com.vaadin.flow.internal.DevModeHandler;
import com.vaadin.flow.internal.UrlUtil;
import com.vaadin.flow.server.StaticFileServer;
import io.netty.handler.codec.http.HttpResponseStatus;
import io.vertx.core.Vertx;
import io.vertx.core.buffer.Buffer;
import io.vertx.core.http.HttpClient;
import io.vertx.core.http.HttpServerRequest;
import io.vertx.core.http.HttpServerResponse;
import io.vertx.ext.web.RoutingContext;
import io.vertx.ext.web.client.HttpRequest;
import io.vertx.ext.web.client.HttpResponse;
import io.vertx.ext.web.client.WebClient;
import io.vertx.ext.web.client.WebClientOptions;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.github.mcollovati.vertx.vaadin.devserver.VertxDevModeHandlerManager;

import static com.vaadin.flow.server.frontend.FrontendUtils.INDEX_HTML;
import static com.vaadin.flow.server.frontend.FrontendUtils.SERVICE_WORKER_SRC_JS;
import static com.vaadin.flow.server.frontend.FrontendUtils.WEB_COMPONENT_HTML;

public class HttpReverseProxy {

    private static final Logger logger = LoggerFactory.getLogger(HttpReverseProxy.class);
    private static final int DEFAULT_TIMEOUT = 120 * 1000;

    private static final String[] FILES_IN_ROOT = new String[] {INDEX_HTML, WEB_COMPONENT_HTML, SERVICE_WORKER_SRC_JS};

    private WebClient client;
    private UnaryOperator uriCustomizer;

    public HttpReverseProxy(CompletableFuture webClientFuture, UnaryOperator uriCustomizer) {
        webClientFuture.whenComplete((wc, err) -> {
            if (err == null) {
                client = wc;
            }
        });
        this.uriCustomizer = uriCustomizer;
    }

    private static UnaryOperator createUriCustomizer(DevModeHandler devModeHandler) {
        boolean isVite = devModeHandler instanceof ViteHandler;
        if (isVite) {
            return path -> {
                if (Stream.of(FILES_IN_ROOT).anyMatch(file -> path.equals("/" + file))) {
                    return "/VAADIN" + path;
                }
                // if ("/index.html".equals(path)) {
                //    return "/VAADIN/index.html";
                // }
                return path;
            };
        }
        return UnaryOperator.identity();
    }

    public static HttpReverseProxy create(Vertx vertx, DevModeHandler devModeHandler) {
        CompletableFuture webClientFuture = VertxDevModeHandlerManager.getDevModeHandlerPort(
                        devModeHandler)
                .thenApply(port -> {
                    logger.debug("Starting DevMode proxy on port {}", port);
                    WebClientOptions options = new WebClientOptions()
                            .setLogActivity(true)
                            .setConnectTimeout(DEFAULT_TIMEOUT)
                            .setIdleTimeout(DEFAULT_TIMEOUT)
                            .setDefaultHost("localhost")
                            .setDefaultPort(port);
                    return WebClient.create(vertx, options);
                });
        return new HttpReverseProxy(webClientFuture, createUriCustomizer(devModeHandler));
    }

    public void forward(RoutingContext routingContext) {
        HttpServerRequest serverRequest = routingContext.request();
        String requestURI = serverRequest.uri();
        if (client == null) {
            logger.debug("DevMode serve not yet started");
            routingContext.next();
        } else if (requestURI.equals("") || requestURI.equals("/")) {
            // Index file must be handled by IndexHtmlRequestHandler
            logger.debug("Not a DevMode server request, Index file must be handled by IndexHtmlRequestHandler");
            routingContext.next();
        } else {

            if (StaticFileServer.APP_THEME_ASSETS_PATTERN.matcher(requestURI).find()) {
                requestURI = "/VAADIN/static" + requestURI;
            }
            requestURI = uriCustomizer.apply(requestURI);

            logger.debug("Forwarding {} to dev-server as {}", serverRequest.uri(), requestURI);

            String devServerRequestPath = UrlUtil.encodeURI(requestURI)
                    + ((serverRequest.query() != null) ? "?" + serverRequest.query() : "");

            HttpRequest clientRequest = client.request(serverRequest.method(), devServerRequestPath);
            serverRequest.headers().forEach(entry -> {
                String valueOk = "Connection".equals(entry.getKey()) ? "close" : entry.getValue();
                clientRequest.putHeader(entry.getKey(), valueOk);
            });

            clientRequest.sendBuffer(routingContext.getBody(), ar -> {
                if (ar.succeeded()) {
                    HttpResponse clientResponse = ar.result();
                    int statusCode = clientResponse.statusCode();
                    HttpServerResponse serverResponse = routingContext.response();
                    if (statusCode == HttpResponseStatus.OK.code()) {
                        logger.debug(
                                "Served resource by dev-server: {} {}",
                                clientResponse.statusCode(),
                                devServerRequestPath);
                        serverResponse.setStatusCode(statusCode).setChunked(true);
                        serverResponse.headers().setAll(clientResponse.headers());
                        serverResponse.end(clientResponse.body());
                    } else if (statusCode == HttpResponseStatus.NOT_FOUND.code()) {
                        logger.debug("Resource not served by dev-server {}", devServerRequestPath);
                        routingContext.next();
                    } else if (statusCode < 400) {
                        logger.debug("dev-server response {} for resource {}", statusCode, devServerRequestPath);
                        serverResponse.headers().setAll(clientResponse.headers());
                        serverResponse.setStatusCode(statusCode).end();
                    } else {
                        logger.debug(
                                "dev-server failed with status {} for resource {}", statusCode, devServerRequestPath);
                        routingContext.fail(statusCode);
                    }
                } else {
                    logger.warn("Request to dev-server failed: {}", devServerRequestPath, ar.cause());
                    routingContext.next();
                }
            });
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy