使用Vert.x实现反向代理

原文发布在:使用Vert.x实现反向代理,欢迎使用 RSS 订阅获取最新更新。

1. 前言

在简单使用Pingora后,我在想既然rust, go都能写代理相关,那为什么不用Java写一个呢?

搜了搜原来用Vert.x实现反向代理这么简单,让我们快速开始吧。

如果你想下载示例代码,可以在这里找到:demo-vertx-reverse-proxy

2. 创建Vert.x项目

Vert.x项目的创建可以在官网找到:Create a new Vert.x application

要选择的依赖则是Vert.x Http Proxy

3. 根据Vert.x的教程开始

官网的教程:Vert.x Http Proxy

3.1 Vert.x Http Proxy

Vert.x Http Proxy 是基于 Vert.x 的反向代理,它旨在实现可重用的反向代理逻辑,以关注更高的问题。

该模块具有技术预览状态,这意味着 API 可以在版本之间更改。

3.2 使用 Vert.x Http Proxy

要使用 Vert.x Http Proxy,请在pom.xml或是其他依赖配置中增加: Maven:

xml 复制代码
<dependency>
    <groupId>io.vertx</groupId>
    <artifactId>vertx-http-proxy</artifactId>
    <version>4.1.8</version>
</dependency>

Gradle:

Gradle 复制代码
dependencies {
    compile 'io.vertx:vertx-http-proxy:4.1.8'
}

3.3 基本 Http 代理

为了使用 Vert.x Http Proxy 完成反向代理,你需要以下内容:

  1. 代理服务器使用 HttpProxy 实例处理出站请求并将其转发给源服务器。
  2. 源服务器处理来自代理服务器的请求并相应地处理响应。

现在,你已经有了总体概念,所以让我们深入实现并从源服务器开始,然后是带有 HttpProxy 的代理服务器:

3.4 源服务

这里可以跳过,因为源服务我们可以直接使用任意你能打开的网站的链接

当然我还是保留了完整步骤。

你只需创建监听端口 7070 的源服务器

Java 复制代码
  @Override
  public void start(Promise<Void> startPromise) throws Exception {
    HttpServer originServer = vertx.createHttpServer();

    originServer.requestHandler(req -> {
      req.response()
        .putHeader("content-type", "text/html")
        .end("<html><body><h1>I'm the target resource!</h1></body></html>");
    }).listen(7070);
  }

3.5 使用 HttpProxy 的代理服务器

使用 HttpProxy 实例创建侦听端口 8080 的代理服务器,该实例相应地处理反向代理逻辑。

java 复制代码
  @Override
  public void start(Promise<Void> startPromise) throws Exception {
    HttpClient proxyClient = vertx.createHttpClient();

    HttpProxy proxy = HttpProxy.reverseProxy(proxyClient);
    proxy.origin(7070, "origin");

    HttpServer proxyServer = vertx.createHttpServer();

    proxyServer.requestHandler(proxy).listen(8080);
  }

最后,所有向外发出的请求都将作为反向代理方便地转发给源服务器。

官方教程到这里就结束了,只是代理某个ip+端口的服务,但是如果想代理某个网站呢?

4. 代理网站

代理某个网站的话,我们需要其实需要代理的是那个网站url+443端口,以及信任网站的证书

所以我们的verticle实现需要改一下,这里我直接使用了自己的博客链接,你可以改为任意地址

java 复制代码
  @Override
  public void start(Promise<Void> startPromise) throws Exception {
    HttpClient proxyClient = vertx.createHttpClient(new HttpClientOptions().setSsl(true).setTrustAll(true));

    // 创建一个 HttpProxy 实例
    HttpProxy proxy = HttpProxy.reverseProxy(proxyClient);
    // 配置目标服务器和端口
    proxy.origin(443, "runnable.run");
    // 设置 HttpServer 监听请求
    HttpServer proxyServer = vertx.createHttpServer();

    proxyServer.requestHandler(req -> {
      // 在此给请求添加 Host 头部
      req.headers().set("Host", "runnable.run");

      // 通过 HttpClient 转发请求到目标服务器
      proxy.handle(req);
    }).listen(8080, res -> {
      if (res.succeeded()) {
        System.out.println("Proxy server is running on port 8080, http://localhost:8080");
        startPromise.complete();
      } else {
        startPromise.fail(res.cause());
      }
    });
  }

