LoginServiceImpl.java 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288
  1. package com.xjrsoft.module.system.service.impl;
  2. import cn.dev33.satoken.context.SaHolder;
  3. import cn.dev33.satoken.oauth2.SaOAuth2Manager;
  4. import cn.dev33.satoken.oauth2.config.SaOAuth2Config;
  5. import cn.dev33.satoken.secure.BCrypt;
  6. import cn.dev33.satoken.session.SaSession;
  7. import cn.dev33.satoken.stp.StpUtil;
  8. import cn.dev33.satoken.temp.SaTempUtil;
  9. import cn.hutool.core.util.IdUtil;
  10. import cn.hutool.core.util.ObjectUtil;
  11. import cn.hutool.core.util.StrUtil;
  12. import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
  13. import com.baomidou.mybatisplus.core.toolkit.CollectionUtils;
  14. import com.baomidou.mybatisplus.core.toolkit.StringPool;
  15. import com.baomidou.mybatisplus.core.toolkit.Wrappers;
  16. import com.xjrsoft.common.constant.GlobalConstant;
  17. import com.xjrsoft.common.enums.EnabledMark;
  18. import com.xjrsoft.common.enums.RoleEnum;
  19. import com.xjrsoft.common.exception.MyException;
  20. import com.xjrsoft.common.model.result.R;
  21. import com.xjrsoft.common.utils.FixedArithmeticCaptcha;
  22. import com.xjrsoft.common.utils.RSAUtil;
  23. import com.xjrsoft.common.utils.RedisUtil;
  24. import com.xjrsoft.common.utils.WeChatUtil;
  25. import com.xjrsoft.config.CommonPropertiesConfig;
  26. import com.xjrsoft.config.LicenseConfig;
  27. import com.xjrsoft.module.organization.entity.*;
  28. import com.xjrsoft.module.organization.mapper.UserRoleRelationMapper;
  29. import com.xjrsoft.module.organization.service.*;
  30. import com.xjrsoft.module.system.dto.CreateTokenDto;
  31. import com.xjrsoft.module.system.dto.LoginByCodeDto;
  32. import com.xjrsoft.module.system.dto.LoginCaptchaDto;
  33. import com.xjrsoft.module.system.dto.LoginDto;
  34. import com.xjrsoft.module.system.service.ILoginService;
  35. import com.xjrsoft.module.system.vo.CreateTokenVo;
  36. import com.xjrsoft.module.system.vo.ImgCaptchaVo;
  37. import com.xjrsoft.module.system.vo.LoginByCodeVo;
  38. import com.xjrsoft.module.system.vo.LoginVo;
  39. import lombok.AllArgsConstructor;
  40. import org.jetbrains.annotations.NotNull;
  41. import org.springframework.beans.factory.annotation.Autowired;
  42. import org.springframework.stereotype.Service;
  43. import java.util.List;
  44. import java.util.UUID;
  45. import java.util.stream.Collectors;
  46. /**
  47. * @Author: tzx
  48. * @Date: 2023/4/21 14:22
  49. */
  50. @Service
  51. @AllArgsConstructor
  52. public class LoginServiceImpl implements ILoginService {
  53. private final IUserService userService;
  54. private final IUserDeptRelationService userDeptRelationService;
  55. private final IUserPostRelationService userPostRelationService;
  56. private final IPostService postService;
  57. private final IDepartmentService departmentService;
  58. private final RedisUtil redisUtil;
  59. private final LicenseConfig licenseConfig;
  60. private final WeChatUtil weChatUtil;
  61. private final UserRoleRelationMapper userRoleRelationMapper;
  62. @Autowired
  63. private CommonPropertiesConfig commonPropertiesConfig;
  64. @Override
  65. public LoginVo login(LoginDto dto) throws Exception {
  66. if (licenseConfig.getEnabled()) {
  67. //查出所有在线用户
  68. List<String> onlineUser = StpUtil.searchSessionId("", 0, Integer.MAX_VALUE);
  69. //如果已经登录人数超过授权人数 不允许登录
  70. if (onlineUser.size() >= licenseConfig.getLoginMax()) {
  71. throw new MyException("登录人数超过授权人数,无法登录,请联系管理员!");
  72. }
  73. }
  74. String captchaCode = redisUtil.get(dto.getKey(), 0);
  75. if (captchaCode == null) {
  76. throw new MyException("验证码已过期,请刷新验证码!");
  77. }
  78. if (!captchaCode.equals(dto.getCode())) {
  79. throw new MyException("验证码不正确,请刷新验证码!");
  80. }
  81. // rsa解密
  82. String decryptData = RSAUtil.decrypt(dto.getPassword());
  83. dto.setPassword(decryptData);
  84. LambdaQueryWrapper<User> queryWrapper = new LambdaQueryWrapper<>();
  85. queryWrapper.eq(User::getUserName, dto.getUserName());
  86. User loginUser = userService.getOne(queryWrapper);
  87. // if (loginUser == null || !StrUtil.equals(loginUser.getPassword(), SaSecureUtil.md5BySalt(dto.getPassword(), GlobalConstant.SECRET_KEY))) {
  88. // throw new MyException("账号或密码不正确");
  89. // }
  90. if (loginUser == null || !BCrypt.checkpw(dto.getPassword(), loginUser.getPassword())) {
  91. throw new MyException("账号或密码不正确");
  92. }
  93. return getLoginInfo(loginUser, "PC");
  94. }
  95. @Override
  96. public LoginByCodeVo loginByCode(LoginByCodeDto dto) throws Exception {
  97. LoginByCodeVo result = new LoginByCodeVo();
  98. String openId = weChatUtil.getOpenid(dto.getCode());
  99. if (openId == null) throw new MyException("code无效");
  100. LambdaQueryWrapper<User> queryWrapper = new LambdaQueryWrapper<>();
  101. queryWrapper.eq(User::getOpenId, openId);
  102. User loginUser = userService.getOne(queryWrapper);
  103. result.setOpenid(openId);
  104. if (loginUser == null) {
  105. return result;
  106. }
  107. LoginVo loginVo = getLoginInfo(loginUser, "WX-MP");
  108. result.setToken(loginVo.getToken());
  109. result.setUserType(loginVo.getUserType());
  110. return result;
  111. }
  112. /**
  113. * 手机验证码登录
  114. *
  115. * @param dto
  116. * @return
  117. * @throws Exception
  118. */
  119. @Override
  120. public LoginVo loginByCaptcha(LoginCaptchaDto dto) throws Exception {
  121. User user = userService.getOne(Wrappers.<User>lambdaQuery().eq(User::getMobile, dto.getMobile()), false);
  122. if (user == null) {
  123. throw new MyException("用户不存在!");
  124. }
  125. return getLoginInfo(user, "Captcha");
  126. }
  127. @Override
  128. public CreateTokenVo createToken(CreateTokenDto dto) {
  129. CreateTokenVo vo = new CreateTokenVo();
  130. if (dto.getExpire() == -1) {
  131. String token = SaTempUtil.createToken(IdUtil.fastSimpleUUID() + StringPool.UNDERSCORE + GlobalConstant.SECRET_KEY, Integer.MAX_VALUE);
  132. vo.setToken(token);
  133. return vo;
  134. } else {
  135. String token = SaTempUtil.createToken(IdUtil.fastSimpleUUID() + StringPool.UNDERSCORE + GlobalConstant.SECRET_KEY, dto.getExpire());
  136. vo.setToken(token);
  137. return vo;
  138. }
  139. }
  140. /**
  141. * 图形验证码
  142. */
  143. @Override
  144. public ImgCaptchaVo imgCaptcha() {
  145. // 算术类型
  146. FixedArithmeticCaptcha captcha = new FixedArithmeticCaptcha(130, 48);
  147. String rKey = GlobalConstant.LOGIN_IMG_CAPTCHA + UUID.randomUUID().toString();
  148. // 存入redis并设置过期时间为30分钟
  149. redisUtil.set(rKey, captcha.text(), 30 * 60);
  150. return new ImgCaptchaVo(rKey, captcha.toBase64());
  151. }
  152. private LoginVo getLoginInfo(User loginUser, String loginType) throws Exception {
  153. LoginVo result = new LoginVo();
  154. if (loginUser.getEnabledMark() == EnabledMark.DISABLED.getCode()) {
  155. throw new MyException("账户未启用");
  156. }
  157. //此登录接口登录web端
  158. StpUtil.login(loginUser.getId(), loginType);
  159. // 获取用户类型(根据固定角色进行匹配)
  160. List<UserRoleRelation> relations = userRoleRelationMapper.selectList(Wrappers.lambdaQuery(UserRoleRelation.class)
  161. .select(UserRoleRelation::getRoleId)
  162. .eq(UserRoleRelation::getUserId, StpUtil.getLoginIdAsLong()));
  163. result.setUserType(roleMatching(relations));
  164. SaSession tokenSession = StpUtil.getTokenSession();
  165. List<UserDeptRelation> userDeptRelations = userDeptRelationService.list(Wrappers.lambdaQuery(UserDeptRelation.class)
  166. .eq(UserDeptRelation::getUserId, StpUtil.getLoginIdAsLong()));
  167. List<UserPostRelation> userPostRelations = userPostRelationService.list(Wrappers.lambdaQuery(UserPostRelation.class)
  168. .eq(UserPostRelation::getUserId, StpUtil.getLoginIdAsLong()));
  169. //获取登陆人所选择的身份缓存
  170. String postId = redisUtil.get(GlobalConstant.LOGIN_IDENTITY_CACHE_PREFIX + loginUser.getId());
  171. Post post = new Post();
  172. if (StrUtil.isNotBlank(postId)) {
  173. post = postService.getById(postId);
  174. }
  175. if (userPostRelations.size() > 0) {
  176. List<Long> postIds = userPostRelations.stream().map(UserPostRelation::getPostId).collect(Collectors.toList());
  177. List<Post> postList = postService.listByIds(postIds);
  178. if (StrUtil.isBlank(postId) && CollectionUtils.isNotEmpty(postList)) {
  179. post = postList.get(0);
  180. }
  181. tokenSession.set(GlobalConstant.LOGIN_USER_POST_INFO_KEY, post);
  182. tokenSession.set(GlobalConstant.LOGIN_USER_POST_LIST_KEY, postList);
  183. loginUser.setPostId(post.getId());
  184. //将登陆人所选择的身份缓存起来
  185. //切换身份的时候 会一起修改
  186. redisUtil.set(GlobalConstant.LOGIN_IDENTITY_CACHE_PREFIX + loginUser.getId(), post.getId());
  187. }
  188. if (userDeptRelations.size() > 0) {
  189. // 存当前用户所有部门到缓存
  190. List<Long> departmentIds = userDeptRelations.stream().map(UserDeptRelation::getDeptId).collect(Collectors.toList());
  191. List<Department> departmentList = departmentService.listByIds(departmentIds);
  192. tokenSession.set(GlobalConstant.LOGIN_USER_DEPT_LIST_KEY, departmentList);
  193. //如果此人有岗位 使用岗位的deptId 找到当前组织机构
  194. if (ObjectUtil.isNotNull(post.getId())) {
  195. Department department = departmentService.getById(post.getDeptId());
  196. tokenSession.set(GlobalConstant.LOGIN_USER_DEPT_INFO_KEY, department);
  197. loginUser.setDepartmentId(department.getId());
  198. } else {
  199. if (departmentList.size() > 0) {
  200. Department department = departmentList.get(0);
  201. tokenSession.set(GlobalConstant.LOGIN_USER_DEPT_INFO_KEY, department);
  202. loginUser.setDepartmentId(department.getId());
  203. }
  204. }
  205. }
  206. //根据登录信息 将post 和 department 信息存入用户信息中
  207. tokenSession.set(GlobalConstant.LOGIN_USER_INFO_KEY, loginUser);
  208. result.setToken(StpUtil.getTokenValue());
  209. // 判断是不是OAuth2
  210. String oauth2Info = SaHolder.getRequest().getCookieValue("Oauth2Info");
  211. if (oauth2Info != null) {
  212. result.setRedirectUri(redisUtil.get(oauth2Info));
  213. }
  214. return result;
  215. }
  216. /**
  217. * 角色匹配
  218. *
  219. * @return
  220. */
  221. private Long roleMatching(List<UserRoleRelation> relations) {
  222. for (UserRoleRelation role : relations) {
  223. if (role.getRoleId() == RoleEnum.PARENT.getCode()) {
  224. return RoleEnum.PARENT.getCode();
  225. }
  226. if (role.getRoleId() == RoleEnum.TEACHER.getCode()) {
  227. return RoleEnum.TEACHER.getCode();
  228. }
  229. if (role.getRoleId() == RoleEnum.STUDENT.getCode()) {
  230. return RoleEnum.STUDENT.getCode();
  231. }
  232. if (role.getRoleId() == RoleEnum.ADMIN.getCode()) {
  233. return RoleEnum.ADMIN.getCode();
  234. }
  235. }
  236. return 0L;
  237. }
  238. }