LoginServiceImpl.java 12 KB

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