然后给服务增加一个main方法启动试试

java 复制代码
  public static void main(String[] args) {
    VertxOptions vertxOptions = new VertxOptions();
    Vertx vertx  = Vertx.vertx(vertxOptions);
    DeploymentOptions deploymentOptions = new DeploymentOptions();
    vertx.deployVerticle(new MainVerticle(), deploymentOptions);
  }

4.1 完整代码如下

java 复制代码
package run.runnable.demo_vertx_reverse_proxy;

import io.vertx.core.*;
import io.vertx.core.http.HttpClient;
import io.vertx.core.http.HttpClientOptions;
import io.vertx.core.http.HttpServer;
import io.vertx.httpproxy.HttpProxy;

public class MainVerticle extends AbstractVerticle {

  @Override
  public void start(Promise<Void> startPromise) throws Exception {
    HttpClient proxyClient = vertx.createHttpClient(new HttpClientOptions().setSsl(true).setTrustAll(true));

    // 创建一个 HttpProxy 实例
    HttpProxy proxy = HttpProxy.reverseProxy(proxyClient);
    // 配置目标服务器和端口
    proxy.origin(443, "runnable.run");
    // 设置 HttpServer 监听请求
    HttpServer proxyServer = vertx.createHttpServer();

    proxyServer.requestHandler(req -> {
      // 在此给请求添加 Host 头部
      req.headers().set("Host", "runnable.run");

      // 通过 HttpClient 转发请求到目标服务器
      proxy.handle(req);
    }).listen(8080, res -> {
      if (res.succeeded()) {
        System.out.println("Proxy server is running on port 8080, http://localhost:8080");
        startPromise.complete();
      } else {
        startPromise.fail(res.cause());
      }
    });
  }

  public static void main(String[] args) {
    VertxOptions vertxOptions = new VertxOptions();
    Vertx vertx  = Vertx.vertx(vertxOptions);
    DeploymentOptions deploymentOptions = new DeploymentOptions();
    vertx.deployVerticle(new MainVerticle(), deploymentOptions);
  }
}

启动之后,使用浏览器打开 http://localhost:8080 就会发现已经是博客的页面了。所以如果你把这个反向代理在香港服务器部署,链接替换为google就。。。

复制代码
相关推荐
magic 2452 分钟前
Lombok 的 @Data 注解失效,未生成 getter/setter 方法引发的HTTP 406 错误
java
爱敲代码的憨仔9 分钟前
分布式协同自动化办公系统-工作流引擎-流程设计
java·flowable·oa
酷爱码22 分钟前
Spring Boot项目中JSON解析库的深度解析与应用实践
spring boot·后端·json
纪元A梦33 分钟前
分布式拜占庭容错算法——PBFT算法深度解析
java·分布式·算法
卿着飞翔35 分钟前
RabbitMQ入门4.1.0版本(基于java、SpringBoot操作)
java·rabbitmq·java-rabbitmq
陈阿土i1 小时前
SpringAI 1.0.0 正式版——利用Redis存储会话(ChatMemory)
java·redis·ai·springai
AI小智1 小时前
Google刀刃向内,开源“深度研究Agent”:Gemini 2.5 + LangGraph 打造搜索终结者!
后端
安全系统学习1 小时前
【网络安全】Qt免杀样本分析
java·网络·安全·web安全·系统安全
java干货1 小时前
虚拟线程与消息队列:Spring Boot 3.5 中异步架构的演进与选择
spring boot·后端·架构
一只叫煤球的猫1 小时前
MySQL 8.0 SQL优化黑科技,面试官都不一定知道!
后端·sql·mysql