Nodejs lambda 请求函数获取 411 Length Required

Nodejs lambda request function gets 411 Length Required

我使用 node.js lambda 向远程 api 发出 post 请求,它安装在多个远程位置。
我无权访问远程 api 代码或日志。
lambda 由我也无法控制的外部应用程序通过 HTTP 网关调用。

它适用于除一个以外的所有位置。对于一个位置,我收到此错误:

411 Length Required.

我已尝试中和 HTTP 网关,并使用测试事件 运行 lambda posts。
我得到了相同的结果。

我已经向其他地点发送了完全相同的请求,并得到了回复。 我发送 ContentLength header.

时找不到问题

这是 lambda 代码:

const https = require("https");
const iconv =  require("iconv-lite");
const charset  =  require("charset");
const qs  =  require("qs");
const url  =  require("url");

exports.handler = (event, context, callback) => {
  event = JSON.parse(JSON.stringify(event));

    let enc ="";
    let multiValueHeaders = event["multiValueHeaders"]["Content-Type"];
    let PostParams = null;
    let domain = event["queryStringParameters"]["domain"] ;

    let buf = Buffer.from(JSON.stringify(event["body"]), "base64"); 
    let tstring =   buf.toString("utf8");
    PostParams  = qs.parse(tstring);
     
    var postData = PostParams ?  qs.stringify(PostParams) : {};
    
    let ContentLength = new Buffer.from(postData).length;
    let headers = "" ;
    headers += (multiValueHeaders) ? (' { "Content-Type": "'+ multiValueHeaders + '",') : '{';
    headers += ('"Content-Length":'+ ContentLength + '}');
    headers = JSON.parse(headers);
   
    var q = url.parse(domain, true);
    let options = {
      'method': 'POST',
      'hostname': q.hostname,
      'path': q.pathname,
      'headers':  {headers}
    };
    
      var req = http.request(options, function (res) {
      let chunks = [];
    
      res.on("data", function (chunk) {
        chunks.push(chunk);
        enc = charset(res.headers, chunk);
      });
    
      res.on("end", function (chunk) {
          var decodedBody = iconv.decode(Buffer.concat(chunks), enc);
   
          const response = {
              statusCode: 200,
              body: decodedBody
          };
          callback(null ,response );
      });
    
      res.on("error", function (error) {
        console.error(error);
      });
    }); 
  
    
    if (PostParams != null) req.write(postData);
    req.end();  
     
  
}

当请求直接发送到端点表单时 postman 没有错误。仅来自 lambda。

除了这个 event = JSON.parse(JSON.stringify(event));?

除此之外,还有一种非常丑陋的构建 headers 对象的方法:

let headers = "";
headers += (multiValueHeaders) ? (' { "Content-Type": "'+ multiValueHeaders + '",') : '{';
headers += ('"Content-Length":'+ ContentLength + '}');
headers = JSON.parse(headers);

我会写成:

const headers = { "Content-Length": ContentLength };
if(multiValueHeaders) headers["Content-Type"] = multiValueHeaders;

问题的根本原因在于这一行:

'headers':  {headers}

需要在以下时间更改:

'headers': headers

希望对您有所帮助

411 返回 when the server demands a valid Content-Length

event 通过 HTTP 网关传递的参数是 entire client request object。您不必解析它。

event.body 是转义字符串。双重转义会给出错误的内容长度。例如,

JSON.stringify({'double': 2}) !== JSON.stringify(JSON.stringify({'double': 2))
// false

考虑到这一点,您可以这样执行您的请求:

exports.handler = (event, context, callback) => {
    let enc = "";
    let multiValueHeaders = event["multiValueHeaders"];
    let domain = event["queryStringParameters"]["domain"] ;
    

    const postHeaders = {...multiValueHeaders};
    let postData = null;
    if (event.body !== null) {
       postData = qs.stringify(
          qs.parse(
             Buffer.from(event.body, "base64").toString("utf-8")
          )
       );
       postHeaders['Content-Length'] = [ Buffer.byteLength(postData) ];
    }
   
    var q = url.parse(domain, true);
    let options = {
      'method': 'POST',
      'hostname': q.hostname,
      'path': q.pathname,
      'headers':  postHeaders
    };
    
    var req = http.request(options, function (res) {
      let chunks = [];
    
      res.on("data", function (chunk) {
        chunks.push(chunk);
        enc = charset(res.headers, chunk);
      });
    
      res.on("end", function (chunk) {
          var decodedBody = iconv.decode(Buffer.concat(chunks), enc);
   
          const response = {
              statusCode: 200,
              body: decodedBody
          };
          callback(null, response);
      });
    
      res.on("error", function (error) {
        console.error(error);
      });
    }); 
  
    if (postData !== null) req.write(postData);
    req.end();  
}