nestjs 中间件获取请求/响应体

Posted

技术标签:

【中文标题】nestjs 中间件获取请求/响应体【英文标题】:nestjs middleware get request/response body 【发布时间】:2019-06-03 17:50:30 【问题描述】:

我正在为一个项目使用nestjs,并希望记录尽可能多的信息,其中之一就是每个http 请求的响应和请求的主体。为此我制作了一个嵌套中间件:

import token from 'gen-uid';
import  inspect  from 'util';
import  Injectable, NestMiddleware, MiddlewareFunction  from '@nestjs/common';
import  Stream  from 'stream';
import  createWriteStream, existsSync, mkdirSync  from 'fs';

@Injectable()
export class LoggerMiddleware implements NestMiddleware 
    logfileStream: Stream;

    constructor() 
        if (!existsSync('./logs')) mkdirSync('./logs');
        this.logfileStream = createWriteStream("./logs/serviceName-"+ new Date().toISOString() + ".log", flags:'a');
    

resolve(...args: any[]): MiddlewareFunction 
    return (req, res, next) => 
        let reqToken = token();
        let startTime = new Date();
        let logreq = 
            "@timestamp": startTime.toISOString(),
            "@Id": reqToken,
            query: req.query,
            params: req.params,
            url: req.url,
            fullUrl: req.originalUrl,
            method: req.method,
            headers: req.headers,
            _parsedUrl: req._parsedUrl,
        

        console.log(
            "timestamp: " + logreq["@timestamp"] + "\t" + 
            "request id: " + logreq["@Id"] + "\t" + 
            "method:  " + req.method + "\t" +
            "URL: " + req.originalUrl);

        this.logfileStream.write(JSON.stringify(logreq));

        const cleanup = () => 
            res.removeListener('finish', logFn)
            res.removeListener('close', abortFn)
            res.removeListener('error', errorFn)
        

        const logFn = () => 
            let endTime = new Date();
            cleanup()
            let logres = 
                "@timestamp": endTime.toISOString(),
                "@Id": reqToken,
                "queryTime": endTime.valueOf() - startTime.valueOf(),
            
            console.log(inspect(res));
        

        const abortFn = () => 
            cleanup()
            console.warn('Request aborted by the client')
        

        const errorFn = err => 
            cleanup()
            console.error(`Request pipeline error: $err`)
        

        res.on('finish', logFn) // successful pipeline (regardless of its response)
        res.on('close', abortFn) // aborted pipeline
        res.on('error', errorFn) // pipeline internal error

        next();
    ;


然后我将此中间件设置为全局中间件来记录所有请求,但是查看 res 和 req 对象,它们都没有属性。

在代码示例中,我设置了要打印的响应对象,在我的项目上运行一个 hello world 端点,该端点返回 "message":"Hello World" 我得到以下输出:

时间戳:2019-01-09T00:37:00.912Z 请求 ID:2852f925f987 方法:GET URL:/hello-world

