From b8a0affd03b5fa9fa33cd6f870e90394c2df86c7 Mon Sep 17 00:00:00 2001
From: 潘志宝 <979469083@qq.com>
Date: 星期一, 06 一月 2025 13:31:07 +0800
Subject: [PATCH] Merge remote-tracking branch 'origin/master'

---
 iailab-module-system/iailab-module-system-biz/src/main/java/com/iailab/module/system/service/app/AppServiceImpl.java |  229 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++
 1 files changed, 229 insertions(+), 0 deletions(-)

diff --git a/iailab-module-system/iailab-module-system-biz/src/main/java/com/iailab/module/system/service/app/AppServiceImpl.java b/iailab-module-system/iailab-module-system-biz/src/main/java/com/iailab/module/system/service/app/AppServiceImpl.java
index cbfe760..43bcb42 100644
--- a/iailab-module-system/iailab-module-system-biz/src/main/java/com/iailab/module/system/service/app/AppServiceImpl.java
+++ b/iailab-module-system/iailab-module-system-biz/src/main/java/com/iailab/module/system/service/app/AppServiceImpl.java
@@ -1,15 +1,47 @@
 package com.iailab.module.system.service.app;
 
+import cn.hutool.core.collection.CollUtil;
+import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
+import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
 import com.iailab.framework.common.pojo.PageResult;
 import com.iailab.framework.common.util.object.BeanUtils;
+import com.iailab.framework.mybatis.core.query.LambdaQueryWrapperX;
+import com.iailab.framework.security.core.util.SecurityFrameworkUtils;
+import com.iailab.framework.tenant.core.aop.TenantIgnore;
+import com.iailab.module.system.api.app.dto.AppMenuRespDTO;
 import com.iailab.module.system.controller.admin.app.vo.AppPageReqVO;
 import com.iailab.module.system.controller.admin.app.vo.AppSaveReqVO;
+import com.iailab.module.system.controller.admin.auth.vo.AuthPermissionInfoRespVO;
+import com.iailab.module.system.convert.auth.AuthConvert;
 import com.iailab.module.system.dal.dataobject.app.AppDO;
+import com.iailab.module.system.dal.dataobject.permission.MenuDO;
+import com.iailab.module.system.dal.dataobject.permission.RoleDO;
+import com.iailab.module.system.dal.dataobject.permission.RoleMenuDO;
+import com.iailab.module.system.dal.dataobject.tenant.TenantDO;
+import com.iailab.module.system.dal.dataobject.tenant.TenantPackageDO;
 import com.iailab.module.system.dal.mysql.app.AppMapper;
+import com.iailab.module.system.dal.mysql.permission.MenuMapper;
+import com.iailab.module.system.dal.mysql.permission.RoleMapper;
+import com.iailab.module.system.dal.mysql.permission.RoleMenuMapper;
+import com.iailab.module.system.dal.mysql.tenant.TenantMapper;
+import com.iailab.module.system.dal.mysql.tenant.TenantPackageMapper;
+import com.iailab.module.system.enums.permission.MenuTypeEnum;
+import com.iailab.module.system.service.permission.MenuService;
+import com.iailab.module.system.service.permission.PermissionService;
 import lombok.extern.slf4j.Slf4j;
+import org.apache.commons.collections.CollectionUtils;
+import org.apache.commons.lang3.ObjectUtils;
 import org.springframework.stereotype.Service;
+import org.springframework.transaction.annotation.Transactional;
 
 import javax.annotation.Resource;
