首页 > 编程语言 >使用 nodejs 从 0 实现简单易用的代理功能之 config.proxy

使用 nodejs 从 0 实现简单易用的代理功能之 config.proxy

时间:2023-05-09 09:33:19浏览次数:56  
标签:const nodejs express 代理 json proxy config options

从 0 实现 config.proxy

config.proxy 类似于 webpack 的 devServe 中的代理, 但更直观易用.

本文为 mockm 的实现过程, 编写此系列文章 1 是为了抛砖引玉, 让想实现类似工具的朋友可以一起学习. 2 是也给自己做一个简单梳理.

image.png

类型: string | object
默认: http://www.httpbin.org/

代理到远程的目标域名,为对象时每个键是分别对应一个要自定义代理的路由.

注: 是对象时, 需要存在键 / 表示默认域名.
此功能可以自定义拦截过程, 类似 webpack 中的 devServer.proxy .

  • string 直接请求转发到指定地址.
  • object 相当于传入 proxy 的配置.

参考 proxy.

快速修改 json response

支持以简便的方式快速处理 json 格式的 response, 使用数组语法: [A, B].
数组中不同个数和类型有不同的处理方式, 参考下表:

个数[A, B] 类型处理方式处理前操作处理后
0, 1 [any] 直接替换 {a: 1} [][undefined]
      {a: 1} [123] 123
2 [string, any] 替换指路径的值 {a: 1} ['a', 2] {a: 2}
      {a: {b: 1, c: 2}} ['a.b', undefined] {a: {c: 2}}
2 [object, ...] 浅合并 {a: {a: 1}} [{a: {b: 2}, c: 1}, '...'] {a: {b: 2}, c: 1}
  [object, deep] 深合并 {a: {a: 1}} [{a: {b: 2}, c: 1}, 'deep'] {a: {a: 1, b: 2}, c: 1}

A 或 B 支持传入函数, 可以接收 {req, json}, 返回值是前端最终收到的值.

::: details 示例
进一步解释表格中的示例.

直接替换

处理前
{
  "a": 1
}

操作, 直接替换为空
[] 或 [undefined]

处理后
undefined

直接替换

处理前
{
  "a": 1
}

操作, 直接替换为 123
[123]

处理后
123

替换指定路径的值

处理前
{
  "a": 1
}

操作, 把 a 的值替换为 2
['a', 2]

处理后
{
  "a": 2
}

替换指定路径的值

处理前
{
  "a": {
    "b": 1,
    "c": 2
  }
}

操作, 把 a 下面 b 的值删除
['a.b', undefined]

处理后
{
  "a": {
    "c": 2
  }
}

浅合并

处理前
{
  "a": {
    "a": 1
  },
  "c": 1
}

操作, 合并时直接替换, 此示例会直接替换掉 a 对象
[
  {
    "a": {
      "b": 2
    },
    "c": 1
  },
  "..."
]

处理后
{
  "a": {
    "b": 2
  },
  "c": 1
}

深合并

处理前
{
  "a": {
    "a": 1
  }
}

操作, 深层合并对象, 此对象会合并 a 对象
[
  {
    "a": {
      "b": 2
    },
    "c": 1
  },
  "deep"
]

处理后
{
  "a": {
    "a": 1,
    "b": 2
  },
  "c": 1
}

:::

请求转发

可以方便的支持任意路径转发, 以下演示转发到其他域名:

``js {3} proxy: { '/':https://httpbin.org/`,
‘/get’: https://www.httpbin.org/ip,
},



## 实现一个简单代理
在 express 中使用 http-proxy-middleware 实现简单代理

``` js
const express = require('express');
const { createProxyMiddleware } = require('http-proxy-middleware');

const app = express();

app.use('/api', createProxyMiddleware({ target: 'http://www.example.org', changeOrigin: true }));
app.listen(3000);

由于需要使用 json-server 自动实现 restfull-api, 并且由于 json-server 是强依赖 express 的, 所以我们直接使用 json-server 中的 express, 而不是再安装一个 express, 这样可以保持一定的兼容性.

const jsonServer = require(`@wll8/json-server`)
const app = jsonServer.create()

解析 config.proxy 对象

默认情况下, 每个代理都需要自己指定 createProxyMiddleware 的 target 参数和 pathRewrite 参数, 才能实现:

:3000/api => http://baidu.com/api 这样的效果.

实际上对于常用的功能, 至少需要以下配置:

const defaultConfig = {
  ws: true,
  target: origin,
  secure: false,
  changeOrigin: true,
  onProxyReq: (proxyReq, req, res) => {
    // https://github.com/chimurai/http-proxy-middleware/pull/492
    this.httpProxyMiddleware.fixRequestBody(proxyReq, req)
  },
  logLevel: `silent`,
  // proxyTimeout: 60 * 1000,
  // timeout: 60 * 1000,
}

所以为了容易使用, 我们需要做一些转换工作. 例如配置 proxy 的时候:

  • proxy 为字符串时, 直接代理到字符串
  • proxy 为对象时,根据对象 key val 做转换
    • val 为字符串, 代理 key 到 val
    • val 为对象, 则表示自定义 createProxyMiddleware
    • val 为数组, 则表示快捷修改响应体

由于默认情况下基本都只代理到一个根服务, 所以我们提取根服务, 如果代理的目标没有指定其他服务时, 那么我们就使用根服务:

主要是提取 val 中的 origin.

this.rootOriginData = this.parseRootOriginData(this.proxy)

转换整个 proxy 对象为 proxy 配置列表:

prepareProxy (proxy = {}) { // 解析 proxy 参数, proxy: string, object
  const isType = tool.type.isType
  let resProxy = []
  function setIndexOfEnd(proxy) { // 需要排序 key:/ 到最后, 否则它生成的拦截器会被其他 key 覆盖
    const indexVal = proxy[`/`]
    delete proxy[`/`]
    proxy[`/`] = indexVal
    return proxy
  }
  proxy = setIndexOfEnd(proxy)
  resProxy = Object.keys(proxy).map(context => {
    let options = proxy[context]
    const optionsType = isType(options)
    if(optionsType === `string`) { // 转换字符串的 value 为对象
      const rootOptions = proxy[`/`]
      options = {
        pathRewrite: { [`^${context}`]: options }, // 原样代理 /a 到 /a
        target: options.includes(`://`) // 如果要代理的目录地址已有主域
          ? new URL(options).origin // 那么就代理到该主域上
          : { // 否则就代理到 / 设定的域上
            string: rootOptions,
            object: rootOptions.target, // 当主域是对象时则取其 target
          }[isType(rootOptions)],
      }
    }
    if(optionsType === `array`) { // 是数组时, 视为设计 res body 的值, 语法为: [k, v]
      const [item1, item2] = options
      const item1Type = isType(item1)
      const item2Type = isType(item2)
      const deepMergeObject = tool.obj.deepMergeObject

      if((item1Type !== `function`) && (options.length <= 1)) { // 只有0个或一个项, 直接替换 res
        options = {
          onProxyRes (proxyRes, req, res) {
            this.midResJson({proxyRes, res, cb: () => item1})
          },
        }
      }
      if((item1Type === `function`) && (options.length <= 1)) {
        options = {
          onProxyRes (proxyRes, req, res) {
            this.midResJson({proxyRes, res, cb: (json) => item1({req, json})})
          },
        }
      }
      // if ...
    }
    return {
      route: context,
      info: options,
    }
  })
  return resProxy
}

