使用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就。。。

复制代码
相关推荐
爱吃南瓜的北瓜10 分钟前
Maven之jjwt依赖爆红
java·pycharm·maven
豌豆花下猫11 分钟前
Python 潮流周刊#90:uv 一周岁了,优缺点分析(摘要)
后端·python·ai
土豆儿@20 分钟前
java之泛型
java·开发语言
橘猫云计算机设计31 分钟前
基于SSM的《计算机网络》题库管理系统(源码+lw+部署文档+讲解),源码可白嫖!
java·数据库·spring boot·后端·python·计算机网络·毕设
熬夜苦读学习1 小时前
Linux文件系统
linux·运维·服务器·开发语言·后端
菜鸟一枚在这1 小时前
深度解析建造者模式:复杂对象构建的优雅之道
java·开发语言·算法
gyeolhada1 小时前
2025蓝桥杯JAVA编程题练习Day5
java·数据结构·算法·蓝桥杯
菜鸟一枚在这1 小时前
深入理解设计模式之代理模式
java·设计模式·代理模式
坚定信念,勇往无前2 小时前
Spring Boot 如何保证接口安全
spring boot·后端·安全
小天努力学java2 小时前
【面试系列】Java开发--AI常见面试题
java·人工智能·面试