/* * 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 com.vci.ubcs.system.service.impl; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.toolkit.Wrappers; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.vci.ubcs.system.cache.SysCache; import com.vci.ubcs.system.dto.MenuDTO; import com.vci.ubcs.system.entity.*; import com.vci.ubcs.system.vo.MenuVO; import com.vci.ubcs.system.mapper.MenuMapper; import com.vci.ubcs.system.service.IMenuService; import com.vci.ubcs.system.service.IRoleMenuService; import com.vci.ubcs.system.service.IRoleScopeService; import com.vci.ubcs.system.service.ITopMenuSettingService; import com.vci.ubcs.system.wrapper.MenuWrapper; import lombok.AllArgsConstructor; import com.vci.ubcs.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.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.springframework.cache.annotation.Cacheable; import org.springframework.stereotype.Service; import java.util.*; import java.util.stream.Collectors; import static com.vci.ubcs.common.constant.CommonConstant.API_SCOPE_CATEGORY; import static com.vci.ubcs.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; @Override public List lazyList(Long parentId, Map param) { if (Func.isEmpty(Func.toStr(param.get(PARENT_ID)))) { parentId = null; } return baseMapper.lazyList(parentId, param); } @Override public List lazyMenuList(Long parentId, Map param) { if (Func.isEmpty(Func.toStr(param.get(PARENT_ID)))) { parentId = null; } return baseMapper.lazyMenuList(parentId, param); } @Override public List routes(String roleId, Long topMenuId) { if (StringUtil.isBlank(roleId)) { return null; } List allMenus = baseMapper.allMenu(); List roleMenus; // 超级管理员并且不是顶部菜单请求则返回全部菜单 if (AuthUtil.isAdministrator() && Func.isEmpty(topMenuId)) { roleMenus = allMenus; } // 非超级管理员并且不是顶部菜单请求则返回对应角色权限菜单 else if (!AuthUtil.isAdministrator() && Func.isEmpty(topMenuId)) { roleMenus = tenantPackageMenu(baseMapper.roleMenuByRoleId(Func.toLongList(roleId))); } // 顶部菜单请求返回对应角色权限菜单 else { // 角色配置对应菜单 List roleIdMenus = baseMapper.roleMenuByRoleId(Func.toLongList(roleId)); // 反向递归角色菜单所有父级 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); } 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) { List menuTree = user.getTenantId().equals(BladeConstant.ADMIN_TENANT_ID) ? baseMapper.grantTree() : baseMapper.grantTreeByRole(Func.toLongList(user.getRoleId())); return ForestNodeMerger.merge(tenantPackageTree(menuTree, user.getTenantId())); } @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(Menu 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()).and( wrapper -> wrapper.eq(Menu::getCode, menu.getCode()).or( o -> o.eq(Menu::getName, menu.getName()).eq(Menu::getCategory, MENU_CATEGORY) ) ); } 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.setIsDeleted(BladeConstant.DB_NOT_DELETED); return saveOrUpdate(menu); } }