然后依次 use 到 express 的实例上即可:

const proxyList = this.prepareProxy(this.proxy)
proxyList.forEach(item => {
  this.use({route: item.route, config: item.info})
})

标签:const,nodejs,express,代理,json,proxy,config,options
From: https://www.cnblogs.com/daysme/p/17383936.html

相关文章

  • 【nodejs基础】Express、路由、中间件详解04
    一、Express简介Express是基于Node.js平台,快速、开放、极简的Web开发框架通俗的理解:Express的作用和Node.js内置的http模块类似,是专门用来创建Web服务器的。本质就是一个npm上的第三方包,提供了快速创建Web服务器的便捷方法中文官网http://www.expressjs.com.cn......
  • Windows系列---【浏览器突然连不上网,报"ERR_PROXY_CONNECTION_FAILED"的解决方案】
    浏览器突然连不上网,报"ERR_PROXY_CONNECTION_FAILED"的解决方案1.问题电脑的浏览器里装的有梯子,突然有一天浏览器访问不了外网了,打开哪个网站都报"ERR_PROXY_CONNECTION_FAILED",但是微信可以正常使用。2.分析微信可以使用,显然是浏览器的问题,再看错误,明显是代理有问题。3.解......
  • 《nodejs跨栏》新建项目
    新建项目1新建一个文件夹(不含中文、空格、大写字母,建议用-分隔法命名)2打开终端。win+R输入cmd,cd到文件夹中。执行命令npminit3依次设置项目名称(跟新建的文件夹名一样)、版本号、描述、关键字等4所有设置完成后,会在根目录生成一个package.json(还有一个附加文件),这个文件保存......
  • 【编程入门】应用市场(NodeJS版)
    背景前面已输出多个系列:《十余种编程语言做个计算器》《十余种编程语言写2048小游戏》《17种编程语言+10种排序算法》《十余种编程语言写博客系统》《十余种编程语言写云笔记》《N种编程语言做个记事本》目标为编程初学者打造入门学习项目,使用各种主流编程语言来实现。让想学......
  • 第6-1讲,Label的config属性
    在Tkinter中,Label组件的config属性可以用来设置或获取Label组件的各种属性。这些属性包括:text:用于设置或获取Label组件的文本内容。font:用于设置或获取Label组件的字体。fg:用于设置或获取Label组件的前景色(即文本颜色)。bg:用于设置或获取Label组件的背景色。w......
  • [230]连接Redis后执行命令错误 MISCONF Redis is configured to save RDB snapshots
    今天在redis中执行setrangename1chun命令时报了如下错误提示:(error)MISCONFRedisisconfiguredtosaveRDBsnapshots,butiscurrentlynotabletopersistondisk.Commandsthatmaymodifythedatasetaredisabled.PleasecheckRedislogsfordetailsabout......
  • [Warning] World-writable config file '/etc/my.cnf' is ignored
    告警信息,全局读写配置文件,那么就把权限调整小。 ......
  • vite.config.ts配置文件
    import{defineConfig}from'vite'importvuefrom'@vitejs/plugin-vue'import{resolve}from'path'importvueSetupExtendfrom'vite-plugin-vue-setup-extend'importAutoImporttfrom'unplugin-auto-import/v......
  • npm ERR! code EPERM npm ERR! syscall mkdir npm ERR! path C:\Program Files\node
    npm项目初始化代码npminit--yesidea代码安装npmnpmiexperss我输入的时候报错了,如下图所示没关系,只需要手动打开C盘的路径文件找到这个文件,并且把他Ctrl+D删除掉即可之后在运行这串代码就可以啦明显成功了......
  • Failed to auto-configure a DataSource: 'spring.datasource.url' is not specified
    导入一个新的springbootmaven项目启动一直报这个错,查出来的答案都说是加注解把数据库扫描给排除掉,这种方式其实有点鸵鸟,项目原先是没问题的,现在导入到自己的环境启动不起来,那肯定是不能去改动代码的。排查了一遍,发现是项目中的resources文件没有指定成资源文件,所以找不到数据库......