java - 使用Netty搭建http代理服务器的简单方法?

标签 java netty

我是 Netty 的新手,正在考虑使用它来制作一个简单的 http 代理服务器,该服务器从客户端接收请求,将请求转发到另一台服务器,然后将响应复制回原始请求的响应.一个额外的要求是我能够支持超时,这样如果代理服务器响应时间太长,代理将自行响应并关闭与代理服务器的连接。

我已经使用 Jetty 实现了这样的应用程序,但是使用 Jetty 我需要使用太多线程来防止入站请求被阻塞(这是一个使用很少内存或 cpu 的轻量级应用程序,但是代理服务器足够高,流量突发会导致代理服务器排队,或者需要太多线程)。

根据我的理解,我可以使用 Netty 构建一个管道,其中每个阶段执行少量计算,然后释放它的线程并等待数据准备好以执行管道中的下一阶段。

我的问题是,是否有此类应用程序的简单示例?到目前为止,我所拥有的是对基本 Netty 教程的服务器代码的简单修改,但它缺乏对客户端的所有支持。我看过 netty 客户端教程,但不确定如何混合使用两者的代码来创建一个简单的代理应用程序。

public static void main(String[] args) throws Exception {
    ChannelFactory factory =
            new NioServerSocketChannelFactory(
                    Executors.newCachedThreadPool(),
                    Executors.newCachedThreadPool());

    ServerBootstrap bootstrap = new ServerBootstrap(factory);

    bootstrap.setPipelineFactory(new ChannelPipelineFactory() {
        public ChannelPipeline getPipeline() {
            return Channels.pipeline(
                    new HttpRequestDecoder(),
                    new HttpResponseEncoder(),
                    /* 
                     * Is there something I can put here to make a
                     * request to another server asynchronously and
                     * copy the result to the response inside
                     * MySimpleChannelHandler?
                     */
                    new MySimpleChannelHandler()
                    );
        }
    });

    bootstrap.setOption("child.tcpNoDelay", true);
    bootstrap.setOption("child.keepAlive", true);

    bootstrap.bind(new InetSocketAddress(8080));
}

private static class MySimpleChannelHandler extends SimpleChannelHandler {

    @Override
    public void messageReceived(ChannelHandlerContext ctx, MessageEvent e) {
        HttpRequest request = (HttpRequest) e.getMessage();
        HttpResponse response = new DefaultHttpResponse(HttpVersion.HTTP_1_1, HttpResponseStatus.OK);
        response.setContent(request.getContent());

        Channel ch = e.getChannel();
        ChannelFuture f = ch.write(response);
        f.addListener(new ChannelFutureListener() {
            public void operationComplete(ChannelFuture future) {
                Channel ch = future.getChannel();
                ch.close();
            }
        });
    }

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx, ExceptionEvent e) {
        e.getCause().printStackTrace();

        Channel ch = e.getChannel();
        ch.close();
    }
}

最佳答案

你必须看看LittleProxy看看他们是如何做到的,因为它是在 Netty 之上编写的。

关于java - 使用Netty搭建http代理服务器的简单方法?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15693158/

相关文章:

Java字符来表示所有字符

java - netty维护连接以及如何通知失败到channelfuture

java - 如何使用 Netty 读取 HTTP 请求的主体?

java - bazel - netty_tcnative 错误

java - 启动简单的非基于 Web 的 Java 应用程序的官方 Spring Boot 方式是什么?

java - 在双链表中插入节点

java - Tomcat 7 GWT 应用 ssl 配置

java - 当客户端更改 IP 地址或断开连接时,是否热衷于捕获 websocket-close 事件?

java - 在 ChannelHandler 中连接到 Netty 中的客户端

java - 无法在 jasperviewer 中以 .html 以外的格式保存报告