0

0

TypeORM与NestJS应用中实现用户密码自动哈希的教程

心靈之曲

心靈之曲

发布时间:2025-12-05 14:05:02

|

739人浏览过

|

来源于php中文网

原创

TypeORM与NestJS应用中实现用户密码自动哈希的教程

本教程详细介绍了如何在typeorm与nestjs应用中,利用实体生命周期钩子(如`@beforeinsert()`和`@beforeupdate()`)实现用户密码的自动哈希。通过在用户实体中集成`bcrypt`库,我们可以在保存用户模型时,无需手动干预,自动将明文密码转换为安全的哈希值,确保数据存储的安全性与便捷性。

引言:保障用户密码安全与自动化处理

在任何用户认证系统中,妥善存储用户密码是至关重要的安全实践。直接存储明文密码是极其危险的,一旦数据库泄露,所有用户账户都将面临风险。因此,对密码进行哈希处理是行业标准做法。为了提升开发效率并确保一致性,我们通常希望在用户模型持久化到数据库时,能够自动完成密码的哈希过程,而无需在每次保存操作时手动调用哈希函数。

TypeORM作为一款强大的ORM框架,提供了实体生命周期钩子(Entity Lifecycle Hooks),这些钩子允许我们在实体被插入、更新或删除等操作前后执行自定义逻辑。结合NestJS的模块化架构,我们可以优雅地实现密码的自动哈希。

TypeORM实体生命周期钩子概览

TypeORM提供了一系列装饰器,用于标记在特定数据库操作前后执行的方法。其中,对于密码哈希场景,@BeforeInsert()和@BeforeUpdate()是核心。

  • @BeforeInsert(): 在实体首次插入数据库之前执行。这非常适合在新用户注册时哈希其密码。
  • @BeforeUpdate(): 在实体更新到数据库之前执行。如果用户有修改密码的功能,此钩子可以确保新密码也被正确哈希。

这些钩子会在使用TypeORM的repository.save()或entityManager.save()方法持久化实体时自动触发。需要注意的是,如果直接使用repository.insert()方法,这些钩子将不会被触发,因为insert()方法旨在提供更底层的、性能导向的插入,它会绕过ORM的一些高级特性,包括生命周期事件。因此,在需要触发实体生命周期钩子的场景下,应优先使用save()方法。

实现步骤:在用户实体中集成自动密码哈希

1. 安装密码哈希库

我们将使用bcrypt库进行密码哈希。bcrypt是一种流行的、安全的密码哈希函数,它设计为计算密集型,以抵御彩虹表攻击和暴力破解。

首先,在您的NestJS项目中安装bcrypt:

npm install bcrypt
npm install -D @types/bcrypt

2. 配置用户实体

接下来,修改您的用户实体(例如User.entity.ts),在其中添加password属性,并实现一个带有@BeforeInsert()和@BeforeUpdate()装饰器的方法来处理密码哈希。

import { Entity, PrimaryGeneratedColumn, Column, BeforeInsert, BeforeUpdate } from 'typeorm';
import * as bcrypt from 'bcrypt';

@Entity()
export class User {
  @PrimaryGeneratedColumn()
  id: number;

  @Column({ unique: true })
  username: string;

  @Column()
  password: string;

  @Column({ default: true })
  isActive: boolean;

  /**
   * 在插入数据库之前哈希密码
   * 如果password属性存在且不为空,则对其进行哈希处理
   */
  @BeforeInsert()
  async hashPasswordOnInsert() {
    if (this.password) {
      this.password = await bcrypt.hash(this.password, 10); // 10是盐值轮次,建议在8-12之间
    }
  }

  /**
   * 在更新数据库之前哈希密码
   * 仅当password属性被修改时才进行哈希处理
   */
  @BeforeUpdate()
  async hashPasswordOnUpdate() {
    // 可以在此处添加逻辑,仅当密码实际发生变化时才进行哈希
    // 例如:通过比较当前密码哈希值与数据库中的旧值,但这需要额外查询
    // 更简单的做法是,如果前端发送了新的明文密码,就重新哈希
    if (this.password && this.password.length < 60) { // 假设哈希后的密码长度远大于60,用于区分明文密码
      this.password = await bcrypt.hash(this.password, 10);
    }
  }
}

