SysLoginService.java 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459
  1. package org.dromara.web.service;
  2. import cn.dev33.satoken.exception.NotLoginException;
  3. import cn.dev33.satoken.secure.BCrypt;
  4. import cn.dev33.satoken.stp.StpUtil;
  5. import cn.hutool.core.bean.BeanUtil;
  6. import cn.hutool.core.util.ObjectUtil;
  7. import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
  8. import jakarta.servlet.http.HttpServletRequest;
  9. import lombok.RequiredArgsConstructor;
  10. import lombok.extern.slf4j.Slf4j;
  11. import me.zhyd.oauth.model.AuthResponse;
  12. import me.zhyd.oauth.model.AuthUser;
  13. import org.dromara.common.core.constant.Constants;
  14. import org.dromara.common.core.constant.GlobalConstants;
  15. import org.dromara.common.core.constant.TenantConstants;
  16. import org.dromara.common.core.domain.R;
  17. import org.dromara.common.core.domain.dto.RoleDTO;
  18. import org.dromara.common.core.domain.model.LoginUser;
  19. import org.dromara.common.core.domain.model.XcxLoginUser;
  20. import org.dromara.common.core.enums.DeviceType;
  21. import org.dromara.common.core.enums.LoginType;
  22. import org.dromara.common.core.enums.TenantStatus;
  23. import org.dromara.common.core.enums.UserStatus;
  24. import org.dromara.common.core.exception.user.CaptchaException;
  25. import org.dromara.common.core.exception.user.CaptchaExpireException;
  26. import org.dromara.common.core.exception.user.UserException;
  27. import org.dromara.common.core.utils.*;
  28. import org.dromara.common.log.event.LogininforEvent;
  29. import org.dromara.common.redis.utils.RedisUtils;
  30. import org.dromara.common.satoken.utils.LoginHelper;
  31. import org.dromara.common.tenant.exception.TenantException;
  32. import org.dromara.common.tenant.helper.TenantHelper;
  33. import org.dromara.common.web.config.properties.CaptchaProperties;
  34. import org.dromara.system.domain.SysUser;
  35. import org.dromara.system.domain.bo.SocialUserBo;
  36. import org.dromara.system.domain.vo.SocialUserVo;
  37. import org.dromara.system.domain.vo.SysTenantVo;
  38. import org.dromara.system.domain.vo.SysUserVo;
  39. import org.dromara.system.mapper.SysUserMapper;
  40. import org.dromara.system.service.ISocialUserService;
  41. import org.dromara.system.service.ISysPermissionService;
  42. import org.dromara.system.service.ISysTenantService;
  43. import org.dromara.system.service.ISysUserService;
  44. import org.springframework.beans.BeanUtils;
  45. import org.springframework.beans.factory.annotation.Value;
  46. import org.springframework.stereotype.Service;
  47. import java.io.IOException;
  48. import java.time.Duration;
  49. import java.util.Date;
  50. import java.util.List;
  51. import java.util.function.Supplier;
  52. /**
  53. * 登录校验方法
  54. *
  55. * @author Lion Li
  56. */
  57. @RequiredArgsConstructor
  58. @Slf4j
  59. @Service
  60. public class SysLoginService {
  61. private final SysUserMapper userMapper;
  62. private final ISocialUserService socialUserService;
  63. private final CaptchaProperties captchaProperties;
  64. private final ISysPermissionService permissionService;
  65. private final ISysTenantService tenantService;
  66. @Value("${user.password.maxRetryCount}")
  67. private Integer maxRetryCount;
  68. @Value("${user.password.lockTime}")
  69. private Integer lockTime;
  70. /**
  71. * 登录验证
  72. *
  73. * @param username 用户名
  74. * @param password 密码
  75. * @param code 验证码
  76. * @param uuid 唯一标识
  77. * @return 结果
  78. */
  79. public String login(String tenantId, String username, String password, String code, String uuid) {
  80. boolean captchaEnabled = captchaProperties.getEnable();
  81. // 验证码开关
  82. if (captchaEnabled) {
  83. validateCaptcha(tenantId, username, code, uuid);
  84. }
  85. // 校验租户
  86. checkTenant(tenantId);
  87. // 框架登录不限制从什么表查询 只要最终构建出 LoginUser 即可
  88. SysUserVo user = loadUserByUsername(tenantId, username);
  89. checkLogin(LoginType.PASSWORD, tenantId, username, () -> !BCrypt.checkpw(password, user.getPassword()));
  90. // 此处可根据登录用户的数据不同 自行创建 loginUser 属性不够用继承扩展就行了
  91. LoginUser loginUser = buildLoginUser(user);
  92. // 生成token
  93. LoginHelper.loginByDevice(loginUser, DeviceType.PC);
  94. recordLogininfor(loginUser.getTenantId(), username, Constants.LOGIN_SUCCESS, MessageUtils.message("user.login.success"));
  95. recordLoginInfo(user.getUserId());
  96. return StpUtil.getTokenValue();
  97. }
  98. public String smsLogin(String tenantId, String phonenumber, String smsCode) {
  99. // 校验租户
  100. checkTenant(tenantId);
  101. // 通过手机号查找用户
  102. SysUserVo user = loadUserByPhonenumber(tenantId, phonenumber);
  103. checkLogin(LoginType.SMS, tenantId, user.getUserName(), () -> !validateSmsCode(tenantId, phonenumber, smsCode));
  104. // 此处可根据登录用户的数据不同 自行创建 loginUser 属性不够用继承扩展就行了
  105. LoginUser loginUser = buildLoginUser(user);
  106. // 生成token
  107. LoginHelper.loginByDevice(loginUser, DeviceType.APP);
  108. recordLogininfor(loginUser.getTenantId(), user.getUserName(), Constants.LOGIN_SUCCESS, MessageUtils.message("user.login.success"));
  109. recordLoginInfo(user.getUserId());
  110. return StpUtil.getTokenValue();
  111. }
  112. public String emailLogin(String tenantId, String email, String emailCode) {
  113. // 校验租户
  114. checkTenant(tenantId);
  115. // 通过邮箱查找用户
  116. SysUserVo user = loadUserByEmail(tenantId, email);
  117. checkLogin(LoginType.EMAIL, tenantId, user.getUserName(), () -> !validateEmailCode(tenantId, email, emailCode));
  118. // 此处可根据登录用户的数据不同 自行创建 loginUser 属性不够用继承扩展就行了
  119. LoginUser loginUser = buildLoginUser(user);
  120. // 生成token
  121. LoginHelper.loginByDevice(loginUser, DeviceType.APP);
  122. recordLogininfor(loginUser.getTenantId(), user.getUserName(), Constants.LOGIN_SUCCESS, MessageUtils.message("user.login.success"));
  123. recordLoginInfo(user.getUserId());
  124. return StpUtil.getTokenValue();
  125. }
  126. public String xcxLogin(String xcxCode) {
  127. // xcxCode 为 小程序调用 wx.login 授权后获取
  128. // todo 以下自行实现
  129. // 校验 appid + appsrcret + xcxCode 调用登录凭证校验接口 获取 session_key 与 openid
  130. String openid = "";
  131. // 框架登录不限制从什么表查询 只要最终构建出 LoginUser 即可
  132. SysUserVo user = loadUserByOpenid(openid);
  133. // 校验租户
  134. checkTenant(user.getTenantId());
  135. // 此处可根据登录用户的数据不同 自行创建 loginUser 属性不够用继承扩展就行了
  136. XcxLoginUser loginUser = new XcxLoginUser();
  137. loginUser.setTenantId(user.getTenantId());
  138. loginUser.setUserId(user.getUserId());
  139. loginUser.setUsername(user.getUserName());
  140. loginUser.setUserType(user.getUserType());
  141. loginUser.setOpenid(openid);
  142. // 生成token
  143. LoginHelper.loginByDevice(loginUser, DeviceType.XCX);
  144. recordLogininfor(loginUser.getTenantId(), user.getUserName(), Constants.LOGIN_SUCCESS, MessageUtils.message("user.login.success"));
  145. recordLoginInfo(user.getUserId());
  146. return StpUtil.getTokenValue();
  147. }
  148. /**
  149. * 社交登录
  150. *
  151. * @param source 登录来源
  152. * @param authUser 授权响应实体
  153. * @param request Http请求对象
  154. * @return 统一响应实体
  155. */
  156. public R<String> socialLogin(String source, AuthResponse<AuthUser> authUser, HttpServletRequest request) {
  157. // 判断授权响应是否成功
  158. if (!authUser.ok()) {
  159. return R.fail("对不起,授权信息验证不通过,请退出重试!");
  160. }
  161. AuthUser authUserData = authUser.getData();
  162. SocialUserVo user = socialUserService.selectSocialUserByAuthId(authUserData.getSource() + authUserData.getUuid());
  163. if (ObjectUtil.isNotNull(user)) {
  164. //执行登录和记录登录信息操作
  165. return loginAndRecord(user.getTenantId(), user.getUserName(), authUserData);
  166. } else {
  167. // 判断是否已登录
  168. if (LoginHelper.getUserId() == null) {
  169. return R.fail("授权失败,请先登录才能绑定");
  170. }
  171. SocialUserBo socialUserBo = new SocialUserBo();
  172. socialUserBo.setUserId(LoginHelper.getUserId());
  173. socialUserBo.setAuthId(authUserData.getSource() + authUserData.getUuid());
  174. socialUserBo.setSource(authUserData.getSource());
  175. socialUserBo.setUserName(authUserData.getUsername());
  176. socialUserBo.setNickName(authUserData.getNickname());
  177. socialUserBo.setAvatar(authUserData.getAvatar());
  178. socialUserBo.setOpenId(authUserData.getUuid());
  179. BeanUtils.copyProperties(authUserData.getToken(), socialUserBo);
  180. socialUserService.insertByBo(socialUserBo);
  181. SysUserVo lodingData = loadUserByUsername(LoginHelper.getTenantId(), LoginHelper.getUsername());
  182. //执行登录和记录登录信息操作
  183. return loginAndRecord(lodingData.getTenantId(), socialUserBo.getUserName(), authUserData);
  184. }
  185. }
  186. /**
  187. * 执行登录和记录登录信息操作
  188. *
  189. * @param tenantId 租户ID
  190. * @param userName 用户名
  191. * @param authUser 授权用户信息
  192. * @return 统一响应实体
  193. */
  194. private R<String> loginAndRecord(String tenantId, String userName, AuthUser authUser) {
  195. checkTenant(tenantId);
  196. SysUserVo dbUser = loadUserByUsername(tenantId, userName);
  197. LoginHelper.loginByDevice(buildLoginUser(dbUser), DeviceType.SOCIAL);
  198. recordLogininfor(dbUser.getTenantId(), userName, Constants.LOGIN_SUCCESS, MessageUtils.message("user.login.success"));
  199. recordLoginInfo(dbUser.getUserId());
  200. return R.ok(StpUtil.getTokenValue());
  201. }
  202. /**
  203. * 退出登录
  204. */
  205. public void logout() {
  206. try {
  207. LoginUser loginUser = LoginHelper.getLoginUser();
  208. if (TenantHelper.isEnable() && LoginHelper.isSuperAdmin()) {
  209. // 超级管理员 登出清除动态租户
  210. TenantHelper.clearDynamic();
  211. }
  212. StpUtil.logout();
  213. recordLogininfor(loginUser.getTenantId(), loginUser.getUsername(), Constants.LOGOUT, MessageUtils.message("user.logout.success"));
  214. } catch (NotLoginException ignored) {
  215. }
  216. }
  217. /**
  218. * 记录登录信息
  219. *
  220. * @param tenantId 租户ID
  221. * @param username 用户名
  222. * @param status 状态
  223. * @param message 消息内容
  224. */
  225. private void recordLogininfor(String tenantId, String username, String status, String message) {
  226. LogininforEvent logininforEvent = new LogininforEvent();
  227. logininforEvent.setTenantId(tenantId);
  228. logininforEvent.setUsername(username);
  229. logininforEvent.setStatus(status);
  230. logininforEvent.setMessage(message);
  231. logininforEvent.setRequest(ServletUtils.getRequest());
  232. SpringUtils.context().publishEvent(logininforEvent);
  233. }
  234. /**
  235. * 校验短信验证码
  236. */
  237. private boolean validateSmsCode(String tenantId, String phonenumber, String smsCode) {
  238. String code = RedisUtils.getCacheObject(GlobalConstants.CAPTCHA_CODE_KEY + phonenumber);
  239. if (StringUtils.isBlank(code)) {
  240. recordLogininfor(tenantId, phonenumber, Constants.LOGIN_FAIL, MessageUtils.message("user.jcaptcha.expire"));
  241. throw new CaptchaExpireException();
  242. }
  243. return code.equals(smsCode);
  244. }
  245. /**
  246. * 校验邮箱验证码
  247. */
  248. private boolean validateEmailCode(String tenantId, String email, String emailCode) {
  249. String code = RedisUtils.getCacheObject(GlobalConstants.CAPTCHA_CODE_KEY + email);
  250. if (StringUtils.isBlank(code)) {
  251. recordLogininfor(tenantId, email, Constants.LOGIN_FAIL, MessageUtils.message("user.jcaptcha.expire"));
  252. throw new CaptchaExpireException();
  253. }
  254. return code.equals(emailCode);
  255. }
  256. /**
  257. * 校验验证码
  258. *
  259. * @param username 用户名
  260. * @param code 验证码
  261. * @param uuid 唯一标识
  262. */
  263. public void validateCaptcha(String tenantId, String username, String code, String uuid) {
  264. String verifyKey = GlobalConstants.CAPTCHA_CODE_KEY + StringUtils.defaultString(uuid, "");
  265. String captcha = RedisUtils.getCacheObject(verifyKey);
  266. RedisUtils.deleteObject(verifyKey);
  267. if (captcha == null) {
  268. recordLogininfor(tenantId, username, Constants.LOGIN_FAIL, MessageUtils.message("user.jcaptcha.expire"));
  269. throw new CaptchaExpireException();
  270. }
  271. if (!code.equalsIgnoreCase(captcha)) {
  272. recordLogininfor(tenantId, username, Constants.LOGIN_FAIL, MessageUtils.message("user.jcaptcha.error"));
  273. throw new CaptchaException();
  274. }
  275. }
  276. private SysUserVo loadUserByUsername(String tenantId, String username) {
  277. SysUser user = userMapper.selectOne(new LambdaQueryWrapper<SysUser>()
  278. .select(SysUser::getUserName, SysUser::getStatus)
  279. .eq(TenantHelper.isEnable(), SysUser::getTenantId, tenantId)
  280. .eq(SysUser::getUserName, username));
  281. if (ObjectUtil.isNull(user)) {
  282. log.info("登录用户:{} 不存在.", username);
  283. throw new UserException("user.not.exists", username);
  284. } else if (UserStatus.DISABLE.getCode().equals(user.getStatus())) {
  285. log.info("登录用户:{} 已被停用.", username);
  286. throw new UserException("user.blocked", username);
  287. }
  288. if (TenantHelper.isEnable()) {
  289. return userMapper.selectTenantUserByUserName(username, tenantId);
  290. }
  291. return userMapper.selectUserByUserName(username);
  292. }
  293. private SysUserVo loadUserByPhonenumber(String tenantId, String phonenumber) {
  294. SysUser user = userMapper.selectOne(new LambdaQueryWrapper<SysUser>()
  295. .select(SysUser::getPhonenumber, SysUser::getStatus)
  296. .eq(TenantHelper.isEnable(), SysUser::getTenantId, tenantId)
  297. .eq(SysUser::getPhonenumber, phonenumber));
  298. if (ObjectUtil.isNull(user)) {
  299. log.info("登录用户:{} 不存在.", phonenumber);
  300. throw new UserException("user.not.exists", phonenumber);
  301. } else if (UserStatus.DISABLE.getCode().equals(user.getStatus())) {
  302. log.info("登录用户:{} 已被停用.", phonenumber);
  303. throw new UserException("user.blocked", phonenumber);
  304. }
  305. if (TenantHelper.isEnable()) {
  306. return userMapper.selectTenantUserByPhonenumber(phonenumber, tenantId);
  307. }
  308. return userMapper.selectUserByPhonenumber(phonenumber);
  309. }
  310. private SysUserVo loadUserByEmail(String tenantId, String email) {
  311. SysUser user = userMapper.selectOne(new LambdaQueryWrapper<SysUser>()
  312. .select(SysUser::getEmail, SysUser::getStatus)
  313. .eq(TenantHelper.isEnable(), SysUser::getTenantId, tenantId)
  314. .eq(SysUser::getEmail, email));
  315. if (ObjectUtil.isNull(user)) {
  316. log.info("登录用户:{} 不存在.", email);
  317. throw new UserException("user.not.exists", email);
  318. } else if (UserStatus.DISABLE.getCode().equals(user.getStatus())) {
  319. log.info("登录用户:{} 已被停用.", email);
  320. throw new UserException("user.blocked", email);
  321. }
  322. if (TenantHelper.isEnable()) {
  323. return userMapper.selectTenantUserByEmail(email, tenantId);
  324. }
  325. return userMapper.selectUserByEmail(email);
  326. }
  327. private SysUserVo loadUserByOpenid(String openid) {
  328. // 使用 openid 查询绑定用户 如未绑定用户 则根据业务自行处理 例如 创建默认用户
  329. // todo 自行实现 userService.selectUserByOpenid(openid);
  330. SysUserVo user = new SysUserVo();
  331. if (ObjectUtil.isNull(user)) {
  332. log.info("登录用户:{} 不存在.", openid);
  333. // todo 用户不存在 业务逻辑自行实现
  334. } else if (UserStatus.DISABLE.getCode().equals(user.getStatus())) {
  335. log.info("登录用户:{} 已被停用.", openid);
  336. // todo 用户已被停用 业务逻辑自行实现
  337. }
  338. return user;
  339. }
  340. /**
  341. * 构建登录用户
  342. */
  343. private LoginUser buildLoginUser(SysUserVo user) {
  344. LoginUser loginUser = new LoginUser();
  345. loginUser.setTenantId(user.getTenantId());
  346. loginUser.setUserId(user.getUserId());
  347. loginUser.setDeptId(user.getDeptId());
  348. loginUser.setUsername(user.getUserName());
  349. loginUser.setUserType(user.getUserType());
  350. loginUser.setMenuPermission(permissionService.getMenuPermission(user.getUserId()));
  351. loginUser.setRolePermission(permissionService.getRolePermission(user.getUserId()));
  352. loginUser.setDeptName(ObjectUtil.isNull(user.getDept()) ? "" : user.getDept().getDeptName());
  353. List<RoleDTO> roles = BeanUtil.copyToList(user.getRoles(), RoleDTO.class);
  354. loginUser.setRoles(roles);
  355. return loginUser;
  356. }
  357. /**
  358. * 记录登录信息
  359. *
  360. * @param userId 用户ID
  361. */
  362. public void recordLoginInfo(Long userId) {
  363. SysUser sysUser = new SysUser();
  364. sysUser.setUserId(userId);
  365. sysUser.setLoginIp(ServletUtils.getClientIP());
  366. sysUser.setLoginDate(DateUtils.getNowDate());
  367. sysUser.setUpdateBy(userId);
  368. userMapper.updateById(sysUser);
  369. }
  370. /**
  371. * 登录校验
  372. */
  373. private void checkLogin(LoginType loginType, String tenantId, String username, Supplier<Boolean> supplier) {
  374. String errorKey = GlobalConstants.PWD_ERR_CNT_KEY + username;
  375. String loginFail = Constants.LOGIN_FAIL;
  376. // 获取用户登录错误次数,默认为0 (可自定义限制策略 例如: key + username + ip)
  377. int errorNumber = ObjectUtil.defaultIfNull(RedisUtils.getCacheObject(errorKey), 0);
  378. // 锁定时间内登录 则踢出
  379. if (errorNumber >= maxRetryCount) {
  380. recordLogininfor(tenantId, username, loginFail, MessageUtils.message(loginType.getRetryLimitExceed(), maxRetryCount, lockTime));
  381. throw new UserException(loginType.getRetryLimitExceed(), maxRetryCount, lockTime);
  382. }
  383. if (supplier.get()) {
  384. // 错误次数递增
  385. errorNumber++;
  386. RedisUtils.setCacheObject(errorKey, errorNumber, Duration.ofMinutes(lockTime));
  387. // 达到规定错误次数 则锁定登录
  388. if (errorNumber >= maxRetryCount) {
  389. recordLogininfor(tenantId, username, loginFail, MessageUtils.message(loginType.getRetryLimitExceed(), maxRetryCount, lockTime));
  390. throw new UserException(loginType.getRetryLimitExceed(), maxRetryCount, lockTime);
  391. } else {
  392. // 未达到规定错误次数
  393. recordLogininfor(tenantId, username, loginFail, MessageUtils.message(loginType.getRetryLimitCount(), errorNumber));
  394. throw new UserException(loginType.getRetryLimitCount(), errorNumber);
  395. }
  396. }
  397. // 登录成功 清空错误次数
  398. RedisUtils.deleteObject(errorKey);
  399. }
  400. private void checkTenant(String tenantId) {
  401. if (!TenantHelper.isEnable()) {
  402. return;
  403. }
  404. if (TenantConstants.DEFAULT_TENANT_ID.equals(tenantId)) {
  405. return;
  406. }
  407. SysTenantVo tenant = tenantService.queryByTenantId(tenantId);
  408. if (ObjectUtil.isNull(tenant)) {
  409. log.info("登录租户:{} 不存在.", tenantId);
  410. throw new TenantException("tenant.not.exists");
  411. } else if (TenantStatus.DISABLE.getCode().equals(tenant.getStatus())) {
  412. log.info("登录租户:{} 已被停用.", tenantId);
  413. throw new TenantException("tenant.blocked");
  414. } else if (ObjectUtil.isNotNull(tenant.getExpireTime())
  415. && new Date().after(tenant.getExpireTime())) {
  416. log.info("登录租户:{} 已超过有效期.", tenantId);
  417. throw new TenantException("tenant.expired");
  418. }
  419. }
  420. }