spring-boot-plus V1.4.0發布 集成用戶角色權限部門管理


摘自:https://www.cnblogs.com/springbootplus/p/11795882.html

 

spring-boot-plus V1.4.0發布 集成用戶角色權限部門管理

RBAC用戶角色權限

用戶角色權限部門管理核心接口介紹

Shiro權限配置

數據庫模型圖

spring-boot-plus數據庫模型圖

獲取驗證碼

  • 可配置是否啟用驗證碼
  • 默認未啟用
  • 如已啟用驗證碼校驗,登陸時,需傳入verifyToken和code

驗證碼演示

spring-boot-plus: # 是否啟用ansi控制台輸出有顏色的字體  enable-ansi: true # 是否啟用驗證碼  enable-verify-code: true

enable-verify-code 設置為 true 啟用驗證碼驗證

兩種方式獲取驗證碼

驗證碼后台保存在Redis中,過期時間默認為5分鍾

驗證碼Swagger

方式一:

輸出圖片流到瀏覽器,驗證碼token輸出到響應頭

http://localhost:8888/verificationCode/getImage

驗證碼圖片

Response Headers
HTTP/1.1 200
verifyToken: 6515b4b798ce49e68b1e40f98ff8eb19
方式二:

獲取Base64編碼圖片和驗證碼token

http://localhost:8888/verificationCode/getBase64Image
{
  "code": 200, "msg": "操作成功", "success": true, "data": { "image": "", "verifyToken": "42ba8abde7bc47b2b1397b4d6676956a" }, "time": "2019-11-01 22:40:37" }

系統用戶登陸

驗證碼Swagger

  • POST請求,Content-Type: application/json
http://127.0.0.1:8888/login
  • 請求參數
{
  "code": "驗證碼", "password": "123456", "username": "admin", "verifyToken": "驗證碼token" }

注意

  • 如果沒有啟用驗證碼登陸,則只需傳入usernamepassword
  • 前端應將密碼加密后進行傳輸

登陸成功

  • 返回登陸用戶信息:部門/角色/權限
  • 返回用戶token
{
  "code": 200, "msg": "登陸成功", "success": true, "data": { "token": "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJhdWQiOiJ3ZWIiLCJpc3MiOiJzcHJpbmctYm9vdC1wbHVzIiwiZXhwIjoxNTcyNjIzMDE5LCJpYXQiOjE1NzI2MTk0MTksImp0aSI6IjdlZmVlM2UwMjc2MTRiYTc5M2I2YmYwZmE4NTgzYmUwIiwidXNlcm5hbWUiOiJhZG1pbiJ9.O3w7CNRqw_Miwp8MDzPND6w490c9Q7yFlKpFJK9ubSU", "loginSysUserVo": { "id": "1", "username": "admin", "nickname": "管理員", "gender": 1, "state": 1, "departmentId": "1", "departmentName": "管理部", "roleId": "1", "roleName": "管理員", "roleCode": "admin", "permissionCodes": [ "sys:permission:codes", "system:management", "sys:department:update", "sys:department:page", "sys:role:management", "sys:permission:add", "sys:user:add", "sys:role:page", "sys:permission:page", "sys:department:delete", "sys:permission:management", "sys:user:delete", "sys:department:management", "sys:user:page", "sys:user:update", "sys:user:update:password", "sys:user:update:head", "sys:role:add", "sys:permission:menu:tree", "sys:department:info", "sys:permission:all:menu:list", "sys:permission:info", "sys:role:info", "sys:permission:all:menu:tree", "sys:permission:update", "sys:permission:menu:list", "sys:role:update", "sys:user:info", "sys:user:management", "sys:role:delete", "sys:permission:delete" ] } }, "time": "2019-11-01 22:43:39" }
  • token默認失效時間為1個小時
  • 設置JWT Token失效時間
  ############################ JWT start #############################  jwt: # 默認過期時間1小時,單位:秒  expire-second: 3600
  • 后台使用Redis緩存登陸用戶信息
  • redis key
