/* * Copyright (c) 2018-2028, Chill Zhuang All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * Neither the name of the dreamlu.net developer nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * Author: Chill 庄骞 (smallchill@163.com) */ package org.springblade.modules.system.service.impl; import com.alibaba.fastjson.JSON; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.toolkit.Wrappers; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import lombok.AllArgsConstructor; import org.apache.commons.collections4.CollectionUtils; import org.apache.commons.lang3.StringUtils; import org.springblade.common.cache.SysCache; import org.springblade.common.constant.CommonConstant; import org.springblade.core.log.exception.ServiceException; import org.springblade.core.secure.BladeUser; import org.springblade.core.secure.utils.AuthUtil; import org.springblade.core.tool.constant.BladeConstant; import org.springblade.core.tool.constant.RoleConstant; import org.springblade.core.tool.node.ForestNodeMerger; import org.springblade.core.tool.node.TreeNode; import org.springblade.core.tool.support.Kv; import org.springblade.core.tool.utils.Func; import org.springblade.core.tool.utils.StringUtil; import org.springblade.modules.house.dto.UserHouseLabelDTO; import org.springblade.modules.house.service.IUserHouseLabelService; import org.springblade.modules.system.dto.MenuDTO; import org.springblade.modules.system.entity.*; import org.springblade.modules.system.mapper.MenuMapper; import org.springblade.modules.system.service.IMenuService; import org.springblade.modules.system.service.IRoleMenuService; import org.springblade.modules.system.service.IRoleScopeService; import org.springblade.modules.system.service.ITopMenuSettingService; import org.springblade.modules.system.vo.MenuVO; import org.springblade.modules.system.wrapper.MenuWrapper; import org.springframework.cache.annotation.Cacheable; import org.springframework.stereotype.Service; import javax.annotation.Resource; import java.io.Serializable; import java.util.*; import java.util.stream.Collectors; import static org.springblade.common.constant.CommonConstant.API_SCOPE_CATEGORY; import static org.springblade.common.constant.CommonConstant.DATA_SCOPE_CATEGORY; import static org.springblade.core.cache.constant.CacheConstant.MENU_CACHE; /** * 服务实现类 * * @author Chill */ @Service @AllArgsConstructor public class MenuServiceImpl extends ServiceImpl implements IMenuService { private final IRoleMenuService roleMenuService; private final IRoleScopeService roleScopeService; private final ITopMenuSettingService topMenuSettingService; private final static String PARENT_ID = "parentId"; private final static Integer MENU_CATEGORY = 1; @Resource private final IUserHouseLabelService iUserHouseLabelService; @Override public List lazyList(Long parentId, Map param) { if (Func.isEmpty(Func.toStr(param.get(PARENT_ID)))) { parentId = null; } List menuVOS = baseMapper.lazyList(parentId, param); for (MenuVO menuVO : menuVOS) { if (StringUtils.isNotBlank(menuVO.getLabelStr())) { List> lists = (List>) JSON.parse(menuVO.getLabelStr()); menuVO.setLabelList(lists); } } return menuVOS; } @Override public List lazyMenuList(Long parentId, Map param) { if (Func.isEmpty(Func.toStr(param.get(PARENT_ID)))) { parentId = null; } List menuVOS = baseMapper.lazyList(parentId, param); for (MenuVO menuVO : menuVOS) { if (StringUtils.isNotBlank(menuVO.getLabelStr())) { List> lists = (List>) JSON.parse(menuVO.getLabelStr()); menuVO.setLabelList(lists); } } return menuVOS; } public static final List list1 = Arrays.asList(1, 2, 3, 4, 5, 6, 11, null); public static final List list2 = Arrays.asList(1, 11, null); /** * * @param roleId * @param topMenuId * @param labelType * @param menuType * @param roleName * @param houseCode * @return */ @Override public List routes(String roleId, Long topMenuId, Integer labelType, Integer menuType, String roleName,String houseCode) { if (StringUtil.isBlank(roleId)) { return null; } List allMenus = baseMapper.selectList(Wrappers.lambdaQuery().eq(Menu::getCategory, 1).eq(Menu::getMenuType, menuType)); List roleMenus; // 超级管理员并且不是顶部菜单请求则返回全部菜单 if (AuthUtil.isAdministrator() && Func.isEmpty(topMenuId)) { roleMenus = allMenus; } // 非超级管理员并且不是顶部菜单请求则返回对应角色权限菜单 else if (!AuthUtil.isAdministrator() && Func.isEmpty(topMenuId)) { roleMenus = tenantPackageMenu(baseMapper.roleMenuByRoleId(Func.toLongList(roleId), menuType)); // 1:居民,居民需要通过人的标签来显示取保候审 if (CommonConstant.RESIDENT.equals(roleName)) { extracted(labelType, roleMenus,houseCode); } } // 顶部菜单请求返回对应角色权限菜单 else { // 角色配置对应菜单 List roleIdMenus = baseMapper.roleMenuByRoleId(Func.toLongList(roleId), menuType); // 反向递归角色菜单所有父级 List routes = new LinkedList<>(roleIdMenus); roleIdMenus.forEach(roleMenu -> recursion(allMenus, routes, roleMenu)); // 顶部配置对应菜单 List topIdMenus = baseMapper.roleMenuByTopMenuId(topMenuId); // 筛选匹配角色对应的权限菜单 roleMenus = topIdMenus.stream().filter(x -> routes.stream().anyMatch(route -> route.getId().longValue() == x.getId().longValue()) ).collect(Collectors.toList()); } return buildRoutes(allMenus, roleMenus); } /** * 房屋的时候才通过人的标签过滤 * * @param roleMenus * @param labelType */ private void extracted(Integer labelType, List roleMenus,String houseCode) { UserHouseLabelDTO userHouseLabelDTO = new UserHouseLabelDTO(); userHouseLabelDTO.setUserId(AuthUtil.getUserId()); userHouseLabelDTO.setLableType(labelType); userHouseLabelDTO.setHouseCode(houseCode); List integers = iUserHouseLabelService.selectUserLabelList(userHouseLabelDTO); Iterator iterator = roleMenus.iterator(); while (iterator.hasNext()) { Menu next = iterator.next(); if (!next.getParentId().equals(0)) { if (StringUtils.isNotBlank(next.getLabelId())) { String[] split = next.getLabelId().split(","); List integerList = Arrays.stream(split).map(Integer::valueOf).collect(Collectors.toList()); Collection union = CollectionUtils.intersection(integerList, integers); if (union.size() == 0) { iterator.remove(); // 场所的时候,删除取保候审 } else if (CommonConstant.NUMBER_TWO.equals(labelType)) { if (next.getName().trim().equals("取保监居")) { iterator.remove(); } } } else { if (next.getName().trim().equals("取保监居")) { iterator.remove(); } } } } } @Override public List routesExt(String roleId, Long topMenuId) { if (StringUtil.isBlank(roleId)) { return null; } List allMenus = baseMapper.allMenuExt(); List roleMenus = baseMapper.roleMenuExt(Func.toLongList(roleId), topMenuId); return buildRoutes(allMenus, roleMenus); } private List buildRoutes(List allMenus, List roleMenus) { List routes = new LinkedList<>(roleMenus); roleMenus.forEach(roleMenu -> recursion(allMenus, routes, roleMenu)); routes.sort(Comparator.comparing(Menu::getSort)); MenuWrapper menuWrapper = new MenuWrapper(); List collect = routes.stream().filter(x -> Func.equals(x.getCategory(), 1)).collect(Collectors.toList()); return menuWrapper.listNodeVO(collect); } private void recursion(List allMenus, List routes, Menu roleMenu) { Optional menu = allMenus.stream().filter(x -> Func.equals(x.getId(), roleMenu.getParentId())).findFirst(); if (menu.isPresent() && !routes.contains(menu.get())) { routes.add(menu.get()); recursion(allMenus, routes, menu.get()); } } @Override public List buttons(String roleId) { List buttons = (AuthUtil.isAdministrator()) ? baseMapper.allButtons() : baseMapper.buttons(Func.toLongList(roleId)); MenuWrapper menuWrapper = new MenuWrapper(); return menuWrapper.listNodeVO(buttons); } @Override public List tree() { return ForestNodeMerger.merge(baseMapper.tree()); } @Override public List grantTree(BladeUser user) { return ForestNodeMerger.merge(AuthUtil.getUser().getRoleName().equals(RoleConstant.ADMINISTRATOR) ? baseMapper.grantTree() : baseMapper.grantTreeByRole(Func.toLongList(user.getRoleId()))); } @Override public List grantTopTree(BladeUser user) { List menuTree = user.getTenantId().equals(BladeConstant.ADMIN_TENANT_ID) ? baseMapper.grantTopTree() : baseMapper.grantTopTreeByRole(Func.toLongList(user.getRoleId())); return ForestNodeMerger.merge(tenantPackageTree(menuTree, user.getTenantId())); } /** * 租户菜单权限自定义筛选 */ private List tenantPackageTree(List menuTree, String tenantId) { TenantPackage tenantPackage = SysCache.getTenantPackage(tenantId); if (!AuthUtil.isAdministrator() && Func.isNotEmpty(tenantPackage) && tenantPackage.getId() > 0L) { List menuIds = Func.toLongList(tenantPackage.getMenuId()); // 筛选出两者菜单交集集合 List collect = menuTree.stream().filter(x -> menuIds.contains(x.getId())).collect(Collectors.toList()); // 创建递归基础集合 List packageTree = new LinkedList<>(collect); // 递归筛选出菜单集合所有父级 collect.forEach(treeNode -> recursionParent(menuTree, packageTree, treeNode)); // 递归筛选出菜单集合所有子级 collect.forEach(treeNode -> recursionChild(menuTree, packageTree, treeNode)); // 合并在一起返回最终集合 return packageTree; } return menuTree; } /** * 父节点递归 */ public void recursionParent(List menuTree, List packageTree, TreeNode treeNode) { Optional node = menuTree.stream().filter(x -> Func.equals(x.getId(), treeNode.getParentId())).findFirst(); if (node.isPresent() && !packageTree.contains(node.get())) { packageTree.add(node.get()); recursionParent(menuTree, packageTree, node.get()); } } /** * 子节点递归 */ public void recursionChild(List menuTree, List packageTree, TreeNode treeNode) { List nodes = menuTree.stream().filter(x -> Func.equals(x.getParentId(), treeNode.getId())).collect(Collectors.toList()); nodes.forEach(node -> { if (!packageTree.contains(node)) { packageTree.add(node); recursionChild(menuTree, packageTree, node); } }); } /** * 租户菜单权限自定义筛选 */ private List tenantPackageMenu(List menu) { TenantPackage tenantPackage = SysCache.getTenantPackage(AuthUtil.getTenantId()); if (Func.isNotEmpty(tenantPackage) && tenantPackage.getId() > 0L) { List menuIds = Func.toLongList(tenantPackage.getMenuId()); menu = menu.stream().filter(x -> menuIds.contains(x.getId())).collect(Collectors.toList()); } return menu; } @Override public List grantDataScopeTree(BladeUser user) { return ForestNodeMerger.merge(user.getTenantId().equals(BladeConstant.ADMIN_TENANT_ID) ? baseMapper.grantDataScopeTree() : baseMapper.grantDataScopeTreeByRole(Func.toLongList(user.getRoleId()))); } @Override public List grantApiScopeTree(BladeUser user) { return ForestNodeMerger.merge(user.getTenantId().equals(BladeConstant.ADMIN_TENANT_ID) ? baseMapper.grantApiScopeTree() : baseMapper.grantApiScopeTreeByRole(Func.toLongList(user.getRoleId()))); } @Override public List roleTreeKeys(String roleIds) { List roleMenus = roleMenuService.list(Wrappers.query().lambda().in(RoleMenu::getRoleId, Func.toLongList(roleIds))); return roleMenus.stream().map(roleMenu -> Func.toStr(roleMenu.getMenuId())).collect(Collectors.toList()); } @Override public List topTreeKeys(String topMenuIds) { List settings = topMenuSettingService.list(Wrappers.query().lambda().in(TopMenuSetting::getTopMenuId, Func.toLongList(topMenuIds))); return settings.stream().map(setting -> Func.toStr(setting.getMenuId())).collect(Collectors.toList()); } @Override public List dataScopeTreeKeys(String roleIds) { List roleScopes = roleScopeService.list(Wrappers.query().lambda().eq(RoleScope::getScopeCategory, DATA_SCOPE_CATEGORY).in(RoleScope::getRoleId, Func.toLongList(roleIds))); return roleScopes.stream().map(roleScope -> Func.toStr(roleScope.getScopeId())).collect(Collectors.toList()); } @Override public List apiScopeTreeKeys(String roleIds) { List roleScopes = roleScopeService.list(Wrappers.query().lambda().eq(RoleScope::getScopeCategory, API_SCOPE_CATEGORY).in(RoleScope::getRoleId, Func.toLongList(roleIds))); return roleScopes.stream().map(roleScope -> Func.toStr(roleScope.getScopeId())).collect(Collectors.toList()); } @Override @Cacheable(cacheNames = MENU_CACHE, key = "'auth:routes:' + #user.roleId") public List authRoutes(BladeUser user) { List routes = baseMapper.authRoutes(Func.toLongList(user.getRoleId())); List list = new ArrayList<>(); routes.forEach(route -> list.add(Kv.create().set(route.getPath(), Kv.create().set("authority", Func.toStrArray(route.getAlias()))))); return list; } @Override public boolean removeMenu(String ids) { Long cnt = baseMapper.selectCount(Wrappers.query().lambda().in(Menu::getParentId, Func.toLongList(ids))); if (cnt > 0L) { throw new ServiceException("请先删除子节点!"); } return removeByIds(Func.toLongList(ids)); } @Override public boolean submit(MenuDTO menu) { LambdaQueryWrapper menuQueryWrapper = Wrappers.lambdaQuery(); if (menu.getId() == null) { menuQueryWrapper.eq(Menu::getCode, menu.getCode()).or( wrapper -> wrapper.eq(Menu::getName, menu.getName()).eq(Menu::getCategory, MENU_CATEGORY) ); } else { menuQueryWrapper.ne(Menu::getId, menu.getId()).eq(Menu::getIsDeleted, 0).and( wrapper -> wrapper.eq(Menu::getCode, menu.getCode()).or( o -> o.eq(Menu::getName, menu.getName()) .eq(Menu::getCategory, MENU_CATEGORY) .eq(Menu::getMenuType, menu.getMenuType()) .eq(Menu::getParentId, menu.getParentId()) ) ); } Long cnt = baseMapper.selectCount(menuQueryWrapper); if (cnt > 0L) { throw new ServiceException("菜单名或编号已存在!"); } if (menu.getParentId() == null && menu.getId() == null) { menu.setParentId(BladeConstant.TOP_PARENT_ID); } if (menu.getParentId() != null && menu.getId() == null) { Menu parentMenu = baseMapper.selectById(menu.getParentId()); if (parentMenu != null && parentMenu.getCategory() != 1) { throw new ServiceException("父节点只可选择菜单类型!"); } } menu.setLabelStr(JSON.toJSONString(menu.getLabelList())); List> labelList = menu.getLabelList(); StringBuffer stringBuffer = new StringBuffer(); for (List strings : labelList) { stringBuffer.append(strings.get(strings.size() - 1)).append(","); } menu.setLabelId(stringBuffer.toString()); menu.setIsDeleted(BladeConstant.DB_NOT_DELETED); return saveOrUpdate(menu); } }