服务器响应 域:空, _events:完成:[功能:绑定resOnFinish], _eventsCount:1, _maxListeners:未定义, 输出: [], 输出编码:[], 输出回调:[], 输出大小:0, 可写:真, _last: 假的, 升级:假, 分块编码:假, 应该保持活动:真, useChunkedEncodingByDefault: true, 发送日期:真, _removedConnection:假, _removedContLen:真, _removedTE:真, _contentLength: 0, _hasBody:假, _预告片: '', 完成:真的, _headerSent:真, 套接字:空, 连接:空, _header: 'HTTP/1.1 304 Not Modified\r\nX-Powered-By: Express\r\nETag: W/"19-c6Hfa5VVP+Ghysj+6y9cPi5QQbk"\r\n日期: 2019 年 1 月 9 日,星期三 00:37:00 GMT\r\n连接:保持活动状态\r\n\r\n', _onPendingData:[功能:绑定updateOutgoingData], _sent100:假, _expect_continue:假, 要求: 传入消息 _可读状态: 可读状态 对象模式:假, 高水位:16384, 缓冲区:[对象], 长度:0, 管道:空, 管道数:0, 流动的:真实的, 结束:真的, endEmitted:假, 阅读:错误, 同步:真, 需要可读:假, 发射可读:真, 可读听力:假, 恢复计划:真, 毁坏:假, 默认编码:'utf8', 等待排水:0, 阅读更多:真的, 解码器:空, 编码:空, 可读:真实, 域:空, _事件:, _eventsCount:0, _maxListeners:未定义, 插座: 插座 连接:假, _hadError:假, _handle:[对象], _父:空, _主机:空, _可读状态:[对象], 可读:真实, 域:空, _events:[对象], _eventsCount:10, _maxListeners:未定义, _writableState:[对象], 可写:真, allowHalfOpen:真, _bytesDispatched: 155, _sockname:空, _pendingData:空, _pendingEncoding: '', 服务器:[对象], _server:[对象], _idleTimeout: 5000, _idleNext: [对象], _idlePrev: [对象], _idleStart:12562, _destroyed:假, 解析器:[对象], 上:[功能:socketOnWrap], _暂停:假, 阅读:[功能], _消费:真的, _httpMessage:空, [符号(asyncId)]:151, [符号(字节读取)]:0, [符号(asyncId)]:153, [符号(triggerAsyncId)]: 151 , 联系: 插座 连接:假, _hadError:假, _handle:[对象], _父:空, _主机:空, _可读状态:[对象], 可读:真实, 域:空, _events:[对象], _eventsCount:10, _maxListeners:未定义, _writableState:[对象], 可写:真, allowHalfOpen:真, _bytesDispatched: 155, _sockname:空, _pendingData:空, _pendingEncoding: '', 服务器:[对象], _server:[对象], _idleTimeout: 5000, _idleNext: [对象], _idlePrev: [对象], _idleStart:12562, _destroyed:假, 解析器:[对象], 上:[功能:socketOnWrap], _暂停:假, 阅读:[功能], _消费:真的, _httpMessage:空, [符号(asyncId)]:151, [符号(字节读取)]:0, [符号(asyncId)]:153, [符号(triggerAsyncId)]: 151 , httpVersionMajor: 1, httpVersionMinor:1, http版本:'1.1', 完整:真实, 标题: 主机:'本地主机:5500', '用户代理': 'Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:64.0) Gecko/20100101 Firefox/64.0', 接受:'text/html,application/xhtml+xml,application/xml;q=0.9,/;q=0.8', '接受语言': 'en-US,en;q=0.5', '接受编码':'gzip,放气', 连接:'保持活动', '升级不安全请求':'1', 'if-none-match': 'W/"19-c6Hfa5VVP+Ghysj+6y9cPi5QQbk"' , 原始标题: [ '主持人', '本地主机:5500', '用户代理', 'Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:64.0) Gecko/20100101 Firefox/64.0', '接受', 'text/html,application/xhtml+xml,application/xml;q=0.9,/;q=0.8', '接受语言', 'en-US,en;q=0.5', '接受编码', 'gzip,放气', '联系', '活着', '升级-不安全-请求', '1', '如果没有匹配', 'W/"19-c6Hfa5VVP+Ghysj+6y9cPi5QQbk"'], 预告片:, rawTrailers: [], 升级:假, 网址:'/你好世界', 方法:'GET', 状态码:空, 状态消息:空, 客户: 插座 连接:假, _hadError:假, _handle:[对象], _父:空, _主机:空, _可读状态:[对象], 可读:真实, 域:空, _events:[对象], _eventsCount:10, _maxListeners:未定义, _writableState:[对象], 可写:真, allowHalfOpen:真, _bytesDispatched: 155, _sockname:空, _pendingData:空, _pendingEncoding: '', 服务器:[对象], _server:[对象], _idleTimeout: 5000, _idleNext: [对象], _idlePrev: [对象], _idleStart:12562, _destroyed:假, 解析器:[对象], 上:[功能:socketOnWrap], _暂停:假, 阅读:[功能], _消费:真的, _httpMessage:空, [符号(asyncId)]:151, [符号(字节读取)]:0, [符号(asyncId)]:153, [符号(triggerAsyncId)]: 151 , _消费:假, _转储:真的, 下一个:[功能:下一个], baseUrl: '', originalUrl: '/hello-world', _parsedUrl: 网址 协议:空, 斜线:空, 身份验证:空, 主机:空, 端口:空, 主机名:空, 哈希:空, 搜索:空, 查询:空, 路径名:'/hello-world', 路径:'/你好世界', href: '/你好世界', _raw: '/hello-world' , 参数:, 询问: , 资源:[循环], 身体: , 路线:路线路径:'/hello-world',堆栈:[数组],方法:[对象], 当地人:, 状态码:304, statusMessage: '未修改', [符号(outHeadersKey)]: 'x-powered-by': [ 'X-Powered-By', 'Express' ], etag: [ 'ETag', 'W/"19-c6Hfa5VVP+Ghysj+6y9cPi5QQbk"' ]