login:user:admin
其它需要授權訪問的接口,請求頭需攜帶token

部門樹形列表

  • 部門可以設置為N級,后台使用遞歸將部門列表轉換為樹形列表

驗證碼Swagger

  • SysDepartmentServiceImpl
    @Override public List<SysDepartmentTreeVo> getAllDepartmentTree() { List<SysDepartment> sysDepartmentList = getAllDepartmentList(); if (CollectionUtils.isEmpty(sysDepartmentList)) { throw new IllegalArgumentException("SysDepartment列表不能為空"); } List<SysDepartmentTreeVo> list = SysDepartmentConvert.INSTANCE.listToTreeVoList(sysDepartmentList); List<SysDepartmentTreeVo> treeVos = new ArrayList<>(); for (SysDepartmentTreeVo treeVo : list) { if (treeVo.getParentId() == null) { treeVos.add(findChildren(treeVo, list)); } } return treeVos; } /** * 遞歸獲取樹形結果列表 * * @param tree * @param list * @return */ public SysDepartmentTreeVo findChildren(SysDepartmentTreeVo tree, List<SysDepartmentTreeVo> list) { for (SysDepartmentTreeVo vo : list) { if (tree.getId().equals(vo.getParentId())) { if (tree.getChildren() == null) { tree.setChildren(new ArrayList<>()); } tree.getChildren().add(findChildren(vo, list)); } } return tree; }
  • 前端JSON結構
http://127.0.0.1:8888/sysDepartment/getAllDepartmentTree

角色管理

驗證碼Swagger

