smiley-http-proxy-servlet 反向代理

简介: 背景 反向代理(Reverse Proxy)方式是指以代理服务器来接受internet上的连接请求,然后将请求转发给内部网络上的服务器,并将从服务器上得到的结果返回给internet上请求连接的客户端,此时代理服务器对外就表现为一个反向代理服务器。

背景

 反向代理(Reverse Proxy)方式是指以代理服务器来接受internet上的连接请求,然后将请求转发给内部网络上的服务器,并将从服务器上得到的结果返回给internet上请求连接的客户端,此时代理服务器对外就表现为一个反向代理服务器。

 简单来说,你的反向代理服务器会接收请求,但其自身不处理该请求,而是对请求经过一些处理,例如添加日志、缓存、身份验证等服务,然后再将请求转发到相应的应用服务器中进行处理,最后将处理结果返回。

 HTTP-Proxy-Servlet是基于Filter进行服务代理,只需要进行相关的配置之后,就能进行服务代理,同时核心文件只有2个,容易理解和进行改动,但是存在的问题是只能代理一个url,不能代理多个不同的url,这部分是可以基于源码进行改造。


源码目录

img_0299b9337bcf040eb34d1d14eb09e498.png
http-proxy-servlet
  • http-proxy-servlet的源码文件非常简单,只有两个文件。
  • ProxyServlet.java重写了spring的HttpServlet,参数初始化都在ProxyServlet初始化。
  • 针对改造我们只需要改造ProxyServlet.java文件即可。


源码改造

  • http-proxy-servlet的代理配置针对代理URL只能定位到一个URL,如果想内部逻辑判断转发多个URL只能进行代码改造。
  • 核心改造逻辑在于改造init()方法、initTarget()方法、service()方法。
  • initTarget()方法内部解析spring的配置初始化核心对象。
  • service()方法内部根据逻辑判断重定向逻辑。
public class ProxyServlet extends HttpServlet {
 
  // 省略相关代码
  protected String targetUri;
  protected URI targetUriObj;//new URI(targetUri)
  protected HttpHost targetHost;//URIUtils.extractHost(targetUriObj);

  // 晴天哥新增代码
  protected String nTargetUri;
  protected URI nTargetUriObj;//new URI(nTargetUri)
  protected HttpHost nTargetHost;//URIUtils.extractHost(nTargetUriObj);

  private static AtomicLong counter = new AtomicLong(0);

  private HttpClient proxyClient;

  @Override
  public void init() throws ServletException {
    // 省略相关代码

    // 核心在于初始化这个链接
    initTarget();//sets target*

    proxyClient = createHttpClient(buildRequestConfig());
  }


  protected RequestConfig buildRequestConfig() {
    // 省略相关代码
  }

  protected void initTarget() throws ServletException {
    targetUri = getConfigParam(P_TARGET_URI);
    if (targetUri == null) {
        throw new ServletException(P_TARGET_URI + " is required.");
    }
    //test it's valid
    try {
      targetUriObj = new URI(targetUri);
    } catch (Exception e) {
      throw new ServletException("Trying to process targetUri init parameter: "+e,e);
    }
    targetHost = URIUtils.extractHost(targetUriObj);

    // 晴天哥新增代码
    nTargetUri = getConfigParam(N_P_TARGET_URI);
    if (null == nTargetUri) {
        throw new ServletException(N_P_TARGET_URI + " is required.");
    }

    try {
        nTargetUriObj = new URI(nTargetUri);
    } catch (Exception e) {
        throw new ServletException("Trying to process targetUri init parameter: "+e,e);
    }

    nTargetHost = URIUtils.extractHost(nTargetUriObj);
  }


  protected void service(HttpServletRequest servletRequest, HttpServletResponse servletResponse)
      throws ServletException, IOException {
      // 晴天哥新增代码
      long flag = counter.incrementAndGet();
    if (servletRequest.getAttribute(ATTR_TARGET_URI) == null) {
        if (0 == flag % 2) {
            servletRequest.setAttribute(ATTR_TARGET_URI, targetUri);
        } else {
            servletRequest.setAttribute(ATTR_TARGET_URI, nTargetUri);
        }
    }
    if (servletRequest.getAttribute(ATTR_TARGET_HOST) == null) {
        if (0 == flag % 2) {
            servletRequest.setAttribute(ATTR_TARGET_HOST, targetHost);
        } else {
            servletRequest.setAttribute(ATTR_TARGET_HOST, nTargetHost);
        }
    }

    String method = servletRequest.getMethod();
    String proxyRequestUri = rewriteUrlFromRequest(servletRequest);
    HttpRequest proxyRequest;

    if (servletRequest.getHeader(HttpHeaders.CONTENT_LENGTH) != null ||
        servletRequest.getHeader(HttpHeaders.TRANSFER_ENCODING) != null) {
      proxyRequest = newProxyRequestWithEntity(method, proxyRequestUri, servletRequest);
    } else {
      proxyRequest = new BasicHttpRequest(method, proxyRequestUri);
    }

    copyRequestHeaders(servletRequest, proxyRequest);

    setXForwardedForHeader(servletRequest, proxyRequest);

    HttpResponse proxyResponse = null;
    try {
      // 执行请求部分逻辑
      proxyResponse = doExecute(servletRequest, servletResponse, proxyRequest);

      // 省略相关代码
    } catch (Exception e) {
      handleRequestException(proxyRequest, e);
    }
  }
}


