首页
学习
活动
专区
工具
TVP
发布
社区首页 >问答首页 >NestJS在拦截器中设置HttpStatus

NestJS在拦截器中设置HttpStatus
EN

Stack Overflow用户
提问于 2019-03-29 04:16:50
回答 1查看 7.7K关注 0票数 5

我正在使用拦截器来转换我的响应。我想在里面设置HttpStatus,但是我现在使用的代码不能工作。

代码语言:javascript
复制
import { CallHandler, ExecutionContext, NestInterceptor, SetMetadata } from '@nestjs/common';
import { Observable } from 'rxjs';
import { map } from 'rxjs/operators';
import { classToPlain } from 'class-transformer';
import { ApiResponse } from '../models/apiResponse';

export class TransformInterceptor implements NestInterceptor {
  intercept(
    context: ExecutionContext,
    next: CallHandler<ApiResponse | any>,
  ): Observable<ApiResponse | any> {
    return next.handle().pipe(
      map(data => {
        const http = context.switchToHttp();
        const res = http.getResponse();

        if(data instanceof ApiResponse) {
          if(data.status !== undefined) {
            res.status(data.status);
          }
        }

        return classToPlain(data);
      }),
    );
  }
}
EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2019-03-29 08:34:59

更新的答案

从nest版本6.1.0开始,可以在拦截器中设置状态代码;它将不再被覆盖(参见this PR):

代码语言:javascript
复制
context.switchToHttp()
      .getResponse()
      .status(205);

过时的答案

从拦截器设置状态代码是不可能的(请参阅此issue),因为:

  • 有时响应状态代码依赖于异常,异常筛选器在拦截器之后执行,
  • 全局响应控制器的逻辑是在通过网络发送最终结果之前执行的最后一步(这是默认状态代码出现的地方)。

因此,您的状态代码将被默认代码200/201或异常筛选器覆盖。

作为一种(hacky)解决方法,您可以使用异常筛选器在拦截器中设置状态代码:

1)创建自己的异常作为HttpException的包装器

代码语言:javascript
复制
export class StatusException extends HttpException {
  constructor(data, status: HttpStatus) {
    super(data, status);
  }
}

2)创建设置响应码并返回数据的异常过滤器:

代码语言:javascript
复制
@Catch(StatusException)
export class StatusFilter implements ExceptionFilter {
  catch(exception: StatusException, host: ArgumentsHost) {
    const ctx = host.switchToHttp();
    const response = ctx.getResponse<Response>();
    const status = exception.getStatus();
    console.log(`Setting status to ${status}`);
    response.status(status).json(exception.message);
  }
}

3)在拦截器中抛出相应的异常,而不是设置响应:

代码语言:javascript
复制
@Injectable()
export class StatusInterceptor implements NestInterceptor {
  intercept(context: ExecutionContext, next): Observable<any> {
    return next.handle().pipe(
      map((data: any) => {
        if (data.text === 'created') {
          throw new StatusException(data, HttpStatus.CREATED);
        } else {
          throw new StatusException(data, HttpStatus.ACCEPTED);
        }
      }),
    );
  }
}

4)在你的控制器中使用它:

代码语言:javascript
复制
@UseFilters(StatusFilter)
@UseInterceptors(StatusInterceptor)
@Controller()
export class AppController {
  @Get(':param')
  async get(@Param('param') param) {
    return { text: param };
  }
}

或者,您可以在控制器中注入@Res()并直接控制响应代码(但也会丢失拦截器、异常过滤器等)。

票数 6
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/55406194

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档