設置角色權限

  • 核心代碼,刪除角色權限,新增角色權限
  • 求集合的差集
  • SysRolePermissionServiceImpl
    @Transactional(rollbackFor = Exception.class) @Override public boolean updateSysRole(UpdateSysRoleParam updateSysRoleParam) throws Exception { Long roleId = updateSysRoleParam.getId(); List<Long> permissionIds = updateSysRoleParam.getPermissionIds(); // 校驗角色是否存在 SysRole sysRole = getById(roleId); if (sysRole == null) { throw new BusinessException("該角色不存在"); } // 校驗權限列表是否存在 if (!sysPermissionService.isExistsByPermissionIds(permissionIds)) { throw new BusinessException("權限列表id匹配失敗"); } // 修改角色 sysRole.setName(updateSysRoleParam.getName()) .setType(updateSysRoleParam.getType()) .setRemark(updateSysRoleParam.getRemark()) .setState(updateSysRoleParam.getState()) .setUpdateTime(new Date()); boolean updateResult = updateById(sysRole); if (!updateResult) { throw new DaoException("修改系統角色失敗"); } // 獲取之前的權限id集合 List<Long> beforeList = sysRolePermissionService.getPermissionIdsByRoleId(roleId); // 差集計算 // before:1,2,3,4,5,6 // after: 1,2,3,4,7,8 // 刪除5,6 新增7,8 // 此處真實刪除,去掉deleted字段的@TableLogic注解 Set<Long> beforeSet = new HashSet<>(beforeList); Set<Long> afterSet = new HashSet<>(permissionIds); SetUtils.SetView deleteSet = SetUtils.difference(beforeSet, afterSet); SetUtils.SetView addSet = SetUtils.difference(afterSet, beforeSet); log.debug("deleteSet = " + deleteSet); log.debug("addSet = " + addSet); // 刪除權限關聯 UpdateWrapper updateWrapper = new UpdateWrapper(); updateWrapper.eq("role_id",roleId); updateWrapper.in("permission_id",deleteSet); boolean deleteResult = sysRolePermissionService.remove(updateWrapper); if (!deleteResult) { throw new DaoException("刪除角色權限關系失敗"); } // 新增權限關聯 boolean addResult = sysRolePermissionService.saveSysRolePermissionBatch(roleId, addSet); if (!addResult) { throw new DaoException("新增角色權限關系失敗"); } return true; }

權限管理

驗證碼Swagger

權限樹形列表

  • 用戶設置角色權限時,選擇權限菜單
  • 權限分為菜單和功能權限
  • 后台獲取三層權限樹
    @Override public List<SysPermissionTreeVo> getAllMenuTree() throws Exception { List<SysPermission> list = getAllMenuList(); // 轉換成樹形菜單 List<SysPermissionTreeVo> treeVos = convertSysPermissionTreeVoList(list); return treeVos; } @Override public List<SysPermissionTreeVo> convertSysPermissionTreeVoList(List<SysPermission> list) { if (CollectionUtils.isEmpty(list)) { throw new IllegalArgumentException("SysPermission列表不能為空"); } // 按level分組獲取map Map<Integer, List<SysPermission>> map = list.stream().collect(Collectors.groupingBy(SysPermission::getLevel)); List<SysPermissionTreeVo> treeVos = new ArrayList<>(); // 循環獲取三級菜單樹形集合 for (SysPermission one : map.get(LevelEnum.ONE.getKey())) { SysPermissionTreeVo oneVo = SysPermissionConvert.INSTANCE.permissionToTreeVo(one); Long oneParentId = oneVo.getParentId(); if (oneParentId == null || oneParentId == 0) { treeVos.add(oneVo); } List<SysPermission> twoList = map.get(LevelEnum.TWO.getKey()); if (CollectionUtils.isNotEmpty(twoList)) { for (SysPermission two : twoList) { SysPermissionTreeVo twoVo = SysPermissionConvert.INSTANCE.permissionToTreeVo(two); if (two.getParentId().equals(one.getId())) { oneVo.getChildren().add(twoVo); } List<SysPermission> threeList = map.get(LevelEnum.THREE.getKey()); if (CollectionUtils.isNotEmpty(threeList)) { for (SysPermission three : threeList) { if (three.getParentId().equals(two.getId())) { SysPermissionTreeVo threeVo = SysPermissionConvert.INSTANCE.permissionToTreeVo(three); twoVo.getChildren().add(threeVo); } } } } } } return treeVos; }
  • 前端JSON格式
http://127.0.0.1:8888/sysPermission/getAllMenuTree

權限編碼列表

返回當前用戶所有的權限編碼,方便前端展示導航菜單和功能按鈕

http://127.0.0.1:8888/sysPermission/getPermissionCodesByUserId/1
{
  "code": 200, "msg": "操作成功", "success": true, "data": [ "system:management", "system:management", "sys:user:management", "sys:user:management", "sys:role:management", "sys:permission:management", "sys:department:management", "sys:user:add", "sys:user:add", "sys:user:update", "sys:user:update", "sys:user:delete", "sys:user:delete", "sys:user:info", "sys:user:info", "sys:user:page", "sys:user:page", "sys:user:update:password", "sys:user:update:head", "sys:role:add", "sys:role:update", "sys:role:delete", "sys:role:info", "sys:role:page", "sys:permission:add", "sys:permission:update", "sys:permission:delete", "sys:permission:info", "sys:permission:page", "sys:permission:all:menu:list", "sys:permission:all:menu:tree", "sys:permission:menu:list", "sys:permission:menu:tree", "sys:permission:codes", "sys:department:update", "sys:department:delete", "sys:department:info", "sys:department:page" ], "time": "2019-11-02 00:32:17" }

注意

  • 使用Shiro注解@RequiresPermissions進行controller方法權限過濾
@RequiresPermissions("sys:department:add")
  • 生成代碼時,可配置生成RequiresPermissions注解
        // 是否生成Shiro RequiresPermissions注解 codeGenerator.setRequiresPermissions(true);
  • 生成或新增的controller方法,需要進行權限管理,需要到sys_permission表新增權限編碼記錄,並給相應角色賦予權限
 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM