NodeJS - 编辑和代理多部分/表单数据请求

And*_*nio 3 proxy multipart form-data express http-proxy-middleware

我有一个微服务,它代理每个请求添加一个字段。对于普通请求,这很容易,只需在request.body 中添加字段并正确设置标头,但对于 multipart/form-data 请求,我几天以来就遇到麻烦,因为如果我在request.body 中添加一个字段,它会消失。

const router = express()
const routes = require('~/routes')
const passport = require('passport')
const proxy = require('http-proxy-middleware')

router.use(passport.initialize())
require('./modules/passport-jwt')(passport)

router.use('/', routes)

router.use(
    '/account',
    passport.authenticate('jwt', { session: false }),
    proxy({
        target: process.env.ACCOUNT_SERVICE,
        pathRewrite: { '/account': '/' },
        onProxyReq: restream
    })
)

const restream = async function (proxyReq, req, res, options) {
    if (req.user) {
        if (
            req.headers['content-type'] &&
            req.headers['content-type'].match(/^multipart\/form-data/)
        ) {
            req.body.reqUser = req.user
        } else {
            const requestBody = JSON.stringify({ ...req.body, reqUser: req.user })
            proxyReq.setHeader('Content-Type', 'application/json')
            proxyReq.setHeader('Content-Length', Buffer.byteLength(requestBody))
            proxyReq.write(requestBody)
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

当请求到达另一个微服务时,request.body为空,然后由multer写入,将 multipart/form-data 参数放入request.body

我真的需要一个解决方案,让我将一个字段附加到代理重功能中的多部分/表单数据请求中。

我想尽一切办法在这方面取得成功,但我被卡住了。我希望一切都从我这边清楚。如果您需要,不要害怕询问更多细节。我请求你的帮助。

And*_*nio 8

我终于知道如何成功了。我的代码现在看起来像这样:

const router = express()
const routes = require('~/routes')
const passport = require('passport')
const proxy = require('http-proxy-middleware')

router.use(passport.initialize())
require('./modules/passport-jwt')(passport)

router.use('/', routes)

router.use(
    '/account',
    passport.authenticate('jwt', { session: false }),
    proxy({
        target: process.env.ACCOUNT_SERVICE,
        pathRewrite: { '/account': '/' },
        onProxyReq: restream
    })
)

const restream = async function (proxyReq, req, res, options) {
    if (req.user) {
        if (
            req.headers['content-type'] &&
            req.headers['content-type'].match(/^multipart\/form-data/)
        ) {
            // build a string in multipart/form-data format with the data you need
            const formdataUser =
                `--${request.headers['content-type'].replace(/^.*boundary=(.*)$/, '$1')}\r\n` +
                `Content-Disposition: form-data; name="reqUser"\r\n` +
                `\r\n` +
                `${JSON.stringify(request.user)}\r\n`

            // set the new content length
            proxyReq.setHeader(
                'Content-Length',
                parseInt(request.headers['content-length']) + Buffer.byteLength(formdataUser)
            )

            proxyReq.write(formdataUser)
        } else {
            const body = JSON.stringify({ ...req.body, reqUser: req.user })
            proxyReq.setHeader('Content-Type', 'application/json')
            proxyReq.setHeader('Content-Length', Buffer.byteLength(body))
            proxyReq.write(body)
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

正如我在代码注释中所写:

  1. 以 multipart/form-data 格式构建一个字符串,必须如下所示:

    ------WebKitFormBoundaryiBtoTWFkpAG6CgXO\r\n
    Content-Disposition: form-data; name="firstname"\r\n
    \r\n\
    Andrea\r\n
    
    Run Code Online (Sandbox Code Playgroud)

    (在我的代码中,我对数据进行了字符串化,因为它是一个对象);

  2. 通过将上述字符串的字节长度添加到原始请求长度来设置标头“Content-Length”;

  3. 使用proxyReq.write函数发送新数据。