LoginController.java 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607
  1. package com.uas.sso.controller;
  2. import com.alibaba.fastjson.JSON;
  3. import com.alibaba.fastjson.JSONObject;
  4. import com.uas.account.AccountConfig;
  5. import com.uas.sso.*;
  6. import com.uas.sso.common.util.HttpUtil;
  7. import com.uas.sso.core.Const;
  8. import com.uas.sso.entity.*;
  9. import com.uas.sso.entity.Token;
  10. import com.uas.sso.service.*;
  11. import com.uas.sso.util.AccountTypeUtils;
  12. import com.uas.sso.util.BeanUtil;
  13. import com.uas.sso.util.CaptchaUtil;
  14. import com.uas.sso.util.encry.HmacUtils;
  15. import com.uas.sso.web.waf.request.WafRequestWrapper;
  16. import org.springframework.beans.factory.annotation.Autowired;
  17. import org.springframework.ui.ModelMap;
  18. import org.springframework.util.CollectionUtils;
  19. import org.springframework.util.StringUtils;
  20. import org.springframework.web.bind.annotation.*;
  21. import org.springframework.web.servlet.ModelAndView;
  22. import java.io.IOException;
  23. import java.util.*;
  24. /**
  25. * 登录controller
  26. *
  27. * @author wangmh
  28. * @date 2018/1/5
  29. */
  30. @RequestMapping("/sso/login")
  31. @RestController
  32. public class LoginController extends BaseController {
  33. /**
  34. * 密码输错3次
  35. */
  36. private static final int PWD_ERROR_FIVE_TIME = 5;
  37. /**
  38. * 密码输错3次
  39. */
  40. private static final int PWD_ERROR_THREE_TIME = 3;
  41. /**
  42. * 登录验证码存session的key值
  43. */
  44. private static final String LOGIN_CAPTCHA = "loginCaptcha";
  45. /**
  46. * 优软云首页地址
  47. */
  48. private static final String HOME_PAGE = "https://www.ubtob.com/";
  49. @Autowired
  50. private AppService appService;
  51. @Autowired
  52. private UserService userService;
  53. @Autowired
  54. private UserspaceService userspaceService;
  55. @Autowired
  56. private UserAccountService userAccountService;
  57. @Autowired
  58. private PersonalAccountService personalAccountService;
  59. @RequestMapping(method = RequestMethod.POST)
  60. public ModelMap login() {
  61. // 获取登录信息
  62. WafRequestWrapper wr = new WafRequestWrapper(request);
  63. String appId = wr.getParameter("appId");
  64. String spaceUU = wr.getParameter("spaceUU");
  65. String username = StringUtils.trimAllWhitespace(wr.getParameter("username"));
  66. String password = wr.getParameter("password");
  67. String captcha = wr.getParameter("captcha");
  68. String returnUrl = wr.getParameter("returnUrl");
  69. String baseUrl = wr.getParameter("baseUrl");
  70. // 校验空参数
  71. if (StringUtils.isEmpty(username)) {
  72. return error("用户名不能为空");
  73. }
  74. if (StringUtils.isEmpty(password)) {
  75. return error("密码不能为空");
  76. }
  77. // 设置baseUrl
  78. if (!StringUtils.isEmpty(baseUrl)) {
  79. request.getSession().setAttribute("baseUrl", baseUrl);
  80. }
  81. // 校验appId,appId为空的话默认为sso
  82. appId = StringUtils.isEmpty(appId) ? SSOHelper.getSSOService().getConfig().getAppName() : appId;
  83. App app = appService.findOne(appId);
  84. if (app == null) {
  85. return error("应用id不存在");
  86. }
  87. // 校验returnUrl,为空默认为优软云
  88. if (StringUtils.isEmpty(returnUrl)) {
  89. returnUrl = HOME_PAGE;
  90. }
  91. // 获取用户基本信息
  92. User user = userService.findByUsername(username);
  93. if (user == null) {
  94. return error("用户名或密码错误");
  95. }
  96. // 校验账户密码输错次数
  97. int pwdErrorCount = user.getUserRecord() == null ? 0 : user.getUserRecord().getPwdErrorCount();
  98. Object loginCaptcha = request.getSession().getAttribute(LOGIN_CAPTCHA);
  99. String checkCode = loginCaptcha == null ? "" : loginCaptcha.toString();
  100. if (pwdErrorCount >= PWD_ERROR_FIVE_TIME) {
  101. return error("403", "密码错误次数已达上限,今日无法登陆");
  102. }
  103. // 校验账号是否被锁定,5次输错密码
  104. if (pwdErrorCount >= PWD_ERROR_THREE_TIME && StringUtils.isEmpty(captcha)) {
  105. return error("404", "验证码不能为空");
  106. }
  107. if (pwdErrorCount >= PWD_ERROR_THREE_TIME && !captcha.equalsIgnoreCase(checkCode)) {
  108. return error("验证码错误");
  109. }
  110. // 校验密码
  111. String encryPwd = userService.getEncryPassword(Const.ENCRY_FORMAT, password, user.getSalt());
  112. if (!encryPwd.equals(user.getPassword())) {
  113. pwdErrorCount = inputErrorPwd(user.getUserRecord());
  114. return error("您输入的账号或密码有误").addAttribute("errorCount", pwdErrorCount);
  115. }
  116. // 校验密码强度,如果和存储的不同,则保存
  117. int strength = checkPasswordLevel(password).getValue();
  118. if (strength != user.getPasswordLevel()) {
  119. user.setPasswordLevel(strength);
  120. userService.save(user);
  121. }
  122. // 登录
  123. App controlApp = StringUtils.isEmpty(app.getUserControl()) ? app : appService.findOne(app.getUserControl());
  124. boolean personalEnable = Const.YES == controlApp.getPersonalEnable();
  125. if (StringUtils.isEmpty(spaceUU)) {
  126. /*企业uu号为空,让用户选择企业*/
  127. // 找到用户账号信息
  128. List<UserAccount> userAccounts = userAccountService.findByUserUU(appId, user.getUserUU());
  129. // 没有记录
  130. if (CollectionUtils.isEmpty(userAccounts)) {
  131. // 没有记录如果当前应用允许个人账号的话,查找个人账号
  132. UserAccount userAccount = personalAccountService.findOneByUserUU(appId, user.getUserUU());
  133. if (!personalEnable) {
  134. // 不支持个人账号则跳转优软云
  135. returnUrl = HOME_PAGE;
  136. }
  137. return loginByUser(userAccount, returnUrl, user.getUserRecord());
  138. }
  139. // 应用允许个人账号,并且账号未绑定企业,或者只绑定了一个企业,直接登录
  140. if (userAccounts.size() == 1) {
  141. return loginByUser(userAccounts.get(0), returnUrl, user.getUserRecord());
  142. }
  143. // 返回企业id和名称
  144. return success(getSpaceSelect(userAccounts, personalEnable));
  145. } else if (personalEnable && Long.valueOf(spaceUU).equals(Const.SPACEUU_PERSONAL)) {
  146. // 使用个人账号登录
  147. UserAccount userAccount = personalAccountService.findOneByUserUU(appId, user.getUserUU());
  148. return loginByUser(userAccount, returnUrl, user.getUserRecord());
  149. } else {
  150. // 带企业登录
  151. UserAccount userAccount = getUserAccountByUserName(controlApp.getUid(), username, spaceUU);
  152. return loginByUser(userAccount, returnUrl, user.getUserRecord());
  153. }
  154. }
  155. /**
  156. * 根据企业uu号和用户uu号获取登录token,有效期1分钟
  157. * @param enUU 企业uu号
  158. * @param userUU 用户uu号
  159. * @param timestamp 请求时间,用于判断请求是否有效,有效期10秒
  160. * @param signature 加密信息,使用企业私钥加密请求参数
  161. * @return tokenId,token存入ModelMap对象,值为 {userUU:用户uu号,spaceUU:企业uu号}
  162. */
  163. @RequestMapping(value = "/getToken", method = RequestMethod.GET)
  164. public ModelMap getToken(String enUU, String userUU, @RequestParam(name = "_timestamp") Long timestamp, @RequestParam(name = "_signature") String signature) {
  165. ModelMap map = new ModelMap();
  166. // 根据企业uu号找到企业信息
  167. Userspace userspace = userspaceService.findOne(Long.valueOf(enUU));
  168. if (userspace == null) {
  169. return error("没有找到企业");
  170. }
  171. // 校验请求
  172. checkRequest(userspace.getAccessSecret());
  173. // 将企业信息、企业管理员信息写入SystemSession
  174. User user = userService.findOne(Long.valueOf(userUU));
  175. if (user == null) {
  176. return error("没有找到用户");
  177. }
  178. if (!userspace.getUsers().contains(user)) {
  179. return error("该用户不在当前企业");
  180. }
  181. map.put("spaceUU", enUU);
  182. map.put("userUU", userUU);
  183. // token有效期,单位秒
  184. int expires_in = 1 * 60;
  185. Token token = new Token(map, expires_in);
  186. tokenService.save(token);
  187. return success(token.getId());
  188. }
  189. /**
  190. * token代理页面
  191. * @return
  192. */
  193. @RequestMapping(value = "/proxy", method = RequestMethod.GET)
  194. public ModelAndView loginProxyByToken() {
  195. WafRequestWrapper wr = new WafRequestWrapper(request);
  196. String returnUrl = wr.getParameter("returnURL");
  197. String appId = wr.getParameter("appId");
  198. String token = wr.getParameter("token");
  199. String baseUrl = wr.getParameter("baseURL");
  200. String isLoginAll = wr.getParameter("isLoginAll");
  201. ModelMap data = new ModelMap();
  202. data.put("returnUrl", returnUrl);
  203. data.put("appId", appId);
  204. data.put("token", token);
  205. data.put("baseUrl", baseUrl);
  206. data.put("isLoginAll", isLoginAll == null ? true : isLoginAll);
  207. return new ModelAndView("/sso/proxyByToken", data);
  208. }
  209. /**
  210. * erp和uu互联跳转
  211. * 代理登录,根据tokenId拿到当前用户登录的用户uu号和企业uu号进行登录
  212. *
  213. * @param token tokenId
  214. * @param appId 应用id
  215. * @param returnUrl 跳转页面
  216. * @param baseUrl 而外登录接口
  217. * @param isLoginAll 是否登录默认应用
  218. * @return
  219. */
  220. @RequestMapping(value = "/proxy", method = RequestMethod.POST)
  221. public ModelMap loginProxyByToken(String token, String appId, String returnUrl, String baseUrl, @RequestParam(defaultValue = "true") boolean isLoginAll) {
  222. Token tk = tokenService.findOne(token);
  223. if (null != tk) {
  224. JSONObject data = JSON.parseObject(JSON.toJSONString(tk.getBind()));
  225. Long userUU = data.getLong("userUU");
  226. Long spaceUU = data.getLong("spaceUU");
  227. request.getSession().setAttribute("baseUrl", baseUrl);
  228. App app = appService.findOne(appId);
  229. if (app != null) {
  230. app = StringUtils.isEmpty(app.getUserControl()) ? app : appService.findOne(app.getUserControl());
  231. }
  232. if (app == null) {
  233. app = appService.findOne(AccountConfig.ACCOUNT_CENTER);
  234. }
  235. UserAccount userAccount = userAccountService.findOneByUserUU(app.getUid(), userUU, spaceUU);
  236. return loginByUser(userAccount, returnUrl, null);
  237. }
  238. return error("验证信息已过期");
  239. }
  240. /**
  241. * 密码输错处理
  242. *
  243. * @param userRecord 用户登录记录
  244. * @return
  245. */
  246. private int inputErrorPwd(UserRecord userRecord) {
  247. // 密码输错次数+1
  248. int pwdErrorCount = userRecord.getPwdErrorCount();
  249. userRecord.setPwdErrorCount(++pwdErrorCount);
  250. userService.save(userRecord);
  251. // 设置返回值
  252. return pwdErrorCount;
  253. }
  254. /**
  255. * 根据用户名获得用户账号信息
  256. *
  257. * @param appId 应用
  258. * @param username 用户名
  259. * @return
  260. */
  261. private List<UserAccount> getUserAccountByUserName(String appId, String username) {
  262. String type = AccountTypeUtils.getAccountType(username);
  263. if (AccountTypeUtils.MOBILE.equals(type)) {
  264. // 手机号登录
  265. return userAccountService.findByMobile(appId, username);
  266. }
  267. if (AccountTypeUtils.EMAIL.equals(type)) {
  268. // 邮箱登录
  269. return userAccountService.findByEmail(appId, username);
  270. }
  271. if (AccountTypeUtils.UU_NUMBER.equals(type)) {
  272. // uu号登录
  273. return userAccountService.findByUserUU(appId, Long.valueOf(username));
  274. }
  275. // 其余情况
  276. return null;
  277. }
  278. /**
  279. * 根据用户名和企业uu找到用户信息
  280. *
  281. * @param appId 应用
  282. * @param username 用户名
  283. * @param spaceUU 企业uu号
  284. * @return
  285. */
  286. private UserAccount getUserAccountByUserName(String appId, String username, String spaceUU) {
  287. String type = AccountTypeUtils.getAccountType(username);
  288. if (AccountTypeUtils.MOBILE.equals(type)) {
  289. // 手机号登录
  290. return userAccountService.findOneByMobile(appId, username, Long.valueOf(spaceUU));
  291. }
  292. if (AccountTypeUtils.EMAIL.equals(type)) {
  293. // 邮箱登录
  294. return userAccountService.findOneByEmail(appId, username, Long.valueOf(spaceUU));
  295. }
  296. if (AccountTypeUtils.UU_NUMBER.equals(type)) {
  297. // uu号登录
  298. return userAccountService.findOneByUserUU(appId, Long.valueOf(username), Long.valueOf(spaceUU));
  299. }
  300. // 其余情况
  301. return null;
  302. }
  303. /**
  304. * 用户信息没问题,直接登录
  305. *
  306. * @param userAccount 用户账号信息
  307. * @param returnUrl 跳转url
  308. * @param userRecord 用户登录记录,便于记录登录时间,不能直接new出来
  309. * @return
  310. */
  311. private ModelMap loginByUser(UserAccount userAccount, String returnUrl, UserRecord userRecord) {
  312. /*
  313. * 设置登录 Cookie 最后一个参数 true 时添加 cookie 同时销毁当前 JSESSIONID
  314. * 创建信任的 JSESSIONID
  315. */
  316. SSOToken st = new SSOToken(request, userAccount.getMobile());
  317. st.setData(JSON.toJSONString(userAccount));
  318. SSOHelper.setSSOCookie(request, response, st, true);
  319. // 设置登录时间,并将密码输错次数设为0,为空则不设置
  320. if (userRecord != null) {
  321. userRecord.setLastLoginTime(System.currentTimeMillis());
  322. userRecord.setPwdErrorCount(0);
  323. userService.save(userRecord);
  324. }
  325. // 设置返回值,通知各个应用用户已经登录
  326. ModelMap data = new ModelMap();
  327. data = addOtherAppRequestData(userAccount, data, request.getSession().getAttribute("baseUrl"), true);
  328. data.put("returnUrl", HttpUtil.decodeURL(returnUrl));
  329. return success(data);
  330. }
  331. private ModelMap addOtherAppRequestData(UserAccount userAccount, ModelMap data, Object loginUrl,
  332. boolean isLoginAll) {
  333. List<App> apps = appService.findAll();
  334. List<String> loginUrls = new ArrayList<>();
  335. boolean hasLoginUrl = false;
  336. if (isLoginAll) {
  337. for (App app : apps) {
  338. if (StringUtils.isEmpty(app.getLoginUrl())) {
  339. continue;
  340. }
  341. if (app.getLoginUrl().equals(loginUrl)) {
  342. hasLoginUrl = true;
  343. }
  344. loginUrls.add(app.getLoginUrl());
  345. }
  346. }
  347. // 添加baseUrl
  348. if (!hasLoginUrl && !StringUtils.isEmpty(loginUrl)) {
  349. loginUrls.add(loginUrl.toString());
  350. }
  351. data.put("loginUrls", loginUrls);
  352. // 添加传递数据
  353. JSONObject jsonObject = JSON.parseObject(JSON.toJSONString(userAccount));
  354. Integer maxage = (Integer) request.getAttribute(SSOConfig.SSO_COOKIE_MAXAGE);
  355. jsonObject.put("maxage", maxage);
  356. data.put("data", jsonObject);
  357. return data;
  358. }
  359. /**
  360. * 获取选择企业信息(id:企业uu号,name:名称)
  361. *
  362. * @param userAccounts 用户账户信息
  363. * @param personalEnable 该应用是否允许个人账户
  364. * @return
  365. */
  366. private ModelMap getSpaceSelect(List<UserAccount> userAccounts, boolean personalEnable) {
  367. List<Map<String, Object>> spaces = new ArrayList<Map<String, Object>>();
  368. Map<String, Object> space = null;
  369. // 设置带企业账号
  370. for (UserAccount userAccount : userAccounts) {
  371. space = new HashMap<String, Object>(2);
  372. space.put("id", userAccount.getSpaceUU());
  373. space.put("name", userAccount.getSpaceName());
  374. spaces.add(space);
  375. }
  376. // 设置个人账号
  377. if (personalEnable) {
  378. space = new HashMap<String, Object>(2);
  379. space.put("id", Const.SPACEUU_PERSONAL);
  380. space.put("name", String.format("%s(个人)", userAccounts.get(0).getVipName()));
  381. spaces.add(space);
  382. }
  383. return new ModelMap("spaces", spaces);
  384. }
  385. @RequestMapping(value = "/checkCode", method = RequestMethod.GET)
  386. public void checkCode() {
  387. try {
  388. // 获取验证码
  389. CaptchaUtil.outputCaptcha(request, response, LOGIN_CAPTCHA);
  390. } catch (IOException e) {
  391. e.printStackTrace();
  392. }
  393. }
  394. /**
  395. * 获得密码输错次数
  396. *
  397. * @param username 用户名
  398. * @return
  399. */
  400. @RequestMapping(value = "/getPwdErrorCount", method = RequestMethod.GET)
  401. public ModelMap getPwdErrorCount(String username) {
  402. if (StringUtils.isEmpty(username)) {
  403. return error("用户名不能为空");
  404. }
  405. return success(userService.getPwdErrorCount(username.trim()));
  406. }
  407. /**
  408. * 退出接口,测试接口
  409. * @return
  410. */
  411. @RequestMapping(value = "/logoutAccount", method = RequestMethod.GET)
  412. public ModelMap logoutAccount() {
  413. SSOHelper.clearLogin(request, response);
  414. return success();
  415. }
  416. /**
  417. * 判断是否登录, 测试接口
  418. * @return
  419. */
  420. @RequestMapping(value = "/isLogin", method = RequestMethod.GET)
  421. public ModelMap isLogin() {
  422. SSOToken token = SSOHelper.getToken(request);
  423. return success(new ModelMap("isLogin", token != null));
  424. }
  425. /**
  426. * 页面样式
  427. * @param appId 应用id
  428. * @return
  429. */
  430. @RequestMapping(value = "/page/style", method = RequestMethod.GET)
  431. public ModelMap getPageStyle(@RequestParam(defaultValue = "sso") String appId) {
  432. App app = appService.findOne(appId);
  433. App defaultApp = appService.findOne("sso");
  434. PageStyle pageStyle = app == null ? defaultApp.getPageStyle() : app.getPageStyle();
  435. BeanUtil.copyProperties(pageStyle, defaultApp.getPageStyle(), true);
  436. return success(defaultApp.getPageStyle());
  437. }
  438. /**
  439. * 跨域询问,回复子系统是否登录
  440. *
  441. * @throws IOException
  442. */
  443. @RequestMapping("/login/ask")
  444. @ResponseBody
  445. public void replyAsk() throws IOException {
  446. String callback = request.getParameter("callback");
  447. SSOToken token = SSOHelper.getToken(request);
  448. if (token != null) {
  449. String askData = request.getParameter("askData");
  450. if (!StringUtils.isEmpty(askData)) {
  451. // 下面开始验证票据,签名新的票据每一步都必须有。
  452. AuthToken at = SSOHelper.replyCiphertext(request, askData);
  453. if (at != null) {
  454. App app = appService.findOne(at.getApp());
  455. if (app != null && StringUtils.isEmpty(app.getPublicKey()) && !StringUtils.isEmpty(app.getUserControl())) {
  456. app = appService.findOne(app.getUserControl());
  457. }
  458. if (app == null) {
  459. printJsonP(callback, error("403", "非法签名"));
  460. return;
  461. }
  462. // 对应系统公钥验证签名
  463. at = at.verify(app.getPublicKey());
  464. if (at != null) {
  465. at.setUid(token.getUid());// 设置绑定用户ID
  466. at.setTime(token.getTime());// 设置登录时间
  467. // 更安全的做法是使用at.getUuid() 作为 key 设置 authToken
  468. // 至分布式缓存中,然后 这里根据UUID验证和赋值
  469. at.setData(token.getData());
  470. // 2、SSO 的私钥签名
  471. at.sign(SSOConfig.getInstance().getCenterPrivateKey());
  472. // 3、生成回复密文票据
  473. printJsonP(callback, success(at.encryptAuthToken()));
  474. } else {
  475. // 非法签名, 可以重定向至无权限界面,App自己处理
  476. printJsonP(callback, error("403", "非法签名"));
  477. }
  478. } else {
  479. // 非法签名, 可以重定向至无权限界面,App自己处理
  480. printJsonP(callback, error("403", "非法签名"));
  481. }
  482. }
  483. } else {
  484. // 未登录
  485. printJsonP(callback, error("404", "未登录"));
  486. }
  487. }
  488. /**
  489. * 跨域询问,随子系统一起退出
  490. *
  491. * @throws IOException
  492. */
  493. @RequestMapping(value = "/logout/ask", method = RequestMethod.GET)
  494. @ResponseBody
  495. public void replyAskOut() throws IOException {
  496. String callback = request.getParameter("callback");
  497. SSOToken token = SSOHelper.getToken(request);
  498. if (token != null) {
  499. // 已登录
  500. printJsonP(callback, error("404", "已登录"));
  501. } else {
  502. printJsonP(callback, success());
  503. }
  504. }
  505. /**
  506. * 随子系统一起退出
  507. *
  508. * @throws IOException
  509. */
  510. @RequestMapping(value = "/logout", method = RequestMethod.GET)
  511. @ResponseBody
  512. public ModelMap logout() throws IOException {
  513. System.err.print(request.getContextPath());
  514. System.err.print(request.getHeaderNames().toString());
  515. System.err.print(request.getCookies());
  516. String callback = request.getParameter("callback");
  517. String returnURL = request.getParameter("returnURL") == null ? HOME_PAGE : request.getParameter("returnURL");
  518. String baseUrl = request.getParameter("baseUrl");
  519. String appId = request.getParameter("appId") == null ? "sso" : request.getParameter("appId");
  520. SSOToken token = SSOHelper.getToken(request);
  521. ModelMap modelMap = new ModelMap();
  522. modelMap.addAttribute("callback", callback);
  523. modelMap.addAttribute("returnURL", HttpUtil.decodeURL(returnURL));
  524. modelMap.addAttribute("appId", appId);
  525. modelMap.addAttribute("logoutUrls", getOtherLogoutUrls(baseUrl));
  526. if (token != null) {
  527. SSOHelper.clearLogin(request, response);
  528. }
  529. return success(modelMap);
  530. }
  531. private Set<String> getOtherLogoutUrls(String baseUrl) {
  532. List<App> apps = appService.findAll();
  533. Set<String> logoutUrls = new HashSet<>();
  534. for (App app : apps) {
  535. if (!StringUtils.isEmpty(app.getLogoutUrl())) {
  536. logoutUrls.add(app.getLogoutUrl());
  537. }
  538. }
  539. logoutUrls.add(baseUrl);
  540. return logoutUrls;
  541. }
  542. }