当前位置: 首页 > news >正文

node和express搭建代理服务器(源码)

实例git 地址:https://github.com/xubaodian/...

本例用node和express搭建的代理服务器。,期望目标如下:

1、开启某服务A,该服务可实现若干功能,例如普通的restful请求,文件上传,静态资源访问等等。

2、开启node代理服务B,指向服务A,访问代理服务B,可访问服务A的任意功能。

就如下图所示:

<img src="./docs.proxy.JPG">

图中上半部分是直接访问服务,下班部分是通过代理服务器访问服务。

使用代理服务器时,浏览器向代理服务器请求数据,代理服务器转发请求,并将接收到的数据返回给浏览器,即所有的数据都通过代理服务器转发。
带着这个目标,我们就讲述下如何实现该功能。

既然是请求和响应转发,那我们就了解一下,什么是请求。

请求和响应简述

http请求和响应主要右报文头部、空行和报文主体三个部分组成。

空行我们不用关心,其实对我们来说,只要完成报文头部和报文主体的转发,就可以说实现了代理功能。

请求和响应通过代理的整个过程如下:

1、代理服务器接收请求后,在将目标服务数据返回给浏览器前要保持请求。

2、提取请求路径、请求头、请求主体等数据。

3、以2中提取的数据为参数,向目标服务器发送请求。

4、接收目标服务器返回数据,提取响应头,响应主体等数据。

5、将4中的提取出来的数据返回给客户端(浏览器)。

6、断开连接。

经过这几个步骤,就实现了代理。

代码实现

下面直接上代码,然后做一些讲解。代理函数如下:

const http = require('http');
const querystring = require('querystring');

//获取请求的cookie和query等
let getHeader = (reqClient) => {
    let headers = reqClient.headers; 
    headers.path = reqClient.path;
    headers.query = reqClient.query;
    headers.cookie = reqClient.get('cookie') || '';

    return headers;
}

//代理函数,options是代理设置,包括目标服务器ip,port等
let proxy = (options) => {
    let reqOptions = {
        hostname: options.host,
        port: options.port
    }
    //返回请求处理函数,reqClient浏览器的请求,resClient是响应浏览器的对象
    return function (reqClient, resClient) {
        //设置目标服务器的请求参数,头中的各项参数
        let headers = getHeader(reqClient);
        reqOptions.headers = reqClient.headers;
        let query = [];
        if (headers.query) {
            Object.keys(headers.query).map(key => {
                query.push(key + '=' + headers.query[key]);
            });
            reqOptions.path = headers.path + (query.length === 0 ? '' : ('?' + query.join('&')));
            
        }
        reqOptions.cookie = headers.cookie;
        reqOptions.method = reqClient.method;
        //向目标服务器发送请求,reqProxy是向目标服务器的请求,resProxy是目标服务器的响应。
        let reqProxy = http.request(reqOptions, (resProxy) => {
            resProxy.setEncoding('utf8');
            //设置返回http头
            resClient.set(resProxy.headers);
            resClient.status(resProxy.statusCode);
            //接收从目标服务器返回的数据
            resProxy.on('data', (chunk) => {
                //接收目标服务器数据后,以流的方式向浏览器返回数据
                resClient.write(chunk);
            });

            //接收目标服务器数据结束
            resProxy.on('end', () => {
                //向浏览器写数据结束。
                resClient.end();
            });
            //目标服务器响应错误
            resProxy.on('error', () => {
                //响应错误,结束向浏览器返回数据
                resClient.end();
            });
        });

        //接收浏览器数据
        reqClient.on('data', (chunk) => {
           //以流的方式向目标服务器发送数据
            reqProxy.write(chunk);
        });
        //接收数据结束
        reqClient.on('end', () => {
          //向目标服务器写数据结束
            reqProxy.end();
        });
        
        //普通JSON数据代理
         if (Object.keys(reqClient.body).length) {
             reqProxy.write(querystring.stringify(reqClient.body));
             reqProxy.end();
         }
    }
}

module.exports = proxy;

上面就是node代理的核心代码。支持普通的请求,静态资源代理,文件上传下载代理等功能。

git 地址:https://github.com/xubaodian/...

demo中,核心代码在common/proxy.js里,我还实现了两个测试服务。

在server文件下的app.js和app2.js是两个服务的入口文件。

app2.js是目标服务器,有三个测试页面

1、http://localhost:20000/json.html post请求测试,对应'/json'接口,可发送数据,f12查看请求是否成功

2、http://localhost:20000/upload.html 文件上传测试,对应接口'/upload'接口,上传文件,f12查看请求是否成功,同时在服务器upload文件夹下会有文件。

