login.js 2.6 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485
  1. 'use strict';
  2. const assert = require('assert');
  3. const _ = require('lodash');
  4. const { ObjectId } = require('mongoose').Types;
  5. const { CrudService } = require('naf-framework-mongoose/lib/service');
  6. const { BusinessError, ErrorCode } = require('naf-core').Error;
  7. const jwt = require('jsonwebtoken');
  8. const uuid = require('uuid');
  9. class LoginService extends CrudService {
  10. constructor(ctx) {
  11. super(ctx, 'login');
  12. this.model = this.ctx.model.User;
  13. this.rmodel = this.ctx.model.Role;
  14. }
  15. // 用户登录
  16. async login(data) {
  17. const { phone, passwd } = data;
  18. // 根据用户输入的手机号查询其他用户表中是否存在相应数据
  19. const user = await this.model.findOne({ phone });
  20. // 如果用户不存在抛出异常
  21. if (!user) {
  22. throw new BusinessError(ErrorCode.USER_NOT_EXIST);
  23. }
  24. const _user = await this.model.findOne({ phone }, '+passwd');
  25. // 将用户输入的密码进行加密并与查询到的用户数据密码相比对
  26. const pas = await this.createJwtPwd(passwd);
  27. // 如果两个密码不一致抛出异常
  28. if (pas !== _user.passwd.secret) {
  29. throw new BusinessError(ErrorCode.BAD_PASSWORD);
  30. }
  31. // 取出用户的类型,根据用户类型返回相应信息
  32. const state = uuid();
  33. const key = `free:auth:state:${state}`;
  34. const _roles = [];
  35. for (const elm of user.roles) {
  36. const role = await this.rmodel.findById({ _id: ObjectId(elm) });
  37. if (role) {
  38. _roles.push(role);
  39. }
  40. }
  41. user.roles = _roles;
  42. const token = await this.createJwt(user);
  43. await this.app.redis.set(key, token, 'EX', 60 * 60 * 24);
  44. return { key };
  45. }
  46. // 创建登录Token
  47. async createJwtPwd(password) {
  48. const { secret, expiresIn, issuer } = this.config.jwt;
  49. const token = await jwt.sign(password, secret);
  50. return token;
  51. }
  52. // 创建登录Token
  53. async createJwt({ id, name, phone, roles, remark, openid }) {
  54. const { secret, expiresIn = '1d', issuer = type } = this.config.jwt;
  55. const subject = phone;
  56. const res = { uid: id, name, phone, roles, openid, remark };
  57. const token = await jwt.sign(res, secret, { expiresIn, issuer, subject });
  58. return token;
  59. }
  60. // 取得redis内token信息
  61. async token({ key }) {
  62. assert(key, 'key不能为空');
  63. const token = await this.app.redis.get(key);
  64. if (!token) {
  65. throw new BusinessError(ErrorCode.SERVICE_FAULT, 'token已经过期');
  66. }
  67. return { token };
  68. }
  69. // 删除操作
  70. async destroy({ key }) {
  71. const res = await this.app.redis.del(key);
  72. console.log(res);
  73. return res;
  74. }
  75. }
  76. module.exports = LoginService;