AclService.java 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. package com.nokia.service;
  2. import com.alibaba.fastjson2.JSON;
  3. import com.nokia.common.R;
  4. import com.nokia.constants.ClientEnum;
  5. import com.nokia.dao.UserDao;
  6. import com.nokia.dao.VerificationLogDao;
  7. import com.nokia.pojo.User;
  8. import com.nokia.pojo.VerificationLog;
  9. import com.nokia.vo.*;
  10. import lombok.extern.slf4j.Slf4j;
  11. import org.springframework.beans.factory.annotation.Autowired;
  12. import org.springframework.beans.factory.annotation.Value;
  13. import org.springframework.data.redis.core.RedisTemplate;
  14. import org.springframework.stereotype.Service;
  15. import org.springframework.transaction.annotation.Transactional;
  16. import java.time.LocalDateTime;
  17. import java.util.HashMap;
  18. import java.util.List;
  19. import java.util.Map;
  20. import java.util.concurrent.TimeUnit;
  21. @Slf4j
  22. @Service
  23. public class AclService {
  24. @Value("${redis.timeoutSeconds:600}")
  25. private Integer timeoutSeconds;
  26. /*
  27. * 重定向的url,解决当token失效时的重定向问题
  28. */
  29. @Value("${dop.redirect.url}")
  30. private String dopRedirectUrl;
  31. @Value("${top.redirect.url}")
  32. private String topRedirectUrl;
  33. private final UserDao userDao;
  34. private final TopService topService;
  35. private final DopService dopService;
  36. private final WoyunweiService woyunweiService;
  37. private final RedisTemplate<String, Object> redisTemplate;
  38. private final VerificationLogDao verificationLogDao;
  39. @Autowired
  40. public AclService(UserDao userDao, TopService topService, DopService dopService, WoyunweiService woyunweiService,
  41. RedisTemplate<String, Object> redisTemplate, VerificationLogDao verificationLogDao) {
  42. this.userDao = userDao;
  43. this.topService = topService;
  44. this.dopService = dopService;
  45. this.woyunweiService = woyunweiService;
  46. this.redisTemplate = redisTemplate;
  47. this.verificationLogDao = verificationLogDao;
  48. }
  49. @Transactional(rollbackFor = Exception.class)
  50. public R verifySystem(TokenVo tokenEntity) {
  51. Map<String, String> map = new HashMap<>();
  52. if (tokenEntity.getFromSystem().equalsIgnoreCase("top")) {
  53. map.put("redirect", topRedirectUrl);
  54. } else if (tokenEntity.getFromSystem().equalsIgnoreCase("dop")) {
  55. map.put("redirect", dopRedirectUrl);
  56. } else {
  57. // 根据接入的系统不同,返回不同的重定向url
  58. switch (tokenEntity.getSystem().trim().toLowerCase()) {
  59. case "daping":
  60. map.put("redirect", dopRedirectUrl);
  61. break;
  62. // case "liucheng":
  63. // case "fenxi":
  64. // case "yuce":
  65. // case "heidianku":
  66. default:
  67. map.put("redirect", topRedirectUrl);
  68. break;
  69. }
  70. }
  71. String token = tokenEntity.getToken();
  72. String fromSystem = tokenEntity.getFromSystem();
  73. String system = tokenEntity.getSystem();
  74. // 为不同system生成不同key
  75. String key = system + ":" + token;
  76. User userEntity;
  77. // 从redis中查询
  78. userEntity = (User) redisTemplate.opsForValue().get(key);
  79. if (userEntity != null) {
  80. log.debug("redis查询成功: {}", userEntity);
  81. // 2.1 可以查到用户信息,重置redis中的用户信息,刷新到期时间
  82. redisTemplate.opsForValue().set(token, userEntity, timeoutSeconds, TimeUnit.SECONDS);
  83. return res(tokenEntity, userEntity, map);
  84. }
  85. // 2.2 无法查到用户信息 可能是已超timeoutSeconds或者是初次登录
  86. // 需要根据fromSystem参数去对应的系统进行鉴权
  87. TokenFlagVo tokenFlagVo = getVerifyResult(token, fromSystem);
  88. if (tokenFlagVo == null || !tokenFlagVo.isValid()) {
  89. // token无效,返回重定向的登陆地址
  90. return R.error().data(map).message("用户不存在");
  91. }
  92. // 如果token状态为可用,根据loginName从数据库查询用户
  93. // 这里存在一个问题,如果是在top已存在的用户,但是我们系统没有,此时返回的userEntity为null,会对后续处理产生影响,需要考虑应该怎么做
  94. // 在前端处理,上面的问题仅影响初次点击入口的用户,不影响超时用户
  95. // 前端需要对入口用户做特别的处理
  96. userEntity = userDao.getByLoginName(tokenFlagVo.getLoginName());
  97. if (userEntity == null) {
  98. return R.error().data(map).message("用户不存在");
  99. }
  100. // 仅当用户信息不为空时,将用户信息存入redis
  101. redisTemplate.opsForValue().set(key, userEntity, timeoutSeconds, TimeUnit.SECONDS);
  102. R r = res(tokenEntity, userEntity, map);
  103. Integer success = Boolean.TRUE.equals(r.getSuccess()) ? 1 : 0;
  104. String res = JSON.toJSONString(r);
  105. // redis保存新token信息,记录登录日志
  106. logVerification(tokenEntity, userEntity, success, res);
  107. log.info("{}登录{}", userEntity.getLoginName(), tokenEntity.getSystem());
  108. return r;
  109. }
  110. private R res(TokenVo tokenEntity, User userEntity, Map<String, String> map) {
  111. switch (tokenEntity.getSystem().trim().toLowerCase()) {
  112. case "liucheng":
  113. return R.ok().data(new LiuchengUserVo(userEntity));
  114. case "fenxi":
  115. FenxiUserVo fenxiUserVo = new FenxiUserVo(userEntity);
  116. if (fenxiUserVo.getCity() == null) {
  117. return R.error().data(map).message("用户不存在");
  118. }
  119. return R.ok().data(fenxiUserVo);
  120. case "daping":
  121. DapingUserVo dapingUserVo = new DapingUserVo(userEntity);
  122. if (dapingUserVo.getCity() == null) {
  123. return R.error().data(map).message("用户不存在");
  124. }
  125. return R.ok().data(dapingUserVo);
  126. case "yuce":
  127. YuceUserVo yuceUserVo = new YuceUserVo(userEntity);
  128. if (yuceUserVo.getCity() == null) {
  129. return R.error().data(map).message("用户不存在");
  130. }
  131. return R.ok().data(yuceUserVo);
  132. case "heidianku": // 黑点库
  133. HeidiankuUserVo heidiankuUserVo = new HeidiankuUserVo(userEntity);
  134. if (heidiankuUserVo.getCity() == null) {
  135. return R.error().data(map).message("用户不存在");
  136. }
  137. return R.ok().data(heidiankuUserVo);
  138. case "volte_t2": // volte分析T2页面
  139. VolteT2UserVo userVo = new VolteT2UserVo(userEntity);
  140. return R.ok().data(userVo);
  141. case "luyin":
  142. LvYinVo lvYinVo = new LvYinVo(userEntity);
  143. return R.ok().data(lvYinVo);
  144. case "app":
  145. List<AppVerificationVO> vo = listUserFunctions(tokenEntity, userEntity.getUserId(),
  146. ClientEnum.APP.value);
  147. return R.ok().data(vo);
  148. default:
  149. return R.error().message("当前允许的system为: liucheng/fenxi/daping/yuce/heidianku/volte_t2/luyin/app");
  150. }
  151. }
  152. private List<AppVerificationVO> listUserFunctions(TokenVo tokenEntity, Integer userId, Integer client) {
  153. List<AppVerificationVO> list = userDao.listUserFunctions(userId, client);
  154. for (AppVerificationVO t : list) {
  155. t.setUrl(t.getUrl() + "&fromSystem=" + tokenEntity.getFromSystem() + "&token=" + tokenEntity.getToken());
  156. }
  157. return list;
  158. }
  159. private void logVerification(TokenVo tokenEntity, User userEntity, Integer success, String res) {
  160. VerificationLog verificationLog = new VerificationLog();
  161. verificationLog.setUserId(userEntity.getUserId());
  162. verificationLog.setLoginName(userEntity.getLoginName());
  163. verificationLog.setUserName(userEntity.getUserName());
  164. verificationLog.setCityId(userEntity.getCityId());
  165. verificationLog.setSystem(tokenEntity.getSystem());
  166. verificationLog.setLoginTime(LocalDateTime.now());
  167. verificationLog.setSuccess(success);
  168. verificationLog.setRes(res);
  169. verificationLogDao.insert(verificationLog);
  170. }
  171. private TokenFlagVo getVerifyResult(String token, String fromSystem) {
  172. if (fromSystem.trim().equalsIgnoreCase("dop")) {
  173. return dopService.dopReturn(token);
  174. } else if (fromSystem.trim().equalsIgnoreCase("top")) {
  175. return topService.topReturn(token);
  176. } else if (fromSystem.trim().equalsIgnoreCase("woyunwei")) {
  177. return woyunweiService.woyunweiReturn(token);
  178. } else if (fromSystem.trim().equalsIgnoreCase("test")) {
  179. // 测试用
  180. return testToken(token);
  181. }
  182. return null;
  183. }
  184. private TokenFlagVo testToken(String token) {
  185. String login_name = token.trim().toLowerCase();
  186. login_name = login_name.replace("test_token_", "test_");
  187. return new TokenFlagVo(0, login_name);
  188. }
  189. }