三簿透、Midway 接口安全認(rèn)證

閱讀本文前沼头,需要提前閱讀前置內(nèi)容:

一爷绘、Midway 增刪改查
二、Midway 增刪改查的封裝及工具類
三进倍、Midway 接口安全認(rèn)證
四土至、Midway 集成 Swagger 以及支持JWT bearer
五、Midway 中環(huán)境變量的使用

樣例源碼
DEMO LIVE

很多時候猾昆,后端接口需要登錄后才能進(jìn)行訪問陶因,甚至有的接口需要擁有相應(yīng)的權(quán)限才能訪問。
這里實現(xiàn)bearer驗證方式(bearerFormat 為 JWT)垂蜗。

安裝JWT組件

>npm i @midwayjs/jwt@3 --save
>npm i @types/jsonwebtoken --save-dev

安裝完后package.json文件中會多出如下配置

{
  "dependencies": {
    "@midwayjs/jwt": "^3.3.11"
  },
  "devDependencies": {
    "@types/jsonwebtoken": "^8.5.8"
  }
}

添加JWT配置

  • 修改src/config/config.default.ts楷扬,添加如下內(nèi)容解幽;
// src/config/config.default.ts
jwt: {
  secret: 'setscrew',
  expiresIn: 60 * 60 * 24,
}
  • 注冊JWT組件;
// src/configuration.ts
import * as jwt from '@midwayjs/jwt';

@Configuration({
  imports: [
    jwt,
    //...
  ],
})
export class ContainerLifeCycle {
    //...
}

關(guān)于JWT的詳細(xì)使用文檔烘苹,見:http://www.midwayjs.org/docs/extensions/jwt

安裝Redis組件

>npm i @midwayjs/redis@3 --save
>npm i @types/ioredis --save-dev

安裝完后package.json文件中會多出如下配置

{
  "dependencies": {
    "@midwayjs/redis": "^3.0.0"
  },
  "devDependencies": {
    "@types/ioredis": "^4.28.7"
  }
}

注冊Redis組件

// src/configuration.ts
import * as redis from '@midwayjs/redis';

@Configuration({
  imports: [
    redis,
    // ...
  ],
})
export class ContainerLifeCycle {
    // ...
}

添加配置

修改src/config/config.default.ts躲株,添加如下內(nèi)容:

添加Redis配置

// src/config/config.default.ts
redis: {
  client: {
    host: 127.0.0.1,
    port: 6379,
    db: 0,
  },
}

關(guān)于Redis的詳細(xì)使用文檔,見:http://www.midwayjs.org/docs/extensions/redis

添加安全攔截配置

// src/config/config.default.ts
app: {
  security: {
    prefix: '/api',         # 指定已/api開頭的接口地址需要攔截
    ignore: ['/api/login'], # 指定該接口地址镣衡,不需要攔截
  },
}

添加接口安全攔截中間件

添加常量定義

// src/common/Constant.ts
export class Constant {
  // 登陸驗證時霜定,緩存用戶登陸狀態(tài)KEY的前綴
  static TOKEM = 'TOKEN';
}

添加用戶訪問上下文類

// src/common/UserContext.ts
/**
 * 登陸后存儲訪問上下文的狀態(tài)數(shù)據(jù),同時也會存在redis緩存中
 */
export class UserContext {
  userId: number;
  username: string;
  phoneNum: string;
  constructor(userId: number, username: string, phoneNum: string) {
    this.userId = userId;
    this.username = username;
    this.phoneNum = phoneNum;
  }
}

新增或者編輯src/interface.ts廊鸥,將UserContext注冊到ApplecationContext

// src/interface.ts
import '@midwayjs/core';
import { UserContext } from './common/UserContext';

declare module '@midwayjs/core' {
  interface Context {
    userContext: UserContext;
  }
}

新增中間件src/middleware/security.middleware.ts

// src/middleware/security.middleware.ts
import { Config, Inject, Middleware } from '@midwayjs/decorator';
import { Context, NextFunction } from '@midwayjs/koa';
import { httpError } from '@midwayjs/core';
import { JwtService } from '@midwayjs/jwt';
import { UserContext } from '../common/UserContext';
import { RedisService } from '@midwayjs/redis';
import { Constant } from '../common/Constant';

/**
 * 安全驗證
 */
@Middleware()
export class SecurityMiddleware {

  @Inject()
  jwtUtil: JwtService;

  @Inject()
  cacheUtil: RedisService;