3、http://localhost:20000/get.html get请求测试,对应接口'/get',同样f12查看

app2为目标服务器,有3个接口。

1、'/upload'接口,测试文件上传功能,上传文件将放在uploads文件夹下,上传的文件,文件名是一个uuid,没有后缀,添加后缀即可查看文件是完整。测试过,传1G的文件没问题,再大的文件没试过,有需要的可以试下

2、'/json',测试POST请求。

3、'/get',测试GET请求。

app.js为代理服务为器,监听端口为18000,将所有请求转发至app2,即所有app2的接口静态资源,app中访问时一致的。

测试步骤:
1、可开启目标服务器,通过三个页面测试功能。

2、开启代理服务器,访问下面三个页面:

http://localhost:18000/json.html

http://localhost:18000/upload.html

http://localhost:18000/get.html

测试同样的功能。若和步骤1实现同样功能,则代理服务功能已经实现了。

经过测试,代理功能是没问题的。

如果问题欢迎留言,或发送邮件至472784995@qq.com。

至于性能,我没测过,因为我自己这边的应用场景,访问量都不大,可以使用。

相关文章:

  • 解决Android8.0之后开启service时报错IllegalStateException: Not allowed to start service Intent ......
  • 技术篇-HBase 最佳实践-读性能优化策略
  • IaaS,PaaS和SaaS,又衍生出CaaS,NaaS和MaaS
  • Spring Cloud 下线微服务
  • 让UITableView的Cell都变成静态的
  • 牛客练习赛37
  • 『原创』设置SQL Server 2005自动备份——数据库邮件设置(图文)
  • Gmail工具和插件网址大全
  • 北京城市生态系统研究站取得2011年国自然科学基金重点基金资助
  • samba 文件共享
  • Nginx的介绍
  • 14个值得推荐的个人提升方法
  • 上海科学家研制出新型“耐火宣纸”
  • Java Web:项目结构和web.xml
  • 调查:澳门96%打工者称奖金、薪酬决定工作去留
  • 「前端早读君006」移动开发必备:那些玩转H5的小技巧
  • canvas 绘制双线技巧
  • ESLint简单操作
  • hadoop集群管理系统搭建规划说明
  • iOS帅气加载动画、通知视图、红包助手、引导页、导航栏、朋友圈、小游戏等效果源码...
  • Linux链接文件
  • SQLServer之创建显式事务
  • SQLServer之索引简介
  • 编写符合Python风格的对象
  • 彻底搞懂浏览器Event-loop
  • 前端js -- this指向总结。
  • 手机app有了短信验证码还有没必要有图片验证码?
  • 树莓派 - 使用须知
  • 一道面试题引发的“血案”
  • 原创:新手布局福音!微信小程序使用flex的一些基础样式属性(一)
  • ​flutter 代码混淆
  • #14vue3生成表单并跳转到外部地址的方式
  • #pragma pack(1)
  • %check_box% in rails :coditions={:has_many , :through}
  • (01)ORB-SLAM2源码无死角解析-(66) BA优化(g2o)→闭环线程:Optimizer::GlobalBundleAdjustemnt→全局优化
  • (1) caustics\
  • (23)Linux的软硬连接
  • (4.10~4.16)
  • (51单片机)第五章-A/D和D/A工作原理-A/D
  • (C#)获取字符编码的类
  • (C语言)strcpy与strcpy详解,与模拟实现
  • (Redis使用系列) Springboot 使用redis实现接口Api限流 十
  • (附源码)ssm户外用品商城 毕业设计 112346
  • (黑客游戏)HackTheGame1.21 过关攻略
  • (每日持续更新)信息系统项目管理(第四版)(高级项目管理)考试重点整理第3章 信息系统治理(一)
  • (原创)Stanford Machine Learning (by Andrew NG) --- (week 9) Anomaly DetectionRecommender Systems...
  • (轉貼)《OOD启思录》:61条面向对象设计的经验原则 (OO)
  • .NET 8.0 中有哪些新的变化?
  • .NET Framework 服务实现监控可观测性最佳实践
  • .NET HttpWebRequest、WebClient、HttpClient
  • .NET Micro Framework 4.2 beta 源码探析
  • .NET/C# 使用 #if 和 Conditional 特性来按条件编译代码的不同原理和适用场景
  • .net网站发布-允许更新此预编译站点
  • .net中应用SQL缓存(实例使用)
  • .sh文件怎么运行_创建优化的Go镜像文件以及踩过的坑