在上述代码中:

  • @BeforeInsert()装饰的hashPasswordOnInsert方法会在每次创建新用户并保存时触发。
  • @BeforeUpdate()装饰的hashPasswordOnUpdate方法会在每次更新现有用户并保存时触发。我们添加了一个简单的长度检查 (this.password.length
  • bcrypt.hash(this.password, 10):10是盐值轮次(salt rounds),它决定了哈希计算的复杂程度。值越大,哈希越安全,但计算时间也越长。通常建议在8到12之间。

3. 在服务层使用

在您的用户服务(例如UserService)中,您可以像往常一样创建和保存用户,无需在服务层手动调用哈希函数。哈希过程将由TypeORM的实体钩子自动处理。

import { Injectable } from '@nestjs/common';
import { InjectRepository } from '@nestjs/typeorm';
import { Repository } from 'typeorm';
import { User } from './user.entity';

@Injectable()
export class UserService {
  constructor(
    @InjectRepository(User)
    private usersRepository: Repository,
  ) {}

  async createUser(username: string, passwordPlain: string): Promise {
    const newUser = this.usersRepository.create({ username, password: passwordPlain });
    // 当调用save()方法时,User实体中的@BeforeInsert()钩子将自动触发,对password进行哈希
    return this.usersRepository.save(newUser);
  }

  async updateUserPassword(userId: number, newPasswordPlain: string): Promise {
    const user = await this.usersRepository.findOne({ where: { id: userId } });
    if (!user) {
      return undefined;
    }
    user.password = newPasswordPlain;
    // 当调用save()方法时,User实体中的@BeforeUpdate()钩子将自动触发,对newPasswordPlain进行哈希
    return this.usersRepository.save(user);
  }

  // ... 其他用户相关方法
}

通过这种方式,UserService中的代码保持简洁,关注于业务逻辑,而密码哈希的底层实现则封装在实体内部。

Detect GPT
Detect GPT

一个Chrome插件,检测您浏览的页面是否包含人工智能生成的内容

下载

注意事项与最佳实践

  1. 安全性考量:

    • 选择合适的盐值轮次: bcrypt的盐值轮次应该根据您的服务器性能和安全需求进行调整。更高的轮次意味着更强的安全性,但也需要更多的计算资源。

    • 永远不要存储盐值: bcrypt生成的哈希值本身就包含了盐值,因此您无需单独存储它。

    • 密码比对: 当用户尝试登录时,您需要将用户输入的明文密码与数据库中存储的哈希密码进行比对。bcrypt提供了compare方法来完成此操作:

      import * as bcrypt from 'bcrypt';
      
      async function validatePassword(plainPassword: string, hashedPasswordFromDb: string): Promise {
        return bcrypt.compare(plainPassword, hashedPasswordFromDb);
      }
  2. 钩子的触发时机:

    • 再次强调,@BeforeInsert()和@BeforeUpdate()钩子仅在使用repository.save()或entityManager.save()方法时触发。如果您出于某种特殊原因需要使用repository.insert()或repository.update()方法,那么您将需要手动处理密码哈希。
    • save()方法会根据实体是否具有主键来判断是执行插入还是更新操作。如果实体没有主键,则执行插入;如果实体有主键,则执行更新。
  3. 密码更新处理:

    • 在@BeforeUpdate()中,判断是否需要重新哈希密码的逻辑需要谨慎。上述示例中的长度检查是一种简易方式,但更健壮的方法可能是在更新操作时,明确只在接收到明文密码时才进行哈希。例如,如果您的更新DTO只包含需要更新的字段,并且password字段仅在用户请求修改密码时才包含明文,那么直接哈希即可。
  4. 性能:

    • bcrypt是一个计算密集型操作,尤其是在高盐值轮次下。对于高并发的注册或密码更新场景,这可能会对服务器性能产生轻微影响。然而,对于大多数Web应用而言,这种开销是可接受的,并且是保障安全性的必要代价。bcrypt.hash是异步的,因此不会阻塞Node.js事件循环。

总结

通过在TypeORM实体中巧妙利用@BeforeInsert()和@BeforeUpdate()生命周期钩子,并结合bcrypt库,我们能够实现NestJS应用中用户密码的自动化哈希。这种方法不仅提高了代码的内聚性和可维护性,将密码安全逻辑封装在实体内部,还确保了每次密码持久化时都能自动进行安全的哈希处理,极大地简化了开发流程并增强了系统的安全性。遵循这些实践,您的NestJS应用将拥有一个健壮且安全的密码管理机制。

相关专题

更多
length函数用法
length函数用法

length函数用于返回指定字符串的字符数或字节数。可以用于计算字符串的长度,以便在查询和处理字符串数据时进行操作和判断。 需要注意的是length函数计算的是字符串的字符数,而不是字节数。对于多字节字符集,一个字符可能由多个字节组成。因此,length函数在计算字符串长度时会将多字节字符作为一个字符来计算。更多关于length函数的用法,大家可以阅读本专题下面的文章。

922

2023.09.19

js正则表达式
js正则表达式

php中文网为大家提供各种js正则表达式语法大全以及各种js正则表达式使用的方法,还有更多js正则表达式的相关文章、相关下载、相关课程,供大家免费下载体验。

510

2023.06.20

js获取当前时间
js获取当前时间

JS全称JavaScript,是一种具有函数优先的轻量级,解释型或即时编译型的编程语言;它是一种属于网络的高级脚本语言,主要用于Web,常用来为网页添加各式各样的动态功能。js怎么获取当前时间呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

244

2023.07.28

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

258

2023.08.03

js是什么意思
js是什么意思

JS是JavaScript的缩写,它是一种广泛应用于网页开发的脚本语言。JavaScript是一种解释性的、基于对象和事件驱动的编程语言,通常用于为网页增加交互性和动态性。它可以在网页上实现复杂的功能和效果,如表单验证、页面元素操作、动画效果、数据交互等。

5286

2023.08.17

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

477

2023.09.01

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

209

2023.09.04

Js中concat和push的区别
Js中concat和push的区别

Js中concat和push的区别:1、concat用于将两个或多个数组合并成一个新数组,并返回这个新数组,而push用于向数组的末尾添加一个或多个元素,并返回修改后的数组的新长度;2、concat不会修改原始数组,是创建新的数组,而push会修改原数组,将新元素添加到原数组的末尾等等。本专题为大家提供concat和push相关的文章、下载、课程内容,供大家免费下载体验。

218

2023.09.14

Java JVM 原理与性能调优实战
Java JVM 原理与性能调优实战

本专题系统讲解 Java 虚拟机(JVM)的核心工作原理与性能调优方法,包括 JVM 内存结构、对象创建与回收流程、垃圾回收器(Serial、CMS、G1、ZGC)对比分析、常见内存泄漏与性能瓶颈排查,以及 JVM 参数调优与监控工具(jstat、jmap、jvisualvm)的实战使用。通过真实案例,帮助学习者掌握 Java 应用在生产环境中的性能分析与优化能力。

19

2026.01.20

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.4万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号