  @Config('app.security')
  securityConfig;

  resolve() {
    return async (ctx: Context, next: NextFunction) => {
      if (!ctx.headers['authorization']) {
        throw new httpError.UnauthorizedError('缺少憑證');
      }
      const parts = ctx.get('authorization').trim().split(' ');
      if (parts.length !== 2) {
        throw new httpError.UnauthorizedError('無效的憑證');
      }
      const [scheme, token] = parts;
      if (!/^Bearer$/i.test(scheme)) {
        throw new httpError.UnauthorizedError('缺少Bearer');
      }
      // 驗證token望浩,過期會拋出異常
      const jwt = await this.jwtUtil.verify(token, { complete: true });
      // jwt中存儲的user信息
      const payload = jwt['payload'];
      const key = Constant.TOKEM + ':' + payload.userId + ':' + token;
      const ucStr = await this.cacheUtil.get(key);
      // 服務(wù)器端緩存中存儲的user信息
      const uc: UserContext = JSON.parse(ucStr);
      if (payload.username !== uc.username) {
        throw new httpError.UnauthorizedError('無效的憑證');
      }
      // 存儲到訪問上下文中
      ctx.userContext = uc;
      return next();
    };
  }

  public match(ctx: Context): boolean {
    const { path } = ctx;
    const { prefix, ignore } = this.securityConfig;
    const exist = ignore.find((item) => {
      return item.match(path);
    });
    return path.indexOf(prefix) === 0 && !exist;
  }

  public static getName(): string {
    return 'SECURITY';
  }

}
  • @Config('app.security')裝飾類,指定加載配置文件src/config/config.**.ts中對應(yīng)的配置信息惰说;
  • 使用JwtService進(jìn)行JWT編碼校驗曾雕;

jwt token將用戶信息編碼在token中,解碼后可以獲取對應(yīng)用戶數(shù)據(jù)助被,通常情況下剖张,不需要存儲到redis中;
但是有個缺點就是揩环,不能人為控制分發(fā)出去的token失效搔弄。所以,有時人們會使用緩存中的用戶信息丰滑;
這里使用了JWT+Redis的方式顾犹,是為了演示兩種做法;

注冊中間件

// src/configuration.ts
this.app.useMiddleware([SecurityMiddleware, FormatMiddleware, ReportMiddleware]);

添加登陸接口

  • 添加DTO;
// src/api/dto/CommonDTO.ts
export class LoginDTO {
  username: string;
  password: string;
}
  • 添加VO;
// src/api/vo/CommonVO.ts
export class LoginVO {
  accessToken: string;
  expiresIn: number;
}
  • 修改src/service/user.service.ts褒墨,添加通過用戶名查找用戶接口炫刷;
import { Provide } from '@midwayjs/decorator';
import { User } from '../eneity/user';
import { InjectEntityModel } from '@midwayjs/orm';
import { Repository } from 'typeorm';
import { BaseService } from '../common/BaseService';

@Provide()
export class UserService extends BaseService<User> {

  @InjectEntityModel(User)
  model: Repository<User>;

  getModel(): Repository<User> {
    return this.model;
  }

  async findByUsername(username: string): Promise<User> {
    return this.model.findOne({ where: { username } });
  }

}
  • 添加Controllersrc/controller/common.controller.ts
// src/controller/common.controller.ts
import { Body, Config, Controller, Inject, Post } from '@midwayjs/decorator';
import { Context } from '@midwayjs/koa';
import { UserService } from '../service/user.service';
import { RedisService } from '@midwayjs/redis';
import { LoginDTO } from '../api/dto/CommonDTO';
import { LoginVO } from '../api/vo/CommonVO';
import { SnowflakeIdGenerate } from '../utils/Snowflake';
import { JwtService } from '@midwayjs/jwt';
import { Assert } from '../common/Assert';
import { ErrorCode } from '../common/ErrorCode';
import { UserContext } from '../common/UserContext';
import { Constant } from '../common/Constant';
import { ILogger } from '@midwayjs/core';
import { decrypt } from '../utils/PasswordEncoder';
import { Validate } from '@midwayjs/validate';
import { ApiResponse, ApiTags } from '@midwayjs/swagger';

@ApiTags(['common'])
@Controller('/api')
export class CommonController {

  @Inject()
  logger: ILogger;

  @Inject()
  ctx: Context;

  @Inject()
  userService: UserService;

  @Inject()
  cacheUtil: RedisService;

