6

我在我的 TypeORM 实体字段电子邮件上设置了一个自定义的唯一验证器装饰器。NestJS 有依赖注入,但服务没有注入。

错误是:

TypeError: Cannot read property 'findByEmail' of undefined

对实施自定义电子邮件验证器有任何帮助吗?

user.entity.ts

@Column()
@Validate(CustomEmail, {
    message: "Title is too short or long!"
})
@IsEmail()
email: string;

我的CustomEmail验证人是

import {ValidatorConstraint, ValidatorConstraintInterface, 
ValidationArguments} from "class-validator";
import {UserService} from "./user.service";

@ValidatorConstraint({ name: "customText", async: true })
export class CustomEmail implements ValidatorConstraintInterface {

  constructor(private userService: UserService) {}
  async validate(text: string, args: ValidationArguments) {

    const user = await this.userService.findByEmail(text);
    return !user; 
  }

  defaultMessage(args: ValidationArguments) { 
    return "Text ($value) is too short or too long!";
  }
}

我知道我可以uniqueColumn选项中设置

@Column({
  unique: true
})

但这会引发 mysql 错误,并且会使ExceptionsHandler我的应用程序崩溃,所以我自己无法处理...

谢谢!

4

2 回答 2

11

我可以在这里提出两种不同的方法,第一种在本地捕获约束违规错误而无需额外的请求,第二种使用全局错误过滤器,在整个应用程序中捕获此类错误。我个人使用后者。

本地no-db请求解决方案

无需提出额外的数据库请求。您可以捕获违反唯一约束的错误并将任何HttpException您想要的内容扔给客户端。在users.service.ts

  public create(newUser: Partial<UserEntity>): Promise<UserEntity> {
    return this.usersRepository.save(newUser).catch((e) => {
      if (/(email)[\s\S]+(already exists)/.test(e.detail)) {
        throw new BadRequestException(
          'Account with this email already exists.',
        );
      }
      return e;
    });
  }

哪个会返回:

Insomnia 的错误截图(MacOS App)

全局错误过滤解决方案

甚至创建一个全局 QueryErrorFilter:

@Catch(QueryFailedError)
export class QueryErrorFilter extends BaseExceptionFilter {
  public catch(exception: any, host: ArgumentsHost): any {
    const detail = exception.detail;
    if (typeof detail === 'string' && detail.includes('already exists')) {
      const messageStart = exception.table.split('_').join(' ') + ' with';
      throw new BadRequestException(
        exception.detail.replace('Key', messageStart),
      );
    }
    return super.catch(exception, host);
  }
}

然后在main.ts

async function bootstrap() {
  const app = await NestFactory.create(/**/);
  /* ... */
  const { httpAdapter } = app.get(HttpAdapterHost);
  app.useGlobalFilters(new QueryErrorFilter(httpAdapter));
  /* ... */
  await app.listen(3000);
}
bootstrap();

这将给出一般$table entity with ($field)=($value) already exists.错误消息。例子:

在此处输入图像描述

于 2021-03-07T17:55:08.457 回答
6

我修改了我的代码。我正在检查用户服务(而不是自定义验证器)中用户名/电子邮件的唯一性,并在用户已插入数据库的情况下返回 HttpExcetion。

于 2018-02-25T11:25:19.363 回答