+import java.util.*;
+
+import static com.iailab.framework.common.exception.util.ServiceExceptionUtil.exception;
+import static com.iailab.framework.common.util.collection.CollectionUtils.convertSet;
+import static com.iailab.framework.tenant.core.context.TenantContextHolder.getTenantId;
+import static com.iailab.module.system.enums.ErrorCodeConstants.MENU_EXISTS_CHILDREN;
+import static com.iailab.module.system.enums.ErrorCodeConstants.MENU_NOT_EXISTS;
 
 /**
  * @author PanZhibao
@@ -23,23 +55,57 @@
     @Resource
     private AppMapper appMapper;
 
+    @Resource
+    private MenuMapper menuMapper;
+
+    @Resource
+    private RoleMapper roleMapper;
+
+    @Resource
+    private RoleMenuMapper roleMenuMapper;
+
+    @Resource
+    private PermissionService permissionService;
+
+    @Resource
+    private TenantPackageMapper tenantPackageMapper;
+
+    @Resource
+    private TenantMapper tenantMapper;
+
+    @Resource
+    private MenuService menuService;
+
 
     @Override
+    @Transactional(rollbackFor = Exception.class)
+    @TenantIgnore
     public Long create(AppSaveReqVO createReqVO) {
         AppDO app = BeanUtils.toBean(createReqVO, AppDO.class);
         appMapper.insert(app);
+//        //为应用创建默认菜单并授权
+        dealAppMenu(1, app);
         return app.getId();
     }
 
     @Override
+    @Transactional(rollbackFor = Exception.class)
+    @TenantIgnore
     public Long update(AppSaveReqVO createReqVO) {
         AppDO app = BeanUtils.toBean(createReqVO, AppDO.class);
         appMapper.updateById(app);
+//        //修改默认菜单并授权
+        dealAppMenu(2, app);
         return app.getId();
     }
 
     @Override
+    @Transactional(rollbackFor = Exception.class)
+    @TenantIgnore
     public void delete(Long id) {
+        AppDO appDO = new AppDO();
+        appDO.setId(id);
+        dealAppMenu(3, appDO);
         appMapper.deleteById(id);
     }
 
@@ -53,4 +119,167 @@
         return appMapper.selectPage(pageReqVO);
     }
 
+    @Override
+    public List<AppDO> getList() {
+        Long tenantId = getTenantId();
+        LambdaQueryWrapperX<MenuDO> menuDOLambdaQueryWrapperX = new LambdaQueryWrapperX<>();
+        menuDOLambdaQueryWrapperX.eq(MenuDO::getParentId, 0)
+                .eq(MenuDO::getAppId, 0);
+        if (tenantId != 1) {
+            //非系统租户查租户套餐
+            TenantDO tenantDO = tenantMapper.selectById(tenantId);
+            TenantPackageDO tenantPackageDO = tenantPackageMapper.selectById(tenantDO.getPackageId());
+            menuDOLambdaQueryWrapperX.in(MenuDO::getId, tenantPackageDO.getMenuIds());
+        }
+        //查询系统应用菜单
+//        List<MenuDO> menuDOS = menuMapper.selectList(menuDOLambdaQueryWrapperX);
+//        List<AppDO> systemApps = convertMenuToApp(menuDOS);
+        List<AppDO> systemApps = new ArrayList<>();
+        List<AppDO> appDOS = appMapper.selectList();
+        //暂时先遍历处理应用菜单和应用类型
+        appDOS.stream().forEach(appDO -> {
+            List<MenuDO> menuDOS = menuMapper.selectList(new LambdaQueryWrapper<MenuDO>().eq(MenuDO::getParentId, 0)
+                    .eq(MenuDO::getAppId, appDO.getId()));
+            appDO.setAppMenuId(menuDOS.get(0).getId());
+            appDO.setAppType(2);
+        });
+        systemApps.addAll(appDOS);
+        return systemApps;
+    }
+
+    @Override
+    public AppDO getAppByTenantId(Long tenantId) {
+        //暂时支持一个租户对应一个应用
+        List<AppDO> appDOS = appMapper.selectList(new LambdaQueryWrapper<AppDO>().eq(AppDO::getTenantId, tenantId));
+        if(CollectionUtils.isNotEmpty(appDOS)) {
+            return appDOS.get(0);
+        } else {
+            AppDO appDO = new AppDO();
+            appDO.setTenantId(tenantId);
+            appDO.setId(0L);
+            return appDO;
+        }
+    }
+
+    @Override
+    public List<AuthPermissionInfoRespVO.MenuVO> getAppMenu(Long id) {
+        MenuDO menuDO = menuMapper.selectById(id);
+        List<MenuDO> children = new LinkedList<>();
+        // 遍历每一层
+        Collection<Long> parentIds = Collections.singleton(id);
+        for (int i = 0; i < Short.MAX_VALUE; i++) { // 使用 Short.MAX_VALUE 避免 bug 场景下,存在死循环
+            // 查询当前层,所有的子应用菜单
+            List<MenuDO> menus = menuMapper.selectListByParentId(parentIds);
+            // 1. 如果没有子菜单,则结束遍历
+            if (CollUtil.isEmpty(menus)) {
+                break;
+            }
+            // 2. 如果有子应用菜单,继续遍历
+            children.addAll(menus);
+            parentIds = convertSet(menus, MenuDO::getId);
+        }
+        children = menuService.filterDisableMenus(children);
+        return AuthConvert.INSTANCE.buildMenuTree(children, id, menuDO.getPath());
+
+    }
+
+    private void dealAppMenu(Integer type, AppDO app){
+        String loginUserNickname = SecurityFrameworkUtils.getLoginUserNickname();
+        MenuDO menuDO = new MenuDO();
+        menuDO.setAppId(app.getId());
+        menuDO.setName(app.getAppName());
+        menuDO.setType(MenuTypeEnum.DIR.getType());
+        menuDO.setSort(app.getOrderNum());
+        menuDO.setPath("/" + app.getAppCode());
+        menuDO.setTenantId(app.getTenantId());
+        if(type == 1){
+            menuDO.setCreator(loginUserNickname);
+            menuDO.setCreateTime(app.getCreateTime());
+            menuDO.setIcon("fa-solid:border-none"); //默认icon
+            menuMapper.insert(menuDO);
+//            //内置租户角色分配菜单
+//            assignRoleMenu(menuDO.getId(), app.getTenantId());
+        } else if(type == 2){
+            LambdaUpdateWrapper<MenuDO> updateWrapper = new LambdaUpdateWrapper<>();
+            updateWrapper.eq(MenuDO::getAppId, app.getId());
+            updateWrapper.eq(MenuDO::getParentId, 0L);
+            List<MenuDO> menuDOS = menuMapper.selectList(updateWrapper);
+            if(menuDOS.size() > 0){
+                menuDO.setUpdater(loginUserNickname);
+                menuDO.setUpdateTime(app.getUpdateTime());
+                menuMapper.update(menuDO, updateWrapper);
+            } else {
+                menuDO.setCreator(loginUserNickname);
+                menuDO.setCreateTime(app.getCreateTime());
+                menuMapper.insert(menuDO);
+//                //内置租户角色分配菜单
+//                assignRoleMenu(menuDO.getId(), app.getTenantId());
+            }
+        } else if(type == 3){
+            //删除租户、角色权限
+            app = appMapper.selectById(app.getId());
+            LambdaQueryWrapperX<MenuDO> menuWrapper = new LambdaQueryWrapperX<>();
+            menuWrapper.eq(MenuDO::getAppId, app.getId());
+            menuWrapper.eq(MenuDO::getType, MenuTypeEnum.DIR.getType());
+            menuWrapper.eq(MenuDO::getParentId, 0);
+            MenuDO menu = menuMapper.selectOne(menuWrapper);
+            TenantDO tenantDO = tenantMapper.selectById(app.getTenantId());
+            if(ObjectUtils.isNotEmpty(menu) && ObjectUtils.isNotEmpty(tenantDO)) {
+                TenantPackageDO tenantPackageDO = tenantPackageMapper.selectById(tenantDO.getPackageId());
+                Set<Long> menuIds = tenantPackageDO.getMenuIds();
+                menuIds.remove(menu.getId());
+                // 校验是否还有子菜单
+                if (menuMapper.selectCountByParentId(menu.getId()) > 0) {
+                    throw exception(MENU_EXISTS_CHILDREN);
+                }
+                // 校验删除的菜单是否存在
+                if (menuMapper.selectById(menu.getId()) == null) {
+                    throw exception(MENU_NOT_EXISTS);
+                }
+                // 标记删除
+                menuMapper.deleteById(menu.getId());
+                // 删除授予给角色的权限
+                permissionService.processMenuDeleted(menu.getId());
+                //删除菜单
+                menuMapper.delete(menuWrapper);
+            }
+        }
+    }
+
+    private void assignRoleMenu(Long menuId, Long tenantId) {
+        //查询内置租户管理员
+        LambdaQueryWrapperX<RoleDO> roleQueryWrapper = new LambdaQueryWrapperX<>();
+        roleQueryWrapper.eq(RoleDO::getCode, "tenant_admin");
+        roleQueryWrapper.eq(RoleDO::getTenantId, tenantId);
+        RoleDO roleDO = roleMapper.selectOne(roleQueryWrapper);
+        RoleMenuDO entity = new RoleMenuDO();
+        entity.setRoleId(roleDO.getId());
+        entity.setMenuId(menuId);
+        entity.setTenantId(tenantId);
+        roleMenuMapper.insert(entity);
+        TenantDO tenantDO = tenantMapper.selectById(tenantId);
+        TenantPackageDO tenantPackageDO = tenantPackageMapper.selectById(tenantDO.getPackageId());
+        Set<Long> menuIds = tenantPackageDO.getMenuIds();
+        menuIds.add(menuId);
+        tenantPackageMapper.updateById(tenantPackageDO);
+    }
+
+    private List<AppDO> convertMenuToApp(List<MenuDO> menuDOS) {
+        List<AppDO> appDOS = new ArrayList<>();
+        menuDOS.stream().forEach(menuDO -> {
+            AppDO appDO = new AppDO();
+            appDO.setAppName(menuDO.getName());
+            appDO.setOrderNum(menuDO.getSort());
+            appDO.setAppMenuId(menuDO.getId());
+            appDO.setAppType(1);
+            appDOS.add(appDO);
+        });
+        return appDOS;
+    }
+
+    public List<AppDO> selectBatchIds(List<Long> ids) {
+        List<AppDO> appDOS = appMapper.selectBatchIds(ids);
+        return appDOS;
+    }
+
 }
\ No newline at end of file

--
Gitblit v1.9.3