在响应对象中没有出现 "message":"Hello World" 消息,如果可能的话,我想知道如何从 res 和 req 对象中获取正文。

注意:我知道nestjs 有Interceptors,但是按照文档的说法,中间件应该是解决这个问题的方法。

【问题讨论】:

还有问题吗? :-) 【参考方案1】:

我无意中遇到了这个问题,它被列在“相关”到my question。

关于回复,我可以进一步扩展Kim Kern's answer。

响应的问题是响应体不是响应对象的属性,而是。为了能够获取它,您需要重写写入该流的方法。

就像 Kim Kern 已经说过的,你可以看看this thread,有公认的答案如何做到这一点。

或者你可以使用express-mung中间件,它会为你做这件事,例如:

var mung = require('express-mung');
app.use(mung.json(
  function transform(body, req, res) 
    console.log(body); // or whatever logger you use
    return body;
  
));

NestJS 还可以为您提供另外两种不同的方式:

Interceptors,就像你说的。文档中有LoggingInterceptor 的示例。 您可以为控制器的方法编写装饰器,这将拦截它们的响应。
import  isObservable, from, of  from 'rxjs';
import  mergeMap  from 'rxjs/operators';

/**
 * Logging decorator for controller's methods
 */
export const LogReponse = (): MethodDecorator =>
  (target: object, propertyKey: string | symbol, descriptor: TypedPropertyDescriptor<any>) => 

    // save original method
    const original = descriptor.value;

    // replace original method
    descriptor.value = function()  // must be ordinary function, not arrow function, to have `this` and `arguments`

      // get original result from original method
      const ret = original.apply(this, arguments);

      // if it is null or undefined -> just pass it further
      if (ret == null) 
        return ret;
      

      // transform result to Observable
      const ret$ = convert(ret);

      // do what you need with response data
      return ret$.pipe(
        map(data => 
          console.log(data); // or whatever logger you use
          return data;
        )
      );
    ;

    // return modified method descriptor
    return descriptor;
  ;

function convert(value: any) 
  // is this already Observable? -> just get it
  if (isObservable(value)) 
    return value;
  

  // is this array? -> convert from array
  if (Array.isArray(value)) 
    return from(value);
  

  // is this Promise-like? -> convert from promise, also convert promise result
  if (typeof value.then === 'function') 
    return from(value).pipe(mergeMap(convert));
  

  // other? -> create stream from given value
  return of(value);

请注意,这将在拦截器之前执行,因为这个装饰器会改变方法的行为。

而且我不认为这是进行日志记录的好方法,只是为了多样性而提到它:)

【讨论】:

【参考方案2】:

response 正文将无法作为属性访问。请参阅此thread 以获得解决方案。

但是,您应该能够使用 req.body 访问 request 正文,因为默认情况下 Nest 使用 bodyParser

【讨论】:

【参考方案3】:

令人难以置信的是,如此微不足道的事情竟然如此难以做到。

记录响应正文的更简单方法是创建一个拦截器 (https://docs.nestjs.com/interceptors):

AppModule

providers: [
    
      provide: APP_INTERCEPTOR,
      useClass: HttpInterceptor,
    
]

HttpInterceptor

import  CallHandler, ExecutionContext, Injectable, Logger, NestInterceptor  from '@nestjs/common';
import  Observable  from 'rxjs';
import  map  from 'rxjs/operators';

@Injectable()
export class HttpInterceptor implements NestInterceptor 
  private readonly logger = new Logger(HttpInterceptor.name);

  intercept(
    context: ExecutionContext,
    next: CallHandler<any>,
  ): Observable<any> | Promise<Observable<any>> 
    return next.handle().pipe(
      map(data => 
        this.logger.debug(data);
        return data;
      ),
    );
  

【讨论】:

通过使用它,我只能看到成功的响应,作为数据结构(我正在使用 GraphQL)。例如,如果 auth 保护阻止了查询(message: "Unauthorized" 是响应),则不会调用拦截器。在所有应用内处理发生后,是否有一种已知的方法可以访问原始响应正文?我想记录可能出错的实际细节。

以上是关于nestjs 中间件获取请求/响应体的主要内容,如果未能解决你的问题,请参考以下文章

Django的请求生命周期

在 NestJs 中何时使用守卫以及何时使用中间件

5. 控制器

架构服务器端 i18n 的最佳方式

Django - 中间件

在nestjs中使用multer-gridfs流