nestjs 使用jwt实现认证

JWT:

JSON WEB TOKEN 用来解决跨域认证的问题,服务器认证后,生成JSON对象返回给客户端,服务器不保存session数据,所有数据都保存在客户端,每次请求都发回给服务器.

JWT结构:

image.png

http://www.ruanyifeng.com/blog/2018/07/json_web_token-tutorial.html

1. header

{
"alg": "HS256", // 签名的算法默认是 HMAC SHA256
"typ": "JWT" // 令牌的类型
}

2. payload

官方规定了7个官方字段

  • iss (issuer):签发人
  • exp (expiration time):过期时间
  • sub (subject):主题
  • aud (audience):受众
  • nbf (Not Before):生效时间
  • iat (Issued At):签发时间
  • jti (JWT ID):编号

除了官方字段,可更具自己的服务自定义字段如:

{
"account": "123",
"role": "superAdmin"
}

3. signature

对前面两个部分的签名。防止数据被篡改。使用header中指定的签名算法(默认是HMAC SHA256),按照下面的公式产生签名

HAMCSHA256(
  base64UrlEncode(header) + '.' +
  base64UrlEncode(payload),
  secret // 密钥,这个密钥只有服务器知道
)

nest实现jwt

流程

  1. 客户端用户进行登录请求
  2. 服务端拿到请求后,根据参数查询用户表
  3. 若匹配到用户,将用户信息进行签证,并颁发token
  4. 客户端拿到token后,存储在某一个地方,在之后的请求中都带上token
  5. 服务端收到带token的请求后,直接根据签证进行校验,无需进行查询用户信息

1 用户登录

user.controller.ts

@Post('login')
  async login(
    @Body()
    data: LoginUserDto,
  ): Promise<THttpResponse<LoginUserCmd>> {
    // 前端加密密码需要解密
    // data.password = this.utilService.encryption(data.password);
    data.password = this.utilService.decryption(data.password);
    let result = await this.userService.login(data);

    // 业务错误直接返回
    if (result.errorCode) {
      return result;
    }

    // 颁发 token
    const tokenInfo = await this.authService.genToken({
      account: result.account,
      name: result.name,
      roles: result.roles,
      permissions: result.permissions,
      department: result.department,
      number: result.number,
      email: result.email,
      type: result.type,
    });
    result.accesstoken = tokenInfo.accesstoken;
    result.refreshToken = tokenInfo.refreshToken;

    return { result };
  }

2. 颁发token

auth.service.ts:

@Injectable()
export class AuthService {
  constructor(
    private readonly userService: UserService,
    private readonly jwtService: JwtService,
  ) // private readonly redisService: RedisService,
  {}

  // 生成 accesstoken refreshToken
  async genToken(payload: JwtPayload): Promise<any> {
    const accesstoken = this.jwtService.sign(payload);
    const refreshToken = this.jwtService.sign(payload, {
      expiresIn: jwtConstants.refreshTokenExpiresIn,
    });
    return { accesstoken, refreshToken };
}

auth.module.ts

import { forwardRef, Module } from '@nestjs/common';
import { JwtModule } from '@nestjs/jwt';
import { PassportModule } from '@nestjs/passport';
import { UserModule } from '../user/user.module';
import { AuthService } from './auth.service';
import { jwtConstants } from './constants';
import { JwtStrategy } from './strategies/jwt.strategy';

@Module({
  imports: [
    forwardRef(() => UserModule),
    PassportModule,
    JwtModule.register({
      secret: jwtConstants.secret,
      signOptions: { expiresIn: jwtConstants.accessTokenExpiresIn },
    }),
  ],
  providers: [AuthService, JwtStrategy],
  exports: [AuthService],
})
export class AuthModule {}

3. 验证获取token

jwt.strategy.ts:

import { Injectable } from '@nestjs/common';
import { PassportStrategy } from '@nestjs/passport';
import { ExtractJwt, Strategy } from 'passport-jwt';
import { jwtConstants } from '../constants';
import { JwtPayload } from '../jwt-payload.interface';
import { getNamespace } from 'cls-hooked';
import { PublicJwtPayload } from '../public-jwt-payload.interface';

@Injectable()
export class JwtStrategy extends PassportStrategy(Strategy) {
  constructor() {
    super({
      jwtFromRequest: ExtractJwt.fromAuthHeaderAsBearerToken(),
      ignoreExpiration: false,
      secretOrKey: jwtConstants.secret,
    });
  }

  async validate(payload: JwtPayload & PublicJwtPayload) {
    if (payload.clientId) {
      // 公共接口部分
      return {
        clientId: payload.clientId,
        secret: payload.secret,
      };
    } else {
      // 全局注入用户信息
      getNamespace('create-nest-app').set('user', {
        id: payload.id,
        account: payload.account,
        number: payload.number,
      });

      // 返回的对象注入到 request.user
      return {
        account: payload.account,
        name: payload.name,
        roles: payload.roles,
        permissions: payload.permissions,
        department: payload.department,
        number: payload.number,
        email: payload.email,
        type: payload.type,
      };
    }
  }
}

参考:
https://juejin.im/post/6844904097317912584

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351