  @Inject()
  jwtUtil: JwtService;

  @Inject()
  idGenerate: SnowflakeIdGenerate;

  @Config('jwt')
  jwtConfig;

  @ApiResponse({ type: LoginVO })
  @Validate()
  @Post('/login', { description: '登陸' })
  async login(@Body() body: LoginDTO): Promise<LoginVO> {
    const user = await this.userService.findByUsername(body.username);
    Assert.notNull(user, ErrorCode.UN_ERROR, '用戶名或者密碼錯誤');
    const flag = decrypt(body.password, user.password);
    Assert.isTrue(flag, ErrorCode.UN_ERROR, '用戶名或者密碼錯誤');
    const uc: UserContext = new UserContext(user.id, user.username, user.phoneNum);
    const at = await this.jwtUtil.sign({ ...uc });
    const key = Constant.TOKEM + ':' + user.id + ':' + at;
    const expiresIn = this.jwtConfig.expiresIn;
    this.cacheUtil.set(key, JSON.stringify(uc), 'EX', expiresIn);
    const vo = new LoginVO();
    vo.accessToken = at;
    vo.expiresIn = expiresIn;
    return vo;
  }

}

使用Postman驗證

  • 調(diào)用接口(未設(shè)置憑證)郁妈;


    未設(shè)置憑證
  • 使用登陸接口獲取token浑玛;


    獲取憑證
  • 調(diào)用接口(使用憑證);


    使用憑證

版權(quán)所有噩咪,轉(zhuǎn)載請注明出處 [碼道功成]

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末顾彰,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子胃碾,更是在濱河造成了極大的恐慌涨享,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,126評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件仆百,死亡現(xiàn)場離奇詭異厕隧,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評論 2 382
  • 文/潘曉璐 我一進(jìn)店門吁讨,熙熙樓的掌柜王于貴愁眉苦臉地迎上來帖族,“玉大人,你說我怎么就攤上這事挡爵∈悖” “怎么了?”我有些...
    開封第一講書人閱讀 152,445評論 0 341
  • 文/不壞的土叔 我叫張陵茶鹃,是天一觀的道長涣雕。 經(jīng)常有香客問我,道長闭翩,這世上最難降的妖魔是什么挣郭? 我笑而不...
    開封第一講書人閱讀 55,185評論 1 278
  • 正文 為了忘掉前任,我火速辦了婚禮疗韵,結(jié)果婚禮上兑障,老公的妹妹穿的比我還像新娘。我一直安慰自己蕉汪,他們只是感情好流译,可當(dāng)我...
    茶點故事閱讀 64,178評論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著者疤,像睡著了一般福澡。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上驹马,一...
    開封第一講書人閱讀 48,970評論 1 284
  • 那天革砸,我揣著相機(jī)與錄音,去河邊找鬼糯累。 笑死算利,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的泳姐。 我是一名探鬼主播效拭,決...
    沈念sama閱讀 38,276評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼仗岸!你這毒婦竟也來了允耿?” 一聲冷哼從身側(cè)響起借笙,我...
    開封第一講書人閱讀 36,927評論 0 259
  • 序言:老撾萬榮一對情侶失蹤扒怖,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后业稼,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體盗痒,經(jīng)...
    沈念sama閱讀 43,400評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,883評論 2 323
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了俯邓。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片骡楼。...
    茶點故事閱讀 37,997評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖稽鞭,靈堂內(nèi)的尸體忽然破棺而出鸟整,到底是詐尸還是另有隱情,我是刑警寧澤朦蕴,帶...
    沈念sama閱讀 33,646評論 4 322
  • 正文 年R本政府宣布篮条,位于F島的核電站,受9級特大地震影響吩抓,放射性物質(zhì)發(fā)生泄漏涉茧。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,213評論 3 307
  • 文/蒙蒙 一疹娶、第九天 我趴在偏房一處隱蔽的房頂上張望伴栓。 院中可真熱鬧,春花似錦雨饺、人聲如沸钳垮。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,204評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽扔枫。三九已至,卻和暖如春锹安,著一層夾襖步出監(jiān)牢的瞬間短荐,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,423評論 1 260
  • 我被黑心中介騙來泰國打工叹哭, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留忍宋,地道東北人。 一個月前我還...
    沈念sama閱讀 45,423評論 2 352
  • 正文 我出身青樓风罩,卻偏偏與公主長得像糠排,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子超升,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,722評論 2 345

推薦閱讀更多精彩內(nèi)容