demo

  • 1、修改源码后mvn打包生成jar包。
  • 2、引入pom.xml依赖包
  <dependencies>
    <dependency>
      <groupId>org.mitre.dsmiley.httpproxy</groupId>
      <artifactId>smiley-http-proxy-servlet</artifactId>
        <version>1.11-SNAPSHOT</version>
    </dependency>
  </dependencies>
  • 3、 配置spring web.xml
<web-app>
  <display-name>Archetype Created Web Application</display-name>

  <servlet>
    <servlet-name>solr</servlet-name>
    <servlet-class>org.mitre.dsmiley.httpproxy.ProxyServlet</servlet-class>
    <init-param>
      <param-name>targetUri</param-name>
      <param-value>http://sports.sina.com.cn:80/nba/</param-value>
    </init-param>
      <init-param>
          <param-name>n_targetUri</param-name>
          <param-value>http://sports.qq.com:80</param-value>
      </init-param>
    <init-param>
      <param-name>log</param-name>
      <param-value>true</param-value>
    </init-param>
  </servlet>

  <servlet-mapping>
    <servlet-name>solr</servlet-name>
    <url-pattern>/solr/*</url-pattern>
  </servlet-mapping>

</web-app>

效果图

  • localhost:8080/solr的请求可以定向到新浪体育或腾讯体育地址
img_7c0a2ad9b93abe3469fefeb3ad4f16c4.png

img_516a0408304cbd98452978c2881452dd.png


参考地址

https://github.com/lebron374/HTTP-Proxy-Servlet

目录
相关文章
SpringBoot使用ProxyServlet实现HTTP反向代理
SpringBoot使用ProxyServlet实现HTTP反向代理
2067 0
|
缓存 负载均衡 应用服务中间件
Nginx七层(应用层)反向代理:HTTP反向代理proxy_pass篇
通过使用Nginx的反向代理功能,可以有效地提高Web应用的性能、安全性和可扩展性。配置过程中需要注意不同场景下的具体需求,如负载均衡、SSL终止和缓存策略等。正确配置和优化Nginx反向代理可以显著提升系统的整体表现。
2055 20
|
负载均衡 Unix 应用服务中间件
深入解析HTTP反向代理:探索NGINX的神奇之处
深入解析HTTP反向代理:探索NGINX的神奇之处
|
前端开发 JavaScript
node反向代理,解决跨域(http-proxy-middleware)
使用node.js和http-proxy-middleware库实现反向代理,解决跨域问题,允许前端请求通过代理访问不同端口的服务。
733 3
|
缓存 算法 应用服务中间件
[Apache,反向代理,日志,默认值,HTTP]Nginx日志配置详细教程
  nginx日志配置   nginx有一个非常灵活的日志记录模式。每个级别的配置可以有各自独立的访问日志。日志格式通过log_format命令来定义。ngx_http_log_module是用来定义请求日志格式的。   access_log指令   access_log path [format [buffer=size [flush=time]]];   access_log path format gzip[=level] [buffer=size] [flush=time];
738 0
|
缓存 应用服务中间件 API
Nginx七层(应用层)反向代理:HTTP反向代理proxy_pass篇(三)
Nginx七层(应用层)反向代理:HTTP反向代理proxy_pass篇(三)
419 3
|
缓存 安全 应用服务中间件
Nginx七层(应用层)反向代理:HTTP反向代理proxy_pass篇(二)
Nginx七层(应用层)反向代理:HTTP反向代理proxy_pass篇(二)
856 1
|
负载均衡 应用服务中间件 网络安全
Nginx七层(应用层)反向代理:HTTP反向代理proxy_pass篇(一)
Nginx七层(应用层)反向代理:HTTP反向代理proxy_pass篇(一)
3075 0
|
Java 应用服务中间件 Apache
Apache HTTP配置反向代理入门
Apache HTTP配置反向代理入门
733 0
Apache HTTP配置反向代理入门
|
负载均衡 网络协议 应用服务中间件
nginx-http反向代理与负载均衡
nginx-http反向代理与负载均衡
217 0

热门文章

最新文章