猫鼬模式预方法中的打字稿错误:“this”意外别名为局部变量

Posted

技术标签:

【中文标题】猫鼬模式预方法中的打字稿错误:“this”意外别名为局部变量【英文标题】:Typescript Error in mongoose schemas pre method: Unexpected aliasing of 'this' to local variable 【发布时间】:2020-07-07 03:15:59 【问题描述】:

我正在尝试为 Ionic 应用程序实现基于 MongoDB 和 NestJS 的身份验证,并在向路由 api/users 发出 POST 请求后收到以下错误消息:

[Nest] 85372 - 03/26/2020, 14:04:49 [ExceptionsHandler] 无法读取未定义的属性“密码”+23790ms

在我的 users.schema.ts 文件中获取错误消息:

'this' 意外别名为局部变量.eslint(@typescript-eslint/no-this-alias)

我的 users.schema.ts 看起来像这样(注释错误的行):

import * as mongoose from 'mongoose';
import * as bcrypt from 'bcryptjs'

export const UserSchema = new mongoose.Schema(
    email: 
        type: String,
        unique: true,
        required: true
    ,
    username: 
        type: String,
        unique: true,
        required: true
    ,
    password: 
        type: String,
        unique: true,
        required: true
    ,
    createdAt: 
        type: Date,
        default: Date.now
    ,
    updatedAt: 
        type: Date,
        default: Date.now
    
);

UserSchema.pre('save', function (next) 
    const user = this; // This is marked as an error in vs code
    if (!user.isModified('password')) return next();

    bcrypt.genSalt(10, (err, salt) => 
        if (err) return next(err);
        bcrypt.hash(this.user.password, salt, (err, hash) => 
            if (err) return next();
            this.user.password = hash;
            next();
        );
    );
);

UserSchema.methods.checkPassword = (attempt, callback) => 
    bcrypt.compare(attempt, this.user.password, (err, isMatch) => 
        if (err) return callback(err);
        callback(null, isMatch);
    )

我尝试使用箭头函数实现相同的架构,但在向 api/users 发出 POST 请求后收到以下错误消息:

[Nest] 85947 - 03/26/2020, 14:09:30 [ExceptionsHandler] 无法读取未定义 +22567ms 的属性“isModified”

UserSchema.pre('save', (next) => 
    if (!this.user.isModified('password')) return next();

    bcrypt.genSalt(10, (err, salt) => 
        if (err) return next(err);
        bcrypt.hash(this.user.password, salt, (err, hash) => 
            if (err) return next();
            this.user.password = hash;
            next();
        );
    );
);

我在这里做错了什么?

【问题讨论】:

【参考方案1】:

你不能使用箭头函数,因为你失去了this的踪迹。阅读箭头函数的官方文档,有一些关于“数组函数中的 this 是什么”的非常好的例子。 MDN Arrow function expressions

有两种解决方案:

移除 bcrypt 的箭头函数回调

UserSchema.pre('save', function (next) 
  if (!this.isModified('password')) return next();

  bcrypt.genSalt(10, function (err, salt) 
    if (err) return next(err);
    bcrypt.hash(this.user.password, salt, (err, hash) => 
      if (err) return next();
      this.user.password = hash;
      next();
    );
  );
);

禁用 eslint 规则并始终使用用户变量

UserSchema.pre('save', function (next) 
  const user = this;
  if (!this.isModified('password')) return next();

  bcrypt.genSalt(10, (err, salt) => 
    if (err) return next(err);
    bcrypt.hash(user.password, salt, (err, hash) => 
      if (err) return next();
      user.password = hash;
      next();
    );
  );
);

【讨论】:

以上是关于猫鼬模式预方法中的打字稿错误:“this”意外别名为局部变量的主要内容,如果未能解决你的问题,请参考以下文章

预中间件中的猫鼬打字稿不存在

如何在打字稿中的猫鼬userschema.methods中使用“this”

使用猫鼬模型设置打字稿

打字稿:猫鼬模式的静态方法

带有打字稿的猫鼬,来自猫鼬的错误“连接”

打字稿 - 在箭头(回调)函数中使用它 - 猫鼬