From c5ecfa1e234f06340a427a8c96b4471fad7b2069 Mon Sep 17 00:00:00 2001 From: fNumb <326122450@qq.com> Date: Tue, 18 Jul 2023 10:03:17 +0800 Subject: [PATCH] =?UTF-8?q?=E9=A6=96=E6=AC=A1=E6=8F=90=E4=BA=A4?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- LICENSE | 20 + README.md | 94 ++ pom.xml | 372 +++++ ruoyi-admin/Dockerfile | 19 + ruoyi-admin/pom.xml | 120 ++ .../main/java/com/ruoyi/RuoYiApplication.java | 32 + .../com/ruoyi/RuoYiServletInitializer.java | 18 + .../com/ruoyi/TestFlowableApplication.java | 17 + .../main/java/com/ruoyi/web/DevinUtil.java | 85 ++ .../controller/common/CaptchaController.java | 92 ++ .../controller/common/CommonController.java | 163 ++ .../controller/monitor/CacheController.java | 120 ++ .../controller/monitor/ServerController.java | 25 + .../monitor/SysLogininforController.java | 83 + .../monitor/SysOperlogController.java | 60 + .../monitor/SysUserOnlineController.java | 92 ++ .../system/SysConfigController.java | 117 ++ .../controller/system/SysDeptController.java | 125 ++ .../system/SysDictDataController.java | 106 ++ .../system/SysDictTypeController.java | 115 ++ .../controller/system/SysIndexController.java | 29 + .../controller/system/SysLoginController.java | 83 + .../controller/system/SysMenuController.java | 156 ++ .../system/SysNoticeController.java | 79 + .../controller/system/SysPostController.java | 123 ++ .../system/SysProfileController.java | 125 ++ .../system/SysRegisterController.java | 38 + .../controller/system/SysRoleController.java | 267 ++++ .../system/SysTreeDictController.java | 84 + .../system/SysTreeDictDataController.java | 87 ++ .../controller/system/SysUserController.java | 260 ++++ .../controller/system/TenantController.java | 96 ++ .../system/TenantPackageController.java | 83 + .../controller/tool/SwaggerController.java | 24 + .../web/controller/tool/TestController.java | 183 +++ .../ruoyi/web/core/config/SwaggerConfig.java | 125 ++ .../META-INF/spring-devtools.properties | 1 + .../src/main/resources/application-dev.yml | 63 + .../src/main/resources/application-druid.yml | 57 + .../src/main/resources/application-test.yml | 61 + .../src/main/resources/application.yml | 154 ++ ruoyi-admin/src/main/resources/banner.txt | 24 + ruoyi-admin/src/main/resources/ehcache.xml | 108 ++ .../main/resources/i18n/messages.properties | 37 + ruoyi-admin/src/main/resources/logback.xml | 93 ++ .../main/resources/mybatis/mybatis-config.xml | 20 + ...防安全重点部位情况登记表.xlsx | Bin 0 -> 9408 bytes .../META-INF/spring-devtools.properties | 1 + .../target/classes/application-dev.yml | 63 + .../target/classes/application-druid.yml | 57 + .../target/classes/application-test.yml | 61 + ruoyi-admin/target/classes/application.yml | 154 ++ ruoyi-admin/target/classes/banner.txt | 24 + .../classes/com/ruoyi/RuoYiApplication.class | Bin 0 -> 1458 bytes .../com/ruoyi/RuoYiServletInitializer.class | Bin 0 -> 880 bytes .../com/ruoyi/TestFlowableApplication.class | Bin 0 -> 1348 bytes .../classes/com/ruoyi/web/DevinUtil.class | Bin 0 -> 5546 bytes .../controller/common/CaptchaController.class | Bin 0 -> 3706 bytes .../controller/common/CommonController.class | Bin 0 -> 7648 bytes .../controller/monitor/ServerController.class | Bin 0 -> 1152 bytes .../monitor/SysLogininforController.class | Bin 0 -> 3901 bytes .../monitor/SysOperlogController.class | Bin 0 -> 3282 bytes .../monitor/SysUserOnlineController.class | Bin 0 -> 1796 bytes .../system/SysConfigController.class | Bin 0 -> 5520 bytes .../controller/system/SysDeptController.class | Bin 0 -> 6903 bytes .../system/SysDictDataController.class | Bin 0 -> 5239 bytes .../system/SysDictTypeController.class | Bin 0 -> 5618 bytes .../system/SysIndexController.class | Bin 0 -> 1088 bytes .../system/SysLoginController.class | Bin 0 -> 3415 bytes .../controller/system/SysMenuController.class | Bin 0 -> 8296 bytes .../system/SysNoticeController.class | Bin 0 -> 3604 bytes .../controller/system/SysPostController.class | Bin 0 -> 6294 bytes .../system/SysProfileController.class | Bin 0 -> 6138 bytes .../system/SysRegisterController.class | Bin 0 -> 1996 bytes .../controller/system/SysRoleController.class | Bin 0 -> 13373 bytes .../system/SysTreeDictController.class | Bin 0 -> 4130 bytes .../system/SysTreeDictDataController.class | Bin 0 -> 3038 bytes .../controller/system/SysUserController.class | Bin 0 -> 13170 bytes .../controller/system/TenantController.class | Bin 0 -> 6846 bytes .../system/TenantPackageController.class | Bin 0 -> 6141 bytes .../controller/tool/SwaggerController.class | Bin 0 -> 899 bytes .../web/controller/tool/TestController.class | Bin 0 -> 4611 bytes .../web/controller/tool/UserEntity.class | Bin 0 -> 1751 bytes .../ruoyi/web/core/config/SwaggerConfig.class | Bin 0 -> 6536 bytes ruoyi-admin/target/classes/ehcache.xml | 108 ++ .../target/classes/i18n/messages.properties | 37 + ruoyi-admin/target/classes/logback.xml | 93 ++ .../target/classes/mybatis/mybatis-config.xml | 20 + ...防安全重点部位情况登记表.xlsx | Bin 0 -> 9408 bytes ruoyi-common/pom.xml | 206 +++ .../ruoyi/common/annotation/Anonymous.java | 19 + .../ruoyi/common/annotation/DataScope.java | 33 + .../ruoyi/common/annotation/DataSource.java | 28 + .../ruoyi/common/annotation/DictFormat.java | 22 + .../com/ruoyi/common/annotation/Excel.java | 187 +++ .../com/ruoyi/common/annotation/Excels.java | 18 + .../java/com/ruoyi/common/annotation/Log.java | 46 + .../ruoyi/common/annotation/RateLimiter.java | 40 + .../ruoyi/common/annotation/RepeatSubmit.java | 31 + .../ruoyi/common/annotation/ReportExcel.java | 26 + .../com/ruoyi/common/config/RuoYiConfig.java | 150 ++ .../ruoyi/common/constant/CacheConstants.java | 54 + .../com/ruoyi/common/constant/Constants.java | 152 ++ .../common/constant/ErrorCodeConstants.java | 145 ++ .../ruoyi/common/constant/GenConstants.java | 117 ++ .../com/ruoyi/common/constant/HttpStatus.java | 93 ++ .../common/constant/ScheduleConstants.java | 50 + .../ruoyi/common/constant/SqlConstants.java | 29 + .../constant/SysErrorCodeConstants.java | 146 ++ .../ruoyi/common/constant/TreeConstants.java | 29 + .../ruoyi/common/constant/UserConstants.java | 78 + .../com/ruoyi/common/convert/DictConvert.java | 71 + .../com/ruoyi/common/convert/JsonConvert.java | 33 + .../core/controller/BaseController.java | 236 +++ .../ruoyi/common/core/domain/AjaxResult.java | 179 +++ .../ruoyi/common/core/domain/BaseEntity.java | 124 ++ .../common/core/domain/DictTreeEntity.java | 118 ++ .../java/com/ruoyi/common/core/domain/R.java | 102 ++ .../common/core/domain/TenantBaseDO.java | 20 + .../ruoyi/common/core/domain/TreeEntity.java | 79 + .../ruoyi/common/core/domain/TreeSelect.java | 77 + .../common/core/domain/entity/SysDept.java | 221 +++ .../core/domain/entity/SysDictData.java | 180 +++ .../core/domain/entity/SysDictType.java | 100 ++ .../common/core/domain/entity/SysMenu.java | 252 +++ .../common/core/domain/entity/SysRole.java | 261 ++++ .../common/core/domain/entity/SysUser.java | 345 +++++ .../common/core/domain/model/LoginBody.java | 69 + .../common/core/domain/model/LoginUser.java | 295 ++++ .../core/domain/model/RegisterBody.java | 11 + .../ruoyi/common/core/page/PageDomain.java | 101 ++ .../ruoyi/common/core/page/TableDataInfo.java | 116 ++ .../ruoyi/common/core/page/TableSupport.java | 56 + .../common/core/service/BaseServiceImpl.java | 12 + .../common/core/service/IBaseService.java | 6 + .../ruoyi/common/core/text/CharsetKit.java | 86 ++ .../com/ruoyi/common/core/text/Convert.java | 1000 ++++++++++++ .../ruoyi/common/core/text/StrFormatter.java | 92 ++ .../ruoyi/common/enums/BusinessStatus.java | 20 + .../com/ruoyi/common/enums/BusinessType.java | 59 + .../ruoyi/common/enums/CommonStatusEnum.java | 38 + .../ruoyi/common/enums/DataSourceType.java | 19 + .../com/ruoyi/common/enums/HttpMethod.java | 36 + .../com/ruoyi/common/enums/LimitType.java | 20 + .../com/ruoyi/common/enums/OperatorType.java | 24 + .../com/ruoyi/common/enums/RoleCodeEnum.java | 30 + .../com/ruoyi/common/enums/UserStatus.java | 30 + .../common/enums/bs/DeviceStatusEnum.java | 27 + .../ruoyi/common/enums/bs/TaskCodeEnum.java | 25 + .../enums/bs/TaskExceptionStatusEnum.java | 27 + .../enums/bs/TaskSettingStatusEnum.java | 26 + .../ruoyi/common/enums/bs/TaskTypeEnum.java | 27 + .../common/exception/DemoModeException.java | 15 + .../com/ruoyi/common/exception/ErrorCode.java | 26 + .../common/exception/GlobalException.java | 58 + .../common/exception/ServiceException.java | 74 + .../ruoyi/common/exception/UtilException.java | 26 + .../common/exception/base/BaseException.java | 97 ++ .../common/exception/file/FileException.java | 19 + .../FileNameLengthLimitExceededException.java | 16 + .../file/FileSizeLimitExceededException.java | 16 + .../file/InvalidExtensionException.java | 81 + .../common/exception/job/TaskException.java | 34 + .../exception/user/CaptchaException.java | 16 + .../user/CaptchaExpireException.java | 16 + .../common/exception/user/UserException.java | 18 + .../user/UserPasswordNotMatchException.java | 16 + ...UserPasswordRetryLimitExceedException.java | 16 + .../exception/util/ServiceExceptionUtil.java | 124 ++ .../ruoyi/common/filter/ApiRequestFilter.java | 30 + .../filter/PropertyPreExcludeFilter.java | 24 + .../ruoyi/common/filter/RepeatableFilter.java | 52 + .../filter/RepeatedlyRequestWrapper.java | 76 + .../com/ruoyi/common/filter/XssFilter.java | 75 + .../filter/XssHttpServletRequestWrapper.java | 111 ++ .../handler/JsonLongSetTypeHandler.java | 31 + .../common/mybatis/dataobject/BaseDO.java | 68 + .../handler/DefaultDBFieldHandler.java | 87 ++ .../common/mybatis/mapper/BaseMapperX.java | 94 ++ .../ruoyi/common/mybatis/pojo/PageParam.java | 30 + .../ruoyi/common/mybatis/pojo/PageResult.java | 42 + .../common/mybatis/pojo/SortingField.java | 56 + .../mybatis/query/LambdaQueryWrapperX.java | 135 ++ .../common/mybatis/query/QueryWrapperX.java | 140 ++ .../ruoyi/common/mybatis/util/JdbcUtils.java | 42 + .../common/mybatis/util/MyBatisUtils.java | 84 + .../java/com/ruoyi/common/utils/Arith.java | 114 ++ .../com/ruoyi/common/utils/DateUtils.java | 189 +++ .../com/ruoyi/common/utils/DictUtils.java | 200 +++ .../com/ruoyi/common/utils/EhcacheUtil.java | 78 + .../com/ruoyi/common/utils/ExcelUtils.java | 48 + .../com/ruoyi/common/utils/ExceptionUtil.java | 39 + .../com/ruoyi/common/utils/JsonUtils.java | 142 ++ .../java/com/ruoyi/common/utils/LogUtils.java | 18 + .../com/ruoyi/common/utils/MessageUtils.java | 26 + .../com/ruoyi/common/utils/NumberUtils.java | 16 + .../com/ruoyi/common/utils/ObjectUtils.java | 61 + .../com/ruoyi/common/utils/PageUtils.java | 48 + .../com/ruoyi/common/utils/SecurityUtils.java | 165 ++ .../com/ruoyi/common/utils/ServletUtils.java | 199 +++ .../com/ruoyi/common/utils/StringUtils.java | 609 ++++++++ .../java/com/ruoyi/common/utils/Threads.java | 99 ++ .../com/ruoyi/common/utils/TreeUtils.java | 177 +++ .../ruoyi/common/utils/ValidationUtils.java | 49 + .../ruoyi/common/utils/bean/BeanUtils.java | 110 ++ .../common/utils/bean/BeanValidators.java | 24 + .../common/utils/collection/ArrayUtils.java | 59 + .../utils/collection/CollectionUtils.java | 187 +++ .../utils/collection/IntArrayValuable.java | 15 + .../common/utils/collection/KeyValue.java | 20 + .../common/utils/collection/MapUtils.java | 65 + .../common/utils/collection/SetUtils.java | 18 + .../common/utils/file/FileTypeUtils.java | 76 + .../common/utils/file/FileUploadUtils.java | 200 +++ .../ruoyi/common/utils/file/FileUtils.java | 293 ++++ .../ruoyi/common/utils/file/ImageUtils.java | 123 ++ .../com/ruoyi/common/utils/file/IoUtils.java | 28 + .../common/utils/file/MimeTypeUtils.java | 59 + .../ruoyi/common/utils/html/EscapeUtil.java | 167 ++ .../ruoyi/common/utils/html/HTMLFilter.java | 570 +++++++ .../ruoyi/common/utils/http/HttpHelper.java | 55 + .../ruoyi/common/utils/http/HttpUtils.java | 274 ++++ .../ruoyi/common/utils/ip/AddressUtils.java | 56 + .../com/ruoyi/common/utils/ip/IpUtils.java | 264 ++++ .../common/utils/poi/ExcelHandlerAdapter.java | 19 + .../com/ruoyi/common/utils/poi/ExcelUtil.java | 1358 +++++++++++++++++ .../common/utils/reflect/ReflectUtils.java | 410 +++++ .../com/ruoyi/common/utils/sign/Base64.java | 291 ++++ .../com/ruoyi/common/utils/sign/Md5Utils.java | 67 + .../common/utils/spring/SpringUtils.java | 158 ++ .../com/ruoyi/common/utils/sql/SqlUtil.java | 61 + .../com/ruoyi/common/utils/uuid/IdUtils.java | 49 + .../java/com/ruoyi/common/utils/uuid/Seq.java | 86 ++ .../com/ruoyi/common/utils/uuid/UUID.java | 484 ++++++ .../main/java/com/ruoyi/common/xss/Xss.java | 27 + .../com/ruoyi/common/xss/XssValidator.java | 34 + .../ruoyi/common/annotation/Anonymous.class | Bin 0 -> 451 bytes .../ruoyi/common/annotation/DataScope.class | Bin 0 -> 573 bytes .../ruoyi/common/annotation/DataSource.class | Bin 0 -> 626 bytes .../ruoyi/common/annotation/DictFormat.class | Bin 0 -> 470 bytes .../common/annotation/Excel$ColumnType.class | Bin 0 -> 1368 bytes .../ruoyi/common/annotation/Excel$Type.class | Bin 0 -> 1323 bytes .../com/ruoyi/common/annotation/Excel.class | Bin 0 -> 2023 bytes .../com/ruoyi/common/annotation/Excels.class | Bin 0 -> 441 bytes .../com/ruoyi/common/annotation/Log.class | Bin 0 -> 850 bytes .../ruoyi/common/annotation/RateLimiter.class | Bin 0 -> 705 bytes .../common/annotation/RepeatSubmit.class | Bin 0 -> 635 bytes .../ruoyi/common/annotation/ReportExcel.class | Bin 0 -> 516 bytes .../com/ruoyi/common/config/RuoYiConfig.class | Bin 0 -> 2596 bytes .../common/constant/CacheConstants.class | Bin 0 -> 820 bytes .../com/ruoyi/common/constant/Constants.class | Bin 0 -> 1850 bytes .../common/constant/ErrorCodeConstants.class | Bin 0 -> 9828 bytes .../ruoyi/common/constant/GenConstants.class | Bin 0 -> 2753 bytes .../ruoyi/common/constant/HttpStatus.class | Bin 0 -> 897 bytes .../constant/ScheduleConstants$Status.class | Bin 0 -> 1449 bytes .../common/constant/ScheduleConstants.class | Bin 0 -> 719 bytes .../ruoyi/common/constant/SqlConstants.class | Bin 0 -> 507 bytes .../constant/SysErrorCodeConstants.class | Bin 0 -> 9872 bytes .../ruoyi/common/constant/TreeConstants.class | Bin 0 -> 673 bytes .../ruoyi/common/constant/UserConstants.class | Bin 0 -> 1145 bytes .../ruoyi/common/convert/DictConvert.class | Bin 0 -> 3703 bytes .../ruoyi/common/convert/JsonConvert.class | Bin 0 -> 1684 bytes .../core/controller/BaseController$1.class | Bin 0 -> 969 bytes .../core/controller/BaseController.class | Bin 0 -> 7222 bytes .../ruoyi/common/core/domain/AjaxResult.class | Bin 0 -> 2825 bytes .../ruoyi/common/core/domain/BaseEntity.class | Bin 0 -> 2691 bytes .../common/core/domain/DictTreeEntity.class | Bin 0 -> 9263 bytes .../com/ruoyi/common/core/domain/R.class | Bin 0 -> 3784 bytes .../common/core/domain/TenantBaseDO.class | Bin 0 -> 1593 bytes .../ruoyi/common/core/domain/TreeEntity.class | Bin 0 -> 1837 bytes .../ruoyi/common/core/domain/TreeSelect.class | Bin 0 -> 3503 bytes .../common/core/domain/entity/SysDept.class | Bin 0 -> 5385 bytes .../core/domain/entity/SysDictData.class | Bin 0 -> 4869 bytes .../core/domain/entity/SysDictType.class | Bin 0 -> 3380 bytes .../common/core/domain/entity/SysMenu.class | Bin 0 -> 5948 bytes .../common/core/domain/entity/SysRole.class | Bin 0 -> 6457 bytes .../common/core/domain/entity/SysUser.class | Bin 0 -> 8361 bytes .../common/core/domain/model/LoginBody.class | Bin 0 -> 1142 bytes .../common/core/domain/model/LoginUser.class | Bin 0 -> 5436 bytes .../core/domain/model/RegisterBody.class | Bin 0 -> 359 bytes .../ruoyi/common/core/page/PageDomain.class | Bin 0 -> 2295 bytes .../common/core/page/TableDataInfo.class | Bin 0 -> 2229 bytes .../ruoyi/common/core/page/TableSupport.class | Bin 0 -> 1516 bytes .../ruoyi/common/core/text/CharsetKit.class | Bin 0 -> 1823 bytes .../com/ruoyi/common/core/text/Convert.class | Bin 0 -> 13735 bytes .../ruoyi/common/core/text/StrFormatter.class | Bin 0 -> 1856 bytes .../ruoyi/common/enums/BusinessStatus.class | Bin 0 -> 1045 bytes .../com/ruoyi/common/enums/BusinessType.class | Bin 0 -> 1462 bytes .../ruoyi/common/enums/CommonStatusEnum.class | Bin 0 -> 2652 bytes .../ruoyi/common/enums/DataSourceType.class | Bin 0 -> 1045 bytes .../com/ruoyi/common/enums/HttpMethod.class | Bin 0 -> 2175 bytes .../com/ruoyi/common/enums/LimitType.class | Bin 0 -> 1008 bytes .../com/ruoyi/common/enums/OperatorType.class | Bin 0 -> 1084 bytes .../com/ruoyi/common/enums/RoleCodeEnum.class | Bin 0 -> 1715 bytes .../com/ruoyi/common/enums/UserStatus.class | Bin 0 -> 1496 bytes .../common/enums/bs/DeviceStatusEnum.class | Bin 0 -> 1642 bytes .../ruoyi/common/enums/bs/TaskCodeEnum.class | Bin 0 -> 1376 bytes .../enums/bs/TaskExceptionStatusEnum.class | Bin 0 -> 1619 bytes .../enums/bs/TaskSettingStatusEnum.class | Bin 0 -> 1520 bytes .../ruoyi/common/enums/bs/TaskTypeEnum.class | Bin 0 -> 1534 bytes .../common/exception/DemoModeException.class | Bin 0 -> 408 bytes .../ruoyi/common/exception/ErrorCode.class | Bin 0 -> 1810 bytes .../common/exception/GlobalException.class | Bin 0 -> 1023 bytes .../common/exception/ServiceException.class | Bin 0 -> 1315 bytes .../common/exception/UtilException.class | Bin 0 -> 816 bytes .../common/exception/base/BaseException.class | Bin 0 -> 1988 bytes .../common/exception/file/FileException.class | Bin 0 -> 633 bytes ...FileNameLengthLimitExceededException.class | Bin 0 -> 714 bytes .../file/FileSizeLimitExceededException.class | Bin 0 -> 671 bytes ...ption$InvalidFlashExtensionException.class | Bin 0 -> 772 bytes ...ption$InvalidImageExtensionException.class | Bin 0 -> 772 bytes ...ption$InvalidMediaExtensionException.class | Bin 0 -> 772 bytes ...ption$InvalidVideoExtensionException.class | Bin 0 -> 772 bytes .../file/InvalidExtensionException.class | Bin 0 -> 1903 bytes .../exception/job/TaskException$Code.class | Bin 0 -> 1442 bytes .../common/exception/job/TaskException.class | Bin 0 -> 1100 bytes .../exception/user/CaptchaException.class | Bin 0 -> 505 bytes .../user/CaptchaExpireException.class | Bin 0 -> 524 bytes .../common/exception/user/UserException.class | Bin 0 -> 633 bytes .../user/UserPasswordNotMatchException.class | Bin 0 -> 548 bytes ...serPasswordRetryLimitExceedException.class | Bin 0 -> 743 bytes .../exception/util/ServiceExceptionUtil.class | Bin 0 -> 4333 bytes .../common/filter/ApiRequestFilter.class | Bin 0 -> 806 bytes .../filter/PropertyPreExcludeFilter.class | Bin 0 -> 815 bytes .../common/filter/RepeatableFilter.class | Bin 0 -> 1702 bytes .../filter/RepeatedlyRequestWrapper$1.class | Bin 0 -> 1433 bytes .../filter/RepeatedlyRequestWrapper.class | Bin 0 -> 1945 bytes .../com/ruoyi/common/filter/XssFilter.class | Bin 0 -> 2853 bytes .../XssHttpServletRequestWrapper$1.class | Bin 0 -> 1419 bytes .../filter/XssHttpServletRequestWrapper.class | Bin 0 -> 2316 bytes .../handler/JsonLongSetTypeHandler$1.class | Bin 0 -> 591 bytes .../handler/JsonLongSetTypeHandler.class | Bin 0 -> 1306 bytes .../common/mybatis/dataobject/BaseDO.class | Bin 0 -> 4590 bytes .../handler/DefaultDBFieldHandler.class | Bin 0 -> 2683 bytes .../common/mybatis/mapper/BaseMapperX.class | Bin 0 -> 7409 bytes .../ruoyi/common/mybatis/pojo/PageParam.class | Bin 0 -> 2929 bytes .../common/mybatis/pojo/PageResult.class | Bin 0 -> 3437 bytes .../common/mybatis/pojo/SortingField.class | Bin 0 -> 1073 bytes .../mybatis/query/LambdaQueryWrapperX.class | Bin 0 -> 7228 bytes .../common/mybatis/query/QueryWrapperX.class | Bin 0 -> 6027 bytes .../ruoyi/common/mybatis/util/JdbcUtils.class | Bin 0 -> 1553 bytes .../common/mybatis/util/MyBatisUtils.class | Bin 0 -> 5527 bytes .../com/ruoyi/common/utils/Arith.class | Bin 0 -> 1944 bytes .../com/ruoyi/common/utils/DateUtils.class | Bin 0 -> 4979 bytes .../com/ruoyi/common/utils/DictUtils.class | Bin 0 -> 4824 bytes .../com/ruoyi/common/utils/EhcacheUtil.class | Bin 0 -> 2459 bytes .../com/ruoyi/common/utils/ExcelUtils.class | Bin 0 -> 3331 bytes .../ruoyi/common/utils/ExceptionUtil.class | Bin 0 -> 1353 bytes .../com/ruoyi/common/utils/JsonUtils.class | Bin 0 -> 5146 bytes .../com/ruoyi/common/utils/LogUtils.class | Bin 0 -> 668 bytes .../com/ruoyi/common/utils/MessageUtils.class | Bin 0 -> 955 bytes .../com/ruoyi/common/utils/NumberUtils.class | Bin 0 -> 612 bytes .../com/ruoyi/common/utils/ObjectUtils.class | Bin 0 -> 2416 bytes .../com/ruoyi/common/utils/PageUtils.class | Bin 0 -> 1825 bytes .../ruoyi/common/utils/SecurityUtils.class | Bin 0 -> 5350 bytes .../com/ruoyi/common/utils/ServletUtils.class | Bin 0 -> 4907 bytes .../com/ruoyi/common/utils/StringUtils.class | Bin 0 -> 8889 bytes .../com/ruoyi/common/utils/Threads.class | Bin 0 -> 2565 bytes .../com/ruoyi/common/utils/TreeUtils.class | Bin 0 -> 6357 bytes .../ruoyi/common/utils/ValidationUtils.class | Bin 0 -> 2262 bytes .../ruoyi/common/utils/bean/BeanUtils.class | Bin 0 -> 2747 bytes .../common/utils/bean/BeanValidators.class | Bin 0 -> 1227 bytes .../common/utils/collection/ArrayUtils.class | Bin 0 -> 2714 bytes .../utils/collection/CollectionUtils.class | Bin 0 -> 15008 bytes .../utils/collection/IntArrayValuable.class | Bin 0 -> 170 bytes .../common/utils/collection/KeyValue.class | Bin 0 -> 2566 bytes .../common/utils/collection/MapUtils.class | Bin 0 -> 4045 bytes .../common/utils/collection/SetUtils.class | Bin 0 -> 724 bytes .../common/utils/file/FileTypeUtils.class | Bin 0 -> 1374 bytes .../common/utils/file/FileUploadUtils.class | Bin 0 -> 5894 bytes .../ruoyi/common/utils/file/FileUtils.class | Bin 0 -> 6368 bytes .../ruoyi/common/utils/file/ImageUtils.class | Bin 0 -> 3585 bytes .../com/ruoyi/common/utils/file/IoUtils.class | Bin 0 -> 754 bytes .../common/utils/file/MimeTypeUtils.class | Bin 0 -> 1893 bytes .../ruoyi/common/utils/html/EscapeUtil.class | Bin 0 -> 2981 bytes .../ruoyi/common/utils/html/HTMLFilter.class | Bin 0 -> 13395 bytes .../ruoyi/common/utils/http/HttpHelper.class | Bin 0 -> 2492 bytes .../ruoyi/common/utils/http/HttpUtils$1.class | Bin 0 -> 231 bytes .../HttpUtils$TrustAnyHostnameVerifier.class | Bin 0 -> 930 bytes .../http/HttpUtils$TrustAnyTrustManager.class | Bin 0 -> 1211 bytes .../ruoyi/common/utils/http/HttpUtils.class | Bin 0 -> 8782 bytes .../ruoyi/common/utils/ip/AddressUtils.class | Bin 0 -> 2195 bytes .../com/ruoyi/common/utils/ip/IpUtils.class | Bin 0 -> 4169 bytes .../utils/poi/ExcelHandlerAdapter.class | Bin 0 -> 223 bytes .../ruoyi/common/utils/poi/ExcelUtil.class | Bin 0 -> 44902 bytes .../common/utils/reflect/ReflectUtils.class | Bin 0 -> 10872 bytes .../com/ruoyi/common/utils/sign/Base64.class | Bin 0 -> 4503 bytes .../ruoyi/common/utils/sign/Md5Utils.class | Bin 0 -> 2116 bytes .../common/utils/spring/SpringUtils.class | Bin 0 -> 3433 bytes .../com/ruoyi/common/utils/sql/SqlUtil.class | Bin 0 -> 1603 bytes .../com/ruoyi/common/utils/uuid/IdUtils.class | Bin 0 -> 713 bytes .../com/ruoyi/common/utils/uuid/Seq.class | Bin 0 -> 1944 bytes .../ruoyi/common/utils/uuid/UUID$Holder.class | Bin 0 -> 569 bytes .../com/ruoyi/common/utils/uuid/UUID.class | Bin 0 -> 6423 bytes .../classes/com/ruoyi/common/xss/Xss.class | Bin 0 -> 807 bytes .../com/ruoyi/common/xss/XssValidator.class | Bin 0 -> 1567 bytes ruoyi-flowable/README.md | 30 + ruoyi-flowable/img_3.png | Bin 0 -> 46288 bytes ruoyi-flowable/pom.xml | 63 + .../config/FlowableConfiguration.java | 44 + .../definition/BpmFormController.java | 81 + .../definition/BpmModelController.java | 93 ++ .../BpmProcessDefinitionController.java | 53 + .../BpmTaskAssignRuleController.java | 57 + .../definition/BpmUserGroupController.java | 83 + .../controller/oa/BpmOALeaveController.java | 62 + .../task/BpmActivityController.java | 35 + .../task/BpmProcessInstanceController.java | 57 + .../controller/task/BpmTaskController.java | 81 + .../convert/definition/BpmFormConvert.java | 34 + .../convert/definition/BpmModelConvert.java | 140 ++ .../BpmProcessDefinitionConvert.java | 82 + .../definition/BpmTaskAssignRuleConvert.java | 40 + .../definition/BpmUserGroupConvert.java | 37 + .../convert/message/BpmMessageConvert.java | 16 + .../convert/oa/BpmOALeaveConvert.java | 30 + .../convert/task/BpmActivityConvert.java | 29 + .../task/BpmProcessInstanceConvert.java | 114 ++ .../flowable/convert/task/BpmTaskConvert.java | 170 +++ .../flowable/convert/user/DeptConvert.java | 25 + .../flowable/convert/user/UserConvert.java | 26 + .../core/enums/DictTypeConstants.java | 13 + .../core/enums/ErrorCodeConstants.java | 65 + .../definition/BpmModelFormTypeEnum.java | 21 + .../definition/BpmTaskAssignRuleTypeEnum.java | 34 + .../definition/BpmTaskRuleScriptEnum.java | 30 + .../core/enums/message/BpmMessageEnum.java | 26 + .../BpmProcessInstanceDeleteReasonEnum.java | 58 + .../task/BpmProcessInstanceResultEnum.java | 48 + .../task/BpmProcessInstanceStatusEnum.java | 27 + .../core/enums/user/ErrorCodeConstants.java | 28 + .../enums/user/SysErrorCodeConstants.java | 146 ++ .../flowable/core/utils/FlowableUtils.java | 84 + .../flowable/core/web/FlowableWebFilter.java | 37 + .../dto/definition/BpmFormFieldRespDTO.java | 27 + .../definition/BpmModelMetaInfoRespDTO.java | 39 + .../BpmProcessDefinitionCreateReqDTO.java | 106 ++ ...eSendWhenProcessInstanceApproveReqDTO.java | 29 + ...geSendWhenProcessInstanceRejectReqDTO.java | 35 + .../BpmMessageSendWhenTaskCreatedReqDTO.java | 48 + .../dto/send/SmsSendSingleToUserReqDTO.java | 34 + .../task/BpmProcessInstanceCreateReqDTO.java | 35 + .../domain/dto/user/AdminUserRespDTO.java | 44 + .../flowable/domain/dto/user/DeptRespDTO.java | 37 + .../domain/entity/definition/BpmFormDO.java | 58 + .../definition/BpmProcessDefinitionExtDO.java | 91 ++ .../definition/BpmTaskAssignRuleDO.java | 84 + .../definition/BpmTaskMessageRuleDO.java | 5 + .../entity/definition/BpmUserGroupDO.java | 52 + .../domain/entity/oa/BpmOALeaveDO.java | 87 ++ .../domain/entity/task/BpmActivityDO.java | 104 ++ .../entity/task/BpmProcessInstanceExtDO.java | 93 ++ .../domain/entity/task/BpmTaskExtDO.java | 88 ++ .../domain/vo/activity/BpmActivityRespVO.java | 26 + .../domain/vo/form/BpmFormBaseVO.java | 26 + .../domain/vo/form/BpmFormCreateReqVO.java | 26 + .../domain/vo/form/BpmFormPageReqVO.java | 19 + .../domain/vo/form/BpmFormRespVO.java | 36 + .../domain/vo/form/BpmFormSimpleRespVO.java | 17 + .../domain/vo/form/BpmFormUpdateReqVO.java | 30 + .../domain/vo/group/BpmUserGroupBaseVO.java | 35 + .../vo/group/BpmUserGroupCreateReqVO.java | 15 + .../vo/group/BpmUserGroupPageReqVO.java | 29 + .../domain/vo/group/BpmUserGroupRespVO.java | 22 + .../vo/group/BpmUserGroupSimpleRespVO.java | 21 + .../vo/group/BpmUserGroupUpdateReqVO.java | 20 + .../BpmProcessInstanceCancelReqVO.java | 21 + .../BpmProcessInstanceCreateReqVO.java | 21 + .../BpmProcessInstanceMyPageReqVO.java | 39 + .../BpmProcessInstancePageItemRespVO.java | 55 + .../vo/instance/BpmProcessInstanceRespVO.java | 97 ++ .../domain/vo/model/BpmModeImportReqVO.java | 22 + .../domain/vo/model/BpmModelBaseVO.java | 41 + .../domain/vo/model/BpmModelCreateReqVO.java | 24 + .../vo/model/BpmModelPageItemRespVO.java | 49 + .../domain/vo/model/BpmModelPageReqVO.java | 26 + .../domain/vo/model/BpmModelRespVO.java | 26 + .../domain/vo/model/BpmModelUpdateReqVO.java | 40 + .../vo/model/BpmModelUpdateStateReqVO.java | 21 + .../domain/vo/oa/BpmOALeaveBaseVO.java | 33 + .../domain/vo/oa/BpmOALeaveCreateReqVO.java | 21 + .../domain/vo/oa/BpmOALeavePageReqVO.java | 33 + .../domain/vo/oa/BpmOALeaveRespVO.java | 33 + .../BpmProcessDefinitionListReqVO.java | 19 + .../BpmProcessDefinitionPageItemRespVO.java | 23 + .../BpmProcessDefinitionPageReqVO.java | 19 + .../process/BpmProcessDefinitionRespVO.java | 51 + .../vo/rule/BpmTaskAssignRuleBaseVO.java | 24 + .../vo/rule/BpmTaskAssignRuleCreateReqVO.java | 25 + .../vo/rule/BpmTaskAssignRuleRespVO.java | 29 + .../vo/rule/BpmTaskAssignRuleUpdateReqVO.java | 21 + .../flowable/domain/vo/task/BackTaskVo.java | 42 + .../domain/vo/task/BpmTaskApproveReqVO.java | 21 + .../vo/task/BpmTaskDonePageItemRespVO.java | 27 + .../domain/vo/task/BpmTaskDonePageReqVO.java | 31 + .../domain/vo/task/BpmTaskRejectReqVO.java | 21 + .../domain/vo/task/BpmTaskRespVO.java | 38 + .../vo/task/BpmTaskTodoPageItemRespVO.java | 61 + .../domain/vo/task/BpmTaskTodoPageReqVO.java | 31 + .../vo/task/BpmTaskUpdateAssigneeReqVO.java | 22 + .../flowable/domain/vo/task/FlowNodeVo.java | 33 + .../bpm/config/BpmCommonConfiguration.java | 19 + .../event/BpmProcessInstanceResultEvent.java | 43 + ...BpmProcessInstanceResultEventListener.java | 34 + ...pmProcessInstanceResultEventPublisher.java | 24 + .../config/BpmFlowableConfiguration.java | 46 + .../behavior/BpmActivityBehaviorFactory.java | 46 + .../BpmParallelMultiInstanceBehavior.java | 60 + .../behavior/BpmUserTaskActivityBehavior.java | 65 + .../behavior/script/BpmTaskAssignScript.java | 34 + .../BpmTaskAssignLeaderAbstractScript.java | 74 + .../impl/BpmTaskAssignLeaderX1Script.java | 27 + .../impl/BpmTaskAssignLeaderX2Script.java | 27 + .../impl/BpmTaskAssignStartUserScript.java | 40 + .../BpmProcessInstanceEventListener.java | 53 + .../core/listener/BpmTaskEventListener.java | 82 + .../mapper/definition/BpmFormMapper.java | 25 + .../BpmProcessDefinitionExtMapper.java | 21 + .../definition/BpmTaskAssignRuleMapper.java | 37 + .../mapper/definition/BpmUserGroupMapper.java | 32 + .../flowable/mapper/oa/BpmOALeaveMapper.java | 32 + .../task/BpmProcessInstanceExtMapper.java | 34 + .../mapper/task/BpmTaskExtMapper.java | 26 + .../service/definition/BpmFormService.java | 99 ++ .../service/definition/BpmModelService.java | 86 ++ .../BpmProcessDefinitionService.java | 160 ++ .../definition/BpmProcessInstanceApi.java | 24 + .../definition/BpmTaskAssignRuleService.java | 97 ++ .../definition/BpmUserGroupService.java | 84 + .../definition/impl/BpmFormServiceImpl.java | 137 ++ .../definition/impl/BpmModelServiceImpl.java | 311 ++++ .../impl/BpmProcessDefinitionServiceImpl.java | 288 ++++ .../impl/BpmProcessInstanceApiImpl.java | 29 + .../impl/BpmTaskAssignRuleServiceImpl.java | 361 +++++ .../impl/BpmUserGroupServiceImpl.java | 113 ++ .../service/message/BpmMessageService.java | 40 + .../message/impl/BpmMessageServiceImpl.java | 69 + .../service/oa/BpmOALeaveService.java | 59 + .../service/oa/BpmOALeaveServiceImpl.java | 99 ++ .../oa/listener/BpmOALeaveResultListener.java | 32 + .../service/task/BpmActivityService.java | 31 + .../task/BpmProcessInstanceService.java | 147 ++ .../flowable/service/task/BpmTaskService.java | 147 ++ .../task/impl/BpmActivityServiceImpl.java | 41 + .../impl/BpmProcessInstanceServiceImpl.java | 1 + .../service/task/impl/BpmTaskServiceImpl.java | 511 +++++++ .../flowable/service/user/AdminUserApi.java | 71 + .../ruoyi/flowable/service/user/DeptApi.java | 54 + .../flowable/service/user/DictDataApi.java | 23 + .../flowable/service/user/PermissionApi.java | 21 + .../ruoyi/flowable/service/user/PostApi.java | 21 + .../ruoyi/flowable/service/user/RoleApi.java | 21 + .../service/user/impl/AdminUserApiImpl.java | 82 + .../service/user/impl/DeptApiImpl.java | 70 + .../service/user/impl/DictDataApiImpl.java | 53 + .../service/user/impl/PermissionApiImpl.java | 26 + .../service/user/impl/PostApiImpl.java | 50 + .../service/user/impl/RoleApiImpl.java | 51 + .../mapper/flowable/BpmOALeaveMapper.xml | 64 + .../config/FlowableConfiguration.class | Bin 0 -> 2020 bytes .../definition/BpmFormController.class | Bin 0 -> 5176 bytes .../definition/BpmModelController.class | Bin 0 -> 5978 bytes .../BpmProcessDefinitionController.class | Bin 0 -> 2899 bytes .../BpmTaskAssignRuleController.class | Bin 0 -> 3415 bytes .../definition/BpmUserGroupController.class | Bin 0 -> 5437 bytes .../controller/oa/BpmOALeaveController.class | Bin 0 -> 4001 bytes .../task/BpmActivityController.class | Bin 0 -> 1989 bytes .../task/BpmProcessInstanceController.class | Bin 0 -> 4058 bytes .../controller/task/BpmTaskController.class | Bin 0 -> 5149 bytes .../convert/definition/BpmFormConvert.class | Bin 0 -> 1407 bytes .../definition/BpmFormConvertImpl.class | Bin 0 -> 6279 bytes .../convert/definition/BpmModelConvert.class | Bin 0 -> 10048 bytes .../definition/BpmModelConvertImpl.class | Bin 0 -> 3694 bytes .../BpmProcessDefinitionConvert.class | Bin 0 -> 7108 bytes .../BpmProcessDefinitionConvertImpl.class | Bin 0 -> 5375 bytes .../definition/BpmTaskAssignRuleConvert.class | Bin 0 -> 3957 bytes .../BpmTaskAssignRuleConvertImpl.class | Bin 0 -> 5272 bytes .../definition/BpmUserGroupConvert.class | Bin 0 -> 1550 bytes .../definition/BpmUserGroupConvertImpl.class | Bin 0 -> 5849 bytes .../convert/message/BpmMessageConvert.class | Bin 0 -> 800 bytes .../message/BpmMessageConvertImpl.class | Bin 0 -> 1534 bytes .../convert/oa/BpmOALeaveConvert.class | Bin 0 -> 1255 bytes .../convert/oa/BpmOALeaveConvertImpl.class | Bin 0 -> 4676 bytes .../convert/task/BpmActivityConvert.class | Bin 0 -> 1021 bytes .../convert/task/BpmActivityConvertImpl.class | Bin 0 -> 2292 bytes .../task/BpmProcessInstanceConvert.class | Bin 0 -> 9704 bytes .../task/BpmProcessInstanceConvertImpl.class | Bin 0 -> 7745 bytes .../convert/task/BpmTaskConvert.class | Bin 0 -> 13736 bytes .../convert/task/BpmTaskConvertImpl.class | Bin 0 -> 5783 bytes .../flowable/convert/user/DeptConvert.class | Bin 0 -> 955 bytes .../convert/user/DeptConvertImpl.class | Bin 0 -> 2236 bytes .../flowable/convert/user/UserConvert.class | Bin 0 -> 1007 bytes .../convert/user/UserConvertImpl.class | Bin 0 -> 3020 bytes .../core/enums/DictTypeConstants.class | Bin 0 -> 318 bytes .../core/enums/ErrorCodeConstants.class | Bin 0 -> 4581 bytes .../definition/BpmModelFormTypeEnum.class | Bin 0 -> 1706 bytes .../BpmTaskAssignRuleTypeEnum.class | Bin 0 -> 2215 bytes .../definition/BpmTaskRuleScriptEnum.class | Bin 0 -> 1848 bytes .../core/enums/message/BpmMessageEnum.class | Bin 0 -> 1555 bytes .../BpmProcessInstanceDeleteReasonEnum.class | Bin 0 -> 2598 bytes .../task/BpmProcessInstanceResultEnum.class | Bin 0 -> 2197 bytes .../task/BpmProcessInstanceStatusEnum.class | Bin 0 -> 1723 bytes .../core/enums/user/ErrorCodeConstants.class | Bin 0 -> 1341 bytes .../enums/user/SysErrorCodeConstants.class | Bin 0 -> 9881 bytes .../flowable/core/utils/FlowableUtils.class | Bin 0 -> 4033 bytes .../flowable/core/web/FlowableWebFilter.class | Bin 0 -> 1477 bytes .../dto/definition/BpmFormFieldRespDTO.class | Bin 0 -> 2077 bytes .../definition/BpmModelMetaInfoRespDTO.class | Bin 0 -> 3587 bytes .../BpmProcessDefinitionCreateReqDTO.class | Bin 0 -> 8848 bytes ...SendWhenProcessInstanceApproveReqDTO.class | Bin 0 -> 3269 bytes ...eSendWhenProcessInstanceRejectReqDTO.class | Bin 0 -> 3746 bytes .../BpmMessageSendWhenTaskCreatedReqDTO.class | Bin 0 -> 5330 bytes .../dto/send/SmsSendSingleToUserReqDTO.class | Bin 0 -> 3449 bytes .../task/BpmProcessInstanceCreateReqDTO.class | Bin 0 -> 3311 bytes .../domain/dto/user/AdminUserRespDTO.class | Bin 0 -> 3988 bytes .../domain/dto/user/DeptRespDTO.class | Bin 0 -> 3310 bytes .../BpmFormDO$BpmFormDOBuilder.class | Bin 0 -> 2494 bytes .../domain/entity/definition/BpmFormDO.class | Bin 0 -> 5147 bytes ...tDO$BpmProcessDefinitionExtDOBuilder.class | Bin 0 -> 3522 bytes .../BpmProcessDefinitionExtDO.class | Bin 0 -> 7684 bytes ...ignRuleDO$BpmTaskAssignRuleDOBuilder.class | Bin 0 -> 2741 bytes .../definition/BpmTaskAssignRuleDO.class | Bin 0 -> 5500 bytes .../definition/BpmTaskMessageRuleDO.class | Bin 0 -> 373 bytes ...BpmUserGroupDO$BpmUserGroupDOBuilder.class | Bin 0 -> 2464 bytes .../entity/definition/BpmUserGroupDO.class | Bin 0 -> 4785 bytes .../oa/BpmOALeaveDO$BpmOALeaveDOBuilder.class | Bin 0 -> 3183 bytes .../domain/entity/oa/BpmOALeaveDO.class | Bin 0 -> 7945 bytes .../domain/entity/task/BpmActivityDO.class | Bin 0 -> 10222 bytes .../entity/task/BpmProcessInstanceExtDO.class | Bin 0 -> 7063 bytes .../domain/entity/task/BpmTaskExtDO.class | Bin 0 -> 5884 bytes .../vo/activity/BpmActivityRespVO.class | Bin 0 -> 3811 bytes .../domain/vo/form/BpmFormBaseVO.class | Bin 0 -> 2876 bytes .../domain/vo/form/BpmFormCreateReqVO.class | Bin 0 -> 2840 bytes .../domain/vo/form/BpmFormPageReqVO.class | Bin 0 -> 1793 bytes .../domain/vo/form/BpmFormRespVO.class | Bin 0 -> 4000 bytes .../domain/vo/form/BpmFormSimpleRespVO.class | Bin 0 -> 2272 bytes .../domain/vo/form/BpmFormUpdateReqVO.class | Bin 0 -> 3398 bytes .../domain/vo/group/BpmUserGroupBaseVO.class | Bin 0 -> 3750 bytes .../vo/group/BpmUserGroupCreateReqVO.class | Bin 0 -> 1247 bytes .../vo/group/BpmUserGroupPageReqVO.class | Bin 0 -> 3045 bytes .../domain/vo/group/BpmUserGroupRespVO.class | Bin 0 -> 2414 bytes .../vo/group/BpmUserGroupSimpleRespVO.class | Bin 0 -> 2434 bytes .../vo/group/BpmUserGroupUpdateReqVO.class | Bin 0 -> 2040 bytes .../BpmProcessInstanceCancelReqVO.class | Bin 0 -> 2437 bytes .../BpmProcessInstanceCreateReqVO.class | Bin 0 -> 2860 bytes .../BpmProcessInstanceMyPageReqVO.class | Bin 0 -> 4738 bytes ...pmProcessInstancePageItemRespVO$Task.class | Bin 0 -> 2317 bytes .../BpmProcessInstancePageItemRespVO.class | Bin 0 -> 6521 bytes ...cessInstanceRespVO$ProcessDefinition.class | Bin 0 -> 6289 bytes .../BpmProcessInstanceRespVO$User.class | Bin 0 -> 3349 bytes .../instance/BpmProcessInstanceRespVO.class | Bin 0 -> 8016 bytes .../domain/vo/model/BpmModeImportReqVO.class | Bin 0 -> 2248 bytes .../domain/vo/model/BpmModelBaseVO.class | Bin 0 -> 5862 bytes .../domain/vo/model/BpmModelCreateReqVO.class | Bin 0 -> 2863 bytes ...odelPageItemRespVO$ProcessDefinition.class | Bin 0 -> 3565 bytes .../vo/model/BpmModelPageItemRespVO.class | Bin 0 -> 3779 bytes .../domain/vo/model/BpmModelPageReqVO.class | Bin 0 -> 2789 bytes .../domain/vo/model/BpmModelRespVO.class | Bin 0 -> 2838 bytes .../domain/vo/model/BpmModelUpdateReqVO.class | Bin 0 -> 6321 bytes .../vo/model/BpmModelUpdateStateReqVO.class | Bin 0 -> 2521 bytes .../domain/vo/oa/BpmOALeaveBaseVO.class | Bin 0 -> 3544 bytes .../domain/vo/oa/BpmOALeaveCreateReqVO.class | Bin 0 -> 1638 bytes .../domain/vo/oa/BpmOALeavePageReqVO.class | Bin 0 -> 3587 bytes .../domain/vo/oa/BpmOALeaveRespVO.class | Bin 0 -> 3781 bytes .../BpmProcessDefinitionListReqVO.class | Bin 0 -> 2052 bytes .../BpmProcessDefinitionPageItemRespVO.class | Bin 0 -> 2578 bytes .../BpmProcessDefinitionPageReqVO.class | Bin 0 -> 1880 bytes .../process/BpmProcessDefinitionRespVO.class | Bin 0 -> 8523 bytes .../vo/rule/BpmTaskAssignRuleBaseVO.class | Bin 0 -> 2656 bytes .../rule/BpmTaskAssignRuleCreateReqVO.class | Bin 0 -> 2686 bytes .../vo/rule/BpmTaskAssignRuleRespVO.class | Bin 0 -> 4145 bytes .../rule/BpmTaskAssignRuleUpdateReqVO.class | Bin 0 -> 2122 bytes .../flowable/domain/vo/task/BackTaskVo.class | Bin 0 -> 3688 bytes .../domain/vo/task/BpmTaskApproveReqVO.class | Bin 0 -> 2371 bytes .../vo/task/BpmTaskDonePageItemRespVO.class | Bin 0 -> 3651 bytes .../domain/vo/task/BpmTaskDonePageReqVO.class | Bin 0 -> 3065 bytes .../domain/vo/task/BpmTaskRejectReqVO.class | Bin 0 -> 2370 bytes .../domain/vo/task/BpmTaskRespVO$User.class | Bin 0 -> 3282 bytes .../domain/vo/task/BpmTaskRespVO.class | Bin 0 -> 2670 bytes ...skTodoPageItemRespVO$ProcessInstance.class | Bin 0 -> 5137 bytes .../vo/task/BpmTaskTodoPageItemRespVO.class | Bin 0 -> 5337 bytes .../domain/vo/task/BpmTaskTodoPageReqVO.class | Bin 0 -> 3065 bytes .../vo/task/BpmTaskUpdateAssigneeReqVO.class | Bin 0 -> 2647 bytes .../flowable/domain/vo/task/FlowNodeVo.class | Bin 0 -> 3625 bytes .../bpm/config/BpmCommonConfiguration.class | Bin 0 -> 1014 bytes .../event/BpmProcessInstanceResultEvent.class | Bin 0 -> 3427 bytes ...pmProcessInstanceResultEventListener.class | Bin 0 -> 1370 bytes ...mProcessInstanceResultEventPublisher.class | Bin 0 -> 1228 bytes .../config/BpmFlowableConfiguration.class | Bin 0 -> 3607 bytes .../behavior/BpmActivityBehaviorFactory.class | Bin 0 -> 3335 bytes .../BpmParallelMultiInstanceBehavior.class | Bin 0 -> 2427 bytes .../BpmUserTaskActivityBehavior.class | Bin 0 -> 4152 bytes .../behavior/script/BpmTaskAssignScript.class | Bin 0 -> 477 bytes .../BpmTaskAssignLeaderAbstractScript.class | Bin 0 -> 3290 bytes .../impl/BpmTaskAssignLeaderX1Script.class | Bin 0 -> 1315 bytes .../impl/BpmTaskAssignLeaderX2Script.class | Bin 0 -> 1315 bytes .../impl/BpmTaskAssignStartUserScript.class | Bin 0 -> 2177 bytes .../BpmProcessInstanceEventListener.class | Bin 0 -> 2669 bytes .../core/listener/BpmTaskEventListener.class | Bin 0 -> 4832 bytes .../mapper/definition/BpmFormMapper.class | Bin 0 -> 1519 bytes .../BpmProcessDefinitionExtMapper.class | Bin 0 -> 1538 bytes .../definition/BpmTaskAssignRuleMapper.class | Bin 0 -> 2154 bytes .../definition/BpmUserGroupMapper.class | Bin 0 -> 4700 bytes .../flowable/mapper/oa/BpmOALeaveMapper.class | Bin 0 -> 5105 bytes .../task/BpmProcessInstanceExtMapper.class | Bin 0 -> 6056 bytes .../mapper/task/BpmTaskExtMapper.class | Bin 0 -> 3699 bytes .../service/definition/BpmFormService.class | Bin 0 -> 2892 bytes .../service/definition/BpmModelService.class | Bin 0 -> 1234 bytes .../BpmProcessDefinitionService.class | Bin 0 -> 3590 bytes .../definition/BpmProcessInstanceApi.class | Bin 0 -> 425 bytes .../definition/BpmTaskAssignRuleService.class | Bin 0 -> 1606 bytes .../definition/BpmUserGroupService.class | Bin 0 -> 1459 bytes .../definition/impl/BpmFormServiceImpl.class | Bin 0 -> 6525 bytes .../definition/impl/BpmModelServiceImpl.class | Bin 0 -> 16115 bytes .../BpmProcessDefinitionServiceImpl.class | Bin 0 -> 15896 bytes .../impl/BpmProcessInstanceApiImpl.class | Bin 0 -> 1267 bytes .../impl/BpmTaskAssignRuleServiceImpl.class | Bin 0 -> 21526 bytes .../impl/BpmUserGroupServiceImpl.class | Bin 0 -> 6534 bytes .../service/message/BpmMessageService.class | Bin 0 -> 727 bytes .../message/impl/BpmMessageServiceImpl.class | Bin 0 -> 2969 bytes .../service/oa/BpmOALeaveService.class | Bin 0 -> 1298 bytes .../service/oa/BpmOALeaveServiceImpl.class | Bin 0 -> 5843 bytes .../listener/BpmOALeaveResultListener.class | Bin 0 -> 1486 bytes .../service/task/BpmActivityService.class | Bin 0 -> 501 bytes .../task/BpmProcessInstanceService.class | Bin 0 -> 3929 bytes .../service/task/BpmTaskService.class | Bin 0 -> 3514 bytes .../task/impl/BpmActivityServiceImpl.class | Bin 0 -> 2169 bytes .../impl/BpmProcessInstanceServiceImpl.class | Bin 0 -> 18141 bytes .../task/impl/BpmTaskServiceImpl$1.class | Bin 0 -> 3088 bytes .../task/impl/BpmTaskServiceImpl$2.class | Bin 0 -> 2673 bytes .../task/impl/BpmTaskServiceImpl.class | Bin 0 -> 30900 bytes .../flowable/service/user/AdminUserApi.class | Bin 0 -> 2073 bytes .../ruoyi/flowable/service/user/DeptApi.class | Bin 0 -> 1941 bytes .../flowable/service/user/DictDataApi.class | Bin 0 -> 292 bytes .../flowable/service/user/PermissionApi.class | Bin 0 -> 316 bytes .../ruoyi/flowable/service/user/PostApi.class | Bin 0 -> 242 bytes .../ruoyi/flowable/service/user/RoleApi.class | Bin 0 -> 242 bytes .../service/user/impl/AdminUserApiImpl.class | Bin 0 -> 6943 bytes .../service/user/impl/DeptApiImpl.class | Bin 0 -> 4602 bytes .../service/user/impl/DictDataApiImpl.class | Bin 0 -> 5763 bytes .../service/user/impl/PermissionApiImpl.class | Bin 0 -> 1057 bytes .../service/user/impl/PostApiImpl.class | Bin 0 -> 3594 bytes .../service/user/impl/RoleApiImpl.class | Bin 0 -> 3655 bytes .../mapper/flowable/BpmOALeaveMapper.xml | 64 + .../definition/BpmFormConvertImpl.java | 137 ++ .../definition/BpmModelConvertImpl.java | 72 + .../BpmProcessDefinitionConvertImpl.java | 107 ++ .../BpmTaskAssignRuleConvertImpl.java | 112 ++ .../definition/BpmUserGroupConvertImpl.java | 123 ++ .../message/BpmMessageConvertImpl.java | 38 + .../convert/oa/BpmOALeaveConvertImpl.java | 86 ++ .../convert/task/BpmActivityConvertImpl.java | 46 + .../task/BpmProcessInstanceConvertImpl.java | 188 +++ .../convert/task/BpmTaskConvertImpl.java | 156 ++ .../convert/user/DeptConvertImpl.java | 47 + .../convert/user/UserConvertImpl.java | 64 + ruoyi-framework/pom.xml | 82 + .../framework/aspectj/DataScopeAspect.java | 167 ++ .../framework/aspectj/DataSourceAspect.java | 72 + .../ruoyi/framework/aspectj/LogAspect.java | 200 +++ .../framework/aspectj/RateLimiterAspect.java | 80 + .../framework/config/ApplicationConfig.java | 30 + .../ruoyi/framework/config/CaptchaConfig.java | 83 + .../ruoyi/framework/config/DruidConfig.java | 126 ++ .../config/FastJson2JsonRedisSerializer.java | 48 + .../ruoyi/framework/config/FilterConfig.java | 58 + .../framework/config/KaptchaTextCreator.java | 68 + .../framework/config/MybatisPlusConfig.java | 70 + .../ruoyi/framework/config/RedisConfig.java | 69 + .../framework/config/ResourcesConfig.java | 71 + .../framework/config/SecurityConfig.java | 204 +++ .../ruoyi/framework/config/ServerConfig.java | 32 + .../framework/config/ThreadPoolConfig.java | 63 + .../config/properties/DruidProperties.java | 77 + .../properties/PermitAllUrlProperties.java | 72 + .../datasource/DynamicDataSource.java | 26 + .../DynamicDataSourceContextHolder.java | 45 + .../interceptor/RepeatSubmitInterceptor.java | 55 + .../impl/SameUrlDataInterceptor.java | 112 ++ .../ruoyi/framework/manager/AsyncManager.java | 55 + .../framework/manager/ShutdownManager.java | 41 + .../manager/factory/AsyncFactory.java | 102 ++ .../context/AuthenticationContextHolder.java | 28 + .../context/PermissionContextHolder.java | 27 + .../filter/JwtAuthenticationTokenFilter.java | 44 + .../handle/AuthenticationEntryPointImpl.java | 34 + .../handle/LogoutSuccessHandlerImpl.java | 50 + .../ruoyi/framework/web/domain/Server.java | 240 +++ .../framework/web/domain/server/Cpu.java | 101 ++ .../framework/web/domain/server/Jvm.java | 130 ++ .../framework/web/domain/server/Mem.java | 61 + .../framework/web/domain/server/Sys.java | 84 + .../framework/web/domain/server/SysFile.java | 114 ++ .../web/exception/GlobalExceptionHandler.java | 114 ++ .../web/service/DataScopeService.java | 163 ++ .../web/service/JimuReportTokenService.java | 47 + .../web/service/PermissionService.java | 169 ++ .../web/service/SysLoginService.java | 136 ++ .../web/service/SysPasswordService.java | 98 ++ .../web/service/SysPermissionService.java | 83 + .../web/service/SysRegisterService.java | 95 ++ .../framework/web/service/TokenService.java | 264 ++++ .../web/service/UserDetailsServiceImpl.java | 71 + .../framework/websocket/SemaphoreUtils.java | 58 + .../framework/websocket/WebSocketConfig.java | 20 + .../framework/websocket/WebSocketServer.java | 103 ++ .../framework/websocket/WebSocketUsers.java | 140 ++ .../framework/aspectj/DataScopeAspect.class | Bin 0 -> 5393 bytes .../framework/aspectj/DataSourceAspect.class | Bin 0 -> 2759 bytes .../ruoyi/framework/aspectj/LogAspect.class | Bin 0 -> 8520 bytes .../framework/config/ApplicationConfig.class | Bin 0 -> 1871 bytes .../framework/config/CaptchaConfig.class | Bin 0 -> 2435 bytes .../framework/config/DruidConfig$1.class | Bin 0 -> 2107 bytes .../ruoyi/framework/config/DruidConfig.class | Bin 0 -> 4773 bytes .../ruoyi/framework/config/FilterConfig.class | Bin 0 -> 2109 bytes .../framework/config/KaptchaTextCreator.class | Bin 0 -> 1711 bytes .../framework/config/MybatisPlusConfig.class | Bin 0 -> 2561 bytes .../framework/config/ResourcesConfig.class | Bin 0 -> 4065 bytes .../framework/config/SecurityConfig$1.class | Bin 0 -> 915 bytes .../framework/config/SecurityConfig.class | Bin 0 -> 13563 bytes .../ruoyi/framework/config/ServerConfig.class | Bin 0 -> 1423 bytes .../framework/config/ThreadPoolConfig$1.class | Bin 0 -> 1292 bytes .../framework/config/ThreadPoolConfig.class | Bin 0 -> 2310 bytes .../config/properties/DruidProperties.class | Bin 0 -> 2407 bytes .../properties/PermitAllUrlProperties.class | Bin 0 -> 5792 bytes .../datasource/DynamicDataSource.class | Bin 0 -> 1151 bytes .../DynamicDataSourceContextHolder.class | Bin 0 -> 1290 bytes .../interceptor/RepeatSubmitInterceptor.class | Bin 0 -> 2087 bytes .../impl/SameUrlDataInterceptor.class | Bin 0 -> 4216 bytes .../framework/manager/AsyncManager.class | Bin 0 -> 1438 bytes .../framework/manager/ShutdownManager.class | Bin 0 -> 1481 bytes .../manager/factory/AsyncFactory$1.class | Bin 0 -> 3197 bytes .../manager/factory/AsyncFactory$2.class | Bin 0 -> 1191 bytes .../manager/factory/AsyncFactory.class | Bin 0 -> 2071 bytes .../context/AuthenticationContextHolder.class | Bin 0 -> 1162 bytes .../context/PermissionContextHolder.class | Bin 0 -> 1153 bytes .../filter/JwtAuthenticationTokenFilter.class | Bin 0 -> 2876 bytes .../handle/AuthenticationEntryPointImpl.class | Bin 0 -> 1868 bytes .../handle/LogoutSuccessHandlerImpl.class | Bin 0 -> 2625 bytes .../ruoyi/framework/web/domain/Server.class | Bin 0 -> 8200 bytes .../framework/web/domain/server/Cpu.class | Bin 0 -> 1605 bytes .../framework/web/domain/server/Jvm.class | Bin 0 -> 2484 bytes .../framework/web/domain/server/Mem.class | Bin 0 -> 1091 bytes .../framework/web/domain/server/Sys.class | Bin 0 -> 1351 bytes .../framework/web/domain/server/SysFile.class | Bin 0 -> 1729 bytes .../exception/GlobalExceptionHandler.class | Bin 0 -> 5307 bytes .../web/service/DataScopeService.class | Bin 0 -> 9663 bytes .../web/service/JimuReportTokenService.class | Bin 0 -> 2272 bytes .../web/service/PermissionService.class | Bin 0 -> 3644 bytes .../web/service/SysLoginService.class | Bin 0 -> 5781 bytes .../web/service/SysPasswordService.class | Bin 0 -> 3837 bytes .../web/service/SysPermissionService.class | Bin 0 -> 2620 bytes .../web/service/SysRegisterService.class | Bin 0 -> 4175 bytes .../framework/web/service/TokenService.class | Bin 0 -> 7511 bytes .../web/service/UserDetailsServiceImpl.class | Bin 0 -> 3912 bytes .../framework/websocket/SemaphoreUtils.class | Bin 0 -> 1231 bytes .../framework/websocket/WebSocketConfig.class | Bin 0 -> 732 bytes .../framework/websocket/WebSocketServer.class | Bin 0 -> 3400 bytes .../framework/websocket/WebSocketUsers.class | Bin 0 -> 3695 bytes ruoyi-generator/pom.xml | 46 + .../com/ruoyi/generator/config/GenConfig.java | 73 + .../generator/controller/GenController.java | 194 +++ .../com/ruoyi/generator/domain/GenTable.java | 372 +++++ .../generator/domain/GenTableColumn.java | 373 +++++ .../mapper/GenTableColumnMapper.java | 60 + .../generator/mapper/GenTableMapper.java | 83 + .../service/GenTableColumnServiceImpl.java | 68 + .../service/GenTableServiceImpl.java | 498 ++++++ .../service/IGenTableColumnService.java | 44 + .../generator/service/IGenTableService.java | 121 ++ .../ruoyi/generator/util/GenTableUtils.java | 56 + .../com/ruoyi/generator/util/GenUtils.java | 257 ++++ .../generator/util/VelocityInitializer.java | 34 + .../ruoyi/generator/util/VelocityUtils.java | 405 +++++ .../src/main/resources/generator.yml | 10 + .../mapper/generator/GenTableColumnMapper.xml | 127 ++ .../mapper/generator/GenTableMapper.xml | 202 +++ .../mybatisplusvm/java/controller.java.vm | 129 ++ .../mybatisplusvm/java/domain.java.vm | 72 + .../mybatisplusvm/java/mapper.java.vm | 36 + .../mybatisplusvm/java/service.java.vm | 71 + .../mybatisplusvm/java/serviceImpl.java.vm | 174 +++ .../mybatisplusvm/java/sub-domain.java.vm | 76 + .../main/resources/mybatisplusvm/js/api.js.vm | 44 + .../main/resources/mybatisplusvm/sql/sql.vm | 22 + .../mybatisplusvm/vue/index-tree.vue.vm | 502 ++++++ .../resources/mybatisplusvm/vue/index.vue.vm | 598 ++++++++ .../mybatisplusvm/vue/v3/index-tree.vue.vm | 486 ++++++ .../mybatisplusvm/vue/v3/index.vue.vm | 596 ++++++++ .../resources/mybatisplusvm/vue/v3/readme.txt | 1 + .../resources/mybatisplusvm/xml/mapper.xml.vm | 109 ++ .../main/resources/vm/java/controller.java.vm | 108 ++ .../src/main/resources/vm/java/domain.java.vm | 105 ++ .../src/main/resources/vm/java/mapper.java.vm | 91 ++ .../main/resources/vm/java/service.java.vm | 61 + .../resources/vm/java/serviceImpl.java.vm | 169 ++ .../main/resources/vm/java/sub-domain.java.vm | 76 + .../src/main/resources/vm/js/api.js.vm | 44 + .../src/main/resources/vm/sql/sql.vm | 22 + .../main/resources/vm/vue/index-tree.vue.vm | 502 ++++++ .../src/main/resources/vm/vue/index.vue.vm | 609 ++++++++ .../resources/vm/vue/v3/index-tree.vue.vm | 486 ++++++ .../src/main/resources/vm/vue/v3/index.vue.vm | 596 ++++++++ .../src/main/resources/vm/vue/v3/readme.txt | 1 + .../src/main/resources/vm/xml/mapper.xml.vm | 135 ++ .../ruoyi/generator/config/GenConfig.class | Bin 0 -> 1587 bytes .../generator/controller/GenController.class | Bin 0 -> 7848 bytes .../com/ruoyi/generator/domain/GenTable.class | Bin 0 -> 7693 bytes .../generator/domain/GenTableColumn.class | Bin 0 -> 7471 bytes .../mapper/GenTableColumnMapper.class | Bin 0 -> 798 bytes .../generator/mapper/GenTableMapper.class | Bin 0 -> 979 bytes .../service/GenTableColumnServiceImpl.class | Bin 0 -> 1633 bytes .../service/GenTableServiceImpl.class | Bin 0 -> 20165 bytes .../service/IGenTableColumnService.class | Bin 0 -> 507 bytes .../generator/service/IGenTableService.class | Bin 0 -> 1295 bytes .../ruoyi/generator/util/GenTableUtils.class | Bin 0 -> 6121 bytes .../com/ruoyi/generator/util/GenUtils.class | Bin 0 -> 5739 bytes .../generator/util/VelocityInitializer.class | Bin 0 -> 1073 bytes .../ruoyi/generator/util/VelocityUtils.class | Bin 0 -> 10931 bytes ruoyi-generator/target/classes/generator.yml | 10 + .../mapper/generator/GenTableColumnMapper.xml | 127 ++ .../mapper/generator/GenTableMapper.xml | 202 +++ .../mybatisplusvm/java/controller.java.vm | 129 ++ .../classes/mybatisplusvm/java/domain.java.vm | 72 + .../classes/mybatisplusvm/java/mapper.java.vm | 36 + .../mybatisplusvm/java/service.java.vm | 71 + .../mybatisplusvm/java/serviceImpl.java.vm | 174 +++ .../mybatisplusvm/java/sub-domain.java.vm | 76 + .../target/classes/mybatisplusvm/js/api.js.vm | 44 + .../target/classes/mybatisplusvm/sql/sql.vm | 22 + .../mybatisplusvm/vue/index-tree.vue.vm | 502 ++++++ .../classes/mybatisplusvm/vue/index.vue.vm | 598 ++++++++ .../mybatisplusvm/vue/v3/index-tree.vue.vm | 486 ++++++ .../classes/mybatisplusvm/vue/v3/index.vue.vm | 596 ++++++++ .../classes/mybatisplusvm/vue/v3/readme.txt | 1 + .../classes/mybatisplusvm/xml/mapper.xml.vm | 109 ++ .../target/classes/vm/java/controller.java.vm | 108 ++ .../target/classes/vm/java/domain.java.vm | 105 ++ .../target/classes/vm/java/mapper.java.vm | 91 ++ .../target/classes/vm/java/service.java.vm | 61 + .../classes/vm/java/serviceImpl.java.vm | 169 ++ .../target/classes/vm/java/sub-domain.java.vm | 76 + .../target/classes/vm/js/api.js.vm | 44 + ruoyi-generator/target/classes/vm/sql/sql.vm | 22 + .../target/classes/vm/vue/index-tree.vue.vm | 502 ++++++ .../target/classes/vm/vue/index.vue.vm | 609 ++++++++ .../classes/vm/vue/v3/index-tree.vue.vm | 486 ++++++ .../target/classes/vm/vue/v3/index.vue.vm | 596 ++++++++ .../target/classes/vm/vue/v3/readme.txt | 1 + .../target/classes/vm/xml/mapper.xml.vm | 135 ++ ruoyi-plugin/pom.xml | 20 + .../ruoyi-springboot-starter-oss/pom.xml | 34 + .../plugin/oss/config/OssConfiguration.java | 69 + .../plugin/oss/controller/OssEndpoint.java | 128 ++ .../ruoyi/plugin/oss/core/OssTemplate.java | 242 +++ .../ruoyi/plugin/oss/props/OssProperties.java | 94 ++ .../main/resources/META-INF/spring.factories | 2 + .../target/classes/META-INF/spring.factories | 2 + .../plugin/oss/config/OssConfiguration.class | Bin 0 -> 1760 bytes .../plugin/oss/controller/OssEndpoint.class | Bin 0 -> 6175 bytes .../ruoyi/plugin/oss/core/OssTemplate.class | Bin 0 -> 9038 bytes .../plugin/oss/props/OssProperties.class | Bin 0 -> 4957 bytes .../ruoyi-springboot-starter-tenant/pom.xml | 39 + .../tenant/config/TenantConfiguration.java | 63 + .../tenant/config/TenantProperties.java | 40 + .../plugin/tenant/core/aop/TenantIgnore.java | 18 + .../tenant/core/aop/TenantIgnoreAspect.java | 32 + .../core/context/TenantContextHolder.java | 66 + .../plugin/tenant/core/db/TenantBaseDO.java | 21 + .../core/db/TenantDatabaseInterceptor.java | 66 + .../security/TenantSecurityWebFilter.java | 102 ++ .../plugin/tenant/core/util/TenantUtils.java | 49 + .../core/web/TenantContextWebFilter.java | 44 + .../main/resources/META-INF/spring.factories | 2 + .../target/classes/META-INF/spring.factories | 2 + .../tenant/config/TenantConfiguration.class | Bin 0 -> 3572 bytes .../tenant/config/TenantProperties.class | Bin 0 -> 2609 bytes .../plugin/tenant/core/aop/TenantIgnore.class | Bin 0 -> 449 bytes .../tenant/core/aop/TenantIgnoreAspect.class | Bin 0 -> 1577 bytes .../core/context/TenantContextHolder.class | Bin 0 -> 1669 bytes .../plugin/tenant/core/db/TenantBaseDO.class | Bin 0 -> 1612 bytes .../core/db/TenantDatabaseInterceptor.class | Bin 0 -> 3119 bytes .../security/TenantSecurityWebFilter.class | Bin 0 -> 3944 bytes .../plugin/tenant/core/util/TenantUtils.class | Bin 0 -> 1696 bytes .../core/web/TenantContextWebFilter.class | Bin 0 -> 2165 bytes ruoyi-quartz/pom.xml | 40 + .../ruoyi/quartz/config/ScheduleConfig.java | 57 + .../quartz/controller/SysJobController.java | 148 ++ .../controller/SysJobLogController.java | 82 + .../java/com/ruoyi/quartz/domain/SysJob.java | 171 +++ .../com/ruoyi/quartz/domain/SysJobLog.java | 155 ++ .../ruoyi/quartz/mapper/SysJobLogMapper.java | 64 + .../com/ruoyi/quartz/mapper/SysJobMapper.java | 67 + .../quartz/service/ISysJobLogService.java | 56 + .../ruoyi/quartz/service/ISysJobService.java | 102 ++ .../service/impl/SysJobLogServiceImpl.java | 87 ++ .../service/impl/SysJobServiceImpl.java | 261 ++++ .../com/ruoyi/quartz/task/CommonTask.java | 30 + .../ruoyi/quartz/util/AbstractQuartzJob.java | 107 ++ .../java/com/ruoyi/quartz/util/CronUtils.java | 63 + .../com/ruoyi/quartz/util/JobInvokeUtil.java | 182 +++ .../QuartzDisallowConcurrentExecution.java | 21 + .../ruoyi/quartz/util/QuartzJobExecution.java | 19 + .../com/ruoyi/quartz/util/ScheduleUtils.java | 139 ++ .../mapper/quartz/SysJobLogMapper.xml | 93 ++ .../resources/mapper/quartz/SysJobMapper.xml | 111 ++ .../quartz/controller/SysJobController.class | Bin 0 -> 6940 bytes .../controller/SysJobLogController.class | Bin 0 -> 3731 bytes .../com/ruoyi/quartz/domain/SysJob.class | Bin 0 -> 4875 bytes .../com/ruoyi/quartz/domain/SysJobLog.class | Bin 0 -> 3384 bytes .../ruoyi/quartz/mapper/SysJobLogMapper.class | Bin 0 -> 719 bytes .../ruoyi/quartz/mapper/SysJobMapper.class | Bin 0 -> 669 bytes .../quartz/service/ISysJobLogService.class | Bin 0 -> 608 bytes .../ruoyi/quartz/service/ISysJobService.class | Bin 0 -> 898 bytes .../service/impl/SysJobLogServiceImpl.class | Bin 0 -> 1775 bytes .../service/impl/SysJobServiceImpl.class | Bin 0 -> 5797 bytes .../com/ruoyi/quartz/task/CommonTask.class | Bin 0 -> 1501 bytes .../ruoyi/quartz/util/AbstractQuartzJob.class | Bin 0 -> 3878 bytes .../com/ruoyi/quartz/util/CronUtils.class | Bin 0 -> 1284 bytes .../com/ruoyi/quartz/util/JobInvokeUtil.class | Bin 0 -> 5489 bytes .../QuartzDisallowConcurrentExecution.class | Bin 0 -> 882 bytes .../quartz/util/QuartzJobExecution.class | Bin 0 -> 754 bytes .../com/ruoyi/quartz/util/ScheduleUtils.class | Bin 0 -> 6271 bytes .../classes/mapper/quartz/SysJobLogMapper.xml | 93 ++ .../classes/mapper/quartz/SysJobMapper.xml | 111 ++ ruoyi-system/pom.xml | 41 + .../system/convert/Device/DeviceConvert.java | 17 + .../system/convert/task/TaskConvert.java | 27 + .../system/convert/tenant/TenantConvert.java | 47 + .../convert/tenant/TenantPackageConvert.java | 37 + .../system/convert/user/UserConvert.java | 19 + .../com/ruoyi/system/domain/SysCache.java | 81 + .../com/ruoyi/system/domain/SysConfig.java | 116 ++ .../ruoyi/system/domain/SysLogininfor.java | 159 ++ .../com/ruoyi/system/domain/SysNotice.java | 106 ++ .../com/ruoyi/system/domain/SysOperLog.java | 259 ++++ .../java/com/ruoyi/system/domain/SysPost.java | 129 ++ .../com/ruoyi/system/domain/SysRoleDept.java | 46 + .../com/ruoyi/system/domain/SysRoleMenu.java | 46 + .../com/ruoyi/system/domain/SysTreeDict.java | 88 ++ .../ruoyi/system/domain/SysTreeDictData.java | 37 + .../ruoyi/system/domain/SysUserOnline.java | 113 ++ .../com/ruoyi/system/domain/SysUserPost.java | 46 + .../com/ruoyi/system/domain/SysUserRole.java | 46 + .../com/ruoyi/system/domain/TenantDO.java | 84 + .../ruoyi/system/domain/TenantPackageDO.java | 50 + .../com/ruoyi/system/domain/vo/ImageVo.java | 18 + .../com/ruoyi/system/domain/vo/MetaVo.java | 106 ++ .../com/ruoyi/system/domain/vo/NameIdVo.java | 17 + .../com/ruoyi/system/domain/vo/RouterVo.java | 148 ++ .../ruoyi/system/domain/vo/UserInfoVo.java | 18 + .../domain/vo/menu/MenuSimpleRespVO.java | 27 + .../vo/packages/TenantPackageBaseVO.java | 31 + .../vo/packages/TenantPackageCreateReqVO.java | 14 + .../vo/packages/TenantPackagePageReqVO.java | 32 + .../vo/packages/TenantPackageRespVO.java | 23 + .../packages/TenantPackageSimpleRespVO.java | 21 + .../vo/packages/TenantPackageUpdateReqVO.java | 21 + .../domain/vo/report/ReportDataRequestVo.java | 16 + .../vo/report/ReportDataResponseVo.java | 19 + .../domain/vo/report/ReportDownLoadVo.java | 18 + .../domain/vo/task/BatchSettingResqVo.java | 63 + .../system/domain/vo/task/PointResqVo.java | 68 + .../domain/vo/task/SettingUserInfoResqVo.java | 51 + .../system/domain/vo/tenant/TenantBaseVO.java | 48 + .../domain/vo/tenant/TenantCreateReqVO.java | 32 + .../domain/vo/tenant/TenantExcelVO.java | 38 + .../domain/vo/tenant/TenantExportReqVO.java | 31 + .../domain/vo/tenant/TenantPageReqVO.java | 36 + .../system/domain/vo/tenant/TenantRespVO.java | 23 + .../domain/vo/tenant/TenantUpdateReqVO.java | 21 + .../domain/vo/user/UserSimpleRespVO.java | 23 + .../ruoyi/system/enums/ReportTypeEnum.java | 64 + .../ruoyi/system/mapper/SysConfigMapper.java | 70 + .../ruoyi/system/mapper/SysDeptMapper.java | 122 ++ .../system/mapper/SysDictDataMapper.java | 99 ++ .../system/mapper/SysDictTypeMapper.java | 87 ++ .../system/mapper/SysLogininforMapper.java | 44 + .../ruoyi/system/mapper/SysMenuMapper.java | 136 ++ .../ruoyi/system/mapper/SysNoticeMapper.java | 62 + .../ruoyi/system/mapper/SysOperLogMapper.java | 50 + .../ruoyi/system/mapper/SysPostMapper.java | 104 ++ .../system/mapper/SysRoleDeptMapper.java | 48 + .../ruoyi/system/mapper/SysRoleMapper.java | 111 ++ .../system/mapper/SysRoleMenuMapper.java | 48 + .../system/mapper/SysTreeDictDataMapper.java | 33 + .../system/mapper/SysTreeDictMapper.java | 34 + .../ruoyi/system/mapper/SysUserMapper.java | 133 ++ .../system/mapper/SysUserPostMapper.java | 48 + .../system/mapper/SysUserRoleMapper.java | 66 + .../com/ruoyi/system/mapper/TenantMapper.java | 47 + .../system/mapper/TenantPackageMapper.java | 28 + .../system/service/ISysConfigService.java | 89 ++ .../ruoyi/system/service/ISysDeptService.java | 126 ++ .../system/service/ISysDictDataService.java | 62 + .../system/service/ISysDictTypeService.java | 100 ++ .../system/service/ISysLogininforService.java | 40 + .../ruoyi/system/service/ISysMenuService.java | 148 ++ .../system/service/ISysNoticeService.java | 60 + .../system/service/ISysOperLogService.java | 48 + .../ruoyi/system/service/ISysPostService.java | 101 ++ .../ruoyi/system/service/ISysRoleService.java | 179 +++ .../service/ISysTreeDictDataService.java | 107 ++ .../system/service/ISysTreeDictService.java | 70 + .../system/service/ISysUserOnlineService.java | 48 + .../ruoyi/system/service/ISysUserService.java | 227 +++ .../system/service/TenantPackageService.java | 74 + .../ruoyi/system/service/TenantService.java | 131 ++ .../service/handler/TenantInfoHandler.java | 22 + .../service/handler/TenantMenuHandler.java | 21 + .../service/impl/SysConfigServiceImpl.java | 235 +++ .../service/impl/SysDeptServiceImpl.java | 344 +++++ .../service/impl/SysDictDataServiceImpl.java | 113 ++ .../service/impl/SysDictTypeServiceImpl.java | 225 +++ .../impl/SysLogininforServiceImpl.java | 68 + .../service/impl/SysMenuServiceImpl.java | 562 +++++++ .../service/impl/SysNoticeServiceImpl.java | 92 ++ .../service/impl/SysOperLogServiceImpl.java | 76 + .../service/impl/SysPostServiceImpl.java | 180 +++ .../service/impl/SysRoleServiceImpl.java | 409 +++++ .../impl/SysUserOnlineServiceImpl.java | 96 ++ .../service/impl/SysUserServiceImpl.java | 518 +++++++ .../impl/TenantPackageServiceImpl.java | 115 ++ .../service/impl/TenantServiceImpl.java | 317 ++++ .../service/impl/TreeDictDataServiceImpl.java | 231 +++ .../service/impl/TreeDictServiceImpl.java | 124 ++ .../mapper/system/SysConfigMapper.xml | 112 ++ .../resources/mapper/system/SysDeptMapper.xml | 161 ++ .../mapper/system/SysDictDataMapper.xml | 124 ++ .../mapper/system/SysDictTypeMapper.xml | 105 ++ .../mapper/system/SysLogininforMapper.xml | 57 + .../resources/mapper/system/SysMenuMapper.xml | 201 +++ .../mapper/system/SysNoticeMapper.xml | 89 ++ .../mapper/system/SysOperLogMapper.xml | 83 + .../resources/mapper/system/SysPostMapper.xml | 122 ++ .../mapper/system/SysRoleDeptMapper.xml | 34 + .../resources/mapper/system/SysRoleMapper.xml | 153 ++ .../mapper/system/SysRoleMenuMapper.xml | 34 + .../mapper/system/SysTreeDictDataMapper.xml | 77 + .../mapper/system/SysTreeDictMapper.xml | 53 + .../resources/mapper/system/SysUserMapper.xml | 232 +++ .../mapper/system/SysUserPostMapper.xml | 34 + .../mapper/system/SysUserRoleMapper.xml | 44 + .../system/convert/Device/DeviceConvert.class | Bin 0 -> 474 bytes .../convert/Device/DeviceConvertImpl.class | Bin 0 -> 393 bytes .../system/convert/task/TaskConvert.class | Bin 0 -> 464 bytes .../system/convert/task/TaskConvertImpl.class | Bin 0 -> 379 bytes .../system/convert/tenant/TenantConvert.class | Bin 0 -> 2206 bytes .../convert/tenant/TenantConvertImpl.class | Bin 0 -> 5997 bytes .../convert/tenant/TenantPackageConvert.class | Bin 0 -> 1553 bytes .../tenant/TenantPackageConvertImpl.class | Bin 0 -> 6222 bytes .../system/convert/user/UserConvert.class | Bin 0 -> 681 bytes .../system/convert/user/UserConvertImpl.class | Bin 0 -> 2046 bytes .../com/ruoyi/system/domain/SysCache.class | Bin 0 -> 1758 bytes .../com/ruoyi/system/domain/SysConfig.class | Bin 0 -> 3589 bytes .../ruoyi/system/domain/SysLogininfor.class | Bin 0 -> 3350 bytes .../com/ruoyi/system/domain/SysNotice.class | Bin 0 -> 2913 bytes .../com/ruoyi/system/domain/SysOperLog.class | Bin 0 -> 5304 bytes .../com/ruoyi/system/domain/SysPost.class | Bin 0 -> 3743 bytes .../com/ruoyi/system/domain/SysRoleDept.class | Bin 0 -> 1293 bytes .../com/ruoyi/system/domain/SysRoleMenu.class | Bin 0 -> 1293 bytes .../com/ruoyi/system/domain/SysTreeDict.class | Bin 0 -> 7229 bytes .../ruoyi/system/domain/SysTreeDictData.class | Bin 0 -> 2551 bytes .../ruoyi/system/domain/SysUserOnline.class | Bin 0 -> 1982 bytes .../com/ruoyi/system/domain/SysUserPost.class | Bin 0 -> 1293 bytes .../com/ruoyi/system/domain/SysUserRole.class | Bin 0 -> 1293 bytes .../domain/TenantDO$TenantDOBuilder.class | Bin 0 -> 2754 bytes .../com/ruoyi/system/domain/TenantDO.class | Bin 0 -> 6856 bytes ...nantPackageDO$TenantPackageDOBuilder.class | Bin 0 -> 2282 bytes .../ruoyi/system/domain/TenantPackageDO.class | Bin 0 -> 4644 bytes .../com/ruoyi/system/domain/vo/ImageVo.class | Bin 0 -> 1882 bytes .../com/ruoyi/system/domain/vo/MetaVo.class | Bin 0 -> 1984 bytes .../com/ruoyi/system/domain/vo/NameIdVo.class | Bin 0 -> 1999 bytes .../com/ruoyi/system/domain/vo/RouterVo.class | Bin 0 -> 2924 bytes .../ruoyi/system/domain/vo/UserInfoVo.class | Bin 0 -> 2487 bytes .../domain/vo/menu/MenuSimpleRespVO.class | Bin 0 -> 3477 bytes .../vo/packages/TenantPackageBaseVO.class | Bin 0 -> 3658 bytes .../packages/TenantPackageCreateReqVO.class | Bin 0 -> 1258 bytes .../vo/packages/TenantPackagePageReqVO.class | Bin 0 -> 3501 bytes .../vo/packages/TenantPackageRespVO.class | Bin 0 -> 2431 bytes .../packages/TenantPackageSimpleRespVO.class | Bin 0 -> 2484 bytes .../packages/TenantPackageUpdateReqVO.class | Bin 0 -> 2063 bytes .../vo/report/ReportDataRequestVo.class | Bin 0 -> 2532 bytes .../vo/report/ReportDataResponseVo.class | Bin 0 -> 2726 bytes .../domain/vo/report/ReportDownLoadVo.class | Bin 0 -> 3377 bytes .../domain/vo/task/BatchSettingResqVo.class | Bin 0 -> 6018 bytes .../system/domain/vo/task/PointResqVo.class | Bin 0 -> 6337 bytes .../vo/task/SettingUserInfoResqVo.class | Bin 0 -> 5044 bytes .../domain/vo/tenant/TenantBaseVO.class | Bin 0 -> 5814 bytes .../domain/vo/tenant/TenantCreateReqVO.class | Bin 0 -> 2940 bytes .../domain/vo/tenant/TenantExcelVO.class | Bin 0 -> 4273 bytes .../domain/vo/tenant/TenantExportReqVO.class | Bin 0 -> 4025 bytes .../domain/vo/tenant/TenantPageReqVO.class | Bin 0 -> 4050 bytes .../domain/vo/tenant/TenantRespVO.class | Bin 0 -> 2384 bytes .../domain/vo/tenant/TenantUpdateReqVO.class | Bin 0 -> 2016 bytes .../domain/vo/user/UserSimpleRespVO.class | Bin 0 -> 2873 bytes .../ruoyi/system/enums/ReportTypeEnum.class | Bin 0 -> 2750 bytes .../ruoyi/system/mapper/SysConfigMapper.class | Bin 0 -> 786 bytes .../ruoyi/system/mapper/SysDeptMapper.class | Bin 0 -> 1749 bytes .../system/mapper/SysDictDataMapper.class | Bin 0 -> 1516 bytes .../system/mapper/SysDictTypeMapper.class | Bin 0 -> 1206 bytes .../system/mapper/SysLogininforMapper.class | Bin 0 -> 595 bytes .../ruoyi/system/mapper/SysMenuMapper.class | Bin 0 -> 4546 bytes .../ruoyi/system/mapper/SysNoticeMapper.class | Bin 0 -> 682 bytes .../system/mapper/SysOperLogMapper.class | Bin 0 -> 650 bytes .../ruoyi/system/mapper/SysPostMapper.class | Bin 0 -> 1327 bytes .../system/mapper/SysRoleDeptMapper.class | Bin 0 -> 652 bytes .../ruoyi/system/mapper/SysRoleMapper.class | Bin 0 -> 1565 bytes .../system/mapper/SysRoleMenuMapper.class | Bin 0 -> 643 bytes .../system/mapper/SysTreeDictDataMapper.class | Bin 0 -> 1151 bytes .../system/mapper/SysTreeDictMapper.class | Bin 0 -> 1103 bytes .../ruoyi/system/mapper/SysUserMapper.class | Bin 0 -> 3518 bytes .../system/mapper/SysUserPostMapper.class | Bin 0 -> 642 bytes .../system/mapper/SysUserRoleMapper.class | Bin 0 -> 920 bytes .../ruoyi/system/mapper/TenantMapper.class | Bin 0 -> 6859 bytes .../system/mapper/TenantPackageMapper.class | Bin 0 -> 4741 bytes .../system/service/ISysConfigService.class | Bin 0 -> 841 bytes .../system/service/ISysDeptService.class | Bin 0 -> 1689 bytes .../system/service/ISysDictDataService.class | Bin 0 -> 918 bytes .../system/service/ISysDictTypeService.class | Bin 0 -> 1418 bytes .../service/ISysLogininforService.class | Bin 0 -> 519 bytes .../system/service/ISysMenuService.class | Bin 0 -> 2166 bytes .../system/service/ISysNoticeService.class | Bin 0 -> 606 bytes .../system/service/ISysOperLogService.class | Bin 0 -> 574 bytes .../system/service/ISysPostService.class | Bin 0 -> 1142 bytes .../system/service/ISysRoleService.class | Bin 0 -> 2022 bytes .../service/ISysTreeDictDataService.class | Bin 0 -> 1414 bytes .../system/service/ISysTreeDictService.class | Bin 0 -> 968 bytes .../service/ISysUserOnlineService.class | Bin 0 -> 597 bytes .../system/service/ISysUserService.class | Bin 0 -> 2478 bytes .../system/service/TenantPackageService.class | Bin 0 -> 1157 bytes .../ruoyi/system/service/TenantService.class | Bin 0 -> 1798 bytes .../service/handler/TenantInfoHandler.class | Bin 0 -> 205 bytes .../service/handler/TenantMenuHandler.class | Bin 0 -> 245 bytes .../service/impl/SysConfigServiceImpl.class | Bin 0 -> 5668 bytes .../service/impl/SysDeptServiceImpl.class | Bin 0 -> 10385 bytes .../service/impl/SysDictDataServiceImpl.class | Bin 0 -> 3002 bytes .../service/impl/SysDictTypeServiceImpl.class | Bin 0 -> 7513 bytes .../impl/SysLogininforServiceImpl.class | Bin 0 -> 1753 bytes .../service/impl/SysMenuServiceImpl.class | Bin 0 -> 15679 bytes .../service/impl/SysNoticeServiceImpl.class | Bin 0 -> 1720 bytes .../service/impl/SysOperLogServiceImpl.class | Bin 0 -> 1603 bytes .../service/impl/SysPostServiceImpl.class | Bin 0 -> 3932 bytes .../service/impl/SysRoleServiceImpl.class | Bin 0 -> 13631 bytes .../impl/SysUserOnlineServiceImpl.class | Bin 0 -> 2682 bytes .../service/impl/SysUserServiceImpl.class | Bin 0 -> 17139 bytes .../impl/TenantPackageServiceImpl.class | Bin 0 -> 6552 bytes .../service/impl/TenantServiceImpl.class | Bin 0 -> 17733 bytes .../impl/TreeDictDataServiceImpl.class | Bin 0 -> 9520 bytes .../service/impl/TreeDictServiceImpl.class | Bin 0 -> 6907 bytes .../classes/mapper/system/SysConfigMapper.xml | 112 ++ .../classes/mapper/system/SysDeptMapper.xml | 161 ++ .../mapper/system/SysDictDataMapper.xml | 124 ++ .../mapper/system/SysDictTypeMapper.xml | 105 ++ .../mapper/system/SysLogininforMapper.xml | 57 + .../classes/mapper/system/SysMenuMapper.xml | 201 +++ .../classes/mapper/system/SysNoticeMapper.xml | 89 ++ .../mapper/system/SysOperLogMapper.xml | 83 + .../classes/mapper/system/SysPostMapper.xml | 122 ++ .../mapper/system/SysRoleDeptMapper.xml | 34 + .../classes/mapper/system/SysRoleMapper.xml | 153 ++ .../mapper/system/SysRoleMenuMapper.xml | 34 + .../mapper/system/SysTreeDictDataMapper.xml | 77 + .../mapper/system/SysTreeDictMapper.xml | 53 + .../classes/mapper/system/SysUserMapper.xml | 232 +++ .../mapper/system/SysUserPostMapper.xml | 34 + .../mapper/system/SysUserRoleMapper.xml | 44 + .../convert/Device/DeviceConvertImpl.java | 11 + .../system/convert/task/TaskConvertImpl.java | 11 + .../convert/tenant/TenantConvertImpl.java | 142 ++ .../tenant/TenantPackageConvertImpl.java | 137 ++ .../system/convert/user/UserConvertImpl.java | 43 + 注意事项.txt | 32 + 1273 files changed, 67817 insertions(+) create mode 100644 LICENSE create mode 100644 README.md create mode 100644 pom.xml create mode 100644 ruoyi-admin/Dockerfile create mode 100644 ruoyi-admin/pom.xml create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/RuoYiApplication.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/RuoYiServletInitializer.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/TestFlowableApplication.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/DevinUtil.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/common/CaptchaController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/common/CommonController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/CacheController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/ServerController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/SysLogininforController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/SysOperlogController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/SysUserOnlineController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysConfigController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysDeptController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysDictDataController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysDictTypeController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysIndexController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysLoginController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysMenuController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysNoticeController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysPostController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysProfileController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysRegisterController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysRoleController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysTreeDictController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysTreeDictDataController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysUserController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/TenantController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/TenantPackageController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/SwaggerController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/TestController.java create mode 100644 ruoyi-admin/src/main/java/com/ruoyi/web/core/config/SwaggerConfig.java create mode 100644 ruoyi-admin/src/main/resources/META-INF/spring-devtools.properties create mode 100644 ruoyi-admin/src/main/resources/application-dev.yml create mode 100644 ruoyi-admin/src/main/resources/application-druid.yml create mode 100644 ruoyi-admin/src/main/resources/application-test.yml create mode 100644 ruoyi-admin/src/main/resources/application.yml create mode 100644 ruoyi-admin/src/main/resources/banner.txt create mode 100644 ruoyi-admin/src/main/resources/ehcache.xml create mode 100644 ruoyi-admin/src/main/resources/i18n/messages.properties create mode 100644 ruoyi-admin/src/main/resources/logback.xml create mode 100644 ruoyi-admin/src/main/resources/mybatis/mybatis-config.xml create mode 100644 ruoyi-admin/src/main/resources/templatefile/消防安全重点部位情况登记表.xlsx create mode 100644 ruoyi-admin/target/classes/META-INF/spring-devtools.properties create mode 100644 ruoyi-admin/target/classes/application-dev.yml create mode 100644 ruoyi-admin/target/classes/application-druid.yml create mode 100644 ruoyi-admin/target/classes/application-test.yml create mode 100644 ruoyi-admin/target/classes/application.yml create mode 100644 ruoyi-admin/target/classes/banner.txt create mode 100644 ruoyi-admin/target/classes/com/ruoyi/RuoYiApplication.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/RuoYiServletInitializer.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/TestFlowableApplication.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/DevinUtil.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/common/CaptchaController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/common/CommonController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/monitor/ServerController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/monitor/SysLogininforController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/monitor/SysOperlogController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/monitor/SysUserOnlineController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysConfigController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysDeptController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysDictDataController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysDictTypeController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysIndexController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysLoginController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysMenuController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysNoticeController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysPostController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysProfileController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysRegisterController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysRoleController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysTreeDictController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysTreeDictDataController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysUserController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/TenantController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/system/TenantPackageController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/tool/SwaggerController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/tool/TestController.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/controller/tool/UserEntity.class create mode 100644 ruoyi-admin/target/classes/com/ruoyi/web/core/config/SwaggerConfig.class create mode 100644 ruoyi-admin/target/classes/ehcache.xml create mode 100644 ruoyi-admin/target/classes/i18n/messages.properties create mode 100644 ruoyi-admin/target/classes/logback.xml create mode 100644 ruoyi-admin/target/classes/mybatis/mybatis-config.xml create mode 100644 ruoyi-admin/target/classes/templatefile/消防安全重点部位情况登记表.xlsx create mode 100644 ruoyi-common/pom.xml create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/annotation/Anonymous.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/annotation/DataScope.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/annotation/DataSource.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/annotation/DictFormat.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/annotation/Excel.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/annotation/Excels.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/annotation/Log.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/annotation/RateLimiter.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/annotation/RepeatSubmit.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/annotation/ReportExcel.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/config/RuoYiConfig.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/constant/CacheConstants.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/constant/Constants.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/constant/ErrorCodeConstants.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/constant/GenConstants.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/constant/HttpStatus.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/constant/ScheduleConstants.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/constant/SqlConstants.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/constant/SysErrorCodeConstants.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/constant/TreeConstants.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/constant/UserConstants.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/convert/DictConvert.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/convert/JsonConvert.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/controller/BaseController.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/domain/AjaxResult.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/domain/BaseEntity.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/domain/DictTreeEntity.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/domain/R.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/domain/TenantBaseDO.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/domain/TreeEntity.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/domain/TreeSelect.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysDept.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysDictData.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysDictType.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysMenu.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysRole.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysUser.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/domain/model/LoginBody.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/domain/model/LoginUser.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/domain/model/RegisterBody.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/page/PageDomain.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/page/TableDataInfo.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/page/TableSupport.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/service/BaseServiceImpl.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/service/IBaseService.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/text/CharsetKit.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/text/Convert.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/core/text/StrFormatter.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/enums/BusinessStatus.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/enums/BusinessType.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/enums/CommonStatusEnum.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/enums/DataSourceType.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/enums/HttpMethod.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/enums/LimitType.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/enums/OperatorType.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/enums/RoleCodeEnum.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/enums/UserStatus.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/DeviceStatusEnum.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/TaskCodeEnum.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/TaskExceptionStatusEnum.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/TaskSettingStatusEnum.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/TaskTypeEnum.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/DemoModeException.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/ErrorCode.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/GlobalException.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/ServiceException.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/UtilException.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/base/BaseException.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/file/FileException.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/file/FileNameLengthLimitExceededException.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/file/FileSizeLimitExceededException.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/file/InvalidExtensionException.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/job/TaskException.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/user/CaptchaException.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/user/CaptchaExpireException.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/user/UserException.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/user/UserPasswordNotMatchException.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/user/UserPasswordRetryLimitExceedException.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/exception/util/ServiceExceptionUtil.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/filter/ApiRequestFilter.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/filter/PropertyPreExcludeFilter.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/filter/RepeatableFilter.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/filter/RepeatedlyRequestWrapper.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/filter/XssFilter.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/filter/XssHttpServletRequestWrapper.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/handler/JsonLongSetTypeHandler.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/mybatis/dataobject/BaseDO.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/mybatis/handler/DefaultDBFieldHandler.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/mybatis/mapper/BaseMapperX.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/mybatis/pojo/PageParam.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/mybatis/pojo/PageResult.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/mybatis/pojo/SortingField.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/mybatis/query/LambdaQueryWrapperX.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/mybatis/query/QueryWrapperX.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/mybatis/util/JdbcUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/mybatis/util/MyBatisUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/Arith.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/DateUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/DictUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/EhcacheUtil.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/ExcelUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/ExceptionUtil.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/JsonUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/LogUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/MessageUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/NumberUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/ObjectUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/PageUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/SecurityUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/ServletUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/StringUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/Threads.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/TreeUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/ValidationUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/bean/BeanUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/bean/BeanValidators.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/ArrayUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/CollectionUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/IntArrayValuable.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/KeyValue.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/MapUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/SetUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/file/FileTypeUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/file/FileUploadUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/file/FileUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/file/ImageUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/file/IoUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/file/MimeTypeUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/html/EscapeUtil.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/html/HTMLFilter.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/http/HttpHelper.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/http/HttpUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/ip/AddressUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/ip/IpUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/poi/ExcelHandlerAdapter.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/poi/ExcelUtil.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/reflect/ReflectUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/sign/Base64.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/sign/Md5Utils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/spring/SpringUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/sql/SqlUtil.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/uuid/IdUtils.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/uuid/Seq.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/utils/uuid/UUID.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/xss/Xss.java create mode 100644 ruoyi-common/src/main/java/com/ruoyi/common/xss/XssValidator.java create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/annotation/Anonymous.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/annotation/DataScope.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/annotation/DataSource.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/annotation/DictFormat.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/annotation/Excel$ColumnType.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/annotation/Excel$Type.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/annotation/Excel.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/annotation/Excels.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/annotation/Log.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/annotation/RateLimiter.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/annotation/RepeatSubmit.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/annotation/ReportExcel.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/config/RuoYiConfig.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/constant/CacheConstants.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/constant/Constants.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/constant/ErrorCodeConstants.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/constant/GenConstants.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/constant/HttpStatus.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/constant/ScheduleConstants$Status.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/constant/ScheduleConstants.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/constant/SqlConstants.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/constant/SysErrorCodeConstants.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/constant/TreeConstants.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/constant/UserConstants.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/convert/DictConvert.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/convert/JsonConvert.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/controller/BaseController$1.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/controller/BaseController.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/domain/AjaxResult.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/domain/BaseEntity.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/domain/DictTreeEntity.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/domain/R.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/domain/TenantBaseDO.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/domain/TreeEntity.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/domain/TreeSelect.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/domain/entity/SysDept.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/domain/entity/SysDictData.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/domain/entity/SysDictType.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/domain/entity/SysMenu.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/domain/entity/SysRole.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/domain/entity/SysUser.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/domain/model/LoginBody.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/domain/model/LoginUser.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/domain/model/RegisterBody.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/page/PageDomain.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/page/TableDataInfo.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/page/TableSupport.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/text/CharsetKit.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/text/Convert.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/core/text/StrFormatter.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/enums/BusinessStatus.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/enums/BusinessType.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/enums/CommonStatusEnum.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/enums/DataSourceType.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/enums/HttpMethod.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/enums/LimitType.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/enums/OperatorType.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/enums/RoleCodeEnum.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/enums/UserStatus.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/enums/bs/DeviceStatusEnum.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/enums/bs/TaskCodeEnum.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/enums/bs/TaskExceptionStatusEnum.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/enums/bs/TaskSettingStatusEnum.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/enums/bs/TaskTypeEnum.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/DemoModeException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/ErrorCode.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/GlobalException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/ServiceException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/UtilException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/base/BaseException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/file/FileException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/file/FileNameLengthLimitExceededException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/file/FileSizeLimitExceededException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/file/InvalidExtensionException$InvalidFlashExtensionException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/file/InvalidExtensionException$InvalidImageExtensionException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/file/InvalidExtensionException$InvalidMediaExtensionException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/file/InvalidExtensionException$InvalidVideoExtensionException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/file/InvalidExtensionException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/job/TaskException$Code.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/job/TaskException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/user/CaptchaException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/user/CaptchaExpireException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/user/UserException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/user/UserPasswordNotMatchException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/user/UserPasswordRetryLimitExceedException.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/exception/util/ServiceExceptionUtil.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/filter/ApiRequestFilter.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/filter/PropertyPreExcludeFilter.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/filter/RepeatableFilter.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/filter/RepeatedlyRequestWrapper$1.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/filter/RepeatedlyRequestWrapper.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/filter/XssFilter.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/filter/XssHttpServletRequestWrapper$1.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/filter/XssHttpServletRequestWrapper.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/handler/JsonLongSetTypeHandler$1.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/handler/JsonLongSetTypeHandler.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/mybatis/dataobject/BaseDO.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/mybatis/handler/DefaultDBFieldHandler.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/mybatis/mapper/BaseMapperX.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/mybatis/pojo/PageParam.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/mybatis/pojo/PageResult.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/mybatis/pojo/SortingField.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/mybatis/query/LambdaQueryWrapperX.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/mybatis/query/QueryWrapperX.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/mybatis/util/JdbcUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/mybatis/util/MyBatisUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/Arith.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/DateUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/DictUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/EhcacheUtil.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/ExcelUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/ExceptionUtil.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/JsonUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/LogUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/MessageUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/NumberUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/ObjectUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/PageUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/SecurityUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/ServletUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/StringUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/Threads.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/TreeUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/ValidationUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/bean/BeanUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/bean/BeanValidators.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/collection/ArrayUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/collection/CollectionUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/collection/IntArrayValuable.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/collection/KeyValue.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/collection/MapUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/collection/SetUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/file/FileTypeUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/file/FileUploadUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/file/FileUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/file/ImageUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/file/IoUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/file/MimeTypeUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/html/EscapeUtil.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/html/HTMLFilter.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/http/HttpHelper.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/http/HttpUtils$1.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/http/HttpUtils$TrustAnyHostnameVerifier.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/http/HttpUtils$TrustAnyTrustManager.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/http/HttpUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/ip/AddressUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/ip/IpUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/poi/ExcelHandlerAdapter.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/poi/ExcelUtil.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/reflect/ReflectUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/sign/Base64.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/sign/Md5Utils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/spring/SpringUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/sql/SqlUtil.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/uuid/IdUtils.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/uuid/Seq.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/uuid/UUID$Holder.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/utils/uuid/UUID.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/xss/Xss.class create mode 100644 ruoyi-common/target/classes/com/ruoyi/common/xss/XssValidator.class create mode 100644 ruoyi-flowable/README.md create mode 100644 ruoyi-flowable/img_3.png create mode 100644 ruoyi-flowable/pom.xml create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/config/FlowableConfiguration.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmFormController.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmModelController.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmProcessDefinitionController.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmTaskAssignRuleController.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmUserGroupController.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/oa/BpmOALeaveController.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/task/BpmActivityController.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/task/BpmProcessInstanceController.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/task/BpmTaskController.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmFormConvert.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmModelConvert.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmProcessDefinitionConvert.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmTaskAssignRuleConvert.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmUserGroupConvert.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/message/BpmMessageConvert.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/oa/BpmOALeaveConvert.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/task/BpmActivityConvert.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/task/BpmProcessInstanceConvert.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/task/BpmTaskConvert.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/user/DeptConvert.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/user/UserConvert.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/DictTypeConstants.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/ErrorCodeConstants.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/definition/BpmModelFormTypeEnum.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/definition/BpmTaskAssignRuleTypeEnum.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/definition/BpmTaskRuleScriptEnum.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/message/BpmMessageEnum.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceDeleteReasonEnum.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceResultEnum.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceStatusEnum.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/user/ErrorCodeConstants.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/user/SysErrorCodeConstants.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/utils/FlowableUtils.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/web/FlowableWebFilter.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/definition/BpmFormFieldRespDTO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/definition/BpmModelMetaInfoRespDTO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/definition/BpmProcessDefinitionCreateReqDTO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenProcessInstanceApproveReqDTO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenProcessInstanceRejectReqDTO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenTaskCreatedReqDTO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/send/SmsSendSingleToUserReqDTO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/task/BpmProcessInstanceCreateReqDTO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/user/AdminUserRespDTO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/user/DeptRespDTO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmFormDO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmProcessDefinitionExtDO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmTaskAssignRuleDO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmTaskMessageRuleDO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmUserGroupDO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/oa/BpmOALeaveDO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/task/BpmActivityDO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/task/BpmProcessInstanceExtDO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/task/BpmTaskExtDO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/activity/BpmActivityRespVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormBaseVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormCreateReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormPageReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormRespVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormSimpleRespVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormUpdateReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupBaseVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupCreateReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupPageReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupRespVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupSimpleRespVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupUpdateReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceCancelReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceCreateReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceMyPageReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstancePageItemRespVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceRespVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModeImportReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelBaseVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelCreateReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelPageItemRespVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelPageReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelRespVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelUpdateReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelUpdateStateReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveBaseVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveCreateReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/oa/BpmOALeavePageReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveRespVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionListReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionPageItemRespVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionPageReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionRespVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleBaseVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleCreateReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleRespVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleUpdateReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BackTaskVo.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskApproveReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskDonePageItemRespVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskDonePageReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskRejectReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskRespVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageItemRespVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskUpdateAssigneeReqVO.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/FlowNodeVo.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/bpm/config/BpmCommonConfiguration.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEvent.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEventListener.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEventPublisher.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/config/BpmFlowableConfiguration.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/BpmActivityBehaviorFactory.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/BpmParallelMultiInstanceBehavior.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/BpmUserTaskActivityBehavior.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/BpmTaskAssignScript.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderAbstractScript.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderX1Script.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderX2Script.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignStartUserScript.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/listener/BpmProcessInstanceEventListener.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/listener/BpmTaskEventListener.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/definition/BpmFormMapper.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/definition/BpmProcessDefinitionExtMapper.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/definition/BpmTaskAssignRuleMapper.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/definition/BpmUserGroupMapper.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/oa/BpmOALeaveMapper.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/task/BpmProcessInstanceExtMapper.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/task/BpmTaskExtMapper.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmFormService.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmModelService.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmProcessDefinitionService.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmProcessInstanceApi.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmTaskAssignRuleService.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmUserGroupService.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmFormServiceImpl.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmModelServiceImpl.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmProcessDefinitionServiceImpl.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmProcessInstanceApiImpl.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmTaskAssignRuleServiceImpl.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmUserGroupServiceImpl.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/message/BpmMessageService.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/message/impl/BpmMessageServiceImpl.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/oa/BpmOALeaveService.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/oa/BpmOALeaveServiceImpl.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/oa/listener/BpmOALeaveResultListener.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/BpmActivityService.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/BpmProcessInstanceService.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/BpmTaskService.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/impl/BpmActivityServiceImpl.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/impl/BpmProcessInstanceServiceImpl.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/impl/BpmTaskServiceImpl.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/AdminUserApi.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/DeptApi.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/DictDataApi.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/PermissionApi.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/PostApi.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/RoleApi.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/AdminUserApiImpl.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/DeptApiImpl.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/DictDataApiImpl.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/PermissionApiImpl.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/PostApiImpl.java create mode 100644 ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/RoleApiImpl.java create mode 100644 ruoyi-flowable/src/main/resources/mapper/flowable/BpmOALeaveMapper.xml create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/config/FlowableConfiguration.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/controller/definition/BpmFormController.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/controller/definition/BpmModelController.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/controller/definition/BpmProcessDefinitionController.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/controller/definition/BpmTaskAssignRuleController.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/controller/definition/BpmUserGroupController.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/controller/oa/BpmOALeaveController.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/controller/task/BpmActivityController.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/controller/task/BpmProcessInstanceController.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/controller/task/BpmTaskController.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmFormConvert.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmFormConvertImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmModelConvert.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmModelConvertImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmProcessDefinitionConvert.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmProcessDefinitionConvertImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmTaskAssignRuleConvert.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmTaskAssignRuleConvertImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmUserGroupConvert.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmUserGroupConvertImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/message/BpmMessageConvert.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/message/BpmMessageConvertImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/oa/BpmOALeaveConvert.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/oa/BpmOALeaveConvertImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/task/BpmActivityConvert.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/task/BpmActivityConvertImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/task/BpmProcessInstanceConvert.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/task/BpmProcessInstanceConvertImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/task/BpmTaskConvert.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/task/BpmTaskConvertImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/user/DeptConvert.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/user/DeptConvertImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/user/UserConvert.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/user/UserConvertImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/DictTypeConstants.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/ErrorCodeConstants.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/definition/BpmModelFormTypeEnum.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/definition/BpmTaskAssignRuleTypeEnum.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/definition/BpmTaskRuleScriptEnum.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/message/BpmMessageEnum.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceDeleteReasonEnum.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceResultEnum.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceStatusEnum.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/user/ErrorCodeConstants.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/user/SysErrorCodeConstants.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/core/utils/FlowableUtils.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/core/web/FlowableWebFilter.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/definition/BpmFormFieldRespDTO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/definition/BpmModelMetaInfoRespDTO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/definition/BpmProcessDefinitionCreateReqDTO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenProcessInstanceApproveReqDTO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenProcessInstanceRejectReqDTO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenTaskCreatedReqDTO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/send/SmsSendSingleToUserReqDTO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/task/BpmProcessInstanceCreateReqDTO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/user/AdminUserRespDTO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/user/DeptRespDTO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/definition/BpmFormDO$BpmFormDOBuilder.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/definition/BpmFormDO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/definition/BpmProcessDefinitionExtDO$BpmProcessDefinitionExtDOBuilder.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/definition/BpmProcessDefinitionExtDO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/definition/BpmTaskAssignRuleDO$BpmTaskAssignRuleDOBuilder.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/definition/BpmTaskAssignRuleDO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/definition/BpmTaskMessageRuleDO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/definition/BpmUserGroupDO$BpmUserGroupDOBuilder.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/definition/BpmUserGroupDO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/oa/BpmOALeaveDO$BpmOALeaveDOBuilder.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/oa/BpmOALeaveDO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/task/BpmActivityDO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/task/BpmProcessInstanceExtDO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/task/BpmTaskExtDO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/activity/BpmActivityRespVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/form/BpmFormBaseVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/form/BpmFormCreateReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/form/BpmFormPageReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/form/BpmFormRespVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/form/BpmFormSimpleRespVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/form/BpmFormUpdateReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/group/BpmUserGroupBaseVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/group/BpmUserGroupCreateReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/group/BpmUserGroupPageReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/group/BpmUserGroupRespVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/group/BpmUserGroupSimpleRespVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/group/BpmUserGroupUpdateReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceCancelReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceCreateReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceMyPageReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstancePageItemRespVO$Task.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstancePageItemRespVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceRespVO$ProcessDefinition.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceRespVO$User.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceRespVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModeImportReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelBaseVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelCreateReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelPageItemRespVO$ProcessDefinition.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelPageItemRespVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelPageReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelRespVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelUpdateReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelUpdateStateReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveBaseVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveCreateReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/oa/BpmOALeavePageReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveRespVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionListReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionPageItemRespVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionPageReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionRespVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleBaseVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleCreateReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleRespVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleUpdateReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BackTaskVo.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskApproveReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskDonePageItemRespVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskDonePageReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskRejectReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskRespVO$User.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskRespVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageItemRespVO$ProcessInstance.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageItemRespVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskUpdateAssigneeReqVO.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/FlowNodeVo.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/bpm/config/BpmCommonConfiguration.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEvent.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEventListener.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEventPublisher.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/config/BpmFlowableConfiguration.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/BpmActivityBehaviorFactory.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/BpmParallelMultiInstanceBehavior.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/BpmUserTaskActivityBehavior.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/script/BpmTaskAssignScript.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderAbstractScript.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderX1Script.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderX2Script.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignStartUserScript.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/listener/BpmProcessInstanceEventListener.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/listener/BpmTaskEventListener.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/mapper/definition/BpmFormMapper.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/mapper/definition/BpmProcessDefinitionExtMapper.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/mapper/definition/BpmTaskAssignRuleMapper.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/mapper/definition/BpmUserGroupMapper.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/mapper/oa/BpmOALeaveMapper.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/mapper/task/BpmProcessInstanceExtMapper.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/mapper/task/BpmTaskExtMapper.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/BpmFormService.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/BpmModelService.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/BpmProcessDefinitionService.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/BpmProcessInstanceApi.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/BpmTaskAssignRuleService.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/BpmUserGroupService.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/impl/BpmFormServiceImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/impl/BpmModelServiceImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/impl/BpmProcessDefinitionServiceImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/impl/BpmProcessInstanceApiImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/impl/BpmTaskAssignRuleServiceImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/impl/BpmUserGroupServiceImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/message/BpmMessageService.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/message/impl/BpmMessageServiceImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/oa/BpmOALeaveService.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/oa/BpmOALeaveServiceImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/oa/listener/BpmOALeaveResultListener.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/task/BpmActivityService.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/task/BpmProcessInstanceService.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/task/BpmTaskService.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/task/impl/BpmActivityServiceImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/task/impl/BpmProcessInstanceServiceImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/task/impl/BpmTaskServiceImpl$1.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/task/impl/BpmTaskServiceImpl$2.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/task/impl/BpmTaskServiceImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/AdminUserApi.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/DeptApi.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/DictDataApi.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/PermissionApi.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/PostApi.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/RoleApi.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/impl/AdminUserApiImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/impl/DeptApiImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/impl/DictDataApiImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/impl/PermissionApiImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/impl/PostApiImpl.class create mode 100644 ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/impl/RoleApiImpl.class create mode 100644 ruoyi-flowable/target/classes/mapper/flowable/BpmOALeaveMapper.xml create mode 100644 ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmFormConvertImpl.java create mode 100644 ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmModelConvertImpl.java create mode 100644 ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmProcessDefinitionConvertImpl.java create mode 100644 ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmTaskAssignRuleConvertImpl.java create mode 100644 ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmUserGroupConvertImpl.java create mode 100644 ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/message/BpmMessageConvertImpl.java create mode 100644 ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/oa/BpmOALeaveConvertImpl.java create mode 100644 ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/task/BpmActivityConvertImpl.java create mode 100644 ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/task/BpmProcessInstanceConvertImpl.java create mode 100644 ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/task/BpmTaskConvertImpl.java create mode 100644 ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/user/DeptConvertImpl.java create mode 100644 ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/user/UserConvertImpl.java create mode 100644 ruoyi-framework/pom.xml create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/aspectj/DataScopeAspect.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/aspectj/DataSourceAspect.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/aspectj/LogAspect.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/aspectj/RateLimiterAspect.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/config/ApplicationConfig.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/config/CaptchaConfig.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/config/DruidConfig.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/config/FastJson2JsonRedisSerializer.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/config/FilterConfig.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/config/KaptchaTextCreator.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/config/MybatisPlusConfig.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/config/RedisConfig.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/config/ResourcesConfig.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/config/SecurityConfig.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/config/ServerConfig.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/config/ThreadPoolConfig.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/config/properties/DruidProperties.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/config/properties/PermitAllUrlProperties.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/datasource/DynamicDataSource.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/datasource/DynamicDataSourceContextHolder.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/interceptor/RepeatSubmitInterceptor.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/interceptor/impl/SameUrlDataInterceptor.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/manager/AsyncManager.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/manager/ShutdownManager.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/manager/factory/AsyncFactory.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/security/context/AuthenticationContextHolder.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/security/context/PermissionContextHolder.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/security/filter/JwtAuthenticationTokenFilter.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/security/handle/AuthenticationEntryPointImpl.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/security/handle/LogoutSuccessHandlerImpl.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/Server.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/Cpu.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/Jvm.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/Mem.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/Sys.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/SysFile.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/web/exception/GlobalExceptionHandler.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/DataScopeService.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/JimuReportTokenService.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/PermissionService.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/SysLoginService.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/SysPasswordService.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/SysPermissionService.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/SysRegisterService.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/TokenService.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/UserDetailsServiceImpl.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/websocket/SemaphoreUtils.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/websocket/WebSocketConfig.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/websocket/WebSocketServer.java create mode 100644 ruoyi-framework/src/main/java/com/ruoyi/framework/websocket/WebSocketUsers.java create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/aspectj/DataScopeAspect.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/aspectj/DataSourceAspect.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/aspectj/LogAspect.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/config/ApplicationConfig.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/config/CaptchaConfig.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/config/DruidConfig$1.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/config/DruidConfig.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/config/FilterConfig.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/config/KaptchaTextCreator.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/config/MybatisPlusConfig.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/config/ResourcesConfig.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/config/SecurityConfig$1.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/config/SecurityConfig.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/config/ServerConfig.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/config/ThreadPoolConfig$1.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/config/ThreadPoolConfig.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/config/properties/DruidProperties.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/config/properties/PermitAllUrlProperties.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/datasource/DynamicDataSource.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/datasource/DynamicDataSourceContextHolder.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/interceptor/RepeatSubmitInterceptor.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/interceptor/impl/SameUrlDataInterceptor.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/manager/AsyncManager.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/manager/ShutdownManager.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/manager/factory/AsyncFactory$1.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/manager/factory/AsyncFactory$2.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/manager/factory/AsyncFactory.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/security/context/AuthenticationContextHolder.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/security/context/PermissionContextHolder.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/security/filter/JwtAuthenticationTokenFilter.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/security/handle/AuthenticationEntryPointImpl.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/security/handle/LogoutSuccessHandlerImpl.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/web/domain/Server.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/web/domain/server/Cpu.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/web/domain/server/Jvm.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/web/domain/server/Mem.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/web/domain/server/Sys.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/web/domain/server/SysFile.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/web/exception/GlobalExceptionHandler.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/web/service/DataScopeService.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/web/service/JimuReportTokenService.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/web/service/PermissionService.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/web/service/SysLoginService.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/web/service/SysPasswordService.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/web/service/SysPermissionService.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/web/service/SysRegisterService.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/web/service/TokenService.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/web/service/UserDetailsServiceImpl.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/websocket/SemaphoreUtils.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/websocket/WebSocketConfig.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/websocket/WebSocketServer.class create mode 100644 ruoyi-framework/target/classes/com/ruoyi/framework/websocket/WebSocketUsers.class create mode 100644 ruoyi-generator/pom.xml create mode 100644 ruoyi-generator/src/main/java/com/ruoyi/generator/config/GenConfig.java create mode 100644 ruoyi-generator/src/main/java/com/ruoyi/generator/controller/GenController.java create mode 100644 ruoyi-generator/src/main/java/com/ruoyi/generator/domain/GenTable.java create mode 100644 ruoyi-generator/src/main/java/com/ruoyi/generator/domain/GenTableColumn.java create mode 100644 ruoyi-generator/src/main/java/com/ruoyi/generator/mapper/GenTableColumnMapper.java create mode 100644 ruoyi-generator/src/main/java/com/ruoyi/generator/mapper/GenTableMapper.java create mode 100644 ruoyi-generator/src/main/java/com/ruoyi/generator/service/GenTableColumnServiceImpl.java create mode 100644 ruoyi-generator/src/main/java/com/ruoyi/generator/service/GenTableServiceImpl.java create mode 100644 ruoyi-generator/src/main/java/com/ruoyi/generator/service/IGenTableColumnService.java create mode 100644 ruoyi-generator/src/main/java/com/ruoyi/generator/service/IGenTableService.java create mode 100644 ruoyi-generator/src/main/java/com/ruoyi/generator/util/GenTableUtils.java create mode 100644 ruoyi-generator/src/main/java/com/ruoyi/generator/util/GenUtils.java create mode 100644 ruoyi-generator/src/main/java/com/ruoyi/generator/util/VelocityInitializer.java create mode 100644 ruoyi-generator/src/main/java/com/ruoyi/generator/util/VelocityUtils.java create mode 100644 ruoyi-generator/src/main/resources/generator.yml create mode 100644 ruoyi-generator/src/main/resources/mapper/generator/GenTableColumnMapper.xml create mode 100644 ruoyi-generator/src/main/resources/mapper/generator/GenTableMapper.xml create mode 100644 ruoyi-generator/src/main/resources/mybatisplusvm/java/controller.java.vm create mode 100644 ruoyi-generator/src/main/resources/mybatisplusvm/java/domain.java.vm create mode 100644 ruoyi-generator/src/main/resources/mybatisplusvm/java/mapper.java.vm create mode 100644 ruoyi-generator/src/main/resources/mybatisplusvm/java/service.java.vm create mode 100644 ruoyi-generator/src/main/resources/mybatisplusvm/java/serviceImpl.java.vm create mode 100644 ruoyi-generator/src/main/resources/mybatisplusvm/java/sub-domain.java.vm create mode 100644 ruoyi-generator/src/main/resources/mybatisplusvm/js/api.js.vm create mode 100644 ruoyi-generator/src/main/resources/mybatisplusvm/sql/sql.vm create mode 100644 ruoyi-generator/src/main/resources/mybatisplusvm/vue/index-tree.vue.vm create mode 100644 ruoyi-generator/src/main/resources/mybatisplusvm/vue/index.vue.vm create mode 100644 ruoyi-generator/src/main/resources/mybatisplusvm/vue/v3/index-tree.vue.vm create mode 100644 ruoyi-generator/src/main/resources/mybatisplusvm/vue/v3/index.vue.vm create mode 100644 ruoyi-generator/src/main/resources/mybatisplusvm/vue/v3/readme.txt create mode 100644 ruoyi-generator/src/main/resources/mybatisplusvm/xml/mapper.xml.vm create mode 100644 ruoyi-generator/src/main/resources/vm/java/controller.java.vm create mode 100644 ruoyi-generator/src/main/resources/vm/java/domain.java.vm create mode 100644 ruoyi-generator/src/main/resources/vm/java/mapper.java.vm create mode 100644 ruoyi-generator/src/main/resources/vm/java/service.java.vm create mode 100644 ruoyi-generator/src/main/resources/vm/java/serviceImpl.java.vm create mode 100644 ruoyi-generator/src/main/resources/vm/java/sub-domain.java.vm create mode 100644 ruoyi-generator/src/main/resources/vm/js/api.js.vm create mode 100644 ruoyi-generator/src/main/resources/vm/sql/sql.vm create mode 100644 ruoyi-generator/src/main/resources/vm/vue/index-tree.vue.vm create mode 100644 ruoyi-generator/src/main/resources/vm/vue/index.vue.vm create mode 100644 ruoyi-generator/src/main/resources/vm/vue/v3/index-tree.vue.vm create mode 100644 ruoyi-generator/src/main/resources/vm/vue/v3/index.vue.vm create mode 100644 ruoyi-generator/src/main/resources/vm/vue/v3/readme.txt create mode 100644 ruoyi-generator/src/main/resources/vm/xml/mapper.xml.vm create mode 100644 ruoyi-generator/target/classes/com/ruoyi/generator/config/GenConfig.class create mode 100644 ruoyi-generator/target/classes/com/ruoyi/generator/controller/GenController.class create mode 100644 ruoyi-generator/target/classes/com/ruoyi/generator/domain/GenTable.class create mode 100644 ruoyi-generator/target/classes/com/ruoyi/generator/domain/GenTableColumn.class create mode 100644 ruoyi-generator/target/classes/com/ruoyi/generator/mapper/GenTableColumnMapper.class create mode 100644 ruoyi-generator/target/classes/com/ruoyi/generator/mapper/GenTableMapper.class create mode 100644 ruoyi-generator/target/classes/com/ruoyi/generator/service/GenTableColumnServiceImpl.class create mode 100644 ruoyi-generator/target/classes/com/ruoyi/generator/service/GenTableServiceImpl.class create mode 100644 ruoyi-generator/target/classes/com/ruoyi/generator/service/IGenTableColumnService.class create mode 100644 ruoyi-generator/target/classes/com/ruoyi/generator/service/IGenTableService.class create mode 100644 ruoyi-generator/target/classes/com/ruoyi/generator/util/GenTableUtils.class create mode 100644 ruoyi-generator/target/classes/com/ruoyi/generator/util/GenUtils.class create mode 100644 ruoyi-generator/target/classes/com/ruoyi/generator/util/VelocityInitializer.class create mode 100644 ruoyi-generator/target/classes/com/ruoyi/generator/util/VelocityUtils.class create mode 100644 ruoyi-generator/target/classes/generator.yml create mode 100644 ruoyi-generator/target/classes/mapper/generator/GenTableColumnMapper.xml create mode 100644 ruoyi-generator/target/classes/mapper/generator/GenTableMapper.xml create mode 100644 ruoyi-generator/target/classes/mybatisplusvm/java/controller.java.vm create mode 100644 ruoyi-generator/target/classes/mybatisplusvm/java/domain.java.vm create mode 100644 ruoyi-generator/target/classes/mybatisplusvm/java/mapper.java.vm create mode 100644 ruoyi-generator/target/classes/mybatisplusvm/java/service.java.vm create mode 100644 ruoyi-generator/target/classes/mybatisplusvm/java/serviceImpl.java.vm create mode 100644 ruoyi-generator/target/classes/mybatisplusvm/java/sub-domain.java.vm create mode 100644 ruoyi-generator/target/classes/mybatisplusvm/js/api.js.vm create mode 100644 ruoyi-generator/target/classes/mybatisplusvm/sql/sql.vm create mode 100644 ruoyi-generator/target/classes/mybatisplusvm/vue/index-tree.vue.vm create mode 100644 ruoyi-generator/target/classes/mybatisplusvm/vue/index.vue.vm create mode 100644 ruoyi-generator/target/classes/mybatisplusvm/vue/v3/index-tree.vue.vm create mode 100644 ruoyi-generator/target/classes/mybatisplusvm/vue/v3/index.vue.vm create mode 100644 ruoyi-generator/target/classes/mybatisplusvm/vue/v3/readme.txt create mode 100644 ruoyi-generator/target/classes/mybatisplusvm/xml/mapper.xml.vm create mode 100644 ruoyi-generator/target/classes/vm/java/controller.java.vm create mode 100644 ruoyi-generator/target/classes/vm/java/domain.java.vm create mode 100644 ruoyi-generator/target/classes/vm/java/mapper.java.vm create mode 100644 ruoyi-generator/target/classes/vm/java/service.java.vm create mode 100644 ruoyi-generator/target/classes/vm/java/serviceImpl.java.vm create mode 100644 ruoyi-generator/target/classes/vm/java/sub-domain.java.vm create mode 100644 ruoyi-generator/target/classes/vm/js/api.js.vm create mode 100644 ruoyi-generator/target/classes/vm/sql/sql.vm create mode 100644 ruoyi-generator/target/classes/vm/vue/index-tree.vue.vm create mode 100644 ruoyi-generator/target/classes/vm/vue/index.vue.vm create mode 100644 ruoyi-generator/target/classes/vm/vue/v3/index-tree.vue.vm create mode 100644 ruoyi-generator/target/classes/vm/vue/v3/index.vue.vm create mode 100644 ruoyi-generator/target/classes/vm/vue/v3/readme.txt create mode 100644 ruoyi-generator/target/classes/vm/xml/mapper.xml.vm create mode 100644 ruoyi-plugin/pom.xml create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-oss/pom.xml create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/java/com/ruoyi/plugin/oss/config/OssConfiguration.java create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/java/com/ruoyi/plugin/oss/controller/OssEndpoint.java create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/java/com/ruoyi/plugin/oss/core/OssTemplate.java create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/java/com/ruoyi/plugin/oss/props/OssProperties.java create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/resources/META-INF/spring.factories create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-oss/target/classes/META-INF/spring.factories create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-oss/target/classes/com/ruoyi/plugin/oss/config/OssConfiguration.class create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-oss/target/classes/com/ruoyi/plugin/oss/controller/OssEndpoint.class create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-oss/target/classes/com/ruoyi/plugin/oss/core/OssTemplate.class create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-oss/target/classes/com/ruoyi/plugin/oss/props/OssProperties.class create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/pom.xml create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/config/TenantConfiguration.java create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/config/TenantProperties.java create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/aop/TenantIgnore.java create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/aop/TenantIgnoreAspect.java create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/context/TenantContextHolder.java create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/db/TenantBaseDO.java create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/db/TenantDatabaseInterceptor.java create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/security/TenantSecurityWebFilter.java create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/util/TenantUtils.java create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/web/TenantContextWebFilter.java create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/resources/META-INF/spring.factories create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/META-INF/spring.factories create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/config/TenantConfiguration.class create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/config/TenantProperties.class create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/aop/TenantIgnore.class create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/aop/TenantIgnoreAspect.class create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/context/TenantContextHolder.class create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/db/TenantBaseDO.class create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/db/TenantDatabaseInterceptor.class create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/security/TenantSecurityWebFilter.class create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/util/TenantUtils.class create mode 100644 ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/web/TenantContextWebFilter.class create mode 100644 ruoyi-quartz/pom.xml create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/config/ScheduleConfig.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/controller/SysJobController.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/controller/SysJobLogController.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/domain/SysJob.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/domain/SysJobLog.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/mapper/SysJobLogMapper.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/mapper/SysJobMapper.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/service/ISysJobLogService.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/service/ISysJobService.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/service/impl/SysJobLogServiceImpl.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/service/impl/SysJobServiceImpl.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/task/CommonTask.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/AbstractQuartzJob.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/CronUtils.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/JobInvokeUtil.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/QuartzDisallowConcurrentExecution.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/QuartzJobExecution.java create mode 100644 ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/ScheduleUtils.java create mode 100644 ruoyi-quartz/src/main/resources/mapper/quartz/SysJobLogMapper.xml create mode 100644 ruoyi-quartz/src/main/resources/mapper/quartz/SysJobMapper.xml create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/controller/SysJobController.class create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/controller/SysJobLogController.class create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/domain/SysJob.class create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/domain/SysJobLog.class create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/mapper/SysJobLogMapper.class create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/mapper/SysJobMapper.class create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/service/ISysJobLogService.class create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/service/ISysJobService.class create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/service/impl/SysJobLogServiceImpl.class create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/service/impl/SysJobServiceImpl.class create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/task/CommonTask.class create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/util/AbstractQuartzJob.class create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/util/CronUtils.class create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/util/JobInvokeUtil.class create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/util/QuartzDisallowConcurrentExecution.class create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/util/QuartzJobExecution.class create mode 100644 ruoyi-quartz/target/classes/com/ruoyi/quartz/util/ScheduleUtils.class create mode 100644 ruoyi-quartz/target/classes/mapper/quartz/SysJobLogMapper.xml create mode 100644 ruoyi-quartz/target/classes/mapper/quartz/SysJobMapper.xml create mode 100644 ruoyi-system/pom.xml create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/convert/Device/DeviceConvert.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/convert/task/TaskConvert.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/convert/tenant/TenantConvert.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/convert/tenant/TenantPackageConvert.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/convert/user/UserConvert.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/SysCache.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/SysConfig.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/SysLogininfor.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/SysNotice.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/SysOperLog.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/SysPost.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/SysRoleDept.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/SysRoleMenu.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/SysTreeDict.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/SysTreeDictData.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/SysUserOnline.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/SysUserPost.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/SysUserRole.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/TenantDO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/TenantPackageDO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/ImageVo.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/MetaVo.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/NameIdVo.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/RouterVo.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/UserInfoVo.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/menu/MenuSimpleRespVO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageBaseVO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageCreateReqVO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackagePageReqVO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageRespVO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageSimpleRespVO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageUpdateReqVO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/report/ReportDataRequestVo.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/report/ReportDataResponseVo.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/report/ReportDownLoadVo.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/task/BatchSettingResqVo.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/task/PointResqVo.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/task/SettingUserInfoResqVo.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantBaseVO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantCreateReqVO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantExcelVO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantExportReqVO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantPageReqVO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantRespVO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantUpdateReqVO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/user/UserSimpleRespVO.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/enums/ReportTypeEnum.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysConfigMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysDeptMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysDictDataMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysDictTypeMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysLogininforMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysMenuMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysNoticeMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysOperLogMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysPostMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysRoleDeptMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysRoleMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysRoleMenuMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysTreeDictDataMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysTreeDictMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysUserMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysUserPostMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysUserRoleMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/TenantMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/mapper/TenantPackageMapper.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/ISysConfigService.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/ISysDeptService.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/ISysDictDataService.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/ISysDictTypeService.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/ISysLogininforService.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/ISysMenuService.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/ISysNoticeService.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/ISysOperLogService.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/ISysPostService.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/ISysRoleService.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/ISysTreeDictDataService.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/ISysTreeDictService.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/ISysUserOnlineService.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/ISysUserService.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/TenantPackageService.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/TenantService.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/handler/TenantInfoHandler.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/handler/TenantMenuHandler.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysConfigServiceImpl.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysDeptServiceImpl.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysDictDataServiceImpl.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysDictTypeServiceImpl.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysLogininforServiceImpl.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysMenuServiceImpl.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysNoticeServiceImpl.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysOperLogServiceImpl.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysPostServiceImpl.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysRoleServiceImpl.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysUserOnlineServiceImpl.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysUserServiceImpl.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TenantPackageServiceImpl.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TenantServiceImpl.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TreeDictDataServiceImpl.java create mode 100644 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TreeDictServiceImpl.java create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysConfigMapper.xml create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysDeptMapper.xml create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysDictDataMapper.xml create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysDictTypeMapper.xml create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysLogininforMapper.xml create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysMenuMapper.xml create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysNoticeMapper.xml create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysOperLogMapper.xml create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysPostMapper.xml create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysRoleDeptMapper.xml create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysRoleMapper.xml create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysRoleMenuMapper.xml create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysTreeDictDataMapper.xml create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysTreeDictMapper.xml create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysUserMapper.xml create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysUserPostMapper.xml create mode 100644 ruoyi-system/src/main/resources/mapper/system/SysUserRoleMapper.xml create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/convert/Device/DeviceConvert.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/convert/Device/DeviceConvertImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/convert/task/TaskConvert.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/convert/task/TaskConvertImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/convert/tenant/TenantConvert.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/convert/tenant/TenantConvertImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/convert/tenant/TenantPackageConvert.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/convert/tenant/TenantPackageConvertImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/convert/user/UserConvert.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/convert/user/UserConvertImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/SysCache.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/SysConfig.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/SysLogininfor.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/SysNotice.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/SysOperLog.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/SysPost.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/SysRoleDept.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/SysRoleMenu.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/SysTreeDict.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/SysTreeDictData.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/SysUserOnline.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/SysUserPost.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/SysUserRole.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/TenantDO$TenantDOBuilder.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/TenantDO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/TenantPackageDO$TenantPackageDOBuilder.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/TenantPackageDO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/ImageVo.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/MetaVo.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/NameIdVo.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/RouterVo.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/UserInfoVo.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/menu/MenuSimpleRespVO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/packages/TenantPackageBaseVO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/packages/TenantPackageCreateReqVO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/packages/TenantPackagePageReqVO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/packages/TenantPackageRespVO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/packages/TenantPackageSimpleRespVO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/packages/TenantPackageUpdateReqVO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/report/ReportDataRequestVo.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/report/ReportDataResponseVo.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/report/ReportDownLoadVo.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/task/BatchSettingResqVo.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/task/PointResqVo.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/task/SettingUserInfoResqVo.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantBaseVO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantCreateReqVO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantExcelVO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantExportReqVO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantPageReqVO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantRespVO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantUpdateReqVO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/domain/vo/user/UserSimpleRespVO.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/enums/ReportTypeEnum.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysConfigMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysDeptMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysDictDataMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysDictTypeMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysLogininforMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysMenuMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysNoticeMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysOperLogMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysPostMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysRoleDeptMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysRoleMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysRoleMenuMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysTreeDictDataMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysTreeDictMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysUserMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysUserPostMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/SysUserRoleMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/TenantMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/mapper/TenantPackageMapper.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/ISysConfigService.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/ISysDeptService.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/ISysDictDataService.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/ISysDictTypeService.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/ISysLogininforService.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/ISysMenuService.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/ISysNoticeService.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/ISysOperLogService.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/ISysPostService.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/ISysRoleService.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/ISysTreeDictDataService.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/ISysTreeDictService.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/ISysUserOnlineService.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/ISysUserService.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/TenantPackageService.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/TenantService.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/handler/TenantInfoHandler.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/handler/TenantMenuHandler.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysConfigServiceImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysDeptServiceImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysDictDataServiceImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysDictTypeServiceImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysLogininforServiceImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysMenuServiceImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysNoticeServiceImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysOperLogServiceImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysPostServiceImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysRoleServiceImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysUserOnlineServiceImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysUserServiceImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/impl/TenantPackageServiceImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/impl/TenantServiceImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/impl/TreeDictDataServiceImpl.class create mode 100644 ruoyi-system/target/classes/com/ruoyi/system/service/impl/TreeDictServiceImpl.class create mode 100644 ruoyi-system/target/classes/mapper/system/SysConfigMapper.xml create mode 100644 ruoyi-system/target/classes/mapper/system/SysDeptMapper.xml create mode 100644 ruoyi-system/target/classes/mapper/system/SysDictDataMapper.xml create mode 100644 ruoyi-system/target/classes/mapper/system/SysDictTypeMapper.xml create mode 100644 ruoyi-system/target/classes/mapper/system/SysLogininforMapper.xml create mode 100644 ruoyi-system/target/classes/mapper/system/SysMenuMapper.xml create mode 100644 ruoyi-system/target/classes/mapper/system/SysNoticeMapper.xml create mode 100644 ruoyi-system/target/classes/mapper/system/SysOperLogMapper.xml create mode 100644 ruoyi-system/target/classes/mapper/system/SysPostMapper.xml create mode 100644 ruoyi-system/target/classes/mapper/system/SysRoleDeptMapper.xml create mode 100644 ruoyi-system/target/classes/mapper/system/SysRoleMapper.xml create mode 100644 ruoyi-system/target/classes/mapper/system/SysRoleMenuMapper.xml create mode 100644 ruoyi-system/target/classes/mapper/system/SysTreeDictDataMapper.xml create mode 100644 ruoyi-system/target/classes/mapper/system/SysTreeDictMapper.xml create mode 100644 ruoyi-system/target/classes/mapper/system/SysUserMapper.xml create mode 100644 ruoyi-system/target/classes/mapper/system/SysUserPostMapper.xml create mode 100644 ruoyi-system/target/classes/mapper/system/SysUserRoleMapper.xml create mode 100644 ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/Device/DeviceConvertImpl.java create mode 100644 ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/task/TaskConvertImpl.java create mode 100644 ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/tenant/TenantConvertImpl.java create mode 100644 ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/tenant/TenantPackageConvertImpl.java create mode 100644 ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/user/UserConvertImpl.java create mode 100644 注意事项.txt diff --git a/LICENSE b/LICENSE new file mode 100644 index 00000000..8564f294 --- /dev/null +++ b/LICENSE @@ -0,0 +1,20 @@ +The MIT License (MIT) + +Copyright (c) 2018 RuoYi + +Permission is hereby granted, free of charge, to any person obtaining a copy of +this software and associated documentation files (the "Software"), to deal in +the Software without restriction, including without limitation the rights to +use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of +the Software, and to permit persons to whom the Software is furnished to do so, +subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS +FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR +COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER +IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN +CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. \ No newline at end of file diff --git a/README.md b/README.md new file mode 100644 index 00000000..88e6d86d --- /dev/null +++ b/README.md @@ -0,0 +1,94 @@ +

+ logo +

+

RuoYi-Vue-Super

+

🎉基于最新若依前后端分离版本,同步更新,并新增功能,集成hutools,mybatis-plus,lombok,knife4j,websocket,minio文件上传,集成flowable工作流,加入可视化大屏。

+

+ + + +

+ +## 平台简介 + +若依是一套全部开源的快速开发平台,毫无保留给个人及企业免费使用。 + +* 前端采用Vue、Element UI。 +* 后端采用Spring Boot、Spring Security、Redis & Jwt。 +* 权限认证使用Jwt,支持多终端认证系统。 +* 支持加载动态权限菜单,多方式轻松权限控制。 +* 高效率开发,使用代码生成器可以一键生成前后端代码。 + +## 新增功能 +* 集成hutools,mybatis-plus,lombok,knife4j,websocket。 +* 代码生成同时适配mybatis跟mybatis-plus。 +* 集成xdh-map,基于openlayers灵活开发地图可视化,各种矢量图层,控件。 +* 在字典管理基础上增加tree树形字典管理功能,方便树形字典开发与后期动态运维。 +* 集成flowable工作流,代码引入芋道源码flowable工作流模块,适配若依官方代码,方便后期同步更新。 +* 集成可视化大屏,代码引入奔跑的面条big-screen-vue-datav大屏vue2版本。 +* [新增 mybatis-plus数据权限解决方案。](https://gitee.com/rainsuper/RuoYi-Vue-Super/wikis/pages?sort_id=5957319&doc_id=2965484) +* [新增 tenant 多租户解决方案(共享数据库方式)。](https://gitee.com/rainsuper/RuoYi-Vue-Super/wikis/pages?sort_id=5960193&doc_id=2965484) +* [新增 oss 文件上传插件,支持所有兼容s3协议的云存储:如阿里云OSS,腾讯云COS,七牛云,京东云,minio等。](https://gitee.com/rainsuper/RuoYi-Vue-Super/wikis/pages?sort_id=5966058&doc_id=2965484) + +## 参考文档 +第三方对接 +
+>[整合Oauth2.0单点方案](https://gitee.com/rainsuper/RuoYi-Vue-Super/wikis/pages?sort_id=5949355&doc_id=2965484) +## 内置功能 + +1. 用户管理:用户是系统操作者,该功能主要完成系统用户配置。 +2. 部门管理:配置系统组织机构(公司、部门、小组),树结构展现支持数据权限。 +3. 岗位管理:配置系统用户所属担任职务。 +4. 菜单管理:配置系统菜单,操作权限,按钮权限标识等。 +5. 角色管理:角色菜单权限分配、设置角色按机构进行数据范围权限划分。 +6. 字典管理:对系统中经常使用的一些较为固定的数据进行维护。 +7. 参数管理:对系统动态配置常用参数。 +8. 通知公告:系统通知公告信息发布维护。 +9. 操作日志:系统正常操作日志记录和查询;系统异常信息日志记录和查询。 +10. 登录日志:系统登录日志记录查询包含登录异常。 +11. 在线用户:当前系统中活跃用户状态监控。 +12. 定时任务:在线(添加、修改、删除)任务调度包含执行结果日志。 +13. 代码生成:前后端代码的生成(java、html、xml、sql)支持CRUD下载 。 +14. 系统接口:根据业务代码自动生成相关的api接口文档。 +15. 服务监控:监视当前系统CPU、内存、磁盘、堆栈等相关信息。 +16. 缓存监控:对系统的缓存信息查询,命令统计等。 +17. 在线构建器:拖动表单元素生成相应的HTML代码。 +18. 连接池监视:监视当前系统数据库连接池状态,可进行分析SQL找出系统性能瓶颈。 + +## 在线体验 + +- admin/admin123 + +旧演示地址:http://vue.ruoyi.vip +旧文档地址:http://doc.ruoyi.vip + +- 新增功能按照若依文档跑下就可以看到了,或者看下边的演示图。 + +## 演示图 +- 若依官方显示图就不放了,这里只放新增的功能图片。 + + + + + + + + + + + + + + + + + + + + +
+ +## 若依Super前后端分离交流群 +QQ群: [![加入QQ群](https://img.shields.io/badge/681646796-blue.svg)](https://jq.qq.com/?_wv=1027&k=bbKX5vcb) 点击按钮入群。 + + diff --git a/pom.xml b/pom.xml new file mode 100644 index 00000000..ea682b43 --- /dev/null +++ b/pom.xml @@ -0,0 +1,372 @@ + + + 4.0.0 + + com.ruoyi + ruoyi + 3.8.4 + + ruoyi + http://www.ruoyi.vip + 数字消防综合管理系统 + + + 3.8.4 + UTF-8 + UTF-8 + 1.8 + 3.1.1 + 1.2.11 + 1.21 + 3.0.0 + 2.3.2 + 2.2.2 + 1.4.6 + 2.0.16 + 6.3.0 + 2.11.0 + 1.4 + 3.2.2 + 4.1.2 + 2.3 + 0.9.1 + 5.8.5 + 3.5.2 + 1.18.12 + 3.0.3 + + 6.7.2 + 1.4.1.Final + 2.11.4 + 1.12.282 + 3.5.2 + 1.5.8 + 3.1.1 + 2.12.2 + + + + + + + xerces + xercesImpl + ${xercesImpl.version} + + + com.alibaba + easyexcel + ${easyexcel.verion} + + + + org.springframework.boot + spring-boot-dependencies + 2.5.14 + pom + import + + + + + com.alibaba + druid-spring-boot-starter + ${druid.version} + + + + + eu.bitwalker + UserAgentUtils + ${bitwalker.version} + + + + + org.mybatis.spring.boot + mybatis-spring-boot-starter + ${mybatis-spring-boot.version} + + + + + com.github.pagehelper + pagehelper-spring-boot-starter + ${pagehelper.boot.version} + + + + + com.github.oshi + oshi-core + ${oshi.version} + + + + + io.springfox + springfox-boot-starter + ${swagger.version} + + + io.swagger + swagger-models + + + + + + + commons-io + commons-io + ${commons.io.version} + + + + + commons-fileupload + commons-fileupload + ${commons.fileupload.version} + + + + + org.apache.poi + poi-ooxml + ${poi.version} + + + + + org.apache.velocity + velocity-engine-core + ${velocity.version} + + + + + commons-collections + commons-collections + ${commons.collections.version} + + + + + com.alibaba.fastjson2 + fastjson2 + ${fastjson.version} + + + + + io.jsonwebtoken + jjwt + ${jwt.version} + + + + + com.github.penggle + kaptcha + ${kaptcha.version} + + + + + com.ruoyi + ruoyi-quartz + ${ruoyi.version} + + + + + com.ruoyi + ruoyi-generator + ${ruoyi.version} + + + + + com.ruoyi + ruoyi-framework + ${ruoyi.version} + + + + + com.ruoyi + ruoyi-system + ${ruoyi.version} + + + + + com.ruoyi + ruoyi-common + ${ruoyi.version} + + + + + com.ruoyi + ruoyi-flowable + ${ruoyi.version} + + + + + cn.hutool + hutool-all + ${hutool.version} + + + + + com.baomidou + mybatis-plus-boot-starter + ${mybatis-plus.version} + + + + + org.projectlombok + lombok + true + ${lombok.version} + + + + + com.github.xiaoymin + knife4j-spring-boot-starter + ${knife4j.version} + + + mapstruct + org.mapstruct + + + + + + + org.flowable + flowable-spring-boot-starter-process + ${flowable.version} + + + org.flowable + flowable-spring-boot-starter-actuator + ${flowable.version} + + + + org.mapstruct + mapstruct + ${mapstruct.version} + + + org.mapstruct + mapstruct-jdk8 + ${mapstruct.version} + + + org.mapstruct + mapstruct-processor + ${mapstruct.version} + + + + + com.alibaba + transmittable-thread-local + ${ttl.version} + + + + com.amazonaws + aws-java-sdk-s3 + ${aws-java-sdk-s3.version} + + + + com.ruoyi + ruoyi-springboot-starter-tenant + ${ruoyi.version} + + + + + com.baomidou + mybatis-plus-generator + ${mybatis-plus-generator.version} + + + + org.jeecgframework.jimureport + jimureport-spring-boot-starter + ${jimureport.version} + + + + + + ruoyi-admin + ruoyi-framework + ruoyi-system + ruoyi-quartz + ruoyi-generator + ruoyi-common + ruoyi-flowable + ruoyi-plugin + + pom + + + + + + + + + + org.apache.maven.plugins + maven-compiler-plugin + 3.1 + + ${java.version} + ${java.version} + ${project.build.sourceEncoding} + + + + + + + + public + aliyun nexus + https://maven.aliyun.com/repository/public + + true + + + + + + + public + aliyun nexus + https://maven.aliyun.com/repository/public + + true + + + false + + + + + diff --git a/ruoyi-admin/Dockerfile b/ruoyi-admin/Dockerfile new file mode 100644 index 00000000..80127187 --- /dev/null +++ b/ruoyi-admin/Dockerfile @@ -0,0 +1,19 @@ +## AdoptOpenJDK 停止发布 OpenJDK 二进制,而 Eclipse Temurin 是它的延伸,提供更好的稳定性 +FROM openjdk:8-jdk-alpine +#FROM eclipse-temurin:8-jre + +## 创建目录,并使用它作为工作目录 +#RUN mkdir -p /project/hzyh-admin +#WORKDIR /project/hzyh-admin +## 将后端项目的 Jar 文件,复制到镜像中 +COPY target/ruoyi-admin.jar hzyh-admin.jar + +## 设置 TZ 时区 +## 设置 JAVA_OPTS 环境变量,可通过 docker run -e "JAVA_OPTS=" 进行覆盖 +ENV TZ=Asia/Shanghai JAVA_OPTS="-Xms512m -Xmx1024m" + +## 暴露后端项目的 48080 端口 +EXPOSE 8080 + +## 启动后端项目 +CMD java ${JAVA_OPTS} -Djava.security.egd=file:/dev/./urandom -Dspring.profiles.active=${HJ} -jar hzyh-admin.jar diff --git a/ruoyi-admin/pom.xml b/ruoyi-admin/pom.xml new file mode 100644 index 00000000..0b8c2a0d --- /dev/null +++ b/ruoyi-admin/pom.xml @@ -0,0 +1,120 @@ + + + + ruoyi + com.ruoyi + 3.8.4 + + 4.0.0 + jar + ruoyi-admin + + + web服务入口 + + + + + + + + + + + org.jsoup + jsoup + 1.14.3 + + + + + + + + + + + mysql + mysql-connector-java + + + + + com.ruoyi + ruoyi-framework + + + + + com.ruoyi + ruoyi-quartz + + + + + com.ruoyi + ruoyi-generator + + + + + com.github.xiaoymin + knife4j-spring-boot-starter + + + + + com.ruoyi + ruoyi-flowable + + + mybatis + org.mybatis + + + + + org.jeecgframework.jimureport + jimureport-spring-boot-starter + + + jsqlparser + com.github.jsqlparser + + + + + + + + + org.springframework.boot + spring-boot-maven-plugin + 2.1.1.RELEASE + + true + + + + + repackage + + + + + + org.apache.maven.plugins + maven-war-plugin + 3.1.0 + + false + ${project.artifactId} + + + + ${project.artifactId} + + + diff --git a/ruoyi-admin/src/main/java/com/ruoyi/RuoYiApplication.java b/ruoyi-admin/src/main/java/com/ruoyi/RuoYiApplication.java new file mode 100644 index 00000000..04873d70 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/RuoYiApplication.java @@ -0,0 +1,32 @@ +package com.ruoyi; + +import org.springframework.boot.SpringApplication; +import org.springframework.boot.autoconfigure.SpringBootApplication; +import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration; +import org.springframework.scheduling.annotation.EnableScheduling; + +/** + * 启动程序 + * + * @author ruoyi + */ +@SpringBootApplication(exclude = { DataSourceAutoConfiguration.class },scanBasePackages = {"org.jeecg.modules.jmreport","com.ruoyi"}) +@EnableScheduling +public class RuoYiApplication +{ + public static void main(String[] args) + { + // System.setProperty("spring.devtools.restart.enabled", "false"); + SpringApplication.run(RuoYiApplication.class, args); + System.out.println("(♥◠‿◠)ノ゙ 泽火启动成功 ლ(´ڡ`ლ)゙ \n" + + " .-------. ____ __ \n" + + " | _ _ \\ \\ \\ / / \n" + + " | ( ' ) | \\ _. / ' \n" + + " |(_ o _) / _( )_ .' \n" + + " | (_,_).' __ ___(_ o _)' \n" + + " | |\\ \\ | || |(_,_)' \n" + + " | | \\ `' /| `-' / \n" + + " | | \\ / \\ / \n" + + " ''-' `'-' `-..-' "); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/RuoYiServletInitializer.java b/ruoyi-admin/src/main/java/com/ruoyi/RuoYiServletInitializer.java new file mode 100644 index 00000000..6de67dc7 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/RuoYiServletInitializer.java @@ -0,0 +1,18 @@ +package com.ruoyi; + +import org.springframework.boot.builder.SpringApplicationBuilder; +import org.springframework.boot.web.servlet.support.SpringBootServletInitializer; + +/** + * web容器中进行部署 + * + * @author ruoyi + */ +public class RuoYiServletInitializer extends SpringBootServletInitializer +{ + @Override + protected SpringApplicationBuilder configure(SpringApplicationBuilder application) + { + return application.sources(RuoYiApplication.class); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/TestFlowableApplication.java b/ruoyi-admin/src/main/java/com/ruoyi/TestFlowableApplication.java new file mode 100644 index 00000000..c028ae2d --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/TestFlowableApplication.java @@ -0,0 +1,17 @@ +package com.ruoyi; + +import org.flowable.engine.ProcessEngine; +import org.flowable.engine.ProcessEngineConfiguration; +import org.flowable.engine.impl.cfg.StandaloneProcessEngineConfiguration; + +public class TestFlowableApplication { + public static void main(String[] args) { + ProcessEngineConfiguration cfg = new StandaloneProcessEngineConfiguration() + .setJdbcUrl("jdbc:mysql://60.204.223.58:3306/flowable?serverTimezone=UTC&nullCatalogMeansCurrent=true") + .setJdbcUsername("root") + .setJdbcPassword("Mysql123!") + .setJdbcDriver("com.mysql.cj.jdbc.Driver") + .setDatabaseSchemaUpdate(ProcessEngineConfiguration.DB_SCHEMA_UPDATE_TRUE); + ProcessEngine processEngine = cfg.buildProcessEngine(); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/DevinUtil.java b/ruoyi-admin/src/main/java/com/ruoyi/web/DevinUtil.java new file mode 100644 index 00000000..77264b63 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/DevinUtil.java @@ -0,0 +1,85 @@ +package com.ruoyi.web; + +import cn.hutool.core.io.FileUtil; +import cn.hutool.core.util.StrUtil; + +import java.io.File; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +/** + * + */ +public class DevinUtil { + public static void main(String[] args) { + copyFileFromModule(); + } + + /** + * 基于某个菜单功能进行快速拷贝生成后端和web代码 + */ + private static void copyFileFromModule() { + Map replaceMap = new HashMap<>(); //要替换掉的内容。注意大小写 + //基于标识管理功能,快速拷贝相关文件 + String newEntityName = "FireStationTrainRecord"; + String oldEntityName = "IdentityManage"; + replaceMap.put("bs_identity_manage", "bs_firestation_trainrecord"); + replaceMap.put("identityManage", "fireStationTrainRecord"); + replaceMap.put("标识管理", "训练记录"); + replaceMap.put(oldEntityName, newEntityName); //实体名 + replaceMap.put(oldEntityName.toLowerCase(), newEntityName.toLowerCase()); //全小写的替换 + List replaceFile = new ArrayList<>(); + //后端代码类 + replaceFile.add("D:\\project\\itcqp\\RuoYi-Vue-Super\\ruoyi-system\\src\\main\\java\\com\\ruoyi\\system\\domain\\Bs" + oldEntityName + ".java"); + replaceFile.add("D:\\project\\itcqp\\RuoYi-Vue-Super\\ruoyi-admin\\src\\main\\java\\com\\ruoyi\\web\\controller\\system\\Bs" + oldEntityName + "Controller.java"); + replaceFile.add("D:\\project\\itcqp\\RuoYi-Vue-Super\\ruoyi-system\\src\\main\\java\\com\\ruoyi\\system\\mapper\\Bs" + oldEntityName + "Mapper.java"); + replaceFile.add("D:\\project\\itcqp\\RuoYi-Vue-Super\\ruoyi-system\\src\\main\\resources\\mapper\\system\\Bs" + oldEntityName + "Mapper.xml"); + replaceFile.add("D:\\project\\itcqp\\RuoYi-Vue-Super\\ruoyi-system\\src\\main\\java\\com\\ruoyi\\system\\service\\impl\\Bs" + oldEntityName + "ServiceImpl.java"); + replaceFile.add("D:\\project\\itcqp\\RuoYi-Vue-Super\\ruoyi-system\\src\\main\\java\\com\\ruoyi\\system\\service\\IBs" + oldEntityName + "Service.java"); + //前端代码类 + //前端vue文件夹路径 + replaceFile.add("D:\\project\\itcqp\\RuoYi-APP\\ruoyi-ui\\src\\views\\system\\"+oldEntityName.toLowerCase()); + //js 路径 + replaceFile.add("D:\\project\\itcqp\\RuoYi-APP\\ruoyi-ui\\src\\api\\system\\"+oldEntityName.toLowerCase()+".js"); + for (String filepath : replaceFile) { + File file = FileUtil.file(filepath); + if (!file.exists()) { + System.out.println("文件不存在:" + filepath); + continue; + } + if (file.isDirectory()) { + //前端目录 + FileUtil.walkFiles(FileUtil.file(filepath),tf->{ + writeNewFile(replaceMap, newEntityName, oldEntityName, tf,true); + }); + }else{ + //文件 + writeNewFile(replaceMap, newEntityName, oldEntityName, file,true); + } + } + } + + private static void writeNewFile(Map replaceMap, String entityName, String oldEntityName, File file, boolean needReplaceFileName) { + String newfilepath = file.getAbsolutePath(); + if (needReplaceFileName) { + newfilepath = file.getAbsolutePath().replace(oldEntityName, entityName).replace(oldEntityName.toLowerCase(),entityName.toLowerCase()); //第二个是为了js文件 + } + List lines = FileUtil.readLines(file.getAbsolutePath(), "utf-8"); + List newlines = new ArrayList<>(); + for (String line : lines) { + String newLine = line; + if (StrUtil.isNotBlank(line)) { + for (String key : replaceMap.keySet()) { + newLine = newLine.replace(key, replaceMap.get(key)); + } + } + newlines.add(newLine); + } + if (!FileUtil.exist(newfilepath)) { + //不存在才创建 + FileUtil.writeLines(newlines,newfilepath,"utf-8"); + } + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/common/CaptchaController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/common/CaptchaController.java new file mode 100644 index 00000000..bec40373 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/common/CaptchaController.java @@ -0,0 +1,92 @@ +package com.ruoyi.web.controller.common; + +import java.awt.image.BufferedImage; +import java.io.IOException; +import java.util.concurrent.TimeUnit; +import javax.annotation.Resource; +import javax.imageio.ImageIO; +import javax.servlet.http.HttpServletResponse; + +import com.ruoyi.common.utils.EhcacheUtil; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.util.FastByteArrayOutputStream; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.RestController; +import com.google.code.kaptcha.Producer; +import com.ruoyi.common.config.RuoYiConfig; +import com.ruoyi.common.constant.CacheConstants; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.utils.sign.Base64; +import com.ruoyi.common.utils.uuid.IdUtils; +import com.ruoyi.system.service.ISysConfigService; + +/** + * 验证码操作处理 + * + * @author ruoyi + */ +@RestController +public class CaptchaController +{ + @Resource(name = "captchaProducer") + private Producer captchaProducer; + + @Resource(name = "captchaProducerMath") + private Producer captchaProducerMath; + + @Autowired + private ISysConfigService configService; + /** + * 生成验证码 + */ + @GetMapping("/captchaImage") + public AjaxResult getCode(HttpServletResponse response) throws IOException + { + AjaxResult ajax = AjaxResult.success(); + boolean captchaEnabled = configService.selectCaptchaEnabled(); + ajax.put("captchaEnabled", captchaEnabled); + if (!captchaEnabled) + { + return ajax; + } + + // 保存验证码信息 + String uuid = IdUtils.simpleUUID(); + String verifyKey = CacheConstants.CAPTCHA_CODE_KEY + uuid; + + String capStr = null, code = null; + BufferedImage image = null; + + // 生成验证码 + String captchaType = RuoYiConfig.getCaptchaType(); + if ("math".equals(captchaType)) + { + String capText = captchaProducerMath.createText(); + capStr = capText.substring(0, capText.lastIndexOf("@")); + code = capText.substring(capText.lastIndexOf("@") + 1); + image = captchaProducerMath.createImage(capStr); + } + else if ("char".equals(captchaType)) + { + capStr = code = captchaProducer.createText(); + image = captchaProducer.createImage(capStr); + } + EhcacheUtil.put("verify",verifyKey,code); +// redisCache.setCacheObject(verifyKey, code, Constants.CAPTCHA_EXPIRATION, TimeUnit.MINUTES); + // 转换流信息写出 + FastByteArrayOutputStream os = new FastByteArrayOutputStream(); + try + { + ImageIO.write(image, "jpg", os); + } + catch (IOException e) + { + return AjaxResult.error(e.getMessage()); + } + + ajax.put("uuid", uuid); + ajax.put("img", Base64.encode(os.toByteArray())); + return ajax; + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/common/CommonController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/common/CommonController.java new file mode 100644 index 00000000..62ea52e4 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/common/CommonController.java @@ -0,0 +1,163 @@ +package com.ruoyi.web.controller.common; + +import cn.hutool.core.util.StrUtil; +import cn.hutool.http.HttpRequest; +import cn.hutool.http.HttpUtil; +import cn.hutool.json.JSONObject; +import com.ruoyi.common.config.RuoYiConfig; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.file.FileUploadUtils; +import com.ruoyi.common.utils.file.FileUtils; +import com.ruoyi.framework.config.ServerConfig; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.http.MediaType; +import org.springframework.web.bind.annotation.*; +import org.springframework.web.multipart.MultipartFile; + +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import java.util.ArrayList; +import java.util.List; + +/** + * 通用请求处理 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/common") +public class CommonController { + private static final Logger log = LoggerFactory.getLogger(CommonController.class); + + @Autowired + private ServerConfig serverConfig; + + private static final String FILE_DELIMETER = ","; + + /** + * 通用下载请求 + * + * @param fileName 文件名称 + * @param delete 是否删除 + */ + @GetMapping("/download") + public void fileDownload(String fileName, Boolean delete, HttpServletResponse response, HttpServletRequest request) { + try { + if (!FileUtils.checkAllowDownload(fileName)) { + throw new Exception(StringUtils.format("文件名称({})非法,不允许下载。 ", fileName)); + } + String realFileName = System.currentTimeMillis() + fileName.substring(fileName.indexOf("_") + 1); + String filePath = RuoYiConfig.getDownloadPath() + fileName; + + response.setContentType(MediaType.APPLICATION_OCTET_STREAM_VALUE); + FileUtils.setAttachmentResponseHeader(response, realFileName); + FileUtils.writeBytes(filePath, response.getOutputStream()); + if (delete) { + FileUtils.deleteFile(filePath); + } + } catch (Exception e) { + log.error("下载文件失败", e); + } + } + + /** + * 通用上传请求(单个) + */ + @PostMapping("/upload") + public AjaxResult uploadFile(MultipartFile file) throws Exception { + try { + // 上传文件路径 + String filePath = RuoYiConfig.getUploadPath(); + // 上传并返回新文件名称 + String fileName = FileUploadUtils.upload(filePath, file); + String url = serverConfig.getUrl() + fileName; + AjaxResult ajax = AjaxResult.success(); + ajax.put("url", url); + ajax.put("fileName", fileName); + ajax.put("newFileName", FileUtils.getName(fileName)); + ajax.put("originalFilename", file.getOriginalFilename()); + return ajax; + } catch (Exception e) { + return AjaxResult.error(e.getMessage()); + } + } + + /** + * 通用上传请求(多个) + */ + @PostMapping("/uploads") + public AjaxResult uploadFiles(List files) throws Exception { + try { + // 上传文件路径 + String filePath = RuoYiConfig.getUploadPath(); + List urls = new ArrayList(); + List fileNames = new ArrayList(); + List newFileNames = new ArrayList(); + List originalFilenames = new ArrayList(); + for (MultipartFile file : files) { + // 上传并返回新文件名称 + String fileName = FileUploadUtils.upload(filePath, file); + String url = serverConfig.getUrl() + fileName; + urls.add(url); + fileNames.add(fileName); + newFileNames.add(FileUtils.getName(fileName)); + originalFilenames.add(file.getOriginalFilename()); + } + AjaxResult ajax = AjaxResult.success(); + ajax.put("urls", StringUtils.join(urls, FILE_DELIMETER)); + ajax.put("fileNames", StringUtils.join(fileNames, FILE_DELIMETER)); + ajax.put("newFileNames", StringUtils.join(newFileNames, FILE_DELIMETER)); + ajax.put("originalFilenames", StringUtils.join(originalFilenames, FILE_DELIMETER)); + return ajax; + } catch (Exception e) { + return AjaxResult.error(e.getMessage()); + } + } + + /** + * 本地资源通用下载 + */ + @GetMapping("/download/resource") + public void resourceDownload(String resource, HttpServletRequest request, HttpServletResponse response) + throws Exception { + try { + if (!FileUtils.checkAllowDownload(resource)) { + throw new Exception(StringUtils.format("资源文件({})非法,不允许下载。 ", resource)); + } + // 本地资源路径 + String localPath = RuoYiConfig.getProfile(); + // 数据库资源地址 + String downloadPath = localPath + StringUtils.substringAfter(resource, Constants.RESOURCE_PREFIX); + // 下载名称 + String downloadName = StringUtils.substringAfterLast(downloadPath, "/"); + response.setContentType(MediaType.APPLICATION_OCTET_STREAM_VALUE); + FileUtils.setAttachmentResponseHeader(response, downloadName); + FileUtils.writeBytes(downloadPath, response.getOutputStream()); + } catch (Exception e) { + log.error("下载文件失败", e); + } + } + + /** + * 本地资源通用下载 + */ + @PostMapping("/chatgpt") + public Object chatgpt(@RequestBody JSONObject param) throws Exception { + HttpRequest post = HttpUtil.createPost("https://api.aioschat.com/"); + post.body(param.toString()); + String body = post.execute().body(); + if (StrUtil.isNotBlank(body) && body.contains("请稍后再试")) { + System.out.println("第2次"); + body = post.execute().body(); + if (StrUtil.isNotBlank(body) && body.contains("请稍后再试")) { + System.out.println("第三次"); + body = post.execute().body(); + } + } + return body; + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/CacheController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/CacheController.java new file mode 100644 index 00000000..9aaf45bf --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/CacheController.java @@ -0,0 +1,120 @@ +//package com.ruoyi.web.controller.monitor; +// +//import java.util.ArrayList; +//import java.util.Collection; +//import java.util.HashMap; +//import java.util.List; +//import java.util.Map; +//import java.util.Properties; +//import java.util.Set; +//import org.springframework.beans.factory.annotation.Autowired; +//import org.springframework.data.redis.core.RedisCallback; +//import org.springframework.data.redis.core.RedisTemplate; +//import org.springframework.security.access.prepost.PreAuthorize; +//import org.springframework.web.bind.annotation.DeleteMapping; +//import org.springframework.web.bind.annotation.GetMapping; +//import org.springframework.web.bind.annotation.PathVariable; +//import org.springframework.web.bind.annotation.RequestMapping; +//import org.springframework.web.bind.annotation.RestController; +//import com.ruoyi.common.constant.CacheConstants; +//import com.ruoyi.common.core.domain.AjaxResult; +//import com.ruoyi.common.utils.StringUtils; +//import com.ruoyi.system.domain.SysCache; +// +///** +// * 缓存监控 +// * +// * @author ruoyi +// */ +//@RestController +//@RequestMapping("/monitor/cache") +//public class CacheController +//{ +// @Autowired +// private RedisTemplate redisTemplate; +// +// private final static List caches = new ArrayList(); +// { +// caches.add(new SysCache(CacheConstants.LOGIN_TOKEN_KEY, "用户信息")); +// caches.add(new SysCache(CacheConstants.SYS_CONFIG_KEY, "配置信息")); +// caches.add(new SysCache(CacheConstants.SYS_DICT_KEY, "数据字典")); +// caches.add(new SysCache(CacheConstants.CAPTCHA_CODE_KEY, "验证码")); +// caches.add(new SysCache(CacheConstants.REPEAT_SUBMIT_KEY, "防重提交")); +// caches.add(new SysCache(CacheConstants.RATE_LIMIT_KEY, "限流处理")); +// caches.add(new SysCache(CacheConstants.PWD_ERR_CNT_KEY, "密码错误次数")); +// } +// +// @PreAuthorize("@ss.hasPermi('monitor:cache:list')") +// @GetMapping() +// public AjaxResult getInfo() throws Exception +// { +// Properties info = (Properties) redisTemplate.execute((RedisCallback) connection -> connection.info()); +// Properties commandStats = (Properties) redisTemplate.execute((RedisCallback) connection -> connection.info("commandstats")); +// Object dbSize = redisTemplate.execute((RedisCallback) connection -> connection.dbSize()); +// +// Map result = new HashMap<>(3); +// result.put("info", info); +// result.put("dbSize", dbSize); +// +// List> pieList = new ArrayList<>(); +// commandStats.stringPropertyNames().forEach(key -> { +// Map data = new HashMap<>(2); +// String property = commandStats.getProperty(key); +// data.put("name", StringUtils.removeStart(key, "cmdstat_")); +// data.put("value", StringUtils.substringBetween(property, "calls=", ",usec")); +// pieList.add(data); +// }); +// result.put("commandStats", pieList); +// return AjaxResult.success(result); +// } +// +// @PreAuthorize("@ss.hasPermi('monitor:cache:list')") +// @GetMapping("/getNames") +// public AjaxResult cache() +// { +// return AjaxResult.success(caches); +// } +// +// @PreAuthorize("@ss.hasPermi('monitor:cache:list')") +// @GetMapping("/getKeys/{cacheName}") +// public AjaxResult getCacheKeys(@PathVariable String cacheName) +// { +// Set cacheKeys = redisTemplate.keys(cacheName + "*"); +// return AjaxResult.success(cacheKeys); +// } +// +// @PreAuthorize("@ss.hasPermi('monitor:cache:list')") +// @GetMapping("/getValue/{cacheName}/{cacheKey}") +// public AjaxResult getCacheValue(@PathVariable String cacheName, @PathVariable String cacheKey) +// { +// String cacheValue = redisTemplate.opsForValue().get(cacheKey); +// SysCache sysCache = new SysCache(cacheName, cacheKey, cacheValue); +// return AjaxResult.success(sysCache); +// } +// +// @PreAuthorize("@ss.hasPermi('monitor:cache:list')") +// @DeleteMapping("/clearCacheName/{cacheName}") +// public AjaxResult clearCacheName(@PathVariable String cacheName) +// { +// Collection cacheKeys = redisTemplate.keys(cacheName + "*"); +// redisTemplate.delete(cacheKeys); +// return AjaxResult.success(); +// } +// +// @PreAuthorize("@ss.hasPermi('monitor:cache:list')") +// @DeleteMapping("/clearCacheKey/{cacheKey}") +// public AjaxResult clearCacheKey(@PathVariable String cacheKey) +// { +// redisTemplate.delete(cacheKey); +// return AjaxResult.success(); +// } +// +// @PreAuthorize("@ss.hasPermi('monitor:cache:list')") +// @DeleteMapping("/clearCacheAll") +// public AjaxResult clearCacheAll() +// { +// Collection cacheKeys = redisTemplate.keys("*"); +// redisTemplate.delete(cacheKeys); +// return AjaxResult.success(); +// } +//} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/ServerController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/ServerController.java new file mode 100644 index 00000000..89ee27d0 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/ServerController.java @@ -0,0 +1,25 @@ +package com.ruoyi.web.controller.monitor; + +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.framework.web.domain.Server; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; + +/** + * 服务器监控 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/monitor/server") +public class ServerController { + @PreAuthorize("@ss.hasPermi('monitor:server:list')") + @GetMapping() + public AjaxResult getInfo() throws Exception { + Server server = new Server(); + server.copyTo(); + return AjaxResult.success(server); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/SysLogininforController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/SysLogininforController.java new file mode 100644 index 00000000..e1112377 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/SysLogininforController.java @@ -0,0 +1,83 @@ +package com.ruoyi.web.controller.monitor; + +import java.util.List; +import javax.servlet.http.HttpServletResponse; + +import com.ruoyi.framework.web.service.SysPasswordService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.web.bind.annotation.DeleteMapping; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.page.TableDataInfo; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.utils.poi.ExcelUtil; +import com.ruoyi.system.domain.SysLogininfor; +import com.ruoyi.system.service.ISysLogininforService; + +/** + * 系统访问记录 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/monitor/logininfor") +public class SysLogininforController extends BaseController +{ + @Autowired + private ISysLogininforService logininforService; + + @Autowired + private SysPasswordService passwordService; + + @PreAuthorize("@ss.hasPermi('monitor:logininfor:list')") + @GetMapping("/list") + public TableDataInfo list(SysLogininfor logininfor) + { + startPage(); + List list = logininforService.selectLogininforList(logininfor); + return getDataTable(list); + } + + @Log(title = "登录日志", businessType = BusinessType.EXPORT) + @PreAuthorize("@ss.hasPermi('monitor:logininfor:export')") + @PostMapping("/export") + public void export(HttpServletResponse response, SysLogininfor logininfor) + { + List list = logininforService.selectLogininforList(logininfor); + ExcelUtil util = new ExcelUtil(SysLogininfor.class); + util.exportExcel(response, list, "登录日志"); + } + + @PreAuthorize("@ss.hasPermi('monitor:logininfor:remove')") + @Log(title = "登录日志", businessType = BusinessType.DELETE) + @DeleteMapping("/{infoIds}") + public AjaxResult remove(@PathVariable Long[] infoIds) + { + return toAjax(logininforService.deleteLogininforByIds(infoIds)); + } + + @PreAuthorize("@ss.hasPermi('monitor:logininfor:remove')") + @Log(title = "登录日志", businessType = BusinessType.CLEAN) + @DeleteMapping("/clean") + public AjaxResult clean() + { + logininforService.cleanLogininfor(); + return success(); + } + + @PreAuthorize("@ss.hasPermi('monitor:logininfor:unlock')") + @Log(title = "账户解锁", businessType = BusinessType.OTHER) + @GetMapping("/unlock/{userName}") + public AjaxResult unlock(@PathVariable("userName") String userName) + { + passwordService.clearLoginRecordCache(userName); + return success(); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/SysOperlogController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/SysOperlogController.java new file mode 100644 index 00000000..4656451f --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/SysOperlogController.java @@ -0,0 +1,60 @@ +package com.ruoyi.web.controller.monitor; + +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.page.TableDataInfo; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.utils.poi.ExcelUtil; +import com.ruoyi.system.domain.SysOperLog; +import com.ruoyi.system.service.ISysOperLogService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.web.bind.annotation.*; + +import javax.servlet.http.HttpServletResponse; +import java.util.List; + +/** + * 操作日志记录 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/monitor/operlog") +public class SysOperlogController extends BaseController { + @Autowired + private ISysOperLogService operLogService; + + @PreAuthorize("@ss.hasPermi('monitor:operlog:list')") + @GetMapping("/list") + public TableDataInfo list(SysOperLog operLog) { + startPage(); + List list = operLogService.selectOperLogList(operLog); + return getDataTable(list); + } + + @Log(title = "操作日志", businessType = BusinessType.EXPORT) + @PreAuthorize("@ss.hasPermi('monitor:operlog:export')") + @PostMapping("/export") + public void export(HttpServletResponse response, SysOperLog operLog) { + List list = operLogService.selectOperLogList(operLog); + ExcelUtil util = new ExcelUtil(SysOperLog.class); + util.exportExcel(response, list, "操作日志"); + } + + @Log(title = "操作日志", businessType = BusinessType.DELETE) + @PreAuthorize("@ss.hasPermi('monitor:operlog:remove')") + @DeleteMapping("/{operIds}") + public AjaxResult remove(@PathVariable Long[] operIds) { + return toAjax(operLogService.deleteOperLogByIds(operIds)); + } + + @Log(title = "操作日志", businessType = BusinessType.CLEAN) + @PreAuthorize("@ss.hasPermi('monitor:operlog:remove')") + @DeleteMapping("/clean") + public AjaxResult clean() { + operLogService.cleanOperLog(); + return success(); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/SysUserOnlineController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/SysUserOnlineController.java new file mode 100644 index 00000000..45ecb03a --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/monitor/SysUserOnlineController.java @@ -0,0 +1,92 @@ +package com.ruoyi.web.controller.monitor; + +import java.util.ArrayList; +import java.util.Collection; +import java.util.Collections; +import java.util.List; + +import com.ruoyi.common.utils.EhcacheUtil; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.web.bind.annotation.DeleteMapping; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.constant.CacheConstants; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.common.core.page.TableDataInfo; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.system.domain.SysUserOnline; +import com.ruoyi.system.service.ISysUserOnlineService; + +/** + * 在线用户监控 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/monitor/online") +public class SysUserOnlineController extends BaseController +{ + @Autowired + private ISysUserOnlineService userOnlineService; + + + @PreAuthorize("@ss.hasPermi('monitor:online:list')") + @GetMapping("/list") + public TableDataInfo list(String ipaddr, String userName) + { +//// Collection keys = redisCache.keys(CacheConstants.LOGIN_TOKEN_KEY + "*"); +// List userOnlineList = new ArrayList(); +// for (String key : keys) +// { +// LoginUser user = redisCache.getCacheObject(key); +// if (StringUtils.isNotEmpty(ipaddr) && StringUtils.isNotEmpty(userName)) +// { +// if (StringUtils.equals(ipaddr, user.getIpaddr()) && StringUtils.equals(userName, user.getUsername())) +// { +// userOnlineList.add(userOnlineService.selectOnlineByInfo(ipaddr, userName, user)); +// } +// } +// else if (StringUtils.isNotEmpty(ipaddr)) +// { +// if (StringUtils.equals(ipaddr, user.getIpaddr())) +// { +// userOnlineList.add(userOnlineService.selectOnlineByIpaddr(ipaddr, user)); +// } +// } +// else if (StringUtils.isNotEmpty(userName) && StringUtils.isNotNull(user.getUser())) +// { +// if (StringUtils.equals(userName, user.getUsername())) +// { +// userOnlineList.add(userOnlineService.selectOnlineByUserName(userName, user)); +// } +// } +// else +// { +// userOnlineList.add(userOnlineService.loginUserToUserOnline(user)); +// } +// } +// Collections.reverse(userOnlineList); +// userOnlineList.removeAll(Collections.singleton(null)); +// return getDataTable(userOnlineList); + return null; + } + + /** + * 强退用户 + */ + @PreAuthorize("@ss.hasPermi('monitor:online:forceLogout')") + @Log(title = "在线用户", businessType = BusinessType.FORCE) + @DeleteMapping("/{tokenId}") + public AjaxResult forceLogout(@PathVariable String tokenId) + { +// redisCache.deleteObject(CacheConstants.LOGIN_TOKEN_KEY + tokenId); + return AjaxResult.success(); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysConfigController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysConfigController.java new file mode 100644 index 00000000..922687c3 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysConfigController.java @@ -0,0 +1,117 @@ +package com.ruoyi.web.controller.system; + +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.page.TableDataInfo; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.utils.poi.ExcelUtil; +import com.ruoyi.system.domain.SysConfig; +import com.ruoyi.system.service.ISysConfigService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import javax.servlet.http.HttpServletResponse; +import java.util.List; + +/** + * 参数配置 信息操作处理 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/system/config") +public class SysConfigController extends BaseController { + @Autowired + private ISysConfigService configService; + + /** + * 获取参数配置列表 + */ + @PreAuthorize("@ss.hasPermi('system:config:list')") + @GetMapping("/list") + public TableDataInfo list(SysConfig config) { + startPage(); + List list = configService.selectConfigList(config); + return getDataTable(list); + } + + @Log(title = "参数管理", businessType = BusinessType.EXPORT) + @PreAuthorize("@ss.hasPermi('system:config:export')") + @PostMapping("/export") + public void export(HttpServletResponse response, SysConfig config) { + List list = configService.selectConfigList(config); + ExcelUtil util = new ExcelUtil(SysConfig.class); + util.exportExcel(response, list, "参数数据"); + } + + /** + * 根据参数编号获取详细信息 + */ + @PreAuthorize("@ss.hasPermi('system:config:query')") + @GetMapping(value = "/{configId}") + public AjaxResult getInfo(@PathVariable Long configId) { + return success(configService.selectConfigById(configId)); + } + + /** + * 根据参数键名查询参数值 + */ + @GetMapping(value = "/configKey/{configKey}") + public AjaxResult getConfigKey(@PathVariable String configKey) { + return success(configService.selectConfigByKey(configKey)); + } + + /** + * 新增参数配置 + */ + @PreAuthorize("@ss.hasPermi('system:config:add')") + @Log(title = "参数管理", businessType = BusinessType.INSERT) + @PostMapping + public AjaxResult add(@Validated @RequestBody SysConfig config) { + if (UserConstants.NOT_UNIQUE.equals(configService.checkConfigKeyUnique(config))) { + return error("新增参数'" + config.getConfigName() + "'失败,参数键名已存在"); + } + config.setCreateBy(getUsername()); + return toAjax(configService.insertConfig(config)); + } + + /** + * 修改参数配置 + */ + @PreAuthorize("@ss.hasPermi('system:config:edit')") + @Log(title = "参数管理", businessType = BusinessType.UPDATE) + @PutMapping + public AjaxResult edit(@Validated @RequestBody SysConfig config) { + if (UserConstants.NOT_UNIQUE.equals(configService.checkConfigKeyUnique(config))) { + return error("修改参数'" + config.getConfigName() + "'失败,参数键名已存在"); + } + config.setUpdateBy(getUsername()); + return toAjax(configService.updateConfig(config)); + } + + /** + * 删除参数配置 + */ + @PreAuthorize("@ss.hasPermi('system:config:remove')") + @Log(title = "参数管理", businessType = BusinessType.DELETE) + @DeleteMapping("/{configIds}") + public AjaxResult remove(@PathVariable Long[] configIds) { + configService.deleteConfigByIds(configIds); + return success(); + } + + /** + * 刷新参数缓存 + */ + @PreAuthorize("@ss.hasPermi('system:config:remove')") + @Log(title = "参数管理", businessType = BusinessType.CLEAN) + @DeleteMapping("/refreshCache") + public AjaxResult refreshCache() { + configService.resetConfigCache(); + return success(); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysDeptController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysDeptController.java new file mode 100644 index 00000000..79a73787 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysDeptController.java @@ -0,0 +1,125 @@ +package com.ruoyi.web.controller.system; + +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.domain.entity.SysDept; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.system.service.ISysDeptService; +import org.apache.commons.lang3.ArrayUtils; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import java.util.Comparator; +import java.util.List; + +/** + * 部门信息 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/system/dept") +public class SysDeptController extends BaseController { + @Autowired + private ISysDeptService deptService; + + /** + * 获取部门列表 + */ + @PreAuthorize("@ss.hasPermi('system:dept:list')") + @GetMapping("/list") + public AjaxResult list(SysDept dept) { + List depts = deptService.selectDeptList(dept); + return success(depts); + } + + /** + * 查询部门列表(排除节点) + */ + @PreAuthorize("@ss.hasPermi('system:dept:list')") + @GetMapping("/list/exclude/{deptId}") + public AjaxResult excludeChild(@PathVariable(value = "deptId", required = false) Long deptId) { + List depts = deptService.selectDeptList(new SysDept()); + depts.removeIf(d -> d.getDeptId().intValue() == deptId || ArrayUtils.contains(StringUtils.split(d.getAncestors(), ","), deptId + "")); + return success(depts); + } + + /** + * 根据部门编号获取详细信息 + */ + @PreAuthorize("@ss.hasPermi('system:dept:query')") + @GetMapping(value = "/{deptId}") + public AjaxResult getInfo(@PathVariable Long deptId) { + deptService.checkDeptDataScope(deptId); + return success(deptService.selectDeptById(deptId)); + } + + /** + * 新增部门 + */ + @PreAuthorize("@ss.hasPermi('system:dept:add')") + @Log(title = "部门管理", businessType = BusinessType.INSERT) + @PostMapping + public AjaxResult add(@Validated @RequestBody SysDept dept) { + if (UserConstants.NOT_UNIQUE.equals(deptService.checkDeptNameUnique(dept))) { + return error("新增部门'" + dept.getDeptName() + "'失败,部门名称已存在"); + } + dept.setCreateBy(getUsername()); + return toAjax(deptService.insertDept(dept)); + } + + /** + * 修改部门 + */ + @PreAuthorize("@ss.hasPermi('system:dept:edit')") + @Log(title = "部门管理", businessType = BusinessType.UPDATE) + @PutMapping + public AjaxResult edit(@Validated @RequestBody SysDept dept) { + Long deptId = dept.getDeptId(); + deptService.checkDeptDataScope(deptId); + if (UserConstants.NOT_UNIQUE.equals(deptService.checkDeptNameUnique(dept))) { + return error("修改部门'" + dept.getDeptName() + "'失败,部门名称已存在"); + } else if (dept.getParentId().equals(deptId)) { + return error("修改部门'" + dept.getDeptName() + "'失败,上级部门不能是自己"); + } else if (StringUtils.equals(UserConstants.DEPT_DISABLE, dept.getStatus()) && deptService.selectNormalChildrenDeptById(deptId) > 0) { + return error("该部门包含未停用的子部门!"); + } + dept.setUpdateBy(getUsername()); + return toAjax(deptService.updateDept(dept)); + } + + /** + * 删除部门 + */ + @PreAuthorize("@ss.hasPermi('system:dept:remove')") + @Log(title = "部门管理", businessType = BusinessType.DELETE) + @DeleteMapping("/{deptId}") + public AjaxResult remove(@PathVariable Long deptId) { + if (deptService.hasChildByDeptId(deptId)) { + return warn("存在下级部门,不允许删除"); + } + if (deptService.checkDeptExistUser(deptId)) { + return warn("部门存在用户,不允许删除"); + } + deptService.checkDeptDataScope(deptId); + return toAjax(deptService.deleteDeptById(deptId)); + } + + /** + * 获取部门精简信息列表 + * 只包含被开启的部门,主要用于前端的下拉选项 + */ + @GetMapping("/list-all-simple") + public AjaxResult getSimpleDepts() { + List list = deptService.selectDeptList(new SysDept()); + // 排序后,返回给前端 + list.sort(Comparator.comparing(SysDept::getOrderNum)); + return success(list); + } + +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysDictDataController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysDictDataController.java new file mode 100644 index 00000000..8f4d4843 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysDictDataController.java @@ -0,0 +1,106 @@ +package com.ruoyi.web.controller.system; + +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.domain.entity.SysDictData; +import com.ruoyi.common.core.page.TableDataInfo; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.poi.ExcelUtil; +import com.ruoyi.system.service.ISysDictDataService; +import com.ruoyi.system.service.ISysDictTypeService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import javax.servlet.http.HttpServletResponse; +import java.util.ArrayList; +import java.util.List; + +/** + * 数据字典信息 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/system/dict/data") +public class SysDictDataController extends BaseController { + @Autowired + private ISysDictDataService dictDataService; + + @Autowired + private ISysDictTypeService dictTypeService; + + @PreAuthorize("@ss.hasPermi('system:dict:list')") + @GetMapping("/list") + public TableDataInfo list(SysDictData dictData) { + startPage(); + List list = dictDataService.selectDictDataList(dictData); + return getDataTable(list); + } + + @Log(title = "字典数据", businessType = BusinessType.EXPORT) + @PreAuthorize("@ss.hasPermi('system:dict:export')") + @PostMapping("/export") + public void export(HttpServletResponse response, SysDictData dictData) { + List list = dictDataService.selectDictDataList(dictData); + ExcelUtil util = new ExcelUtil(SysDictData.class); + util.exportExcel(response, list, "字典数据"); + } + + /** + * 查询字典数据详细 + */ + @PreAuthorize("@ss.hasPermi('system:dict:query')") + @GetMapping(value = "/{dictCode}") + public AjaxResult getInfo(@PathVariable Long dictCode) { + return success(dictDataService.selectDictDataById(dictCode)); + } + + /** + * 根据字典类型查询字典数据信息 + */ + @GetMapping(value = "/type/{dictType}") + public AjaxResult dictType(@PathVariable String dictType) { + List data = dictTypeService.selectDictDataByType(dictType); + if (StringUtils.isNull(data)) { + data = new ArrayList(); + } + return success(data); + } + + /** + * 新增字典类型 + */ + @PreAuthorize("@ss.hasPermi('system:dict:add')") + @Log(title = "字典数据", businessType = BusinessType.INSERT) + @PostMapping + public AjaxResult add(@Validated @RequestBody SysDictData dict) { + dict.setCreateBy(getUsername()); + return toAjax(dictDataService.insertDictData(dict)); + } + + /** + * 修改保存字典类型 + */ + @PreAuthorize("@ss.hasPermi('system:dict:edit')") + @Log(title = "字典数据", businessType = BusinessType.UPDATE) + @PutMapping + public AjaxResult edit(@Validated @RequestBody SysDictData dict) { + dict.setUpdateBy(getUsername()); + return toAjax(dictDataService.updateDictData(dict)); + } + + /** + * 删除字典类型 + */ + @PreAuthorize("@ss.hasPermi('system:dict:remove')") + @Log(title = "字典类型", businessType = BusinessType.DELETE) + @DeleteMapping("/{dictCodes}") + public AjaxResult remove(@PathVariable Long[] dictCodes) { + dictDataService.deleteDictDataByIds(dictCodes); + return success(); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysDictTypeController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysDictTypeController.java new file mode 100644 index 00000000..7ac33a21 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysDictTypeController.java @@ -0,0 +1,115 @@ +package com.ruoyi.web.controller.system; + +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.domain.entity.SysDictType; +import com.ruoyi.common.core.page.TableDataInfo; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.utils.poi.ExcelUtil; +import com.ruoyi.system.service.ISysDictTypeService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import javax.servlet.http.HttpServletResponse; +import java.util.List; + +/** + * 数据字典信息 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/system/dict/type") +public class SysDictTypeController extends BaseController { + @Autowired + private ISysDictTypeService dictTypeService; + + @PreAuthorize("@ss.hasPermi('system:dict:list')") + @GetMapping("/list") + public TableDataInfo list(SysDictType dictType) { + startPage(); + List list = dictTypeService.selectDictTypeList(dictType); + return getDataTable(list); + } + + @Log(title = "字典类型", businessType = BusinessType.EXPORT) + @PreAuthorize("@ss.hasPermi('system:dict:export')") + @PostMapping("/export") + public void export(HttpServletResponse response, SysDictType dictType) { + List list = dictTypeService.selectDictTypeList(dictType); + ExcelUtil util = new ExcelUtil(SysDictType.class); + util.exportExcel(response, list, "字典类型"); + } + + /** + * 查询字典类型详细 + */ + @PreAuthorize("@ss.hasPermi('system:dict:query')") + @GetMapping(value = "/{dictId}") + public AjaxResult getInfo(@PathVariable Long dictId) { + return success(dictTypeService.selectDictTypeById(dictId)); + } + + /** + * 新增字典类型 + */ + @PreAuthorize("@ss.hasPermi('system:dict:add')") + @Log(title = "字典类型", businessType = BusinessType.INSERT) + @PostMapping + public AjaxResult add(@Validated @RequestBody SysDictType dict) { + if (UserConstants.NOT_UNIQUE.equals(dictTypeService.checkDictTypeUnique(dict))) { + return error("新增字典'" + dict.getDictName() + "'失败,字典类型已存在"); + } + dict.setCreateBy(getUsername()); + return toAjax(dictTypeService.insertDictType(dict)); + } + + /** + * 修改字典类型 + */ + @PreAuthorize("@ss.hasPermi('system:dict:edit')") + @Log(title = "字典类型", businessType = BusinessType.UPDATE) + @PutMapping + public AjaxResult edit(@Validated @RequestBody SysDictType dict) { + if (UserConstants.NOT_UNIQUE.equals(dictTypeService.checkDictTypeUnique(dict))) { + return error("修改字典'" + dict.getDictName() + "'失败,字典类型已存在"); + } + dict.setUpdateBy(getUsername()); + return toAjax(dictTypeService.updateDictType(dict)); + } + + /** + * 删除字典类型 + */ + @PreAuthorize("@ss.hasPermi('system:dict:remove')") + @Log(title = "字典类型", businessType = BusinessType.DELETE) + @DeleteMapping("/{dictIds}") + public AjaxResult remove(@PathVariable Long[] dictIds) { + dictTypeService.deleteDictTypeByIds(dictIds); + return success(); + } + + /** + * 刷新字典缓存 + */ + @PreAuthorize("@ss.hasPermi('system:dict:remove')") + @Log(title = "字典类型", businessType = BusinessType.CLEAN) + @DeleteMapping("/refreshCache") + public AjaxResult refreshCache() { + dictTypeService.resetDictCache(); + return success(); + } + + /** + * 获取字典选择框列表 + */ + @GetMapping("/optionselect") + public AjaxResult optionselect() { + List dictTypes = dictTypeService.selectDictTypeAll(); + return success(dictTypes); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysIndexController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysIndexController.java new file mode 100644 index 00000000..13007eb1 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysIndexController.java @@ -0,0 +1,29 @@ +package com.ruoyi.web.controller.system; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; +import com.ruoyi.common.config.RuoYiConfig; +import com.ruoyi.common.utils.StringUtils; + +/** + * 首页 + * + * @author ruoyi + */ +@RestController +public class SysIndexController +{ + /** 系统基础配置 */ + @Autowired + private RuoYiConfig ruoyiConfig; + + /** + * 访问首页,提示语 + */ + @RequestMapping("/") + public String index() + { + return StringUtils.format("欢迎使用{}后台管理框架,当前版本:v{},请通过前端地址访问。", ruoyiConfig.getName(), ruoyiConfig.getVersion()); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysLoginController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysLoginController.java new file mode 100644 index 00000000..f2e59cc8 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysLoginController.java @@ -0,0 +1,83 @@ +package com.ruoyi.web.controller.system; + +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.domain.entity.SysMenu; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.core.domain.model.LoginBody; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.framework.web.service.SysLoginService; +import com.ruoyi.framework.web.service.SysPermissionService; +import com.ruoyi.system.service.ISysMenuService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RestController; + +import java.util.List; +import java.util.Set; + +/** + * 登录验证 + * + * @author ruoyi + */ +@RestController +public class SysLoginController { + @Autowired + private SysLoginService loginService; + + @Autowired + private ISysMenuService menuService; + + @Autowired + private SysPermissionService permissionService; + + /** + * 登录方法 + * + * @param loginBody 登录信息 + * @return 结果 + */ + @PostMapping("/login") + public AjaxResult login(@RequestBody LoginBody loginBody) { + AjaxResult ajax = AjaxResult.success(); + // 生成令牌 + String token = loginService.login(loginBody.getUsername(), loginBody.getPassword(), loginBody.getCode(), + loginBody.getUuid()); + ajax.put(Constants.TOKEN, token); + return ajax; + } + + /** + * 获取用户信息 + * + * @return 用户信息 + */ + @GetMapping("getInfo") + public AjaxResult getInfo() { + SysUser user = SecurityUtils.getLoginUser().getUser(); + // 角色集合 + Set roles = permissionService.getRolePermission(user); + // 权限集合 + Set permissions = permissionService.getMenuPermission(user); + AjaxResult ajax = AjaxResult.success(); + ajax.put("user", user); + ajax.put("roles", roles); + ajax.put("permissions", permissions); + return ajax; + } + + /** + * 获取路由信息 + * + * @return 路由信息 + */ + @GetMapping("getRouters") + public AjaxResult getRouters() { + Long userId = SecurityUtils.getUserId(); + List menus = menuService.selectMenuTreeByUserId(userId); + return AjaxResult.success(menuService.buildMenus(menus)); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysMenuController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysMenuController.java new file mode 100644 index 00000000..9da3fa4f --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysMenuController.java @@ -0,0 +1,156 @@ +package com.ruoyi.web.controller.system; + +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.domain.R; +import com.ruoyi.common.core.domain.entity.SysMenu; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.enums.CommonStatusEnum; +import com.ruoyi.common.mybatis.query.LambdaQueryWrapperX; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.system.domain.vo.menu.MenuSimpleRespVO; +import com.ruoyi.system.service.ISysMenuService; +import io.swagger.annotations.ApiOperation; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import java.util.ArrayList; +import java.util.Comparator; +import java.util.List; + +/** + * 菜单信息 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/system/menu") +public class SysMenuController extends BaseController { + @Autowired + private ISysMenuService menuService; + + /** + * 获取菜单列表 + */ + @PreAuthorize("@ss.hasPermi('system:menu:list')") + @GetMapping("/list") + public AjaxResult list(SysMenu menu) { + List menus = menuService.selectMenuList(menu, getUserId()); + return success(menus); + } + + /** + * 根据菜单编号获取详细信息 + */ + @PreAuthorize("@ss.hasPermi('system:menu:query')") + @GetMapping(value = "/{menuId}") + public AjaxResult getInfo(@PathVariable Long menuId) { + return success(menuService.selectMenuById(menuId)); + } + + /** + * 获取菜单下拉树列表 + */ + @GetMapping("/treeselect") + public AjaxResult treeselect(SysMenu menu) { + List menus = menuService.selectMenuList(menu, getUserId()); + return success(menuService.buildMenuTreeSelect(menus)); + } + + /** + * 加载对应角色菜单列表树 + */ + @GetMapping(value = "/roleMenuTreeselect/{roleId}") + public AjaxResult roleMenuTreeselect(@PathVariable("roleId") Long roleId) { + List menus = menuService.selectMenuList(getUserId()); + AjaxResult ajax = success(); + ajax.put("checkedKeys", menuService.selectMenuListByRoleId(roleId)); + ajax.put("menus", menuService.buildMenuTreeSelect(menus)); + return ajax; + } + + /** + * 新增菜单 + */ + @PreAuthorize("@ss.hasPermi('system:menu:add')") + @Log(title = "菜单管理", businessType = BusinessType.INSERT) + @PostMapping + public AjaxResult add(@Validated @RequestBody SysMenu menu) { + if (UserConstants.NOT_UNIQUE.equals(menuService.checkMenuNameUnique(menu))) { + return error("新增菜单'" + menu.getMenuName() + "'失败,菜单名称已存在"); + } else if (UserConstants.YES_FRAME.equals(menu.getIsFrame()) && !StringUtils.ishttp(menu.getPath())) { + return error("新增菜单'" + menu.getMenuName() + "'失败,地址必须以http(s)://开头"); + } + menu.setCreateBy(getUsername()); + return toAjax(menuService.insertMenu(menu)); + } + + /** + * 修改菜单 + */ + @PreAuthorize("@ss.hasPermi('system:menu:edit')") + @Log(title = "菜单管理", businessType = BusinessType.UPDATE) + @PutMapping + public AjaxResult edit(@Validated @RequestBody SysMenu menu) { + if (UserConstants.NOT_UNIQUE.equals(menuService.checkMenuNameUnique(menu))) { + return error("修改菜单'" + menu.getMenuName() + "'失败,菜单名称已存在"); + } else if (UserConstants.YES_FRAME.equals(menu.getIsFrame()) && !StringUtils.ishttp(menu.getPath())) { + return error("修改菜单'" + menu.getMenuName() + "'失败,地址必须以http(s)://开头"); + } else if (menu.getMenuId().equals(menu.getParentId())) { + return error("修改菜单'" + menu.getMenuName() + "'失败,上级菜单不能选择自己"); + } + menu.setUpdateBy(getUsername()); + return toAjax(menuService.updateMenu(menu)); + } + + /** + * 删除菜单 + */ + @PreAuthorize("@ss.hasPermi('system:menu:remove')") + @Log(title = "菜单管理", businessType = BusinessType.DELETE) + @DeleteMapping("/{menuId}") + public AjaxResult remove(@PathVariable("menuId") Long menuId) { + if (menuService.hasChildByMenuId(menuId)) { + return warn("存在子菜单,不允许删除"); + } + if (menuService.checkMenuExistRole(menuId)) { + return warn("菜单已分配,不允许删除"); + } + return toAjax(menuService.deleteMenuById(menuId)); + } + + @GetMapping("/list-all-simple") + @ApiOperation(value = "获取菜单精简信息列表", notes = "只包含被开启的菜单,用于【角色分配菜单】功能的选项。" + "在多租户的场景下,会只返回租户所在套餐有的菜单") + public R> getSimpleMenus() { + // 获得菜单列表,只要开启状态的 +// List list = menuService.list(new LambdaQueryWrapperX() +// .eqIfPresent(SysMenu::getStatus, "0")); + SysMenu sysMenu = new SysMenu(); +// sysMenu.setStatus("0"); +// sysMenu.setVisible("0"); + List list = menuService.getTenantMenus(sysMenu); + // 排序后,返回给前端 + list.sort(Comparator.comparing(SysMenu::getOrderNum)); + List results = new ArrayList<>(); + for (SysMenu menu : list) { + MenuSimpleRespVO vo = new MenuSimpleRespVO(); + vo.setId(menu.getMenuId()); + vo.setName(menu.getMenuName()); + vo.setParentId(menu.getParentId()); + String menuType = menu.getMenuType(); + if ("M".equals(menuType)) { + vo.setType(1); + }else if ("C".equals(menuType)) { + vo.setType(2); + }else if ("F".equals(menuType)) { + vo.setType(3); + } + results.add(vo); + } + return R.ok(results); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysNoticeController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysNoticeController.java new file mode 100644 index 00000000..61c72cf0 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysNoticeController.java @@ -0,0 +1,79 @@ +package com.ruoyi.web.controller.system; + +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.page.TableDataInfo; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.system.domain.SysNotice; +import com.ruoyi.system.service.ISysNoticeService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import java.util.List; + +/** + * 公告 信息操作处理 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/system/notice") +public class SysNoticeController extends BaseController { + @Autowired + private ISysNoticeService noticeService; + + /** + * 获取通知公告列表 + */ + @PreAuthorize("@ss.hasPermi('system:notice:list')") + @GetMapping("/list") + public TableDataInfo list(SysNotice notice) { + startPage(); + List list = noticeService.selectNoticeList(notice); + return getDataTable(list); + } + + /** + * 根据通知公告编号获取详细信息 + */ + @PreAuthorize("@ss.hasPermi('system:notice:query')") + @GetMapping(value = "/{noticeId}") + public AjaxResult getInfo(@PathVariable Long noticeId) { + return success(noticeService.selectNoticeById(noticeId)); + } + + /** + * 新增通知公告 + */ + @PreAuthorize("@ss.hasPermi('system:notice:add')") + @Log(title = "通知公告", businessType = BusinessType.INSERT) + @PostMapping + public AjaxResult add(@Validated @RequestBody SysNotice notice) { + notice.setCreateBy(getUsername()); + return toAjax(noticeService.insertNotice(notice)); + } + + /** + * 修改通知公告 + */ + @PreAuthorize("@ss.hasPermi('system:notice:edit')") + @Log(title = "通知公告", businessType = BusinessType.UPDATE) + @PutMapping + public AjaxResult edit(@Validated @RequestBody SysNotice notice) { + notice.setUpdateBy(getUsername()); + return toAjax(noticeService.updateNotice(notice)); + } + + /** + * 删除通知公告 + */ + @PreAuthorize("@ss.hasPermi('system:notice:remove')") + @Log(title = "通知公告", businessType = BusinessType.DELETE) + @DeleteMapping("/{noticeIds}") + public AjaxResult remove(@PathVariable Long[] noticeIds) { + return toAjax(noticeService.deleteNoticeByIds(noticeIds)); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysPostController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysPostController.java new file mode 100644 index 00000000..5beba7d0 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysPostController.java @@ -0,0 +1,123 @@ +package com.ruoyi.web.controller.system; + +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.page.TableDataInfo; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.utils.poi.ExcelUtil; +import com.ruoyi.system.domain.SysPost; +import com.ruoyi.system.service.ISysPostService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import javax.servlet.http.HttpServletResponse; +import java.util.Comparator; +import java.util.List; + +/** + * 岗位信息操作处理 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/system/post") +public class SysPostController extends BaseController { + @Autowired + private ISysPostService postService; + + /** + * 获取岗位列表 + */ + @PreAuthorize("@ss.hasPermi('system:post:list')") + @GetMapping("/list") + public TableDataInfo list(SysPost post) { + startPage(); + List list = postService.selectPostList(post); + return getDataTable(list); + } + + @Log(title = "岗位管理", businessType = BusinessType.EXPORT) + @PreAuthorize("@ss.hasPermi('system:post:export')") + @PostMapping("/export") + public void export(HttpServletResponse response, SysPost post) { + List list = postService.selectPostList(post); + ExcelUtil util = new ExcelUtil(SysPost.class); + util.exportExcel(response, list, "岗位数据"); + } + + /** + * 根据岗位编号获取详细信息 + */ + @PreAuthorize("@ss.hasPermi('system:post:query')") + @GetMapping(value = "/{postId}") + public AjaxResult getInfo(@PathVariable Long postId) { + return success(postService.selectPostById(postId)); + } + + /** + * 新增岗位 + */ + @PreAuthorize("@ss.hasPermi('system:post:add')") + @Log(title = "岗位管理", businessType = BusinessType.INSERT) + @PostMapping + public AjaxResult add(@Validated @RequestBody SysPost post) { + if (UserConstants.NOT_UNIQUE.equals(postService.checkPostNameUnique(post))) { + return error("新增岗位'" + post.getPostName() + "'失败,岗位名称已存在"); + } else if (UserConstants.NOT_UNIQUE.equals(postService.checkPostCodeUnique(post))) { + return error("新增岗位'" + post.getPostName() + "'失败,岗位编码已存在"); + } + post.setCreateBy(getUsername()); + return toAjax(postService.insertPost(post)); + } + + /** + * 修改岗位 + */ + @PreAuthorize("@ss.hasPermi('system:post:edit')") + @Log(title = "岗位管理", businessType = BusinessType.UPDATE) + @PutMapping + public AjaxResult edit(@Validated @RequestBody SysPost post) { + if (UserConstants.NOT_UNIQUE.equals(postService.checkPostNameUnique(post))) { + return error("修改岗位'" + post.getPostName() + "'失败,岗位名称已存在"); + } else if (UserConstants.NOT_UNIQUE.equals(postService.checkPostCodeUnique(post))) { + return error("修改岗位'" + post.getPostName() + "'失败,岗位编码已存在"); + } + post.setUpdateBy(getUsername()); + return toAjax(postService.updatePost(post)); + } + + /** + * 删除岗位 + */ + @PreAuthorize("@ss.hasPermi('system:post:remove')") + @Log(title = "岗位管理", businessType = BusinessType.DELETE) + @DeleteMapping("/{postIds}") + public AjaxResult remove(@PathVariable Long[] postIds) { + return toAjax(postService.deletePostByIds(postIds)); + } + + /** + * 获取岗位选择框列表 + */ + @GetMapping("/optionselect") + public AjaxResult optionselect() { + List posts = postService.selectPostAll(); + return success(posts); + } + + /** + * 获取岗位精简信息列表 + * 只包含被开启的岗位,主要用于前端的下拉选项 + */ + @GetMapping("/list-all-simple") + public AjaxResult getSimplePosts() { + List list = postService.selectPostList(new SysPost()); + // 排序后,返回给前端 + list.sort(Comparator.comparing(SysPost::getPostSort)); + return success(list); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysProfileController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysProfileController.java new file mode 100644 index 00000000..e3d10c0d --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysProfileController.java @@ -0,0 +1,125 @@ +package com.ruoyi.web.controller.system; + +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.config.RuoYiConfig; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.file.FileUploadUtils; +import com.ruoyi.common.utils.file.MimeTypeUtils; +import com.ruoyi.framework.web.service.TokenService; +import com.ruoyi.system.service.ISysUserService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.*; +import org.springframework.web.multipart.MultipartFile; + +/** + * 个人信息 业务处理 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/system/user/profile") +public class SysProfileController extends BaseController { + @Autowired + private ISysUserService userService; + + @Autowired + private TokenService tokenService; + + /** + * 个人信息 + */ + @GetMapping + public AjaxResult profile() { + LoginUser loginUser = getLoginUser(); + SysUser user = loginUser.getUser(); + AjaxResult ajax = success(user); + ajax.put("roleGroup", userService.selectUserRoleGroup(loginUser.getUsername())); + ajax.put("postGroup", userService.selectUserPostGroup(loginUser.getUsername())); + return ajax; + } + + /** + * 修改用户 + */ + @Log(title = "个人信息", businessType = BusinessType.UPDATE) + @PutMapping + public AjaxResult updateProfile(@RequestBody SysUser user) { + LoginUser loginUser = getLoginUser(); + SysUser sysUser = loginUser.getUser(); + user.setUserName(sysUser.getUserName()); + if (StringUtils.isNotEmpty(user.getPhonenumber()) + && UserConstants.NOT_UNIQUE.equals(userService.checkPhoneUnique(user))) { + return error("修改用户'" + user.getUserName() + "'失败,手机号码已存在"); + } + if (StringUtils.isNotEmpty(user.getEmail()) + && UserConstants.NOT_UNIQUE.equals(userService.checkEmailUnique(user))) { + return error("修改用户'" + user.getUserName() + "'失败,邮箱账号已存在"); + } + user.setUserId(sysUser.getUserId()); + user.setPassword(null); + user.setAvatar(null); + user.setDeptId(null); + if (userService.updateUserProfile(user) > 0) { + // 更新缓存用户信息 + sysUser.setNickName(user.getNickName()); + sysUser.setPhonenumber(user.getPhonenumber()); + sysUser.setEmail(user.getEmail()); + sysUser.setSex(user.getSex()); + tokenService.setLoginUser(loginUser); + return success(); + } + return error("修改个人信息异常,请联系管理员"); + } + + /** + * 重置密码 + */ + @Log(title = "个人信息", businessType = BusinessType.UPDATE) + @PutMapping("/updatePwd") + public AjaxResult updatePwd(String oldPassword, String newPassword) { + LoginUser loginUser = getLoginUser(); + String userName = loginUser.getUsername(); + String password = loginUser.getPassword(); + if (!SecurityUtils.matchesPassword(oldPassword, password)) { + return error("修改密码失败,旧密码错误"); + } + if (SecurityUtils.matchesPassword(newPassword, password)) { + return error("新密码不能与旧密码相同"); + } + if (userService.resetUserPwd(userName, SecurityUtils.encryptPassword(newPassword)) > 0) { + // 更新缓存用户密码 + loginUser.getUser().setPassword(SecurityUtils.encryptPassword(newPassword)); + tokenService.setLoginUser(loginUser); + return success(); + } + return error("修改密码异常,请联系管理员"); + } + + /** + * 头像上传 + */ + @Log(title = "用户头像", businessType = BusinessType.UPDATE) + @PostMapping("/avatar") + public AjaxResult avatar(@RequestParam("avatarfile") MultipartFile file) throws Exception { + if (!file.isEmpty()) { + LoginUser loginUser = getLoginUser(); + String avatar = FileUploadUtils.upload(RuoYiConfig.getAvatarPath(), file, MimeTypeUtils.IMAGE_EXTENSION); + if (userService.updateUserAvatar(loginUser.getUsername(), avatar)) { + AjaxResult ajax = success(); + ajax.put("imgUrl", avatar); + // 更新缓存用户头像 + loginUser.getUser().setAvatar(avatar); + tokenService.setLoginUser(loginUser); + return ajax; + } + } + return error("上传图片异常,请联系管理员"); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysRegisterController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysRegisterController.java new file mode 100644 index 00000000..fe192492 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysRegisterController.java @@ -0,0 +1,38 @@ +package com.ruoyi.web.controller.system; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RestController; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.domain.model.RegisterBody; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.framework.web.service.SysRegisterService; +import com.ruoyi.system.service.ISysConfigService; + +/** + * 注册验证 + * + * @author ruoyi + */ +@RestController +public class SysRegisterController extends BaseController +{ + @Autowired + private SysRegisterService registerService; + + @Autowired + private ISysConfigService configService; + + @PostMapping("/register") + public AjaxResult register(@RequestBody RegisterBody user) + { + if (!("true".equals(configService.selectConfigByKey("sys.account.registerUser")))) + { + return error("当前系统没有开启注册功能!"); + } + String msg = registerService.register(user); + return StringUtils.isEmpty(msg) ? success() : error(msg); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysRoleController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysRoleController.java new file mode 100644 index 00000000..64dc2adf --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysRoleController.java @@ -0,0 +1,267 @@ +package com.ruoyi.web.controller.system; + +import cn.hutool.core.collection.CollUtil; +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.domain.entity.SysDept; +import com.ruoyi.common.core.domain.entity.SysRole; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.common.core.page.TableDataInfo; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.poi.ExcelUtil; +import com.ruoyi.framework.web.service.DataScopeService; +import com.ruoyi.framework.web.service.SysPermissionService; +import com.ruoyi.framework.web.service.TokenService; +import com.ruoyi.system.domain.SysUserRole; +import com.ruoyi.system.service.ISysDeptService; +import com.ruoyi.system.service.ISysRoleService; +import com.ruoyi.system.service.ISysUserService; +import com.ruoyi.system.service.TenantService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.context.annotation.Lazy; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import javax.annotation.Resource; +import javax.servlet.http.HttpServletResponse; +import java.util.*; +import java.util.stream.Collectors; + +/** + * 角色信息 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/system/role") +public class SysRoleController extends BaseController { + @Autowired + private ISysRoleService roleService; + + @Autowired + private TokenService tokenService; + + @Autowired + private SysPermissionService permissionService; + + @Autowired + private ISysUserService userService; + + @Autowired + private ISysDeptService deptService; + + @Autowired + private DataScopeService dataScopeService; + @Resource + @Lazy // 延迟,避免循环依赖报错 + private TenantService tenantService; + + @PreAuthorize("@ss.hasPermi('system:role:list')") + @GetMapping("/list") + public TableDataInfo list(SysRole role) { + startPage(); + List list = roleService.selectRoleList(role); + return getDataTable(list); + } + + @Log(title = "角色管理", businessType = BusinessType.EXPORT) + @PreAuthorize("@ss.hasPermi('system:role:export')") + @PostMapping("/export") + public void export(HttpServletResponse response, SysRole role) { + List list = roleService.selectRoleList(role); + ExcelUtil util = new ExcelUtil(SysRole.class); + util.exportExcel(response, list, "角色数据"); + } + + /** + * 根据角色编号获取详细信息 + */ + @PreAuthorize("@ss.hasPermi('system:role:query')") + @GetMapping(value = "/{roleId}") + public AjaxResult getInfo(@PathVariable Long roleId) { + roleService.checkRoleDataScope(roleId); + return success(roleService.selectRoleById(roleId)); + } + + /** + * 新增角色 + */ + @PreAuthorize("@ss.hasPermi('system:role:add')") + @Log(title = "角色管理", businessType = BusinessType.INSERT) + @PostMapping + public AjaxResult add(@Validated @RequestBody SysRole role) { + if (UserConstants.NOT_UNIQUE.equals(roleService.checkRoleNameUnique(role))) { + return error("新增角色'" + role.getRoleName() + "'失败,角色名称已存在"); + } else if (UserConstants.NOT_UNIQUE.equals(roleService.checkRoleKeyUnique(role))) { + return error("新增角色'" + role.getRoleName() + "'失败,角色权限已存在"); + } + role.setCreateBy(getUsername()); + // 开启多租户的情况下,需要过滤掉未开通的菜单 + Set tmids = Arrays.stream(role.getMenuIds()).collect(Collectors.toSet()); + tenantService.handleTenantMenu(menuIds -> tmids.removeIf(menuId -> !CollUtil.contains(menuIds, menuId))); + role.setMenuIds(tmids.toArray(new Long[tmids.size()])); + return toAjax(roleService.insertRole(role)); + + } + + /** + * 修改保存角色 + */ + @PreAuthorize("@ss.hasPermi('system:role:edit')") + @Log(title = "角色管理", businessType = BusinessType.UPDATE) + @PutMapping + public AjaxResult edit(@Validated @RequestBody SysRole role) { + roleService.checkRoleAllowed(role); + roleService.checkRoleDataScope(role.getRoleId()); + // 开启多租户的情况下,需要过滤掉未开通的菜单 + Set tmids = Arrays.stream(role.getMenuIds()).collect(Collectors.toSet()); + tenantService.handleTenantMenu(menuIds -> tmids.removeIf(menuId -> !CollUtil.contains(menuIds, menuId))); + role.setMenuIds(tmids.toArray(new Long[tmids.size()])); + if (UserConstants.NOT_UNIQUE.equals(roleService.checkRoleNameUnique(role))) { + return error("修改角色'" + role.getRoleName() + "'失败,角色名称已存在"); + } else if (UserConstants.NOT_UNIQUE.equals(roleService.checkRoleKeyUnique(role))) { + return error("修改角色'" + role.getRoleName() + "'失败,角色权限已存在"); + } + role.setUpdateBy(getUsername()); + + if (roleService.updateRole(role) > 0) { + // 更新缓存用户权限 + LoginUser loginUser = getLoginUser(); + if (StringUtils.isNotNull(loginUser.getUser()) && !loginUser.getUser().isAdmin()) { + loginUser.setPermissions(permissionService.getMenuPermission(loginUser.getUser())); + loginUser.setUser(userService.selectUserByUserName(loginUser.getUser().getUserName())); + loginUser.setDataScopes(dataScopeService.getUserDataScopeIdList(loginUser.getUserId(), loginUser.getDeptId())); + tokenService.setLoginUser(loginUser); + } + return success(); + } + return error("修改角色'" + role.getRoleName() + "'失败,请联系管理员"); + } + + /** + * 修改保存数据权限 + */ + @PreAuthorize("@ss.hasPermi('system:role:edit')") + @Log(title = "角色管理", businessType = BusinessType.UPDATE) + @PutMapping("/dataScope") + public AjaxResult dataScope(@RequestBody SysRole role) { + roleService.checkRoleAllowed(role); + roleService.checkRoleDataScope(role.getRoleId()); + return toAjax(roleService.authDataScope(role)); + } + + /** + * 状态修改 + */ + @PreAuthorize("@ss.hasPermi('system:role:edit')") + @Log(title = "角色管理", businessType = BusinessType.UPDATE) + @PutMapping("/changeStatus") + public AjaxResult changeStatus(@RequestBody SysRole role) { + roleService.checkRoleAllowed(role); + roleService.checkRoleDataScope(role.getRoleId()); + role.setUpdateBy(getUsername()); + return toAjax(roleService.updateRoleStatus(role)); + } + + /** + * 删除角色 + */ + @PreAuthorize("@ss.hasPermi('system:role:remove')") + @Log(title = "角色管理", businessType = BusinessType.DELETE) + @DeleteMapping("/{roleIds}") + public AjaxResult remove(@PathVariable Long[] roleIds) { + return toAjax(roleService.deleteRoleByIds(roleIds)); + } + + /** + * 获取角色选择框列表 + */ + @PreAuthorize("@ss.hasPermi('system:role:query')") + @GetMapping("/optionselect") + public AjaxResult optionselect() { + return success(roleService.selectRoleAll()); + } + + /** + * 查询已分配用户角色列表 + */ + @PreAuthorize("@ss.hasPermi('system:role:list')") + @GetMapping("/authUser/allocatedList") + public TableDataInfo allocatedList(SysUser user) { + startPage(); + List list = userService.selectAllocatedList(user); + return getDataTable(list); + } + + /** + * 查询未分配用户角色列表 + */ + @PreAuthorize("@ss.hasPermi('system:role:list')") + @GetMapping("/authUser/unallocatedList") + public TableDataInfo unallocatedList(SysUser user) { + startPage(); + List list = userService.selectUnallocatedList(user); + return getDataTable(list); + } + + /** + * 取消授权用户 + */ + @PreAuthorize("@ss.hasPermi('system:role:edit')") + @Log(title = "角色管理", businessType = BusinessType.GRANT) + @PutMapping("/authUser/cancel") + public AjaxResult cancelAuthUser(@RequestBody SysUserRole userRole) { + return toAjax(roleService.deleteAuthUser(userRole)); + } + + /** + * 批量取消授权用户 + */ + @PreAuthorize("@ss.hasPermi('system:role:edit')") + @Log(title = "角色管理", businessType = BusinessType.GRANT) + @PutMapping("/authUser/cancelAll") + public AjaxResult cancelAuthUserAll(Long roleId, Long[] userIds) { + return toAjax(roleService.deleteAuthUsers(roleId, userIds)); + } + + /** + * 批量选择用户授权 + */ + @PreAuthorize("@ss.hasPermi('system:role:edit')") + @Log(title = "角色管理", businessType = BusinessType.GRANT) + @PutMapping("/authUser/selectAll") + public AjaxResult selectAuthUserAll(Long roleId, Long[] userIds) { + roleService.checkRoleDataScope(roleId); + return toAjax(roleService.insertAuthUsers(roleId, userIds)); + } + + /** + * 获取对应角色部门树列表 + */ + @PreAuthorize("@ss.hasPermi('system:role:query')") + @GetMapping(value = "/deptTree/{roleId}") + public AjaxResult deptTree(@PathVariable("roleId") Long roleId) { + AjaxResult ajax = success(); + ajax.put("checkedKeys", deptService.selectDeptListByRoleId(roleId)); + ajax.put("depts", deptService.selectDeptTreeList(new SysDept())); + return ajax; + } + + /** + * 获取角色精简信息列表 + * 只包含被开启的角色,主要用于前端的下拉选项 + */ + @GetMapping("/list-all-simple") + public AjaxResult getSimpleRoles() { + List list = roleService.selectRoleList(new SysRole()); + // 排序后,返回给前端 + list.sort(Comparator.comparing(SysRole::getRoleSort)); + return success(list); + } + +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysTreeDictController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysTreeDictController.java new file mode 100644 index 00000000..e0eeeb3e --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysTreeDictController.java @@ -0,0 +1,84 @@ +package com.ruoyi.web.controller.system; + +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.page.TableDataInfo; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.utils.poi.ExcelUtil; +import com.ruoyi.system.domain.SysTreeDict; +import com.ruoyi.system.service.ISysTreeDictService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.*; + +import java.util.List; + +/** + * 系统树形Controller + * + * @author wangzongrun + * @date 2021-06-04 + */ +@RestController +@RequestMapping("system/dict/tree") +public class SysTreeDictController extends BaseController { + + @Autowired + private ISysTreeDictService SysTreeDictService; + + /** + * 分页查询系统树形列表 + */ + @GetMapping("/list") + public TableDataInfo list(SysTreeDict sysTreeDict) { + IPage list = SysTreeDictService.selectList(getPage(), sysTreeDict); + return getDataTable(list); + } + + /** + * 查询字典类型详细 + */ + @GetMapping(value = "/{dictId}") + public AjaxResult getInfo(@PathVariable String dictId) { + return success(SysTreeDictService.selectById(dictId)); + } + + /** + * 导出系统树形列表 + */ + @Log(title = "系统树形", businessType = BusinessType.EXPORT) + @GetMapping("/export") + public AjaxResult export(SysTreeDict sysTreeDict) { + List list = SysTreeDictService.selectListAll(sysTreeDict); + ExcelUtil util = new ExcelUtil(SysTreeDict.class); + return util.exportExcel(list, "系统树形"); + } + + /** + * 新增系统树形 + */ + @Log(title = "系统树形", businessType = BusinessType.INSERT) + @PostMapping + public AjaxResult add(@RequestBody SysTreeDict sysTreeDict) { + return toAjax(SysTreeDictService.insert(sysTreeDict)); + } + + /** + * 修改系统树形 + */ + @Log(title = "系统树形", businessType = BusinessType.UPDATE) + @PutMapping + public AjaxResult edit(@RequestBody SysTreeDict sysTreeDict) { + return toAjax(SysTreeDictService.update(sysTreeDict)); + } + + /** + * 删除系统树形 + */ + @Log(title = "系统树形", businessType = BusinessType.DELETE) + @DeleteMapping("/{sids}") + public AjaxResult remove(@PathVariable String[] sids) { + return toAjax(SysTreeDictService.deleteByIds(sids)); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysTreeDictDataController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysTreeDictDataController.java new file mode 100644 index 00000000..2cfad468 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysTreeDictDataController.java @@ -0,0 +1,87 @@ +package com.ruoyi.web.controller.system; + +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.system.domain.SysTreeDictData; +import com.ruoyi.system.service.ISysTreeDictDataService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.*; + +import java.util.List; + +/** + * 树形字典数据Controller + * + * @author wangzongrun + * @date 2021-05-31 + */ +@RestController +@RequestMapping("system/tree/dict/data") +public class SysTreeDictDataController extends BaseController { + + @Autowired + private ISysTreeDictDataService sysTreeDictDataService; + + /** + * 分页查询树形字典数据列表 + */ + @GetMapping("list") + public AjaxResult list(SysTreeDictData sysTreeDictData) { + List list = sysTreeDictDataService.buildTree(sysTreeDictData); + return success(list); + } + + /** + * 查询树形字典数据列表 + */ + @GetMapping("/tree") + public AjaxResult tree(SysTreeDictData sysTreeDictData) { + List list = sysTreeDictDataService.buildTree(sysTreeDictData); + return success(list); + } + + /** + * 新增树形字典数据 + */ + @Log(title = "树形字典数据", businessType = BusinessType.INSERT) + @PostMapping + public AjaxResult add(@RequestBody SysTreeDictData sysTreeDictData) { + return toAjax(sysTreeDictDataService.insert(sysTreeDictData)); + } + + /** + * 修改树形字典数据 + */ + @Log(title = "树形字典数据", businessType = BusinessType.UPDATE) + @PutMapping + public AjaxResult edit(@RequestBody SysTreeDictData sysTreeDictData) { + return toAjax(sysTreeDictDataService.update(sysTreeDictData)); + } + + /** + * 删除树形字典数据 + */ + @Log(title = "树形字典数据", businessType = BusinessType.DELETE) + @DeleteMapping("/{ids}") + public AjaxResult remove(@PathVariable String[] ids) { + return toAjax(sysTreeDictDataService.deleteByIds(ids)); + } + + /** + * 检查名称唯一性 + */ + @PostMapping(value = "/check/unique/label") + public AjaxResult checkUniqueByLabel(@RequestBody SysTreeDictData sysTreeDictData) { + return sysTreeDictDataService.checkUniqueByLabel(sysTreeDictData); + } + + /** + * 校验编码 + */ + @PostMapping(value = "/check/code") + public AjaxResult checkCode(@RequestBody SysTreeDictData sysTreeDictData) { + return sysTreeDictDataService.checkCode(sysTreeDictData); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysUserController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysUserController.java new file mode 100644 index 00000000..590ad98b --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/SysUserController.java @@ -0,0 +1,260 @@ +package com.ruoyi.web.controller.system; + +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.domain.entity.SysDept; +import com.ruoyi.common.core.domain.entity.SysRole; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.core.page.TableDataInfo; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.enums.CommonStatusEnum; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.poi.ExcelUtil; +import com.ruoyi.system.convert.user.UserConvert; +import com.ruoyi.system.service.ISysDeptService; +import com.ruoyi.system.service.ISysPostService; +import com.ruoyi.system.service.ISysRoleService; +import com.ruoyi.system.service.ISysUserService; +import org.apache.commons.lang3.ArrayUtils; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; +import org.springframework.web.multipart.MultipartFile; + +import javax.servlet.http.HttpServletResponse; +import java.util.ArrayList; +import java.util.List; +import java.util.stream.Collectors; + +/** + * 用户信息 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/system/user") +public class SysUserController extends BaseController { + @Autowired + private ISysUserService userService; + + @Autowired + private ISysRoleService roleService; + + @Autowired + private ISysDeptService deptService; + + @Autowired + private ISysPostService postService; + + /** + * 获取用户列表 + */ + @PreAuthorize("@ss.hasPermi('system:user:list')") + @GetMapping("/list") + public TableDataInfo list(SysUser user) { + startPage(); + List list = userService.selectUserList(user); + return getDataTable(list); + } + /** + * 获取用户列表 + */ + @PreAuthorize("@ss.hasPermi('system:user:xalist')") + @GetMapping("/xalist") + public TableDataInfo xalist(SysUser user) { + SysRole xiaoan = roleService.selectByRoleKey("xiaoan"); + if (xiaoan == null) { + return getDataTable(new ArrayList<>()); + } + startPage(); + user.setRoleId(xiaoan.getRoleId()); //角色 + List list = userService.selectUserList(user); + return getDataTable(list); + } + + @Log(title = "用户管理", businessType = BusinessType.EXPORT) + @PreAuthorize("@ss.hasPermi('system:user:export')") + @PostMapping("/export") + public void export(HttpServletResponse response, SysUser user) { + List list = userService.selectUserList(user); + ExcelUtil util = new ExcelUtil(SysUser.class); + util.exportExcel(response, list, "用户数据"); + } + + @Log(title = "用户管理", businessType = BusinessType.IMPORT) + @PreAuthorize("@ss.hasPermi('system:user:import')") + @PostMapping("/importData") + public AjaxResult importData(MultipartFile file, boolean updateSupport) throws Exception { + ExcelUtil util = new ExcelUtil(SysUser.class); + List userList = util.importExcel(file.getInputStream()); + String operName = getUsername(); + String message = userService.importUser(userList, updateSupport, operName); + return success(message); + } + + @PostMapping("/importTemplate") + public void importTemplate(HttpServletResponse response) { + ExcelUtil util = new ExcelUtil(SysUser.class); + util.importTemplateExcel(response, "用户数据"); + } + + /** + * 根据用户编号获取详细信息 + */ + @PreAuthorize("@ss.hasPermi('system:user:query')") + @GetMapping(value = {"/", "/{userId}"}) + public AjaxResult getInfo(@PathVariable(value = "userId", required = false) Long userId) { + userService.checkUserDataScope(userId); + AjaxResult ajax = success(); + List roles = roleService.selectRoleAll(); + ajax.put("roles", SysUser.isAdmin(userId) ? roles : roles.stream().filter(r -> !r.isAdmin()).collect(Collectors.toList())); + ajax.put("posts", postService.selectPostAll()); + if (StringUtils.isNotNull(userId)) { + SysUser sysUser = userService.selectUserById(userId); + ajax.put(AjaxResult.DATA_TAG, sysUser); + ajax.put("postIds", postService.selectPostListByUserId(userId)); + ajax.put("roleIds", sysUser.getRoles().stream().map(SysRole::getRoleId).collect(Collectors.toList())); + } + return ajax; + } + + /** + * 新增用户 + */ + @PreAuthorize("@ss.hasPermi('system:user:add')") + @Log(title = "用户管理", businessType = BusinessType.INSERT) + @PostMapping + public AjaxResult add(@Validated @RequestBody SysUser user) { + if (UserConstants.NOT_UNIQUE.equals(userService.checkUserNameUnique(user))) { + return error("新增用户'" + user.getUserName() + "'失败,登录账号已存在"); + } else if (StringUtils.isNotEmpty(user.getPhonenumber()) + && UserConstants.NOT_UNIQUE.equals(userService.checkPhoneUnique(user))) { + return error("新增用户'" + user.getUserName() + "'失败,手机号码已存在"); + } else if (StringUtils.isNotEmpty(user.getEmail()) + && UserConstants.NOT_UNIQUE.equals(userService.checkEmailUnique(user))) { + return error("新增用户'" + user.getUserName() + "'失败,邮箱账号已存在"); + } + if (user.getRoleIds() == null || user.getRoleIds().length ==0) { + return error("角色不能为空,请分配角色"); + } + user.setCreateBy(getUsername()); + user.setPassword(SecurityUtils.encryptPassword(user.getPassword())); + return toAjax(userService.insertUser(user)); + } + + /** + * 修改用户 + */ + @PreAuthorize("@ss.hasPermi('system:user:edit')") + @Log(title = "用户管理", businessType = BusinessType.UPDATE) + @PutMapping + public AjaxResult edit(@Validated @RequestBody SysUser user) { + userService.checkUserAllowed(user); + userService.checkUserDataScope(user.getUserId()); + if (UserConstants.NOT_UNIQUE.equals(userService.checkUserNameUnique(user))) { + return error("修改用户'" + user.getUserName() + "'失败,登录账号已存在"); + } else if (StringUtils.isNotEmpty(user.getPhonenumber()) + && UserConstants.NOT_UNIQUE.equals(userService.checkPhoneUnique(user))) { + return error("修改用户'" + user.getUserName() + "'失败,手机号码已存在"); + } else if (StringUtils.isNotEmpty(user.getEmail()) + && UserConstants.NOT_UNIQUE.equals(userService.checkEmailUnique(user))) { + return error("修改用户'" + user.getUserName() + "'失败,邮箱账号已存在"); + } + if (user.getRoleIds() == null || user.getRoleIds().length ==0) { + return error("角色不能为空,请分配角色"); + } + user.setUpdateBy(getUsername()); + return toAjax(userService.updateUser(user)); + } + + /** + * 删除用户 + */ + @PreAuthorize("@ss.hasPermi('system:user:remove')") + @Log(title = "用户管理", businessType = BusinessType.DELETE) + @DeleteMapping("/{userIds}") + public AjaxResult remove(@PathVariable Long[] userIds) { + if (ArrayUtils.contains(userIds, getUserId())) { + return error("当前用户不能删除"); + } + return toAjax(userService.deleteUserByIds(userIds)); + } + + /** + * 重置密码 + */ + @PreAuthorize("@ss.hasPermi('system:user:resetPwd')") + @Log(title = "用户管理", businessType = BusinessType.UPDATE) + @PutMapping("/resetPwd") + public AjaxResult resetPwd(@RequestBody SysUser user) { + userService.checkUserAllowed(user); + userService.checkUserDataScope(user.getUserId()); + user.setPassword(SecurityUtils.encryptPassword(user.getPassword())); + user.setUpdateBy(getUsername()); + return toAjax(userService.resetPwd(user)); + } + + /** + * 状态修改 + */ + @PreAuthorize("@ss.hasPermi('system:user:edit')") + @Log(title = "用户管理", businessType = BusinessType.UPDATE) + @PutMapping("/changeStatus") + public AjaxResult changeStatus(@RequestBody SysUser user) { + userService.checkUserAllowed(user); + userService.checkUserDataScope(user.getUserId()); + user.setUpdateBy(getUsername()); + return toAjax(userService.updateUserStatus(user)); + } + + /** + * 根据用户编号获取授权角色 + */ + @PreAuthorize("@ss.hasPermi('system:user:query')") + @GetMapping("/authRole/{userId}") + public AjaxResult authRole(@PathVariable("userId") Long userId) { + AjaxResult ajax = success(); + SysUser user = userService.selectUserById(userId); + List roles = roleService.selectRolesByUserId(userId); + ajax.put("user", user); + ajax.put("roles", SysUser.isAdmin(userId) ? roles : roles.stream().filter(r -> !r.isAdmin()).collect(Collectors.toList())); + return ajax; + } + + /** + * 用户授权角色 + */ + @PreAuthorize("@ss.hasPermi('system:user:edit')") + @Log(title = "用户管理", businessType = BusinessType.GRANT) + @PutMapping("/authRole") + public AjaxResult insertAuthRole(Long userId, Long[] roleIds) { + userService.checkUserDataScope(userId); + userService.insertUserAuth(userId, roleIds); + return success(); + } + + /** + * 获取部门树列表 + */ + @PreAuthorize("@ss.hasPermi('system:user:list')") + @GetMapping("/deptTree") + public AjaxResult deptTree(SysDept dept) { + return success(deptService.selectDeptTreeList(dept)); + } + + /** + * 获取用户精简信息列表 + * 只包含被开启的用户,主要用于前端的下拉选项 + */ + @GetMapping("/list-all-simple") + public AjaxResult getSimpleUsers() { + // 获用户列表,只要开启状态的 + List list = userService.getUserListByStatus(CommonStatusEnum.ENABLE.getStatus()); + // 排序后,返回给前端 + return success(UserConvert.INSTANCE.convertList04(list)); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/TenantController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/TenantController.java new file mode 100644 index 00000000..3ee24562 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/TenantController.java @@ -0,0 +1,96 @@ +package com.ruoyi.web.controller.system; + + +import static com.ruoyi.common.core.domain.R.*; + +import com.ruoyi.common.core.domain.R; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.utils.ExcelUtils; +import com.ruoyi.system.convert.tenant.TenantConvert; +import com.ruoyi.system.domain.TenantDO; +import com.ruoyi.system.domain.vo.tenant.*; +import com.ruoyi.system.service.TenantService; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiImplicitParam; +import io.swagger.annotations.ApiOperation; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.web.bind.annotation.*; + +import javax.annotation.Resource; +import javax.annotation.security.PermitAll; +import javax.servlet.http.HttpServletResponse; +import javax.validation.Valid; +import java.io.IOException; +import java.util.List; + + +@Api(tags = "管理后台 - 租户") +@RestController +@RequestMapping("/system/tenant") +public class TenantController { + + @Resource + private TenantService tenantService; + + @GetMapping("/get-id-by-name") + @PermitAll + @ApiOperation(value = "使用租户名,获得租户编号", notes = "登录界面,根据用户的租户名,获得租户编号") + @ApiImplicitParam(name = "name", value = "租户名", required = true, example = "1024", dataTypeClass = Long.class) + public R getTenantIdByName(@RequestParam("name") String name) { + TenantDO tenantDO = tenantService.getTenantByName(name); + return ok(tenantDO != null ? tenantDO.getId() : null); + } + + @PostMapping("/create") + @ApiOperation("创建租户") + @PreAuthorize("@ss.hasPermi('system:tenant:create')") + public R createTenant(@Valid @RequestBody TenantCreateReqVO createReqVO) { + return ok(tenantService.createTenant(createReqVO)); + } + + @PutMapping("/update") + @ApiOperation("更新租户") + @PreAuthorize("@ss.hasPermi('system:tenant:update')") + public R updateTenant(@Valid @RequestBody TenantUpdateReqVO updateReqVO) { + tenantService.updateTenant(updateReqVO); + return ok(true); + } + + @DeleteMapping("/delete") + @ApiOperation("删除租户") + @ApiImplicitParam(name = "id", value = "编号", required = true, example = "1024", dataTypeClass = Long.class) + @PreAuthorize("@ss.hasPermi('system:tenant:delete')") + public R deleteTenant(@RequestParam("id") Long id) { + tenantService.deleteTenant(id); + return ok(true); + } + + @GetMapping("/get") + @ApiOperation("获得租户") + @ApiImplicitParam(name = "id", value = "编号", required = true, example = "1024", dataTypeClass = Long.class) + @PreAuthorize("@ss.hasPermi('system:tenant:query')") + public R getTenant(@RequestParam("id") Long id) { + TenantDO tenant = tenantService.getTenant(id); + return ok(TenantConvert.INSTANCE.convert(tenant)); + } + + @GetMapping("/page") + @ApiOperation("获得租户分页") + @PreAuthorize("@ss.hasPermi('system:tenant:query')") + public R> getTenantPage(@Valid TenantPageReqVO pageVO) { + PageResult pageResult = tenantService.getTenantPage(pageVO); + return ok(TenantConvert.INSTANCE.convertPage(pageResult)); + } + + @GetMapping("/export-excel") + @ApiOperation("导出租户 Excel") + @PreAuthorize("@ss.hasPermi('system:tenant:export')") + public void exportTenantExcel(@Valid TenantExportReqVO exportReqVO, HttpServletResponse response) throws IOException { + List list = tenantService.getTenantList(exportReqVO); + // 导出 Excel + List datas = TenantConvert.INSTANCE.convertList02(list); + ExcelUtils.write(response, "租户.xls", "数据", TenantExcelVO.class, datas); + } + + +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/TenantPackageController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/TenantPackageController.java new file mode 100644 index 00000000..06f616a6 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/system/TenantPackageController.java @@ -0,0 +1,83 @@ +package com.ruoyi.web.controller.system; + + +import static com.ruoyi.common.core.domain.R.*; + +import com.ruoyi.common.core.domain.R; +import com.ruoyi.common.enums.CommonStatusEnum; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.system.convert.tenant.TenantPackageConvert; +import com.ruoyi.system.domain.TenantPackageDO; +import com.ruoyi.system.domain.vo.packages.*; +import com.ruoyi.system.service.TenantPackageService; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiImplicitParam; +import io.swagger.annotations.ApiOperation; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import javax.annotation.Resource; +import javax.validation.Valid; +import java.util.List; + + +@Api(tags = "管理后台 - 租户套餐") +@RestController +@RequestMapping("/system/tenant-package") +@Validated +public class TenantPackageController { + + @Resource + private TenantPackageService tenantPackageService; + + @PostMapping("/create") + @ApiOperation("创建租户套餐") + @PreAuthorize("@ss.hasPermi('system:tenant-package:create')") + public R createTenantPackage(@Valid @RequestBody TenantPackageCreateReqVO createReqVO) { + return ok(tenantPackageService.createTenantPackage(createReqVO)); + } + + @PutMapping("/update") + @ApiOperation("更新租户套餐") + @PreAuthorize("@ss.hasPermi('system:tenant-package:update')") + public R updateTenantPackage(@Valid @RequestBody TenantPackageUpdateReqVO updateReqVO) { + tenantPackageService.updateTenantPackage(updateReqVO); + return ok(true); + } + + @DeleteMapping("/delete") + @ApiOperation("删除租户套餐") + @ApiImplicitParam(name = "id", value = "编号", required = true, dataTypeClass = Long.class) + @PreAuthorize("@ss.hasPermi('system:tenant-package:delete')") + public R deleteTenantPackage(@RequestParam("id") Long id) { + tenantPackageService.deleteTenantPackage(id); + return ok(true); + } + + @GetMapping("/get") + @ApiOperation("获得租户套餐") + @ApiImplicitParam(name = "id", value = "编号", required = true, example = "1024", dataTypeClass = Long.class) + @PreAuthorize("@ss.hasPermi('system:tenant-package:query')") + public R getTenantPackage(@RequestParam("id") Long id) { + TenantPackageDO tenantPackage = tenantPackageService.getTenantPackage(id); + return ok(TenantPackageConvert.INSTANCE.convert(tenantPackage)); + } + + @GetMapping("/page") + @ApiOperation("获得租户套餐分页") + @PreAuthorize("@ss.hasPermi('system:tenant-package:query')") + public R> getTenantPackagePage(@Valid TenantPackagePageReqVO pageVO) { + PageResult pageResult = tenantPackageService.getTenantPackagePage(pageVO); + return ok(TenantPackageConvert.INSTANCE.convertPage(pageResult)); + } + + @GetMapping("/get-simple-list") + @ApiOperation(value = "获取租户套餐精简信息列表", notes = "只包含被开启的租户套餐,主要用于前端的下拉选项") + public R> getTenantPackageList() { + // 获得角色列表,只要开启状态的 + List list = tenantPackageService.getTenantPackageListByStatus(CommonStatusEnum.ENABLE.getStatus()); + return ok(TenantPackageConvert.INSTANCE.convertList02(list)); + } + +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/SwaggerController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/SwaggerController.java new file mode 100644 index 00000000..f66ca24e --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/SwaggerController.java @@ -0,0 +1,24 @@ +package com.ruoyi.web.controller.tool; + +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.stereotype.Controller; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.RequestMapping; +import com.ruoyi.common.core.controller.BaseController; + +/** + * swagger 接口 + * + * @author ruoyi + */ +@Controller +@RequestMapping("/tool/swagger") +public class SwaggerController extends BaseController +{ + @PreAuthorize("@ss.hasPermi('tool:swagger:view')") + @GetMapping() + public String index() + { + return redirect("/swagger-ui.html"); + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/TestController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/TestController.java new file mode 100644 index 00000000..b4f6bac7 --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/TestController.java @@ -0,0 +1,183 @@ +package com.ruoyi.web.controller.tool; + +import java.util.ArrayList; +import java.util.LinkedHashMap; +import java.util.List; +import java.util.Map; +import org.springframework.web.bind.annotation.DeleteMapping; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.PutMapping; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.R; +import com.ruoyi.common.utils.StringUtils; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiImplicitParam; +import io.swagger.annotations.ApiImplicitParams; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import io.swagger.annotations.ApiOperation; + +/** + * swagger 用户测试方法 + * + * @author ruoyi + */ +@Api("用户信息管理") +@RestController +@RequestMapping("/test/user") +public class TestController extends BaseController +{ + private final static Map users = new LinkedHashMap(); + { + users.put(1, new UserEntity(1, "admin", "admin123", "15888888888")); + users.put(2, new UserEntity(2, "ry", "admin123", "15666666666")); + } + + @ApiOperation("获取用户列表") + @GetMapping("/list") + public R> userList() + { + List userList = new ArrayList(users.values()); + return R.ok(userList); + } + + @ApiOperation("获取用户详细") + @ApiImplicitParam(name = "userId", value = "用户ID", required = true, dataType = "int", paramType = "path", dataTypeClass = Integer.class) + @GetMapping("/{userId}") + public R getUser(@PathVariable Integer userId) + { + if (!users.isEmpty() && users.containsKey(userId)) + { + return R.ok(users.get(userId)); + } + else + { + return R.fail("用户不存在"); + } + } + + @ApiOperation("新增用户") + @ApiImplicitParams({ + @ApiImplicitParam(name = "userId", value = "用户id", dataType = "Integer", dataTypeClass = Integer.class), + @ApiImplicitParam(name = "username", value = "用户名称", dataType = "String", dataTypeClass = String.class), + @ApiImplicitParam(name = "password", value = "用户密码", dataType = "String", dataTypeClass = String.class), + @ApiImplicitParam(name = "mobile", value = "用户手机", dataType = "String", dataTypeClass = String.class) + }) + @PostMapping("/save") + public R save(UserEntity user) + { + if (StringUtils.isNull(user) || StringUtils.isNull(user.getUserId())) + { + return R.fail("用户ID不能为空"); + } + users.put(user.getUserId(), user); + return R.ok(); + } + + @ApiOperation("更新用户") + @PutMapping("/update") + public R update(@RequestBody UserEntity user) + { + if (StringUtils.isNull(user) || StringUtils.isNull(user.getUserId())) + { + return R.fail("用户ID不能为空"); + } + if (users.isEmpty() || !users.containsKey(user.getUserId())) + { + return R.fail("用户不存在"); + } + users.remove(user.getUserId()); + users.put(user.getUserId(), user); + return R.ok(); + } + + @ApiOperation("删除用户信息") + @ApiImplicitParam(name = "userId", value = "用户ID", required = true, dataType = "int", paramType = "path", dataTypeClass = Integer.class) + @DeleteMapping("/{userId}") + public R delete(@PathVariable Integer userId) + { + if (!users.isEmpty() && users.containsKey(userId)) + { + users.remove(userId); + return R.ok(); + } + else + { + return R.fail("用户不存在"); + } + } +} + +@ApiModel(value = "UserEntity", description = "用户实体") +class UserEntity +{ + @ApiModelProperty("用户ID") + private Integer userId; + + @ApiModelProperty("用户名称") + private String username; + + @ApiModelProperty("用户密码") + private String password; + + @ApiModelProperty("用户手机") + private String mobile; + + public UserEntity() + { + + } + + public UserEntity(Integer userId, String username, String password, String mobile) + { + this.userId = userId; + this.username = username; + this.password = password; + this.mobile = mobile; + } + + public Integer getUserId() + { + return userId; + } + + public void setUserId(Integer userId) + { + this.userId = userId; + } + + public String getUsername() + { + return username; + } + + public void setUsername(String username) + { + this.username = username; + } + + public String getPassword() + { + return password; + } + + public void setPassword(String password) + { + this.password = password; + } + + public String getMobile() + { + return mobile; + } + + public void setMobile(String mobile) + { + this.mobile = mobile; + } +} diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/core/config/SwaggerConfig.java b/ruoyi-admin/src/main/java/com/ruoyi/web/core/config/SwaggerConfig.java new file mode 100644 index 00000000..3901ec6d --- /dev/null +++ b/ruoyi-admin/src/main/java/com/ruoyi/web/core/config/SwaggerConfig.java @@ -0,0 +1,125 @@ +package com.ruoyi.web.core.config; + +import java.util.ArrayList; +import java.util.List; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import com.ruoyi.common.config.RuoYiConfig; +import io.swagger.annotations.ApiOperation; +import io.swagger.models.auth.In; +import springfox.documentation.builders.ApiInfoBuilder; +import springfox.documentation.builders.PathSelectors; +import springfox.documentation.builders.RequestHandlerSelectors; +import springfox.documentation.service.ApiInfo; +import springfox.documentation.service.ApiKey; +import springfox.documentation.service.AuthorizationScope; +import springfox.documentation.service.Contact; +import springfox.documentation.service.SecurityReference; +import springfox.documentation.service.SecurityScheme; +import springfox.documentation.spi.DocumentationType; +import springfox.documentation.spi.service.contexts.SecurityContext; +import springfox.documentation.spring.web.plugins.Docket; + +/** + * Swagger2的接口配置 + * + * @author ruoyi + */ +@Configuration +public class SwaggerConfig +{ + /** 系统基础配置 */ + @Autowired + private RuoYiConfig ruoyiConfig; + + /** 是否开启swagger */ + @Value("${swagger.enabled}") + private boolean enabled; + + /** 设置请求的统一前缀 */ + @Value("${swagger.pathMapping}") + private String pathMapping; + + /** + * 创建API + */ + @Bean + public Docket createRestApi() + { + return new Docket(DocumentationType.OAS_30) + // 是否启用Swagger + .enable(enabled) + // 用来创建该API的基本信息,展示在文档的页面中(自定义展示的信息) + .apiInfo(apiInfo()) + // 设置哪些接口暴露给Swagger展示 + .select() + // 扫描所有有注解的api,用这种方式更灵活 + .apis(RequestHandlerSelectors.withMethodAnnotation(ApiOperation.class)) + // 扫描指定包中的swagger注解 + // .apis(RequestHandlerSelectors.basePackage("com.ruoyi.project.tool.swagger")) + // 扫描所有 .apis(RequestHandlerSelectors.any()) + .paths(PathSelectors.any()) + .build() + /* 设置安全模式,swagger可以设置访问token */ + .securitySchemes(securitySchemes()) + .securityContexts(securityContexts()) + .pathMapping(pathMapping); + } + + /** + * 安全模式,这里指定token通过Authorization头请求头传递 + */ + private List securitySchemes() + { + List apiKeyList = new ArrayList(); + apiKeyList.add(new ApiKey("Authorization", "Authorization", In.HEADER.toValue())); + return apiKeyList; + } + + /** + * 安全上下文 + */ + private List securityContexts() + { + List securityContexts = new ArrayList<>(); + securityContexts.add( + SecurityContext.builder() + .securityReferences(defaultAuth()) + .operationSelector(o -> o.requestMappingPattern().matches("/.*")) + .build()); + return securityContexts; + } + + /** + * 默认的安全上引用 + */ + private List defaultAuth() + { + AuthorizationScope authorizationScope = new AuthorizationScope("global", "accessEverything"); + AuthorizationScope[] authorizationScopes = new AuthorizationScope[1]; + authorizationScopes[0] = authorizationScope; + List securityReferences = new ArrayList<>(); + securityReferences.add(new SecurityReference("Authorization", authorizationScopes)); + return securityReferences; + } + + /** + * 添加摘要信息 + */ + private ApiInfo apiInfo() + { + // 用ApiInfoBuilder进行定制 + return new ApiInfoBuilder() + // 设置标题 + .title("标题:数字消防综合管理系统_接口文档") + // 描述 + .description("描述:用于管理集团旗下公司的人员信息,具体包括XXX,XXX模块...") + // 作者信息 + .contact(new Contact(ruoyiConfig.getName(), null, null)) + // 版本 + .version("版本号:" + ruoyiConfig.getVersion()) + .build(); + } +} diff --git a/ruoyi-admin/src/main/resources/META-INF/spring-devtools.properties b/ruoyi-admin/src/main/resources/META-INF/spring-devtools.properties new file mode 100644 index 00000000..2b23f85a --- /dev/null +++ b/ruoyi-admin/src/main/resources/META-INF/spring-devtools.properties @@ -0,0 +1 @@ +restart.include.json=/com.alibaba.fastjson.*.jar \ No newline at end of file diff --git a/ruoyi-admin/src/main/resources/application-dev.yml b/ruoyi-admin/src/main/resources/application-dev.yml new file mode 100644 index 00000000..0683a1ef --- /dev/null +++ b/ruoyi-admin/src/main/resources/application-dev.yml @@ -0,0 +1,63 @@ +ruoyi: + # 文件路径 + profile: D:/ruoyi/uploadPath +# 数据源配置 +spring: + datasource: + type: com.alibaba.druid.pool.DruidDataSource + driverClassName: com.mysql.cj.jdbc.Driver + druid: + # 主库数据源 + master: +# url: jdbc:mysql://121.40.191.30:3306/ry-base-tenant?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8 + url: jdbc:mysql://43.139.7.36:3306/ry-base-tenant?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8 + username: root + password: it@luokeng +# password: 123456 + + # 从库数据源 + slave: + # 从数据源开关/默认关闭 + enabled: false + url: + username: + password: + # 初始连接数 + initialSize: 5 + # 最小连接池数量 + minIdle: 10 + # 最大连接池数量 + maxActive: 20 + # 配置获取连接等待超时的时间 + maxWait: 60000 + # 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 + timeBetweenEvictionRunsMillis: 60000 + # 配置一个连接在池中最小生存的时间,单位是毫秒 + minEvictableIdleTimeMillis: 300000 + # 配置一个连接在池中最大生存的时间,单位是毫秒 + maxEvictableIdleTimeMillis: 900000 + # 配置检测连接是否有效 + validationQuery: SELECT 1 FROM DUAL + testWhileIdle: true + testOnBorrow: false + testOnReturn: false + webStatFilter: + enabled: true + statViewServlet: + enabled: true + # 设置白名单,不填则允许所有访问 + allow: + url-pattern: /druid/* + # 控制台管理用户名和密码 + login-username: ruoyi + login-password: 123456 + filter: + stat: + enabled: true + # 慢SQL记录 + log-slow-sql: true + slow-sql-millis: 1000 + merge-sql: true + wall: + config: + multi-statement-allow: true diff --git a/ruoyi-admin/src/main/resources/application-druid.yml b/ruoyi-admin/src/main/resources/application-druid.yml new file mode 100644 index 00000000..b4aa52c5 --- /dev/null +++ b/ruoyi-admin/src/main/resources/application-druid.yml @@ -0,0 +1,57 @@ +# 数据源配置 +spring: + datasource: + type: com.alibaba.druid.pool.DruidDataSource + driverClassName: com.mysql.cj.jdbc.Driver + druid: + # 主库数据源 + master: + url: jdbc:mysql://localhost:3306/ry-vue?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8 + username: root + password: root + # 从库数据源 + slave: + # 从数据源开关/默认关闭 + enabled: false + url: + username: + password: + # 初始连接数 + initialSize: 5 + # 最小连接池数量 + minIdle: 10 + # 最大连接池数量 + maxActive: 20 + # 配置获取连接等待超时的时间 + maxWait: 60000 + # 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 + timeBetweenEvictionRunsMillis: 60000 + # 配置一个连接在池中最小生存的时间,单位是毫秒 + minEvictableIdleTimeMillis: 300000 + # 配置一个连接在池中最大生存的时间,单位是毫秒 + maxEvictableIdleTimeMillis: 900000 + # 配置检测连接是否有效 + validationQuery: SELECT 1 FROM DUAL + testWhileIdle: true + testOnBorrow: false + testOnReturn: false + webStatFilter: + enabled: true + statViewServlet: + enabled: true + # 设置白名单,不填则允许所有访问 + allow: + url-pattern: /druid/* + # 控制台管理用户名和密码 + login-username: ruoyi + login-password: 123456 + filter: + stat: + enabled: true + # 慢SQL记录 + log-slow-sql: true + slow-sql-millis: 1000 + merge-sql: true + wall: + config: + multi-statement-allow: true diff --git a/ruoyi-admin/src/main/resources/application-test.yml b/ruoyi-admin/src/main/resources/application-test.yml new file mode 100644 index 00000000..8217380e --- /dev/null +++ b/ruoyi-admin/src/main/resources/application-test.yml @@ -0,0 +1,61 @@ +ruoyi: + # 文件路径 + profile: /project/xinchuang/xiaofang-dm/uploadPath + +# 数据源配置 +spring: + datasource: + type: com.alibaba.druid.pool.DruidDataSource + driverClassName: com.mysql.cj.jdbc.Driver + druid: + # 主库数据源 + master: + url: jdbc:mysql://192.168.254.125:3306/xiaofang?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8 + username: root + password: abcd1234 + # 从库数据源 + slave: + # 从数据源开关/默认关闭 + enabled: false + url: + username: + password: + # 初始连接数 + initialSize: 5 + # 最小连接池数量 + minIdle: 10 + # 最大连接池数量 + maxActive: 20 + # 配置获取连接等待超时的时间 + maxWait: 60000 + # 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 + timeBetweenEvictionRunsMillis: 60000 + # 配置一个连接在池中最小生存的时间,单位是毫秒 + minEvictableIdleTimeMillis: 300000 + # 配置一个连接在池中最大生存的时间,单位是毫秒 + maxEvictableIdleTimeMillis: 900000 + # 配置检测连接是否有效 + validationQuery: SELECT 1 FROM DUAL + testWhileIdle: true + testOnBorrow: false + testOnReturn: false + webStatFilter: + enabled: true + statViewServlet: + enabled: true + # 设置白名单,不填则允许所有访问 + allow: + url-pattern: /druid/* + # 控制台管理用户名和密码 + login-username: ruoyi + login-password: 123456 + filter: + stat: + enabled: true + # 慢SQL记录 + log-slow-sql: true + slow-sql-millis: 1000 + merge-sql: true + wall: + config: + multi-statement-allow: true diff --git a/ruoyi-admin/src/main/resources/application.yml b/ruoyi-admin/src/main/resources/application.yml new file mode 100644 index 00000000..c05620e3 --- /dev/null +++ b/ruoyi-admin/src/main/resources/application.yml @@ -0,0 +1,154 @@ +# 项目相关配置 +ruoyi: + # 名称 + name: RuoYi + # 版本 + version: 1.0.0 + # 版权年份 + copyrightYear: 2022 + # 实例演示开关 + demoEnabled: true + # 获取ip地址开关 + addressEnabled: false + # 验证码类型 math 数组计算 char 字符验证 + captchaType: math + tenant: + enable: true + ignore-urls: + - /system/tenant/get-id-by-name # 基于名字获取租户,不许带租户编号 + - /captcha/get # 获取图片验证码,和租户无关 + - /captcha/check # 校验图片验证码,和租户无关 + - /infra/file/*/get/** # 获取图片,和租户无关 + - /system/sms/callback/* # 短信回调接口,无法带上租户编号 + - /app-api/pay/order/notify/* # 支付回调通知,不携带租户编号 + - /jmreport/** # 积木报表,无法携带租户编号。*匹配任意字符(除了路径分隔符),而**匹配任意字符 +# 开发环境配置 +server: + # 服务器的HTTP端口,默认为8080 + port: 8080 + servlet: + # 应用的访问路径 + context-path: / + tomcat: + # tomcat的URI编码 + uri-encoding: UTF-8 + # 连接数满后的排队数,默认为100 + accept-count: 1000 + threads: + # tomcat最大线程数,默认为200 + max: 800 + # Tomcat启动初始化的线程数,默认值10 + min-spare: 100 + +# 日志配置 +logging: + level: + com.ruoyi: debug + org.springframework: warn + +# 用户配置 +user: + password: + # 密码最大错误次数 + maxRetryCount: 5 + # 密码锁定时间(默认10分钟) + lockTime: 10 + +# Spring配置 +spring: + # 资源信息 + messages: + # 国际化资源文件路径 + basename: i18n/messages + profiles: + active: dev + # 文件上传 + servlet: + multipart: + # 单个文件大小 + max-file-size: 10MB + # 设置总上传的文件大小 + max-request-size: 20MB + # 服务模块 + devtools: + restart: + # 热部署开关 + enabled: false + cache: + type: ehcache + ehcache: + config: classpath:/ehcache.xml + # redis 配置 +# redis: +# # 地址 +# host: localhost +# # 端口,默认为6379 +# port: 6379 +# # 数据库索引 +# database: 0 +# # 密码 +# password: +# # 连接超时时间 +# timeout: 10s +# lettuce: +# pool: +# # 连接池中的最小空闲连接 +# min-idle: 0 +# # 连接池中的最大空闲连接 +# max-idle: 8 +# # 连接池的最大数据库连接数 +# max-active: 8 +# # #连接池最大阻塞等待时间(使用负值表示没有限制) +# max-wait: -1ms + +# token配置 +token: + # 令牌自定义标识 + header: Authorization + # 令牌密钥 + secret: abcytrmiuumwefsadfdfafahvwxyz + # 令牌有效期(默认30分钟) + expireTime: 30 + +# MyBatis Plus配置 +mybatis-plus: + # 搜索指定包别名 + typeAliasesPackage: com.ruoyi.**.domain + # 配置mapper的扫描,找到所有的mapper.xml映射文件 + mapperLocations: classpath*:mapper/**/*Mapper.xml + # 加载全局的配置文件 + configLocation: classpath:mybatis/mybatis-config.xml + +# PageHelper分页插件 +pagehelper: + helperDialect: mysql + supportMethodsArguments: true + params: count=countSql + +# Swagger配置 +swagger: + # 是否开启swagger + enabled: true + # 请求前缀 + pathMapping: /dev-api + +# 防止XSS攻击 +xss: + # 过滤开关 + enabled: true + # 排除链接(多个用逗号分隔) + excludes: /system/notice + # 匹配链接 + urlPatterns: /system/*,/monitor/*,/tool/* + +# 工作流 Flowable 配置 +flowable: + # 1. false: 默认值,Flowable 启动时,对比数据库表中保存的版本,如果不匹配。将抛出异常 + # 2. true: 启动时会对数据库中所有表进行更新操作,如果表存在,不做处理,反之,自动创建表 + # 3. create_drop: 启动时自动创建表,关闭时自动删除表 + # 4. drop_create: 启动时,删除旧表,再创建新表 + database-schema-update: true # 设置为 false,可通过 https://github.com/flowable/flowable-sql 初始化 + db-history-used: true # flowable6 默认 true 生成信息表,无需手动设置 + check-process-definitions: false # 设置为 false,禁用 /resources/processes 自动部署 BPMN XML 流程 + history-level: full # full:保存历史数据的最高级别,可保存全部流程相关细节,包括流程流转各节点参数 + diff --git a/ruoyi-admin/src/main/resources/banner.txt b/ruoyi-admin/src/main/resources/banner.txt new file mode 100644 index 00000000..0931cb84 --- /dev/null +++ b/ruoyi-admin/src/main/resources/banner.txt @@ -0,0 +1,24 @@ +Application Version: ${ruoyi.version} +Spring Boot Version: ${spring-boot.version} +//////////////////////////////////////////////////////////////////// +// _ooOoo_ // +// o8888888o // +// 88" . "88 // +// (| ^_^ |) // +// O\ = /O // +// ____/`---'\____ // +// .' \\| |// `. // +// / \\||| : |||// \ // +// / _||||| -:- |||||- \ // +// | | \\\ - /// | | // +// | \_| ''\---/'' | | // +// \ .-\__ `-` ___/-. / // +// ___`. .' /--.--\ `. . ___ // +// ."" '< `.___\_<|>_/___.' >'"". // +// | | : `- \`.;`\ _ /`;.`/ - ` : | | // +// \ \ `-. \_ __\ /__ _/ .-` / / // +// ========`-.____`-.___\_____/___.-`____.-'======== // +// `=---=' // +// ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ // +// 佛祖保佑 永不宕机 永无BUG // +//////////////////////////////////////////////////////////////////// \ No newline at end of file diff --git a/ruoyi-admin/src/main/resources/ehcache.xml b/ruoyi-admin/src/main/resources/ehcache.xml new file mode 100644 index 00000000..c2bad019 --- /dev/null +++ b/ruoyi-admin/src/main/resources/ehcache.xml @@ -0,0 +1,108 @@ + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/ruoyi-admin/src/main/resources/i18n/messages.properties b/ruoyi-admin/src/main/resources/i18n/messages.properties new file mode 100644 index 00000000..b0bc5ed0 --- /dev/null +++ b/ruoyi-admin/src/main/resources/i18n/messages.properties @@ -0,0 +1,37 @@ +#错误消息 +not.null=* 必须填写 +user.jcaptcha.error=验证码错误 +user.jcaptcha.expire=验证码已失效 +user.not.exists=用户不存在/密码错误 +user.password.not.match=用户不存在/密码错误 +user.password.retry.limit.count=密码输入错误{0}次 +user.password.retry.limit.exceed=密码输入错误{0}次,帐户锁定{1}分钟 +user.password.delete=对不起,您的账号已被删除 +user.blocked=用户已封禁,请联系管理员 +role.blocked=角色已封禁,请联系管理员 +user.logout.success=退出成功 + +length.not.valid=长度必须在{min}到{max}个字符之间 + +user.username.not.valid=* 2到20个汉字、字母、数字或下划线组成,且必须以非数字开头 +user.password.not.valid=* 5-50个字符 + +user.email.not.valid=邮箱格式错误 +user.mobile.phone.number.not.valid=手机号格式错误 +user.login.success=登录成功 +user.register.success=注册成功 +user.notfound=请重新登录 +user.forcelogout=管理员强制退出,请重新登录 +user.unknown.error=未知错误,请重新登录 + +##文件上传消息 +upload.exceed.maxSize=上传的文件大小超出限制的文件大小!
允许的文件最大大小是:{0}MB! +upload.filename.exceed.length=上传的文件名最长{0}个字符 + +##权限 +no.permission=您没有数据的权限,请联系管理员添加权限 [{0}] +no.create.permission=您没有创建数据的权限,请联系管理员添加权限 [{0}] +no.update.permission=您没有修改数据的权限,请联系管理员添加权限 [{0}] +no.delete.permission=您没有删除数据的权限,请联系管理员添加权限 [{0}] +no.export.permission=您没有导出数据的权限,请联系管理员添加权限 [{0}] +no.view.permission=您没有查看数据的权限,请联系管理员添加权限 [{0}] diff --git a/ruoyi-admin/src/main/resources/logback.xml b/ruoyi-admin/src/main/resources/logback.xml new file mode 100644 index 00000000..9a26a231 --- /dev/null +++ b/ruoyi-admin/src/main/resources/logback.xml @@ -0,0 +1,93 @@ + + + + + + + + + + + ${log.pattern} + + + + + + ${log.path}/sys-info.log + + + + ${log.path}/sys-info.%d{yyyy-MM-dd}.log + + 60 + + + ${log.pattern} + + + + INFO + + ACCEPT + + DENY + + + + + ${log.path}/sys-error.log + + + + ${log.path}/sys-error.%d{yyyy-MM-dd}.log + + 60 + + + ${log.pattern} + + + + ERROR + + ACCEPT + + DENY + + + + + + ${log.path}/sys-user.log + + + ${log.path}/sys-user.%d{yyyy-MM-dd}.log + + 60 + + + ${log.pattern} + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/ruoyi-admin/src/main/resources/mybatis/mybatis-config.xml b/ruoyi-admin/src/main/resources/mybatis/mybatis-config.xml new file mode 100644 index 00000000..ac47c038 --- /dev/null +++ b/ruoyi-admin/src/main/resources/mybatis/mybatis-config.xml @@ -0,0 +1,20 @@ + + + + + + + + + + + + + + + + + + diff --git a/ruoyi-admin/src/main/resources/templatefile/消防安全重点部位情况登记表.xlsx b/ruoyi-admin/src/main/resources/templatefile/消防安全重点部位情况登记表.xlsx new file mode 100644 index 0000000000000000000000000000000000000000..8e58d705ea3abea9fa6799b5fdb73617764b7d7b GIT binary patch literal 9408 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/___.' >'"". // +// | | : `- \`.;`\ _ /`;.`/ - ` : | | // +// \ \ `-. \_ __\ /__ _/ .-` / / // +// ========`-.____`-.___\_____/___.-`____.-'======== // +// `=---=' // +// ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ // +// 佛祖保佑 永不宕机 永无BUG // +//////////////////////////////////////////////////////////////////// \ No newline at end of file diff --git a/ruoyi-admin/target/classes/com/ruoyi/RuoYiApplication.class b/ruoyi-admin/target/classes/com/ruoyi/RuoYiApplication.class new file mode 100644 index 0000000000000000000000000000000000000000..622bd74ce88defb4c0b81e0c4a20c655ff533e7f GIT binary patch literal 1458 zcma)6OK($06#m8uF>zgD5|Z+4hNrJXa;FP+K}bnxrAii!a1~Sq$r;}aW{mHh=sqY& zSroCMtk_ks;IXJwsf(%)wN)iji6uXQ9g#wp{DX=!_Zr(lL1OGXk8{5H&Np+!Suj+{5aMd%u2orNti;Ic*M$fISwY4W>ZGu_i^ra3M zCY0y2RB2ZW|GiTUuY|0{9Sg}5;7dPm;|x*)sl`h zshbM|6Q|LM_FUD3O%bixa92XLhSyH) zt$g{RRISy9C(g@QX5!~4&O#^7+e;}$o($$?q!y)jO*R!NJxD^`mQr5R_04M1R*~EC zL$zkYm|>W%*iEmP;p~n&BJZlAU35$#yB8on5A6G@b$zo%?8Yz!d;crTqg;f8N*+^1 zoW?X`x17!UKW_f;c=Nk|sjs}=`2O|AL(X6Q{`9|FKX2XpW$W%wukPI2y8DQ8{_^$% z=h+`Go~+P>kOV7mcP48t`E9FH>rB#tx&mJ(JN-DnoZ0MPACdkugu~0c!YwpylJIwTHWE5I(OefIz;v9dWCFO2Z`xy>R~cH>zAzpineYe5;bk=VN%KK zgjlgQVIeCsH0?$vLkE+ci5x^{+jdB~Y_}0jIuMELX0sHVc6rPc;X*K!dNuS_qoq8` zjj8u$c$U+B`Al^lCILgk>8r(swv3|L9m=n=W1>;X#Og*Yx+!f-u2=D7wr(o?KB={v zQLKE5ZW0nao(0!NEK*s<^es@*rwwE2*|wV*^0qMJHmM%(SKa-B;3#tRk+UraX49Mc z0<8p%jK*D4e}Vm)x<8_RC^e3tNb_O_z{l7_t6_|wL>9YwJ|zQ6qq(WSF>vWO436&l z6GJ_oJs2R|J_fFDYHk`_9;m0dI!=Q+>YFFgMzd3wHo1n?2>t0-g5#=4DW6|b>jp*8m@7s^i!(SEiZZG0X_Muq-`5kq;U%@6Ksop3i*!MQYi zkSC9#C{UhAln%S2M5jGRFGEdChzy;%Sk-|}jVF&)rqI@m(f7WKOHqf71=2OU^FV|? z?*$?z?KY>R`(s`kZ|m5?JBID4qQCWJ=+6}NO1eBAZG^{Z6ls%;#qE;b|6=XWWc_bB z40fDrLZf|ZeRZbhoojd4Mv1=S>?wd=gatY)& zMV^I4EKwwD?h>>>c5&BwK;f*oQXJgE`bCifCX_}_X^tw#6zV|(%fzAIBd=e50YEbT AGXMYp literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/TestFlowableApplication.class b/ruoyi-admin/target/classes/com/ruoyi/TestFlowableApplication.class new file mode 100644 index 0000000000000000000000000000000000000000..b756b93ea4ccd86e7802cb8f0bff5f607dc809f2 GIT binary patch literal 1348 zcma)6YflqF6g^W4eXs>A3aFqKq!iHI(gI=&7*ZtB6vgm}#!u7jP`26b*4Zrtzxk8= zQW8!40sbiCotB3rteNQPZd)ezE!W8tbd6gM z!$|6LvC5CRVRNTq?E2DjD)|;QmlXmV?ocLm=3#{)U36u|I6Q|pgi|4z#*TDN;d!rH zQNeW%tx7{)u*T~DR+ZKpAxLX<*KJ5sY+0)H@h&~-stpZs3^QzWLbYmj+aTf8C3ni) zb{)}OE5pZXxn$;Qr`{JkXBf+AJ(FJ4GnuTuw35qa)618Qzw(4UCTFag_)3^-`+J3? z)3EIV_lcykEx6+q8d3_!U-M-{Xf!QLWPo&CKZIosFR;Q8*;Y&onQTJCOO+TS7xfms zZdP?AM&Fbc5oyS&y56bB|1}hBljpIbVHIl(i5usHVd!c&-j}Mv^vMlR_-{$hzO)%W zq&kdGM^i5!e|L><$RV;j->I`(-cB#eg*>vw;We6=&0A~iE{$Ye0Pk?mJ8@;1A$CsH z_Jv`T;!SeB#67WV9*G*?ua~*6-q%vYvdh=t8KzUS9iP;-V|gSnM4uM*3xJR?OzT~W z4B8p(y>n+^-&^H-wDz@=!plL=#BR^ z5sHVKh{U5!^v7dO3>4>^7~H;spG3F?6+sjO=qEKAfyXdGQ38WVl3GRmqz#XU7^f&f VvL^6|$dmMcOjXrOf+_(_{srDTbNK)O literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/DevinUtil.class b/ruoyi-admin/target/classes/com/ruoyi/web/DevinUtil.class new file mode 100644 index 0000000000000000000000000000000000000000..4a06690e853a8bff500eb5a3a5c2170d5b7a1bb1 GIT binary patch literal 5546 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 literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/common/CaptchaController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/common/CaptchaController.class new file mode 100644 index 0000000000000000000000000000000000000000..e250ac12706215e2d991dbc5c06386f1f98c5687 GIT binary patch literal 3706 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 literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/monitor/ServerController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/monitor/ServerController.class new file mode 100644 index 0000000000000000000000000000000000000000..54c33012f80186978b62abde915776eb34faf8ef GIT binary patch literal 1152 zcmb7DTTc@~6#k}^uGGp^K)fL$wxtOZBM%1DNQfGfq6A12g6fm2Lxiaoz-orf)3k;>l zDpKw#!$7UxVkoZbUCA)gRFQn0gl%bdL^~kMcvJf#XbGcIy=N@CePtOQHhmp(ljsx0 zk7S$sI&ww_fiyhS#I6nBmgZ2J_5Q^wgV&MnRkWuW8nyZv2I3+Vkr>H$btqKC*AB$- zmb6LW2v@Qh7en<7XwQgH9%=JFOP~kq1|>-O`LQo!r*vdJ+-I2CN+PF1*-}P%vSwJU6Hcsu=Q6tDufW zh6g$5f=k6PnikIk5q0>R_JQ;r!$fZCZvicluj7*)+P!KX<)4d3`)ZSH_6$#I&78Cw zXZ7M6j5(C(O{QM~ENu|UGHIMnM(5z-DcI-i`ZCE{wrMvT){AD^iDB~fegGs zWCgN~eMjN#DF&L2FDN#?V(=3g3rLBE$rYHSh*Lx~oe@rTK|R7WmobbJ@dSAm>AXt* ipfCl(Ficxq!weY;xSq)y+4VT-H%Y&hy1WCkxcwWzB0z!w literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/monitor/SysLogininforController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/monitor/SysLogininforController.class new file mode 100644 index 0000000000000000000000000000000000000000..3a42803c0fc70661bc8fbc205faf1dbbb95b8b4a GIT binary patch literal 3901 zcmb7GTT>KA6h6JcE-Ztq2pSWUpix;aGREKyh{|OJk_A>kqA|B&w}H{wnRR-GizW|w z%MVD(N?ub*UXv;d%UDU?^FK7^Po(no%C z1zqTi1|K#}%h4R&G6kcHS<7BlMakC9K5;eR`x-l-q3}^ zXf7QA34B|yikfb2l^yJgoI^CA0QzjMmb7KAdc4LoN1H_40G^YxNtBUZC%&*pV6vkdO2D~?lAC-5h;5eu_il#tvRy~Oa?PQ=KaMM;nzbaY2vBITd&pFaEJ zyXOyodiK{td@i~I_e_X+q!Xi)JCV5Q7KJ(zIR>2X^4p zVv|op+7utlWHa*_Mi*)}Zj2*6d1zizeGt(3iqY|{wK}KCn(T0UOVK`Gv#y-xtdt9; zOoF_D%Fd`Ky%%pajAk>#)38+y#N)1MSoyn*-uJJG>P2DBkxgVzacFeWyQX2GEbd;t ziz3w$37?6Kg!1Ci&(FVk{Nk4%|NZuBm^w2*k(mXq>K{=bly$+&$vMll^Zc4FZ^Y4E z8bhBXv|ER)R9quHD0H75z(XPA-dZms z^v&{PCLrxK(n$9n(F6~j%BV>=n(gG!X&Lo}(t#~6eF`yZ6S#YM2Nf}RR>{uF=u+*C ze@9bRmboLleefI^3T{N$w|Bc~RUZp&(F-+d;Hfky8&k7i0KG24XsYJswC!Cmxbu!l z^!9>RkwUMGYX`_Sxg!ArYCnZu3FJedDSD-f-)>z?|H%KY#7N}ACBNH z0c{U(mLqQpqgs9iY0(~aV!0|U&@sHPTk%wn;fL!(IEb$}#zWpJa(dgE+{2Z_{~I`= zVq_Q_x*rq!%@a9_e|d=LZF&>`ihqFK!Z%^w0i`idxQMY1Ywg{e6zhIObv;k0K1NgB zy_?kVD>mv#g6;DT^-w+aQi@J{pdG$o015~?PVWFb1=SsN0>CFBDOLL`r?4U|JB+at zt7$=9)Ga5si3#e#tQXKeuOFbQ0f##zr(TcK^izu8YE3+*#)0}H^*yIsn>Hya?e!_P zLW1ZVHPd-G?gA{m=+W%=aVj)BsLyj5usy;tNVI!|>mZk;GxRQyB=tk!;_sb;4u9_~ z_M{P=Sa}co{Q@e_bLhKRHtG@$RE!$LY>3{+VHv4Wj4_O9LM@Dt z0Q*8FNl*JYHPJ*QgZ2e%Ucq4^!eJ6>KA;Z+4`cu<&1!+#RL>^uuckFejdVTYgO-X9 yS}03X0Nkd3y6l4)KNPu0T3z>ip!h!Uk}eAdv)~Y;IS;56vw3{q0I;8m3;zRWY(~id literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/monitor/SysOperlogController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/monitor/SysOperlogController.class new file mode 100644 index 0000000000000000000000000000000000000000..3e08e4626cfcdf7a1da10e2d81407f3c2b153bdd GIT binary patch literal 3282 zcmb7GTT@$A6ka=oq=X1GmR7VXAT8n2Zd+U00@iX#TQoofimml_l5E1MCuhgAcPQxi z6MXWaGmfvm>x@IEI_kT>L;EuvzkSXL2_?~VGIREAt#94e%Af!I@;eb-q6Z1;qxTZj zOW7`bmb+<%R?RUNr}yKumY^7|AEOIq-7t@L%;TonePDKX<8&`hA2RBZ72#!NXx??mW1Ebk z3$h61db74GR;uN^@HQ++W0cCug5_*lo^8(IWmIq55Pzk?l!qeE3)0n|bR6OFvIM&H zcxaI$OA7~&kREkxr5WYY4G5t#in46kZjBM1n;B~$g6{wgNuJ;pt0cIu(W0fTC3i~# zIUf3((MSVZOKe7c*$38+#jDzOcotd#kX`b^0LbWYD`%3}l&K=9CA`~Mxlsdm{shsf2c6u~)@w71*{+%{19UWtI zsi`azpSoY zu64O0GIg?6P1?w@l9Z=Hl8Pjfv_+*PZ5y#REg<{r5YI7%7-5Rk263pNKUNy1*u14g z-F=L@l(syb!`wlz>)fH@t;aa3K_NX0dNr8tX3j90YIFJ>nNP`*&}QU?(|9&$rVS&k zRT?&(m?uJZzu*@fOQ{*tth1m<&(tyT zgrX5z9-rK&&L8p6;e++~M1vIdH5>Lp2Lr|s6cF?py$QI}~7@PScRu)9jvKx~X!z#75YJ05zvJ3{-EG;VH; z*+pM(BORm%diT;a%>b^s^x2y9x9R$U^mph6AmS!H=CENj4;2Fc&f;Wz&|@@tLN72v v0`$~1N%+#QQ5U4vGpHxjrnwG_&2-2B=q3b3=$5Z~3cK4lFF}0Zqj&!WS_YER literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/monitor/SysUserOnlineController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/monitor/SysUserOnlineController.class new file mode 100644 index 0000000000000000000000000000000000000000..81014f7b00fa036010a2ecc0ede1002d6ea2c7dc GIT binary patch literal 1796 zcmb7E%~BIV5bjBW!4O4JLHq?36F?bMR0@Aee#D|GVhWIho+g_i4D9a2nOQJe-hBX% zJ^)8OVR6vA@1eYfWzQyrg#-&P)6+BE{dIrSJwN|!eFuOMxSN6m+)Tj`&?F2e;TD0D zf#&Lk?+TyKbG0TMPGC6aNRO&ORs_`*ZMa9#$llSJ`HFtI-{S#+?zzA>!sClVi-OC? zeP0@8g!DCm(VSEzs>@3Fr6t8YzAn`&EpX;*y2KnKRfV#;104^HTo;NLG3=3t!WZT- zfq1rmkw9Wn7BRo}9A-Nkcm=K&7@`sA$VrE}i%bct*A5eASs?saq?&bJppNv7lCH}Y z^(01@itZ_DvVM;VBwV2lfqXW%!qymdnO~yw##Zv6@uWY3gbZSI1XG-rS&7pysVQdI zjK3spX`;-EMP+ku2u+~XKFcg}W%F-FpXN@WghA#UhigsCikGD}G_N=kxGa@;#}TDv zjky5^x%x!wp=GA?TzMj!sTDpZ!-pTUMP~X5j5LL^q!ol;j6TvcZl1Gp+46uOw8~Q3 zl2i`Q$&w6^vqw!hJBXp8^qBDJ_zHVF$93QujG8oZmHRVA0t5d{J_2fM=xuz2NONIw zHen--ZJYG3&l}rYKej(@{QkI!Y9Y|rSz0et%6Qw?7>WCVr|Cq*%to7jF*o^?!2LtB zH^p6U_%4Ad`o31|2Lf02j#8f4kuhBDnW=k+#>z9ZyfYY>Y-(Nx_(;>j)_>uE#vigB zdC=fCYX^KZvuAFa&wO*&HvxL@lF?ot%W8L~;V7h%a61imU<8xe{hR)8nhD@E?tE*} z@r-;36HN2nItXN%do;8wcB>A;K_oDkJ$#o#FU0Xn1|SfF7C3@3iJFzIDB~!xcwF01 zqbLc=mVr$mUvU$IG_KYd;0heWHC;7;(c@0EEH}1~Cc?@BpMp28QOnigbey~>|SZDRV z4wirmA(*W_gHdV_7vT~bUe?e9y&NmN3fJn8eQ+I(la`STZeoxP*Zwfg4%7!wAFP|- F_zM902Gall literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysConfigController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysConfigController.class new file mode 100644 index 0000000000000000000000000000000000000000..e33e637eea7996cae45fe2bb2d756a46b894e0c4 GIT binary patch literal 5520 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 TwQi;>8*vM6_1~VxZtVFV8gA9y literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysDictDataController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysDictDataController.class new file mode 100644 index 0000000000000000000000000000000000000000..20df591197a0343ef864e3eddadc318ac1339242 GIT binary patch literal 5239 zcmbtY3v(N16+J5}mLqTN#C2kl0&z-OKbs{Gnqs@5w(O<`#ZmpDX%b*9eX%!@wCe84 zv0BQzw6v6WDJ@Xm&l#9@hH(m+;RoFTFFvshsk7D-|OCU?>+Z^cm1#b z{^OqjPUFu>+=oIEM^W6*huw`5o{^!GfSZ7qL?_B!I4kc>S-c^OTe7#(jc4&(5?{pg z34AGu1A&|`Cy~HcWb0RD>(^xXbs2s`9xuq;Hxqa>HM&aun+$TdtJUUXVrhQ$**K zRkXab8ajqYRyD*^c8&)2rmdn{C>Pe0vuqNuhQ73&HS?>cW68KW8TU3UT5u-fjvH#t z$l68EvGaN5gx-lbB~@Fb_ITcMJq=feBNFJ^f?cE#M;ST0U|K~((Rp6Qh)9_nX>+1v zt}DZbClQ-3UbU&FD~zCqQ*F+**QDV<`kJ|E8fDMQ8)*Wj#KVp5UE4!TPaSgjxL~=>MG(5Odr7X~Wwq?w1WmSHKLt1rl zR~zpxpvQO0j^uRO_Pt7yORDL&|ML3n7v6pEwYT4U^{*VRm0f0|>q_Ekcwna_RV)`= zVP$!&Aq5ia%9C`{Fd4+i$eYD=BW;s0 z?UJ>cbko<&EfS=hm(22WHK;<>ysOL6aJ;=S&y2|-$B=9X3 zCBq{h0#gM5P3SM7AcFy@eYiZ znbly{zuj7&YgL>U6@%}3Gj^`Rn8a1iBEmbAKiS|zT*=H%FVF3z4ra;`_IEm}U~e)h zr-m;@qPJ(x?+^LP%kujGmws!YW%cikETF+B?({(4M z7wHZzJil?!a6Z_yo5rUJZlqp>mj$Iad1t0F&x%g(_=oEkVLL65N+3qE>}22n?-9jJ2;$uWI@QSsI*Jkxq`q&T<#Z zc{Y28qC!K{o@UfYjTL^Q6?yq<=pP=5c(iuecDu?;Im-0ZOr=Rwh>s!avDn8tH3IYU zxC`>m6SHSa`#-~UVok~URXm&!2GK$NEiNe9GDb)2 z@Ne|XE20_4`B8iyzsFTfOI*`j7ssr?e4gMAgKv%ReWN#_z2hHw zfKL$sihGbxJ=lOD40EJm1fx_YhYNgn@O@}>8?n*9p=0dth{te#bbK58-sVCFL|Bi% zX$Wx&=;d({Iv5haT8q=@gwoEhx+z!BPsR<5c_xn6x@wdI84dM zC|h8j4&>Be4&n?Rp)Shy0mpdakPoFH@H^+Q82Qi!Uaf^0tt>Ny) zIXppeO)i{|xbV~X438v)4r%^i8O1_X^~;ndaTb5x!x;B4#^djz_YHL4!hwr7anKLB zk0tJ73H+I&2WHVnBaTvkJ?w)uT!Ykdfx5&~nTD_jGZOYEshOnx4~{0V=v&ak6ZP0H zH3MJ9N)`Bps?P?-$k4uR+#C9?Uwn6{?z^P#uP3QzG2*|08qNW%`t%h49iz}H1wY#q z#-|u1_;9*4O1#Ly)BLaDQd>xeLr4QHAzdb--xe3x0e6BU5&*KYKFP$UQfLTEb ze;@|B>qxVA*2(XU2$U$hOsvt$klR)7WJ4&qfaD5SxhdfGdSlRHlq@{9d>MWGb&cce Iwe9);0i$TO+W-In literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysDictTypeController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysDictTypeController.class new file mode 100644 index 0000000000000000000000000000000000000000..6c2c3dbff630ce34f7c8c0a72d42f6c46622282e GIT binary patch literal 5618 zcmbtYX?q)26+YLSYgGOH(^`k|80Di=BpeODa$t*Bi}JU@TqR<47~=%*e4? zO4-WRvX#<}5}>qDmb6I;cHNW~zVbV8;T!*ehxg2kq>((zW1A12J9C$_y!Sne{Ez>> z_W=30|dOAYij#4}28tFScem6=H`d%Y_pFf`A6VLMF2R!{lKJ_F1{Ba}wglC@PnV<6G zd7l0mzrDbZ7bEmigkBcZn$nZbWIih=q_wCgrJy5;q?u8zoSD~EJ8wHOquSn@8lTA9 z!{z(?1??Ws8IGQj)4HwCq~(BNn2zS?reOcGV-cvT~TMGX4vYi zmUK)juWH2u>Ojshuj-aeL2ScG-O!y=f@-@CP7A6VGE=awHK7}FG?$r?)}#h)f?^3X zsimhiOXqKYvd)>;p}sfZ%T+m}CQZY!%ye2>1=j)|4HXxmy)LcWj-W@o0uq2r#xx+r zl4{D#Xu6?Fc<$uYfRz4&RZe8JIjOql4Qq}TH)c(!X)M4m=xCL5)inufODt%MnwoR; zw3>jykhrrP4&1B{$xy}KlbhCe3gY`hFVvRFoW($B$)vPxHEYSNX*=qeC2^(mrlmhF zakzd_OXqN9drsT-k$KG?lU7FW>hxIF$58Y!?41V%9S_yRcQ~UPsUT<0O6Q!G&4OU8 zLG|vfK`v!Y3m15~%XNFn<#1X$>b&D*)rat4!ee4w+F4Ll_HP;vW@V%09~ZP?MIaYn zn4tUCtKjOg!?y4gfTF##Ds`i zpg?3!Ivj3-`n@|+)0#1-CQLAew$^(`< znbKkL9kLH^a7~;V8y=V(*-RQ5%LT~SV9AWRh?qFpb;(P+C9-UWauJU(sATQy*kd~Z>gFRiF@uz+$F)}a;(x~N9 z6f^5K3{cFRYGSuJslf5O)bh3HgqgFF@{G<2>Fz+`EtT&hTokMI9^o{xozihICA@73 zKk7291Xg>YvtS=o2Tmx|O+5;|!jJo?GeSRC=vDfKLf7b*g6>@xqaigw5W0>+Q?{W{ zW~IMU=-2d`LcgKc6?%hyE2uL>ieWpN;iyw+@_>Q+0ocV+g?>kGM(Fnn{ek`{XqVe* zxHO({xSbo!>FJcTa6=!xaAWPYYhG>WROnB1U7@$=Z9yHKYp>t@_pLYn^Zv8J%CWZc z&f4{t)?R&6p&RtJLO1Chh5pQMx9BelEz^oZ@6swH|MSDQK6-w&DEU2JzRk<;(_h)8 z1%*CfSU;q{A)S<*IK>sTZKy^5{p@3@u zy|T!KvSy|Xs5px5L(q2Q@=ER4)aaN6OX{U%nF#qZ2#b13k$|_#UrJ@I6zk=LW4L;o zZRreIQgh^BzQW0ZY%s4yN0)B7=6e0+@hu=82Uw2Dwjgr5;?*J|tlwcPr?TiwTq|30 z?rVVsg0`jHs#%y|>f*+>htz-e1r^jo-Y=u=App^J*!k#L+82=EmQpsEY!NfTHH7&ULL>3#@4fnBWHo9V?2d$9}S1DO4cO?}kQFPBKzbP{h5(kU#mU9Uh@Ju6|G4!Jgs z-lm2NvB(NF_SNmK>p2{2TBa!b>zNqC2?8k1IKLA{cfd$C^SCFcXyzXJES-if$aW2` z!HFHN!L>LSr2!g*BsP5~Z1L85p~G7n!Wui#iJ4)nkJxmE&PsZS;svKZN1umyǓ z&e13~vr|JDYcRHU`$#v}pb9DsbMVeixOTI4Y%SW|O6O?|;wm6~H~`@|P2fm`=`e{G zK~psCtA5T`9gns6x?5Ry+sm}S`!=;-SfN;K+cItMS*9HrVmt9zrd`YQu^T<0#_nah zyI}M^eDLEXZg{|-rWl|*1|RzX3bjDJ^W z;jKp#FCO>Q5UyZvi6&8Z`7+4*Xuzf*Rg#qQslr6{0hSe}%+ee-c|@JBq>fGt(88?c zz~pdoiL*-kkb6R`nL}2ZcdhY~O^bi;g-!ASBY*Y)zQuZ;SD~s70$l}!)-YkqZc6kUn F{{ye)D?|VQ literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysIndexController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysIndexController.class new file mode 100644 index 0000000000000000000000000000000000000000..4e6fdb0616c4131fb086b45be2a79d2316d03b30 GIT binary patch literal 1088 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 UkoFV`ev*&U?OmeZYqJ;s2UMlw%K!iX literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysPostController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysPostController.class new file mode 100644 index 0000000000000000000000000000000000000000..f4e3641e4a0dddf9fbbd8fccbe595bee2ba7799c GIT binary patch literal 6294 zcmbVQ`F|AU8UId5vSBll5IKsq0tzN1;FL=(K`RMS(k_@15(Rrrc3;TA?#wzf8;I1l z_OO?dFMTz>-!ww!z^7&M8QK1h;BEJ2foXS+aerpaw0^8p^9Jw(a<)Z#lN7;mVBT?lZiiYuWqux@JKg zbld~RkTPx0*lT8e#~n4y3W2eul1byR6|DAv3=Lc=apMFE3&4$vO&qUdCT)P zY;Q|YM=0c+f@#@CMAf+}ae{~y9GfCsWfaYQ$_P~1YWilcz1I=80Zc|mqcq&_REuTKgW|ij|MOTT^MxU$b znqkMaZc!wz8!_`GLci=9&%12c^!k)ru-aNfYc>hRCV{YZm4*#f<%lnaEIXI<{dLOU zZWfCS6p>(rKB!lRijGUvGHrpGhk`iFE8iIQ{i3mr4+&Z>>{MQnA*VXeYQ98MgBu;! zFl)y2pg2Usr3n#{*JXonIhN6LD68@VoYIQkSsk36!;)bW2~8wU#tBgKEniYXpLqJ- z*Iqbs`sgF4@4TJUp_0d}@w{D(8VxI_Yo+W`!85uN$EbaM&y9W8@7$%~;+fqXxJ(P8nqvr}imdQbj{&=o2Gv+WU-*!-$wE(V~3Zvfn(!04U`pQ|cuEdvld6 zin%cPW;W&Kl5$7IlDWp=ki)GseR6H{c!)=*e^)3|E6vU7fC4lJ8W6 z+4H-c+$im@RXK~feTFh32Lcrb`nGP_)pIWOq_32SvU*n)oDl~9=C+%{dKpK>xx`i~ zj0|2a^QZASW1@ue15>v4WO~GuD`p)wTNMkOigPAij!=&wsyoHtvgMg9!R%-|7klc& z9$IsrF`3Ie=3>89DCSjPc%V5b$ZO1eevKEbYe@GyB{!?CwWQ=Mimt@?BE3wmu$el5 z>apSZF*(Dddh<}7GzhC0A-#ORi+wsSgQ4RId9;*-T+B5r zjgM8mwMEF2mC}3a?y*xxj_NpyZ|E4qQ#zi;H#M}zOd?tANIc)PePe(P7Of8^bcnWA z$1`{~g>fCn@SKLlK}QfoZND$=R9DH$=afs|ow{di;?ak~n%}D9IG)#W0w<|+>%?PE z{rj26|8w$=aChR)Bd4Djn>hCL#FO_;JV+Q`z)2l1;w2s5!n0KfPM^Hz^uxC$2)-=@ z-@$ivd=KB(@dNyj3cU8x(NlLFuPE>%+5WL?zl@*gcm=Pr7I!(0@AUa~=R5_Gnr#%!9YDe5L_XBdS6_XHAMqjK{Si8x%R zUEo#ohxwlJN*-ER#w@BVomZu8M6dB;Zl^$6xPUaC@ZdE0jv2`={(fnJ7rDk zl5C;pvdkd&?sFBUPH-Ex!~56i3BBmwV=w~Pub_i zdAudh_P(Wzd7Of2`~iPtdrNj!?rY=P=!NcF z5U>cubvuP3n@oNfvo6tRoUqmhu^S(n&LfJFDJ;l(UQ87hsah}io>~%_GL>T0Uv?Oa}YD?-A~=~16pnM5cbHOb?%-wZ{bnKqez5$A3;LH(95AHwBi z?I)%wOz_)k8d}w?({Rz5&tW?QV7q*Y1ru`Yr={Szs8fUX!Kj-g>w?ULG#H7#?;krsPH6_ema`1UsJDs0a;!ZkX;01fB*~}`u)d9&tMG-PB zI4`a!nGDWcLZM`E?!~?+I3+bAzYI@N!@_)bFd$PxZMe~6W5G@;bWNU8Z)5LB61sc@ybQa3Zu tT`FG#F9`M)zWJP&sBDPe2>#A_n)AQnU-%oA;B%adu@BOO&*LF{;eW1A3ts>L literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysProfileController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysProfileController.class new file mode 100644 index 0000000000000000000000000000000000000000..9e6981d7998ba0dda5cf5e2d36cdf542fa5cde36 GIT binary patch literal 6138 zcmbVQ33yc175;BB$xJ2>f)TYrt3^;kLKsj$Bti`sDQyxWAuQt3m&qecn3*@udjqD{ zwqWb7sHG}Ot+Z~n6;SCULBy`MwXL<)T3fqVn@O;n-8ZZ4f8M-Vk_=1w<;#8d-gnQr z|M}0k=bksuy|e!yfH|tdkA=9}kMr;mKR$|&NpFh}Tg&*oMrJ-reWz9iQ^zPCq8%Y9H>B=iNT+kmo(3|6UQ>DQiFN z!+kRNj5PO4vrC%KO7l5c@PPCll;$C69+u`2X+AH_qk{4aetZ#+iQe5l>=DR&{dgQt z`0%6;`=t9N(UkO|w;X-w_hUbvD#HOh?Z-hpBk#WK!&iLxsuy4L;_C|K3CoDL8Szb_ zprKIH6f`4&c*5)n1*{&+HX;GOmUn?=Z9P_dZf(6n)rv&a4n>SrAuH4oHX5Q))7I^f z8MPGVH<|ISfE9~}qFw9bdc@dl#y18!3_WTE*6TsrjQ0fetU#b4VVj#naif!JHQU^1 zM6-})=0VD7+H7=Wk*&;IPx;dGLeY@DNTIlT`YHv_60?(*RWya7MoS{nVZ>MJM5|EQ zWCr!{Dm@;Ou{-IpH-rd(Uf!m%RnUyuaWfn?;u%`VMr+($9|{{wvhTRLHQdL$`kA3d2f-ZOWB}&8|>X_*KZaxPV-Q_JjjXne`O*hz}L& zj=Y~iXW2ag*Nq~k6M@25V9qJ%R$-_k6z$CS=u*RO)?+ccpF539#5#4`a0ydbQ#~fA zV?pR;*rsV5*JkU%jRfJi){Ad=@vOq+(L3kGAz3Bd5)vg5q7C#tIgl~uTs;`f*5AEqav}OCylTQup-ZgaFwW-_h*6<%8^KA|9;J*q} z)557e*NS0TYwvnEJ@dx)9f$TX*-jq3b8mWPpzoGL*E~JYck5tg&CtDlsoQSW6!4;^ zlq#Z!hABNf7OyJSlt-0#RjH=Dgr$6%DpP*1D%X^z#%XH2U@Q$qy4vGm(c5?9z=8Wy z_Z}R&;kqN($mE`jkz+3y6j%{t>aA>Ou|&wLHx}zwUWuVFqp*ZqRE#JNScV*N`S6-R zVZz98RX9F3mCa?$V_k-wEkP7!9j(|Ifs6UaOAYCZI&d`M;V4DcN(6(vh!qy*-&xB$ zx{aVcrqo{UqKG2-3q7Y zXk<^y;e;IuTW+b*E`450A*;o-8zV8hN1>qViy5ZD4MuRIqr5#D;$@-GFxHZ?fS*fQ z8tDRJ*uEJOxo;^eS|iF*rmm39qe92c^$b{_ZKmDMO2&xCO|mg8a}3hR=HD0xzLSrBbbQu1I&%{iimDAimjaBw0tkP0Wm71>Zdad zD~xC`-V@8)RErz5!*POC^2ex(yax(>rppC(n7Wy{H)Oe|li$|_Gi8*$yu8+jx&kW_ z=GqWzBlcPCkfU_?wCeT-esqqKO0y*8XrP%tD#$lhhN@-F4NDs@Z(Osov88R<@)ic{ zy3xcGy6WI(oWnBXVmj!T*jOy-X{3;FX@5&u}N zLg9RlO8IPXW?6oXm@DC!r7znHxgx+PD7aKDJtV0hZzY94UMN zP25a`?}Uecz=E12ikfPZC~mHqk%XtEuCkC~DF>xhP677tLrP8b4-XIa;Y@0;Mj(3KVB4Xro4j zm1yT2@v3cn3f|WRZ!?h=x75}pp(QbnP{;S7Vh83{9_PU9N2Opqeg~%1JcWsCdvQYL ziAkKqgT58IgwN@}Z0`VhDW{+h~JLc$DZ zyhFybWh|^#d1mzCtbUwbOlnW2T(OKvzOcw)hQ&FN+ZoO)B%HU~e`HQyDpGdUA#RUQUf{V^G#?2kocPVBh=h|C!4wgUWS<^KzLLEUaeP%) H=EMI3BJX^* literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysRegisterController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysRegisterController.class new file mode 100644 index 0000000000000000000000000000000000000000..0fc9d05c0953ea73e6d18b5f1fd03d4146567f85 GIT binary patch literal 1996 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 OReUC|cAh@YtN#yehxAMU literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysTreeDictController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysTreeDictController.class new file mode 100644 index 0000000000000000000000000000000000000000..5ce95268671ccf81028fac4f0511875be84d7852 GIT binary patch literal 4130 zcmbtX>vj`W6yB4h38VwX0##6uyQLHkDo~(M=?#pgZRkxd-X@vTGG#Io=A3C8kzfAu zr%#~EC(xyfO)Kry2SzOIiAS2VGU|7i#`XwZl-oqWY9FWM+tK z)H_!d^B(6DR!PorZ`CSc@?5^;mYL_fRf~zLki5)9xW;m=`-e3;FzY+gD)U84SVfzU zI*uz1X}ONj=yKlmmYJw{mb1L%8D+ladJkEV8;)R0MoGF}l^Kl#Y}A+Tn&oj5V$;_w z$C4u&CHqe=YLpsxO=#DXw;Vq0my6t+H-H8s@@~nn7Y)x+1#=z+TyP0OfI3F_t+CRGKiP zk#m+@i0+1*I_nypj2vwbPNFE5`aP7rreaIc*}YbEXQB7C9R2WRJu3CmJ2w9Kr=dg{tJ< zrdYnzv3&(YKBxzz(u1#8To2Jyc(rJXvvE&onlCMzJwDYDMCbC3w#thnhrAnvi?pP=_WFxop1pYX{og-)_t*0uF)aE5 znL&to1p4M%RWqJDep#@w$TsDg$vcIa*?EmJtnQ?wVIpMGx#_t{wQ#X>ue%}=)}5c5 z<&S(W#f**lTd%Wzff{Bs-`>oO6E?>n(SV~wXQt93p zF;H`2GC!%*9O{^+32x(RHSd3hJyr;~h1ogR_e%VxrTlv^l3$yd`JB4Q?aE4AszEZ4 z&Ew!Vz9PjR4gAv7bWc4$Bdsurx~bCvI;hir>eJ}Fs+t`n)fvy^UJg}NrzKj>&;y+; zTG8nt**cZU(aEKXPLIgb=!B9PR^s+ISfk_3ZFLP>N|aJn=p@P4X_eM=TBoW`U#Z?> z`Wj*0l(CIMsS%3L78QfS7b&!3hTu&P;HDwJs}iNrBZQ_pz8%(T1t9haUzgkwcz7_B z#bpam4AfuC3Em>CRuu=E`qsF5$>RngDG;qCsWPiZ--PKq7CW$dOAU@>v$lSDnLTkb zk?FvndQH?`I`_7zRJ*GWs#>2aHS@=+s)}Cuk8<4JMmSygr4q99mZCmWTmie=D(btM z9vQV!=7u`0slgKm#i z(pB#ojRyO3?c5iaxzm0HFG5_jZZpWa%8l?qUu;Fl(NX+J*oU7S8YNUpL;E^LX|!*k z&7eiL!}m>$7+MW&*T7TKenuxjZ{e#p6Sf_~R}VMPVR{=Qjg9~y5gaa45`Wk?@GB(- zlFuo*L8&vpQCE_t1_n1M{Sy|Gfxs+|5FLkRCjy~I!hwF1?4t3h`SYrqCX$0q#uD)HBN7?d^U_`t;es-d!8CyGg?C z!03zAO_$)9p+Jv;P(VYELp370sTV}N51{?D1HT?G;;f=X=^d_JYOD=mP3e6YGnesy zMbK3m=5&p&N5~q1>K{P$He~fEvi58x>qZ;0LT@QhrO;;38#Idf25Dm$XmCzjT*f+ky$Z_=%HHuuIN&$dDSka932p6HM0W2}WynQz*z0tPV=e5Zq>y%^76Txe{+ F{V(a5hIs%0 literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysTreeDictDataController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysTreeDictDataController.class new file mode 100644 index 0000000000000000000000000000000000000000..480aaec0b0571788037e237b6ce81ce118f7f235 GIT binary patch literal 3038 zcmb_e+fy4=82?TZl0sH#sm0=@LTwdN?5Qoi38fN})@Yze0$%#G**%3rH@mod7R)$4 z`sjlXj<1gII(5dWml^#7{6FB&aQvN3NEQf693OViCEw-u{l3fQ&%b{AgNUZ+;{h^g zuAj1$&(b^<2CysY-95d#uXpe0S4qF#)vxz5RL;->qv44h^^~ck=LU>!l|6rr2b;ch z*B<$1U2J>)V_p-c8}LV_r98jE&9(xc3zfGmePP3FdPcfZ%`!@kPpmRZ&3iTw4wj`W z7Q=c?_*E0$7!8#@%XC&vU+Qz~GNslf+)u~i*%mc!d9Lz3#}WQscn6}IZwmlH%8`L$ zR2`35X|ZB^byK?hfcEr6OcG4iJr^#0!CT^U8|J6WA_yG?>w_>APXeR3_CNb!bh5l* zZkaq((&1&mgu&tcL^L+{$hdGocBXlDOm~exsBH*KP0vDqs%x$^Z8hoI@ig5RYQfyx zL{6dBw`DpZ@{m%fC>(inO`5jNXzVZ{v2>Tc$bgb6O62ahU+w(<-Ol&l?0oUd?$=Lt zzx;ubQ40fPJP4|Yhtb8u_JkYO1D=nG0aUuQSSc=78C^eiP)lAAi|YLTqFyp}@hai( ztM20)UlyN)B2anHZU9|M*b;@6URf&4Rg1kazZAwWrhQTOw!q5l_`_IAD@u1;Pqmno zHoE0>7b`$nD3*&F<;`PLSr87m>Xb;De+CDiGrDnX6HBIA-|J~GHe{^}>+y;!L1VsA zHfzEGULM`@FuLYW^R_>_Y)*@8OTvm~$3VpkeM{Vvy4ubi=-p1|U)G&<)adHg4rpoB zQ3u-YQ?I*fR2Lnp&a@g|HxainpSg_zZO|ziGUyq4HbaXBJ)k9n-lt`QDpWOSg;ovv zK<_@Jj|{p+BL=-iBaCi$rg@Lkj}Zpf-j^Ug$zl^UZw8_hHlr&YjaCr<^6g;MUkjyU z>nVs?-OHFgH2g9_XjvE&jP8ubdZ9Jm;*&KVO}Dw6mM(CkJtunrR6?I9PtSL)SL(@M zxQP^lfUF(0O}j|tV6}(U5#IZ80eYu~B z{fJG1p2J%U6AjZDyp5&+PS4}S=q!yuN-GPjNvx-GKT+b+uar#CLT>UYrJmq285tUY z2+=4IjYY;rn+h#sP^A~>MHtP( z&^S#1Hqj+K8J(_4Y#6<9wC7A$&olqQ^JP@XApbs| zd5W&%w`RU?bYj05#f|~iLU0#~n)Pgp^!@}LWPBm!J=HXdkG=pdobSI)^2)kLF@3z6c{Wq(!?kNBO literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysUserController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/SysUserController.class new file mode 100644 index 0000000000000000000000000000000000000000..246ba21d5574c64272676a609cf4685ca1c80135 GIT binary patch literal 13170 zcmbVTd0-Uv^?xtdE}MabMZh5C5+&qdRS+e?BMDKXNk|Ex;yG*vSlH~YdjQc=FSJml z;zh+;>xFnCA_ju@-Kwo;YcJ0lzn!Nec_`v`r8ov{JoI=L0`(_KjraXURqm7|EB+V2os0M#Q)O&`DhsZ z&qH5(n0z#p*~>X{%`N0S&X>CaFkp`lFNT00`{c1uUTE?-R33+cEf;y|Au!|OLLSb3 zKM@~T$R#|+#|QCPx%-Zfrt&y1kB2Zkp^zu?Bp)BllZDM89zN8|Q@mU%<5Ru#j<728 z6Z15o4S4u4FCQ*bkMQwyp5fsmeRL#!;pLgQ@=-GVXfMx_hbob&T1IBe0>=o|v0gq- zMvj+<8ZXz%L!FoB$hBU6PVn)Ge3INX$efdf$tiNzC}T}tZkFp@8EKJsO+NlE&+~Gt zjLesx1@f~{Y;mfWI{cIa3t;4Q(RWcHpTTDe-C0BUY(B?_pL6A{K7qpsjM*=Z4bm#(T-3c(UC|R?E$=&cY%i1j>H0IZZ%WMyi_C^ zYBv^!5~0Omqb3rGCiP?}8c8r6-58B84J2amP-N+nxZZB8ipEz278`mb5m=%JlhJrb zKu;3{YEsGQs!-f$1KYxQG;9o1qIu3-A#F2a$$^s2apnqXESeZFX^S%#qy@7>kx=qj zrrgr03z_ojqHQo`QDZ1#G^g4Z8}a!%%){hwj0W}aLOmXmXKOGoxhw>GO?Mf7m9aPw zj7E~ML)eJhez446mwthj^TMG-lIiqP7d0H%9*uxO+z7Nq+x1W+V8Bhuj)29XdTJlD zWBO7fV2U+IPwEYkB~e*I@WE8s2Tk@Bn2H;h>nrs@Dj5m~8X+V&jmlKNBW9c6xE^z7 z4{EJr!MZ9d+7d=E6&JHeVyopua8Dq;w6QuCjRQ2Nm6~Q>Z3cMQNCuWAld-@_ z_>lm@z&s-nLo6HBgK~#NfH!WDG7WeUK^nlqbZGD80*lYBkeMSdT0Xv?itKU}aA`Jb|2sjze%0GfR_)Loj$l(>EGi*+>jt7;xKbn0lCR zw8z2_j;VaW7?;>6vY0vao~1@ok}XrU;uJ8|XpBOM!5G?ujJ3c_)&*jthPHGTcKE>^ zI-J8a7{>WhkWr()b)Ghp*Rj=R%ds3Cn1@$*IPBqerip!Luk%tyyhH5I0j42=3zSl8 z5G9#?M~g1~Mbe08G$1qk#9Pl)-J&O#*+m(sLGn5;C9OyD zgh1=3GRg!~#04#LYUbAurbM=+T*;{*ZnQ^NBARBGp5cn+!3Tjt3D7#c7oUFft6JYw zWlnu#y{K~3fFU%;K)p1aRO_?RngCp-8XC!#RY1|?zFlok5($O`>?f3z;4&mN19d-{ zf>uKVna^Gwos7g6_EkuO0sGMw0P7D6p@_$m((=eerp8`rrkAM(lO8;h9va#pT>gpk zYMKGlA!#w7y|m=>;|A6pcnnEvKB(N6C#wvioiY~?K^^IaAq}@8w}%Z$qR^(ibTC#Z zHha8EJshq~m|TFsu-?A7O`mN0Wb)xmhYdK7h_9-rqT(^y?_!IURVpSQhMLwIO~r$T z6lC<)y>`R~pXJ9m09NruDrcf`EKM zx|iE~{r0^ZwrTu5UgP2KYkVPJr18akiN=@mS`S~Q@#Xvjjo0ydjep2jXnZAa(D+Au zJ|1bG##ix1jj!fwP~>nB=tx#AW;Imm=A~+ zK3wC6cqi(*#t-u*rqMk+vf5NA+$MX;p?f!N@4oG3)jChnxRZBjyqniDji1te$AkNK zZu$JtEB9@Fr2Db!_wT&3`>I{tyLNQne?#|8cWKC$+ZTq%i$sWr*#ZRMJ?B9CL{`EUP+`VD{rH_5Md(XbR z_f+hEU|09L%RjnogL%#V(4dkz zc}*Cgv(k{s=O7fVQ=r*aNlt6jXZ6c^NYlSAV%b*vgs({HQrjV;rx+bfHSTu6+B|(j z<#P^{%qj)LLLIr4O@|+K)A5Gc!HMOa2_xJKstykuqXXQdbzwb`sFwXwq1v>X8-J!I zHEAG+7ClVv63NyP+ymK8DJ2;C*!ltsW(kg^lHg+K*ujjp)eS`h4UzN<__$E*6fM!G zSo-ytmIWJySVC|UF}Jn{*!(j$x5FJ1qr8N1u!^j1o@C8kf_|VBd-Ps2Ad|KA7xqjI zmPiijve3AOY{T^)!rF*mFbHyM!Z>?ML)WngGz%qa+S)^~Zx-O9KPyOBNI1y$lu4~d!EqUa_M8bZJwZigw!MAxwDY6s zGRr8~cO#UT=DH0ATgu6f3x2i<``?^94G@ItKUd0;Svpr*@Q^VR9+P^3SX_{>PMj9V zJu+P=(Hu=Srvy)~!cp5HxekV?CJuhs2dMYWrGQYO~0TLXB^GFh^fIvsUXf zj1?b6MkE;Th`9(ekZ3FBiW6%{N#(5PQ|B6j|H165%{ZFcVCFGN3``ZJ4d2l3WF!z} zQDK9m5M5nY+CIm)~{BVF*#aNigUSl|w=rx8jtR7?d+Z#C`#BUs&0=P1+`DMF^A2eAf z@GA=ul~NMFnwmf`u9wyI%Oa*>V7kMF6#S2ta_6Q9*eHZyeJdwE^f);j?=E&Y$R%ywe zXUaZ~JYZB0Nlu_*I+4aw15KclVVYBDIyIWQl&Qtiy3De4(WsKv(q)i%x}{ZaS}QfO z27+D={$Aq$(pV4upr?-OTsp3Yjz6R;z)oy`BNolab%MxVVcNa<5gK~Bf7mW6s>&E-iadW-A23BmD%EMX8^I zSnNGBtx0nwFy&|Jq!CqlaM=M>`J!-1-q_tVvLwHYM&(geL0L&bCyjOxH_BA$ObB}x z#5x;ZI|oQV*Hrf;wL)6mnRFBV)KqsS-Aq4&$Udqtl@ygG&@Dh?9ypAlTj@6N4S@c3 z+!drvpyqEhO(65vVE!G@A`e`*^kIxUZDVY88RISx+)ejb9$pTH5?%hZF`6pM#is{4 zK2@P*K;$Jf9BD%0RfU#6t)SoXrw=d_-#MYBMmAzZJT?rgJC3;*PxsON$}xN%lo7OT zu_C1;hv1AcO?WKmqJx;4EBs^2chh$^lc#bk9aPyxdD=47CUbT7r}o1+*_h>bM>GN6aVTumo9Z8=fVJY)Kn( z3q3&FVU{95`9azNvz5TK4*?%eT&ocv)QsyY%KaLVkmsR?J=E!;T^?8>$GmoZja-)b zB3Hfe(4Ma-Xu4z!*08eOBX%apNzkJSdW_^xLx)T9egcz_4j-T=7tvFAewv=Kyw-|q z4yJwJKfDbM5WSaK4 zerb=-psndaTiY*fT?Xx(=cy3O&w2bt@|F4PO=%?ZJL!aSjGwrhPU@tFGE1YAh4U#+ z=+C5~(0DERk$8`Q6Y7vun`s?B=B%d+kv}Da-h=$sg;;o!HX`L*O>fXOrt2l2!CGmD ze_$nYNzoq4Kcc#NSb^!6@i46AmuGuqx#ti=WICsi!R~oHynug1JNmRmlvRA@vn zN%8EYrY=Hcr7hmuTg|W(=6x$~xoHRMEjR5zy*1Z%QA-aORUy?7vaSz#cQkVDRAk)- zgWr)R1$e_0(_MW1x^eWv&Z^3Er(!DM>eU^69XD&Alq5Q9?A)BLq(Zf+M z(o5ML^)mfZ#ShYeil2$)z~6Uw)4a`s$W}q5t=s$_x~YWxF!D|(Ah_El@5Hpc6X{p< z3J@W%$|d%vON0ffS9?hM8oiF$s$RZfLd*Q)gC|T_y=lV=Wk*RXU5|8PJz?bv9?=K! zl=jeqv=%vL*Q4s^)9}zU@X)g~iJqe}U|I5`3X)lAZR~ExF8Y=dzwV)gnA=w3H;xi2 zHr`4r5wVn*)|V1JBwyG=a@FL#3dvuC0d7xKMeQ{7mQw$$O6e`h+e%|g@~m2RdbV2W)Uq+?IEem; z>hv}`(Ra|1yi1emJ!+x%O$}8WH!rQ>JgZveQJ7Az}ui=oxq$d%Foo(um= z(=*2Mzcd;GXLPuWa?PGYETT}VV(nx2`V%y$pPE)s&Ci&$6~?_s*}dN1v> zR$FGRHcr&vL1%Z;IbC#a4*H}Wq+{{L>BUud?*kneP%-DY1Q=%tP>hr-+bT|PK-9<@ zOC;(2oZZy^VxR2xF(xjl#TATcD8L=OkvsrxUhaJuPzx)q-!m6gTg9RDF>*MWNXn<2 z!s8kIaD*@ze(`?!0j8f~fgJkGG${T{27Qj_zhDKGCjZJ`;@OA)X7YFb7ypAwcm$pS Y4n6??M)D{gZQhl01y@>Qr}60j2Y~a;&Hw-a literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/TenantController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/TenantController.class new file mode 100644 index 0000000000000000000000000000000000000000..91628cfccdc3e09078fac5957730a9af6df8366b GIT binary patch literal 6846 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 literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/TenantPackageController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/system/TenantPackageController.class new file mode 100644 index 0000000000000000000000000000000000000000..48214ecefd6359ce01d0e37267675fe3274781ca GIT binary patch literal 6141 zcmcIoTXz%J75@(fKnQQ4VWR+0b&yp+c<<&*p6+~mb7Wc(m@`OW|X-Y zoZg^Ks3`=yZ3;6=ntE`n!R$4A$(Wk!jcOVNt^&fO~pP7+nMo3upY92I~GiUGp z?fX9Z_kVBR2GE5+Hlh`|20V<{nqa_egaunXO*Y~fd(X3%5c}z+3QEp7CNzNe0lX_M*Y<)_V%_-C3 zS=n^I$zv4=dKexZu{BH0@`P%t;~CzoX}YCYs;-#~kHwEFlgbHMDJ{w)+|+F&$%%5s z9#vDVy$p5j9SMeLzn&rl55-lD58ByrZj33!kfAlMCzVV>F;wyHEk>;gl_WeJzsq>JFX{7CTsP9PnKM)^eZ)|*{J3r$lZ89RRV@|t{;+OZ2bEloHWFR^q>`}-bWC;) z!mu%}>auxUNvFwye$<$9Z%!S`al;8LqHN7iU&?=Zv+$P>=cljb|8#NTlUauDiZD!` zv<=mol9gnVo2HyIc#crW!v^oQtqI*wUl+l+^*PhrHldiq+{mi!n_Ne9yGhdNdA-|( zxv7I;Z8+-3rgGuPZ}TFbG*nX#YB?buNz#Te>98U!W^p5wIklUVqR8f^)u*SXi1|u8 zN8#&MsnXSkv{q1;t5hAM`#9f;(Z?5aI2m=c~?Rc_E>nqC-QSw^`Klr@*P6*=0S7-HD7^lt47VrUe6$ijAp4754G z%NrH4Q{&Vvs7vWbb=f5pd&w88!&4B&vNs7Gs+mi5?B$^*od2({)TTi!_t!nRGn^n* z=dn@<3q9&-XcIBgc5gW5r%x};{k=j|#Z{C1v!OkhoK=RoXvW#=)mYn7GqRv2!98x- zEiF}7q>}mVMTL76xd#4K4K=EYTnw43MI>*h>3ryPb$eV`ZstNTkqcLE7d|_A=bv-) zr#{b5U%Wf_DH)n3e@?}9AV2fh{JA&tvzPCFd^vyXWPbMZ!h3H7ssD59+@0%R-u?5b z!rPzTxp_W+=B>hIvFFb9v-4-qESx;EFn5D;Wz;z-JgTYHnIn;>UM1O+cbyCgB2eA9buxn^fgx!?JhVwC>1vCRQmp*ebB zu(vNx?yA7QG7WjCf%=Bagy9itUVhjf&@4^|dfKz5e0&qFvTv$LwlNr;MJkU|sm&3a zX3t=PdixI1sLNU{<)$OvC&D-BKeSeH8bCx;N4h^kZ!74&n(i^WQ{klRqx2@zozcC1 z^EI%)(L)4Zr>ocuti@w=mD~+jL+^s=I(!554sjptj?mhJo4-J0%MH{;a2?Srv=Zd< zgu`XEyVK)BG+0kSNg-$Vw;YJ;QHRHAZR5*eCZ511@ob_U9qz+sY@yu^U_y_r^y{Qw zlTcH8j*iLPYZHsUN$H=!8^Anf2;`8y8&X9#9o0C22)u!w(>VKP9nHU>?e>fR8e9Khdd<`_%8AN9w{kOMyQp*_e*LW zE(MRT*2_h;eo!?3OO=8rCcGg+oZ*Lfh0G{2H%*))bYCkRaTU$~z=JpNPy})E2IXQ) zMMZ|tLZ76SVtyh;L!@ZA-KR$Q-)JMh<#NFRQZ;73G=q`*xaq31LKieSRo ZyMa8Y68%xfgU67eZ&`YGJ*q(m{s$S_WBLF9 literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/tool/SwaggerController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/tool/SwaggerController.class new file mode 100644 index 0000000000000000000000000000000000000000..57181b864c742661403cca81ca93b0aced56f144 GIT binary patch literal 899 zcmb7COK(y^6#jKnqt_v8ZJi|z4s2z%mwjR=|&S5 z{s4cJ@mwG+P1(4a^Zd^F&Ybh-@2}qg4)CCa9JY(NRYDfG3%FCjU4i^F9c%wwV58CO z31pAVfCMTX9n;$+>XUt^`XPI29TTXqr>suxr8DbC+6nA;0uxD_n3dU~y-kLBZ zE4?wH>`v8iNcQN|{Zt^Q;{kmU*lIL8AJs(3P{l*p^;XA24sV|%vDXpxwA0-0FpiB^ zUYnTPwL8WRr5mTIAFPUKYV0TJC`hdKGpT}roRecqW8=I$wse^Ik+J$KrF|zVOn9E1 z7tXat${mx9bYnN=*>($_?TM!8Zd2f3ZKHI;KC_qVl$3l;{*4-s7u-u-lh}Y>>Ga>~ z7lv=8;|e@lTikiFWc+MQ^4xblS)09%$_af=$gR-XTr6QZTp4*31gdgrKTdRO3-24aIK=p(G literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/tool/TestController.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/tool/TestController.class new file mode 100644 index 0000000000000000000000000000000000000000..e1ba511e0908e5ca9cccd5282940304a91d53d05 GIT binary patch literal 4611 zcmb_f?|T&06@KsTCbL-VX`|ULw0A@nc2kD zT1&MQwJ2>Z3Q`Mdm42wu*2e^R+J5)%APM~^`t-ebW_M>pz}0p??A$r`-gDmbo^#JV z^VNSY{}aGN_$-CBD7Ih?>=Yc7G`yIEn}nBypF#rVC78fu5-%k&rQs*??d25G_-P8S z;AisnRr&XG`FA9VUr6z5DZGwf%H2nk_?46#lhbc#cr%6DQPl93hF=Rbl|9q-1eRqF z8xw|J_N{{cj8W)9udYoTBcQoheJ7G)MHD zWBaaCD44G9J5E90OEWud-}0xrw^4a>k7Zl_Hi3A1$FM-6&&kuwvaDsB1Lfj~>FzN` z3Y28BPR=L{8?GhaLt(-nvj}!0l`5D*N)&%`ZBImf1AH9(T0H-g0ZIAl!7}+Tn!}c7k;`7& zc6`IP9Gg^b$Xbr>O&X(&q;Aw^^xl%St7N)r9d$HK7=^M7%eA+sXWuzF|L*y@W7D(8 z-n(}0yud?Q#~szZl55$chg_p*PCD+mN`?{3&PUYhPnbS)ti*Lvr(RGY(MC;QW{JSV z?Th4B1MzB>@$JtGEMH{5z|tY#$c@u*m`298G4l=MT1V8i3!lzkIV#X`6Sn({rGk~S z{M`l_Bs&S4L^5>)s{40v=$bE-YhVU1^U0; zF~5#gSwl0k$Ikxy)F**+SO0Zx?#Km!Ee%t*LqV#QzGP3X5Z;5SA*bx-%*R!1me*fh z8zmNjH8)8oi7kq1{_}J5Cs=B>HjHg>;cfX%wkqOlex-Du+H{$`X{^WHX{^I{G#pRk zZM-AU-e{}9%fZMOEt?6go|_(cP+-ZXhqqRKQu_%Fze(e__?dF~elNG4)bL&!@8eV&r}04=XYhwKKExl{iG*a%9H0Hm`?F`xr*RhNG<=lC z$8yal_>;h`!J7UZS7(l2d+i@rXD-kG`La~knZ~D5$*P)dHa6pC{z=0dBTEAJ-JF5? z@C&drsq|LSRwdG7Z8to#*6Rqg)zY_GOf+?Q?>fXTroF!-QZ_0roGR1TZS0P$3=2#n z64}-*tr4Kyj+e^3pSDI1V9UlHSuMWh)j_%2jc; zJp%VGHXEV2%0S|fVHFlY45QKj#P!$tFIn>RK*L@+CdrmJP%h9&OW2XgmTE>9(F$Oj9IkRX9O1GViMFQI^=w9x_(xP#N#X;w1UN{W0!B5X{V=1u z2s?CZt@I$w=(b=iU6H|GkkUa4j1m4}9_(?ZfD?h;RfXMEpVk|~3>aCM*0~1KI$!l_ zL0TtaR?FAjDye}bS8)5jY25L5#AG4j7c~Ey`2?57aEQ+sMiq=Z81^x#+k-`SKzAzi G75)oI%?pVD literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/com/ruoyi/web/controller/tool/UserEntity.class b/ruoyi-admin/target/classes/com/ruoyi/web/controller/tool/UserEntity.class new file mode 100644 index 0000000000000000000000000000000000000000..dec262a7faa7007f95924affabce8cc029b923ad GIT binary patch literal 1751 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 F{|DZo7oPwC literal 0 HcmV?d00001 diff --git a/ruoyi-admin/target/classes/ehcache.xml b/ruoyi-admin/target/classes/ehcache.xml new file mode 100644 index 00000000..c2bad019 --- /dev/null +++ b/ruoyi-admin/target/classes/ehcache.xml @@ -0,0 +1,108 @@ + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/ruoyi-admin/target/classes/i18n/messages.properties b/ruoyi-admin/target/classes/i18n/messages.properties new file mode 100644 index 00000000..b0bc5ed0 --- /dev/null +++ b/ruoyi-admin/target/classes/i18n/messages.properties @@ -0,0 +1,37 @@ +#错误消息 +not.null=* 必须填写 +user.jcaptcha.error=验证码错误 +user.jcaptcha.expire=验证码已失效 +user.not.exists=用户不存在/密码错误 +user.password.not.match=用户不存在/密码错误 +user.password.retry.limit.count=密码输入错误{0}次 +user.password.retry.limit.exceed=密码输入错误{0}次,帐户锁定{1}分钟 +user.password.delete=对不起,您的账号已被删除 +user.blocked=用户已封禁,请联系管理员 +role.blocked=角色已封禁,请联系管理员 +user.logout.success=退出成功 + +length.not.valid=长度必须在{min}到{max}个字符之间 + +user.username.not.valid=* 2到20个汉字、字母、数字或下划线组成,且必须以非数字开头 +user.password.not.valid=* 5-50个字符 + +user.email.not.valid=邮箱格式错误 +user.mobile.phone.number.not.valid=手机号格式错误 +user.login.success=登录成功 +user.register.success=注册成功 +user.notfound=请重新登录 +user.forcelogout=管理员强制退出,请重新登录 +user.unknown.error=未知错误,请重新登录 + +##文件上传消息 +upload.exceed.maxSize=上传的文件大小超出限制的文件大小!
允许的文件最大大小是:{0}MB! +upload.filename.exceed.length=上传的文件名最长{0}个字符 + +##权限 +no.permission=您没有数据的权限,请联系管理员添加权限 [{0}] +no.create.permission=您没有创建数据的权限,请联系管理员添加权限 [{0}] +no.update.permission=您没有修改数据的权限,请联系管理员添加权限 [{0}] +no.delete.permission=您没有删除数据的权限,请联系管理员添加权限 [{0}] +no.export.permission=您没有导出数据的权限,请联系管理员添加权限 [{0}] +no.view.permission=您没有查看数据的权限,请联系管理员添加权限 [{0}] diff --git a/ruoyi-admin/target/classes/logback.xml b/ruoyi-admin/target/classes/logback.xml new file mode 100644 index 00000000..9a26a231 --- /dev/null +++ b/ruoyi-admin/target/classes/logback.xml @@ -0,0 +1,93 @@ + + + + + + + + + + + ${log.pattern} + + + + + + ${log.path}/sys-info.log + + + + ${log.path}/sys-info.%d{yyyy-MM-dd}.log + + 60 + + + ${log.pattern} + + + + INFO + + ACCEPT + + DENY + + + + + ${log.path}/sys-error.log + + + + ${log.path}/sys-error.%d{yyyy-MM-dd}.log + + 60 + + + ${log.pattern} + + + + ERROR + + ACCEPT + + DENY + + + + + + ${log.path}/sys-user.log + + + ${log.path}/sys-user.%d{yyyy-MM-dd}.log + + 60 + + + ${log.pattern} + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/ruoyi-admin/target/classes/mybatis/mybatis-config.xml b/ruoyi-admin/target/classes/mybatis/mybatis-config.xml new file mode 100644 index 00000000..ac47c038 --- /dev/null +++ b/ruoyi-admin/target/classes/mybatis/mybatis-config.xml @@ -0,0 +1,20 @@ + + + + + + + + + + + + + + + + + + diff --git a/ruoyi-admin/target/classes/templatefile/消防安全重点部位情况登记表.xlsx b/ruoyi-admin/target/classes/templatefile/消防安全重点部位情况登记表.xlsx new file mode 100644 index 0000000000000000000000000000000000000000..8e58d705ea3abea9fa6799b5fdb73617764b7d7b GIT binary patch literal 9408 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 + + + ruoyi + com.ruoyi + 3.8.4 + + 4.0.0 + + ruoyi-common + + + common通用工具 + + + + + com.alibaba + easyexcel + + + + org.springframework + spring-context-support + + + + + org.springframework + spring-web + + + + + org.springframework.boot + spring-boot-starter-security + + + + + com.github.pagehelper + pagehelper-spring-boot-starter + + + org.mybatis + mybatis + + + org.mybatis + mybatis-spring + + + com.github.jsqlparser + jsqlparser + + + + + + + org.springframework.boot + spring-boot-starter-validation + + + + + org.apache.commons + commons-lang3 + + + + + com.fasterxml.jackson.core + jackson-databind + + + + + com.alibaba.fastjson2 + fastjson2 + + + + + commons-io + commons-io + + + + + commons-fileupload + commons-fileupload + + + + + org.apache.poi + poi-ooxml + + + + + org.yaml + snakeyaml + + + + + io.jsonwebtoken + jjwt + + + + + javax.xml.bind + jaxb-api + + + + net.sf.ehcache + ehcache + 2.10.6 + + + org.springframework.boot + spring-boot-starter-cache + + + + + + + + + + + org.apache.commons + commons-pool2 + + + + + eu.bitwalker + UserAgentUtils + + + + + javax.servlet + javax.servlet-api + + + + + cn.hutool + hutool-all + + + + + com.baomidou + mybatis-plus-boot-starter + + + + + org.projectlombok + lombok + true + + + + + com.github.xiaoymin + knife4j-spring-boot-starter + + + + + com.alibaba + druid-spring-boot-starter + + + + org.mapstruct + mapstruct + + + org.mapstruct + mapstruct-jdk8 + + + org.mapstruct + mapstruct-processor + + + + com.alibaba + transmittable-thread-local + + + + + diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/annotation/Anonymous.java b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/Anonymous.java new file mode 100644 index 00000000..1d6d4f44 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/Anonymous.java @@ -0,0 +1,19 @@ +package com.ruoyi.common.annotation; + +import java.lang.annotation.Documented; +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +/** + * 匿名访问不鉴权注解 + * + * @author ruoyi + */ +@Target({ ElementType.METHOD, ElementType.TYPE }) +@Retention(RetentionPolicy.RUNTIME) +@Documented +public @interface Anonymous +{ +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/annotation/DataScope.java b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/DataScope.java new file mode 100644 index 00000000..aae7f72c --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/DataScope.java @@ -0,0 +1,33 @@ +package com.ruoyi.common.annotation; + +import java.lang.annotation.Documented; +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +/** + * 数据权限过滤注解 + * + * @author ruoyi + */ +@Target(ElementType.METHOD) +@Retention(RetentionPolicy.RUNTIME) +@Documented +public @interface DataScope +{ + /** + * 部门表的别名 + */ + public String deptAlias() default ""; + + /** + * 用户表的别名 + */ + public String userAlias() default ""; + + /** + * 权限字符(用于多个角色匹配符合要求的权限)默认根据权限注解@ss获取,多个权限用逗号分隔开来 + */ + public String permission() default ""; +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/annotation/DataSource.java b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/DataSource.java new file mode 100644 index 00000000..79cd191f --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/DataSource.java @@ -0,0 +1,28 @@ +package com.ruoyi.common.annotation; + +import java.lang.annotation.Documented; +import java.lang.annotation.ElementType; +import java.lang.annotation.Inherited; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; +import com.ruoyi.common.enums.DataSourceType; + +/** + * 自定义多数据源切换注解 + * + * 优先级:先方法,后类,如果方法覆盖了类上的数据源类型,以方法的为准,否则以类上的为准 + * + * @author ruoyi + */ +@Target({ ElementType.METHOD, ElementType.TYPE }) +@Retention(RetentionPolicy.RUNTIME) +@Documented +@Inherited +public @interface DataSource +{ + /** + * 切换数据源名称 + */ + public DataSourceType value() default DataSourceType.MASTER; +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/annotation/DictFormat.java b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/DictFormat.java new file mode 100644 index 00000000..5ff76a07 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/DictFormat.java @@ -0,0 +1,22 @@ +package com.ruoyi.common.annotation; + +import java.lang.annotation.*; + +/** + * 字典格式化 + * + * 实现将字典数据的值,格式化成字典数据的标签 + */ +@Target({ElementType.FIELD}) +@Retention(RetentionPolicy.RUNTIME) +@Inherited +public @interface DictFormat { + + /** + * 例如说,SysDictTypeConstants、InfDictTypeConstants + * + * @return 字典类型 + */ + String value(); + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/annotation/Excel.java b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/Excel.java new file mode 100644 index 00000000..a8fa282b --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/Excel.java @@ -0,0 +1,187 @@ +package com.ruoyi.common.annotation; + +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; +import java.math.BigDecimal; +import org.apache.poi.ss.usermodel.HorizontalAlignment; +import org.apache.poi.ss.usermodel.IndexedColors; +import com.ruoyi.common.utils.poi.ExcelHandlerAdapter; + +/** + * 自定义导出Excel数据注解 + * + * @author ruoyi + */ +@Retention(RetentionPolicy.RUNTIME) +@Target(ElementType.FIELD) +public @interface Excel +{ + /** + * 导出时在excel中排序 + */ + public int sort() default Integer.MAX_VALUE; + + /** + * 导出到Excel中的名字. + */ + public String name() default ""; + + /** + * 日期格式, 如: yyyy-MM-dd + */ + public String dateFormat() default ""; + + /** + * 如果是字典类型,请设置字典的type值 (如: sys_user_sex) + */ + public String dictType() default ""; + + /** + * 读取内容转表达式 (如: 0=男,1=女,2=未知) + */ + public String readConverterExp() default ""; + + /** + * 分隔符,读取字符串组内容 + */ + public String separator() default ","; + + /** + * BigDecimal 精度 默认:-1(默认不开启BigDecimal格式化) + */ + public int scale() default -1; + + /** + * BigDecimal 舍入规则 默认:BigDecimal.ROUND_HALF_EVEN + */ + public int roundingMode() default BigDecimal.ROUND_HALF_EVEN; + + /** + * 导出时在excel中每个列的高度 单位为字符 + */ + public double height() default 14; + + /** + * 导出时在excel中每个列的宽 单位为字符 + */ + public double width() default 16; + + /** + * 文字后缀,如% 90 变成90% + */ + public String suffix() default ""; + + /** + * 当值为空时,字段的默认值 + */ + public String defaultValue() default ""; + + /** + * 提示信息 + */ + public String prompt() default ""; + + /** + * 设置只能选择不能输入的列内容. + */ + public String[] combo() default {}; + + /** + * 是否需要纵向合并单元格,应对需求:含有list集合单元格) + */ + public boolean needMerge() default false; + + /** + * 是否导出数据,应对需求:有时我们需要导出一份模板,这是标题需要但内容需要用户手工填写. + */ + public boolean isExport() default true; + + /** + * 另一个类中的属性名称,支持多级获取,以小数点隔开 + */ + public String targetAttr() default ""; + + /** + * 是否自动统计数据,在最后追加一行统计数据总和 + */ + public boolean isStatistics() default false; + + /** + * 导出类型(0数字 1字符串 2图片) + */ + public ColumnType cellType() default ColumnType.STRING; + + /** + * 导出列头背景色 + */ + public IndexedColors headerBackgroundColor() default IndexedColors.GREY_50_PERCENT; + + /** + * 导出列头字体颜色 + */ + public IndexedColors headerColor() default IndexedColors.WHITE; + + /** + * 导出单元格背景色 + */ + public IndexedColors backgroundColor() default IndexedColors.WHITE; + + /** + * 导出单元格字体颜色 + */ + public IndexedColors color() default IndexedColors.BLACK; + + /** + * 导出字段对齐方式 + */ + public HorizontalAlignment align() default HorizontalAlignment.CENTER; + + /** + * 自定义数据处理器 + */ + public Class handler() default ExcelHandlerAdapter.class; + + /** + * 自定义数据处理器参数 + */ + public String[] args() default {}; + + /** + * 字段类型(0:导出导入;1:仅导出;2:仅导入) + */ + Type type() default Type.ALL; + + public enum Type + { + ALL(0), EXPORT(1), IMPORT(2); + private final int value; + + Type(int value) + { + this.value = value; + } + + public int value() + { + return this.value; + } + } + + public enum ColumnType + { + NUMERIC(0), STRING(1), IMAGE(2); + private final int value; + + ColumnType(int value) + { + this.value = value; + } + + public int value() + { + return this.value; + } + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/annotation/Excels.java b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/Excels.java new file mode 100644 index 00000000..1f1cc81b --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/Excels.java @@ -0,0 +1,18 @@ +package com.ruoyi.common.annotation; + +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +/** + * Excel注解集 + * + * @author ruoyi + */ +@Target(ElementType.FIELD) +@Retention(RetentionPolicy.RUNTIME) +public @interface Excels +{ + public Excel[] value(); +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/annotation/Log.java b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/Log.java new file mode 100644 index 00000000..ca02c6c4 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/Log.java @@ -0,0 +1,46 @@ +package com.ruoyi.common.annotation; + +import java.lang.annotation.Documented; +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.enums.OperatorType; + +/** + * 自定义操作日志记录注解 + * + * @author ruoyi + * + */ +@Target({ ElementType.PARAMETER, ElementType.METHOD }) +@Retention(RetentionPolicy.RUNTIME) +@Documented +public @interface Log +{ + /** + * 模块 + */ + public String title() default ""; + + /** + * 功能 + */ + public BusinessType businessType() default BusinessType.OTHER; + + /** + * 操作人类别 + */ + public OperatorType operatorType() default OperatorType.MANAGE; + + /** + * 是否保存请求的参数 + */ + public boolean isSaveRequestData() default true; + + /** + * 是否保存响应的参数 + */ + public boolean isSaveResponseData() default true; +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/annotation/RateLimiter.java b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/RateLimiter.java new file mode 100644 index 00000000..0f024c7d --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/RateLimiter.java @@ -0,0 +1,40 @@ +package com.ruoyi.common.annotation; + +import java.lang.annotation.Documented; +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; +import com.ruoyi.common.constant.CacheConstants; +import com.ruoyi.common.enums.LimitType; + +/** + * 限流注解 + * + * @author ruoyi + */ +@Target(ElementType.METHOD) +@Retention(RetentionPolicy.RUNTIME) +@Documented +public @interface RateLimiter +{ + /** + * 限流key + */ + public String key() default CacheConstants.RATE_LIMIT_KEY; + + /** + * 限流时间,单位秒 + */ + public int time() default 60; + + /** + * 限流次数 + */ + public int count() default 100; + + /** + * 限流类型 + */ + public LimitType limitType() default LimitType.DEFAULT; +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/annotation/RepeatSubmit.java b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/RepeatSubmit.java new file mode 100644 index 00000000..b7697481 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/RepeatSubmit.java @@ -0,0 +1,31 @@ +package com.ruoyi.common.annotation; + +import java.lang.annotation.Documented; +import java.lang.annotation.ElementType; +import java.lang.annotation.Inherited; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +/** + * 自定义注解防止表单重复提交 + * + * @author ruoyi + * + */ +@Inherited +@Target(ElementType.METHOD) +@Retention(RetentionPolicy.RUNTIME) +@Documented +public @interface RepeatSubmit +{ + /** + * 间隔时间(ms),小于此时间视为重复提交 + */ + public int interval() default 5000; + + /** + * 提示消息 + */ + public String message() default "不允许重复提交,请稍候再试"; +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/annotation/ReportExcel.java b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/ReportExcel.java new file mode 100644 index 00000000..f1e13972 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/annotation/ReportExcel.java @@ -0,0 +1,26 @@ +package com.ruoyi.common.annotation; + +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +/** + * 自定义easy excel报表下载表单时对数据进行格式转换 + * + * @author ruoyi + */ +@Retention(RetentionPolicy.RUNTIME) +@Target(ElementType.FIELD) +public @interface ReportExcel +{ + + /** + * 如果是字典类型,请设置字典的type值 (如: sys_user_sex) + */ + public String dictType() default ""; + /** + * 如果是字典类型,值的转方式。0-为展示所有字典值并且带□和☑ + */ + public String dictTransferType() default "0"; +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/config/RuoYiConfig.java b/ruoyi-common/src/main/java/com/ruoyi/common/config/RuoYiConfig.java new file mode 100644 index 00000000..9f69c358 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/config/RuoYiConfig.java @@ -0,0 +1,150 @@ +package com.ruoyi.common.config; + +import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.stereotype.Component; + +/** + * 读取项目相关配置 + * + * @author ruoyi + */ +@Component +@ConfigurationProperties(prefix = "ruoyi") +public class RuoYiConfig +{ + /** 项目名称 */ + private String name; + + /** 版本 */ + private String version; + + /** 版权年份 */ + private String copyrightYear; + + /** 实例演示开关 */ + private boolean demoEnabled; + + /** 上传路径 */ + private static String profile; + + /** 获取地址开关 */ + private static boolean addressEnabled; + + /** 验证码类型 */ + private static String captchaType; + + public String getName() + { + return name; + } + + public void setName(String name) + { + this.name = name; + } + + public String getVersion() + { + return version; + } + + public void setVersion(String version) + { + this.version = version; + } + + public String getCopyrightYear() + { + return copyrightYear; + } + + public void setCopyrightYear(String copyrightYear) + { + this.copyrightYear = copyrightYear; + } + + public boolean isDemoEnabled() + { + return demoEnabled; + } + + public void setDemoEnabled(boolean demoEnabled) + { + this.demoEnabled = demoEnabled; + } + + public static String getProfile() + { + return profile; + } + + public void setProfile(String profile) + { + RuoYiConfig.profile = profile; + } + + public static boolean isAddressEnabled() + { + return addressEnabled; + } + + public void setAddressEnabled(boolean addressEnabled) + { + RuoYiConfig.addressEnabled = addressEnabled; + } + + public static String getCaptchaType() { + return captchaType; + } + + public void setCaptchaType(String captchaType) { + RuoYiConfig.captchaType = captchaType; + } + + /** + * 获取导入上传路径 + */ + public static String getImportPath() + { + return getProfile() + "/import"; + } + + /** + * 获取头像上传路径 + */ + public static String getAvatarPath() + { + return getProfile() + "/avatar"; + } + + /** + * 获取下载路径 + */ + public static String getDownloadPath() + { + return getProfile() + "/download/"; + } + /** + * 获取临时下载路径,改文件夹下的文件会被定时删除 + */ + public static String getTempDownloadPath() + { + return getProfile() + "/download/temp/"; + } + /** + * 获取上传路径 + */ + public static String getUploadPath() + { + return getProfile() + "/upload"; + } + + /** + * 获取文件模板路径 + */ + public static String getTemplateFilePath() + { + return getProfile() + "/template"; + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/constant/CacheConstants.java b/ruoyi-common/src/main/java/com/ruoyi/common/constant/CacheConstants.java new file mode 100644 index 00000000..296733b9 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/constant/CacheConstants.java @@ -0,0 +1,54 @@ +package com.ruoyi.common.constant; + +/** + * 缓存的key 常量 + * + * @author ruoyi + */ +public class CacheConstants +{ + /** + * 登录用户 redis key + */ + public static final String LOGIN_TOKEN_KEY = "login_tokens:"; + + /** + * 验证码 redis key + */ + public static final String CAPTCHA_CODE_KEY = "captcha_codes:"; + + /** + * 参数管理 cache key + */ + public static final String SYS_CONFIG_KEY = "sys_config:"; + + /** + * 字典管理 cache key + */ + public static final String SYS_DICT_KEY = "sys_dict:"; + + /** + * 分类管理-list cache key + */ + public static final String TREE_DICT_KEY = "sys_tree_dict:"; + + /** + * 防重提交 redis key + */ + public static final String REPEAT_SUBMIT_KEY = "repeat_submit:"; + + /** + * 限流 redis key + */ + public static final String RATE_LIMIT_KEY = "rate_limit:"; + + /** + * 登录账户密码错误次数 redis key + */ + public static final String PWD_ERR_CNT_KEY = "pwd_err_cnt:"; + + /** + * 租户管理 cache key + */ + public static final String SYS_TENANT_KEY = "sys_tenant:"; +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/constant/Constants.java b/ruoyi-common/src/main/java/com/ruoyi/common/constant/Constants.java new file mode 100644 index 00000000..ed687257 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/constant/Constants.java @@ -0,0 +1,152 @@ +package com.ruoyi.common.constant; + +import io.jsonwebtoken.Claims; + +/** + * 通用常量信息 + * + * @author ruoyi + */ +public class Constants +{ + /** + * UTF-8 字符集 + */ + public static final String UTF8 = "UTF-8"; + + /** + * GBK 字符集 + */ + public static final String GBK = "GBK"; + + /** + * www主域 + */ + public static final String WWW = "www."; + + /** + * http请求 + */ + public static final String HTTP = "http://"; + + /** + * https请求 + */ + public static final String HTTPS = "https://"; + + /** + * 通用成功标识 + */ + public static final String SUCCESS = "0"; + + /** + * 通用失败标识 + */ + public static final String FAIL = "1"; + + /** + * 通用true标识 + */ + public static final String TRUE = "1"; + + /** + * 通用false标识 + */ + public static final String FALSE = "0"; + + /** + * 登录成功 + */ + public static final String LOGIN_SUCCESS = "Success"; + + /** + * 注销 + */ + public static final String LOGOUT = "Logout"; + + /** + * 注册 + */ + public static final String REGISTER = "Register"; + + /** + * 登录失败 + */ + public static final String LOGIN_FAIL = "Error"; + + /** + * 验证码有效期(分钟) + */ + public static final Integer CAPTCHA_EXPIRATION = 2; + + /** + * 令牌 + */ + public static final String TOKEN = "token"; + + /** + * 令牌前缀 + */ + public static final String TOKEN_PREFIX = "Bearer "; + + /** + * 令牌前缀 + */ + public static final String LOGIN_USER_KEY = "login_user_key"; + + /** + * 用户ID + */ + public static final String JWT_USERID = "userid"; + + /** + * 用户名称 + */ + public static final String JWT_USERNAME = Claims.SUBJECT; + + /** + * 用户头像 + */ + public static final String JWT_AVATAR = "avatar"; + + /** + * 创建时间 + */ + public static final String JWT_CREATED = "created"; + + /** + * 用户权限 + */ + public static final String JWT_AUTHORITIES = "authorities"; + + /** + * 资源映射路径 前缀 + */ + public static final String RESOURCE_PREFIX = "/profile"; + + /** + * RMI 远程方法调用 + */ + public static final String LOOKUP_RMI = "rmi:"; + + /** + * LDAP 远程方法调用 + */ + public static final String LOOKUP_LDAP = "ldap:"; + + /** + * LDAPS 远程方法调用 + */ + public static final String LOOKUP_LDAPS = "ldaps:"; + + /** + * 定时任务白名单配置(仅允许访问的包名,如其他需要可以自行添加) + */ + public static final String[] JOB_WHITELIST_STR = { "com.ruoyi" }; + + /** + * 定时任务违规的字符 + */ + public static final String[] JOB_ERROR_STR = { "java.net.URL", "javax.naming.InitialContext", "org.yaml.snakeyaml", + "org.springframework", "org.apache", "com.ruoyi.common.utils.file" }; +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/constant/ErrorCodeConstants.java b/ruoyi-common/src/main/java/com/ruoyi/common/constant/ErrorCodeConstants.java new file mode 100644 index 00000000..66485bea --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/constant/ErrorCodeConstants.java @@ -0,0 +1,145 @@ +package com.ruoyi.common.constant; + + +import com.ruoyi.common.exception.ErrorCode; + +/** + * Member 错误码枚举类 + *

+ * member 系统,使用 1-004-000-000 段 + */ +public interface ErrorCodeConstants { + + // ========== AUTH 模块 1002000000 ========== + ErrorCode AUTH_LOGIN_BAD_CREDENTIALS = new ErrorCode(1002000000, "登录失败,账号密码不正确"); + ErrorCode AUTH_LOGIN_USER_DISABLED = new ErrorCode(1002000001, "登录失败,账号被禁用"); + ErrorCode AUTH_LOGIN_CAPTCHA_CODE_ERROR = new ErrorCode(1002000004, "验证码不正确,原因:{}"); + ErrorCode AUTH_THIRD_LOGIN_NOT_BIND = new ErrorCode(1002000005, "未绑定账号,需要进行绑定"); + ErrorCode AUTH_TOKEN_EXPIRED = new ErrorCode(1002000006, "Token 已经过期"); + ErrorCode AUTH_MOBILE_NOT_EXISTS = new ErrorCode(1002000007, "手机号不存在"); + + // ========== 菜单模块 1002001000 ========== + ErrorCode MENU_NAME_DUPLICATE = new ErrorCode(1002001000, "已经存在该名字的菜单"); + ErrorCode MENU_PARENT_NOT_EXISTS = new ErrorCode(1002001001, "父菜单不存在"); + ErrorCode MENU_PARENT_ERROR = new ErrorCode(1002001002, "不能设置自己为父菜单"); + ErrorCode MENU_NOT_EXISTS = new ErrorCode(1002001003, "菜单不存在"); + ErrorCode MENU_EXISTS_CHILDREN = new ErrorCode(1002001004, "存在子菜单,无法删除"); + ErrorCode MENU_PARENT_NOT_DIR_OR_MENU = new ErrorCode(1002001005, "父菜单的类型必须是目录或者菜单"); + + // ========== 角色模块 1002002000 ========== + ErrorCode ROLE_NOT_EXISTS = new ErrorCode(1002002000, "角色不存在"); + ErrorCode ROLE_NAME_DUPLICATE = new ErrorCode(1002002001, "已经存在名为【{}】的角色"); + ErrorCode ROLE_CODE_DUPLICATE = new ErrorCode(1002002002, "已经存在编码为【{}】的角色"); + ErrorCode ROLE_CAN_NOT_UPDATE_SYSTEM_TYPE_ROLE = new ErrorCode(1002002003, "不能操作类型为系统内置的角色"); + ErrorCode ROLE_IS_DISABLE = new ErrorCode(1002002004, "名字为【{}】的角色已被禁用"); + ErrorCode ROLE_ADMIN_CODE_ERROR = new ErrorCode(1002002005, "编码【{}】不能使用"); + + // ========== 用户模块 1002003000 ========== + ErrorCode USER_USERNAME_EXISTS = new ErrorCode(1002003000, "用户账号已经存在"); + ErrorCode USER_MOBILE_EXISTS = new ErrorCode(1002003001, "手机号已经存在"); + ErrorCode USER_EMAIL_EXISTS = new ErrorCode(1002003002, "邮箱已经存在"); + ErrorCode USER_NOT_EXISTS = new ErrorCode(1002003003, "用户不存在"); + ErrorCode USER_IMPORT_LIST_IS_EMPTY = new ErrorCode(1002003004, "导入用户数据不能为空!"); + ErrorCode USER_PASSWORD_FAILED = new ErrorCode(1002003005, "用户密码校验失败"); + ErrorCode USER_IS_DISABLE = new ErrorCode(1002003006, "名字为【{}】的用户已被禁用"); + ErrorCode USER_COUNT_MAX = new ErrorCode(1002003008, "创建用户失败,原因:超过租户最大租户配额({})!"); + + // ========== 部门模块 1002004000 ========== + ErrorCode DEPT_NAME_DUPLICATE = new ErrorCode(1002004000, "已经存在该名字的部门"); + ErrorCode DEPT_PARENT_NOT_EXITS = new ErrorCode(1002004001,"父级部门不存在"); + ErrorCode DEPT_NOT_FOUND = new ErrorCode(1002004002, "当前部门不存在"); + ErrorCode DEPT_EXITS_CHILDREN = new ErrorCode(1002004003, "存在子部门,无法删除"); + ErrorCode DEPT_PARENT_ERROR = new ErrorCode(1002004004, "不能设置自己为父部门"); + ErrorCode DEPT_EXISTS_USER = new ErrorCode(1002004005, "部门中存在员工,无法删除"); + ErrorCode DEPT_NOT_ENABLE = new ErrorCode(1002004006, "部门不处于开启状态,不允许选择"); + ErrorCode DEPT_PARENT_IS_CHILD = new ErrorCode(1002004007, "不能设置自己的子部门为父部门"); + + // ========== 岗位模块 1002005000 ========== + ErrorCode POST_NOT_FOUND = new ErrorCode(1002005000, "当前岗位不存在"); + ErrorCode POST_NOT_ENABLE = new ErrorCode(1002005001, "岗位({}) 不处于开启状态,不允许选择"); + ErrorCode POST_NAME_DUPLICATE = new ErrorCode(1002005002, "已经存在该名字的岗位"); + ErrorCode POST_CODE_DUPLICATE = new ErrorCode(1002005003, "已经存在该标识的岗位"); + + // ========== 字典类型 1002006000 ========== + ErrorCode DICT_TYPE_NOT_EXISTS = new ErrorCode(1002006001, "当前字典类型不存在"); + ErrorCode DICT_TYPE_NOT_ENABLE = new ErrorCode(1002006002, "字典类型不处于开启状态,不允许选择"); + ErrorCode DICT_TYPE_NAME_DUPLICATE = new ErrorCode(1002006003, "已经存在该名字的字典类型"); + ErrorCode DICT_TYPE_TYPE_DUPLICATE = new ErrorCode(1002006004, "已经存在该类型的字典类型"); + ErrorCode DICT_TYPE_HAS_CHILDREN = new ErrorCode(1002006005, "无法删除,该字典类型还有字典数据"); + + // ========== 字典数据 1002007000 ========== + ErrorCode DICT_DATA_NOT_EXISTS = new ErrorCode(1002007001, "当前字典数据不存在"); + ErrorCode DICT_DATA_NOT_ENABLE = new ErrorCode(1002007002, "字典数据({})不处于开启状态,不允许选择"); + ErrorCode DICT_DATA_VALUE_DUPLICATE = new ErrorCode(1002007003, "已经存在该值的字典数据"); + + // ========== 通知公告 1002008000 ========== + ErrorCode NOTICE_NOT_FOUND = new ErrorCode(1002008001, "当前通知公告不存在"); + + // ========== 短信渠道 1002011000 ========== + ErrorCode SMS_CHANNEL_NOT_EXISTS = new ErrorCode(1002011000, "短信渠道不存在"); + ErrorCode SMS_CHANNEL_DISABLE = new ErrorCode(1002011001, "短信渠道不处于开启状态,不允许选择"); + ErrorCode SMS_CHANNEL_HAS_CHILDREN = new ErrorCode(1002011002, "无法删除,该短信渠道还有短信模板"); + + // ========== 短信模板 1002012000 ========== + ErrorCode SMS_TEMPLATE_NOT_EXISTS = new ErrorCode(1002012000, "短信模板不存在"); + ErrorCode SMS_TEMPLATE_CODE_DUPLICATE = new ErrorCode(1002012001, "已经存在编码为【{}】的短信模板"); + + // ========== 短信发送 1002013000 ========== + ErrorCode SMS_SEND_MOBILE_NOT_EXISTS = new ErrorCode(1002013000, "手机号不存在"); + ErrorCode SMS_SEND_MOBILE_TEMPLATE_PARAM_MISS = new ErrorCode(1002013001, "模板参数({})缺失"); + ErrorCode SMS_SEND_TEMPLATE_NOT_EXISTS = new ErrorCode(1002013002, "短信模板不存在"); + + // ========== 短信验证码 1002014000 ========== + ErrorCode SMS_CODE_NOT_FOUND = new ErrorCode(1002014000, "验证码不存在"); + ErrorCode SMS_CODE_EXPIRED = new ErrorCode(1002014001, "验证码已过期"); + ErrorCode SMS_CODE_USED = new ErrorCode(1002014002, "验证码已使用"); + ErrorCode SMS_CODE_NOT_CORRECT = new ErrorCode(1002014003, "验证码不正确"); + ErrorCode SMS_CODE_EXCEED_SEND_MAXIMUM_QUANTITY_PER_DAY = new ErrorCode(1002014004, "超过每日短信发送数量"); + ErrorCode SMS_CODE_SEND_TOO_FAST = new ErrorCode(1002014005, "短信发送过于频率"); + ErrorCode SMS_CODE_IS_EXISTS = new ErrorCode(1002014006, "手机号已被使用"); + ErrorCode SMS_CODE_IS_UNUSED = new ErrorCode(1002014007, "验证码未被使用"); + + // ========== 租户信息 1002015000 ========== + ErrorCode TENANT_NOT_EXISTS = new ErrorCode(1002015000, "租户不存在"); + ErrorCode TENANT_DISABLE = new ErrorCode(1002015001, "名字为【%s】的租户已被禁用"); + ErrorCode TENANT_EXPIRE = new ErrorCode(1002015002, "名字为【%s】的租户已过期"); + ErrorCode TENANT_CAN_NOT_UPDATE_SYSTEM = new ErrorCode(1002015003, "系统租户不能进行修改、删除等操作!"); + + // ========== 租户套餐 1002016000 ========== + ErrorCode TENANT_PACKAGE_NOT_EXISTS = new ErrorCode(1002016000, "租户套餐不存在"); + ErrorCode TENANT_PACKAGE_USED = new ErrorCode(1002016001, "租户正在使用该套餐,请给租户重新设置套餐后再尝试删除"); + ErrorCode TENANT_PACKAGE_DISABLE = new ErrorCode(1002016002, "名字为【{}】的租户套餐已被禁用"); + + // ========== 错误码模块 1002017000 ========== + ErrorCode ERROR_CODE_NOT_EXISTS = new ErrorCode(1002017000, "错误码不存在"); + ErrorCode ERROR_CODE_DUPLICATE = new ErrorCode(1002017001, "已经存在编码为【{}】的错误码"); + + // ========== 社交用户 1002018000 ========== + ErrorCode SOCIAL_USER_AUTH_FAILURE = new ErrorCode(1002018000, "社交授权失败,原因是:{}"); + ErrorCode SOCIAL_USER_UNBIND_NOT_SELF = new ErrorCode(1002018001, "社交解绑失败,非当前用户绑定"); + ErrorCode SOCIAL_USER_NOT_FOUND = new ErrorCode(1002018002, "社交授权失败,找不到对应的用户"); + + // ========== 系统敏感词 1002019000 ========= + ErrorCode SENSITIVE_WORD_NOT_EXISTS = new ErrorCode(1002019000, "系统敏感词在所有标签中都不存在"); + ErrorCode SENSITIVE_WORD_EXISTS = new ErrorCode(1002019001, "系统敏感词已在标签中存在"); + + // ========== OAuth2 客户端 1002020000 ========= + ErrorCode OAUTH2_CLIENT_NOT_EXISTS = new ErrorCode(1002020000, "OAuth2 客户端不存在"); + ErrorCode OAUTH2_CLIENT_EXISTS = new ErrorCode(1002020001, "OAuth2 客户端编号已存在"); + ErrorCode OAUTH2_CLIENT_DISABLE = new ErrorCode(1002020002, "OAuth2 客户端已禁用"); + ErrorCode OAUTH2_CLIENT_AUTHORIZED_GRANT_TYPE_NOT_EXISTS = new ErrorCode(1002020003, "不支持该授权类型"); + ErrorCode OAUTH2_CLIENT_SCOPE_OVER = new ErrorCode(1002020004, "授权范围过大"); + ErrorCode OAUTH2_CLIENT_REDIRECT_URI_NOT_MATCH = new ErrorCode(1002020005, "无效 redirect_uri: {}"); + ErrorCode OAUTH2_CLIENT_CLIENT_SECRET_ERROR = new ErrorCode(1002020006, "无效 client_secret: {}"); + + // ========== OAuth2 授权 1002021000 ========= + ErrorCode OAUTH2_GRANT_CLIENT_ID_MISMATCH = new ErrorCode(1002021000, "client_id 不匹配"); + ErrorCode OAUTH2_GRANT_REDIRECT_URI_MISMATCH = new ErrorCode(1002021001, "redirect_uri 不匹配"); + ErrorCode OAUTH2_GRANT_STATE_MISMATCH = new ErrorCode(1002021002, "state 不匹配"); + ErrorCode OAUTH2_GRANT_CODE_NOT_EXISTS = new ErrorCode(1002021003, "code 不存在"); + + // ========== OAuth2 授权 1002022000 ========= + ErrorCode OAUTH2_CODE_NOT_EXISTS = new ErrorCode(1002022000, "code 不存在"); + ErrorCode OAUTH2_CODE_EXPIRE = new ErrorCode(1002022000, "code 已过期"); + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/constant/GenConstants.java b/ruoyi-common/src/main/java/com/ruoyi/common/constant/GenConstants.java new file mode 100644 index 00000000..7d899d49 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/constant/GenConstants.java @@ -0,0 +1,117 @@ +package com.ruoyi.common.constant; + +/** + * 代码生成通用常量 + * + * @author ruoyi + */ +public class GenConstants +{ + /** 单表(增删改查) */ + public static final String TPL_CRUD = "crud"; + + /** 树表(增删改查) */ + public static final String TPL_TREE = "tree"; + + /** 主子表(增删改查) */ + public static final String TPL_SUB = "sub"; + + /** 树编码字段 */ + public static final String TREE_CODE = "treeCode"; + + /** 树父编码字段 */ + public static final String TREE_PARENT_CODE = "treeParentCode"; + + /** 树名称字段 */ + public static final String TREE_NAME = "treeName"; + + /** 上级菜单ID字段 */ + public static final String PARENT_MENU_ID = "parentMenuId"; + + /** 上级菜单名称字段 */ + public static final String PARENT_MENU_NAME = "parentMenuName"; + + /** 数据库字符串类型 */ + public static final String[] COLUMNTYPE_STR = { "char", "varchar", "nvarchar", "varchar2" }; + + /** 数据库文本类型 */ + public static final String[] COLUMNTYPE_TEXT = { "tinytext", "text", "mediumtext", "longtext" }; + + /** 数据库时间类型 */ + public static final String[] COLUMNTYPE_TIME = { "datetime", "time", "date", "timestamp" }; + + /** 数据库数字类型 */ + public static final String[] COLUMNTYPE_NUMBER = { "tinyint", "smallint", "mediumint", "int", "number", "integer", + "bit", "bigint", "float", "double", "decimal" }; + + /** 页面不需要编辑字段 */ + public static final String[] COLUMNNAME_NOT_EDIT = { "id", "create_by", "create_time", "del_flag" }; + + /** 页面不需要显示的列表字段 */ + public static final String[] COLUMNNAME_NOT_LIST = { "id", "create_by", "create_time", "del_flag", "update_by", + "update_time" }; + + /** 页面不需要查询字段 */ + public static final String[] COLUMNNAME_NOT_QUERY = { "id", "create_by", "create_time", "del_flag", "update_by", + "update_time", "remark" }; + + /** Entity基类字段 */ + public static final String[] BASE_ENTITY = { "createBy", "createTime", "updateBy", "updateTime", "remark" }; + + /** Tree基类字段 */ + public static final String[] TREE_ENTITY = { "parentName", "parentId", "orderNum", "ancestors", "children" }; + + /** 文本框 */ + public static final String HTML_INPUT = "input"; + + /** 文本域 */ + public static final String HTML_TEXTAREA = "textarea"; + + /** 下拉框 */ + public static final String HTML_SELECT = "select"; + + /** 单选框 */ + public static final String HTML_RADIO = "radio"; + + /** 复选框 */ + public static final String HTML_CHECKBOX = "checkbox"; + + /** 日期控件 */ + public static final String HTML_DATETIME = "datetime"; + + /** 图片上传控件 */ + public static final String HTML_IMAGE_UPLOAD = "imageUpload"; + + /** 文件上传控件 */ + public static final String HTML_FILE_UPLOAD = "fileUpload"; + + /** 富文本控件 */ + public static final String HTML_EDITOR = "editor"; + + /** 字符串类型 */ + public static final String TYPE_STRING = "String"; + + /** 整型 */ + public static final String TYPE_INTEGER = "Integer"; + + /** 长整型 */ + public static final String TYPE_LONG = "Long"; + + /** 浮点型 */ + public static final String TYPE_DOUBLE = "Double"; + + /** 高精度计算类型 */ + public static final String TYPE_BIGDECIMAL = "BigDecimal"; + + /** 时间类型 */ + public static final String TYPE_DATE = "Date"; + + /** 模糊查询 */ + public static final String QUERY_LIKE = "LIKE"; + + /** 相等查询 */ + public static final String QUERY_EQ = "EQ"; + + /** 需要 */ + public static final String REQUIRE = "1"; +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/constant/HttpStatus.java b/ruoyi-common/src/main/java/com/ruoyi/common/constant/HttpStatus.java new file mode 100644 index 00000000..54a24394 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/constant/HttpStatus.java @@ -0,0 +1,93 @@ +package com.ruoyi.common.constant; + +/** + * 返回状态码 + * + * @author ruoyi + */ +public class HttpStatus { + /** + * 操作成功 + */ + public static final int SUCCESS = 200; + + /** + * 对象创建成功 + */ + public static final int CREATED = 201; + + /** + * 请求已经被接受 + */ + public static final int ACCEPTED = 202; + + /** + * 操作已经执行成功,但是没有返回数据 + */ + public static final int NO_CONTENT = 204; + + /** + * 资源已被移除 + */ + public static final int MOVED_PERM = 301; + + /** + * 重定向 + */ + public static final int SEE_OTHER = 303; + + /** + * 资源没有被修改 + */ + public static final int NOT_MODIFIED = 304; + + /** + * 参数列表错误(缺少,格式不匹配) + */ + public static final int BAD_REQUEST = 400; + + /** + * 未授权 + */ + public static final int UNAUTHORIZED = 401; + + /** + * 访问受限,授权过期 + */ + public static final int FORBIDDEN = 403; + + /** + * 资源,服务未找到 + */ + public static final int NOT_FOUND = 404; + + /** + * 不允许的http方法 + */ + public static final int BAD_METHOD = 405; + + /** + * 资源冲突,或者资源被锁 + */ + public static final int CONFLICT = 409; + + /** + * 不支持的数据,媒体类型 + */ + public static final int UNSUPPORTED_TYPE = 415; + + /** + * 系统内部错误 + */ + public static final int ERROR = 500; + + /** + * 接口未实现 + */ + public static final int NOT_IMPLEMENTED = 501; + + /** + * 系统警告消息 + */ + public static final int WARN = 601; +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/constant/ScheduleConstants.java b/ruoyi-common/src/main/java/com/ruoyi/common/constant/ScheduleConstants.java new file mode 100644 index 00000000..62ad8154 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/constant/ScheduleConstants.java @@ -0,0 +1,50 @@ +package com.ruoyi.common.constant; + +/** + * 任务调度通用常量 + * + * @author ruoyi + */ +public class ScheduleConstants +{ + public static final String TASK_CLASS_NAME = "TASK_CLASS_NAME"; + + /** 执行目标key */ + public static final String TASK_PROPERTIES = "TASK_PROPERTIES"; + + /** 默认 */ + public static final String MISFIRE_DEFAULT = "0"; + + /** 立即触发执行 */ + public static final String MISFIRE_IGNORE_MISFIRES = "1"; + + /** 触发一次执行 */ + public static final String MISFIRE_FIRE_AND_PROCEED = "2"; + + /** 不触发立即执行 */ + public static final String MISFIRE_DO_NOTHING = "3"; + + public enum Status + { + /** + * 正常 + */ + NORMAL("0"), + /** + * 暂停 + */ + PAUSE("1"); + + private String value; + + private Status(String value) + { + this.value = value; + } + + public String getValue() + { + return value; + } + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/constant/SqlConstants.java b/ruoyi-common/src/main/java/com/ruoyi/common/constant/SqlConstants.java new file mode 100644 index 00000000..85dcdbfa --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/constant/SqlConstants.java @@ -0,0 +1,29 @@ +package com.ruoyi.common.constant; + +/** + * 数据库常量信息 + * + * @author wangzongrun + */ +public class SqlConstants { + /** + * 数据库排序 升序 + */ + public static final String ASC = "ASC"; + + /** + * 数据库排序 降序 + */ + public static final String DESC = "DESC"; + + /** + * 数据库 开始时间字段名 + */ + public static final String FIELD_NAME_BEGIN_TIME = "beginTime"; + + /** + * 数据库 开始时间字段名 + */ + public static final String FIELD_NAME_END_TIME = "endTime"; + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/constant/SysErrorCodeConstants.java b/ruoyi-common/src/main/java/com/ruoyi/common/constant/SysErrorCodeConstants.java new file mode 100644 index 00000000..83da849c --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/constant/SysErrorCodeConstants.java @@ -0,0 +1,146 @@ +package com.ruoyi.common.constant; + + +import com.ruoyi.common.exception.ErrorCode; + +/** + * System 错误码枚举类 + *

+ * system 系统,使用 1-002-000-000 段 + */ +public interface SysErrorCodeConstants { + + // ========== AUTH 模块 1002000000 ========== + ErrorCode AUTH_LOGIN_BAD_CREDENTIALS = new ErrorCode(1002000000, "登录失败,账号密码不正确"); + ErrorCode AUTH_LOGIN_USER_DISABLED = new ErrorCode(1002000001, "登录失败,账号被禁用"); + ErrorCode AUTH_LOGIN_CAPTCHA_NOT_FOUND = new ErrorCode(1002000003, "验证码不存在"); + ErrorCode AUTH_LOGIN_CAPTCHA_CODE_ERROR = new ErrorCode(1002000004, "验证码不正确"); + ErrorCode AUTH_THIRD_LOGIN_NOT_BIND = new ErrorCode(1002000005, "未绑定账号,需要进行绑定"); + ErrorCode AUTH_TOKEN_EXPIRED = new ErrorCode(1002000006, "Token 已经过期"); + ErrorCode AUTH_MOBILE_NOT_EXISTS = new ErrorCode(1002000007, "手机号不存在"); + + // ========== 菜单模块 1002001000 ========== + ErrorCode MENU_NAME_DUPLICATE = new ErrorCode(1002001000, "已经存在该名字的菜单"); + ErrorCode MENU_PARENT_NOT_EXISTS = new ErrorCode(1002001001, "父菜单不存在"); + ErrorCode MENU_PARENT_ERROR = new ErrorCode(1002001002, "不能设置自己为父菜单"); + ErrorCode MENU_NOT_EXISTS = new ErrorCode(1002001003, "菜单不存在"); + ErrorCode MENU_EXISTS_CHILDREN = new ErrorCode(1002001004, "存在子菜单,无法删除"); + ErrorCode MENU_PARENT_NOT_DIR_OR_MENU = new ErrorCode(1002001005, "父菜单的类型必须是目录或者菜单"); + + // ========== 角色模块 1002002000 ========== + ErrorCode ROLE_NOT_EXISTS = new ErrorCode(1002002000, "角色不存在"); + ErrorCode ROLE_NAME_DUPLICATE = new ErrorCode(1002002001, "已经存在名为【{}】的角色"); + ErrorCode ROLE_CODE_DUPLICATE = new ErrorCode(1002002002, "已经存在编码为【{}】的角色"); + ErrorCode ROLE_CAN_NOT_UPDATE_SYSTEM_TYPE_ROLE = new ErrorCode(1002002003, "不能操作类型为系统内置的角色"); + ErrorCode ROLE_IS_DISABLE = new ErrorCode(1002002004, "名字为【{}】的角色已被禁用"); + ErrorCode ROLE_ADMIN_CODE_ERROR = new ErrorCode(1002002005, "编码【{}】不能使用"); + + // ========== 用户模块 1002003000 ========== + ErrorCode USER_USERNAME_EXISTS = new ErrorCode(1002003000, "用户账号已经存在"); + ErrorCode USER_MOBILE_EXISTS = new ErrorCode(1002003001, "手机号已经存在"); + ErrorCode USER_EMAIL_EXISTS = new ErrorCode(1002003002, "邮箱已经存在"); + ErrorCode USER_NOT_EXISTS = new ErrorCode(1002003003, "用户不存在"); + ErrorCode USER_IMPORT_LIST_IS_EMPTY = new ErrorCode(1002003004, "导入用户数据不能为空!"); + ErrorCode USER_PASSWORD_FAILED = new ErrorCode(1002003005, "用户密码校验失败"); + ErrorCode USER_IS_DISABLE = new ErrorCode(1002003006, "名字为【{}】的用户已被禁用"); + ErrorCode USER_COUNT_MAX = new ErrorCode(1002003008, "创建用户失败,原因:超过租户最大租户配额({})!"); + + // ========== 部门模块 1002004000 ========== + ErrorCode DEPT_NAME_DUPLICATE = new ErrorCode(1002004000, "已经存在该名字的部门"); + ErrorCode DEPT_PARENT_NOT_EXITS = new ErrorCode(1002004001, "父级部门不存在"); + ErrorCode DEPT_NOT_FOUND = new ErrorCode(1002004002, "当前部门不存在"); + ErrorCode DEPT_EXITS_CHILDREN = new ErrorCode(1002004003, "存在子部门,无法删除"); + ErrorCode DEPT_PARENT_ERROR = new ErrorCode(1002004004, "不能设置自己为父部门"); + ErrorCode DEPT_EXISTS_USER = new ErrorCode(1002004005, "部门中存在员工,无法删除"); + ErrorCode DEPT_NOT_ENABLE = new ErrorCode(1002004006, "部门不处于开启状态,不允许选择"); + ErrorCode DEPT_PARENT_IS_CHILD = new ErrorCode(1002004007, "不能设置自己的子部门为父部门"); + + // ========== 岗位模块 1002005000 ========== + ErrorCode POST_NOT_FOUND = new ErrorCode(1002005000, "当前岗位不存在"); + ErrorCode POST_NOT_ENABLE = new ErrorCode(1002005001, "岗位({}) 不处于开启状态,不允许选择"); + ErrorCode POST_NAME_DUPLICATE = new ErrorCode(1002005002, "已经存在该名字的岗位"); + ErrorCode POST_CODE_DUPLICATE = new ErrorCode(1002005003, "已经存在该标识的岗位"); + + // ========== 字典类型 1002006000 ========== + ErrorCode DICT_TYPE_NOT_EXISTS = new ErrorCode(1002006001, "当前字典类型不存在"); + ErrorCode DICT_TYPE_NOT_ENABLE = new ErrorCode(1002006002, "字典类型不处于开启状态,不允许选择"); + ErrorCode DICT_TYPE_NAME_DUPLICATE = new ErrorCode(1002006003, "已经存在该名字的字典类型"); + ErrorCode DICT_TYPE_TYPE_DUPLICATE = new ErrorCode(1002006004, "已经存在该类型的字典类型"); + ErrorCode DICT_TYPE_HAS_CHILDREN = new ErrorCode(1002006005, "无法删除,该字典类型还有字典数据"); + + // ========== 字典数据 1002007000 ========== + ErrorCode DICT_DATA_NOT_EXISTS = new ErrorCode(1002007001, "当前字典数据不存在"); + ErrorCode DICT_DATA_NOT_ENABLE = new ErrorCode(1002007002, "字典数据({})不处于开启状态,不允许选择"); + ErrorCode DICT_DATA_VALUE_DUPLICATE = new ErrorCode(1002007003, "已经存在该值的字典数据"); + + // ========== 通知公告 1002008000 ========== + ErrorCode NOTICE_NOT_FOUND = new ErrorCode(1002008001, "当前通知公告不存在"); + + // ========== 短信渠道 1002011000 ========== + ErrorCode SMS_CHANNEL_NOT_EXISTS = new ErrorCode(1002011000, "短信渠道不存在"); + ErrorCode SMS_CHANNEL_DISABLE = new ErrorCode(1002011001, "短信渠道不处于开启状态,不允许选择"); + ErrorCode SMS_CHANNEL_HAS_CHILDREN = new ErrorCode(1002011002, "无法删除,该短信渠道还有短信模板"); + + // ========== 短信模板 1002012000 ========== + ErrorCode SMS_TEMPLATE_NOT_EXISTS = new ErrorCode(1002012000, "短信模板不存在"); + ErrorCode SMS_TEMPLATE_CODE_DUPLICATE = new ErrorCode(1002012001, "已经存在编码为【{}】的短信模板"); + + // ========== 短信发送 1002013000 ========== + ErrorCode SMS_SEND_MOBILE_NOT_EXISTS = new ErrorCode(1002013000, "手机号不存在"); + ErrorCode SMS_SEND_MOBILE_TEMPLATE_PARAM_MISS = new ErrorCode(1002013001, "模板参数({})缺失"); + ErrorCode SMS_SEND_TEMPLATE_NOT_EXISTS = new ErrorCode(1002013002, "短信模板不存在"); + + // ========== 短信验证码 1002014000 ========== + ErrorCode SMS_CODE_NOT_FOUND = new ErrorCode(1002014000, "验证码不存在"); + ErrorCode SMS_CODE_EXPIRED = new ErrorCode(1002014001, "验证码已过期"); + ErrorCode SMS_CODE_USED = new ErrorCode(1002014002, "验证码已使用"); + ErrorCode SMS_CODE_NOT_CORRECT = new ErrorCode(1002014003, "验证码不正确"); + ErrorCode SMS_CODE_EXCEED_SEND_MAXIMUM_QUANTITY_PER_DAY = new ErrorCode(1002014004, "超过每日短信发送数量"); + ErrorCode SMS_CODE_SEND_TOO_FAST = new ErrorCode(1002014005, "短信发送过于频率"); + ErrorCode SMS_CODE_IS_EXISTS = new ErrorCode(1002014006, "手机号已被使用"); + ErrorCode SMS_CODE_IS_UNUSED = new ErrorCode(1002014007, "验证码未被使用"); + + // ========== 租户信息 1002015000 ========== + ErrorCode TENANT_NOT_EXISTS = new ErrorCode(1002015000, "租户不存在"); + ErrorCode TENANT_DISABLE = new ErrorCode(1002015001, "名字为【{}】的租户已被禁用"); + ErrorCode TENANT_EXPIRE = new ErrorCode(1002015002, "名字为【{}】的租户已过期"); + ErrorCode TENANT_CAN_NOT_UPDATE_SYSTEM = new ErrorCode(1002015003, "系统租户不能进行修改、删除等操作!"); + + // ========== 租户套餐 1002016000 ========== + ErrorCode TENANT_PACKAGE_NOT_EXISTS = new ErrorCode(1002016000, "租户套餐不存在"); + ErrorCode TENANT_PACKAGE_USED = new ErrorCode(1002016001, "租户正在使用该套餐,请给租户重新设置套餐后再尝试删除"); + ErrorCode TENANT_PACKAGE_DISABLE = new ErrorCode(1002016002, "名字为【{}】的租户套餐已被禁用"); + + // ========== 错误码模块 1002017000 ========== + ErrorCode ERROR_CODE_NOT_EXISTS = new ErrorCode(1002017000, "错误码不存在"); + ErrorCode ERROR_CODE_DUPLICATE = new ErrorCode(1002017001, "已经存在编码为【{}】的错误码"); + + // ========== 社交用户 1002018000 ========== + ErrorCode SOCIAL_USER_AUTH_FAILURE = new ErrorCode(1002018000, "社交授权失败,原因是:{}"); + ErrorCode SOCIAL_USER_UNBIND_NOT_SELF = new ErrorCode(1002018001, "社交解绑失败,非当前用户绑定"); + ErrorCode SOCIAL_USER_NOT_FOUND = new ErrorCode(1002018002, "社交授权失败,找不到对应的用户"); + + // ========== 系统敏感词 1002019000 ========= + ErrorCode SENSITIVE_WORD_NOT_EXISTS = new ErrorCode(1002019000, "系统敏感词在所有标签中都不存在"); + ErrorCode SENSITIVE_WORD_EXISTS = new ErrorCode(1002019001, "系统敏感词已在标签中存在"); + + // ========== OAuth2 客户端 1002020000 ========= + ErrorCode OAUTH2_CLIENT_NOT_EXISTS = new ErrorCode(1002020000, "OAuth2 客户端不存在"); + ErrorCode OAUTH2_CLIENT_EXISTS = new ErrorCode(1002020001, "OAuth2 客户端编号已存在"); + ErrorCode OAUTH2_CLIENT_DISABLE = new ErrorCode(1002020002, "OAuth2 客户端已禁用"); + ErrorCode OAUTH2_CLIENT_AUTHORIZED_GRANT_TYPE_NOT_EXISTS = new ErrorCode(1002020003, "不支持该授权类型"); + ErrorCode OAUTH2_CLIENT_SCOPE_OVER = new ErrorCode(1002020004, "授权范围过大"); + ErrorCode OAUTH2_CLIENT_REDIRECT_URI_NOT_MATCH = new ErrorCode(1002020005, "无效 redirect_uri: {}"); + ErrorCode OAUTH2_CLIENT_CLIENT_SECRET_ERROR = new ErrorCode(1002020006, "无效 client_secret: {}"); + + // ========== OAuth2 授权 1002021000 ========= + ErrorCode OAUTH2_GRANT_CLIENT_ID_MISMATCH = new ErrorCode(1002021000, "client_id 不匹配"); + ErrorCode OAUTH2_GRANT_REDIRECT_URI_MISMATCH = new ErrorCode(1002021001, "redirect_uri 不匹配"); + ErrorCode OAUTH2_GRANT_STATE_MISMATCH = new ErrorCode(1002021002, "state 不匹配"); + ErrorCode OAUTH2_GRANT_CODE_NOT_EXISTS = new ErrorCode(1002021003, "code 不存在"); + + // ========== OAuth2 授权 1002022000 ========= + ErrorCode OAUTH2_CODE_NOT_EXISTS = new ErrorCode(1002022000, "code 不存在"); + ErrorCode OAUTH2_CODE_EXPIRE = new ErrorCode(1002022000, "code 已过期"); + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/constant/TreeConstants.java b/ruoyi-common/src/main/java/com/ruoyi/common/constant/TreeConstants.java new file mode 100644 index 00000000..3752848b --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/constant/TreeConstants.java @@ -0,0 +1,29 @@ +package com.ruoyi.common.constant; + +/** + * 树形通用常量信息 + * + * @author wangzongrun + */ +public class TreeConstants { + /** + * 无父级的情况下,祖籍的值 + */ + public static final String ANCESTORS_ROOT_VALUE = "0"; + + /** + * 无父级的情况下, + */ + public static final String PARENT_ROOT_VALUE = "0"; + + /** + * 无父级的情况下,层次默认值 + */ + public static final Integer LAYOUT_DEPTH_INIT_VALUE = 1; + + /** + * 层次码分隔符 + * .不能直接转义,添加转移字符 + */ + public static final String LAYOUT_CODE_SEPARATOR = "."; +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/constant/UserConstants.java b/ruoyi-common/src/main/java/com/ruoyi/common/constant/UserConstants.java new file mode 100644 index 00000000..a936cd83 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/constant/UserConstants.java @@ -0,0 +1,78 @@ +package com.ruoyi.common.constant; + +/** + * 用户常量信息 + * + * @author ruoyi + */ +public class UserConstants +{ + /** + * 平台内系统用户的唯一标志 + */ + public static final String SYS_USER = "SYS_USER"; + + /** 正常状态 */ + public static final String NORMAL = "0"; + + /** 异常状态 */ + public static final String EXCEPTION = "1"; + + /** 用户封禁状态 */ + public static final String USER_DISABLE = "1"; + + /** 角色封禁状态 */ + public static final String ROLE_DISABLE = "1"; + + /** 部门正常状态 */ + public static final String DEPT_NORMAL = "0"; + + /** 部门停用状态 */ + public static final String DEPT_DISABLE = "1"; + + /** 字典正常状态 */ + public static final String DICT_NORMAL = "0"; + + /** 是否为系统默认(是) */ + public static final String YES = "Y"; + + /** 是否菜单外链(是) */ + public static final String YES_FRAME = "0"; + + /** 是否菜单外链(否) */ + public static final String NO_FRAME = "1"; + + /** 菜单类型(目录) */ + public static final String TYPE_DIR = "M"; + + /** 菜单类型(菜单) */ + public static final String TYPE_MENU = "C"; + + /** 菜单类型(按钮) */ + public static final String TYPE_BUTTON = "F"; + + /** Layout组件标识 */ + public final static String LAYOUT = "Layout"; + + /** ParentView组件标识 */ + public final static String PARENT_VIEW = "ParentView"; + + /** InnerLink组件标识 */ + public final static String INNER_LINK = "InnerLink"; + + /** 校验返回结果码 */ + public final static String UNIQUE = "0"; + public final static String NOT_UNIQUE = "1"; + + /** + * 用户名长度限制 + */ + public static final int USERNAME_MIN_LENGTH = 2; + public static final int USERNAME_MAX_LENGTH = 20; + + /** + * 密码长度限制 + */ + public static final int PASSWORD_MIN_LENGTH = 5; + public static final int PASSWORD_MAX_LENGTH = 20; +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/convert/DictConvert.java b/ruoyi-common/src/main/java/com/ruoyi/common/convert/DictConvert.java new file mode 100644 index 00000000..cf40ccdd --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/convert/DictConvert.java @@ -0,0 +1,71 @@ +package com.ruoyi.common.convert; + +import cn.hutool.core.convert.Convert; + +import com.alibaba.excel.converters.Converter; +import com.alibaba.excel.enums.CellDataTypeEnum; +import com.alibaba.excel.metadata.GlobalConfiguration; +import com.alibaba.excel.metadata.data.ReadCellData; +import com.alibaba.excel.metadata.data.WriteCellData; +import com.alibaba.excel.metadata.property.ExcelContentProperty; +import com.ruoyi.common.annotation.DictFormat; +import com.ruoyi.common.utils.DictUtils; +import lombok.extern.slf4j.Slf4j; + +/** + * Excel 数据字典转换器 + * + * @author 芋道源码 + */ +@Slf4j +public class DictConvert implements Converter { + + @Override + public Class supportJavaTypeKey() { + throw new UnsupportedOperationException("暂不支持,也不需要"); + } + + @Override + public CellDataTypeEnum supportExcelTypeKey() { + throw new UnsupportedOperationException("暂不支持,也不需要"); + } + + @Override + public Object convertToJavaData(ReadCellData readCellData, ExcelContentProperty contentProperty, GlobalConfiguration globalConfiguration) { + // 使用字典解析 + String type = getType(contentProperty); + String label = readCellData.getStringValue(); + String value = DictUtils.getDictValue(type, label); + if (value == null) { + log.error("[convertToJavaData][type({}) 解析不掉 label({})]", type, label); + return null; + } + // 将 String 的 value 转换成对应的属性 + Class fieldClazz = contentProperty.getField().getType(); + return Convert.convert(fieldClazz, value); + } + + @Override + public WriteCellData convertToExcelData(Object object, ExcelContentProperty contentProperty, GlobalConfiguration globalConfiguration) { + // 空时,返回空 + if (object == null) { + return new WriteCellData<>(""); + } + + // 使用字典格式化 + String type = getType(contentProperty); + String value = String.valueOf(object); + String label = DictUtils.getDictLabel(type, value); + if (label == null) { + log.error("[convertToExcelData][type({}) 转换不了 label({})]", type, value); + return new WriteCellData<>(""); + } + // 生成 Excel 小表格 + return new WriteCellData<>(label); + } + + private static String getType(ExcelContentProperty contentProperty) { + return contentProperty.getField().getAnnotation(DictFormat.class).value(); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/convert/JsonConvert.java b/ruoyi-common/src/main/java/com/ruoyi/common/convert/JsonConvert.java new file mode 100644 index 00000000..5b512b3b --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/convert/JsonConvert.java @@ -0,0 +1,33 @@ +package com.ruoyi.common.convert; + +import com.alibaba.excel.converters.Converter; +import com.alibaba.excel.enums.CellDataTypeEnum; +import com.alibaba.excel.metadata.GlobalConfiguration; +import com.alibaba.excel.metadata.data.WriteCellData; +import com.alibaba.excel.metadata.property.ExcelContentProperty; +import com.ruoyi.common.utils.JsonUtils; + +/** + * Excel Json 转换器 + * + * @author 芋道源码 + */ +public class JsonConvert implements Converter { + + @Override + public Class supportJavaTypeKey() { + throw new UnsupportedOperationException("暂不支持,也不需要"); + } + + @Override + public CellDataTypeEnum supportExcelTypeKey() { + throw new UnsupportedOperationException("暂不支持,也不需要"); + } + + @Override + public WriteCellData convertToExcelData(Object value, ExcelContentProperty contentProperty, GlobalConfiguration globalConfiguration) { + // 生成 Excel 小表格 + return new WriteCellData<>(JsonUtils.toJsonString(value)); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/controller/BaseController.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/controller/BaseController.java new file mode 100644 index 00000000..0645d64b --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/controller/BaseController.java @@ -0,0 +1,236 @@ +package com.ruoyi.common.core.controller; + +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.baomidou.mybatisplus.core.metadata.OrderItem; +import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.github.pagehelper.PageHelper; +import com.github.pagehelper.PageInfo; +import com.ruoyi.common.constant.HttpStatus; +import com.ruoyi.common.constant.SqlConstants; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.common.core.page.PageDomain; +import com.ruoyi.common.core.page.TableDataInfo; +import com.ruoyi.common.core.page.TableSupport; +import com.ruoyi.common.utils.DateUtils; +import com.ruoyi.common.utils.PageUtils; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.sql.SqlUtil; +import org.apache.poi.ss.formula.functions.T; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.web.bind.WebDataBinder; +import org.springframework.web.bind.annotation.InitBinder; + +import java.beans.PropertyEditorSupport; +import java.util.Date; +import java.util.List; + +/** + * web层通用数据处理 + * + * @author ruoyi + */ +public class BaseController { + protected final Logger logger = LoggerFactory.getLogger(this.getClass()); + + /** + * 将前台传递过来的日期格式的字符串,自动转化为Date类型 + */ + @InitBinder + public void initBinder(WebDataBinder binder) { + // Date 类型转换 + binder.registerCustomEditor(Date.class, new PropertyEditorSupport() { + @Override + public void setAsText(String text) { + setValue(DateUtils.parseDate(text)); + } + }); + } + + /** + * 设置请求分页数据 + */ + protected void startPage() { + PageUtils.startPage(); + } + protected void startPage(String orderBy) { + PageUtils.startPage(orderBy); + } + /** + * 设置mybatis-plus请求分页数据 + */ + protected Page getPage(OrderItem... sorts) { + Page page = new Page<>(); + PageDomain pageDomain = TableSupport.buildPageRequest(); + Integer pageNum = pageDomain.getPageNum(); + Integer pageSize = pageDomain.getPageSize(); + String orderColumn = pageDomain.getOrderByColumn(); + String isAsc = pageDomain.getIsAsc(); + + if (StringUtils.isNotNull(pageNum) && StringUtils.isNotNull(pageSize)) { + page.setCurrent(pageNum); + page.setSize(pageSize); + page.setOptimizeCountSql(false); + page.setMaxLimit(500L); + } + //排序 + if (StringUtils.isNotBlank(orderColumn)) { + if (SqlConstants.ASC.equalsIgnoreCase(isAsc)) { + page.addOrder(OrderItem.asc(orderColumn)); + } else { + page.addOrder(OrderItem.desc(orderColumn)); + } + } + if (sorts.length >0) { + page.addOrder(sorts); + } + return page; + } + + /** + * mybatis-plus响应请求分页数据 + * + * @param page + */ + protected TableDataInfo getDataTable(IPage page) { + TableDataInfo rspData = new TableDataInfo(); + rspData.setCode(HttpStatus.SUCCESS); + rspData.setMsg("查询成功"); + rspData.setRows(page.getRecords()); + rspData.setTotal(page.getTotal()); + rspData.setPageNum(page.getCurrent()); + rspData.setPageSize(page.getSize()); + rspData.setTotalPageNum(page.getPages()); + return rspData; + } + + /** + * 设置请求排序数据 + */ + protected void startOrderBy() { + PageDomain pageDomain = TableSupport.buildPageRequest(); + if (StringUtils.isNotEmpty(pageDomain.getOrderBy())) { + String orderBy = SqlUtil.escapeOrderBySql(pageDomain.getOrderBy()); + PageHelper.orderBy(orderBy); + } + } + + /** + * 清理分页的线程变量 + */ + protected void clearPage() { + PageUtils.clearPage(); + } + + /** + * 响应请求分页数据 + */ + @SuppressWarnings({"rawtypes", "unchecked"}) + protected TableDataInfo getDataTable(List list) { + TableDataInfo rspData = new TableDataInfo(); + rspData.setCode(HttpStatus.SUCCESS); + rspData.setMsg("查询成功"); + rspData.setRows(list); + rspData.setTotal(new PageInfo(list).getTotal()); + return rspData; + } + + /** + * 返回成功 + */ + public AjaxResult success() { + return AjaxResult.success(); + } + + /** + * 返回失败消息 + */ + public AjaxResult error() { + return AjaxResult.error(); + } + + /** + * 返回成功消息 + */ + public AjaxResult success(String message) { + return AjaxResult.success(message); + } + + /** + * 返回成功消息 + */ + public AjaxResult success(Object data) { + return AjaxResult.success(data); + } + + /** + * 返回失败消息 + */ + public AjaxResult error(String message) { + return AjaxResult.error(message); + } + + /** + * 返回警告消息 + */ + public AjaxResult warn(String message) { + return AjaxResult.warn(message); + } + + /** + * 响应返回结果 + * + * @param rows 影响行数 + * @return 操作结果 + */ + protected AjaxResult toAjax(int rows) { + return rows > 0 ? AjaxResult.success() : AjaxResult.error(); + } + + /** + * 响应返回结果 + * + * @param result 结果 + * @return 操作结果 + */ + protected AjaxResult toAjax(boolean result) { + return result ? success() : error(); + } + + /** + * 页面跳转 + */ + public String redirect(String url) { + return StringUtils.format("redirect:{}", url); + } + + /** + * 获取用户缓存信息 + */ + public LoginUser getLoginUser() { + return SecurityUtils.getLoginUser(); + } + + /** + * 获取登录用户id + */ + public Long getUserId() { + return getLoginUser().getUserId(); + } + + /** + * 获取登录部门id + */ + public Long getDeptId() { + return getLoginUser().getDeptId(); + } + + /** + * 获取登录用户名 + */ + public String getUsername() { + return getLoginUser().getUsername(); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/AjaxResult.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/AjaxResult.java new file mode 100644 index 00000000..3dc5afc9 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/AjaxResult.java @@ -0,0 +1,179 @@ +package com.ruoyi.common.core.domain; + +import com.ruoyi.common.constant.HttpStatus; +import com.ruoyi.common.utils.StringUtils; + +import java.util.HashMap; + +/** + * 操作消息提醒 + * + * @author ruoyi + */ +public class AjaxResult extends HashMap { + private static final long serialVersionUID = 1L; + + /** + * 状态码 + */ + public static final String CODE_TAG = "code"; + + /** + * 返回内容 + */ + public static final String MSG_TAG = "msg"; + + /** + * 数据对象 + */ + public static final String DATA_TAG = "data"; + + /** + * 初始化一个新创建的 AjaxResult 对象,使其表示一个空消息。 + */ + public AjaxResult() { + } + + /** + * 初始化一个新创建的 AjaxResult 对象 + * + * @param code 状态码 + * @param msg 返回内容 + */ + public AjaxResult(int code, String msg) { + super.put(CODE_TAG, code); + super.put(MSG_TAG, msg); + } + + /** + * 初始化一个新创建的 AjaxResult 对象 + * + * @param code 状态码 + * @param msg 返回内容 + * @param data 数据对象 + */ + public AjaxResult(int code, String msg, Object data) { + super.put(CODE_TAG, code); + super.put(MSG_TAG, msg); + if (StringUtils.isNotNull(data)) { + super.put(DATA_TAG, data); + } + } + + /** + * 返回成功消息 + * + * @return 成功消息 + */ + public static AjaxResult success() { + return AjaxResult.success("操作成功"); + } + + /** + * 返回成功数据 + * + * @return 成功消息 + */ + public static AjaxResult success(Object data) { + return AjaxResult.success("操作成功", data); + } + + /** + * 返回成功消息 + * + * @param msg 返回内容 + * @return 成功消息 + */ + public static AjaxResult success(String msg) { + return AjaxResult.success(msg, null); + } + + /** + * 返回成功消息 + * + * @param msg 返回内容 + * @param data 数据对象 + * @return 成功消息 + */ + public static AjaxResult success(String msg, Object data) { + return new AjaxResult(HttpStatus.SUCCESS, msg, data); + } + + /** + * 返回警告消息 + * + * @param msg 返回内容 + * @return 警告消息 + */ + public static AjaxResult warn(String msg) { + return AjaxResult.warn(msg, null); + } + + /** + * 返回警告消息 + * + * @param msg 返回内容 + * @param data 数据对象 + * @return 警告消息 + */ + public static AjaxResult warn(String msg, Object data) { + return new AjaxResult(HttpStatus.WARN, msg, data); + } + + /** + * 返回错误消息 + * + * @return + */ + public static AjaxResult error() { + return AjaxResult.error("操作失败"); + } + + /** + * 返回错误消息 + * + * @param msg 返回内容 + * @return 警告消息 + */ + public static AjaxResult error(String msg) { + return AjaxResult.error(msg, null); + } + + /** + * 返回错误消息 + * + * @param msg 返回内容 + * @param data 数据对象 + * @return 警告消息 + */ + public static AjaxResult error(String msg, Object data) { + return new AjaxResult(HttpStatus.ERROR, msg, data); + } + + /** + * 返回错误消息 + * + * @param code 状态码 + * @param msg 返回内容 + * @return 警告消息 + */ + public static AjaxResult error(int code, String msg) { + return new AjaxResult(code, msg, null); + } + + /** + * 方便链式调用 + * + * @param key 键 + * @param value 值 + * @return 数据对象 + */ + @Override + public AjaxResult put(String key, Object value) { + super.put(key, value); + return this; + } + public boolean isSuccess(){ + return super.get("code") != null && "200".equals(super.get("code").toString()); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/BaseEntity.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/BaseEntity.java new file mode 100644 index 00000000..9ae0db89 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/BaseEntity.java @@ -0,0 +1,124 @@ +package com.ruoyi.common.core.domain; + +import java.io.Serializable; +import java.util.Date; +import java.util.HashMap; +import java.util.Map; + +import com.baomidou.mybatisplus.annotation.FieldFill; +import com.baomidou.mybatisplus.annotation.TableField; +import com.fasterxml.jackson.annotation.JsonFormat; + +/** + * Entity基类 + * + * @author ruoyi + */ +public class BaseEntity implements Serializable +{ + private static final long serialVersionUID = 1L; + + /** 搜索值 */ + @TableField(exist = false) + private String searchValue; + + /** 创建者 */ + @TableField(fill = FieldFill.INSERT) + private String createBy; + + /** 创建时间 */ + @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") + @TableField(fill = FieldFill.INSERT) + private Date createTime; + + /** 更新者 */ + @TableField(fill = FieldFill.INSERT_UPDATE) + private String updateBy; + + /** 更新时间 */ + @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") + @TableField(fill = FieldFill.INSERT_UPDATE) + private Date updateTime; + + /** 备注 */ + @TableField(exist = false) + private String remark; + + /** 请求参数 */ + @TableField(exist = false) + private Map params; + + public String getSearchValue() + { + return searchValue; + } + + public void setSearchValue(String searchValue) + { + this.searchValue = searchValue; + } + + public String getCreateBy() + { + return createBy; + } + + public void setCreateBy(String createBy) + { + this.createBy = createBy; + } + + public Date getCreateTime() + { + return createTime; + } + + public void setCreateTime(Date createTime) + { + this.createTime = createTime; + } + + public String getUpdateBy() + { + return updateBy; + } + + public void setUpdateBy(String updateBy) + { + this.updateBy = updateBy; + } + + public Date getUpdateTime() + { + return updateTime; + } + + public void setUpdateTime(Date updateTime) + { + this.updateTime = updateTime; + } + + public String getRemark() + { + return remark; + } + + public void setRemark(String remark) + { + this.remark = remark; + } + + public Map getParams() + { + if (params == null) + { + params = new HashMap<>(); + } + return params; + } + + public void setParams(Map params) + { + this.params = params; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/DictTreeEntity.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/DictTreeEntity.java new file mode 100644 index 00000000..16f21c82 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/DictTreeEntity.java @@ -0,0 +1,118 @@ +package com.ruoyi.common.core.domain; + +import com.baomidou.mybatisplus.annotation.*; +import com.fasterxml.jackson.annotation.JsonFormat; +import com.fasterxml.jackson.annotation.JsonInclude; +import com.ruoyi.common.annotation.Excel; +import lombok.Data; + +import java.util.ArrayList; +import java.util.Date; +import java.util.List; + +/** + * Tree基类 + * + * @author ruoyi + */ +@Data +public class DictTreeEntity { + private static final long serialVersionUID = 1L; + + /** + * 主键 + */ + @Excel(name = "主键") + @TableId(value = "ID", type = IdType.ASSIGN_ID) + private String id; + + /** + * 名称 + */ + @Excel(name = "名称") + private String label; + + /** + * 编码 + */ + @Excel(name = "编码") + private String code; + + /** + * 备注 + */ + @Excel(name = "备注") + private String remark; + + /** + * 父节点ID + */ + private String pid; + + /** + * 显示顺序 + */ + private Integer orderNum; + + /** + * 层次码 + * 说明: + * 顶级层次码为0 + * 非顶级使用4位数字显示,起始位1000 + */ + private String levelCode; + + /** + * 层次 + * 说明: + * 结合层次码,计算树形结构的的层次 + */ + private Integer levelDepth; + + /** + * 是否叶子节点 + *

+ * 说明: + * 新建节点,默认为末级节点,并更新父节点为非末级节点 + */ + private String isLeaf; + + /** + * 创建者 + */ + @TableField(fill = FieldFill.INSERT) + private String createBy; + + /** + * 创建时间 + */ + @TableField(fill = FieldFill.INSERT) + @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") + private Date createTime; + + /** + * 更新者 + */ + @TableField(fill = FieldFill.INSERT_UPDATE) + private String updateBy; + + /** + * 更新时间 + */ + @TableField(fill = FieldFill.INSERT_UPDATE) + @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") + private Date updateTime; + + /** + * 删除标志 + */ + @TableLogic + private String delFlag; + + /** + * 子节点 + */ + @TableField(exist = false) + @JsonInclude(JsonInclude.Include.NON_EMPTY) + private List children = new ArrayList<>(); +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/R.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/R.java new file mode 100644 index 00000000..a1290d99 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/R.java @@ -0,0 +1,102 @@ +package com.ruoyi.common.core.domain; + +import com.ruoyi.common.constant.HttpStatus; + +import java.io.Serializable; + +/** + * 响应信息主体 + * + * @author ruoyi + */ +public class R implements Serializable { + private static final long serialVersionUID = 1L; + + /** + * 成功 + */ + public static final int SUCCESS = HttpStatus.SUCCESS; + + /** + * 失败 + */ + public static final int FAIL = HttpStatus.ERROR; + + private int code; + + private String msg; + + private T data; + + public static R ok() { + return restResult(null, SUCCESS, "操作成功"); + } + + public static R ok(T data) { + return restResult(data, SUCCESS, "操作成功"); + } + + public static R ok(T data, String msg) { + return restResult(data, SUCCESS, msg); + } + + public static R fail() { + return restResult(null, FAIL, "操作失败"); + } + + public static R fail(String msg) { + return restResult(null, FAIL, msg); + } + + public static R fail(T data) { + return restResult(data, FAIL, "操作失败"); + } + + public static R fail(T data, String msg) { + return restResult(data, FAIL, msg); + } + + public static R fail(int code, String msg) { + return restResult(null, code, msg); + } + + private static R restResult(T data, int code, String msg) { + R apiResult = new R<>(); + apiResult.setCode(code); + apiResult.setData(data); + apiResult.setMsg(msg); + return apiResult; + } + + public int getCode() { + return code; + } + + public void setCode(int code) { + this.code = code; + } + + public String getMsg() { + return msg; + } + + public void setMsg(String msg) { + this.msg = msg; + } + + public T getData() { + return data; + } + + public void setData(T data) { + this.data = data; + } + + public static Boolean isError(R ret) { + return !isSuccess(ret); + } + + public static Boolean isSuccess(R ret) { + return R.SUCCESS == ret.getCode(); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/TenantBaseDO.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/TenantBaseDO.java new file mode 100644 index 00000000..f5609b0a --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/TenantBaseDO.java @@ -0,0 +1,20 @@ +package com.ruoyi.common.core.domain; + +import lombok.Data; +import lombok.EqualsAndHashCode; + +/** + * 拓展多租户的 BaseDO 基类 + * + * @author 芋道源码 + */ +@Data +@EqualsAndHashCode(callSuper = true) +public abstract class TenantBaseDO extends BaseEntity { + + /** + * 多租户编号 + */ + private Long tenantId; + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/TreeEntity.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/TreeEntity.java new file mode 100644 index 00000000..a180a18c --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/TreeEntity.java @@ -0,0 +1,79 @@ +package com.ruoyi.common.core.domain; + +import java.util.ArrayList; +import java.util.List; + +/** + * Tree基类 + * + * @author ruoyi + */ +public class TreeEntity extends BaseEntity +{ + private static final long serialVersionUID = 1L; + + /** 父菜单名称 */ + private String parentName; + + /** 父菜单ID */ + private Long parentId; + + /** 显示顺序 */ + private Integer orderNum; + + /** 祖级列表 */ + private String ancestors; + + /** 子部门 */ + private List children = new ArrayList<>(); + + public String getParentName() + { + return parentName; + } + + public void setParentName(String parentName) + { + this.parentName = parentName; + } + + public Long getParentId() + { + return parentId; + } + + public void setParentId(Long parentId) + { + this.parentId = parentId; + } + + public Integer getOrderNum() + { + return orderNum; + } + + public void setOrderNum(Integer orderNum) + { + this.orderNum = orderNum; + } + + public String getAncestors() + { + return ancestors; + } + + public void setAncestors(String ancestors) + { + this.ancestors = ancestors; + } + + public List getChildren() + { + return children; + } + + public void setChildren(List children) + { + this.children = children; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/TreeSelect.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/TreeSelect.java new file mode 100644 index 00000000..bd835db9 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/TreeSelect.java @@ -0,0 +1,77 @@ +package com.ruoyi.common.core.domain; + +import java.io.Serializable; +import java.util.List; +import java.util.stream.Collectors; +import com.fasterxml.jackson.annotation.JsonInclude; +import com.ruoyi.common.core.domain.entity.SysDept; +import com.ruoyi.common.core.domain.entity.SysMenu; + +/** + * Treeselect树结构实体类 + * + * @author ruoyi + */ +public class TreeSelect implements Serializable +{ + private static final long serialVersionUID = 1L; + + /** 节点ID */ + private Long id; + + /** 节点名称 */ + private String label; + + /** 子节点 */ + @JsonInclude(JsonInclude.Include.NON_EMPTY) + private List children; + + public TreeSelect() + { + + } + + public TreeSelect(SysDept dept) + { + this.id = dept.getDeptId(); + this.label = dept.getDeptName(); + this.children = dept.getChildren().stream().map(TreeSelect::new).collect(Collectors.toList()); + } + + public TreeSelect(SysMenu menu) + { + this.id = menu.getMenuId(); + this.label = menu.getMenuName(); + this.children = menu.getChildren().stream().map(TreeSelect::new).collect(Collectors.toList()); + } + + public Long getId() + { + return id; + } + + public void setId(Long id) + { + this.id = id; + } + + public String getLabel() + { + return label; + } + + public void setLabel(String label) + { + this.label = label; + } + + public List getChildren() + { + return children; + } + + public void setChildren(List children) + { + this.children = children; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysDept.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysDept.java new file mode 100644 index 00000000..16ffd7f8 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysDept.java @@ -0,0 +1,221 @@ +package com.ruoyi.common.core.domain.entity; + +import java.util.ArrayList; +import java.util.List; +import javax.validation.constraints.Email; +import javax.validation.constraints.NotBlank; +import javax.validation.constraints.NotNull; +import javax.validation.constraints.Size; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableField; +import com.baomidou.mybatisplus.annotation.TableId; +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import com.ruoyi.common.core.domain.BaseEntity; + +/** + * 部门表 sys_dept + * + * @author ruoyi + */ +public class SysDept extends BaseEntity +{ + private static final long serialVersionUID = 1L; + + /** 部门ID */ + @TableId(type = IdType.AUTO) + private Long deptId; + + /** 父部门ID */ + private Long parentId; + + /** 祖级列表 */ + private String ancestors; + + /** 部门名称 */ + private String deptName; + + /** 显示顺序 */ + private Integer orderNum; + + /** 负责人 */ + private String leader; + + /** 联系电话 */ + private String phone; + + /** 邮箱 */ + private String email; + + /** 部门状态:0正常,1停用 */ + private String status; + + /** 删除标志(0代表存在 2代表删除) */ + private String delFlag; + + /** 父部门名称 */ + @TableField(exist = false) + private String parentName; + /** + * 多租户编号 + */ + private Long tenantId; + /** 子部门 */ + @TableField(exist = false) + private List children = new ArrayList(); + + public Long getDeptId() + { + return deptId; + } + + public void setDeptId(Long deptId) + { + this.deptId = deptId; + } + + public Long getParentId() + { + return parentId; + } + + public void setParentId(Long parentId) + { + this.parentId = parentId; + } + + public String getAncestors() + { + return ancestors; + } + + public void setAncestors(String ancestors) + { + this.ancestors = ancestors; + } + + @NotBlank(message = "部门名称不能为空") + @Size(min = 0, max = 30, message = "部门名称长度不能超过30个字符") + public String getDeptName() + { + return deptName; + } + + public void setDeptName(String deptName) + { + this.deptName = deptName; + } + + @NotNull(message = "显示顺序不能为空") + public Integer getOrderNum() + { + return orderNum; + } + + public void setOrderNum(Integer orderNum) + { + this.orderNum = orderNum; + } + + public String getLeader() + { + return leader; + } + + public void setLeader(String leader) + { + this.leader = leader; + } + + @Size(min = 0, max = 11, message = "联系电话长度不能超过11个字符") + public String getPhone() + { + return phone; + } + + public void setPhone(String phone) + { + this.phone = phone; + } + + @Email(message = "邮箱格式不正确") + @Size(min = 0, max = 50, message = "邮箱长度不能超过50个字符") + public String getEmail() + { + return email; + } + + public void setEmail(String email) + { + this.email = email; + } + + public String getStatus() + { + return status; + } + + public void setStatus(String status) + { + this.status = status; + } + + public String getDelFlag() + { + return delFlag; + } + + public void setDelFlag(String delFlag) + { + this.delFlag = delFlag; + } + + public String getParentName() + { + return parentName; + } + + public void setParentName(String parentName) + { + this.parentName = parentName; + } + + public List getChildren() + { + return children; + } + + public void setChildren(List children) + { + this.children = children; + } + + public Long getTenantId() { + return this.tenantId; + } + + public void setTenantId(Long tenantId) { + this.tenantId = tenantId; + } + + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) + .append("deptId", getDeptId()) + .append("parentId", getParentId()) + .append("ancestors", getAncestors()) + .append("deptName", getDeptName()) + .append("orderNum", getOrderNum()) + .append("leader", getLeader()) + .append("phone", getPhone()) + .append("email", getEmail()) + .append("status", getStatus()) + .append("delFlag", getDelFlag()) + .append("createBy", getCreateBy()) + .append("createTime", getCreateTime()) + .append("updateBy", getUpdateBy()) + .append("updateTime", getUpdateTime()) + .toString(); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysDictData.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysDictData.java new file mode 100644 index 00000000..0d88bbcc --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysDictData.java @@ -0,0 +1,180 @@ +package com.ruoyi.common.core.domain.entity; + +import javax.validation.constraints.NotBlank; +import javax.validation.constraints.Size; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableId; +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import com.ruoyi.common.annotation.Excel; +import com.ruoyi.common.annotation.Excel.ColumnType; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.core.domain.BaseEntity; + +/** + * 字典数据表 sys_dict_data + * + * @author ruoyi + */ +public class SysDictData extends BaseEntity +{ + private static final long serialVersionUID = 1L; + + /** 字典编码 */ + @Excel(name = "字典编码", cellType = ColumnType.NUMERIC) + @TableId(type = IdType.AUTO) + private Long dictCode; + + /** 字典排序 */ + @Excel(name = "字典排序", cellType = ColumnType.NUMERIC) + private Long dictSort; + + /** 字典标签 */ + @Excel(name = "字典标签") + private String dictLabel; + + /** 字典键值 */ + @Excel(name = "字典键值") + private String dictValue; + + /** 字典类型 */ + @Excel(name = "字典类型") + private String dictType; + + /** 样式属性(其他样式扩展) */ + private String cssClass; + + /** 表格字典样式 */ + private String listClass; + + /** 是否默认(Y是 N否) */ + @Excel(name = "是否默认", readConverterExp = "Y=是,N=否") + private String isDefault; + + /** 状态(0正常 1停用) */ + @Excel(name = "状态", readConverterExp = "0=正常,1=停用") + private String status; + + public Long getDictCode() + { + return dictCode; + } + + public void setDictCode(Long dictCode) + { + this.dictCode = dictCode; + } + + public Long getDictSort() + { + return dictSort; + } + + public void setDictSort(Long dictSort) + { + this.dictSort = dictSort; + } + + @NotBlank(message = "字典标签不能为空") + @Size(min = 0, max = 100, message = "字典标签长度不能超过100个字符") + public String getDictLabel() + { + return dictLabel; + } + + public void setDictLabel(String dictLabel) + { + this.dictLabel = dictLabel; + } + + @NotBlank(message = "字典键值不能为空") + @Size(min = 0, max = 100, message = "字典键值长度不能超过100个字符") + public String getDictValue() + { + return dictValue; + } + + public void setDictValue(String dictValue) + { + this.dictValue = dictValue; + } + + @NotBlank(message = "字典类型不能为空") + @Size(min = 0, max = 100, message = "字典类型长度不能超过100个字符") + public String getDictType() + { + return dictType; + } + + public void setDictType(String dictType) + { + this.dictType = dictType; + } + + @Size(min = 0, max = 100, message = "样式属性长度不能超过100个字符") + public String getCssClass() + { + return cssClass; + } + + public void setCssClass(String cssClass) + { + this.cssClass = cssClass; + } + + public String getListClass() + { + return listClass; + } + + public void setListClass(String listClass) + { + this.listClass = listClass; + } + + public boolean getDefault() + { + return UserConstants.YES.equals(this.isDefault); + } + + public String getIsDefault() + { + return isDefault; + } + + public void setIsDefault(String isDefault) + { + this.isDefault = isDefault; + } + + public String getStatus() + { + return status; + } + + public void setStatus(String status) + { + this.status = status; + } + + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) + .append("dictCode", getDictCode()) + .append("dictSort", getDictSort()) + .append("dictLabel", getDictLabel()) + .append("dictValue", getDictValue()) + .append("dictType", getDictType()) + .append("cssClass", getCssClass()) + .append("listClass", getListClass()) + .append("isDefault", getIsDefault()) + .append("status", getStatus()) + .append("createBy", getCreateBy()) + .append("createTime", getCreateTime()) + .append("updateBy", getUpdateBy()) + .append("updateTime", getUpdateTime()) + .append("remark", getRemark()) + .toString(); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysDictType.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysDictType.java new file mode 100644 index 00000000..4c8b7424 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysDictType.java @@ -0,0 +1,100 @@ +package com.ruoyi.common.core.domain.entity; + +import javax.validation.constraints.NotBlank; +import javax.validation.constraints.Pattern; +import javax.validation.constraints.Size; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableId; +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import com.ruoyi.common.annotation.Excel; +import com.ruoyi.common.annotation.Excel.ColumnType; +import com.ruoyi.common.core.domain.BaseEntity; + +/** + * 字典类型表 sys_dict_type + * + * @author ruoyi + */ +public class SysDictType extends BaseEntity +{ + private static final long serialVersionUID = 1L; + + /** 字典主键 */ + @Excel(name = "字典主键", cellType = ColumnType.NUMERIC) + @TableId(type = IdType.AUTO) + private Long dictId; + + /** 字典名称 */ + @Excel(name = "字典名称") + private String dictName; + + /** 字典类型 */ + @Excel(name = "字典类型") + private String dictType; + + /** 状态(0正常 1停用) */ + @Excel(name = "状态", readConverterExp = "0=正常,1=停用") + private String status; + + public Long getDictId() + { + return dictId; + } + + public void setDictId(Long dictId) + { + this.dictId = dictId; + } + + @NotBlank(message = "字典名称不能为空") + @Size(min = 0, max = 100, message = "字典类型名称长度不能超过100个字符") + public String getDictName() + { + return dictName; + } + + public void setDictName(String dictName) + { + this.dictName = dictName; + } + + @NotBlank(message = "字典类型不能为空") + @Size(min = 0, max = 100, message = "字典类型类型长度不能超过100个字符") + @Pattern(regexp = "^[a-z][a-z0-9_]*$", message = "字典类型必须以字母开头,且只能为(小写字母,数字,下滑线)") + public String getDictType() + { + return dictType; + } + + public void setDictType(String dictType) + { + this.dictType = dictType; + } + + public String getStatus() + { + return status; + } + + public void setStatus(String status) + { + this.status = status; + } + + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) + .append("dictId", getDictId()) + .append("dictName", getDictName()) + .append("dictType", getDictType()) + .append("status", getStatus()) + .append("createBy", getCreateBy()) + .append("createTime", getCreateTime()) + .append("updateBy", getUpdateBy()) + .append("updateTime", getUpdateTime()) + .append("remark", getRemark()) + .toString(); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysMenu.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysMenu.java new file mode 100644 index 00000000..44eae08f --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysMenu.java @@ -0,0 +1,252 @@ +package com.ruoyi.common.core.domain.entity; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableField; +import com.baomidou.mybatisplus.annotation.TableId; +import com.ruoyi.common.core.domain.BaseEntity; +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; + +import javax.validation.constraints.NotBlank; +import javax.validation.constraints.NotNull; +import javax.validation.constraints.Size; +import java.util.ArrayList; +import java.util.List; + +/** + * 菜单权限表 sys_menu + * + * @author ruoyi + */ +public class SysMenu extends BaseEntity { + private static final long serialVersionUID = 1L; + + /** + * 菜单ID + */ + @TableId(type = IdType.AUTO) + private Long menuId; + + /** + * 菜单名称 + */ + private String menuName; + + /** + * 父菜单ID + */ + private Long parentId; + + /** + * 显示顺序 + */ + private Integer orderNum; + + /** + * 路由地址 + */ + private String path; + + /** + * 组件路径 + */ + private String component; + + /** + * 路由参数 + */ + private String query; + + /** + * 是否为外链(0是 1否) + */ + private String isFrame; + + /** + * 是否缓存(0缓存 1不缓存) + */ + private String isCache; + + /** + * 类型(M目录 C菜单 F按钮) + */ + private String menuType; + + /** + * 显示状态(0显示 1隐藏) + */ + private String visible; + + /** + * 菜单状态(0正常 1停用) + */ + private String status; + + /** + * 权限字符串 + */ + private String perms; + + /** + * 菜单图标 + */ + private String icon; + + /** + * 子菜单 + */ + @TableField(exist = false) + private List children = new ArrayList(); + + public Long getMenuId() { + return menuId; + } + + public void setMenuId(Long menuId) { + this.menuId = menuId; + } + + @NotBlank(message = "菜单名称不能为空") + @Size(min = 0, max = 50, message = "菜单名称长度不能超过50个字符") + public String getMenuName() { + return menuName; + } + + public void setMenuName(String menuName) { + this.menuName = menuName; + } + + + public Long getParentId() { + return parentId; + } + + public void setParentId(Long parentId) { + this.parentId = parentId; + } + + @NotNull(message = "显示顺序不能为空") + public Integer getOrderNum() { + return orderNum; + } + + public void setOrderNum(Integer orderNum) { + this.orderNum = orderNum; + } + + @Size(min = 0, max = 200, message = "路由地址不能超过200个字符") + public String getPath() { + return path; + } + + public void setPath(String path) { + this.path = path; + } + + @Size(min = 0, max = 200, message = "组件路径不能超过255个字符") + public String getComponent() { + return component; + } + + public void setComponent(String component) { + this.component = component; + } + + public String getQuery() { + return query; + } + + public void setQuery(String query) { + this.query = query; + } + + public String getIsFrame() { + return isFrame; + } + + public void setIsFrame(String isFrame) { + this.isFrame = isFrame; + } + + public String getIsCache() { + return isCache; + } + + public void setIsCache(String isCache) { + this.isCache = isCache; + } + + @NotBlank(message = "菜单类型不能为空") + public String getMenuType() { + return menuType; + } + + public void setMenuType(String menuType) { + this.menuType = menuType; + } + + public String getVisible() { + return visible; + } + + public void setVisible(String visible) { + this.visible = visible; + } + + public String getStatus() { + return status; + } + + public void setStatus(String status) { + this.status = status; + } + + @Size(min = 0, max = 100, message = "权限标识长度不能超过100个字符") + public String getPerms() { + return perms; + } + + public void setPerms(String perms) { + this.perms = perms; + } + + public String getIcon() { + return icon; + } + + public void setIcon(String icon) { + this.icon = icon; + } + + public List getChildren() { + return children; + } + + public void setChildren(List children) { + this.children = children; + } + + @Override + public String toString() { + return new ToStringBuilder(this, ToStringStyle.MULTI_LINE_STYLE) + .append("menuId", getMenuId()) + .append("menuName", getMenuName()) + .append("parentId", getParentId()) + .append("orderNum", getOrderNum()) + .append("path", getPath()) + .append("component", getComponent()) + .append("isFrame", getIsFrame()) + .append("IsCache", getIsCache()) + .append("menuType", getMenuType()) + .append("visible", getVisible()) + .append("status ", getStatus()) + .append("perms", getPerms()) + .append("icon", getIcon()) + .append("createBy", getCreateBy()) + .append("createTime", getCreateTime()) + .append("updateBy", getUpdateBy()) + .append("updateTime", getUpdateTime()) + .append("remark", getRemark()) + .toString(); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysRole.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysRole.java new file mode 100644 index 00000000..bf07e8d1 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysRole.java @@ -0,0 +1,261 @@ +package com.ruoyi.common.core.domain.entity; + +import javax.validation.constraints.NotBlank; +import javax.validation.constraints.Size; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableField; +import com.baomidou.mybatisplus.annotation.TableId; +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import com.ruoyi.common.annotation.Excel; +import com.ruoyi.common.annotation.Excel.ColumnType; +import com.ruoyi.common.core.domain.BaseEntity; + +import java.util.Set; + +/** + * 角色表 sys_role + * + * @author ruoyi + */ +public class SysRole extends BaseEntity +{ + private static final long serialVersionUID = 1L; + + /** 角色ID */ + @Excel(name = "角色序号", cellType = ColumnType.NUMERIC) + @TableId(type = IdType.AUTO) + private Long roleId; + + /** 角色名称 */ + @Excel(name = "角色名称") + private String roleName; + + /** 角色权限 */ + @Excel(name = "角色权限") + private String roleKey; + + /** 角色排序 */ + @Excel(name = "角色排序") + private String roleSort; + + /** 数据范围(1:所有数据权限;2:自定义数据权限;3:本部门数据权限;4:本部门及以下数据权限;5:仅本人数据权限) */ + @Excel(name = "数据范围", readConverterExp = "1=所有数据权限,2=自定义数据权限,3=本部门数据权限,4=本部门及以下数据权限,5=仅本人数据权限") + private String dataScope; + + /** 菜单树选择项是否关联显示( 0:父子不互相关联显示 1:父子互相关联显示) */ + private boolean menuCheckStrictly; + + /** 部门树选择项是否关联显示(0:父子不互相关联显示 1:父子互相关联显示 ) */ + private boolean deptCheckStrictly; + + /** 角色状态(0正常 1停用) */ + @Excel(name = "角色状态", readConverterExp = "0=正常,1=停用") + private String status; + + /** 删除标志(0代表存在 2代表删除) */ + private String delFlag; + + /** 用户是否存在此角色标识 默认不存在 */ + @TableField(exist = false) + private boolean flag = false; + + /** 菜单组 */ + @TableField(exist = false) + private Long[] menuIds; + + /** 部门组(数据权限) */ + @TableField(exist = false) + private Long[] deptIds; + + /** 角色菜单权限 */ + @TableField(exist = false) + private Set permissions; + /** + * 多租户编号 + */ + private Long tenantId; + public SysRole() + { + + } + + public SysRole(Long roleId) + { + this.roleId = roleId; + } + + public Long getRoleId() + { + return roleId; + } + + public void setRoleId(Long roleId) + { + this.roleId = roleId; + } + + public boolean isAdmin() + { + return isAdmin(this.roleId); + } + + public static boolean isAdmin(Long roleId) + { + return roleId != null && 1L == roleId; + } + + @NotBlank(message = "角色名称不能为空") + @Size(min = 0, max = 30, message = "角色名称长度不能超过30个字符") + public String getRoleName() + { + return roleName; + } + + public void setRoleName(String roleName) + { + this.roleName = roleName; + } + + @NotBlank(message = "权限字符不能为空") + @Size(min = 0, max = 100, message = "权限字符长度不能超过100个字符") + public String getRoleKey() + { + return roleKey; + } + + public void setRoleKey(String roleKey) + { + this.roleKey = roleKey; + } + + @NotBlank(message = "显示顺序不能为空") + public String getRoleSort() + { + return roleSort; + } + + public void setRoleSort(String roleSort) + { + this.roleSort = roleSort; + } + + public String getDataScope() + { + return dataScope; + } + + public void setDataScope(String dataScope) + { + this.dataScope = dataScope; + } + + public boolean isMenuCheckStrictly() + { + return menuCheckStrictly; + } + + public void setMenuCheckStrictly(boolean menuCheckStrictly) + { + this.menuCheckStrictly = menuCheckStrictly; + } + + public boolean isDeptCheckStrictly() + { + return deptCheckStrictly; + } + + public void setDeptCheckStrictly(boolean deptCheckStrictly) + { + this.deptCheckStrictly = deptCheckStrictly; + } + + public String getStatus() + { + return status; + } + + public void setStatus(String status) + { + this.status = status; + } + + public String getDelFlag() + { + return delFlag; + } + + public void setDelFlag(String delFlag) + { + this.delFlag = delFlag; + } + + public boolean isFlag() + { + return flag; + } + + public void setFlag(boolean flag) + { + this.flag = flag; + } + + public Long[] getMenuIds() + { + return menuIds; + } + + public void setMenuIds(Long[] menuIds) + { + this.menuIds = menuIds; + } + + public Long[] getDeptIds() + { + return deptIds; + } + + public void setDeptIds(Long[] deptIds) + { + this.deptIds = deptIds; + } + + public Set getPermissions() + { + return permissions; + } + + public void setPermissions(Set permissions) + { + this.permissions = permissions; + } + + public Long getTenantId() { + return this.tenantId; + } + + public void setTenantId(Long tenantId) { + this.tenantId = tenantId; + } + + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) + .append("roleId", getRoleId()) + .append("roleName", getRoleName()) + .append("roleKey", getRoleKey()) + .append("roleSort", getRoleSort()) + .append("dataScope", getDataScope()) + .append("menuCheckStrictly", isMenuCheckStrictly()) + .append("deptCheckStrictly", isDeptCheckStrictly()) + .append("status", getStatus()) + .append("delFlag", getDelFlag()) + .append("createBy", getCreateBy()) + .append("createTime", getCreateTime()) + .append("updateBy", getUpdateBy()) + .append("updateTime", getUpdateTime()) + .append("remark", getRemark()) + .toString(); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysUser.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysUser.java new file mode 100644 index 00000000..de21286e --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/entity/SysUser.java @@ -0,0 +1,345 @@ +package com.ruoyi.common.core.domain.entity; + +import java.util.Date; +import java.util.List; +import javax.validation.constraints.*; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableField; +import com.baomidou.mybatisplus.annotation.TableId; +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import com.ruoyi.common.annotation.Excel; +import com.ruoyi.common.annotation.Excel.ColumnType; +import com.ruoyi.common.annotation.Excel.Type; +import com.ruoyi.common.annotation.Excels; +import com.ruoyi.common.core.domain.BaseEntity; +import com.ruoyi.common.xss.Xss; + +/** + * 用户对象 sys_user + * + * @author ruoyi + */ +public class SysUser extends BaseEntity +{ + private static final long serialVersionUID = 1L; + + /** 用户ID */ + @Excel(name = "用户序号", cellType = ColumnType.NUMERIC, prompt = "用户编号") + @TableId(type = IdType.AUTO) + private Long userId; + + /** 部门ID */ + @Excel(name = "部门编号", type = Type.IMPORT) + private Long deptId; + + /** 用户账号 */ + @Excel(name = "登录名称") + private String userName; + + /** 用户昵称 */ + @Excel(name = "用户名称") + private String nickName; + + /** 用户邮箱 */ + @Excel(name = "用户邮箱") + private String email; + + /** 手机号码 */ + @Excel(name = "手机号码") + private String phonenumber; + + /** 用户性别 */ + @Excel(name = "用户性别", readConverterExp = "0=男,1=女,2=未知") + private String sex; + + /** 用户头像 */ + private String avatar; + + /** 密码 */ + private String password; + + /** 帐号状态(0正常 1停用) */ + @Excel(name = "帐号状态", readConverterExp = "0=正常,1=停用") + private String status; + + /** 删除标志(0代表存在 2代表删除) */ + private String delFlag; + + /** 最后登录IP */ + @Excel(name = "最后登录IP", type = Type.EXPORT) + private String loginIp; + + /** 最后登录时间 */ + @Excel(name = "最后登录时间", width = 30, dateFormat = "yyyy-MM-dd HH:mm:ss", type = Type.EXPORT) + private Date loginDate; + + /** 部门对象 */ + @Excels({ + @Excel(name = "部门名称", targetAttr = "deptName", type = Type.EXPORT), + @Excel(name = "部门负责人", targetAttr = "leader", type = Type.EXPORT) + }) + @TableField(exist = false) + private SysDept dept; + + /** 角色对象 */ + @TableField(exist = false) + private List roles; + + /** 角色组 */ + @TableField(exist = false) + private Long[] roleIds; + + /** 岗位组 */ + @TableField(exist = false) + private Long[] postIds; + + /** 角色ID */ + @TableField(exist = false) + private Long roleId; + /** + * 多租户编号 + */ + private Long tenantId; + public SysUser() + { + + } + + public SysUser(Long userId) + { + this.userId = userId; + } + + public Long getUserId() + { + return userId; + } + + public void setUserId(Long userId) + { + this.userId = userId; + } + + public boolean isAdmin() + { + return isAdmin(this.userId); + } + + public static boolean isAdmin(Long userId) + { + return userId != null && 1L == userId; + } + + public Long getDeptId() + { + return deptId; + } + + public void setDeptId(Long deptId) + { + this.deptId = deptId; + } + + @Xss(message = "用户昵称不能包含脚本字符") + @Size(min = 0, max = 30, message = "用户昵称长度不能超过30个字符") + public String getNickName() + { + return nickName; + } + + public void setNickName(String nickName) + { + this.nickName = nickName; + } + + @Xss(message = "用户账号不能包含脚本字符") + @NotBlank(message = "用户账号不能为空") + @Size(min = 0, max = 30, message = "用户账号长度不能超过30个字符") + public String getUserName() + { + return userName; + } + + public void setUserName(String userName) + { + this.userName = userName; + } + + @Email(message = "邮箱格式不正确") + @Size(min = 0, max = 50, message = "邮箱长度不能超过50个字符") + public String getEmail() + { + return email; + } + + public void setEmail(String email) + { + this.email = email; + } + + @Size(min = 0, max = 11, message = "手机号码长度不能超过11个字符") + public String getPhonenumber() + { + return phonenumber; + } + + public void setPhonenumber(String phonenumber) + { + this.phonenumber = phonenumber; + } + + public String getSex() + { + return sex; + } + + public void setSex(String sex) + { + this.sex = sex; + } + + public String getAvatar() + { + return avatar; + } + + public void setAvatar(String avatar) + { + this.avatar = avatar; + } + + public String getPassword() + { + return password; + } + + public void setPassword(String password) + { + this.password = password; + } + + public String getStatus() + { + return status; + } + + public void setStatus(String status) + { + this.status = status; + } + + public String getDelFlag() + { + return delFlag; + } + + public void setDelFlag(String delFlag) + { + this.delFlag = delFlag; + } + + public String getLoginIp() + { + return loginIp; + } + + public void setLoginIp(String loginIp) + { + this.loginIp = loginIp; + } + + public Date getLoginDate() + { + return loginDate; + } + + public void setLoginDate(Date loginDate) + { + this.loginDate = loginDate; + } + + public SysDept getDept() + { + return dept; + } + + public void setDept(SysDept dept) + { + this.dept = dept; + } + + public List getRoles() + { + return roles; + } + + public void setRoles(List roles) + { + this.roles = roles; + } + + public Long[] getRoleIds() + { + return roleIds; + } + + public void setRoleIds(Long[] roleIds) + { + this.roleIds = roleIds; + } + + public Long[] getPostIds() + { + return postIds; + } + + public void setPostIds(Long[] postIds) + { + this.postIds = postIds; + } + + public Long getRoleId() + { + return roleId; + } + + public void setRoleId(Long roleId) + { + this.roleId = roleId; + } + + public Long getTenantId() { + return this.tenantId; + } + + public void setTenantId(Long tenantId) { + this.tenantId = tenantId; + } + + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) + .append("userId", getUserId()) + .append("deptId", getDeptId()) + .append("userName", getUserName()) + .append("nickName", getNickName()) + .append("email", getEmail()) + .append("phonenumber", getPhonenumber()) + .append("sex", getSex()) + .append("avatar", getAvatar()) + .append("password", getPassword()) + .append("status", getStatus()) + .append("delFlag", getDelFlag()) + .append("loginIp", getLoginIp()) + .append("loginDate", getLoginDate()) + .append("createBy", getCreateBy()) + .append("createTime", getCreateTime()) + .append("updateBy", getUpdateBy()) + .append("updateTime", getUpdateTime()) + .append("remark", getRemark()) + .append("dept", getDept()) + .toString(); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/model/LoginBody.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/model/LoginBody.java new file mode 100644 index 00000000..b5bc8c8e --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/model/LoginBody.java @@ -0,0 +1,69 @@ +package com.ruoyi.common.core.domain.model; + +/** + * 用户登录对象 + * + * @author ruoyi + */ +public class LoginBody +{ + /** + * 用户名 + */ + private String username; + + /** + * 用户密码 + */ + private String password; + + /** + * 验证码 + */ + private String code; + + /** + * 唯一标识 + */ + private String uuid; + + public String getUsername() + { + return username; + } + + public void setUsername(String username) + { + this.username = username; + } + + public String getPassword() + { + return password; + } + + public void setPassword(String password) + { + this.password = password; + } + + public String getCode() + { + return code; + } + + public void setCode(String code) + { + this.code = code; + } + + public String getUuid() + { + return uuid; + } + + public void setUuid(String uuid) + { + this.uuid = uuid; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/model/LoginUser.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/model/LoginUser.java new file mode 100644 index 00000000..79dba4e7 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/model/LoginUser.java @@ -0,0 +1,295 @@ +package com.ruoyi.common.core.domain.model; + +import com.alibaba.fastjson2.annotation.JSONField; +import com.ruoyi.common.core.domain.entity.SysUser; +import org.springframework.security.core.GrantedAuthority; +import org.springframework.security.core.userdetails.UserDetails; + +import java.util.Collection; +import java.util.List; +import java.util.Set; + +/** + * 登录用户身份权限 + * + * @author ruoyi + */ +public class LoginUser implements UserDetails +{ + private static final long serialVersionUID = 1L; + + /** + * 用户ID + */ + private Long userId; + + /** + * 部门ID + */ + private Long deptId; + + /** + * 用户唯一标识 + */ + private String token; + + /** + * 登录时间 + */ + private Long loginTime; + + /** + * 过期时间 + */ + private Long expireTime; + + /** + * 登录IP地址 + */ + private String ipaddr; + + /** + * 登录地点 + */ + private String loginLocation; + + /** + * 浏览器类型 + */ + private String browser; + + /** + * 操作系统 + */ + private String os; + + /** + * 权限列表 + */ + private Set permissions; + + /** + * 数据范围信息 + */ + private List dataScopes; + + /** + * 用户信息 + */ + private SysUser user; + /** + * 租户编号 + */ + private Long tenantId; + + public Long getUserId() + { + return userId; + } + + public void setUserId(Long userId) + { + this.userId = userId; + } + + public Long getDeptId() + { + return deptId; + } + + public void setDeptId(Long deptId) + { + this.deptId = deptId; + } + + public String getToken() + { + return token; + } + + public void setToken(String token) + { + this.token = token; + } + + public LoginUser() + { + } + + public LoginUser(SysUser user, Set permissions) + { + this.user = user; + this.permissions = permissions; + } + + public LoginUser(Long userId, Long deptId, SysUser user, Set permissions, List dataScopes,Long tenantId) + { + this.userId = userId; + this.deptId = deptId; + this.user = user; + this.permissions = permissions; + this.dataScopes = dataScopes; + this.tenantId = tenantId; + } + + @JSONField(serialize = false) + @Override + public String getPassword() + { + return user.getPassword(); + } + + @Override + public String getUsername() + { + return user.getUserName(); + } + + /** + * 账户是否未过期,过期无法验证 + */ + @JSONField(serialize = false) + @Override + public boolean isAccountNonExpired() + { + return true; + } + + /** + * 指定用户是否解锁,锁定的用户无法进行身份验证 + * + * @return + */ + @JSONField(serialize = false) + @Override + public boolean isAccountNonLocked() + { + return true; + } + + /** + * 指示是否已过期的用户的凭据(密码),过期的凭据防止认证 + * + * @return + */ + @JSONField(serialize = false) + @Override + public boolean isCredentialsNonExpired() + { + return true; + } + + /** + * 是否可用 ,禁用的用户不能身份验证 + * + * @return + */ + @JSONField(serialize = false) + @Override + public boolean isEnabled() + { + return true; + } + + public Long getLoginTime() + { + return loginTime; + } + + public void setLoginTime(Long loginTime) + { + this.loginTime = loginTime; + } + + public String getIpaddr() + { + return ipaddr; + } + + public void setIpaddr(String ipaddr) + { + this.ipaddr = ipaddr; + } + + public String getLoginLocation() + { + return loginLocation; + } + + public void setLoginLocation(String loginLocation) + { + this.loginLocation = loginLocation; + } + + public String getBrowser() + { + return browser; + } + + public void setBrowser(String browser) + { + this.browser = browser; + } + + public String getOs() + { + return os; + } + + public void setOs(String os) + { + this.os = os; + } + + public Long getExpireTime() + { + return expireTime; + } + + public void setExpireTime(Long expireTime) + { + this.expireTime = expireTime; + } + + public Set getPermissions() + { + return permissions; + } + + public void setPermissions(Set permissions) + { + this.permissions = permissions; + } + + public List getDataScopes() { + return dataScopes; + } + + public void setDataScopes(List dataScopes) { + this.dataScopes = dataScopes; + } + + public SysUser getUser() + { + return user; + } + + public void setUser(SysUser user) + { + this.user = user; + } + + public Long getTenantId() { + return this.tenantId; + } + + public void setTenantId(Long tenantId) { + this.tenantId = tenantId; + } + + @Override + public Collection getAuthorities() + { + return null; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/model/RegisterBody.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/model/RegisterBody.java new file mode 100644 index 00000000..868a1fc5 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/domain/model/RegisterBody.java @@ -0,0 +1,11 @@ +package com.ruoyi.common.core.domain.model; + +/** + * 用户注册对象 + * + * @author ruoyi + */ +public class RegisterBody extends LoginBody +{ + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/page/PageDomain.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/page/PageDomain.java new file mode 100644 index 00000000..8966cb4c --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/page/PageDomain.java @@ -0,0 +1,101 @@ +package com.ruoyi.common.core.page; + +import com.ruoyi.common.utils.StringUtils; + +/** + * 分页数据 + * + * @author ruoyi + */ +public class PageDomain +{ + /** 当前记录起始索引 */ + private Integer pageNum; + + /** 每页显示记录数 */ + private Integer pageSize; + + /** 排序列 */ + private String orderByColumn; + + /** 排序的方向desc或者asc */ + private String isAsc = "asc"; + + /** 分页参数合理化 */ + private Boolean reasonable = true; + + public String getOrderBy() + { + if (StringUtils.isEmpty(orderByColumn)) + { + return ""; + } + return StringUtils.toUnderScoreCase(orderByColumn) + " " + isAsc; + } + + public Integer getPageNum() + { + return pageNum; + } + + public void setPageNum(Integer pageNum) + { + this.pageNum = pageNum; + } + + public Integer getPageSize() + { + return pageSize; + } + + public void setPageSize(Integer pageSize) + { + this.pageSize = pageSize; + } + + public String getOrderByColumn() + { + return orderByColumn; + } + + public void setOrderByColumn(String orderByColumn) + { + this.orderByColumn = orderByColumn; + } + + public String getIsAsc() + { + return isAsc; + } + + public void setIsAsc(String isAsc) + { + if (StringUtils.isNotEmpty(isAsc)) + { + // 兼容前端排序类型 + if ("ascending".equals(isAsc)) + { + isAsc = "asc"; + } + else if ("descending".equals(isAsc)) + { + isAsc = "desc"; + } + this.isAsc = isAsc; + } + } + + public Boolean getReasonable() + { + if (StringUtils.isNull(reasonable)) + { + return Boolean.TRUE; + } + return reasonable; + } + + public void setReasonable(Boolean reasonable) + { + this.reasonable = reasonable; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/page/TableDataInfo.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/page/TableDataInfo.java new file mode 100644 index 00000000..96294659 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/page/TableDataInfo.java @@ -0,0 +1,116 @@ +package com.ruoyi.common.core.page; + +import java.io.Serializable; +import java.util.List; + +/** + * 表格分页数据对象 + * + * @author ruoyi + */ +public class TableDataInfo implements Serializable +{ + private static final long serialVersionUID = 1L; + + /** 总记录数 */ + private long total; + + /** 列表数据 */ + private List rows; + + /** 消息状态码 */ + private int code; + + /** 消息内容 */ + private String msg; + /** 当前记录起始索引 */ + private long pageNum; + + /** 每页显示记录数 */ + private long pageSize; + + /** 总页数 */ + private long totalPageNum; + /** + * 表格数据对象 + */ + public TableDataInfo() + { + } + + /** + * 分页 + * + * @param list 列表数据 + * @param total 总记录数 + */ + public TableDataInfo(List list, int total) + { + this.rows = list; + this.total = total; + } + + public long getTotal() + { + return total; + } + + public void setTotal(long total) + { + this.total = total; + } + + public List getRows() + { + return rows; + } + + public void setRows(List rows) + { + this.rows = rows; + } + + public int getCode() + { + return code; + } + + public void setCode(int code) + { + this.code = code; + } + + public String getMsg() + { + return msg; + } + + public void setMsg(String msg) + { + this.msg = msg; + } + + public long getPageNum() { + return pageNum; + } + + public void setPageNum(long pageNum) { + this.pageNum = pageNum; + } + + public long getPageSize() { + return pageSize; + } + + public void setPageSize(long pageSize) { + this.pageSize = pageSize; + } + + public long getTotalPageNum() { + return totalPageNum; + } + + public void setTotalPageNum(long totalPageNum) { + this.totalPageNum = totalPageNum; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/page/TableSupport.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/page/TableSupport.java new file mode 100644 index 00000000..a120c300 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/page/TableSupport.java @@ -0,0 +1,56 @@ +package com.ruoyi.common.core.page; + +import com.ruoyi.common.core.text.Convert; +import com.ruoyi.common.utils.ServletUtils; + +/** + * 表格数据处理 + * + * @author ruoyi + */ +public class TableSupport +{ + /** + * 当前记录起始索引 + */ + public static final String PAGE_NUM = "pageNum"; + + /** + * 每页显示记录数 + */ + public static final String PAGE_SIZE = "pageSize"; + + /** + * 排序列 + */ + public static final String ORDER_BY_COLUMN = "orderByColumn"; + + /** + * 排序的方向 "desc" 或者 "asc". + */ + public static final String IS_ASC = "isAsc"; + + /** + * 分页参数合理化 + */ + public static final String REASONABLE = "reasonable"; + + /** + * 封装分页对象 + */ + public static PageDomain getPageDomain() + { + PageDomain pageDomain = new PageDomain(); + pageDomain.setPageNum(Convert.toInt(ServletUtils.getParameter(PAGE_NUM), 1)); + pageDomain.setPageSize(Convert.toInt(ServletUtils.getParameter(PAGE_SIZE), 10)); + pageDomain.setOrderByColumn(ServletUtils.getParameter(ORDER_BY_COLUMN)); + pageDomain.setIsAsc(ServletUtils.getParameter(IS_ASC)); + pageDomain.setReasonable(ServletUtils.getParameterToBool(REASONABLE)); + return pageDomain; + } + + public static PageDomain buildPageRequest() + { + return getPageDomain(); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/service/BaseServiceImpl.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/service/BaseServiceImpl.java new file mode 100644 index 00000000..7d3c417d --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/service/BaseServiceImpl.java @@ -0,0 +1,12 @@ +//package com.ruoyi.common.core.service; +// +//import com.baomidou.mybatisplus.core.mapper.BaseMapper; +//import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +//import org.springframework.beans.factory.annotation.Autowired; +// +//public class BaseServiceImpl, T> extends ServiceImpl implements IBaseService { +// @Autowired +// private BaseMapper mapper; +// +// +//} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/service/IBaseService.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/service/IBaseService.java new file mode 100644 index 00000000..592bcfa6 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/service/IBaseService.java @@ -0,0 +1,6 @@ +//package com.ruoyi.common.core.service; +// +//import com.baomidou.mybatisplus.extension.service.IService; +// +//public interface IBaseService extends IService { +//} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/text/CharsetKit.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/text/CharsetKit.java new file mode 100644 index 00000000..84124aac --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/text/CharsetKit.java @@ -0,0 +1,86 @@ +package com.ruoyi.common.core.text; + +import java.nio.charset.Charset; +import java.nio.charset.StandardCharsets; +import com.ruoyi.common.utils.StringUtils; + +/** + * 字符集工具类 + * + * @author ruoyi + */ +public class CharsetKit +{ + /** ISO-8859-1 */ + public static final String ISO_8859_1 = "ISO-8859-1"; + /** UTF-8 */ + public static final String UTF_8 = "UTF-8"; + /** GBK */ + public static final String GBK = "GBK"; + + /** ISO-8859-1 */ + public static final Charset CHARSET_ISO_8859_1 = Charset.forName(ISO_8859_1); + /** UTF-8 */ + public static final Charset CHARSET_UTF_8 = Charset.forName(UTF_8); + /** GBK */ + public static final Charset CHARSET_GBK = Charset.forName(GBK); + + /** + * 转换为Charset对象 + * + * @param charset 字符集,为空则返回默认字符集 + * @return Charset + */ + public static Charset charset(String charset) + { + return StringUtils.isEmpty(charset) ? Charset.defaultCharset() : Charset.forName(charset); + } + + /** + * 转换字符串的字符集编码 + * + * @param source 字符串 + * @param srcCharset 源字符集,默认ISO-8859-1 + * @param destCharset 目标字符集,默认UTF-8 + * @return 转换后的字符集 + */ + public static String convert(String source, String srcCharset, String destCharset) + { + return convert(source, Charset.forName(srcCharset), Charset.forName(destCharset)); + } + + /** + * 转换字符串的字符集编码 + * + * @param source 字符串 + * @param srcCharset 源字符集,默认ISO-8859-1 + * @param destCharset 目标字符集,默认UTF-8 + * @return 转换后的字符集 + */ + public static String convert(String source, Charset srcCharset, Charset destCharset) + { + if (null == srcCharset) + { + srcCharset = StandardCharsets.ISO_8859_1; + } + + if (null == destCharset) + { + destCharset = StandardCharsets.UTF_8; + } + + if (StringUtils.isEmpty(source) || srcCharset.equals(destCharset)) + { + return source; + } + return new String(source.getBytes(srcCharset), destCharset); + } + + /** + * @return 系统字符集编码 + */ + public static String systemCharset() + { + return Charset.defaultCharset().name(); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/text/Convert.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/text/Convert.java new file mode 100644 index 00000000..b82321cd --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/text/Convert.java @@ -0,0 +1,1000 @@ +package com.ruoyi.common.core.text; + +import java.math.BigDecimal; +import java.math.BigInteger; +import java.nio.ByteBuffer; +import java.nio.charset.Charset; +import java.text.NumberFormat; +import java.util.Set; +import com.ruoyi.common.utils.StringUtils; +import org.apache.commons.lang3.ArrayUtils; + +/** + * 类型转换器 + * + * @author ruoyi + */ +public class Convert +{ + /** + * 转换为字符串
+ * 如果给定的值为null,或者转换失败,返回默认值
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @param defaultValue 转换错误时的默认值 + * @return 结果 + */ + public static String toStr(Object value, String defaultValue) + { + if (null == value) + { + return defaultValue; + } + if (value instanceof String) + { + return (String) value; + } + return value.toString(); + } + + /** + * 转换为字符串
+ * 如果给定的值为null,或者转换失败,返回默认值null
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @return 结果 + */ + public static String toStr(Object value) + { + return toStr(value, null); + } + + /** + * 转换为字符
+ * 如果给定的值为null,或者转换失败,返回默认值
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @param defaultValue 转换错误时的默认值 + * @return 结果 + */ + public static Character toChar(Object value, Character defaultValue) + { + if (null == value) + { + return defaultValue; + } + if (value instanceof Character) + { + return (Character) value; + } + + final String valueStr = toStr(value, null); + return StringUtils.isEmpty(valueStr) ? defaultValue : valueStr.charAt(0); + } + + /** + * 转换为字符
+ * 如果给定的值为null,或者转换失败,返回默认值null
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @return 结果 + */ + public static Character toChar(Object value) + { + return toChar(value, null); + } + + /** + * 转换为byte
+ * 如果给定的值为null,或者转换失败,返回默认值
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @param defaultValue 转换错误时的默认值 + * @return 结果 + */ + public static Byte toByte(Object value, Byte defaultValue) + { + if (value == null) + { + return defaultValue; + } + if (value instanceof Byte) + { + return (Byte) value; + } + if (value instanceof Number) + { + return ((Number) value).byteValue(); + } + final String valueStr = toStr(value, null); + if (StringUtils.isEmpty(valueStr)) + { + return defaultValue; + } + try + { + return Byte.parseByte(valueStr); + } + catch (Exception e) + { + return defaultValue; + } + } + + /** + * 转换为byte
+ * 如果给定的值为null,或者转换失败,返回默认值null
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @return 结果 + */ + public static Byte toByte(Object value) + { + return toByte(value, null); + } + + /** + * 转换为Short
+ * 如果给定的值为null,或者转换失败,返回默认值
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @param defaultValue 转换错误时的默认值 + * @return 结果 + */ + public static Short toShort(Object value, Short defaultValue) + { + if (value == null) + { + return defaultValue; + } + if (value instanceof Short) + { + return (Short) value; + } + if (value instanceof Number) + { + return ((Number) value).shortValue(); + } + final String valueStr = toStr(value, null); + if (StringUtils.isEmpty(valueStr)) + { + return defaultValue; + } + try + { + return Short.parseShort(valueStr.trim()); + } + catch (Exception e) + { + return defaultValue; + } + } + + /** + * 转换为Short
+ * 如果给定的值为null,或者转换失败,返回默认值null
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @return 结果 + */ + public static Short toShort(Object value) + { + return toShort(value, null); + } + + /** + * 转换为Number
+ * 如果给定的值为空,或者转换失败,返回默认值
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @param defaultValue 转换错误时的默认值 + * @return 结果 + */ + public static Number toNumber(Object value, Number defaultValue) + { + if (value == null) + { + return defaultValue; + } + if (value instanceof Number) + { + return (Number) value; + } + final String valueStr = toStr(value, null); + if (StringUtils.isEmpty(valueStr)) + { + return defaultValue; + } + try + { + return NumberFormat.getInstance().parse(valueStr); + } + catch (Exception e) + { + return defaultValue; + } + } + + /** + * 转换为Number
+ * 如果给定的值为空,或者转换失败,返回默认值null
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @return 结果 + */ + public static Number toNumber(Object value) + { + return toNumber(value, null); + } + + /** + * 转换为int
+ * 如果给定的值为空,或者转换失败,返回默认值
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @param defaultValue 转换错误时的默认值 + * @return 结果 + */ + public static Integer toInt(Object value, Integer defaultValue) + { + if (value == null) + { + return defaultValue; + } + if (value instanceof Integer) + { + return (Integer) value; + } + if (value instanceof Number) + { + return ((Number) value).intValue(); + } + final String valueStr = toStr(value, null); + if (StringUtils.isEmpty(valueStr)) + { + return defaultValue; + } + try + { + return Integer.parseInt(valueStr.trim()); + } + catch (Exception e) + { + return defaultValue; + } + } + + /** + * 转换为int
+ * 如果给定的值为null,或者转换失败,返回默认值null
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @return 结果 + */ + public static Integer toInt(Object value) + { + return toInt(value, null); + } + + /** + * 转换为Integer数组
+ * + * @param str 被转换的值 + * @return 结果 + */ + public static Integer[] toIntArray(String str) + { + return toIntArray(",", str); + } + + /** + * 转换为Long数组
+ * + * @param str 被转换的值 + * @return 结果 + */ + public static Long[] toLongArray(String str) + { + return toLongArray(",", str); + } + + /** + * 转换为Integer数组
+ * + * @param split 分隔符 + * @param split 被转换的值 + * @return 结果 + */ + public static Integer[] toIntArray(String split, String str) + { + if (StringUtils.isEmpty(str)) + { + return new Integer[] {}; + } + String[] arr = str.split(split); + final Integer[] ints = new Integer[arr.length]; + for (int i = 0; i < arr.length; i++) + { + final Integer v = toInt(arr[i], 0); + ints[i] = v; + } + return ints; + } + + /** + * 转换为Long数组
+ * + * @param split 分隔符 + * @param str 被转换的值 + * @return 结果 + */ + public static Long[] toLongArray(String split, String str) + { + if (StringUtils.isEmpty(str)) + { + return new Long[] {}; + } + String[] arr = str.split(split); + final Long[] longs = new Long[arr.length]; + for (int i = 0; i < arr.length; i++) + { + final Long v = toLong(arr[i], null); + longs[i] = v; + } + return longs; + } + + /** + * 转换为String数组
+ * + * @param str 被转换的值 + * @return 结果 + */ + public static String[] toStrArray(String str) + { + return toStrArray(",", str); + } + + /** + * 转换为String数组
+ * + * @param split 分隔符 + * @param split 被转换的值 + * @return 结果 + */ + public static String[] toStrArray(String split, String str) + { + return str.split(split); + } + + /** + * 转换为long
+ * 如果给定的值为空,或者转换失败,返回默认值
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @param defaultValue 转换错误时的默认值 + * @return 结果 + */ + public static Long toLong(Object value, Long defaultValue) + { + if (value == null) + { + return defaultValue; + } + if (value instanceof Long) + { + return (Long) value; + } + if (value instanceof Number) + { + return ((Number) value).longValue(); + } + final String valueStr = toStr(value, null); + if (StringUtils.isEmpty(valueStr)) + { + return defaultValue; + } + try + { + // 支持科学计数法 + return new BigDecimal(valueStr.trim()).longValue(); + } + catch (Exception e) + { + return defaultValue; + } + } + + /** + * 转换为long
+ * 如果给定的值为null,或者转换失败,返回默认值null
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @return 结果 + */ + public static Long toLong(Object value) + { + return toLong(value, null); + } + + /** + * 转换为double
+ * 如果给定的值为空,或者转换失败,返回默认值
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @param defaultValue 转换错误时的默认值 + * @return 结果 + */ + public static Double toDouble(Object value, Double defaultValue) + { + if (value == null) + { + return defaultValue; + } + if (value instanceof Double) + { + return (Double) value; + } + if (value instanceof Number) + { + return ((Number) value).doubleValue(); + } + final String valueStr = toStr(value, null); + if (StringUtils.isEmpty(valueStr)) + { + return defaultValue; + } + try + { + // 支持科学计数法 + return new BigDecimal(valueStr.trim()).doubleValue(); + } + catch (Exception e) + { + return defaultValue; + } + } + + /** + * 转换为double
+ * 如果给定的值为空,或者转换失败,返回默认值null
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @return 结果 + */ + public static Double toDouble(Object value) + { + return toDouble(value, null); + } + + /** + * 转换为Float
+ * 如果给定的值为空,或者转换失败,返回默认值
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @param defaultValue 转换错误时的默认值 + * @return 结果 + */ + public static Float toFloat(Object value, Float defaultValue) + { + if (value == null) + { + return defaultValue; + } + if (value instanceof Float) + { + return (Float) value; + } + if (value instanceof Number) + { + return ((Number) value).floatValue(); + } + final String valueStr = toStr(value, null); + if (StringUtils.isEmpty(valueStr)) + { + return defaultValue; + } + try + { + return Float.parseFloat(valueStr.trim()); + } + catch (Exception e) + { + return defaultValue; + } + } + + /** + * 转换为Float
+ * 如果给定的值为空,或者转换失败,返回默认值null
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @return 结果 + */ + public static Float toFloat(Object value) + { + return toFloat(value, null); + } + + /** + * 转换为boolean
+ * String支持的值为:true、false、yes、ok、no,1,0 如果给定的值为空,或者转换失败,返回默认值
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @param defaultValue 转换错误时的默认值 + * @return 结果 + */ + public static Boolean toBool(Object value, Boolean defaultValue) + { + if (value == null) + { + return defaultValue; + } + if (value instanceof Boolean) + { + return (Boolean) value; + } + String valueStr = toStr(value, null); + if (StringUtils.isEmpty(valueStr)) + { + return defaultValue; + } + valueStr = valueStr.trim().toLowerCase(); + switch (valueStr) + { + case "true": + case "yes": + case "ok": + case "1": + return true; + case "false": + case "no": + case "0": + return false; + default: + return defaultValue; + } + } + + /** + * 转换为boolean
+ * 如果给定的值为空,或者转换失败,返回默认值null
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @return 结果 + */ + public static Boolean toBool(Object value) + { + return toBool(value, null); + } + + /** + * 转换为Enum对象
+ * 如果给定的值为空,或者转换失败,返回默认值
+ * + * @param clazz Enum的Class + * @param value 值 + * @param defaultValue 默认值 + * @return Enum + */ + public static > E toEnum(Class clazz, Object value, E defaultValue) + { + if (value == null) + { + return defaultValue; + } + if (clazz.isAssignableFrom(value.getClass())) + { + @SuppressWarnings("unchecked") + E myE = (E) value; + return myE; + } + final String valueStr = toStr(value, null); + if (StringUtils.isEmpty(valueStr)) + { + return defaultValue; + } + try + { + return Enum.valueOf(clazz, valueStr); + } + catch (Exception e) + { + return defaultValue; + } + } + + /** + * 转换为Enum对象
+ * 如果给定的值为空,或者转换失败,返回默认值null
+ * + * @param clazz Enum的Class + * @param value 值 + * @return Enum + */ + public static > E toEnum(Class clazz, Object value) + { + return toEnum(clazz, value, null); + } + + /** + * 转换为BigInteger
+ * 如果给定的值为空,或者转换失败,返回默认值
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @param defaultValue 转换错误时的默认值 + * @return 结果 + */ + public static BigInteger toBigInteger(Object value, BigInteger defaultValue) + { + if (value == null) + { + return defaultValue; + } + if (value instanceof BigInteger) + { + return (BigInteger) value; + } + if (value instanceof Long) + { + return BigInteger.valueOf((Long) value); + } + final String valueStr = toStr(value, null); + if (StringUtils.isEmpty(valueStr)) + { + return defaultValue; + } + try + { + return new BigInteger(valueStr); + } + catch (Exception e) + { + return defaultValue; + } + } + + /** + * 转换为BigInteger
+ * 如果给定的值为空,或者转换失败,返回默认值null
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @return 结果 + */ + public static BigInteger toBigInteger(Object value) + { + return toBigInteger(value, null); + } + + /** + * 转换为BigDecimal
+ * 如果给定的值为空,或者转换失败,返回默认值
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @param defaultValue 转换错误时的默认值 + * @return 结果 + */ + public static BigDecimal toBigDecimal(Object value, BigDecimal defaultValue) + { + if (value == null) + { + return defaultValue; + } + if (value instanceof BigDecimal) + { + return (BigDecimal) value; + } + if (value instanceof Long) + { + return new BigDecimal((Long) value); + } + if (value instanceof Double) + { + return new BigDecimal((Double) value); + } + if (value instanceof Integer) + { + return new BigDecimal((Integer) value); + } + final String valueStr = toStr(value, null); + if (StringUtils.isEmpty(valueStr)) + { + return defaultValue; + } + try + { + return new BigDecimal(valueStr); + } + catch (Exception e) + { + return defaultValue; + } + } + + /** + * 转换为BigDecimal
+ * 如果给定的值为空,或者转换失败,返回默认值
+ * 转换失败不会报错 + * + * @param value 被转换的值 + * @return 结果 + */ + public static BigDecimal toBigDecimal(Object value) + { + return toBigDecimal(value, null); + } + + /** + * 将对象转为字符串
+ * 1、Byte数组和ByteBuffer会被转换为对应字符串的数组 2、对象数组会调用Arrays.toString方法 + * + * @param obj 对象 + * @return 字符串 + */ + public static String utf8Str(Object obj) + { + return str(obj, CharsetKit.CHARSET_UTF_8); + } + + /** + * 将对象转为字符串
+ * 1、Byte数组和ByteBuffer会被转换为对应字符串的数组 2、对象数组会调用Arrays.toString方法 + * + * @param obj 对象 + * @param charsetName 字符集 + * @return 字符串 + */ + public static String str(Object obj, String charsetName) + { + return str(obj, Charset.forName(charsetName)); + } + + /** + * 将对象转为字符串
+ * 1、Byte数组和ByteBuffer会被转换为对应字符串的数组 2、对象数组会调用Arrays.toString方法 + * + * @param obj 对象 + * @param charset 字符集 + * @return 字符串 + */ + public static String str(Object obj, Charset charset) + { + if (null == obj) + { + return null; + } + + if (obj instanceof String) + { + return (String) obj; + } + else if (obj instanceof byte[]) + { + return str((byte[]) obj, charset); + } + else if (obj instanceof Byte[]) + { + byte[] bytes = ArrayUtils.toPrimitive((Byte[]) obj); + return str(bytes, charset); + } + else if (obj instanceof ByteBuffer) + { + return str((ByteBuffer) obj, charset); + } + return obj.toString(); + } + + /** + * 将byte数组转为字符串 + * + * @param bytes byte数组 + * @param charset 字符集 + * @return 字符串 + */ + public static String str(byte[] bytes, String charset) + { + return str(bytes, StringUtils.isEmpty(charset) ? Charset.defaultCharset() : Charset.forName(charset)); + } + + /** + * 解码字节码 + * + * @param data 字符串 + * @param charset 字符集,如果此字段为空,则解码的结果取决于平台 + * @return 解码后的字符串 + */ + public static String str(byte[] data, Charset charset) + { + if (data == null) + { + return null; + } + + if (null == charset) + { + return new String(data); + } + return new String(data, charset); + } + + /** + * 将编码的byteBuffer数据转换为字符串 + * + * @param data 数据 + * @param charset 字符集,如果为空使用当前系统字符集 + * @return 字符串 + */ + public static String str(ByteBuffer data, String charset) + { + if (data == null) + { + return null; + } + + return str(data, Charset.forName(charset)); + } + + /** + * 将编码的byteBuffer数据转换为字符串 + * + * @param data 数据 + * @param charset 字符集,如果为空使用当前系统字符集 + * @return 字符串 + */ + public static String str(ByteBuffer data, Charset charset) + { + if (null == charset) + { + charset = Charset.defaultCharset(); + } + return charset.decode(data).toString(); + } + + // ----------------------------------------------------------------------- 全角半角转换 + /** + * 半角转全角 + * + * @param input String. + * @return 全角字符串. + */ + public static String toSBC(String input) + { + return toSBC(input, null); + } + + /** + * 半角转全角 + * + * @param input String + * @param notConvertSet 不替换的字符集合 + * @return 全角字符串. + */ + public static String toSBC(String input, Set notConvertSet) + { + char[] c = input.toCharArray(); + for (int i = 0; i < c.length; i++) + { + if (null != notConvertSet && notConvertSet.contains(c[i])) + { + // 跳过不替换的字符 + continue; + } + + if (c[i] == ' ') + { + c[i] = '\u3000'; + } + else if (c[i] < '\177') + { + c[i] = (char) (c[i] + 65248); + + } + } + return new String(c); + } + + /** + * 全角转半角 + * + * @param input String. + * @return 半角字符串 + */ + public static String toDBC(String input) + { + return toDBC(input, null); + } + + /** + * 替换全角为半角 + * + * @param text 文本 + * @param notConvertSet 不替换的字符集合 + * @return 替换后的字符 + */ + public static String toDBC(String text, Set notConvertSet) + { + char[] c = text.toCharArray(); + for (int i = 0; i < c.length; i++) + { + if (null != notConvertSet && notConvertSet.contains(c[i])) + { + // 跳过不替换的字符 + continue; + } + + if (c[i] == '\u3000') + { + c[i] = ' '; + } + else if (c[i] > '\uFF00' && c[i] < '\uFF5F') + { + c[i] = (char) (c[i] - 65248); + } + } + String returnString = new String(c); + + return returnString; + } + + /** + * 数字金额大写转换 先写个完整的然后将如零拾替换成零 + * + * @param n 数字 + * @return 中文大写数字 + */ + public static String digitUppercase(double n) + { + String[] fraction = { "角", "分" }; + String[] digit = { "零", "壹", "贰", "叁", "肆", "伍", "陆", "柒", "捌", "玖" }; + String[][] unit = { { "元", "万", "亿" }, { "", "拾", "佰", "仟" } }; + + String head = n < 0 ? "负" : ""; + n = Math.abs(n); + + String s = ""; + for (int i = 0; i < fraction.length; i++) + { + s += (digit[(int) (Math.floor(n * 10 * Math.pow(10, i)) % 10)] + fraction[i]).replaceAll("(零.)+", ""); + } + if (s.length() < 1) + { + s = "整"; + } + int integerPart = (int) Math.floor(n); + + for (int i = 0; i < unit[0].length && integerPart > 0; i++) + { + String p = ""; + for (int j = 0; j < unit[1].length && n > 0; j++) + { + p = digit[integerPart % 10] + unit[1][j] + p; + integerPart = integerPart / 10; + } + s = p.replaceAll("(零.)*零$", "").replaceAll("^$", "零") + unit[0][i] + s; + } + return head + s.replaceAll("(零.)*零元", "元").replaceFirst("(零.)+", "").replaceAll("(零.)+", "零").replaceAll("^整$", "零元整"); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/core/text/StrFormatter.java b/ruoyi-common/src/main/java/com/ruoyi/common/core/text/StrFormatter.java new file mode 100644 index 00000000..c78ac776 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/core/text/StrFormatter.java @@ -0,0 +1,92 @@ +package com.ruoyi.common.core.text; + +import com.ruoyi.common.utils.StringUtils; + +/** + * 字符串格式化 + * + * @author ruoyi + */ +public class StrFormatter +{ + public static final String EMPTY_JSON = "{}"; + public static final char C_BACKSLASH = '\\'; + public static final char C_DELIM_START = '{'; + public static final char C_DELIM_END = '}'; + + /** + * 格式化字符串
+ * 此方法只是简单将占位符 {} 按照顺序替换为参数
+ * 如果想输出 {} 使用 \\转义 { 即可,如果想输出 {} 之前的 \ 使用双转义符 \\\\ 即可
+ * 例:
+ * 通常使用:format("this is {} for {}", "a", "b") -> this is a for b
+ * 转义{}: format("this is \\{} for {}", "a", "b") -> this is \{} for a
+ * 转义\: format("this is \\\\{} for {}", "a", "b") -> this is \a for b
+ * + * @param strPattern 字符串模板 + * @param argArray 参数列表 + * @return 结果 + */ + public static String format(final String strPattern, final Object... argArray) + { + if (StringUtils.isEmpty(strPattern) || StringUtils.isEmpty(argArray)) + { + return strPattern; + } + final int strPatternLength = strPattern.length(); + + // 初始化定义好的长度以获得更好的性能 + StringBuilder sbuf = new StringBuilder(strPatternLength + 50); + + int handledPosition = 0; + int delimIndex;// 占位符所在位置 + for (int argIndex = 0; argIndex < argArray.length; argIndex++) + { + delimIndex = strPattern.indexOf(EMPTY_JSON, handledPosition); + if (delimIndex == -1) + { + if (handledPosition == 0) + { + return strPattern; + } + else + { // 字符串模板剩余部分不再包含占位符,加入剩余部分后返回结果 + sbuf.append(strPattern, handledPosition, strPatternLength); + return sbuf.toString(); + } + } + else + { + if (delimIndex > 0 && strPattern.charAt(delimIndex - 1) == C_BACKSLASH) + { + if (delimIndex > 1 && strPattern.charAt(delimIndex - 2) == C_BACKSLASH) + { + // 转义符之前还有一个转义符,占位符依旧有效 + sbuf.append(strPattern, handledPosition, delimIndex - 1); + sbuf.append(Convert.utf8Str(argArray[argIndex])); + handledPosition = delimIndex + 2; + } + else + { + // 占位符被转义 + argIndex--; + sbuf.append(strPattern, handledPosition, delimIndex - 1); + sbuf.append(C_DELIM_START); + handledPosition = delimIndex + 1; + } + } + else + { + // 正常占位符 + sbuf.append(strPattern, handledPosition, delimIndex); + sbuf.append(Convert.utf8Str(argArray[argIndex])); + handledPosition = delimIndex + 2; + } + } + } + // 加入最后一个占位符后所有的字符 + sbuf.append(strPattern, handledPosition, strPattern.length()); + + return sbuf.toString(); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/enums/BusinessStatus.java b/ruoyi-common/src/main/java/com/ruoyi/common/enums/BusinessStatus.java new file mode 100644 index 00000000..10b7306f --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/enums/BusinessStatus.java @@ -0,0 +1,20 @@ +package com.ruoyi.common.enums; + +/** + * 操作状态 + * + * @author ruoyi + * + */ +public enum BusinessStatus +{ + /** + * 成功 + */ + SUCCESS, + + /** + * 失败 + */ + FAIL, +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/enums/BusinessType.java b/ruoyi-common/src/main/java/com/ruoyi/common/enums/BusinessType.java new file mode 100644 index 00000000..2e17c4a5 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/enums/BusinessType.java @@ -0,0 +1,59 @@ +package com.ruoyi.common.enums; + +/** + * 业务操作类型 + * + * @author ruoyi + */ +public enum BusinessType +{ + /** + * 其它 + */ + OTHER, + + /** + * 新增 + */ + INSERT, + + /** + * 修改 + */ + UPDATE, + + /** + * 删除 + */ + DELETE, + + /** + * 授权 + */ + GRANT, + + /** + * 导出 + */ + EXPORT, + + /** + * 导入 + */ + IMPORT, + + /** + * 强退 + */ + FORCE, + + /** + * 生成代码 + */ + GENCODE, + + /** + * 清空数据 + */ + CLEAN, +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/enums/CommonStatusEnum.java b/ruoyi-common/src/main/java/com/ruoyi/common/enums/CommonStatusEnum.java new file mode 100644 index 00000000..707cf90e --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/enums/CommonStatusEnum.java @@ -0,0 +1,38 @@ +package com.ruoyi.common.enums; + +import com.ruoyi.common.utils.collection.IntArrayValuable; +import lombok.AllArgsConstructor; +import lombok.Getter; + +import java.util.Arrays; + +/** + * 通用状态枚举 + * + * hasPermi + */ +@Getter +@AllArgsConstructor +public enum CommonStatusEnum implements IntArrayValuable { + + ENABLE(0, "开启"), + DISABLE(1, "关闭"); + + public static final int[] ARRAYS = Arrays.stream(values()).mapToInt(CommonStatusEnum::getStatus).toArray(); + + + /** + * 状态值 + */ + private final Integer status; + /** + * 状态名 + */ + private final String name; + + @Override + public int[] array() { + return ARRAYS; + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/enums/DataSourceType.java b/ruoyi-common/src/main/java/com/ruoyi/common/enums/DataSourceType.java new file mode 100644 index 00000000..0d945be5 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/enums/DataSourceType.java @@ -0,0 +1,19 @@ +package com.ruoyi.common.enums; + +/** + * 数据源 + * + * @author ruoyi + */ +public enum DataSourceType +{ + /** + * 主库 + */ + MASTER, + + /** + * 从库 + */ + SLAVE +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/enums/HttpMethod.java b/ruoyi-common/src/main/java/com/ruoyi/common/enums/HttpMethod.java new file mode 100644 index 00000000..be6f7392 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/enums/HttpMethod.java @@ -0,0 +1,36 @@ +package com.ruoyi.common.enums; + +import java.util.HashMap; +import java.util.Map; +import org.springframework.lang.Nullable; + +/** + * 请求方式 + * + * @author ruoyi + */ +public enum HttpMethod +{ + GET, HEAD, POST, PUT, PATCH, DELETE, OPTIONS, TRACE; + + private static final Map mappings = new HashMap<>(16); + + static + { + for (HttpMethod httpMethod : values()) + { + mappings.put(httpMethod.name(), httpMethod); + } + } + + @Nullable + public static HttpMethod resolve(@Nullable String method) + { + return (method != null ? mappings.get(method) : null); + } + + public boolean matches(String method) + { + return (this == resolve(method)); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/enums/LimitType.java b/ruoyi-common/src/main/java/com/ruoyi/common/enums/LimitType.java new file mode 100644 index 00000000..c609fd8a --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/enums/LimitType.java @@ -0,0 +1,20 @@ +package com.ruoyi.common.enums; + +/** + * 限流类型 + * + * @author ruoyi + */ + +public enum LimitType +{ + /** + * 默认策略全局限流 + */ + DEFAULT, + + /** + * 根据请求者IP进行限流 + */ + IP +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/enums/OperatorType.java b/ruoyi-common/src/main/java/com/ruoyi/common/enums/OperatorType.java new file mode 100644 index 00000000..bdd143c1 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/enums/OperatorType.java @@ -0,0 +1,24 @@ +package com.ruoyi.common.enums; + +/** + * 操作人类别 + * + * @author ruoyi + */ +public enum OperatorType +{ + /** + * 其它 + */ + OTHER, + + /** + * 后台用户 + */ + MANAGE, + + /** + * 手机端用户 + */ + MOBILE +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/enums/RoleCodeEnum.java b/ruoyi-common/src/main/java/com/ruoyi/common/enums/RoleCodeEnum.java new file mode 100644 index 00000000..09ab14a3 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/enums/RoleCodeEnum.java @@ -0,0 +1,30 @@ +package com.ruoyi.common.enums; + +import com.ruoyi.common.utils.ObjectUtils; +import lombok.AllArgsConstructor; +import lombok.Getter; + +/** + * 角色标识枚举 + */ +@Getter +@AllArgsConstructor +public enum RoleCodeEnum { + + SUPER_ADMIN("admin", "超级管理员"), TENANT_ADMIN("tenant_admin", "租户管理员"), + ; + + /** + * 角色编码 + */ + private final String code; + /** + * 名字 + */ + private final String name; + + public static boolean isSuperAdmin(String code) { + return ObjectUtils.equalsAny(code, SUPER_ADMIN.getCode()); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/enums/UserStatus.java b/ruoyi-common/src/main/java/com/ruoyi/common/enums/UserStatus.java new file mode 100644 index 00000000..d7ff44a9 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/enums/UserStatus.java @@ -0,0 +1,30 @@ +package com.ruoyi.common.enums; + +/** + * 用户状态 + * + * @author ruoyi + */ +public enum UserStatus +{ + OK("0", "正常"), DISABLE("1", "停用"), DELETED("2", "删除"); + + private final String code; + private final String info; + + UserStatus(String code, String info) + { + this.code = code; + this.info = info; + } + + public String getCode() + { + return code; + } + + public String getInfo() + { + return info; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/DeviceStatusEnum.java b/ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/DeviceStatusEnum.java new file mode 100644 index 00000000..25f1c0b7 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/DeviceStatusEnum.java @@ -0,0 +1,27 @@ +package com.ruoyi.common.enums.bs; + +import lombok.AllArgsConstructor; +import lombok.Getter; + +/** + * 角色标识枚举 + */ +@Getter +@AllArgsConstructor +public enum DeviceStatusEnum { + + EXCEPTION(0, "异常"), + NORMAL(1, "正常"), + INVALID(1, "失效"), + ; + + /** + * 编码 + */ + private final Integer code; + /** + * 名称 + */ + private final String name; + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/TaskCodeEnum.java b/ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/TaskCodeEnum.java new file mode 100644 index 00000000..46806184 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/TaskCodeEnum.java @@ -0,0 +1,25 @@ +package com.ruoyi.common.enums.bs; + +import lombok.AllArgsConstructor; +import lombok.Getter; + +/** + * 任务编码 + */ +@Getter +@AllArgsConstructor +public enum TaskCodeEnum { + + POINT("point", "点位巡查"), + ; + + /** + * 编码 + */ + private final String code; + /** + * 名称 + */ + private final String name; + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/TaskExceptionStatusEnum.java b/ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/TaskExceptionStatusEnum.java new file mode 100644 index 00000000..1e55dea5 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/TaskExceptionStatusEnum.java @@ -0,0 +1,27 @@ +package com.ruoyi.common.enums.bs; + +import lombok.AllArgsConstructor; +import lombok.Getter; + +/** + * 角色标识枚举 + */ +@Getter +@AllArgsConstructor +public enum TaskExceptionStatusEnum { + + PENDING("pending", "待检"), + NORMAL("normal", "正常"), + EXCEPTION("exception", "异常"), + ; + + /** + * 编码 + */ + private final String code; + /** + * 名称 + */ + private final String name; + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/TaskSettingStatusEnum.java b/ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/TaskSettingStatusEnum.java new file mode 100644 index 00000000..18341e22 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/TaskSettingStatusEnum.java @@ -0,0 +1,26 @@ +package com.ruoyi.common.enums.bs; + +import lombok.AllArgsConstructor; +import lombok.Getter; + +/** + * 角色标识枚举 + */ +@Getter +@AllArgsConstructor +public enum TaskSettingStatusEnum { + + NOTSET("notset", "未设置"), + SETED("seted", "已设置"), + ; + + /** + * 编码 + */ + private final String code; + /** + * 名称 + */ + private final String name; + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/TaskTypeEnum.java b/ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/TaskTypeEnum.java new file mode 100644 index 00000000..75eff0b9 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/enums/bs/TaskTypeEnum.java @@ -0,0 +1,27 @@ +package com.ruoyi.common.enums.bs; + +import lombok.AllArgsConstructor; +import lombok.Getter; + +/** + * 任务编码 + */ +@Getter +@AllArgsConstructor +public enum TaskTypeEnum { + + STANDARD("standard", "标准"), + CUSTOM("custom", "定制"), + SELF("self", "自制"), + ; + + /** + * 编码 + */ + private final String code; + /** + * 名称 + */ + private final String name; + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/DemoModeException.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/DemoModeException.java new file mode 100644 index 00000000..f6ad2ab4 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/DemoModeException.java @@ -0,0 +1,15 @@ +package com.ruoyi.common.exception; + +/** + * 演示模式异常 + * + * @author ruoyi + */ +public class DemoModeException extends RuntimeException +{ + private static final long serialVersionUID = 1L; + + public DemoModeException() + { + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/ErrorCode.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/ErrorCode.java new file mode 100644 index 00000000..43920680 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/ErrorCode.java @@ -0,0 +1,26 @@ +package com.ruoyi.common.exception; + +import lombok.Data; + +/** + * 错误码对象 + * TODO 错误码设计成对象的原因,为未来的 i18 国际化做准备 + */ +@Data +public class ErrorCode { + + /** + * 错误码 + */ + private final Integer code; + /** + * 错误提示 + */ + private final String msg; + + public ErrorCode(Integer code, String message) { + this.code = code; + this.msg = message; + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/GlobalException.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/GlobalException.java new file mode 100644 index 00000000..81a71b5d --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/GlobalException.java @@ -0,0 +1,58 @@ +package com.ruoyi.common.exception; + +/** + * 全局异常 + * + * @author ruoyi + */ +public class GlobalException extends RuntimeException +{ + private static final long serialVersionUID = 1L; + + /** + * 错误提示 + */ + private String message; + + /** + * 错误明细,内部调试错误 + * + * 和 {@link CommonResult#getDetailMessage()} 一致的设计 + */ + private String detailMessage; + + /** + * 空构造方法,避免反序列化问题 + */ + public GlobalException() + { + } + + public GlobalException(String message) + { + this.message = message; + } + + public String getDetailMessage() + { + return detailMessage; + } + + public GlobalException setDetailMessage(String detailMessage) + { + this.detailMessage = detailMessage; + return this; + } + + @Override + public String getMessage() + { + return message; + } + + public GlobalException setMessage(String message) + { + this.message = message; + return this; + } +} \ No newline at end of file diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/ServiceException.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/ServiceException.java new file mode 100644 index 00000000..fcc7ab6e --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/ServiceException.java @@ -0,0 +1,74 @@ +package com.ruoyi.common.exception; + +/** + * 业务异常 + * + * @author ruoyi + */ +public final class ServiceException extends RuntimeException +{ + private static final long serialVersionUID = 1L; + + /** + * 错误码 + */ + private Integer code; + + /** + * 错误提示 + */ + private String message; + + /** + * 错误明细,内部调试错误 + * + * 和 {@link CommonResult#getDetailMessage()} 一致的设计 + */ + private String detailMessage; + + /** + * 空构造方法,避免反序列化问题 + */ + public ServiceException() + { + } + + public ServiceException(String message) + { + this.message = message; + } + + public ServiceException(String message, Integer code) + { + this.message = message; + this.code = code; + } + + public String getDetailMessage() + { + return detailMessage; + } + + @Override + public String getMessage() + { + return message; + } + + public Integer getCode() + { + return code; + } + + public ServiceException setMessage(String message) + { + this.message = message; + return this; + } + + public ServiceException setDetailMessage(String detailMessage) + { + this.detailMessage = detailMessage; + return this; + } +} \ No newline at end of file diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/UtilException.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/UtilException.java new file mode 100644 index 00000000..980fa465 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/UtilException.java @@ -0,0 +1,26 @@ +package com.ruoyi.common.exception; + +/** + * 工具类异常 + * + * @author ruoyi + */ +public class UtilException extends RuntimeException +{ + private static final long serialVersionUID = 8247610319171014183L; + + public UtilException(Throwable e) + { + super(e.getMessage(), e); + } + + public UtilException(String message) + { + super(message); + } + + public UtilException(String message, Throwable throwable) + { + super(message, throwable); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/base/BaseException.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/base/BaseException.java new file mode 100644 index 00000000..b55d72e1 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/base/BaseException.java @@ -0,0 +1,97 @@ +package com.ruoyi.common.exception.base; + +import com.ruoyi.common.utils.MessageUtils; +import com.ruoyi.common.utils.StringUtils; + +/** + * 基础异常 + * + * @author ruoyi + */ +public class BaseException extends RuntimeException +{ + private static final long serialVersionUID = 1L; + + /** + * 所属模块 + */ + private String module; + + /** + * 错误码 + */ + private String code; + + /** + * 错误码对应的参数 + */ + private Object[] args; + + /** + * 错误消息 + */ + private String defaultMessage; + + public BaseException(String module, String code, Object[] args, String defaultMessage) + { + this.module = module; + this.code = code; + this.args = args; + this.defaultMessage = defaultMessage; + } + + public BaseException(String module, String code, Object[] args) + { + this(module, code, args, null); + } + + public BaseException(String module, String defaultMessage) + { + this(module, null, null, defaultMessage); + } + + public BaseException(String code, Object[] args) + { + this(null, code, args, null); + } + + public BaseException(String defaultMessage) + { + this(null, null, null, defaultMessage); + } + + @Override + public String getMessage() + { + String message = null; + if (!StringUtils.isEmpty(code)) + { + message = MessageUtils.message(code, args); + } + if (message == null) + { + message = defaultMessage; + } + return message; + } + + public String getModule() + { + return module; + } + + public String getCode() + { + return code; + } + + public Object[] getArgs() + { + return args; + } + + public String getDefaultMessage() + { + return defaultMessage; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/file/FileException.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/file/FileException.java new file mode 100644 index 00000000..871f09b5 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/file/FileException.java @@ -0,0 +1,19 @@ +package com.ruoyi.common.exception.file; + +import com.ruoyi.common.exception.base.BaseException; + +/** + * 文件信息异常类 + * + * @author ruoyi + */ +public class FileException extends BaseException +{ + private static final long serialVersionUID = 1L; + + public FileException(String code, Object[] args) + { + super("file", code, args, null); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/file/FileNameLengthLimitExceededException.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/file/FileNameLengthLimitExceededException.java new file mode 100644 index 00000000..70e0ec9b --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/file/FileNameLengthLimitExceededException.java @@ -0,0 +1,16 @@ +package com.ruoyi.common.exception.file; + +/** + * 文件名称超长限制异常类 + * + * @author ruoyi + */ +public class FileNameLengthLimitExceededException extends FileException +{ + private static final long serialVersionUID = 1L; + + public FileNameLengthLimitExceededException(int defaultFileNameLength) + { + super("upload.filename.exceed.length", new Object[] { defaultFileNameLength }); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/file/FileSizeLimitExceededException.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/file/FileSizeLimitExceededException.java new file mode 100644 index 00000000..ec6ab054 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/file/FileSizeLimitExceededException.java @@ -0,0 +1,16 @@ +package com.ruoyi.common.exception.file; + +/** + * 文件名大小限制异常类 + * + * @author ruoyi + */ +public class FileSizeLimitExceededException extends FileException +{ + private static final long serialVersionUID = 1L; + + public FileSizeLimitExceededException(long defaultMaxSize) + { + super("upload.exceed.maxSize", new Object[] { defaultMaxSize }); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/file/InvalidExtensionException.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/file/InvalidExtensionException.java new file mode 100644 index 00000000..8b2af16c --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/file/InvalidExtensionException.java @@ -0,0 +1,81 @@ +package com.ruoyi.common.exception.file; + +import java.util.Arrays; +import org.apache.commons.fileupload.FileUploadException; + +/** + * 文件上传 误异常类 + * + * @author ruoyi + */ +public class InvalidExtensionException extends FileUploadException +{ + private static final long serialVersionUID = 1L; + + private String[] allowedExtension; + private String extension; + private String filename; + + public InvalidExtensionException(String[] allowedExtension, String extension, String filename) + { + super("文件[" + filename + "]后缀[" + extension + "]不正确,请上传" + Arrays.toString(allowedExtension) + "格式"); + this.allowedExtension = allowedExtension; + this.extension = extension; + this.filename = filename; + } + + public String[] getAllowedExtension() + { + return allowedExtension; + } + + public String getExtension() + { + return extension; + } + + public String getFilename() + { + return filename; + } + + public static class InvalidImageExtensionException extends InvalidExtensionException + { + private static final long serialVersionUID = 1L; + + public InvalidImageExtensionException(String[] allowedExtension, String extension, String filename) + { + super(allowedExtension, extension, filename); + } + } + + public static class InvalidFlashExtensionException extends InvalidExtensionException + { + private static final long serialVersionUID = 1L; + + public InvalidFlashExtensionException(String[] allowedExtension, String extension, String filename) + { + super(allowedExtension, extension, filename); + } + } + + public static class InvalidMediaExtensionException extends InvalidExtensionException + { + private static final long serialVersionUID = 1L; + + public InvalidMediaExtensionException(String[] allowedExtension, String extension, String filename) + { + super(allowedExtension, extension, filename); + } + } + + public static class InvalidVideoExtensionException extends InvalidExtensionException + { + private static final long serialVersionUID = 1L; + + public InvalidVideoExtensionException(String[] allowedExtension, String extension, String filename) + { + super(allowedExtension, extension, filename); + } + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/job/TaskException.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/job/TaskException.java new file mode 100644 index 00000000..a567b408 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/job/TaskException.java @@ -0,0 +1,34 @@ +package com.ruoyi.common.exception.job; + +/** + * 计划策略异常 + * + * @author ruoyi + */ +public class TaskException extends Exception +{ + private static final long serialVersionUID = 1L; + + private Code code; + + public TaskException(String msg, Code code) + { + this(msg, code, null); + } + + public TaskException(String msg, Code code, Exception nestedEx) + { + super(msg, nestedEx); + this.code = code; + } + + public Code getCode() + { + return code; + } + + public enum Code + { + TASK_EXISTS, NO_TASK_EXISTS, TASK_ALREADY_STARTED, UNKNOWN, CONFIG_ERROR, TASK_NODE_NOT_AVAILABLE + } +} \ No newline at end of file diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/user/CaptchaException.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/user/CaptchaException.java new file mode 100644 index 00000000..389dbc75 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/user/CaptchaException.java @@ -0,0 +1,16 @@ +package com.ruoyi.common.exception.user; + +/** + * 验证码错误异常类 + * + * @author ruoyi + */ +public class CaptchaException extends UserException +{ + private static final long serialVersionUID = 1L; + + public CaptchaException() + { + super("user.jcaptcha.error", null); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/user/CaptchaExpireException.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/user/CaptchaExpireException.java new file mode 100644 index 00000000..85f94861 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/user/CaptchaExpireException.java @@ -0,0 +1,16 @@ +package com.ruoyi.common.exception.user; + +/** + * 验证码失效异常类 + * + * @author ruoyi + */ +public class CaptchaExpireException extends UserException +{ + private static final long serialVersionUID = 1L; + + public CaptchaExpireException() + { + super("user.jcaptcha.expire", null); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/user/UserException.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/user/UserException.java new file mode 100644 index 00000000..c292d70e --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/user/UserException.java @@ -0,0 +1,18 @@ +package com.ruoyi.common.exception.user; + +import com.ruoyi.common.exception.base.BaseException; + +/** + * 用户信息异常类 + * + * @author ruoyi + */ +public class UserException extends BaseException +{ + private static final long serialVersionUID = 1L; + + public UserException(String code, Object[] args) + { + super("user", code, args, null); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/user/UserPasswordNotMatchException.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/user/UserPasswordNotMatchException.java new file mode 100644 index 00000000..a7f3e5ff --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/user/UserPasswordNotMatchException.java @@ -0,0 +1,16 @@ +package com.ruoyi.common.exception.user; + +/** + * 用户密码不正确或不符合规范异常类 + * + * @author ruoyi + */ +public class UserPasswordNotMatchException extends UserException +{ + private static final long serialVersionUID = 1L; + + public UserPasswordNotMatchException() + { + super("user.password.not.match", null); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/user/UserPasswordRetryLimitExceedException.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/user/UserPasswordRetryLimitExceedException.java new file mode 100644 index 00000000..3f95c0b1 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/user/UserPasswordRetryLimitExceedException.java @@ -0,0 +1,16 @@ +package com.ruoyi.common.exception.user; + +/** + * 用户错误最大次数异常类 + * + * @author ruoyi + */ +public class UserPasswordRetryLimitExceedException extends UserException +{ + private static final long serialVersionUID = 1L; + + public UserPasswordRetryLimitExceedException(int retryLimitCount, int lockTime) + { + super("user.password.retry.limit.exceed", new Object[] { retryLimitCount, lockTime }); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/exception/util/ServiceExceptionUtil.java b/ruoyi-common/src/main/java/com/ruoyi/common/exception/util/ServiceExceptionUtil.java new file mode 100644 index 00000000..767a62c4 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/exception/util/ServiceExceptionUtil.java @@ -0,0 +1,124 @@ +package com.ruoyi.common.exception.util; + +import com.google.common.annotations.VisibleForTesting; +import com.ruoyi.common.exception.ErrorCode; +import com.ruoyi.common.exception.ServiceException; +import lombok.extern.slf4j.Slf4j; + +import java.util.Map; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.ConcurrentMap; + +/** + * {@link ServiceException} 工具类 + *

+ * 目的在于,格式化异常信息提示。 + * 考虑到 String.format 在参数不正确时会报错,因此使用 {} 作为占位符,并使用 {@link #doFormat(int, String, Object...)} 方法来格式化 + *

+ * 因为 {@link #MESSAGES} 里面默认是没有异常信息提示的模板的,所以需要使用方自己初始化进去。目前想到的有几种方式: + *

+ * 1. 异常提示信息,写在枚举类中,例如说,cn.iocoder.oceans.user.api.constants.ErrorCodeEnum 类 + ServiceExceptionConfiguration + * 2. 异常提示信息,写在 .properties 等等配置文件 + * 3. 异常提示信息,写在 Apollo 等等配置中心中,从而实现可动态刷新 + * 4. 异常提示信息,存储在 db 等等数据库中,从而实现可动态刷新 + * + * @author wangzongrun + */ +@Slf4j +public class ServiceExceptionUtil { + + /** + * 错误码提示模板 + */ + private static final ConcurrentMap MESSAGES = new ConcurrentHashMap<>(); + + public static void putAll(Map messages) { + ServiceExceptionUtil.MESSAGES.putAll(messages); + } + + public static void put(Integer code, String message) { + ServiceExceptionUtil.MESSAGES.put(code, message); + } + + public static void delete(Integer code, String message) { + ServiceExceptionUtil.MESSAGES.remove(code, message); + } + + // ========== 和 ServiceException 的集成 ========== + + public static ServiceException exception(ErrorCode errorCode) { + String messagePattern = MESSAGES.getOrDefault(errorCode.getCode(), errorCode.getMsg()); + return exception0(errorCode.getCode(), messagePattern); + } + + public static ServiceException exception(ErrorCode errorCode, Object... params) { + String messagePattern = MESSAGES.getOrDefault(errorCode.getCode(), errorCode.getMsg()); + return exception0(errorCode.getCode(), messagePattern, params); + } + + /** + * 创建指定编号的 ServiceException 的异常 + * + * @param code 编号 + * @return 异常 + */ + public static ServiceException exception(Integer code) { + return exception0(code, MESSAGES.get(code)); + } + + /** + * 创建指定编号的 ServiceException 的异常 + * + * @param code 编号 + * @param params 消息提示的占位符对应的参数 + * @return 异常 + */ + public static ServiceException exception(Integer code, Object... params) { + return exception0(code, MESSAGES.get(code), params); + } + + public static ServiceException exception0(Integer code, String messagePattern, Object... params) { + String message = doFormat(code, messagePattern, params); + return new ServiceException(message, code); + } + + // ========== 格式化方法 ========== + + /** + * 将错误编号对应的消息使用 params 进行格式化。 + * + * @param code 错误编号 + * @param messagePattern 消息模版 + * @param params 参数 + * @return 格式化后的提示 + */ + @VisibleForTesting + public static String doFormat(int code, String messagePattern, Object... params) { + StringBuilder sbuf = new StringBuilder(messagePattern.length() + 50); + int i = 0; + int j; + int l; + for (l = 0; l < params.length; l++) { + j = messagePattern.indexOf("{}", i); + if (j == -1) { + log.error("[doFormat][参数过多:错误码({})|错误内容({})|参数({})", code, messagePattern, params); + if (i == 0) { + return messagePattern; + } else { + sbuf.append(messagePattern.substring(i)); + return sbuf.toString(); + } + } else { + sbuf.append(messagePattern, i, j); + sbuf.append(params[l]); + i = j + 2; + } + } + if (messagePattern.indexOf("{}", i) != -1) { + log.error("[doFormat][参数过少:错误码({})|错误内容({})|参数({})", code, messagePattern, params); + } + sbuf.append(messagePattern.substring(i)); + return sbuf.toString(); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/filter/ApiRequestFilter.java b/ruoyi-common/src/main/java/com/ruoyi/common/filter/ApiRequestFilter.java new file mode 100644 index 00000000..ea1bb9e8 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/filter/ApiRequestFilter.java @@ -0,0 +1,30 @@ +package com.ruoyi.common.filter; + +import cn.hutool.core.util.StrUtil; +import lombok.RequiredArgsConstructor; +import org.springframework.web.filter.OncePerRequestFilter; + +import javax.servlet.http.HttpServletRequest; + +/** + * 过滤 /admin-api、/app-api 等 API 请求的过滤器 + * + * @author 芋道源码 + */ +@RequiredArgsConstructor +public abstract class ApiRequestFilter extends OncePerRequestFilter { + + + @Override + protected boolean shouldNotFilter(HttpServletRequest request) { + if (StrUtil.startWith(request.getRequestURI(),"/profile/",false)) { + //静态资源,不需要走TenantSecurityWebFilter过滤器 + return true; + } + // 只过滤 API 请求的地址 +// return !StrUtil.startWithAny(request.getRequestURI(), webProperties.getAdminApi().getPrefix(), +// webProperties.getAppApi().getPrefix()); + return false; + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/filter/PropertyPreExcludeFilter.java b/ruoyi-common/src/main/java/com/ruoyi/common/filter/PropertyPreExcludeFilter.java new file mode 100644 index 00000000..a145789d --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/filter/PropertyPreExcludeFilter.java @@ -0,0 +1,24 @@ +package com.ruoyi.common.filter; + +import com.alibaba.fastjson2.filter.SimplePropertyPreFilter; + +/** + * 排除JSON敏感属性 + * + * @author ruoyi + */ +public class PropertyPreExcludeFilter extends SimplePropertyPreFilter +{ + public PropertyPreExcludeFilter() + { + } + + public PropertyPreExcludeFilter addExcludes(String... filters) + { + for (int i = 0; i < filters.length; i++) + { + this.getExcludes().add(filters[i]); + } + return this; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/filter/RepeatableFilter.java b/ruoyi-common/src/main/java/com/ruoyi/common/filter/RepeatableFilter.java new file mode 100644 index 00000000..a1bcfe2a --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/filter/RepeatableFilter.java @@ -0,0 +1,52 @@ +package com.ruoyi.common.filter; + +import java.io.IOException; +import javax.servlet.Filter; +import javax.servlet.FilterChain; +import javax.servlet.FilterConfig; +import javax.servlet.ServletException; +import javax.servlet.ServletRequest; +import javax.servlet.ServletResponse; +import javax.servlet.http.HttpServletRequest; +import org.springframework.http.MediaType; +import com.ruoyi.common.utils.StringUtils; + +/** + * Repeatable 过滤器 + * + * @author ruoyi + */ +public class RepeatableFilter implements Filter +{ + @Override + public void init(FilterConfig filterConfig) throws ServletException + { + + } + + @Override + public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) + throws IOException, ServletException + { + ServletRequest requestWrapper = null; + if (request instanceof HttpServletRequest + && StringUtils.startsWithIgnoreCase(request.getContentType(), MediaType.APPLICATION_JSON_VALUE)) + { + requestWrapper = new RepeatedlyRequestWrapper((HttpServletRequest) request, response); + } + if (null == requestWrapper) + { + chain.doFilter(request, response); + } + else + { + chain.doFilter(requestWrapper, response); + } + } + + @Override + public void destroy() + { + + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/filter/RepeatedlyRequestWrapper.java b/ruoyi-common/src/main/java/com/ruoyi/common/filter/RepeatedlyRequestWrapper.java new file mode 100644 index 00000000..407d1ba4 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/filter/RepeatedlyRequestWrapper.java @@ -0,0 +1,76 @@ +package com.ruoyi.common.filter; + +import java.io.BufferedReader; +import java.io.ByteArrayInputStream; +import java.io.IOException; +import java.io.InputStreamReader; +import javax.servlet.ReadListener; +import javax.servlet.ServletInputStream; +import javax.servlet.ServletResponse; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletRequestWrapper; +import com.ruoyi.common.utils.http.HttpHelper; +import com.ruoyi.common.constant.Constants; + +/** + * 构建可重复读取inputStream的request + * + * @author ruoyi + */ +public class RepeatedlyRequestWrapper extends HttpServletRequestWrapper +{ + private final byte[] body; + + public RepeatedlyRequestWrapper(HttpServletRequest request, ServletResponse response) throws IOException + { + super(request); + request.setCharacterEncoding(Constants.UTF8); + response.setCharacterEncoding(Constants.UTF8); + + body = HttpHelper.getBodyString(request).getBytes(Constants.UTF8); + } + + @Override + public BufferedReader getReader() throws IOException + { + return new BufferedReader(new InputStreamReader(getInputStream())); + } + + @Override + public ServletInputStream getInputStream() throws IOException + { + final ByteArrayInputStream bais = new ByteArrayInputStream(body); + return new ServletInputStream() + { + @Override + public int read() throws IOException + { + return bais.read(); + } + + @Override + public int available() throws IOException + { + return body.length; + } + + @Override + public boolean isFinished() + { + return false; + } + + @Override + public boolean isReady() + { + return false; + } + + @Override + public void setReadListener(ReadListener readListener) + { + + } + }; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/filter/XssFilter.java b/ruoyi-common/src/main/java/com/ruoyi/common/filter/XssFilter.java new file mode 100644 index 00000000..9052f0da --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/filter/XssFilter.java @@ -0,0 +1,75 @@ +package com.ruoyi.common.filter; + +import java.io.IOException; +import java.util.ArrayList; +import java.util.List; +import javax.servlet.Filter; +import javax.servlet.FilterChain; +import javax.servlet.FilterConfig; +import javax.servlet.ServletException; +import javax.servlet.ServletRequest; +import javax.servlet.ServletResponse; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.enums.HttpMethod; + +/** + * 防止XSS攻击的过滤器 + * + * @author ruoyi + */ +public class XssFilter implements Filter +{ + /** + * 排除链接 + */ + public List excludes = new ArrayList<>(); + + @Override + public void init(FilterConfig filterConfig) throws ServletException + { + String tempExcludes = filterConfig.getInitParameter("excludes"); + if (StringUtils.isNotEmpty(tempExcludes)) + { + String[] url = tempExcludes.split(","); + for (int i = 0; url != null && i < url.length; i++) + { + excludes.add(url[i]); + } + } + } + + @Override + public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) + throws IOException, ServletException + { + HttpServletRequest req = (HttpServletRequest) request; + HttpServletResponse resp = (HttpServletResponse) response; + if (handleExcludeURL(req, resp)) + { + chain.doFilter(request, response); + return; + } + XssHttpServletRequestWrapper xssRequest = new XssHttpServletRequestWrapper((HttpServletRequest) request); + chain.doFilter(xssRequest, response); + } + + private boolean handleExcludeURL(HttpServletRequest request, HttpServletResponse response) + { + String url = request.getServletPath(); + String method = request.getMethod(); + // GET DELETE 不过滤 + if (method == null || HttpMethod.GET.matches(method) || HttpMethod.DELETE.matches(method)) + { + return true; + } + return StringUtils.matches(url, excludes); + } + + @Override + public void destroy() + { + + } +} \ No newline at end of file diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/filter/XssHttpServletRequestWrapper.java b/ruoyi-common/src/main/java/com/ruoyi/common/filter/XssHttpServletRequestWrapper.java new file mode 100644 index 00000000..b1eeb65c --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/filter/XssHttpServletRequestWrapper.java @@ -0,0 +1,111 @@ +package com.ruoyi.common.filter; + +import java.io.ByteArrayInputStream; +import java.io.IOException; +import javax.servlet.ReadListener; +import javax.servlet.ServletInputStream; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletRequestWrapper; +import org.apache.commons.io.IOUtils; +import org.springframework.http.HttpHeaders; +import org.springframework.http.MediaType; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.html.EscapeUtil; + +/** + * XSS过滤处理 + * + * @author ruoyi + */ +public class XssHttpServletRequestWrapper extends HttpServletRequestWrapper +{ + /** + * @param request + */ + public XssHttpServletRequestWrapper(HttpServletRequest request) + { + super(request); + } + + @Override + public String[] getParameterValues(String name) + { + String[] values = super.getParameterValues(name); + if (values != null) + { + int length = values.length; + String[] escapseValues = new String[length]; + for (int i = 0; i < length; i++) + { + // 防xss攻击和过滤前后空格 + escapseValues[i] = EscapeUtil.clean(values[i]).trim(); + } + return escapseValues; + } + return super.getParameterValues(name); + } + + @Override + public ServletInputStream getInputStream() throws IOException + { + // 非json类型,直接返回 + if (!isJsonRequest()) + { + return super.getInputStream(); + } + + // 为空,直接返回 + String json = IOUtils.toString(super.getInputStream(), "utf-8"); + if (StringUtils.isEmpty(json)) + { + return super.getInputStream(); + } + + // xss过滤 + json = EscapeUtil.clean(json).trim(); + byte[] jsonBytes = json.getBytes("utf-8"); + final ByteArrayInputStream bis = new ByteArrayInputStream(jsonBytes); + return new ServletInputStream() + { + @Override + public boolean isFinished() + { + return true; + } + + @Override + public boolean isReady() + { + return true; + } + + @Override + public int available() throws IOException + { + return jsonBytes.length; + } + + @Override + public void setReadListener(ReadListener readListener) + { + } + + @Override + public int read() throws IOException + { + return bis.read(); + } + }; + } + + /** + * 是否是Json请求 + * + * @param request + */ + public boolean isJsonRequest() + { + String header = super.getHeader(HttpHeaders.CONTENT_TYPE); + return StringUtils.startsWithIgnoreCase(header, MediaType.APPLICATION_JSON_VALUE); + } +} \ No newline at end of file diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/handler/JsonLongSetTypeHandler.java b/ruoyi-common/src/main/java/com/ruoyi/common/handler/JsonLongSetTypeHandler.java new file mode 100644 index 00000000..80d546bf --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/handler/JsonLongSetTypeHandler.java @@ -0,0 +1,31 @@ +package com.ruoyi.common.handler; + +import com.baomidou.mybatisplus.extension.handlers.AbstractJsonTypeHandler; +import com.fasterxml.jackson.core.type.TypeReference; +import com.ruoyi.common.utils.JsonUtils; + +import java.util.Set; + +/** + * 参考 {@link com.baomidou.mybatisplus.extension.handlers.JacksonTypeHandler} 实现 + * 在我们将字符串反序列化为 Set 并且泛型为 Long 时,如果每个元素的数值太小,会被处理成 Integer 类型,导致可能存在隐性的 BUG。 + * + * 例如说哦,SysUserDO 的 postIds 属性 + * + * hasPermi + */ +public class JsonLongSetTypeHandler extends AbstractJsonTypeHandler { + + private static final TypeReference> typeReference = new TypeReference>(){}; + + @Override + protected Object parse(String json) { + return JsonUtils.parseObject(json, typeReference); + } + + @Override + protected String toJson(Object obj) { + return JsonUtils.toJsonString(obj); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/dataobject/BaseDO.java b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/dataobject/BaseDO.java new file mode 100644 index 00000000..43bbb1d1 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/dataobject/BaseDO.java @@ -0,0 +1,68 @@ +package com.ruoyi.common.mybatis.dataobject; + +import com.baomidou.mybatisplus.annotation.FieldFill; +import com.baomidou.mybatisplus.annotation.TableField; +import com.baomidou.mybatisplus.annotation.TableLogic; +import lombok.Data; +import org.apache.ibatis.type.JdbcType; + +import java.io.Serializable; +import java.util.Date; +import java.util.HashMap; +import java.util.Map; + +/** + * 基础实体对象 + * + * hasPermi + */ +@Data +public abstract class BaseDO implements Serializable { + + /** + * 创建时间 + */ + @TableField(fill = FieldFill.INSERT) + private Date createTime; + /** + * 最后更新时间 + */ + @TableField(fill = FieldFill.INSERT_UPDATE) + private Date updateTime; + /** + * 创建者,目前使用 SysUser 的 id 编号 + * + * 使用 String 类型的原因是,未来可能会存在非数值的情况,留好拓展性。 + */ + @TableField(fill = FieldFill.INSERT, jdbcType = JdbcType.VARCHAR) + private String creator; + /** + * 更新者,目前使用 SysUser 的 id 编号 + * + * 使用 String 类型的原因是,未来可能会存在非数值的情况,留好拓展性。 + */ + @TableField(fill = FieldFill.INSERT_UPDATE, jdbcType = JdbcType.VARCHAR) + private String updater; + /** + * 是否删除 + */ + @TableLogic + private Boolean deleted; + /** 请求参数 */ + @TableField(exist = false) + private Map params ; + + public Map getParams() + { + if (params == null) + { + params = new HashMap<>(); + } + return params; + } + + public void setParams(Map params) + { + this.params = params; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/handler/DefaultDBFieldHandler.java b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/handler/DefaultDBFieldHandler.java new file mode 100644 index 00000000..b4797ca3 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/handler/DefaultDBFieldHandler.java @@ -0,0 +1,87 @@ +package com.ruoyi.common.mybatis.handler; + +import com.baomidou.mybatisplus.core.handlers.MetaObjectHandler; +import com.ruoyi.common.mybatis.dataobject.BaseDO; +import com.ruoyi.common.utils.SecurityUtils; +import org.apache.ibatis.reflection.MetaObject; + +import java.util.Date; +import java.util.Objects; + +/** + * 通用参数填充实现类 + *

+ * 如果没有显式的对通用参数进行赋值,这里会对通用参数进行填充、赋值 + * + * @author hexiaowu + */ +public class DefaultDBFieldHandler implements MetaObjectHandler { + + @Override + public void insertFill(MetaObject metaObject) { + if (Objects.nonNull(metaObject)) { + if (metaObject.getOriginalObject() instanceof BaseDO) { + BaseDO baseDO = (BaseDO) metaObject.getOriginalObject(); + + Date current = new Date(); + // 创建时间为空,则以当前时间为插入时间 + if (Objects.isNull(baseDO.getCreateTime())) { + baseDO.setCreateTime(current); + } + // 更新时间为空,则以当前时间为更新时间 + if (Objects.isNull(baseDO.getUpdateTime())) { + baseDO.setUpdateTime(current); + } + + Long userId = SecurityUtils.getLoginUserId(); + // 当前登录用户不为空,创建人为空,则当前登录用户为创建人 + if (Objects.nonNull(userId) && Objects.isNull(baseDO.getCreator())) { + baseDO.setCreator(userId.toString()); + } + // 当前登录用户不为空,更新人为空,则当前登录用户为更新人 + if (Objects.nonNull(userId) && Objects.isNull(baseDO.getUpdater())) { + baseDO.setUpdater(userId.toString()); + } + } + //创建人 + Object createBy = getFieldValByName("createBy", metaObject); + String username = SecurityUtils.getUsernameNoExcePtion(); + if (Objects.nonNull(username) && Objects.isNull(createBy)) { + setFieldValByName("updateBy", username, metaObject); + setFieldValByName("createBy", username, metaObject); + } + if (metaObject.hasGetter("updateTime") && metaObject.getValue("updateTime") == null) { + setFieldValByName("updateTime", new Date(), metaObject); + } + //强制设置租户字段 +// String tenantId = metaObject.findProperty("tenantId", true); +// if (StrUtil.isNotBlank(tenantId)) { +// setFieldValByName(tenantId, TenantContextHolder, metaObject); +// } +// System.out.println(1); + } + } + + @Override + public void updateFill(MetaObject metaObject) { + // 更新时间为空,则以当前时间为更新时间 + Object modifyTime = getFieldValByName("updateTime", metaObject); + if (Objects.isNull(modifyTime)) { + setFieldValByName("updateTime", new Date(), metaObject); + } + + // 当前登录用户不为空,更新人为空,则当前登录用户为更新人 + Object modifier = getFieldValByName("updater", metaObject); + Long userId = SecurityUtils.getLoginUserId(); + if (Objects.nonNull(userId) && Objects.isNull(modifier)) { + setFieldValByName("updater", userId.toString(), metaObject); + } + // 当前登录用户不为空,更新人为空,则当前登录用户为更新人 + Object updateBy = getFieldValByName("updateBy", metaObject); + String username = SecurityUtils.getUsernameNoExcePtion(); + if (Objects.nonNull(username) && Objects.isNull(updateBy)) { + setFieldValByName("updateBy", username, metaObject); + } + //强制设置租户字段 + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/mapper/BaseMapperX.java b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/mapper/BaseMapperX.java new file mode 100644 index 00000000..667af281 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/mapper/BaseMapperX.java @@ -0,0 +1,94 @@ +package com.ruoyi.common.mybatis.mapper; + +import com.baomidou.mybatisplus.core.conditions.Wrapper; +import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.baomidou.mybatisplus.core.toolkit.support.SFunction; +import com.ruoyi.common.mybatis.pojo.PageParam; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.mybatis.util.MyBatisUtils; +import org.apache.ibatis.annotations.Param; + +import java.util.Collection; +import java.util.List; + +/** + * 在 MyBatis Plus 的 BaseMapper 的基础上拓展,提供更多的能力 + */ +public interface BaseMapperX extends BaseMapper { + + default PageResult selectPage(PageParam pageParam, @Param("ew") Wrapper queryWrapper) { + // MyBatis Plus 查询 + IPage mpPage = MyBatisUtils.buildPage(pageParam); + selectPage(mpPage, queryWrapper); + // 转换返回 + return new PageResult<>(mpPage.getRecords(), mpPage.getTotal()); + } + + default T selectOne(String field, Object value) { + return selectOne(new QueryWrapper().eq(field, value)); + } + + default T selectOne(SFunction field, Object value) { + return selectOne(new LambdaQueryWrapper().eq(field, value)); + } + + default T selectOne(String field1, Object value1, String field2, Object value2) { + return selectOne(new QueryWrapper().eq(field1, value1).eq(field2, value2)); + } + + default T selectOne(SFunction field1, Object value1, SFunction field2, Object value2) { + return selectOne(new LambdaQueryWrapper().eq(field1, value1).eq(field2, value2)); + } + + default Long selectCount() { + return selectCount(new QueryWrapper()); + } + + default Long selectCount(String field, Object value) { + return selectCount(new QueryWrapper().eq(field, value)); + } + + default Long selectCount(SFunction field, Object value) { + return selectCount(new LambdaQueryWrapper().eq(field, value)); + } + + default List selectList() { + return selectList(new QueryWrapper<>()); + } + + default List selectList(String field, Object value) { + return selectList(new QueryWrapper().eq(field, value)); + } + + default List selectList(SFunction field, Object value) { + return selectList(new LambdaQueryWrapper().eq(field, value)); + } + + default List selectList(String field, Collection values) { + return selectList(new QueryWrapper().in(field, values)); + } + + default List selectList(SFunction field, Collection values) { + return selectList(new LambdaQueryWrapper().in(field, values)); + } + + /** + * 逐条插入,适合少量数据插入,或者对性能要求不高的场景 + * + * 如果大量,请使用 {@link com.baomidou.mybatisplus.extension.service.impl.ServiceImpl#saveBatch(Collection)} 方法 + * 使用示例,可见 RoleMenuBatchInsertMapper、UserRoleBatchInsertMapper 类 + * + * @param entities 实体们 + */ + default void insertBatch(Collection entities) { + entities.forEach(this::insert); + } + + default void updateBatch(T update) { + update(update, new QueryWrapper<>()); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/pojo/PageParam.java b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/pojo/PageParam.java new file mode 100644 index 00000000..0d8df561 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/pojo/PageParam.java @@ -0,0 +1,30 @@ +package com.ruoyi.common.mybatis.pojo; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.Max; +import javax.validation.constraints.Min; +import javax.validation.constraints.NotNull; +import java.io.Serializable; + +@ApiModel("分页参数") +@Data +public class PageParam implements Serializable { + + private static final Integer PAGE_NO = 1; + private static final Integer PAGE_SIZE = 10; + + @ApiModelProperty(value = "页码,从 1 开始", required = true,example = "1") + @NotNull(message = "页码不能为空") + @Min(value = 1, message = "页码最小值为 1") + private Integer pageNo = PAGE_NO; + + @ApiModelProperty(value = "每页条数,最大值为 100", required = true, example = "10") + @NotNull(message = "每页条数不能为空") + @Min(value = 1, message = "每页条数最小值为 1") + @Max(value = 100, message = "每页条数最大值为 100") + private Integer pageSize = PAGE_SIZE; + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/pojo/PageResult.java b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/pojo/PageResult.java new file mode 100644 index 00000000..a20fb18f --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/pojo/PageResult.java @@ -0,0 +1,42 @@ +package com.ruoyi.common.mybatis.pojo; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.io.Serializable; +import java.util.ArrayList; +import java.util.List; + +@ApiModel("分页结果") +@Data +public final class PageResult implements Serializable { + + @ApiModelProperty(value = "数据", required = true) + private List list; + + @ApiModelProperty(value = "总量", required = true) + private Long total; + + public PageResult() { + } + + public PageResult(List list, Long total) { + this.list = list; + this.total = total; + } + + public PageResult(Long total) { + this.list = new ArrayList<>(); + this.total = total; + } + + public static PageResult empty() { + return new PageResult<>(0L); + } + + public static PageResult empty(Long total) { + return new PageResult<>(total); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/pojo/SortingField.java b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/pojo/SortingField.java new file mode 100644 index 00000000..fbe472f8 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/pojo/SortingField.java @@ -0,0 +1,56 @@ +package com.ruoyi.common.mybatis.pojo; + +import java.io.Serializable; + +/** + * 排序字段 DTO + * + * 类名加了 ing 的原因是,避免和 ES SortField 重名。 + */ +public class SortingField implements Serializable { + + /** + * 顺序 - 升序 + */ + public static final String ORDER_ASC = "asc"; + /** + * 顺序 - 降序 + */ + public static final String ORDER_DESC = "desc"; + + /** + * 字段 + */ + private String field; + /** + * 顺序 + */ + private String order; + + // 空构造方法,解决反序列化 + public SortingField() { + } + + public SortingField(String field, String order) { + this.field = field; + this.order = order; + } + + public String getField() { + return field; + } + + public SortingField setField(String field) { + this.field = field; + return this; + } + + public String getOrder() { + return order; + } + + public SortingField setOrder(String order) { + this.order = order; + return this; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/query/LambdaQueryWrapperX.java b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/query/LambdaQueryWrapperX.java new file mode 100644 index 00000000..b6abbcac --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/query/LambdaQueryWrapperX.java @@ -0,0 +1,135 @@ +package com.ruoyi.common.mybatis.query; + +import cn.hutool.core.util.ArrayUtil; +import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; +import com.baomidou.mybatisplus.core.toolkit.CollectionUtils; +import com.baomidou.mybatisplus.core.toolkit.support.SFunction; +import com.ruoyi.common.utils.collection.ArrayUtils; +import org.springframework.util.StringUtils; + +import java.util.Collection; + +/** + * 拓展 MyBatis Plus QueryWrapper 类,主要增加如下功能: + * + * 1. 拼接条件的方法,增加 xxxIfPresent 方法,用于判断值不存在的时候,不要拼接到条件中。 + * + * @param 数据类型 + */ +public class LambdaQueryWrapperX extends LambdaQueryWrapper { + + public LambdaQueryWrapperX likeIfPresent(SFunction column, String val) { + if (StringUtils.hasText(val)) { + return (LambdaQueryWrapperX) super.like(column, val); + } + return this; + } + + public LambdaQueryWrapperX inIfPresent(SFunction column, Collection values) { + if (!CollectionUtils.isEmpty(values)) { + return (LambdaQueryWrapperX) super.in(column, values); + } + return this; + } + + public LambdaQueryWrapperX inIfPresent(SFunction column, Object... values) { + if (!ArrayUtil.isEmpty(values)) { + return (LambdaQueryWrapperX) super.in(column, values); + } + return this; + } + + public LambdaQueryWrapperX eqIfPresent(SFunction column, Object val) { + if (val != null) { + return (LambdaQueryWrapperX) super.eq(column, val); + } + return this; + } + + public LambdaQueryWrapperX neIfPresent(SFunction column, Object val) { + if (val != null) { + return (LambdaQueryWrapperX) super.ne(column, val); + } + return this; + } + + public LambdaQueryWrapperX gtIfPresent(SFunction column, Object val) { + if (val != null) { + return (LambdaQueryWrapperX) super.gt(column, val); + } + return this; + } + + public LambdaQueryWrapperX geIfPresent(SFunction column, Object val) { + if (val != null) { + return (LambdaQueryWrapperX) super.ge(column, val); + } + return this; + } + + public LambdaQueryWrapperX ltIfPresent(SFunction column, Object val) { + if (val != null) { + return (LambdaQueryWrapperX) super.lt(column, val); + } + return this; + } + + public LambdaQueryWrapperX leIfPresent(SFunction column, Object val) { + if (val != null) { + return (LambdaQueryWrapperX) super.le(column, val); + } + return this; + } + + public LambdaQueryWrapperX betweenIfPresent(SFunction column, Object val1, Object val2) { + if (val1 != null && val2 != null) { + return (LambdaQueryWrapperX) super.between(column, val1, val2); + } + if (val1 != null) { + return (LambdaQueryWrapperX) ge(column, val1); + } + if (val2 != null) { + return (LambdaQueryWrapperX) le(column, val2); + } + return this; + } + + public LambdaQueryWrapperX betweenIfPresent(SFunction column, Object[] values) { + Object val1 = ArrayUtils.get(values, 0); + Object val2 = ArrayUtils.get(values, 1); + return betweenIfPresent(column, val1, val2); + } + + // ========== 重写父类方法,方便链式调用 ========== + + @Override + public LambdaQueryWrapperX eq(boolean condition, SFunction column, Object val) { + super.eq(condition, column, val); + return this; + } + + @Override + public LambdaQueryWrapperX eq(SFunction column, Object val) { + super.eq(column, val); + return this; + } + + @Override + public LambdaQueryWrapperX orderByDesc(SFunction column) { + super.orderByDesc(true, column); + return this; + } + + @Override + public LambdaQueryWrapperX last(String lastSql) { + super.last(lastSql); + return this; + } + + @Override + public LambdaQueryWrapperX in(SFunction column, Collection coll) { + super.in(column, coll); + return this; + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/query/QueryWrapperX.java b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/query/QueryWrapperX.java new file mode 100644 index 00000000..a496f0a6 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/query/QueryWrapperX.java @@ -0,0 +1,140 @@ +package com.ruoyi.common.mybatis.query; + +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.baomidou.mybatisplus.core.toolkit.ArrayUtils; +import com.baomidou.mybatisplus.core.toolkit.CollectionUtils; +import org.springframework.util.StringUtils; + +import java.util.Collection; + +/** + * 拓展 MyBatis Plus QueryWrapper 类,主要增加如下功能: + * + * 1. 拼接条件的方法,增加 xxxIfPresent 方法,用于判断值不存在的时候,不要拼接到条件中。 + * + * @param 数据类型 + */ +public class QueryWrapperX extends QueryWrapper { + + public QueryWrapperX likeIfPresent(String column, String val) { + if (StringUtils.hasText(val)) { + return (QueryWrapperX) super.like(column, val); + } + return this; + } + + public QueryWrapperX inIfPresent(String column, Collection values) { + if (!CollectionUtils.isEmpty(values)) { + return (QueryWrapperX) super.in(column, values); + } + return this; + } + + public QueryWrapperX inIfPresent(String column, Object... values) { + if (!ArrayUtils.isEmpty(values)) { + return (QueryWrapperX) super.in(column, values); + } + return this; + } + + public QueryWrapperX eqIfPresent(String column, Object val) { + if (val != null) { + return (QueryWrapperX) super.eq(column, val); + } + return this; + } + + public QueryWrapperX neIfPresent(String column, Object val) { + if (val != null) { + return (QueryWrapperX) super.ne(column, val); + } + return this; + } + + public QueryWrapperX gtIfPresent(String column, Object val) { + if (val != null) { + return (QueryWrapperX) super.gt(column, val); + } + return this; + } + + public QueryWrapperX geIfPresent(String column, Object val) { + if (val != null) { + return (QueryWrapperX) super.ge(column, val); + } + return this; + } + + public QueryWrapperX ltIfPresent(String column, Object val) { + if (val != null) { + return (QueryWrapperX) super.lt(column, val); + } + return this; + } + + public QueryWrapperX leIfPresent(String column, Object val) { + if (val != null) { + return (QueryWrapperX) super.le(column, val); + } + return this; + } + + public QueryWrapperX betweenIfPresent(String column, Object val1, Object val2) { + if (val1 != null && val2 != null) { + return (QueryWrapperX) super.between(column, val1, val2); + } + if (val1 != null) { + return (QueryWrapperX) ge(column, val1); + } + if (val2 != null) { + return (QueryWrapperX) le(column, val2); + } + return this; + } + + public QueryWrapperX betweenIfPresent(String column, Object[] values) { + if (values!= null && values.length != 0 && values[0] != null && values[1] != null) { + return (QueryWrapperX) super.between(column, values[0], values[1]); + } + if (values!= null && values.length != 0 && values[0] != null) { + return (QueryWrapperX) ge(column, values[0]); + } + if (values!= null && values.length != 0 && values[1] != null) { + return (QueryWrapperX) le(column, values[1]); + } + return this; + } + + // ========== 重写父类方法,方便链式调用 ========== + + @Override + public QueryWrapperX eq(boolean condition, String column, Object val) { + super.eq(condition, column, val); + return this; + } + + @Override + public QueryWrapperX eq(String column, Object val) { + super.eq(column, val); + return this; + } + + @Override + public QueryWrapperX orderByDesc(String column) { + super.orderByDesc(true, column); + return this; + } + + @Override + public QueryWrapperX last(String lastSql) { + super.last(lastSql); + return this; + } + + @Override + public QueryWrapperX in(String column, Collection coll) { + super.in(column, coll); + return this; + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/util/JdbcUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/util/JdbcUtils.java new file mode 100644 index 00000000..0b82336e --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/util/JdbcUtils.java @@ -0,0 +1,42 @@ +package com.ruoyi.common.mybatis.util; + +import com.baomidou.mybatisplus.annotation.DbType; + +import java.sql.Connection; +import java.sql.DriverManager; + +/** + * JDBC 工具类 + * + * hasPermi + */ +public class JdbcUtils { + + /** + * 判断连接是否正确 + * + * @param url 数据源连接 + * @param username 账号 + * @param password 密码 + * @return 是否正确 + */ + public static boolean isConnectionOK(String url, String username, String password) { + try (Connection ignored = DriverManager.getConnection(url, username, password)) { + return true; + } catch (Exception ex) { + return false; + } + } + + /** + * 获得 URL 对应的 DB 类型 + * + * @param url URL + * @return DB 类型 + */ + public static DbType getDbType(String url) { + String name = com.alibaba.druid.util.JdbcUtils.getDbType(url, null); + return DbType.getDbType(name); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/util/MyBatisUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/util/MyBatisUtils.java new file mode 100644 index 00000000..f1e91ad3 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/mybatis/util/MyBatisUtils.java @@ -0,0 +1,84 @@ +package com.ruoyi.common.mybatis.util; + +import cn.hutool.core.collection.CollectionUtil; +import com.baomidou.mybatisplus.core.metadata.OrderItem; +import com.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor; +import com.baomidou.mybatisplus.extension.plugins.inner.InnerInterceptor; +import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.ruoyi.common.mybatis.pojo.PageParam; +import com.ruoyi.common.mybatis.pojo.SortingField; +import net.sf.jsqlparser.expression.Alias; +import net.sf.jsqlparser.schema.Column; +import net.sf.jsqlparser.schema.Table; + +import java.util.ArrayList; +import java.util.Collection; +import java.util.List; +import java.util.stream.Collectors; + +/** + * MyBatis 工具类 + */ +public class MyBatisUtils { + + private static final String MYSQL_ESCAPE_CHARACTER = "`"; + + public static Page buildPage(PageParam pageParam) { + return buildPage(pageParam, null); + } + + public static Page buildPage(PageParam pageParam, Collection sortingFields) { + // 页码 + 数量 + Page page = new Page<>(pageParam.getPageNo(), pageParam.getPageSize()); + // 排序字段 + if (!CollectionUtil.isEmpty(sortingFields)) { + page.addOrder(sortingFields.stream().map(sortingField -> SortingField.ORDER_ASC.equals(sortingField.getOrder()) ? + OrderItem.asc(sortingField.getField()) : OrderItem.desc(sortingField.getField())) + .collect(Collectors.toList())); + } + return page; + } + + /** + * 将拦截器添加到链中 + * 由于 MybatisPlusInterceptor 不支持添加拦截器,所以只能全量设置 + * + * @param interceptor 链 + * @param inner 拦截器 + * @param index 位置 + */ + public static void addInterceptor(MybatisPlusInterceptor interceptor, InnerInterceptor inner, int index) { + List inners = new ArrayList<>(interceptor.getInterceptors()); + inners.add(index, inner); + interceptor.setInterceptors(inners); + } + + /** + * 获得 Table 对应的表名 + * + * 兼容 MySQL 转义表名 `t_xxx` + * + * @param table 表 + * @return 去除转移字符后的表名 + */ + public static String getTableName(Table table) { + String tableName = table.getName(); + if (tableName.startsWith(MYSQL_ESCAPE_CHARACTER) && tableName.endsWith(MYSQL_ESCAPE_CHARACTER)) { + tableName = tableName.substring(1, tableName.length() - 1); + } + return tableName; + } + + /** + * 构建 Column 对象 + * + * @param tableName 表名 + * @param tableAlias 别名 + * @param column 字段名 + * @return Column 对象 + */ + public static Column buildColumn(String tableName, Alias tableAlias, String column) { + return new Column(tableAlias != null ? tableAlias.getName() + "." + column : column); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/Arith.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/Arith.java new file mode 100644 index 00000000..b6326c2b --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/Arith.java @@ -0,0 +1,114 @@ +package com.ruoyi.common.utils; + +import java.math.BigDecimal; +import java.math.RoundingMode; + +/** + * 精确的浮点数运算 + * + * @author ruoyi + */ +public class Arith +{ + + /** 默认除法运算精度 */ + private static final int DEF_DIV_SCALE = 10; + + /** 这个类不能实例化 */ + private Arith() + { + } + + /** + * 提供精确的加法运算。 + * @param v1 被加数 + * @param v2 加数 + * @return 两个参数的和 + */ + public static double add(double v1, double v2) + { + BigDecimal b1 = new BigDecimal(Double.toString(v1)); + BigDecimal b2 = new BigDecimal(Double.toString(v2)); + return b1.add(b2).doubleValue(); + } + + /** + * 提供精确的减法运算。 + * @param v1 被减数 + * @param v2 减数 + * @return 两个参数的差 + */ + public static double sub(double v1, double v2) + { + BigDecimal b1 = new BigDecimal(Double.toString(v1)); + BigDecimal b2 = new BigDecimal(Double.toString(v2)); + return b1.subtract(b2).doubleValue(); + } + + /** + * 提供精确的乘法运算。 + * @param v1 被乘数 + * @param v2 乘数 + * @return 两个参数的积 + */ + public static double mul(double v1, double v2) + { + BigDecimal b1 = new BigDecimal(Double.toString(v1)); + BigDecimal b2 = new BigDecimal(Double.toString(v2)); + return b1.multiply(b2).doubleValue(); + } + + /** + * 提供(相对)精确的除法运算,当发生除不尽的情况时,精确到 + * 小数点以后10位,以后的数字四舍五入。 + * @param v1 被除数 + * @param v2 除数 + * @return 两个参数的商 + */ + public static double div(double v1, double v2) + { + return div(v1, v2, DEF_DIV_SCALE); + } + + /** + * 提供(相对)精确的除法运算。当发生除不尽的情况时,由scale参数指 + * 定精度,以后的数字四舍五入。 + * @param v1 被除数 + * @param v2 除数 + * @param scale 表示表示需要精确到小数点以后几位。 + * @return 两个参数的商 + */ + public static double div(double v1, double v2, int scale) + { + if (scale < 0) + { + throw new IllegalArgumentException( + "The scale must be a positive integer or zero"); + } + BigDecimal b1 = new BigDecimal(Double.toString(v1)); + BigDecimal b2 = new BigDecimal(Double.toString(v2)); + if (b1.compareTo(BigDecimal.ZERO) == 0) + { + return BigDecimal.ZERO.doubleValue(); + } + return b1.divide(b2, scale, RoundingMode.HALF_UP).doubleValue(); + } + + /** + * 提供精确的小数位四舍五入处理。 + * @param v 需要四舍五入的数字 + * @param scale 小数点后保留几位 + * @return 四舍五入后的结果 + */ + public static double round(double v, int scale) + { + if (scale < 0) + { + throw new IllegalArgumentException( + "The scale must be a positive integer or zero"); + } + BigDecimal b = new BigDecimal(Double.toString(v)); + BigDecimal one = BigDecimal.ONE; + return b.divide(one, scale, RoundingMode.HALF_UP).doubleValue(); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/DateUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/DateUtils.java new file mode 100644 index 00000000..e677b849 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/DateUtils.java @@ -0,0 +1,189 @@ +package com.ruoyi.common.utils; + +import java.lang.management.ManagementFactory; +import java.text.ParseException; +import java.text.SimpleDateFormat; +import java.time.LocalDate; +import java.time.LocalDateTime; +import java.time.LocalTime; +import java.time.ZoneId; +import java.time.ZonedDateTime; +import java.util.Date; +import org.apache.commons.lang3.time.DateFormatUtils; + +/** + * 时间工具类 + * + * @author ruoyi + */ +public class DateUtils extends org.apache.commons.lang3.time.DateUtils +{ + public static String YYYY = "yyyy"; + + public static String YYYY_MM = "yyyy-MM"; + + public static String YYYY_MM_DD = "yyyy-MM-dd"; + + public static String YYYYMMDDHHMMSS = "yyyyMMddHHmmss"; + + public static String YYYY_MM_DD_HH_MM_SS = "yyyy-MM-dd HH:mm:ss"; + + private static String[] parsePatterns = { + "yyyy-MM-dd", "yyyy-MM-dd HH:mm:ss", "yyyy-MM-dd HH:mm", "yyyy-MM", + "yyyy/MM/dd", "yyyy/MM/dd HH:mm:ss", "yyyy/MM/dd HH:mm", "yyyy/MM", + "yyyy.MM.dd", "yyyy.MM.dd HH:mm:ss", "yyyy.MM.dd HH:mm", "yyyy.MM"}; + + /** + * 获取当前Date型日期 + * + * @return Date() 当前日期 + */ + public static Date getNowDate() + { + return new Date(); + } + public static boolean isExpired(Date time) { + return System.currentTimeMillis() > time.getTime(); + } + /** + * 获取当前日期, 默认格式为yyyy-MM-dd + * + * @return String + */ + public static String getDate() + { + return dateTimeNow(YYYY_MM_DD); + } + + public static final String getTime() + { + return dateTimeNow(YYYY_MM_DD_HH_MM_SS); + } + + public static final String dateTimeNow() + { + return dateTimeNow(YYYYMMDDHHMMSS); + } + + public static final String dateTimeNow(final String format) + { + return parseDateToStr(format, new Date()); + } + + public static final String dateTime(final Date date) + { + return parseDateToStr(YYYY_MM_DD, date); + } + + public static final String parseDateToStr(final String format, final Date date) + { + return new SimpleDateFormat(format).format(date); + } + + public static final Date dateTime(final String format, final String ts) + { + try + { + return new SimpleDateFormat(format).parse(ts); + } + catch (ParseException e) + { + throw new RuntimeException(e); + } + } + + /** + * 日期路径 即年/月/日 如2018/08/08 + */ + public static final String datePath() + { + Date now = new Date(); + return DateFormatUtils.format(now, "yyyy/MM/dd"); + } + + /** + * 日期路径 即年/月/日 如20180808 + */ + public static final String dateTime() + { + Date now = new Date(); + return DateFormatUtils.format(now, "yyyyMMdd"); + } + + /** + * 日期型字符串转化为日期 格式 + */ + public static Date parseDate(Object str) + { + if (str == null) + { + return null; + } + try + { + return parseDate(str.toString(), parsePatterns); + } + catch (ParseException e) + { + return null; + } + } + + /** + * 获取服务器启动时间 + */ + public static Date getServerStartDate() + { + long time = ManagementFactory.getRuntimeMXBean().getStartTime(); + return new Date(time); + } + + /** + * 计算相差天数 + */ + public static int differentDaysByMillisecond(Date date1, Date date2) + { + return Math.abs((int) ((date2.getTime() - date1.getTime()) / (1000 * 3600 * 24))); + } + + /** + * 计算两个时间差 + */ + public static String getDatePoor(Date endDate, Date nowDate) + { + long nd = 1000 * 24 * 60 * 60; + long nh = 1000 * 60 * 60; + long nm = 1000 * 60; + // long ns = 1000; + // 获得两个时间的毫秒时间差异 + long diff = endDate.getTime() - nowDate.getTime(); + // 计算差多少天 + long day = diff / nd; + // 计算差多少小时 + long hour = diff % nd / nh; + // 计算差多少分钟 + long min = diff % nd % nh / nm; + // 计算差多少秒//输出结果 + // long sec = diff % nd % nh % nm / ns; + return day + "天" + hour + "小时" + min + "分钟"; + } + + /** + * 增加 LocalDateTime ==> Date + */ + public static Date toDate(LocalDateTime temporalAccessor) + { + ZonedDateTime zdt = temporalAccessor.atZone(ZoneId.systemDefault()); + return Date.from(zdt.toInstant()); + } + + /** + * 增加 LocalDate ==> Date + */ + public static Date toDate(LocalDate temporalAccessor) + { + LocalDateTime localDateTime = LocalDateTime.of(temporalAccessor, LocalTime.of(0, 0, 0)); + ZonedDateTime zdt = localDateTime.atZone(ZoneId.systemDefault()); + return Date.from(zdt.toInstant()); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/DictUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/DictUtils.java new file mode 100644 index 00000000..13b3a83d --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/DictUtils.java @@ -0,0 +1,200 @@ +package com.ruoyi.common.utils; + +import java.util.ArrayList; +import java.util.Collection; +import java.util.List; + +import cn.hutool.core.collection.CollUtil; +import cn.hutool.json.JSONUtil; +import com.alibaba.fastjson2.JSONArray; +import com.ruoyi.common.constant.CacheConstants; +import com.ruoyi.common.core.domain.entity.SysDictData; +import com.ruoyi.common.utils.spring.SpringUtils; +import net.sf.ehcache.CacheManager; + +/** + * 字典工具类 + * + * @author ruoyi + */ +public class DictUtils +{ + /** + * 分隔符 + */ + public static final String SEPARATOR = ","; + + /** + * 设置字典缓存 + * + * @param key 参数键 + * @param dictDatas 字典数据列表 + */ + public static void setDictCache(String key, List dictDatas) + { + EhcacheUtil.put("dict",getCacheKey(key),dictDatas); +// SpringUtils.getBean(RedisCache.class).setCacheObject(getCacheKey(key), dictDatas); + } + + /** + * 获取字典缓存 + * + * @param key 参数键 + * @return dictDatas 字典数据列表 + */ + public static List getDictCache(String key) + { + Object dict = EhcacheUtil.get("dict", getCacheKey(key)); + if (dict != null) { + JSONArray arrayCache = JSONArray.parseArray(JSONUtil.toJsonStr(dict)); + if (CollUtil.isNotEmpty(arrayCache)) { + return arrayCache.toList(SysDictData.class); + } + } +// JSONArray arrayCache = SpringUtils.getBean(RedisCache.class).getCacheObject(getCacheKey(key)); +// if (StringUtils.isNotNull(arrayCache)) +// { +// return arrayCache.toList(SysDictData.class); +// } + return new ArrayList<>(); + } + + /** + * 根据字典类型和字典值获取字典标签 + * + * @param dictType 字典类型 + * @param dictValue 字典值 + * @return 字典标签 + */ + public static String getDictLabel(String dictType, String dictValue) + { + return getDictLabel(dictType, dictValue, SEPARATOR); + } + + /** + * 根据字典类型和字典标签获取字典值 + * + * @param dictType 字典类型 + * @param dictLabel 字典标签 + * @return 字典值 + */ + public static String getDictValue(String dictType, String dictLabel) + { + return getDictValue(dictType, dictLabel, SEPARATOR); + } + + /** + * 根据字典类型和字典值获取字典标签 + * + * @param dictType 字典类型 + * @param dictValue 字典值 + * @param separator 分隔符 + * @return 字典标签 + */ + public static String getDictLabel(String dictType, String dictValue, String separator) + { + StringBuilder propertyString = new StringBuilder(); + List datas = getDictCache(dictType); + + if (StringUtils.isNotNull(datas)) + { + if (StringUtils.containsAny(separator, dictValue)) + { + for (SysDictData dict : datas) + { + for (String value : dictValue.split(separator)) + { + if (value.equals(dict.getDictValue())) + { + propertyString.append(dict.getDictLabel()).append(separator); + break; + } + } + } + } + else + { + for (SysDictData dict : datas) + { + if (dictValue.equals(dict.getDictValue())) + { + return dict.getDictLabel(); + } + } + } + } + return StringUtils.stripEnd(propertyString.toString(), separator); + } + + /** + * 根据字典类型和字典标签获取字典值 + * + * @param dictType 字典类型 + * @param dictLabel 字典标签 + * @param separator 分隔符 + * @return 字典值 + */ + public static String getDictValue(String dictType, String dictLabel, String separator) + { + StringBuilder propertyString = new StringBuilder(); + List datas = getDictCache(dictType); + + if (StringUtils.containsAny(separator, dictLabel) && StringUtils.isNotEmpty(datas)) + { + for (SysDictData dict : datas) + { + for (String label : dictLabel.split(separator)) + { + if (label.equals(dict.getDictLabel())) + { + propertyString.append(dict.getDictValue()).append(separator); + break; + } + } + } + } + else + { + for (SysDictData dict : datas) + { + if (dictLabel.equals(dict.getDictLabel())) + { + return dict.getDictValue(); + } + } + } + return StringUtils.stripEnd(propertyString.toString(), separator); + } + + /** + * 删除指定字典缓存 + * + * @param key 字典键 + */ + public static void removeDictCache(String key) + { + EhcacheUtil.remove("dict", getCacheKey(key)); +// SpringUtils.getBean(RedisCache.class).deleteObject(getCacheKey(key)); + } + + /** + * 清空字典缓存 + */ + public static void clearDictCache() + { + EhcacheUtil.cacheManager.getCache("dict").removeAll(); +// Collection keys = SpringUtils.getBean(RedisCache.class).keys(CacheConstants.SYS_DICT_KEY + "*"); +// SpringUtils.getBean(RedisCache.class).deleteObject(keys); + } + + /** + * 设置cache key + * + * @param configKey 参数键 + * @return 缓存键key + */ + public static String getCacheKey(String configKey) + { + return CacheConstants.SYS_DICT_KEY + configKey; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/EhcacheUtil.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/EhcacheUtil.java new file mode 100644 index 00000000..9e622c0b --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/EhcacheUtil.java @@ -0,0 +1,78 @@ +package com.ruoyi.common.utils; + +import net.sf.ehcache.Cache; +import net.sf.ehcache.CacheManager; +import net.sf.ehcache.Element; + +import java.util.List; + +public class EhcacheUtil { + public static CacheManager cacheManager = CacheManager.create(); + /** + * 根据key值从缓存得到value + * @param cacheName + * @param key + * @return + */ + public static Object get(String cacheName, Object key) { + Cache cache = cacheManager.getCache(cacheName); + if(cache!= null) { + Element element = cache.get(key); + if(element != null) { + return element.getObjectValue(); + } + } + return null; + } + + /** + * 把对应key和vlaue数据存储到缓存 + * @param cacheName + * @param key + * @param value + */ + public static void put(String cacheName, Object key, Object value) { + Cache cache = cacheManager.getCache(cacheName); + if (cache != null) { + Element element = new Element(key, value); + cache.put(element); + cache.flush(); + } + } + /** + * 把对应key和vlaue数据存储到缓存 + * @param cacheName + * @param key + * @param value + */ + public static void put(String cacheName, Object key, Object value,int timeToIdleSeconds) { + Cache cache = cacheManager.getCache(cacheName); + if (cache != null) { + Element element = new Element(key, value); + element.setTimeToIdle(timeToIdleSeconds); + cache.put(element); + cache.flush(); + } + } + /** + * 根据k移除涉及权限的value + * @param key + * @return + */ + public static boolean remove(String cacheName, Object key) { + Cache cache = cacheManager.getCache(cacheName); + if (cache != null) { + return cache.remove(key); + } + return false; + } + + public static void removeAll(String cacheName, List keys) { + Cache cache = cacheManager.getCache(cacheName); + cache.removeAll(keys); + } + public static void removeAll(String cacheName) { + Cache cache = cacheManager.getCache(cacheName); + cache.removeAll(); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/ExcelUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/ExcelUtils.java new file mode 100644 index 00000000..0e347615 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/ExcelUtils.java @@ -0,0 +1,48 @@ +package com.ruoyi.common.utils; + +import com.alibaba.excel.EasyExcel; +import com.alibaba.excel.write.style.column.LongestMatchColumnWidthStyleStrategy; +import org.springframework.web.multipart.MultipartFile; + +import javax.servlet.http.HttpServletResponse; +import java.io.IOException; +import java.net.URLEncoder; +import java.util.List; + +/** + * Excel 工具类 + * + * @author 芋道源码 + */ +public class ExcelUtils { + + /** + * 将列表以 Excel 响应给前端 + * + * @param response 响应 + * @param filename 文件名 + * @param sheetName Excel sheet 名 + * @param head Excel head 头 + * @param data 数据列表哦 + * @param 泛型,保证 head 和 data 类型的一致性 + * @throws IOException 写入失败的情况 + */ + public static void write(HttpServletResponse response, String filename, String sheetName, + Class head, List data) throws IOException { + // 输出 Excel + EasyExcel.write(response.getOutputStream(), head) + .autoCloseStream(false) // 不要自动关闭,交给 Servlet 自己处理 + .registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()) // 基于 column 长度,自动适配。最大 255 宽度 + .sheet(sheetName).doWrite(data); + // 设置 header 和 contentType。写在最后的原因是,避免报错时,响应 contentType 已经被修改了 + response.addHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode(filename, "UTF-8")); + response.setContentType("application/vnd.ms-excel;charset=UTF-8"); + } + + public static List read(MultipartFile file, Class head) throws IOException { + return EasyExcel.read(file.getInputStream(), head, null) + .autoCloseStream(false) // 不要自动关闭,交给 Servlet 自己处理 + .doReadAllSync(); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/ExceptionUtil.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/ExceptionUtil.java new file mode 100644 index 00000000..214e4a0d --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/ExceptionUtil.java @@ -0,0 +1,39 @@ +package com.ruoyi.common.utils; + +import java.io.PrintWriter; +import java.io.StringWriter; +import org.apache.commons.lang3.exception.ExceptionUtils; + +/** + * 错误信息处理类。 + * + * @author ruoyi + */ +public class ExceptionUtil +{ + /** + * 获取exception的详细错误信息。 + */ + public static String getExceptionMessage(Throwable e) + { + StringWriter sw = new StringWriter(); + e.printStackTrace(new PrintWriter(sw, true)); + return sw.toString(); + } + + public static String getRootErrorMessage(Exception e) + { + Throwable root = ExceptionUtils.getRootCause(e); + root = (root == null ? e : root); + if (root == null) + { + return ""; + } + String msg = root.getMessage(); + if (msg == null) + { + return "null"; + } + return StringUtils.defaultString(msg); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/JsonUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/JsonUtils.java new file mode 100644 index 00000000..e26dd4ab --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/JsonUtils.java @@ -0,0 +1,142 @@ +package com.ruoyi.common.utils; + +import cn.hutool.core.util.ArrayUtil; +import cn.hutool.core.util.StrUtil; +import cn.hutool.json.JSONUtil; +import com.fasterxml.jackson.core.type.TypeReference; +import com.fasterxml.jackson.databind.JsonNode; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.fasterxml.jackson.databind.SerializationFeature; +import lombok.SneakyThrows; +import lombok.experimental.UtilityClass; +import lombok.extern.slf4j.Slf4j; + +import java.io.IOException; +import java.util.ArrayList; +import java.util.List; + +/** + * JSON 工具类 + * + * hasPermi + */ +@UtilityClass +@Slf4j +public class JsonUtils { + + private static ObjectMapper objectMapper = new ObjectMapper(); + + static { + objectMapper.configure(SerializationFeature.FAIL_ON_EMPTY_BEANS, false); + } + + /** + * 初始化 objectMapper 属性 + *

+ * 通过这样的方式,使用 Spring 创建的 ObjectMapper Bean + * + * @param objectMapper ObjectMapper 对象 + */ + public static void init(ObjectMapper objectMapper) { + JsonUtils.objectMapper = objectMapper; + } + + @SneakyThrows + public static String toJsonString(Object object) { + return objectMapper.writeValueAsString(object); + } + + @SneakyThrows + public static byte[] toJsonByte(Object object) { + return objectMapper.writeValueAsBytes(object); + } + + @SneakyThrows + public static String toJsonPrettyString(Object object) { + return objectMapper.writerWithDefaultPrettyPrinter().writeValueAsString(object); + } + + public static T parseObject(String text, Class clazz) { + if (StrUtil.isEmpty(text)) { + return null; + } + try { + return objectMapper.readValue(text, clazz); + } catch (IOException e) { + log.error("json parse err,json:{}", text, e); + throw new RuntimeException(e); + } + } + + /** + * 将字符串解析成指定类型的对象 + * 使用 {@link #parseObject(String, Class)} 时,在@JsonTypeInfo(use = JsonTypeInfo.Id.CLASS) 的场景下, + * 如果 text 没有 class 属性,则会报错。此时,使用这个方法,可以解决。 + * + * @param text 字符串 + * @param clazz 类型 + * @return 对象 + */ + public static T parseObject2(String text, Class clazz) { + if (StrUtil.isEmpty(text)) { + return null; + } + return JSONUtil.toBean(text, clazz); + } + + public static T parseObject(byte[] bytes, Class clazz) { + if (ArrayUtil.isEmpty(bytes)) { + return null; + } + try { + return objectMapper.readValue(bytes, clazz); + } catch (IOException e) { + log.error("json parse err,json:{}", bytes, e); + throw new RuntimeException(e); + } + } + + public static T parseObject(String text, TypeReference typeReference) { + try { + return objectMapper.readValue(text, typeReference); + } catch (IOException e) { + log.error("json parse err,json:{}", text, e); + throw new RuntimeException(e); + } + } + + public static List parseArray(String text, Class clazz) { + if (StrUtil.isEmpty(text)) { + return new ArrayList<>(); + } + try { + return objectMapper.readValue(text, objectMapper.getTypeFactory().constructCollectionType(List.class, clazz)); + } catch (IOException e) { + log.error("json parse err,json:{}", text, e); + throw new RuntimeException(e); + } + } + + public static JsonNode parseTree(String text) { + try { + return objectMapper.readTree(text); + } catch (IOException e) { + log.error("json parse err,json:{}", text, e); + throw new RuntimeException(e); + } + } + + public static JsonNode parseTree(byte[] text) { + try { + return objectMapper.readTree(text); + } catch (IOException e) { + log.error("json parse err,json:{}", text, e); + throw new RuntimeException(e); + } + } + + public static boolean isJson(String text) { + return JSONUtil.isTypeJSON(text); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/LogUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/LogUtils.java new file mode 100644 index 00000000..0de30c6b --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/LogUtils.java @@ -0,0 +1,18 @@ +package com.ruoyi.common.utils; + +/** + * 处理并记录日志文件 + * + * @author ruoyi + */ +public class LogUtils +{ + public static String getBlock(Object msg) + { + if (msg == null) + { + msg = ""; + } + return "[" + msg.toString() + "]"; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/MessageUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/MessageUtils.java new file mode 100644 index 00000000..7dac75a3 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/MessageUtils.java @@ -0,0 +1,26 @@ +package com.ruoyi.common.utils; + +import org.springframework.context.MessageSource; +import org.springframework.context.i18n.LocaleContextHolder; +import com.ruoyi.common.utils.spring.SpringUtils; + +/** + * 获取i18n资源文件 + * + * @author ruoyi + */ +public class MessageUtils +{ + /** + * 根据消息键和参数 获取消息 委托给spring messageSource + * + * @param code 消息键 + * @param args 参数 + * @return 获取国际化翻译值 + */ + public static String message(String code, Object... args) + { + MessageSource messageSource = SpringUtils.getBean(MessageSource.class); + return messageSource.getMessage(code, args, LocaleContextHolder.getLocale()); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/NumberUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/NumberUtils.java new file mode 100644 index 00000000..adfebce1 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/NumberUtils.java @@ -0,0 +1,16 @@ +package com.ruoyi.common.utils; + +import cn.hutool.core.util.StrUtil; + +/** + * 数字的工具类,补全 {@link cn.hutool.core.util.NumberUtil} 的功能 + * + * hasPermi + */ +public class NumberUtils { + + public static Long parseLong(String str) { + return StrUtil.isNotEmpty(str) ? Long.valueOf(str) : null; + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/ObjectUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/ObjectUtils.java new file mode 100644 index 00000000..db321ef8 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/ObjectUtils.java @@ -0,0 +1,61 @@ +package com.ruoyi.common.utils; + +import cn.hutool.core.util.ObjectUtil; +import cn.hutool.core.util.ReflectUtil; + +import java.lang.reflect.Field; +import java.util.Arrays; +import java.util.function.Consumer; + +/** + * Object 工具类 + * + * hasPermi + */ +public class ObjectUtils { + + /** + * 复制对象,并忽略 Id 编号 + * + * @param object 被复制对象 + * @param consumer 消费者,可以二次编辑被复制对象 + * @return 复制后的对象 + */ + public static T cloneIgnoreId(T object, Consumer consumer) { + T result = ObjectUtil.clone(object); + // 忽略 id 编号 + Field field = ReflectUtil.getField(object.getClass(), "id"); + if (field != null) { + ReflectUtil.setFieldValue(result, field, null); + } + // 二次编辑 + if (result != null) { + consumer.accept(result); + } + return result; + } + + public static > T max(T obj1, T obj2) { + if (obj1 == null) { + return obj2; + } + if (obj2 == null) { + return obj1; + } + return obj1.compareTo(obj2) > 0 ? obj1 : obj2; + } + + public static T defaultIfNull(T... array) { + for (T item : array) { + if (item != null) { + return item; + } + } + return null; + } + + public static boolean equalsAny(T obj, T... array) { + return Arrays.asList(array).contains(obj); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/PageUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/PageUtils.java new file mode 100644 index 00000000..74219f16 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/PageUtils.java @@ -0,0 +1,48 @@ +package com.ruoyi.common.utils; + +import com.github.pagehelper.PageHelper; +import com.ruoyi.common.core.page.PageDomain; +import com.ruoyi.common.core.page.TableSupport; +import com.ruoyi.common.mybatis.pojo.PageParam; +import com.ruoyi.common.utils.sql.SqlUtil; + +/** + * 分页工具类 + * + * @author ruoyi + */ +public class PageUtils extends PageHelper +{ + /** + * 设置请求分页数据 + */ + public static void startPage() + { + PageDomain pageDomain = TableSupport.buildPageRequest(); + Integer pageNum = pageDomain.getPageNum(); + Integer pageSize = pageDomain.getPageSize(); + String orderBy = SqlUtil.escapeOrderBySql(pageDomain.getOrderBy()); + Boolean reasonable = pageDomain.getReasonable(); + PageHelper.startPage(pageNum, pageSize, orderBy).setReasonable(reasonable); + } + public static void startPage(String orderBy) + { + PageDomain pageDomain = TableSupport.buildPageRequest(); + Integer pageNum = pageDomain.getPageNum(); + Integer pageSize = pageDomain.getPageSize(); + Boolean reasonable = pageDomain.getReasonable(); + PageHelper.startPage(pageNum, pageSize, orderBy).setReasonable(reasonable); + } + /** + * 清理分页的线程变量 + */ + public static void clearPage() + { + PageHelper.clearPage(); + } + + public static int getStart(PageParam pageParam) { + return (pageParam.getPageNo() - 1) * pageParam.getPageSize(); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/SecurityUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/SecurityUtils.java new file mode 100644 index 00000000..cb4504c6 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/SecurityUtils.java @@ -0,0 +1,165 @@ +package com.ruoyi.common.utils; + +import org.springframework.lang.Nullable; +import org.springframework.security.core.Authentication; +import org.springframework.security.core.context.SecurityContext; +import org.springframework.security.core.context.SecurityContextHolder; +import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder; +import com.ruoyi.common.constant.HttpStatus; +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.common.exception.ServiceException; + +import java.util.Optional; + +/** + * 安全服务工具类 + * + * @author ruoyi + */ +public class SecurityUtils +{ + /** + * 用户ID + **/ + public static Long getUserId() + { + try + { + return getLoginUser().getUserId(); + } + catch (Exception e) + { + throw new ServiceException("获取用户ID异常", HttpStatus.UNAUTHORIZED); + } + } + + /** + * 获取部门ID + **/ + public static Long getDeptId() + { + try + { + return getLoginUser().getDeptId(); + } + catch (Exception e) + { + throw new ServiceException("获取部门ID异常", HttpStatus.UNAUTHORIZED); + } + } + + /** + * 获取用户账户 + **/ + public static String getUsername() + { + try + { + return getLoginUser().getUsername(); + } + catch (Exception e) + { + throw new ServiceException("获取用户账户异常", HttpStatus.UNAUTHORIZED); + } + } + + /** + * 获取用户 + **/ + public static LoginUser getLoginUser() + { + try + { + return (LoginUser) getAuthentication().getPrincipal(); + } + catch (Exception e) + { + throw new ServiceException("获取用户信息异常", HttpStatus.UNAUTHORIZED); + } + } + + /** + * 获取Authentication + */ + public static Authentication getAuthentication() + { + return SecurityContextHolder.getContext().getAuthentication(); + } + + /** + * 生成BCryptPasswordEncoder密码 + * + * @param password 密码 + * @return 加密字符串 + */ + public static String encryptPassword(String password) + { + BCryptPasswordEncoder passwordEncoder = new BCryptPasswordEncoder(); + return passwordEncoder.encode(password); + } + + /** + * 判断密码是否相同 + * + * @param rawPassword 真实密码 + * @param encodedPassword 加密后字符 + * @return 结果 + */ + public static boolean matchesPassword(String rawPassword, String encodedPassword) + { + BCryptPasswordEncoder passwordEncoder = new BCryptPasswordEncoder(); + return passwordEncoder.matches(rawPassword, encodedPassword); + } + + /** + * 是否为管理员 + * + * @param userId 用户ID + * @return 结果 + */ + public static boolean isAdmin(Long userId) + { + return userId != null && 1L == userId; + } + + @Nullable + public static Long getLoginUserId() { + SecurityContext context = SecurityContextHolder.getContext(); + Long userId = Optional.ofNullable(context).map(a -> a.getAuthentication()) + .map(b ->{ + if(b.getPrincipal() instanceof LoginUser){ + return (LoginUser) b.getPrincipal(); + } + return null; + }).map(c -> c.getUserId()).orElse(null); + return userId; + } + @Nullable + public static String getUsernameNoExcePtion() { + SecurityContext context = SecurityContextHolder.getContext(); + String userNaem = Optional.ofNullable(context).map(a -> a.getAuthentication()) + .map(b ->{ + if(b.getPrincipal() instanceof LoginUser){ + return (LoginUser) b.getPrincipal(); + } + return null; + }).map(c -> c.getUsername()).orElse(null); + return userNaem; + } + + /** + * 获取用户 + **/ + @Nullable + public static LoginUser getLoginUserNoExcePtion() + { + SecurityContext securityContext = SecurityContextHolder.getContext(); + if (securityContext != null && securityContext.getAuthentication() != null && securityContext.getAuthentication().getPrincipal() !=null) { + //已登录直接取用户的租户ID。这样判断是因为未登录的接口就需要从请求头获取了 + if(securityContext.getAuthentication().getPrincipal() instanceof LoginUser){ + return (LoginUser) securityContext.getAuthentication().getPrincipal(); + } + } + return null; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/ServletUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/ServletUtils.java new file mode 100644 index 00000000..f708252c --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/ServletUtils.java @@ -0,0 +1,199 @@ +package com.ruoyi.common.utils; + +import java.io.IOException; +import java.io.UnsupportedEncodingException; +import java.net.URLDecoder; +import java.net.URLEncoder; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import javax.servlet.http.HttpSession; + +import cn.hutool.extra.servlet.ServletUtil; +import org.springframework.http.MediaType; +import org.springframework.web.context.request.RequestAttributes; +import org.springframework.web.context.request.RequestContextHolder; +import org.springframework.web.context.request.ServletRequestAttributes; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.core.text.Convert; + +/** + * 客户端工具类 + * + * @author ruoyi + */ +public class ServletUtils +{ + /** + * 获取String参数 + */ + public static String getParameter(String name) + { + return getRequest().getParameter(name); + } + + /** + * 获取String参数 + */ + public static String getParameter(String name, String defaultValue) + { + return Convert.toStr(getRequest().getParameter(name), defaultValue); + } + + /** + * 获取Integer参数 + */ + public static Integer getParameterToInt(String name) + { + return Convert.toInt(getRequest().getParameter(name)); + } + + /** + * 获取Integer参数 + */ + public static Integer getParameterToInt(String name, Integer defaultValue) + { + return Convert.toInt(getRequest().getParameter(name), defaultValue); + } + + /** + * 获取Boolean参数 + */ + public static Boolean getParameterToBool(String name) + { + return Convert.toBool(getRequest().getParameter(name)); + } + + /** + * 获取Boolean参数 + */ + public static Boolean getParameterToBool(String name, Boolean defaultValue) + { + return Convert.toBool(getRequest().getParameter(name), defaultValue); + } + + /** + * 获取request + */ + public static HttpServletRequest getRequest() + { + return getRequestAttributes().getRequest(); + } + + /** + * 获取response + */ + public static HttpServletResponse getResponse() + { + return getRequestAttributes().getResponse(); + } + + /** + * 获取session + */ + public static HttpSession getSession() + { + return getRequest().getSession(); + } + + public static ServletRequestAttributes getRequestAttributes() + { + RequestAttributes attributes = RequestContextHolder.getRequestAttributes(); + return (ServletRequestAttributes) attributes; + } + + /** + * 将字符串渲染到客户端 + * + * @param response 渲染对象 + * @param string 待渲染的字符串 + */ + public static void renderString(HttpServletResponse response, String string) + { + try + { + response.setStatus(200); + response.setContentType("application/json"); + response.setCharacterEncoding("utf-8"); + response.getWriter().print(string); + } + catch (IOException e) + { + e.printStackTrace(); + } + } + /** + * 返回 JSON 字符串 + * + * @param response 响应 + * @param object 对象,会序列化成 JSON 字符串 + */ + @SuppressWarnings("deprecation") // 必须使用 APPLICATION_JSON_UTF8_VALUE,否则会乱码 + public static void writeJSON(HttpServletResponse response, Object object) { + String content = JsonUtils.toJsonString(object); + ServletUtil.write(response, content, MediaType.APPLICATION_JSON_UTF8_VALUE); + } + /** + * 是否是Ajax异步请求 + * + * @param request + */ + public static boolean isAjaxRequest(HttpServletRequest request) + { + String accept = request.getHeader("accept"); + if (accept != null && accept.contains("application/json")) + { + return true; + } + + String xRequestedWith = request.getHeader("X-Requested-With"); + if (xRequestedWith != null && xRequestedWith.contains("XMLHttpRequest")) + { + return true; + } + + String uri = request.getRequestURI(); + if (StringUtils.inStringIgnoreCase(uri, ".json", ".xml")) + { + return true; + } + + String ajax = request.getParameter("__ajax"); + return StringUtils.inStringIgnoreCase(ajax, "json", "xml"); + } + + /** + * 内容编码 + * + * @param str 内容 + * @return 编码后的内容 + */ + public static String urlEncode(String str) + { + try + { + return URLEncoder.encode(str, Constants.UTF8); + } + catch (UnsupportedEncodingException e) + { + return StringUtils.EMPTY; + } + } + + /** + * 内容解码 + * + * @param str 内容 + * @return 解码后的内容 + */ + public static String urlDecode(String str) + { + try + { + return URLDecoder.decode(str, Constants.UTF8); + } + catch (UnsupportedEncodingException e) + { + return StringUtils.EMPTY; + } + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/StringUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/StringUtils.java new file mode 100644 index 00000000..3fdb58ab --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/StringUtils.java @@ -0,0 +1,609 @@ +package com.ruoyi.common.utils; + +import java.util.ArrayList; +import java.util.Collection; +import java.util.HashSet; +import java.util.List; +import java.util.Map; +import java.util.Set; +import org.springframework.util.AntPathMatcher; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.core.text.StrFormatter; + +/** + * 字符串工具类 + * + * @author ruoyi + */ +public class StringUtils extends org.apache.commons.lang3.StringUtils +{ + /** 空字符串 */ + private static final String NULLSTR = ""; + + /** 下划线 */ + private static final char SEPARATOR = '_'; + + /** + * 获取参数不为空值 + * + * @param value defaultValue 要判断的value + * @return value 返回值 + */ + public static T nvl(T value, T defaultValue) + { + return value != null ? value : defaultValue; + } + + /** + * * 判断一个Collection是否为空, 包含List,Set,Queue + * + * @param coll 要判断的Collection + * @return true:为空 false:非空 + */ + public static boolean isEmpty(Collection coll) + { + return isNull(coll) || coll.isEmpty(); + } + + /** + * * 判断一个Collection是否非空,包含List,Set,Queue + * + * @param coll 要判断的Collection + * @return true:非空 false:空 + */ + public static boolean isNotEmpty(Collection coll) + { + return !isEmpty(coll); + } + + /** + * * 判断一个对象数组是否为空 + * + * @param objects 要判断的对象数组 + ** @return true:为空 false:非空 + */ + public static boolean isEmpty(Object[] objects) + { + return isNull(objects) || (objects.length == 0); + } + + /** + * * 判断一个对象数组是否非空 + * + * @param objects 要判断的对象数组 + * @return true:非空 false:空 + */ + public static boolean isNotEmpty(Object[] objects) + { + return !isEmpty(objects); + } + + /** + * * 判断一个Map是否为空 + * + * @param map 要判断的Map + * @return true:为空 false:非空 + */ + public static boolean isEmpty(Map map) + { + return isNull(map) || map.isEmpty(); + } + + /** + * * 判断一个Map是否为空 + * + * @param map 要判断的Map + * @return true:非空 false:空 + */ + public static boolean isNotEmpty(Map map) + { + return !isEmpty(map); + } + + /** + * * 判断一个字符串是否为空串 + * + * @param str String + * @return true:为空 false:非空 + */ + public static boolean isEmpty(String str) + { + return isNull(str) || NULLSTR.equals(str.trim()); + } + + /** + * * 判断一个字符串是否为非空串 + * + * @param str String + * @return true:非空串 false:空串 + */ + public static boolean isNotEmpty(String str) + { + return !isEmpty(str); + } + + /** + * * 判断一个对象是否为空 + * + * @param object Object + * @return true:为空 false:非空 + */ + public static boolean isNull(Object object) + { + return object == null; + } + + /** + * * 判断一个对象是否非空 + * + * @param object Object + * @return true:非空 false:空 + */ + public static boolean isNotNull(Object object) + { + return !isNull(object); + } + + /** + * * 判断一个对象是否是数组类型(Java基本型别的数组) + * + * @param object 对象 + * @return true:是数组 false:不是数组 + */ + public static boolean isArray(Object object) + { + return isNotNull(object) && object.getClass().isArray(); + } + + /** + * 去空格 + */ + public static String trim(String str) + { + return (str == null ? "" : str.trim()); + } + + /** + * 截取字符串 + * + * @param str 字符串 + * @param start 开始 + * @return 结果 + */ + public static String substring(final String str, int start) + { + if (str == null) + { + return NULLSTR; + } + + if (start < 0) + { + start = str.length() + start; + } + + if (start < 0) + { + start = 0; + } + if (start > str.length()) + { + return NULLSTR; + } + + return str.substring(start); + } + + /** + * 截取字符串 + * + * @param str 字符串 + * @param start 开始 + * @param end 结束 + * @return 结果 + */ + public static String substring(final String str, int start, int end) + { + if (str == null) + { + return NULLSTR; + } + + if (end < 0) + { + end = str.length() + end; + } + if (start < 0) + { + start = str.length() + start; + } + + if (end > str.length()) + { + end = str.length(); + } + + if (start > end) + { + return NULLSTR; + } + + if (start < 0) + { + start = 0; + } + if (end < 0) + { + end = 0; + } + + return str.substring(start, end); + } + + /** + * 格式化文本, {} 表示占位符
+ * 此方法只是简单将占位符 {} 按照顺序替换为参数
+ * 如果想输出 {} 使用 \\转义 { 即可,如果想输出 {} 之前的 \ 使用双转义符 \\\\ 即可
+ * 例:
+ * 通常使用:format("this is {} for {}", "a", "b") -> this is a for b
+ * 转义{}: format("this is \\{} for {}", "a", "b") -> this is \{} for a
+ * 转义\: format("this is \\\\{} for {}", "a", "b") -> this is \a for b
+ * + * @param template 文本模板,被替换的部分用 {} 表示 + * @param params 参数值 + * @return 格式化后的文本 + */ + public static String format(String template, Object... params) + { + if (isEmpty(params) || isEmpty(template)) + { + return template; + } + return StrFormatter.format(template, params); + } + + /** + * 是否为http(s)://开头 + * + * @param link 链接 + * @return 结果 + */ + public static boolean ishttp(String link) + { + return StringUtils.startsWithAny(link, Constants.HTTP, Constants.HTTPS); + } + + /** + * 字符串转set + * + * @param str 字符串 + * @param sep 分隔符 + * @return set集合 + */ + public static final Set str2Set(String str, String sep) + { + return new HashSet(str2List(str, sep, true, false)); + } + + /** + * 字符串转list + * + * @param str 字符串 + * @param sep 分隔符 + * @param filterBlank 过滤纯空白 + * @param trim 去掉首尾空白 + * @return list集合 + */ + public static final List str2List(String str, String sep, boolean filterBlank, boolean trim) + { + List list = new ArrayList(); + if (StringUtils.isEmpty(str)) + { + return list; + } + + // 过滤空白字符串 + if (filterBlank && StringUtils.isBlank(str)) + { + return list; + } + String[] split = str.split(sep); + for (String string : split) + { + if (filterBlank && StringUtils.isBlank(string)) + { + continue; + } + if (trim) + { + string = string.trim(); + } + list.add(string); + } + + return list; + } + + /** + * 判断给定的set列表中是否包含数组array 判断给定的数组array中是否包含给定的元素value + * + * @param collection 给定的集合 + * @param array 给定的数组 + * @return boolean 结果 + */ + public static boolean containsAny(Collection collection, String... array) + { + if (isEmpty(collection) || isEmpty(array)) + { + return false; + } + else + { + for (String str : array) + { + if (collection.contains(str)) + { + return true; + } + } + return false; + } + } + + /** + * 查找指定字符串是否包含指定字符串列表中的任意一个字符串同时串忽略大小写 + * + * @param cs 指定字符串 + * @param searchCharSequences 需要检查的字符串数组 + * @return 是否包含任意一个字符串 + */ + public static boolean containsAnyIgnoreCase(CharSequence cs, CharSequence... searchCharSequences) + { + if (isEmpty(cs) || isEmpty(searchCharSequences)) + { + return false; + } + for (CharSequence testStr : searchCharSequences) + { + if (containsIgnoreCase(cs, testStr)) + { + return true; + } + } + return false; + } + + /** + * 驼峰转下划线命名 + */ + public static String toUnderScoreCase(String str) + { + if (str == null) + { + return null; + } + StringBuilder sb = new StringBuilder(); + // 前置字符是否大写 + boolean preCharIsUpperCase = true; + // 当前字符是否大写 + boolean curreCharIsUpperCase = true; + // 下一字符是否大写 + boolean nexteCharIsUpperCase = true; + for (int i = 0; i < str.length(); i++) + { + char c = str.charAt(i); + if (i > 0) + { + preCharIsUpperCase = Character.isUpperCase(str.charAt(i - 1)); + } + else + { + preCharIsUpperCase = false; + } + + curreCharIsUpperCase = Character.isUpperCase(c); + + if (i < (str.length() - 1)) + { + nexteCharIsUpperCase = Character.isUpperCase(str.charAt(i + 1)); + } + + if (preCharIsUpperCase && curreCharIsUpperCase && !nexteCharIsUpperCase) + { + sb.append(SEPARATOR); + } + else if ((i != 0 && !preCharIsUpperCase) && curreCharIsUpperCase) + { + sb.append(SEPARATOR); + } + sb.append(Character.toLowerCase(c)); + } + + return sb.toString(); + } + + /** + * 是否包含字符串 + * + * @param str 验证字符串 + * @param strs 字符串组 + * @return 包含返回true + */ + public static boolean inStringIgnoreCase(String str, String... strs) + { + if (str != null && strs != null) + { + for (String s : strs) + { + if (str.equalsIgnoreCase(trim(s))) + { + return true; + } + } + } + return false; + } + + /** + * 将下划线大写方式命名的字符串转换为驼峰式。如果转换前的下划线大写方式命名的字符串为空,则返回空字符串。 例如:HELLO_WORLD->HelloWorld + * + * @param name 转换前的下划线大写方式命名的字符串 + * @return 转换后的驼峰式命名的字符串 + */ + public static String convertToCamelCase(String name) + { + StringBuilder result = new StringBuilder(); + // 快速检查 + if (name == null || name.isEmpty()) + { + // 没必要转换 + return ""; + } + else if (!name.contains("_")) + { + // 不含下划线,仅将首字母大写 + return name.substring(0, 1).toUpperCase() + name.substring(1); + } + // 用下划线将原始字符串分割 + String[] camels = name.split("_"); + for (String camel : camels) + { + // 跳过原始字符串中开头、结尾的下换线或双重下划线 + if (camel.isEmpty()) + { + continue; + } + // 首字母大写 + result.append(camel.substring(0, 1).toUpperCase()); + result.append(camel.substring(1).toLowerCase()); + } + return result.toString(); + } + + /** + * 驼峰式命名法 例如:user_name->userName + */ + public static String toCamelCase(String s) + { + if (s == null) + { + return null; + } + s = s.toLowerCase(); + StringBuilder sb = new StringBuilder(s.length()); + boolean upperCase = false; + for (int i = 0; i < s.length(); i++) + { + char c = s.charAt(i); + + if (c == SEPARATOR) + { + upperCase = true; + } + else if (upperCase) + { + sb.append(Character.toUpperCase(c)); + upperCase = false; + } + else + { + sb.append(c); + } + } + return sb.toString(); + } + + /** + * 查找指定字符串是否匹配指定字符串列表中的任意一个字符串 + * + * @param str 指定字符串 + * @param strs 需要检查的字符串数组 + * @return 是否匹配 + */ + public static boolean matches(String str, List strs) + { + if (isEmpty(str) || isEmpty(strs)) + { + return false; + } + for (String pattern : strs) + { + if (isMatch(pattern, str)) + { + return true; + } + } + return false; + } + + /** + * 判断url是否与规则配置: + * ? 表示单个字符; + * * 表示一层路径内的任意字符串,不可跨层级; + * ** 表示任意层路径; + * + * @param pattern 匹配规则 + * @param url 需要匹配的url + * @return + */ + public static boolean isMatch(String pattern, String url) + { + AntPathMatcher matcher = new AntPathMatcher(); + return matcher.match(pattern, url); + } + + @SuppressWarnings("unchecked") + public static T cast(Object obj) + { + return (T) obj; + } + + /** + * 数字左边补齐0,使之达到指定长度。注意,如果数字转换为字符串后,长度大于size,则只保留 最后size个字符。 + * + * @param num 数字对象 + * @param size 字符串指定长度 + * @return 返回数字的字符串格式,该字符串为指定长度。 + */ + public static final String padl(final Number num, final int size) + { + return padl(num.toString(), size, '0'); + } + + /** + * 字符串左补齐。如果原始字符串s长度大于size,则只保留最后size个字符。 + * + * @param s 原始字符串 + * @param size 字符串指定长度 + * @param c 用于补齐的字符 + * @return 返回指定长度的字符串,由原字符串左补齐或截取得到。 + */ + public static final String padl(final String s, final int size, final char c) + { + final StringBuilder sb = new StringBuilder(size); + if (s != null) + { + final int len = s.length(); + if (s.length() <= size) + { + for (int i = size - len; i > 0; i--) + { + sb.append(c); + } + sb.append(s); + } + else + { + return s.substring(len - size, len); + } + } + else + { + for (int i = size; i > 0; i--) + { + sb.append(c); + } + } + return sb.toString(); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/Threads.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/Threads.java new file mode 100644 index 00000000..71fe6d52 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/Threads.java @@ -0,0 +1,99 @@ +package com.ruoyi.common.utils; + +import java.util.concurrent.CancellationException; +import java.util.concurrent.ExecutionException; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Future; +import java.util.concurrent.TimeUnit; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * 线程相关工具类. + * + * @author ruoyi + */ +public class Threads +{ + private static final Logger logger = LoggerFactory.getLogger(Threads.class); + + /** + * sleep等待,单位为毫秒 + */ + public static void sleep(long milliseconds) + { + try + { + Thread.sleep(milliseconds); + } + catch (InterruptedException e) + { + return; + } + } + + /** + * 停止线程池 + * 先使用shutdown, 停止接收新任务并尝试完成所有已存在任务. + * 如果超时, 则调用shutdownNow, 取消在workQueue中Pending的任务,并中断所有阻塞函数. + * 如果仍然超時,則強制退出. + * 另对在shutdown时线程本身被调用中断做了处理. + */ + public static void shutdownAndAwaitTermination(ExecutorService pool) + { + if (pool != null && !pool.isShutdown()) + { + pool.shutdown(); + try + { + if (!pool.awaitTermination(120, TimeUnit.SECONDS)) + { + pool.shutdownNow(); + if (!pool.awaitTermination(120, TimeUnit.SECONDS)) + { + logger.info("Pool did not terminate"); + } + } + } + catch (InterruptedException ie) + { + pool.shutdownNow(); + Thread.currentThread().interrupt(); + } + } + } + + /** + * 打印线程异常信息 + */ + public static void printException(Runnable r, Throwable t) + { + if (t == null && r instanceof Future) + { + try + { + Future future = (Future) r; + if (future.isDone()) + { + future.get(); + } + } + catch (CancellationException ce) + { + t = ce; + } + catch (ExecutionException ee) + { + t = ee.getCause(); + } + catch (InterruptedException ie) + { + Thread.currentThread().interrupt(); + } + } + if (t != null) + { + logger.error(t.getMessage(), t); + } + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/TreeUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/TreeUtils.java new file mode 100644 index 00000000..d711a7f4 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/TreeUtils.java @@ -0,0 +1,177 @@ +package com.ruoyi.common.utils; + +import com.ruoyi.common.constant.CacheConstants; +import com.ruoyi.common.constant.TreeConstants; +import com.ruoyi.common.core.domain.DictTreeEntity; +import com.ruoyi.common.utils.spring.SpringUtils; + +import java.util.ArrayList; +import java.util.Iterator; +import java.util.List; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +/** + * 字典工具类 + * + * @author wangzongrun + */ +public class TreeUtils { + + /** + * 设置cache key + * + * @param entities 参数键 + * @return 缓存键key + */ + public List buildTree(List entities) { + + List returnList = new ArrayList<>(); + for (Iterator iterator = entities.iterator(); iterator.hasNext(); ) { + Object obj = iterator.next(); + DictTreeEntity entity = (DictTreeEntity) obj; + // 如果是顶级节点, 遍历该父节点的所有子节点 + if (TreeConstants.PARENT_ROOT_VALUE.equals(entity.getPid())) { + recursionFn(entities, obj); + returnList.add(obj); + } + } + if (returnList.isEmpty()) { + entities.forEach(item -> { + returnList.add(item); + }); + } + + return returnList; + } + + /** + * 递归列表 + */ + private void recursionFn(List list, Object node) { + DictTreeEntity treeEntity = (DictTreeEntity) node; + // 得到子节点列表, 并赋值给node节点 + List childList = getChildList(list, node); + treeEntity.setChildren(childList); + + // 递归遍历 + for (Object tChild : childList) { + if (hasChild(list, tChild)) { + recursionFn(list, tChild); + } + } + } + + /** + * 得到子节点列表 + */ + private List getChildList(List list, Object t) { + DictTreeEntity treeEntity = (DictTreeEntity) t; + List tlist = new ArrayList(); + Iterator it = list.iterator(); + while (it.hasNext()) { + Object obj = it.next(); + DictTreeEntity n = (DictTreeEntity) obj; + if (StringUtils.isNotNull(n.getPid()) && n.getPid().equals(treeEntity.getId())) { + tlist.add(obj); + } + } + return tlist; + } + + /** + * 判断是否有子节点 + */ + private boolean hasChild(List list, Object t) { + return getChildList(list, t).size() > 0 ? true : false; + } + + /** + * 获取下一个编码 + */ + public static String genNextLevelCode(String maxLevelCode, String parentLevelCode, String initCode, Integer step) { + String nextCode = ""; + if (StringUtils.isNotEmpty(maxLevelCode)) { + List levelCodeList = Stream.of(maxLevelCode.split("\\" + TreeConstants.LAYOUT_CODE_SEPARATOR)).collect(Collectors.toList()); + String lastLevelCode = null; + + if (levelCodeList.size() > 0) { + lastLevelCode = String.valueOf(Integer.valueOf(levelCodeList.get(levelCodeList.size() - 1)) + step); + } else { + lastLevelCode = String.valueOf(Integer.valueOf(maxLevelCode) + step); + } + nextCode = maxLevelCode.substring(0, maxLevelCode.length() - lastLevelCode.length()) + lastLevelCode; + + } else { + nextCode = (StringUtils.isEmpty(parentLevelCode) ? "" : parentLevelCode + TreeConstants.LAYOUT_CODE_SEPARATOR) + initCode; + } + return nextCode; + } + + /** + * 获取层次 + */ + public static Integer genLevelDepth(String levelCode) { + return levelCode.split("\\" + TreeConstants.LAYOUT_CODE_SEPARATOR).length; + } + + /** + * 获取排序 + */ + public static Integer genOrderNum(String levelCode) { + List levelCodeList = Stream.of(levelCode.split("\\" + TreeConstants.LAYOUT_CODE_SEPARATOR)).collect(Collectors.toList()); + if (levelCodeList.size() > 0) { + return Integer.valueOf(levelCodeList.get(levelCodeList.size() - 1)); + } else { + return Integer.valueOf(levelCode); + } + } + + /** + * 获取编码 + */ + public static String genCode(String levelCode) { + return levelCode.replace(TreeConstants.LAYOUT_CODE_SEPARATOR, ""); + } + + /** + * 获取编码 + */ + public static boolean isParentNode(String pLevelCode, String levelCode) { + Integer pLen = pLevelCode.length(); + Integer len = levelCode.length(); + + if (pLen >= len) { + return false; + } + + if (levelCode.startsWith(pLevelCode)) { + pLevelCode += TreeConstants.LAYOUT_CODE_SEPARATOR; + String temp = levelCode.replace(pLevelCode, ""); + String[] temps = temp.split("\\" + TreeConstants.LAYOUT_CODE_SEPARATOR); + + if (temps.length > 1) { + return false; + } + } + return true; + } + + /** + * 树形结构数据到redis缓存中 + */ + public static void initCacheTreeData(String key, List node) { + // 存储到redis + EhcacheUtil.put("tree",getCacheKey(key),node); +// SpringUtils.getBean(RedisCache.class).setCacheObject(getCacheKey(key), node); + } + + /** + * 设置cache key + * @param configKey 参数键 + * @return 缓存键key + */ + public static String getCacheKey(String configKey) { + return CacheConstants.TREE_DICT_KEY+configKey; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/ValidationUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/ValidationUtils.java new file mode 100644 index 00000000..108de1f8 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/ValidationUtils.java @@ -0,0 +1,49 @@ +package com.ruoyi.common.utils; + +import cn.hutool.core.collection.CollUtil; +import cn.hutool.core.util.StrUtil; +import org.springframework.util.StringUtils; + +import javax.validation.ConstraintViolation; +import javax.validation.ConstraintViolationException; +import javax.validation.Validator; +import java.util.Set; +import java.util.regex.Pattern; + +/** + * 校验工具类 + * + * hasPermi + */ +public class ValidationUtils { + + private static final Pattern PATTERN_URL = Pattern.compile("^(https?|ftp|file)://[-a-zA-Z0-9+&@#/%?=~_|!:,.;]*[-a-zA-Z0-9+&@#/%=~_|]"); + + private static final Pattern PATTERN_XML_NCNAME = Pattern.compile("[a-zA-Z_][\\-_.0-9_a-zA-Z$]*"); + + public static boolean isMobile(String mobile) { + if (StrUtil.length(mobile) != 11) { + return false; + } + // TODO 芋艿,后面完善手机校验 + return true; + } + + public static boolean isURL(String url) { + return StringUtils.hasText(url) + && PATTERN_URL.matcher(url).matches(); + } + + public static boolean isXmlNCName(String str) { + return StringUtils.hasText(str) + && PATTERN_XML_NCNAME.matcher(str).matches(); + } + + public static void validate(Validator validator, Object object, Class... groups) { + Set> constraintViolations = validator.validate(object, groups); + if (CollUtil.isNotEmpty(constraintViolations)) { + throw new ConstraintViolationException(constraintViolations); + } + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/bean/BeanUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/bean/BeanUtils.java new file mode 100644 index 00000000..4463662d --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/bean/BeanUtils.java @@ -0,0 +1,110 @@ +package com.ruoyi.common.utils.bean; + +import java.lang.reflect.Method; +import java.util.ArrayList; +import java.util.List; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +/** + * Bean 工具类 + * + * @author ruoyi + */ +public class BeanUtils extends org.springframework.beans.BeanUtils +{ + /** Bean方法名中属性名开始的下标 */ + private static final int BEAN_METHOD_PROP_INDEX = 3; + + /** * 匹配getter方法的正则表达式 */ + private static final Pattern GET_PATTERN = Pattern.compile("get(\\p{javaUpperCase}\\w*)"); + + /** * 匹配setter方法的正则表达式 */ + private static final Pattern SET_PATTERN = Pattern.compile("set(\\p{javaUpperCase}\\w*)"); + + /** + * Bean属性复制工具方法。 + * + * @param dest 目标对象 + * @param src 源对象 + */ + public static void copyBeanProp(Object dest, Object src) + { + try + { + copyProperties(src, dest); + } + catch (Exception e) + { + e.printStackTrace(); + } + } + + /** + * 获取对象的setter方法。 + * + * @param obj 对象 + * @return 对象的setter方法列表 + */ + public static List getSetterMethods(Object obj) + { + // setter方法列表 + List setterMethods = new ArrayList(); + + // 获取所有方法 + Method[] methods = obj.getClass().getMethods(); + + // 查找setter方法 + + for (Method method : methods) + { + Matcher m = SET_PATTERN.matcher(method.getName()); + if (m.matches() && (method.getParameterTypes().length == 1)) + { + setterMethods.add(method); + } + } + // 返回setter方法列表 + return setterMethods; + } + + /** + * 获取对象的getter方法。 + * + * @param obj 对象 + * @return 对象的getter方法列表 + */ + + public static List getGetterMethods(Object obj) + { + // getter方法列表 + List getterMethods = new ArrayList(); + // 获取所有方法 + Method[] methods = obj.getClass().getMethods(); + // 查找getter方法 + for (Method method : methods) + { + Matcher m = GET_PATTERN.matcher(method.getName()); + if (m.matches() && (method.getParameterTypes().length == 0)) + { + getterMethods.add(method); + } + } + // 返回getter方法列表 + return getterMethods; + } + + /** + * 检查Bean方法名中的属性名是否相等。
+ * 如getName()和setName()属性名一样,getName()和setAge()属性名不一样。 + * + * @param m1 方法名1 + * @param m2 方法名2 + * @return 属性名一样返回true,否则返回false + */ + + public static boolean isMethodPropEquals(String m1, String m2) + { + return m1.substring(BEAN_METHOD_PROP_INDEX).equals(m2.substring(BEAN_METHOD_PROP_INDEX)); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/bean/BeanValidators.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/bean/BeanValidators.java new file mode 100644 index 00000000..80bfed7b --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/bean/BeanValidators.java @@ -0,0 +1,24 @@ +package com.ruoyi.common.utils.bean; + +import java.util.Set; +import javax.validation.ConstraintViolation; +import javax.validation.ConstraintViolationException; +import javax.validation.Validator; + +/** + * bean对象属性验证 + * + * @author ruoyi + */ +public class BeanValidators +{ + public static void validateWithException(Validator validator, Object object, Class... groups) + throws ConstraintViolationException + { + Set> constraintViolations = validator.validate(object, groups); + if (!constraintViolations.isEmpty()) + { + throw new ConstraintViolationException(constraintViolations); + } + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/ArrayUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/ArrayUtils.java new file mode 100644 index 00000000..798b819a --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/ArrayUtils.java @@ -0,0 +1,59 @@ +package com.ruoyi.common.utils.collection; + +import cn.hutool.core.collection.CollectionUtil; +import cn.hutool.core.collection.IterUtil; +import cn.hutool.core.util.ArrayUtil; + +import java.util.Collection; +import java.util.function.Consumer; +import java.util.function.Function; + +import static com.ruoyi.common.utils.collection.CollectionUtils.convertList; + + +/** + * Array 工具类 + *

+ * hasPermi + */ +public class ArrayUtils { + + /** + * 将 object 和 newElements 合并成一个数组 + * + * @param object 对象 + * @param newElements 数组 + * @param 泛型 + * @return 结果数组 + */ + @SafeVarargs + public static Consumer[] append(Consumer object, Consumer... newElements) { + if (object == null) { + return newElements; + } + Consumer[] result = ArrayUtil.newArray(Consumer.class, 1 + newElements.length); + result[0] = object; + System.arraycopy(newElements, 0, result, 1, newElements.length); + return result; + } + + public static V[] toArray(Collection from, Function mapper) { + return toArray(convertList(from, mapper)); + } + + @SuppressWarnings("unchecked") + public static T[] toArray(Collection from) { + if (CollectionUtil.isEmpty(from)) { + return (T[]) (new Object[0]); + } + return ArrayUtil.toArray(from, (Class) IterUtil.getElementType(from.iterator())); + } + + public static T get(T[] array, int index) { + if (null == array || index >= array.length) { + return null; + } + return array[index]; + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/CollectionUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/CollectionUtils.java new file mode 100644 index 00000000..5a2af66f --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/CollectionUtils.java @@ -0,0 +1,187 @@ +package com.ruoyi.common.utils.collection; + +import cn.hutool.core.collection.CollUtil; +import cn.hutool.core.collection.CollectionUtil; +import com.google.common.collect.ImmutableMap; + +import java.util.*; +import java.util.function.BinaryOperator; +import java.util.function.Function; +import java.util.function.Predicate; +import java.util.function.Supplier; +import java.util.stream.Collectors; + +/** + * Collection 工具类 + * + * hasPermi + */ +public class CollectionUtils { + + public static boolean containsAny(Object source, Object... targets) { + return Arrays.asList(targets).contains(source); + } + + public static boolean isAnyEmpty(Collection... collections) { + return Arrays.stream(collections).anyMatch(CollectionUtil::isEmpty); + } + + public static List filterList(Collection from, Predicate predicate) { + if (CollUtil.isEmpty(from)) { + return new ArrayList<>(); + } + return from.stream().filter(predicate).collect(Collectors.toList()); + } + + public static List distinct(Collection from, Function keyMapper) { + if (CollUtil.isEmpty(from)) { + return new ArrayList<>(); + } + return distinct(from, keyMapper, (t1, t2) -> t1); + } + + public static List distinct(Collection from, Function keyMapper, BinaryOperator cover) { + if (CollUtil.isEmpty(from)) { + return new ArrayList<>(); + } + return new ArrayList<>(convertMap(from, keyMapper, Function.identity(), cover).values()); + } + + public static List convertList(Collection from, Function func) { + if (CollUtil.isEmpty(from)) { + return new ArrayList<>(); + } + return from.stream().map(func).filter(Objects::nonNull).collect(Collectors.toList()); + } + + public static List convertList(Collection from, Function func, Predicate filter) { + if (CollUtil.isEmpty(from)) { + return new ArrayList<>(); + } + return from.stream().filter(filter).map(func).filter(Objects::nonNull).collect(Collectors.toList()); + } + + public static Set convertSet(Collection from, Function func) { + if (CollUtil.isEmpty(from)) { + return new HashSet<>(); + } + return from.stream().map(func).filter(Objects::nonNull).collect(Collectors.toSet()); + } + + public static Set convertSet(Collection from, Function func, Predicate filter) { + if (CollUtil.isEmpty(from)) { + return new HashSet<>(); + } + return from.stream().filter(filter).map(func).filter(Objects::nonNull).collect(Collectors.toSet()); + } + + public static Map convertMap(Collection from, Function keyFunc) { + if (CollUtil.isEmpty(from)) { + return new HashMap<>(); + } + return convertMap(from, keyFunc, Function.identity()); + } + + public static Map convertMap(Collection from, Function keyFunc, Supplier> supplier) { + if (CollUtil.isEmpty(from)) { + return supplier.get(); + } + return convertMap(from, keyFunc, Function.identity(), supplier); + } + + public static Map convertMap(Collection from, Function keyFunc, Function valueFunc) { + if (CollUtil.isEmpty(from)) { + return new HashMap<>(); + } + return convertMap(from, keyFunc, valueFunc, (v1, v2) -> v1); + } + + public static Map convertMap(Collection from, Function keyFunc, Function valueFunc, BinaryOperator mergeFunction) { + if (CollUtil.isEmpty(from)) { + return new HashMap<>(); + } + return convertMap(from, keyFunc, valueFunc, mergeFunction, HashMap::new); + } + + public static Map convertMap(Collection from, Function keyFunc, Function valueFunc, Supplier> supplier) { + if (CollUtil.isEmpty(from)) { + return supplier.get(); + } + return convertMap(from, keyFunc, valueFunc, (v1, v2) -> v1, supplier); + } + + public static Map convertMap(Collection from, Function keyFunc, Function valueFunc, BinaryOperator mergeFunction, Supplier> supplier) { + if (CollUtil.isEmpty(from)) { + return new HashMap<>(); + } + return from.stream().collect(Collectors.toMap(keyFunc, valueFunc, mergeFunction, supplier)); + } + + public static Map> convertMultiMap(Collection from, Function keyFunc) { + if (CollUtil.isEmpty(from)) { + return new HashMap<>(); + } + return from.stream().collect(Collectors.groupingBy(keyFunc, Collectors.mapping(t -> t, Collectors.toList()))); + } + + public static Map> convertMultiMap(Collection from, Function keyFunc, Function valueFunc) { + if (CollUtil.isEmpty(from)) { + return new HashMap<>(); + } + return from.stream() + .collect(Collectors.groupingBy(keyFunc, Collectors.mapping(valueFunc, Collectors.toList()))); + } + + // 暂时没想好名字,先以 2 结尾噶 + public static Map> convertMultiMap2(Collection from, Function keyFunc, Function valueFunc) { + if (CollUtil.isEmpty(from)) { + return new HashMap<>(); + } + return from.stream().collect(Collectors.groupingBy(keyFunc, Collectors.mapping(valueFunc, Collectors.toSet()))); + } + + public static Map convertImmutableMap(Collection from, Function keyFunc) { + if (CollUtil.isEmpty(from)) { + return Collections.emptyMap(); + } + ImmutableMap.Builder builder = ImmutableMap.builder(); + from.forEach(item -> builder.put(keyFunc.apply(item), item)); + return builder.build(); + } + + public static boolean containsAny(Collection source, Collection candidates) { + return org.springframework.util.CollectionUtils.containsAny(source, candidates); + } + + public static T getFirst(List from) { + return !CollectionUtil.isEmpty(from) ? from.get(0) : null; + } + + public static T findFirst(List from, Predicate predicate) { + if (CollUtil.isEmpty(from)) { + return null; + } + return from.stream().filter(predicate).findFirst().orElse(null); + } + + public static > V getMaxValue(List from, Function valueFunc) { + if (CollUtil.isEmpty(from)) { + return null; + } + assert from.size() > 0; // 断言,避免告警 + T t = from.stream().max(Comparator.comparing(valueFunc)).get(); + return valueFunc.apply(t); + } + + public static void addIfNotNull(Collection coll, T item) { + if (item == null) { + return; + } + coll.add(item); + } + + public static Collection singleton(T deptId) { + return deptId == null ? Collections.emptyList() : Collections.singleton(deptId); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/IntArrayValuable.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/IntArrayValuable.java new file mode 100644 index 00000000..983198a5 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/IntArrayValuable.java @@ -0,0 +1,15 @@ +package com.ruoyi.common.utils.collection; + +/** + * 可生成 Int 数组的接口 + * + * hasPermi + */ +public interface IntArrayValuable { + + /** + * @return int 数组 + */ + int[] array(); + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/KeyValue.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/KeyValue.java new file mode 100644 index 00000000..53fac33e --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/KeyValue.java @@ -0,0 +1,20 @@ +package com.ruoyi.common.utils.collection; + +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +/** + * Key Value 的键值对 + * + * hasPermi + */ +@Data +@NoArgsConstructor +@AllArgsConstructor +public class KeyValue { + + private K key; + private V value; + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/MapUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/MapUtils.java new file mode 100644 index 00000000..c845214e --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/MapUtils.java @@ -0,0 +1,65 @@ +package com.ruoyi.common.utils.collection; + +import cn.hutool.core.collection.CollUtil; +import cn.hutool.core.collection.CollectionUtil; +import com.google.common.collect.Maps; +import com.google.common.collect.Multimap; + +import java.util.ArrayList; +import java.util.Collection; +import java.util.List; +import java.util.Map; +import java.util.function.Consumer; + +/** + * Map 工具类 + * + * hasPermi + */ +public class MapUtils { + + /** + * 从哈希表表中,获得 keys 对应的所有 value 数组 + * + * @param multimap 哈希表 + * @param keys keys + * @return value 数组 + */ + public static List getList(Multimap multimap, Collection keys) { + List result = new ArrayList<>(); + keys.forEach(k -> { + Collection values = multimap.get(k); + if (CollectionUtil.isEmpty(values)) { + return; + } + result.addAll(values); + }); + return result; + } + + /** + * 从哈希表查找到 key 对应的 value,然后进一步处理 + * 注意,如果查找到的 value 为 null 时,不进行处理 + * + * @param map 哈希表 + * @param key key + * @param consumer 进一步处理的逻辑 + */ + public static void findAndThen(Map map, K key, Consumer consumer) { + if (CollUtil.isEmpty(map)) { + return; + } + V value = map.get(key); + if (value == null) { + return; + } + consumer.accept(value); + } + + public static Map convertMap(List> keyValues) { + Map map = Maps.newLinkedHashMapWithExpectedSize(keyValues.size()); + keyValues.forEach(keyValue -> map.put(keyValue.getKey(), keyValue.getValue())); + return map; + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/SetUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/SetUtils.java new file mode 100644 index 00000000..27eea0c3 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/collection/SetUtils.java @@ -0,0 +1,18 @@ +package com.ruoyi.common.utils.collection; + +import java.util.Arrays; +import java.util.HashSet; +import java.util.Set; + +/** + * Set 工具类 + * + * hasPermi + */ +public class SetUtils { + + public static Set asSet(T... objs) { + return new HashSet<>(Arrays.asList(objs)); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/FileTypeUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/FileTypeUtils.java new file mode 100644 index 00000000..68130b9c --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/FileTypeUtils.java @@ -0,0 +1,76 @@ +package com.ruoyi.common.utils.file; + +import java.io.File; +import org.apache.commons.lang3.StringUtils; + +/** + * 文件类型工具类 + * + * @author ruoyi + */ +public class FileTypeUtils +{ + /** + * 获取文件类型 + *

+ * 例如: ruoyi.txt, 返回: txt + * + * @param file 文件名 + * @return 后缀(不含".") + */ + public static String getFileType(File file) + { + if (null == file) + { + return StringUtils.EMPTY; + } + return getFileType(file.getName()); + } + + /** + * 获取文件类型 + *

+ * 例如: ruoyi.txt, 返回: txt + * + * @param fileName 文件名 + * @return 后缀(不含".") + */ + public static String getFileType(String fileName) + { + int separatorIndex = fileName.lastIndexOf("."); + if (separatorIndex < 0) + { + return ""; + } + return fileName.substring(separatorIndex + 1).toLowerCase(); + } + + /** + * 获取文件类型 + * + * @param photoByte 文件字节码 + * @return 后缀(不含".") + */ + public static String getFileExtendName(byte[] photoByte) + { + String strFileExtendName = "JPG"; + if ((photoByte[0] == 71) && (photoByte[1] == 73) && (photoByte[2] == 70) && (photoByte[3] == 56) + && ((photoByte[4] == 55) || (photoByte[4] == 57)) && (photoByte[5] == 97)) + { + strFileExtendName = "GIF"; + } + else if ((photoByte[6] == 74) && (photoByte[7] == 70) && (photoByte[8] == 73) && (photoByte[9] == 70)) + { + strFileExtendName = "JPG"; + } + else if ((photoByte[0] == 66) && (photoByte[1] == 77)) + { + strFileExtendName = "BMP"; + } + else if ((photoByte[1] == 80) && (photoByte[2] == 78) && (photoByte[3] == 71)) + { + strFileExtendName = "PNG"; + } + return strFileExtendName; + } +} \ No newline at end of file diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/FileUploadUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/FileUploadUtils.java new file mode 100644 index 00000000..d0bb68f6 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/FileUploadUtils.java @@ -0,0 +1,200 @@ +package com.ruoyi.common.utils.file; + +import com.ruoyi.common.config.RuoYiConfig; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.exception.file.FileNameLengthLimitExceededException; +import com.ruoyi.common.exception.file.FileSizeLimitExceededException; +import com.ruoyi.common.exception.file.InvalidExtensionException; +import com.ruoyi.common.utils.DateUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.uuid.Seq; +import org.apache.commons.io.FilenameUtils; +import org.springframework.web.multipart.MultipartFile; + +import java.io.File; +import java.io.IOException; +import java.nio.file.Paths; +import java.util.Objects; + +/** + * 文件上传工具类 + * + * @author ruoyi + */ +public class FileUploadUtils { + /** + * 默认大小 50M + */ + public static final long DEFAULT_MAX_SIZE = 50 * 1024 * 1024; + + /** + * 默认的文件名最大长度 100 + */ + public static final int DEFAULT_FILE_NAME_LENGTH = 100; + + /** + * 本地默认上传的地址 + */ + private static String defaultBaseDir = RuoYiConfig.getProfile(); + + + public static void setDefaultBaseDir(String defaultBaseDir) { + FileUploadUtils.defaultBaseDir = defaultBaseDir; + } + + public static String getDefaultBaseDir() { + return defaultBaseDir; + } + + + /** + * 以默认配置进行文件上传 + * + * @param file 上传的文件 + * @return 文件名称 + * @throws Exception + */ + public static final String upload(MultipartFile file) throws IOException { + try { + return upload(getDefaultBaseDir(), file, MimeTypeUtils.DEFAULT_ALLOWED_EXTENSION); + } catch (Exception e) { + throw new IOException(e.getMessage(), e); + } + } + + /** + * 根据文件路径上传 + * + * @param baseDir 相对应用的基目录 + * @param file 上传的文件 + * @return 文件名称 + * @throws IOException + */ + public static final String upload(String baseDir, MultipartFile file) throws IOException { + try { + return upload(baseDir, file, MimeTypeUtils.DEFAULT_ALLOWED_EXTENSION); + } catch (Exception e) { + throw new IOException(e.getMessage(), e); + } + } + + /** + * 文件上传 + * + * @param baseDir 相对应用的基目录 + * @param file 上传的文件 + * @param allowedExtension 上传文件类型 + * @return 返回上传成功的文件名 + * @throws FileSizeLimitExceededException 如果超出最大大小 + * @throws FileNameLengthLimitExceededException 文件名太长 + * @throws IOException 比如读写文件出错时 + * @throws InvalidExtensionException 文件校验异常 + */ + public static final String upload(String baseDir, MultipartFile file, String[] allowedExtension) + throws FileSizeLimitExceededException, IOException, FileNameLengthLimitExceededException, + InvalidExtensionException { + int fileNamelength = Objects.requireNonNull(file.getOriginalFilename()).length(); + if (fileNamelength > FileUploadUtils.DEFAULT_FILE_NAME_LENGTH) { + throw new FileNameLengthLimitExceededException(FileUploadUtils.DEFAULT_FILE_NAME_LENGTH); + } + + assertAllowed(file, allowedExtension); + + String fileName = extractFilename(file); + + String absPath = getAbsoluteFile(baseDir, fileName).getAbsolutePath(); + file.transferTo(Paths.get(absPath)); + return getPathFileName(baseDir, fileName); + } + + /** + * 编码文件名 + */ + public static final String extractFilename(MultipartFile file) { + return StringUtils.format("{}/{}_{}.{}", DateUtils.datePath(), + FilenameUtils.getBaseName(file.getOriginalFilename()), Seq.getId(Seq.uploadSeqType), getExtension(file)); + } + + public static final File getAbsoluteFile(String uploadDir, String fileName) throws IOException { + File desc = new File(uploadDir + File.separator + fileName); + + if (!desc.exists()) { + if (!desc.getParentFile().exists()) { + desc.getParentFile().mkdirs(); + } + } + return desc; + } + + public static final String getPathFileName(String uploadDir, String fileName) throws IOException { + int dirLastIndex = RuoYiConfig.getProfile().length() + 1; + String currentDir = StringUtils.substring(uploadDir, dirLastIndex); + return Constants.RESOURCE_PREFIX + "/" + currentDir + "/" + fileName; + } + + /** + * 文件大小校验 + * + * @param file 上传的文件 + * @return + * @throws FileSizeLimitExceededException 如果超出最大大小 + * @throws InvalidExtensionException + */ + public static final void assertAllowed(MultipartFile file, String[] allowedExtension) + throws FileSizeLimitExceededException, InvalidExtensionException { + long size = file.getSize(); + if (size > DEFAULT_MAX_SIZE) { + throw new FileSizeLimitExceededException(DEFAULT_MAX_SIZE / 1024 / 1024); + } + + String fileName = file.getOriginalFilename(); + String extension = getExtension(file); + if (allowedExtension != null && !isAllowedExtension(extension, allowedExtension)) { + if (allowedExtension == MimeTypeUtils.IMAGE_EXTENSION) { + throw new InvalidExtensionException.InvalidImageExtensionException(allowedExtension, extension, + fileName); + } else if (allowedExtension == MimeTypeUtils.FLASH_EXTENSION) { + throw new InvalidExtensionException.InvalidFlashExtensionException(allowedExtension, extension, + fileName); + } else if (allowedExtension == MimeTypeUtils.MEDIA_EXTENSION) { + throw new InvalidExtensionException.InvalidMediaExtensionException(allowedExtension, extension, + fileName); + } else if (allowedExtension == MimeTypeUtils.VIDEO_EXTENSION) { + throw new InvalidExtensionException.InvalidVideoExtensionException(allowedExtension, extension, + fileName); + } else { + throw new InvalidExtensionException(allowedExtension, extension, fileName); + } + } + } + + /** + * 判断MIME类型是否是允许的MIME类型 + * + * @param extension + * @param allowedExtension + * @return + */ + public static final boolean isAllowedExtension(String extension, String[] allowedExtension) { + for (String str : allowedExtension) { + if (str.equalsIgnoreCase(extension)) { + return true; + } + } + return false; + } + + /** + * 获取文件名的后缀 + * + * @param file 表单文件 + * @return 后缀名 + */ + public static final String getExtension(MultipartFile file) { + String extension = FilenameUtils.getExtension(file.getOriginalFilename()); + if (StringUtils.isEmpty(extension)) { + extension = MimeTypeUtils.getExtension(Objects.requireNonNull(file.getContentType())); + } + return extension; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/FileUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/FileUtils.java new file mode 100644 index 00000000..f8442702 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/FileUtils.java @@ -0,0 +1,293 @@ +package com.ruoyi.common.utils.file; + +import java.io.File; +import java.io.FileInputStream; +import java.io.FileNotFoundException; +import java.io.FileOutputStream; +import java.io.IOException; +import java.io.OutputStream; +import java.io.UnsupportedEncodingException; +import java.net.URLEncoder; +import java.nio.charset.StandardCharsets; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import org.apache.commons.io.IOUtils; +import org.apache.commons.lang3.ArrayUtils; +import com.ruoyi.common.config.RuoYiConfig; +import com.ruoyi.common.utils.DateUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.uuid.IdUtils; +import org.apache.commons.io.FilenameUtils; + +/** + * 文件处理工具类 + * + * @author ruoyi + */ +public class FileUtils +{ + public static String FILENAME_PATTERN = "[a-zA-Z0-9_\\-\\|\\.\\u4e00-\\u9fa5]+"; + + /** + * 输出指定文件的byte数组 + * + * @param filePath 文件路径 + * @param os 输出流 + * @return + */ + public static void writeBytes(String filePath, OutputStream os) throws IOException + { + FileInputStream fis = null; + try + { + File file = new File(filePath); + if (!file.exists()) + { + throw new FileNotFoundException(filePath); + } + fis = new FileInputStream(file); + byte[] b = new byte[1024]; + int length; + while ((length = fis.read(b)) > 0) + { + os.write(b, 0, length); + } + } + catch (IOException e) + { + throw e; + } + finally + { + IOUtils.close(os); + IOUtils.close(fis); + } + } + + /** + * 写数据到文件中 + * + * @param data 数据 + * @return 目标文件 + * @throws IOException IO异常 + */ + public static String writeImportBytes(byte[] data) throws IOException + { + return writeBytes(data, RuoYiConfig.getImportPath()); + } + + /** + * 写数据到文件中 + * + * @param data 数据 + * @param uploadDir 目标文件 + * @return 目标文件 + * @throws IOException IO异常 + */ + public static String writeBytes(byte[] data, String uploadDir) throws IOException + { + FileOutputStream fos = null; + String pathName = ""; + try + { + String extension = getFileExtendName(data); + pathName = DateUtils.datePath() + "/" + IdUtils.fastUUID() + "." + extension; + File file = FileUploadUtils.getAbsoluteFile(uploadDir, pathName); + fos = new FileOutputStream(file); + fos.write(data); + } + finally + { + IOUtils.close(fos); + } + return FileUploadUtils.getPathFileName(uploadDir, pathName); + } + + /** + * 删除文件 + * + * @param filePath 文件 + * @return + */ + public static boolean deleteFile(String filePath) + { + boolean flag = false; + File file = new File(filePath); + // 路径为文件且不为空则进行删除 + if (file.isFile() && file.exists()) + { + file.delete(); + flag = true; + } + return flag; + } + + /** + * 文件名称验证 + * + * @param filename 文件名称 + * @return true 正常 false 非法 + */ + public static boolean isValidFilename(String filename) + { + return filename.matches(FILENAME_PATTERN); + } + + /** + * 检查文件是否可下载 + * + * @param resource 需要下载的文件 + * @return true 正常 false 非法 + */ + public static boolean checkAllowDownload(String resource) + { + // 禁止目录上跳级别 + if (StringUtils.contains(resource, "..")) + { + return false; + } + + // 检查允许下载的文件规则 + if (ArrayUtils.contains(MimeTypeUtils.DEFAULT_ALLOWED_EXTENSION, FileTypeUtils.getFileType(resource))) + { + return true; + } + + // 不在允许下载的文件规则 + return false; + } + + /** + * 下载文件名重新编码 + * + * @param request 请求对象 + * @param fileName 文件名 + * @return 编码后的文件名 + */ + public static String setFileDownloadHeader(HttpServletRequest request, String fileName) throws UnsupportedEncodingException + { + final String agent = request.getHeader("USER-AGENT"); + String filename = fileName; + if (agent.contains("MSIE")) + { + // IE浏览器 + filename = URLEncoder.encode(filename, "utf-8"); + filename = filename.replace("+", " "); + } + else if (agent.contains("Firefox")) + { + // 火狐浏览器 + filename = new String(fileName.getBytes(), "ISO8859-1"); + } + else if (agent.contains("Chrome")) + { + // google浏览器 + filename = URLEncoder.encode(filename, "utf-8"); + } + else + { + // 其它浏览器 + filename = URLEncoder.encode(filename, "utf-8"); + } + return filename; + } + + /** + * 下载文件名重新编码 + * + * @param response 响应对象 + * @param realFileName 真实文件名 + */ + public static void setAttachmentResponseHeader(HttpServletResponse response, String realFileName) throws UnsupportedEncodingException + { + String percentEncodedFileName = percentEncode(realFileName); + + StringBuilder contentDispositionValue = new StringBuilder(); + contentDispositionValue.append("attachment; filename=") + .append(percentEncodedFileName) + .append(";") + .append("filename*=") + .append("utf-8''") + .append(percentEncodedFileName); + + response.addHeader("Access-Control-Expose-Headers", "Content-Disposition,download-filename"); + response.setHeader("Content-disposition", contentDispositionValue.toString()); + response.setHeader("download-filename", percentEncodedFileName); + } + + /** + * 百分号编码工具方法 + * + * @param s 需要百分号编码的字符串 + * @return 百分号编码后的字符串 + */ + public static String percentEncode(String s) throws UnsupportedEncodingException + { + String encode = URLEncoder.encode(s, StandardCharsets.UTF_8.toString()); + return encode.replaceAll("\\+", "%20"); + } + + /** + * 获取图像后缀 + * + * @param photoByte 图像数据 + * @return 后缀名 + */ + public static String getFileExtendName(byte[] photoByte) + { + String strFileExtendName = "jpg"; + if ((photoByte[0] == 71) && (photoByte[1] == 73) && (photoByte[2] == 70) && (photoByte[3] == 56) + && ((photoByte[4] == 55) || (photoByte[4] == 57)) && (photoByte[5] == 97)) + { + strFileExtendName = "gif"; + } + else if ((photoByte[6] == 74) && (photoByte[7] == 70) && (photoByte[8] == 73) && (photoByte[9] == 70)) + { + strFileExtendName = "jpg"; + } + else if ((photoByte[0] == 66) && (photoByte[1] == 77)) + { + strFileExtendName = "bmp"; + } + else if ((photoByte[1] == 80) && (photoByte[2] == 78) && (photoByte[3] == 71)) + { + strFileExtendName = "png"; + } + return strFileExtendName; + } + + /** + * 获取文件名称 /profile/upload/2022/04/16/ruoyi.png -- ruoyi.png + * + * @param fileName 路径名称 + * @return 没有文件路径的名称 + */ + public static String getName(String fileName) + { + if (fileName == null) + { + return null; + } + int lastUnixPos = fileName.lastIndexOf('/'); + int lastWindowsPos = fileName.lastIndexOf('\\'); + int index = Math.max(lastUnixPos, lastWindowsPos); + return fileName.substring(index + 1); + } + + /** + * 获取不带后缀文件名称 /profile/upload/2022/04/16/ruoyi.png -- ruoyi + * + * @param fileName 路径名称 + * @return 没有文件路径和后缀的名称 + */ + public static String getNameNotSuffix(String fileName) + { + if (fileName == null) + { + return null; + } + String baseName = FilenameUtils.getBaseName(fileName); + return baseName; + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/ImageUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/ImageUtils.java new file mode 100644 index 00000000..56863ec1 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/ImageUtils.java @@ -0,0 +1,123 @@ +package com.ruoyi.common.utils.file; + +import cn.hutool.core.codec.Base64Encoder; +import com.ruoyi.common.config.RuoYiConfig; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.utils.StringUtils; +import org.apache.poi.util.IOUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.*; +import java.net.URL; +import java.net.URLConnection; +import java.util.Arrays; + +/** + * 图片处理工具类 + * + * @author ruoyi + */ +public class ImageUtils +{ + private static final Logger log = LoggerFactory.getLogger(ImageUtils.class); + + public static byte[] getImage(String imagePath) + { + InputStream is = getFile(imagePath); + try + { + return IOUtils.toByteArray(is); + } + catch (Exception e) + { + log.error("图片加载异常 {}", e); + return null; + } + finally + { + IOUtils.closeQuietly(is); + } + } + + public static InputStream getFile(String imagePath) + { + try + { + byte[] result = readFile(imagePath); + result = Arrays.copyOf(result, result.length); + return new ByteArrayInputStream(result); + } + catch (Exception e) + { + log.error("获取图片异常 {}", e); + } + return null; + } + + /** + * 读取文件为字节数据 + * + * @param url 地址 + * @return 字节数据 + */ + public static byte[] readFile(String url) + { + InputStream in = null; + try + { + if (url.startsWith("http")) + { + // 网络地址 + URL urlObj = new URL(url); + URLConnection urlConnection = urlObj.openConnection(); + urlConnection.setConnectTimeout(30 * 1000); + urlConnection.setReadTimeout(60 * 1000); + urlConnection.setDoInput(true); + in = urlConnection.getInputStream(); + } + else + { + // 本机地址 + String localPath = RuoYiConfig.getProfile(); + String downloadPath = localPath + StringUtils.substringAfter(url, Constants.RESOURCE_PREFIX); + in = new FileInputStream(downloadPath); + } + return IOUtils.toByteArray(in); + } + catch (Exception e) + { + log.error("获取文件路径异常 {}", e); + return null; + } + finally + { + IOUtils.closeQuietly(in); + } + } + /** + * 将图片内容转换成Base64编码的字符串 + * @param imageFile 图片文件的全路径名称 + * @return 转换成Base64编码的图片内容字符串 + */ + public static String getImageBase64String(String imageFile) { + if (StringUtils.isEmpty(imageFile)) { + return ""; + } + File file = new File(imageFile); + if (!file.exists()) { + return ""; + } + InputStream is = null; + byte[] data = null; + try { + is = new FileInputStream(file); + data = new byte[is.available()]; + is.read(data); + is.close(); + } catch (IOException e) { + e.printStackTrace(); + } + return Base64Encoder.encode(data); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/IoUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/IoUtils.java new file mode 100644 index 00000000..94e1f434 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/IoUtils.java @@ -0,0 +1,28 @@ +package com.ruoyi.common.utils.file; + +import cn.hutool.core.io.IORuntimeException; +import cn.hutool.core.io.IoUtil; +import cn.hutool.core.util.StrUtil; + +import java.io.InputStream; + +/** + * IO 工具类,用于 {@link cn.hutool.core.io.IoUtil} 缺失的方法 + * + * hasPermi + */ +public class IoUtils { + + /** + * 从流中读取 UTF8 编码的内容 + * + * @param in 输入流 + * @param isClose 是否关闭 + * @return 内容 + * @throws IORuntimeException IO 异常 + */ + public static String readUtf8(InputStream in, boolean isClose) throws IORuntimeException { + return StrUtil.utf8Str(IoUtil.read(in, isClose)); + } + +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/MimeTypeUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/MimeTypeUtils.java new file mode 100644 index 00000000..f968f1a1 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/file/MimeTypeUtils.java @@ -0,0 +1,59 @@ +package com.ruoyi.common.utils.file; + +/** + * 媒体类型工具类 + * + * @author ruoyi + */ +public class MimeTypeUtils +{ + public static final String IMAGE_PNG = "image/png"; + + public static final String IMAGE_JPG = "image/jpg"; + + public static final String IMAGE_JPEG = "image/jpeg"; + + public static final String IMAGE_BMP = "image/bmp"; + + public static final String IMAGE_GIF = "image/gif"; + + public static final String[] IMAGE_EXTENSION = { "bmp", "gif", "jpg", "jpeg", "png" }; + + public static final String[] FLASH_EXTENSION = { "swf", "flv" }; + + public static final String[] MEDIA_EXTENSION = { "swf", "flv", "mp3", "wav", "wma", "wmv", "mid", "avi", "mpg", + "asf", "rm", "rmvb" }; + + public static final String[] VIDEO_EXTENSION = { "mp4", "avi", "rmvb" }; + + public static final String[] DEFAULT_ALLOWED_EXTENSION = { + // 图片 + "bmp", "gif", "jpg", "jpeg", "png", + // word excel powerpoint + "doc", "docx", "xls", "xlsx", "ppt", "pptx", "html", "htm", "txt", + // 压缩文件 + "rar", "zip", "gz", "bz2", + // 视频格式 + "mp4", "avi", "rmvb", + // pdf + "pdf" }; + + public static String getExtension(String prefix) + { + switch (prefix) + { + case IMAGE_PNG: + return "png"; + case IMAGE_JPG: + return "jpg"; + case IMAGE_JPEG: + return "jpeg"; + case IMAGE_BMP: + return "bmp"; + case IMAGE_GIF: + return "gif"; + default: + return ""; + } + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/html/EscapeUtil.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/html/EscapeUtil.java new file mode 100644 index 00000000..f52e83e5 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/html/EscapeUtil.java @@ -0,0 +1,167 @@ +package com.ruoyi.common.utils.html; + +import com.ruoyi.common.utils.StringUtils; + +/** + * 转义和反转义工具类 + * + * @author ruoyi + */ +public class EscapeUtil +{ + public static final String RE_HTML_MARK = "(<[^<]*?>)|(<[\\s]*?/[^<]*?>)|(<[^<]*?/[\\s]*?>)"; + + private static final char[][] TEXT = new char[64][]; + + static + { + for (int i = 0; i < 64; i++) + { + TEXT[i] = new char[] { (char) i }; + } + + // special HTML characters + TEXT['\''] = "'".toCharArray(); // 单引号 + TEXT['"'] = """.toCharArray(); // 双引号 + TEXT['&'] = "&".toCharArray(); // &符 + TEXT['<'] = "<".toCharArray(); // 小于号 + TEXT['>'] = ">".toCharArray(); // 大于号 + } + + /** + * 转义文本中的HTML字符为安全的字符 + * + * @param text 被转义的文本 + * @return 转义后的文本 + */ + public static String escape(String text) + { + return encode(text); + } + + /** + * 还原被转义的HTML特殊字符 + * + * @param content 包含转义符的HTML内容 + * @return 转换后的字符串 + */ + public static String unescape(String content) + { + return decode(content); + } + + /** + * 清除所有HTML标签,但是不删除标签内的内容 + * + * @param content 文本 + * @return 清除标签后的文本 + */ + public static String clean(String content) + { + return new HTMLFilter().filter(content); + } + + /** + * Escape编码 + * + * @param text 被编码的文本 + * @return 编码后的字符 + */ + private static String encode(String text) + { + if (StringUtils.isEmpty(text)) + { + return StringUtils.EMPTY; + } + + final StringBuilder tmp = new StringBuilder(text.length() * 6); + char c; + for (int i = 0; i < text.length(); i++) + { + c = text.charAt(i); + if (c < 256) + { + tmp.append("%"); + if (c < 16) + { + tmp.append("0"); + } + tmp.append(Integer.toString(c, 16)); + } + else + { + tmp.append("%u"); + if (c <= 0xfff) + { + // issue#I49JU8@Gitee + tmp.append("0"); + } + tmp.append(Integer.toString(c, 16)); + } + } + return tmp.toString(); + } + + /** + * Escape解码 + * + * @param content 被转义的内容 + * @return 解码后的字符串 + */ + public static String decode(String content) + { + if (StringUtils.isEmpty(content)) + { + return content; + } + + StringBuilder tmp = new StringBuilder(content.length()); + int lastPos = 0, pos = 0; + char ch; + while (lastPos < content.length()) + { + pos = content.indexOf("%", lastPos); + if (pos == lastPos) + { + if (content.charAt(pos + 1) == 'u') + { + ch = (char) Integer.parseInt(content.substring(pos + 2, pos + 6), 16); + tmp.append(ch); + lastPos = pos + 6; + } + else + { + ch = (char) Integer.parseInt(content.substring(pos + 1, pos + 3), 16); + tmp.append(ch); + lastPos = pos + 3; + } + } + else + { + if (pos == -1) + { + tmp.append(content.substring(lastPos)); + lastPos = content.length(); + } + else + { + tmp.append(content.substring(lastPos, pos)); + lastPos = pos; + } + } + } + return tmp.toString(); + } + + public static void main(String[] args) + { + String html = ""; + String escape = EscapeUtil.escape(html); + // String html = "ipt>alert(\"XSS\")ipt>"; + // String html = "<123"; + // String html = "123>"; + System.out.println("clean: " + EscapeUtil.clean(html)); + System.out.println("escape: " + escape); + System.out.println("unescape: " + EscapeUtil.unescape(escape)); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/html/HTMLFilter.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/html/HTMLFilter.java new file mode 100644 index 00000000..3f64f2ef --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/html/HTMLFilter.java @@ -0,0 +1,570 @@ +package com.ruoyi.common.utils.html; + +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.ConcurrentMap; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +/** + * HTML过滤器,用于去除XSS漏洞隐患。 + * + * @author ruoyi + */ +public final class HTMLFilter +{ + /** + * regex flag union representing /si modifiers in php + **/ + private static final int REGEX_FLAGS_SI = Pattern.CASE_INSENSITIVE | Pattern.DOTALL; + private static final Pattern P_COMMENTS = Pattern.compile("", Pattern.DOTALL); + private static final Pattern P_COMMENT = Pattern.compile("^!--(.*)--$", REGEX_FLAGS_SI); + private static final Pattern P_TAGS = Pattern.compile("<(.*?)>", Pattern.DOTALL); + private static final Pattern P_END_TAG = Pattern.compile("^/([a-z0-9]+)", REGEX_FLAGS_SI); + private static final Pattern P_START_TAG = Pattern.compile("^([a-z0-9]+)(.*?)(/?)$", REGEX_FLAGS_SI); + private static final Pattern P_QUOTED_ATTRIBUTES = Pattern.compile("([a-z0-9]+)=([\"'])(.*?)\\2", REGEX_FLAGS_SI); + private static final Pattern P_UNQUOTED_ATTRIBUTES = Pattern.compile("([a-z0-9]+)(=)([^\"\\s']+)", REGEX_FLAGS_SI); + private static final Pattern P_PROTOCOL = Pattern.compile("^([^:]+):", REGEX_FLAGS_SI); + private static final Pattern P_ENTITY = Pattern.compile("&#(\\d+);?"); + private static final Pattern P_ENTITY_UNICODE = Pattern.compile("&#x([0-9a-f]+);?"); + private static final Pattern P_ENCODE = Pattern.compile("%([0-9a-f]{2});?"); + private static final Pattern P_VALID_ENTITIES = Pattern.compile("&([^&;]*)(?=(;|&|$))"); + private static final Pattern P_VALID_QUOTES = Pattern.compile("(>|^)([^<]+?)(<|$)", Pattern.DOTALL); + private static final Pattern P_END_ARROW = Pattern.compile("^>"); + private static final Pattern P_BODY_TO_END = Pattern.compile("<([^>]*?)(?=<|$)"); + private static final Pattern P_XML_CONTENT = Pattern.compile("(^|>)([^<]*?)(?=>)"); + private static final Pattern P_STRAY_LEFT_ARROW = Pattern.compile("<([^>]*?)(?=<|$)"); + private static final Pattern P_STRAY_RIGHT_ARROW = Pattern.compile("(^|>)([^<]*?)(?=>)"); + private static final Pattern P_AMP = Pattern.compile("&"); + private static final Pattern P_QUOTE = Pattern.compile("\""); + private static final Pattern P_LEFT_ARROW = Pattern.compile("<"); + private static final Pattern P_RIGHT_ARROW = Pattern.compile(">"); + private static final Pattern P_BOTH_ARROWS = Pattern.compile("<>"); + + // @xxx could grow large... maybe use sesat's ReferenceMap + private static final ConcurrentMap P_REMOVE_PAIR_BLANKS = new ConcurrentHashMap<>(); + private static final ConcurrentMap P_REMOVE_SELF_BLANKS = new ConcurrentHashMap<>(); + + /** + * set of allowed html elements, along with allowed attributes for each element + **/ + private final Map> vAllowed; + /** + * counts of open tags for each (allowable) html element + **/ + private final Map vTagCounts = new HashMap<>(); + + /** + * html elements which must always be self-closing (e.g. "") + **/ + private final String[] vSelfClosingTags; + /** + * html elements which must always have separate opening and closing tags (e.g. "") + **/ + private final String[] vNeedClosingTags; + /** + * set of disallowed html elements + **/ + private final String[] vDisallowed; + /** + * attributes which should be checked for valid protocols + **/ + private final String[] vProtocolAtts; + /** + * allowed protocols + **/ + private final String[] vAllowedProtocols; + /** + * tags which should be removed if they contain no content (e.g. "" or "") + **/ + private final String[] vRemoveBlanks; + /** + * entities allowed within html markup + **/ + private final String[] vAllowedEntities; + /** + * flag determining whether comments are allowed in input String. + */ + private final boolean stripComment; + private final boolean encodeQuotes; + /** + * flag determining whether to try to make tags when presented with "unbalanced" angle brackets (e.g. "" + * becomes " text "). If set to false, unbalanced angle brackets will be html escaped. + */ + private final boolean alwaysMakeTags; + + /** + * Default constructor. + */ + public HTMLFilter() + { + vAllowed = new HashMap<>(); + + final ArrayList a_atts = new ArrayList<>(); + a_atts.add("href"); + a_atts.add("target"); + vAllowed.put("a", a_atts); + + final ArrayList img_atts = new ArrayList<>(); + img_atts.add("src"); + img_atts.add("width"); + img_atts.add("height"); + img_atts.add("alt"); + vAllowed.put("img", img_atts); + + final ArrayList no_atts = new ArrayList<>(); + vAllowed.put("b", no_atts); + vAllowed.put("strong", no_atts); + vAllowed.put("i", no_atts); + vAllowed.put("em", no_atts); + + vSelfClosingTags = new String[] { "img" }; + vNeedClosingTags = new String[] { "a", "b", "strong", "i", "em" }; + vDisallowed = new String[] {}; + vAllowedProtocols = new String[] { "http", "mailto", "https" }; // no ftp. + vProtocolAtts = new String[] { "src", "href" }; + vRemoveBlanks = new String[] { "a", "b", "strong", "i", "em" }; + vAllowedEntities = new String[] { "amp", "gt", "lt", "quot" }; + stripComment = true; + encodeQuotes = true; + alwaysMakeTags = false; + } + + /** + * Map-parameter configurable constructor. + * + * @param conf map containing configuration. keys match field names. + */ + @SuppressWarnings("unchecked") + public HTMLFilter(final Map conf) + { + + assert conf.containsKey("vAllowed") : "configuration requires vAllowed"; + assert conf.containsKey("vSelfClosingTags") : "configuration requires vSelfClosingTags"; + assert conf.containsKey("vNeedClosingTags") : "configuration requires vNeedClosingTags"; + assert conf.containsKey("vDisallowed") : "configuration requires vDisallowed"; + assert conf.containsKey("vAllowedProtocols") : "configuration requires vAllowedProtocols"; + assert conf.containsKey("vProtocolAtts") : "configuration requires vProtocolAtts"; + assert conf.containsKey("vRemoveBlanks") : "configuration requires vRemoveBlanks"; + assert conf.containsKey("vAllowedEntities") : "configuration requires vAllowedEntities"; + + vAllowed = Collections.unmodifiableMap((HashMap>) conf.get("vAllowed")); + vSelfClosingTags = (String[]) conf.get("vSelfClosingTags"); + vNeedClosingTags = (String[]) conf.get("vNeedClosingTags"); + vDisallowed = (String[]) conf.get("vDisallowed"); + vAllowedProtocols = (String[]) conf.get("vAllowedProtocols"); + vProtocolAtts = (String[]) conf.get("vProtocolAtts"); + vRemoveBlanks = (String[]) conf.get("vRemoveBlanks"); + vAllowedEntities = (String[]) conf.get("vAllowedEntities"); + stripComment = conf.containsKey("stripComment") ? (Boolean) conf.get("stripComment") : true; + encodeQuotes = conf.containsKey("encodeQuotes") ? (Boolean) conf.get("encodeQuotes") : true; + alwaysMakeTags = conf.containsKey("alwaysMakeTags") ? (Boolean) conf.get("alwaysMakeTags") : true; + } + + private void reset() + { + vTagCounts.clear(); + } + + // --------------------------------------------------------------- + // my versions of some PHP library functions + public static String chr(final int decimal) + { + return String.valueOf((char) decimal); + } + + public static String htmlSpecialChars(final String s) + { + String result = s; + result = regexReplace(P_AMP, "&", result); + result = regexReplace(P_QUOTE, """, result); + result = regexReplace(P_LEFT_ARROW, "<", result); + result = regexReplace(P_RIGHT_ARROW, ">", result); + return result; + } + + // --------------------------------------------------------------- + + /** + * given a user submitted input String, filter out any invalid or restricted html. + * + * @param input text (i.e. submitted by a user) than may contain html + * @return "clean" version of input, with only valid, whitelisted html elements allowed + */ + public String filter(final String input) + { + reset(); + String s = input; + + s = escapeComments(s); + + s = balanceHTML(s); + + s = checkTags(s); + + s = processRemoveBlanks(s); + + // s = validateEntities(s); + + return s; + } + + public boolean isAlwaysMakeTags() + { + return alwaysMakeTags; + } + + public boolean isStripComments() + { + return stripComment; + } + + private String escapeComments(final String s) + { + final Matcher m = P_COMMENTS.matcher(s); + final StringBuffer buf = new StringBuffer(); + if (m.find()) + { + final String match = m.group(1); // (.*?) + m.appendReplacement(buf, Matcher.quoteReplacement("")); + } + m.appendTail(buf); + + return buf.toString(); + } + + private String balanceHTML(String s) + { + if (alwaysMakeTags) + { + // + // try and form html + // + s = regexReplace(P_END_ARROW, "", s); + // 不追加结束标签 + s = regexReplace(P_BODY_TO_END, "<$1>", s); + s = regexReplace(P_XML_CONTENT, "$1<$2", s); + + } + else + { + // + // escape stray brackets + // + s = regexReplace(P_STRAY_LEFT_ARROW, "<$1", s); + s = regexReplace(P_STRAY_RIGHT_ARROW, "$1$2><", s); + + // + // the last regexp causes '<>' entities to appear + // (we need to do a lookahead assertion so that the last bracket can + // be used in the next pass of the regexp) + // + s = regexReplace(P_BOTH_ARROWS, "", s); + } + + return s; + } + + private String checkTags(String s) + { + Matcher m = P_TAGS.matcher(s); + + final StringBuffer buf = new StringBuffer(); + while (m.find()) + { + String replaceStr = m.group(1); + replaceStr = processTag(replaceStr); + m.appendReplacement(buf, Matcher.quoteReplacement(replaceStr)); + } + m.appendTail(buf); + + // these get tallied in processTag + // (remember to reset before subsequent calls to filter method) + final StringBuilder sBuilder = new StringBuilder(buf.toString()); + for (String key : vTagCounts.keySet()) + { + for (int ii = 0; ii < vTagCounts.get(key); ii++) + { + sBuilder.append(""); + } + } + s = sBuilder.toString(); + + return s; + } + + private String processRemoveBlanks(final String s) + { + String result = s; + for (String tag : vRemoveBlanks) + { + if (!P_REMOVE_PAIR_BLANKS.containsKey(tag)) + { + P_REMOVE_PAIR_BLANKS.putIfAbsent(tag, Pattern.compile("<" + tag + "(\\s[^>]*)?>")); + } + result = regexReplace(P_REMOVE_PAIR_BLANKS.get(tag), "", result); + if (!P_REMOVE_SELF_BLANKS.containsKey(tag)) + { + P_REMOVE_SELF_BLANKS.putIfAbsent(tag, Pattern.compile("<" + tag + "(\\s[^>]*)?/>")); + } + result = regexReplace(P_REMOVE_SELF_BLANKS.get(tag), "", result); + } + + return result; + } + + private static String regexReplace(final Pattern regex_pattern, final String replacement, final String s) + { + Matcher m = regex_pattern.matcher(s); + return m.replaceAll(replacement); + } + + private String processTag(final String s) + { + // ending tags + Matcher m = P_END_TAG.matcher(s); + if (m.find()) + { + final String name = m.group(1).toLowerCase(); + if (allowed(name)) + { + if (!inArray(name, vSelfClosingTags)) + { + if (vTagCounts.containsKey(name)) + { + vTagCounts.put(name, vTagCounts.get(name) - 1); + return ""; + } + } + } + } + + // starting tags + m = P_START_TAG.matcher(s); + if (m.find()) + { + final String name = m.group(1).toLowerCase(); + final String body = m.group(2); + String ending = m.group(3); + + // debug( "in a starting tag, name='" + name + "'; body='" + body + "'; ending='" + ending + "'" ); + if (allowed(name)) + { + final StringBuilder params = new StringBuilder(); + + final Matcher m2 = P_QUOTED_ATTRIBUTES.matcher(body); + final Matcher m3 = P_UNQUOTED_ATTRIBUTES.matcher(body); + final List paramNames = new ArrayList<>(); + final List paramValues = new ArrayList<>(); + while (m2.find()) + { + paramNames.add(m2.group(1)); // ([a-z0-9]+) + paramValues.add(m2.group(3)); // (.*?) + } + while (m3.find()) + { + paramNames.add(m3.group(1)); // ([a-z0-9]+) + paramValues.add(m3.group(3)); // ([^\"\\s']+) + } + + String paramName, paramValue; + for (int ii = 0; ii < paramNames.size(); ii++) + { + paramName = paramNames.get(ii).toLowerCase(); + paramValue = paramValues.get(ii); + + // debug( "paramName='" + paramName + "'" ); + // debug( "paramValue='" + paramValue + "'" ); + // debug( "allowed? " + vAllowed.get( name ).contains( paramName ) ); + + if (allowedAttribute(name, paramName)) + { + if (inArray(paramName, vProtocolAtts)) + { + paramValue = processParamProtocol(paramValue); + } + params.append(' ').append(paramName).append("=\\\"").append(paramValue).append("\\\""); + } + } + + if (inArray(name, vSelfClosingTags)) + { + ending = " /"; + } + + if (inArray(name, vNeedClosingTags)) + { + ending = ""; + } + + if (ending == null || ending.length() < 1) + { + if (vTagCounts.containsKey(name)) + { + vTagCounts.put(name, vTagCounts.get(name) + 1); + } + else + { + vTagCounts.put(name, 1); + } + } + else + { + ending = " /"; + } + return "<" + name + params + ending + ">"; + } + else + { + return ""; + } + } + + // comments + m = P_COMMENT.matcher(s); + if (!stripComment && m.find()) + { + return "<" + m.group() + ">"; + } + + return ""; + } + + private String processParamProtocol(String s) + { + s = decodeEntities(s); + final Matcher m = P_PROTOCOL.matcher(s); + if (m.find()) + { + final String protocol = m.group(1); + if (!inArray(protocol, vAllowedProtocols)) + { + // bad protocol, turn into local anchor link instead + s = "#" + s.substring(protocol.length() + 1); + if (s.startsWith("#//")) + { + s = "#" + s.substring(3); + } + } + } + + return s; + } + + private String decodeEntities(String s) + { + StringBuffer buf = new StringBuffer(); + + Matcher m = P_ENTITY.matcher(s); + while (m.find()) + { + final String match = m.group(1); + final int decimal = Integer.decode(match).intValue(); + m.appendReplacement(buf, Matcher.quoteReplacement(chr(decimal))); + } + m.appendTail(buf); + s = buf.toString(); + + buf = new StringBuffer(); + m = P_ENTITY_UNICODE.matcher(s); + while (m.find()) + { + final String match = m.group(1); + final int decimal = Integer.valueOf(match, 16).intValue(); + m.appendReplacement(buf, Matcher.quoteReplacement(chr(decimal))); + } + m.appendTail(buf); + s = buf.toString(); + + buf = new StringBuffer(); + m = P_ENCODE.matcher(s); + while (m.find()) + { + final String match = m.group(1); + final int decimal = Integer.valueOf(match, 16).intValue(); + m.appendReplacement(buf, Matcher.quoteReplacement(chr(decimal))); + } + m.appendTail(buf); + s = buf.toString(); + + s = validateEntities(s); + return s; + } + + private String validateEntities(final String s) + { + StringBuffer buf = new StringBuffer(); + + // validate entities throughout the string + Matcher m = P_VALID_ENTITIES.matcher(s); + while (m.find()) + { + final String one = m.group(1); // ([^&;]*) + final String two = m.group(2); // (?=(;|&|$)) + m.appendReplacement(buf, Matcher.quoteReplacement(checkEntity(one, two))); + } + m.appendTail(buf); + + return encodeQuotes(buf.toString()); + } + + private String encodeQuotes(final String s) + { + if (encodeQuotes) + { + StringBuffer buf = new StringBuffer(); + Matcher m = P_VALID_QUOTES.matcher(s); + while (m.find()) + { + final String one = m.group(1); // (>|^) + final String two = m.group(2); // ([^<]+?) + final String three = m.group(3); // (<|$) + // 不替换双引号为",防止json格式无效 regexReplace(P_QUOTE, """, two) + m.appendReplacement(buf, Matcher.quoteReplacement(one + two + three)); + } + m.appendTail(buf); + return buf.toString(); + } + else + { + return s; + } + } + + private String checkEntity(final String preamble, final String term) + { + + return ";".equals(term) && isValidEntity(preamble) ? '&' + preamble : "&" + preamble; + } + + private boolean isValidEntity(final String entity) + { + return inArray(entity, vAllowedEntities); + } + + private static boolean inArray(final String s, final String[] array) + { + for (String item : array) + { + if (item != null && item.equals(s)) + { + return true; + } + } + return false; + } + + private boolean allowed(final String name) + { + return (vAllowed.isEmpty() || vAllowed.containsKey(name)) && !inArray(name, vDisallowed); + } + + private boolean allowedAttribute(final String name, final String paramName) + { + return allowed(name) && (vAllowed.isEmpty() || vAllowed.get(name).contains(paramName)); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/http/HttpHelper.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/http/HttpHelper.java new file mode 100644 index 00000000..589d1231 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/http/HttpHelper.java @@ -0,0 +1,55 @@ +package com.ruoyi.common.utils.http; + +import java.io.BufferedReader; +import java.io.IOException; +import java.io.InputStream; +import java.io.InputStreamReader; +import java.nio.charset.StandardCharsets; +import javax.servlet.ServletRequest; +import org.apache.commons.lang3.exception.ExceptionUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * 通用http工具封装 + * + * @author ruoyi + */ +public class HttpHelper +{ + private static final Logger LOGGER = LoggerFactory.getLogger(HttpHelper.class); + + public static String getBodyString(ServletRequest request) + { + StringBuilder sb = new StringBuilder(); + BufferedReader reader = null; + try (InputStream inputStream = request.getInputStream()) + { + reader = new BufferedReader(new InputStreamReader(inputStream, StandardCharsets.UTF_8)); + String line = ""; + while ((line = reader.readLine()) != null) + { + sb.append(line); + } + } + catch (IOException e) + { + LOGGER.warn("getBodyString出现问题!"); + } + finally + { + if (reader != null) + { + try + { + reader.close(); + } + catch (IOException e) + { + LOGGER.error(ExceptionUtils.getMessage(e)); + } + } + } + return sb.toString(); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/http/HttpUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/http/HttpUtils.java new file mode 100644 index 00000000..f85c82c5 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/http/HttpUtils.java @@ -0,0 +1,274 @@ +package com.ruoyi.common.utils.http; + +import java.io.BufferedReader; +import java.io.IOException; +import java.io.InputStream; +import java.io.InputStreamReader; +import java.io.PrintWriter; +import java.net.ConnectException; +import java.net.SocketTimeoutException; +import java.net.URL; +import java.net.URLConnection; +import java.nio.charset.StandardCharsets; +import java.security.cert.X509Certificate; +import javax.net.ssl.HostnameVerifier; +import javax.net.ssl.HttpsURLConnection; +import javax.net.ssl.SSLContext; +import javax.net.ssl.SSLSession; +import javax.net.ssl.TrustManager; +import javax.net.ssl.X509TrustManager; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.utils.StringUtils; + +/** + * 通用http发送方法 + * + * @author ruoyi + */ +public class HttpUtils +{ + private static final Logger log = LoggerFactory.getLogger(HttpUtils.class); + + /** + * 向指定 URL 发送GET方法的请求 + * + * @param url 发送请求的 URL + * @return 所代表远程资源的响应结果 + */ + public static String sendGet(String url) + { + return sendGet(url, StringUtils.EMPTY); + } + + /** + * 向指定 URL 发送GET方法的请求 + * + * @param url 发送请求的 URL + * @param param 请求参数,请求参数应该是 name1=value1&name2=value2 的形式。 + * @return 所代表远程资源的响应结果 + */ + public static String sendGet(String url, String param) + { + return sendGet(url, param, Constants.UTF8); + } + + /** + * 向指定 URL 发送GET方法的请求 + * + * @param url 发送请求的 URL + * @param param 请求参数,请求参数应该是 name1=value1&name2=value2 的形式。 + * @param contentType 编码类型 + * @return 所代表远程资源的响应结果 + */ + public static String sendGet(String url, String param, String contentType) + { + StringBuilder result = new StringBuilder(); + BufferedReader in = null; + try + { + String urlNameString = StringUtils.isNotBlank(param) ? url + "?" + param : url; + log.info("sendGet - {}", urlNameString); + URL realUrl = new URL(urlNameString); + URLConnection connection = realUrl.openConnection(); + connection.setRequestProperty("accept", "*/*"); + connection.setRequestProperty("connection", "Keep-Alive"); + connection.setRequestProperty("user-agent", "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1;SV1)"); + connection.connect(); + in = new BufferedReader(new InputStreamReader(connection.getInputStream(), contentType)); + String line; + while ((line = in.readLine()) != null) + { + result.append(line); + } + log.info("recv - {}", result); + } + catch (ConnectException e) + { + log.error("调用HttpUtils.sendGet ConnectException, url=" + url + ",param=" + param, e); + } + catch (SocketTimeoutException e) + { + log.error("调用HttpUtils.sendGet SocketTimeoutException, url=" + url + ",param=" + param, e); + } + catch (IOException e) + { + log.error("调用HttpUtils.sendGet IOException, url=" + url + ",param=" + param, e); + } + catch (Exception e) + { + log.error("调用HttpsUtil.sendGet Exception, url=" + url + ",param=" + param, e); + } + finally + { + try + { + if (in != null) + { + in.close(); + } + } + catch (Exception ex) + { + log.error("调用in.close Exception, url=" + url + ",param=" + param, ex); + } + } + return result.toString(); + } + + /** + * 向指定 URL 发送POST方法的请求 + * + * @param url 发送请求的 URL + * @param param 请求参数,请求参数应该是 name1=value1&name2=value2 的形式。 + * @return 所代表远程资源的响应结果 + */ + public static String sendPost(String url, String param) + { + PrintWriter out = null; + BufferedReader in = null; + StringBuilder result = new StringBuilder(); + try + { + log.info("sendPost - {}", url); + URL realUrl = new URL(url); + URLConnection conn = realUrl.openConnection(); + conn.setRequestProperty("accept", "*/*"); + conn.setRequestProperty("connection", "Keep-Alive"); + conn.setRequestProperty("user-agent", "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1;SV1)"); + conn.setRequestProperty("Accept-Charset", "utf-8"); + conn.setRequestProperty("contentType", "utf-8"); + conn.setDoOutput(true); + conn.setDoInput(true); + out = new PrintWriter(conn.getOutputStream()); + out.print(param); + out.flush(); + in = new BufferedReader(new InputStreamReader(conn.getInputStream(), StandardCharsets.UTF_8)); + String line; + while ((line = in.readLine()) != null) + { + result.append(line); + } + log.info("recv - {}", result); + } + catch (ConnectException e) + { + log.error("调用HttpUtils.sendPost ConnectException, url=" + url + ",param=" + param, e); + } + catch (SocketTimeoutException e) + { + log.error("调用HttpUtils.sendPost SocketTimeoutException, url=" + url + ",param=" + param, e); + } + catch (IOException e) + { + log.error("调用HttpUtils.sendPost IOException, url=" + url + ",param=" + param, e); + } + catch (Exception e) + { + log.error("调用HttpsUtil.sendPost Exception, url=" + url + ",param=" + param, e); + } + finally + { + try + { + if (out != null) + { + out.close(); + } + if (in != null) + { + in.close(); + } + } + catch (IOException ex) + { + log.error("调用in.close Exception, url=" + url + ",param=" + param, ex); + } + } + return result.toString(); + } + + public static String sendSSLPost(String url, String param) + { + StringBuilder result = new StringBuilder(); + String urlNameString = url + "?" + param; + try + { + log.info("sendSSLPost - {}", urlNameString); + SSLContext sc = SSLContext.getInstance("SSL"); + sc.init(null, new TrustManager[] { new TrustAnyTrustManager() }, new java.security.SecureRandom()); + URL console = new URL(urlNameString); + HttpsURLConnection conn = (HttpsURLConnection) console.openConnection(); + conn.setRequestProperty("accept", "*/*"); + conn.setRequestProperty("connection", "Keep-Alive"); + conn.setRequestProperty("user-agent", "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1;SV1)"); + conn.setRequestProperty("Accept-Charset", "utf-8"); + conn.setRequestProperty("contentType", "utf-8"); + conn.setDoOutput(true); + conn.setDoInput(true); + + conn.setSSLSocketFactory(sc.getSocketFactory()); + conn.setHostnameVerifier(new TrustAnyHostnameVerifier()); + conn.connect(); + InputStream is = conn.getInputStream(); + BufferedReader br = new BufferedReader(new InputStreamReader(is)); + String ret = ""; + while ((ret = br.readLine()) != null) + { + if (ret != null && !"".equals(ret.trim())) + { + result.append(new String(ret.getBytes(StandardCharsets.ISO_8859_1), StandardCharsets.UTF_8)); + } + } + log.info("recv - {}", result); + conn.disconnect(); + br.close(); + } + catch (ConnectException e) + { + log.error("调用HttpUtils.sendSSLPost ConnectException, url=" + url + ",param=" + param, e); + } + catch (SocketTimeoutException e) + { + log.error("调用HttpUtils.sendSSLPost SocketTimeoutException, url=" + url + ",param=" + param, e); + } + catch (IOException e) + { + log.error("调用HttpUtils.sendSSLPost IOException, url=" + url + ",param=" + param, e); + } + catch (Exception e) + { + log.error("调用HttpsUtil.sendSSLPost Exception, url=" + url + ",param=" + param, e); + } + return result.toString(); + } + + private static class TrustAnyTrustManager implements X509TrustManager + { + @Override + public void checkClientTrusted(X509Certificate[] chain, String authType) + { + } + + @Override + public void checkServerTrusted(X509Certificate[] chain, String authType) + { + } + + @Override + public X509Certificate[] getAcceptedIssuers() + { + return new X509Certificate[] {}; + } + } + + private static class TrustAnyHostnameVerifier implements HostnameVerifier + { + @Override + public boolean verify(String hostname, SSLSession session) + { + return true; + } + } +} \ No newline at end of file diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/ip/AddressUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/ip/AddressUtils.java new file mode 100644 index 00000000..edfe419a --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/ip/AddressUtils.java @@ -0,0 +1,56 @@ +package com.ruoyi.common.utils.ip; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import com.alibaba.fastjson2.JSON; +import com.alibaba.fastjson2.JSONObject; +import com.ruoyi.common.config.RuoYiConfig; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.http.HttpUtils; + +/** + * 获取地址类 + * + * @author ruoyi + */ +public class AddressUtils +{ + private static final Logger log = LoggerFactory.getLogger(AddressUtils.class); + + // IP地址查询 + public static final String IP_URL = "http://whois.pconline.com.cn/ipJson.jsp"; + + // 未知地址 + public static final String UNKNOWN = "XX XX"; + + public static String getRealAddressByIP(String ip) + { + // 内网不查询 + if (IpUtils.internalIp(ip)) + { + return "内网IP"; + } + if (RuoYiConfig.isAddressEnabled()) + { + try + { + String rspStr = HttpUtils.sendGet(IP_URL, "ip=" + ip + "&json=true", Constants.GBK); + if (StringUtils.isEmpty(rspStr)) + { + log.error("获取地理位置异常 {}", ip); + return UNKNOWN; + } + JSONObject obj = JSON.parseObject(rspStr); + String region = obj.getString("pro"); + String city = obj.getString("city"); + return String.format("%s %s", region, city); + } + catch (Exception e) + { + log.error("获取地理位置异常 {}", ip); + } + } + return UNKNOWN; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/ip/IpUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/ip/IpUtils.java new file mode 100644 index 00000000..c18c56a4 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/ip/IpUtils.java @@ -0,0 +1,264 @@ +package com.ruoyi.common.utils.ip; + +import java.net.InetAddress; +import java.net.UnknownHostException; +import javax.servlet.http.HttpServletRequest; +import com.ruoyi.common.utils.StringUtils; + +/** + * 获取IP方法 + * + * @author ruoyi + */ +public class IpUtils +{ + /** + * 获取客户端IP + * + * @param request 请求对象 + * @return IP地址 + */ + public static String getIpAddr(HttpServletRequest request) + { + if (request == null) + { + return "unknown"; + } + String ip = request.getHeader("x-forwarded-for"); + if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) + { + ip = request.getHeader("Proxy-Client-IP"); + } + if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) + { + ip = request.getHeader("X-Forwarded-For"); + } + if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) + { + ip = request.getHeader("WL-Proxy-Client-IP"); + } + if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) + { + ip = request.getHeader("X-Real-IP"); + } + + if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) + { + ip = request.getRemoteAddr(); + } + + return "0:0:0:0:0:0:0:1".equals(ip) ? "127.0.0.1" : getMultistageReverseProxyIp(ip); + } + + /** + * 检查是否为内部IP地址 + * + * @param ip IP地址 + * @return 结果 + */ + public static boolean internalIp(String ip) + { + byte[] addr = textToNumericFormatV4(ip); + return internalIp(addr) || "127.0.0.1".equals(ip); + } + + /** + * 检查是否为内部IP地址 + * + * @param addr byte地址 + * @return 结果 + */ + private static boolean internalIp(byte[] addr) + { + if (StringUtils.isNull(addr) || addr.length < 2) + { + return true; + } + final byte b0 = addr[0]; + final byte b1 = addr[1]; + // 10.x.x.x/8 + final byte SECTION_1 = 0x0A; + // 172.16.x.x/12 + final byte SECTION_2 = (byte) 0xAC; + final byte SECTION_3 = (byte) 0x10; + final byte SECTION_4 = (byte) 0x1F; + // 192.168.x.x/16 + final byte SECTION_5 = (byte) 0xC0; + final byte SECTION_6 = (byte) 0xA8; + switch (b0) + { + case SECTION_1: + return true; + case SECTION_2: + if (b1 >= SECTION_3 && b1 <= SECTION_4) + { + return true; + } + case SECTION_5: + switch (b1) + { + case SECTION_6: + return true; + } + default: + return false; + } + } + + /** + * 将IPv4地址转换成字节 + * + * @param text IPv4地址 + * @return byte 字节 + */ + public static byte[] textToNumericFormatV4(String text) + { + if (text.length() == 0) + { + return null; + } + + byte[] bytes = new byte[4]; + String[] elements = text.split("\\.", -1); + try + { + long l; + int i; + switch (elements.length) + { + case 1: + l = Long.parseLong(elements[0]); + if ((l < 0L) || (l > 4294967295L)) + { + return null; + } + bytes[0] = (byte) (int) (l >> 24 & 0xFF); + bytes[1] = (byte) (int) ((l & 0xFFFFFF) >> 16 & 0xFF); + bytes[2] = (byte) (int) ((l & 0xFFFF) >> 8 & 0xFF); + bytes[3] = (byte) (int) (l & 0xFF); + break; + case 2: + l = Integer.parseInt(elements[0]); + if ((l < 0L) || (l > 255L)) + { + return null; + } + bytes[0] = (byte) (int) (l & 0xFF); + l = Integer.parseInt(elements[1]); + if ((l < 0L) || (l > 16777215L)) + { + return null; + } + bytes[1] = (byte) (int) (l >> 16 & 0xFF); + bytes[2] = (byte) (int) ((l & 0xFFFF) >> 8 & 0xFF); + bytes[3] = (byte) (int) (l & 0xFF); + break; + case 3: + for (i = 0; i < 2; ++i) + { + l = Integer.parseInt(elements[i]); + if ((l < 0L) || (l > 255L)) + { + return null; + } + bytes[i] = (byte) (int) (l & 0xFF); + } + l = Integer.parseInt(elements[2]); + if ((l < 0L) || (l > 65535L)) + { + return null; + } + bytes[2] = (byte) (int) (l >> 8 & 0xFF); + bytes[3] = (byte) (int) (l & 0xFF); + break; + case 4: + for (i = 0; i < 4; ++i) + { + l = Integer.parseInt(elements[i]); + if ((l < 0L) || (l > 255L)) + { + return null; + } + bytes[i] = (byte) (int) (l & 0xFF); + } + break; + default: + return null; + } + } + catch (NumberFormatException e) + { + return null; + } + return bytes; + } + + /** + * 获取IP地址 + * + * @return 本地IP地址 + */ + public static String getHostIp() + { + try + { + return InetAddress.getLocalHost().getHostAddress(); + } + catch (UnknownHostException e) + { + } + return "127.0.0.1"; + } + + /** + * 获取主机名 + * + * @return 本地主机名 + */ + public static String getHostName() + { + try + { + return InetAddress.getLocalHost().getHostName(); + } + catch (UnknownHostException e) + { + } + return "未知"; + } + + /** + * 从多级反向代理中获得第一个非unknown IP地址 + * + * @param ip 获得的IP地址 + * @return 第一个非unknown IP地址 + */ + public static String getMultistageReverseProxyIp(String ip) + { + // 多级反向代理检测 + if (ip != null && ip.indexOf(",") > 0) + { + final String[] ips = ip.trim().split(","); + for (String subIp : ips) + { + if (false == isUnknown(subIp)) + { + ip = subIp; + break; + } + } + } + return ip; + } + + /** + * 检测给定字符串是否为未知,多用于检测HTTP请求相关 + * + * @param checkString 被检测的字符串 + * @return 是否未知 + */ + public static boolean isUnknown(String checkString) + { + return StringUtils.isBlank(checkString) || "unknown".equalsIgnoreCase(checkString); + } +} \ No newline at end of file diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/poi/ExcelHandlerAdapter.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/poi/ExcelHandlerAdapter.java new file mode 100644 index 00000000..5ea74c11 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/poi/ExcelHandlerAdapter.java @@ -0,0 +1,19 @@ +package com.ruoyi.common.utils.poi; + +/** + * Excel数据格式处理适配器 + * + * @author ruoyi + */ +public interface ExcelHandlerAdapter +{ + /** + * 格式化 + * + * @param value 单元格数据值 + * @param args excel注解args参数组 + * + * @return 处理后的值 + */ + Object format(Object value, String[] args); +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/poi/ExcelUtil.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/poi/ExcelUtil.java new file mode 100644 index 00000000..8e8c3fab --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/poi/ExcelUtil.java @@ -0,0 +1,1358 @@ +package com.ruoyi.common.utils.poi; + +import com.ruoyi.common.annotation.Excel; +import com.ruoyi.common.annotation.Excel.ColumnType; +import com.ruoyi.common.annotation.Excel.Type; +import com.ruoyi.common.annotation.Excels; +import com.ruoyi.common.config.RuoYiConfig; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.text.Convert; +import com.ruoyi.common.exception.UtilException; +import com.ruoyi.common.utils.DateUtils; +import com.ruoyi.common.utils.DictUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.file.FileTypeUtils; +import com.ruoyi.common.utils.file.FileUtils; +import com.ruoyi.common.utils.file.ImageUtils; +import com.ruoyi.common.utils.reflect.ReflectUtils; +import org.apache.commons.lang3.ArrayUtils; +import org.apache.commons.lang3.RegExUtils; +import org.apache.commons.lang3.reflect.FieldUtils; +import org.apache.poi.hssf.usermodel.*; +import org.apache.poi.ooxml.POIXMLDocumentPart; +import org.apache.poi.ss.usermodel.*; +import org.apache.poi.ss.util.CellRangeAddress; +import org.apache.poi.ss.util.CellRangeAddressList; +import org.apache.poi.util.IOUtils; +import org.apache.poi.xssf.streaming.SXSSFWorkbook; +import org.apache.poi.xssf.usermodel.*; +import org.openxmlformats.schemas.drawingml.x2006.spreadsheetDrawing.CTMarker; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import javax.servlet.http.HttpServletResponse; +import java.io.*; +import java.lang.reflect.Field; +import java.lang.reflect.Method; +import java.lang.reflect.ParameterizedType; +import java.math.BigDecimal; +import java.text.DecimalFormat; +import java.time.LocalDate; +import java.time.LocalDateTime; +import java.util.*; +import java.util.stream.Collectors; + +/** + * Excel相关处理 + * + * @author ruoyi + */ +public class ExcelUtil { + private static final Logger log = LoggerFactory.getLogger(ExcelUtil.class); + + public static final String FORMULA_REGEX_STR = "=|-|\\+|@"; + + public static final String[] FORMULA_STR = {"=", "-", "+", "@"}; + + /** + * Excel sheet最大行数,默认65536 + */ + public static final int sheetSize = 65536; + + /** + * 工作表名称 + */ + private String sheetName; + + /** + * 导出类型(EXPORT:导出数据;IMPORT:导入模板) + */ + private Type type; + + /** + * 工作薄对象 + */ + private Workbook wb; + + /** + * 工作表对象 + */ + private Sheet sheet; + + /** + * 样式列表 + */ + private Map styles; + + /** + * 导入导出数据列表 + */ + private List list; + + /** + * 注解列表 + */ + private List fields; + + /** + * 当前行号 + */ + private int rownum; + + /** + * 标题 + */ + private String title; + + /** + * 最大高度 + */ + private short maxHeight; + + /** + * 合并后最后行数 + */ + private int subMergedLastRowNum = 0; + + /** + * 合并后开始行数 + */ + private int subMergedFirstRowNum = 1; + + /** + * 对象的子列表方法 + */ + private Method subMethod; + + /** + * 对象的子列表属性 + */ + private List subFields; + + /** + * 统计列表 + */ + private Map statistics = new HashMap(); + + /** + * 数字格式 + */ + private static final DecimalFormat DOUBLE_FORMAT = new DecimalFormat("######0.00"); + + /** + * 实体对象 + */ + public Class clazz; + + /** + * 需要排除列属性 + */ + public String[] excludeFields; + + public ExcelUtil(Class clazz) { + this.clazz = clazz; + } + + /** + * 隐藏Excel中列属性 + * + * @param fields 列属性名 示例[单个"name"/多个"id","name"] + * @throws Exception + */ + public void hideColumn(String... fields) { + this.excludeFields = fields; + } + + public void init(List list, String sheetName, String title, Type type) { + if (list == null) { + list = new ArrayList(); + } + this.list = list; + this.sheetName = sheetName; + this.type = type; + this.title = title; + createExcelField(); + createWorkbook(); + createTitle(); + createSubHead(); + } + + /** + * 创建excel第一行标题 + */ + public void createTitle() { + if (StringUtils.isNotEmpty(title)) { + subMergedFirstRowNum++; + subMergedLastRowNum++; + int titleLastCol = this.fields.size() - 1; + if (isSubList()) { + titleLastCol = titleLastCol + subFields.size() - 1; + } + Row titleRow = sheet.createRow(rownum == 0 ? rownum++ : 0); + titleRow.setHeightInPoints(30); + Cell titleCell = titleRow.createCell(0); + titleCell.setCellStyle(styles.get("title")); + titleCell.setCellValue(title); + sheet.addMergedRegion(new CellRangeAddress(titleRow.getRowNum(), titleRow.getRowNum(), titleRow.getRowNum(), titleLastCol)); + } + } + + /** + * 创建对象的子列表名称 + */ + public void createSubHead() { + if (isSubList()) { + subMergedFirstRowNum++; + subMergedLastRowNum++; + Row subRow = sheet.createRow(rownum); + int excelNum = 0; + for (Object[] objects : fields) { + Excel attr = (Excel) objects[1]; + Cell headCell1 = subRow.createCell(excelNum); + headCell1.setCellValue(attr.name()); + headCell1.setCellStyle(styles.get(StringUtils.format("header_{}_{}", attr.headerColor(), attr.headerBackgroundColor()))); + excelNum++; + } + int headFirstRow = excelNum - 1; + int headLastRow = headFirstRow + subFields.size() - 1; + if (headLastRow > headFirstRow) { + sheet.addMergedRegion(new CellRangeAddress(rownum, rownum, headFirstRow, headLastRow)); + } + rownum++; + } + } + + /** + * 对excel表单默认第一个索引名转换成list + * + * @param is 输入流 + * @return 转换后集合 + */ + public List importExcel(InputStream is) throws Exception { + return importExcel(is, 0); + } + + /** + * 对excel表单默认第一个索引名转换成list + * + * @param is 输入流 + * @param titleNum 标题占用行数 + * @return 转换后集合 + */ + public List importExcel(InputStream is, int titleNum) throws Exception { + return importExcel(StringUtils.EMPTY, is, titleNum); + } + + /** + * 对excel表单指定表格索引名转换成list + * + * @param sheetName 表格索引名 + * @param titleNum 标题占用行数 + * @param is 输入流 + * @return 转换后集合 + */ + public List importExcel(String sheetName, InputStream is, int titleNum) throws Exception { + this.type = Type.IMPORT; + this.wb = WorkbookFactory.create(is); + List list = new ArrayList(); + // 如果指定sheet名,则取指定sheet中的内容 否则默认指向第1个sheet + Sheet sheet = StringUtils.isNotEmpty(sheetName) ? wb.getSheet(sheetName) : wb.getSheetAt(0); + if (sheet == null) { + throw new IOException("文件sheet不存在"); + } + boolean isXSSFWorkbook = !(wb instanceof HSSFWorkbook); + Map pictures; + if (isXSSFWorkbook) { + pictures = getSheetPictures07((XSSFSheet) sheet, (XSSFWorkbook) wb); + } else { + pictures = getSheetPictures03((HSSFSheet) sheet, (HSSFWorkbook) wb); + } + // 获取最后一个非空行的行下标,比如总行数为n,则返回的为n-1 + int rows = sheet.getLastRowNum(); + + if (rows > 0) { + // 定义一个map用于存放excel列的序号和field. + Map cellMap = new HashMap(); + // 获取表头 + Row heard = sheet.getRow(titleNum); + for (int i = 0; i < heard.getPhysicalNumberOfCells(); i++) { + Cell cell = heard.getCell(i); + if (StringUtils.isNotNull(cell)) { + String value = this.getCellValue(heard, i).toString(); + cellMap.put(value, i); + } else { + cellMap.put(null, i); + } + } + // 有数据时才处理 得到类的所有field. + List fields = this.getFields(); + Map fieldsMap = new HashMap(); + for (Object[] objects : fields) { + Excel attr = (Excel) objects[1]; + Integer column = cellMap.get(attr.name()); + if (column != null) { + fieldsMap.put(column, objects); + } + } + for (int i = titleNum + 1; i <= rows; i++) { + // 从第2行开始取数据,默认第一行是表头. + Row row = sheet.getRow(i); + // 判断当前行是否是空行 + if (isRowEmpty(row)) { + continue; + } + T entity = null; + for (Map.Entry entry : fieldsMap.entrySet()) { + Object val = this.getCellValue(row, entry.getKey()); + + // 如果不存在实例则新建. + entity = (entity == null ? clazz.newInstance() : entity); + // 从map中得到对应列的field. + Field field = (Field) entry.getValue()[0]; + Excel attr = (Excel) entry.getValue()[1]; + // 取得类型,并根据对象类型设置值. + Class fieldType = field.getType(); + if (String.class == fieldType) { + String s = Convert.toStr(val); + if (StringUtils.endsWith(s, ".0")) { + val = StringUtils.substringBefore(s, ".0"); + } else { + String dateFormat = field.getAnnotation(Excel.class).dateFormat(); + if (StringUtils.isNotEmpty(dateFormat)) { + val = parseDateToStr(dateFormat, val); + } else { + val = Convert.toStr(val); + } + } + } else if ((Integer.TYPE == fieldType || Integer.class == fieldType) && StringUtils.isNumeric(Convert.toStr(val))) { + val = Convert.toInt(val); + } else if ((Long.TYPE == fieldType || Long.class == fieldType) && StringUtils.isNumeric(Convert.toStr(val))) { + val = Convert.toLong(val); + } else if (Double.TYPE == fieldType || Double.class == fieldType) { + val = Convert.toDouble(val); + } else if (Float.TYPE == fieldType || Float.class == fieldType) { + val = Convert.toFloat(val); + } else if (BigDecimal.class == fieldType) { + val = Convert.toBigDecimal(val); + } else if (Date.class == fieldType) { + if (val instanceof String) { + val = DateUtils.parseDate(val); + } else if (val instanceof Double) { + val = DateUtil.getJavaDate((Double) val); + } + } else if (Boolean.TYPE == fieldType || Boolean.class == fieldType) { + val = Convert.toBool(val, false); + } + if (StringUtils.isNotNull(fieldType)) { + String propertyName = field.getName(); + if (StringUtils.isNotEmpty(attr.targetAttr())) { + propertyName = field.getName() + "." + attr.targetAttr(); + } else if (StringUtils.isNotEmpty(attr.readConverterExp())) { + val = reverseByExp(Convert.toStr(val), attr.readConverterExp(), attr.separator()); + } else if (StringUtils.isNotEmpty(attr.dictType())) { + val = reverseDictByExp(Convert.toStr(val), attr.dictType(), attr.separator()); + } else if (!attr.handler().equals(ExcelHandlerAdapter.class)) { + val = dataFormatHandlerAdapter(val, attr); + } else if (ColumnType.IMAGE == attr.cellType() && StringUtils.isNotEmpty(pictures)) { + PictureData image = pictures.get(row.getRowNum() + "_" + entry.getKey()); + if (image == null) { + val = ""; + } else { + byte[] data = image.getData(); + val = FileUtils.writeImportBytes(data); + } + } + ReflectUtils.invokeSetter(entity, propertyName, val); + } + } + list.add(entity); + } + } + return list; + } + + /** + * 对list数据源将其里面的数据导入到excel表单 + * + * @param list 导出数据集合 + * @param sheetName 工作表的名称 + * @return 结果 + */ + public AjaxResult exportExcel(List list, String sheetName) { + return exportExcel(list, sheetName, StringUtils.EMPTY); + } + + /** + * 对list数据源将其里面的数据导入到excel表单 + * + * @param list 导出数据集合 + * @param sheetName 工作表的名称 + * @param title 标题 + * @return 结果 + */ + public AjaxResult exportExcel(List list, String sheetName, String title) { + this.init(list, sheetName, title, Type.EXPORT); + return exportExcel(); + } + + /** + * 对list数据源将其里面的数据导入到excel表单 + * + * @param response 返回数据 + * @param list 导出数据集合 + * @param sheetName 工作表的名称 + * @return 结果 + */ + public void exportExcel(HttpServletResponse response, List list, String sheetName) { + exportExcel(response, list, sheetName, StringUtils.EMPTY); + } + + /** + * 对list数据源将其里面的数据导入到excel表单 + * + * @param response 返回数据 + * @param list 导出数据集合 + * @param sheetName 工作表的名称 + * @param title 标题 + * @return 结果 + */ + public void exportExcel(HttpServletResponse response, List list, String sheetName, String title) { + response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"); + response.setCharacterEncoding("utf-8"); + this.init(list, sheetName, title, Type.EXPORT); + exportExcel(response); + } + + /** + * 对list数据源将其里面的数据导入到excel表单 + * + * @param sheetName 工作表的名称 + * @return 结果 + */ + public AjaxResult importTemplateExcel(String sheetName) { + return importTemplateExcel(sheetName, StringUtils.EMPTY); + } + + /** + * 对list数据源将其里面的数据导入到excel表单 + * + * @param sheetName 工作表的名称 + * @param title 标题 + * @return 结果 + */ + public AjaxResult importTemplateExcel(String sheetName, String title) { + this.init(null, sheetName, title, Type.IMPORT); + return exportExcel(); + } + + /** + * 对list数据源将其里面的数据导入到excel表单 + * + * @param sheetName 工作表的名称 + * @return 结果 + */ + public void importTemplateExcel(HttpServletResponse response, String sheetName) { + importTemplateExcel(response, sheetName, StringUtils.EMPTY); + } + + /** + * 对list数据源将其里面的数据导入到excel表单 + * + * @param sheetName 工作表的名称 + * @param title 标题 + * @return 结果 + */ + public void importTemplateExcel(HttpServletResponse response, String sheetName, String title) { + response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"); + response.setCharacterEncoding("utf-8"); + this.init(null, sheetName, title, Type.IMPORT); + exportExcel(response); + } + + /** + * 对list数据源将其里面的数据导入到excel表单 + * + * @return 结果 + */ + public void exportExcel(HttpServletResponse response) { + try { + writeSheet(); + wb.write(response.getOutputStream()); + } catch (Exception e) { + log.error("导出Excel异常{}", e.getMessage()); + } finally { + IOUtils.closeQuietly(wb); + } + } + + /** + * 对list数据源将其里面的数据导入到excel表单 + * + * @return 结果 + */ + public AjaxResult exportExcel() { + OutputStream out = null; + try { + writeSheet(); + String filename = encodingFilename(sheetName); + out = new FileOutputStream(getAbsoluteFile(filename)); + wb.write(out); + return AjaxResult.success(filename); + } catch (Exception e) { + log.error("导出Excel异常{}", e.getMessage()); + throw new UtilException("导出Excel失败,请联系网站管理员!"); + } finally { + IOUtils.closeQuietly(wb); + IOUtils.closeQuietly(out); + } + } + + /** + * 创建写入数据到Sheet + */ + public void writeSheet() { + // 取出一共有多少个sheet. + int sheetNo = Math.max(1, (int) Math.ceil(list.size() * 1.0 / sheetSize)); + for (int index = 0; index < sheetNo; index++) { + createSheet(sheetNo, index); + + // 产生一行 + Row row = sheet.createRow(rownum); + int column = 0; + // 写入各个字段的列头名称 + for (Object[] os : fields) { + Field field = (Field) os[0]; + Excel excel = (Excel) os[1]; + if (Collection.class.isAssignableFrom(field.getType())) { + for (Field subField : subFields) { + Excel subExcel = subField.getAnnotation(Excel.class); + this.createHeadCell(subExcel, row, column++); + } + } else { + this.createHeadCell(excel, row, column++); + } + } + if (Type.EXPORT.equals(type)) { + fillExcelData(index, row); + addStatisticsRow(); + } + } + } + + /** + * 填充excel数据 + * + * @param index 序号 + * @param row 单元格行 + */ + @SuppressWarnings("unchecked") + public void fillExcelData(int index, Row row) { + int startNo = index * sheetSize; + int endNo = Math.min(startNo + sheetSize, list.size()); + int rowNo = (1 + rownum) - startNo; + for (int i = startNo; i < endNo; i++) { + rowNo = i > 1 ? rowNo + 1 : rowNo + i; + row = sheet.createRow(rowNo); + // 得到导出对象. + T vo = (T) list.get(i); + Collection subList = null; + if (isSubList()) { + if (isSubListValue(vo)) { + subList = getListCellValue(vo); + subMergedLastRowNum = subMergedLastRowNum + subList.size(); + } else { + subMergedFirstRowNum++; + subMergedLastRowNum++; + } + } + + int column = 0; + for (Object[] os : fields) { + Field field = (Field) os[0]; + Excel excel = (Excel) os[1]; + if (Collection.class.isAssignableFrom(field.getType()) && StringUtils.isNotNull(subList)) { + boolean subFirst = false; + for (Object obj : subList) { + if (subFirst) { + rowNo++; + row = sheet.createRow(rowNo); + } + List subFields = FieldUtils.getFieldsListWithAnnotation(obj.getClass(), Excel.class); + int subIndex = 0; + for (Field subField : subFields) { + if (subField.isAnnotationPresent(Excel.class)) { + subField.setAccessible(true); + Excel attr = subField.getAnnotation(Excel.class); + this.addCell(attr, row, (T) obj, subField, column + subIndex); + } + subIndex++; + } + subFirst = true; + } + this.subMergedFirstRowNum = this.subMergedFirstRowNum + subList.size(); + } else { + this.addCell(excel, row, vo, field, column++); + } + } + } + } + + /** + * 创建表格样式 + * + * @param wb 工作薄对象 + * @return 样式列表 + */ + private Map createStyles(Workbook wb) { + // 写入各条记录,每条记录对应excel表中的一行 + Map styles = new HashMap(); + CellStyle style = wb.createCellStyle(); + style.setAlignment(HorizontalAlignment.CENTER); + style.setVerticalAlignment(VerticalAlignment.CENTER); + Font titleFont = wb.createFont(); + titleFont.setFontName("Arial"); + titleFont.setFontHeightInPoints((short) 16); + titleFont.setBold(true); + style.setFont(titleFont); + styles.put("title", style); + + style = wb.createCellStyle(); + style.setAlignment(HorizontalAlignment.CENTER); + style.setVerticalAlignment(VerticalAlignment.CENTER); + style.setBorderRight(BorderStyle.THIN); + style.setRightBorderColor(IndexedColors.GREY_50_PERCENT.getIndex()); + style.setBorderLeft(BorderStyle.THIN); + style.setLeftBorderColor(IndexedColors.GREY_50_PERCENT.getIndex()); + style.setBorderTop(BorderStyle.THIN); + style.setTopBorderColor(IndexedColors.GREY_50_PERCENT.getIndex()); + style.setBorderBottom(BorderStyle.THIN); + style.setBottomBorderColor(IndexedColors.GREY_50_PERCENT.getIndex()); + Font dataFont = wb.createFont(); + dataFont.setFontName("Arial"); + dataFont.setFontHeightInPoints((short) 10); + style.setFont(dataFont); + styles.put("data", style); + + style = wb.createCellStyle(); + style.setAlignment(HorizontalAlignment.CENTER); + style.setVerticalAlignment(VerticalAlignment.CENTER); + Font totalFont = wb.createFont(); + totalFont.setFontName("Arial"); + totalFont.setFontHeightInPoints((short) 10); + style.setFont(totalFont); + styles.put("total", style); + + styles.putAll(annotationHeaderStyles(wb, styles)); + + styles.putAll(annotationDataStyles(wb)); + + return styles; + } + + /** + * 根据Excel注解创建表格头样式 + * + * @param wb 工作薄对象 + * @return 自定义样式列表 + */ + private Map annotationHeaderStyles(Workbook wb, Map styles) { + Map headerStyles = new HashMap(); + for (Object[] os : fields) { + Excel excel = (Excel) os[1]; + String key = StringUtils.format("header_{}_{}", excel.headerColor(), excel.headerBackgroundColor()); + if (!headerStyles.containsKey(key)) { + CellStyle style = wb.createCellStyle(); + style.cloneStyleFrom(styles.get("data")); + style.setAlignment(HorizontalAlignment.CENTER); + style.setVerticalAlignment(VerticalAlignment.CENTER); + style.setFillForegroundColor(excel.headerBackgroundColor().index); + style.setFillPattern(FillPatternType.SOLID_FOREGROUND); + Font headerFont = wb.createFont(); + headerFont.setFontName("Arial"); + headerFont.setFontHeightInPoints((short) 10); + headerFont.setBold(true); + headerFont.setColor(excel.headerColor().index); + style.setFont(headerFont); + headerStyles.put(key, style); + } + } + return headerStyles; + } + + /** + * 根据Excel注解创建表格列样式 + * + * @param wb 工作薄对象 + * @return 自定义样式列表 + */ + private Map annotationDataStyles(Workbook wb) { + Map styles = new HashMap(); + for (Object[] os : fields) { + Excel excel = (Excel) os[1]; + String key = StringUtils.format("data_{}_{}_{}", excel.align(), excel.color(), excel.backgroundColor()); + if (!styles.containsKey(key)) { + CellStyle style = wb.createCellStyle(); + style.setAlignment(excel.align()); + style.setVerticalAlignment(VerticalAlignment.CENTER); + style.setBorderRight(BorderStyle.THIN); + style.setRightBorderColor(IndexedColors.GREY_50_PERCENT.getIndex()); + style.setBorderLeft(BorderStyle.THIN); + style.setLeftBorderColor(IndexedColors.GREY_50_PERCENT.getIndex()); + style.setBorderTop(BorderStyle.THIN); + style.setTopBorderColor(IndexedColors.GREY_50_PERCENT.getIndex()); + style.setBorderBottom(BorderStyle.THIN); + style.setBottomBorderColor(IndexedColors.GREY_50_PERCENT.getIndex()); + style.setFillPattern(FillPatternType.SOLID_FOREGROUND); + style.setFillForegroundColor(excel.backgroundColor().getIndex()); + Font dataFont = wb.createFont(); + dataFont.setFontName("Arial"); + dataFont.setFontHeightInPoints((short) 10); + dataFont.setColor(excel.color().index); + style.setFont(dataFont); + styles.put(key, style); + } + } + return styles; + } + + /** + * 创建单元格 + */ + public Cell createHeadCell(Excel attr, Row row, int column) { + // 创建列 + Cell cell = row.createCell(column); + // 写入列信息 + cell.setCellValue(attr.name()); + setDataValidation(attr, row, column); + cell.setCellStyle(styles.get(StringUtils.format("header_{}_{}", attr.headerColor(), attr.headerBackgroundColor()))); + if (isSubList()) { + // 填充默认样式,防止合并单元格样式失效 + sheet.setDefaultColumnStyle(column, styles.get(StringUtils.format("data_{}_{}_{}", attr.align(), attr.color(), attr.backgroundColor()))); + if (attr.needMerge()) { + sheet.addMergedRegion(new CellRangeAddress(rownum - 1, rownum, column, column)); + } + } + return cell; + } + + /** + * 设置单元格信息 + * + * @param value 单元格值 + * @param attr 注解相关 + * @param cell 单元格信息 + */ + public void setCellVo(Object value, Excel attr, Cell cell) { + if (ColumnType.STRING == attr.cellType()) { + String cellValue = Convert.toStr(value); + // 对于任何以表达式触发字符 =-+@开头的单元格,直接使用tab字符作为前缀,防止CSV注入。 + if (StringUtils.startsWithAny(cellValue, FORMULA_STR)) { + cellValue = RegExUtils.replaceFirst(cellValue, FORMULA_REGEX_STR, "\t$0"); + } + cell.setCellValue(StringUtils.isNull(cellValue) ? attr.defaultValue() : cellValue + attr.suffix()); + } else if (ColumnType.NUMERIC == attr.cellType()) { + if (StringUtils.isNotNull(value)) { + cell.setCellValue(StringUtils.contains(Convert.toStr(value), ".") ? Convert.toDouble(value) : Convert.toInt(value)); + } + } else if (ColumnType.IMAGE == attr.cellType()) { + ClientAnchor anchor = new XSSFClientAnchor(0, 0, 0, 0, (short) cell.getColumnIndex(), cell.getRow().getRowNum(), (short) (cell.getColumnIndex() + 1), cell.getRow().getRowNum() + 1); + String imagePath = Convert.toStr(value); + if (StringUtils.isNotEmpty(imagePath)) { + byte[] data = ImageUtils.getImage(imagePath); + getDrawingPatriarch(cell.getSheet()).createPicture(anchor, + cell.getSheet().getWorkbook().addPicture(data, getImageType(data))); + } + } + } + + /** + * 获取画布 + */ + public static Drawing getDrawingPatriarch(Sheet sheet) { + if (sheet.getDrawingPatriarch() == null) { + sheet.createDrawingPatriarch(); + } + return sheet.getDrawingPatriarch(); + } + + /** + * 获取图片类型,设置图片插入类型 + */ + public int getImageType(byte[] value) { + String type = FileTypeUtils.getFileExtendName(value); + if ("JPG".equalsIgnoreCase(type)) { + return Workbook.PICTURE_TYPE_JPEG; + } else if ("PNG".equalsIgnoreCase(type)) { + return Workbook.PICTURE_TYPE_PNG; + } + return Workbook.PICTURE_TYPE_JPEG; + } + + /** + * 创建表格样式 + */ + public void setDataValidation(Excel attr, Row row, int column) { + if (attr.name().indexOf("注:") >= 0) { + sheet.setColumnWidth(column, 6000); + } else { + // 设置列宽 + sheet.setColumnWidth(column, (int) ((attr.width() + 0.72) * 256)); + } + if (StringUtils.isNotEmpty(attr.prompt()) || attr.combo().length > 0) { + // 提示信息或只能选择不能输入的列内容. + setPromptOrValidation(sheet, attr.combo(), attr.prompt(), 1, 100, column, column); + } + } + + /** + * 添加单元格 + */ + public Cell addCell(Excel attr, Row row, T vo, Field field, int column) { + Cell cell = null; + try { + // 设置行高 + row.setHeight(maxHeight); + // 根据Excel中设置情况决定是否导出,有些情况需要保持为空,希望用户填写这一列. + if (attr.isExport()) { + // 创建cell + cell = row.createCell(column); + if (isSubListValue(vo) && getListCellValue(vo).size() > 1 && attr.needMerge()) { + CellRangeAddress cellAddress = new CellRangeAddress(subMergedFirstRowNum, subMergedLastRowNum, column, column); + sheet.addMergedRegion(cellAddress); + } + cell.setCellStyle(styles.get(StringUtils.format("data_{}_{}_{}", attr.align(), attr.color(), attr.backgroundColor()))); + + // 用于读取对象中的属性 + Object value = getTargetValue(vo, field, attr); + String dateFormat = attr.dateFormat(); + String readConverterExp = attr.readConverterExp(); + String separator = attr.separator(); + String dictType = attr.dictType(); + if (StringUtils.isNotEmpty(dateFormat) && StringUtils.isNotNull(value)) { + cell.setCellValue(parseDateToStr(dateFormat, value)); + } else if (StringUtils.isNotEmpty(readConverterExp) && StringUtils.isNotNull(value)) { + cell.setCellValue(convertByExp(Convert.toStr(value), readConverterExp, separator)); + } else if (StringUtils.isNotEmpty(dictType) && StringUtils.isNotNull(value)) { + cell.setCellValue(convertDictByExp(Convert.toStr(value), dictType, separator)); + } else if (value instanceof BigDecimal && -1 != attr.scale()) { + cell.setCellValue((((BigDecimal) value).setScale(attr.scale(), attr.roundingMode())).doubleValue()); + } else if (!attr.handler().equals(ExcelHandlerAdapter.class)) { + cell.setCellValue(dataFormatHandlerAdapter(value, attr)); + } else { + // 设置列类型 + setCellVo(value, attr, cell); + } + addStatisticsData(column, Convert.toStr(value), attr); + } + } catch (Exception e) { + log.error("导出Excel失败{}", e); + } + return cell; + } + + /** + * 设置 POI XSSFSheet 单元格提示或选择框 + * + * @param sheet 表单 + * @param textlist 下拉框显示的内容 + * @param promptContent 提示内容 + * @param firstRow 开始行 + * @param endRow 结束行 + * @param firstCol 开始列 + * @param endCol 结束列 + */ + public void setPromptOrValidation(Sheet sheet, String[] textlist, String promptContent, int firstRow, int endRow, + int firstCol, int endCol) { + DataValidationHelper helper = sheet.getDataValidationHelper(); + DataValidationConstraint constraint = textlist.length > 0 ? helper.createExplicitListConstraint(textlist) : helper.createCustomConstraint("DD1"); + CellRangeAddressList regions = new CellRangeAddressList(firstRow, endRow, firstCol, endCol); + DataValidation dataValidation = helper.createValidation(constraint, regions); + if (StringUtils.isNotEmpty(promptContent)) { + // 如果设置了提示信息则鼠标放上去提示 + dataValidation.createPromptBox("", promptContent); + dataValidation.setShowPromptBox(true); + } + // 处理Excel兼容性问题 + if (dataValidation instanceof XSSFDataValidation) { + dataValidation.setSuppressDropDownArrow(true); + dataValidation.setShowErrorBox(true); + } else { + dataValidation.setSuppressDropDownArrow(false); + } + sheet.addValidationData(dataValidation); + } + + /** + * 解析导出值 0=男,1=女,2=未知 + * + * @param propertyValue 参数值 + * @param converterExp 翻译注解 + * @param separator 分隔符 + * @return 解析后值 + */ + public static String convertByExp(String propertyValue, String converterExp, String separator) { + StringBuilder propertyString = new StringBuilder(); + String[] convertSource = converterExp.split(","); + for (String item : convertSource) { + String[] itemArray = item.split("="); + if (StringUtils.containsAny(propertyValue, separator)) { + for (String value : propertyValue.split(separator)) { + if (itemArray[0].equals(value)) { + propertyString.append(itemArray[1] + separator); + break; + } + } + } else { + if (itemArray[0].equals(propertyValue)) { + return itemArray[1]; + } + } + } + return StringUtils.stripEnd(propertyString.toString(), separator); + } + + /** + * 反向解析值 男=0,女=1,未知=2 + * + * @param propertyValue 参数值 + * @param converterExp 翻译注解 + * @param separator 分隔符 + * @return 解析后值 + */ + public static String reverseByExp(String propertyValue, String converterExp, String separator) { + StringBuilder propertyString = new StringBuilder(); + String[] convertSource = converterExp.split(","); + for (String item : convertSource) { + String[] itemArray = item.split("="); + if (StringUtils.containsAny(propertyValue, separator)) { + for (String value : propertyValue.split(separator)) { + if (itemArray[1].equals(value)) { + propertyString.append(itemArray[0] + separator); + break; + } + } + } else { + if (itemArray[1].equals(propertyValue)) { + return itemArray[0]; + } + } + } + return StringUtils.stripEnd(propertyString.toString(), separator); + } + + /** + * 解析字典值 + * + * @param dictValue 字典值 + * @param dictType 字典类型 + * @param separator 分隔符 + * @return 字典标签 + */ + public static String convertDictByExp(String dictValue, String dictType, String separator) { + return DictUtils.getDictLabel(dictType, dictValue, separator); + } + + /** + * 反向解析值字典值 + * + * @param dictLabel 字典标签 + * @param dictType 字典类型 + * @param separator 分隔符 + * @return 字典值 + */ + public static String reverseDictByExp(String dictLabel, String dictType, String separator) { + return DictUtils.getDictValue(dictType, dictLabel, separator); + } + + /** + * 数据处理器 + * + * @param value 数据值 + * @param excel 数据注解 + * @return + */ + public String dataFormatHandlerAdapter(Object value, Excel excel) { + try { + Object instance = excel.handler().newInstance(); + Method formatMethod = excel.handler().getMethod("format", new Class[]{Object.class, String[].class}); + value = formatMethod.invoke(instance, value, excel.args()); + } catch (Exception e) { + log.error("不能格式化数据 " + excel.handler(), e.getMessage()); + } + return Convert.toStr(value); + } + + /** + * 合计统计信息 + */ + private void addStatisticsData(Integer index, String text, Excel entity) { + if (entity != null && entity.isStatistics()) { + Double temp = 0D; + if (!statistics.containsKey(index)) { + statistics.put(index, temp); + } + try { + temp = Double.valueOf(text); + } catch (NumberFormatException e) { + } + statistics.put(index, statistics.get(index) + temp); + } + } + + /** + * 创建统计行 + */ + public void addStatisticsRow() { + if (statistics.size() > 0) { + Row row = sheet.createRow(sheet.getLastRowNum() + 1); + Set keys = statistics.keySet(); + Cell cell = row.createCell(0); + cell.setCellStyle(styles.get("total")); + cell.setCellValue("合计"); + + for (Integer key : keys) { + cell = row.createCell(key); + cell.setCellStyle(styles.get("total")); + cell.setCellValue(DOUBLE_FORMAT.format(statistics.get(key))); + } + statistics.clear(); + } + } + + /** + * 编码文件名 + */ + public String encodingFilename(String filename) { + filename = UUID.randomUUID().toString() + "_" + filename + ".xlsx"; + return filename; + } + + /** + * 获取下载路径 + * + * @param filename 文件名称 + */ + public String getAbsoluteFile(String filename) { + String downloadPath = RuoYiConfig.getDownloadPath() + filename; + File desc = new File(downloadPath); + if (!desc.getParentFile().exists()) { + desc.getParentFile().mkdirs(); + } + return downloadPath; + } + + /** + * 获取bean中的属性值 + * + * @param vo 实体对象 + * @param field 字段 + * @param excel 注解 + * @return 最终的属性值 + * @throws Exception + */ + private Object getTargetValue(T vo, Field field, Excel excel) throws Exception { + Object o = field.get(vo); + if (StringUtils.isNotEmpty(excel.targetAttr())) { + String target = excel.targetAttr(); + if (target.contains(".")) { + String[] targets = target.split("[.]"); + for (String name : targets) { + o = getValue(o, name); + } + } else { + o = getValue(o, target); + } + } + return o; + } + + /** + * 以类的属性的get方法方法形式获取值 + * + * @param o + * @param name + * @return value + * @throws Exception + */ + private Object getValue(Object o, String name) throws Exception { + if (StringUtils.isNotNull(o) && StringUtils.isNotEmpty(name)) { + Class clazz = o.getClass(); + Field field = clazz.getDeclaredField(name); + field.setAccessible(true); + o = field.get(o); + } + return o; + } + + /** + * 得到所有定义字段 + */ + private void createExcelField() { + this.fields = getFields(); + this.fields = this.fields.stream().sorted(Comparator.comparing(objects -> ((Excel) objects[1]).sort())).collect(Collectors.toList()); + this.maxHeight = getRowHeight(); + } + + /** + * 获取字段注解信息 + */ + public List getFields() { + List fields = new ArrayList(); + List tempFields = new ArrayList<>(); + tempFields.addAll(Arrays.asList(clazz.getSuperclass().getDeclaredFields())); + tempFields.addAll(Arrays.asList(clazz.getDeclaredFields())); + for (Field field : tempFields) { + if (!ArrayUtils.contains(this.excludeFields, field.getName())) { + // 单注解 + if (field.isAnnotationPresent(Excel.class)) { + Excel attr = field.getAnnotation(Excel.class); + if (attr != null && (attr.type() == Type.ALL || attr.type() == type)) { + field.setAccessible(true); + fields.add(new Object[]{field, attr}); + } + if (Collection.class.isAssignableFrom(field.getType())) { + subMethod = getSubMethod(field.getName(), clazz); + ParameterizedType pt = (ParameterizedType) field.getGenericType(); + Class subClass = (Class) pt.getActualTypeArguments()[0]; + this.subFields = FieldUtils.getFieldsListWithAnnotation(subClass, Excel.class); + } + } + + // 多注解 + if (field.isAnnotationPresent(Excels.class)) { + Excels attrs = field.getAnnotation(Excels.class); + Excel[] excels = attrs.value(); + for (Excel attr : excels) { + if (attr != null && (attr.type() == Type.ALL || attr.type() == type)) { + field.setAccessible(true); + fields.add(new Object[]{field, attr}); + } + } + } + } + } + return fields; + } + + /** + * 根据注解获取最大行高 + */ + public short getRowHeight() { + double maxHeight = 0; + for (Object[] os : this.fields) { + Excel excel = (Excel) os[1]; + maxHeight = Math.max(maxHeight, excel.height()); + } + return (short) (maxHeight * 20); + } + + /** + * 创建一个工作簿 + */ + public void createWorkbook() { + this.wb = new SXSSFWorkbook(500); + this.sheet = wb.createSheet(); + wb.setSheetName(0, sheetName); + this.styles = createStyles(wb); + } + + /** + * 创建工作表 + * + * @param sheetNo sheet数量 + * @param index 序号 + */ + public void createSheet(int sheetNo, int index) { + // 设置工作表的名称. + if (sheetNo > 1 && index > 0) { + this.sheet = wb.createSheet(); + this.createTitle(); + wb.setSheetName(index, sheetName + index); + } + } + + /** + * 获取单元格值 + * + * @param row 获取的行 + * @param column 获取单元格列号 + * @return 单元格值 + */ + public Object getCellValue(Row row, int column) { + if (row == null) { + return row; + } + Object val = ""; + try { + Cell cell = row.getCell(column); + if (StringUtils.isNotNull(cell)) { + if (cell.getCellType() == CellType.NUMERIC || cell.getCellType() == CellType.FORMULA) { + val = cell.getNumericCellValue(); + if (DateUtil.isCellDateFormatted(cell)) { + val = DateUtil.getJavaDate((Double) val); // POI Excel 日期格式转换 + } else { + if ((Double) val % 1 != 0) { + val = new BigDecimal(val.toString()); + } else { + val = new DecimalFormat("0").format(val); + } + } + } else if (cell.getCellType() == CellType.STRING) { + val = cell.getStringCellValue(); + } else if (cell.getCellType() == CellType.BOOLEAN) { + val = cell.getBooleanCellValue(); + } else if (cell.getCellType() == CellType.ERROR) { + val = cell.getErrorCellValue(); + } + + } + } catch (Exception e) { + return val; + } + return val; + } + + /** + * 判断是否是空行 + * + * @param row 判断的行 + * @return + */ + private boolean isRowEmpty(Row row) { + if (row == null) { + return true; + } + for (int i = row.getFirstCellNum(); i < row.getLastCellNum(); i++) { + Cell cell = row.getCell(i); + if (cell != null && cell.getCellType() != CellType.BLANK) { + return false; + } + } + return true; + } + + /** + * 获取Excel2003图片 + * + * @param sheet 当前sheet对象 + * @param workbook 工作簿对象 + * @return Map key:图片单元格索引(1_1)String,value:图片流PictureData + */ + public static Map getSheetPictures03(HSSFSheet sheet, HSSFWorkbook workbook) { + Map sheetIndexPicMap = new HashMap(); + List pictures = workbook.getAllPictures(); + if (!pictures.isEmpty()) { + for (HSSFShape shape : sheet.getDrawingPatriarch().getChildren()) { + HSSFClientAnchor anchor = (HSSFClientAnchor) shape.getAnchor(); + if (shape instanceof HSSFPicture) { + HSSFPicture pic = (HSSFPicture) shape; + int pictureIndex = pic.getPictureIndex() - 1; + HSSFPictureData picData = pictures.get(pictureIndex); + String picIndex = String.valueOf(anchor.getRow1()) + "_" + String.valueOf(anchor.getCol1()); + sheetIndexPicMap.put(picIndex, picData); + } + } + return sheetIndexPicMap; + } else { + return sheetIndexPicMap; + } + } + + /** + * 获取Excel2007图片 + * + * @param sheet 当前sheet对象 + * @param workbook 工作簿对象 + * @return Map key:图片单元格索引(1_1)String,value:图片流PictureData + */ + public static Map getSheetPictures07(XSSFSheet sheet, XSSFWorkbook workbook) { + Map sheetIndexPicMap = new HashMap(); + for (POIXMLDocumentPart dr : sheet.getRelations()) { + if (dr instanceof XSSFDrawing) { + XSSFDrawing drawing = (XSSFDrawing) dr; + List shapes = drawing.getShapes(); + for (XSSFShape shape : shapes) { + if (shape instanceof XSSFPicture) { + XSSFPicture pic = (XSSFPicture) shape; + XSSFClientAnchor anchor = pic.getPreferredSize(); + CTMarker ctMarker = anchor.getFrom(); + String picIndex = ctMarker.getRow() + "_" + ctMarker.getCol(); + sheetIndexPicMap.put(picIndex, pic.getPictureData()); + } + } + } + } + return sheetIndexPicMap; + } + + /** + * 格式化不同类型的日期对象 + * + * @param dateFormat 日期格式 + * @param val 被格式化的日期对象 + * @return 格式化后的日期字符 + */ + public String parseDateToStr(String dateFormat, Object val) { + if (val == null) { + return ""; + } + String str; + if (val instanceof Date) { + str = DateUtils.parseDateToStr(dateFormat, (Date) val); + } else if (val instanceof LocalDateTime) { + str = DateUtils.parseDateToStr(dateFormat, DateUtils.toDate((LocalDateTime) val)); + } else if (val instanceof LocalDate) { + str = DateUtils.parseDateToStr(dateFormat, DateUtils.toDate((LocalDate) val)); + } else { + str = val.toString(); + } + return str; + } + + /** + * 是否有对象的子列表 + */ + public boolean isSubList() { + return StringUtils.isNotNull(subFields) && subFields.size() > 0; + } + + /** + * 是否有对象的子列表,集合不为空 + */ + public boolean isSubListValue(T vo) { + return StringUtils.isNotNull(subFields) && subFields.size() > 0 && StringUtils.isNotNull(getListCellValue(vo)) && getListCellValue(vo).size() > 0; + } + + /** + * 获取集合的值 + */ + public Collection getListCellValue(Object obj) { + Object value; + try { + value = subMethod.invoke(obj, new Object[]{}); + } catch (Exception e) { + return new ArrayList(); + } + return (Collection) value; + } + + /** + * 获取对象的子列表方法 + * + * @param name 名称 + * @param pojoClass 类对象 + * @return 子列表方法 + */ + public Method getSubMethod(String name, Class pojoClass) { + StringBuffer getMethodName = new StringBuffer("get"); + getMethodName.append(name.substring(0, 1).toUpperCase()); + getMethodName.append(name.substring(1)); + Method method = null; + try { + method = pojoClass.getMethod(getMethodName.toString(), new Class[]{}); + } catch (Exception e) { + log.error("获取对象异常{}", e.getMessage()); + } + return method; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/reflect/ReflectUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/reflect/ReflectUtils.java new file mode 100644 index 00000000..b19953e0 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/reflect/ReflectUtils.java @@ -0,0 +1,410 @@ +package com.ruoyi.common.utils.reflect; + +import java.lang.reflect.Field; +import java.lang.reflect.InvocationTargetException; +import java.lang.reflect.Method; +import java.lang.reflect.Modifier; +import java.lang.reflect.ParameterizedType; +import java.lang.reflect.Type; +import java.util.Date; +import org.apache.commons.lang3.StringUtils; +import org.apache.commons.lang3.Validate; +import org.apache.poi.ss.usermodel.DateUtil; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import com.ruoyi.common.core.text.Convert; +import com.ruoyi.common.utils.DateUtils; + +/** + * 反射工具类. 提供调用getter/setter方法, 访问私有变量, 调用私有方法, 获取泛型类型Class, 被AOP过的真实类等工具函数. + * + * @author ruoyi + */ +@SuppressWarnings("rawtypes") +public class ReflectUtils +{ + private static final String SETTER_PREFIX = "set"; + + private static final String GETTER_PREFIX = "get"; + + private static final String CGLIB_CLASS_SEPARATOR = "$$"; + + private static Logger logger = LoggerFactory.getLogger(ReflectUtils.class); + + /** + * 调用Getter方法. + * 支持多级,如:对象名.对象名.方法 + */ + @SuppressWarnings("unchecked") + public static E invokeGetter(Object obj, String propertyName) + { + Object object = obj; + for (String name : StringUtils.split(propertyName, ".")) + { + String getterMethodName = GETTER_PREFIX + StringUtils.capitalize(name); + object = invokeMethod(object, getterMethodName, new Class[] {}, new Object[] {}); + } + return (E) object; + } + + /** + * 调用Setter方法, 仅匹配方法名。 + * 支持多级,如:对象名.对象名.方法 + */ + public static void invokeSetter(Object obj, String propertyName, E value) + { + Object object = obj; + String[] names = StringUtils.split(propertyName, "."); + for (int i = 0; i < names.length; i++) + { + if (i < names.length - 1) + { + String getterMethodName = GETTER_PREFIX + StringUtils.capitalize(names[i]); + object = invokeMethod(object, getterMethodName, new Class[] {}, new Object[] {}); + } + else + { + String setterMethodName = SETTER_PREFIX + StringUtils.capitalize(names[i]); + invokeMethodByName(object, setterMethodName, new Object[] { value }); + } + } + } + + /** + * 直接读取对象属性值, 无视private/protected修饰符, 不经过getter函数. + */ + @SuppressWarnings("unchecked") + public static E getFieldValue(final Object obj, final String fieldName) + { + Field field = getAccessibleField(obj, fieldName); + if (field == null) + { + logger.debug("在 [" + obj.getClass() + "] 中,没有找到 [" + fieldName + "] 字段 "); + return null; + } + E result = null; + try + { + result = (E) field.get(obj); + } + catch (IllegalAccessException e) + { + logger.error("不可能抛出的异常{}", e.getMessage()); + } + return result; + } + + /** + * 直接设置对象属性值, 无视private/protected修饰符, 不经过setter函数. + */ + public static void setFieldValue(final Object obj, final String fieldName, final E value) + { + Field field = getAccessibleField(obj, fieldName); + if (field == null) + { + // throw new IllegalArgumentException("在 [" + obj.getClass() + "] 中,没有找到 [" + fieldName + "] 字段 "); + logger.debug("在 [" + obj.getClass() + "] 中,没有找到 [" + fieldName + "] 字段 "); + return; + } + try + { + field.set(obj, value); + } + catch (IllegalAccessException e) + { + logger.error("不可能抛出的异常: {}", e.getMessage()); + } + } + + /** + * 直接调用对象方法, 无视private/protected修饰符. + * 用于一次性调用的情况,否则应使用getAccessibleMethod()函数获得Method后反复调用. + * 同时匹配方法名+参数类型, + */ + @SuppressWarnings("unchecked") + public static E invokeMethod(final Object obj, final String methodName, final Class[] parameterTypes, + final Object[] args) + { + if (obj == null || methodName == null) + { + return null; + } + Method method = getAccessibleMethod(obj, methodName, parameterTypes); + if (method == null) + { + logger.debug("在 [" + obj.getClass() + "] 中,没有找到 [" + methodName + "] 方法 "); + return null; + } + try + { + return (E) method.invoke(obj, args); + } + catch (Exception e) + { + String msg = "method: " + method + ", obj: " + obj + ", args: " + args + ""; + throw convertReflectionExceptionToUnchecked(msg, e); + } + } + + /** + * 直接调用对象方法, 无视private/protected修饰符, + * 用于一次性调用的情况,否则应使用getAccessibleMethodByName()函数获得Method后反复调用. + * 只匹配函数名,如果有多个同名函数调用第一个。 + */ + @SuppressWarnings("unchecked") + public static E invokeMethodByName(final Object obj, final String methodName, final Object[] args) + { + Method method = getAccessibleMethodByName(obj, methodName, args.length); + if (method == null) + { + // 如果为空不报错,直接返回空。 + logger.debug("在 [" + obj.getClass() + "] 中,没有找到 [" + methodName + "] 方法 "); + return null; + } + try + { + // 类型转换(将参数数据类型转换为目标方法参数类型) + Class[] cs = method.getParameterTypes(); + for (int i = 0; i < cs.length; i++) + { + if (args[i] != null && !args[i].getClass().equals(cs[i])) + { + if (cs[i] == String.class) + { + args[i] = Convert.toStr(args[i]); + if (StringUtils.endsWith((String) args[i], ".0")) + { + args[i] = StringUtils.substringBefore((String) args[i], ".0"); + } + } + else if (cs[i] == Integer.class) + { + args[i] = Convert.toInt(args[i]); + } + else if (cs[i] == Long.class) + { + args[i] = Convert.toLong(args[i]); + } + else if (cs[i] == Double.class) + { + args[i] = Convert.toDouble(args[i]); + } + else if (cs[i] == Float.class) + { + args[i] = Convert.toFloat(args[i]); + } + else if (cs[i] == Date.class) + { + if (args[i] instanceof String) + { + args[i] = DateUtils.parseDate(args[i]); + } + else + { + args[i] = DateUtil.getJavaDate((Double) args[i]); + } + } + else if (cs[i] == boolean.class || cs[i] == Boolean.class) + { + args[i] = Convert.toBool(args[i]); + } + } + } + return (E) method.invoke(obj, args); + } + catch (Exception e) + { + String msg = "method: " + method + ", obj: " + obj + ", args: " + args + ""; + throw convertReflectionExceptionToUnchecked(msg, e); + } + } + + /** + * 循环向上转型, 获取对象的DeclaredField, 并强制设置为可访问. + * 如向上转型到Object仍无法找到, 返回null. + */ + public static Field getAccessibleField(final Object obj, final String fieldName) + { + // 为空不报错。直接返回 null + if (obj == null) + { + return null; + } + Validate.notBlank(fieldName, "fieldName can't be blank"); + for (Class superClass = obj.getClass(); superClass != Object.class; superClass = superClass.getSuperclass()) + { + try + { + Field field = superClass.getDeclaredField(fieldName); + makeAccessible(field); + return field; + } + catch (NoSuchFieldException e) + { + continue; + } + } + return null; + } + + /** + * 循环向上转型, 获取对象的DeclaredMethod,并强制设置为可访问. + * 如向上转型到Object仍无法找到, 返回null. + * 匹配函数名+参数类型。 + * 用于方法需要被多次调用的情况. 先使用本函数先取得Method,然后调用Method.invoke(Object obj, Object... args) + */ + public static Method getAccessibleMethod(final Object obj, final String methodName, + final Class... parameterTypes) + { + // 为空不报错。直接返回 null + if (obj == null) + { + return null; + } + Validate.notBlank(methodName, "methodName can't be blank"); + for (Class searchType = obj.getClass(); searchType != Object.class; searchType = searchType.getSuperclass()) + { + try + { + Method method = searchType.getDeclaredMethod(methodName, parameterTypes); + makeAccessible(method); + return method; + } + catch (NoSuchMethodException e) + { + continue; + } + } + return null; + } + + /** + * 循环向上转型, 获取对象的DeclaredMethod,并强制设置为可访问. + * 如向上转型到Object仍无法找到, 返回null. + * 只匹配函数名。 + * 用于方法需要被多次调用的情况. 先使用本函数先取得Method,然后调用Method.invoke(Object obj, Object... args) + */ + public static Method getAccessibleMethodByName(final Object obj, final String methodName, int argsNum) + { + // 为空不报错。直接返回 null + if (obj == null) + { + return null; + } + Validate.notBlank(methodName, "methodName can't be blank"); + for (Class searchType = obj.getClass(); searchType != Object.class; searchType = searchType.getSuperclass()) + { + Method[] methods = searchType.getDeclaredMethods(); + for (Method method : methods) + { + if (method.getName().equals(methodName) && method.getParameterTypes().length == argsNum) + { + makeAccessible(method); + return method; + } + } + } + return null; + } + + /** + * 改变private/protected的方法为public,尽量不调用实际改动的语句,避免JDK的SecurityManager抱怨。 + */ + public static void makeAccessible(Method method) + { + if ((!Modifier.isPublic(method.getModifiers()) || !Modifier.isPublic(method.getDeclaringClass().getModifiers())) + && !method.isAccessible()) + { + method.setAccessible(true); + } + } + + /** + * 改变private/protected的成员变量为public,尽量不调用实际改动的语句,避免JDK的SecurityManager抱怨。 + */ + public static void makeAccessible(Field field) + { + if ((!Modifier.isPublic(field.getModifiers()) || !Modifier.isPublic(field.getDeclaringClass().getModifiers()) + || Modifier.isFinal(field.getModifiers())) && !field.isAccessible()) + { + field.setAccessible(true); + } + } + + /** + * 通过反射, 获得Class定义中声明的泛型参数的类型, 注意泛型必须定义在父类处 + * 如无法找到, 返回Object.class. + */ + @SuppressWarnings("unchecked") + public static Class getClassGenricType(final Class clazz) + { + return getClassGenricType(clazz, 0); + } + + /** + * 通过反射, 获得Class定义中声明的父类的泛型参数的类型. + * 如无法找到, 返回Object.class. + */ + public static Class getClassGenricType(final Class clazz, final int index) + { + Type genType = clazz.getGenericSuperclass(); + + if (!(genType instanceof ParameterizedType)) + { + logger.debug(clazz.getSimpleName() + "'s superclass not ParameterizedType"); + return Object.class; + } + + Type[] params = ((ParameterizedType) genType).getActualTypeArguments(); + + if (index >= params.length || index < 0) + { + logger.debug("Index: " + index + ", Size of " + clazz.getSimpleName() + "'s Parameterized Type: " + + params.length); + return Object.class; + } + if (!(params[index] instanceof Class)) + { + logger.debug(clazz.getSimpleName() + " not set the actual class on superclass generic parameter"); + return Object.class; + } + + return (Class) params[index]; + } + + public static Class getUserClass(Object instance) + { + if (instance == null) + { + throw new RuntimeException("Instance must not be null"); + } + Class clazz = instance.getClass(); + if (clazz != null && clazz.getName().contains(CGLIB_CLASS_SEPARATOR)) + { + Class superClass = clazz.getSuperclass(); + if (superClass != null && !Object.class.equals(superClass)) + { + return superClass; + } + } + return clazz; + + } + + /** + * 将反射时的checked exception转换为unchecked exception. + */ + public static RuntimeException convertReflectionExceptionToUnchecked(String msg, Exception e) + { + if (e instanceof IllegalAccessException || e instanceof IllegalArgumentException + || e instanceof NoSuchMethodException) + { + return new IllegalArgumentException(msg, e); + } + else if (e instanceof InvocationTargetException) + { + return new RuntimeException(msg, ((InvocationTargetException) e).getTargetException()); + } + return new RuntimeException(msg, e); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/sign/Base64.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/sign/Base64.java new file mode 100644 index 00000000..ca1cd924 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/sign/Base64.java @@ -0,0 +1,291 @@ +package com.ruoyi.common.utils.sign; + +/** + * Base64工具类 + * + * @author ruoyi + */ +public final class Base64 +{ + static private final int BASELENGTH = 128; + static private final int LOOKUPLENGTH = 64; + static private final int TWENTYFOURBITGROUP = 24; + static private final int EIGHTBIT = 8; + static private final int SIXTEENBIT = 16; + static private final int FOURBYTE = 4; + static private final int SIGN = -128; + static private final char PAD = '='; + static final private byte[] base64Alphabet = new byte[BASELENGTH]; + static final private char[] lookUpBase64Alphabet = new char[LOOKUPLENGTH]; + + static + { + for (int i = 0; i < BASELENGTH; ++i) + { + base64Alphabet[i] = -1; + } + for (int i = 'Z'; i >= 'A'; i--) + { + base64Alphabet[i] = (byte) (i - 'A'); + } + for (int i = 'z'; i >= 'a'; i--) + { + base64Alphabet[i] = (byte) (i - 'a' + 26); + } + + for (int i = '9'; i >= '0'; i--) + { + base64Alphabet[i] = (byte) (i - '0' + 52); + } + + base64Alphabet['+'] = 62; + base64Alphabet['/'] = 63; + + for (int i = 0; i <= 25; i++) + { + lookUpBase64Alphabet[i] = (char) ('A' + i); + } + + for (int i = 26, j = 0; i <= 51; i++, j++) + { + lookUpBase64Alphabet[i] = (char) ('a' + j); + } + + for (int i = 52, j = 0; i <= 61; i++, j++) + { + lookUpBase64Alphabet[i] = (char) ('0' + j); + } + lookUpBase64Alphabet[62] = (char) '+'; + lookUpBase64Alphabet[63] = (char) '/'; + } + + private static boolean isWhiteSpace(char octect) + { + return (octect == 0x20 || octect == 0xd || octect == 0xa || octect == 0x9); + } + + private static boolean isPad(char octect) + { + return (octect == PAD); + } + + private static boolean isData(char octect) + { + return (octect < BASELENGTH && base64Alphabet[octect] != -1); + } + + /** + * Encodes hex octects into Base64 + * + * @param binaryData Array containing binaryData + * @return Encoded Base64 array + */ + public static String encode(byte[] binaryData) + { + if (binaryData == null) + { + return null; + } + + int lengthDataBits = binaryData.length * EIGHTBIT; + if (lengthDataBits == 0) + { + return ""; + } + + int fewerThan24bits = lengthDataBits % TWENTYFOURBITGROUP; + int numberTriplets = lengthDataBits / TWENTYFOURBITGROUP; + int numberQuartet = fewerThan24bits != 0 ? numberTriplets + 1 : numberTriplets; + char encodedData[] = null; + + encodedData = new char[numberQuartet * 4]; + + byte k = 0, l = 0, b1 = 0, b2 = 0, b3 = 0; + + int encodedIndex = 0; + int dataIndex = 0; + + for (int i = 0; i < numberTriplets; i++) + { + b1 = binaryData[dataIndex++]; + b2 = binaryData[dataIndex++]; + b3 = binaryData[dataIndex++]; + + l = (byte) (b2 & 0x0f); + k = (byte) (b1 & 0x03); + + byte val1 = ((b1 & SIGN) == 0) ? (byte) (b1 >> 2) : (byte) ((b1) >> 2 ^ 0xc0); + byte val2 = ((b2 & SIGN) == 0) ? (byte) (b2 >> 4) : (byte) ((b2) >> 4 ^ 0xf0); + byte val3 = ((b3 & SIGN) == 0) ? (byte) (b3 >> 6) : (byte) ((b3) >> 6 ^ 0xfc); + + encodedData[encodedIndex++] = lookUpBase64Alphabet[val1]; + encodedData[encodedIndex++] = lookUpBase64Alphabet[val2 | (k << 4)]; + encodedData[encodedIndex++] = lookUpBase64Alphabet[(l << 2) | val3]; + encodedData[encodedIndex++] = lookUpBase64Alphabet[b3 & 0x3f]; + } + + // form integral number of 6-bit groups + if (fewerThan24bits == EIGHTBIT) + { + b1 = binaryData[dataIndex]; + k = (byte) (b1 & 0x03); + byte val1 = ((b1 & SIGN) == 0) ? (byte) (b1 >> 2) : (byte) ((b1) >> 2 ^ 0xc0); + encodedData[encodedIndex++] = lookUpBase64Alphabet[val1]; + encodedData[encodedIndex++] = lookUpBase64Alphabet[k << 4]; + encodedData[encodedIndex++] = PAD; + encodedData[encodedIndex++] = PAD; + } + else if (fewerThan24bits == SIXTEENBIT) + { + b1 = binaryData[dataIndex]; + b2 = binaryData[dataIndex + 1]; + l = (byte) (b2 & 0x0f); + k = (byte) (b1 & 0x03); + + byte val1 = ((b1 & SIGN) == 0) ? (byte) (b1 >> 2) : (byte) ((b1) >> 2 ^ 0xc0); + byte val2 = ((b2 & SIGN) == 0) ? (byte) (b2 >> 4) : (byte) ((b2) >> 4 ^ 0xf0); + + encodedData[encodedIndex++] = lookUpBase64Alphabet[val1]; + encodedData[encodedIndex++] = lookUpBase64Alphabet[val2 | (k << 4)]; + encodedData[encodedIndex++] = lookUpBase64Alphabet[l << 2]; + encodedData[encodedIndex++] = PAD; + } + return new String(encodedData); + } + + /** + * Decodes Base64 data into octects + * + * @param encoded string containing Base64 data + * @return Array containind decoded data. + */ + public static byte[] decode(String encoded) + { + if (encoded == null) + { + return null; + } + + char[] base64Data = encoded.toCharArray(); + // remove white spaces + int len = removeWhiteSpace(base64Data); + + if (len % FOURBYTE != 0) + { + return null;// should be divisible by four + } + + int numberQuadruple = (len / FOURBYTE); + + if (numberQuadruple == 0) + { + return new byte[0]; + } + + byte decodedData[] = null; + byte b1 = 0, b2 = 0, b3 = 0, b4 = 0; + char d1 = 0, d2 = 0, d3 = 0, d4 = 0; + + int i = 0; + int encodedIndex = 0; + int dataIndex = 0; + decodedData = new byte[(numberQuadruple) * 3]; + + for (; i < numberQuadruple - 1; i++) + { + + if (!isData((d1 = base64Data[dataIndex++])) || !isData((d2 = base64Data[dataIndex++])) + || !isData((d3 = base64Data[dataIndex++])) || !isData((d4 = base64Data[dataIndex++]))) + { + return null; + } // if found "no data" just return null + + b1 = base64Alphabet[d1]; + b2 = base64Alphabet[d2]; + b3 = base64Alphabet[d3]; + b4 = base64Alphabet[d4]; + + decodedData[encodedIndex++] = (byte) (b1 << 2 | b2 >> 4); + decodedData[encodedIndex++] = (byte) (((b2 & 0xf) << 4) | ((b3 >> 2) & 0xf)); + decodedData[encodedIndex++] = (byte) (b3 << 6 | b4); + } + + if (!isData((d1 = base64Data[dataIndex++])) || !isData((d2 = base64Data[dataIndex++]))) + { + return null;// if found "no data" just return null + } + + b1 = base64Alphabet[d1]; + b2 = base64Alphabet[d2]; + + d3 = base64Data[dataIndex++]; + d4 = base64Data[dataIndex++]; + if (!isData((d3)) || !isData((d4))) + {// Check if they are PAD characters + if (isPad(d3) && isPad(d4)) + { + if ((b2 & 0xf) != 0)// last 4 bits should be zero + { + return null; + } + byte[] tmp = new byte[i * 3 + 1]; + System.arraycopy(decodedData, 0, tmp, 0, i * 3); + tmp[encodedIndex] = (byte) (b1 << 2 | b2 >> 4); + return tmp; + } + else if (!isPad(d3) && isPad(d4)) + { + b3 = base64Alphabet[d3]; + if ((b3 & 0x3) != 0)// last 2 bits should be zero + { + return null; + } + byte[] tmp = new byte[i * 3 + 2]; + System.arraycopy(decodedData, 0, tmp, 0, i * 3); + tmp[encodedIndex++] = (byte) (b1 << 2 | b2 >> 4); + tmp[encodedIndex] = (byte) (((b2 & 0xf) << 4) | ((b3 >> 2) & 0xf)); + return tmp; + } + else + { + return null; + } + } + else + { // No PAD e.g 3cQl + b3 = base64Alphabet[d3]; + b4 = base64Alphabet[d4]; + decodedData[encodedIndex++] = (byte) (b1 << 2 | b2 >> 4); + decodedData[encodedIndex++] = (byte) (((b2 & 0xf) << 4) | ((b3 >> 2) & 0xf)); + decodedData[encodedIndex++] = (byte) (b3 << 6 | b4); + + } + return decodedData; + } + + /** + * remove WhiteSpace from MIME containing encoded Base64 data. + * + * @param data the byte array of base64 data (with WS) + * @return the new length + */ + private static int removeWhiteSpace(char[] data) + { + if (data == null) + { + return 0; + } + + // count characters that's not whitespace + int newSize = 0; + int len = data.length; + for (int i = 0; i < len; i++) + { + if (!isWhiteSpace(data[i])) + { + data[newSize++] = data[i]; + } + } + return newSize; + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/sign/Md5Utils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/sign/Md5Utils.java new file mode 100644 index 00000000..c1c58dbc --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/sign/Md5Utils.java @@ -0,0 +1,67 @@ +package com.ruoyi.common.utils.sign; + +import java.nio.charset.StandardCharsets; +import java.security.MessageDigest; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * Md5加密方法 + * + * @author ruoyi + */ +public class Md5Utils +{ + private static final Logger log = LoggerFactory.getLogger(Md5Utils.class); + + private static byte[] md5(String s) + { + MessageDigest algorithm; + try + { + algorithm = MessageDigest.getInstance("MD5"); + algorithm.reset(); + algorithm.update(s.getBytes("UTF-8")); + byte[] messageDigest = algorithm.digest(); + return messageDigest; + } + catch (Exception e) + { + log.error("MD5 Error...", e); + } + return null; + } + + private static final String toHex(byte hash[]) + { + if (hash == null) + { + return null; + } + StringBuffer buf = new StringBuffer(hash.length * 2); + int i; + + for (i = 0; i < hash.length; i++) + { + if ((hash[i] & 0xff) < 0x10) + { + buf.append("0"); + } + buf.append(Long.toString(hash[i] & 0xff, 16)); + } + return buf.toString(); + } + + public static String hash(String s) + { + try + { + return new String(toHex(md5(s)).getBytes(StandardCharsets.UTF_8), StandardCharsets.UTF_8); + } + catch (Exception e) + { + log.error("not supported charset...{}", e); + return s; + } + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/spring/SpringUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/spring/SpringUtils.java new file mode 100644 index 00000000..f290ec37 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/spring/SpringUtils.java @@ -0,0 +1,158 @@ +package com.ruoyi.common.utils.spring; + +import org.springframework.aop.framework.AopContext; +import org.springframework.beans.BeansException; +import org.springframework.beans.factory.NoSuchBeanDefinitionException; +import org.springframework.beans.factory.config.BeanFactoryPostProcessor; +import org.springframework.beans.factory.config.ConfigurableListableBeanFactory; +import org.springframework.context.ApplicationContext; +import org.springframework.context.ApplicationContextAware; +import org.springframework.stereotype.Component; +import com.ruoyi.common.utils.StringUtils; + +/** + * spring工具类 方便在非spring管理环境中获取bean + * + * @author ruoyi + */ +@Component +public final class SpringUtils implements BeanFactoryPostProcessor, ApplicationContextAware +{ + /** Spring应用上下文环境 */ + private static ConfigurableListableBeanFactory beanFactory; + + private static ApplicationContext applicationContext; + + @Override + public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException + { + SpringUtils.beanFactory = beanFactory; + } + + @Override + public void setApplicationContext(ApplicationContext applicationContext) throws BeansException + { + SpringUtils.applicationContext = applicationContext; + } + + /** + * 获取对象 + * + * @param name + * @return Object 一个以所给名字注册的bean的实例 + * @throws org.springframework.beans.BeansException + * + */ + @SuppressWarnings("unchecked") + public static T getBean(String name) throws BeansException + { + return (T) beanFactory.getBean(name); + } + + /** + * 获取类型为requiredType的对象 + * + * @param clz + * @return + * @throws org.springframework.beans.BeansException + * + */ + public static T getBean(Class clz) throws BeansException + { + T result = (T) beanFactory.getBean(clz); + return result; + } + + /** + * 如果BeanFactory包含一个与所给名称匹配的bean定义,则返回true + * + * @param name + * @return boolean + */ + public static boolean containsBean(String name) + { + return beanFactory.containsBean(name); + } + + /** + * 判断以给定名字注册的bean定义是一个singleton还是一个prototype。 如果与给定名字相应的bean定义没有被找到,将会抛出一个异常(NoSuchBeanDefinitionException) + * + * @param name + * @return boolean + * @throws org.springframework.beans.factory.NoSuchBeanDefinitionException + * + */ + public static boolean isSingleton(String name) throws NoSuchBeanDefinitionException + { + return beanFactory.isSingleton(name); + } + + /** + * @param name + * @return Class 注册对象的类型 + * @throws org.springframework.beans.factory.NoSuchBeanDefinitionException + * + */ + public static Class getType(String name) throws NoSuchBeanDefinitionException + { + return beanFactory.getType(name); + } + + /** + * 如果给定的bean名字在bean定义中有别名,则返回这些别名 + * + * @param name + * @return + * @throws org.springframework.beans.factory.NoSuchBeanDefinitionException + * + */ + public static String[] getAliases(String name) throws NoSuchBeanDefinitionException + { + return beanFactory.getAliases(name); + } + + /** + * 获取aop代理对象 + * + * @param invoker + * @return + */ + @SuppressWarnings("unchecked") + public static T getAopProxy(T invoker) + { + return (T) AopContext.currentProxy(); + } + + /** + * 获取当前的环境配置,无配置返回null + * + * @return 当前的环境配置 + */ + public static String[] getActiveProfiles() + { + return applicationContext.getEnvironment().getActiveProfiles(); + } + + /** + * 获取当前的环境配置,当有多个环境配置时,只获取第一个 + * + * @return 当前的环境配置 + */ + public static String getActiveProfile() + { + final String[] activeProfiles = getActiveProfiles(); + return StringUtils.isNotEmpty(activeProfiles) ? activeProfiles[0] : null; + } + + /** + * 获取配置文件中的值 + * + * @param key 配置文件的key + * @return 当前的配置文件的值 + * + */ + public static String getRequiredProperty(String key) + { + return applicationContext.getEnvironment().getRequiredProperty(key); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/sql/SqlUtil.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/sql/SqlUtil.java new file mode 100644 index 00000000..246a9cfc --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/sql/SqlUtil.java @@ -0,0 +1,61 @@ +package com.ruoyi.common.utils.sql; + +import com.ruoyi.common.exception.UtilException; +import com.ruoyi.common.utils.StringUtils; + +/** + * sql操作工具类 + * + * @author ruoyi + */ +public class SqlUtil +{ + /** + * 定义常用的 sql关键字 + */ + public static String SQL_REGEX = "select |insert |delete |update |drop |count |exec |chr |mid |master |truncate |char |and |declare "; + + /** + * 仅支持字母、数字、下划线、空格、逗号、小数点(支持多个字段排序) + */ + public static String SQL_PATTERN = "[a-zA-Z0-9_\\ \\,\\.]+"; + + /** + * 检查字符,防止注入绕过 + */ + public static String escapeOrderBySql(String value) + { + if (StringUtils.isNotEmpty(value) && !isValidOrderBySql(value)) + { + throw new UtilException("参数不符合规范,不能进行查询"); + } + return value; + } + + /** + * 验证 order by 语法是否符合规范 + */ + public static boolean isValidOrderBySql(String value) + { + return value.matches(SQL_PATTERN); + } + + /** + * SQL关键字检查 + */ + public static void filterKeyword(String value) + { + if (StringUtils.isEmpty(value)) + { + return; + } + String[] sqlKeywords = StringUtils.split(SQL_REGEX, "\\|"); + for (String sqlKeyword : sqlKeywords) + { + if (StringUtils.indexOfIgnoreCase(value, sqlKeyword) > -1) + { + throw new UtilException("参数存在SQL注入风险"); + } + } + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/uuid/IdUtils.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/uuid/IdUtils.java new file mode 100644 index 00000000..2c844271 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/uuid/IdUtils.java @@ -0,0 +1,49 @@ +package com.ruoyi.common.utils.uuid; + +/** + * ID生成器工具类 + * + * @author ruoyi + */ +public class IdUtils +{ + /** + * 获取随机UUID + * + * @return 随机UUID + */ + public static String randomUUID() + { + return UUID.randomUUID().toString(); + } + + /** + * 简化的UUID,去掉了横线 + * + * @return 简化的UUID,去掉了横线 + */ + public static String simpleUUID() + { + return UUID.randomUUID().toString(true); + } + + /** + * 获取随机UUID,使用性能更好的ThreadLocalRandom生成UUID + * + * @return 随机UUID + */ + public static String fastUUID() + { + return UUID.fastUUID().toString(); + } + + /** + * 简化的UUID,去掉了横线,使用性能更好的ThreadLocalRandom生成UUID + * + * @return 简化的UUID,去掉了横线 + */ + public static String fastSimpleUUID() + { + return UUID.fastUUID().toString(true); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/uuid/Seq.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/uuid/Seq.java new file mode 100644 index 00000000..528f3c1b --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/uuid/Seq.java @@ -0,0 +1,86 @@ +package com.ruoyi.common.utils.uuid; + +import java.util.concurrent.atomic.AtomicInteger; +import com.ruoyi.common.utils.DateUtils; +import com.ruoyi.common.utils.StringUtils; + +/** + * @author ruoyi 序列生成类 + */ +public class Seq +{ + // 通用序列类型 + public static final String commSeqType = "COMMON"; + + // 上传序列类型 + public static final String uploadSeqType = "UPLOAD"; + + // 通用接口序列数 + private static AtomicInteger commSeq = new AtomicInteger(1); + + // 上传接口序列数 + private static AtomicInteger uploadSeq = new AtomicInteger(1); + + // 机器标识 + private static String machineCode = "A"; + + /** + * 获取通用序列号 + * + * @return 序列值 + */ + public static String getId() + { + return getId(commSeqType); + } + + /** + * 默认16位序列号 yyMMddHHmmss + 一位机器标识 + 3长度循环递增字符串 + * + * @return 序列值 + */ + public static String getId(String type) + { + AtomicInteger atomicInt = commSeq; + if (uploadSeqType.equals(type)) + { + atomicInt = uploadSeq; + } + return getId(atomicInt, 3); + } + + /** + * 通用接口序列号 yyMMddHHmmss + 一位机器标识 + length长度循环递增字符串 + * + * @param atomicInt 序列数 + * @param length 数值长度 + * @return 序列值 + */ + public static String getId(AtomicInteger atomicInt, int length) + { + String result = DateUtils.dateTimeNow(); + result += machineCode; + result += getSeq(atomicInt, length); + return result; + } + + /** + * 序列循环递增字符串[1, 10 的 (length)幂次方), 用0左补齐length位数 + * + * @return 序列值 + */ + private synchronized static String getSeq(AtomicInteger atomicInt, int length) + { + // 先取值再+1 + int value = atomicInt.getAndIncrement(); + + // 如果更新后值>=10 的 (length)幂次方则重置为1 + int maxSeq = (int) Math.pow(10, length); + if (atomicInt.get() >= maxSeq) + { + atomicInt.set(1); + } + // 转字符串,用0左补齐 + return StringUtils.padl(value, length); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/utils/uuid/UUID.java b/ruoyi-common/src/main/java/com/ruoyi/common/utils/uuid/UUID.java new file mode 100644 index 00000000..dfda46cf --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/utils/uuid/UUID.java @@ -0,0 +1,484 @@ +package com.ruoyi.common.utils.uuid; + +import java.security.MessageDigest; +import java.security.NoSuchAlgorithmException; +import java.security.SecureRandom; +import java.util.Random; +import java.util.concurrent.ThreadLocalRandom; +import com.ruoyi.common.exception.UtilException; + +/** + * 提供通用唯一识别码(universally unique identifier)(UUID)实现 + * + * @author ruoyi + */ +public final class UUID implements java.io.Serializable, Comparable +{ + private static final long serialVersionUID = -1185015143654744140L; + + /** + * SecureRandom 的单例 + * + */ + private static class Holder + { + static final SecureRandom numberGenerator = getSecureRandom(); + } + + /** 此UUID的最高64有效位 */ + private final long mostSigBits; + + /** 此UUID的最低64有效位 */ + private final long leastSigBits; + + /** + * 私有构造 + * + * @param data 数据 + */ + private UUID(byte[] data) + { + long msb = 0; + long lsb = 0; + assert data.length == 16 : "data must be 16 bytes in length"; + for (int i = 0; i < 8; i++) + { + msb = (msb << 8) | (data[i] & 0xff); + } + for (int i = 8; i < 16; i++) + { + lsb = (lsb << 8) | (data[i] & 0xff); + } + this.mostSigBits = msb; + this.leastSigBits = lsb; + } + + /** + * 使用指定的数据构造新的 UUID。 + * + * @param mostSigBits 用于 {@code UUID} 的最高有效 64 位 + * @param leastSigBits 用于 {@code UUID} 的最低有效 64 位 + */ + public UUID(long mostSigBits, long leastSigBits) + { + this.mostSigBits = mostSigBits; + this.leastSigBits = leastSigBits; + } + + /** + * 获取类型 4(伪随机生成的)UUID 的静态工厂。 使用加密的本地线程伪随机数生成器生成该 UUID。 + * + * @return 随机生成的 {@code UUID} + */ + public static UUID fastUUID() + { + return randomUUID(false); + } + + /** + * 获取类型 4(伪随机生成的)UUID 的静态工厂。 使用加密的强伪随机数生成器生成该 UUID。 + * + * @return 随机生成的 {@code UUID} + */ + public static UUID randomUUID() + { + return randomUUID(true); + } + + /** + * 获取类型 4(伪随机生成的)UUID 的静态工厂。 使用加密的强伪随机数生成器生成该 UUID。 + * + * @param isSecure 是否使用{@link SecureRandom}如果是可以获得更安全的随机码,否则可以得到更好的性能 + * @return 随机生成的 {@code UUID} + */ + public static UUID randomUUID(boolean isSecure) + { + final Random ng = isSecure ? Holder.numberGenerator : getRandom(); + + byte[] randomBytes = new byte[16]; + ng.nextBytes(randomBytes); + randomBytes[6] &= 0x0f; /* clear version */ + randomBytes[6] |= 0x40; /* set to version 4 */ + randomBytes[8] &= 0x3f; /* clear variant */ + randomBytes[8] |= 0x80; /* set to IETF variant */ + return new UUID(randomBytes); + } + + /** + * 根据指定的字节数组获取类型 3(基于名称的)UUID 的静态工厂。 + * + * @param name 用于构造 UUID 的字节数组。 + * + * @return 根据指定数组生成的 {@code UUID} + */ + public static UUID nameUUIDFromBytes(byte[] name) + { + MessageDigest md; + try + { + md = MessageDigest.getInstance("MD5"); + } + catch (NoSuchAlgorithmException nsae) + { + throw new InternalError("MD5 not supported"); + } + byte[] md5Bytes = md.digest(name); + md5Bytes[6] &= 0x0f; /* clear version */ + md5Bytes[6] |= 0x30; /* set to version 3 */ + md5Bytes[8] &= 0x3f; /* clear variant */ + md5Bytes[8] |= 0x80; /* set to IETF variant */ + return new UUID(md5Bytes); + } + + /** + * 根据 {@link #toString()} 方法中描述的字符串标准表示形式创建{@code UUID}。 + * + * @param name 指定 {@code UUID} 字符串 + * @return 具有指定值的 {@code UUID} + * @throws IllegalArgumentException 如果 name 与 {@link #toString} 中描述的字符串表示形式不符抛出此异常 + * + */ + public static UUID fromString(String name) + { + String[] components = name.split("-"); + if (components.length != 5) + { + throw new IllegalArgumentException("Invalid UUID string: " + name); + } + for (int i = 0; i < 5; i++) + { + components[i] = "0x" + components[i]; + } + + long mostSigBits = Long.decode(components[0]).longValue(); + mostSigBits <<= 16; + mostSigBits |= Long.decode(components[1]).longValue(); + mostSigBits <<= 16; + mostSigBits |= Long.decode(components[2]).longValue(); + + long leastSigBits = Long.decode(components[3]).longValue(); + leastSigBits <<= 48; + leastSigBits |= Long.decode(components[4]).longValue(); + + return new UUID(mostSigBits, leastSigBits); + } + + /** + * 返回此 UUID 的 128 位值中的最低有效 64 位。 + * + * @return 此 UUID 的 128 位值中的最低有效 64 位。 + */ + public long getLeastSignificantBits() + { + return leastSigBits; + } + + /** + * 返回此 UUID 的 128 位值中的最高有效 64 位。 + * + * @return 此 UUID 的 128 位值中最高有效 64 位。 + */ + public long getMostSignificantBits() + { + return mostSigBits; + } + + /** + * 与此 {@code UUID} 相关联的版本号. 版本号描述此 {@code UUID} 是如何生成的。 + *

+ * 版本号具有以下含意: + *

    + *
  • 1 基于时间的 UUID + *
  • 2 DCE 安全 UUID + *
  • 3 基于名称的 UUID + *
  • 4 随机生成的 UUID + *
+ * + * @return 此 {@code UUID} 的版本号 + */ + public int version() + { + // Version is bits masked by 0x000000000000F000 in MS long + return (int) ((mostSigBits >> 12) & 0x0f); + } + + /** + * 与此 {@code UUID} 相关联的变体号。变体号描述 {@code UUID} 的布局。 + *

+ * 变体号具有以下含意: + *

    + *
  • 0 为 NCS 向后兼容保留 + *
  • 2 IETF RFC 4122(Leach-Salz), 用于此类 + *
  • 6 保留,微软向后兼容 + *
  • 7 保留供以后定义使用 + *
+ * + * @return 此 {@code UUID} 相关联的变体号 + */ + public int variant() + { + // This field is composed of a varying number of bits. + // 0 - - Reserved for NCS backward compatibility + // 1 0 - The IETF aka Leach-Salz variant (used by this class) + // 1 1 0 Reserved, Microsoft backward compatibility + // 1 1 1 Reserved for future definition. + return (int) ((leastSigBits >>> (64 - (leastSigBits >>> 62))) & (leastSigBits >> 63)); + } + + /** + * 与此 UUID 相关联的时间戳值。 + * + *

+ * 60 位的时间戳值根据此 {@code UUID} 的 time_low、time_mid 和 time_hi 字段构造。
+ * 所得到的时间戳以 100 毫微秒为单位,从 UTC(通用协调时间) 1582 年 10 月 15 日零时开始。 + * + *

+ * 时间戳值仅在在基于时间的 UUID(其 version 类型为 1)中才有意义。
+ * 如果此 {@code UUID} 不是基于时间的 UUID,则此方法抛出 UnsupportedOperationException。 + * + * @throws UnsupportedOperationException 如果此 {@code UUID} 不是 version 为 1 的 UUID。 + */ + public long timestamp() throws UnsupportedOperationException + { + checkTimeBase(); + return (mostSigBits & 0x0FFFL) << 48// + | ((mostSigBits >> 16) & 0x0FFFFL) << 32// + | mostSigBits >>> 32; + } + + /** + * 与此 UUID 相关联的时钟序列值。 + * + *

+ * 14 位的时钟序列值根据此 UUID 的 clock_seq 字段构造。clock_seq 字段用于保证在基于时间的 UUID 中的时间唯一性。 + *

+ * {@code clockSequence} 值仅在基于时间的 UUID(其 version 类型为 1)中才有意义。 如果此 UUID 不是基于时间的 UUID,则此方法抛出 + * UnsupportedOperationException。 + * + * @return 此 {@code UUID} 的时钟序列 + * + * @throws UnsupportedOperationException 如果此 UUID 的 version 不为 1 + */ + public int clockSequence() throws UnsupportedOperationException + { + checkTimeBase(); + return (int) ((leastSigBits & 0x3FFF000000000000L) >>> 48); + } + + /** + * 与此 UUID 相关的节点值。 + * + *

+ * 48 位的节点值根据此 UUID 的 node 字段构造。此字段旨在用于保存机器的 IEEE 802 地址,该地址用于生成此 UUID 以保证空间唯一性。 + *

+ * 节点值仅在基于时间的 UUID(其 version 类型为 1)中才有意义。
+ * 如果此 UUID 不是基于时间的 UUID,则此方法抛出 UnsupportedOperationException。 + * + * @return 此 {@code UUID} 的节点值 + * + * @throws UnsupportedOperationException 如果此 UUID 的 version 不为 1 + */ + public long node() throws UnsupportedOperationException + { + checkTimeBase(); + return leastSigBits & 0x0000FFFFFFFFFFFFL; + } + + /** + * 返回此{@code UUID} 的字符串表现形式。 + * + *

+ * UUID 的字符串表示形式由此 BNF 描述: + * + *

+     * {@code
+     * UUID                   = ----
+     * time_low               = 4*
+     * time_mid               = 2*
+     * time_high_and_version  = 2*
+     * variant_and_sequence   = 2*
+     * node                   = 6*
+     * hexOctet               = 
+     * hexDigit               = [0-9a-fA-F]
+     * }
+     * 
+ * + * + * + * @return 此{@code UUID} 的字符串表现形式 + * @see #toString(boolean) + */ + @Override + public String toString() + { + return toString(false); + } + + /** + * 返回此{@code UUID} 的字符串表现形式。 + * + *

+ * UUID 的字符串表示形式由此 BNF 描述: + * + *

+     * {@code
+     * UUID                   = ----
+     * time_low               = 4*
+     * time_mid               = 2*
+     * time_high_and_version  = 2*
+     * variant_and_sequence   = 2*
+     * node                   = 6*
+     * hexOctet               = 
+     * hexDigit               = [0-9a-fA-F]
+     * }
+     * 
+ * + * + * + * @param isSimple 是否简单模式,简单模式为不带'-'的UUID字符串 + * @return 此{@code UUID} 的字符串表现形式 + */ + public String toString(boolean isSimple) + { + final StringBuilder builder = new StringBuilder(isSimple ? 32 : 36); + // time_low + builder.append(digits(mostSigBits >> 32, 8)); + if (!isSimple) + { + builder.append('-'); + } + // time_mid + builder.append(digits(mostSigBits >> 16, 4)); + if (!isSimple) + { + builder.append('-'); + } + // time_high_and_version + builder.append(digits(mostSigBits, 4)); + if (!isSimple) + { + builder.append('-'); + } + // variant_and_sequence + builder.append(digits(leastSigBits >> 48, 4)); + if (!isSimple) + { + builder.append('-'); + } + // node + builder.append(digits(leastSigBits, 12)); + + return builder.toString(); + } + + /** + * 返回此 UUID 的哈希码。 + * + * @return UUID 的哈希码值。 + */ + @Override + public int hashCode() + { + long hilo = mostSigBits ^ leastSigBits; + return ((int) (hilo >> 32)) ^ (int) hilo; + } + + /** + * 将此对象与指定对象比较。 + *

+ * 当且仅当参数不为 {@code null}、而是一个 UUID 对象、具有与此 UUID 相同的 varriant、包含相同的值(每一位均相同)时,结果才为 {@code true}。 + * + * @param obj 要与之比较的对象 + * + * @return 如果对象相同,则返回 {@code true};否则返回 {@code false} + */ + @Override + public boolean equals(Object obj) + { + if ((null == obj) || (obj.getClass() != UUID.class)) + { + return false; + } + UUID id = (UUID) obj; + return (mostSigBits == id.mostSigBits && leastSigBits == id.leastSigBits); + } + + // Comparison Operations + + /** + * 将此 UUID 与指定的 UUID 比较。 + * + *

+ * 如果两个 UUID 不同,且第一个 UUID 的最高有效字段大于第二个 UUID 的对应字段,则第一个 UUID 大于第二个 UUID。 + * + * @param val 与此 UUID 比较的 UUID + * + * @return 在此 UUID 小于、等于或大于 val 时,分别返回 -1、0 或 1。 + * + */ + @Override + public int compareTo(UUID val) + { + // The ordering is intentionally set up so that the UUIDs + // can simply be numerically compared as two numbers + return (this.mostSigBits < val.mostSigBits ? -1 : // + (this.mostSigBits > val.mostSigBits ? 1 : // + (this.leastSigBits < val.leastSigBits ? -1 : // + (this.leastSigBits > val.leastSigBits ? 1 : // + 0)))); + } + + // ------------------------------------------------------------------------------------------------------------------- + // Private method start + /** + * 返回指定数字对应的hex值 + * + * @param val 值 + * @param digits 位 + * @return 值 + */ + private static String digits(long val, int digits) + { + long hi = 1L << (digits * 4); + return Long.toHexString(hi | (val & (hi - 1))).substring(1); + } + + /** + * 检查是否为time-based版本UUID + */ + private void checkTimeBase() + { + if (version() != 1) + { + throw new UnsupportedOperationException("Not a time-based UUID"); + } + } + + /** + * 获取{@link SecureRandom},类提供加密的强随机数生成器 (RNG) + * + * @return {@link SecureRandom} + */ + public static SecureRandom getSecureRandom() + { + try + { + return SecureRandom.getInstance("SHA1PRNG"); + } + catch (NoSuchAlgorithmException e) + { + throw new UtilException(e); + } + } + + /** + * 获取随机数生成器对象
+ * ThreadLocalRandom是JDK 7之后提供并发产生随机数,能够解决多个线程发生的竞争争夺。 + * + * @return {@link ThreadLocalRandom} + */ + public static ThreadLocalRandom getRandom() + { + return ThreadLocalRandom.current(); + } +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/xss/Xss.java b/ruoyi-common/src/main/java/com/ruoyi/common/xss/Xss.java new file mode 100644 index 00000000..7bfdf04b --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/xss/Xss.java @@ -0,0 +1,27 @@ +package com.ruoyi.common.xss; + +import javax.validation.Constraint; +import javax.validation.Payload; +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +/** + * 自定义xss校验注解 + * + * @author ruoyi + */ +@Retention(RetentionPolicy.RUNTIME) +@Target(value = { ElementType.METHOD, ElementType.FIELD, ElementType.CONSTRUCTOR, ElementType.PARAMETER }) +@Constraint(validatedBy = { XssValidator.class }) +public @interface Xss +{ + String message() + + default "不允许任何脚本运行"; + + Class[] groups() default {}; + + Class[] payload() default {}; +} diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/xss/XssValidator.java b/ruoyi-common/src/main/java/com/ruoyi/common/xss/XssValidator.java new file mode 100644 index 00000000..ed9ec1f5 --- /dev/null +++ b/ruoyi-common/src/main/java/com/ruoyi/common/xss/XssValidator.java @@ -0,0 +1,34 @@ +package com.ruoyi.common.xss; + +import com.ruoyi.common.utils.StringUtils; +import javax.validation.ConstraintValidator; +import javax.validation.ConstraintValidatorContext; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +/** + * 自定义xss校验注解实现 + * + * @author ruoyi + */ +public class XssValidator implements ConstraintValidator +{ + private static final String HTML_PATTERN = "<(\\S*?)[^>]*>.*?|<.*? />"; + + @Override + public boolean isValid(String value, ConstraintValidatorContext constraintValidatorContext) + { + if (StringUtils.isBlank(value)) + { + return true; + } + return !containsHtml(value); + } + + public static boolean containsHtml(String value) + { + Pattern pattern = Pattern.compile(HTML_PATTERN); + Matcher matcher = pattern.matcher(value); + return matcher.matches(); + } +} \ No newline at end of file diff --git a/ruoyi-common/target/classes/com/ruoyi/common/annotation/Anonymous.class b/ruoyi-common/target/classes/com/ruoyi/common/annotation/Anonymous.class new file mode 100644 index 0000000000000000000000000000000000000000..8ed89cdab608113211c18889af2d8a0eb73c0248 GIT binary patch literal 451 zcmah`Jx{|h5PdFf0|iQfi2?B;=tveerb-P0iP9bx_XVQ?*EZkKAZ(#i Og>CFq5MdWx>;b>pDT5LK literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/annotation/DataScope.class b/ruoyi-common/target/classes/com/ruoyi/common/annotation/DataScope.class new file mode 100644 index 0000000000000000000000000000000000000000..43210a55b0cec5da3c49ce54c86691a4f0a8e798 GIT binary patch literal 573 zcmaixO-sW-5Qg9Fhp|;_tM%(3Rl%YcbMfY>kcxt}(nc?yw&_^5B-xN`3jJ#y`~m(b zaRSm*DR@|RXL#OsW@kRX-ai0bo+B#a1DhdPZtx>Y_2)EsH#(9;oJW!3^m zgETZMpcj>>kx#idH*maEUM`B-u8b#SxRdfs`YD^9t>(Hu1r!>09udQ9y{>!LcLYvW z(*rULbH~iK|3RL#ue>>f*5LWkz3+7ePFG7D?WI{0G-mqT(}5kQdai7K1v<2sezh2k zKs}#&e>5eJ2i(e2OPYLIZNV!j@G0_2izR^)${ej_Q38t->wgoKUkMgh87^Q0n|!y} WXp0SMuMmW7)X~Ty!VY%P1ik?)$(&sP literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/annotation/DataSource.class b/ruoyi-common/target/classes/com/ruoyi/common/annotation/DataSource.class new file mode 100644 index 0000000000000000000000000000000000000000..ae50afc978296408a0978f7e2902cd0aa21fc3b1 GIT binary patch literal 626 zcma)4%TB^T6g`7L6(4|a7skf|VPiM$T#>X!5)e|ViE%a5E0(k~rnDsdnhQU`k22l{ zV@bs5VjlO-IrnkyynnpD0l2`vfgJ<82KEFB6B#5V(5N@so(|0@(No{#Td0*uDhbD? zC1cswN#v0`eW6Q%YEvm4%h=b-q9>UIvB25C*~Lz?@7j()X%_U#4>*50%W^)F6De@u zBr5hpy7R~WAfPPgSm3aoV$49Qp(($MFAkBt5$b4xu_G?|mDnhCD)p~1@B{c! zh+78Ip@_luwa<5t<>Sw<_YVL)90h0xI0$e^@aHP8m~eb?Ih?Aw3Ug&9;mDQROs)yd zk*&&%Bb^J=>g&u!wk(vpTC#+bq%uwy{G?}kob!z_)+wj0nGsG`{M4@yrmCE<6YKN! z`XJ|mjY}7AERY|?gJGX=wjNH{2`l#r+WiN4u({3_0tU(BT^ipG{&bDaE0@~w%hENo zMOapLp+l((Yr>Vc)Oirvb$a*XDQAw*sgG8g>X5nwLPN%ed<%I55580lZ2rsvwq)&y Vx-E?N3}Q5~gWcthv4?%MfNvMdhvEPL literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/annotation/Excel$ColumnType.class b/ruoyi-common/target/classes/com/ruoyi/common/annotation/Excel$ColumnType.class new file mode 100644 index 0000000000000000000000000000000000000000..b55aa1f9aea18d6966b35ee2662367963d2218fd GIT binary patch literal 1368 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 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/annotation/Excels.class b/ruoyi-common/target/classes/com/ruoyi/common/annotation/Excels.class new file mode 100644 index 0000000000000000000000000000000000000000..fed578d4a45b618ce05e6e2e833a163905331c8a GIT binary patch literal 441 zcmaiwO-sW-5Qg8WiP6@N>PH;?0v_3WfMVj9z-$hM{cPY^2!~^VdB11N>3q zvpz{#==w3E|cCD9Otrn$=~aBT+?ZBV}#r zl+z;P&z#L2q5E3avz!OoNL6o{t!}1jt_UY-ZJjRoT~~T+`2N4RBAg`)CNj!SR=Z|u zHen}iFW3Lah6}bXYd%;Ye=v-b2g1dAIAtfS3@d230eLBn&Km)P^zA7dKE^V_M*HHU z-`aQ{PdRsl^ES1zuhxA?Aavwx$@a-3c<@Dau)Vwk?8tj6@vbo52*l{3hrLC_*vG*i DY-)d; literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/annotation/Log.class b/ruoyi-common/target/classes/com/ruoyi/common/annotation/Log.class new file mode 100644 index 0000000000000000000000000000000000000000..c762b72e86f149623bcb1366759924da952391a6 GIT binary patch literal 850 zcma)4O>@&Q6kH{B+?E1uDIWz&`b~Sm7jB%)Bv59WrsJ3%%1JSgj5@ZIT5@Lc*EsYC z^hYr~4-^~7P!6`F)$Utq^{&5ty#nwWJ3cmjZ28z0@SJif39KC)9$(0b3{t5_!NBE8 zkKPEZgjyRXoie&ZUu2Ow0SGJ&3#&9)8&Ah9e#zp*WI_zeV1>l>_AQ^qRWD;>qsXBng@sHO^!?m ztln(fvC)=(>@N;Xktg&{l?l!to6)N>LV@+D&`xD^rmPyKRAp!dHfk};U*QI^%tz$- z;NC>01((~N&FiL=k#=PqOkOw*qp;VF%cxpB9`!o{o3oV>ITrFmvTpwt%Be|JGG(G4 zeLRUj^f;fb*^-V)it<48nc3~XnxJ+=jz?g*>aqUtf)Z}0b_LbgRc8Bw8?M1$`4Zlo zz#KeYwRqLE@EMrL!cV{>EaDDBOSoG>_i&$~WvtxD&aukCgWmwQSmTlnJj5d&ciHYS Y^PhY|gRqXL@w~x=acL0YdIaqbD=3reQ^GqFq;@*C(C10f*NZoXAY@~GaSYY~8 zYh$HVMwjVH#(@EwGlFS^iAwdNOK~k>Ir_E;h^vH2E%u%KdhlLT0_Jk_(ER3avaTX>p`Ay^qQz)?CH0KXt b2DAK@*t^KU+&u(g9t&7ZBEk}uu>yPnWmmr7 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/annotation/RepeatSubmit.class b/ruoyi-common/target/classes/com/ruoyi/common/annotation/RepeatSubmit.class new file mode 100644 index 0000000000000000000000000000000000000000..471494f7be2ef905550224b099502d9a0050345a GIT binary patch literal 635 zcmZ{hJ5L)y5Xb+Ef#Vgxyg@?YA%SVQCLI-$9bqXvWE+}P>tisB?)K2#9+JDH2vBe$ zO^WmsR8WFY#0TJWU@#Tm05L`ayKsut?9BdV=0CG{FE_UUzF@+_sD&{L;{qK@8ww7j zE09PnWCfDTTKh&Cg6ASuDl-w4t-TFS4-w-pQOo0 zMNgTlkSRCj;=X)ST7qi_p5F@V9EM;O8|Mu1n+&BA{G literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/annotation/ReportExcel.class b/ruoyi-common/target/classes/com/ruoyi/common/annotation/ReportExcel.class new file mode 100644 index 0000000000000000000000000000000000000000..9ce16f8f95d0beb35c887cef3f406247e29a70a3 GIT binary patch literal 516 zcmah`O-lnY5S{6VTkBV=RS^-zL)DAx!J8+cTPV~Iw$+QLbe)!zY*Mn>V*i>4e}F$q zoK@&nD0m2&nY{OA=H>J2{R055Vc&y>2fH5ZAyi&vjURq$~ zRy2|QE$}JwcNIzqB`9;Y!dWo^j|-K>!YX5{84}iDo$o`AHWuuTu2TqPXic zg+&v^$IhN(HypQaZy7D!`VEUxdqVqCxfbOr%G=knzH7FHqb(Y#il$n1ot=iLT4X)K zSx>8)Y~ZRP1fzrmfDtuV9y2Ag;@{kJC&b{ ztk6F=FquF4o%v;OMfpI6432l5aeOKC%QBAdp-@ixE4^BwYN$1egMHz6{?P&2#(|KG z)^*0EzZ}x_1ipu3qfDw-Zbs@0x5Zbw!E!EPMAYjpWPj?o1-Yh!3*>a6+in8=2+Pv(#$~Xoy zhc3?-m9E>q$J1ETXyX&5zqfrOv`_}KWET|baD{8o>ea}@J4zWI4I`c!8x!gYZKUv# z$-&oyGslB>tLG6k~C)i?Kez z3W8&^9wI1b_ksbq6?Iui@^<3?Nc3irp9$M>-k6` zLt!G9BZ(|T6B!Bybn*MH7jKiQEQ}PX-X-&NGq4pr7H^ zBz_6CW+jryWqJ=+aPnbmW?oPtOk@p2?hPQqqld@BPp@;=8La$^|4vC3#Oj%^b$B0i z%fAJW2O_cXjFMr{Es9gA2Q<;0heR;v40@$oeGh1gi|++}2=8jQ)Gp}EGfIU)zXj-b z13>vvG3d3ZO4H#=9i06zveI-IQ~VRbiRj!Pk$dyF+==MipOAa&xZIP`xj#eM XCCpBYZsXk%jbXQp^$t2v@M+~ghz`I_ literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/constant/CacheConstants.class b/ruoyi-common/target/classes/com/ruoyi/common/constant/CacheConstants.class new file mode 100644 index 0000000000000000000000000000000000000000..53e523864d8cd1e1f333274d982f30b39a8ce921 GIT binary patch literal 820 zcma))OLN*V5XV;p0s-em`byvFlhB7RJ*Az_xJF4mA%+2$PEHzOR8VSU#un}5WA)UT z9{K_Lp*md|lDfBE*82UmTJ8R=U%!9+1b|oYq6#apSB3{=ct~Kg-8u4n7IcoO&yMLS zfqMIlf8ln-)!6RqM5^&?0_$$9Qq7ef@Mtb7@QA>c>vRM6y~Es2i<+YqcuZiUf7(aI zKlF|k+7kkn(ORAxER?5MF6dF3lNtoJd$dcPfb~DTJMos8`&h5%1eCSCORb4LcR#im z?e&=JFB6|FFAk{hTxib;tiF*->bC?|8v6qRMK>M^GqO~|pHGJ(3HUHVrQVK19u0UR z4KKx_p2(CyvmM4$JDJDl(#F*^R`^+XVY@t>hzn;@w5i&U=Se6IrK!5}@AgY$s|7b; zmq6oxYog76|3+tcCPIzwMe$fFrsK~-r3W~A$Y(m7a2DcxMyRFdDT4YW$4G02Ml#eU z)vR?Q#HEmknc$kG^WjvQj9S9AV3AyqHTybZB1u@Nvd!$cQ1}oIU=IrTk^t6V7m82< z^c$%FW#meZ&3-k&P29-bCKY@iuZj$2-VZa(or}T8^({C&c)S@g{a;XADqk heh2a`Yw;GwRg?i6y^Ybz)+M+DcaaF(!&rd(e*hrxv3>vm literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/constant/Constants.class b/ruoyi-common/target/classes/com/ruoyi/common/constant/Constants.class new file mode 100644 index 0000000000000000000000000000000000000000..efe5fed1472646d22b8924da7a4d9bb53a67af76 GIT binary patch literal 1850 zcmai!+jiST6o&USK1>|vWN09SLrWJy^Yf#ai$HJQRz6>?9p}!;NNtTYF~DKF#0%{P7nNJ)ocRWEW_LoC3L2AE#;B z$UJs~zYJ>m3})7PB7;q)!1rci&ECofO?j52k%@{m!-2q2Ek6)*V0_78T7f zIUPZA?Ya)bie@@kCrxYJs5Y-`14cQhth&qTm=U-2b;EIWo6`xS!pL}ZRtGmOYc;pL zp*8irnqh0MVO1HuB!yvwso(Zv#F2HaExpRA%cwXMn>Aan7<-&TMpL76>yBVEr>=|? z(R7UFf=E41qHy5vr_qTYs^yLsbdw@&kddeJBK<%=7U_NZfYHp0ZOUjIcTk46DUwbV z~;M?6pIcNTCeA|ju6Dcl~-ky1Erq^okTfAh(%f#_qc^0|Cgr!L}cwB`z_olitW4_HGweuyjT~*h~rMhTVmmjm|HZKq?WxLSGtMkX$BA5 z>oiZZR3elEbd6pHUzd0e{EEb{g6AcE4YVM!IPVRK-vqxU@eT0J7x-<7h5apw-vQs2 z_+9XO5-)<6B)$Vi4+ts$zPX{YKqlnk z$bhz~3gTp4ZGJ)-Xo!}xzth+=%4GheY-6mHbDmLtCN~1^G%}@er;#lcoJOuR;WYB4 hqSN4|Nmxw@kBo%JDH_AKMtS-fE%XbtHSjh1>R&x&ZnXda literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/constant/ErrorCodeConstants.class b/ruoyi-common/target/classes/com/ruoyi/common/constant/ErrorCodeConstants.class new file mode 100644 index 0000000000000000000000000000000000000000..f8754e5efc762955eb901da39f5a5be776326403 GIT binary patch literal 9828 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 C4v6&t literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/constant/GenConstants.class b/ruoyi-common/target/classes/com/ruoyi/common/constant/GenConstants.class new file mode 100644 index 0000000000000000000000000000000000000000..82a4183eb01728ebdfe493508146db305f006458 GIT binary patch literal 2753 zcma)-_m>k@6vyxEW|O-iz!I<^Hb4;&SwIvOL{o-kWRmPA6C|R>Niv%qNG9uSf*|(Z zJNDkYAS#N2z2WcvIUc`vCOHhK=lILK`?>Aici(-}QM|(RD${EszRvW9tZy>C#q>7QJ52A2;ytGKnLc3pkm)0) zk45)Mg!V>gpF(Vm>g80vm{usRZ!>q8!wu7^4;R|5?bO#QG(FXFrrM^{-fT8H7SpHQ z5F@XuO#2mzNPFsSoAzk+8frx!C>qcp0P3r1d{*<1guS_u;-zenA4c#g{>l7pGKmqj zTpZPN33T*ZbktLC(Db9)?|K``L34Sg@1^8!bG7UU(+>*8JcX7u)DaafKg#uqi=GpQ zpL!ALTrsKRK7U5DJ$q6cNvkO>lhB!d=>Ucbt^wh8Y%a5KN{3noiJS-V~6Sb5<5r5uCyvfBeW6wwQXlt+uGSin7>4uR?Y4-ec(pRsf!pR z&2z7rZL4iJQBkP7hz1Q7!b5FN#^?(1j7VbJL0+C}nhhKZ;-H_~ll#4I*zuYOf1p*j zP-SUoNRkv4KJ8=oPp}JL^#N8oVbR0w6gD~a`M@PK>E04R^dUqevF>kP> zyLTN;+V!-5-8fd-8?}%;Tim-krjW9d=Hu^VHU9n<(p%f-=cs=;C0ACJH1ULy!1vm{>1H1u@CxyM|%3JYPyaaE>pn5AN&|5KT-ijgfR?LmJ zVgS5#ERZ*ke1!lT;9`I`gRczm7H}!RSAn+%csls%0AB;XHo#@@bpbZPl>k@4wE$b- z@c`Gs69Klt+XB2D+z4;$+4o(%ByU^l>1xLZYj`|#a?ZVh8qunsIeK*~M{eRKo9 zVNc1l6LQRVpj~t$h-r5(SlLY>Xh=y#4pCpJKi*#`h2o(?X&^pOD23zU!hVW$O)REC zF-65R4W?MvG+j(H#57Y(Lom%6n&xMWi*2^p=7{ZtLMakIu~6dpNrlo-IDRs|k@#GE zIerShgYkLzM&qX<$7$X4r%Q@6B*mGMVm?d@hQfaOf7pesd+}2Lq%}$(tyLD#dU)RizdpM8Zwcm5NdN!< literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/constant/HttpStatus.class b/ruoyi-common/target/classes/com/ruoyi/common/constant/HttpStatus.class new file mode 100644 index 0000000000000000000000000000000000000000..3e48329ecfa405ed68b54496fedc1c0a9fedbca9 GIT binary patch literal 897 zcma*lOK;Oa5C`ztJeoF5Nh#0rE|k(1S{LXIgp}GF)l%#=egu_^F}Fy3Re1 z5i=Pr=vg|>qcrbEgGpRQ{&%G4E^qnVko>`DwuJ)c*hh@&wg`1$``q?r#!o~n(d9YX2b>uPYk93&e z=lM%7kMha*0j}j`lTklzC$eMn->2=v=vCC9b=qLG`aivJe_5gpqV%~YdWUg8rxhyU zZ$WgDmZ?Hj!rh<})!)oA8?ATbKlsv&FiJiD-q1s+(VkeJ)0H8|w_hghAcG R+9=+uv_;zxLx~*Z&R>4YkFWp$ literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/constant/ScheduleConstants$Status.class b/ruoyi-common/target/classes/com/ruoyi/common/constant/ScheduleConstants$Status.class new file mode 100644 index 0000000000000000000000000000000000000000..94a75c4744de6001c15727125f19ddd1dd633268 GIT binary patch literal 1449 zcmbVL?@!ZE6g_Y2)~(cm6(&Cwaq2dN0uE7`FhU@ZOfXcI5I>bsM@iQvTSwx5rJ!Oo z8b13+8Sm{H9l{sZ4}JH(ciz3{+*tgee8n&}9tDm{rh=IsQH2lR^T0 z3Ct^GK4pI)f&K)ZvH4ua3xW97_Rg!7G8u+%`mLsby6L)BW7RR6O^e(@x$4#RM$3C| z>txkEmln6_o35`L)kEu`(H~fa} z)(9kLDl6sPHA7(jZFgznL3pCMkj-@&LR$3>h+uEocCD>eecx)lG4~zXk%QRV?*ulo zm)YmKBdnFZs_9hBhRsRBOp5ySBe1ki8<`<`onw0+N4dll-J%*IepfgTf44 zaXUxQM9uOm?QZmEbC<&rqF8l83ehZIS?@rtxvhG!8_`SDX~pnbjjFY7b79k$=4P6Q zL&a^L;6W9OD9U&#Fn9gYQ!$1m75A{LVgP9YF~i=3iay*TZ&pP=8>xTe$iiq)tB&Uq zxr?RQ-aoRcJ_(q-1{E6T&K3kuX|bb{;rw^DfH_c>bmzSgwwbGxX6e25(tDRmL#zX! zm1NLPgjE84!(2S_V@P5M!w?uDZ-Q3-loNSIRO|;GM^DgW>`5m`8K)2|pk1`KfN^R@hKe@wmu8;C SFx|({XjKR;VFou#Lw^8}Ay6a$ literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/constant/ScheduleConstants.class b/ruoyi-common/target/classes/com/ruoyi/common/constant/ScheduleConstants.class new file mode 100644 index 0000000000000000000000000000000000000000..8ea01aa9e6313e208843d479ae531ad24e08858f GIT binary patch literal 719 zcmb7?TTj$b5Xb+sTw0e!U{x+&0fC@KtLU4?*mS$tSX#2D?rU?loIpvpO?ol$WBFv_ zgCD>TWt^=vHt~svGpE0~&z%1H{o^NqH#q2G4i7av(y*;zhp;G0Sylz1=cK91twa`u zVq6JK@hP-iQ;5(rPi#Wpy_8?1k;wGi5M>^x=WhwiR+bi}Ov|xMDy2hfB2P#DsXYoD zTj*F~<%uJP&d3f2_R#$720C^L8;#OA_I$QoAxAtWY&OO^%=88!r?hN)pkt4lX{-Z3 z^!(tX;~neR=W*V}X-r0Lsao9RL6T literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/constant/SqlConstants.class b/ruoyi-common/target/classes/com/ruoyi/common/constant/SqlConstants.class new file mode 100644 index 0000000000000000000000000000000000000000..048aa0672ad0494365bc70e5786498c2910e0bfb GIT binary patch literal 507 zcma)(%TB^T6o&tSA_YNi-cUDg47#uzcN!Bc#gLS`C}B&cWr9Q6X)LXYkLAk5g%99E z8BY;omoDba`OeAw=Q5vP?;ilJa8gAP2W1?V(Pb$1U6Y}0FZnY!06Yd2RY1@&{gDHVTV7HBRX0jwUkqiY*~%v`vuSJCL4)D!e|ikfzZE@rOW`LB6;CXrn#hm} zl!cmS9!^o91p^wobV{JPi3*@~L(g%AxSHc_;#!XD^pa&gl5LWLv1Xuj@doy~mTZyT WqA*CfO}4i7OXy&ih(3vI0sG&uHEV$Y literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/constant/SysErrorCodeConstants.class b/ruoyi-common/target/classes/com/ruoyi/common/constant/SysErrorCodeConstants.class new file mode 100644 index 0000000000000000000000000000000000000000..e0e05585c5e10ced8d80b70c91d840e70fbe8302 GIT binary patch literal 9872 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 KFlsU?;K^@1+K}J? literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/constant/UserConstants.class b/ruoyi-common/target/classes/com/ruoyi/common/constant/UserConstants.class new file mode 100644 index 0000000000000000000000000000000000000000..768177d25aaa65428a5aa8f604db6905b9f27bba GIT binary patch literal 1145 zcma))TT|0O6vxlf7D8JD0Y$vyjf+Y^@s5|!rkXL`)LaUE%7mK1V4KXOtqvc{Cue-{ z1NfmF{~MqzJo#`o``iCHXHU-AKYxGyCZY{`SEL+mjM0-Zddf&~T8{5Iy3J_X+z-A5 zYA1;IRHvUt@!l&&6O|c?4zw*(KfzKJuKr;(#Egc%rK(Xm5pylwQRoFOz{aoHS{>;WZVk6?t06Mm3cZ9= zMs=Ng3ZdpwhFhNNin2CQv8lBz&sFF(#D-?;-1VD=zOB$3Mn!{j6m1&(qe5@t>~Z6h zhwF;ma{XhN5r4)}hF>?hZ|eNL`+*TNa_H#@BT3eFj$|RtLTYG^vu)W`DN;y?L@IXOB*y1(kgC@U)8nc3(5|AB4AOR3i?XDr)K&2!%57ZJ(C;ITeL$JLDS iYFB;|`;m=c73(OD34>6vma~0<)@U8X=n>Wtdi)Qcn!FVN literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/convert/DictConvert.class b/ruoyi-common/target/classes/com/ruoyi/common/convert/DictConvert.class new file mode 100644 index 0000000000000000000000000000000000000000..59cc5ed060c7062c8525db52203c768f7bfd07a8 GIT binary patch literal 3703 zcmdT{-)|IE6#j0z-PsP)0^1f`P%KD)QW&sBO1B``Qm~X3X(=M2&Tgm6lo-P;Dhl$Af9_?Iy>84M_*0b`El<(-#zC$-#K^s z+v9sb0qDVX83`Pfa4d!wWmI8U#zq{MF@h5^Mln`PuP0-8X$?+cT*d@W%b3I&8873k zgjZrXC!?NQy^8bP;I$ZDm+%H(q^&_CQWDZKrl3ohmXKknHZ8iwM=U#|=FRDz8Fj?U zWOO^p5bZazhI5eCw0BH0LtZEmmc|)bASu0Do?3`{p>Y$N!2Ha=bES@jqa+d83YjfJfd`>^E&oeZ%cZ|%? zfNE;lj5=Uy`8+MH9XB#r%_-Ql_lEKn{hbGCQGgvH(BlMJ`oD*;nqyemWRR<|)Qp~XCC~8Bgf(hAEapUU6uDI&P9W8zFgq9P_BwQdze5r_Bx?1_VJz(#pBrMc$ z_QEVHkyJ_Qr9wSp8;{md#p@piG=mLzX?OISQ`g9R+(qePw%@iN{FrOq1}gf@@fy zd=>NJrqtWC^bo3~h7wy>XberkGW68P&*!dxBQSU@G1OBfxYm&B)@TS1HpCVpmD(l+9rmUR^AVPE=G=Rq9Tm*CEUDk)lV z9#*GtNJ~4GO%+~C8@L~r3@w$wS6e2rl|FY9VAv%7sQu7KvP93+@96q8y{UAkW`LT` zJ7C|^Llv6o8Wn?01OO=R0&K%G^hV{27FxxPdr&Rr

    P8?U~D$j>wyL`Vb>WDlWL ztZ5_5c68A7zaed@fW**=9Uj7;h#*GD$*v{T^hI_oBHCAzh$L#3AoWEP(FbVw4zYMG zAG!-U0(QG=5o;Eq^hpUR5nV#vZ8DFD@V5!5?LK9; zk^MB%MHgAuKMr_l_?<8%S~I-k0qX9f{%m~h9jv?U5wV2zD@4RyA~tg( z8hj#(7W^p@<+g-YEtqIUEj@Fud&q}g`WI+FG0{g-ND>?Uq8~1lW3CrI8^bt= z=LH+XJ{!Xx8;xY?vayLqI0rR^*(P|2iip>us{S$d6hn4(a|kj&TnU!zrk^NabdPsZ zF5``hX!-?>cd>zihY(caU_d`{;SPxM0lyHo3LKy|zHtdpe2*9}1|~Rgi=mO=To(p$ IM1T+d4dX5aF#rGn literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/convert/JsonConvert.class b/ruoyi-common/target/classes/com/ruoyi/common/convert/JsonConvert.class new file mode 100644 index 0000000000000000000000000000000000000000..514c120fc9f60c1a692d42cd97d025cfb41129d0 GIT binary patch literal 1684 zcmdT^%}*0i5dS@BDXdkH3ZnP{icreOqTyr-mx?CXfC&Y|@wGhbBir3(`B8B1V1feD7T8EH#3%g~x0U15mjoHfDFnU}U$@G3>&7I@JjNq62U zachOUQr|;iOg)q}hT(k4shFA zZ{74UC}JgSbuSbiX?e$TirgZL2eRzBTuH~yHsgQb%J+BQl}h{$hJuK1aLe<*=`fWc zZTd}$Fv^GF=D){6SuZzIW|%q(vRRM}U;d{g&qNAhNx9N4XLSiKM8tould!~uOBEP& zgK(pOp``{imK@J5iQ7_lU00-kCiTxIaRNz(k;7kJw!ypAWr(3CS|MT9^_UuHaP28+)%=%EM&vSz6s9{mcS3LsqMarueFKfr z810m9T2Io=q@B?o&uoLeqeBZ$(c0z-)V+ISkR{+qFjmeuao zZNxs4G>VXlAQ&OS8DFP|EPHX5*5e`dH9=zN!?}=4fw;s7Ik}7Y>evq2wvl-I2b)Vi mMkde;+35Jrqu-Asi3_+$cv^M|mkBZ8IzY1EG>9uc=G9*eQsD;x literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/controller/BaseController$1.class b/ruoyi-common/target/classes/com/ruoyi/common/core/controller/BaseController$1.class new file mode 100644 index 0000000000000000000000000000000000000000..44e164981ea6c5b3c627566c99370e1796ad8a2c GIT binary patch literal 969 zcmb7C%We}f6g^Ht(isLCQp)=gV1Oh=xDnk2Rcb?#kU~L932RSc(kT-=@^~oyEEYV( zf)C)M5Z9AXT6L9?#=h4+_uR+v?>|3(0oXvx!#UjWQN~Rlw{Y7h_f7?OE2uM+?S7nw z>kRGQP*1qY^ij;onrKCvk+djlbdpHJw?!(u$8Lv$pT{b;n+zL`g`=AThEi9LB*Stq zR&qC+MAGbuD4~;DPY*>h5GHp1{IKLoWe8p=B~3RGX)4KmwrJ+CO_h78v|H(({A?Lk z8@&VZN$^CdG4ERwt8pi<&gR#(v$qU>UuR}0U&gN5mD7WH>e#5O9W_jJN^GyC-PfZE z?ghAy2LUQr3g98E;9-DAcpRXCCMmF>0AZVv}8p;G?V=&3aJXlE2#ZTi`CstvX0 z)LcYYYR;*;9jlQv6yE4*Gv?{k&E^Qt1vD!jmGri4Nq@cAB%vySO}zbXy|b1|rEQ;*@OvA_l0_>n_u~ zXVt13G+71uJ@{ZaPu3>+?!<4ceFbZMN8wA331}@726%!Cv|gE^a1mAV2%obtLR>;^ z&b2k?@>**@Q2d6{e=0u@=TyM@2|DkX&cg~W6HO7TL|&kDS7|Td8m?nGPsgZ{{QC=5 CO!#B~ literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/controller/BaseController.class b/ruoyi-common/target/classes/com/ruoyi/common/core/controller/BaseController.class new file mode 100644 index 0000000000000000000000000000000000000000..aeb3b0a194bb9dbd050df05ed4d5aa9697b20ef6 GIT binary patch literal 7222 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 I@yVb61E>&6Hvj+t literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/domain/AjaxResult.class b/ruoyi-common/target/classes/com/ruoyi/common/core/domain/AjaxResult.class new file mode 100644 index 0000000000000000000000000000000000000000..5850fac937e591c6e0046964a05d2a7d92332f21 GIT binary patch literal 2825 zcmbtVYfl_i7=F&~va`D^0^61%HI&v`mRVqFt#U2x3KVF$XcuUympUxtGIVzq=Th5` zelr>qlO`rjnivy}U(`rqfM_&+GREIw8-IYG)%Tn^><)CAN`!DObKduPpZCh|fB*0^ zfOA++pcR)COva!lFokJ;zZ^q%0#`5-LvIYT34}1mmw5%R^8GcwT#X^m_XWNza37^|4JqnXZf7A^*Oh^qFW2vN{POtW-0~Zq7fsy%a0Lf&9E7|W(t*Z)^aO%jVztY6_Ykg z&#qR=nqg*h>)M^Gy6u)6BGHnbew>@ITo9JoG*KzrMTT^cq)&H#WnC{g!$NmebF{r) z%U23l-q8xTX0;8$nUY2^R`3>u!*&Y=-6sB-^g%p0pAH~#fC%i-m&rMZ{dvEAssH~N zMc&galc7$xtct}j8Wi}0Fbr?F4rTF{ewRRR^D1SC8+Lv--c)Kg7zX@)$aDqyQLE%? zPUCr_XljmY5%8d=<2ptuJEhrcl<5&4%o77NO2wGSS6r*0Uov=wH`eO3m)lI@B%WvJ z{ExM*;ARqUgVaWlQ?Qam0jmtjNB2JdBc=YpSfZ|`Tdtk3~{6(3?!lB9g6nZzCJ4JB#Kx|!r-QxYLWAzz^M^X zv-s{0*dAFg%DZA>giM=`Ueu{H6p_t&vPXT~p zMQ)fVoEJ1y54K99vr3~=(jcQmIZFik2dM+ID3C{_JUJ?|+!86q_)%&D+RAm9$*L~Vi z;ftp|AhmO-BYmLZM}(G}@td02M#Fvz1p(6H2}%%B5oD-0D4>C4p4pF64kSavB9KgI zIgreqr;r&6NJ8Tdn)WB*eIiaXDJY7~`-y)PNSuaQAaV9rAn{M1N_<$tU64ts=SgYH zuz-|&?gOhA)k;Z3y+{u4Y8spp>D$5lyMqobx*U8z}fzztqq xAIVT5wqImTIsX4hnm8x9C)CU@XzAX<(HeOVticDCkYFLaLU19BVO;DJ{{qHV8)N_g literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/domain/BaseEntity.class b/ruoyi-common/target/classes/com/ruoyi/common/core/domain/BaseEntity.class new file mode 100644 index 0000000000000000000000000000000000000000..a59632b4b2f47128771a9e38830bc95e5ee7532d GIT binary patch literal 2691 zcma)-`A-{16vyBAwy{kx1e)|n(ks|0+4M-00wo1(3WTe+p-rW#8rvJQWDj?bDo+2Y zR!S79mHKO>{!vxGv$ObE))h#eotbxi=Djz=KmY#uHxYe7-{&Y#kFzvIdt%rZ!+{tM z#c(8s@5E3~QzJ)NdYql+8WnSRNK)k^LVGqi0v{eY+9z{a??HGx~C=) zM*J~GQ;y2*_Nl+ksI+&+f8~nK&92gLZO!biFuGRvOjk416V1_Dy1H(fmdjmucNpE< zYg>lW;+COxEKf1|El8Z6?l}q{=qXJOyDd%CJ8+j&pKFdA_v2HhZL8c>H~NecPc>a< zq+H6Q?`#VOaG2UTXw>RWMmg!M305RR9>sGtU8y1j6qkJ0Pr2i&_H#p5&UpKoW0|9* zyI5~ow!vNKr+eIms>x`!kH4k;{iROl=JxioVJtg1e8%f_0{DVt`1$y#y56h-MT#U! z%2o|-KLcrx+uVTV^a$KO?;%#Up><8}dN$DSk4slC7G-4Tuyv-k-IX#XBM)-9Q#K>b!W!VWPJ+_ z7u|+=knRUzol93)L_LpG8aObG9Q@GiZVr}xF|LoxeE%s!^ujAs1j#jnOTcTPdf=(c!4F41tf(!<9| zc0Hpxe@nBJhX3qoFGNSvEj*5S!mCFax=ux!#{3E%oEXjE2TrW7Vm*cLJbnv5(pwOK zJrEQE!F841rgtF1i2~6)RyQ!e=^%_S9_Zk}fzFuu2QWYy{?}aSAWMNx2n(5mc+uxC zU*r~tL;f*-0u=>chHtX+n%Ez{%@W3xzd1v9F&CME0#xte%ji@34CA=*ON#vtVT^{x z_aluz$Nwv7yaw$UY*LjwuP8PQo%45>5pc!#AQC8mWOy-?1;M))2@(&3JcIMG7oC1u`FvGNyXpS-5rE6Ru+ffo1gQ4wFFUnorumB<@G zTnKK%rmxFN!ELX?Z;iHsI7j#nE?7n|OSkfZy#VuX6PU#d!suIA$h)7yYh1>Td{eeD K+kq@byZ-^)3g{^S literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/domain/DictTreeEntity.class b/ruoyi-common/target/classes/com/ruoyi/common/core/domain/DictTreeEntity.class new file mode 100644 index 0000000000000000000000000000000000000000..5b1179a73f84efa780e2c1b890dc261f64c0306e GIT binary patch literal 9263 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 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/domain/R.class b/ruoyi-common/target/classes/com/ruoyi/common/core/domain/R.class new file mode 100644 index 0000000000000000000000000000000000000000..55b990954c80ef905d24bbd45b4916f80b5da4b4 GIT binary patch literal 3784 zcmbtWT~8cU7=F)w4Z9uM0xbs6T9N9qK(|`Z+ERq_QMN80vJ1A@YKLW@1IrGXoo!No zfiY=f;+-+k3yn8k*rcV6Mz0%xhi&`;TpOP==M0?P&Soh>I5X#+_dM_Oejfh%_qRWZ zXppuvT1-$YEot;g6J1YGm~J%DGJh`devSpVRGZT#1HAcY?Hl9I6UsbAIn-LbGEXzVmf(ar3mfTtX+T?27JL@3M1|v ztIX(hwqS3ior-Q44H zP>1H_;rvAXhN%CIem{xhxacqatP_ofHP+4EG?V@Rv5ZSO}B2&#d z*C=ex7~683)vJsKHbgLk4aKc;u2LwNWsJ#@;1nuhmCg!@Wv6rA)nR?XM=n?=a(2Zj zm{S%X$gzd9JRgjD0*0@+TGG?5yA2=RN?E5f^cJI20lLbdW~4v==KkM*eE0mzub+MO zy-pViW~J=ej}QKN_>)efG^SIAuIV&ElRBNEPMuz-(>lFD-8y|tGaB8|$)XLNHmRu7 z2z`iKaKCnhlc*?9RO2P1mVHf`FECgUne*;S@2Xlx72M@F=iC}Dm*jF#^g$?$>zq*= z#i)}`&`G?v<4O(DE7XDC&b@sXcCRY?*PsK(JYD$aJ>eeo-}K&lJY0knlD8ol!;|3| zO+BX21H6POi8ba4b?U{tE)*z5ec1I^Nd~Zs(^*QxEQjmHdH$1NE%dZC@`R$lV#gHB z(~`(J@7z1`+!W4n8i|y8PUX_0a7pltL3^JIm-Ij)oP5{AJ}+a=S`ps{@J&#>jEcfs!CKvY59!D+ zdvrwiO)L3|*6lvV$C`wjhsV6qW2vr3bY#Dq6P`2+Cufu>#v~^1c0MxHQs>`zZ!-jA zJb>=kOP$(9u!f-aPE#>f5sy6811E(I_oH)VH^K QSbHJa3Ja72nGogw13-M%ZU6uP literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/domain/TenantBaseDO.class b/ruoyi-common/target/classes/com/ruoyi/common/core/domain/TenantBaseDO.class new file mode 100644 index 0000000000000000000000000000000000000000..833845657726537377b285094c63aadecea7723c GIT binary patch literal 1593 zcma)6+in|G6kTULlbP6)#Bm#!6a&3dU*Z_rK-)MekT#Vf*(qwOO6^l8&M2PZ%mjOk z)bHpUA3$I7&`RJ)R6_6q65j;0YkQ{gI7A4^I%l8leOYVmZU6bt&%Xdz!g?ArxH^dp zu9;X$BZaFLK9l?NG@aMQx?$m_+{-5F;#^5%6}RO0!o-&fR_OYUA8za@WE)SN@0?o4 z@po&Dz~8MaOssgm7v54x6-y5l^xMIXt1#2>e0Q_Ix9xTxINKe1a*d$nbRIfgPu{Up z58GZ(q0(ptd$n#qc;?k;?FBxsu3Os)_8iZzJs9?1b9(Nbd+eCmb;HOdkaES+zd>bs z$&gdU;}L??3WKe%>-oD1r^jkUUZLiG+jly|H96l^G}qai z_C(k1^*f=$WTIqxE9iGy?p;q*pG|7FC`fJOaLPs&vkGTM&?C}oeXp}4!1Kwx;yVJX zHr8<4#yrl7b%!WQHa@_I3Z?%&K^rBUBTXA0;hc>!DhgNrGg~IV?uTCZj2k$D9H~te zoc(>*=PFc_V~VGkT#hl0obOotyRjz5IUcW{=Wk<%zbXYy2SLpF#dYp7Mi>-cBlYMIdd?WI<^bjZR&J8^0O{k{lEex}#5mfx=4;k8ShHCv z{{TZjsQiY^_b^fim0yu*F0i^ie3mPRn2Ibm++5(XtXKa)W?ugtvoBy*=k-Axp&zhd zDxt1D6lf`GbVRX<^SD45bGU(vR9wR%ckU8>DgHIi<1#HBAG1~-rfL0!X%kl_|3=k(<0lWJ4GTfaG{10FlPV5ded21%Q$*kIOt~p3dwG+4>3>WqMWg-7IA5 zZIZRJl@~Cp157_zEu2iU9A(EnQY5s1Dr;L@rA1mArbxIpBq`J`LVLrnbW&4_tf15z P@*ll5B>+De68iKl?2Reu literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/domain/TreeEntity.class b/ruoyi-common/target/classes/com/ruoyi/common/core/domain/TreeEntity.class new file mode 100644 index 0000000000000000000000000000000000000000..bef2aee75695dab6c76e46190e04041d82a4a57f GIT binary patch literal 1837 zcma)-`%lwQ6vw~a6GoBuRD9tB*oIP6eDH7xLnCG)2}t6PWxE+QTT5;`ll@nkNJupC z5AcsNp3`<+SG(wv-Fw@6&i8!oIp=P_et-W7U==$#8> zTUor!Vq3wuEu^70kA<`i%lxpnt3Y|HV6+r7)d+y(JV( zRzK;Vb*-tJ4eij8hS}IqFyvY6)k!?GU$wklOV)+lZ?_aog?9H$M>K>a_hH?v3EQ!x zO;WW}qgkhTfmGWunp)Mc9kx6)8m8{FC6V}A!S-gUOct3SQ}s zHB3PSCqf?SCr$SpR*iC?Qzq!yf0=Sl4Pq!(YgS8>ZL4Ewv}#!v&x5+x<)Lf64Q3CLMnNM9+|ij)GQzl%0f8`;s-CiT5fS5%B+ zLd7JeR7_(=MII9h7XPy!{$;tN+kWTZHr>b^-LDL0F#;9TN%WM!CHfMAyCw>ECJ=Wf z7I!92vcKxC*<0a5(RhR2l#!$rFkZTVQu=|!#|tFSk^1Ud-=vX$BtTv%^0`GTj!pKM zBPP1|*$mnU*SbI>bpt0vt}^IG*C0qqX}@~*-Y0`b~7Lzj4#w7n=#Y4wcu5_tXxQaVI@ literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/domain/TreeSelect.class b/ruoyi-common/target/classes/com/ruoyi/common/core/domain/TreeSelect.class new file mode 100644 index 0000000000000000000000000000000000000000..91db6a142e6186532c1374096984f26d969a798f GIT binary patch literal 3503 zcmbVOYgg1(6y0}#2@FvvD%RTiMCGAGTPw7KR7I)ifYw2(ec8**1p=9bCK)Q)_xtr@ z+O=w1{m|9VUHwg6-8TvIKwvSQwI(^a_ndw9J$K)e{Qd8fKLMP@PZ{jO)eI(Ztqonc zuC1K5Cbc!y4ih)J=y|&xGk7zDJGiT@w+y_kz23>-UA(6+_w;3|3j((d%o->dC<=52 z%D1I6qx``3+}jhE1;jOh9b=vwgwhRXq*GQc#MeI|kg|19Zc(mC%aQK9mGj*BtU!w+ zXO$z+RWp?jeLI?LEiBkh(N}IIv>e)wm9vA8p>274-j!k5R|45MyfQtKD|jW#FMF%D zMXThwy!py1dL?PQ*0irwUOB1|W=Dy%?^f9j?UI_Y1AEp{7hTs2Wyo~}0)r9KoD4$c z-!D1VqAV;0gp(B~>l&XY+=5dsD!rl=y>tU}0!A__U{1KM^2Zz*1d5>N8UpR9g11fG zn7Vgm^5*nif%J&&+To}`Q*YmlK=YVaWF_Otd_%U#HmP2;8a6_jbxfsZaL0WqSZjAD3F^qufho` zv#YnSMu2PD4C2XMz3b=n+FBW!i^aRUF1jjT#wQ+oY9SH_4bo1Kv5LqmEGjytp3bj< zR|XpscRN{BQ>?9}0v1)TyIvxz%WE zd|<#g5g;^BHnD;aP29(-z|N@SROp*HfP)4en0Tl&euNQXeB6mo1Ws(TV~|d@i`|%A zWD^nC9Szx@m5r&^+%1ZkTTMTPD@!!O50Pi#FD33 zHYG~}`|D&Bk6KhBC$kxRhR+3#*NQY=S+@3ABtsd#9=;H`(_lN;x(8I1picXeHtKHT_lgXctY=g3VO_Q^ee;j**Mdo zr(Z(?2kJN`AQeyI7=<`2YHOQgZdB>*%RslYSO(f!lYzFi4D^nD*~pGX-La8(!^uWe zZs02uU*kJ{{LkpiS%D)_Wuh+w+gjJ0Q1^?e^7reLh{|@m>6m^Q%3^Bk?+f1ZET4wA73#a*{0nYHM z0Un|$@*UCu{R6)t^$1P+-u#H5yr&<-5O{)C;4#`#_yd{8XiwoUn2)fdnz9@EWAi9e z_AuQsF84Uj;Y5_aJ5G_L2Zr!s!kp6EjL6P}NJm1XGa>TKwutnT$N-6)B#~1jGPntm zmpP|PIE3MZh!s8SlKzUOpSh+K0f01zLEamc^A)_B1dq~NmpF3{#2Yfi8V%PF zNmnY8j)b6zWHFLp)e>o0Ph@06BIhx>5s{YlL@tuZrA>%vC2Ask`*SH;Beg_!tS2(Q z0g*!tO5q|wE@3PIxygH_GLdeYqPoVFxlTg4O$g~DA#jBM>;@;|TufB66a literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/domain/entity/SysDept.class b/ruoyi-common/target/classes/com/ruoyi/common/core/domain/entity/SysDept.class new file mode 100644 index 0000000000000000000000000000000000000000..4602bf58b52212e07db967d69d25b4f5d40a19e7 GIT binary patch literal 5385 zcmb7Hdw3Mp6+e??vy;sPVjuynQePDklm)5nr=ih2b#V#QO;Vu`oMgw%!0yaCJ4=c7 zp;EQ2O|=#aZM7A(hzbZQfi4KP*6P=<@Au;BQo8| z<58LJ;_(fc5?56y`Ya~wQj>OZPkX_fj!i#%MOTWyY+%yE8I@9tr-Xg`&}Lz*RyGu=Y^nX>)Ch`!r0Z2(lZ92 zSO9msV}tj3g8*R_%?Uk@VB|~#gQdmOCX6uRaF}jS#8VT6K6s?Fi|yT@8@k;kQSGYs z5qP)POhfPDq;6>N(x_$$#Ozpe3^}AClMp|5vCbaXy}&Q@R9KMFYzDx#wybLR#lg+8 zhC*3j$*SOY#@PrJTf7WtgnW6X;9GY&C6-p4;3b8EJwB;c|1qx6{6UP>3GzyTWp{j+ zn$^=H@8Mrmkp=JCh4`Rp_n=yK!mrVsRw$?$_;UH&ot?S)S}QzPu8|uq4jWz=EM~JnzWl=evokN3<_?sm zOHWmr&gYg-YV&)4{vdWgr%KYTNY09RUgr37fwD^);gZ=$_RhR@dS>s5**Bj5s7uzZ z^SMNZ9DSGd?zED*cjiOBiQR*A6|8NHv~-Ro)4?H zcZK2O9bSvGa&>c*=sYjt+CVkYe&;WCH zGP#RMg2_fEcQe^cw@I|tv@&ruua1pt?oC_}jdXo{w1~@Hv*JT8#vW%LrFRrMOXnEY zy9#}tzM#-A=$8s@XHKP2g)}C4dO)FF%##J{)_NusdXMKZXFi0M ztNQuYML zta;OLawxGqnHcQfo*LSk?1yiYm!-a-<63bPveG=4~N%sVk}!ngsy&uG*|x8oBMUx*?1#fbD5BPu|QC;~B}4#bF3 z5F@HVj3@{(q9)uU8ZjpE+q8`aaRBN>2%y$5yGQUTfu%z5O6)+Tf{s9>i)z6UHFumO zspB-&Y&}V}C#deY*#0?Ce!~;BQ5~xGUT}&dPg`jlW)fGHt1FI?bwxVfrpPfcHN*<) z#o~Z(bHv;HMf#GvxeL;KP<_Yh6BG#^e>X*7Dj(nGNVytVyW?|-3)|J@2T~sdvLzJA zec?drgFuD>WTXlZjxc)wmrDqcXb{M@P#|NJb~jgeBpL*=9YF4@0)$hHf#8Becsa~| zkH|qF_lE%C6kY*W-w!pJ0*#XnO*;}uw4Djc@=}fr$XrV`OUaW08zLu>8E%vuG-@Y| z%2qLo>xGTNg&ks4L(r%Psu;z!U^zw=gas&6!1k6TvZ;vgPI35hvLOg|7r=hG3Ro^` z28+hx>heb*76kiH2v`ZqLsKCZ1oAL|JW>S+-zp3Q%`L=hjX@xfhXCPJXoU)YWG6TS z9u)ptCj1xjQKE;!4QLD+@FWcQP89?AI%EUTm_rOuf(ATO#Q@IbuR;xAf33ojuk+Hw zzFeM{3gogYC&AO9QBi_M?SoPKs~E+1I~#?E257KAn}S9?7h;q|d%}S<1%Vs@kQb`} z;pW0X@Yo3fvM31TU?`Bs!htLb0(l)k4p#xf-Hm}f9tvb}5XiAmAWwt?SsVm{218R- zfH>_{qHof-kTSv0)CGZ*LV!4}_uH`MNu;%-WD7>Gy@~ExE1ILzUC#l~yHx;jJC|q@ zMd)U$wjC3-r%@ihm1_5mP%SOp@#JUDoG>4LJ){)^`V zKLsXF(`8#b-lUe3w4`Ne#}u_rQCrJ0Fw2>_9LyEWd<@K$%v=R#1v4K9b2T&9fVq~L z>%d&k%qPIyz|2Z8tC(pAga2(Ut3hvM<|Z&}n2Cc~3ucPel^xPU%>egXJmG#v%joxX zHT{9w>5q8W{Rt1IKjW{GztAT7D@^$to)v#b`}+sF&OdSE{R?IHZ=}Y5e9oNoo-@xt IYYlz>|E3b9w*UYD literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/domain/entity/SysDictData.class b/ruoyi-common/target/classes/com/ruoyi/common/core/domain/entity/SysDictData.class new file mode 100644 index 0000000000000000000000000000000000000000..0cc9231c346bf5771e10d68e4ec1ebd78dfb0136 GIT binary patch literal 4869 zcmb7I>w6UC75~lVvXjlA7(v7qD{2eG$b{O*UNBU$3A(xo$|fnX7dqJ;!ocp#vNM}z zm9{~OR4n2J6kCh5AU6pRiy;jNy?yH6!6xvz-~05OnRm0B+1Unp9^Uu7`<`>&^E=mh z^6&pn-vIC({40WXyx9a52YGpmm*4Snh?lo{In2vDVZ1BjJtp6e;0TT~^+6aPHsKgP zir^mnUdGcA`tc8Z_*lj!KKxO}^SmFI@gnb^$mrqygpAF+|4GJ6y#HCo%e__LQ7fgBXndzOqN!=k99H9&IlNxN@-2DOF*5pqVH-nf zeWPhwj^>bJTf$@Uq?J)~d8=TkL}e^f)#mQi?$M;4CJptbmLc6ras0#L{;ApPAIzTE zORv&py04Jc!}yDYNYqN_GbW#jGiI84E}GVCTc=mnF8s>lWpR2i+}q#Ny(Jcv@T4m` zq*)mwW#!dOVTeq(v+2BDX=tBDz+x#j)o~4IyDPC+ip^!Ijs1O_g&z}E&Y8nH^Y(ki z>BAD5*u!ycNKZ>>sc><^$r-Mbg>Pm~9GD%yA^6=e&8fO` z>`>cweMa(W!*)uvVRz}i_Pa$ZWJvcctMp3)3y<=wlWN~V$ZCyuk@5i%8&eG*GhG{q(B-E{5 zGeF+(Qt95fVd}m4%#faQ{maO$B&DTB$w~eeKh`@t40>Q)b&Qi%PFGV_Ml(!RrxbJw zYNB9wkte$}Md@_dU*4LU zo;^3sRmDYEQ(eM%m6NV6W0-Yr1rw=_GMSRlTB-KV(VNBT)1LC%llyPqJg~04{npfZ z;+efb;axGmv^lHhmA4|iC(tjgbe{mv4pfAbM8XPWUc$~tCam+W7P7ANx=>pRlb8GA zky$tQ10V6O7J8%9OAjxbd3gyj39G6)x|_;0Hq?HaE2SAiB`cA_mk|r&ih?gNsi2?b+pXNN zs%5q04qc1{+Z`uQsY7`qozin^pRl>pJBM*q!4&?=CQd8ZhMz0AhU*HFOzGIJz~epx z(Q>;K{DalL!AxxZ{L%LR8&a_A$}Ox@v-NJbjfL?~32jyC{gbm|rcp%8bGa1pMtyf) zOH&}`)#c`)oqEz)&pEaBu0xS<3U1o;^vC;R+v2g_?rn*_!FV@?Gyc_-P-E(@Y!kwq zud0RHN~4y|>L&HHt(6Kz->GQiCB%3ChR8fILZTvl=nIL<-#u|^(7-CS`@CbMRhl~7 zcezLOi+u7#ypt!wmAb{vB?&7R3^7*H$6Gsf?g$>jI(!=vo}yETzDPq{taSb!ombH5 z`}F?@HrC@A9nWGTI_WN|qYK@1K_x9%TWA-o&y`q1TPVly0ai(fSH$QfZfHZ)m*= zX~I?gV_F;CAD7@KL@OQxeu|&bN#a&P>wMuW2wnCGcLWM+7!tx8$j%Ty)%wI22$i3X zxW9Iy0Z~`O10I)no=x*2Bypz-Fr?0J$a8^)q=F5p^BWQ;Lwaf$;>BO$Zi%`okJS4O zNdy?edG;_N93~_?Y#7){ns&vIMkWT6f3lDzo<|W{f;lcC1NbEDMlQ&^^?p!W3DjT> zP+rnX-2DSUHTXeoTNsp6ZEU_8loJFhPjE(pK{fb6r3jQ>1C&>A5>K-LP>p_2yA}pD zS`DgD4Qez9RDnRf8Vst@4=PKbcGm#qH4BNSa{#EYAJj+?cFss9^|qRW@V{dxar9 z3AistS|Y7i&^!s{0-ZuAH^+k&)C@_s0XD})>7g!ctQY4$@oLQsy@ zfZ`1XiNDbT3|Zti+%wqvIqtoR z#Vz->PNHQJ_qQw|Vkr|35V4Gj2Z>nD#6v`Ujft-lv4V+*iTDN+j}WnviARZejETpI z_$CwIB4QO2tBF{{L@N4Nq^Egg=P2@kKi=c;0%2poTd4H zj%NOOngpL=02gQmj$;QdQfXeIQof9Rn82I(yuz7>L{vjK;3m#*@k350azF?1ZFqM1 F{{UL1n&1Ec literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/domain/entity/SysDictType.class b/ruoyi-common/target/classes/com/ruoyi/common/core/domain/entity/SysDictType.class new file mode 100644 index 0000000000000000000000000000000000000000..02a151958a3bb6d5b1f4645ca4cb8f9063766243 GIT binary patch literal 3380 zcmb7G`%@EF6h6xG0KOSo3`>nb<=jxvZZl66DT)J}h^K*D~37PV-q zVM?mmB}LO(iCGDTdM;mG`Qu zW=ba98Uno%2x~D>k824n%~qFl4+^|MXWX=FIV+tG%<$*WqswsLnF&m9&@= z(c)q((TmWH{%G7t4>TkJm_NdynpOa{&t&G0aJX!>w=;AAh?vE83u{>#k3mycEG8WS zQeO_w{5;~QJAiF!X6}xszPpNaAW3E%$ctvKj830DhrH;r6hZa|WZjf?-(bH$OB;4g z-}xbxOx8B;N}c<5=Eg73->ImIxm%$8s_HJp)uKf}=aPV;%I)!3udG{01u6?@VU&l2 zA^~#eV}W^6!JCcq*$!*EEJn1Lq^P2dVlWe8Fk!R;twTU@pKS7xOQ@`>It{#2M8(FO z!m_IDdycilabfdUp5bJc!Ai?=P#ln=O2jHclpRW1M`WfUwrgfHO7}FEKuk6asSoH^ z+1%Q!PbG&Rp8snqIX3g_7?&?s`-1L*${EgXeoRsGtww<;4YCNB4e2Sfoj8tQiR9IL4i=uieVlS7ZbAD!ow_Ir}{CrB?EJ=?fb2(%aKvz+dLs6Q?5=9FtC`Hf$67WSej|bXavE(1xd`i#Fgt4eKz<3TX|! zMC%DgIOWmH_@Nw-qj2%#i<_FtNhXMMCK}Q+ldltmals^aY zP%enw_?T%Y(6QW*3$h{m0eSxdknQpncxLA)sKRW>?KvO|)RQ}q!t6kT2qd&XAa=V5 zJmsB0?ok0Gq7&m(bhzdoxgLBJfl=n=7q3`g~bq-uuuwNDGOx~ma$L{VL1yc zAgp9z6@=9+tby#_EH^oZu3p_Db?;-{U2(Jf}8*V literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/domain/entity/SysMenu.class b/ruoyi-common/target/classes/com/ruoyi/common/core/domain/entity/SysMenu.class new file mode 100644 index 0000000000000000000000000000000000000000..1a8b719ec50591a43c39cd75bbd99d7fd8ba8aaa GIT binary patch literal 5948 zcmb7Hdw3LA6+e^LPLcuAK!8?TX^Vw~mZh}#3Z*6?=+Y%g^D3!ToFvm^V0UJjokg}< zTgywO2r5{rP(cw&1xad2Xd+;%)fTmCt5sXAS_uih->*OX{GB_K&15>um++Bu&YgSj zxxaJnIp>}|_rHNNM07K~Rz$bZp%8tkh;ryqAuXlDOpY*ll*v&hk1_c&lVePdGx-XW z6HI!T^fGyz$yb>?5u%fY)JOeAG@VY#R8xeHPx4?urjHExoAkz<-zrbK#4AGB5^ka!ilA5Kfu~yAW>ZZ{WSuT;ZQlcp}rjfK&!){e$DJ>r^ zntwhGVMH6^@!lSEWRkH>M0*k87=ZQg4rGRfb{OstvVCiEfB? znDKDCYR2_WGZl`f+cA?SzynpqcaRApmIXBG{a^yWYg-@th!WOqUoNc5yRHHG>cUxRJ&WELcmX$26jo5 zzdof|X^BF5a)l*MfmEY*bmKg6G^4@v2FDL@8eD0o;D!8zX2rqt^bXU&;*M@T)(I#X zccyGT7LMvk8z&bw>0Ji)SgDo??RQSb8nQ1)?WO35~v$Sxh8CP{944Ar|4mYKf ztKf;sWze>0LY@>CutL{jd_9xJh$`Hr`!Ub%)-ypcQPl82mu9m8d{L$I%RTqwMqFWXY{B&T{&~h(?5c30 zVa7NmnRu8dc5V!BP-A+hh*)?N@o&MGb~0RN+SQ2jb#O^2t|gOd7c8GWvUA_ijy*%W zb_^fsIe%)$$fnchPYnzo9gyhSORH@u%wT&qW&(~wC8-s2g;arK$p zC>!UvvFy0+(x$?5b*Wek*3Q1L_ssC2fr|$Rh6Z+ycNd>KcG65Q*3p9)YzTlFkc;fC zXC<0D(tl!jckj@?o}vBsd;PR{k=IcnVH9Vug8emPNj!IWaP#@WK4=*_vw7V9B}+Vx zR+8ep#&ParBgJON z{}EBP*r$%fM$}|#SSCd?WeOKxH5xgYkCU%m93f-Jj3#a|EZM8EF(Mt;bgnr5Zf2L3ubu08Ty`s=MiYc^_hxgD26}q2?o0)u!Vz}{$ z^QtMdoxkm1vP(=j@7WxMKE>a5Gug`%a)+gmVQ3KaSUy^0m71S+gp z@k_D789OU}#VZOdEv{PY6#6x*J;;j|;E4-Q@Y#@P-X&L8t4Xa^JdYABoUkLC>@;6@ z$*PuUb7W04QdhgCsktp$%kS#slprN)y(O-`eXZ7ES58(72W5epNN5I9dbL-fTi-p8 zp?gI6#LsZ!23SRgyO)Z#kn>%P_`*DovWWKOxL9{fi+jt?7H8ya;WE`IZkVn{H+l<- zQVCr_%ji~;=$#nlP!;}=DHzvcJOz9O{@$h0yD5Sx1V>&8j?@wy*(Epn$}0)Eh|Dv{$x`MYp$W38I)NCF<_lU8|VW;MiuytvSCyz zi%}db>`fFoSC`kQkl(0ufKd{qgS{K_1GyJKKAZ&zM=JwCp$+h^><98d01);rhv0|s z2Up%JT#596|1zH>0JsnHPE2}2LH2{)3b5O483=lj03cKRK#m0hc`z8r6hDw20O`#Fgx?Vcf_^9k3$}Vk#M;KeKkaG$~kpyLMhUumh|fR_RckWhC5-_}w;ke31El`KHG zQDPvUrO)Bb_XElE19>$7h|^u|gOdI9h^wRlJO>^Y9gJw8az+2+G*GVr&~LH;;*L$C z&ttCw&OJ>V-37@xX&xFeUeh~B)7vT@ry2cJRywnyk7o7J?9$62T*1N{ zAiR-Vc}{B*RXIcgt;ud8Nzie%!4qWg>ne^Kf81R==Chz0AV2uVF-&@ zxDmo)7H)!YGYd-~yoH6gLU2z^vJ=GI2S%TL2Jl}2b5U8FhmTPmmD(PH{N zRni~OaQqQJG5>_0kAFr7_7`+Of2H;GH?$ys$0Pd>dXWA}d+1+q{J-f4{fCaze?3>! On`th;fKzhli~k2`iz~JO literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/domain/entity/SysRole.class b/ruoyi-common/target/classes/com/ruoyi/common/core/domain/entity/SysRole.class new file mode 100644 index 0000000000000000000000000000000000000000..fcf76eb5d507d600d467d0d2972efa4d9abbff7d GIT binary patch literal 6457 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 UVsQ0hB3-^{G<_1|rxuO=FL<5EDgXcg literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/domain/model/LoginBody.class b/ruoyi-common/target/classes/com/ruoyi/common/core/domain/model/LoginBody.class new file mode 100644 index 0000000000000000000000000000000000000000..541f90ed0076ea6c5c87c37592b68648bf04f364 GIT binary patch literal 1142 zcmb7?NpBM|6vzLbOfp%9wv@K)&;l)KKsAS6Sd;@roFYJK>D`%(f~(9J%>wGT0ttk~ zfe*lkLcHfh2@KAG%iH{x-?RPm*Y_U)4spMY0NX7zu*2dGi@PlD1=tO+C(tbOI7?L$ z3#<>$)CVOeDjmz?BGc*kDV@JpdH&I4BLP1$BTDnjQj^zsrc+(?1!~>>Q;Hth!_|RK z<5y)ejI$FpoY<}=Qj=4aY4&G9zj&u}frCM05}B3eTuZVNlak6}IWmdTsZ6NlL=MbY zr_askoM<}ZxOh7U-stYTkUROGNt@k&a~POgy_xUEH487Gv*|to#b*|ewwrToYPF9| znMLtS&5N$gK@a#ILtMa0h$dPg+AKOOLbL=P{?FC|Yl~-jJv@t}0z34A0X?t}j|Y(D z@eHeJjMX%WbPIwxNUoA6Nej~T-ZzNPHf@b$!v>qUNOm|)z&b9GC$ND{l7AO}f%mCW zyzMOB!sS_Up9no3)qC(2-ePpyMt4!ip6#$T&6V literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/domain/model/LoginUser.class b/ruoyi-common/target/classes/com/ruoyi/common/core/domain/model/LoginUser.class new file mode 100644 index 0000000000000000000000000000000000000000..13791ca5bb72ec1b077b5492a12a1587f4d03fbe GIT binary patch literal 5436 zcmcJSOK=oL7{|ZaWOuX4=0SKSs34FKHi1R(#VZgJ9>Fzll?T0Wk%JdKc+rCwJ$TWB7cI;G*E74BZDwL9%2cIa)BW#XfBp5> zz4!k4@+%^GlHN(s3bHt{IdPh)lgeDW$jK#6E^|_8rYm%nlh?WW1}ATFa*dPgoZR5# zW}I%t>2{p%DAZwVmaZ12G|Sda(y6BZCn6tVusTVbcnzhR%-O}7j zOfRYVyoJd{kBpg99WW^rpSH|Pm{y^PX)DxH(yXFxGcRiG?i^)DFQl`Y1B^sgpEXpc zY+-`U^X7o&3V;PvpKV*LC{;65X|T{D{k_saFbs~ter&~*)j@k&*aQv+A5nB zJ&jV)G>}KV9k*vw;5 z?HS$B#>&NM&6-rF3ux(NV-==UY$fVmW219k$9i@w4OE0=0Xv92@rV-kBF4%-nPS_N z3p(QMHTSNTo3MaMck5Tx@z*xL3BA1@tNqem2NsX**A3knRH&hks|Wj*=9diW+bQ-2 zR#;PSoJy%`pzOm)`+dNp9~9GF^RG*6oNU5}S&(mi!Z~GaUhoJ1Cqdx4B`SX(?|cMt zw!mE`RNKB}TJZR;lVzNZqBf=5I-GdOFib}T0Qm5LFqvA=r`2gSJ)_#roNXGr(rT?! z`b2ho?5M64@;FoQNe*iI6|F&pE#jHjyA742&Pv@L%H>Q$ z9b&ROa4tN@My5i|}!fxMO!dSis?l*>fjrH$)4h7HhnfLuKc@Y3Yv6SJ%2a(+r!1 zf|g^K!jTvc`Lw@3V_LImyTrHijD_=o(>R~DwOkph75DBsX5og?^3}O2`^2T0HOp2` zJF4>q)KTp$$8CD^@5$|O&f8` z3iU5Jd;KlsZW|9DgF&&F6?m~B?ob zyNDd75jm_Qau`VDFf4R?MR_Z9VdGjog*vP+zQOFsx8U0jN+aEg8s`e~XzDXkJ{H}c zKpGQ`r)d|;Nw)*-#)A|uL3)awM#_T+As*r0ss060K0zr$Ji1x5o~0M3k;;KE2k0Qo%`2Hl&evc&3-A_2Qv(YWW2pwQPn{4U zx{S0!3H*3yqpL!4qdTm&U+uz$*iG8`6hrn9%C28<3p<_benVmfhdqG1$m8hQTLn-U zAd=e7C|!+UNp(dRD4yzyEl|@U*CuMhs5ilN4P4j3bpuD}CcNer0>W(&+!36|UG*Ag zujf~oks}ah=MuX=o1s^*^NM@L5#ir9y&Q|+kB3lpy_9=x6?WlXDeRcLvSF+Y@yc7- zE^lv|aBQNVQ$qHWcYRya2KgNze}sVWmChh91v+bo4Dx3n$eCbg?T|tK0?6MXApG`VkeBJK7h(L)8k0b9 zHQfJFaO37Vh`d6tdLk3ZYcB@hO(b3nN|d5XECk4XXKY zjG72V%|RqjS`ccSZRcK^E@~@K+d@#?_m>})FOn+i3`(G4riZ!*`(OemCK%?Mri7q@8J2LJ#7 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/domain/model/RegisterBody.class b/ruoyi-common/target/classes/com/ruoyi/common/core/domain/model/RegisterBody.class new file mode 100644 index 0000000000000000000000000000000000000000..e4e717e16b5e91ebd5b23f3100f2e88c61f550f1 GIT binary patch literal 359 zcmb7A%Sr=55Uk$pL)X=)f8b3;90YHM927h)9#nFl><+PIX9gy_LO#n=z=I#)M~OYb zgU2`x)m_zebu(YzAD;lOaadx6{Q?IC#)N#rj{7NLbb2-;q<6kGgmKNzJPdYj!n2-t zq8!)0(cMf3j{H|l`xjmkuIk2H6$ZazC6)C~QZTCZR_`CBrp~a7QW#-RFhTVLdyiTv literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/page/PageDomain.class b/ruoyi-common/target/classes/com/ruoyi/common/core/page/PageDomain.class new file mode 100644 index 0000000000000000000000000000000000000000..b4530eb6d56ce5596a560f16305748c05f4e2fb4 GIT binary patch literal 2295 zcma)7>vGd(5IvtSvaQI8QHW{UatV~+*bvhTv?UNA7uvMON#i6fDWz3xVMJxgm1Snq zKYfloLI0*55;AnA5739|bXSsX*%dSV+3(`rvuAhDC;#*B>%Ref0z<|$?xeAp!8^Ds z;~wtwZi&sZj1{b=@gRjSWMpuM&tLLxEsd{u{hHk>Y##D&)il;+Y~Y(THq)r_)mDZe zw^P_j;ah=JTWjjoZcAXY@?3kNl}*iRme(v-Z|cs1K$?HA8NccRS=(vo&eHy}ZFXCh zfEo&@xsG8q$v%Ld9f8EM-5}qI zieWLKJ>A*zqt%LC*UTNwG58!@CfsL6N8n1OZnw%#*WNeEv}@TGEskDh*5wT{D|SmW zIGfzm-F3eJfzyS#DDv5wtJQygsI`M69K#@9f2M|F1|5<+A*XyHgxmAs>c#>nzm;P_ zjDN}vsP%fABZq)(?`kRx9rT-#(-ghtH$6>J!8S{`o1qa5vmXp0OCe#up#pktoJOwM zU8k-;FnFL7gVxToY7|_=B^spGsVg{-b18hM;1PBeT)>QivwZWtg2(tl00L(QO7ja@ z>KZ0#OTiQTsNg4P0^)*#JrotZi}w_~&!*1d8Oots4N_YQ4Fx)$61RrlyCfa)K|vEw z6+FX?z~#4+h(K<@YJKmyUUvmf4X%P-(2IGYnO%LI(&h`hVHx$hC2;Y78N04wcFKOS z+q@?nqqEv-yZZu>liejkuDxwhTWT!nWvxSvpN$ehNOJvmC~56BwMpRGTSWu|;0N#8 zJ_xnd=u~YtAQ|dT?{`W6rK@o%Lk0M~TbQE)Bm(G~Ce37Pb9)st^i`UsZ(I^Fo+|BP zJXmifiFhlC$6HA^v??SPQWn_{=(I}yq#sNcUqKYr*b(Bz#0e5dNdDmkT_!t@I9VA9 z29QPoIbVDU=>(|*NNW1DoDHEQ`<&;&LEMH# z@{Z9e6-!5$7*sBMw@` zXCfc<2NCXwG)m-FB$1El&oZJT(kPL;MC9HZh_LLK$c;!M@lhfxkwk9xiToCfrcgS; zsRLY6)zUF050Ft~mESR;p7xKDI#uD}#hI!vq;p8oevNDeIaEo!8ziYsQvVhWa3@5+ z&>v`lKea@jC6+`yKbD}wr}&I4>!eDn|6E=od4l`_Qh(yiAxSRDb%h1IO9FmOPv=Rf qzTR^Q1UybypW{}5J*2_V>+>~OCpwJ~WvNeD0zYi#wLlbNxcwhBhO4mv literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/core/page/TableDataInfo.class b/ruoyi-common/target/classes/com/ruoyi/common/core/page/TableDataInfo.class new file mode 100644 index 0000000000000000000000000000000000000000..977ed1ef1339dddf236ec1c0df1a73eac860ce99 GIT binary patch literal 2229 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 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/enums/BusinessStatus.class b/ruoyi-common/target/classes/com/ruoyi/common/enums/BusinessStatus.class new file mode 100644 index 0000000000000000000000000000000000000000..d587b37d4b1ee20e9ea2a16edebf53d90c5985ae GIT binary patch literal 1045 zcma)4%Wl&^6g?9=@uP0iCJ<;Tlt)YL6k;ls1xi%4M5$ECY8R%FDprm$!Ifi6epKSG zAeATrfx70S5O^>wk$UqnsF9=^iBF3Mxp72*)TC*XNezrNn)o?nk6j; z@$lftqSTGvfpyxYRQ>JTxupvy$h?H1p&RpOiJj=3$563I$Z0d(_20q_rM4gX=^jJZ>J^MKrTgGutMsnT6CyFI7iJ-P+ej2QMb;L;wH) literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/enums/BusinessType.class b/ruoyi-common/target/classes/com/ruoyi/common/enums/BusinessType.class new file mode 100644 index 0000000000000000000000000000000000000000..3d39fac222df4262ee932f93a46f498425836274 GIT binary patch literal 1462 zcma)5ZBr6a6n+*Kc0m-xw^V$811-_g%penCeJKRTMJJ}dtVFgBEDbMC{(*i^;Y>C) zHGb|(|ETGl3zInYf!&$&+~?lso_+2)cYpr=`UAiU4qT8Cb>j&(6oeE!bt8aH4lxxY zZd#17VoSvsV_d~JV?xCQqo!h#aa+X{>}rP+ zW|Fy5FdF3d2EiX|(Yw`#>oc&k=1t$d}DxpLWR6LzvZ9J*PQZzGt97`T)x`f%6?K(I&3blMi zLNFL=caIz|z0OPUCg~`tTJa=b&KM^Jno&fu^cxB5!Q`27ZiEYlbsE;IWz#y1gxa*^ zPnL2<;n*mfoVA%+wu~Zu!T)bsDpAz5%<6`O{{MV+FLW$HuD&(tj4s_gwTx=5OquLK zUQnaCLUZ$4OKOdp6>ZDz26fS`mulr)e%s_4bhoy6k&EC#kS}1!gMCnwrrNj~9xNj5 z!2_&#FoIDJ#yE^~nBXwUVT!{vhZzpD9Of`8q3fpfd=R=HxkAYzZ@s}#%OTSzXZc)} z{I9l_EnYAxl}L;1Z})CzjchK{wGYt0*0DLrCILe7DUhrRB)0;|ut4%GkZcPi=K{&R zK=KdxhPYysS%wSqSb&7P#3l*(an~VzhV2^mkHjS0qv#ZCD(M#yZt7QH6SHo=JpTob zYsjp(-O?}7jHjW(GVY6YsyeCoi_5Un3n=pofh#yaiRISM-E3E|%GU8vXx5c z0|YcVNZ_htSigkC)BxK{5 zz-<{LGVa9DkGI%3$&VaArr0(e!$2H&F%!dVoQ!Yt$-NkcV#xE?ci8xD90lA@P~>7e z7V)qhOL$+#2QrogqBC>3sltpvs!*~j8M|if=o$J}vCNES)+$bB+S?Re)vY-*WXTH1 zck{(MDwMS^E+FGMjIt#j~&DAQan!Tv58f1~Zfb;7Dx0B6z zo`s}D!78c7l4|Q5*$*F4ZCl+TDkxndOma7LO1`bExpjgCdXuTZy$$kY@|pAaJ9ot6 zkiqDLZtCtGfozM!^DSZqOn#8m66I3V%O>^OCNh%2NC&C3hQ%gIhJSxJ$#JZJAIoOq z)Lp%kZFa4>y5W#64MQurI%$)#$uX5n)PLreBor@NHM^wE>fFcP6DUJkisR(tq~**U!HFfvvA77{DN#zS;Zd zyYCeYVNk&+F3b2#fr?dut|`lM9oJT?_ceFJT5~$1D0N}2GXkv(>k7))kfAHs#Fh+0 zK?P$19gUZY85OwEl>h(oBoHYXmPrg9mC^j_CNWDa-U{}7+E5)Q8|dVLPy!}@CiDey zzFVG!(A~(|mykeTGT&O|U7j;nk@4C~F9!`3TT`jp@#IS2T0dbwjEAT@X6SPAr}1gR z;`gZqD9>}T%2T6RY8p}7V(O*nb&adls8!saKKmOL z9m!81R#B^-x4@*IA;NnhI3%Z~Fuzuyq*q{Xc$kVD}6m?h2%AnFOH}TV|9)Sxd`%PXawe>^b zc|<1%csKyG(paG1Rvq9ym(hJ=0_N(mG6{x5j~C_ zK?5t{npY3Kr^xtNUKtxn{|xy!cqjml9D-lQ^%mgRaqz_!;1JJVs*QPifG{GoCel~W zKSo>n+9Sy60qGH9Kan+9f2SAe68Uov>NOm6qi$nPB>Jz6akmh|t9Z>%aho{j&%$qz zmJ{)CcpvQvs?LI-JDMk{arx^-azkRHm}o;vw9OXTH}Op@C97b)Ew=(r}I?T1H6ZY{{UG6 BqU!(v literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/enums/DataSourceType.class b/ruoyi-common/target/classes/com/ruoyi/common/enums/DataSourceType.class new file mode 100644 index 0000000000000000000000000000000000000000..ef976f48ddbb6ef1713c69331e9deba34cc81c27 GIT binary patch literal 1045 zcma)4?`zXg6g{s=n>0(?Ez|1ebpF^}(rM#hevp>2GAjsi15*pbPic%bBTXbt2K%qH z11E~I&;C*3eQ9wm`k@Kr<-L2}J?EZ#fBpXU6TmK>6d__m!6qI`Sdy@%K!wgfhK!Oz zL4zY|GAexCmhgz7@Tz5Z+OHY5EH@gMaT=ZZCjAYf(DcG|Ff^Y#iDO4;?0TKEk6wcz zZ(FTyn?c&>wyZa8n?Zj&_vON=6Qtgdp{5)2-x3d^cOFB<@f^lX5fT<(@tVP>^F?LWocIAIziWoeV%ucTnwE7acKW0Cf}NV=qGy&)&KCm zG`b98^1-J%i?-hnog|GZlc4i}+;oG<08KRGU2PZF~RR5mz^q_So@0nB}9(T&ERj+i8@AM1*=({+#nQZ+<`zZAl0_l zE>QTKg{SGRu1mPhrQ=S9S%-|fSf2CDcVnQy+i9RL32mzMy>QEY}E z4}wTxQAJ$EQV>^>=3|*HpIyKTj39kwH1xBS*Q*)^*}SGgDAJ9Y{y!v6xXd%##}GZjVhtin*z5Z zZx;|f7oBj*&6$NYvt%=79oX-fMZyoi4NCkIwqtwu1Uk>ixe#5Wp?P_m>Qf@Ds3oiH z7WN1&P?WqWWXBVwCyKEDUX~%Ir)3aTNa0XLuX)9jd<*96z zh}gGoX(uj6wNxn-m^RH};55Rr$)287It8T`O)r<{=69SyKKh6jD3L4F>kH0fCDr4`bde=V=fOY3q0oV>)hQTgMIx z0#Dx}T z%kubnGz8akGzK51^+)8^ohDV!Mqs-Y0=sht*ty`R#^^WfD*ciMr@Fbp07>exl+*k)YwSh z0RqpYxiQ^IxQasAK2P8m%t>kBbQve09k?es zeasx-yELU@3e%);xu1}nVS}hE0<%ccH>X^rls=NJv5`YG{{)qDDL>OqW3B<2tFz`= zFMwiXRsqi11fM=dZy$3=)S$%;8oQrNf4zV|8xAT5xUdghDlLAgv`VF|Nh%?!TntDh zER{>DRNAG|p-H7vD({49eqB=Q4%HlcLfWsCOQ}LPvi>SsjcijDAtM{8B5Y*UD%y># zRz;_g4bfE>3A*bs_90%NHya8apl=`QAud0Ny-X-~`+9#sZ>;au63t@tIj)Qz;avp} zzxRnJ|NhNGuJz~&^_;u0O&jtfLU@XHyhbPfAo(Y{j_356eD2fm71^;C_}<`1g(Ad1 R=%d!wHewzN1m?r1{{iS1+j{^2 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/enums/LimitType.class b/ruoyi-common/target/classes/com/ruoyi/common/enums/LimitType.class new file mode 100644 index 0000000000000000000000000000000000000000..de21e9f50f5bd60954641fde6abf66fabea249bf GIT binary patch literal 1008 zcma)4?@!ZE6g_Y4*0q!khE72g@L%aj z#Aq~p_Kz~&*DWxj342Z6>wEXybI!f*{r>a)7l1lm$q=!oU>(mS%u0B!Kt+X5S{~C1 z1vQSS=27J9hJ+Uk(%aVF&au&ESTXF-*P|pncXj&n!$5a}#P92d>$`FL{FBpQ$m}0c z&RS>3IBuB?tM4XB%AHwW;`ABTE80W};%@lfVJI4I;2b1=*NNIz*P|0DWjg%8@TOv% zT4$E-SwT-X0eHUT+*5=(_x2WHW877hK?76)9!u3tmcIR{>;wA+80I9mMc!}JX zG(0YMlP=*gw~i+%W(j#LV`*gMfC}NG_56fxaQWq(pg7|??`5eJwu dFTl9ku((2MqCicWBa1Sb8>&2|mhggB{{lcR+dlvR literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/enums/OperatorType.class b/ruoyi-common/target/classes/com/ruoyi/common/enums/OperatorType.class new file mode 100644 index 0000000000000000000000000000000000000000..f6a702fa0da6b5842f9ca3a6eca367e42aa2e4b4 GIT binary patch literal 1084 zcma)4ZEw<06n-ugTDtO9aSG12y#kx7D8?AbGF(uX)R7315I+^H*rl{ffhGG_$`WTZ zntk?18PB~VjzvGvCimQP&vT#ioO92w-`{=$D5D&MgbfvW6l7#%6jda!$(Jn!NtJ?I z9PvOwn%iv!I=6<5hYXQM>s9>?!E)+0)OG52buPI(^bl&(9|t}A zg~NN0o_(cl-UJQT_$*{Jl<T;HR_ znS5cUf=2hy?u|&?li8l^xz=!4nSuR%nyajp-4cyuf@WIU&2j|*SrQ>E=>SqWUKU3n zhU-`a!wvG5$?{K~fPF#e9N|yoG2EmzDq;%Zv&c>HWrW13s!PSMh@3;>_|O&nEjrOI zP*}ok5vSG=if1e#Og|tOH=`OF!xW}dAzQ8O)!HN*fX>2Nlf3@$BI=UP_ zgK~EFk!`h$3xWjqc(nm8FYy7p;XMzzpk!6KWRYr0H&(IO7k{ z@lx-+bG-7#aTFCCcXYh=f4J12WF4=*Kx3=h=QirC~xv4i|Yl$-}Pyvg=D{no}U9{96v)mT{#}vyPnOIsT-;g%*@E%@oa+%#uK->8yH|AJj;N zu@*$Fy_8(pjiM*hOmjAd<1WaxCD|l(fNEBQTG_tra2bdRaUc$j zQkcmsO`W1&LqGj;Qd2{$CO|V?nk+hN8O|gp(7TrlM|&BaIEWu1Fig7drL!V$LaKLP-R&EuFh|IuEvR zPH@Z7O@}Gs9FlYk(l~o;{|2<|@wZU3L(*Gxy&`K|jjoVsnDAVYMjBhSY1mjH3CBaf z6Hs8_WOToIssMjy{)M!x?@H?3dd~)WUqcQJyCq%Me+UhIsSgBUDa~bR4SnWo2+pKY WN0l-Hx~Yo1YD7APpUIZMFW&$x&wnET literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/enums/UserStatus.class b/ruoyi-common/target/classes/com/ruoyi/common/enums/UserStatus.class new file mode 100644 index 0000000000000000000000000000000000000000..40fa95069853a80c33282a3256e7391a35d798c6 GIT binary patch literal 1496 zcma)5OHUI~7(KV0wnG^n9bSS}9#&|JP(Tm`L|#QJRTcvzL^nen>X3Gt^g-g%oj<^h zi3<~5xG~WpLNprI{tq<%3FG%os}yJ=v$%Kex!>cQbMO8B`|~#dqlni)!k8D=Fz!L8 z2iLu5#SI=NcxRG_DH&}tZpxVU(&`pNGcwv`%*vSK)x3-j84EI^yjt|&wt!=KN#IgE zkx56gh0GH(LVxLu6)~(rIu}{Z8Cfl_=L@+>0nb8Ao1KeC`7aueu0$6Eq(tVCA)v;$ z^ld$o(ye4f%V$k1N%4|tZDuGqur?cCjcNjc2WRTz-quqEBPY-s44o}Qnzvar#7)b% zUr28l*%f^wMZ7$AspU<9@!)BSp)=7@TGO*8*DM#7EIm#2-BYGz=BEWFs+5nN z?sysrt&vgr$0o_G(afYpUT3LrUzK{*lRZfzUy|P(44n=k4=5mJpM*S>NR|0IgWPzm zB0fq7K6M5|rs-t8TBeXq7>g!*+;DVB2Kg2!xX3rOQ^6g?1iJrwwG;$!S3wX<3eLl? z-~#*tVuNd3`eWK{yL zL@Hwu)#DB=Z)_QfJjofWG%3#+2`KrMFy_@>(nN%3-n4+RQM$u2y=s?wNKb z4bV(f?U@MDj>P!uxO?K;n4EvG`s>lu{J>Yz_`~m!i1iLpb50Sq=C5_uvi>?7|anmv$k4 z_*Y3KbwrsM_)tqqd}sn!?V?;(7Pb}s)mXwoek7WYJ=Kn13nv2 zU22osYy&Osuax5~qE1_vituUct|D5r9SF9mbU2=7pa`gV!Ao1fcmWkb1fgoR7RM+f<5Q@y#%a@;UT}Q%$sh0w z^wk*!5yso_?0>*#IPRSW3D^foXL`e-C$zsVW*6pf* zkx1?{RGRVFwq+gH&ecnccIBb9SR{(Fmvio!z*OR2w#oew)U)NhRh+jf4kzEfpvs|y_?VAoaBEEO*&hlVsIKuPaui++c z38emiYBdbtwuWBxX*h(#8U~S(afji%G|-#hUvI9jGdiRpj&6dsKD_gwgWhTL{im(B zZ#4AqsdZ<_xl(~hzF2mN_3lp1ExxexHL7ehDCSqZtyGh@QZ;1JSr%_4Wq|fRR+%FZ!%-ZAz;QaA zpp_qW8R82<8wh`-lOUWu*%qVU08aY;BM5n@s!OSLL^dF?e`uHgDMGY{7k7nD0;h4t z&tTBXc_XRMi0%!JV<$L*gja|_5~(J^^4{RyCcz06!(QB3C@^r&r}&a|18dM$ z(BeU@lEhG(2eo@pN5q4459*BmpvdqtTFeD$8SUmmWEq|23Ix+^v__}PffjV4m85OM Q7&R&6(>ad|WDDWqAE&l%fdBvi literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/enums/bs/TaskCodeEnum.class b/ruoyi-common/target/classes/com/ruoyi/common/enums/bs/TaskCodeEnum.class new file mode 100644 index 0000000000000000000000000000000000000000..9b3ccd1f3ede1a3c3ebed63294e457ef012091fb GIT binary patch literal 1376 zcmb7DTTc^V5ItYJ?Y8T3TW*3VhzM;FRusI@hCm1=Bn^-hn;4(A>slAvUDFGRufF>m zj1Rsmh2E@7d4sQd6;&Np*r=FIGmzhA!tm`6Ge2`L5BxFMro#*Bha z%rdC+a+9xfF?7XnD~8(&LGz3+$hadA+1gy+*cM1+O}DI9>h4Qhr&HN=bjzuit9qfT zZyVL;tL_tPjexX(WYSqc%kCJvhF&t9qMoZ&Y^TU?j!~vJc_P1(eYlnrNIq_#IJ#?; z>Q+@?I`Kbl66Yq+m9-sfqh2mpm2IO?;w(JA&8GsHM02vq_7l^xu4$C=M#X006ZWE2 zYeeo#B%A3;i`p}rijm1~!&4=iu^qd%B(T^bo%LplW+a&xi0AC0W7O&us&J|WQERVQ z#w_{K!wIHY4A^mFGwsT;L~T}bZoOh!4{VmXCpaIoJQOPWco6$l+{HbC$@WG|#W3!x z=z*r9mzS$pqyc^5+D=VC{qyR}&!gAB4&VNM^N!!fRdk_Sp!0MLxeoF$ORhtCoo&lz zVaGCSl>0(Zv(=JOt)>G~_>&xLs6X}tK|kMYJ(*vgqkpNBene?`h!did#x;05l>u}Q zvmA7fB8~wJLf|6Z#%bqI*@O6m&@saA=_W8lYt(y-ll?F*`S$Y&d8ndEsn3WULt^{T z8T%1J)P@yH=2A^2u8g^1jbrA%jca3TR4ZfN0cAex`i+z zv}>v9kwe5%GY5#KhNJ_?9|#TBpm5wp3S*Ix@hOyu!8XBO7=3e`@k@Ai zMnRPE!tm^Wz-Kt_otB|MeULn4=d68Sd!PLN^L-aUKOXxbp+A5D4EoUO!|ebXafiXX z{2Jopu#6@dBQov<2pVPlzKo!ZF&X20dLW}&#)OPXK1F?)67bDM6O*yTv_M}xWoNW( z!G3FM^p&wK&9DlYoVJwH=JnjG=$n+WmN#uHnb-4$T$FUf0^Y>zlbP|jfIs?dB04u8 zn@tEvDSO!vP~)rmx~`>lYeh@uv!=B|4w9v3NayQV7>_@VCI$MRpDppao-Pq}{_%?UmGLSQ;{x`ptW%2nfbN>4B7S(9@dCR#D_#UkoM z;nSIjto%!pTJ(`!!lBJOVp?W?RA8t=WwFyCP9xz3npM(Vv1o`{YPPdNpvqqINGe^* z?hbL7Xqg?a!^jz!8Ksp7B<(^rWlWh&e#<|0t%r9>!DZh5Rt3{|DA4=g8>^rLF$EXV zqTmw0y6}kjT{Em@T2ET%{oM}-8}Asup`ZyD2(s*KMo-hn!N-mLogD>1hW*CjePFMj z?;TAuYpRd7o>>v_q|&xUtsNio?9!@{%2WFT$%);8L_2{Q@$RhtJhfWS2Bx* z7sh(&ayQan!#ft_4giu=vLmTL(x^;;BZ&i`wwY;V5BSlBD-dX>Q#(n10-F$D;o5`y z6P*OE(%b9k{AAyOYbE=BxEwT~N|7CS_8_sn>zMs@LX@Hv^Y6s#M5q)&Bgr{EkuUI` zbZ)?rvj<_P5rIfWXWvQZUnAJbKjtBp~?ZOCACJab$~ht zP(A8-wcY_5yuZlDy@{&iqO^&+KjFK&^} Gg}%Sb)onol literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/enums/bs/TaskSettingStatusEnum.class b/ruoyi-common/target/classes/com/ruoyi/common/enums/bs/TaskSettingStatusEnum.class new file mode 100644 index 0000000000000000000000000000000000000000..67e9a1cb691677a4d9c656e8858325138e964a78 GIT binary patch literal 1520 zcmbVL+fEZv6kVsCw$pJaFkFO+f{4-DL( zZK;M;t2k=GQFFTUPBUEBv`U(*yEP|6o+X9=MP{Be2t|9_U{JC<`mU~)b*rRmZk5o4 z5iGqzzR*N|E&D2?G0eTbLZ#rYUalDqLoyb>O1zNvI_%Dxma$Q*6pU(4FO+#-UgFI+ z44GJa`|&HZs$}h=Ue4=PlUFz`Q!?B}3;nTpdmWO=eQOfE5aBjF!GbB%GF`f%#SVR~ zx2I^g#PbYc%`91TV^yL$*&$KqC^1zmH>>+&JWaX<$L~IMX??dH@PDEc!!25ao-a)JQH(Y5N%%agqzH*`2%JFRc;7kvZL-J>6j#{GH;y~a z3@YusW+3qe!HdEDI13J7!Xv_Pw_|YVV(?IhU?0Ck>W_1M2|og~D~YL*Lx_p#0|XPp z!T~~`$=b?6@-mGPp6l3X<5baxjq?%^^Wx$Vk-$`wU!HX1-@*?FTT0lkNXoTC$ouH> y3=u);R(d=`Z{Ua^{yMs}EuoHHZ6Azx(ojF*RqsMCB6Le(>aS7lKBfud!_0rUY*%vt literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/enums/bs/TaskTypeEnum.class b/ruoyi-common/target/classes/com/ruoyi/common/enums/bs/TaskTypeEnum.class new file mode 100644 index 0000000000000000000000000000000000000000..f477905c00962791fdc785721e40fde3345b3978 GIT binary patch literal 1534 zcmb7D+foxj5IvJ@l1*GLS-BWQK}1M`up;0EA`(MTG=SxXR8c-{!WtKojb;<7_yO@7 zeDc``OF_WW(&E|w!160Bdp4m6BoD9;(>rInd(P?Z`Tg(vF973s;)Q^5A0}|igB}lV z`_O?q9K6eyNxq#D(J3M#;+~J7X^!6)5fCvWBFc9UM0AOWiJ0ZPxCe6#qMD2@&PJcj zG6WNuLS8AB3mb+)fBAx`=w>-@DQQbdYSydd#%n!Jz!ZZ!wyY+X78nFIo}es3rm(6r z$cZ&=T~l(JnN`$M(J-^*E0|iI&OH68Xkt08GK8L=;i7ddSJo|tq2O713f}#5KREyF}gT#2q zZ_%!bJ7SndX_{fO0iF5!6!l0b#o$$qtf`gCMN&A}AW`EeF_OtuZ+8ZHn#meJ-ss3_ z%P~!D_Nj$(F{95Jyv+XEc^TnDA>lHg+a3uI@rYsIbfYGrAM+9}qFcfxz6{|p&8$_@ z%vG(pN`v_4!{))}TYhj|LMJW|lqp-KLY{sOKD|G9w<{sQVZqXK)OPpZ&5t(N#lhyI zdF6r`T$x4Sz1-lTk zg)oF02768h2O0!Bcz=l>*ZLGrxX8-kq3%71;o%*)!vSFjo-c&fYVg^addZ*Hu|ngp zq7{tm5-@DXjX;Eg8&!R8sBZjQ_yJ)>_Bv&sd~OfYHkxdpS&&=gRvYl!fb5db%WXE$ y?*2(J&Mh>lE5a81>WXU%?dmodCs=8UOxFTU@S~YTwP1|ePxacogaA0HL@b9f&3Y2^1)Jq`#OG9J23_eMazER7ezj03U@I zM<5C+u9)2!&CcH4$LHHSfFaf+G|+2fwT(5wa;XcWvZ*dg3I>zZ z(L}r_BOewtT*}ZUTh-ZPx000qlu8UMHZmVPU>yMMJju94um_eke9h7CPuUZ9gZ#$ QftWU$$hxfjd18Q-FWHP>M*si- literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/ErrorCode.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/ErrorCode.class new file mode 100644 index 0000000000000000000000000000000000000000..990604a2ef35ac452cbef0c1cea1aa272a8f6e36 GIT binary patch literal 1810 zcmah~U2hvj6g}ge^{(Ab>^gN*Lx8jm#h*3M7O0&BXq&bvc8VHQsXVc9R`wQug!LN1 zU*QLkmpp((T?r%vFRjFH;AikiDQ9-qj$IWYMKgEq+tkhIGjKhLBx)vZsC`QGzuD>ae&(B;>QfFLIM&UlOt_N7D zhTC^`M!h}9`^?_!5wA(Cav>iW$%H|6*R3_}mo)Jye+o`#LV0tv-_EfhBy_@aXuTm-@(d#b$hDxV=?}u51d_Xhk=Jr=x=QfH zsc;w7yIWYsCl;1)%0d>W1z6?(HMda3nuRy8YT>-H&f=VfxABg^^6@l<_}4~mcVFp} znj+_wI2#rkXsTt0U6fbJdV?($EWD3%0_o#Ap~9TWyc(sreQ@CP_XR4m4>XyznKz$< z=akIFPagC?(A&5rOqOV`BuM!VyfBySmehKOy!5jj(t;aHfKS zRu~W!6zWmvQ5c z@07XJWaTMNRtIn50vTqpi?^s?jC^k(PoIV&Hc_A@af>1pY3XRPN{Ln+W!7Is!Z82FqJhdY zoHlTgf2MebGohK_FP0kdNP(-|W6aW3>x$m2_8~p>Tnw1f(@9Fv6Y8so%KXU~ol&)P zeO+z@a;>E69g1b9ONTHjV3_h`vLhC7hY>5!B`2(}I(o-^q998-=g^hv-ZM$Wq!Mm&DEQK$Ea vsfIQG3z82W<3z?9V?i}M#$qNF+9tP5n#&j`UmVF$({P^i(|P_NQ2y$_N909i literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/GlobalException.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/GlobalException.class new file mode 100644 index 0000000000000000000000000000000000000000..a1a03aeb728e86053a53915217e53078686039fb GIT binary patch literal 1023 zcmb7?Z)+1l5XPUq{5gAG+q9-tW2;StlBUH26%-T|S}RKQh1I;@q)S}6yOp~=(Ql=o z2)^(G_@Ts^%O$DJ33`y(oy*KKzkT+8{`&p{z-?UjP{Y*)RB?^NmW#HFZGlFr5-r2N zN>Xj&*G~@x#4~}#0~4oK#;2-)GxH@ zW8mwppoKbI2%MwL!2=vzp)>`))#OJ(hg`!7ozy^tKVwHCO zxCj@}q>KgDD6+s2RdTJz0^Pl@aAqn?gad4mN}E8obEWlhb)m$Q7>^;=ak0=0@|>Z* zpz?_%9557<13RTamvFggepmpx3{&aO+Q(RC4o24XI W%1N9#k~>6l_b((9Pd4(yZ~g|AaJqs3 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/ServiceException.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/ServiceException.class new file mode 100644 index 0000000000000000000000000000000000000000..a798c4c9f0b0cd8279dae49987b5608cb5e2b94d GIT binary patch literal 1315 zcmb7?TTc@~7>3{JX=T&WQYdOY0SdHRa6>flLZU$s#Z)h>Ht}*PlRBo|HM?7)|4I{y zi5LC=f0Xf^-JU4x1)Jyaq8t6UX65lGkSZKC%hbg`*i`DV~Rmi{~ExEH~^ z4pVfPDA&WyO!!d+0^7}w*SGzF_f^@P^*z^?Upn%0$ZG7C^iNesJ|Ar`hf3{YW;%E+ zVSh?Fs!>ZzT^SynuVkrKpAwG?HmMv}Yj6=)SUIO2p#{ufO1_tnQ2mm{@^WW0HCx`m z@5n=?8(W!e(0zTYc`Tru$2^L85`7k^PM*ixfg7s+nFtbNVY_)-GNnN*HGasAB(Vzt6XWo zHChcgq*7~54X_$#NKGj7Nb_W!lnth}8IhJ_^^l|^>GX6G3%EXHexOEs7RO9$>kR1$ zxJ)Eq(KZO)B4Rs&uf%R+cyS0PD(Xgx0jjt$3}bVQYyW}FHv&`8wdbPEoy59tCOW5P zogXDS-@@&L$mv<<-9+a*SQ$D$WuCg@M$~O<;vQex9$&<>2obm178WwOXUv3Au6JS?hJynU8 zQC}sgw&wignSgjHuySlo>ZEaf8D(lA$<(jMi$CuR_)oOa?nt23?p?|EGK{1dhG&||@<8-VhFG8R(ElxfG z6%CbpH8$I9AI#V&?P3pA{sjQ}@DXrdWaME96`q$-=eokoQ^p?mTBrL3qWcxz=b~>l z&lQzb-dW=VTpx8}9oHD;^$j%0$$8C^cf_8Y=Xc1e{L5-kQ*(ypPgp)S$se({O4bRr jRmciD-3N1owPLA$|^+ScDohQN1f!`0}p1+a}qquiN!RjU(d}js2UaD~HyIq*C)}MskjF^KP z;Sa&6pFIlGH0TjsU5w%=yQR=+?7|nElRE~|NznWnT`XxchdNOlemNSfhr@4!_1t>W z<&gs(oX^FkmHimev)}0^1AjP5o<=^SK@$7nFWqo6V|D!XAPxOHT$Z=Dc%JKT>smaX zf30LeHVjBZDFixgfXcqx3&&MI-k6^(n}R*qZp(p{Ea?99DA?2mvT;qp5rU%sHsid& z65ZAfpKi<>EGZlz;_jZVyDQ0P*bVPRx+DK9>ViyG4Kui2!(kk$;ixRfaJ+_TiDvMw zjT<#A;--yTHPrE$Li>&SrZ6`i-)}~777fNt<*gXM%;!*{+>jEUo=_QOQ9mtiz^cTo zVU#WpHnXSubmFl>eIJ_QP82-X41F}z*U(3t;%#?%V=I`$JE(y-nolJhHwS~%aojWTSRJopkg|k#Y zN3HX0S_89~=R5g#(oA>{@AFCU;za34yp(h#Zpu0mKbCg$LE&bV3>M!xUbEF!FHs$f zyE?s%DJ*C)F9+_3IXhyGjFUz?GE)?j1AkaZJfwukQ$nehLE-`>E^3LBInXE}-bSBXo7u@m`?(3ifX}BoV*XD@=8oe_@JorPKVKurk}b-TFIBT;i}Tb9C;r$qz{T zaF=ksC{>C3buD~Ge53RaPS`j*MF;10eE8ZICzx6#Q7UPrDz|~JHLHxZjCg(={+Sw+ k%X^bojmZs8X+HD2m94$WkB!NmQMs}=`Fmq>Qy;GN9}!7#X#fBK literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/file/FileException.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/file/FileException.class new file mode 100644 index 0000000000000000000000000000000000000000..7c9cb9eb57e4be0c919c10174cacbeb4b300e57c GIT binary patch literal 633 zcmb7ByG{c!5FF=0!jX`0;Tb5POL)1V3!)%^1Uf+xxQ5DinBeGQqkACuEGi@lK7fxx z?1g}W5+UJYy*nQ7&e-oCuWtbMuvWy9g&dX(SSg^+P>p1$MBvIWQbu2$9x<>phH~5J zC>A<)MUcom-Axx`nA%rb#Rm+HhSL)df(Jr(c_$8)?zXN+pbM`j{kYY18FFoNBN-}= z((*j%dosKfUOW$T+F#Vgp(FZu&aZk;fG61*$SAQbx`(L$O4NL#J8VtxcP4ZL=3- oo(H^giV+{EVj6RprEL9+i3(xqvIw`SPny**Pi%$-sxnyo1l4||0{{R3 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/file/FileNameLengthLimitExceededException.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/file/FileNameLengthLimitExceededException.class new file mode 100644 index 0000000000000000000000000000000000000000..418d10edd3b366141d4389a34482397067351c51 GIT binary patch literal 714 zcmb_aO;6iE5Pg#bOszmdTGH~>0;dG2@ujy^y+DzIEI0%rAx;~65*FFJ$XQ42&*B6^ zf*;TyRdpPy7StYks1Gyyp5Hvp%h_9Tq5R7!f!GVM$m2${t>@Gr` zYWG|1UPbmO)tl-B!;y9&<+U zrp5Igj|fe79_tARDv%jlF=6> zE(voes*>c!Zi6cKD4mtpZ&B{wpmIIa*ChHg#S)fLLtSw5u|jY|kksNOUd?=01eMVI E0qsz~&;S4c literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/file/FileSizeLimitExceededException.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/file/FileSizeLimitExceededException.class new file mode 100644 index 0000000000000000000000000000000000000000..867ca4e8b73a42c5d830439a1dd56cd01ef1b66f GIT binary patch literal 671 zcmbtSO>fgc5Pg%Rb!w$eaA+u{P$Di#rN)=uf-0l{5|K*}DUlGTjlBsQ*}IXmj#B;= zCr~B$0sN?{;{a{Bpi14t%)I${JF~ODE-!upI6||GhJ!~gcJSE6I(8jAaj-|I&1I@Y zJe28N8U5kiYXbENn?0lFR%kmEaVFQ8BR@n~f2p*x$AnU=-yX8+nMe|f;%u-8p^zo@Z$gVovBy|!ge-`jfgt=-7@MVf2nGD z*pG#t_@|?(46TEw6+FX!1r3+K9`aM>+Jwy|0{2h3;*oA{K>kUQ)xNH%RyUB8kJ$;TAS$hW_m42Y` rvAF*o#lbg}zOM4CEM2YxXFos{HMSZn>@oHjaV_er8wd{>6|j8;x>~SC literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/file/InvalidExtensionException$InvalidFlashExtensionException.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/file/InvalidExtensionException$InvalidFlashExtensionException.class new file mode 100644 index 0000000000000000000000000000000000000000..c52a5b0200ed62d033e8de1be464f1c33143b2c6 GIT binary patch literal 772 zcmcIiO;1xn6g{^*X!~sCqlyZI#KeX2;oGp08aJgRH7{%^4bj!mjxc08L*9GE^0Qo# zux)Ac$a3%WgTS)dX|Ct5y9L?<4pERu>1CFe6B84WR*LO4zt9#)B3+K?>d5@;d@#z z9~-9+2K^JAwAG0vvlRO-yHV_k{)h~+Gm{F|TkcX>)7!TyJ!=j#?ednJ!>+!_2n3au zb2@48In~sEiMzfZB?EtDBBH)`ksgbqZ^r6qlyExNn}{4iKS zfLYo(dIc1TxDtfCpO9(uHJ~kRY;J8|Lw?+5dZ)H literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/file/InvalidExtensionException$InvalidImageExtensionException.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/file/InvalidExtensionException$InvalidImageExtensionException.class new file mode 100644 index 0000000000000000000000000000000000000000..bb4ba7196ab3c9e2f763f0de54cfd3ec545bd22e GIT binary patch literal 772 zcmcIi$w~u35Pg-5CK;oNabFMw4;q(I@Zy4(s1ONW)F685ByDWz>6V#^%V&8KJoo{A zlvtxC9wH)wbLf8c=2g|JqF-L0o&l_5+C>r*DNLp?MaYLTP(oK_5Grd<%DV(QAoQ24 z2_s>ms?aUj!-?@ngx+msRJ23boIkHzi5uZ*VVd4i6e!c&`VF&CB{(J9kc3=C8F|?9 zYce<%HO)++V(aWiuqTc`Nhi8gAz`L$ZiH5ivM-u)?=F%iw!C*&m)8*kLbhy-3`$yr zp=AA8rEYyMXxV$^G4iePWLxZAC@sCug59;|er~nJdlOpQTiNL35{AB62q7cePUGpi zHbzLtR*dj@diTh-g1X#i;%nSaht8xcag{7Ga0#dXH6vl;&!sVo1V1u}y zN+89ER|1#MK4y;jufun`xVW_Z2zq!|W%+bj4Xkp;8u}R(KXe$tAV-{m0*1IsjD|7t Yu1;f=QIhc(^9hV2&v)D_c9TQq4PBAe+yDRo literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/file/InvalidExtensionException$InvalidMediaExtensionException.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/file/InvalidExtensionException$InvalidMediaExtensionException.class new file mode 100644 index 0000000000000000000000000000000000000000..abf13a0882f5a61c2614c6000b63229181bffd5f GIT binary patch literal 772 zcmcIi%T5A85UfTJSV4S0(3qHb@Vyr=it&P&V9*OljGhJ0by_TvUDTvgsX*Vy5+!ArkVQ|4N zNQ735Xz}z;*|xp9JYwQ&+A literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/file/InvalidExtensionException$InvalidVideoExtensionException.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/file/InvalidExtensionException$InvalidVideoExtensionException.class new file mode 100644 index 0000000000000000000000000000000000000000..10f91b3db7f7a9cd19bfdfecc78f3239bb634e26 GIT binary patch literal 772 zcmcIi%T5A85Ul1Qu%aON)|i-hP<-sgi;s9gO)vxxDlvK*V8k)QjM-ffKg*Me2S31% zGWM$A0b`7bdzh)Nt?sVw%**T3Gk{IZJBVT?f!PG+2w7ixO6aQed}Ym9X`etxgpr~( zejrRx6}m0soEUyY7}`}v1$%_;<;(Jo=m=K}({xV)PnqVZSR%KsAY{Sdt&!WY3Y6z9IUnUbE_TRo6y?c$wohykpE^Ogp}+#4W}E} z7$F&2F`~uOJ14g7)#V`*U*q;W^e5d2tE7>FLpcAh83|i|E{#P*_>u90!7O5k^F72- z1PMmG5;%MgGjqy+V|*tIt843zpoe!=nooz-K!r1oF~X?up~EP~IN}WCkmo8P8pp)D XI*Ca}QN~luM=*^n-(j!NO$Mnq(m~f$ literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/file/InvalidExtensionException.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/file/InvalidExtensionException.class new file mode 100644 index 0000000000000000000000000000000000000000..43ee6a9648c30b6eeb333b1a0f697126f38a3e25 GIT binary patch literal 1903 zcmb_cSx*yD6#j03PFn`SB8wXeE|fwUQCzFYB51YhgW70_50~~@hD>Lg>D1-THxuLb zWa68~XSYO_NaGLicOd%aKQNv<)S^s@i8{%gbMDFc&T`K=_tW{0;V_N_a8#;}h0ut25F2n@k`s~)DLAQMSiuNGN5Z_x4Ly_6Gr}6b?Fh4= zTju!fw8%RoFsKRB6!w_G3k6Y7FuDpkFETp!LKKXx+Ds5`x|+?YRhvYPXNA{>p{XEj zof|1(%bczx2*b`YG>lnh!QrNp;zm)_(M`yYkv%sI>z2q=d@(dnCTI9ft{L3SYL^^a zH?whup!iuRL%5nj;D&Apljj73|F~}I&Jf*Xq>3*u*sJMDG1QJ(8FJQ`)J<`rn41#z zWj0DOUI@5ZAA)ZWIIn6FwcXW-UoMmd_XJa&(p;m5QrQk~6sL1B5 za7H{A(h}*ZQUq`p{mtVd7^5XTk(u1#W=9d=k zFO{AvIHMu~8VLXL^U~+iV|j{6Z|78WV26sG=v2{#PKKj@eUS_UmYvmjo~NfpxeEpN z@{4)H;u%d=f5jCm1v6az>v+5Ww2%MpdY!%fA6c)n?*GH}y01h{L@t=0=kvnMFvKF> z!S&8^dG=^cn%uE0I=Z2a*fzgIdlzu5m8XKC`?saPI5i{EPTcE`{Ipbyn?fhOw~h2X z)nW@a)6cM#PCjg-pN7~coxo*-% zSArpe9y(E2H%aD7GEJW=$wX@?!6fe&!gAC<&mD^p{>WV?xHA5d_ zq>(lL3Rc&GyQ+fwtAZQWg1f7LeH2gTw+~_BT~u%r^=Ko4XePdicnk4X;%&s+5&QxF CMhV6M literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/job/TaskException$Code.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/job/TaskException$Code.class new file mode 100644 index 0000000000000000000000000000000000000000..4053363075794d8590fc09a426b869d598c59484 GIT binary patch literal 1442 zcmbVMZBNrs6n@Ie+La$O{{wU+QmnmEH3u}{m`rLc&InQ(Mx&8U;>kk0SNQNLHuHZhFBuq-qIjC1^2|1&eoVk87yGp&3} z@BxlABdu%OxxAgx(iweQfV6M!nbuR2(k3l)C$*c`(`hR$V3_-x*0!EEtxR6aYAHk8 zGIZK9o7Iecod_>KZ;M;-#3|O8wADzoy$C&q=euUDbl_Gp&OwoiCh?foO98tP$S=N~4bZSf2n<$_bpIu(!QXe1JSr$qU}|0WR-T=#shO6NNC9}cN#mX5gk z$|K<++dK4~YOO+%0ui<#uNR7q&3aR4-LI9_+aSL|Euh%tTBYFbc&t9*mM&Rj6;d(9 z8hKj96KqP*1jMi0X)0#0r6Pi;ib0I17-hp48^+l%!G=kUP(xlG3KJ!uvrsJi#B3lE zZ4zu99Jz%mv0u0f7Xd5H73X&+(G>M>8?=&0Y{uy-4$%FUHb)tMfGkPimQ-*{LbxR@ z+>#uicZ^k!c8Un$I&OdtozP9P>{03vKBMCtfscd)ng@b>ZHoM7Fx&87MhBl&!s6T) zbe=Cj4A?$ho4Pq`V1SmvO=H~5|61?trC zIkAo|JJ(r9x19^t(PQVNI{NIKTt~ls3V~6-sFXt5KvgxV#(@UiMZ{>X1NSgYRv}N3 K{>np}qrU;x-Aup$ literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/job/TaskException.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/job/TaskException.class new file mode 100644 index 0000000000000000000000000000000000000000..fd1270bef2c75259d554ffa1b2496b48bf9f9f5d GIT binary patch literal 1100 zcmb`G&u-H|5XQf;lel$F8(IP_>oeBeFyLmcYTy_(?=O=vmC8^xaHxthYdoh zsShO~*i%Yo%~a&M%n8+8W{F5UGRqUKUhO?2&_3ZpQ>(lXs_2MxAj`})7cnKaln1R? z_oHl}KPC}JeXSyS63cf5L!x8djoKoAyL*<|Fh@5CmB)!niYJ5@o2_HI%Y?Leer&jEEPuy+ufP!6fW8A=RJe`t6ajH z|6}4e^tfRS<7`IBypV^xCxqI$B7)bG#c-^THn*m-b~Y0KKo7E5K2L1t{+Qa1@f@Iv zWwYD~FpmX7eVV}&mL`oGyCdA624YJkgy*$;LT?{s`hyub_YUj)?VFF*!Bxz{2Vu_c z0?x1tj$O`GagkS#cl)e&`92eF*Xb1W*{TBb5PidkkP`$!)Y#d9iJURU24jPoXf9eH*H|s*29_MVWN**t&$2SH@CW## zjC1%AV`JRn&CJVtGqby|Z_h6P4zO89(?bDkCA3OdC)5*Zm54)W6J_-9{DeRkglgaD z#0l*}5vQ`q-s})z=}2kijtTj8CnOa5=2jAFfzt9aosFcu7NeM%MqnatE4Vq=pJc&J zR6^JfA~W-CY95r&+01BP-bZrocmqG>h5WvlyJ#X#zt|qZ8<^Ba@=RsjTfYU}skjq_ zX6EirqYrymT5D_t9ySPj|D8$L{mU}sY2P~%+U+1)!;gg?`vYf{9`|m3P*1b$F#p@=1pC9ZNPbH-Q2GGm3AL)H<;@=k;33Ax9Gs>kY literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/user/CaptchaExpireException.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/user/CaptchaExpireException.class new file mode 100644 index 0000000000000000000000000000000000000000..d0258147f6185e578e832fbaf472b245ca7f7426 GIT binary patch literal 524 zcmbVJO-lnY5Pj2c*4EZqKhTr6RU$KvZsV>v*i%wvxzc4G@m@Nt6FnjGxMEcp9cBaT+*f4; QvsXOMYMc=^7&EB91CyGF(*OVf literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/user/UserException.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/user/UserException.class new file mode 100644 index 0000000000000000000000000000000000000000..ef0d67452925f2ba163385db055a626c78beec50 GIT binary patch literal 633 zcmb7BxlRKy5Pi45MTyE%sD7Ylh3W|_zsJx369Bpj02ZGO{LZaXU_$b6| z2q-8K5-!Fw&(AaO+3z2(ZvghOUc$1499D{0Euz6ti)5%o;L0#kMt9Fo7}y0v<=E&b z7CLrCkjOmUO&4P*94M{gLxyJ4>5B)!1EG7o6NgIo+Baj+mDiVk+-|uHxnpxH8LE!b z@-i8CGQ1XEK#1)apJoeEK=laB#dj)V;ErzwJWR}!;*<@H=JL^(_c6)j3905?3a8&t zo5uLVG4_~Drzm8wIwlEyd8US%ezo08J<|1bU>O$H7u^ z{PuOd85XI^VCfUetfajF literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/user/UserPasswordNotMatchException.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/user/UserPasswordNotMatchException.class new file mode 100644 index 0000000000000000000000000000000000000000..cd4e9b135275cc81bf7969a4910d59621bee3600 GIT binary patch literal 548 zcmbtRyG{Z@6g`)BRuIGo#!f3Raknux_$W{l4WbFMjn%MBUWo2UF2l!FO zy9AAuF>#7>A368TnLFXgACU4SkGgFP>E!y#Kf0jq>S!eoD%4gP-+<+ zS)r{jCb7)&Y;uT@J5*ZPV?w&#@Clig8Aw98t+edK)4mLEMSsGbYTE=XD_ETCMKfbZ zDk7Y;12gr)*gPqZv#HUZd<^8wvI8$>Ltc;JMnutr2?rfxuY?UoXJ3>i!ReYf4CJ{= zuG;yP?T*F0Ak-3Fcec=UwXyCrQ7ytjjd1iYM})n<%zwXtP_MTWU*1IMq1Uyc(!=K6 zkJfd6ECbtY_}E68|BH8zEo717nCB{m0%yDkEHM_jbHF^}SlF!+y&&~GXO$QoW&_JS TS7rpW=RB?|ToG0nQ>cCd=$(@6 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/exception/user/UserPasswordRetryLimitExceedException.class b/ruoyi-common/target/classes/com/ruoyi/common/exception/user/UserPasswordRetryLimitExceedException.class new file mode 100644 index 0000000000000000000000000000000000000000..2dd4989de2b91b4f072e3c89eb892091b21b6e53 GIT binary patch literal 743 zcmb_aO>fgc5Pg%6I#?-b1C*8)Na2!H!oF||;t~onvM3^Lq^hTlvxF`7Zna()`m;EJ zkl+XKqY$%cRVs1m0Uu`HjA!1ynVnz1zyAdA6pve253mv99yUWPVJpD>01pT&rL;=q z6KP9j^g;51K(7hQBcn?vw3~>$l1=9D9}$+GDXrXdLbIPFg9)KAGBZi&q)N*-RWX%z zUrck3bW@Wtv0!@kFMFimm3gvb=NzS}oqz?IBd--!@D(zT)LTKeC`?#+PmJyCkWi$Cwc}+XlqK%HlsUY;c zkm%$>M6|dl%Ds%pi@)cDmI3f!bkx^Bd~p!^W0BzM}DEQPyD_ayhtY9UZJN>YW#2 Y6Tes-GBV|&tCx8gHL40cs7n5cy1lOjqmU<#e*ti8hIXOc&SO4l~9mLagNuzjCnlI zOG83YMmtOimW((`GB)B=8*E;b8QEz=h1D0<;k!7^;zic^9uH@Dc!7trZTP-~7iGMJ zmnFO+V<(;!2pd*GATeUug-qF)-7}vVu?hvv9uR079m?enJwB8ZxMyTuT~IR>M>jHg z%gk47TQi-^U~M|8mS{&jrx#4sso0vp6KmMA-_tQvvyjP}j>dKaqMT#vW?|p}A;T88n*1BN+tYAD!WD#bpmRIDdq)4mYPU;x-u)U~OE6cCtTj_`OcfSrt344eQ+5wlr5gH>dkKJOw>SE7*nmB>X_Z zgV?QLAI>Rw6|YHnUBM4=Ucry>hJrV7PGCbbCk-O6o)u(qRKZ*Lv4po3`~>ePcmg8= z=~a4Yvxn^1PX#tN+w=KV@H2+(!Mg&XGiMdNho39B2lomLPF8dM+~oDQU-{>Im%sem z>(@WN_@BRA{P(*TzWnU7e_eVxb>?jM3*PGbYp-7afCmw34{U3PCz0!eZ&5JHX3m|7}n=8QmJ3wLkcQE!TEj^%*} zJQUnIIH%eWSaPCb>zUXoWB+37$6}r6+HeKS!u%1>q1FAq+;4=^e<0I5|;Ak?{@s-2gm9{~j(& z`f264x0m$y;bB5${Kp83Q542cuOJe}c-=?OLW%#$99A&_2;eRlK2@=%*;er(NV8Z%+-m!zY?a zXF5WpDZ5n@#(BOHkc8?`d`jPgMAnwCVdF%435gB6r>s{K z8CgQIKbnk&E@AKI=uB*mQSK+gTbJ>b+(aaun~3)2RC1g--m-NG+s6Br zu-#oJqf7YecwcWawtES8k0+ziss6~hnE3sT3sn=L53qybzjiD5W?b^!`zntjL35RU zaZl4s!*P%^2~UqdkuzdV@rgkZ@eW z2?^BgsSIxrXiz3d)_#LZI4ESr6vhdKYw26`Y$LzwZH_At*p%3@jIJx_yoQ|%p=_g#T_kyiBq2=x4;i9q5&!@I literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/filter/ApiRequestFilter.class b/ruoyi-common/target/classes/com/ruoyi/common/filter/ApiRequestFilter.class new file mode 100644 index 0000000000000000000000000000000000000000..c83ebd8eba4b4e86382f4d4bda821f99cf8bcf4b GIT binary patch literal 806 zcma)4%We}f6g|#NG9{fhw4~t?UIihNBH;zy1tFnI2&7O@hZdBbGj%hjlkwp3wD2kX zLpQ8=EcgIE3UTLwL|MfP-+PbkbMC#qzkdJt31Ay{m$8A~5-zuK1y|c>;hKlOhwBWT zEYf*=xU20m6-$2#Q zq^+Os@4aB?4u)l?cr20`4{RYzjqHaBk8-PZOieeE=T^nV={{L1Gqb|jx5`Eg+tmbA z?{^|$g5nD($I(B$=kUpJ;%=fzgH@~sTzFUlu~PI74k5s$2j9VHFdEJ`IG@pU&zv)Y z9kgj*B@1-XL65rI$1+8|>KS~jP(%cDt3#w0$#=lk$fb{{RXABj2hFXoaKDh#D0y2Y RxLWf0)rFdm5npku`v)Ya&y4^8 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/filter/PropertyPreExcludeFilter.class b/ruoyi-common/target/classes/com/ruoyi/common/filter/PropertyPreExcludeFilter.class new file mode 100644 index 0000000000000000000000000000000000000000..a197dbb95d590874f38fe79caf4693f60c33ac68 GIT binary patch literal 815 zcmbVKO>fgc5Pf5ZxN#Etnb6XNw!k4tC0s?7I0U#LKu7^98MRV5t(|Sy;MkF~uIRB6 zzk~}{xPU;SKY*XYPe2HmIIgHrIe=E0-FZ9vX6DU)`SIxlz&5U1xMU-bWg8c;Qp9Df zTBup56Y{rIsPr9z*=X((a(APiB+PbIC?6+7PsUG$7cjEWjeHU8idaqj49w~0YD{Q% z{b=aKNpz$fJ`JPLIZ%O?v9l9LBN^+Xomk#`=?6(q-cK(&grey6GK?u6G@f+_;!rq& z2>Xt!V-@y0&Hv#gMWBZSYgz(xFP=k9D7#wtFCK|e){E^%N$gMR<8JGJEBwZ!NW$Eo zwU%JvN(pONx3E#dRa_%%{i`a3?a31nC{K98IS^w#7)N3IR83b6M}a(3+Eh$J^<4FM zoW;J>zeS#JG^ZjbS_O_PHIJd3c5({vH{h`lDvdMKc;XGDuRG0sZ1DHV^N#`yu)>&O zTVU^I>H~$A`5E*9#=A7cW^3~mQ0CB!DDtVqPdJeGGVjNX&SoTsFnIxW^LB0XZ#lqZQeXkO)XrcD74}w9O%ts1y@3_XVLnY=&r&flM=f%) UC4N_c@f`+z!!iRE1`Jex0R>peZ2$lO literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/filter/RepeatableFilter.class b/ruoyi-common/target/classes/com/ruoyi/common/filter/RepeatableFilter.class new file mode 100644 index 0000000000000000000000000000000000000000..f85340dab0219d21038379dbdf9a641f1d6969e7 GIT binary patch literal 1702 zcma)6TT|0O6#lj?q(Vg$P%8?8f-MDuf}-FB?C8i0>Woz!^=V0$Hc*nd*(&@L{(wIF zYiFvHM(&GlS$gCSigE;D4xepN6WT5>&czqzp@-{FmA zE?s?&g&9?MLx!;>$KSAJ)8BG!+HLrrz3MiUkoH3n2(Glnok%c82pW%Ju&}hoU-C^m z6!K+5DBCEMeQ(vR&1ot*i7tyq&alp1oTM;5Dg3>3V^D@RM?*05Cp35BGjzr1 zPb3clAsG%<6nEC|@gORZg{u~38LTP=D*Y|WQ^jveN0)<&+Lx@)d0mqn4x)=;WOse) zN(J^^T3Y#u!Xktm=5UQ+*q1ds45aJTRwdsMuY9>~M^^Vl)#Z=20z&L2nhn?Cx={97 z=zA8f=Wqiz87B8{B2jH@#TAQ#)ja1dEaY$kw-^fh+y7}oVxdE`8{Dhe4_4NML;cvh zJKEp9Gq776$KL8SYRi8r#6%f-A|%(h7a!~}>Z%DveXWG2Oj>#i#i-1NQ$^_{(yR>s z6;e~VjnFmuKpLQseKCoz1X(<~6z|O5-1J zAdNT3P2@9XugRpND29=x{UDuAQ}{y!d{< Jj#N`9`~omEzBB*; literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/filter/RepeatedlyRequestWrapper$1.class b/ruoyi-common/target/classes/com/ruoyi/common/filter/RepeatedlyRequestWrapper$1.class new file mode 100644 index 0000000000000000000000000000000000000000..cd1c90501a478a41c27d0707151da8c213ebb46c GIT binary patch literal 1433 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 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/filter/XssHttpServletRequestWrapper$1.class b/ruoyi-common/target/classes/com/ruoyi/common/filter/XssHttpServletRequestWrapper$1.class new file mode 100644 index 0000000000000000000000000000000000000000..b5ae9b1def7efe676740948d2bd622746c3be6e7 GIT binary patch literal 1419 zcmbVMYflqF6g|_{F4W~AC@La?R&C2;0l}DHOoTvaQY8V!2VY#;5r%Ac>&}+Y|Kt}F z(Zmn@0DqM6%u=)#LQHqF^SE;#=bn3KfByP$3}7A$-ALo18FFUGXJDa_!2~8POj(!~ z=s9q`?7j~CrA8#RKw^7Iz%s%e#qDTm-#u_06*%Vnc^JBlqF<{=n^7p;>Y_k8+EY3^ zBk;IX4ysOA4;sqhsv7vtN99E_blz&c7Dcs986J2t+K`{>Qb%t>w^oy(@$gjnDtadH zF4vih?WK!B=C=e=%Rxm7^p%t^U)HNTGJNIkc$5s3g0kyvxuG)qrZ8oa60nQDFT-We z)!I<=q$7CQS>~8gdX-t~Jy{V*=JM~!vC$*}TH;P9+S^_9c2tVJsDwFQT; zwc*XJl$4GnAqm{c#W^23`h-=d^?1=_Y_mWu1|t{glOLnaz?(rmEX!48s2RVgaMPwQ zh-}3#dx2K|?h6_11r-Z3HfAwr!$yydE_B<-;IfTgJhCuvV;sZml}=hEJ1a0|qW(>~ z*-|YXDKL22u*LO1EDFqb5>cRcS4RJM&*t)NM4rXKdb5ly!N@6;^X?ADcp)v`(qwR;U$`iQ4F->h?1p^$1 zD5E|GagDQC8NY(>5Uw}9PR4s$r#@4o)kiV*Ve{O`Pkad6_z%nVb1iRTn0`zqGtHRk z!dE1Y+vC1&u}(MR-lA2CRwMjwW0a%uvk)T#M0@(%E$Oo(n(6b_kaUwio3IIGF5UxO mgp&M*OYLAjw1DY4115>PF@$@_#sKf*0Z9hS$ln@gKEDBg13<9= literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/filter/XssHttpServletRequestWrapper.class b/ruoyi-common/target/classes/com/ruoyi/common/filter/XssHttpServletRequestWrapper.class new file mode 100644 index 0000000000000000000000000000000000000000..270adc32bf78c973157f0335f15b5bb3fe08714c GIT binary patch literal 2316 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 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/mybatis/handler/DefaultDBFieldHandler.class b/ruoyi-common/target/classes/com/ruoyi/common/mybatis/handler/DefaultDBFieldHandler.class new file mode 100644 index 0000000000000000000000000000000000000000..b03bc1559c1b466674b5c897c53812f504c2a9f8 GIT binary patch literal 2683 zcmbVN-E$LF6#p$rvtixx5ok+FMMQ))ZCOC5*dj=2%U7BfN`v4BZjx=gu*t^FE)x7; zh6iSR_oWX$>bozcqce_<4>K*>2kO_qCm4NHOuoSY{yYB(3^F=dBd!l#Rbc-Lj=#7bF89o6Liw@ z&8fmYY9k`oM8T|CV^fUVdjFE?o30EoMx{6RREvSiuOV_HP zea1J7_b1J&fscn8g4BI#jb^^ zwbxp`_e80#I*Y?9FlXfihL|F1t$uhzr$l_sn6Y?TZNEM%G%`t;tji*1qhs34t;w#q zRT>4joPpLNn};#!yAA5KMg_vXh3Uv!6d*9egy|$Anu=gnUTb;N8Lj8!`JYqmOzKP2>jXP?d3zF^iaS-vkWp-su< z7$;dA*$a3d9oZ!HO$fp3eAs%R;8i}PG5+(a;`bmmgZ~Qu+cM9f{6JwFoq>d0OyLk^ zJ#@euI805!n>fPnL5kn;5pCnz@BPczwus;Qmk|3E@kKa7`UUaHpU`&u84|gHC8*0t zE+RQFxP?%W$454bXScah;8+>b{j^a(wl;sn0INqmd9@I6jr8AEu^2*KxNqNZUo zs#mgo8-0w95o?GD1LX7x#xaOvv_8cJA|9ueEJqy>(g|WK#0K+7tl+qcQz}mHcp;vk zp+C{Rf_@cmt2px)lU49evt@?}uQII4(j{>rTkDy~q`HMQx3)7~ z_iu(zT)s&g^KOz*jYvrHZ5)iJ5pTbwG9hW}M(hO+kfRm! TN~VTZa5SWhe7(yIx8dAB*wnA7 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/mybatis/mapper/BaseMapperX.class b/ruoyi-common/target/classes/com/ruoyi/common/mybatis/mapper/BaseMapperX.class new file mode 100644 index 0000000000000000000000000000000000000000..a66544eb17dc873171feb774490c7aadf4973cec GIT binary patch literal 7409 zcmcIoX?q)26@IU5OSUX8q^(O@h)vQ^FL9DmHsPprq9 z71*c+Do7r&lNTNS{PObLaq@P!4SaMWgPw=l^lT;`^cq+)%a-o>4$b9jcQM=~UMI6E zV=myg`?7C&X4P0RU6TS#Ou6A1&PmI%Jz5A9(F}VFwo}&hnqFKtG*hk*z0rV{311=% zeYf?BPXkREFHp3}m3F}*!;#6t4gI#RRdlPYEqRV%b z5?=_7m&!x4vEE~9lVZxh$RXKiVd2k^vY_ zblaED?lcegm9EF9T!d5)W=RNly(5uBy|*{(X))^uC~t_6;t#@dFtE_W;{T6=7}C34 zYZVg<-OgN$tJr1nPS<1;C$?M4i4LcPtb;NSthi%CSTtMAN)o;Ri;Z@l^oUJrGZso@k34lhfMTg<)sxi+PcqPjw2- zo429|{PnJ-Qfr`sB$5tUF8sw)ir1|48L z)1XOsFdjxNm0j^pB;pi#8uV*)@-!6Yec{Oo_C9KrmaQ0iM0!h4(vA0iZhrZg#22R= z)#G^7)F#hH&aQzZ*(KX|ipFV^*9D`Yqy2tKwyxV;=b)&5#kt&CA{Rr~2i4Y2CJO8k$Uul=1X6z3AD_25okH1*v}-yh}nAZ2P8P zqt87HmSs5kitf66Q~ghvr1D3eD0iOKty0BsACeuzC+cXi*VJ*DKH`L%cm7gBZrwM$ zJeB9XUa2gZ9{-$_mIn)l*!`WP{0n{WhzJ-rV1`{=w@cH(9UliBU+8n>2;CpRgY-@F z6ZG$G^r4sTduQGUJ3l?UZxg8=T=@_^*Jd}-OW*c3(Z}#U((fVjH)_~}PttXm?jVW zL)H}bVV3U0DY`K1r=RBH5xO0~L3+yvu|~+8&)D>b=)d+J24?pC1KEFJa1*%{8m!aQ z3xvnnO7yWh+c6xX@4)>17&JUao;k;yvPh1S&*Nk}fWr#A7Q!AP>|u^Ql7zh{5&K94 zdyEh`n?N7V@F=yCERG2lbBe`F#G;Q{kGCk$KfAMc3n?BJbRLym>r#wkUP&>I&j=~F zXvfLMMIOfqvh{}K8CMh~vCj%(Cl#@`Ld3%I3<+Y1I?W^!lR7mMI}#x#XEz|mNzV)I zDG<1X?s$H1f3EC4wx-`GxsBO?*~xMe)(;&a>yx%=8Di7udvo zDg2!ba{J1PmscpgUlmm3P>0lFrfy`vL=GD=@OP2`e|L@ENf|yzPtV}Gf-R8@9Cmbi zM(sqK7^Cg975N$3g|CN@N9)L=RP8<1gv{SbWd82CI`Z>`B{_T^I$@@;ia*djOD%qf WKjZhbHQ%QHJW;(sL?9M{V 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 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/mybatis/pojo/SortingField.class b/ruoyi-common/target/classes/com/ruoyi/common/mybatis/pojo/SortingField.class new file mode 100644 index 0000000000000000000000000000000000000000..356a0463f9bbdb146fa25620d1fc6828791c1604 GIT binary patch literal 1073 zcmb7?+iuf95QhJaliCR>X+tTLQ_pP@0v6(M1LB~xy^0#CL6NvZj-UeJ0G?pKgaX!!$YVpBO%q!twoU8^6gwve zhbQmfbXx+|_Q?6<*q-AL?QR?@fB0Nru@(4H?D%od@g~y3u0Sd6IXLXLEbIvsJ}Bu8 z=!0+|LxF`C%2)AA0i#y$3FKSBKnhga%9qEJabJdSoxaCRwH>&Q*K>k$`fLp= zqKk^xb7brPN2c%VqZrf9z(-$zD$mIhKWaMI#e?|d(J^D46uy`)=_sOBe_{ycY%pB2@ zZ=|DNU@eJhml-{#7xab3cZjb^S(UcLHeg-_W{ivR!s literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/mybatis/query/LambdaQueryWrapperX.class b/ruoyi-common/target/classes/com/ruoyi/common/mybatis/query/LambdaQueryWrapperX.class new file mode 100644 index 0000000000000000000000000000000000000000..339190548d46a05b41b445d0ab14126f2d6e2bb9 GIT binary patch literal 7228 zcmcJT+kew!7{{NqWnH^&4yHv^a4N#+I^sA8V-pyYqvNu*oBs3fpMMe24*Dfc>*=F3 zeM|~p6_Qj;(-f(Eb16+Fy3AJ^4_A1Y=0WG7%tM6-gNJIG5Tm29_l<=KK_X*GGKu9~y*s8TLWDX01R zoT*f+syR6*QMOU!Y}FMz5JBG1`1s&n2yNH3Y4zx(Q>I!|E0#pJdu(9^#VBi2M%@Lg z>h&5}O;xrGL!Z_xxmK@M4bzgx4%I6~OEW5iqgRxximWS@l00UaTBS6&B?1_jT8f5V zFIOaT42!~o_*_+V4nqJ+qJ2?KWT)k9uZ>wsar(GY6@g8SX{Cx{)lEeHeq;{{@j>J$ zeslxerd535{US1)2yfleba}+kb=3)a6f_PoqElB<3t7k7g<>UI7y4kxcKWnWbpH0tMp5Oe9 zLP&bHrMwJ0H^JH}{!ZOivV1;lD>XdR8=mR@XPpJrx~8iBt`NPRL|3qzzOe9_rE&s2 zq90ogt#&a4thYUXjG;&-jgMg&U5ughHB`S z{wPO5^YYj*r&RO6?7M2MD3K94uu*&DCy_M@cPz*Fgs#+Z&N|ZL%f96umy#@TYzC*D zpCu3`*cM2H6UdX#Omu|5Yy*OD*&oddR{dzr+TZ1!oSVaLDTyhsO5==l3qrS$wWt>@&g*YpWvUEqpjVT0D?^{r=NY<9lM-$F|DP^Nx|5;1bT32WbRkJ!@XeQe^Hqk9 z&}4?b=HVM2zUARN9`5t-fWA-C4;lKAe#+1Yz02z-GIX4N&d@1(U!q-1vv{&C`mMr( zj`A?bf=;u@6HJ7CX1KY?r*N`F8x6B0*Q)#tbjei8>NUfhmhH$p*>x6&Xqwe0}p2-MJyrcrCC(QDrq4GC}wXTqr^cwz3SdTw8h}M$C1%|O3cfBa<{n}?VGHlDmt-jkmesV*R?_XfCX_$`ZK0fi=%qZwv*Ij%`Pm!9ik($` zShW?`agtV(jEfk(g{gkpCYZj)BbRB_Z-S{MlxaPfHh`&nNv7@ii^yS`CvRk8y6vapJE@n38i06}<1Sw13t@){;X44@+XN^b3bYMC+X1w5Wq|fI z0cr~c8UWB901d4S(EcVsnNXnp02&6+$jSf>Hvwu71v(6%BLF(KGC&8K0Cj`{odD2D z0G(bLAe<*el|6={1nQ&C%)w#2e5Nh|W?g zO`zLMqMyEx4toJz;seytMcaa$qr@xAey=S1>7ZSfK%hgoiu-zOzgM8V-Yw8o6z^(p z7bSWaB|75uvHMW73NYDx{4uRLkxyXUlK}O)#2j^T^LKU@r}-DK!xX6Z(f6m^L9I7J)cP%mX6WExFz~ zda0SbmVn_r;~0LN&I@@5g*+4s80YBaW+L5OoktYG10p^G7c={nW-=Ew+YA2(^dVvw Hql^Cmmr0Ic literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/mybatis/query/QueryWrapperX.class b/ruoyi-common/target/classes/com/ruoyi/common/mybatis/query/QueryWrapperX.class new file mode 100644 index 0000000000000000000000000000000000000000..22c247194b8e28df306108dfde0decb71bfd11aa GIT binary patch literal 6027 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 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/mybatis/util/MyBatisUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/mybatis/util/MyBatisUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..6d94a0a7671a7c4e150a9d6f16e92028202b6681 GIT binary patch literal 5527 zcmcgw`F|8=9e&7hWhO99J-@VhWau6F4b%wu)M0RAk{aA%|0K*k)6~>eb>rP_6js7oUG>?S^uzM=S_izJ}Ybay98=G zIu8rfC7d)*Tas4R9LZ-UOn1zfuzAv!bW(8kyeCfO5ic zGFIBj&uYloc~3Xb_-59#sF$ykR@T#V1{W&jk(3~{@+g!f5Q?e@c#cx?jarkGly`~t zp1!f2F{kdEIANyz-d+D68~Vn^dUw%~I~T4l#2UmP;n+68(jnLC$Y#%ReJ0bOW!h;* zzrC^+=92WdK;N~+t-UnyrgO8&cUwihp|%lfy)|cU*z{!~9!dAr6O)U_Z+=CZ+dkoNi^armGLh zXI#XP>{m@z99rUP#8yEjP#-81xGHH?KnBv%<{1Go#AX{%v1VFh(M?*O&$2!==jf`+ zriymiF0@J057KhP$gmoBbR@H;uY1S!6W%F1XSg1H>!qg5jG+f)Y!4N#je0-42G&0p zHydU^Dzz#27$X#Pd$yl^CM$5Vqg;9gk>;74YkEOP@3$?(>n-Q%tiTr85h|nYY5*mW zAh4+dLzy1NC>20e@DMLO+sI6$jZIfBtWDbmjy+fZEom|HlwitnO+91!M%wTVeV?0V ze*0!d#w^}q=e=Kp>#9Vdf{)YiQ76xW9kk@ty(E7BZj-^)pkup+7obmIr@XrtSj=;7 zRRy2W@JT$Z;8Pktjn8N}hli!?&nozwhR@>*3cje}OZYM~ez)WJp6?pDVbh;-(q2mg zzS4?ExIEg5qZ+=7uPOMthHs#tU`E4vT+r|sE($CT@-eu@WUXlUCLR}9T0HG{U1M7Q zq~Hk+-@=m;!?!hj2jA83J$zrmB@I8oWr41R4=SwKV@zNTKg3fSek31HOA~*Lp9rk0 zT0^smzI<*G?UqB2=`yc0{1iW<_-z_~j>{73FXls57Gln7hL_UtOUe8z{940raG7D( z@Bp6C@LN1XcNf)%Q%cEEI1LJHC^utnMF?!KnjlpYSYS(4HWJazGn#1g`6d zw;tK*Cry{tu%u{9bSTisz8_7n99-F2n<7D@)*Vdw`i{MO<@L~!%IZ^j-*IdXrmQyB zb~FX)3{|f8KP|8_>QQBL)9nJwidk0_z^1{>HjRwH>RB5C=I~Sw-~6#x z#qaQYfg9s3Fd8A_hDS&8qvQ{?#K`4rf?8K1_;@}W%yRl*w4s~Aj4Rj6bjcVwfm`Bc zl?#XY~+=Ns;l6%ge06cVzJitS!Mfmy&Xn$p1{LIG8fZ3hrRYIfw*JQeY&` z9o(PuwAqDx9Q0Ss@pA6Ua|6Gbn1M1sqd|#|X)EzZfODSgV!GDSL=My)L$T{_HbeMx zi#J!p(sD?n?VL3A zWH49oCNz#4DNfZa@FqD4L~jcIK`$j8=VYFRs}0G~H(?u|XVRLd<4_}c;hnn2$fj-6 z+Y~ZZn3-a(EnytvS5=&R;Xj5d=jV;*KqNU~*!HO9o4pGDDX_WBKb7oZUe8a58`v_t z`Cl4=Rp`N1uG{!lgXK`T=g7n7O+3-L3a<5CGZ2pkB+uisAvkQu&D?7t0slqxe7-H> zsPsY#lFZw=O6GsOxrV>(zVcgwXSeaR5#ryt zRlz`H(jZNGDfW;~I_7T5uj9JvB3h+MOJ>kEvULp&SXzs&tpzNTuC5KRwgr?o)6F}u zj4obD2TQkig%U7?y?moAybMXom%Cw%5QoNpuI&?)lq4Ko5F&rXo zJ>7gbNU$qz*{I+M0lXaJ5rBsXKt{OhBG!y|U&8u}xPE*FYpHf!0qfgt;G>6s1#CEn zx}F(qETRZA@jxtWU4cG|ql<3GD*~5v;g!S?hUO;JJj;035`}_SNh~#uGGKU?EV0wt z@RS6ts%%aX*l~)HUS8WK{dHpjTh5}HyN>hdJO?%0brk`sfp0gWng5S0mdjckgFz|M oDGF-Z2Ng`v2PwYRAdSCrRjKCB_&ff>kh^@B0rW`YBahSn1*Z>c0ssI2 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/Arith.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/Arith.class new file mode 100644 index 0000000000000000000000000000000000000000..93c58578d3ee5d9edef89aa92a01ad7ef98e5fc2 GIT binary patch literal 1944 zcmbVNe^VP(6g@B5WRqq2ArvGPF`{6AhS(Oh{IE(%Xd?kC0Tgs(Hen}hyV=cbb{Xr3 z@B=u0hmQO~J9b9@bjI(b<9Uw^U^5 zB3CR<76oEQH%Ul2x45ctHB@(adz>FORPS@^V<9>p(SCI^;lda*`{M!BS;qfBT`p@CB(|%;EWx)RrUVfB%XW<*RD1JZXGm zy@Yr9>JIRA6S$6R802$^s~FydPD{P19pE;ff;)++XAr-rl(%T*s|^BZEFPg}o+jaK zjB=Hx4znGAh&f5d#$O<|JMkRxXVB&~xyW@EiHw#>oPfXNQzphB8}vYa3`WsSsOtoi zO{NG%AaWn$Q+n^wi!o;y?{lRwvkMdS5~T3Kt5eV~p)gJO8wAe328G7$oZod)4R1xE52?YX&h9wXx#P6_PIN~42XS;Ftqy8G zqx%KAcKgj|=>GlUgcBDhoXjMS(Q|^{pI|5_`81QrBnP!)^qt_1&XMD4@S_U4O0wJN za*9%?vEIKhts~DL zMKZ%Y`3e9(vPz=nN3J^4`Z4ZC;U# cF=qfbsn$3%dvHrBSdxrG^pMqUx-rcD1CI}5Z~y=R literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/DateUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/DateUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..3010a99bc2bcdf124cfea1c270e9d4c363a36d16 GIT binary patch literal 4979 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 VIE=HBoFhp)hN~pY(5^0w{|8cd+KK=G literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/ExceptionUtil.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/ExceptionUtil.class new file mode 100644 index 0000000000000000000000000000000000000000..cd62101e038192fed9fc9198093a718877e46fc6 GIT binary patch literal 1353 zcma)5+fvg|6kVscl#l{)69EOvwHJv9UJHmKJ{5HYaqumaSb}ZSNm9{Y@dtdx7hq(@ zS7-bt#&Mmbw8g>E&NMk^pMBO^d+(K>zrKA3uz)oc9hg=yqaq~VDHRcDwek_5S~eG2}D-yT~nYlZ&l3awepVXY#KWyM!NHM z!64!7l4g&HiNRT$SyLUh}MytFL`5m{rfRE3diD3G@_AuMt}} zUDqfQIyIBuH$E78$*2_d%{|9HkQBL8OLW6?tV)pxg86+d2hDVWkb5A|9~@e?9^853 zSf1(BxnJ3~x8^vub2)cI+U4A>yv*l!&FZz+j)#yj* ziB+TKl0W-zBewcYwFs|1*KJzVaPNz>0 z&u4x>Nq<2kBUiERwnLJr>sh&OKg^y$lZ0Wvgd_nejS`1g?=T5(Q2Ho$V;I3Wl9=>G tj0W%&afA6$#z!beFi&Il7$qhd??L1Y2?gUT{3~Z@S1`e(NgjnT^&4dGOY;B# literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/JsonUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/JsonUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..3ad60a163e23ead968a9da5fb841519f96a37b33 GIT binary patch literal 5146 zcmbtY`CAnC9exLdVTTbSL^N7qb10{7BIc6C76>#F1S6Md8apfl3@$r3I~xUJuN*zn zHoe>QPVXaaiMD9#^ECare^j5opPAX&S=be9pGW8T&gXbP@8|e_mw*4~AO8e!3V)5^ zQOs(194QrZ8cfW~%Ys}MHCRY#Si)6#O{-WI=}ZmmTCCuj3P(d7vKn0EG`xoEB6dSA zt18wsJdAaDd0oX#4WGeh<>hm7`Mg}dAbM|TXvMmUH*4^ghGx92;)@!-gfFZ3iU_U8 z@D9EzPhV5oGINQsh&a{vSQ*Pz@MPNq>9tQ$bHJW671WPd8FMtZ zJZm}=#%!7^gqbqZQ-)*7d$6dx_AqgbyN;D%D974HmW1N89W#?MOQ&nE2!=8W5@i~toKXd*9!$dzN`X;^h6tnix_h87qoiqG!ulM?SXJu-7Ixk>ZWcC!OR(5cC#pP%^91VG3(QwAiYdL>rvp`~wY0P;P zlH#pLBn4G0Z=1n9A2pI6om(@~6B)05Xg$*32cx`&gjF9LVUo-w|-Dio{^EC!Ac$ z9kA1B9w*sX0z|>ZXyE=95Me04C+x;yv1P&>i<-{oV+Tva9E<|2sM>=(ak5Gw_#1|1 zWkpVsQD%rs>b4e3NPLW$@Q>IEjz)3jBi^dXc)EAMK5p_voVM1ysWoHB1h_k{uzF`7Ra z7?`G{$kyJ1(y@t(jgiieJYf|zcn+0#mLIF3yr}Z6VEP2Vc6p|os2=10ZvFxkB(-=N z&+s0@^Eh4FN!2c$bkoUGJ2>eHBe@hrQpZR(QO#h~(2qt~kzigb4(2(2eJ1pyM}nyf zgGr*7>q7baWVMp_wvM}~c@I^}hEUhJiP*-uTc~Z&JG%aky?byEbu(ReP%m`Gun!Fw zqNdQPq9R56%y-Y+GXkj8GhPt8e z`fa{trPy~8!}Pe7vVBp?8vYN;E)w7j0lq@YK1GUNC1qF2Q1%i=80XHEouj=ZMg!RI z2AOUN65LoGEm*3tFv0AXv-$is&Twju!U$NR^;7u} z{1ROz-x-#`HHtYLPFc3R8-~&v2_+eZlH67%vQj9MNrp0oX@(|c;*csS(PrU$6NffN z{C?^T`>8YNC+VRYf{l(Rzny%#*Z6gvwqGZzn_&UR0*o@IngfjD|6*zXhyQo~Z_xjn z^#2w;zFpS8)MSh`d@1ms;N2hQck!_g_|YHctR_Cc*y<&C>eG literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/LogUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/LogUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..c798f8178133b7f32497be7ae8df58a0347e9bdc GIT binary patch literal 668 zcmah`O;6iE5PcJe*aU|U^VRPV!bgY%zHkfb0i<4V+e2EUsw&dPvCM+wjT{?^e+8Ey zfy5PwABC7rQUxgoe3*GN`)1z0_0#j+1Au*O7cqkm7CxHzRD_}7XPv+3e%ZpBF3Kh< zCe{gs0~yQoE5WGkoDlLysvQvKJQ)W+vQ8^Veu-AZ9mi9?h)zTzbvbaf@e{0YnSYePJM#{e+Y}LGTaVgwL#G(7MbsqR>W9LnAa%OLss^5Zc2FG(83xlSGdB*mTfCV^QO=A&D+|k3QgJnRq{)mE;JHChU z=N5Tqs*ggw50jH)ap+_En&}4g>RHZuX`YeE0%M6Pma)QTs{<~Xtn!Xw4)0$8eLIqe literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/MessageUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/MessageUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..ca75c7cf9ce266592cf33d40b74b9a09bbd096c9 GIT binary patch literal 955 zcma)4%Wl&^6g}g(F>zhmrY$Y+mbQ?jh!Ij3pdc1RLP(^r2$Th@i90n@{6J%e$0zVN zu%fU)tobO!9gh)4N>RP=o%_D`ocr_Fx9d`JqbXlSwj``k5HVWN8nxP{jK}E2CFNw|J#;hV5aPC0tLk z4~o+($x=QkRGjlYndc&s&yCV&@Di&t-0cjH#9P5*kw$z}Xq86&mo?MV@sSLRez#_- zqG-gh9I-vsCv-#uElsJ#Fjuu7WfL9JN;|_$M?8OHx*uwh$ak53O$(%jd|&*T-3HU} zzI{DZ293q~X1B~L0nVVs&^>iY57z>0V#~vJfa|yspo1>M<|&|LSon*cg^D_#kZAYg zqjJrX2_hnr#RDl)qOMNFAQpN459k#S47>l^OWnSc^7209z@qmkiw~q`Xi+2UU|8-{ zbeh|!!VRtzQlZK4xIWIwcVVfo*O_uD*g%tR-+T_3p@;rHX`Cb@In(oa1}#KX}%dwA~Q1zvKz;%G3Gj#Q%T zF+;x9K4&QO^o3;D2~{Hd*=!`uXEBNisfRie@wqU{)qlZ)ov4)IeHiIkU^2Z_0cA6t z1esNF8mt|@xV+0yo(q%8P$y%Ow8E)ah#(fpI2c-^l5w}a1vwT;@~JiM><3(i>d=bl z+b1z!^}_L*!5iw#MDk3zhW)L>KRA5_C2)@Y3J&mw_!YcGjp5z@dndC^1B202MwX$r zfz~56q9mAPR_mB9!N}DWxY6AB)SF8C+MdqlcFEA#p1wB`W+=aBGKplj{gr_)M4ZXN zC7LMEqq#2!dMR*PA5j%hWE6{?8?fuucA3`F3iRMnURe|Hm@=S@Dr$r{@kfeI?00@4 p_X8e<{7)3v6=8W|{`0AB`S{p@PdiWO=g8GgIO=5bl*$~Q-2?l&ibwze literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/ObjectUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/ObjectUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..50741c33cf63fb588733257884486cccd42036e3 GIT binary patch literal 2416 zcmah~&sQ5&6#iZk81e%s&<3#(LZxj;peeNe0vf?o6b)2MiVFT3hJj9<%;03kBYQU< z*KS?7>Ba?a6kM>xbNpBQ2i#cw-b^5a1j;$gd+*-+zWd$pz5DX#v)>N^+=Z4v3eyS9 zU{=oVb)b~M8<>-y`6S*%If?stD}e>Poy0p>RPe5X_XOH!4a;y#0@2*qsz9u4Z|VZw z6~oe(z1oKERMicWmfni3sphKc7;@b(#@xq7U0}4L*|nnM*}F!Ovzl!cJ=ZYn#RnTt zbj^Jv=Q9GGnrU15;q#im`2}uIEtnx!LM;h(sXgOesfz=O-BR znM7;qHv;#=QkU)8j_Sx`Y@qlD5VD4K4w-p7sP&{QzNz#X|CE9yfdZL_sW-!d>irr> zFd*5vsc)&&esOErGfjb;x%E(rM%iIj4HFp|u3qDjsg9%W3iP$u$Dnm)QgB;f_#8oH zKtLs~f90uWecmEkF05K3Ws3x3YwR8>Bal4@p6GrC1}0W)&(ZV+LpIf=0Mu`gTT;^$ zI?<(IIfVy!KZOtQP#|J#Dp*OOibpA|;zI==rSLJ{6?~C`iVfaA z?YZs9+|4r#SP1ktO{ea=z~!c=d6EJd%_=_jT-!F;_m1v&eW>YqE`G}k+-RX>%k%8m z*4?tH*7>UR<-!rZkxet)D&@nVJEe|nc|!PYDo_fi4G=>2YQ=R7Yx~rza;j|EqV)#W zs%m<&A;W-;8xcZ0y*=b;gvuB8sxM{h7mkmRs%iQTAE7HPGRf_6o-MmquiEFtEFK@N zpJHD21;uPv>lLF;-0K&#mcK)WvGT4a&0wQ^NnO>j2-cF@8b*-hPmla-1EoV3zujCF zISOgb{|503ClOrYx7{~(qlfcU5P)9vaTUj9Tw#>le26Gw9B<_J5n0OrifC_aA8pGM zM~IJ4h<&t=A0bf~KY$X!_#Pu;K6(ZU=L4j2l{kaM9pWm>?+8ZmlAkpnAR~=_#xe(9 zU<_|9V_u|9GU(_2L2?+PKg&@@rIC^0e*ExBe1`E^T2qMpg8>C23a%xXN!B~cnyxcG zsOfu-5!wgyf1s_fSKv3lmoFTkBZ42HV7E0=~mIK?8z6aFj_$@|D8jy=ZA@@F%1T zLsQeSf!Gn+2V#5QM#a;A{}TUaIZ0Mi%zTGBPGT6ZoUrUc2X0YKd6px*61O=XvkZKh z6W=6#S^jq+`V5MKDIt3PCZT}p9a6oEmm5^qIZ7>)g?+^H2S`NlGndiO6`MXG!7x$^ c4m;_S2ume>fmG5rNvVQ_uX6SpF(Y{WKMk5WD*ylh literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/PageUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/PageUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..02dd97f001a7604ac8cee959f11e3e319b0c981b GIT binary patch literal 1825 zcma)6?NSqG6n+*~)~rhugtmOtB2-Ai$5u;i6{;;-H5!l-OMh>|J7Jy8ZrG2`(0lL# zya)gE4`uw}1$a@NKIcs!VUnrMOwOM7>~qd}&d0kS|9Srbz&c*pxQe2U5|(YOU{%DL z2-n6pxGmZp8{cAGaNpUui|U zL$#ANYbt)_)dE_^D^cAG_PyAbbH`|=hkjyUzEY2xZk$EOzDuncg>IJmLE`Ru4fUI7 z8wPSo>cy#84A?C`9!E{j4-G8!d)K2_xq{IV+CYY|#6(sGu2)`re|T=-g$=hHrmCT0 zzP~K*tNx$NU`6qPinor1co$ktV?S&VX~)V-qOe_BZ?LT>3X~TTGFj+%ln&+U1jM_{ z#Doo%R;4ThD~0|tn#VOS^%J)hy^eGxyI$-yH;QFO5vuxVp!mN%NxT|maa}#}WhQxH zq5HcgH9EM4F$+IBcz}l%esb_Lwj4YX@fg2Yc;etGo;fJvxrG-FD%f`LId&|(bg+wG z9n4FBJqJ~68JPRjR6CG{pB`p4J+DI*v{bBXeWq28w=YU}x_XVOS*sOE3$N8OKRA$@ z_S8|P5}JyI0nO?93-)|FZv981E|4wc(B!NaX_5)L`_WycD;-dKo5bbi*+NZRa5-P8 z`M&h+JCGa&ZuKaTF|kCI)V-GK#G={2;-Ee2;dGv0lqB|g*7hRx?&6O1)x$;g!*t&Z zGA<48Irr~QpG3(!1ST*BQG{Bd)=WK`(ZX(Z5 zoBS?;9mW{tI8QF0hFD8~!}yEJ5Ux{>Xk#7|)SWf}llY9M9Ph%J^IHrv9D|%<=tb!S z!xaj1o8rM3$l<_YZIyJ>E~qWdIFhL*s2d_iprvzW#jW_0>> zrZBP0H&Z(OBHzuBQl2%X``$_ZZoMJ$|C1C}7kLDYgK$-eU}B!}oB#mtbvL=QE7Kswas z=qEq(6j!7z{Ag3!^aP6ZDlygqzV4tmX_sM*v$GC4n+H@zsT{tewRdu>nqPZzo!i|8wOd0Gsf)C|1CXq8-U7Vz4wE(J&Ojum(GdqZo;z9x1t$j=+(l8$m|yKh}gS zavHn{@=+Yei6)%HF%6?p%$JDs@u+-!CyFKbZWQ0c_gkp(w0wK21p=pAP!oLqP{Xq= zsFgEy@==e|8h)hVf`*F%4foiN?QIdL?N~J+P}l1wErI5Q?O6Nr=|L-d&=^c{rX}H; zMry#w+VWdX)_KErPGDuibkp%{-W|2$d`r7dJnz}5T)f{h^I6*)JtV)n1saDe?@-Rl z_9Y3}u_|%YIBvvKhBFjTxXuta2x_`5G`;hrX=Oayb+|UC-!sgSy++1YPEQ7SJGepO zH7RVzNE2j!33|VmwS9b@n-cDj?MM~^>$qzMb<@pS@uZtJY$u+kH&P_0B* zU6BrZsbEJWQ;=zyDMIL}^_`AM6KLx8DhwqNH$LbSif``?^HLa+aoX_AVJlZ;d}kTP zGYPGFnDNOPCyH{?b`nb#XEZz4lT6!mXKNVKPYfyFkXMNl3*oD@nHvtPOa{$#9nU)H z33OMnTSi80E~IO3AWVZ7KA2~m)7F5Uvzhiij^lcM(3$ni%Od;Pyf2?h$zwuja|xZa!;TyKKKXloa~4vaIV^i92pFgo-{^P;zvaa5Vt9zdyD>bX<2{TC z%q<2{<_;I%=il%7_lFqziC$a}dR4jLl(`#}NbVmzYMCtEchByGSt>W%bdg9OJHe1_ zf!c8+mACfu-q6<37cO~f2di~Qxb>)R8)`RG8|6)DU3GS)9bZ7+J`V7TVA>fYHT{kf zSWOOFs{>akFT358-|M5)qZ)?5y*G6^senj24MC@7qKAFcJ8We8tz&sETHRq^7Op6o zx+@5*sVEq7t&)?SzA?qr8j!K_Z>9J1uINWo=1qLRe;t(Wi@Haq`s}p8DOW05C1TxC ze1JbryJ?l~_dB?nk~i^jfqEm8NsY2QOx-Gd%p-Znl)XH@OTExrwP_95GDK&C-;nbn#2WU#K(Tx+8g#*ZUmD%J!ys!?bb&{|HA<;RU5* zx8WpHR<1n|?4B9S&`gQo!O@Hr-hT6ofy&us!`Y~idyQ18-}bC-4Sy9_F~dLQ>=9Tz z`#vmVP*(6q7a)IdvEA|6!LK++!Lhz`0^)tX)L<2#4gO>+I{B^#8_guAdHPjWas*9bptS(*Ynp0KRX6m|!y0%o+#jA$W zWhx&on#A18XcqVwF{NOr4GTdlUA?6PX z4y)sd?8E)s(?K1QhG1NSW0>IB$Ozw6na5J4f0-KZ2=Q1xlgG2+n%x{me1i=WKbu{`wk31TmyB{U&(T}U?tF$vuC z3hBMd|0#Nn_26~p&YOi8)sfHv-%T<`5r}_dtA+={u8KuvUm>y5;x-Pw7>ywG7Ynd5 z+y`+8OcvRhl&}Pm_AMY^9!sD5oZx*L?VzZg<5@h9He3kdF2L7tm>807Ekqj?u>UCq z>y-j6aL9CiXG1s|z9G$~ID4oPFIH|@dZ{2>xzB`zi^j1FE8$x;?i7uClEyt1!YyMQ zd;QHB_jm|L8Ak^hz7>Nkfk2c%AK}lqhJ`#h;4yFD;i=`}IYlPH+Zi6A;O%KVqu$OI MR-Y@Zp2PY70$cfJQvd(} literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/ServletUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/ServletUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..1905c835f074452b5e69194f3fcf87350ec1994a GIT binary patch literal 4907 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 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/ValidationUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/ValidationUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..97524d816f411acffcf94778dc0f4b1ac1d6aa80 GIT binary patch literal 2262 zcmbtV-BTM?6#p#=Y_eGjjX+C^A8jc@AZ)d*T0;3S2B}R_B!Co3S( zeD^<)Cm((3i#pSy&gi>;ljC^q-3{Fk8J#f?x%Zxr-#Pbp&pG?YU%&hY;3iZF{dix( z2e>U^2H80JFe~8><|NExfgKkU_z*b>A0aPdNrHrrV_0T3cVqZO#61xO0g08xwYB_e zsj|LGr{TgA^_i+PJ;PKSy{hjkE2`(|j+GUVY*}s=3zbr?v{=jw=$SPv!{aj7esw!pAF|2BKVv91W2L{Pby@1}br)p1& zY9mOY+i>~CBTdI7G7NXKZqnVX$ipXi#G z-42Ozrs}#`#!)=*wh4TMTbuz%cd+Qs$qE> zhHY{h$+pi&;xAm*J>qnz{b>#wt>+o+fSpaxk;A=qh}q6NHl)fa%42eW;HXYO{r`*9Zg^2bu)0Ps-_vUG6paxq9o%!Rz!R%V-+JZ z&Lb(JEMpDpA~s}vhD{l-G2sIlqu3I$E#o0}1Y{UW+dKXP2l;(XZ?JM^;T{QCel%Y5 zyoNh}xaTzv_b9!I8AaL7sF|-8Gh5d)w zxU}tiR(7@@W-8MJQ1Jz$J81!9|EDh_90!D48s<;i-xqA+E+?TfV^aP27yy*pqHPv0y z_dNj@US-iXoXDoGdRk4Vj=FJn$WYn(?nU2;O1c9%P@N!C=g@?s@sjFz=%AihJKqWi zKYIv-67gH`$|*WE+car_u=$dst%@F!bsq}OxDb~H+_t*{=|3ajy#V(CqaaRoTC-w7AMC9UJrrw z(&-J-y~%Bb1ev^bf?5txQ-m5h!oZJ+reC1@1$vGkM!!ci@+yQxwxG)-%7N^;9dN^puEEJJiDJW$$4z!(K)zZAwX!P!nBjY*0Wr-z*)j|iqo zKg5|}I^{%nopg-i7=%ADEaDx!OQ$Z}Aj!>>63B2=;Cxa(M(>oWku_$rPbydOs= zK9G@#BZ#bwTpVE`oIyU0%eWGwx)4VZvvJJfY8=;aU2Jb`!A;x}+7D&ij$EA^p%{1-psOkPfay!y^=Duvi2aAS8ZD}YooL*_$?!2dQG>-B?J@6ISHXD zW0CPYGI~wRHL44mSyUG)lyqf`l3JNlOPP1H>=j2yxBGhon_4~XabY0W-l3wEaT8ivPrs59UwEh zWiy;qeHp%4)h;b*CiRi3<4{7MFCkOAQ(<)}HyEvg#=uYx5Uk+5xpQBA@r(6M)24uYp(8Fyt=6jV`Dum`(k7z*lG zkzp#ZK%<6&d$_OQ0al5%f{*b~!7l8UaQKbpNX91$KE-DWKF33Hktil^)gOvnXX|y% zoKh|A(_8oZlM24Tmoy;BUTz@apkbC%R$ZWa$5gA@eZ#!#jIFgXx9r^p4(^l^HAmE( z3kt+&i^gfDt!r#thwy4UlUFp=5|NHF)2WJTQMoPQO_npRj*vsqu@T3!i}Q%f2_D*$ zNWO9ec5WPRoi_>4}re{{h%a;V+$|UouXalFB$O)!~>H2G4M=Gc;F7ih7C*<9q!jZ3EZh@R; ztFd4?(<6>e`_XS=5Mepa%*o_o-5HKSe+18PbFu3m7wqSktw(&2(1#t^36hVy0C#kYz?8uo_&L$oyOaxvUELY4;9UqlXA2`T_+=QAah5-cjT-)kLB@F`fcFT-xh5EYd0@=G28@w) zFgiAY(d~lq?Tc9au?dUI%(lQ9W?AJ`)_9Ew+(4H50&e@T=xbtew28%R6N@Y+JS@_i zun@Y}VKLN-#U!Q(-acY8NlrOj>SuSi4v5DZx_)p_+s~&M0ifa}+T&WXfHY--br-l2 r?YP(+l0nBF*xI%2DYpNNj;?O!9TNj1IUI>QZnpE-4oWUj9>DZ}bH;+x literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/bean/BeanValidators.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/bean/BeanValidators.class new file mode 100644 index 0000000000000000000000000000000000000000..89e8abf318bf8c29499e9084e5fca135f0aec49a GIT binary patch literal 1227 zcmbtTOK;Oa5dJoC<2-3en--ezD98oxTEY{m&5*3t`V2-#?mG;NUEw)03QV5!=;h<4(pG%`3%VTlF>tJdZ-O;v72h z)KjkrgLYG+6?%%>?_cn7I5HDk4B2_lso+dc3_Y$UKFxUde+Qrm=ow^ta^l<0uF#va z99@8I-ReBj$e3v4u?>rg%HshZGBjfI{{z)|^%~?6L;c^Ek74QeK~sl9)y2dZ)?-D? zmyjVn+b-hublR;*&toHP)}qs-iI^hT8I9FR_{+Ve^_t;aR6rfX22%9f>MsFGiVU4u zdRcTcx)`$sj?Pr5EFe#Ee(Hb%F45})7O{k67<@{%LApwPbL$+5B=#ETNHUxv z{~5-c#u*mAYD@zfvl#joDfKeRC6uui;@77bxJ+98Rmi>)=B$v{Diy3F)^LTgT_wBL lXs;61=e9E0Dx^(=eMi~EHC(5cK_~`pkd(kpk~c!>)=!OIK?48) literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/collection/ArrayUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/collection/ArrayUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..86b0c3d0fc8266e2ea38aa5e673b6727021563dd GIT binary patch literal 2714 zcmb7GO>^5+6g|&&qS|r3OG+Bj7`LIe9Aeti59$yIA)yARh1xZ=;Y)FpRKb=!vRufH zEkA)J!>|u43^Qc9V}>8aaGoUFO0b=>uqEAh-#zEvd)}4){^yTl0B_=}7^1kVAQ!_5 z-c~Rd!z$j1;a#l7FoySHSjYPcHWb_#!6rV4;U4nR_d^9A2@J0ors1p#gj4CVKxEz8 z)dj{2hN<6g)^>Eer0rB`nJQQnty

    Lq7Y)i1XBF2rL&WRxN8ctwSTryP9QYn~qU! z@S$4OD~>_Knr&-`kEAt6P_16q&0T@*RN+8-rn&Cfy{75;u3KiKS<~&@_Q2Kj;AO&C zI|qapm>(ENV8qnF-Kgp{-E^4Fr8DE&dZSro+?kU_Ijpw}lN=^uiHHdo>0$eGQ=qnC%msy{i&3%qF@{HX^(KwCNQMQ<1aAXV+U6ab65XP zK;+p(49P7(s=IFkG$a@Fsu%VE8$H5x2I&q4;c77rVa&eITeD_^ndtiBoy!+)KW z_)ZXLchJqM6D@dBytWFg^r0Q-lIPk>0C<%z{3yQ)fFbZqV1oaXe9H1Fq*eV5;#b~; zFvb62*Eo%7-p4%$oX1Oi`T;YzfUvvyX9f$?mQ?@3Q08bV6v-Ta-9q>|B3nygwU`Mn z(bB@u(ha`xk%me{$)>xO>8ver&4Yg3gFzxP?*N38X)X>$t(q+|KXIn|}Hpx?H=(l>~nggqVw${%v~Qp)G{B{so#$ B%QgT2 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/collection/CollectionUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/collection/CollectionUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..535cfba2fc5a30530763b0cc51e41172ecc6d6a1 GIT binary patch literal 15008 zcmdU03wT_`bv|b$>sqU8NtUs(EhF0?8%dTd8ynjoVQkBGY%D)w$>J3VYiT7fTJ0+P zFc=6ifdHl{fdty%HZ&xpgoYMF8Y4TD(55XdP10A|rVrZEr0;jqcbl~TnVEavX8KMI{W<-G-2G(@EvCPc zPJb;_f73uQ`ldsF7pIFw*jx1X4*i2e{}`u6dEQL_R73wv-*xDF^6)PWR6*Zvpi25r zdHk;injr$-qW@{2D*9jCG07i}bBrtET*)&U$l;m5$JKE<&ha?cup8%D^ergGwW65- z=eju8^XzKc%ni~o$Ki|Ov<{l_T%IQc7lVMl3q5$g!wZDrLJ`s^e-;VrOB`;B^I~4& z@KUBZ&53+InJc8Snf%sNK5;Oe9Ae^|n5w!{nN(pTQ$o$SuCW|c|4@k$w86Y z+2h|HkftuCn!#+QkVs|nn=?n4T3UK=38 zV4{%hYPFh$4-ixi=dxo0Us*BC4hYA+-b_svQ3Q9UMvEZub$5S%mneIg)mq-&qng!M zNDPkdPK=xI8tv}CT6eUgVq>)an#nZege~DopE(4xq+mFvZP6mSqPTB2wvMuJ^=R@4 z)EQ6a;NvMfGcB8{T`>0VSEjr#(kT1ShoL-Gd}))J>d?8^CrZC9sZ1hwWG@VqC}eX{ zCY&*tJuK#njG$0=WRWuq@!b9V1+?RIQB7uVNtLidK_`Gp89~ztm;+M;%z^VBFe0s7 z0gYkMc1(~SrXEmHR+}=RRp~Uz^lYSW-x-ysM5%*qGONUmTlAm&Vz@+1s--VsYs@tP6pVe zGQjxGk{35V*+@)`*(Gj8X$4L!lJJ@KAuh`OQ#EMo0)mjHpzVJcV${1Bi`QiqTi6ZPbBxByOV& z?Ww5hA%PiXRt^7w^A+5l%*d%RRPTh~!1A-kl34tBJf>L>X$`*fZg4=9mr9M&-9NRX z?!6$6@`h+QV&I~)&pV=!bRmVwl91h=`nz98RbYE6&M_+P92+YZWN%VMm~*h0N)IJ-4qu9jd5eBv za(8Aj$y`rbc0;kJw@PhXj%2eV>7=)FX*T^jZGO#$0YYTMS$44s;LbB=GHYq9_WcGE znW5AW_U{08f#v0ZPH!e&8O>T6C+#HyW@4vwdnzZ}ip@3)wL@rfvc31lRB!P!`-pQx zuk?6Q`_8K(gRNvx2P3c*3k4L9!>P=W!L+3elO<}{tB1?>1WFP2oehR5`D?{A%ob+gTkR$k@TYa;lKTzViA=4N4#e z9JOhox$vpZr!p9kg=~hY%^)a|WhoS}0Q#z-~6?u$5|u|OPU|Eekkb16767t`K1`VnX`Ef)AGp;VspXqM~Fvaycy|# zLqzuo4CSwwVux4DA25kvji%W~ptq0Syb9z6=~F7Tw;88vT!Inl&`De74_UwS4B=e zx4YcID;@51c@?jA_zIWTa69k@Cmfq|xx^6}p$HP;xz?p3U+M5Vm)G+Km#^ZhnQ8-i zA+4v^A*RdaZobB)B#k<}(dBD-lS}u|BQ9^|Edb0tE^p;+pxh2y1RQoG@`o^Ex%?j9 zfE8NMfR|;2n7oAPgAVU-c_&|2(!n%73K6^N`MSE9w5XnYg>JWWdK~ZfxV)G5)zfRZ zf3E=E{0|t5xUJi(Z ztmSe>9Ghk6e3C!la*i?2t(M_+R)>iW!@5O2jPMA==rcs7m1j3vJ*c7MFpsr*tzrmB zs}gw$4{XGF=Xdr9w6lzw5-uf;*W&`doP?o|>?;PtkY9`3}Al zMS@`)g0D^yD^W%O?2;B6nV*q@SrliTL6DY6fd+{TPP7Vx@c&XUD zn@yv<%1Mpoov5nARvGXp&r*4F9NSm|twc1k`&DNWXeHWk-^MbHOxnrj#F2%pc4Si& z0G|>V9HRkhM;nb17f?pjPWVKCapXJ$=YeTlmz9X}?N_=YgG39tPqf{XKJZ6sD48jw z z%6w6ZdFxEbLVbm?=mE6{mKgFLWKOS50YPpA`bliyvF(zsA5;)P&-6*#hvlkR@ie{0 zkn^JY`%vsbSQb8l^Ik+73_gxpi0S%iBQ4`Kr67D;Fp|p_MaUM!(A;2{1W|#2nVFRD zC6vvUPcNS`T><}0kJegVMW)38F~^4_-wqCd$R-Vx8NvGa?V+i;t3#vv*uh6%uK{Xb9Zzy@l1Gd^(Rxtumjw3m1GSv@4b#~#H%>fxxAr7 zOEUvsX1Ln<#O8?1B+SN(h_ZnQ+VZ-1x?Z+|yiS=}K~P%}@W@aypTgHRej$Y~VG2cf z8^*rL;iE3!#}Byli}bp~A949Xluh(idflb3$e*8+!tBfRD(2gOb1HKrm zhdQ@LEpqC4=Yf((JI1qH3)+`nQTVZwhkY|99(9uC~fie)6{gd6kq@K z;9EqlpDzV6?v~(NLo8wNcL$z2ab;X*w7o+75^iF&6Mw7J%QDoEzE`AERHYeKSeO=CnCj>N4WP^Q_^(mQn;Y3v4P!?+6C%OSvd zifX=0GjDu_s!srT9Qz5Us3t}efJ&njDnY5KGN1|>Fbllnz5zcD0x{s~5MJNC zb7OQGzCl<`?dUbYz82WmS$19RV_!`l2-#KIG-7GXmD@G1a29}0;`{|%#jfikakeXT z2*&B6y&7n{fc6^8((8S+>kV3nqaxh~(SfCfO5+wP!L6wdf+}R`!_eRky3=S-Zt06L zkmd&iDTIB28Bzw*dRr^JVEH3g=x0XSt&JVv@pqKUNxH=|v~h_?w_h!Uo+7i2DF>zh%(*elX0ZdqbfsZ8`*`?L+0EdsM-> z&%k*M2~k3~QNx)96E-n&E{GK1}yROc0(z zrLhT>VAE6&fXalYu$4Uj5ry)>Xq3qFp3&M*BYPv2HiS{$0+hGHZnvEgig+dEFvr2_M^Fw7yqOakVJg4w3~`^H@48*>q9nBY!7 zg#22e%o@;ZvkJX7tI!KzP5)8QhoJwkg8rz1{+xl{WWu1s3Imw9nW!TI{3yVu5eXR! ze3K8p$p_!$gKttXVS^WX4PNLqc%j$eh5lm!crQ90Q{W#r;J*s+s4ZxD*zURtXzy51 zIYG-z##$WqKn_ae;en!3Vy))nE3w>HV!5xxa^nF=RO(bJUj#k1I?_|CBR#b`(o=k) zdwvYw4teF{Q0x$CK%ktNFl2F)DUr}H??YO6R!t77yVRdQev zy$<@660BDkbD-Pj&`q=r-uxt-$L*n!FQ%2%ZTt&OZ+9NYqDkdC4 z>7~ah!ci2Tq&0zbz3_MjTm|~Gmimhg3NR~Mcu1Sz(QVekquQYQ0+_u(KW$LGimSxQ zMjzG7wDve%1Z-C_9EfI|)0d&b~sXG9zgXr@B{XFD+ zftEnl8b}M~|BXKQjcSPEXIcoipF!nf2a!-|HH7M6_)1f~2r4l(vR(B&qtOXSdWl{( z8s%{nJJ+swgDPKYgL$t3V_Dql^_)&6FX8EB1pLp^Ec%jV+FJUeiL6?*?JJ2aY14r% zZ7-tDg!2pofZnY39-7iH>i4oHaB zW8jRv8^gKX=n-oZ=r%V!Pv~(FQ!XU*#MY1LxeixxMR`wM2Iv_kMnA8V7T&)gCI=7f zA)S$jF4rdKflaA1@~~IWG5RGR$1e*G{QB|@f2(9Ip5@={hRr8w%W-llkt-^`NNdZt zO~h9-DCTa$HW~d29%A&X^y9e7uQ-o$CH)vx^N0B%zMmG*ujBoGNcs)X{wDnv{kFP$ qoBoX6Ht+rx@1^x^`d9Pr-{{}zKg_%THt)VqKcMf+JFa4ftNsraIi)KA literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/collection/IntArrayValuable.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/collection/IntArrayValuable.class new file mode 100644 index 0000000000000000000000000000000000000000..da59a9cd1b20228c4420244e40cc378a78062217 GIT binary patch literal 170 zcmZ8bJqp4w7=2%D8+CH=4!Rf+FW@3{au8gczJ?$vUmJ*x(5pFk01qW5tIOm4!uxz* zZvY1@6qLf8P;m%65z5W>))1B#AA_+cCzh~ohVhu=D@){7?5^V{6ZXdUIz&G?omlS& r9Y-gRBt)zkT{8ZcgjL>Z;o+{&ZI@~_fl#D30F{Ip%FIwDEKvUdic~D} literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/collection/KeyValue.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/collection/KeyValue.class new file mode 100644 index 0000000000000000000000000000000000000000..4222a1099e98f7970c6d80810d8d45b8544f8a59 GIT binary patch literal 2566 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 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/file/FileTypeUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/file/FileTypeUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..d78a82f4671ba2007e9834fa1e8cf21ccb9ad9fb GIT binary patch literal 1374 zcma)5O>@&$6g|&hl5NFN1PLFel#i6!N{I>zC20Z_2NE2dkeL7*W;#_Ikq8{iBg;Uh ztNx5m8LCy;(oNc7!4Kd!Wf;zr3^6lau(?m~zI)EO_q?a~AKtzLFoVScMp0E@s+cO^ z2V5+mglQF40YBoB43}lFRm`ZkqT(kNR~5`Em=nk?cx^AdE|9KN9|~j^{Vi8uwBfbg zwchTg8{Bs`Tg(^@zv;9dI)Nwm{bVNG@wx(2ji$eA1wDVyv-q{^x2<02wYt`0ujN`x z4EOgs?gN>c7btGK;isIyxk}@S^USe4KQ5fFCfkVz*OwS; ztLu#GYs^%?`WglFL)Y$YK5?7j7pdgt1hm1s6;SA4;@i>6P}hnrryItW-+L@DKI}Sq z5&3R!lWy{D$eQ|arqFNr&)s0r>AE<}|6!EB3XlQBN&oyC==A#45z@h?A8IKr5V zlSLfmx0V!O924B-Y3p%jB=|0?C0G~cKgb#BH|9Pv|0DZ`xpd4^;tC?NH~|&q0Y06X zlRPWOwL*=FPxXmi!csgqVH&x8C=2D634J;9+dkBNjiOxVB8-`;tveC&MzQf4znVJBRw7KAI?stKn>q_fWSDwB&g3cb8Z%O- zCh^o+m3WQW({i^SVM3~|MJP#Cc9Szk8dHaJn<#LSSsFEmdA|3XeBv%&_)|Xc-*F3n sUilrk4fUv_zU0BnZyqb*#p=LE{s&iPcrvCD^jR@1cd+}DgXcg literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/file/FileUploadUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/file/FileUploadUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..92d46aceaaacbb056ffd25c720f0bc3289c21342 GIT binary patch literal 5894 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 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/file/FileUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/file/FileUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..547dce63755349f3b41c61d3e2c1c1aa9ff79ae9 GIT binary patch literal 6368 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 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/file/ImageUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/file/ImageUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..d7e158e0427e4615c4f9b4e3ab45e986687f6bb9 GIT binary patch literal 3585 zcmZ`+3tJTD6@I^EcV`(V3WyjqF(6)mWiwheNyS)&U_}=+f)_MxGVBh^=(4k!ofT=C z)TU{zX`A$FYtl=4wfB&uT{c0n_e*;D6{Yp*Z^+a1{bqLAMS$nwyPxx&^PcydZ}#2) zy)y@3H~y(&Cq`B1NT^7{l+BoeR4rG=Bd}KCJTA!Xw2BF26xb?OAsfNd5jZMxa24cL z)ZwCRCRJR*R1NUor{w5kvUx@}A6M}#o>TA%70vRxW?WYA$q1fT@hQ9@=bu*a85J$K z7{L{}_}K`qDtJ-FR;hX`J|{Onui^`MNj5Lb=8Foxq@oL7R`C^lRl(O(T*KEDd_$ln zZKnhp;UqN|{c#KzN^(vE2PU)827PAlPdsO@TFWD`O7kCq_+Y*ceT7 zg~ACVeadhwIWH{+-Ek`?&>m0N6S|YPr!1YL2|J_bT`Qf_$E>ue_fHrp^Q4^fx=6}& zy#;}$_V{_@qM@gaOiCYe9V?US?l{*YAb8!ns=dBT2{Y?jc82n`meh9KaL1)dOXLLt z(x8&GW$XQ!Y~H1E)0m+2JwvXMxG-R3J^K}0SMUnMtKd}ug_a$ny#gJ}8Rp-$L}S=7 zbNRH(d!^+O-jno|g!4}N*ywqI)xKoLboG-b;?fB2^xBz>nUFpS+`H&NWk-6OmZ1CI zYDs%CleUedKdqq_=`9sBmND%ya^^$3eJBOCt?WiM!*}qSz}iYo z7i`h+0qmlk^RLeR=jxUD7hk(`>(>0u=jV&XmS>(-@Ldhx!}m4Bu}{GdH2e@h61dOP zLqmI}T=Ssg7*kbLkhkBU!aFa`%)flCq_(V(PK~>6mh8~*WBf!w^Hh}rC`Es&;b-`{ zf?sI(C0^GsEt@y+riQohD+RySP{549`id)lOnUNGI%zr@ig;U~ZAsE3>`cxzGOkV% zcmq;7qGugjg0J8m4YRnZ;Z{X?DG*hz32gMe`uDXf|C%k_nR$JF?u7-gG|WklPAFR9LiBrm#;2d2=O=%l3F29v*k>NxwjCl*P@+8i{dJ&)Sw(foQQ;cm!6t z_5u+EnoFs%ShHV!v_#s@nUCi!(@jrRsuYj*$_;s0x2%+PcNbHEkm)$KBha}_=J`p! z*l1smhlGSn_h`zPhZA;o>ev`-Qv11{ey=8mIFh%jpW98!0R89M8srsDhmgd9OglXp)ykL93G;cMxPOt_ptgBpj%=W@351sW6 z$$C{C(Dh|A?uy3>IWlIY^b>h|#G(-#6BH^u?q>f%B<%q-GlHF+KXVy-Bm64^g{BsoMn6C!ghd9&OvyvgyIRo(q2LU`>Is~l4BOGh~ z0eld9IjcoC_LYQhqXr?a#X5@!#G^&j46q4yO%J?@aD&qIHX=0;gK{l4y*DWP=HP7v zDI~w_LSDax3hJ;4vfqMjUSH!Gu?Y&-HuBUap5M+>EvUz4+=n)?T1+2M_=q&0+7_a()t^yhcWA*tR z1+0xVG#0Qf7A&ACR>1oB4YU_=@4zgYM`o~rCpUIB++V~d4hnc+a272iu^F^33ESqC z!xq3tkf4vUpQODBJni+9nDTlbiz8^@UJp<8(a*zZB*e}1yoJ7Rqi0bJ5T!vPbb{y% z;TVQ-2B)a=NoqX99}8z;;~aHA>0x)6G-zbd2PhMuruCFK?&02w#~JJZ=e5*zfNRMr zDqN=S1GFT7=XgVz3iSxShh1dBmLnAwoaFUhK8QdG?jvMD+jYi#%k(U^O0>7dqXle# z#aj{s!UK@hXqQ3gxEUGjBI=!|If>qYx|c*RAQmg)K}nWYCe42SsPF;Wk2N%FoNig< z^#z)oCV?h!Kiz%+S)O*#$)7AgI2F1_e*NFUMX=JV87Z27rn2(J&$ezFZ$#sY(9T aHt0p{cne`!BLpw;3ketvfu}eN!1zCo#D7)* literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/file/IoUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/file/IoUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..c6edb3423b588074f14ee4b21e7110dfe34aa681 GIT binary patch literal 754 zcma)4O-~d-5PdZ)J1~xbAPBCC2pHIdeeqzjYN7-Zl8u{)aNyQEG|*(Go6K|*_h0Er z;=v!_k1|%zt{>4gagyn(S6^NAs=t2w`~|>MY$lk*odnBRNpKgf824h_Cp4dHquoow z#Mbr^p|N966k)cjje6sYzAE3ze$JNpuFYhABugFe$k=e_x*}|LGh2w#+kqCW71jvv zbY6*5ohz|#4+H2Brb;DG4&CW9LVK%wA+My+R_vQg?+zUMiq7%&(40$iCfKIU8RtdX z5EjQK5Mo{J6no0zR%R9)7Dfm9xy(zzRc7fH$q59tvm@=r)VOMv6|u`)?z$L zv5pPG`hScd$Mw`2Tr{zOF8Iy_lrqF>yPlTlE7)$2Gizw4~z(pNB8|UO=rv+u!aUdq3}iE zcY~PEDMo@Nv7Bsw0)4FA(|k5-;|vnk(_sK^um()yCT6%uNPfXG3buCZ8qqZ-J}`(l i#f2(y@DPbhH<^IUX+^#;q^{4cT|Vd*e$YE!&)5ubVer6Pu-# zO((VHnAN6Lt137({QN~no;N87)*EKYT4T3$$0**<8g=g!d=U!7M6t4SkBDAtHHzjm zOY_uwil22yr{Nu3(eN0bas-xbl+8rFTGsFxKHoL0)XN-PcXOp~lI(aE?E0==wo2?8 zV~JjXzw4@cPO7&J1+jm=7Y%RWZN@ZL-S|?atzq0V*!7DvtQu_>ZNo!Xx2+|j(Y9RO zJ(+4W8R~1;oL9qc7c@-lBpA~$>8VS#A~%Wbm%3Krko-iK)awojk;K33*cI9+h|U^! z)P~U@Zdkmiue?FEuyKJ{Un*(1?B*W%-^1Q>`MJW1S!4mRY&4gpnhm2PgPbJkNiN2!?P07unJ=G9Og(*L!$?{EZ$SB!9DqhsZDWa6!&VFxQYf z08g@yo@5g}$qQ4etjR}1Op+xi70STqk5GP~;=@&9&@&2Lqpo!wFohIZ7}J=cg~5vS z4i0RMW_*+JE%?5p1HV z|1~Dn$tSoJmjRC@{A8nX8T8nR0NIha40&uINVYF7g~tM+$sXt*J-X% zF1Ow}8ScnPNA`83){*@kxvxX3>fQDjLe%5<7H9E<1A5BQe9ysjID%(v^WRzcZYcfyA}4+B2Z4A0YOZp{I|-CS;%7wjs?_wnU-s+yfQ;z*pzPE0AsN=6 zAr|QF?|1hl4FfT4kA)Ej+=zp2#GsxR@&!E~3(7;pkQ^ck`64ka4-+Hu2(eEdWu{}n zZlagm)N~AZ*5k3k|2h%#pJ0Zde3=-MuMmYiNes(Zi4l2<*e6dDH91V|mm|b|a+DaA zabirqM%*vY5T#`G*noG=1FWBq9h7HzwuJlM6mJ592zVVmjzRuB4&fmV<4YXnZiaRr GvVQ`P5_<{& literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/html/EscapeUtil.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/html/EscapeUtil.class new file mode 100644 index 0000000000000000000000000000000000000000..1005c8df352b34f8645eb7e14fddc75695e6c62c GIT binary patch literal 2981 zcmaJ?>su7(75~ldvNKFZSymPUvJDZzy%?9MwTZw+r6i$&XaqsT+pvr}?oqcwEbmZ*lp<~Y}h>lz^FPfRW>C9!u+_LS=^(oji zSad3`>9`YSzG`W>svtBzbYfgVaC&-BLAc*`Y&T0bk({7zusCZeXd1B{>-lP7#ww4S zGkHp)BgLGVpD@d|@HJt`owq9r5+k`{Aycjv7winFLea@oT{~aN%)5nrW~h=gOV)7; z7*5#oI0Ze4jnO7IiWre=U2+v@RcF;y%@rNjawu%b14E!{vru2nYhmYBSacNA>@KPG;YNM#r;6l{I4e`+uh5(DqzClbqh2HwX9M7P_( zhqy+fSH(vLev0b~0^L=iJ~r?(d}83|xNhJVLVbu|3VTCAdw(TYwo7i-%v)tQ@p!VY zKU0$!IDkGCzcTP^{D!D|#XZnz;3*OR*1)IujKKMAaG+CxzSDs%GdRd?F*ZsY9@pZGG0%+-*MqxIajgP@ zDpu()=I^$PnWMZKT_$9i1(tq`m+*Qq>rj$Sc}{uVIuP-P+~S~MUM`!gRw$929>k-( zv3K!J37`#+VK+xA<8i%cm#!qOQO1Gh|GVTeTse7YNZp3=M{nzX{==S7$M;AZ{sugO zC&@L+h)}ZLcm;v49x&dy&bXhkh%slRj&blo<2~z)vo&K*Va<3y6O^!xZxEWg&FGgA ze$dCjeLf8MK6JUpd>_hAKCU79xRP3ddIRBT<5%B6pQ5XWBH{VlFh+vexNu82k%}*4 z(?CbFf+cjPqnnq}G}#e}EJ064Bf_M~t)N-RTj)whqu!3@rn@)mZMhm$Zr%Mdy^Jjq zbPxjhVMO?U7Ah}U9iyz@5p>}NBykMA7-uraiQXi>k10w{^M3~CU}6qAl)3j3-+3ul zib*BOAEL(qwsGemZGzNXfnS9-%wh;n6NYCf?S!h|K}6-z%3UN7QZYoq(}#D-Tk}lj@RAb=@;XHMa0r6x_?+;mIG<_E+s5eOhj?= z3+wSch8D*C67j7tql<{)Wn#R*iFt*1U1p2D%GQ33$iGgs-@qw8Uk+cdD&C>?14^&q zJ$!_Z`LcY9PjC}AJiN0UmQgg}IpQ6n#pW7CjA~66qm7UERz6K5tcFbIHCjnsqz26h zH{L;$2m60;_>tBohWI>-=+D_7xJ?RpGTJthUP1fl*VrbldK(>4X52*Ccet>fmhs4a zx=spr)l)nwiWO|Xr(2u+cBTU+7{q71DsEy2zF^z^j_vlPmu{yY368QgrFAFSjB?Fr zTn~^6kQ%#py{y!3lIh67zoP3;*b(e&j~rY=>(|)P9{NkeRIvT}@v*DmIQ)~M$j)f@ zGJ3{@+#cPzO7=t(t7LyPxk_fEEPT(HgzisJigN?vuOZ4=YC)W97anF%ZM?eLiS{-^ h+JTeQtX^z7buu6D>d=V5f1xY?rjf!XkZbh#{{V6oeZ&9& literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/html/HTMLFilter.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/html/HTMLFilter.class new file mode 100644 index 0000000000000000000000000000000000000000..7fd6568614e3110e3a3ee20c4d8321537f684fc9 GIT binary patch literal 13395 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 cY3AeCA`9^oqD9&jK>bX}fRVz=JJlKg2gwRpF literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/http/HttpHelper.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/http/HttpHelper.class new file mode 100644 index 0000000000000000000000000000000000000000..6bef188c4ac31885952a823aed72efd7f8160aea GIT binary patch literal 2492 zcma)7OKenC82-+@?#%R-$FxAFA_xMulyc>zbjm|pCgZ+3z!J7&?Lg>VSFakIjh96x)9Fljp6gwQo5xk`! z9>x;%NSKeKAvB;jjAMAa27TxcVnE)5QsKCQ6Jeahkb-vv0`cBMhYt1$MB=8E)(Y9- zougXZOs5U2o!mW{T*lrf;B9I?F5v4lQ-;9OcqV5YEsiG)Yd}wAiBLJIXOHVvM!rj+ z&mPGX1e)SWb6m5E=J||9YTV3eMLUx%Xd||r*SdLh8`-?$t~PDh9cJo$zinl5X@Tuc z@lkz3Kc^K8Ya(mdTEBbu8E1+{!ESGM*jiT4r8QTFG5fNNPKG2^t9Uh;T)t>i)X>KT zR=6S=Q#)LSX+coEg4cWST++zf88b(oV!6Hr*8F#D0-i!bAi5}wj$$U8a~59WQ1K$x2&}5kOerKx*M*f~ zm)qzWJM;a#gh5v%PGCcm+ro=%S4go5);S^Nn3&{%`x*24s@N~ zEzs!pQl)T7h=xi7rB${3^Ht>3*SS+Dl}c?X(V%Qzxd;U6OJ7knYymx=H*#!)){=qB zIbA%u<+<=Dvu1&IYxPvBznITkMxkKPPD7=gh2CkG(QK2;OV8`c5u>zZ3QjaTG-G~v zwS_Srl>HM4a*|h6nklU$XLFv_tsIkPoAbS} zs@h1Gp0rKtyg-O?xf@tuW2L>$tX?Q^=~ZjCV;$FhBmY8JemO3ryrenoJlFE2@#Z+f zzj+$s2NE8<#53T45vt2kbp)^;FY^_`25clkWzTPv3ln>K1D>I2c;ljH^L6+HH&4uW z1OA~K2n@Aeg))ucG(tU*aLm7D3N=G9|14@-qW&qUCno2fw9evK)HjQjty8F-#DSPU z>XXc+li2M-+X(q%1aHLxlwMmQT^Ca*tvJ$(l#T{&ZJEVsI$Ab~lSi*2(s~Vb*HQ1m zB)a80vf?^cdO)c*S898e)cRPE%&5zZ#6r}GhUSe#gST3yu<8mE(~UTK`R`iKKjcK| z9%Di#@F70pJHgc?-=E_WzQS@Q>mn}mp21DrF1_!Pe;HGd_dVRl1LAk_tMm3c#rW8% zR0D#D(n2G>t;Q>?VGFCcixT@;ngsDN-WSnKZ>{vVogOuOjcxdr^ko+BI~Hz={2B6Z z61z?Q9rAx6eV6z>;`d2EApVf}W72=ogKUCREPM++Z=z>EBm4wS)bP^2ip}JBX+2z` z?kr#rQR`qKT6l^rv=4s8V{CQ$E{s3%D*brq2lxWZ=V&OPV26U83U(=YO~LC5b_0K- zu9tL87msfK<`{ueB?#gWQp!6=lime3a6~~H_1gyBo9Ze;6?}RxDYq@D)@8i0g=tym di~@%s(r^vY9}$rgBb*|+6I0JuJ24OT{0sl!YzhDX literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/http/HttpUtils$1.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/http/HttpUtils$1.class new file mode 100644 index 0000000000000000000000000000000000000000..3df88e35d444e2a56034b10f89d0dfd0e98eb122 GIT binary patch literal 231 zcma)$O$x$54256(YptR^f(yl+gWwHB(S^7W4=~oDKTKz&GZnm=3lHF-#B}4@gybcV z_mTJK`2w)QAV3hHPl!&ot`fD)wIYmm-rvqX*KSeByCjS^CehaACOs&h*=vA^Fi&hL zs@gts!Bc6Cs2hvO*erg8>5?$q8>6bVmd+_hh#RQT(xl>eDOBPKv%gsJ1cVkp-b9-d TymioJ?V!g!OY-#?J zFFpxA_yhb=;+;)sg<1&sa=$p|-gEEFuirm@0(gbZ2tL*#1Xv64IK+C0CxrSNrIpQ_5FF&6C1EjET7D?U16drUgN!TfI3K22KP{BYXT`vdlp(Af6{WFn_2e)&R;Odx zmxcPGWI<@Y*IE__S!#@AmCytqnSRp&NVI}WQ`6?SiQMX zhz-K?&YX+v_S{vyi&W};efD1P)!Dz8ixtz2b@eIJMR(Dzx_MZ_J$NlQX?fP|HG7Zo mP73{7%JLL&*|3HOSgB|at3j){!@kWZACFjgg!`PM)%XKHf8<91 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/http/HttpUtils$TrustAnyTrustManager.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/http/HttpUtils$TrustAnyTrustManager.class new file mode 100644 index 0000000000000000000000000000000000000000..9777bc4373b2540c9f590276c8ade4fdf58b1405 GIT binary patch literal 1211 zcmbVLO>Yx15Pi_%R1)%+_s zAtAvJ;16*@i18+bLXo=S5RYdvZ)V<`_17Ptz5sZDdI>qKl#s`Yhnj~~54Sz6F%-6S zpzULZY@@lykZXtgiea&%1N9;v^_A&~zE6>ACzQh96Gl6K7R=e9jun1Rco5#i(tUJ*63i+a$NIr zD0INEo~i~zQN(uGn~W1RlCH-q{+h6>%ps}f6mb()2g>e9smA2(Mp3K`Nl|Y!Q}y() z&Ty}hQn<~8Rpw}ep_v-OQ0j)Uk?N^-YhIY|;l5*7hKC}<>ggJLxI^ps{|z%#eoOkY zf1spgSazW!9w^JB$WPY#=O0 zjIq?kF5}8aW8*%^u literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/http/HttpUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/http/HttpUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..c333664674b88c9a77e209e862b2ce461455c4b1 GIT binary patch literal 8782 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 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/ip/AddressUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/ip/AddressUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..9cdd6e7233a7bfd9a331a522b695b88dc330a8af GIT binary patch literal 2195 zcma)8ZBr9h6n>V3?6PbvELtkI7PKHBWYt<IyW8E3N@>4z zrhe<&m(DQL-_W7$jHWO22lRK8u|J~I_Sqz8G@)Z=_MUssz2~0uJm=hd^T$8G{0`tG zK9SLdQ5olOUcv<#@8De#7bT2|7?&YpRKlfDPDr>M%1IGdBuvSe#*B;vuC|~VNeQ#9 zn8P&*^Q}nXx`YK8i?|`-xP&DUX&EFJ-p2<5B9@aApe7tIr}@^($-I_uayi2r z6%d=6yS*@<2xs&9U0t(uJEx@r&$M%#eQDgW{Xn;aMcpbI0>p*nOmg-{l6L1z+YByn z@BV>B0nu@1ne#6vOxs8n3mL<^u4gQ2=rpTai@Ilq&tlNWHP z>{qwUjGoc872OZ=zGI)zrc<-Y*-YNZ1|cz?kux2e(6VN*CO{yVeKng;+|L?rz!hBE zWV+-k73kSJS9P}xQp9cYFF-n%wW>_XDW~XVjmu`pOM5NAcu2j1H_A-D$#&=Y0m<`FzsH zil((}gjAXCc?BzA;mCaVQ3| z$SJp6jHJV~1H-d*Yl;N!YgB_bcVTG7%xUvQCvB1>_8EccSLD8Jf^#@ zVRJa#X#birtOQ=o8cR_MoE?%jRt?A^jkY$rLf6~`;+B=>9shqr3D+<{S}s_emJcRo+fWjz}>9%&0*Rql8J zy|o_3E#3D=2kM^!^zdcr;v*Gd%QMQ}esojI`;z@8#5g`&Gv3RULU?Df?c)lKQlM-e zd<5Z#vdIy)v9huq{p>511sK2}M=dytA!=wWoT2k(>edHKXxczbjck?>-M}3+_AA8n zBS>n?L&$^t)K)eRp{RRGd~c4H(6)j0AJML=gFmAq0(I{umHVy^sjoalXL3k=m0~2S z?r+{gS2{ABN=NrcQzabOOgw8vjBU6C3$I}!ys{`G2qBDM58H!$j=Ld~jW&jSj5DVR zWQIKR|sCy`<4Wn;t}I z*NPLg4Vhf$ABG;vxA}D3&275?tbd-G<-&t~vKSWJu|0efsStR}&%c32F9A+|9TD0Do2*bl}t!_2d> zLmL_b36KP`&^AEWoW(4zYNCpgs3q_xl`6HBTJ>*RwNk4}m7+#nl$PRt_r1ZMI)>oU zz4yFx&prD&_xfk|KE4HD0j~K`3B!xsUbN?+1EwE)uvbnoFFL)5dy()WDYy1{k&+u} zFDySQ&?TpCIqmnO2fbcAC$|sy@lAY7C3g(*&juvy03(>8AwQnSK@H#b;{_a&(~BAo z%k_vCFM08@Jn{+z__p(HyXM+ zTa8q6q%}rNemD`0#9AULQ+{WViq&DJ1!jb!iOx`}E75C)2z4gnp)Sjer9);iRG-`| z!5RTyyJ6KQSGKjK1Zv8{dm`PDo>1CIb;k@V)L~i4&{|GS4x-W6*JY%wnsS>i6pO^$ zLrqr7jJK1MmU4ClW<4Ss0hgJSMo6inxaBrkk?7tHk)&;uhLZw*Gj18FcqCTOLq(3R z|97f<8(kH#O>u3n)v#PZE!$oz(XQ4x0-{y|b7^$bs=DU-jSai9;dxp3`C0ISEO=oS zTumh`qsMAaFeFCGjIK_kIwMvK=_ib(+;&PAF^+W{CPzI7V_2GHz7t%B(W;)Pk+jT2 zoW510AX?MfYZ(l!*N7RNM%*G;us!Q|CKei6n0!=~v0a-;Grj_aWn%=5St*c{p=yY9 zQkyCK+RzoV%(NA0HyVv@Bb7Ebr4l{8lFQ7o4EwSI)HL1IDi65LB$f1;>CN%I@x=Z( z#YH=e=w3%t#ne9%#a+03f0`v=GMRP9cr_D51}Mw^5*ysqO7 z>|}1}coT2wIEAGw5?k4-x|m4^Rn>1IW@{BanmI^_8Mh@|^(8Xj)~ZG$65|ma-vM)G z+?>VPzqvZTi|^5xx$_pyp2HszU&l@z>k-zm1S>VXt>d&LaqXC`;|$Jfct^)MoY!zc z$3crhF=MTvPsh7>PsjUGg{d}qoTZrYpA6@4V(8c*i9COL=jxv?yw7Z^)bRl> z>FCEcDgUyL2$t%&f~!okuj~c_ff4OCw(c>aOt~!7{g^!|slKvFWYW)wv>EicCuYRk ztq!}PG4BIx6_HrFzCE5u8Fi5~gP+U9Ycx6&mf zGu_Y?W7VnYWXz;JWn*M;efbzgB#<}ajc_8~PJ@yW7J~zi*?(Bs<3|qFvt6}I@iwDp<8IkJ zQf4Q+@)(wnZMbHpYuVfPvKwTdsq2WOnq-fUM-4^*rM$K(Mv^n4cq4}oC5_OMOw!!~^n+w?zh(;xf@IRfE)ojT;6hBrU2A6mstj2ooZZrjqu z$Rqq16ikAKDF~n#PvCL7Xc|^xI@a-PCuX7@WjKH!4s+*iT7QB5klvZ#cmUN{#IFGQ z@D!e=e-_(ZbG5rD)==XuaDJZLI)2fKOR&^&BHA)eoD#f=K$yynKSQtO1#OKrr5}oW zfV>=S{kF8FP_Zma`V?CxgGgU9AN!S9_;iBH9zJ+ez!i?R@o$cIVBgZa$1Ff)Npqr0Eslg7%Cffm&!<|~x(c;J0SDzu|BG;W* zMaWIw+pwCDLJ2LjsgACjNzQ8tc_^)cI@K{9W;6Wj2zl*rPKK5%hOh!YI#p!;Xj*Oo z+l{ZCxN}q5j98)=x9KRuIaix3S zO%&CZ`c9oiPH7<~oH{Q#RJcn66Au((;=!YDG*>9xn0zo07)Gc>6dw$DhY=nkY}$U| zvROWUWQ6C$VV)VHN`*qRDJEwql+s5Nf^+U_wMbp$2eAN$Sko^uXO58b%gocGFma5#$Js7kA?7ul z#0fqwUT52Q6IbvSuJZx(3EwDp*i3%Ghsf{v1o#Wi;;*Fo2dVDiqR`PN^6;LRgma<< zABb7#7YlKPZTFgZ1_Po2-xtr~2Vxtpizq%6`|v}N#trd2afk7-c*(ZzC*n9h5wB8` ztdb;hsB4(D*1$SA$y#!td%PiRh*E+X+-Iz?jjTD9-0$x;vuQfGSu<{#>8#bK z$vs!Rz>1X(s7Ul;Gi%FbSI!|!^!*n>mW_rjcTuQ_f1`%YNf9N4g7-+q4#Cc=vO79d z3af*t06fDeBKRL1T|SEH3ru<-hc6Hi{O@9`?08>e)jPM|8m_YXa&BVU4U`V9&(9mQ z5grD2dZyRrNz?FiVt&a6@GIMW^0t$mo%iD!jm^j>Uv*p zcdz&331EqFj5x-K5S^{7xF@XU`;uQd)7%v7a4Mzn>9Hqe-YHY0i=Om#5hj^!oRC|k zC1JL?38{D7%CfoMoD35c#;ENkzI zy<-y6QIT+IQ966?NUJTU0FkKkqp+_s&f&0e8R8?|J_2GI!?8Dermv`<`HPK##^;$)nrofKlQ^5p&T*}c{Ch5!oR?{x zZyoDe7i3a}^%JgmVN^`AF5<_FUF(u49b;YUT9-ww%dIOS)|K4zReZUc-(15Jx|Tb; zE(1-k=NFr}>J1sR(z@QYZsdoXGN^_hZsvzuGN_gxZsmvDT zjqjRWYdfQ&g-cp9tS77;T)dM3ev+F%6|tV?-(9ZtOeS4wJsY)tYCRXVer7$-lzoBQ zy~srRIp6()FTdo=uOim3Gp(1bmm}6I{QD{wzQ&i=`SKgSyumNtWNN+@wcfVgajkcm z$nQn0-$v<9>vx&f`_>0s@k7`8J^%h8O7~ffQR|P^N3OLyVtpK?hnW1II8(r%{9n}i z)cO-&KI8g-cCEiK)&9y4e~SW2JP%@&dOr?~~7l$t{zl&t33>D?O z%naqJtSBy>_^~rzviZ`5FFAb4`JlEch^WGdDvGM%s+gZg zaGgW=GLkDE%4MS<<7%|4#_;c0z8n^%@6@=cDpBM45!z>qn#e;;;=9R=jwzX{R88fw zX?&R;RWsDgh=Q_NqGq|OETWExs3ZBQoZrvRq(U{vnfys~;@UgkiaJx)8p?9Ad1(z-0vW0v(nlFpE>=?d45!IE7(S5-$vAv6~r;+bltTE&;uuBzeR=h^mv-DG~K!emE5}t)P)=)akA|gLTf~Ow}_Z>MWkz*->$p zI)_U(M%B6MJX|ciJ3p!}P(N`Mw9$Nsf|#!^;>*Q+(MqX;;kq=UF5{_Q&O=`T{#IA! z5UJ}k)h2aArn*tx6je8?Tlnr)SKY={Z|ACa@a4{^x=Y>7jqZu6&FWqT_CAQXx}WPk zz`#8SQBV)L>fuaLrygCs-bna50 zx$4gm1(ovvNP~`{ca|vVod*;rRmyaG_H=vn<{EfL48ps$UM)f zT!UxTbn@D`tG>qF0^AAm%IfOk4by5X8yn+|f`*n?)vqaTSXX~?O)>7))YlbP*45QF zRW{Y&cE*OPcrX&!TLUF%K!*(~7;@Bh8k z8y$`|EnZt&!2^w-gpO5hO=FXw>`q9e6|XL@Y;2lef6^ST z-8E4KU4PLwsc|Ml7Bk=XZ=URGjI9TF@5MG!1oK4R92^*90Wg zRAH85LQO%7Ft*F;n&Qmaq0Xk)uUp|`Hf#Fag;UFCEMwj+T_7lCsx`$oG!;*eSJkYk ztesiku%?o6ZCBM+o^lE#B-EAWPQ2^Zwk#k-R(wNM?YioCFiB@ZO)ss;1R7nEv4QqQPw0sM%gi#zH)5 zFzdseKwA8c3J$1;U{bMmu$wnLf#~X*>iD$!+I4H{fR>?QcJoz-o+e{K&@#UxEkv57 zLrC$frztr=MZg1PRWwyrodBNjCtBThL<4!6LE5Rj_9#lPe|Xr;s)l%FQ+$DuC2J^` zd0MhqdVn555&Z~364bv#U4SYqDY`oA1wjXQSfA^Rchw%atC+t5q+;EQS@B9pqpECb zYG7TrUmWt7tI>wLJ4BFMuayP3Ek(8aDungox)pjd5paCGR%-{3+XspPNa06cd${Tw zSAC1Jnl)?d8=5qe3L4^5r>4HRtZwbPCa^%fa?SW*hCE3wfr@9o_qAV}+TcO%ZY|pd zR4P2MJ_Ft!n!b8j8W;=(^-PNnuvIWhiC;f#Qq3gWOeXEPYip|5%3yN#dSiSQP!7fr zH6TQlFMPQnrVg|qa8nJwG&&9S(S<@mB@nuFD!PNhTLon^tV0HyXNI4Sc3={3ZA1Oq zctg|4#*ABlDAi*~u!yaVQbNh^=yK9I5+E7f$=i-T}qnK%g#|nrQ)}0|!ZNbKOXz~JBlNv1Ok%u3fI{X0- z4ENnt-ysv>s_$L37a0{KEs&rls0l|7)%e zva0%qcyV?8n#!8G;?m zGyV^dG#{b^Mn~8VvajSr^k214137vuIA9*+@db>aZ*DYb35Noi?{{4r8NGs`KLo>( zl{K~TIvwX)_3PmI`T+y1{oHj;Nl(YM1x)lw4K+=19dJU5;8*!%=38wiVcFvBWWj3J zvlCELR~_E~!>0q;IrZ@RFkG!}#YP%Y;Lx%TEjqiHJZ&DK%yeeK-!lc;fx}Si(XopP zENqDhXe9uDmcu&uTTrqt0<(rDh647xTK~ej@~ku8@WQQG0gO^}J?7+F6Szm{Vb?jVo9Sjj+r^(dMXhIGGQ6jf(`0rp2 z+u%UUbmV9xL1%;BEg6Q)f`c7z(6a(x%sfD7LW-0T81{4xLZj@s#~aKXtrOyKvb1ZT z9BSu(HPrty?>12WFI~Y?OQMhkCO}{(!;y?tM!BGx14u%R?~vpl}P)jr8z(goW{j85HdHn-Z$+vR zKpI@1`+qWZ1Vo|P|2K2Z1Fh#Z)UR3FG`As4Z}=hmXSHyg7O3we>iv>3{HIMU=N4R`4Ssv;=U2_po!Kq|D5vpQajh+ELe4uMPM-C4Q~h!d^i zynO?LBMdp_2lSxx{kk1`aU0^ROn#*(T^H#XL+AGdb%HFV$W&pv4ibs_)LD#oZRy$; zmd{*UvCCkXZ0gCH$)IK0;r&2Ld|MKBLRqOb#n+TJG*rUcDZYnW6{I?RZ5a&j^LVG= zwIO-J;R*FSbzMy@EGZC}<%3>PzpkMwjs7-&!hqCmBTSEu~czJ9j>n>B96iYN3@3Mg8x z6ucN{weTWNgcazY8^RF#-2tH{dU4_0V`QMtF?WcS;Gk3t?MMgzw_`?=sL=79HuMLjG^RDgKX z!B7T4K}3~Uz3?MYd#JHo+@6iSuYbmLY#Mz>H_3(xdPHj)@p1+%A2jR0{r55q3ECWt zt-LXzO<~;s6>?OLfk%yj{u5>Fpf2G~-rqRg8FGbyNMZU6m|M#=I~Kw>=EhDZ$Fhl{!{&;PGI;4mO>+wj*e2c+Tx^&4X9GuIxhQ4ZNGhTRBKr+pFpC=+!V&%2ZM z3>P9~9nK8piBHAXo_JS0>xuWovj}98-r=mu#??@up7>C_xX)d?#?RT|455-=MOcVMjfyuXT_o-siTz z6u_&(5+td3I+z+Y`SQFJVR2Q-86$z*%kMRiqdnW>}rn#KNs- z=h(TP-PO*6^O&Yly3pOTy8$xssomYPd)P70?rHbJ_6^VO4Os0yc3B9|$+zr- z;24=?^0FVFhW|sR+6u8}_p|$Z_QANQb@l*HX31P+<=b`7y0Ed>upbi-8k@LsH_sj@ zUUcn2o;}ze;)$P&UwHOVdzfeE+XVrsc$dXorn}xHf&}o#*Pj2;j%IClK5DsS_e0n3 zzUf|`Fp!kr21#lBra%q0$7mXzsZBupZqw8Zy7esAinQ-L4IsL2PyR=K>Dh%KnO$Ui zo;{rJior7W2&=y*zmWg*>_co&ZlryvXOFT+d-4PMVZiBO3G?K?>BF+2idADO5KUbnS_rJ;|O7vpal*XHT(9 zJ$tG>&9kT5Gd$S`z*@bm-kv=ZGG`xd&+_at`v~apzl%&py^Z&a;2Ss9wgrAHs^6wx193?BzDnBVxEGuR~>d zJzrYo7Ef;CyPdeRS1{g&??ViDf2&qwskt7{u3~~eC7*WfYR_J214FCr8df>S^Oz@K z%w>Fogi$(OkjTP?yjRa&W7oNMy=SkrPju}D&u+AV$#p!?Oj+jH>si_TOZ=N3m-1yU z{QAnZYinz&%wFB~b=AXD-NB;zl`Cti;?=%CH9X`6uBjcarP8%e^6U*PkM_Ezl|_en z_Q`f0_kOe|kCGL9S?I~-tWQs|f9%<(+NZhp>7IRteP$r^nm!4k_rw!o2VU)dWasYl zpU}wLz4PqdEiFGj&9l$44|46ZJ^LJcqbFaIFC(9(nZ!@Tgx736`HFnmwa@kJ^CTE% zXgdnU12t33YAP|lpyKe*cRl*)<9Gjk=f$5svi;MuuKwh)9iKdT#U~GJ`sCr;Ke_z8 z-B(=y_nl`UT5i|5(Six`^9ZhezGq)x|HQK|v@i1HKjh~Di)%JcIvJjQv3-drH_CH8 zxk|3~`xZuRWj?Aty24*O1!tP~4D0PHT$z8h4t@3A*~_PyxBzR$kj zwIA^82knPE`(gVL*9NOUYH#)I$K+o<`*E(g&3?qQoB3`#-?i{vE8jg~Z${sG{neAF zNl<%-z0;HD$aS9mB!IV{f~Avj@s4LdZEtq%U7r1n{j6vI)PByhf5wfosu5xvQ-rH$ zKX1R_*)OvF`8h7~3VEex|HA&IXa9-~-LD}(;%R|yUb4Xwk%^}koqBA+sZ-!%W(*wR z+OK%_t1POo@z}4k)c=MrZ?M?C$r`Pz{T5`njXg=D_`|`JXTNO}`8#Znu=(IU&;G6b zJGN=>d-7rlVtinK=*i3E<(~a}`wy=DN6-Gq-i>$m$DaL({jqC*>e+v?PXw4p%sbq( zKeJ&>2DX7B#a{>v4_eHdb2vQUj~~1D?>les>_6Lo@$A3K&prEZ_9D;zJ5TQ)_UE3w zQQqW_*F*k5|`JbAt+pOrstBZ68Em=x#O zUwQV|_8w&XJ^LFQdqu73(+~0N@9ggp(xrhOtlw@iN;_iqUWc$;;5ovPo@3e9BBE{6 zJ`D{x`GUZ}U;6VL1$K38zBrBxy;{|Sn~8Q) z43GAXJ*Tsi?KxfS&*4yePL2Z)ryKl^PtSSsB_7( z(t+5Fbh-&TC~3sZ262<*Pv{lT>24#H;OxHqf=?g5-4lNhe`H5$vgh;=U%O7sb9y?x zAU4A{)HZH#o!*`U>k79ltlvDRuXB*+^ux$be;)o#dlvA&bog;F@A7w^bFeeOa|Swt zU~6IUni{d5wAKU@jh!85aJJnAKSQ#u5uP*D8Rk0qo>Slyx=xYj40noM2jz!2BR%I( zXH<%f)%22of#^Go)?f=>-fsn< zDs`}jbwCn7f#=aKY3n)D93(#5JC$vh&>KD3!BnGLnz%<-JLe3^&doueFB591)TKYjo<8v*^p zl`9}~&H}z$$mBVi?-n`7xXxl`kR_h8l(Ba#UykEwF z+^2OWkqz{mDhFF=+bXRz)x$~)$&fY@mZpL4RCrF@jtVL|;07$2p0koQ$w+6F=d9+r z)}))%zLGWV!AqwAOlrz=j!)Nhng%-IazI@tW`2TG>p5$jI@f_eRfHKiYx#1b(*S?h zK}h2>M9rEw^2Rv3!A|G`Jb6x&vo3IKno?f#tM$+)NjV67O%CTF#t8g9=Okx?=bX&? zd%1H8Kdyq8@ad)7cVBkx?nhcbef0K(Gvxf(j=IjNo^zUWI{2^%{}4wBSiR@~{ojb; zBM`I?Zs(h{6()T=3z1?QLZ?_BKR7z5qdIy4FGo*Ldo*QOjxjq6vXX^5oowP24)&6# z2?X`)sB;dL1H5X7alz__`jaa485|QRpu4dD*WQ?y?xBs z!AS6JlE}b9X5hipG@{CXJnY@iA#lWHa z*-(O7NRRS<8ngIRRyVI6+Yi9rYUoVf6cIJO8Qf#DV!KRg%0#&u`b$}CO_L<@i(uL? zFt^be5FoML`arwhbxplFDrlQ7MHt`)6W|3!%u=5Paj{-zR?IqM-PxR(o3D3E=))*! zD4))A8P8abVBdky+TBvt`2v;)8r;>DjdQT4gqLXbPuHNPCUpV}sl`6*{-GQkv#hV> ztvN+#n+Cu{9Hhww_;{sb*b1_Xn|Z5GZbS&pN2_owaPCTGX^4=ES4MD<;)WE@ zS;q=2(!_alcqFK>p7-|6T?v!Ga3m*`%%WP@f!rp??(*6Ps)nT)69d91tS(q(hxj}o)zS>E-(*Kc#8HwEz;h(In#bPn*OJ9=EApC+T?i zddFO-t6>zcsXm#%(|Ueiu(iqcp>4l*-V3vDV=?<%3;7NRO{__EIB(7N+zh>DR_mk2 zzVU$NM9|S;;2jKyreKC6upGf9@JEJD_gN;~eYc@Y!i?mn9x*QH!w-T|&Kb5B>7YgW zURWm0+ECQ&^(ob7M;+wyAUA9*LUh?DV<)$~!BXjsXn+s0RmnIlY=`y`SL>YIK-e0b zB{sSt8h)_LW|tlgMjyWatuvucfSpJcG^gos1eDDW0RgRZ{8?*4@M=2if!?W_)gU7| z^At>_5tFv{8S@}KQo?HJFg}<$s2z|zd}HxS>99gY{B$*<3T9{d#N7!Am64WNAZueg{y(KwniT| zZu&HCN#5 z?lSxx4nPubXkbSwMJIQN)gkT3nf@FkY;EJZsw%yyJG6W#a`a5GAwC!%zdF2m?-N{Q!?Lt*9D(O=m@HCYo1u! zlwJyODUl&0m81j(j2g|lF((4P=KugHt_h7Dody-D37m% zsN~?8?@}tU2$+He^=nNA0yimDIwz`3t#4}5N==vWeM&_KF;XdqR_z@C;3gBB^_y^} zqEUxAonQs(;#x%*McEB~&cvBaX->ab=l}@E{$L|J{fGRkgS5TKXgB`mR?2KRVMAw zc}3d<0?XL+_{vHg+&4a-uWt+XbCNZwZ<%8=>f&*efq^c?cd5#FF)K?SP}N~yX&r2I zYM2!cj=~v>l#q~(+<=hiq}nbdduD71{DFozPHI1RxNi>-gNOHk0#11AILN zE8u*Fch17uGv=2~V=vX`C}WN>8{YVDkl{^NfeTWhE1r2I;GvN5(h0aC}Yv;IyC<5`* z#}a*_bkQzQEe`(U$XK;8N7*KrDqB?tJDshm;@p6muVCE_5G10O)Jo-x1kZ{g3T&5 zX0K=HwA_&?`w{~|=$QT~10i~8G)M1}3^8q8BTP;5jXo@!OkDi0qxh#^k|+{%MaP1W ziqRxs?e+t+t%B}IyXd~3G->J>K!)#G&(kL&r`AIg?kDmI@B+P<&Ahd`{-jXj9$d6; zEjE)kHcrR)iKgRl6?SIwiApy3TzduwSb8Krsg}f#iMxw`2#z!}+Kpfu2tYcx(@Rjo zXV7LS0?w#eJA)G|`%FgRNQtiq_%IHi{PR!y1&_Wt;pK>6ZD&e2M)R>27U-17NNe~Y z_A)(yI0|RJkV$Y%3ZPAP-;e9UT?Vm!NNpsdKM%XE6Q=@My99|XJ#-@bm$JAM&^pv# zqN0O?d=@7g3#%daj2(6<>N8W%-&^1nYAC(;6f=<3`lSOxnHIIrsusHba6i-9fvXj;@c!@-BAw>W13H@y80uFgRz z^;fQ|(;w|9p6TD9YqP{@#2SZ_gCpA+zz`nvdctU*p>(D8eS{&JmgoXqo2Bgbkn((B zI^saYA>pn=jl+GW&Js9C(%XLpABjIlf)y%sjZ8HNAgFK3)Z5kscrJWVZl*orefOD? zf=_TYa{}~Dya9CO)ADSmQ;3YR-l>MPEm6%zXpJ7dhn#`0Y&5_RN-;dC;-y>o$qWaY zOvfle0!zR2@xh;G>DY1A<1S=(a36c{Ljo880Z%v|sBI@f?ee%fdO&Wdw7lH7vWLg< zdDogMlgjHEkS9e0`E9xZo258SSE;kMr46gFuA!HY!T({s6QI;@HkjY;7ciEm)&7T0 z?V;0k;!7cQ1;fJOqjq79qlM+O5CR}(5C^UL*}dh!!%Ss=?rl&L19!L<5Y)q(Nrx^* zP>*@o{{}CDW`N``|L}X?E+E;(PcU)*Cjhn7?I$_GEi~ikBWrrrp)TF2bLW=NC`BrQ zhx8W|LhtPv^XJb+28$&^o7~VlEQlbt9ZEuS${HOFO-WdDA?C*FS4UCYfnI3mc!*a^ z69x9v^3pj+8t#G^_+C1g*@b^<&YKp4&)WFw>2@XsUoOG8Z6oxKs+(n-X{&L_A6CI^ z3ou_v;idlr89DJy6`rm?0YBQ~MFB)~_!eiI%&O0cm?mmK*il2D^^M~bVdfhR><{SO zNx%i$ALKc)6$l2%e2RHwd(hHsZ%PS}%wQ(DafIJIbx-$ydBhJHl2yh3druH*gABtB zJ|>NCcgO)vH`D}NFeZ%`Wt4$du`E$3mLWkC7$ANLKJt4=-K@oXxq^nto|0Z9=WNe8$2rfliY@FNJ6PLWJ+|>Ni_7)tFnFMqD}xOlnQM|y zu5$q>T#gTTuR~t%S7GA#9T?};9EWHeXljmcuf9ViDGjv=`LAA@A;Uwp?>LNX|LwF& zY*4~>AE4$Zz+yXuq?-@CnvUyHf7(LD_(Fa+>dD6)a2FFh#gili|G5`Q|H3miwCnO` z#IuR=oh13>8rR{&Yx)-lsq`;SrRiTBKSTYO_5CZThvROE=hyL!6K#p-H}H&uY`QMa z!RcS@7T3QxX{Ud2*iQdqE3f{=5x7MA56~WG?h?=yu#MbDQNzUM{Eo*HeD=nl_N(Z{M%l=M7btTF zWniw+a-L~se&IIq@?$_kR({bo>IA%W&Tpn{{Of{$ISh%?m>7;fI6|};n79`s-A`TU z0nGkE>P-)Uy&k4P^au?Dn-$TcbO>$L(2g>rCZHW@n1dnFY{fGk$p<1q7QSN^Q+h52 zuOt1r@Vi&V>zLDjP!E4d#q*K)C5mu>5?3Hm{D@qDN3=Enaq84Uxtr+z{Kb!xrIs^| zaVPms;%(Pf!U-Z(JppsrtJo557Q@u7R_ZBfc41Dh zR_ZPBcWYXykEA(8sN6T_pjPT9X&3cxrGq7LS*KPSAc=e1K?4^Tw$dP&|IoG9b{gDD zLnJLqqInEuVFoP}rB>=nPk_;OP(JMhN}r_3^fWEQ2&?EBs;6hc%|E3}=s6(#d5!LT z%s!va7keN<_}D4U7vG3)LHc@{D!vonYb0+q#0W7Amw;h3YI~FVmL`j1T(NgAz>uy) zM-oqB4=p0W=Tv=M{~Wk>QRM@J!FTt8XpKjWwQ_4BFag956~I_44HFQfYTV=tOsSwm z#grPkiwaBZg5DafB~HwV*{y_)#u|??N27VT{UjAHw)hP!M#W;qcW`woT4-bn1vW~t zdGDs)s(Ml%G)z!;jLK7e?t%vJ$%5Y)|Lc>d66DZS*F;Ye<_N-HG@)N%(!U1pzeK(0 zW$^neR6ws%8NEjH>2+F4zo8m>0~33bZlbs7R(hN6q<3g5y$h1Qhqu3@_vr&otoal{ znfF6?(5LKeL@iGH<7WyodQQR@Zf6$Jo zToMqqFRAZov@0`(E4@8b^Bs*6VlUJgp3xX&&61st`w@hRZO+aYNaau@V|MV;=+O>4CeMfg`MMVt{N`{#F{xs;{tx4%35# za!g5XvcQ1eHI~~xz zYD5g%L8ril__2P~OsD3YhKpYTcg=KqNw>}PGMAj;H|eXf;+OExnf&4`hG8>5p53k< zN)q+X;d*EA<3_BzpW965-2fLQmJ{pROy~2vs;zVZzlO_v4;>wITj;_~G$-Z;fPyz+ zdz)WN*qAgQ-Yc*iLdVytqDS+a2a`n(T|aZt7r|vx8(m>Y=jmw!b`>x?=_rIe_1a zZ8%>%P!6IT@s3<92g@OpD?XF6^cY>m=dx4|MHrGNZj)EaVX}a_h&A#xSqLlMNt`Nw zDvQvzvsfxW)9_`{ChI}q{3RIUJFPopG3rI>d+Tr=G{Ih5ljI1z_1IPg_78U)l7Ye0 zENuy4uW`qg&;ryWw2!UmFqO?84uNlRjP(zBh#U#u;%w`AA8Qv`Ps&5lBBE{QN{p&8 zDo25H^(^G*#4N^O76$gQ)Y(8R4+F#o&NqN%4W}HJfD@+Qhvapbcs;7_8Q!}nrQzRF zp({(u;8uQ1*|=s+!MuCoU$!g9@1+6cxe`>E;L3?`QA*hT#0&n%P7KkvG|`ol_E6Ry zvcIS9T!mkG-_X=OB)^3yeoK8&^*ioxGG2T~<6Jof_oeu^m*#MreGXu{!)TeNv1#c_ zw4Lh8X@H9%7=+)i$&sD9;(r;PGO~Mg?-c!(P-()DA-%q(xvrePhdS?tm)GS73`T?* zd`%g9LAQ*w?b3KvGvM6Jl!wC+Vi&F|;Z+qB=8SHptJ-)~;`H=hRU$UaFtM5D4CCtg zUKKje@Z&HY!^&BD;4**UlK?Y&qn+~$3X8VV)y;Gb{ETap!|PyHTu4!Ikp_X|6;_i$ z@Z328Lgz$y*x7`KAz-*ZGZ?otL7@TW2o21UJ}?bFFi{4E(QqBR^2V1na9oGUTpxlX znt&rJ%jIkqFhI)L40ed48X_wM2iG@na4pVs7&zu=IOh6r+zL22^zL7fbA2mql2nj$ zLo3~gu&}TwrhF_FG}BE<@VapWaTjFsZfL7}sF&DG{lvW*R1;D6Pk`D#0c!sQsQnG7 zJg7!qIHKm3257FI1bQBTK1$Aq6VEI(2bzQ@9my|X)ZLsUp$-GYqZ$^&8<7Mo5j}1= z4B&S>8uwk{_g#hAvD4o;%T2)WaC}Kg=b*`UCwrbr5#{1K!{P7x)7n z*G3+0Nn+YC!VAFki=m;I+f6nnBZ1wpOqe`NAQL>)LVu_WF*j!Ee(i`YSR(XX)6x1h zK=3+f{Tpb(H;_YmQ$u6KqhA7=ehFy$CB&m&NIVQ&T;Jdv-$e1@dLI~PxJn_9w{rA> zpyoDqy>Hj)iE%A-$7UKldn?_Qb2nVyds=C;#d!i>#_upSovFwq>o`{~KE{7*C{1ElsQM9EE8g|e_i*p{{PLDiYqI^7Xy_kAh3rPVT23$a_ z=_m|(7**(BgA1L@0V_#B?TF98uU}Aar2Gbpe^CK4DI<|pISfgc=Dn|O<`^InE;=3h6d+|aQ!ApM{N6>)^aPk-hfW2R zu-qHM%tvaCkavkmg`AyDgJpz9$qX7RGwB%VQ6&&_JaQr@$QiLAog z2pvPUvKo0Fhepx;+C#KxENzy(1m|>m(a-V08qWBDI6lE+^eed%WnqHzyMW+RDC0XA z3wXeT{bbKxqzt2=zhG|$|2&rp0!Id9-$RGUuLvn2cpTl}OMx};{U5^`%7bs+4Z`+N zubiE$C-uqMxjexlE%f9V#k7wt?^8X-?!)qDsu}CXDBZxD$o2h%c{38A~ zJu3PP2v+eK=u5uy888!nxmsrgkmE26m<0xeLNE;25{u{@3jWM}Cm!!vaAIeU?sRtY ze7-|RCE0nMHc`$-TZAVQ%WkHh=0JZuSJK5AlfywJGjeV$*KaAR(ZOlQE<_R^X)8@VF7#CI&7Z#F8`LV3R zm>cu*GB##F7+!!d*c%Ekhf;qzj0$8vO-4|1q$~oP52vNF zm{!OUbb>sD8WA9z2PgeJ#V{IgxhDN%X0lER>ugGGwXnY{_FJTa3-XtC)k1|nSM)t@e62rWb?`YggOMH}St2%@s-QwYlm zfVvaCK)==E<Nu90;RzMC-z7PJf+C@#>Un8U2@ z3mWdU@WBJb9S3h%J`wlIdbF?*EM6%lwJ6mgDXlx(i50D@>qmE6C+JI5b z2!XghK-qg~Ah|B&lf@JN!K6+X-_a}`cr@mIM_DYidubL2GKpH|ZHhSa14ktg{K+OF z;*xI(0ZZxEWOFppc|Sd{I>`BoVS>E~)#Y-VB!9C8h78^zOpd~|5QqEu!`bjAUgV_y zR{D7>{Q_$rTj`grP*#K&zlKO4wV(4+E4?f!rm~UUdxdXQE4_-UG5o#;Gn{h}mI63` zxX;ee>+7vhyWIU7&Gcp~y`}F__;#r9ouKgDP~m$);cr8Qze@lJ;}rmCXG3`az#_I{ z8nE~G2be!g*t(z-C(#L%o$llVzmr0QBirf2R{A~WoL{h={$OsmK+l`t2yZo9fjEtnE0KR*g2%qp*k@3NwS{Kk!uNT3A@z* z_pXs1l1JPKM1oh5jZ_rGLxQ=zDoO_~i`IS)K`@K1=kMXNzIT z`y49I6_e%p$p8FA3+gOG_k@^~^H((yC`Iu3cwPpUaB(Ke^%%v11|1+bU>rriptI#E zxU-S(nkavaJ0}@NiE?AX{62UA*@RK2g6HrpH^Xyqm~3_|>2sgw*k$t-Ke)?RxNq>C zulPYyqwk%ZV8sZu@K=m*=dT!*xJ5CymHsH{2D-|yUP&YtDa?64xt8=1*fkRHHsAeH zGj}W!GyObHyIbjFQ|XgdXvu)@;nM^hk>PVhECV2d?~zG2;ONGG;>PUufL8zyYkQMQ zA5z4O5q4Ux0%KeaX1E5-a4j7ruLA>IPt|f0t(P~@nes-uP~L?2@@BeQ-U8y^3gX{J zPs`h($?t%m+zHORi~b_-p})(`nyAM@3s%zwnrsm)=1!JpfJ7NIlAh4I)PYBPBN*dJ z5Pul`RBKgsus;vj0C$EBxXiFYg#H9JdI>DezNv}exIT2BZux5gLPH~RtyfTsmvy_4 z^&RB~nVh%bmH3=+HaG#_^7FON3~)jQw9lkK`y@CaSXC4QA#6SyBv%!4K64Tahks5i z9R7vxwD0;?vxHdU6f$KrL%icNgbR{D8=#d|Z!tH)0=P3eiOUj7i64wsN)|&c`Nhyn z$zmubzZjY+Sq#<04~Cn}=zvcvJusTDaKNVldqo2FiZGyc+v#N40-e%Im&hmJ80>&Ayc53gle9xV zMK8*y!LqyP1NjU{`7G_hXZ)mmPGriTiLS^G50Wp4;qpZ>PX0nnkiXRI#_OIZ&`X-# zTp{6evu=n`v50DRL%tF$%IxMKP|DTphS*h13fK*Ifz}yg*e#?@xE3>6MDRP-CJENc z@mWg@PO{diaC!0FOXKp2!vO|!_O?PSzF}dX4Hs}7EI>nb_i3iT#n|)h8)y~0H%bLJ z=~khQi(7>zJ$M^$$Li=C)J?vLnY;yS@iube@4_;?N7u;T(pF^bIm|Tqq-_CivEt0@(Qrg? zXB;2CV^_&SuMg_{y2&S*kepir-x|vif~T|U($OB-2RBL?K8QD{v*E;S#l3#<@DycnfA1T?5)fVtc?%2JLajT!7dUcf6uv)VQX&t{+hR2#S15Ly_`63RK@q z-N}aIsNGXIo|90{*$4jJgRQr^3O4A3tLXQZ`wg8AZpAlDjgBYeAvF(n&u=Fb7+;bf(ssnfEP!Rr=SD3}1~B|bOn1NaAMBu?C|h8BPt$`zfP zMfQ!9%etUXm+os@MHlAd-uk(qRpj90hRChzD)LUaRKG)6cfGWF1H__X%?|${AecH3zO_k2U!QvVOD>dV-28H)*wAmlbl=~Fb}OxD2~+$#j)Bb z4y~RIRk(h@JlNuLP>;MGm@mE`YzS|aK~*4-5f4L(3iBTqJzGUDws)d8RG>JokP`{4 zn4Ns2c91a`&%e8Lo&cGLqCi&;^!22Lc}nCVZ0N}nmnY3Oa0pgz{mn$aPk?p@pp3Dr zD<8_KV7us3;-blND1ZolvBq7Jf!{On8^teHu1hjGv49O|eRwSz3Mr4JKf-yAN?&W; zAIs1+`ujCvY@ne{;2B0+L|?zw##mNe4E2~_nH*mpFL zd)q~I<{8)iP@3Z4W--85eFXqCxtQLS!E=$ChGx_~Vo#u3!0KvXejb!txdsh0AGDcr zO5mB?l`94VmtqK#XwDWexFt4;n~(`LVzM4tVX_LyvW8QZHG;ZXBdMQtC{4CT(E@8U zEw{$fbyf-8ZjGn=tO@j}HIZ7aN%V|0nO?P~&>yW*`lmHjWLVQg%$hEWtr?=!nu!JD z!^K)_mN?5gQe0@w7MrZ2#7)**af>xiJYdb&oOvlS0t=u>uaVb+n{KDCv|4lNKKh4V z?2TZ*+^6z72xA7+`-k#++(ogY?rpgVcbVcAB=9&J%^~i?0e_2!1OCRHZy0gMc?BpC z-0@N{yD1NAp2rf0ctESpaHzw*f%|o>S~(VV*g089ST|g*wd2|T$|tm^S$r%-SO)!x zHB2mq`u`^yG-zbs$X-m7<8-|$8LCYj^e^t@>$@8uUic=nQ77UoSOJI^oezfKTvenH z+5m~K!d8TP;G?WqBv(Mk7C^)@c8J2od5W`mdG2;mgj_5_6r|R~a0~ohq}ehrmYFje zHm(?Kkf~b52>7j;&xk`dQ8)cOlAlq2DEH2}xK@Fp0_-;uqa^4Ms8yXQw^mTZs-l5b zHI267G}T&3W!7q{uxeHI}s;Ekf5oOcOzQL zpwV!kHp_d#D^p=y@00g~SCX0c^Fx}gC)wXp4_7`==E?`Zr^9e{|9BOeG0s?T=tLf?zj+U^jzcw}4=` zf?&6SV7G%{cYt7bf?&6SV4Fd(dqJ?9L9hq^e?~BTX5J8tW`I3d>fHjCmhsHJQtxKh z>)2L{o|y63T)D&NHCGz~<~7&nwG95#8)>_o(6-CxQo4L{pIr+6YaBjXMWK6G53NBg z0-KC&jdRRhu9#>)DJCtpa*DChIaf@sXckjgI~3VEmr;OH=Es;_nj9VKU=?plzx5BOpwEHTFX(XVpES?d?x;Gr*Bdy#N5aVxy2Uu%C0KGGhV`xR`Akoc=FK0DTY|}C@!ZG+R3i;kl!tihRn5MAsVlZJzMjIqTt8snOv2LD2yI(`12M*5{XkW$7l|r> z&S!^e3JaRWA|!Z{JBg1r_w4Lo6R|jE6P3j5T(Ouv8my;ad%XM9mVK-T5f8bx}AKrVlMCC^mHv<*I(vOZBHg*y35N22hzA zNEP^9tOnDuY6z`V`5F`Kn3Pcz+gdp;D5iY*Hk=CwOY%p`cTmQfx+a0A8XW_$nAhkS zfElDlD?1D7bcK8uiZZ+bi(4{nxfaI&=19{NQhRB%6IUZ0j1^4O7zWLcXmkKptr48|2K)SUP7(Y^os5HC zd>-K~Xk^$ol<`nwsFNBC6dp!B)i~;>N@$3hkdVtHt-&F+DV%$?ph0HJvvjy@8KH3ACn_`9i#|A5w6_fTIf zvVx=q##e|beO9WXRqtHKCJ4MUn;^#KLwGQh0r`Y&q85*0%`Q>LGT6$0*Bg5%#3W=; zVy@osunmV>;Et@-8ycGY+hx1NS{CG1aU#q~8tFaKm)=;E!A2mN1lv698|(is z?fJd@1GK`&lqvrR)^}+govL#LwkX6REhkW;0xN6sBe@$(pE=lV@v-~_E#`?nfry9( zUqm9}cymq`n*dkHPkGO&<~-KQdIX3FoHJdYdC=5<19D*M^SM~gB`l?k?+JMVY>3AH zvF{oC<5*Sn+(T3LVxJyw21>luum0r9&+vbajkNQD9n?K}yMPMUWOEp-#{C&)34fT! z7e+)~+AEpTX%?twR%8Q)rojP2hfntnVGe2F^myZ}&3oS((p5B_aOqWTGOP-#eQG`Q z%t#(wrwH2N7O!;s!#f6}IYlozZJ(G(wR-)$GC zY@@zO)nW|oSxV74=FY{J92*rc+V#w5zmy9XK*<>zyTd_7+EO1!$jZ}Jm{=R&-4 z;kRtTYCJyXK`-ObNf0zG;Z1Vt7E^7;%74+&PHPx*$H&WWD5WERH;=cU-=sl45-@ zCv?>Fz|#xB)6c29`UM@NehE+PS9A!T$EueT;(ZX7!1S^c4)x^g-C68ajZ4gToSAQk zDfwM6w@gybx7cDbfcGtBNc;^{W30s$kR|p*?+A7dz}2}F%1Yp%*9~+Dl3tfLi!017 zzGO7i2%gYpSzO81tyx^POI!`}WM!XTS6|o&a$a@Vcg!MEcIJR?(ZN= z@6&Yk0hOx{;lTbL&g&mw=Dm` z6gAjB@(q@pGU##`uR; z6ILf}nm`RWhxZ!jIhZc6x?q)(W%)7CGndIpkAa?DH9a3N^wa{yRMai4JhY4u-Z4G3 zdohjd@5tXru@@VJBCgfVwQ!h;trrD)X{9cti&=hTPs~jID}-{hEx>S4^)(gghNhV9!{zuTm~ZWr@nne zRDmAc6R+sv)G=Z;Uh6dW9@_ zb6V^&V4k(>sKkH;p#cxdM~Jvf!_igW>ZR6w0aBm=G5L*A3G|FI5psgLFqQ#{M-9cm z0B+=zKqVXuAn_+>eIOqM6_`ExVoA8Rz(Vd5>CV}k$(!THTU)>=%_gIng{{GXan?SM zJ&LmIv6O9(rY`mvirI&0Fgc%;g;S)w=Rsl(A(F^n_Dz7^S8p}O)(tEN=z1Z%a;*Tb z@F?fOcz(;P4wHSQhMg4rCn?PgL?^({*}L)`;sLlR4{jF^%_%Cd5NAEyEFNhVTNV`+ z;Nele6D6t;KFn4Nsb8J%-As8UwqA36jI(hjhs$XWd=d^jc*Z8?t23AOG|IH6W1cf; zfIX82+p}nteI$6ijK=g*GCILg+uWhJ#JdPl;ina=y%WGOQ=4(2h@uIUAkxopZ5Q>nhZPbgHYd MQ{@_Ut-9|212jSjwg3PC literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/reflect/ReflectUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/reflect/ReflectUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..a1eb773b87276bff8eb9ac19977a493b631a0b61 GIT binary patch literal 10872 zcmb_i3w%`7ng4&u%-qRv2{4dAc*yV&@(u`sB#68ML`lFTfM5|PlOdTfnF%u!6nraH zvG{(1_@Wi8yS71X2tE*ZTXfZ3*KOTyyDz&HY~8imZg<|5l*Se zKdQ;kyg1G60sZz?!@S#j}HJa&@b7Ynp%~nr&nRRKK`9rW?&iBd!q9+%4WuaFVd1bLCOT4nwhZd<-3uky` znL2s7CM$esmpV-@^r1s8QY-3wvQp{=Lz`-wn`;|4t!}JcR(FZOzp^8^Em#!|#@ecy zQi({ct%i%1#AC@+FqT>yjCO~4zdUlSZXvT{*V)261{)r%SzHLq$Ec%t#P zws1m8jVIcwlF`kxJE~S1bCl|tABja$3k12vC2Iw)CGl3u9=0+P3paFkwuBSS!Imgz zxH%Mzt_>z4>fV}krP?D&LB+~Yyt69N9p4_Q;;1totLjcgqRFa6cylxyN>w$Q*EMRA zXX=sIw)mFt@^C7p^g6HDVeYDy4(_ON%vb=FIHs&cg5hn3jQVh@J>J?7?4+Wum~qLC zw{&C!nck##CE{J-L~6Ua!NZk2Qh6a23~i|ob{W<&YRFs-cMI~GB5ko?syjjP7tF7% z9;nHJGc%#NHe-F0W&OM}+B)z^<*nh#0p?3YAaw%wHj^c81qXdJ@=}M0G^Yx)y}G^2 z_H3@9>cT(4SPD=o4yA{dMZ(cmi?u=wCp%QBi$=q3!RVq;D4a~zUKt8^r6TbdVanUA zc9|}68)HF{W24P2L%1SxBAo1wrZj0_;^Mi4#(A#AIg=@Zjc0_q6SF1JU@}?boUsTs zC^2(8lWl?mhrUkRe4P%$=63}X^btd$nfTLHe&-?Pv7~h{(bnIMX-0cYd4jQy18smz zlZjBl0J-x^7u0C7ib-71_9ysyP{Gbg8!CfUelz?{y%t+hc+ppOh%M7$avG4FP*TH_ z8V(8?|DWu#k26gmt4+#Am<1IcBtys{AJbcbA-%q zZmwCtBy;w_1GBJ8pBRa?hOZQ8ZQ+=y>KF%MZ0pq%mD`OJNqCC}gV$K3$!ZNhVsAkd z)+EyrlBJFv!j6M%b-WRyw+a!mVy6Y4w?$Eo#)jgtZR{i(&HjdkE6wia_?lR#JsjE+ zZWYuzMOwwzp+tI97KyA-H~XkYQW0ALi@%>kqGP3ENgUGE9-Q*MAoxm%m_CiWp9$1 zF+tSv7#`<#7G@pa!FP2$kMHS7f~}YP#J;Bjm+G=X_|q(36BKR?yx;fC$1m+Z`Ro%X z_w72l>*$GFdjglTFW4A3@yvrKzw^C-CL49ROg6FP>o_Tw3j&!l2k$FbA>Z%2{luN$ z{l!f$pSN=n|Cmy0l10$45f9jsLv0)d8-?OL*B(=i~!N zPCj=()mf#pyV=Yvn^7a}Jvz@!64e2czB~|b=};qYc|hd^HROxfKjNQs{006}mo{nF z*~+nnb>L_+N0TkOM5R-ZQ#nJIn8bDIlC5Mx=GeMeD$Jf=mxM5C@-q)s#$#=|q@+7b z@X~lU%a|_P$y`l_YD6!Y~CU9OU=b-6~a)#N%|u9q8hxeT^^D>y6oi~-o)FwJS_VJQ!--hjJ4Ws<{~@7ttvC=@`wtMDbteyqjZN<9~p?n zQ-ML2J}Qrq9d#--RtHohJTCB+2bw4>5Z_FajHZ~ZeF1fbqR%`-CW9-GY7Ym3p;UJ; z8ZZxw$DBtq;f7fs0!B_%Z6|{wV@Ln2Lxk(B)EMaOPNs}g=(bpQG|D6+n@Gc{Q+0Vl z_Or3i$g$%u(bnA=j-@h4bI&)KaxyDz1H)MnR`WzHZ+SW^Un%GUE8G1x{B z+~GtbPU21jhkRX)#x$QHhVQ8l1EhvJEFRVsXCs*2x)R0Bdp3~zrqo@q9ENyX&4NUB{h%ZcKW z_F$rkDurXAaE-&XtoahEH7wcPk~BU}Ee>y{Lj6AESO+;ISTloCk?ON%4eloA2CJ}< zyq;8C6|;UVt__>xShdOw2eZt5IY#C>?$J_lqXhPAYX>>RSjHWJ0lrcdGij7eE0yFi z@g~EnhJK;$=}JqfLt2djZGA%Fm%63diK{CfsY)iRy4m-3^81}xq1AERN+Mp!hR{53 zWbsl9O{aKQ^ZM1b2BNB5f4LK^qE5voeQx%ou8~&5MlWmKVpZe?>&}E|7EPR?G6>*V zbYwF_Gna;0?h@fvs~Th_uR*w%RXk{m`cl=TUapOzX@;;Z(k>$lodyh|s2S5v!rG^p z5_pVyP0JR`(4UZ<8U}3QTT;sW6sF#9>f@~uc2Ws$Bp3Lx&r!eATdg(B@{#0fws(;b zKOz;^smN8y$y9+~mJOBFH`%1~U6gXUP<=#m&bo>>s5eb5l3W&v1<90Al*HFf)lWE~ z8=F;4Y?V|Rv~iAfc16RcMTL~H$Y@v8f<;!tK-Js z{084Ix7ZBozq1CD_!k?UfBtb{EMka5(8aqPoF$BU^~aUxd1zG74PzJHLl@-YV!}NhL_j+()u!gP9Jg` zDvr)^<<4;zxC>nSF}I??eNMIKMYz@6U}1UoY>Wz+5OL#!?NSgol-40?IK56{zL-B5JJ^n+-SThT&S;sc}UyuH#7UsmJxW zfmb!U(Lf=GyKdrK4t8>OEOJg`I6NA@uHk0m_ZA@xzehE0XSMkWw|y zUsu{-8d_~yS>!spAJGCguddepn3-j&i`HkX^;8sj9M(1*M5VntUjZ|4wTrl{&n<82 z!?1M*Huot1xD042F@)pusNOs}wi*SPkBL}-8eD)2v5;mjK?F;=wwmxPH{fcduWBr~ zJbXF~yA8O8k~p{H4uVMo)4*M40jK8fG=R&Y-xY+pJh2;h;cn`35BHe-P^<2>;2iUcxnc~l998Y&R{z){S1-mL!T4NM^pLjIgklN*c4>lEO2vUy z^yO-f8p)j|Lb!&jF5$Y3T)Bb#`Wl*XS(;z9gy{i1ND0(44bECTWUxv}+Jn6`yBg#1 zumO$@i8*sGZB&e#h}=^+Ps6^`JlbXRZXX8_4UY_W(C$&2cKEdgP@@4*L&a;D@l^mi zR8d;qiwS9T&dQ*3h~Uh{`83-GXBu4tY$iBu)Ubo#Y~hMH*QB^=D;6_-Tui^)fw$NJ zOk*0DMhh4aUXwK*tTZ6Xb#O&Jv$--<0W+7ttZ=}fc*5^QR`ks$3@Z9|mL7*z_8Q#M zQ(9Kuhl$m$BG(y^CXhv%B3EfeFDCU=yKSaT;d2GeDF0NB$f!~NvpL|ZQT}P>-fB;g zr^wxh>8}vDTAEak$yuzrp22k^f!hf`T{;@KkWII8b_p)qSYp+!SNi zvzY%RhMEG-KY;wSf(x2dqW| z|0CqgqfE9>FlX&&zB$OSe2P`=X|&SIF?x49{d+48F%LaY&h+9r^xa+>|90|@($qY}`nf#1_LS-&OLr`-KY03tsM0YW)7elgp%m4MnDzDq?>?Xs zzho2jAy@p0F8UbL@CjG`hK9nSEGZ!b=Bcz-#cwactDbn{6K1)Z0~T7d842L^PFY3MrA&~>Dt>%ehV zn4ENZEoA9^1q!;I)M+j}7I99qDlw?olikd)c|pT#{8u%-Wv9sYGsG%3Dc(WJ7c(ZyhE2pa$7porCY~TG<9#VrbLbGH6rm&69 zyiZIu3{_=K0hcp+h;iZ@)`>Ce#7c^8G)^4LQImP%9t<_7oA(%Jud&Wv+yCt4JbML- z`>;+qHpj>T#ygR$amexDugPgx6#N@Q z%G(wNXAlM2j{{WolKS#jv3@V6X1X`dZeVNSztrq6jE2^)_mm&Q`Xk_9z8c0=@YO;a zw-0kOoBSKOdR#9q^Kat77I8Vp6-PO1H^!7xF^&~&$eg^?QmepU(5`!W8(^1<_5rEKaHZx8?aHuGbp>?1@tqe1wf;z|ygUX1*ASxXZ5K!y~@ps?bq+4hQ z|1h1tbIv{Yo^$TG_x$eJ=RbbxIDmSLdJ#mQ2GautF*o`pjcZ7F;li!*GvMXOp!_7| zXNwnGk&@hyhO~gUwWYhgv%PCg&sqV|ArNTGq(^dQI=8`0jV4rj*)8DjT(|C;-u3o8 zIo2#t+_SO0t7p^db-mZOcJ!>deqHZ+sVNn3w|A^r+rvRAyXm95$hC{&7-La-iotm1WFLiy(Du&V~RgD6{m^qS&*0-dF2hCU_C!lO@6_}mMWVZGW zw|+|3Cg5sFrjxlQTB@$uK-+DZctRl5nM^0TMu%dFY>ydBaiq92({H9WnAxP1^Mj7u zU~)vDs`2y<8_J|>M{~*4NbN{+AYE&_UBSF0M>Y;7bBXR@v!50n)onF5 zGJs5fF450W1Kl~Zf9tj8aNf8xIkMi2b3$@tm6bS2fBTWdPEnA^d9hY%^1rGacNsWS82cjw4AXHkM4A*&Vj6U@DOw$PG%NHJRfcLboPv=Vk}Z z^wRp6H4;qQ`_CqaQwcdGU=Lh3YG!jXw>~?bxNT8vrJEEp9%BSbjJ}l^J^S;MI@0mP zb^%YE9xYzOr5ZlRUBwgAyZZE&YUJ^$xuIbJEpJF3#d)^wwNKuTMdj|erI~o+*-`F+ z5!=4<5i7BJ1{s&(#$~v1hO5NuSq}nwHZhdhmMA18}<{^7N4_lU2KbW%dB1SqWf~0wm~!7lFgbsxTorx%`ApO zM>|Gxi6I`CDQEO&hB+%zU66U(;EIm(40Y6OU@!{72Jj+`5OfF>QBqJ$c~*fuo3c?L zmr%Z_KrW?RRv?Edmlwztlq(D5Ig~HP+!=rKC^MP#9hS)?byLTK;J2DmEh#e%XXFWp z$H*wC;n!six)CAow!N zcTL`0Nw^5R4?;f-HC(6cyFj2<7Z@%%*YIo__MncBij@gRh|%+am5RvE;S`gR!PpYY6FFciwJ|?9y>JZDFhC>@X=3MSnqb@b7sZqBY^{7#=8uh7B zzZwk~fy#q;A!7K){Gu-Eo@PALjMwn!p$7JYpMLCE~yJxOc72~sh2qo-Rj37$6%g#kVd*QPS(6VN7S>fl-HzW2g`jl4HEvcfjAgl{G= z^q~|c3p&QD*Uw5zU>OFm3Q4TN7F>-Kx{=0B$WSv(E{ko%yE`$8Z<9NM+wmfH;5F>T zd$>b5@C9KYCCb^B%){McKE5OtV@!159-%{MzqZDJWRT|G@HvQho7^4KcBaHeAav{dhjw{LO zxE8;`Ripv>Tx!h=(Bg(X_X<{6D+rWI1(sVda1^SS5bPOh$<>YsAZpP=^m! ziX z^d!Ufpo7O{gF|B;9NO~W@HpI^3y#9mSfLH?m2gp^rJ5=ypsB~)lVZ6;jb0ED)M!wR zhALfy$K1;uga=3bARb;|gvLUmepA$Q6yCk4n~}YieCdqrv*cMbvY+yV41hRjc9ft&LtdZ+hoXs)xCoaeXniF`=gpGt;4Y3KI;l*=;5SqwSBdI7%6 zN_d#p>k;1NAMhUk5GD8#G2}2R@MFHLkMjyY$;bXFzEw}NUVg%=IL^EOQ>?`^Schkc zKtID5@f`Nx7p&?RZ~(uewO`{gR_>FmQLP zvO51I-o?Mg`#2>&WG`|?2*oKB(Dg$+(DQc5graAAou%fbc?b;O_R zE#!zQ0ix|(mhx4Km%YwhB1eOwp_iNwD~V1U$obK#2pi+jE?$>WO+Edn@2yg7xwi^b(-)pJkD-;Q3{+$>`_ zVP>`h%K^-igXH3b z5KJAKdRyN~g?Ruaw!)0-;l8}?gnkuuOJ;OS;(Gmzj??u8dQ*?2wff2)dezGwlJ=c6 zAMSgw*K+)}9$uj~g!Q_Iqz9_Za$Ch@9-2C#*IU}gY3-78v|OM!=?jP-PJN|b+bbJs zyXDQY5W%kB&Br~&&c2l0{2bCMcJYgt-YZBOd97D7ncbwDsENs+H>9I%n(yRc+{5JG z&mMLU>37&1e~-m{nDl90)93imzDW9e!oll&IN##iBg^F`9(56Y%Hso1q9o5X)4x4- z(+qd-jCMbxv9$-Oy_stT;bMC1nVB(p;L8w{kKjoZpN2wW*rarZNS$$pmreW!rPGuK KZ{cDEpZ_oJ7fc`k literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/sign/Md5Utils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/sign/Md5Utils.class new file mode 100644 index 0000000000000000000000000000000000000000..d8cc407d8cb849d57154c37bc90990493c3f94f7 GIT binary patch literal 2116 zcma)7Yf~F#6n@^2WJ6d=8t#`;s4bFkNozq-pj2)aOj?yoi>P%;mSkzN8+SJ?)X`7+ zL+ls7Vx0kOXZY%jAN@aCpLY{V3h9iK$;o@pp38HdbKcEg|Gqc|FovgL^x#$qGhy7u z$5PCOFc(HM=B2VA#iD{w>d}QeAuKgu8J{XhhS7*r2zTZ0y)Zt*=koN0f|W2mSEX31 z$2#uIR1c)s2;oZwn_+x~uN8bFP?xu|0@0*pXA@3-du%6>w6a;7aqo+%>>?Kmw{>(23w6!Yv!fskGx`t)SX9OCpdR8gSsw=LR zew5ORUQ--^K)|)`=(`f>=4`B1j6bJ2Irg(v+7{?I=P_H_-uBoP1`987h$N|2*XJS$ zG&GgYS7s7ku}XGYUo_+d8c&EEmF%gwj6Nbr&5tX1sN!4Dv?Wb}wbjLu2^DF)t0Dtk z#Wpzo=2OmJ)1u-MdIeM_>07XE%N`vaRghC*V29nGMwH1awn z$&1{|hZCH5xuI(3iWBqqR(h!Q)FbaQNqu_)$w&KHdKa~{YuVh7dJd`l+7q}|P1amq zbDYW8`L`qb`EKpu%bT#}P2nlPGcO)xKeYsZg|x;GA%5e*hbxpp&)A4}cvi~}xQc7k z!tyrIB7=WH9X~)U>N`SRa_9j5p?EZKfZzcXYN6Ede(E{G(T3qapw>a`d#Sq6jd}*k zLRS!?Y-7Q8rtUxsI{A?vJ^b{tRG$YT4nOTa3?fDuL@VMLVzwSMU>GB`5e7!t4{)7d zHn6GS1~c*AR^Yy14heTT{wD%r|0NnX;>Us9Auc5ArhDSY&oCPAxp@(>tPsBkYi4s*8@eB~n2rG4Ljz k4Pe?-;(G*O$P&4=m#0PaNKasU7T literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/spring/SpringUtils.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/spring/SpringUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..75e11ade8958b8b0c8ce15fff56fe2cbdb9d4edd GIT binary patch literal 3433 zcmbVOTXz#x6#h;U7?TbyrBFo$m1~*;Ljf<)S}3IzArvH~D0pEyoQ6Y_Im67PfIj=I zudL;}FFq(fNPX}J_?uiV_nw&~(_{)=KJ-k^IeUNm+t<^-|NZkX0B5jP!Z1E7Vkd6$ zQX9Y`Zt=3j%WYmh=j97t?v!vBUk>0ad_90~aIZufwAfhYtz+Q60cl{xKwV&9S=#Qr z?S$HYDsZK${kj!+zH;j;zTK3MwZCeyNMNnRLYAZ56;-!pqfg7Xmm9LGf{?!#yI7L~ zyKK*ED8~+!c1b#vPeOr{nHC6>udSIr4Z_(zrCb$G3*^gVO9F*ieP0R;Rh26jTFqtY zFS2d2SJjT)Sh9V^`{b|?K2U+cc-7HO%WvtYiX3P*wQIFP)d&(_ty<*rEj}WPqn-}J z8@_gA5Ttw(xK_@j|GxsnzLe&kIMU+)1qL1n>+zNHbytGr=VkDssdjItp-(4np=nW=*HYX_1gI#?S+~mKGVb~AMu)a zwI30ISZgfOF*)qEa--6qoTXiB$Mz0&mI`VV=Y|ZmYe2>6SR`gW(P*pm5+#o1Ea+aj zsS3MOqFm23lpRQ__M=&PcT0{4x>>>{#P=mNDxY2>X0gghHKN)u5`mm6K`OziQU*^Vid0%ST%7NM|c@G>1BA| z#0R{5gtI0lF=e1(q6w-vZNkGN1HOqFTr?1v2+=Ze1y=>ewmc5Wx$Z28YeHaV%g;n= zeBS5|RZW$1Ityd-i8k}t=D}0W(u(IVn+iGuRp9*QqV2bALDyQ&16I5wSNN5oNt{AR zc^+q9;PR_M$%idGkOIfDplt0~-IteIbGH3*noi62sU(tK-PjnNL3knybn~bR3{o@9 zxogVTZj;c77Ij=Uh%YVau30H*o>5zs6*#a-(Z#Wi9$)G!6)fm*uIX`4j+WDH-JVEy zFhT#)3?qkK7@@Vq?Lq%1mAFIby90u|hSoP}XVF*CcU$ETh~MZ(4sX%AJvto0+w|Lv z1+WkMX(#Xw-X$qk{+M<-`feNl9l5S@DLUMT0x2-SXe^Wfka)yM;61u45X1rcJBULW zp!r_VgQRe16Hs<94O-+;fPBRC41u$orRT_1)=|jeSK6h0ok;mw!cmmzzDsZ{lCFR; z#}mwZq|e@)G3K@$s^g5Bk0KZcVVc5zKE>C>F&rluzC95I|0M0=0P)>K=kC-7Ut;^G z$vzD=y5Z>e__RHTU+6UL)0GULiaZfXDg(jjN4TAV(2mN7#D)fIc7$K+kMQzKv0@L; z5hl&)Y<|UlO#M>KB30;gf^~tS#uZR_fl@b1ah-pcLMc$bERy5A#Fft{r)U>5yg&@; z#Yl2+B*Fv+pCiAH0fBXxBC0Ks>Z9*TXBL+dofCL=Gf<7(vv2A`o=$)EZe1<@rI zm(LGW?mu?O6)=Q3%#-RT5xLwy7&dVEf`L!T0KHs^fwPH$p*AV1_@VyT{nP;l0av4n S&*2(D=dgh5^yTQ>0R9796g;Z{ literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/sql/SqlUtil.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/sql/SqlUtil.class new file mode 100644 index 0000000000000000000000000000000000000000..26050ee3a03555fc8212c0c67422271d65dae898 GIT binary patch literal 1603 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 literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/uuid/UUID$Holder.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/uuid/UUID$Holder.class new file mode 100644 index 0000000000000000000000000000000000000000..b9ef6fafa9e9014774d6a2456fc0c0f3ef30ce93 GIT binary patch literal 569 zcma)4OHTqZ5dO+*VO<6B0r=!#B6zSDZxSyUm1Lud2=}tvV9Byg_F=-$G9EM@`~m(b zGf+3sU?}_XbR>gI>B~SwI$+qb zC*ncyP&#p-q8aZej=UAVtEZ<7g^u!7bjFZvwTBG3u68Arv6U~c(^7Y0WW>xnRwgtu)+Pk9jQnd0P4wa?>7c_EKuR<1IQd`#KIBd7%>BQgZA#xq^ivN(QzpY-7hl0S3e2KLr>n=@dNSkNM4LBAtk# z`4^eyFqYBJfidj0+P@7HO?r(yJrP4aX_34PE$W;QlOj%*kR@;K}uD^GFU^ItV!7%eUTNIpYke!lmGw# literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/utils/uuid/UUID.class b/ruoyi-common/target/classes/com/ruoyi/common/utils/uuid/UUID.class new file mode 100644 index 0000000000000000000000000000000000000000..fb526f05408ce3743d0165e032b2f6173c0db04f GIT binary patch literal 6423 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 J>*0Jv{s&{-s*wNy literal 0 HcmV?d00001 diff --git a/ruoyi-common/target/classes/com/ruoyi/common/xss/Xss.class b/ruoyi-common/target/classes/com/ruoyi/common/xss/Xss.class new file mode 100644 index 0000000000000000000000000000000000000000..404ec7a498df5127b5db6135be791bfa38c560ce GIT binary patch literal 807 zcmah{&ubGw6n@jD*|zaVtJSvF+Nw}m1qVUARD_srv5+=lvn@hS)6KX{nVktcv!#0! z^rqm&i+J%ao{L4j`*%diUm?ETh=~~KVcxuX?|a{S?_<9G{PYC??!k2jt~s#ez%l_R z;>NI$6PQ_Ds~)gJMkP}rZP-|-aGk((St)Iq6fmabP@-AQ%Gzw{=}F&*=hvly+m0N@jYS0zU2Mg zaNOmAS=pbZsmCqH8m*9a4BI8hxN-`#uB8alYzuF9$KTp^2`mmT@mU;li^vf_k24)e|`H7U;)cAx)GO=KvG5kDG4)On3XXH zRV>$JT*nOw^CGw);iiOJ49cToKKEvQxmbMgWQ`%3+tKzlwW^tAwZJXIET)rQUx${41>vkd-tVuK;DTHhvbU(PV~wMK1Aw~E?Ul|0d0y`)t)G|LeGJ!gPd z44YvpSE|=kt5JV%sI;rqO|@y;>T}zsBF47H>lRg(4EHa?O1zVq<7ukna>rHjt4T%BNG+wIc)=61;H$+)&!+E^5qnCL4OkD^<;et^% zHQumjKyO`2jttI%m-*+dx_sJ%l_lnKq291c`a?tLCDMBPb3%v;`Vo_YG~-={u~wC=Ry3=izia4bNq2NR5}4y+-uD&~Fqy=q1}n&y){A ziqK>cp*2b#l`JD0OnwFX;-s9THRL#Zah~=9gc2}-L9$HT4XxsEg{+_KVCo3Ij|c`1 zXyg9`DexHq{{cn(5NN$l)&L|r8^(x(8goy2sDwd&K~wBTF-BA`&^`ozSYD#fL#WTI z`ix?tY9tywLMWF?9zx2`9HL7U^tBWmr{kB1eypWn#3>_yyD1hUNQGLPz$ASA%S3O= iE3!Z(L=ice`hhT^%g5-Q!;!r= 1}。 +节点XML 示例: + + +${coll_userList.size()} +${nrOfCompletedInstances/nrOfInstances == 1} + + +配置截图示例: +![img_3.png](img_3.png) + +2、并签配置方式(没有业务,需后续验证): +回路特性:并行多重事件 +元素变量:固定节点ID_assignee,意义不大,代码里强设了 +循环基数:固定为${coll_userList.size()} +完成条件配置为:${nrOfCompletedInstances/nrOfInstances == 1}。 +`nrOfCompletedInstances` 和 `nrOfInstances` 是并行多实例活动(ParallelMultiInstanceActivity)中的两个内置变量。 +- `nrOfCompletedInstances`:表示已完成的实例数量。在并行多实例活动中,每当一个实例完成时,`nrOfCompletedInstances` 的值就会增加。 +- `nrOfInstances`:表示总实例数量。在并行多实例活动中,`nrOfInstances` 的值等于要创建的实例总数。 +节点XML示例: + + +${coll_userList.size()} +${nrOfCompletedInstances/nrOfInstances == 1} + + \ No newline at end of file diff --git a/ruoyi-flowable/img_3.png b/ruoyi-flowable/img_3.png new file mode 100644 index 0000000000000000000000000000000000000000..905869415b844e0a25bf48436570a1a3cc61fd02 GIT binary patch literal 46288 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 + + + ruoyi + com.ruoyi + 3.8.4 + + 4.0.0 + + ruoyi-flowable + + + + + + xerces + xercesImpl + + + + com.ruoyi + ruoyi-common + + + + + com.ruoyi + ruoyi-system + + + + + cn.hutool + hutool-all + + + + + com.github.xiaoymin + knife4j-spring-boot-starter + + + + + org.projectlombok + lombok + true + + + + + org.flowable + flowable-spring-boot-starter-process + + + org.flowable + flowable-spring-boot-starter-actuator + + + + diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/config/FlowableConfiguration.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/config/FlowableConfiguration.java new file mode 100644 index 00000000..427aea9d --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/config/FlowableConfiguration.java @@ -0,0 +1,44 @@ +package com.ruoyi.flowable.config; + +import com.ruoyi.flowable.core.web.FlowableWebFilter; +import org.springframework.boot.web.servlet.FilterRegistrationBean; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.core.task.AsyncListenableTaskExecutor; +import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor; + +@Configuration +public class FlowableConfiguration { + + /** + * 参考 {@link org.flowable.spring.boot.FlowableJobConfiguration} 类,创建对应的 AsyncListenableTaskExecutor Bean + * + * 如果不创建,会导致项目启动时,Flowable 报错的问题 + */ + @Bean + public AsyncListenableTaskExecutor taskExecutor() { + ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor(); + executor.setCorePoolSize(8); + executor.setMaxPoolSize(8); + executor.setQueueCapacity(100); + executor.setThreadNamePrefix("flowable-task-Executor-"); + executor.setAwaitTerminationSeconds(30); + executor.setWaitForTasksToCompleteOnShutdown(true); + executor.setAllowCoreThreadTimeOut(true); + executor.initialize(); + return executor; + } + + /** + * 配置 flowable Web 过滤器 + */ + @Bean + public FilterRegistrationBean flowableWebFilter() { + FilterRegistrationBean registrationBean = new FilterRegistrationBean<>(); + registrationBean.setFilter(new FlowableWebFilter()); + // Spring Security Filter 默认为 -100,可见 org.springframework.boot.autoconfigure.security.SecurityProperties 配置属性类 + // 需要保证在 Spring Security 过滤后面 + registrationBean.setOrder(Integer.MAX_VALUE); + return registrationBean; + } +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmFormController.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmFormController.java new file mode 100644 index 00000000..e1c06ba7 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmFormController.java @@ -0,0 +1,81 @@ +package com.ruoyi.flowable.controller.definition; + +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.flowable.convert.definition.BpmFormConvert; +import com.ruoyi.flowable.domain.entity.definition.BpmFormDO; +import com.ruoyi.flowable.domain.vo.form.BpmFormCreateReqVO; +import com.ruoyi.flowable.domain.vo.form.BpmFormPageReqVO; +import com.ruoyi.flowable.domain.vo.form.BpmFormRespVO; +import com.ruoyi.flowable.domain.vo.form.BpmFormUpdateReqVO; +import com.ruoyi.flowable.service.definition.BpmFormService; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiImplicitParam; +import io.swagger.annotations.ApiOperation; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import javax.annotation.Resource; +import javax.validation.Valid; +import java.util.List; + +@Api(tags = "管理后台 - 动态表单") +@RestController +@RequestMapping("/bpm/form") +@Validated +public class BpmFormController { + + @Resource + private BpmFormService formService; + + @PostMapping("/create") + @ApiOperation("创建动态表单") + @PreAuthorize("@ss.hasPermi('bpm:form:create')") + public AjaxResult createForm(@Valid @RequestBody BpmFormCreateReqVO createReqVO) { + return AjaxResult.success(formService.createForm(createReqVO)); + } + + @PutMapping("/update") + @ApiOperation("更新动态表单") + @PreAuthorize("@ss.hasPermi('bpm:form:update')") + public AjaxResult updateForm(@Valid @RequestBody BpmFormUpdateReqVO updateReqVO) { + formService.updateForm(updateReqVO); + return AjaxResult.success(true); + } + + @DeleteMapping("/delete") + @ApiOperation("删除动态表单") + @ApiImplicitParam(name = "id", value = "编号", required = true, dataTypeClass = Long.class) + @PreAuthorize("@ss.hasPermi('bpm:form:delete')") + public AjaxResult deleteForm(@RequestParam("id") Long id) { + formService.deleteForm(id); + return AjaxResult.success(true); + } + + @GetMapping("/get") + @ApiOperation("获得动态表单") + @ApiImplicitParam(name = "id", value = "编号", required = true, example = "1024", dataTypeClass = Long.class) + @PreAuthorize("@ss.hasPermi('bpm:form:query')") + public AjaxResult getForm(@RequestParam("id") Long id) { + BpmFormDO form = formService.getForm(id); + return AjaxResult.success(BpmFormConvert.INSTANCE.convert(form)); + } + + @GetMapping("/list-all-simple") + @ApiOperation(value = "获得动态表单的精简列表", notes = "用于表单下拉框") + public AjaxResult getSimpleForms() { + List list = formService.getFormList(); + return AjaxResult.success(BpmFormConvert.INSTANCE.convertList2(list)); + } + + @GetMapping("/page") + @ApiOperation("获得动态表单分页") + @PreAuthorize("@ss.hasPermi('bpm:form:query')") + public AjaxResult getFormPage(@Valid BpmFormPageReqVO pageVO) { + PageResult pageResult = formService.getFormPage(pageVO); + PageResult bpmFormRespVOPageResult = BpmFormConvert.INSTANCE.convertPage(pageResult); + return AjaxResult.success(bpmFormRespVOPageResult); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmModelController.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmModelController.java new file mode 100644 index 00000000..4a4f1b9c --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmModelController.java @@ -0,0 +1,93 @@ +package com.ruoyi.flowable.controller.definition; + +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.utils.file.IoUtils; +import com.ruoyi.flowable.convert.definition.BpmModelConvert; +import com.ruoyi.flowable.domain.vo.model.*; +import com.ruoyi.flowable.service.definition.BpmModelService; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiImplicitParam; +import io.swagger.annotations.ApiOperation; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import javax.annotation.Resource; +import javax.validation.Valid; +import java.io.IOException; + +@Api(tags = "管理后台 - 流程模型") +@RestController +@RequestMapping("/bpm/model") +@Validated +public class BpmModelController { + + @Resource + private BpmModelService modelService; + + @GetMapping("/page") + @ApiOperation(value = "获得模型分页") + public AjaxResult getModelPage(BpmModelPageReqVO pageVO) { + return AjaxResult.success(modelService.getModelPage(pageVO)); + } + + @GetMapping("/get") + @ApiOperation("获得模型") + @ApiImplicitParam(name = "id", value = "编号", required = true, example = "1024", dataTypeClass = String.class) + @PreAuthorize("@ss.hasPermi('bpm:model:query')") + public AjaxResult getModel(@RequestParam("id") String id) { + BpmModelRespVO model = modelService.getModel(id); + return AjaxResult.success(model); + } + + @PostMapping("/create") + @ApiOperation(value = "新建模型") + @PreAuthorize("@ss.hasPermi('bpm:model:create')") + public AjaxResult createModel(@Valid @RequestBody BpmModelCreateReqVO createRetVO) { + return AjaxResult.success(modelService.createModel(createRetVO, null)); + } + + @PutMapping("/update") + @ApiOperation(value = "修改模型") + @PreAuthorize("@ss.hasPermi('bpm:model:update')") + public AjaxResult updateModel(@Valid @RequestBody BpmModelUpdateReqVO modelVO) { + modelService.updateModel(modelVO); + return AjaxResult.success(true); + } + + @PostMapping("/import") + @ApiOperation(value = "导入模型") + @PreAuthorize("@ss.hasPermi('bpm:model:import')") + public AjaxResult importModel(@Valid BpmModeImportReqVO importReqVO) throws IOException { + BpmModelCreateReqVO createReqVO = BpmModelConvert.INSTANCE.convert(importReqVO); + // 读取文件 + String bpmnXml = IoUtils.readUtf8(importReqVO.getBpmnFile().getInputStream(), false); + return AjaxResult.success(modelService.createModel(createReqVO, bpmnXml)); + } + + @PostMapping("/deploy") + @ApiOperation(value = "部署模型") + @ApiImplicitParam(name = "id", value = "编号", required = true, example = "1024", dataTypeClass = String.class) + @PreAuthorize("@ss.hasPermi('bpm:model:deploy')") + public AjaxResult deployModel(@RequestParam("id") String id) { + modelService.deployModel(id); + return AjaxResult.success(true); + } + + @PutMapping("/update-state") + @ApiOperation(value = "修改模型的状态", notes = "实际更新的部署的流程定义的状态") + @PreAuthorize("@ss.hasPermi('bpm:model:update')") + public AjaxResult updateModelState(@Valid @RequestBody BpmModelUpdateStateReqVO reqVO) { + modelService.updateModelState(reqVO.getId(), reqVO.getState()); + return AjaxResult.success(true); + } + + @DeleteMapping("/delete") + @ApiOperation("删除模型") + @ApiImplicitParam(name = "id", value = "编号", required = true, example = "1024", dataTypeClass = String.class) + @PreAuthorize("@ss.hasPermi('bpm:model:delete')") + public AjaxResult deleteModel(@RequestParam("id") String id) { + modelService.deleteModel(id); + return AjaxResult.success(true); + } +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmProcessDefinitionController.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmProcessDefinitionController.java new file mode 100644 index 00000000..ba053b3c --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmProcessDefinitionController.java @@ -0,0 +1,53 @@ +package com.ruoyi.flowable.controller.definition; + +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.flowable.domain.vo.process.BpmProcessDefinitionListReqVO; +import com.ruoyi.flowable.domain.vo.process.BpmProcessDefinitionPageReqVO; +import com.ruoyi.flowable.service.definition.BpmProcessDefinitionService; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiImplicitParam; +import io.swagger.annotations.ApiOperation; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RequestParam; +import org.springframework.web.bind.annotation.RestController; + +import javax.annotation.Resource; + + +@Api(tags = "管理后台 - 流程定义") +@RestController +@RequestMapping("/bpm/process-definition") +@Validated +public class BpmProcessDefinitionController { + + @Resource + private BpmProcessDefinitionService bpmDefinitionService; + + @GetMapping("/page") + @ApiOperation(value = "获得流程定义分页") + @PreAuthorize("@ss.hasPermi('bpm:process-definition:query')") + public AjaxResult getProcessDefinitionPage( + BpmProcessDefinitionPageReqVO pageReqVO) { + return AjaxResult.success(bpmDefinitionService.getProcessDefinitionPage(pageReqVO)); + } + + @GetMapping("/list") + @ApiOperation(value = "获得流程定义列表") + @PreAuthorize("@ss.hasPermi('bpm:process-definition:query')") + public AjaxResult getProcessDefinitionList( + BpmProcessDefinitionListReqVO listReqVO) { + return AjaxResult.success(bpmDefinitionService.getProcessDefinitionList(listReqVO)); + } + + @GetMapping("/get-bpmn-xml") + @ApiOperation(value = "获得流程定义的 BPMN XML") + @ApiImplicitParam(name = "id", value = "编号", required = true, example = "1024", dataTypeClass = String.class) + @PreAuthorize("@ss.hasPermi('bpm:process-definition:query')") + public AjaxResult getProcessDefinitionBpmnXML(@RequestParam("id") String id) { + String bpmnXML = bpmDefinitionService.getProcessDefinitionBpmnXML(id); + return AjaxResult.success(bpmnXML); + } +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmTaskAssignRuleController.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmTaskAssignRuleController.java new file mode 100644 index 00000000..fd08eccc --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmTaskAssignRuleController.java @@ -0,0 +1,57 @@ +package com.ruoyi.flowable.controller.definition; + +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.flowable.domain.vo.rule.BpmTaskAssignRuleCreateReqVO; +import com.ruoyi.flowable.domain.vo.rule.BpmTaskAssignRuleRespVO; +import com.ruoyi.flowable.domain.vo.rule.BpmTaskAssignRuleUpdateReqVO; +import com.ruoyi.flowable.service.definition.BpmTaskAssignRuleService; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiImplicitParam; +import io.swagger.annotations.ApiImplicitParams; +import io.swagger.annotations.ApiOperation; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import javax.annotation.Resource; +import javax.validation.Valid; +import java.util.List; + + +@Api(tags = "管理后台 - 任务分配规则") +@RestController +@RequestMapping("/bpm/task-assign-rule") +@Validated +public class BpmTaskAssignRuleController { + + @Resource + private BpmTaskAssignRuleService taskAssignRuleService; + + @GetMapping("/list") + @ApiOperation(value = "获得任务分配规则列表") + @ApiImplicitParams({ + @ApiImplicitParam(name = "modelId", value = "模型编号", example = "1024", dataTypeClass = String.class), + @ApiImplicitParam(name = "processDefinitionId", value = "流程定义的编号", example = "2048", dataTypeClass = String.class) + }) + @PreAuthorize("@ss.hasPermi('bpm:task-assign-rule:query')") + public AjaxResult getTaskAssignRuleList( + @RequestParam(value = "modelId", required = false) String modelId, + @RequestParam(value = "processDefinitionId", required = false) String processDefinitionId) { + return AjaxResult.success(taskAssignRuleService.getTaskAssignRuleList(modelId, processDefinitionId)); + } + + @PostMapping("/create") + @ApiOperation(value = "创建任务分配规则") + @PreAuthorize("@ss.hasPermi('bpm:task-assign-rule:create')") + public AjaxResult createTaskAssignRule(@Valid @RequestBody BpmTaskAssignRuleCreateReqVO reqVO) { + return AjaxResult.success(taskAssignRuleService.createTaskAssignRule(reqVO)); + } + + @PutMapping("/update") + @ApiOperation(value = "更新任务分配规则") + @PreAuthorize("@ss.hasPermi('bpm:task-assign-rule:update')") + public AjaxResult updateTaskAssignRule(@Valid @RequestBody BpmTaskAssignRuleUpdateReqVO reqVO) { + taskAssignRuleService.updateTaskAssignRule(reqVO); + return AjaxResult.success(true); + } +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmUserGroupController.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmUserGroupController.java new file mode 100644 index 00000000..cd4ae3b1 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/definition/BpmUserGroupController.java @@ -0,0 +1,83 @@ +package com.ruoyi.flowable.controller.definition; + +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.enums.CommonStatusEnum; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.flowable.convert.definition.BpmUserGroupConvert; +import com.ruoyi.flowable.domain.entity.definition.BpmUserGroupDO; +import com.ruoyi.flowable.domain.vo.group.BpmUserGroupCreateReqVO; +import com.ruoyi.flowable.domain.vo.group.BpmUserGroupPageReqVO; +import com.ruoyi.flowable.domain.vo.group.BpmUserGroupUpdateReqVO; +import com.ruoyi.flowable.service.definition.BpmUserGroupService; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiImplicitParam; +import io.swagger.annotations.ApiOperation; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import javax.annotation.Resource; +import javax.validation.Valid; +import java.util.List; + + +@Api(tags = "管理后台 - 用户组") +@RestController +@RequestMapping("/bpm/user-group") +@Validated +public class BpmUserGroupController { + + @Resource + private BpmUserGroupService userGroupService; + + @PostMapping("/create") + @ApiOperation("创建用户组") + @PreAuthorize("@ss.hasPermi('bpm:user-group:create')") + public AjaxResult createUserGroup(@Valid @RequestBody BpmUserGroupCreateReqVO createReqVO) { + return AjaxResult.success(userGroupService.createUserGroup(createReqVO)); + } + + @PutMapping("/update") + @ApiOperation("更新用户组") + @PreAuthorize("@ss.hasPermi('bpm:user-group:update')") + public AjaxResult updateUserGroup(@Valid @RequestBody BpmUserGroupUpdateReqVO updateReqVO) { + userGroupService.updateUserGroup(updateReqVO); + return AjaxResult.success(true); + } + + @DeleteMapping("/delete") + @ApiOperation("删除用户组") + @ApiImplicitParam(name = "id", value = "编号", required = true, dataTypeClass = Long.class) + @PreAuthorize("@ss.hasPermi('bpm:user-group:delete')") + public AjaxResult deleteUserGroup(@RequestParam("id") Long id) { + userGroupService.deleteUserGroup(id); + return AjaxResult.success(true); + } + + @GetMapping("/get") + @ApiOperation("获得用户组") + @ApiImplicitParam(name = "id", value = "编号", required = true, example = "1024", dataTypeClass = Long.class) + @PreAuthorize("@ss.hasPermi('bpm:user-group:query')") + public AjaxResult getUserGroup(@RequestParam("id") Long id) { + BpmUserGroupDO userGroup = userGroupService.getUserGroup(id); + return AjaxResult.success(BpmUserGroupConvert.INSTANCE.convert(userGroup)); + } + + @GetMapping("/page") + @ApiOperation("获得用户组分页") + @PreAuthorize("@ss.hasPermi('bpm:user-group:query')") + public AjaxResult getUserGroupPage(@Valid BpmUserGroupPageReqVO pageVO) { + PageResult pageResult = userGroupService.getUserGroupPage(pageVO); + return AjaxResult.success(BpmUserGroupConvert.INSTANCE.convertPage(pageResult)); + } + + @GetMapping("/list-all-simple") + @ApiOperation(value = "获取用户组精简信息列表", notes = "只包含被开启的用户组,主要用于前端的下拉选项") + public AjaxResult getSimpleUserGroups() { + // 获用户门列表,只要开启状态的 + List list = userGroupService.getUserGroupListByStatus(CommonStatusEnum.ENABLE.getStatus()); + // 排序后,返回给前端 + return AjaxResult.success(BpmUserGroupConvert.INSTANCE.convertList2(list)); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/oa/BpmOALeaveController.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/oa/BpmOALeaveController.java new file mode 100644 index 00000000..fd3c4943 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/oa/BpmOALeaveController.java @@ -0,0 +1,62 @@ +package com.ruoyi.flowable.controller.oa; + +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.baomidou.mybatisplus.core.metadata.OrderItem; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.page.TableDataInfo; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.flowable.convert.oa.BpmOALeaveConvert; +import com.ruoyi.flowable.domain.entity.oa.BpmOALeaveDO; +import com.ruoyi.flowable.domain.vo.oa.BpmOALeaveCreateReqVO; +import com.ruoyi.flowable.service.oa.BpmOALeaveService; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiImplicitParam; +import io.swagger.annotations.ApiOperation; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import javax.annotation.Resource; +import javax.validation.Valid; + +/** + * OA 请假申请 Controller,用于演示自己存储数据,接入工作流的例子 + * + * @author jason + * hasPermi + */ +@Api(tags = "管理后台 - OA 请假申请") +@RestController +@RequestMapping("/bpm/oa/leave") +@Validated +public class BpmOALeaveController extends BaseController { + + @Resource + private BpmOALeaveService leaveService; + + @PostMapping("/create") + @PreAuthorize("@ss.hasPermi('bpm:oa-leave:create')") + @ApiOperation("创建请求申请") + public AjaxResult createLeave(@Valid @RequestBody BpmOALeaveCreateReqVO createReqVO) { + return AjaxResult.success(leaveService.createLeave(SecurityUtils.getUserId(), createReqVO)); + } + + @GetMapping("/get") + @PreAuthorize("@ss.hasPermi('bpm:oa-leave:query')") + @ApiOperation("获得请假申请") + @ApiImplicitParam(name = "id", value = "编号", required = true, example = "1024", dataTypeClass = Long.class) + public AjaxResult getLeave(@RequestParam("id") Long id) { + BpmOALeaveDO leave = leaveService.getLeave(id); + return AjaxResult.success(BpmOALeaveConvert.INSTANCE.convert(leave)); + } + + @GetMapping("/page") + @PreAuthorize("@ss.hasPermi('bpm:oa-leave:query')") + @ApiOperation("获得请假申请分页") + public TableDataInfo list(BpmOALeaveDO bpmOaLeave) { + IPage list = leaveService.selectList(getPage(OrderItem.asc("result"),OrderItem.desc("id")), bpmOaLeave); + return getDataTable(list); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/task/BpmActivityController.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/task/BpmActivityController.java new file mode 100644 index 00000000..2d1bc487 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/task/BpmActivityController.java @@ -0,0 +1,35 @@ +package com.ruoyi.flowable.controller.task; + +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.flowable.service.task.BpmActivityService; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiImplicitParam; +import io.swagger.annotations.ApiOperation; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RequestParam; +import org.springframework.web.bind.annotation.RestController; + +import javax.annotation.Resource; + + +@Api(tags = "管理后台 - 流程活动实例") +@RestController +@RequestMapping("/bpm/activity") +@Validated +public class BpmActivityController { + + @Resource + private BpmActivityService activityService; + + @GetMapping("/list") + @ApiOperation(value = "生成指定流程实例的高亮流程图", + notes = "只高亮进行中的任务。不过要注意,该接口暂时没用,通过前端的 ProcessViewer.vue 界面的 highlightDiagram 方法生成") + @ApiImplicitParam(name = "processInstanceId", value = "流程实例的编号", required = true, dataTypeClass = String.class) + @PreAuthorize("@ss.hasPermi('bpm:task:query')") + public AjaxResult getActivityList(@RequestParam("processInstanceId") String processInstanceId) { + return AjaxResult.success(activityService.getActivityListByProcessInstanceId(processInstanceId)); + } +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/task/BpmProcessInstanceController.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/task/BpmProcessInstanceController.java new file mode 100644 index 00000000..e84702ba --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/task/BpmProcessInstanceController.java @@ -0,0 +1,57 @@ +package com.ruoyi.flowable.controller.task; + +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.flowable.domain.vo.instance.*; +import com.ruoyi.flowable.service.task.BpmProcessInstanceService; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiImplicitParam; +import io.swagger.annotations.ApiOperation; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import javax.annotation.Resource; +import javax.validation.Valid; + + +@Api(tags = "管理后台 - 流程实例") // 流程实例,通过流程定义创建的一次“申请” +@RestController +@RequestMapping("/bpm/process-instance") +@Validated +public class BpmProcessInstanceController { + + @Resource + private BpmProcessInstanceService processInstanceService; + + @GetMapping("/my-page") + @ApiOperation(value = "获得我的实例分页列表", notes = "在【我的流程】菜单中,进行调用") + @PreAuthorize("@ss.hasPermi('bpm:process-instance:query')") + public AjaxResult getMyProcessInstancePage( + @Valid BpmProcessInstanceMyPageReqVO pageReqVO) { + return AjaxResult.success(processInstanceService.getMyProcessInstancePage(SecurityUtils.getUserId(), pageReqVO)); + } + + @PostMapping("/create") + @ApiOperation("新建流程实例") + @PreAuthorize("@ss.hasPermi('bpm:process-instance:query')") + public AjaxResult createProcessInstance(@Valid @RequestBody BpmProcessInstanceCreateReqVO createReqVO) { + return AjaxResult.success(processInstanceService.createProcessInstance(SecurityUtils.getUserId(), createReqVO)); + } + + @GetMapping("/get") + @ApiOperation(value = "获得指定流程实例", notes = "在【流程详细】界面中,进行调用") + @ApiImplicitParam(name = "id", value = "流程实例的编号", required = true, dataTypeClass = String.class) + @PreAuthorize("@ss.hasPermi('bpm:process-instance:query')") + public AjaxResult getProcessInstance(@RequestParam("id") String id) { + return AjaxResult.success(processInstanceService.getProcessInstanceVO(id)); + } + + @DeleteMapping("/cancel") + @ApiOperation(value = "取消流程实例", notes = "撤回发起的流程") + @PreAuthorize("@ss.hasPermi('bpm:process-instance:cancel')") + public AjaxResult cancelProcessInstance(@Valid @RequestBody BpmProcessInstanceCancelReqVO cancelReqVO) { + processInstanceService.cancelProcessInstance(SecurityUtils.getUserId(), cancelReqVO); + return AjaxResult.success(); + } +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/task/BpmTaskController.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/task/BpmTaskController.java new file mode 100644 index 00000000..9d2cd26f --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/controller/task/BpmTaskController.java @@ -0,0 +1,81 @@ +package com.ruoyi.flowable.controller.task; + +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.flowable.domain.vo.task.*; +import com.ruoyi.flowable.service.task.BpmTaskService; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiImplicitParam; +import io.swagger.annotations.ApiOperation; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import javax.annotation.Resource; +import javax.validation.Valid; + + +@Api(tags = "管理后台 - 流程任务实例") +@RestController +@RequestMapping("/bpm/task") +@Validated +public class BpmTaskController { + + @Resource + private BpmTaskService taskService; + + @GetMapping("todo-page") + @ApiOperation("获取 Todo 待办任务分页") + @PreAuthorize("@ss.hasPermi('bpm:task:query')") + public AjaxResult getTodoTaskPage(@Valid BpmTaskTodoPageReqVO pageVO) { + return AjaxResult.success(taskService.getTodoTaskPage(SecurityUtils.getUserId(), pageVO)); + } + + @GetMapping("done-page") + @ApiOperation("获取 Done 已办任务分页") + @PreAuthorize("@ss.hasPermi('bpm:task:query')") + public AjaxResult getDoneTaskPage(@Valid BpmTaskDonePageReqVO pageVO) { + return AjaxResult.success(taskService.getDoneTaskPage(SecurityUtils.getUserId(), pageVO)); + } + + @GetMapping("/list-by-process-instance-id") + @ApiOperation(value = "获得指定流程实例的任务列表", notes = "包括完成的、未完成的") + @ApiImplicitParam(name = "processInstanceId", value = "流程实例的编号", required = true, dataTypeClass = String.class) + @PreAuthorize("@ss.hasPermi('bpm:task:query')") + public AjaxResult getTaskListByProcessInstanceId( + @RequestParam("processInstanceId") String processInstanceId) { + return AjaxResult.success(taskService.getTaskListByProcessInstanceId(processInstanceId)); + } + + @PutMapping("/approve") + @ApiOperation("通过任务") + @PreAuthorize("@ss.hasPermi('bpm:task:update')") + public AjaxResult approveTask(@Valid @RequestBody BpmTaskApproveReqVO reqVO) { + taskService.approveTask(SecurityUtils.getUserId(), reqVO); + return AjaxResult.success(true); + } + + @PutMapping("/reject") + @ApiOperation("不通过任务") + @PreAuthorize("@ss.hasPermi('bpm:task:update')") + public AjaxResult rejectTask(@Valid @RequestBody BpmTaskRejectReqVO reqVO) { + taskService.rejectTask(SecurityUtils.getUserId(), reqVO); + return AjaxResult.success(true); + } + + @PutMapping("/update-assignee") + @ApiOperation(value = "更新任务的负责人", notes = "用于【流程详情】的【转派】按钮") + @PreAuthorize("@ss.hasPermi('bpm:task:update')") + public AjaxResult updateTaskAssignee(@Valid @RequestBody BpmTaskUpdateAssigneeReqVO reqVO) { + taskService.updateTaskAssignee(SecurityUtils.getUserId(), reqVO); + return AjaxResult.success(true); + } + + @ApiOperation(value = "获取可驳回节点列表", notes = "获取可驳回节点列表") + @GetMapping(value = "/getBackNodesByProcessInstanceId/{processInstanceId}/{taskId}") + public AjaxResult getBackNodesByProcessInstanceId(@PathVariable String processInstanceId,@PathVariable String taskId) { + return AjaxResult.success(taskService.getBackNodesByProcessInstanceId(processInstanceId,taskId)); + } + + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmFormConvert.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmFormConvert.java new file mode 100644 index 00000000..762f6e15 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmFormConvert.java @@ -0,0 +1,34 @@ +package com.ruoyi.flowable.convert.definition; + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.flowable.domain.entity.definition.BpmFormDO; +import com.ruoyi.flowable.domain.vo.form.BpmFormCreateReqVO; +import com.ruoyi.flowable.domain.vo.form.BpmFormRespVO; +import com.ruoyi.flowable.domain.vo.form.BpmFormSimpleRespVO; +import com.ruoyi.flowable.domain.vo.form.BpmFormUpdateReqVO; +import org.mapstruct.Mapper; +import org.mapstruct.factory.Mappers; + +import java.util.List; + +/** + * 动态表单 Convert + * + * @author 芋艿 + */ +@Mapper +public interface BpmFormConvert { + + BpmFormConvert INSTANCE = Mappers.getMapper(BpmFormConvert.class); + + BpmFormDO convert(BpmFormCreateReqVO bean); + + BpmFormDO convert(BpmFormUpdateReqVO bean); + + BpmFormRespVO convert(BpmFormDO bean); + + List convertList2(List list); + + PageResult convertPage(PageResult page); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmModelConvert.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmModelConvert.java new file mode 100644 index 00000000..4ebc9bf9 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmModelConvert.java @@ -0,0 +1,140 @@ +package com.ruoyi.flowable.convert.definition; + +import cn.hutool.core.util.StrUtil; +import com.ruoyi.common.utils.JsonUtils; +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.domain.dto.definition.BpmModelMetaInfoRespDTO; +import com.ruoyi.flowable.domain.dto.definition.BpmProcessDefinitionCreateReqDTO; +import com.ruoyi.flowable.domain.entity.definition.BpmFormDO; +import com.ruoyi.flowable.domain.vo.model.*; +import org.flowable.common.engine.impl.db.SuspensionState; +import org.flowable.engine.repository.Deployment; +import org.flowable.engine.repository.Model; +import org.flowable.engine.repository.ProcessDefinition; +import org.mapstruct.Mapper; +import org.mapstruct.MappingTarget; +import org.mapstruct.factory.Mappers; + +import java.util.List; +import java.util.Map; +import java.util.Objects; + +/** + * 流程模型 Convert + * + * @author yunlongn + */ +@Mapper +public interface BpmModelConvert { + + BpmModelConvert INSTANCE = Mappers.getMapper(BpmModelConvert.class); + + default List convertList(List list, Map formMap, + Map deploymentMap, + Map processDefinitionMap) { + return CollectionUtils.convertList(list, model -> { + BpmModelMetaInfoRespDTO metaInfo = JsonUtils.parseObject(model.getMetaInfo(), BpmModelMetaInfoRespDTO.class); + BpmFormDO form = metaInfo != null ? formMap.get(metaInfo.getFormId()) : null; + Deployment deployment = model.getDeploymentId() != null ? deploymentMap.get(model.getDeploymentId()) : null; + ProcessDefinition processDefinition = model.getDeploymentId() != null ? processDefinitionMap.get(model.getDeploymentId()) : null; + return convert(model, form, deployment, processDefinition); + }); + } + + default BpmModelPageItemRespVO convert(Model model, BpmFormDO form, Deployment deployment, ProcessDefinition processDefinition) { + BpmModelPageItemRespVO modelRespVO = new BpmModelPageItemRespVO(); + modelRespVO.setId(model.getId()); + modelRespVO.setCreateTime(model.getCreateTime()); + // 通用 copy + copyTo(model, modelRespVO); + // Form + if (form != null) { + modelRespVO.setFormId(form.getId()); + modelRespVO.setFormName(form.getName()); + } + // ProcessDefinition + modelRespVO.setProcessDefinition(this.convert(processDefinition)); + if (modelRespVO.getProcessDefinition() != null) { + modelRespVO.getProcessDefinition().setSuspensionState(processDefinition.isSuspended() ? + SuspensionState.SUSPENDED.getStateCode() : SuspensionState.ACTIVE.getStateCode()); + modelRespVO.getProcessDefinition().setDeploymentTime(deployment.getDeploymentTime()); + } + return modelRespVO; + } + + default BpmModelRespVO convert(Model model) { + BpmModelRespVO modelRespVO = new BpmModelRespVO(); + modelRespVO.setId(model.getId()); + modelRespVO.setCreateTime(model.getCreateTime()); + // 通用 copy + copyTo(model, modelRespVO); + return modelRespVO; + } + + default void copyTo(Model model, BpmModelBaseVO to) { + to.setName(model.getName()); + to.setKey(model.getKey()); + to.setCategory(model.getCategory()); + // metaInfo + BpmModelMetaInfoRespDTO metaInfo = JsonUtils.parseObject(model.getMetaInfo(), BpmModelMetaInfoRespDTO.class); + copyTo(metaInfo, to); + } + + BpmModelCreateReqVO convert(BpmModeImportReqVO bean); + + default BpmProcessDefinitionCreateReqDTO convert2(Model model, BpmFormDO form) { + BpmProcessDefinitionCreateReqDTO createReqDTO = new BpmProcessDefinitionCreateReqDTO(); + createReqDTO.setModelId(model.getId()); + createReqDTO.setName(model.getName()); + createReqDTO.setKey(model.getKey()); + createReqDTO.setCategory(model.getCategory()); + BpmModelMetaInfoRespDTO metaInfo = JsonUtils.parseObject(model.getMetaInfo(), BpmModelMetaInfoRespDTO.class); + // metaInfo + copyTo(metaInfo, createReqDTO); + // form + if (form != null) { + createReqDTO.setFormConf(form.getConf()); + createReqDTO.setFormFields(form.getFields()); + } + return createReqDTO; + } + + void copyTo(BpmModelMetaInfoRespDTO from, @MappingTarget BpmProcessDefinitionCreateReqDTO to); + + void copyTo(BpmModelMetaInfoRespDTO from, @MappingTarget BpmModelBaseVO to); + + BpmModelPageItemRespVO.ProcessDefinition convert(ProcessDefinition bean); + + default void copy(Model model, BpmModelCreateReqVO bean) { + model.setName(bean.getName()); + model.setKey(bean.getKey()); + model.setMetaInfo(buildMetaInfoStr(null, bean.getDescription(), null, null, + null, null)); + } + + default void copy(Model model, BpmModelUpdateReqVO bean) { + model.setName(bean.getName()); + model.setCategory(bean.getCategory()); + model.setMetaInfo(buildMetaInfoStr(JsonUtils.parseObject(model.getMetaInfo(), BpmModelMetaInfoRespDTO.class), + bean.getDescription(), bean.getFormType(), bean.getFormId(), + bean.getFormCustomCreatePath(), bean.getFormCustomViewPath())); + } + + default String buildMetaInfoStr(BpmModelMetaInfoRespDTO metaInfo, String description, Integer formType, + Long formId, String formCustomCreatePath, String formCustomViewPath) { + if (metaInfo == null) { + metaInfo = new BpmModelMetaInfoRespDTO(); + } + // 只有非空,才进行设置,避免更新时的覆盖 + if (StrUtil.isNotEmpty(description)) { + metaInfo.setDescription(description); + } + if (Objects.nonNull(formType)) { + metaInfo.setFormType(formType); + metaInfo.setFormId(formId); + metaInfo.setFormCustomCreatePath(formCustomCreatePath); + metaInfo.setFormCustomViewPath(formCustomViewPath); + } + return JsonUtils.toJsonString(metaInfo); + } +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmProcessDefinitionConvert.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmProcessDefinitionConvert.java new file mode 100644 index 00000000..927c6cb0 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmProcessDefinitionConvert.java @@ -0,0 +1,82 @@ +package com.ruoyi.flowable.convert.definition; + +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.domain.dto.definition.BpmProcessDefinitionCreateReqDTO; +import com.ruoyi.flowable.domain.entity.definition.BpmFormDO; +import com.ruoyi.flowable.domain.entity.definition.BpmProcessDefinitionExtDO; +import com.ruoyi.flowable.domain.vo.process.BpmProcessDefinitionPageItemRespVO; +import com.ruoyi.flowable.domain.vo.process.BpmProcessDefinitionRespVO; +import org.flowable.common.engine.impl.db.SuspensionState; +import org.flowable.engine.repository.Deployment; +import org.flowable.engine.repository.ProcessDefinition; +import org.mapstruct.Mapper; +import org.mapstruct.Mapping; +import org.mapstruct.MappingTarget; +import org.mapstruct.Named; +import org.mapstruct.factory.Mappers; + +import java.util.List; +import java.util.Map; + +/** + * Bpm 流程定义的 Convert + * + * @author yunlong.li + */ +@Mapper +public interface BpmProcessDefinitionConvert { + + BpmProcessDefinitionConvert INSTANCE = Mappers.getMapper(BpmProcessDefinitionConvert.class); + + BpmProcessDefinitionPageItemRespVO convert(ProcessDefinition bean); + + BpmProcessDefinitionExtDO convert2(BpmProcessDefinitionCreateReqDTO bean); + + default List convertList(List list, Map deploymentMap, + Map processDefinitionDOMap, Map formMap) { + return CollectionUtils.convertList(list, definition -> { + Deployment deployment = definition.getDeploymentId() != null ? deploymentMap.get(definition.getDeploymentId()) : null; + BpmProcessDefinitionExtDO definitionDO = processDefinitionDOMap.get(definition.getId()); + BpmFormDO form = definitionDO != null ? formMap.get(definitionDO.getFormId()) : null; + return convert(definition, deployment, definitionDO, form); + }); + } + + default List convertList3(List list, + Map processDefinitionDOMap) { + return CollectionUtils.convertList(list, processDefinition -> { + BpmProcessDefinitionRespVO respVO = convert3(processDefinition); + BpmProcessDefinitionExtDO processDefinitionExtDO = processDefinitionDOMap.get(processDefinition.getId()); + // 复制通用属性 + copyTo(processDefinitionExtDO, respVO); + return respVO; + }); + } + + @Mapping(source = "suspended", target = "suspensionState", qualifiedByName = "convertSuspendedToSuspensionState") + BpmProcessDefinitionRespVO convert3(ProcessDefinition bean); + + @Named("convertSuspendedToSuspensionState") + default Integer convertSuspendedToSuspensionState(boolean suspended) { + return suspended ? SuspensionState.SUSPENDED.getStateCode() : + SuspensionState.ACTIVE.getStateCode(); + } + + default BpmProcessDefinitionPageItemRespVO convert(ProcessDefinition bean, Deployment deployment, + BpmProcessDefinitionExtDO processDefinitionExtDO, BpmFormDO form) { + BpmProcessDefinitionPageItemRespVO respVO = convert(bean); + respVO.setSuspensionState(bean.isSuspended() ? SuspensionState.SUSPENDED.getStateCode() : SuspensionState.ACTIVE.getStateCode()); + if (deployment != null) { + respVO.setDeploymentTime(deployment.getDeploymentTime()); + } + if (form != null) { + respVO.setFormName(form.getName()); + } + // 复制通用属性 + copyTo(processDefinitionExtDO, respVO); + return respVO; + } + + @Mapping(source = "from.id", target = "to.id", ignore = true) + void copyTo(BpmProcessDefinitionExtDO from, @MappingTarget BpmProcessDefinitionRespVO to); +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmTaskAssignRuleConvert.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmTaskAssignRuleConvert.java new file mode 100644 index 00000000..b5d4b4c9 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmTaskAssignRuleConvert.java @@ -0,0 +1,40 @@ +package com.ruoyi.flowable.convert.definition; + +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.domain.entity.definition.BpmTaskAssignRuleDO; +import com.ruoyi.flowable.domain.vo.rule.BpmTaskAssignRuleCreateReqVO; +import com.ruoyi.flowable.domain.vo.rule.BpmTaskAssignRuleRespVO; +import com.ruoyi.flowable.domain.vo.rule.BpmTaskAssignRuleUpdateReqVO; +import org.flowable.bpmn.model.UserTask; +import org.mapstruct.Mapper; +import org.mapstruct.factory.Mappers; + +import java.util.List; +import java.util.Map; + +@Mapper +public interface BpmTaskAssignRuleConvert { + BpmTaskAssignRuleConvert INSTANCE = Mappers.getMapper(BpmTaskAssignRuleConvert.class); + + default List convertList(List tasks, List rules) { + Map ruleMap = CollectionUtils.convertMap(rules, BpmTaskAssignRuleDO::getTaskDefinitionKey); + // 以 UserTask 为主维度,原因是:流程图编辑后,一些规则实际就没用了。 + return CollectionUtils.convertList(tasks, task -> { + BpmTaskAssignRuleRespVO respVO = convert(ruleMap.get(task.getId())); + if (respVO == null) { + respVO = new BpmTaskAssignRuleRespVO(); + respVO.setTaskDefinitionKey(task.getId()); + } + respVO.setTaskDefinitionName(task.getName()); + return respVO; + }); + } + + BpmTaskAssignRuleRespVO convert(BpmTaskAssignRuleDO bean); + + BpmTaskAssignRuleDO convert(BpmTaskAssignRuleCreateReqVO bean); + + BpmTaskAssignRuleDO convert(BpmTaskAssignRuleUpdateReqVO bean); + + List convertList2(List list); +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmUserGroupConvert.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmUserGroupConvert.java new file mode 100644 index 00000000..a9cc96ba --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/definition/BpmUserGroupConvert.java @@ -0,0 +1,37 @@ +package com.ruoyi.flowable.convert.definition; + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.flowable.domain.entity.definition.BpmUserGroupDO; +import com.ruoyi.flowable.domain.vo.group.BpmUserGroupCreateReqVO; +import com.ruoyi.flowable.domain.vo.group.BpmUserGroupRespVO; +import com.ruoyi.flowable.domain.vo.group.BpmUserGroupUpdateReqVO; +import org.mapstruct.Mapper; +import org.mapstruct.Named; +import org.mapstruct.factory.Mappers; + +import java.util.List; + +/** + * 用户组 Convert + * + * hasPermi + */ +@Mapper +public interface BpmUserGroupConvert { + + BpmUserGroupConvert INSTANCE = Mappers.getMapper(BpmUserGroupConvert.class); + + BpmUserGroupDO convert(BpmUserGroupCreateReqVO bean); + + BpmUserGroupDO convert(BpmUserGroupUpdateReqVO bean); + + BpmUserGroupRespVO convert(BpmUserGroupDO bean); + + List convertList(List list); + + PageResult convertPage(PageResult page); + + @Named("convertList2") + List convertList2(List list); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/message/BpmMessageConvert.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/message/BpmMessageConvert.java new file mode 100644 index 00000000..c3267eb3 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/message/BpmMessageConvert.java @@ -0,0 +1,16 @@ +package com.ruoyi.flowable.convert.message; + +import com.ruoyi.flowable.domain.dto.send.SmsSendSingleToUserReqDTO; +import org.mapstruct.Mapper; +import org.mapstruct.factory.Mappers; + +import java.util.Map; + +@Mapper +public interface BpmMessageConvert { + + BpmMessageConvert INSTANCE = Mappers.getMapper(BpmMessageConvert.class); + + SmsSendSingleToUserReqDTO convert(Long userId, String templateCode, Map templateParams); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/oa/BpmOALeaveConvert.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/oa/BpmOALeaveConvert.java new file mode 100644 index 00000000..ac8ee5a5 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/oa/BpmOALeaveConvert.java @@ -0,0 +1,30 @@ +package com.ruoyi.flowable.convert.oa; + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.flowable.domain.entity.oa.BpmOALeaveDO; +import com.ruoyi.flowable.domain.vo.oa.BpmOALeaveCreateReqVO; +import com.ruoyi.flowable.domain.vo.oa.BpmOALeaveRespVO; +import org.mapstruct.Mapper; +import org.mapstruct.factory.Mappers; + +import java.util.List; + +/** + * 请假申请 Convert + * + * @author 芋艿 + */ +@Mapper +public interface BpmOALeaveConvert { + + BpmOALeaveConvert INSTANCE = Mappers.getMapper(BpmOALeaveConvert.class); + + BpmOALeaveDO convert(BpmOALeaveCreateReqVO bean); + + BpmOALeaveRespVO convert(BpmOALeaveDO bean); + + List convertList(List list); + + PageResult convertPage(PageResult page); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/task/BpmActivityConvert.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/task/BpmActivityConvert.java new file mode 100644 index 00000000..285c5511 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/task/BpmActivityConvert.java @@ -0,0 +1,29 @@ +package com.ruoyi.flowable.convert.task; + +import com.ruoyi.flowable.domain.vo.activity.BpmActivityRespVO; +import org.flowable.engine.history.HistoricActivityInstance; +import org.mapstruct.Mapper; +import org.mapstruct.Mapping; +import org.mapstruct.Mappings; +import org.mapstruct.factory.Mappers; + +import java.util.List; + +/** + * BPM 活动 Convert + * + * hasPermi + */ +@Mapper +public interface BpmActivityConvert { + + BpmActivityConvert INSTANCE = Mappers.getMapper(BpmActivityConvert.class); + + List convertList(List list); + + @Mappings({ + @Mapping(source = "activityId", target = "key"), + @Mapping(source = "activityType", target = "type") + }) + BpmActivityRespVO convert(HistoricActivityInstance bean); +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/task/BpmProcessInstanceConvert.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/task/BpmProcessInstanceConvert.java new file mode 100644 index 00000000..0b0a7391 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/task/BpmProcessInstanceConvert.java @@ -0,0 +1,114 @@ +package com.ruoyi.flowable.convert.task; + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.utils.NumberUtils; +import com.ruoyi.flowable.domain.dto.message.BpmMessageSendWhenProcessInstanceApproveReqDTO; +import com.ruoyi.flowable.domain.dto.message.BpmMessageSendWhenProcessInstanceRejectReqDTO; +import com.ruoyi.flowable.domain.dto.user.AdminUserRespDTO; +import com.ruoyi.flowable.domain.dto.user.DeptRespDTO; +import com.ruoyi.flowable.domain.entity.definition.BpmProcessDefinitionExtDO; +import com.ruoyi.flowable.domain.entity.task.BpmProcessInstanceExtDO; +import com.ruoyi.flowable.domain.vo.instance.BpmProcessInstancePageItemRespVO; +import com.ruoyi.flowable.domain.vo.instance.BpmProcessInstanceRespVO; +import com.ruoyi.flowable.framework.bpm.core.event.BpmProcessInstanceResultEvent; +import org.flowable.engine.history.HistoricProcessInstance; +import org.flowable.engine.repository.ProcessDefinition; +import org.flowable.engine.runtime.ProcessInstance; +import org.flowable.task.api.Task; +import org.mapstruct.Mapper; +import org.mapstruct.Mapping; +import org.mapstruct.MappingTarget; +import org.mapstruct.factory.Mappers; + +import java.util.List; +import java.util.Map; + +/** + * 流程实例 Convert + * + * hasPermi + */ +@Mapper +public interface BpmProcessInstanceConvert { + + BpmProcessInstanceConvert INSTANCE = Mappers.getMapper(BpmProcessInstanceConvert.class); + + default PageResult convertPage(PageResult page, + Map> taskMap) { + List list = convertList(page.getList()); + list.forEach(respVO -> respVO.setTasks(convertList2(taskMap.get(respVO.getId())))); + return new PageResult<>(list, page.getTotal()); + } + + List convertList(List list); + + @Mapping(source = "processInstanceId", target = "id") + BpmProcessInstancePageItemRespVO convert(BpmProcessInstanceExtDO bean); + + List convertList2(List tasks); + + default BpmProcessInstanceRespVO convert2(HistoricProcessInstance processInstance, BpmProcessInstanceExtDO processInstanceExt, + ProcessDefinition processDefinition, BpmProcessDefinitionExtDO processDefinitionExt, + String bpmnXml, AdminUserRespDTO startUser, DeptRespDTO dept) { + BpmProcessInstanceRespVO respVO = convert2(processInstance); + copyTo(processInstanceExt, respVO); + // definition + respVO.setProcessDefinition(convert2(processDefinition)); + copyTo(processDefinitionExt, respVO.getProcessDefinition()); + respVO.getProcessDefinition().setBpmnXml(bpmnXml); + // user + if (startUser != null) { + respVO.setStartUser(convert2(startUser)); + if (dept != null) { + respVO.getStartUser().setDeptName(dept.getName()); + } + } + return respVO; + } + + BpmProcessInstanceRespVO convert2(HistoricProcessInstance bean); + + @Mapping(source = "from.id", target = "to.id", ignore = true) + void copyTo(BpmProcessInstanceExtDO from, @MappingTarget BpmProcessInstanceRespVO to); + + BpmProcessInstanceRespVO.ProcessDefinition convert2(ProcessDefinition bean); + + @Mapping(source = "from.id", target = "to.id", ignore = true) + void copyTo(BpmProcessDefinitionExtDO from, @MappingTarget BpmProcessInstanceRespVO.ProcessDefinition to); + + BpmProcessInstanceRespVO.User convert2(AdminUserRespDTO bean); + + default BpmProcessInstanceResultEvent convert(Object source, HistoricProcessInstance instance, Integer result) { + BpmProcessInstanceResultEvent event = new BpmProcessInstanceResultEvent(source); + event.setId(instance.getId()); + event.setProcessDefinitionKey(instance.getProcessDefinitionKey()); + event.setBusinessKey(instance.getBusinessKey()); + event.setResult(result); + return event; + } + + default BpmProcessInstanceResultEvent convert(Object source, ProcessInstance instance, Integer result) { + BpmProcessInstanceResultEvent event = new BpmProcessInstanceResultEvent(source); + event.setId(instance.getId()); + event.setProcessDefinitionKey(instance.getProcessDefinitionKey()); + event.setBusinessKey(instance.getBusinessKey()); + event.setResult(result); + return event; + } + + default BpmMessageSendWhenProcessInstanceApproveReqDTO convert2ApprovedReq(ProcessInstance instance){ + return new BpmMessageSendWhenProcessInstanceApproveReqDTO() + .setStartUserId(NumberUtils.parseLong(instance.getStartUserId())) + .setProcessInstanceId(instance.getId()) + .setProcessInstanceName(instance.getName()); + } + + default BpmMessageSendWhenProcessInstanceRejectReqDTO convert2RejectReq(ProcessInstance instance, String reason) { + return new BpmMessageSendWhenProcessInstanceRejectReqDTO() + .setProcessInstanceName(instance.getName()) + .setProcessInstanceId(instance.getId()) + .setReason(reason) + .setStartUserId(NumberUtils.parseLong(instance.getStartUserId())); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/task/BpmTaskConvert.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/task/BpmTaskConvert.java new file mode 100644 index 00000000..719ab2ba --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/task/BpmTaskConvert.java @@ -0,0 +1,170 @@ +package com.ruoyi.flowable.convert.task; + +import com.ruoyi.common.utils.NumberUtils; +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.domain.dto.message.BpmMessageSendWhenTaskCreatedReqDTO; +import com.ruoyi.flowable.domain.dto.user.AdminUserRespDTO; +import com.ruoyi.flowable.domain.dto.user.DeptRespDTO; +import com.ruoyi.flowable.domain.entity.task.BpmTaskExtDO; +import com.ruoyi.flowable.domain.vo.task.BpmTaskDonePageItemRespVO; +import com.ruoyi.flowable.domain.vo.task.BpmTaskRespVO; +import com.ruoyi.flowable.domain.vo.task.BpmTaskTodoPageItemRespVO; +import org.flowable.common.engine.impl.db.SuspensionState; +import org.flowable.engine.history.HistoricProcessInstance; +import org.flowable.engine.runtime.ProcessInstance; +import org.flowable.task.api.Task; +import org.flowable.task.api.history.HistoricTaskInstance; +import org.mapstruct.*; +import org.mapstruct.factory.Mappers; +import org.springframework.beans.BeanUtils; + +import java.util.Collections; +import java.util.List; +import java.util.Map; +import java.util.stream.Collectors; + +/** + * Bpm 任务 Convert + * + * hasPermi + */ +@Mapper +public interface BpmTaskConvert { + + BpmTaskConvert INSTANCE = Mappers.getMapper(BpmTaskConvert.class); + + /** + * 复制对象 + * + * @param source 源 要复制的对象 + * @param target 目标 复制到此对象 + * @param + * + * @return + */ + public static T copy(Object source, Class target) { + if (source == null || target == null) { + return null; + } + try { + T newInstance = target.newInstance(); + BeanUtils.copyProperties(source, newInstance); + return newInstance; + } catch (Exception e) { + throw new RuntimeException(e); + } + } + + default List copyList(List source, Class target) { + if (null == source || source.isEmpty()) { + return Collections.emptyList(); + } + return source.stream().map(e -> copy(e, target)).collect(Collectors.toList()); + } + + default List convertList1(List tasks, + Map processInstanceMap, Map userMap) { + return CollectionUtils.convertList(tasks, task -> { + BpmTaskTodoPageItemRespVO respVO = convert1(task); + ProcessInstance processInstance = processInstanceMap.get(task.getProcessInstanceId()); + if (processInstance != null) { + AdminUserRespDTO startUser = userMap.get(NumberUtils.parseLong(processInstance.getStartUserId())); + respVO.setProcessInstance(convert(processInstance, startUser)); + } + return respVO; + }); + } + + @Mapping(source = "suspended", target = "suspensionState", qualifiedByName = "convertSuspendedToSuspensionState") + BpmTaskTodoPageItemRespVO convert1(Task bean); + + @Named("convertSuspendedToSuspensionState") + default Integer convertSuspendedToSuspensionState(boolean suspended) { + return suspended ? SuspensionState.SUSPENDED.getStateCode() : SuspensionState.ACTIVE.getStateCode(); + } + + default List convertList2(List tasks, + Map bpmTaskExtDOMap, Map historicProcessInstanceMap, + Map userMap) { + return CollectionUtils.convertList(tasks, task -> { + BpmTaskDonePageItemRespVO respVO = convert2(task); + BpmTaskExtDO taskExtDO = bpmTaskExtDOMap.get(task.getId()); + copyTo(taskExtDO, respVO); + HistoricProcessInstance processInstance = historicProcessInstanceMap.get(task.getProcessInstanceId()); + if (processInstance != null) { + AdminUserRespDTO startUser = userMap.get(NumberUtils.parseLong(processInstance.getStartUserId())); + respVO.setProcessInstance(convert(processInstance, startUser)); + } + return respVO; + }); + } + + BpmTaskDonePageItemRespVO convert2(HistoricTaskInstance bean); + + @Mappings({@Mapping(source = "processInstance.id", target = "id"), + @Mapping(source = "processInstance.name", target = "name"), + @Mapping(source = "processInstance.startUserId", target = "startUserId"), + @Mapping(source = "processInstance.processDefinitionId", target = "processDefinitionId"), + @Mapping(source = "processInstance.businessKey", target = "businessKey"), + @Mapping(source = "processInstance.processDefinitionKey", target = "processDefinitionKey"), + @Mapping(source = "startUser.nickname", target = "startUserNickname")}) + BpmTaskTodoPageItemRespVO.ProcessInstance convert(ProcessInstance processInstance, AdminUserRespDTO startUser); + + default List convertList3(List tasks, + Map bpmTaskExtDOMap, HistoricProcessInstance processInstance, + Map userMap, Map deptMap) { + return CollectionUtils.convertList(tasks, task -> { + BpmTaskRespVO respVO = convert3(task); + BpmTaskExtDO taskExtDO = bpmTaskExtDOMap.get(task.getId()); + copyTo(taskExtDO, respVO); + if (processInstance != null) { + AdminUserRespDTO startUser = userMap.get(NumberUtils.parseLong(processInstance.getStartUserId())); + respVO.setProcessInstance(convert(processInstance, startUser)); + } + AdminUserRespDTO assignUser = userMap.get(NumberUtils.parseLong(task.getAssignee())); + if (assignUser != null) { + respVO.setAssigneeUser(convert3(assignUser)); + DeptRespDTO dept = deptMap.get(assignUser.getDeptId()); + if (dept != null) { + respVO.getAssigneeUser().setDeptName(dept.getName()); + } + } + return respVO; + }); + } + + @Mapping(source = "taskDefinitionKey", target = "definitionKey") + BpmTaskRespVO convert3(HistoricTaskInstance bean); + + BpmTaskRespVO.User convert3(AdminUserRespDTO bean); + + @Mapping(target = "id", ignore = true) + void copyTo(BpmTaskExtDO from, @MappingTarget BpmTaskDonePageItemRespVO to); + + @Mappings({@Mapping(source = "processInstance.id", target = "id"), + @Mapping(source = "processInstance.name", target = "name"), + @Mapping(source = "processInstance.startUserId", target = "startUserId"), + @Mapping(source = "processInstance.processDefinitionId", target = "processDefinitionId"), + @Mapping(source = "startUser.nickname", target = "startUserNickname")}) + BpmTaskTodoPageItemRespVO.ProcessInstance convert(HistoricProcessInstance processInstance, + AdminUserRespDTO startUser); + + default BpmTaskExtDO convert2TaskExt(Task task) { + BpmTaskExtDO taskExtDO = new BpmTaskExtDO().setTaskId(task.getId()) + .setAssigneeUserId(NumberUtils.parseLong(task.getAssignee())).setName(task.getName()) + .setProcessDefinitionId(task.getProcessDefinitionId()).setProcessInstanceId(task.getProcessInstanceId()); + taskExtDO.setCreateTime(task.getCreateTime()); + return taskExtDO; + } + + default BpmMessageSendWhenTaskCreatedReqDTO convert(ProcessInstance processInstance, AdminUserRespDTO startUser, + Task task) { + BpmMessageSendWhenTaskCreatedReqDTO reqDTO = new BpmMessageSendWhenTaskCreatedReqDTO(); + reqDTO.setProcessInstanceId(processInstance.getProcessInstanceId()) + .setProcessInstanceName(processInstance.getName()).setStartUserId(startUser.getId()) + .setStartUserNickname(startUser.getNickname()).setTaskId(task.getId()).setTaskName(task.getName()) + .setAssigneeUserId(NumberUtils.parseLong(task.getAssignee())); + return reqDTO; + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/user/DeptConvert.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/user/DeptConvert.java new file mode 100644 index 00000000..2a7dc669 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/user/DeptConvert.java @@ -0,0 +1,25 @@ +package com.ruoyi.flowable.convert.user; + +import com.ruoyi.common.core.domain.entity.SysDept; +import com.ruoyi.flowable.domain.dto.user.DeptRespDTO; +import org.mapstruct.Mapper; +import org.mapstruct.Mapping; +import org.mapstruct.Mappings; +import org.mapstruct.factory.Mappers; + +import java.util.List; + + +@Mapper +public interface DeptConvert { + + DeptConvert INSTANCE = Mappers.getMapper(DeptConvert.class); + + @Mappings({ + @Mapping(source = "deptId", target = "id"), + @Mapping(source = "deptName", target = "name") + }) + DeptRespDTO convert(SysDept bean); + + List convertList(List list); +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/user/UserConvert.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/user/UserConvert.java new file mode 100644 index 00000000..0184c341 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/convert/user/UserConvert.java @@ -0,0 +1,26 @@ +package com.ruoyi.flowable.convert.user; + +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.flowable.domain.dto.user.AdminUserRespDTO; +import org.mapstruct.Mapper; +import org.mapstruct.Mapping; +import org.mapstruct.Mappings; +import org.mapstruct.factory.Mappers; + +import java.util.List; + +@Mapper +public interface UserConvert { + + UserConvert INSTANCE = Mappers.getMapper(UserConvert.class); + + @Mappings({ + @Mapping(source = "userId", target = "id"), + @Mapping(source = "nickName", target = "nickname"), + @Mapping(source = "phonenumber", target = "mobile") + }) + AdminUserRespDTO convert(SysUser bean); + + List convertList(List users); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/DictTypeConstants.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/DictTypeConstants.java new file mode 100644 index 00000000..d3665cc1 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/DictTypeConstants.java @@ -0,0 +1,13 @@ +package com.ruoyi.flowable.core.enums; + +/** + * BPM 字典类型的枚举类 + * + * hasPermi + */ +public interface DictTypeConstants { + + String TASK_ASSIGN_RULE_TYPE = "bpm_task_assign_rule_type"; // 任务分配规则类型 + String TASK_ASSIGN_SCRIPT = "bpm_task_assign_script"; // 任务分配自定义脚本 + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/ErrorCodeConstants.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/ErrorCodeConstants.java new file mode 100644 index 00000000..351903ea --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/ErrorCodeConstants.java @@ -0,0 +1,65 @@ +package com.ruoyi.flowable.core.enums; + + +import com.ruoyi.common.exception.ErrorCode; + +/** + * 工作流 错误码枚举类 + * + * 工作流系统,使用 1-009-000-000 段 + */ +public interface ErrorCodeConstants { + + // ========== 通用流程处理 模块 1-009-000-000 ========== + ErrorCode HIGHLIGHT_IMG_ERROR = new ErrorCode(1009000002, "获取高亮流程图异常"); + + // ========== OA 流程模块 1-009-001-000 ========== + ErrorCode OA_LEAVE_NOT_EXISTS = new ErrorCode(1009001001, "请假申请不存在"); + ErrorCode OA_PM_POST_NOT_EXISTS = new ErrorCode(1009001002, "项目经理岗位未设置"); + ErrorCode OA_DEPART_PM_POST_NOT_EXISTS = new ErrorCode(1009001009, "部门的项目经理不存在"); + ErrorCode OA_BM_POST_NOT_EXISTS = new ErrorCode(1009001004, "部门经理岗位未设置"); + ErrorCode OA_DEPART_BM_POST_NOT_EXISTS = new ErrorCode(1009001005, "部门的部门经理不存在"); + ErrorCode OA_HR_POST_NOT_EXISTS = new ErrorCode(1009001006, "HR岗位未设置"); + ErrorCode OA_DAY_LEAVE_ERROR = new ErrorCode(1009001007, "请假天数必须>=1"); + + // ========== 流程模型 1-009-002-000 ========== + ErrorCode MODEL_KEY_EXISTS = new ErrorCode(1009002000, "已经存在流程标识为【{}】的流程"); + ErrorCode MODEL_NOT_EXISTS = new ErrorCode(1009002001, "流程模型不存在"); + ErrorCode MODEL_KEY_VALID = new ErrorCode(1009002002, "流程标识格式不正确,需要以字母或下划线开头,后接任意字母、数字、中划线、下划线、句点!"); + ErrorCode MODEL_DEPLOY_FAIL_FORM_NOT_CONFIG = new ErrorCode(1009002003, "部署流程失败,原因:流程表单未配置,请点击【修改流程】按钮进行配置"); + ErrorCode MODEL_DEPLOY_FAIL_TASK_ASSIGN_RULE_NOT_CONFIG = new ErrorCode(1009002004, "部署流程失败," + + "原因:用户任务({})未配置分配规则,请点击【修改流程】按钮进行配置"); + ErrorCode MODEL_DEPLOY_FAIL_TASK_INFO_EQUALS = new ErrorCode(1009003005, "流程定义部署失败,原因:信息未发生变化"); + + // ========== 流程定义 1-009-003-000 ========== + ErrorCode PROCESS_DEFINITION_KEY_NOT_MATCH = new ErrorCode(1009003000, "流程定义的标识期望是({}),当前是({}),请修改 BPMN 流程图"); + ErrorCode PROCESS_DEFINITION_NAME_NOT_MATCH = new ErrorCode(1009003001, "流程定义的名字期望是({}),当前是({}),请修改 BPMN 流程图"); + ErrorCode PROCESS_DEFINITION_NOT_EXISTS = new ErrorCode(1009003002, "流程定义不存在"); + ErrorCode PROCESS_DEFINITION_IS_SUSPENDED = new ErrorCode(1009003003, "流程定义处于挂起状态"); + ErrorCode PROCESS_DEFINITION_BPMN_MODEL_NOT_EXISTS = new ErrorCode(1009003004, "流程定义的模型不存在"); + + // ========== 流程实例 1-009-004-000 ========== + ErrorCode PROCESS_INSTANCE_NOT_EXISTS = new ErrorCode(1009004000, "流程实例不存在"); + ErrorCode PROCESS_INSTANCE_CANCEL_FAIL_NOT_EXISTS = new ErrorCode(1009004001, "流程取消失败,流程不处于运行中"); + ErrorCode PROCESS_INSTANCE_CANCEL_FAIL_NOT_SELF = new ErrorCode(1009004002, "流程取消失败,该流程不是你发起的"); + + // ========== 流程任务 1-009-005-000 ========== + ErrorCode TASK_COMPLETE_FAIL_NOT_EXISTS = new ErrorCode(1009005000, "审批任务失败,原因:该任务不处于未审批"); + ErrorCode TASK_COMPLETE_FAIL_ASSIGN_NOT_SELF = new ErrorCode(1009005001, "审批任务失败,原因:该任务的审批人不是你"); + + // ========== 流程任务分配规则 1-009-006-000 ========== + ErrorCode TASK_ASSIGN_RULE_EXISTS = new ErrorCode(1009006000, "流程({}) 的任务({}) 已经存在分配规则"); + ErrorCode TASK_ASSIGN_RULE_NOT_EXISTS = new ErrorCode(1009006001, "流程任务分配规则不存在"); + ErrorCode TASK_UPDATE_FAIL_NOT_MODEL = new ErrorCode(1009006002, "只有流程模型的任务分配规则,才允许被修改"); + ErrorCode TASK_CREATE_FAIL_NO_CANDIDATE_USER = new ErrorCode(1009006003, "操作失败,原因:找不到任务的审批人!"); + ErrorCode TASK_ASSIGN_SCRIPT_NOT_EXISTS = new ErrorCode(1009006004, "操作失败,原因:任务分配脚本({}) 不存在"); + + // ========== 动态表单模块 1-009-010-000 ========== + ErrorCode FORM_NOT_EXISTS = new ErrorCode(1009010000, "动态表单不存在"); + ErrorCode FORM_FIELD_REPEAT = new ErrorCode(1009010001, "表单项({}) 和 ({}) 使用了相同的字段名({})"); + + // ========== 用户组模块 1-009-011-000 ========== + ErrorCode USER_GROUP_NOT_EXISTS = new ErrorCode(1009011000, "用户组不存在"); + ErrorCode USER_GROUP_IS_DISABLE = new ErrorCode(1009011001, "名字为【{}】的用户组已被禁用"); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/definition/BpmModelFormTypeEnum.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/definition/BpmModelFormTypeEnum.java new file mode 100644 index 00000000..c1ce540f --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/definition/BpmModelFormTypeEnum.java @@ -0,0 +1,21 @@ +package com.ruoyi.flowable.core.enums.definition; + +import lombok.AllArgsConstructor; +import lombok.Getter; + +/** + * BPM 模型的表单类型的枚举 + * + * hasPermi + */ +@Getter +@AllArgsConstructor +public enum BpmModelFormTypeEnum { + + NORMAL(10, "流程表单"), // 对应 BpmFormDO + CUSTOM(20, "业务表单") // 业务自己定义的表单,自己进行数据的存储 + ; + + private final Integer type; + private final String desc; +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/definition/BpmTaskAssignRuleTypeEnum.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/definition/BpmTaskAssignRuleTypeEnum.java new file mode 100644 index 00000000..cfc1ac0d --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/definition/BpmTaskAssignRuleTypeEnum.java @@ -0,0 +1,34 @@ +package com.ruoyi.flowable.core.enums.definition; + +import lombok.AllArgsConstructor; +import lombok.Getter; + +/** + * BPM 任务分配规则的类型枚举 + * + * hasPermi + */ +@Getter +@AllArgsConstructor +public enum BpmTaskAssignRuleTypeEnum { + + ROLE(10, "角色"), + DEPT_MEMBER(20, "部门的成员"), // 包括负责人 + DEPT_LEADER(21, "部门的负责人"), + POST(22, "岗位"), + USER(30, "用户"), + USER_GROUP(40, "用户组"), + SCRIPT(50, "自定义脚本"), // 例如说,发起人所在部门的领导、发起人所在部门的领导的领导 + VARIABLE(60, "流程变量"), // 目前固定取流程key+"_assignees",FlowableUtils.formatCollectionVariable(execution.getCurrentActivityId()) + ; + + /** + * 类型 + */ + private final Integer type; + /** + * 描述 + */ + private final String desc; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/definition/BpmTaskRuleScriptEnum.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/definition/BpmTaskRuleScriptEnum.java new file mode 100644 index 00000000..0aee6396 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/definition/BpmTaskRuleScriptEnum.java @@ -0,0 +1,30 @@ +package com.ruoyi.flowable.core.enums.definition; + +import lombok.AllArgsConstructor; +import lombok.Getter; + +/** + * BPM 任务规则的脚本枚举 + * 目前暂时通过 TODO 芋艿:硬编码,未来可以考虑 Groovy 动态脚本的方式 + * + * hasPermi + */ +@Getter +@AllArgsConstructor +public enum BpmTaskRuleScriptEnum { + + START_USER(10L, "流程发起人"), + + LEADER_X1(20L, "流程发起人的一级领导"), + LEADER_X2(21L, "流程发起人的二级领导"); + + /** + * 脚本编号 + */ + private final Long id; + /** + * 脚本描述 + */ + private final String desc; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/message/BpmMessageEnum.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/message/BpmMessageEnum.java new file mode 100644 index 00000000..cf806a6b --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/message/BpmMessageEnum.java @@ -0,0 +1,26 @@ +package com.ruoyi.flowable.core.enums.message; + +import lombok.AllArgsConstructor; +import lombok.Getter; + +/** + * Bpm 消息的枚举 + * + * hasPermi + */ +@AllArgsConstructor +@Getter +public enum BpmMessageEnum { + + PROCESS_INSTANCE_APPROVE("bpm_process_instance_approve"), // 流程任务被审批通过时,发送给申请人 + PROCESS_INSTANCE_REJECT("bpm_process_instance_reject"), // 流程任务被审批不通过时,发送给申请人 + TASK_ASSIGNED("bpm_task_assigned"); // 任务被分配时,发送给审批人 + + /** + * 短信模板的标识 + * + * 关联 SmsTemplateDO 的 code 属性 + */ + private final String smsTemplateCode; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceDeleteReasonEnum.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceDeleteReasonEnum.java new file mode 100644 index 00000000..a64b0e7c --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceDeleteReasonEnum.java @@ -0,0 +1,58 @@ +package com.ruoyi.flowable.core.enums.task; + +import cn.hutool.core.util.StrUtil; +import lombok.AllArgsConstructor; +import lombok.Getter; + +/** + * 流程实例的删除原因 + * + * hasPermi + */ +@Getter +@AllArgsConstructor +public enum BpmProcessInstanceDeleteReasonEnum { + + REJECT_TASK("不通过任务,原因:{}"), // 修改文案时,需要注意 isRejectReason 方法 + CANCEL_TASK("主动取消任务,原因:{}"), + + // ========== 流程任务的独有原因 ========== + MULTI_TASK_END("系统自动取消,原因:多任务审批已经满足条件,无需审批该任务"), // 多实例满足 condition 而结束时,其它任务实例任务会被取消,对应的删除原因是 MI_END + + ; + + private final String reason; + + /** + * 格式化理由 + * + * @param args 参数 + * @return 理由 + */ + public String format(Object... args) { + return StrUtil.format(reason, args); + } + + // ========== 逻辑 ========== + + public static boolean isRejectReason(String reason) { + return StrUtil.startWith(reason, "不通过任务,原因:"); + } + + /** + * 将 Flowable 的删除原因,翻译成对应的中文原因 + * + * @param reason 原始原因 + * @return 原因 + */ + public static String translateReason(String reason) { + if (StrUtil.isEmpty(reason)) { + return reason; + } + switch (reason) { + case "MI_END": return MULTI_TASK_END.getReason(); + default: return reason; + } + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceResultEnum.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceResultEnum.java new file mode 100644 index 00000000..f594db69 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceResultEnum.java @@ -0,0 +1,48 @@ +package com.ruoyi.flowable.core.enums.task; + +import com.ruoyi.common.utils.ObjectUtils; +import lombok.AllArgsConstructor; +import lombok.Getter; + +/** + * 流程实例的结果 + * + * @author jason + */ +@Getter +@AllArgsConstructor +public enum BpmProcessInstanceResultEnum { + + PROCESS(1, "处理中"), + APPROVE(2, "通过"), + REJECT(3, "驳回"), + CANCEL(4, "已取消"), + + // ========== 流程任务独有的状态 ========== + + BACK(5, "退回/驳回"); + + /** + * 结果 + * + * 如果新增时,注意 {@link #isEndResult(Integer)} 是否需要变更 + */ + private final Integer result; + /** + * 描述 + */ + private final String desc; + + /** + * 判断该结果是否已经处于 End 最终结果 + * + * 主要用于一些结果更新的逻辑,如果已经是最终结果,就不再进行更新 + * + * @param result 结果 + * @return 是否 + */ + public static boolean isEndResult(Integer result) { + return ObjectUtils.equalsAny(result, APPROVE.getResult(), REJECT.getResult(), CANCEL.getResult(), BACK.getResult()); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceStatusEnum.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceStatusEnum.java new file mode 100644 index 00000000..1210fd26 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceStatusEnum.java @@ -0,0 +1,27 @@ +package com.ruoyi.flowable.core.enums.task; + +import lombok.AllArgsConstructor; +import lombok.Getter; + +/** + * 流程实例的状态 + * + * hasPermi + */ +@Getter +@AllArgsConstructor +public enum BpmProcessInstanceStatusEnum { + + RUNNING(1, "进行中"), + FINISH(2, "已完成"); + + /** + * 状态 + */ + private final Integer status; + /** + * 描述 + */ + private final String desc; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/user/ErrorCodeConstants.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/user/ErrorCodeConstants.java new file mode 100644 index 00000000..195294f0 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/user/ErrorCodeConstants.java @@ -0,0 +1,28 @@ +package com.ruoyi.flowable.core.enums.user; + + +import com.ruoyi.common.exception.ErrorCode; + +/** + * Member 错误码枚举类 + *

    + * member 系统,使用 1-004-000-000 段 + */ +public interface ErrorCodeConstants { + + // ========== 用户相关 1004001000============ + ErrorCode USER_NOT_EXISTS = new ErrorCode(1004001000, "用户不存在"); + ErrorCode USER_PASSWORD_FAILED = new ErrorCode(1004001001, "密码校验失败"); + + // ========== AUTH 模块 1004003000 ========== + ErrorCode AUTH_LOGIN_BAD_CREDENTIALS = new ErrorCode(1004003000, "登录失败,账号密码不正确"); + ErrorCode AUTH_LOGIN_USER_DISABLED = new ErrorCode(1004003001, "登录失败,账号被禁用"); + ErrorCode AUTH_TOKEN_EXPIRED = new ErrorCode(1004003004, "Token 已经过期"); + ErrorCode AUTH_THIRD_LOGIN_NOT_BIND = new ErrorCode(1004003005, "未绑定账号,需要进行绑定"); + ErrorCode AUTH_WEIXIN_MINI_APP_PHONE_CODE_ERROR = new ErrorCode(1004003006, "获得手机号失败"); + + // ========== 用户收件地址 1004004000 ========== + ErrorCode ADDRESS_NOT_EXISTS = new ErrorCode(1004004000, "用户收件地址不存在"); + ErrorCode ADDRESS_FORBIDDEN = new ErrorCode(1004004001, "没有该操作权限"); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/user/SysErrorCodeConstants.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/user/SysErrorCodeConstants.java new file mode 100644 index 00000000..bc65ec64 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/enums/user/SysErrorCodeConstants.java @@ -0,0 +1,146 @@ +package com.ruoyi.flowable.core.enums.user; + + +import com.ruoyi.common.exception.ErrorCode; + +/** + * System 错误码枚举类 + *

    + * system 系统,使用 1-002-000-000 段 + */ +public interface SysErrorCodeConstants { + + // ========== AUTH 模块 1002000000 ========== + ErrorCode AUTH_LOGIN_BAD_CREDENTIALS = new ErrorCode(1002000000, "登录失败,账号密码不正确"); + ErrorCode AUTH_LOGIN_USER_DISABLED = new ErrorCode(1002000001, "登录失败,账号被禁用"); + ErrorCode AUTH_LOGIN_CAPTCHA_NOT_FOUND = new ErrorCode(1002000003, "验证码不存在"); + ErrorCode AUTH_LOGIN_CAPTCHA_CODE_ERROR = new ErrorCode(1002000004, "验证码不正确"); + ErrorCode AUTH_THIRD_LOGIN_NOT_BIND = new ErrorCode(1002000005, "未绑定账号,需要进行绑定"); + ErrorCode AUTH_TOKEN_EXPIRED = new ErrorCode(1002000006, "Token 已经过期"); + ErrorCode AUTH_MOBILE_NOT_EXISTS = new ErrorCode(1002000007, "手机号不存在"); + + // ========== 菜单模块 1002001000 ========== + ErrorCode MENU_NAME_DUPLICATE = new ErrorCode(1002001000, "已经存在该名字的菜单"); + ErrorCode MENU_PARENT_NOT_EXISTS = new ErrorCode(1002001001, "父菜单不存在"); + ErrorCode MENU_PARENT_ERROR = new ErrorCode(1002001002, "不能设置自己为父菜单"); + ErrorCode MENU_NOT_EXISTS = new ErrorCode(1002001003, "菜单不存在"); + ErrorCode MENU_EXISTS_CHILDREN = new ErrorCode(1002001004, "存在子菜单,无法删除"); + ErrorCode MENU_PARENT_NOT_DIR_OR_MENU = new ErrorCode(1002001005, "父菜单的类型必须是目录或者菜单"); + + // ========== 角色模块 1002002000 ========== + ErrorCode ROLE_NOT_EXISTS = new ErrorCode(1002002000, "角色不存在"); + ErrorCode ROLE_NAME_DUPLICATE = new ErrorCode(1002002001, "已经存在名为【{}】的角色"); + ErrorCode ROLE_CODE_DUPLICATE = new ErrorCode(1002002002, "已经存在编码为【{}】的角色"); + ErrorCode ROLE_CAN_NOT_UPDATE_SYSTEM_TYPE_ROLE = new ErrorCode(1002002003, "不能操作类型为系统内置的角色"); + ErrorCode ROLE_IS_DISABLE = new ErrorCode(1002002004, "名字为【{}】的角色已被禁用"); + ErrorCode ROLE_ADMIN_CODE_ERROR = new ErrorCode(1002002005, "编码【{}】不能使用"); + + // ========== 用户模块 1002003000 ========== + ErrorCode USER_USERNAME_EXISTS = new ErrorCode(1002003000, "用户账号已经存在"); + ErrorCode USER_MOBILE_EXISTS = new ErrorCode(1002003001, "手机号已经存在"); + ErrorCode USER_EMAIL_EXISTS = new ErrorCode(1002003002, "邮箱已经存在"); + ErrorCode USER_NOT_EXISTS = new ErrorCode(1002003003, "用户不存在"); + ErrorCode USER_IMPORT_LIST_IS_EMPTY = new ErrorCode(1002003004, "导入用户数据不能为空!"); + ErrorCode USER_PASSWORD_FAILED = new ErrorCode(1002003005, "用户密码校验失败"); + ErrorCode USER_IS_DISABLE = new ErrorCode(1002003006, "名字为【{}】的用户已被禁用"); + ErrorCode USER_COUNT_MAX = new ErrorCode(1002003008, "创建用户失败,原因:超过租户最大租户配额({})!"); + + // ========== 部门模块 1002004000 ========== + ErrorCode DEPT_NAME_DUPLICATE = new ErrorCode(1002004000, "已经存在该名字的部门"); + ErrorCode DEPT_PARENT_NOT_EXITS = new ErrorCode(1002004001, "父级部门不存在"); + ErrorCode DEPT_NOT_FOUND = new ErrorCode(1002004002, "当前部门不存在"); + ErrorCode DEPT_EXITS_CHILDREN = new ErrorCode(1002004003, "存在子部门,无法删除"); + ErrorCode DEPT_PARENT_ERROR = new ErrorCode(1002004004, "不能设置自己为父部门"); + ErrorCode DEPT_EXISTS_USER = new ErrorCode(1002004005, "部门中存在员工,无法删除"); + ErrorCode DEPT_NOT_ENABLE = new ErrorCode(1002004006, "部门不处于开启状态,不允许选择"); + ErrorCode DEPT_PARENT_IS_CHILD = new ErrorCode(1002004007, "不能设置自己的子部门为父部门"); + + // ========== 岗位模块 1002005000 ========== + ErrorCode POST_NOT_FOUND = new ErrorCode(1002005000, "当前岗位不存在"); + ErrorCode POST_NOT_ENABLE = new ErrorCode(1002005001, "岗位({}) 不处于开启状态,不允许选择"); + ErrorCode POST_NAME_DUPLICATE = new ErrorCode(1002005002, "已经存在该名字的岗位"); + ErrorCode POST_CODE_DUPLICATE = new ErrorCode(1002005003, "已经存在该标识的岗位"); + + // ========== 字典类型 1002006000 ========== + ErrorCode DICT_TYPE_NOT_EXISTS = new ErrorCode(1002006001, "当前字典类型不存在"); + ErrorCode DICT_TYPE_NOT_ENABLE = new ErrorCode(1002006002, "字典类型不处于开启状态,不允许选择"); + ErrorCode DICT_TYPE_NAME_DUPLICATE = new ErrorCode(1002006003, "已经存在该名字的字典类型"); + ErrorCode DICT_TYPE_TYPE_DUPLICATE = new ErrorCode(1002006004, "已经存在该类型的字典类型"); + ErrorCode DICT_TYPE_HAS_CHILDREN = new ErrorCode(1002006005, "无法删除,该字典类型还有字典数据"); + + // ========== 字典数据 1002007000 ========== + ErrorCode DICT_DATA_NOT_EXISTS = new ErrorCode(1002007001, "当前字典数据不存在"); + ErrorCode DICT_DATA_NOT_ENABLE = new ErrorCode(1002007002, "字典数据({})不处于开启状态,不允许选择"); + ErrorCode DICT_DATA_VALUE_DUPLICATE = new ErrorCode(1002007003, "已经存在该值的字典数据"); + + // ========== 通知公告 1002008000 ========== + ErrorCode NOTICE_NOT_FOUND = new ErrorCode(1002008001, "当前通知公告不存在"); + + // ========== 短信渠道 1002011000 ========== + ErrorCode SMS_CHANNEL_NOT_EXISTS = new ErrorCode(1002011000, "短信渠道不存在"); + ErrorCode SMS_CHANNEL_DISABLE = new ErrorCode(1002011001, "短信渠道不处于开启状态,不允许选择"); + ErrorCode SMS_CHANNEL_HAS_CHILDREN = new ErrorCode(1002011002, "无法删除,该短信渠道还有短信模板"); + + // ========== 短信模板 1002012000 ========== + ErrorCode SMS_TEMPLATE_NOT_EXISTS = new ErrorCode(1002012000, "短信模板不存在"); + ErrorCode SMS_TEMPLATE_CODE_DUPLICATE = new ErrorCode(1002012001, "已经存在编码为【{}】的短信模板"); + + // ========== 短信发送 1002013000 ========== + ErrorCode SMS_SEND_MOBILE_NOT_EXISTS = new ErrorCode(1002013000, "手机号不存在"); + ErrorCode SMS_SEND_MOBILE_TEMPLATE_PARAM_MISS = new ErrorCode(1002013001, "模板参数({})缺失"); + ErrorCode SMS_SEND_TEMPLATE_NOT_EXISTS = new ErrorCode(1002013002, "短信模板不存在"); + + // ========== 短信验证码 1002014000 ========== + ErrorCode SMS_CODE_NOT_FOUND = new ErrorCode(1002014000, "验证码不存在"); + ErrorCode SMS_CODE_EXPIRED = new ErrorCode(1002014001, "验证码已过期"); + ErrorCode SMS_CODE_USED = new ErrorCode(1002014002, "验证码已使用"); + ErrorCode SMS_CODE_NOT_CORRECT = new ErrorCode(1002014003, "验证码不正确"); + ErrorCode SMS_CODE_EXCEED_SEND_MAXIMUM_QUANTITY_PER_DAY = new ErrorCode(1002014004, "超过每日短信发送数量"); + ErrorCode SMS_CODE_SEND_TOO_FAST = new ErrorCode(1002014005, "短信发送过于频率"); + ErrorCode SMS_CODE_IS_EXISTS = new ErrorCode(1002014006, "手机号已被使用"); + ErrorCode SMS_CODE_IS_UNUSED = new ErrorCode(1002014007, "验证码未被使用"); + + // ========== 租户信息 1002015000 ========== + ErrorCode TENANT_NOT_EXISTS = new ErrorCode(1002015000, "租户不存在"); + ErrorCode TENANT_DISABLE = new ErrorCode(1002015001, "名字为【{}】的租户已被禁用"); + ErrorCode TENANT_EXPIRE = new ErrorCode(1002015002, "名字为【{}】的租户已过期"); + ErrorCode TENANT_CAN_NOT_UPDATE_SYSTEM = new ErrorCode(1002015003, "系统租户不能进行修改、删除等操作!"); + + // ========== 租户套餐 1002016000 ========== + ErrorCode TENANT_PACKAGE_NOT_EXISTS = new ErrorCode(1002016000, "租户套餐不存在"); + ErrorCode TENANT_PACKAGE_USED = new ErrorCode(1002016001, "租户正在使用该套餐,请给租户重新设置套餐后再尝试删除"); + ErrorCode TENANT_PACKAGE_DISABLE = new ErrorCode(1002016002, "名字为【{}】的租户套餐已被禁用"); + + // ========== 错误码模块 1002017000 ========== + ErrorCode ERROR_CODE_NOT_EXISTS = new ErrorCode(1002017000, "错误码不存在"); + ErrorCode ERROR_CODE_DUPLICATE = new ErrorCode(1002017001, "已经存在编码为【{}】的错误码"); + + // ========== 社交用户 1002018000 ========== + ErrorCode SOCIAL_USER_AUTH_FAILURE = new ErrorCode(1002018000, "社交授权失败,原因是:{}"); + ErrorCode SOCIAL_USER_UNBIND_NOT_SELF = new ErrorCode(1002018001, "社交解绑失败,非当前用户绑定"); + ErrorCode SOCIAL_USER_NOT_FOUND = new ErrorCode(1002018002, "社交授权失败,找不到对应的用户"); + + // ========== 系统敏感词 1002019000 ========= + ErrorCode SENSITIVE_WORD_NOT_EXISTS = new ErrorCode(1002019000, "系统敏感词在所有标签中都不存在"); + ErrorCode SENSITIVE_WORD_EXISTS = new ErrorCode(1002019001, "系统敏感词已在标签中存在"); + + // ========== OAuth2 客户端 1002020000 ========= + ErrorCode OAUTH2_CLIENT_NOT_EXISTS = new ErrorCode(1002020000, "OAuth2 客户端不存在"); + ErrorCode OAUTH2_CLIENT_EXISTS = new ErrorCode(1002020001, "OAuth2 客户端编号已存在"); + ErrorCode OAUTH2_CLIENT_DISABLE = new ErrorCode(1002020002, "OAuth2 客户端已禁用"); + ErrorCode OAUTH2_CLIENT_AUTHORIZED_GRANT_TYPE_NOT_EXISTS = new ErrorCode(1002020003, "不支持该授权类型"); + ErrorCode OAUTH2_CLIENT_SCOPE_OVER = new ErrorCode(1002020004, "授权范围过大"); + ErrorCode OAUTH2_CLIENT_REDIRECT_URI_NOT_MATCH = new ErrorCode(1002020005, "无效 redirect_uri: {}"); + ErrorCode OAUTH2_CLIENT_CLIENT_SECRET_ERROR = new ErrorCode(1002020006, "无效 client_secret: {}"); + + // ========== OAuth2 授权 1002021000 ========= + ErrorCode OAUTH2_GRANT_CLIENT_ID_MISMATCH = new ErrorCode(1002021000, "client_id 不匹配"); + ErrorCode OAUTH2_GRANT_REDIRECT_URI_MISMATCH = new ErrorCode(1002021001, "redirect_uri 不匹配"); + ErrorCode OAUTH2_GRANT_STATE_MISMATCH = new ErrorCode(1002021002, "state 不匹配"); + ErrorCode OAUTH2_GRANT_CODE_NOT_EXISTS = new ErrorCode(1002021003, "code 不存在"); + + // ========== OAuth2 授权 1002022000 ========= + ErrorCode OAUTH2_CODE_NOT_EXISTS = new ErrorCode(1002022000, "code 不存在"); + ErrorCode OAUTH2_CODE_EXPIRE = new ErrorCode(1002022000, "code 已过期"); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/utils/FlowableUtils.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/utils/FlowableUtils.java new file mode 100644 index 00000000..6401e6f5 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/utils/FlowableUtils.java @@ -0,0 +1,84 @@ +package com.ruoyi.flowable.core.utils; + +import org.flowable.bpmn.converter.BpmnXMLConverter; +import org.flowable.bpmn.model.BpmnModel; +import org.flowable.bpmn.model.FlowElement; +import org.flowable.common.engine.impl.identity.Authentication; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; + +/** + * Flowable 相关的工具方法 + * + * hasPermi + */ +public class FlowableUtils { + + public static String assignees = "_assignees";; + + // ========== User 相关的工具方法 ========== + public static void setAuthenticatedUserId(Long userId) { + Authentication.setAuthenticatedUserId(String.valueOf(userId)); + } + + public static void clearAuthenticatedUserId() { + Authentication.setAuthenticatedUserId(null); + } + + // ========== BPMN 相关的工具方法 ========== + + /** + * 获得 BPMN 流程中,指定的元素们 + * + * @param model + * @param clazz 指定元素。例如说,{@link org.flowable.bpmn.model.UserTask}、{@link org.flowable.bpmn.model.Gateway} 等等 + * @return 元素们 + */ + public static List getBpmnModelElements(BpmnModel model, Class clazz) { + List result = new ArrayList<>(); + model.getProcesses().forEach(process -> { + process.getFlowElements().forEach(flowElement -> { + if (flowElement.getClass().isAssignableFrom(clazz)) { + result.add((T) flowElement); + } + }); + }); + return result; + } + + /** + * 比较 两个bpmnModel 是否相同 + * @param oldModel 老的bpmn model + * @param newModel 新的bpmn model + */ + public static boolean equals(BpmnModel oldModel, BpmnModel newModel) { + // 由于 BpmnModel 未提供 equals 方法,所以只能转成字节数组,进行比较 + return Arrays.equals(getBpmnBytes(oldModel), getBpmnBytes(newModel)); + } + + /** + * 把 bpmnModel 转换成 byte[] + * @param model bpmnModel + */ + public static byte[] getBpmnBytes(BpmnModel model) { + if (model == null) { + return new byte[0]; + } + BpmnXMLConverter converter = new BpmnXMLConverter(); + return converter.convertToXML(model); + } + + // ========== Execution 相关的工具方法 ========== + + public static String formatCollectionVariable(String activityId) { + + return activityId + assignees; + } + + public static String formatCollectionElementVariable(String activityId) { + return activityId + "_assignee"; + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/web/FlowableWebFilter.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/web/FlowableWebFilter.java new file mode 100644 index 00000000..c3ee83f8 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/core/web/FlowableWebFilter.java @@ -0,0 +1,37 @@ +package com.ruoyi.flowable.core.web; + +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.flowable.core.utils.FlowableUtils; +import org.springframework.web.filter.OncePerRequestFilter; + +import javax.servlet.FilterChain; +import javax.servlet.ServletException; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import java.io.IOException; + +/** + * flowable Web 过滤器,将 userId 设置到 {@link org.flowable.common.engine.impl.identity.Authentication} 中 + * + * @author jason + */ +public class FlowableWebFilter extends OncePerRequestFilter { + + @Override + protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain chain) + throws ServletException, IOException { + try { + // 设置工作流的用户 + Long userId = SecurityUtils.getLoginUserId(); + if (userId != null) { + FlowableUtils.setAuthenticatedUserId(userId); + } + // 过滤 + chain.doFilter(request, response); + } finally { + // 清理 + FlowableUtils.clearAuthenticatedUserId(); + } + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/definition/BpmFormFieldRespDTO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/definition/BpmFormFieldRespDTO.java new file mode 100644 index 00000000..192667f4 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/definition/BpmFormFieldRespDTO.java @@ -0,0 +1,27 @@ +package com.ruoyi.flowable.domain.dto.definition; + +import com.fasterxml.jackson.annotation.JsonProperty; +import lombok.Data; +import lombok.experimental.Accessors; + +/** + * Bpm 表单的 Field 表单项 Response DTO + * 字段的定义,可见 https://github.com/JakHuang/form-generator/issues/46 文档 + * + * hasPermi + */ +@Data +@Accessors(chain = true) +public class BpmFormFieldRespDTO { + + /** + * 表单标题 + */ + private String label; + /** + * 表单字段的属性名,可自定义 + */ + @JsonProperty(value = "vModel") + private String vModel; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/definition/BpmModelMetaInfoRespDTO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/definition/BpmModelMetaInfoRespDTO.java new file mode 100644 index 00000000..4afa340b --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/definition/BpmModelMetaInfoRespDTO.java @@ -0,0 +1,39 @@ +package com.ruoyi.flowable.domain.dto.definition; + +import com.ruoyi.flowable.core.enums.definition.BpmModelFormTypeEnum; +import lombok.Data; + +/** + * BPM 流程 MetaInfo Response DTO + * 主要用于 { Model#setMetaInfo(String)} 的存储 + * + * hasPermi + */ +@Data +public class BpmModelMetaInfoRespDTO { + + /** + * 流程描述 + */ + private String description; + /** + * 表单类型 + */ + private Integer formType; + /** + * 表单编号 + * 在表单类型为 {@link BpmModelFormTypeEnum#NORMAL} 时 + */ + private Long formId; + /** + * 自定义表单的提交路径,使用 Vue 的路由地址 + * 在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时 + */ + private String formCustomCreatePath; + /** + * 自定义表单的查看路径,使用 Vue 的路由地址 + * 在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时 + */ + private String formCustomViewPath; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/definition/BpmProcessDefinitionCreateReqDTO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/definition/BpmProcessDefinitionCreateReqDTO.java new file mode 100644 index 00000000..f1780f74 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/definition/BpmProcessDefinitionCreateReqDTO.java @@ -0,0 +1,106 @@ +package com.ruoyi.flowable.domain.dto.definition; + +import cn.hutool.core.collection.CollUtil; +import cn.hutool.core.util.StrUtil; +import com.ruoyi.flowable.core.enums.definition.BpmModelFormTypeEnum; +import lombok.Data; +import lombok.experimental.Accessors; + +import javax.validation.constraints.AssertTrue; +import javax.validation.constraints.NotEmpty; +import javax.validation.constraints.NotNull; +import java.util.List; +import java.util.Objects; + +/** + * 流程定义创建 Request DTO + */ +@Data +@Accessors(chain = true) +public class BpmProcessDefinitionCreateReqDTO { + + // ========== 模型相关 ========== + + /** + * 流程模型的编号 + */ + @NotEmpty(message = "流程模型编号不能为空") + private String modelId; + /** + * 流程标识 + */ + @NotEmpty(message = "流程标识不能为空") + private String key; + /** + * 流程名称 + */ + @NotEmpty(message = "流程名称不能为空") + private String name; + /** + * 流程描述 + */ + private String description; + /** + * 流程分类 + * 参见 bpm_model_category 数据字典 + */ + @NotEmpty(message = "流程分类不能为空") + private String category; + /** + * BPMN XML + */ + @NotEmpty(message = "BPMN XML 不能为空") + private byte[] bpmnBytes; + + // ========== 表单相关 ========== + + /** + * 表单类型 + */ + @NotNull(message = "表单类型不能为空") + private Integer formType; + /** + * 动态表单编号 + * 在表单类型为 {@link BpmModelFormTypeEnum#NORMAL} 时 + */ + private Long formId; + /** + * 表单的配置 + * 在表单类型为 {@link BpmModelFormTypeEnum#NORMAL} 时 + */ + private String formConf; + /** + * 表单项的数组 + * 在表单类型为 {@link BpmModelFormTypeEnum#NORMAL} 时 + */ + private List formFields; + /** + * 自定义表单的提交路径,使用 Vue 的路由地址 + * 在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时 + */ + private String formCustomCreatePath; + /** + * 自定义表单的查看路径,使用 Vue 的路由地址 + * 在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时 + */ + private String formCustomViewPath; + + @AssertTrue(message = "流程表单信息不全") + public boolean isNormalFormTypeValid() { + // 如果非业务表单,则直接通过 + if (!Objects.equals(formType, BpmModelFormTypeEnum.NORMAL.getType())) { + return true; + } + return formId != null && StrUtil.isNotEmpty(formConf) && CollUtil.isNotEmpty(formFields); + } + + @AssertTrue(message = "业务表单信息不全") + public boolean isNormalCustomTypeValid() { + // 如果非业务表单,则直接通过 + if (!Objects.equals(formType, BpmModelFormTypeEnum.CUSTOM.getType())) { + return true; + } + return StrUtil.isNotEmpty(formCustomCreatePath) && StrUtil.isNotEmpty(formCustomViewPath); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenProcessInstanceApproveReqDTO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenProcessInstanceApproveReqDTO.java new file mode 100644 index 00000000..fdff5466 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenProcessInstanceApproveReqDTO.java @@ -0,0 +1,29 @@ +package com.ruoyi.flowable.domain.dto.message; + +import lombok.Data; +import lombok.experimental.Accessors; + +import javax.validation.constraints.NotEmpty; +import javax.validation.constraints.NotNull; + +/** + * BPM 发送流程实例被通过 Request DTO + */ +@Data +@Accessors(chain = true) +public class BpmMessageSendWhenProcessInstanceApproveReqDTO { + + /** + * 流程实例的编号 + */ + @NotEmpty(message = "流程实例的编号不能为空") + private String processInstanceId; + /** + * 流程实例的名字 + */ + @NotEmpty(message = "流程实例的名字不能为空") + private String processInstanceName; + @NotNull(message = "发起人的用户编号") + private Long startUserId; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenProcessInstanceRejectReqDTO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenProcessInstanceRejectReqDTO.java new file mode 100644 index 00000000..7b583a10 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenProcessInstanceRejectReqDTO.java @@ -0,0 +1,35 @@ +package com.ruoyi.flowable.domain.dto.message; + +import lombok.Data; +import lombok.experimental.Accessors; + +import javax.validation.constraints.NotEmpty; +import javax.validation.constraints.NotNull; + +/** + * BPM 发送流程实例被不通过 Request DTO + */ +@Data +@Accessors(chain = true) +public class BpmMessageSendWhenProcessInstanceRejectReqDTO { + + /** + * 流程实例的编号 + */ + @NotEmpty(message = "流程实例的编号不能为空") + private String processInstanceId; + /** + * 流程实例的名字 + */ + @NotEmpty(message = "流程实例的名字不能为空") + private String processInstanceName; + @NotNull(message = "发起人的用户编号") + private Long startUserId; + + /** + * 不通过理由 + */ + @NotEmpty(message = "不通过理由不能为空") + private String reason; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenTaskCreatedReqDTO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenTaskCreatedReqDTO.java new file mode 100644 index 00000000..b39fee36 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenTaskCreatedReqDTO.java @@ -0,0 +1,48 @@ +package com.ruoyi.flowable.domain.dto.message; + +import lombok.Data; +import lombok.experimental.Accessors; + +import javax.validation.constraints.NotEmpty; +import javax.validation.constraints.NotNull; + +/** + * BPM 发送任务被分配 Request DTO + */ +@Data +@Accessors(chain = true) +public class BpmMessageSendWhenTaskCreatedReqDTO { + + /** + * 流程实例的编号 + */ + @NotEmpty(message = "流程实例的编号不能为空") + private String processInstanceId; + /** + * 流程实例的名字 + */ + @NotEmpty(message = "流程实例的名字不能为空") + private String processInstanceName; + @NotNull(message = "发起人的用户编号") + private Long startUserId; + @NotEmpty(message = "发起人的昵称") + private String startUserNickname; + + /** + * 流程任务的编号 + */ + @NotEmpty(message = "流程任务的编号不能为空") + private String taskId; + /** + * 流程任务的名字 + */ + @NotEmpty(message = "流程任务的名字不能为空") + private String taskName; + + /** + * 审批人的用户编号 + */ + @NotNull(message = "审批人的用户编号不能为空") + private Long assigneeUserId; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/send/SmsSendSingleToUserReqDTO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/send/SmsSendSingleToUserReqDTO.java new file mode 100644 index 00000000..2aff7c41 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/send/SmsSendSingleToUserReqDTO.java @@ -0,0 +1,34 @@ +package com.ruoyi.flowable.domain.dto.send; + +import lombok.Data; + +import javax.validation.constraints.NotEmpty; +import java.util.Map; + +/** + * 短信发送给 Admin 或者 Member 用户 + * + * hasPermi + */ +@Data +public class SmsSendSingleToUserReqDTO { + + /** + * 用户编号 + */ + private Long userId; + /** + * 手机号 + */ + private String mobile; + /** + * 短信模板编号 + */ + @NotEmpty(message = "短信模板编号不能为空") + private String templateCode; + /** + * 短信模板参数 + */ + private Map templateParams; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/task/BpmProcessInstanceCreateReqDTO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/task/BpmProcessInstanceCreateReqDTO.java new file mode 100644 index 00000000..25bdc5e2 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/task/BpmProcessInstanceCreateReqDTO.java @@ -0,0 +1,35 @@ +package com.ruoyi.flowable.domain.dto.task; + +import lombok.Data; +import lombok.experimental.Accessors; + +import javax.validation.constraints.NotEmpty; +import java.util.Map; + +/** + * 流程实例的创建 Request DTO + * + * hasPermi + */ +@Data +@Accessors(chain = true) +public class BpmProcessInstanceCreateReqDTO { + + /** + * 流程定义的标识 + */ + @NotEmpty(message = "流程定义的标识不能为空") + private String processDefinitionKey; + /** + * 变量实例 + */ + private Map variables; + + /** + * 业务的唯一标识 + * + * 例如说,请假申请的编号。通过它,可以查询到对应的实例 + */ + @NotEmpty(message = "业务的唯一标识") + private String businessKey; +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/user/AdminUserRespDTO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/user/AdminUserRespDTO.java new file mode 100644 index 00000000..dba9811f --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/user/AdminUserRespDTO.java @@ -0,0 +1,44 @@ +package com.ruoyi.flowable.domain.dto.user; + +import com.ruoyi.common.enums.CommonStatusEnum; +import lombok.Data; + +import java.util.Set; + +/** + * Admin 用户 Response DTO + * + * hasPermi + */ +@Data +public class AdminUserRespDTO { + + /** + * 用户ID + */ + private Long id; + /** + * 用户昵称 + */ + private String nickname; + /** + * 帐号状态 + * + * 枚举 {@link CommonStatusEnum} + */ + private Integer status; + + /** + * 部门ID + */ + private Long deptId; + /** + * 岗位编号数组 + */ + private Set postIds; + /** + * 手机号码 + */ + private String mobile; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/user/DeptRespDTO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/user/DeptRespDTO.java new file mode 100644 index 00000000..699213a5 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/dto/user/DeptRespDTO.java @@ -0,0 +1,37 @@ +package com.ruoyi.flowable.domain.dto.user; + +import com.ruoyi.common.enums.CommonStatusEnum; +import lombok.Data; + +/** + * 部门 Response DTO + * + * hasPermi + */ +@Data +public class DeptRespDTO { + + /** + * 部门编号 + */ + private Long id; + /** + * 部门名称 + */ + private String name; + /** + * 父部门编号 + */ + private Long parentId; + /** + * 负责人的用户编号 + */ + private Long leaderUserId; + /** + * 部门状态 + * + * 枚举 {@link CommonStatusEnum} + */ + private Integer status; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmFormDO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmFormDO.java new file mode 100644 index 00000000..835deec6 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmFormDO.java @@ -0,0 +1,58 @@ +package com.ruoyi.flowable.domain.entity.definition; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableField; +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import com.baomidou.mybatisplus.extension.handlers.JacksonTypeHandler; +import com.ruoyi.common.mybatis.dataobject.BaseDO; +import lombok.*; + +import java.util.List; + +/** + * 工作流的表单定义 + * 用于工作流的申请表单,需要动态配置的场景 + * + * hasPermi + */ +@TableName(value = "bpm_form", autoResultMap = true) +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +@Builder +@NoArgsConstructor +@AllArgsConstructor +public class BpmFormDO extends BaseDO { + + /** + * 编号 + */ + @TableId(type = IdType.AUTO) + private Long id; + /** + * 表单名 + */ + private String name; + /** + * 状态 + */ + private Integer status; + /** + * 表单的配置 + */ + private String conf; + /** + * 表单项的数组 + * + * 目前直接将 https://github.com/JakHuang/form-generator 生成的 JSON 串,直接保存 + * 定义:https://github.com/JakHuang/form-generator/issues/46 + */ + @TableField(typeHandler = JacksonTypeHandler.class) + private List fields; + /** + * 备注 + */ + private String remark; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmProcessDefinitionExtDO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmProcessDefinitionExtDO.java new file mode 100644 index 00000000..4d5f9df3 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmProcessDefinitionExtDO.java @@ -0,0 +1,91 @@ +package com.ruoyi.flowable.domain.entity.definition; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableField; +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import com.baomidou.mybatisplus.extension.handlers.JacksonTypeHandler; +import com.ruoyi.common.mybatis.dataobject.BaseDO; +import com.ruoyi.flowable.core.enums.definition.BpmModelFormTypeEnum; +import lombok.*; + +import java.util.List; + +/** + * Bpm 流程定义的拓展表 + * 主要解决 Activiti {@link ProcessDefinition} 不支持拓展字段,所以新建拓展表 + * + * hasPermi + */ +@TableName(value = "bpm_process_definition_ext", autoResultMap = true) +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +@Builder +@NoArgsConstructor +@AllArgsConstructor +public class BpmProcessDefinitionExtDO extends BaseDO { + + /** + * 编号 + */ + @TableId(type = IdType.AUTO) + private Long id; + /** + * 流程定义的编号 + * + * 关联 ProcessDefinition 的 id 属性 + */ + private String processDefinitionId; + /** + * 流程模型的编号 + * + * 关联 Model 的 id 属性 + */ + private String modelId; + /** + * 描述 + */ + private String description; + + /** + * 表单类型 + * + * 关联 {@link BpmModelFormTypeEnum} + */ + private Integer formType; + /** + * 动态表单编号 + * 在表单类型为 {@link BpmModelFormTypeEnum#NORMAL} 时 + * + * 关联 {@link BpmFormDO#getId()} + */ + private Long formId; + /** + * 表单的配置 + * 在表单类型为 {@link BpmModelFormTypeEnum#NORMAL} 时 + * + * 冗余 {@link BpmFormDO#getConf()} + */ + private String formConf; + /** + * 表单项的数组 + * 在表单类型为 {@link BpmModelFormTypeEnum#NORMAL} 时 + * + * 冗余 {@link BpmFormDO#getFields()} ()} + */ + @TableField(typeHandler = JacksonTypeHandler.class) + private List formFields; + /** + * 自定义表单的提交路径,使用 Vue 的路由地址 + * 在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时 + */ + private String formCustomCreatePath; + /** + * 自定义表单的查看路径,使用 Vue 的路由地址 + * 在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时 + */ + private String formCustomViewPath; + + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmTaskAssignRuleDO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmTaskAssignRuleDO.java new file mode 100644 index 00000000..cb3c2abf --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmTaskAssignRuleDO.java @@ -0,0 +1,84 @@ +package com.ruoyi.flowable.domain.entity.definition; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableField; +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import com.ruoyi.common.handler.JsonLongSetTypeHandler; +import com.ruoyi.common.mybatis.dataobject.BaseDO; +import com.ruoyi.flowable.core.enums.definition.BpmTaskAssignRuleTypeEnum; +import com.ruoyi.flowable.core.enums.definition.BpmTaskRuleScriptEnum; +import lombok.*; + +import java.util.Set; + +/** + * Bpm 任务分配的规则表,用于自定义配置每个任务的负责人、候选人的分配规则。 + * 也就是说,废弃 BPMN 原本的 UserTask 设置的 assignee、candidateUsers 等配置,而是通过使用该规则进行计算对应的负责人。 + * + * 1. 默认情况下,{@link #processDefinitionId} 为 {@link #PROCESS_DEFINITION_ID_NULL} 值,表示贵改则与流程模型关联 + * 2. 在流程模型部署后,会将他的所有规则记录,复制出一份新部署出来的流程定义,通过设置 {@link #processDefinitionId} 为新的流程定义的编号进行关联 + * + * hasPermi + */ +@TableName(value = "bpm_task_assign_rule", autoResultMap = true) +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +@Builder +@NoArgsConstructor +@AllArgsConstructor +public class BpmTaskAssignRuleDO extends BaseDO { + + /** + * {@link #processDefinitionId} 空串,用于标识属于流程模型,而不属于流程定义 + */ + public static final String PROCESS_DEFINITION_ID_NULL = ""; + + /** + * 编号 + */ +@TableId(type = IdType.AUTO) + private Long id; + + /** + * 流程模型编号 + * + * 关联 Model 的 id 属性 + */ + private String modelId; + /** + * 流程定义编号 + * + * 关联 ProcessDefinition 的 id 属性 + */ + private String processDefinitionId; + /** + * 流程任务的定义 Key + * + * 关联 Task 的 taskDefinitionKey 属性 + */ + private String taskDefinitionKey; + + /** + * 规则类型 + * + * 枚举 {@link BpmTaskAssignRuleTypeEnum} + */ + @TableField("`type`") + private Integer type; + /** + * 规则值数组,一般关联指定表的编号 + * 根据 type 不同,对应的值是不同的: + * + * 1. {@link BpmTaskAssignRuleTypeEnum#ROLE} 时:角色编号 + * 2. {@link BpmTaskAssignRuleTypeEnum#DEPT_MEMBER} 时:部门编号 + * 3. {@link BpmTaskAssignRuleTypeEnum#DEPT_LEADER} 时:部门编号 + * 4. {@link BpmTaskAssignRuleTypeEnum#USER} 时:用户编号 + * 5. {@link BpmTaskAssignRuleTypeEnum#USER_GROUP} 时:用户组编号 + * 6. {@link BpmTaskAssignRuleTypeEnum#SCRIPT} 时:脚本编号,目前通过 {@link BpmTaskRuleScriptEnum#getId()} 标识 + */ + @TableField(typeHandler = JsonLongSetTypeHandler.class) + private Set options; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmTaskMessageRuleDO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmTaskMessageRuleDO.java new file mode 100644 index 00000000..e52b9d31 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmTaskMessageRuleDO.java @@ -0,0 +1,5 @@ +package com.ruoyi.flowable.domain.entity.definition; + +// TODO 芋艿:先埋个坑。任务消息的配置规则。说白了,就是不同的 +public class BpmTaskMessageRuleDO { +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmUserGroupDO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmUserGroupDO.java new file mode 100644 index 00000000..1c1b81ba --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/definition/BpmUserGroupDO.java @@ -0,0 +1,52 @@ +package com.ruoyi.flowable.domain.entity.definition; + +import com.ruoyi.common.handler.JsonLongSetTypeHandler; +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableField; +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import com.ruoyi.common.mybatis.dataobject.BaseDO; +import lombok.*; + +import java.util.Set; + +/** + * Bpm 用户组 + * + * hasPermi + */ +@TableName(value = "bpm_user_group", autoResultMap = true) +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +@Builder +@NoArgsConstructor +@AllArgsConstructor +public class BpmUserGroupDO extends BaseDO { + + /** + * 编号,自增 + */ + @TableId(type = IdType.AUTO) + private Long id; + /** + * 组名 + */ + private String name; + /** + * 描述 + */ + private String description; + /** + * 状态 + * + * 枚举 {@link CommonStatusEnum} + */ + private Integer status; + /** + * 成员用户编号数组 + */ + @TableField(typeHandler = JsonLongSetTypeHandler.class) + private Set memberUserIds; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/oa/BpmOALeaveDO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/oa/BpmOALeaveDO.java new file mode 100644 index 00000000..fa671334 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/oa/BpmOALeaveDO.java @@ -0,0 +1,87 @@ +package com.ruoyi.flowable.domain.entity.oa; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableField; +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import com.ruoyi.common.mybatis.dataobject.BaseDO; +import com.ruoyi.flowable.core.enums.task.BpmProcessInstanceResultEnum; +import lombok.*; +import lombok.experimental.Accessors; + +import java.util.Date; + +/** + * OA 请假申请 DO + * + * {@link #day} 请假天数,目前先简单做。一般是分成请假上午和下午,可以是 1 整天,可以是 0.5 半天 + * + * @author jason + * hasPermi + */ +@TableName("bpm_oa_leave") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +@Builder +@NoArgsConstructor +@AllArgsConstructor +@Accessors(chain = true) +public class BpmOALeaveDO extends BaseDO { + + /** + * 请假表单主键 + */ + @TableId(type = IdType.AUTO) + private Long id; + /** + * 申请人的用户编号 + * + * 关联 AdminUserDO 的 id 属性 + */ + private Long userId; + /** + * 请假类型 + */ + @TableField("`type`") + private String type; + /** + * 原因 + */ + private String reason; + /** + * 开始时间 + */ + private Date startTime; + /** + * 结束时间 + */ + private Date endTime; + /** + * 请假天数 + */ + private Long day; + /** + * 请假的结果 + * + * 枚举 {@link BpmProcessInstanceResultEnum} + * 考虑到简单,所以直接复用了 BpmProcessInstanceResultEnum 枚举,也可以自己定义一个枚举哈 + */ + private Integer result; + + /** + * 对应的流程编号 + * + * 关联 ProcessInstance 的 id 属性 + */ + private String processInstanceId; + /** + * 租户ID + */ + private String tenantId; + + @TableField(exist = false) + private String nickName; + @TableField(exist = false) + private Long deptId; +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/task/BpmActivityDO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/task/BpmActivityDO.java new file mode 100644 index 00000000..2a87b29d --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/task/BpmActivityDO.java @@ -0,0 +1,104 @@ +package com.ruoyi.flowable.domain.entity.task; + +import com.fasterxml.jackson.annotation.JsonFormat; +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; +import org.springframework.format.annotation.DateTimeFormat; + +import java.util.Date; + +/** + * 任务流程关联表 + * + * @author kemengkai + * @create 2022-05-09 10:33 + */ +@Data +@AllArgsConstructor +@NoArgsConstructor +public class BpmActivityDO { + + /** + * 任务流程关联id + */ + private String id; + + /** + * 审批结果 + */ + private Integer rev; + + /** + * 任务流程部署id + */ + private String procDefId; + + /** + * 任务流程id + */ + private String processInstanceId; + + /** + * 任务执行id + */ + private String executionId; + + /** + * 任务key + */ + private String activityId; + + /** + * 任务id + */ + private String taskId; + + /** + * 调用流程id + */ + private String callProcInstId; + + /** + * 任务名称 + */ + private String activityName; + + /** + * 任务类型 + */ + private String activityType; + + /** + * 任务审批人id + */ + private String assignee; + + /** + * 任务开始时间 + */ + @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") + @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone = "GMT+8") + private Date startTime; + + /** + * 任务结束时间 + */ + @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") + @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone = "GMT+8") + private Date endTime; + + private Integer transactionOrder; + + private Long duration; + + /** + * 删除结果 + */ + private String deleteReason; + + /** + * 租户id + */ + private String tenantId; +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/task/BpmProcessInstanceExtDO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/task/BpmProcessInstanceExtDO.java new file mode 100644 index 00000000..3e1fc67e --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/task/BpmProcessInstanceExtDO.java @@ -0,0 +1,93 @@ +package com.ruoyi.flowable.domain.entity.task; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableField; +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import com.baomidou.mybatisplus.extension.handlers.JacksonTypeHandler; +import com.ruoyi.common.mybatis.dataobject.BaseDO; +import com.ruoyi.flowable.core.enums.task.BpmProcessInstanceResultEnum; +import com.ruoyi.flowable.core.enums.task.BpmProcessInstanceStatusEnum; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; +import lombok.experimental.Accessors; + +import java.util.Date; +import java.util.Map; + +/** + * Bpm 流程实例的拓展表 + * 主要解决 Activiti ProcessInstance 和 HistoricProcessInstance 不支持拓展字段,所以新建拓展表 + * + * hasPermi + */ +@TableName(value = "bpm_process_instance_ext", autoResultMap = true) +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +@Accessors(chain = true) +public class BpmProcessInstanceExtDO extends BaseDO { + + /** + * 编号,自增 + */ + @TableId(type = IdType.AUTO) + private Long id; + /** + * 发起流程的用户编号 + * + * 冗余 HistoricProcessInstance 的 startUserId 属性 + */ + private Long startUserId; + /** + * 流程实例的名字 + * + * 冗余 ProcessInstance 的 name 属性,用于筛选 + */ + private String name; + /** + * 流程实例的编号 + * + * 关联 ProcessInstance 的 id 属性 + */ + private String processInstanceId; + /** + * 流程定义的编号 + * + * 关联 ProcessDefinition 的 id 属性 + */ + private String processDefinitionId; + /** + * 流程分类 + * + * 冗余 ProcessDefinition 的 category 属性 + * 数据字典 bpm_model_category + */ + private String category; + /** + * 流程实例的状态 + * + * 枚举 {@link BpmProcessInstanceStatusEnum} + */ + private Integer status; + /** + * 流程实例的结果 + * + * 枚举 {@link BpmProcessInstanceResultEnum} + */ + private Integer result; + /** + * 结束时间 + * + * 冗余 HistoricProcessInstance 的 endTime 属性 + */ + private Date endTime; + + /** + * 提交的表单值 + */ + @TableField(typeHandler = JacksonTypeHandler.class) + private Map formVariables; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/task/BpmTaskExtDO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/task/BpmTaskExtDO.java new file mode 100644 index 00000000..0517300f --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/entity/task/BpmTaskExtDO.java @@ -0,0 +1,88 @@ +package com.ruoyi.flowable.domain.entity.task; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import com.ruoyi.common.mybatis.dataobject.BaseDO; +import com.ruoyi.flowable.core.enums.task.BpmProcessInstanceResultEnum; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; +import lombok.experimental.Accessors; + +import java.util.Date; + +/** + * Bpm 流程任务的拓展表 + * 主要解决 Flowable Task 和 HistoricTaskInstance 不支持拓展字段,所以新建拓展表 + * + * hasPermi + */ +@TableName(value = "bpm_task_ext", autoResultMap = true) +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +@Accessors(chain = true) +public class BpmTaskExtDO extends BaseDO { + + /** + * 编号,自增 + */ + @TableId(type = IdType.AUTO) + private Long id; + + /** + * 任务的审批人 + * + * 冗余 Task 的 assignee 属性 + */ + private Long assigneeUserId; + /** + * 任务的名字 + * + * 冗余 Task 的 name 属性,为了筛选 + */ + private String name; + /** + * 任务的编号 + * + * 关联 Task 的 id 属性 + */ + private String taskId; +// /** +// * 任务的标识 +// * +// * 关联 {@link Task#getTaskDefinitionKey()} +// */ +// private String definitionKey; + /** + * 任务的结果 + * + * 枚举 {@link BpmProcessInstanceResultEnum} + */ + private Integer result; + /** + * 审批建议 + */ + private String reason; + /** + * 任务的结束时间 + * + * 冗余 HistoricTaskInstance 的 endTime 属性 + */ + private Date endTime; + + /** + * 流程实例的编号 + * + * 关联 ProcessInstance 的 id 属性 + */ + private String processInstanceId; + /** + * 流程定义的编号 + * + * 关联 ProcessDefinition 的 id 属性 + */ + private String processDefinitionId; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/activity/BpmActivityRespVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/activity/BpmActivityRespVO.java new file mode 100644 index 00000000..2bed8387 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/activity/BpmActivityRespVO.java @@ -0,0 +1,26 @@ +package com.ruoyi.flowable.domain.vo.activity; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.util.Date; + +@ApiModel("管理后台 - 流程活动的 Response VO") +@Data +public class BpmActivityRespVO { + + @ApiModelProperty(value = "流程活动的标识", required = true, example = "1024") + private String key; + @ApiModelProperty(value = "流程活动的类型", required = true, example = "StartEvent") + private String type; + + @ApiModelProperty(value = "流程活动的开始时间", required = true) + private Date startTime; + @ApiModelProperty(value = "流程活动的结束时间", required = true) + private Date endTime; + + @ApiModelProperty(value = "关联的流程任务的编号", example = "2048", notes = "关联的流程任务,只有 UserTask 等类型才有") + private String taskId; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormBaseVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormBaseVO.java new file mode 100644 index 00000000..1f50e8f8 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormBaseVO.java @@ -0,0 +1,26 @@ +package com.ruoyi.flowable.domain.vo.form; + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotNull; + +/** +* 动态表单 Base VO,提供给添加、修改、详细的子 VO 使用 +* 如果子 VO 存在差异的字段,请不要添加到这里,影响 Swagger 文档生成 +*/ +@Data +public class BpmFormBaseVO { + + @ApiModelProperty(value = "表单名称", required = true, example = "芋道") + @NotNull(message = "表单名称不能为空") + private String name; + + @ApiModelProperty(value = "表单状态", required = true, notes = "参见 CommonStatusEnum 枚举", example = "1") + @NotNull(message = "表单状态不能为空") + private Integer status; + + @ApiModelProperty(value = "备注", example = "我是备注") + private String remark; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormCreateReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormCreateReqVO.java new file mode 100644 index 00000000..71207c23 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormCreateReqVO.java @@ -0,0 +1,26 @@ +package com.ruoyi.flowable.domain.vo.form; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.validation.constraints.NotNull; +import java.util.List; + +@ApiModel("管理后台 - 动态表单创建 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmFormCreateReqVO extends BpmFormBaseVO { + + @ApiModelProperty(value = "表单的配置", required = true, notes = "JSON 字符串") + @NotNull(message = "表单的配置不能为空") + private String conf; + + @ApiModelProperty(value = "表单项的数组", required = true, notes = "JSON 字符串的数组") + @NotNull(message = "表单项的数组不能为空") + private List fields; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormPageReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormPageReqVO.java new file mode 100644 index 00000000..eccfba64 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormPageReqVO.java @@ -0,0 +1,19 @@ +package com.ruoyi.flowable.domain.vo.form; + +import com.ruoyi.common.mybatis.pojo.PageParam; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +@ApiModel("管理后台 - 动态表单分页 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmFormPageReqVO extends PageParam { + + @ApiModelProperty(value = "表单名称", example = "芋道") + private String name; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormRespVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormRespVO.java new file mode 100644 index 00000000..e9b7b8e6 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormRespVO.java @@ -0,0 +1,36 @@ +package com.ruoyi.flowable.domain.vo.form; + +import com.fasterxml.jackson.databind.annotation.JsonSerialize; +import com.fasterxml.jackson.databind.ser.std.ToStringSerializer; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.validation.constraints.NotNull; +import java.util.Date; +import java.util.List; + +@ApiModel("管理后台 - 动态表单 Response VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmFormRespVO extends BpmFormBaseVO { + + @ApiModelProperty(value = "表单编号", required = true, example = "1024") + @JsonSerialize(using = ToStringSerializer.class) + private Long id; + + @ApiModelProperty(value = "表单的配置", required = true, notes = "JSON 字符串") + @NotNull(message = "表单的配置不能为空") + private String conf; + + @ApiModelProperty(value = "表单项的数组", required = true, notes = "JSON 字符串的数组") + @NotNull(message = "表单项的数组不能为空") + private List fields; + + @ApiModelProperty(value = "创建时间", required = true) + private Date createTime; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormSimpleRespVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormSimpleRespVO.java new file mode 100644 index 00000000..d8adb1cc --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormSimpleRespVO.java @@ -0,0 +1,17 @@ +package com.ruoyi.flowable.domain.vo.form; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +@ApiModel("管理后台 - 流程表单精简 Response VO") +@Data +public class BpmFormSimpleRespVO { + + @ApiModelProperty(value = "表单编号", required = true, example = "1024") + private Long id; + + @ApiModelProperty(value = "表单名称", required = true, example = "芋道") + private String name; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormUpdateReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormUpdateReqVO.java new file mode 100644 index 00000000..1a6f6389 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/form/BpmFormUpdateReqVO.java @@ -0,0 +1,30 @@ +package com.ruoyi.flowable.domain.vo.form; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.validation.constraints.NotNull; +import java.util.List; + +@ApiModel("管理后台 - 动态表单更新 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmFormUpdateReqVO extends BpmFormBaseVO { + + @ApiModelProperty(value = "表单编号", required = true, example = "1024") + @NotNull(message = "表单编号不能为空") + private Long id; + + @ApiModelProperty(value = "表单的配置", required = true, notes = "JSON 字符串") + @NotNull(message = "表单的配置不能为空") + private String conf; + + @ApiModelProperty(value = "表单项的数组", required = true, notes = "JSON 字符串的数组") + @NotNull(message = "表单项的数组不能为空") + private List fields; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupBaseVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupBaseVO.java new file mode 100644 index 00000000..07c75a13 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupBaseVO.java @@ -0,0 +1,35 @@ +package com.ruoyi.flowable.domain.vo.group; + +import javax.validation.constraints.NotNull; +import java.util.Set; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + + +/** +* 用户组 Base VO,提供给添加、修改、详细的子 VO 使用 +* 如果子 VO 存在差异的字段,请不要添加到这里,影响 Swagger 文档生成 +*/ +@Data +public class BpmUserGroupBaseVO { + + @ApiModelProperty(value = "组名", required = true, example = "芋道") + @NotNull(message = "组名不能为空") + private String name; + + @ApiModelProperty(value = "描述", required = true, example = "芋道源码") + @NotNull(message = "描述不能为空") + private String description; + + @ApiModelProperty(value = "成员编号数组", required = true, example = "1,2,3") + @NotNull(message = "成员编号数组不能为空") + private Set memberUserIds; + + @ApiModelProperty(value = "状态", required = true, example = "1") + @NotNull(message = "状态不能为空") + private Integer status; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupCreateReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupCreateReqVO.java new file mode 100644 index 00000000..3f3a6dc4 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupCreateReqVO.java @@ -0,0 +1,15 @@ +package com.ruoyi.flowable.domain.vo.group; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +@ApiModel("管理后台 - 用户组创建 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmUserGroupCreateReqVO extends BpmUserGroupBaseVO { + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupPageReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupPageReqVO.java new file mode 100644 index 00000000..037f7f32 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupPageReqVO.java @@ -0,0 +1,29 @@ +package com.ruoyi.flowable.domain.vo.group; + +import com.ruoyi.common.mybatis.pojo.PageParam; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; +import org.springframework.format.annotation.DateTimeFormat; + +import java.util.Date; + +@ApiModel("管理后台 - 用户组分页 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmUserGroupPageReqVO extends PageParam { + + @ApiModelProperty(value = "组名", example = "芋道") + private String name; + + @ApiModelProperty(value = "状态", example = "1") + private Integer status; + + @DateTimeFormat(pattern ="yyyy-MM-dd HH:mm:ss") + @ApiModelProperty(value = "创建时间") + private Date[] createTime; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupRespVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupRespVO.java new file mode 100644 index 00000000..fdcab9a2 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupRespVO.java @@ -0,0 +1,22 @@ +package com.ruoyi.flowable.domain.vo.group; + +import java.util.Date; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +@ApiModel("管理后台 - 用户组 Response VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmUserGroupRespVO extends BpmUserGroupBaseVO { + + @ApiModelProperty(value = "编号", required = true, example = "1024") + private Long id; + + @ApiModelProperty(value = "创建时间", required = true) + private Date createTime; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupSimpleRespVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupSimpleRespVO.java new file mode 100644 index 00000000..89a7f104 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupSimpleRespVO.java @@ -0,0 +1,21 @@ +package com.ruoyi.flowable.domain.vo.group; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@ApiModel("管理后台 - 用户组精简信息 Response VO") +@Data +@NoArgsConstructor +@AllArgsConstructor +public class BpmUserGroupSimpleRespVO { + + @ApiModelProperty(value = "用户组编号", required = true, example = "1024") + private Long id; + + @ApiModelProperty(value = "用户组名字", required = true, example = "芋道") + private String name; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupUpdateReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupUpdateReqVO.java new file mode 100644 index 00000000..2b0e947b --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/group/BpmUserGroupUpdateReqVO.java @@ -0,0 +1,20 @@ +package com.ruoyi.flowable.domain.vo.group; + +import javax.validation.constraints.NotNull; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +@ApiModel("管理后台 - 用户组更新 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmUserGroupUpdateReqVO extends BpmUserGroupBaseVO { + + @ApiModelProperty(value = "编号", required = true, example = "1024") + @NotNull(message = "编号不能为空") + private Long id; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceCancelReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceCancelReqVO.java new file mode 100644 index 00000000..e7147dfe --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceCancelReqVO.java @@ -0,0 +1,21 @@ +package com.ruoyi.flowable.domain.vo.instance; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotEmpty; + +@ApiModel("管理后台 - 流程实例的取消 Request VO") +@Data +public class BpmProcessInstanceCancelReqVO { + + @ApiModelProperty(value = "流程实例的编号", required = true, example = "1024") + @NotEmpty(message = "流程实例的编号不能为空") + private String id; + + @ApiModelProperty(value = "取消原因", required = true, example = "不请假了!") + @NotEmpty(message = "取消原因不能为空") + private String reason; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceCreateReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceCreateReqVO.java new file mode 100644 index 00000000..c6da8e2a --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceCreateReqVO.java @@ -0,0 +1,21 @@ +package com.ruoyi.flowable.domain.vo.instance; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotEmpty; +import java.util.Map; + +@ApiModel("管理后台 - 流程实例的创建 Request VO") +@Data +public class BpmProcessInstanceCreateReqVO { + + @ApiModelProperty(value = "流程定义的编号", required = true, example = "1024") + @NotEmpty(message = "流程定义编号不能为空") + private String processDefinitionId; + + @ApiModelProperty(value = "变量实例") + private Map variables; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceMyPageReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceMyPageReqVO.java new file mode 100644 index 00000000..e1e393df --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceMyPageReqVO.java @@ -0,0 +1,39 @@ +package com.ruoyi.flowable.domain.vo.instance; + +import com.ruoyi.common.mybatis.pojo.PageParam; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; +import org.springframework.format.annotation.DateTimeFormat; + +import java.util.Date; + + +@ApiModel("管理后台 - 流程实例的分页 Item Response VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmProcessInstanceMyPageReqVO extends PageParam { + + @ApiModelProperty(value = "流程名称", example = "芋道") + private String name; + + @ApiModelProperty(value = "流程定义的编号", example = "2048") + private String processDefinitionId; + + @ApiModelProperty(value = "流程实例的状态", notes = "参见 bpm_process_instance_status", example = "1") + private Integer status; + + @ApiModelProperty(value = "流程实例的结果", notes = "参见 bpm_process_instance_result", example = "2") + private Integer result; + + @ApiModelProperty(value = "流程分类", notes = "参见 bpm_model_category 数据字典", example = "1") + private String category; + + @ApiModelProperty(value = "创建时间") + @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") + private Date[] createTime; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstancePageItemRespVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstancePageItemRespVO.java new file mode 100644 index 00000000..ff40fa13 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstancePageItemRespVO.java @@ -0,0 +1,55 @@ +package com.ruoyi.flowable.domain.vo.instance; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.util.Date; +import java.util.List; + +@ApiModel("管理后台 - 流程实例的分页 Item Response VO") +@Data +public class BpmProcessInstancePageItemRespVO { + + @ApiModelProperty(value = "流程实例的编号", required = true, example = "1024") + private String id; + + @ApiModelProperty(value = "流程名称", required = true, example = "芋道") + private String name; + + @ApiModelProperty(value = "流程定义的编号", required = true, example = "2048") + private String processDefinitionId; + + @ApiModelProperty(value = "流程分类", required = true, notes = "参见 bpm_model_category 数据字典", example = "1") + private String category; + + @ApiModelProperty(value = "流程实例的状态", required = true, notes = "参见 bpm_process_instance_status", example = "1") + private Integer status; + + @ApiModelProperty(value = "流程实例的结果", required = true, notes = "参见 bpm_process_instance_result", example = "2") + private Integer result; + + @ApiModelProperty(value = "提交时间", required = true) + private Date createTime; + + @ApiModelProperty(value = "结束时间", required = true) + private Date endTime; + + /** + * 当前任务 + */ + private List tasks; + + @ApiModel("流程任务") + @Data + public static class Task { + + @ApiModelProperty(value = "流程任务的编号", required = true, example = "1024") + private String id; + + @ApiModelProperty(value = "任务名称", required = true, example = "芋道") + private String name; + + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceRespVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceRespVO.java new file mode 100644 index 00000000..d122d777 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceRespVO.java @@ -0,0 +1,97 @@ +package com.ruoyi.flowable.domain.vo.instance; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.util.Date; +import java.util.List; +import java.util.Map; + +@ApiModel("管理后台 - 流程实例的 Response VO") +@Data +public class BpmProcessInstanceRespVO { + + @ApiModelProperty(value = "流程实例的编号", required = true, example = "1024") + private String id; + + @ApiModelProperty(value = "流程名称", required = true, example = "芋道") + private String name; + + @ApiModelProperty(value = "流程分类", required = true, notes = "参见 bpm_model_category 数据字典", example = "1") + private String category; + + @ApiModelProperty(value = "流程实例的状态", required = true, notes = "参见 bpm_process_instance_status", example = "1") + private Integer status; + + @ApiModelProperty(value = "流程实例的结果", required = true, notes = "参见 bpm_process_instance_result", example = "2") + private Integer result; + + @ApiModelProperty(value = "提交时间", required = true) + private Date createTime; + + @ApiModelProperty(value = "结束时间", required = true) + private Date endTime; + + @ApiModelProperty(value = "提交的表单值", required = true) + private Map formVariables; + + @ApiModelProperty(value = "业务的唯一标识", example = "1", notes = "例如说,请假申请的编号") + private String businessKey; + + /** + * 发起流程的用户 + */ + private User startUser; + + /** + * 流程定义 + */ + private ProcessDefinition processDefinition; + + @ApiModel("用户信息") + @Data + public static class User { + + @ApiModelProperty(value = "用户编号", required = true, example = "1") + private Long id; + @ApiModelProperty(value = "用户昵称", required = true, example = "芋艿") + private String nickname; + + @ApiModelProperty(value = "部门编号", required = true, example = "1") + private Long deptId; + @ApiModelProperty(value = "部门名称", required = true, example = "研发部") + private String deptName; + + } + + @ApiModel("流程定义信息") + @Data + public static class ProcessDefinition { + + @ApiModelProperty(value = "编号", required = true, example = "1024") + private String id; + + @ApiModelProperty(value = "表单类型", notes = "参见 bpm_model_form_type 数据字典", example = "1") + private Integer formType; + @ApiModelProperty(value = "表单编号", example = "1024", notes = "在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时,必须非空") + private Long formId; + @ApiModelProperty(value = "表单的配置", required = true, + notes = "JSON 字符串。在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时,必须非空") + private String formConf; + @ApiModelProperty(value = "表单项的数组", required = true, + notes = "JSON 字符串的数组。在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时,必须非空") + private List formFields; + @ApiModelProperty(value = "自定义表单的提交路径,使用 Vue 的路由地址", example = "/bpm/oa/leave/create", + notes = "在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时,必须非空") + private String formCustomCreatePath; + @ApiModelProperty(value = "自定义表单的查看路径,使用 Vue 的路由地址", example = "/bpm/oa/leave/view", + notes = "在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时,必须非空") + private String formCustomViewPath; + + @ApiModelProperty(value = "BPMN XML", required = true) + private String bpmnXml; + + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModeImportReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModeImportReqVO.java new file mode 100644 index 00000000..4139f027 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModeImportReqVO.java @@ -0,0 +1,22 @@ +package com.ruoyi.flowable.domain.vo.model; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; +import org.springframework.web.multipart.MultipartFile; + +import javax.validation.constraints.NotNull; + +@ApiModel(value = "管理后台 - 流程模型的导入 Request VO", description = "相比流程模型的新建来说,只是多了一个 bpmnFile 文件") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmModeImportReqVO extends BpmModelCreateReqVO { + + @ApiModelProperty(value = "BPMN 文件", required = true) + @NotNull(message = "BPMN 文件不能为空") + private MultipartFile bpmnFile; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelBaseVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelBaseVO.java new file mode 100644 index 00000000..cf7ce48e --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelBaseVO.java @@ -0,0 +1,41 @@ +package com.ruoyi.flowable.domain.vo.model; + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotEmpty; + +/** +* 流程模型 Base VO,提供给添加、修改、详细的子 VO 使用 +* 如果子 VO 存在差异的字段,请不要添加到这里,影响 Swagger 文档生成 +*/ +@Data +public class BpmModelBaseVO { + + @ApiModelProperty(value = "流程标识", required = true, example = "process_yudao") + @NotEmpty(message = "流程标识不能为空") + private String key; + + @ApiModelProperty(value = "流程名称", required = true, example = "芋道") + @NotEmpty(message = "流程名称不能为空") + private String name; + + @ApiModelProperty(value = "流程描述", example = "我是描述") + private String description; + + @ApiModelProperty(value = "流程分类", notes = "参见 bpm_model_category 数据字典", example = "1") + @NotEmpty(message = "流程分类不能为空") + private String category; + + @ApiModelProperty(value = "表单类型", notes = "参见 bpm_model_form_type 数据字典", example = "1") + private Integer formType; + @ApiModelProperty(value = "表单编号", example = "1024", notes = "在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时,必须非空") + private Long formId; + @ApiModelProperty(value = "自定义表单的提交路径,使用 Vue 的路由地址", example = "/bpm/oa/leave/create", + notes = "在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时,必须非空") + private String formCustomCreatePath; + @ApiModelProperty(value = "自定义表单的查看路径,使用 Vue 的路由地址", example = "/bpm/oa/leave/view", + notes = "在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时,必须非空") + private String formCustomViewPath; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelCreateReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelCreateReqVO.java new file mode 100644 index 00000000..25d1de52 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelCreateReqVO.java @@ -0,0 +1,24 @@ +package com.ruoyi.flowable.domain.vo.model; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotEmpty; + +@ApiModel("管理后台 - 流程模型的创建 Request VO") +@Data +public class BpmModelCreateReqVO { + + @ApiModelProperty(value = "流程标识", required = true, example = "process_yudao") + @NotEmpty(message = "流程标识不能为空") + private String key; + + @ApiModelProperty(value = "流程名称", required = true, example = "芋道") + @NotEmpty(message = "流程名称不能为空") + private String name; + + @ApiModelProperty(value = "流程描述", example = "我是描述") + private String description; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelPageItemRespVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelPageItemRespVO.java new file mode 100644 index 00000000..accc6a28 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelPageItemRespVO.java @@ -0,0 +1,49 @@ +package com.ruoyi.flowable.domain.vo.model; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.util.Date; + +@ApiModel("管理后台 - 流程模型的分页的每一项 Response VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmModelPageItemRespVO extends BpmModelBaseVO { + + @ApiModelProperty(value = "编号", required = true, example = "1024") + private String id; + + @ApiModelProperty(value = "表单名字", example = "请假表单") + private String formName; + + @ApiModelProperty(value = "创建时间", required = true) + private Date createTime; + + /** + * 最新部署的流程定义 + */ + private ProcessDefinition processDefinition; + + @ApiModel("流程定义") + @Data + public static class ProcessDefinition { + + @ApiModelProperty(value = "编号", required = true, example = "1024") + private String id; + + @ApiModelProperty(value = "版本", required = true, example = "1") + private Integer version; + + @ApiModelProperty(value = "部署时间", required = true) + private Date deploymentTime; + + @ApiModelProperty(value = "中断状态", required = true, example = "1", notes = "参见 SuspensionState 枚举") + private Integer suspensionState; + + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelPageReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelPageReqVO.java new file mode 100644 index 00000000..6db07bcb --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelPageReqVO.java @@ -0,0 +1,26 @@ +package com.ruoyi.flowable.domain.vo.model; + +import com.ruoyi.common.mybatis.pojo.PageParam; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + + +@ApiModel("管理后台 - 流程模型分页 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmModelPageReqVO extends PageParam { + + @ApiModelProperty(value = "标识", example = "process1641042089407", notes = "精准匹配") + private String key; + + @ApiModelProperty(value = "名字", example = "芋道", notes = "模糊匹配") + private String name; + + @ApiModelProperty(value = "流程分类", notes = "参见 bpm_model_category 数据字典", example = "1") + private String category; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelRespVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelRespVO.java new file mode 100644 index 00000000..7a5d0ae6 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelRespVO.java @@ -0,0 +1,26 @@ +package com.ruoyi.flowable.domain.vo.model; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.util.Date; + +@ApiModel("管理后台 - 流程模型的创建 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmModelRespVO extends BpmModelBaseVO { + + @ApiModelProperty(value = "编号", required = true, example = "1024") + private String id; + + @ApiModelProperty(value = "BPMN XML", required = true) + private String bpmnXml; + + @ApiModelProperty(value = "创建时间", required = true) + private Date createTime; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelUpdateReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelUpdateReqVO.java new file mode 100644 index 00000000..c4b6fb2a --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelUpdateReqVO.java @@ -0,0 +1,40 @@ +package com.ruoyi.flowable.domain.vo.model; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotEmpty; + +@ApiModel("管理后台 - 流程模型的更新 Request VO") +@Data +public class BpmModelUpdateReqVO { + + @ApiModelProperty(value = "编号", required = true, example = "1024") + @NotEmpty(message = "编号不能为空") + private String id; + + @ApiModelProperty(value = "流程名称", example = "芋道") + private String name; + + @ApiModelProperty(value = "流程描述", example = "我是描述") + private String description; + + @ApiModelProperty(value = "流程分类", notes = "参见 bpm_model_category 数据字典", example = "1") + private String category; + + @ApiModelProperty(value = "BPMN XML", required = true) + private String bpmnXml; + + @ApiModelProperty(value = "表单类型", notes = "参见 bpm_model_form_type 数据字典", example = "1") + private Integer formType; + @ApiModelProperty(value = "表单编号", example = "1024", notes = "在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时,必须非空") + private Long formId; + @ApiModelProperty(value = "自定义表单的提交路径,使用 Vue 的路由地址", example = "/bpm/oa/leave/create", + notes = "在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时,必须非空") + private String formCustomCreatePath; + @ApiModelProperty(value = "自定义表单的查看路径,使用 Vue 的路由地址", example = "/bpm/oa/leave/view", + notes = "在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时,必须非空") + private String formCustomViewPath; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelUpdateStateReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelUpdateStateReqVO.java new file mode 100644 index 00000000..e3e77bf2 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/model/BpmModelUpdateStateReqVO.java @@ -0,0 +1,21 @@ +package com.ruoyi.flowable.domain.vo.model; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotNull; + +@ApiModel("管理后台 - 流程模型更新状态 Request VO") +@Data +public class BpmModelUpdateStateReqVO { + + @ApiModelProperty(value = "编号", required = true, example = "1024") + @NotNull(message = "编号不能为空") + private String id; + + @ApiModelProperty(value = "状态", required = true, example = "1", notes = "见 SuspensionState 枚举") + @NotNull(message = "状态不能为空") + private Integer state; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveBaseVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveBaseVO.java new file mode 100644 index 00000000..45eba61b --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveBaseVO.java @@ -0,0 +1,33 @@ +package com.ruoyi.flowable.domain.vo.oa; + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import org.springframework.format.annotation.DateTimeFormat; + +import javax.validation.constraints.NotNull; +import java.util.Date; + + +/** +* 请假申请 Base VO,提供给添加、修改、详细的子 VO 使用 +* 如果子 VO 存在差异的字段,请不要添加到这里,影响 Swagger 文档生成 +*/ +@Data +public class BpmOALeaveBaseVO { + + @ApiModelProperty(value = "请假的开始时间", required = true) + @NotNull(message = "开始时间不能为空") + @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") + private Date startTime; + @ApiModelProperty(value = "请假的结束时间", required = true) + @NotNull(message = "结束时间不能为空") + @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") + private Date endTime; + + @ApiModelProperty(value = "请假类型", required = true, example = "1", notes = "参见 bpm_oa_type 枚举") + private Integer type; + + @ApiModelProperty(value = "原因", required = true, example = "阅读芋道源码") + private String reason; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveCreateReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveCreateReqVO.java new file mode 100644 index 00000000..ea9bb572 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveCreateReqVO.java @@ -0,0 +1,21 @@ +package com.ruoyi.flowable.domain.vo.oa; + +import io.swagger.annotations.ApiModel; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.validation.constraints.AssertTrue; + +@ApiModel("管理后台 - 请假申请创建 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmOALeaveCreateReqVO extends BpmOALeaveBaseVO { + + @AssertTrue(message = "结束时间,需要在开始时间之后") + public boolean isEndTimeValid() { + return !getEndTime().before(getStartTime()); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/oa/BpmOALeavePageReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/oa/BpmOALeavePageReqVO.java new file mode 100644 index 00000000..98adaf67 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/oa/BpmOALeavePageReqVO.java @@ -0,0 +1,33 @@ +package com.ruoyi.flowable.domain.vo.oa; + +import com.ruoyi.common.mybatis.pojo.PageParam; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; +import org.springframework.format.annotation.DateTimeFormat; + +import java.util.Date; + + +@ApiModel("管理后台 - 请假申请分页 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmOALeavePageReqVO extends PageParam { + + @ApiModelProperty(value = "状态", example = "1", notes = "参见 bpm_process_instance_result 枚举") + private Integer result; + + @ApiModelProperty(value = "请假类型", example = "1", notes = "参见 bpm_oa_type") + private Integer type; + + @ApiModelProperty(value = "原因", example = "阅读芋道源码", notes = "模糊匹配") + private String reason; + + @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") + @ApiModelProperty(value = "申请时间") + private Date[] createTime; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveRespVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveRespVO.java new file mode 100644 index 00000000..c2d98bd3 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveRespVO.java @@ -0,0 +1,33 @@ +package com.ruoyi.flowable.domain.vo.oa; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; +import org.springframework.format.annotation.DateTimeFormat; + +import javax.validation.constraints.NotNull; +import java.util.Date; + +@ApiModel("管理后台 - 请假申请 Response VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmOALeaveRespVO extends BpmOALeaveBaseVO { + + @ApiModelProperty(value = "请假表单主键", required = true, example = "1024") + private Long id; + + @ApiModelProperty(value = "状态", required = true, example = "1", notes = "参见 bpm_process_instance_result 枚举") + private Integer result; + + @ApiModelProperty(value = "申请时间", required = true) + @NotNull(message = "申请时间不能为空") + @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") + private Date createTime; + + @ApiModelProperty(value = "流程id") + private String processInstanceId; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionListReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionListReqVO.java new file mode 100644 index 00000000..debd90cb --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionListReqVO.java @@ -0,0 +1,19 @@ +package com.ruoyi.flowable.domain.vo.process; + +import com.ruoyi.common.mybatis.pojo.PageParam; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +@ApiModel("管理后台 - 流程定义列表 Request VO") +@Data +@ToString(callSuper = true) +@EqualsAndHashCode(callSuper = true) +public class BpmProcessDefinitionListReqVO extends PageParam { + + @ApiModelProperty(value = "中断状态", example = "1", notes = "参见 SuspensionState 枚举") + private Integer suspensionState; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionPageItemRespVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionPageItemRespVO.java new file mode 100644 index 00000000..8e685872 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionPageItemRespVO.java @@ -0,0 +1,23 @@ +package com.ruoyi.flowable.domain.vo.process; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.util.Date; + +@ApiModel("管理后台 - 流程定义的分页的每一项 Response VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmProcessDefinitionPageItemRespVO extends BpmProcessDefinitionRespVO { + + @ApiModelProperty(value = "表单名字", example = "请假表单") + private String formName; + + @ApiModelProperty(value = "部署时间", required = true) + private Date deploymentTime; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionPageReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionPageReqVO.java new file mode 100644 index 00000000..47bab053 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionPageReqVO.java @@ -0,0 +1,19 @@ +package com.ruoyi.flowable.domain.vo.process; + +import com.ruoyi.common.mybatis.pojo.PageParam; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +@ApiModel("管理后台 - 流程定义分页 Request VO") +@Data +@ToString(callSuper = true) +@EqualsAndHashCode(callSuper = true) +public class BpmProcessDefinitionPageReqVO extends PageParam { + + @ApiModelProperty(value = "标识", example = "process1641042089407", notes = "精准匹配") + private String key; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionRespVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionRespVO.java new file mode 100644 index 00000000..544854e7 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionRespVO.java @@ -0,0 +1,51 @@ +package com.ruoyi.flowable.domain.vo.process; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotEmpty; +import java.util.List; + +@ApiModel("管理后台 - 流程定义 Response VO") +@Data +public class BpmProcessDefinitionRespVO { + + @ApiModelProperty(value = "编号", required = true, example = "1024") + private String id; + + @ApiModelProperty(value = "版本", required = true, example = "1") + private Integer version; + + @ApiModelProperty(value = "流程名称", required = true, example = "芋道") + @NotEmpty(message = "流程名称不能为空") + private String name; + + @ApiModelProperty(value = "流程描述", example = "我是描述") + private String description; + + @ApiModelProperty(value = "流程分类", notes = "参见 bpm_model_category 数据字典", example = "1") + @NotEmpty(message = "流程分类不能为空") + private String category; + + @ApiModelProperty(value = "表单类型", notes = "参见 bpm_model_form_type 数据字典", example = "1") + private Integer formType; + @ApiModelProperty(value = "表单编号", example = "1024", notes = "在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时,必须非空") + private Long formId; + @ApiModelProperty(value = "表单的配置", required = true, + notes = "JSON 字符串。在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时,必须非空") + private String formConf; + @ApiModelProperty(value = "表单项的数组", required = true, + notes = "JSON 字符串的数组。在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时,必须非空") + private List formFields; + @ApiModelProperty(value = "自定义表单的提交路径,使用 Vue 的路由地址", example = "/bpm/oa/leave/create", + notes = "在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时,必须非空") + private String formCustomCreatePath; + @ApiModelProperty(value = "自定义表单的查看路径,使用 Vue 的路由地址", example = "/bpm/oa/leave/view", + notes = "在表单类型为 {@link BpmModelFormTypeEnum#CUSTOM} 时,必须非空") + private String formCustomViewPath; + + @ApiModelProperty(value = "中断状态", required = true, example = "1", notes = "参见 SuspensionState 枚举") + private Integer suspensionState; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleBaseVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleBaseVO.java new file mode 100644 index 00000000..73c91824 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleBaseVO.java @@ -0,0 +1,24 @@ +package com.ruoyi.flowable.domain.vo.rule; + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotNull; +import java.util.Set; + +/** + * 流程任务分配规则 Base VO,提供给添加、修改、详细的子 VO 使用 + * 如果子 VO 存在差异的字段,请不要添加到这里,影响 Swagger 文档生成 + */ +@Data +public class BpmTaskAssignRuleBaseVO { + + @ApiModelProperty(value = "规则类型", required = true, example = "bpm_task_assign_rule_type") + @NotNull(message = "规则类型不能为空") + private Integer type; + + @ApiModelProperty(value = "规则值数组", required = true, example = "1,2,3") + @NotNull(message = "规则值数组不能为空") + private Set options; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleCreateReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleCreateReqVO.java new file mode 100644 index 00000000..45a16421 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleCreateReqVO.java @@ -0,0 +1,25 @@ +package com.ruoyi.flowable.domain.vo.rule; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.validation.constraints.NotEmpty; + +@ApiModel("管理后台 - 流程任务分配规则的创建 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmTaskAssignRuleCreateReqVO extends BpmTaskAssignRuleBaseVO { + + @ApiModelProperty(value = "流程模型的编号", required = true, example = "1024") + @NotEmpty(message = "流程模型的编号不能为空") + private String modelId; + + @ApiModelProperty(value = "流程任务定义的编号", required = true, example = "2048") + @NotEmpty(message = "流程任务定义的编号不能为空") + private String taskDefinitionKey; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleRespVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleRespVO.java new file mode 100644 index 00000000..dd43e126 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleRespVO.java @@ -0,0 +1,29 @@ +package com.ruoyi.flowable.domain.vo.rule; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +@ApiModel("管理后台 - 流程任务分配规则的 Response VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmTaskAssignRuleRespVO extends BpmTaskAssignRuleBaseVO { + + @ApiModelProperty(value = "任务分配规则的编号", required = true, example = "1024") + private Long id; + + @ApiModelProperty(value = "流程模型的编号", required = true, example = "2048") + private String modelId; + + @ApiModelProperty(value = "流程定义的编号", required = true, example = "4096") + private String processDefinitionId; + + @ApiModelProperty(value = "流程任务定义的编号", required = true, example = "2048") + private String taskDefinitionKey; + @ApiModelProperty(value = "流程任务定义的名字", required = true, example = "关注芋道") + private String taskDefinitionName; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleUpdateReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleUpdateReqVO.java new file mode 100644 index 00000000..1feae03f --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleUpdateReqVO.java @@ -0,0 +1,21 @@ +package com.ruoyi.flowable.domain.vo.rule; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.validation.constraints.NotNull; + +@ApiModel("管理后台 - 流程任务分配规则的更新 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmTaskAssignRuleUpdateReqVO extends BpmTaskAssignRuleBaseVO { + + @ApiModelProperty(value = "任务分配规则的编号", required = true, example = "1024") + @NotNull(message = "任务分配规则的编号不能为空") + private Long id; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BackTaskVo.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BackTaskVo.java new file mode 100644 index 00000000..81ddcb93 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BackTaskVo.java @@ -0,0 +1,42 @@ +package com.ruoyi.flowable.domain.vo.task; + + +import lombok.Data; + +/** + * @author wangzongrun + */ +@Data +public class BackTaskVo { + + /** + * 需要驳回的节点id 必填 + */ + private String distFlowElementId; + + /** + * 操作人code 必填 + */ + private String userCode; + + /** + * 任务id 必填 + */ + private String taskId; + + /** + * 流程实例的id 必填 + */ + private String processInstanceId; + + /** + * 审批意见 必填 + */ + private String message; + + /** + * 审批类型 必填 + */ + private String type; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskApproveReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskApproveReqVO.java new file mode 100644 index 00000000..7e013dd5 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskApproveReqVO.java @@ -0,0 +1,21 @@ +package com.ruoyi.flowable.domain.vo.task; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotEmpty; + +@ApiModel("管理后台 - 通过流程任务的 Request VO") +@Data +public class BpmTaskApproveReqVO { + + @ApiModelProperty(value = "任务编号", required = true, example = "1024") + @NotEmpty(message = "任务编号不能为空") + private String id; + + @ApiModelProperty(value = "审批意见", required = true, example = "不错不错!") + @NotEmpty(message = "审批意见不能为空") + private String reason; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskDonePageItemRespVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskDonePageItemRespVO.java new file mode 100644 index 00000000..c96fa73b --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskDonePageItemRespVO.java @@ -0,0 +1,27 @@ +package com.ruoyi.flowable.domain.vo.task; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.util.Date; + +@ApiModel("管理后台 - 流程任务的 Done 已完成的分页项 Response VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmTaskDonePageItemRespVO extends BpmTaskTodoPageItemRespVO { + + @ApiModelProperty(value = "结束时间", required = true) + private Date endTime; + @ApiModelProperty(value = "持续时间", required = true, example = "1000") + private Long durationInMillis; + + @ApiModelProperty(value = "任务结果", required = true, notes = "参见 bpm_process_instance_result", example = "2") + private Integer result; + @ApiModelProperty(value = "审批建议", required = true, example = "不请假了!") + private String reason; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskDonePageReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskDonePageReqVO.java new file mode 100644 index 00000000..912cad1e --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskDonePageReqVO.java @@ -0,0 +1,31 @@ +package com.ruoyi.flowable.domain.vo.task; + +import com.ruoyi.common.mybatis.pojo.PageParam; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; +import org.springframework.format.annotation.DateTimeFormat; + +import java.util.Date; + + +@ApiModel("管理后台 - 流程任务的 Done 已办的分页 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmTaskDonePageReqVO extends PageParam { + + @ApiModelProperty(value = "流程任务名", example = "芋道") + private String name; + + @ApiModelProperty(value = "开始的创建收间") + @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") + private Date beginCreateTime; + + @ApiModelProperty(value = "结束的创建时间") + @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") + private Date endCreateTime; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskRejectReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskRejectReqVO.java new file mode 100644 index 00000000..7770f038 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskRejectReqVO.java @@ -0,0 +1,21 @@ +package com.ruoyi.flowable.domain.vo.task; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotEmpty; + +@ApiModel("管理后台 - 不通过流程任务的 Request VO") +@Data +public class BpmTaskRejectReqVO { + + @ApiModelProperty(value = "任务编号", required = true, example = "1024") + @NotEmpty(message = "任务编号不能为空") + private String id; + + @ApiModelProperty(value = "审批意见", required = true, example = "不错不错!") + @NotEmpty(message = "审批意见不能为空") + private String reason; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskRespVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskRespVO.java new file mode 100644 index 00000000..5851b57a --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskRespVO.java @@ -0,0 +1,38 @@ +package com.ruoyi.flowable.domain.vo.task; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +@ApiModel("管理后台 - 流程任务的 Response VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmTaskRespVO extends BpmTaskDonePageItemRespVO { + + @ApiModelProperty(value = "任务定义的标识", required = true, example = "user-001") + private String definitionKey; + + /** + * 审核的用户信息 + */ + private User assigneeUser; + + @ApiModel("用户信息") + @Data + public static class User { + + @ApiModelProperty(value = "用户编号", required = true, example = "1") + private Long id; + @ApiModelProperty(value = "用户昵称", required = true, example = "芋艿") + private String nickname; + + @ApiModelProperty(value = "部门编号", required = true, example = "1") + private Long deptId; + @ApiModelProperty(value = "部门名称", required = true, example = "研发部") + private String deptName; + + } +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageItemRespVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageItemRespVO.java new file mode 100644 index 00000000..d269b0bb --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageItemRespVO.java @@ -0,0 +1,61 @@ +package com.ruoyi.flowable.domain.vo.task; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.util.Date; + +@ApiModel("管理后台 - 流程任务的 Running 进行中的分页项 Response VO") +@Data +public class BpmTaskTodoPageItemRespVO { + + @ApiModelProperty(value = "任务编号", required = true, example = "1024") + private String id; + + @ApiModelProperty(value = "任务名字", required = true, example = "芋道") + private String name; + + @ApiModelProperty(value = "接收时间", required = true) + private Date claimTime; + + @ApiModelProperty(value = "创建时间", required = true) + private Date createTime; + + @ApiModelProperty(value = "激活状态", required = true, example = "1", notes = "参见 SuspensionState 枚举") + private Integer suspensionState; + + @ApiModelProperty(value = "任务定义的标识") + private String taskDefinitionKey; + + /** + * 所属流程实例 + */ + private ProcessInstance processInstance; + + @Data + @ApiModel("流程实例") + public static class ProcessInstance { + + @ApiModelProperty(value = "流程实例编号", required = true, example = "1024") + private String id; + + @ApiModelProperty(value = "流程实例名称", required = true, example = "芋道") + private String name; + + @ApiModelProperty(value = "发起人的用户编号", required = true, example = "1024") + private Long startUserId; + + @ApiModelProperty(value = "发起人的用户昵称", required = true, example = "芋艿") + private String startUserNickname; + + @ApiModelProperty(value = "流程定义的编号", required = true, example = "2048") + private String processDefinitionId; + + @ApiModelProperty(value = "业务的唯一标识") + private String businessKey; + @ApiModelProperty(value = "流程定义标识") + private String processDefinitionKey; + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageReqVO.java new file mode 100644 index 00000000..7546889e --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageReqVO.java @@ -0,0 +1,31 @@ +package com.ruoyi.flowable.domain.vo.task; + +import com.ruoyi.common.mybatis.pojo.PageParam; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; +import org.springframework.format.annotation.DateTimeFormat; + +import java.util.Date; + + +@ApiModel("管理后台 - 流程任务的 TODO 待办的分页 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmTaskTodoPageReqVO extends PageParam { + + @ApiModelProperty(value = "流程任务名", example = "芋道") + private String name; + + @ApiModelProperty(value = "开始的创建收间") + @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") + private Date beginCreateTime; + + @ApiModelProperty(value = "结束的创建时间") + @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") + private Date endCreateTime; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskUpdateAssigneeReqVO.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskUpdateAssigneeReqVO.java new file mode 100644 index 00000000..9294ee3c --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/BpmTaskUpdateAssigneeReqVO.java @@ -0,0 +1,22 @@ +package com.ruoyi.flowable.domain.vo.task; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotEmpty; +import javax.validation.constraints.NotNull; + +@ApiModel("管理后台 - 流程任务的更新负责人的 Request VO") +@Data +public class BpmTaskUpdateAssigneeReqVO { + + @ApiModelProperty(value = "任务编号", required = true, example = "1024") + @NotEmpty(message = "任务编号不能为空") + private String id; + + @ApiModelProperty(value = "新审批人的用户编号", required = true, example = "2048") + @NotNull(message = "新审批人的用户编号不能为空") + private Long assigneeUserId; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/FlowNodeVo.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/FlowNodeVo.java new file mode 100644 index 00000000..11152c8f --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/domain/vo/task/FlowNodeVo.java @@ -0,0 +1,33 @@ +package com.ruoyi.flowable.domain.vo.task; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.io.Serializable; +import java.util.Date; + +/** + * + * @author wangzongrun + */ +@ApiModel("流程节点对象") +@Data +public class FlowNodeVo implements Serializable { + + @ApiModelProperty(value = "节点id") + private String nodeId; + + @ApiModelProperty(value = "节点名称") + private String nodeName; + + @ApiModelProperty(value = "执行人的code") + private String userCode; + + @ApiModelProperty(value = "执行人姓名") + private String userName; + + @ApiModelProperty(value = "任务节点结束时间") + private Date endTime; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/bpm/config/BpmCommonConfiguration.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/bpm/config/BpmCommonConfiguration.java new file mode 100644 index 00000000..ff709354 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/bpm/config/BpmCommonConfiguration.java @@ -0,0 +1,19 @@ +package com.ruoyi.flowable.framework.bpm.config; + +import com.ruoyi.flowable.framework.bpm.core.event.BpmProcessInstanceResultEventPublisher; +import org.springframework.context.ApplicationEventPublisher; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; + +/** + * BPM 通用的 Configuration 配置类,提供给 Activiti 和 Flowable + */ +@Configuration +public class BpmCommonConfiguration { + + @Bean + public BpmProcessInstanceResultEventPublisher processInstanceResultEventPublisher(ApplicationEventPublisher publisher) { + return new BpmProcessInstanceResultEventPublisher(publisher); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEvent.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEvent.java new file mode 100644 index 00000000..a8b6c37f --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEvent.java @@ -0,0 +1,43 @@ +package com.ruoyi.flowable.framework.bpm.core.event; + +import lombok.Data; +import org.springframework.context.ApplicationEvent; + +import javax.validation.constraints.NotNull; + +/** + * 流程实例的结果发生变化的 Event + * 定位:由于额外增加了 {@link BpmProcessInstanceExtDO#getResult()} 结果,所以增加该事件 + * + * hasPermi + */ +@SuppressWarnings("ALL") +@Data +public class BpmProcessInstanceResultEvent extends ApplicationEvent { + + /** + * 流程实例的编号 + */ + @NotNull(message = "流程实例的编号不能为空") + private String id; + /** + * 流程实例的 key + */ + @NotNull(message = "流程实例的 key 不能为空") + private String processDefinitionKey; + /** + * 流程实例的结果 + */ + @NotNull(message = "流程实例的结果不能为空") + private Integer result; + /** + * 流程实例对应的业务标识 + * 例如说,请假 + */ + private String businessKey; + + public BpmProcessInstanceResultEvent(Object source) { + super(source); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEventListener.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEventListener.java new file mode 100644 index 00000000..89da1355 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEventListener.java @@ -0,0 +1,34 @@ +package com.ruoyi.flowable.framework.bpm.core.event; + +import cn.hutool.core.util.StrUtil; +import org.springframework.context.ApplicationListener; + +/** + * {@link BpmProcessInstanceResultEvent} 的监听器 + * + * hasPermi + */ +public abstract class BpmProcessInstanceResultEventListener + implements ApplicationListener { + + @Override + public final void onApplicationEvent(BpmProcessInstanceResultEvent event) { + if (!StrUtil.equals(event.getProcessDefinitionKey(), getProcessDefinitionKey())) { + return; + } + onEvent(event); + } + + /** + * @return 返回监听的流程定义 Key + */ + protected abstract String getProcessDefinitionKey(); + + /** + * 处理事件 + * + * @param event 事件 + */ + protected abstract void onEvent(BpmProcessInstanceResultEvent event); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEventPublisher.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEventPublisher.java new file mode 100644 index 00000000..59e0a36b --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEventPublisher.java @@ -0,0 +1,24 @@ +package com.ruoyi.flowable.framework.bpm.core.event; + +import lombok.AllArgsConstructor; +import org.springframework.context.ApplicationEventPublisher; +import org.springframework.validation.annotation.Validated; + +import javax.validation.Valid; + +/** + * {@link BpmProcessInstanceResultEvent} 的生产者 + * + * hasPermi + */ +@AllArgsConstructor +@Validated +public class BpmProcessInstanceResultEventPublisher { + + private final ApplicationEventPublisher publisher; + + public void sendProcessInstanceResultEvent(@Valid BpmProcessInstanceResultEvent event) { + publisher.publishEvent(event); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/config/BpmFlowableConfiguration.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/config/BpmFlowableConfiguration.java new file mode 100644 index 00000000..dec897e6 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/config/BpmFlowableConfiguration.java @@ -0,0 +1,46 @@ +package com.ruoyi.flowable.framework.flowable.config; + +import cn.hutool.core.collection.ListUtil; +import com.ruoyi.flowable.framework.flowable.core.behavior.BpmActivityBehaviorFactory; +import com.ruoyi.flowable.service.definition.BpmTaskAssignRuleService; +import org.flowable.common.engine.api.delegate.event.FlowableEventListener; +import org.flowable.spring.SpringProcessEngineConfiguration; +import org.flowable.spring.boot.EngineConfigurationConfigurer; +import org.springframework.beans.factory.ObjectProvider; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; + +/** + * BPM 模块的 Flowable 配置类 + * + * @author jason + */ +@Configuration +public class BpmFlowableConfiguration { + + /** + * BPM 模块的 ProcessEngineConfigurationConfigurer 实现类: + * + * 1. 设置各种监听器 + * 2. 设置自定义的 ActivityBehaviorFactory 实现 + */ + @Bean + public EngineConfigurationConfigurer bpmProcessEngineConfigurationConfigurer( + ObjectProvider listeners, + BpmActivityBehaviorFactory bpmActivityBehaviorFactory) { + return configuration -> { + // 注册监听器,例如说 BpmActivityEventListener + configuration.setEventListeners(ListUtil.toList(listeners.iterator())); + // 设置 ActivityBehaviorFactory 实现类,用于流程任务的审核人的自定义 + configuration.setActivityBehaviorFactory(bpmActivityBehaviorFactory); + }; + } + + @Bean + public BpmActivityBehaviorFactory bpmActivityBehaviorFactory(BpmTaskAssignRuleService taskRuleService) { + BpmActivityBehaviorFactory bpmActivityBehaviorFactory = new BpmActivityBehaviorFactory(); + bpmActivityBehaviorFactory.setBpmTaskRuleService(taskRuleService); + return bpmActivityBehaviorFactory; + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/BpmActivityBehaviorFactory.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/BpmActivityBehaviorFactory.java new file mode 100644 index 00000000..33c8d2f3 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/BpmActivityBehaviorFactory.java @@ -0,0 +1,46 @@ +package com.ruoyi.flowable.framework.flowable.core.behavior; + +import com.ruoyi.flowable.service.definition.BpmTaskAssignRuleService; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.Setter; +import lombok.ToString; +import org.flowable.bpmn.model.Activity; +import org.flowable.bpmn.model.UserTask; +import org.flowable.engine.impl.bpmn.behavior.AbstractBpmnActivityBehavior; +import org.flowable.engine.impl.bpmn.behavior.ParallelMultiInstanceBehavior; +import org.flowable.engine.impl.bpmn.behavior.UserTaskActivityBehavior; +import org.flowable.engine.impl.bpmn.parser.factory.DefaultActivityBehaviorFactory; + +/** + * 自定义的 ActivityBehaviorFactory 实现类,目的如下: + * 1. 自定义 {@link #createUserTaskActivityBehavior(UserTask)}:实现自定义的流程任务的 assignee 负责人的分配 + * + * hasPermi + */ +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class BpmActivityBehaviorFactory extends DefaultActivityBehaviorFactory { + + @Setter + private BpmTaskAssignRuleService bpmTaskRuleService; + + @Override + public UserTaskActivityBehavior createUserTaskActivityBehavior(UserTask userTask) { + BpmUserTaskActivityBehavior bpmUserTaskActivityBehavior = new BpmUserTaskActivityBehavior(userTask); + bpmUserTaskActivityBehavior.setBpmTaskRuleService(bpmTaskRuleService); + return bpmUserTaskActivityBehavior; + } + + @Override + public ParallelMultiInstanceBehavior createParallelMultiInstanceBehavior(Activity activity, + AbstractBpmnActivityBehavior innerActivityBehavior) { + BpmParallelMultiInstanceBehavior bpmParallelMultiInstanceBehavior = new BpmParallelMultiInstanceBehavior(activity, innerActivityBehavior); + bpmParallelMultiInstanceBehavior.setBpmTaskRuleService(bpmTaskRuleService); + return bpmParallelMultiInstanceBehavior; + } + + // TODO @ke:SequentialMultiInstanceBehavior 这个抽空也可以看看 + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/BpmParallelMultiInstanceBehavior.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/BpmParallelMultiInstanceBehavior.java new file mode 100644 index 00000000..3746a4cb --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/BpmParallelMultiInstanceBehavior.java @@ -0,0 +1,60 @@ +package com.ruoyi.flowable.framework.flowable.core.behavior; + +import com.ruoyi.flowable.core.utils.FlowableUtils; +import com.ruoyi.flowable.service.definition.BpmTaskAssignRuleService; +import lombok.Setter; +import lombok.extern.slf4j.Slf4j; +import org.flowable.bpmn.model.Activity; +import org.flowable.engine.delegate.DelegateExecution; +import org.flowable.engine.impl.bpmn.behavior.AbstractBpmnActivityBehavior; +import org.flowable.engine.impl.bpmn.behavior.ParallelMultiInstanceBehavior; + +import java.util.Set; + +/** + * 自定义的【并行】的【多个】流程任务的 assignee 负责人的分配 + * 第一步,基于分配规则,计算出分配任务的【多个】候选人们。 + * 第二步,将【多个】任务候选人们,设置到 DelegateExecution 的 collectionVariable 变量中,以便 BpmUserTaskActivityBehavior 使用它 + * + * @author kemengkai + * @date 2022-04-21 16:57 + */ +@Slf4j +public class BpmParallelMultiInstanceBehavior extends ParallelMultiInstanceBehavior { + + @Setter + private BpmTaskAssignRuleService bpmTaskRuleService; + + public BpmParallelMultiInstanceBehavior(Activity activity, + AbstractBpmnActivityBehavior innerActivityBehavior) { + super(activity, innerActivityBehavior); + } + + /** + * 重写该方法,主要实现两个功能: + * 1. 忽略原有的 collectionVariable、collectionElementVariable 表达式,而是采用自己定义的 + * 2. 获得任务的处理人,并设置到 collectionVariable 中,用于 BpmUserTaskActivityBehavior 从中可以获取任务的处理人 + * + * 注意,多个任务实例,每个任务实例对应一个处理人,所以返回的数量就是任务处理人的数量 + * + * @param execution 执行任务 + * @return 数量 + */ + @Override + protected int resolveNrOfInstances(DelegateExecution execution) { + // 第一步,设置 collectionVariable 和 CollectionVariable + // 从 execution.getVariable() 读取所有任务处理人的 key + super.collectionExpression = null; // collectionExpression 和 collectionVariable 是互斥的 + String cv = FlowableUtils.formatCollectionVariable(execution.getCurrentActivityId()); + super.collectionVariable = cv; + // 从 execution.getVariable() 读取当前所有任务处理的人的 key + super.collectionElementVariable = FlowableUtils.formatCollectionElementVariable(execution.getCurrentActivityId()); + //todo 并签支持,并签的暂无业务所以先注释了 + execution.setVariable("coll_userList", cv); + // 第二步,获取任务的所有处理人 + Set assigneeUserIds = bpmTaskRuleService.calculateTaskCandidateUsers(execution); + execution.setVariable(super.collectionVariable, assigneeUserIds); + return assigneeUserIds.size(); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/BpmUserTaskActivityBehavior.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/BpmUserTaskActivityBehavior.java new file mode 100644 index 00000000..428e1d9f --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/BpmUserTaskActivityBehavior.java @@ -0,0 +1,65 @@ +package com.ruoyi.flowable.framework.flowable.core.behavior; + +import cn.hutool.core.collection.CollUtil; +import cn.hutool.core.lang.Assert; +import cn.hutool.core.util.RandomUtil; +import com.ruoyi.flowable.service.definition.BpmTaskAssignRuleService; +import lombok.Setter; +import lombok.extern.slf4j.Slf4j; +import org.flowable.bpmn.model.UserTask; +import org.flowable.common.engine.impl.el.ExpressionManager; +import org.flowable.engine.delegate.DelegateExecution; +import org.flowable.engine.impl.bpmn.behavior.UserTaskActivityBehavior; +import org.flowable.engine.impl.cfg.ProcessEngineConfigurationImpl; +import org.flowable.engine.impl.util.TaskHelper; +import org.flowable.task.service.TaskService; +import org.flowable.task.service.impl.persistence.entity.TaskEntity; + +import java.util.List; +import java.util.Set; + +/** + * 自定义的【单个】流程任务的 assignee 负责人的分配 + * 第一步,基于分配规则,计算出分配任务的【单个】候选人。如果找不到,则直接报业务异常,不继续执行后续的流程; + * 第二步,随机选择一个候选人,则选择作为 assignee 负责人。 + * + * hasPermi + */ +@Slf4j +public class BpmUserTaskActivityBehavior extends UserTaskActivityBehavior { + + @Setter + private BpmTaskAssignRuleService bpmTaskRuleService; + + public BpmUserTaskActivityBehavior(UserTask userTask) { + super(userTask); + } + + @Override + protected void handleAssignments(TaskService taskService, String assignee, String owner, + List candidateUsers, List candidateGroups, TaskEntity task, ExpressionManager expressionManager, + DelegateExecution execution, ProcessEngineConfigurationImpl processEngineConfiguration) { + // 第一步,获得任务的候选用户 + Long assigneeUserId = calculateTaskCandidateUsers(execution); + Assert.notNull(assigneeUserId, "任务处理人不能为空"); + // 第二步,设置作为负责人 + TaskHelper.changeTaskAssignee(task, String.valueOf(assigneeUserId)); + } + + private Long calculateTaskCandidateUsers(DelegateExecution execution) { + // 情况一,如果是多实例的任务,例如说会签、或签等情况,则从 Variable 中获取。它的任务处理人在 BpmParallelMultiInstanceBehavior 中已经被分配了 + if (super.multiInstanceActivityBehavior != null) { + return execution.getVariable(super.multiInstanceActivityBehavior.getCollectionElementVariable(), Long.class); + } + // 情况二,如果非多实例的任务,则计算任务处理人 + // 第一步,先计算可处理该任务的处理人们 + Set candidateUserIds = bpmTaskRuleService.calculateTaskCandidateUsers(execution); + // 第二步,后随机选择一个任务的处理人 + // 疑问:为什么一定要选择一个任务处理人? + // 解答:项目对 bpm 的任务是责任到人,所以每个任务有且仅有一个处理人。 + // 如果希望一个任务可以同时被多个人处理,可以考虑使用 BpmParallelMultiInstanceBehavior 实现的会签 or 或签。 + int index = RandomUtil.randomInt(candidateUserIds.size()); + return CollUtil.get(candidateUserIds, index); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/BpmTaskAssignScript.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/BpmTaskAssignScript.java new file mode 100644 index 00000000..30567f1b --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/BpmTaskAssignScript.java @@ -0,0 +1,34 @@ +package com.ruoyi.flowable.framework.flowable.core.behavior.script; + +import com.ruoyi.flowable.core.enums.definition.BpmTaskRuleScriptEnum; +import org.flowable.engine.delegate.DelegateExecution; + +import java.util.Set; + +/** + * Bpm 任务分配的自定义 Script 脚本 + * 使用场景: + * 1. 设置审批人为发起人 + * 2. 设置审批人为发起人的 Leader + * 3. 甚至审批人为发起人的 Leader 的 Leader + * + * hasPermi + */ +public interface BpmTaskAssignScript { + + /** + * 基于执行任务,获得任务的候选用户们 + * + * @param execution 执行任务 + * @return 候选人用户的编号数组 + */ + Set calculateTaskCandidateUsers(DelegateExecution execution); + + /** + * 获得枚举值 + * + * @return 枚举值 + */ + BpmTaskRuleScriptEnum getEnum(); +} + diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderAbstractScript.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderAbstractScript.java new file mode 100644 index 00000000..a029308c --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderAbstractScript.java @@ -0,0 +1,74 @@ +package com.ruoyi.flowable.framework.flowable.core.behavior.script.impl; + +import com.ruoyi.common.utils.NumberUtils; +import com.ruoyi.flowable.domain.dto.user.AdminUserRespDTO; +import com.ruoyi.flowable.domain.dto.user.DeptRespDTO; +import com.ruoyi.flowable.framework.flowable.core.behavior.script.BpmTaskAssignScript; +import com.ruoyi.flowable.service.task.BpmProcessInstanceService; +import com.ruoyi.flowable.service.user.AdminUserApi; +import com.ruoyi.flowable.service.user.DeptApi; +import org.flowable.engine.delegate.DelegateExecution; +import org.flowable.engine.runtime.ProcessInstance; +import org.springframework.context.annotation.Lazy; +import org.springframework.util.Assert; + +import javax.annotation.Resource; +import java.util.Set; + +import static com.ruoyi.common.utils.collection.SetUtils.asSet; +import static java.util.Collections.emptySet; + +/** + * 分配给发起人的 Leader 审批的 Script 实现类 + * 目前 Leader 的定义是, + * + * hasPermi + */ +public abstract class BpmTaskAssignLeaderAbstractScript implements BpmTaskAssignScript { + + @Resource + private AdminUserApi adminUserApi; + @Resource + private DeptApi deptApi; + @Resource + @Lazy // 解决循环依赖 + private BpmProcessInstanceService bpmProcessInstanceService; + + protected Set calculateTaskCandidateUsers(DelegateExecution execution, int level) { + Assert.isTrue(level > 0, "level 必须大于 0"); + // 获得发起人 + ProcessInstance processInstance = bpmProcessInstanceService.getProcessInstance(execution.getProcessInstanceId()); + Long startUserId = NumberUtils.parseLong(processInstance.getStartUserId()); + // 获得对应 leve 的部门 + + DeptRespDTO dept = null; + for (int i = 0; i < level; i++) { + // 获得 level 对应的部门 + if (dept == null) { + dept = getStartUserDept(startUserId); + // 找不到发起人的部门,所以无法使用该规则 + if (dept == null) { + return emptySet(); + } + } else { + DeptRespDTO parentDept = deptApi.getDept(dept.getParentId()); + // 找不到父级部门,所以只好结束寻找。原因是:例如说,级别比较高的人,所在部门层级比较少 + if (parentDept == null) { + break; + } + dept = parentDept; + } + } + return dept.getLeaderUserId() != null ? asSet(dept.getLeaderUserId()) : emptySet(); + } + + private DeptRespDTO getStartUserDept(Long startUserId) { + AdminUserRespDTO startUser = adminUserApi.getUser(startUserId); + // 找不到部门,所以无法使用该规则 + if (startUser.getDeptId() == null) { + return null; + } + return deptApi.getDept(startUser.getDeptId()); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderX1Script.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderX1Script.java new file mode 100644 index 00000000..efa93cd9 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderX1Script.java @@ -0,0 +1,27 @@ +package com.ruoyi.flowable.framework.flowable.core.behavior.script.impl; + +import com.ruoyi.flowable.core.enums.definition.BpmTaskRuleScriptEnum; +import org.flowable.engine.delegate.DelegateExecution; +import org.springframework.stereotype.Component; + +import java.util.Set; + +/** + * 分配给发起人的一级 Leader 审批的 Script 实现类 + * + * hasPermi + */ +@Component +public class BpmTaskAssignLeaderX1Script extends BpmTaskAssignLeaderAbstractScript { + + @Override + public Set calculateTaskCandidateUsers(DelegateExecution execution) { + return calculateTaskCandidateUsers(execution, 1); + } + + @Override + public BpmTaskRuleScriptEnum getEnum() { + return BpmTaskRuleScriptEnum.LEADER_X1; + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderX2Script.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderX2Script.java new file mode 100644 index 00000000..6e32b39f --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderX2Script.java @@ -0,0 +1,27 @@ +package com.ruoyi.flowable.framework.flowable.core.behavior.script.impl; + +import com.ruoyi.flowable.core.enums.definition.BpmTaskRuleScriptEnum; +import org.flowable.engine.delegate.DelegateExecution; +import org.springframework.stereotype.Component; + +import java.util.Set; + +/** + * 分配给发起人的二级 Leader 审批的 Script 实现类 + * + * hasPermi + */ +@Component +public class BpmTaskAssignLeaderX2Script extends BpmTaskAssignLeaderAbstractScript { + + @Override + public Set calculateTaskCandidateUsers(DelegateExecution execution) { + return calculateTaskCandidateUsers(execution, 2); + } + + @Override + public BpmTaskRuleScriptEnum getEnum() { + return BpmTaskRuleScriptEnum.LEADER_X2; + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignStartUserScript.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignStartUserScript.java new file mode 100644 index 00000000..de64b321 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignStartUserScript.java @@ -0,0 +1,40 @@ +package com.ruoyi.flowable.framework.flowable.core.behavior.script.impl; + +import com.ruoyi.common.utils.NumberUtils; +import com.ruoyi.common.utils.collection.SetUtils; +import com.ruoyi.flowable.core.enums.definition.BpmTaskRuleScriptEnum; +import com.ruoyi.flowable.framework.flowable.core.behavior.script.BpmTaskAssignScript; +import com.ruoyi.flowable.service.task.BpmProcessInstanceService; +import org.flowable.engine.delegate.DelegateExecution; +import org.flowable.engine.runtime.ProcessInstance; +import org.springframework.context.annotation.Lazy; +import org.springframework.stereotype.Component; + +import javax.annotation.Resource; +import java.util.Set; + +/** + * 分配给发起人审批的 Script 实现类 + * + * hasPermi + */ +@Component +public class BpmTaskAssignStartUserScript implements BpmTaskAssignScript { + + @Resource + @Lazy // 解决循环依赖 + private BpmProcessInstanceService bpmProcessInstanceService; + + @Override + public Set calculateTaskCandidateUsers(DelegateExecution execution) { + ProcessInstance processInstance = bpmProcessInstanceService.getProcessInstance(execution.getProcessInstanceId()); + Long startUserId = NumberUtils.parseLong(processInstance.getStartUserId()); + return SetUtils.asSet(startUserId); + } + + @Override + public BpmTaskRuleScriptEnum getEnum() { + return BpmTaskRuleScriptEnum.START_USER; + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/listener/BpmProcessInstanceEventListener.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/listener/BpmProcessInstanceEventListener.java new file mode 100644 index 00000000..22a4964b --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/listener/BpmProcessInstanceEventListener.java @@ -0,0 +1,53 @@ +package com.ruoyi.flowable.framework.flowable.core.listener; + +import com.google.common.collect.ImmutableSet; +import com.ruoyi.flowable.domain.entity.task.BpmProcessInstanceExtDO; +import com.ruoyi.flowable.service.task.BpmProcessInstanceService; +import org.flowable.common.engine.api.delegate.event.FlowableEngineEntityEvent; +import org.flowable.common.engine.api.delegate.event.FlowableEngineEventType; +import org.flowable.engine.delegate.event.AbstractFlowableEngineEventListener; +import org.flowable.engine.delegate.event.FlowableCancelledEvent; +import org.flowable.engine.runtime.ProcessInstance; +import org.springframework.context.annotation.Lazy; +import org.springframework.stereotype.Component; + +import javax.annotation.Resource; +import java.util.Set; + +/** + * 监听 {@link ProcessInstance} 的开始与完成,创建与更新对应的 {@link BpmProcessInstanceExtDO} 记录 + * + * @author jason + */ +@Component +public class BpmProcessInstanceEventListener extends AbstractFlowableEngineEventListener { + + @Resource + @Lazy + private BpmProcessInstanceService processInstanceService; + + public static final Set PROCESS_INSTANCE_EVENTS = ImmutableSet.builder() + .add(FlowableEngineEventType.PROCESS_CREATED) + .add(FlowableEngineEventType.PROCESS_CANCELLED) + .add(FlowableEngineEventType.PROCESS_COMPLETED) + .build(); + + public BpmProcessInstanceEventListener(){ + super(PROCESS_INSTANCE_EVENTS); + } + + @Override + protected void processCreated(FlowableEngineEntityEvent event) { + processInstanceService.createProcessInstanceExt((ProcessInstance)event.getEntity()); + } + + @Override + protected void processCancelled(FlowableCancelledEvent event) { + processInstanceService.updateProcessInstanceExtCancel(event); + } + + @Override + protected void processCompleted(FlowableEngineEntityEvent event) { + processInstanceService.updateProcessInstanceExtComplete((ProcessInstance)event.getEntity()); + } +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/listener/BpmTaskEventListener.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/listener/BpmTaskEventListener.java new file mode 100644 index 00000000..ac81716a --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/framework/flowable/core/listener/BpmTaskEventListener.java @@ -0,0 +1,82 @@ +package com.ruoyi.flowable.framework.flowable.core.listener; + +import cn.hutool.core.collection.CollUtil; +import cn.hutool.core.util.StrUtil; +import com.google.common.collect.ImmutableSet; +import com.ruoyi.flowable.domain.entity.task.BpmTaskExtDO; +import com.ruoyi.flowable.service.task.BpmActivityService; +import com.ruoyi.flowable.service.task.BpmTaskService; +import lombok.extern.slf4j.Slf4j; +import org.flowable.common.engine.api.delegate.event.FlowableEngineEntityEvent; +import org.flowable.common.engine.api.delegate.event.FlowableEngineEventType; +import org.flowable.engine.delegate.event.AbstractFlowableEngineEventListener; +import org.flowable.engine.delegate.event.FlowableActivityCancelledEvent; +import org.flowable.engine.history.HistoricActivityInstance; +import org.flowable.task.api.Task; +import org.springframework.context.annotation.Lazy; +import org.springframework.stereotype.Component; + +import javax.annotation.Resource; +import java.util.List; +import java.util.Set; + +/** + * 监听 {@link Task} 的开始与完成,创建与更新对应的 {@link BpmTaskExtDO} 记录 + * + * @author jason + */ +@Component +@Slf4j +public class BpmTaskEventListener extends AbstractFlowableEngineEventListener { + + @Resource + @Lazy // 解决循环依赖 + private BpmTaskService taskService; + + @Resource + @Lazy // 解决循环依赖 + private BpmActivityService activityService; + + public static final Set TASK_EVENTS = ImmutableSet.builder() + .add(FlowableEngineEventType.TASK_CREATED) + .add(FlowableEngineEventType.TASK_ASSIGNED) + .add(FlowableEngineEventType.TASK_COMPLETED) + .add(FlowableEngineEventType.ACTIVITY_CANCELLED) + .build(); + + public BpmTaskEventListener(){ + super(TASK_EVENTS); + } + + @Override + protected void taskCreated(FlowableEngineEntityEvent event) { + taskService.createTaskExt((Task) event.getEntity()); + } + + @Override + protected void taskCompleted(FlowableEngineEntityEvent event) { + taskService.updateTaskExtComplete((Task)event.getEntity()); + } + + @Override + protected void taskAssigned(FlowableEngineEntityEvent event) { + taskService.updateTaskExtAssign((Task)event.getEntity()); + } + + @Override + protected void activityCancelled(FlowableActivityCancelledEvent event) { + List activityList = activityService.getHistoricActivityListByExecutionId(event.getExecutionId()); + if (CollUtil.isEmpty(activityList)) { + log.error("[activityCancelled][使用 executionId({}) 查找不到对应的活动实例]", event.getExecutionId()); + return; + } + // 遍历处理 + activityList.forEach(activity -> { + if (StrUtil.isEmpty(activity.getTaskId())) { + return; + } + taskService.updateTaskExtCancel(activity.getTaskId()); + }); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/definition/BpmFormMapper.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/definition/BpmFormMapper.java new file mode 100644 index 00000000..e2704484 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/definition/BpmFormMapper.java @@ -0,0 +1,25 @@ +package com.ruoyi.flowable.mapper.definition; + + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.mybatis.query.QueryWrapperX; +import com.ruoyi.flowable.domain.entity.definition.BpmFormDO; +import com.ruoyi.flowable.domain.vo.form.BpmFormPageReqVO; +import org.apache.ibatis.annotations.Mapper; + +/** + * 动态表单 Mapper + * + * @author 风里雾里 + */ +@Mapper +public interface BpmFormMapper extends BaseMapperX { + + default PageResult selectPage(BpmFormPageReqVO reqVO) { + return selectPage(reqVO, new QueryWrapperX() + .likeIfPresent("name", reqVO.getName()) + .orderByDesc("id")); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/definition/BpmProcessDefinitionExtMapper.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/definition/BpmProcessDefinitionExtMapper.java new file mode 100644 index 00000000..5415eb31 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/definition/BpmProcessDefinitionExtMapper.java @@ -0,0 +1,21 @@ +package com.ruoyi.flowable.mapper.definition; + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.ruoyi.flowable.domain.entity.definition.BpmProcessDefinitionExtDO; +import org.apache.ibatis.annotations.Mapper; + +import java.util.Collection; +import java.util.List; + +@Mapper +public interface BpmProcessDefinitionExtMapper extends BaseMapperX { + + default List selectListByProcessDefinitionIds(Collection processDefinitionIds) { + return selectList("process_definition_id", processDefinitionIds); + } + + default BpmProcessDefinitionExtDO selectByProcessDefinitionId(String processDefinitionId) { + return selectOne("process_definition_id", processDefinitionId); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/definition/BpmTaskAssignRuleMapper.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/definition/BpmTaskAssignRuleMapper.java new file mode 100644 index 00000000..58464030 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/definition/BpmTaskAssignRuleMapper.java @@ -0,0 +1,37 @@ +package com.ruoyi.flowable.mapper.definition; + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.ruoyi.common.mybatis.query.QueryWrapperX; +import com.ruoyi.flowable.domain.entity.definition.BpmTaskAssignRuleDO; +import org.apache.ibatis.annotations.Mapper; +import org.springframework.lang.Nullable; + +import java.util.List; + +@Mapper +public interface BpmTaskAssignRuleMapper extends BaseMapperX { + + default List selectListByProcessDefinitionId(String processDefinitionId, + @Nullable String taskDefinitionKey) { + return selectList(new QueryWrapperX() + .eq("process_definition_id", processDefinitionId) + .eqIfPresent("task_definition_key", taskDefinitionKey)); + } + + default List selectListByModelId(String modelId) { + return selectList(new QueryWrapperX() + .eq("model_id", modelId) + .eq("process_definition_id", BpmTaskAssignRuleDO.PROCESS_DEFINITION_ID_NULL)); + } + + default BpmTaskAssignRuleDO selectListByModelIdAndTaskDefinitionKey(String modelId, + String taskDefinitionKey) { + return selectOne(new QueryWrapperX() + .eq("model_id", modelId) + .eq("process_definition_id", BpmTaskAssignRuleDO.PROCESS_DEFINITION_ID_NULL) + .eq("task_definition_key", taskDefinitionKey)); + } + + + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/definition/BpmUserGroupMapper.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/definition/BpmUserGroupMapper.java new file mode 100644 index 00000000..fd790984 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/definition/BpmUserGroupMapper.java @@ -0,0 +1,32 @@ +package com.ruoyi.flowable.mapper.definition; + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.mybatis.query.LambdaQueryWrapperX; +import com.ruoyi.flowable.domain.entity.definition.BpmUserGroupDO; +import com.ruoyi.flowable.domain.vo.group.BpmUserGroupPageReqVO; +import org.apache.ibatis.annotations.Mapper; + +import java.util.List; + +/** + * 用户组 Mapper + * + * hasPermi + */ +@Mapper +public interface BpmUserGroupMapper extends BaseMapperX { + + default PageResult selectPage(BpmUserGroupPageReqVO reqVO) { + return selectPage(reqVO, new LambdaQueryWrapperX() + .likeIfPresent(BpmUserGroupDO::getName, reqVO.getName()) + .eqIfPresent(BpmUserGroupDO::getStatus, reqVO.getStatus()) + .betweenIfPresent(BpmUserGroupDO::getCreateTime, reqVO.getCreateTime()) + .orderByDesc(BpmUserGroupDO::getId)); + } + + default List selectListByStatus(Integer status) { + return selectList(BpmUserGroupDO::getStatus, status); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/oa/BpmOALeaveMapper.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/oa/BpmOALeaveMapper.java new file mode 100644 index 00000000..4e0bb6a2 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/oa/BpmOALeaveMapper.java @@ -0,0 +1,32 @@ +package com.ruoyi.flowable.mapper.oa; + +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.mybatis.query.LambdaQueryWrapperX; +import com.ruoyi.flowable.domain.entity.oa.BpmOALeaveDO; +import com.ruoyi.flowable.domain.vo.oa.BpmOALeavePageReqVO; +import org.apache.ibatis.annotations.Mapper; +import org.apache.ibatis.annotations.Param; + +/** + * 请假申请 Mapper + * + * @author jason + * hasPermi + */ +@Mapper +public interface BpmOALeaveMapper extends BaseMapperX { + + default PageResult selectPage(Long userId, BpmOALeavePageReqVO reqVO) { + return selectPage(reqVO, new LambdaQueryWrapperX() + .eqIfPresent(BpmOALeaveDO::getUserId, userId) + .eqIfPresent(BpmOALeaveDO::getResult, reqVO.getResult()) + .eqIfPresent(BpmOALeaveDO::getType, reqVO.getType()) + .likeIfPresent(BpmOALeaveDO::getReason, reqVO.getReason()) + .betweenIfPresent(BpmOALeaveDO::getCreateTime, reqVO.getCreateTime()) + .orderByDesc(BpmOALeaveDO::getId)); + } + + IPage selectBpmOaLeavePage(IPage page,@Param("entity") BpmOALeaveDO bpmOaLeave); +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/task/BpmProcessInstanceExtMapper.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/task/BpmProcessInstanceExtMapper.java new file mode 100644 index 00000000..4433b424 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/task/BpmProcessInstanceExtMapper.java @@ -0,0 +1,34 @@ +package com.ruoyi.flowable.mapper.task; + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.mybatis.query.LambdaQueryWrapperX; +import com.ruoyi.flowable.domain.entity.task.BpmProcessInstanceExtDO; +import com.ruoyi.flowable.domain.vo.instance.BpmProcessInstanceMyPageReqVO; +import org.apache.ibatis.annotations.Mapper; + +@Mapper +public interface BpmProcessInstanceExtMapper extends BaseMapperX { + + default PageResult selectPage(Long userId, BpmProcessInstanceMyPageReqVO reqVO) { + return selectPage(reqVO, new LambdaQueryWrapperX() + .eqIfPresent(BpmProcessInstanceExtDO::getStartUserId, userId) + .likeIfPresent(BpmProcessInstanceExtDO::getName, reqVO.getName()) + .eqIfPresent(BpmProcessInstanceExtDO::getProcessDefinitionId, reqVO.getProcessDefinitionId()) + .eqIfPresent(BpmProcessInstanceExtDO::getCategory, reqVO.getCategory()) + .eqIfPresent(BpmProcessInstanceExtDO::getStatus, reqVO.getStatus()) + .eqIfPresent(BpmProcessInstanceExtDO::getResult, reqVO.getResult()) + .betweenIfPresent(BpmProcessInstanceExtDO::getCreateTime, reqVO.getCreateTime()) + .orderByDesc(BpmProcessInstanceExtDO::getId)); + } + + default BpmProcessInstanceExtDO selectByProcessInstanceId(String processInstanceId) { + return selectOne(BpmProcessInstanceExtDO::getProcessInstanceId, processInstanceId); + } + + default void updateByProcessInstanceId(BpmProcessInstanceExtDO updateObj) { + update(updateObj, new LambdaQueryWrapperX() + .eq(BpmProcessInstanceExtDO::getProcessInstanceId, updateObj.getProcessInstanceId())); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/task/BpmTaskExtMapper.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/task/BpmTaskExtMapper.java new file mode 100644 index 00000000..b1d0db0a --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/mapper/task/BpmTaskExtMapper.java @@ -0,0 +1,26 @@ +package com.ruoyi.flowable.mapper.task; + +import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.ruoyi.flowable.domain.entity.task.BpmTaskExtDO; +import org.apache.ibatis.annotations.Mapper; + +import java.util.Collection; +import java.util.List; + +@Mapper +public interface BpmTaskExtMapper extends BaseMapperX { + + default void updateByTaskId(BpmTaskExtDO entity) { + update(entity, new LambdaQueryWrapper().eq(BpmTaskExtDO::getTaskId, entity.getTaskId())); + } + + default List selectListByTaskIds(Collection taskIds) { + return selectList(BpmTaskExtDO::getTaskId, taskIds); + } + + default BpmTaskExtDO selectByTaskId(String taskId) { + return selectOne(BpmTaskExtDO::getTaskId, taskId); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmFormService.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmFormService.java new file mode 100644 index 00000000..9d561d40 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmFormService.java @@ -0,0 +1,99 @@ +package com.ruoyi.flowable.service.definition; + +import cn.hutool.core.collection.CollUtil; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.domain.entity.definition.BpmFormDO; +import com.ruoyi.flowable.domain.vo.form.BpmFormCreateReqVO; +import com.ruoyi.flowable.domain.vo.form.BpmFormPageReqVO; +import com.ruoyi.flowable.domain.vo.form.BpmFormUpdateReqVO; + +import javax.validation.Valid; +import java.util.Collection; +import java.util.Collections; +import java.util.List; +import java.util.Map; + + +/** + * 动态表单 Service 接口 + * + * @author @风里雾里 + */ +public interface BpmFormService { + + /** + * 创建动态表单 + * + * @param createReqVO 创建信息 + * @return 编号 + */ + Long createForm(@Valid BpmFormCreateReqVO createReqVO); + + /** + * 更新动态表单 + * + * @param updateReqVO 更新信息 + */ + void updateForm(@Valid BpmFormUpdateReqVO updateReqVO); + + /** + * 删除动态表单 + * + * @param id 编号 + */ + void deleteForm(Long id); + + /** + * 获得动态表单 + * + * @param id 编号 + * @return 动态表单 + */ + BpmFormDO getForm(Long id); + + /** + * 获得动态表单列表 + * + * @return 动态表单列表 + */ + List getFormList(); + + /** + * 获得动态表单列表 + * + * @param ids 编号 + * @return 动态表单列表 + */ + List getFormList(Collection ids); + + /** + * 获得动态表单 Map + * + * @param ids 编号 + * @return 动态表单 Map + */ + default Map getFormMap(Collection ids) { + if (CollUtil.isEmpty(ids)) { + return Collections.emptyMap(); + } + return CollectionUtils.convertMap(this.getFormList(ids), BpmFormDO::getId); + } + + /** + * 获得动态表单分页 + * + * @param pageReqVO 分页查询 + * @return 动态表单分页 + */ + PageResult getFormPage(BpmFormPageReqVO pageReqVO); + + /** + * 校验流程表单已配置 + * + * @param configStr configStr 字段 + * @return 流程表单 + */ + BpmFormDO checkFormConfig(String configStr); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmModelService.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmModelService.java new file mode 100644 index 00000000..23d84892 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmModelService.java @@ -0,0 +1,86 @@ +package com.ruoyi.flowable.service.definition; + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.flowable.domain.vo.model.*; +import org.flowable.bpmn.model.BpmnModel; +import org.flowable.bpmn.model.FlowNode; + +import javax.validation.Valid; + +/** + * Flowable流程模型接口 + * + * @author yunlongn + */ +public interface BpmModelService { + /** + * 获得流程模型分页 + * + * @param pageVO 分页查询 + * @return 流程模型分页 + */ + PageResult getModelPage(BpmModelPageReqVO pageVO); + + /** + * 创建流程模型 + * + * @param modelVO 创建信息 + * @param bpmnXml BPMN XML + * @return 创建的流程模型的编号 + */ + String createModel(@Valid BpmModelCreateReqVO modelVO, String bpmnXml); + + /** + * 获得流程模块 + * + * @param id 编号 + * @return 流程模型 + */ + BpmModelRespVO getModel(String id); + + /** + * 修改流程模型 + * + * @param updateReqVO 更新信息 + */ + void updateModel(@Valid BpmModelUpdateReqVO updateReqVO); + + /** + * 将流程模型,部署成一个流程定义 + * + * @param id 编号 + */ + void deployModel(String id); + + /** + * 删除模型 + * + * @param id 编号 + */ + void deleteModel(String id); + + /** + * 修改模型的状态,实际更新的部署的流程定义的状态 + * + * @param id 编号 + * @param state 状态 + */ + void updateModelState(String id, Integer state); + + /** + * 获得流程模型编号对应的 BPMN Model + * + * @param id 流程模型编号 + * @return BPMN Model + */ + BpmnModel getBpmnModel(String id); + + /** + * 查找节点 + * @param processDefId 流程定义id + * @param activityId 节点id + * @return + */ + FlowNode findFlowNodeByActivityId(String processDefId, String activityId) ; + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmProcessDefinitionService.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmProcessDefinitionService.java new file mode 100644 index 00000000..6e912659 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmProcessDefinitionService.java @@ -0,0 +1,160 @@ +package com.ruoyi.flowable.service.definition; + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.domain.dto.definition.BpmProcessDefinitionCreateReqDTO; +import com.ruoyi.flowable.domain.entity.definition.BpmProcessDefinitionExtDO; +import com.ruoyi.flowable.domain.vo.process.BpmProcessDefinitionListReqVO; +import com.ruoyi.flowable.domain.vo.process.BpmProcessDefinitionPageItemRespVO; +import com.ruoyi.flowable.domain.vo.process.BpmProcessDefinitionPageReqVO; +import com.ruoyi.flowable.domain.vo.process.BpmProcessDefinitionRespVO; +import org.flowable.bpmn.model.BpmnModel; +import org.flowable.engine.repository.Deployment; +import org.flowable.engine.repository.ProcessDefinition; + +import javax.validation.Valid; +import java.util.List; +import java.util.Map; +import java.util.Set; + +/** + * Flowable流程定义接口 + * + * @author yunlong.li + * @author ZJQ + * hasPermi + */ +public interface BpmProcessDefinitionService { + + /** + * 获得流程定义分页 + * + * @param pageReqVO 分页入参 + * @return 流程定义 Page + */ + PageResult getProcessDefinitionPage(BpmProcessDefinitionPageReqVO pageReqVO); + + /** + * 获得流程定义列表 + * + * @param listReqVO 列表入参 + * @return 流程定义列表 + */ + List getProcessDefinitionList(BpmProcessDefinitionListReqVO listReqVO); + + /** + * 创建流程定义 + * + * @param createReqDTO 创建信息 + * @return 流程编号 + */ + String createProcessDefinition(@Valid BpmProcessDefinitionCreateReqDTO createReqDTO); + + /** + * 更新流程定义状态 + * + * @param id 流程定义的编号 + * @param state 状态 + */ + void updateProcessDefinitionState(String id, Integer state); + + /** + * 获得流程定义对应的 BPMN XML + * + * @param id 流程定义编号 + * @return BPMN XML + */ + String getProcessDefinitionBpmnXML(String id); + + /** + * 获得需要创建的流程定义,是否和当前激活的流程定义相等 + * + * @param createReqDTO 创建信息 + * @return 是否相等 + */ + boolean isProcessDefinitionEquals(@Valid BpmProcessDefinitionCreateReqDTO createReqDTO); + + /** + * 获得编号对应的 BpmProcessDefinitionExtDO + * + * @param id 编号 + * @return 流程定义拓展 + */ + BpmProcessDefinitionExtDO getProcessDefinitionExt(String id); + + /** + * 获得编号对应的 ProcessDefinition + * + * @param id 编号 + * @return 流程定义 + */ + ProcessDefinition getProcessDefinition(String id); + + /** + * 获得编号对应的 ProcessDefinition + * + * 相比 {@link #getProcessDefinition(String)} 方法,category 的取值是正确 + * + * @param id 编号 + * @return 流程定义 + */ + ProcessDefinition getProcessDefinition2(String id); + + /** + * 获得 deploymentId 对应的 ProcessDefinition + * + * @param deploymentId 部署编号 + * @return 流程定义 + */ + ProcessDefinition getProcessDefinitionByDeploymentId(String deploymentId); + + /** + * 获得 deploymentIds 对应的 ProcessDefinition 数组 + * + * @param deploymentIds 部署编号的数组 + * @return 流程定义的数组 + */ + List getProcessDefinitionListByDeploymentIds(Set deploymentIds); + + /** + * 获得流程定义标识对应的激活的流程定义 + * + * @param key 流程定义的标识 + * @return 流程定义 + */ + ProcessDefinition getActiveProcessDefinition(String key); + + /** + * 获得 ids 对应的 Deployment Map + * + * @param ids 部署编号的数组 + * @return 流程部署 Map + */ + default Map getDeploymentMap(Set ids) { + return CollectionUtils.convertMap(getDeployments(ids), Deployment::getId); + } + + /** + * 获得 ids 对应的 Deployment 数组 + * + * @param ids 部署编号的数组 + * @return 流程部署的数组 + */ + List getDeployments(Set ids); + + /** + * 获得 id 对应的 Deployment + * + * @param id 部署编号 + * @return 流程部署 + */ + Deployment getDeployment(String id); + + /** + * 获得 Bpmn 模型 + * + * @param processDefinitionId 流程定义的编号 + * @return Bpmn 模型 + */ + BpmnModel getBpmnModel(String processDefinitionId); +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmProcessInstanceApi.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmProcessInstanceApi.java new file mode 100644 index 00000000..1ab0d7c8 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmProcessInstanceApi.java @@ -0,0 +1,24 @@ +package com.ruoyi.flowable.service.definition; + + +import com.ruoyi.flowable.domain.dto.task.BpmProcessInstanceCreateReqDTO; + +import javax.validation.Valid; + +/** + * 流程实例 Api 接口 + * + * hasPermi + */ +public interface BpmProcessInstanceApi { + + /** + * 创建流程实例(提供给内部) + * + * @param userId 用户编号 + * @param reqDTO 创建信息 + * @return 实例的编号 + */ + String createProcessInstance(Long userId, @Valid BpmProcessInstanceCreateReqDTO reqDTO); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmTaskAssignRuleService.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmTaskAssignRuleService.java new file mode 100644 index 00000000..c2710312 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmTaskAssignRuleService.java @@ -0,0 +1,97 @@ +package com.ruoyi.flowable.service.definition; + +import com.ruoyi.flowable.domain.entity.definition.BpmTaskAssignRuleDO; +import com.ruoyi.flowable.domain.vo.rule.BpmTaskAssignRuleCreateReqVO; +import com.ruoyi.flowable.domain.vo.rule.BpmTaskAssignRuleRespVO; +import com.ruoyi.flowable.domain.vo.rule.BpmTaskAssignRuleUpdateReqVO; +import org.flowable.engine.delegate.DelegateExecution; +import org.springframework.lang.Nullable; + +import javax.validation.Valid; +import java.util.List; +import java.util.Set; + +/** + * BPM 任务分配规则 Service 接口 + * + * hasPermi + */ +public interface BpmTaskAssignRuleService { + + /** + * 获得流程定义的任务分配规则数组 + * + * @param processDefinitionId 流程定义的编号 + * @param taskDefinitionKey 流程任务定义的 Key。允许空 + * @return 任务规则数组 + */ + List getTaskAssignRuleListByProcessDefinitionId(String processDefinitionId, + @Nullable String taskDefinitionKey); + + /** + * 获得流程模型的任务规则数组 + * + * @param modelId 流程模型的编号 + * @return 任务规则数组 + */ + List getTaskAssignRuleListByModelId(String modelId); + + /** + * 获得流程定义的任务分配规则数组 + * + * @param modelId 流程模型的编号 + * @param processDefinitionId 流程定义的编号 + * @return 任务规则数组 + */ + List getTaskAssignRuleList(String modelId, String processDefinitionId); + + /** + * 创建任务分配规则 + * + * @param reqVO 创建信息 + * @return 规则编号 + */ + Long createTaskAssignRule(@Valid BpmTaskAssignRuleCreateReqVO reqVO); + + /** + * 更新任务分配规则 + * + * @param reqVO 创建信息 + */ + void updateTaskAssignRule(@Valid BpmTaskAssignRuleUpdateReqVO reqVO); + + /** + * 判断指定流程模型和流程定义的分配规则是否相等 + * + * @param modelId 流程模型编号 + * @param processDefinitionId 流程定义编号 + * @return 是否相等 + */ + boolean isTaskAssignRulesEquals(String modelId, String processDefinitionId); + + /** + * 将流程流程模型的任务分配规则,复制一份给流程定义 + * 目的:每次流程模型部署时,都会生成一个新的流程定义,此时考虑到每次部署的流程不可变性,所以需要复制一份给该流程定义 + * + * @param fromModelId 流程模型编号 + * @param toProcessDefinitionId 流程定义编号 + */ + void copyTaskAssignRules(String fromModelId, String toProcessDefinitionId); + + /** + * 校验流程模型的任务分配规则全部都配置了 + * 目的:如果有规则未配置,会导致流程任务找不到负责人,进而流程无法进行下去! + * + * @param id 流程模型编号 + */ + void checkTaskAssignRuleAllConfig(String id); + + /** + * 计算当前执行任务的处理人 + * + * @param execution 执行任务 + * @return 处理人的编号数组 + */ + Set calculateTaskCandidateUsers(DelegateExecution execution); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmUserGroupService.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmUserGroupService.java new file mode 100644 index 00000000..8556cbcd --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/BpmUserGroupService.java @@ -0,0 +1,84 @@ +package com.ruoyi.flowable.service.definition; + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.flowable.domain.entity.definition.BpmUserGroupDO; +import com.ruoyi.flowable.domain.vo.group.BpmUserGroupCreateReqVO; +import com.ruoyi.flowable.domain.vo.group.BpmUserGroupPageReqVO; +import com.ruoyi.flowable.domain.vo.group.BpmUserGroupUpdateReqVO; + +import javax.validation.Valid; +import java.util.Collection; +import java.util.List; +import java.util.Set; + +/** + * 用户组 Service 接口 + * + * hasPermi + */ +public interface BpmUserGroupService { + + /** + * 创建用户组 + * + * @param createReqVO 创建信息 + * @return 编号 + */ + Long createUserGroup(@Valid BpmUserGroupCreateReqVO createReqVO); + + /** + * 更新用户组 + * + * @param updateReqVO 更新信息 + */ + void updateUserGroup(@Valid BpmUserGroupUpdateReqVO updateReqVO); + + /** + * 删除用户组 + * + * @param id 编号 + */ + void deleteUserGroup(Long id); + + /** + * 获得用户组 + * + * @param id 编号 + * @return 用户组 + */ + BpmUserGroupDO getUserGroup(Long id); + + /** + * 获得用户组列表 + * + * @param ids 编号 + * @return 用户组列表 + */ + List getUserGroupList(Collection ids); + + /** + * 获得指定状态的用户组列表 + * + * @param status 状态 + * @return 用户组列表 + */ + List getUserGroupListByStatus(Integer status); + + /** + * 获得用户组分页 + * + * @param pageReqVO 分页查询 + * @return 用户组分页 + */ + PageResult getUserGroupPage(BpmUserGroupPageReqVO pageReqVO); + + /** + * 校验用户组们是否有效。如下情况,视为无效: + * 1. 用户组编号不存在 + * 2. 用户组被禁用 + * + * @param ids 用户组编号数组 + */ + void validUserGroups(Set ids); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmFormServiceImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmFormServiceImpl.java new file mode 100644 index 00000000..eba49101 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmFormServiceImpl.java @@ -0,0 +1,137 @@ +package com.ruoyi.flowable.service.definition.impl; + +import cn.hutool.core.lang.Assert; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.utils.JsonUtils; +import com.ruoyi.common.utils.ValidationUtils; +import com.ruoyi.flowable.convert.definition.BpmFormConvert; +import com.ruoyi.flowable.core.enums.ErrorCodeConstants; +import com.ruoyi.flowable.core.enums.definition.BpmModelFormTypeEnum; +import com.ruoyi.flowable.domain.dto.definition.BpmFormFieldRespDTO; +import com.ruoyi.flowable.domain.dto.definition.BpmModelMetaInfoRespDTO; +import com.ruoyi.flowable.domain.entity.definition.BpmFormDO; +import com.ruoyi.flowable.domain.vo.form.BpmFormCreateReqVO; +import com.ruoyi.flowable.domain.vo.form.BpmFormPageReqVO; +import com.ruoyi.flowable.domain.vo.form.BpmFormUpdateReqVO; +import com.ruoyi.flowable.mapper.definition.BpmFormMapper; +import com.ruoyi.flowable.service.definition.BpmFormService; +import org.springframework.stereotype.Service; +import org.springframework.validation.annotation.Validated; + +import javax.annotation.Resource; +import java.util.*; + +import static com.ruoyi.common.exception.util.ServiceExceptionUtil.exception; +import static com.ruoyi.flowable.core.enums.ErrorCodeConstants.*; + + +/** + * 动态表单 Service 实现类 + * + * @author 风里雾里 + */ +@Service +@Validated +public class BpmFormServiceImpl implements BpmFormService { + + @Resource + private BpmFormMapper formMapper; + + @Override + public Long createForm(BpmFormCreateReqVO createReqVO) { + // 插入 + BpmFormDO form = BpmFormConvert.INSTANCE.convert(createReqVO); + formMapper.insert(form); + // 返回 + return form.getId(); + } + + @Override + public void updateForm(BpmFormUpdateReqVO updateReqVO) { + // 校验存在 + this.validateFormExists(updateReqVO.getId()); + // 更新 + BpmFormDO updateObj = BpmFormConvert.INSTANCE.convert(updateReqVO); + formMapper.updateById(updateObj); + } + + @Override + public void deleteForm(Long id) { + // 校验存在 + this.validateFormExists(id); + // 删除 + formMapper.deleteById(id); + } + + private void validateFormExists(Long id) { + if (formMapper.selectById(id) == null) { + throw exception(ErrorCodeConstants.FORM_NOT_EXISTS); + } + } + + @Override + public BpmFormDO getForm(Long id) { + return formMapper.selectById(id); + } + + @Override + public List getFormList() { + return formMapper.selectList(); + } + + @Override + public List getFormList(Collection ids) { + return formMapper.selectBatchIds(ids); + } + + @Override + public PageResult getFormPage(BpmFormPageReqVO pageReqVO) { + return formMapper.selectPage(pageReqVO); + } + + + @Override + public BpmFormDO checkFormConfig(String configStr) { + BpmModelMetaInfoRespDTO metaInfo = JsonUtils.parseObject(configStr, BpmModelMetaInfoRespDTO.class); + if (metaInfo == null || metaInfo.getFormType() == null) { + throw exception(MODEL_DEPLOY_FAIL_FORM_NOT_CONFIG); + } + // 校验表单存在 + if (Objects.equals(metaInfo.getFormType(), BpmModelFormTypeEnum.NORMAL.getType())) { + BpmFormDO form = getForm(metaInfo.getFormId()); + if (form == null) { + throw exception(FORM_NOT_EXISTS); + } + return form; + } + return null; + } + + private void checkKeyNCName(String key) { + if (!ValidationUtils.isXmlNCName(key)) { + throw exception(MODEL_KEY_VALID); + } + } + + /** + * 校验 Field,避免 field 重复 + * + * @param fields field 数组 + */ + private void checkFields(List fields) { + // key 是 vModel,value 是 label + Map fieldMap = new HashMap<>(); + for (String field : fields) { + BpmFormFieldRespDTO fieldDTO = JsonUtils.parseObject(field, BpmFormFieldRespDTO.class); + Assert.notNull(fieldDTO); + String oldLabel = fieldMap.put(fieldDTO.getVModel(), fieldDTO.getLabel()); + // 如果不存在,则直接返回 + if (oldLabel == null) { + continue; + } + // 如果存在,则报错 + throw exception(ErrorCodeConstants.FORM_FIELD_REPEAT, oldLabel, fieldDTO.getLabel(), fieldDTO.getVModel()); + } + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmModelServiceImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmModelServiceImpl.java new file mode 100644 index 00000000..c9d53d44 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmModelServiceImpl.java @@ -0,0 +1,311 @@ +package com.ruoyi.flowable.service.definition.impl; + +import cn.hutool.core.util.ArrayUtil; +import cn.hutool.core.util.StrUtil; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.utils.JsonUtils; +import com.ruoyi.common.utils.PageUtils; +import com.ruoyi.common.utils.ValidationUtils; +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.convert.definition.BpmModelConvert; +import com.ruoyi.flowable.core.enums.definition.BpmModelFormTypeEnum; +import com.ruoyi.flowable.domain.dto.definition.BpmModelMetaInfoRespDTO; +import com.ruoyi.flowable.domain.dto.definition.BpmProcessDefinitionCreateReqDTO; +import com.ruoyi.flowable.domain.entity.definition.BpmFormDO; +import com.ruoyi.flowable.domain.vo.model.*; +import com.ruoyi.flowable.service.definition.BpmFormService; +import com.ruoyi.flowable.service.definition.BpmModelService; +import com.ruoyi.flowable.service.definition.BpmProcessDefinitionService; +import com.ruoyi.flowable.service.definition.BpmTaskAssignRuleService; +import com.ruoyi.plugin.tenant.core.context.TenantContextHolder; +import lombok.extern.slf4j.Slf4j; +import org.flowable.bpmn.converter.BpmnXMLConverter; +import org.flowable.bpmn.model.BpmnModel; +import org.flowable.bpmn.model.FlowElement; +import org.flowable.bpmn.model.FlowNode; +import org.flowable.bpmn.model.Process; +import org.flowable.common.engine.impl.db.SuspensionState; +import org.flowable.common.engine.impl.util.io.BytesStreamSource; +import org.flowable.engine.RepositoryService; +import org.flowable.engine.repository.Deployment; +import org.flowable.engine.repository.Model; +import org.flowable.engine.repository.ModelQuery; +import org.flowable.engine.repository.ProcessDefinition; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; +import org.springframework.util.ObjectUtils; +import org.springframework.validation.annotation.Validated; + +import javax.annotation.Resource; +import javax.validation.Valid; +import java.util.*; + +import static com.ruoyi.common.exception.util.ServiceExceptionUtil.exception; +import static com.ruoyi.common.utils.collection.CollectionUtils.convertMap; +import static com.ruoyi.flowable.core.enums.ErrorCodeConstants.*; + +/** + * Flowable流程模型实现 + * 主要进行 Flowable {@link Model} 的维护 + */ +@Service +@Validated +@Slf4j +public class BpmModelServiceImpl implements BpmModelService { + + @Resource + private RepositoryService repositoryService; + @Resource + private BpmProcessDefinitionService processDefinitionService; + @Resource + private BpmFormService bpmFormService; + @Resource + private BpmTaskAssignRuleService taskAssignRuleService; + + @Override + public PageResult getModelPage(BpmModelPageReqVO pageVO) { + ModelQuery modelQuery = repositoryService.createModelQuery(); + if (StrUtil.isNotBlank(pageVO.getKey())) { + modelQuery.modelKey(pageVO.getKey()); + } + if (StrUtil.isNotBlank(pageVO.getName())) { + // 模糊匹配 + modelQuery.modelNameLike("%" + pageVO.getName() + "%"); + } + if (StrUtil.isNotBlank(pageVO.getCategory())) { + modelQuery.modelCategory(pageVO.getCategory()); + } + modelQuery.modelTenantId(TenantContextHolder.getTenantId().toString()); + // 执行查询 + List models = modelQuery.orderByCreateTime().desc() + .listPage(PageUtils.getStart(pageVO), pageVO.getPageSize()); + + // 获得 Form Map + Set formIds = CollectionUtils.convertSet(models, model -> { + BpmModelMetaInfoRespDTO metaInfo = JsonUtils.parseObject(model.getMetaInfo(), BpmModelMetaInfoRespDTO.class); + return metaInfo != null ? metaInfo.getFormId() : null; + }); + Map formMap = bpmFormService.getFormMap(formIds); + + // 获得 Deployment Map + Set deploymentIds = new HashSet<>(); + models.forEach(model -> CollectionUtils.addIfNotNull(deploymentIds, model.getDeploymentId())); + Map deploymentMap = processDefinitionService.getDeploymentMap(deploymentIds); + // 获得 ProcessDefinition Map + List processDefinitions = processDefinitionService.getProcessDefinitionListByDeploymentIds(deploymentIds); + Map processDefinitionMap = convertMap(processDefinitions, ProcessDefinition::getDeploymentId); + + // 拼接结果 + long modelCount = modelQuery.count(); + return new PageResult<>(BpmModelConvert.INSTANCE.convertList(models, formMap, deploymentMap, processDefinitionMap), modelCount); + } + + @Override + @Transactional(rollbackFor = Exception.class) + public String createModel(@Valid BpmModelCreateReqVO createReqVO, String bpmnXml) { + checkKeyNCName(createReqVO.getKey()); + // 校验流程标识已经存在666 + Model keyModel = getModelByKey(createReqVO.getKey()); + if (keyModel != null) { + throw exception(MODEL_KEY_EXISTS, createReqVO.getKey()); + } + + // 创建流程定义 + Model model = repositoryService.newModel(); + BpmModelConvert.INSTANCE.copy(model, createReqVO); + model.setTenantId(TenantContextHolder.getTenantId().toString()); + // 保存流程定义 + repositoryService.saveModel(model); + // 保存 BPMN XML + saveModelBpmnXml(model, bpmnXml); + return model.getId(); + } + + private Model getModelByKey(String key) { + return repositoryService.createModelQuery().modelKey(key).modelTenantId(TenantContextHolder.getTenantId().toString()).singleResult(); + } + + @Override + public BpmModelRespVO getModel(String id) { + Model model = repositoryService.getModel(id); + if (model == null) { + return null; + } + BpmModelRespVO modelRespVO = BpmModelConvert.INSTANCE.convert(model); + // 拼接 bpmn XML + byte[] bpmnBytes = repositoryService.getModelEditorSource(id); + modelRespVO.setBpmnXml(StrUtil.utf8Str(bpmnBytes)); + return modelRespVO; + } + + @Override + @Transactional(rollbackFor = Exception.class) // 因为进行多个操作,所以开启事务 + public void updateModel(@Valid BpmModelUpdateReqVO updateReqVO) { + // 校验流程模型存在 + Model model = repositoryService.getModel(updateReqVO.getId()); + if (model == null) { + throw exception(MODEL_NOT_EXISTS); + } + + // 修改流程定义 + BpmModelConvert.INSTANCE.copy(model, updateReqVO); + // 更新模型 + repositoryService.saveModel(model); + // 更新 BPMN XML + saveModelBpmnXml(model, updateReqVO.getBpmnXml()); + } + + @Override + @Transactional(rollbackFor = Exception.class) // 因为进行多个操作,所以开启事务 + public void deployModel(String id) { + // 1.1 校验流程模型存在 + Model model = repositoryService.getModel(id); + if (ObjectUtils.isEmpty(model)) { + throw exception(MODEL_NOT_EXISTS); + } + // 1.2 校验流程图 + // TODO 芋艿:校验流程图的有效性;例如说,是否有开始的元素,是否有结束的元素; + byte[] bpmnBytes = repositoryService.getModelEditorSource(model.getId()); + if (bpmnBytes == null) { + throw exception(MODEL_NOT_EXISTS); + } + // 1.3 校验表单已配 + BpmFormDO form = checkFormConfig(model.getMetaInfo()); + // 1.4 校验任务分配规则已配置 + taskAssignRuleService.checkTaskAssignRuleAllConfig(id); + + // 1.5 校验模型是否发生修改。如果未修改,则不允许创建 + BpmProcessDefinitionCreateReqDTO definitionCreateReqDTO = BpmModelConvert.INSTANCE.convert2(model, form).setBpmnBytes(bpmnBytes); + // 流程定义的信息相等 + if (processDefinitionService.isProcessDefinitionEquals(definitionCreateReqDTO)) { + ProcessDefinition oldProcessDefinition = processDefinitionService.getProcessDefinitionByDeploymentId(model.getDeploymentId()); + if (oldProcessDefinition != null && taskAssignRuleService.isTaskAssignRulesEquals(model.getId(), oldProcessDefinition.getId())) { + throw exception(MODEL_DEPLOY_FAIL_TASK_INFO_EQUALS); + } + } + + // 2.1 创建流程定义 + String definitionId = processDefinitionService.createProcessDefinition(definitionCreateReqDTO); + + // 2.2 将老的流程定义进行挂起。也就是说,只有最新部署的流程定义,才可以发起任务。 + updateProcessDefinitionSuspended(model.getDeploymentId()); + + // 2.3 更新 model 的 deploymentId,进行关联 + ProcessDefinition definition = processDefinitionService.getProcessDefinition(definitionId); + model.setDeploymentId(definition.getDeploymentId()); + repositoryService.saveModel(model); + + // 2.4 复制任务分配规则 + taskAssignRuleService.copyTaskAssignRules(id, definition.getId()); + } + + + @Override + @Transactional(rollbackFor = Exception.class) + public void deleteModel(String id) { + // 校验流程模型存在 + Model model = repositoryService.getModel(id); + if (model == null) { + throw exception(MODEL_NOT_EXISTS); + } + // 执行删除 + repositoryService.deleteModel(id); + // 禁用流程实例 + updateProcessDefinitionSuspended(model.getDeploymentId()); + } + + @Override + public void updateModelState(String id, Integer state) { + // 校验流程模型存在 + Model model = repositoryService.getModel(id); + if (model == null) { + throw exception(MODEL_NOT_EXISTS); + } + // 校验流程定义存在 + ProcessDefinition definition = processDefinitionService.getProcessDefinitionByDeploymentId(model.getDeploymentId()); + if (definition == null) { + throw exception(PROCESS_DEFINITION_NOT_EXISTS); + } + + // 更新状态 + processDefinitionService.updateProcessDefinitionState(definition.getId(), state); + } + + @Override + public BpmnModel getBpmnModel(String id) { + byte[] bpmnBytes = repositoryService.getModelEditorSource(id); + if (ArrayUtil.isEmpty(bpmnBytes)) { + return null; + } + BpmnXMLConverter converter = new BpmnXMLConverter(); + return converter.convertToBpmnModel(new BytesStreamSource(bpmnBytes), true, true); + } + + @Override + public FlowNode findFlowNodeByActivityId(String processDefId, String activityId) { + FlowNode activity = null; + BpmnModel bpmnModel = repositoryService.getBpmnModel(processDefId); + List processes = bpmnModel.getProcesses(); + for (Process process : processes) { + FlowElement flowElement = process.getFlowElementMap().get(activityId); + if (flowElement != null) { + activity = (FlowNode) flowElement; + break; + } + } + return activity; + } + + private void checkKeyNCName(String key) { + if (!ValidationUtils.isXmlNCName(key)) { + throw exception(MODEL_KEY_VALID); + } + } + + /** + * 校验流程表单已配置 + * + * @param metaInfoStr 流程模型 metaInfo 字段 + * @return 流程表单 + */ + private BpmFormDO checkFormConfig(String metaInfoStr) { + BpmModelMetaInfoRespDTO metaInfo = JsonUtils.parseObject(metaInfoStr, BpmModelMetaInfoRespDTO.class); + if (metaInfo == null || metaInfo.getFormType() == null) { + throw exception(MODEL_DEPLOY_FAIL_FORM_NOT_CONFIG); + } + // 校验表单存在 + if (Objects.equals(metaInfo.getFormType(), BpmModelFormTypeEnum.NORMAL.getType())) { + BpmFormDO form = bpmFormService.getForm(metaInfo.getFormId()); + if (form == null) { + throw exception(FORM_NOT_EXISTS); + } + return form; + } + return null; + } + + private void saveModelBpmnXml(Model model, String bpmnXml) { + if (StrUtil.isEmpty(bpmnXml)) { + return; + } + repositoryService.addModelEditorSource(model.getId(), StrUtil.utf8Bytes(bpmnXml)); + } + + /** + * 挂起 deploymentId 对应的流程定义。 这里一个deploymentId 只关联一个流程定义 + * + * @param deploymentId 流程发布Id. + */ + private void updateProcessDefinitionSuspended(String deploymentId) { + if (StrUtil.isEmpty(deploymentId)) { + return; + } + ProcessDefinition oldDefinition = processDefinitionService.getProcessDefinitionByDeploymentId(deploymentId); + if (oldDefinition == null) { + return; + } + processDefinitionService.updateProcessDefinitionState(oldDefinition.getId(), SuspensionState.SUSPENDED.getStateCode()); + } + + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmProcessDefinitionServiceImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmProcessDefinitionServiceImpl.java new file mode 100644 index 00000000..2d4536a2 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmProcessDefinitionServiceImpl.java @@ -0,0 +1,288 @@ +package com.ruoyi.flowable.service.definition.impl; + +import cn.hutool.core.collection.CollUtil; +import cn.hutool.core.util.ObjectUtil; +import cn.hutool.core.util.StrUtil; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.utils.PageUtils; +import com.ruoyi.flowable.convert.definition.BpmProcessDefinitionConvert; +import com.ruoyi.flowable.core.utils.FlowableUtils; +import com.ruoyi.flowable.domain.dto.definition.BpmProcessDefinitionCreateReqDTO; +import com.ruoyi.flowable.domain.entity.definition.BpmFormDO; +import com.ruoyi.flowable.domain.entity.definition.BpmProcessDefinitionExtDO; +import com.ruoyi.flowable.domain.vo.process.BpmProcessDefinitionListReqVO; +import com.ruoyi.flowable.domain.vo.process.BpmProcessDefinitionPageItemRespVO; +import com.ruoyi.flowable.domain.vo.process.BpmProcessDefinitionPageReqVO; +import com.ruoyi.flowable.domain.vo.process.BpmProcessDefinitionRespVO; +import com.ruoyi.flowable.mapper.definition.BpmProcessDefinitionExtMapper; +import com.ruoyi.flowable.service.definition.BpmFormService; +import com.ruoyi.flowable.service.definition.BpmProcessDefinitionService; +import com.ruoyi.plugin.tenant.core.context.TenantContextHolder; +import lombok.extern.slf4j.Slf4j; +import org.flowable.bpmn.converter.BpmnXMLConverter; +import org.flowable.bpmn.model.BpmnModel; +import org.flowable.common.engine.impl.db.SuspensionState; +import org.flowable.common.engine.impl.util.io.BytesStreamSource; +import org.flowable.engine.RepositoryService; +import org.flowable.engine.repository.Deployment; +import org.flowable.engine.repository.ProcessDefinition; +import org.flowable.engine.repository.ProcessDefinitionQuery; +import org.springframework.stereotype.Service; +import org.springframework.validation.annotation.Validated; + +import javax.annotation.Resource; +import javax.validation.Valid; +import java.util.*; + +import static com.ruoyi.common.exception.util.ServiceExceptionUtil.exception; +import static com.ruoyi.common.utils.collection.CollectionUtils.*; +import static com.ruoyi.flowable.core.enums.ErrorCodeConstants.PROCESS_DEFINITION_KEY_NOT_MATCH; +import static com.ruoyi.flowable.core.enums.ErrorCodeConstants.PROCESS_DEFINITION_NAME_NOT_MATCH; +import static java.util.Collections.emptyList; + +/** + * 流程定义实现 + * 主要进行 Flowable {@link ProcessDefinition} 和 {@link Deployment} 的维护 + * + * @author yunlongn + * @author ZJQ + * hasPermi + */ +@Service +@Validated +@Slf4j +public class BpmProcessDefinitionServiceImpl implements BpmProcessDefinitionService { + + private static final String BPMN_FILE_SUFFIX = ".bpmn"; + + @Resource + private RepositoryService repositoryService; + + @Resource + private BpmProcessDefinitionExtMapper processDefinitionMapper; + + @Resource + private BpmFormService formService; + + @Override + public ProcessDefinition getProcessDefinition(String id) { + return repositoryService.getProcessDefinition(id); + } + + @Override + public ProcessDefinition getProcessDefinition2(String id) { + return repositoryService.createProcessDefinitionQuery().processDefinitionId(id).singleResult(); + } + + @Override + public ProcessDefinition getProcessDefinitionByDeploymentId(String deploymentId) { + if (StrUtil.isEmpty(deploymentId)) { + return null; + } + return repositoryService.createProcessDefinitionQuery().deploymentId(deploymentId).singleResult(); + } + + @Override + public List getProcessDefinitionListByDeploymentIds(Set deploymentIds) { + if (CollUtil.isEmpty(deploymentIds)) { + return emptyList(); + } + return repositoryService.createProcessDefinitionQuery().deploymentIds(deploymentIds).list(); + } + + @Override + public ProcessDefinition getActiveProcessDefinition(String key) { + return repositoryService.createProcessDefinitionQuery().processDefinitionKey(key).active().singleResult(); + } + + @Override + public List getDeployments(Set ids) { + if (CollUtil.isEmpty(ids)) { + return emptyList(); + } + List list = new ArrayList<>(ids.size()); + for (String id : ids) { + addIfNotNull(list, getDeployment(id)); + } + return list; + } + + @Override + public Deployment getDeployment(String id) { + if (StrUtil.isEmpty(id)) { + return null; + } + return repositoryService.createDeploymentQuery().deploymentId(id).singleResult(); + } + + @Override + public BpmnModel getBpmnModel(String processDefinitionId) { + return repositoryService.getBpmnModel(processDefinitionId); + } + + @Override + public String createProcessDefinition(@Valid BpmProcessDefinitionCreateReqDTO createReqDTO) { + // 创建 Deployment 部署 + Deployment deploy = repositoryService.createDeployment() + .key(createReqDTO.getKey()).name(createReqDTO.getName()).category(createReqDTO.getCategory()).tenantId(TenantContextHolder.getTenantId().toString()) + .addBytes(createReqDTO.getKey() + BPMN_FILE_SUFFIX, createReqDTO.getBpmnBytes()) + .deploy(); + + // 设置 ProcessDefinition 的 category 分类 + ProcessDefinition definition = repositoryService.createProcessDefinitionQuery() + .deploymentId(deploy.getId()).singleResult(); + repositoryService.setProcessDefinitionCategory(definition.getId(), createReqDTO.getCategory()); + // 注意 1,ProcessDefinition 的 key 和 name 是通过 BPMN 中的 的 id 和 name 决定 + // 注意 2,目前该项目的设计上,需要保证 Model、Deployment、ProcessDefinition 使用相同的 key,保证关联性。 + // 否则,会导致 ProcessDefinition 的分页无法查询到。 + if (!Objects.equals(definition.getKey(), createReqDTO.getKey())) { + throw exception(PROCESS_DEFINITION_KEY_NOT_MATCH, createReqDTO.getKey(), definition.getKey()); + } + if (!Objects.equals(definition.getName(), createReqDTO.getName())) { + throw exception(PROCESS_DEFINITION_NAME_NOT_MATCH, createReqDTO.getName(), definition.getName()); + } + + // 插入拓展表 + BpmProcessDefinitionExtDO definitionDO = BpmProcessDefinitionConvert.INSTANCE.convert2(createReqDTO); + definitionDO.setProcessDefinitionId(definition.getId()); + processDefinitionMapper.insert(definitionDO); + return definition.getId(); + } + + @Override + public void updateProcessDefinitionState(String id, Integer state) { + // 激活 + if (Objects.equals(SuspensionState.ACTIVE.getStateCode(), state)) { + repositoryService.activateProcessDefinitionById(id, false, null); + return; + } + // 挂起 + if (Objects.equals(SuspensionState.SUSPENDED.getStateCode(), state)) { + // suspendProcessInstances = false,进行中的任务,不进行挂起。 + // 原因:只要新的流程不允许发起即可,老流程继续可以执行。 + repositoryService.suspendProcessDefinitionById(id, false, null); + return; + } + log.error("[updateProcessDefinitionState][流程定义({}) 修改未知状态({})]", id, state); + } + + @Override + public String getProcessDefinitionBpmnXML(String id) { + BpmnModel bpmnModel = repositoryService.getBpmnModel(id); + if (bpmnModel == null) { + return null; + } + BpmnXMLConverter converter = new BpmnXMLConverter(); + return StrUtil.utf8Str(converter.convertToXML(bpmnModel)); + } + + @Override + public boolean isProcessDefinitionEquals(@Valid BpmProcessDefinitionCreateReqDTO createReqDTO) { + // 校验 name、description 是否更新 + ProcessDefinition oldProcessDefinition = getActiveProcessDefinition(createReqDTO.getKey()); + if (oldProcessDefinition == null) { + return false; + } + BpmProcessDefinitionExtDO oldProcessDefinitionExt = getProcessDefinitionExt(oldProcessDefinition.getId()); + if (!StrUtil.equals(createReqDTO.getName(), oldProcessDefinition.getName()) + || !StrUtil.equals(createReqDTO.getDescription(), oldProcessDefinitionExt.getDescription()) + || !StrUtil.equals(createReqDTO.getCategory(), oldProcessDefinition.getCategory())) { + return false; + } + // 校验 form 信息是否更新 + if (!ObjectUtil.equal(createReqDTO.getFormType(), oldProcessDefinitionExt.getFormType()) + || !ObjectUtil.equal(createReqDTO.getFormId(), oldProcessDefinitionExt.getFormId()) + || !ObjectUtil.equal(createReqDTO.getFormConf(), oldProcessDefinitionExt.getFormConf()) + || !ObjectUtil.equal(createReqDTO.getFormFields(), oldProcessDefinitionExt.getFormFields()) + || !ObjectUtil.equal(createReqDTO.getFormCustomCreatePath(), oldProcessDefinitionExt.getFormCustomCreatePath()) + || !ObjectUtil.equal(createReqDTO.getFormCustomViewPath(), oldProcessDefinitionExt.getFormCustomViewPath())) { + return false; + } + // 校验 BPMN XML 信息 + BpmnModel newModel = buildBpmnModel(createReqDTO.getBpmnBytes()); + BpmnModel oldModel = getBpmnModel(oldProcessDefinition.getId()); + // TODO 貌似 flowable 不修改这个也不同。需要看看。 sourceSystemId 不同 + if (!FlowableUtils.equals(oldModel, newModel)) { + return false; + } + // 最终发现都一致,则返回 true + return true; + } + + /** + * 构建对应的 BPMN Model + * + * @param bpmnBytes 原始的 BPMN XML 字节数组 + * @return BPMN Model + */ + private BpmnModel buildBpmnModel(byte[] bpmnBytes) { + // 转换成 BpmnModel 对象 + BpmnXMLConverter converter = new BpmnXMLConverter(); + return converter.convertToBpmnModel(new BytesStreamSource(bpmnBytes), true, true); + } + + @Override + public BpmProcessDefinitionExtDO getProcessDefinitionExt(String id) { + return processDefinitionMapper.selectByProcessDefinitionId(id); + } + + @Override + public List getProcessDefinitionList(BpmProcessDefinitionListReqVO listReqVO) { + // 拼接查询条件 + ProcessDefinitionQuery definitionQuery = repositoryService.createProcessDefinitionQuery(); + if (Objects.equals(SuspensionState.SUSPENDED.getStateCode(), listReqVO.getSuspensionState())) { + definitionQuery.suspended(); + } else if (Objects.equals(SuspensionState.ACTIVE.getStateCode(), listReqVO.getSuspensionState())) { + definitionQuery.active(); + } + // 执行查询 + List processDefinitions = definitionQuery.list(); + if (CollUtil.isEmpty(processDefinitions)) { + return Collections.emptyList(); + } + + // 获得 BpmProcessDefinitionDO Map + List processDefinitionDOs = processDefinitionMapper.selectListByProcessDefinitionIds( + convertList(processDefinitions, ProcessDefinition::getId)); + Map processDefinitionDOMap = convertMap(processDefinitionDOs, + BpmProcessDefinitionExtDO::getProcessDefinitionId); + // 执行查询,并返回 + return BpmProcessDefinitionConvert.INSTANCE.convertList3(processDefinitions, processDefinitionDOMap); + } + + @Override + public PageResult getProcessDefinitionPage(BpmProcessDefinitionPageReqVO pageVO) { + ProcessDefinitionQuery definitionQuery = repositoryService.createProcessDefinitionQuery(); + if (StrUtil.isNotBlank(pageVO.getKey())) { + definitionQuery.processDefinitionKey(pageVO.getKey()); + } + + // 执行查询 + List processDefinitions = definitionQuery.orderByProcessDefinitionVersion().desc() + .listPage(PageUtils.getStart(pageVO), pageVO.getPageSize()); + + if (CollUtil.isEmpty(processDefinitions)) { + return new PageResult<>(emptyList(), definitionQuery.count()); + } + // 获得 Deployment Map + Set deploymentIds = new HashSet<>(); + processDefinitions.forEach(definition -> addIfNotNull(deploymentIds, definition.getDeploymentId())); + Map deploymentMap = getDeploymentMap(deploymentIds); + + // 获得 BpmProcessDefinitionDO Map + List processDefinitionDOs = processDefinitionMapper.selectListByProcessDefinitionIds( + convertList(processDefinitions, ProcessDefinition::getId)); + Map processDefinitionDOMap = convertMap(processDefinitionDOs, + BpmProcessDefinitionExtDO::getProcessDefinitionId); + + // 获得 Form Map + Set formIds = convertSet(processDefinitionDOs, BpmProcessDefinitionExtDO::getFormId); + Map formMap = formService.getFormMap(formIds); + + // 拼接结果 + long definitionCount = definitionQuery.count(); + return new PageResult<>(BpmProcessDefinitionConvert.INSTANCE.convertList(processDefinitions, deploymentMap, + processDefinitionDOMap, formMap), definitionCount); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmProcessInstanceApiImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmProcessInstanceApiImpl.java new file mode 100644 index 00000000..b0413e14 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmProcessInstanceApiImpl.java @@ -0,0 +1,29 @@ +package com.ruoyi.flowable.service.definition.impl; + +import com.ruoyi.flowable.domain.dto.task.BpmProcessInstanceCreateReqDTO; +import com.ruoyi.flowable.service.definition.BpmProcessInstanceApi; +import com.ruoyi.flowable.service.task.BpmProcessInstanceService; +import org.springframework.stereotype.Service; +import org.springframework.validation.annotation.Validated; + +import javax.annotation.Resource; +import javax.validation.Valid; + +/** + * Flowable 流程实例 Api 实现类 + * + * hasPermi + * @author jason + */ +@Service +@Validated +public class BpmProcessInstanceApiImpl implements BpmProcessInstanceApi { + + @Resource + private BpmProcessInstanceService processInstanceService; + + @Override + public String createProcessInstance(Long userId, @Valid BpmProcessInstanceCreateReqDTO reqDTO) { + return processInstanceService.createProcessInstance(userId, reqDTO); + } +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmTaskAssignRuleServiceImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmTaskAssignRuleServiceImpl.java new file mode 100644 index 00000000..297e5c30 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmTaskAssignRuleServiceImpl.java @@ -0,0 +1,361 @@ +package com.ruoyi.flowable.service.definition.impl; + +import cn.hutool.core.collection.CollUtil; +import cn.hutool.core.util.ObjectUtil; +import cn.hutool.core.util.StrUtil; +import cn.hutool.json.JSONArray; +import cn.hutool.json.JSONUtil; +import com.google.common.annotations.VisibleForTesting; +import com.ruoyi.common.enums.CommonStatusEnum; +import com.ruoyi.common.utils.ObjectUtils; +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.convert.definition.BpmTaskAssignRuleConvert; +import com.ruoyi.flowable.core.enums.DictTypeConstants; +import com.ruoyi.flowable.core.enums.definition.BpmTaskAssignRuleTypeEnum; +import com.ruoyi.flowable.core.utils.FlowableUtils; +import com.ruoyi.flowable.domain.dto.user.AdminUserRespDTO; +import com.ruoyi.flowable.domain.dto.user.DeptRespDTO; +import com.ruoyi.flowable.domain.entity.definition.BpmTaskAssignRuleDO; +import com.ruoyi.flowable.domain.entity.definition.BpmUserGroupDO; +import com.ruoyi.flowable.domain.vo.rule.BpmTaskAssignRuleCreateReqVO; +import com.ruoyi.flowable.domain.vo.rule.BpmTaskAssignRuleRespVO; +import com.ruoyi.flowable.domain.vo.rule.BpmTaskAssignRuleUpdateReqVO; +import com.ruoyi.flowable.framework.flowable.core.behavior.script.BpmTaskAssignScript; +import com.ruoyi.flowable.mapper.definition.BpmTaskAssignRuleMapper; +import com.ruoyi.flowable.service.definition.BpmModelService; +import com.ruoyi.flowable.service.definition.BpmProcessDefinitionService; +import com.ruoyi.flowable.service.definition.BpmTaskAssignRuleService; +import com.ruoyi.flowable.service.definition.BpmUserGroupService; +import com.ruoyi.flowable.service.user.*; +import lombok.extern.slf4j.Slf4j; +import org.flowable.bpmn.model.BpmnModel; +import org.flowable.bpmn.model.UserTask; +import org.flowable.common.engine.api.FlowableException; +import org.flowable.engine.delegate.DelegateExecution; +import org.springframework.context.annotation.Lazy; +import org.springframework.stereotype.Service; +import org.springframework.validation.annotation.Validated; + +import javax.annotation.Resource; +import javax.validation.Valid; +import java.util.*; + +import static cn.hutool.core.text.CharSequenceUtil.format; +import static com.ruoyi.common.exception.util.ServiceExceptionUtil.exception; +import static com.ruoyi.common.utils.JsonUtils.toJsonString; +import static com.ruoyi.common.utils.collection.CollectionUtils.convertSet; +import static com.ruoyi.flowable.core.enums.ErrorCodeConstants.*; + +/** + * BPM 任务分配规则 Service 实现类 + */ +@Service +@Validated +@Slf4j +public class BpmTaskAssignRuleServiceImpl implements BpmTaskAssignRuleService { + + @Resource + private BpmTaskAssignRuleMapper taskRuleMapper; + @Resource + @Lazy // 解决循环依赖 + private BpmModelService modelService; + @Resource + @Lazy // 解决循环依赖 + private BpmProcessDefinitionService processDefinitionService; + + @Resource + private BpmUserGroupService userGroupService; + @Resource + private AdminUserApi adminUserApi; + @Resource + private RoleApi roleApi; + @Resource + private DeptApi deptApi; + @Resource + private PostApi postApi; + @Resource + private DictDataApi dictDataApi; + @Resource + private PermissionApi permissionApi; + + /** + * 任务分配脚本 + */ + private Map scriptMap = Collections.emptyMap(); + + + @Override + public List getTaskAssignRuleListByProcessDefinitionId(String processDefinitionId, + String taskDefinitionKey) { + return taskRuleMapper.selectListByProcessDefinitionId(processDefinitionId, taskDefinitionKey); + } + + @Override + public List getTaskAssignRuleListByModelId(String modelId) { + return taskRuleMapper.selectListByModelId(modelId); + } + + @Override + public List getTaskAssignRuleList(String modelId, String processDefinitionId) { + // 获得规则 + List rules = Collections.emptyList(); + BpmnModel model = null; + if (StrUtil.isNotEmpty(modelId)) { + rules = getTaskAssignRuleListByModelId(modelId); + model = modelService.getBpmnModel(modelId); + } else if (StrUtil.isNotEmpty(processDefinitionId)) { + rules = getTaskAssignRuleListByProcessDefinitionId(processDefinitionId, null); + model = processDefinitionService.getBpmnModel(processDefinitionId); + } + if (model == null) { + return Collections.emptyList(); + } + // 获得用户任务,只有用户任务才可以设置分配规则 + List userTasks = FlowableUtils.getBpmnModelElements(model, UserTask.class); + if (CollUtil.isEmpty(userTasks)) { + return Collections.emptyList(); + } + // 转换数据 + return BpmTaskAssignRuleConvert.INSTANCE.convertList(userTasks, rules); + } + + @Override + public Long createTaskAssignRule(@Valid BpmTaskAssignRuleCreateReqVO reqVO) { + // 校验参数 + validTaskAssignRuleOptions(reqVO.getType(), reqVO.getOptions()); + // 校验是否已经配置 + BpmTaskAssignRuleDO existRule = + taskRuleMapper.selectListByModelIdAndTaskDefinitionKey(reqVO.getModelId(), reqVO.getTaskDefinitionKey()); + if (existRule != null) { + throw exception(TASK_ASSIGN_RULE_EXISTS, reqVO.getModelId(), reqVO.getTaskDefinitionKey()); + } + + // 存储 + BpmTaskAssignRuleDO rule = BpmTaskAssignRuleConvert.INSTANCE.convert(reqVO); + // 只有流程模型,才允许新建 + rule.setProcessDefinitionId(BpmTaskAssignRuleDO.PROCESS_DEFINITION_ID_NULL); + taskRuleMapper.insert(rule); + return rule.getId(); + } + + @Override + public void updateTaskAssignRule(@Valid BpmTaskAssignRuleUpdateReqVO reqVO) { + // 校验参数 + validTaskAssignRuleOptions(reqVO.getType(), reqVO.getOptions()); + // 校验是否存在 + BpmTaskAssignRuleDO existRule = taskRuleMapper.selectById(reqVO.getId()); + if (existRule == null) { + throw exception(TASK_ASSIGN_RULE_NOT_EXISTS); + } + // 只允许修改流程模型的规则 + if (!Objects.equals(BpmTaskAssignRuleDO.PROCESS_DEFINITION_ID_NULL, existRule.getProcessDefinitionId())) { + throw exception(TASK_UPDATE_FAIL_NOT_MODEL); + } + + // 执行更新 + taskRuleMapper.updateById(BpmTaskAssignRuleConvert.INSTANCE.convert(reqVO)); + } + + @Override + public boolean isTaskAssignRulesEquals(String modelId, String processDefinitionId) { + // 调用 VO 接口的原因是,过滤掉流程模型不需要的规则,保持和 copyTaskAssignRules 方法的一致性 + List modelRules = getTaskAssignRuleList(modelId, null); + List processInstanceRules = getTaskAssignRuleList(null, processDefinitionId); + if (modelRules.size() != processInstanceRules.size()) { + return false; + } + + // 遍历,匹配对应的规则 + Map processInstanceRuleMap = + CollectionUtils.convertMap(processInstanceRules, BpmTaskAssignRuleRespVO::getTaskDefinitionKey); + for (BpmTaskAssignRuleRespVO modelRule : modelRules) { + BpmTaskAssignRuleRespVO processInstanceRule = processInstanceRuleMap.get(modelRule.getTaskDefinitionKey()); + if (processInstanceRule == null) { + return false; + } + if (!ObjectUtil.equals(modelRule.getType(), processInstanceRule.getType()) || !ObjectUtil.equal( + modelRule.getOptions(), processInstanceRule.getOptions())) { + return false; + } + } + return true; + } + + @Override + public void copyTaskAssignRules(String fromModelId, String toProcessDefinitionId) { + List rules = getTaskAssignRuleList(fromModelId, null); + if (CollUtil.isEmpty(rules)) { + return; + } + // 开始复制 + List newRules = BpmTaskAssignRuleConvert.INSTANCE.convertList2(rules); + newRules.forEach(rule -> { + rule.setProcessDefinitionId(toProcessDefinitionId); + rule.setId(null); + rule.setCreateTime(null); + rule.setUpdateTime(null); + }); + taskRuleMapper.insertBatch(newRules); + } + + @Override + public void checkTaskAssignRuleAllConfig(String id) { + // 一个用户任务都没配置,所以无需配置规则 + List taskAssignRules = getTaskAssignRuleList(id, null); + if (CollUtil.isEmpty(taskAssignRules)) { + return; + } + // 校验未配置规则的任务 + taskAssignRules.forEach(rule -> { + if (CollUtil.isEmpty(rule.getOptions()) && !rule.getType().equals(60)) { + throw exception(MODEL_DEPLOY_FAIL_TASK_ASSIGN_RULE_NOT_CONFIG, rule.getTaskDefinitionName()); + } + }); + } + + private void validTaskAssignRuleOptions(Integer type, Set options) { + if (Objects.equals(type, BpmTaskAssignRuleTypeEnum.ROLE.getType())) { + roleApi.validRoles(options); + } else if (ObjectUtils.equalsAny(type, BpmTaskAssignRuleTypeEnum.DEPT_MEMBER.getType(), + BpmTaskAssignRuleTypeEnum.DEPT_LEADER.getType())) { + deptApi.validDepts(options); + } else if (Objects.equals(type, BpmTaskAssignRuleTypeEnum.POST.getType())) { + postApi.validPosts(options); + } else if (Objects.equals(type, BpmTaskAssignRuleTypeEnum.USER.getType())) { + adminUserApi.validUsers(options); + } else if (Objects.equals(type, BpmTaskAssignRuleTypeEnum.USER_GROUP.getType())) { + userGroupService.validUserGroups(options); + } else if (Objects.equals(type, BpmTaskAssignRuleTypeEnum.SCRIPT.getType())) { + dictDataApi.validDictDatas(DictTypeConstants.TASK_ASSIGN_SCRIPT, CollectionUtils.convertSet(options, String::valueOf)); + } else if (Objects.equals(type, BpmTaskAssignRuleTypeEnum.VARIABLE.getType())) { + //流程变量。 + } else { + throw new IllegalArgumentException(format("未知的规则类型({})", type)); + } + } + + @Override + //@DataPermission(enable = false) // 忽略数据权限,不然分配会存在问题 + public Set calculateTaskCandidateUsers(DelegateExecution execution) { + BpmTaskAssignRuleDO rule = getTaskRule(execution); + return calculateTaskCandidateUsers(execution, rule); + } + + @VisibleForTesting + BpmTaskAssignRuleDO getTaskRule(DelegateExecution execution) { + List taskRules = getTaskAssignRuleListByProcessDefinitionId( + execution.getProcessDefinitionId(), execution.getCurrentActivityId()); + if (CollUtil.isEmpty(taskRules)) { + throw new FlowableException(format("流程任务({}/{}/{}) 找不到符合的任务规则", + execution.getId(), execution.getProcessDefinitionId(), execution.getCurrentActivityId())); + } + if (taskRules.size() > 1) { + throw new FlowableException(format("流程任务({}/{}/{}) 找到过多任务规则({})", + execution.getId(), execution.getProcessDefinitionId(), execution.getCurrentActivityId())); + } + return taskRules.get(0); + } + + @VisibleForTesting + Set calculateTaskCandidateUsers(DelegateExecution execution, BpmTaskAssignRuleDO rule) { + Set assigneeUserIds = null; + if (Objects.equals(BpmTaskAssignRuleTypeEnum.ROLE.getType(), rule.getType())) { + assigneeUserIds = calculateTaskCandidateUsersByRole(rule); + } else if (Objects.equals(BpmTaskAssignRuleTypeEnum.DEPT_MEMBER.getType(), rule.getType())) { + assigneeUserIds = calculateTaskCandidateUsersByDeptMember(rule); + } else if (Objects.equals(BpmTaskAssignRuleTypeEnum.DEPT_LEADER.getType(), rule.getType())) { + assigneeUserIds = calculateTaskCandidateUsersByDeptLeader(rule); + } else if (Objects.equals(BpmTaskAssignRuleTypeEnum.POST.getType(), rule.getType())) { + assigneeUserIds = calculateTaskCandidateUsersByPost(rule); + } else if (Objects.equals(BpmTaskAssignRuleTypeEnum.USER.getType(), rule.getType())) { + assigneeUserIds = calculateTaskCandidateUsersByUser(rule); + } else if (Objects.equals(BpmTaskAssignRuleTypeEnum.USER_GROUP.getType(), rule.getType())) { + assigneeUserIds = calculateTaskCandidateUsersByUserGroup(rule); + } else if (Objects.equals(BpmTaskAssignRuleTypeEnum.SCRIPT.getType(), rule.getType())) { + assigneeUserIds = calculateTaskCandidateUsersByScript(execution, rule); + }else if (Objects.equals(BpmTaskAssignRuleTypeEnum.VARIABLE.getType(), rule.getType())) { + assigneeUserIds = calculateTaskCandidateUsersByVariable(execution, rule); + } + + // 移除被禁用的用户 + removeDisableUsers(assigneeUserIds); + // 如果候选人为空,抛出异常 + if (CollUtil.isEmpty(assigneeUserIds)) { + log.error("[calculateTaskCandidateUsers][流程任务({}/{}/{}) 任务规则({}) 找不到候选人]", execution.getId(), + execution.getProcessDefinitionId(), execution.getCurrentActivityId(), toJsonString(rule)); + throw exception(TASK_CREATE_FAIL_NO_CANDIDATE_USER); + } + return assigneeUserIds; + } + + private Set calculateTaskCandidateUsersByRole(BpmTaskAssignRuleDO rule) { + return permissionApi.getUserRoleIdListByRoleIds(rule.getOptions()); + } + + private Set calculateTaskCandidateUsersByDeptMember(BpmTaskAssignRuleDO rule) { + List users = adminUserApi.getUsersByDeptIds(rule.getOptions()); + return convertSet(users, AdminUserRespDTO::getId); + } + + private Set calculateTaskCandidateUsersByDeptLeader(BpmTaskAssignRuleDO rule) { + List depts = deptApi.getDepts(rule.getOptions()); + return convertSet(depts, DeptRespDTO::getLeaderUserId); + } + + private Set calculateTaskCandidateUsersByPost(BpmTaskAssignRuleDO rule) { + List users = adminUserApi.getUsersByPostIds(rule.getOptions()); + return convertSet(users, AdminUserRespDTO::getId); + } + + private Set calculateTaskCandidateUsersByUser(BpmTaskAssignRuleDO rule) { + return rule.getOptions(); + } + + private Set calculateTaskCandidateUsersByUserGroup(BpmTaskAssignRuleDO rule) { + List userGroups = userGroupService.getUserGroupList(rule.getOptions()); + Set userIds = new HashSet<>(); + userGroups.forEach(group -> userIds.addAll(group.getMemberUserIds())); + return userIds; + } + + private Set calculateTaskCandidateUsersByScript(DelegateExecution execution, BpmTaskAssignRuleDO rule) { + // 获得对应的脚本 + List scripts = new ArrayList<>(rule.getOptions().size()); + rule.getOptions().forEach(id -> { + BpmTaskAssignScript script = scriptMap.get(id); + if (script == null) { + throw exception(TASK_ASSIGN_SCRIPT_NOT_EXISTS, id); + } + scripts.add(script); + }); + // 逐个计算任务 + Set userIds = new HashSet<>(); + scripts.forEach(script -> CollUtil.addAll(userIds, script.calculateTaskCandidateUsers(execution))); + return userIds; + } + private Set calculateTaskCandidateUsersByVariable(DelegateExecution execution, BpmTaskAssignRuleDO rule) { + // 获得对应的脚本 + Object variable = execution.getVariable(FlowableUtils.formatCollectionVariable(execution.getCurrentActivityId())); + if (variable == null || StrUtil.isBlank(variable.toString())) { + return new HashSet<>(); + } + JSONArray jsonArray = JSONUtil.parseArray(variable.toString()); + Set ids = new HashSet<>(); + for (int i = 0; i < jsonArray.size(); i++) { + ids.add(Long.valueOf(jsonArray.get(i).toString())); + } + return ids; + } + @VisibleForTesting + void removeDisableUsers(Set assigneeUserIds) { + if (CollUtil.isEmpty(assigneeUserIds)) { + return; + } + Map userMap = adminUserApi.getUserMap(assigneeUserIds); + assigneeUserIds.removeIf(id -> { + AdminUserRespDTO user = userMap.get(id); + return user == null || !CommonStatusEnum.ENABLE.getStatus().equals(user.getStatus()); + }); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmUserGroupServiceImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmUserGroupServiceImpl.java new file mode 100644 index 00000000..15716203 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/definition/impl/BpmUserGroupServiceImpl.java @@ -0,0 +1,113 @@ +package com.ruoyi.flowable.service.definition.impl; + +import cn.hutool.core.collection.CollUtil; +import com.ruoyi.common.enums.CommonStatusEnum; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.convert.definition.BpmUserGroupConvert; +import com.ruoyi.flowable.domain.entity.definition.BpmUserGroupDO; +import com.ruoyi.flowable.domain.vo.group.BpmUserGroupCreateReqVO; +import com.ruoyi.flowable.domain.vo.group.BpmUserGroupPageReqVO; +import com.ruoyi.flowable.domain.vo.group.BpmUserGroupUpdateReqVO; +import com.ruoyi.flowable.mapper.definition.BpmUserGroupMapper; +import com.ruoyi.flowable.service.definition.BpmUserGroupService; +import org.springframework.stereotype.Service; +import org.springframework.validation.annotation.Validated; + +import javax.annotation.Resource; +import java.util.Collection; +import java.util.List; +import java.util.Map; +import java.util.Set; + +import static com.ruoyi.common.exception.util.ServiceExceptionUtil.exception; +import static com.ruoyi.flowable.core.enums.ErrorCodeConstants.USER_GROUP_IS_DISABLE; +import static com.ruoyi.flowable.core.enums.ErrorCodeConstants.USER_GROUP_NOT_EXISTS; + + +/** + * 用户组 Service 实现类 + *

    + * hasPermi + */ +@Service +@Validated +public class BpmUserGroupServiceImpl implements BpmUserGroupService { + + @Resource + private BpmUserGroupMapper userGroupMapper; + + @Override + public Long createUserGroup(BpmUserGroupCreateReqVO createReqVO) { + // 插入 + BpmUserGroupDO userGroup = BpmUserGroupConvert.INSTANCE.convert(createReqVO); + userGroupMapper.insert(userGroup); + // 返回 + return userGroup.getId(); + } + + @Override + public void updateUserGroup(BpmUserGroupUpdateReqVO updateReqVO) { + // 校验存在 + this.validateUserGroupExists(updateReqVO.getId()); + // 更新 + BpmUserGroupDO updateObj = BpmUserGroupConvert.INSTANCE.convert(updateReqVO); + userGroupMapper.updateById(updateObj); + } + + @Override + public void deleteUserGroup(Long id) { + // 校验存在 + this.validateUserGroupExists(id); + // 删除 + userGroupMapper.deleteById(id); + } + + private void validateUserGroupExists(Long id) { + if (userGroupMapper.selectById(id) == null) { + throw exception(USER_GROUP_NOT_EXISTS); + } + } + + @Override + public BpmUserGroupDO getUserGroup(Long id) { + return userGroupMapper.selectById(id); + } + + @Override + public List getUserGroupList(Collection ids) { + return userGroupMapper.selectBatchIds(ids); + } + + + @Override + public List getUserGroupListByStatus(Integer status) { + return userGroupMapper.selectListByStatus(status); + } + + @Override + public PageResult getUserGroupPage(BpmUserGroupPageReqVO pageReqVO) { + return userGroupMapper.selectPage(pageReqVO); + } + + @Override + public void validUserGroups(Set ids) { + if (CollUtil.isEmpty(ids)) { + return; + } + // 获得用户组信息 + List userGroups = userGroupMapper.selectBatchIds(ids); + Map userGroupMap = CollectionUtils.convertMap(userGroups, BpmUserGroupDO::getId); + // 校验 + ids.forEach(id -> { + BpmUserGroupDO userGroup = userGroupMap.get(id); + if (userGroup == null) { + throw exception(USER_GROUP_NOT_EXISTS); + } + if (!CommonStatusEnum.ENABLE.getStatus().equals(userGroup.getStatus())) { + throw exception(USER_GROUP_IS_DISABLE, userGroup.getName()); + } + }); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/message/BpmMessageService.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/message/BpmMessageService.java new file mode 100644 index 00000000..f1767dab --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/message/BpmMessageService.java @@ -0,0 +1,40 @@ +package com.ruoyi.flowable.service.message; + + +import com.ruoyi.flowable.domain.dto.message.BpmMessageSendWhenProcessInstanceApproveReqDTO; +import com.ruoyi.flowable.domain.dto.message.BpmMessageSendWhenProcessInstanceRejectReqDTO; +import com.ruoyi.flowable.domain.dto.message.BpmMessageSendWhenTaskCreatedReqDTO; + +import javax.validation.Valid; + +/** + * BPM 消息 Service 接口 + * + * TODO 芋艿:未来支持消息的可配置;不同的流程,在什么场景下,需要发送什么消息,消息的内容是什么; + * + * hasPermi + */ +public interface BpmMessageService { + + /** + * 发送流程实例被通过的消息 + * + * @param reqDTO 发送信息 + */ + void sendMessageWhenProcessInstanceApprove(@Valid BpmMessageSendWhenProcessInstanceApproveReqDTO reqDTO); + + /** + * 发送流程实例被不通过的消息 + * + * @param reqDTO 发送信息 + */ + void sendMessageWhenProcessInstanceReject(@Valid BpmMessageSendWhenProcessInstanceRejectReqDTO reqDTO); + + /** + * 发送任务被分配的消息 + * + * @param reqDTO 发送信息 + */ + void sendMessageWhenTaskAssigned(@Valid BpmMessageSendWhenTaskCreatedReqDTO reqDTO); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/message/impl/BpmMessageServiceImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/message/impl/BpmMessageServiceImpl.java new file mode 100644 index 00000000..449974a8 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/message/impl/BpmMessageServiceImpl.java @@ -0,0 +1,69 @@ +package com.ruoyi.flowable.service.message.impl; + +import com.ruoyi.flowable.convert.message.BpmMessageConvert; +import com.ruoyi.flowable.core.enums.message.BpmMessageEnum; +import com.ruoyi.flowable.domain.dto.message.BpmMessageSendWhenProcessInstanceApproveReqDTO; +import com.ruoyi.flowable.domain.dto.message.BpmMessageSendWhenProcessInstanceRejectReqDTO; +import com.ruoyi.flowable.domain.dto.message.BpmMessageSendWhenTaskCreatedReqDTO; +import com.ruoyi.flowable.service.message.BpmMessageService; +import lombok.extern.slf4j.Slf4j; +import org.springframework.boot.autoconfigure.web.WebProperties; +import org.springframework.stereotype.Service; +import org.springframework.validation.annotation.Validated; + +import javax.annotation.Resource; +import java.util.HashMap; +import java.util.Map; + +/** + * BPM 消息 Service 实现类 + * + * hasPermi + */ +@Service +@Validated +@Slf4j +public class BpmMessageServiceImpl implements BpmMessageService { + + //@Resource + //private SmsSendApi smsSendApi; + + @Resource + private WebProperties webProperties; + + @Override + public void sendMessageWhenProcessInstanceApprove(BpmMessageSendWhenProcessInstanceApproveReqDTO reqDTO) { + Map templateParams = new HashMap<>(); + templateParams.put("processInstanceName", reqDTO.getProcessInstanceName()); + templateParams.put("detailUrl", getProcessInstanceDetailUrl(reqDTO.getProcessInstanceId())); + //smsSendApi.sendSingleSmsToAdmin(BpmMessageConvert.INSTANCE.convert(reqDTO.getStartUserId(), + // BpmMessageEnum.PROCESS_INSTANCE_APPROVE.getSmsTemplateCode(), templateParams)); + } + + @Override + public void sendMessageWhenProcessInstanceReject(BpmMessageSendWhenProcessInstanceRejectReqDTO reqDTO) { + Map templateParams = new HashMap<>(); + templateParams.put("processInstanceName", reqDTO.getProcessInstanceName()); + templateParams.put("reason", reqDTO.getReason()); + templateParams.put("detailUrl", getProcessInstanceDetailUrl(reqDTO.getProcessInstanceId())); + //smsSendApi.sendSingleSmsToAdmin(BpmMessageConvert.INSTANCE.convert(reqDTO.getStartUserId(), + // BpmMessageEnum.PROCESS_INSTANCE_REJECT.getSmsTemplateCode(), templateParams)); + } + + @Override + public void sendMessageWhenTaskAssigned(BpmMessageSendWhenTaskCreatedReqDTO reqDTO) { + Map templateParams = new HashMap<>(); + templateParams.put("processInstanceName", reqDTO.getProcessInstanceName()); + templateParams.put("taskName", reqDTO.getTaskName()); + templateParams.put("startUserNickname", reqDTO.getStartUserNickname()); + templateParams.put("detailUrl", getProcessInstanceDetailUrl(reqDTO.getProcessInstanceId())); + //smsSendApi.sendSingleSmsToAdmin(BpmMessageConvert.INSTANCE.convert(reqDTO.getStartUserId(), + // BpmMessageEnum.TASK_ASSIGNED.getSmsTemplateCode(), templateParams)); + } + + private String getProcessInstanceDetailUrl(String taskId) { + //return webProperties.getAdminUi().getUrl() + "/bpm/process-instance/detail?id=" + taskId; + return null; + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/oa/BpmOALeaveService.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/oa/BpmOALeaveService.java new file mode 100644 index 00000000..7f3c7b45 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/oa/BpmOALeaveService.java @@ -0,0 +1,59 @@ +package com.ruoyi.flowable.service.oa; + +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.flowable.domain.entity.oa.BpmOALeaveDO; +import com.ruoyi.flowable.domain.vo.oa.BpmOALeaveCreateReqVO; +import com.ruoyi.flowable.domain.vo.oa.BpmOALeavePageReqVO; + +import javax.validation.Valid; + +/** + * 请假申请 Service 接口 + * + * @author jason + * hasPermi + */ +public interface BpmOALeaveService { + + /** + * 创建请假申请 + * + * @param userId 用户编号 + * @param createReqVO 创建信息 + * @return 编号 + */ + Long createLeave(Long userId, @Valid BpmOALeaveCreateReqVO createReqVO); + + /** + * 更新请假申请的状态 + * + * @param id 编号 + * @param result 结果 + */ + void updateLeaveResult(Long id, Integer result); + + /** + * 获得请假申请 + * + * @param id 编号 + * @return 请假申请 + */ + BpmOALeaveDO getLeave(Long id); + + /** + * 获得请假申请分页 + * + * @param userId 用户编号 + * @param pageReqVO 分页查询 + * @return 请假申请分页 + */ + PageResult getLeavePage(Long userId, BpmOALeavePageReqVO pageReqVO); + /** + * 分页查询OA 请假申请列表 + * + * @param bpmOaLeave OA 请假申请 + * @return OA 请假申请集合 + */ + public IPage selectList(IPage page, BpmOALeaveDO bpmOaLeave); +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/oa/BpmOALeaveServiceImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/oa/BpmOALeaveServiceImpl.java new file mode 100644 index 00000000..8b47612b --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/oa/BpmOALeaveServiceImpl.java @@ -0,0 +1,99 @@ +package com.ruoyi.flowable.service.oa; + +import cn.hutool.core.date.DateUtil; +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.flowable.convert.oa.BpmOALeaveConvert; +import com.ruoyi.flowable.core.enums.task.BpmProcessInstanceResultEnum; +import com.ruoyi.flowable.domain.dto.task.BpmProcessInstanceCreateReqDTO; +import com.ruoyi.flowable.domain.entity.oa.BpmOALeaveDO; +import com.ruoyi.flowable.domain.vo.oa.BpmOALeaveCreateReqVO; +import com.ruoyi.flowable.domain.vo.oa.BpmOALeavePageReqVO; +import com.ruoyi.flowable.mapper.oa.BpmOALeaveMapper; +import com.ruoyi.flowable.service.definition.BpmProcessInstanceApi; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; +import org.springframework.validation.annotation.Validated; + +import javax.annotation.Resource; +import java.util.HashMap; +import java.util.Map; + +import static com.ruoyi.common.exception.util.ServiceExceptionUtil.exception; +import static com.ruoyi.flowable.core.enums.ErrorCodeConstants.OA_LEAVE_NOT_EXISTS; + +/** + * OA 请假申请 Service 实现类 + * + * @author jason + * hasPermi + */ +@Service +@Validated +public class BpmOALeaveServiceImpl implements BpmOALeaveService { + + /** + * OA 请假对应的流程定义 KEY + */ + public static final String PROCESS_KEY = "oa_leave"; + + @Resource + private BpmOALeaveMapper leaveMapper; + + @Resource + private BpmProcessInstanceApi processInstanceApi; + + @Override + @Transactional(rollbackFor = Exception.class) + public Long createLeave(Long userId, BpmOALeaveCreateReqVO createReqVO) { + // 插入 OA 请假单 + long day = DateUtil.betweenDay(createReqVO.getStartTime(), createReqVO.getEndTime(), false); + BpmOALeaveDO leave = BpmOALeaveConvert.INSTANCE.convert(createReqVO).setUserId(userId).setDay(day) + .setResult(BpmProcessInstanceResultEnum.PROCESS.getResult()); + leaveMapper.insert(leave); + + // 发起 BPM 流程 + Map processInstanceVariables = new HashMap<>(); + processInstanceVariables.put("day", day); + String processInstanceId = processInstanceApi.createProcessInstance(userId, + new BpmProcessInstanceCreateReqDTO().setProcessDefinitionKey(PROCESS_KEY) + .setVariables(processInstanceVariables).setBusinessKey(String.valueOf(leave.getId()))); + + // 将工作流的编号,更新到 OA 请假单中 + leaveMapper.updateById(new BpmOALeaveDO().setId(leave.getId()).setProcessInstanceId(processInstanceId)); + return leave.getId(); + } + + @Override + public void updateLeaveResult(Long id, Integer result) { + validateLeaveExists(id); + leaveMapper.updateById(new BpmOALeaveDO().setId(id).setResult(result)); + } + + private void validateLeaveExists(Long id) { + if (leaveMapper.selectById(id) == null) { + throw exception(OA_LEAVE_NOT_EXISTS); + } + } + + @Override + public BpmOALeaveDO getLeave(Long id) { + return leaveMapper.selectById(id); + } + + @Override + public PageResult getLeavePage(Long userId, BpmOALeavePageReqVO pageReqVO) { + return leaveMapper.selectPage(userId, pageReqVO); + } + + /** + * 分页查询OA 请假申请列表 + * + * @param bpmOaLeave OA 请假申请 + * @return OA 请假申请 + */ + @Override + public IPage selectList(IPage page, BpmOALeaveDO bpmOaLeave) { + return leaveMapper.selectBpmOaLeavePage(page, bpmOaLeave); + } +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/oa/listener/BpmOALeaveResultListener.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/oa/listener/BpmOALeaveResultListener.java new file mode 100644 index 00000000..8723d233 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/oa/listener/BpmOALeaveResultListener.java @@ -0,0 +1,32 @@ +package com.ruoyi.flowable.service.oa.listener; + +import com.ruoyi.flowable.framework.bpm.core.event.BpmProcessInstanceResultEvent; +import com.ruoyi.flowable.framework.bpm.core.event.BpmProcessInstanceResultEventListener; +import com.ruoyi.flowable.service.oa.BpmOALeaveService; +import com.ruoyi.flowable.service.oa.BpmOALeaveServiceImpl; +import org.springframework.stereotype.Component; + +import javax.annotation.Resource; + +/** + * OA 请假单的结果的监听器实现类 + * + * hasPermi + */ +@Component +public class BpmOALeaveResultListener extends BpmProcessInstanceResultEventListener { + + @Resource + private BpmOALeaveService leaveService; + + @Override + protected String getProcessDefinitionKey() { + return BpmOALeaveServiceImpl.PROCESS_KEY; + } + + @Override + protected void onEvent(BpmProcessInstanceResultEvent event) { + leaveService.updateLeaveResult(Long.parseLong(event.getBusinessKey()), event.getResult()); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/BpmActivityService.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/BpmActivityService.java new file mode 100644 index 00000000..39d01a5a --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/BpmActivityService.java @@ -0,0 +1,31 @@ +package com.ruoyi.flowable.service.task; + +import com.ruoyi.flowable.domain.vo.activity.BpmActivityRespVO; +import org.flowable.engine.history.HistoricActivityInstance; + +import java.util.List; + +/** + * BPM 活动实例 Service 接口 + * + * hasPermi + */ +public interface BpmActivityService { + + /** + * 获得指定流程实例的活动实例列表 + * + * @param processInstanceId 流程实例的编号 + * @return 活动实例列表 + */ + List getActivityListByProcessInstanceId(String processInstanceId); + + /** + * 获得执行编号对应的活动实例 + * + * @param executionId 执行编号 + * @return 活动实例 + */ + List getHistoricActivityListByExecutionId(String executionId); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/BpmProcessInstanceService.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/BpmProcessInstanceService.java new file mode 100644 index 00000000..531e729d --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/BpmProcessInstanceService.java @@ -0,0 +1,147 @@ +package com.ruoyi.flowable.service.task; + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.domain.dto.task.BpmProcessInstanceCreateReqDTO; +import com.ruoyi.flowable.domain.vo.instance.*; +import org.flowable.engine.delegate.event.FlowableCancelledEvent; +import org.flowable.engine.history.HistoricProcessInstance; +import org.flowable.engine.runtime.ProcessInstance; + +import javax.validation.Valid; +import java.util.List; +import java.util.Map; +import java.util.Set; + +/** + * 流程实例 Service 接口 + * + * hasPermi + */ +public interface BpmProcessInstanceService { + + /** + * 获得流程实例 + * + * @param id 流程实例的编号 + * @return 流程实例 + */ + ProcessInstance getProcessInstance(String id); + + /** + * 获得流程实例列表 + * + * @param ids 流程实例的编号集合 + * @return 流程实例列表 + */ + List getProcessInstances(Set ids); + + /** + * 获得流程实例 Map + * + * @param ids 流程实例的编号集合 + * @return 流程实例列表 Map + */ + default Map getProcessInstanceMap(Set ids) { + return CollectionUtils.convertMap(getProcessInstances(ids), ProcessInstance::getProcessInstanceId); + } + + /** + * 获得流程实例的分页 + * + * @param userId 用户编号 + * @param pageReqVO 分页请求 + * @return 流程实例的分页 + */ + PageResult getMyProcessInstancePage(Long userId, + @Valid BpmProcessInstanceMyPageReqVO pageReqVO); + /** + * 创建流程实例(提供给前端) + * + * @param userId 用户编号 + * @param createReqVO 创建信息 + * @return 实例的编号 + */ + String createProcessInstance(Long userId, @Valid BpmProcessInstanceCreateReqVO createReqVO); + + /** + * 创建流程实例(提供给内部) + * + * @param userId 用户编号 + * @param createReqDTO 创建信息 + * @return 实例的编号 + */ + String createProcessInstance(Long userId, @Valid BpmProcessInstanceCreateReqDTO createReqDTO); + + /** + * 获得流程实例 VO 信息 + * + * @param id 流程实例的编号 + * @return 流程实例 + */ + BpmProcessInstanceRespVO getProcessInstanceVO(String id); + + /** + * 取消流程实例 + * + * @param userId 用户编号 + * @param cancelReqVO 取消信息 + */ + void cancelProcessInstance(Long userId, @Valid BpmProcessInstanceCancelReqVO cancelReqVO); + + /** + * 获得历史的流程实例 + * + * @param id 流程实例的编号 + * @return 历史的流程实例 + */ + HistoricProcessInstance getHistoricProcessInstance(String id); + + /** + * 获得历史的流程实例列表 + * + * @param ids 流程实例的编号集合 + * @return 历史的流程实例列表 + */ + List getHistoricProcessInstances(Set ids); + + /** + * 获得历史的流程实例 Map + * + * @param ids 流程实例的编号集合 + * @return 历史的流程实例列表 Map + */ + default Map getHistoricProcessInstanceMap(Set ids) { + return CollectionUtils.convertMap(getHistoricProcessInstances(ids), HistoricProcessInstance::getId); + } + + /** + * 创建 ProcessInstance 拓展记录 + * + * @param instance 流程任务 + */ + void createProcessInstanceExt(ProcessInstance instance); + + /** + * 更新 ProcessInstance 拓展记录为取消 + * + * @param event 流程取消事件 + */ + void updateProcessInstanceExtCancel(FlowableCancelledEvent event); + + /** + * 更新 ProcessInstance 拓展记录为完成 + * + * @param instance 流程任务 + */ + void updateProcessInstanceExtComplete(ProcessInstance instance); + + /** + * 更新 ProcessInstance 拓展记录为不通过 + * + * @param id 流程编号 + * @param reason 理由。例如说,审批不通过时,需要传递该值 + */ + void updateProcessInstanceExtReject(String id, String reason); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/BpmTaskService.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/BpmTaskService.java new file mode 100644 index 00000000..c8f750c4 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/BpmTaskService.java @@ -0,0 +1,147 @@ +package com.ruoyi.flowable.service.task; + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.domain.vo.task.*; +import org.flowable.task.api.Task; + +import javax.validation.Valid; +import java.util.List; +import java.util.Map; + +/** + * 流程任务实例 Service 接口 + * + * @author jason + * hasPermi + */ +public interface BpmTaskService { + + /** + * 获得待办的流程任务分页 + * + * @param userId 用户编号 + * @param pageReqVO 分页请求 + * + * @return 流程任务分页 + */ + PageResult getTodoTaskPage(Long userId, BpmTaskTodoPageReqVO pageReqVO); + + /** + * 获得已办的流程任务分页 + * + * @param userId 用户编号 + * @param pageReqVO 分页请求 + * + * @return 流程任务分页 + */ + PageResult getDoneTaskPage(Long userId, BpmTaskDonePageReqVO pageReqVO); + + /** + * 获得流程任务 Map + * + * @param processInstanceIds 流程实例的编号数组 + * + * @return 流程任务 Map + */ + default Map> getTaskMapByProcessInstanceIds(List processInstanceIds) { + return CollectionUtils.convertMultiMap(getTasksByProcessInstanceIds(processInstanceIds), + Task::getProcessInstanceId); + } + + /** + * 获得流程任务列表 + * + * @param processInstanceIds 流程实例的编号数组 + * + * @return 流程任务列表 + */ + List getTasksByProcessInstanceIds(List processInstanceIds); + + /** + * 获得指令流程实例的流程任务列表,包括所有状态的 + * + * @param processInstanceId 流程实例的编号 + * + * @return 流程任务列表 + */ + List getTaskListByProcessInstanceId(String processInstanceId); + + /** + * 通过任务 + * + * @param userId 用户编号 + * @param reqVO 通过请求 + */ + void approveTask(Long userId, @Valid BpmTaskApproveReqVO reqVO); + + /** + * 不通过任务 + * + * @param userId 用户编号 + * @param reqVO 不通过请求 + */ + void rejectTask(Long userId, @Valid BpmTaskRejectReqVO reqVO); + + /** + * 将流程任务分配给指定用户 + * + * @param userId 用户编号 + * @param reqVO 分配请求 + */ + void updateTaskAssignee(Long userId, BpmTaskUpdateAssigneeReqVO reqVO); + + /** + * 将流程任务分配给指定用户 + * + * @param id 流程任务编号 + * @param userId 用户编号 + */ + void updateTaskAssignee(String id, Long userId); + + /** + * 创建 Task 拓展记录 + * + * @param task 任务实体 + */ + void createTaskExt(Task task); + + /** + * 更新 Task 拓展记录为完成 + * + * @param task 任务实体 + */ + void updateTaskExtComplete(Task task); + + /** + * 更新 Task 拓展记录为已取消 + * + * @param taskId 任务的编号 + */ + void updateTaskExtCancel(String taskId); + + /** + * 更新 Task 拓展记录,并发送通知 + * + * @param task 任务实体 + */ + void updateTaskExtAssign(Task task); + + + /** + * 获取可驳回节点列表 + * @param taskId 任务id + * @param processInstanceId 流程实例id + * @return + */ + public List getBackNodesByProcessInstanceId(String processInstanceId,String taskId) ; + + /** + * 驳回任意节点 暂时没有考虑子流程 + * + * @param backTaskVo 参数 + * @return + */ + public Boolean backToStepTask(BackTaskVo backTaskVo); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/impl/BpmActivityServiceImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/impl/BpmActivityServiceImpl.java new file mode 100644 index 00000000..6472c416 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/impl/BpmActivityServiceImpl.java @@ -0,0 +1,41 @@ +package com.ruoyi.flowable.service.task.impl; + +import com.ruoyi.flowable.convert.task.BpmActivityConvert; +import com.ruoyi.flowable.domain.vo.activity.BpmActivityRespVO; +import com.ruoyi.flowable.service.task.BpmActivityService; +import lombok.extern.slf4j.Slf4j; +import org.flowable.engine.HistoryService; +import org.flowable.engine.history.HistoricActivityInstance; +import org.springframework.stereotype.Service; +import org.springframework.validation.annotation.Validated; + +import javax.annotation.Resource; +import java.util.List; + + +/** + * BPM 活动实例 Service 实现类 + * + * hasPermi + */ +@Service +@Slf4j +@Validated +public class BpmActivityServiceImpl implements BpmActivityService { + + @Resource + private HistoryService historyService; + + @Override + public List getActivityListByProcessInstanceId(String processInstanceId) { + List activityList = historyService.createHistoricActivityInstanceQuery() + .processInstanceId(processInstanceId).list(); + return BpmActivityConvert.INSTANCE.convertList(activityList); + } + + @Override + public List getHistoricActivityListByExecutionId(String executionId) { + return historyService.createHistoricActivityInstanceQuery().executionId(executionId).list(); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/impl/BpmProcessInstanceServiceImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/impl/BpmProcessInstanceServiceImpl.java new file mode 100644 index 00000000..e22cc9b2 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/impl/BpmProcessInstanceServiceImpl.java @@ -0,0 +1 @@ +package com.ruoyi.flowable.service.task.impl; import cn.hutool.core.collection.CollUtil; import cn.hutool.core.lang.Assert; import cn.hutool.core.util.StrUtil; import com.ruoyi.common.mybatis.pojo.PageResult; import com.ruoyi.common.utils.NumberUtils; import com.ruoyi.flowable.convert.task.BpmProcessInstanceConvert; import com.ruoyi.flowable.core.enums.task.BpmProcessInstanceDeleteReasonEnum; import com.ruoyi.flowable.core.enums.task.BpmProcessInstanceResultEnum; import com.ruoyi.flowable.core.enums.task.BpmProcessInstanceStatusEnum; import com.ruoyi.flowable.domain.dto.task.BpmProcessInstanceCreateReqDTO; import com.ruoyi.flowable.domain.dto.user.AdminUserRespDTO; import com.ruoyi.flowable.domain.dto.user.DeptRespDTO; import com.ruoyi.flowable.domain.entity.definition.BpmProcessDefinitionExtDO; import com.ruoyi.flowable.domain.entity.task.BpmProcessInstanceExtDO; import com.ruoyi.flowable.domain.vo.instance.*; import com.ruoyi.flowable.framework.bpm.core.event.BpmProcessInstanceResultEventPublisher; import com.ruoyi.flowable.mapper.task.BpmProcessInstanceExtMapper; import com.ruoyi.flowable.service.definition.BpmProcessDefinitionService; import com.ruoyi.flowable.service.message.BpmMessageService; import com.ruoyi.flowable.service.task.BpmProcessInstanceService; import com.ruoyi.flowable.service.task.BpmTaskService; import com.ruoyi.flowable.service.user.AdminUserApi; import com.ruoyi.flowable.service.user.DeptApi; import com.ruoyi.plugin.tenant.core.context.TenantContextHolder; import lombok.extern.slf4j.Slf4j; import org.flowable.engine.HistoryService; import org.flowable.engine.RuntimeService; import org.flowable.engine.delegate.event.FlowableCancelledEvent; import org.flowable.engine.history.HistoricProcessInstance; import org.flowable.engine.repository.ProcessDefinition; import org.flowable.engine.runtime.ProcessInstance; import org.flowable.engine.runtime.ProcessInstanceBuilder; import org.flowable.task.api.Task; import org.springframework.context.annotation.Lazy; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import org.springframework.validation.annotation.Validated; import javax.annotation.Resource; import javax.validation.Valid; import java.util.*; import static com.ruoyi.common.exception.util.ServiceExceptionUtil.exception; import static com.ruoyi.common.utils.collection.CollectionUtils.convertList; import static com.ruoyi.flowable.core.enums.ErrorCodeConstants.*; /** * 流程实例 Service 实现类 *

    * ProcessDefinition & ProcessInstance & Execution & Task 的关系: * 1. *

    * HistoricProcessInstance & ProcessInstance 的关系: * 1. *

    * 简单来说,前者 = 历史 + 运行中的流程实例,后者仅是运行中的流程实例 *

    * hasPermi */ @Service @Validated @Slf4j public class BpmProcessInstanceServiceImpl implements BpmProcessInstanceService { @Resource private RuntimeService runtimeService; @Resource private BpmProcessInstanceExtMapper processInstanceExtMapper; @Resource @Lazy // 解决循环依赖 private BpmTaskService taskService; @Resource private BpmProcessDefinitionService processDefinitionService; @Resource private HistoryService historyService; @Resource private AdminUserApi adminUserApi; @Resource private DeptApi deptApi; @Resource private BpmProcessInstanceResultEventPublisher processInstanceResultEventPublisher; @Resource private BpmMessageService messageService; @Override public ProcessInstance getProcessInstance(String id) { return runtimeService.createProcessInstanceQuery().processInstanceId(id).processInstanceTenantId(TenantContextHolder.getTenantId().toString()).singleResult(); } @Override public List getProcessInstances(Set ids) { return runtimeService.createProcessInstanceQuery().processInstanceIds(ids).processInstanceTenantId(TenantContextHolder.getTenantId().toString()).list(); } @Override public PageResult getMyProcessInstancePage(Long userId, BpmProcessInstanceMyPageReqVO pageReqVO) { // 通过 BpmProcessInstanceExtDO 表,先查询到对应的分页 PageResult pageResult = processInstanceExtMapper.selectPage(userId, pageReqVO); if (CollUtil.isEmpty(pageResult.getList())) { return new PageResult<>(pageResult.getTotal()); } // 获得流程 Task Map List processInstanceIds = convertList(pageResult.getList(), BpmProcessInstanceExtDO::getProcessInstanceId); Map> taskMap = taskService.getTaskMapByProcessInstanceIds(processInstanceIds); // 转换返回 return BpmProcessInstanceConvert.INSTANCE.convertPage(pageResult, taskMap); } @Override @Transactional(rollbackFor = Exception.class) public String createProcessInstance(Long userId, @Valid BpmProcessInstanceCreateReqVO createReqVO) { // 获得流程定义 ProcessDefinition definition = processDefinitionService.getProcessDefinition(createReqVO.getProcessDefinitionId()); // 发起流程 return createProcessInstance0(userId, definition, createReqVO.getVariables(), null); } @Override public String createProcessInstance(Long userId, @Valid BpmProcessInstanceCreateReqDTO createReqDTO) { // 获得流程定义 ProcessDefinition definition = processDefinitionService.getActiveProcessDefinition(createReqDTO.getProcessDefinitionKey()); // 发起流程 return createProcessInstance0(userId, definition, createReqDTO.getVariables(), createReqDTO.getBusinessKey()); } @Override public BpmProcessInstanceRespVO getProcessInstanceVO(String id) { // 获得流程实例 HistoricProcessInstance processInstance = getHistoricProcessInstance(id); if (processInstance == null) { return null; } BpmProcessInstanceExtDO processInstanceExt = processInstanceExtMapper.selectByProcessInstanceId(id); Assert.notNull(processInstanceExt, "流程实例拓展({}) 不存在", id); // 获得流程定义 ProcessDefinition processDefinition = processDefinitionService .getProcessDefinition(processInstance.getProcessDefinitionId()); Assert.notNull(processDefinition, "流程定义({}) 不存在", processInstance.getProcessDefinitionId()); BpmProcessDefinitionExtDO processDefinitionExt = processDefinitionService.getProcessDefinitionExt( processInstance.getProcessDefinitionId()); Assert.notNull(processDefinitionExt, "流程定义拓展({}) 不存在", id); String bpmnXml = processDefinitionService.getProcessDefinitionBpmnXML(processInstance.getProcessDefinitionId()); // 获得 User AdminUserRespDTO startUser = adminUserApi.getUser(NumberUtils.parseLong(processInstance.getStartUserId())); DeptRespDTO dept = null; if (startUser != null) { dept = deptApi.getDept(startUser.getDeptId()); } // 拼接结果 return BpmProcessInstanceConvert.INSTANCE.convert2(processInstance, processInstanceExt, processDefinition, processDefinitionExt, bpmnXml, startUser, dept); } @Override public void cancelProcessInstance(Long userId, @Valid BpmProcessInstanceCancelReqVO cancelReqVO) { // 校验流程实例存在 ProcessInstance instance = getProcessInstance(cancelReqVO.getId()); if (instance == null) { throw exception(PROCESS_INSTANCE_CANCEL_FAIL_NOT_EXISTS); } // 只能取消自己的 if (!Objects.equals(instance.getStartUserId(), String.valueOf(userId))) { throw exception(PROCESS_INSTANCE_CANCEL_FAIL_NOT_SELF); } // 通过删除流程实例,实现流程实例的取消, // 删除流程实例,正则执行任务 ACT_RU_TASK. 任务会被删除。通过历史表查询 deleteProcessInstance(cancelReqVO.getId(), BpmProcessInstanceDeleteReasonEnum.CANCEL_TASK.format(cancelReqVO.getReason())); } /** * 获得历史的流程实例 * * @param id 流程实例的编号 * @return 历史的流程实例 */ @Override public HistoricProcessInstance getHistoricProcessInstance(String id) { return historyService.createHistoricProcessInstanceQuery().processInstanceId(id).processInstanceTenantId(TenantContextHolder.getTenantId().toString()).singleResult(); } @Override public List getHistoricProcessInstances(Set ids) { return historyService.createHistoricProcessInstanceQuery().processInstanceIds(ids).processInstanceTenantId(TenantContextHolder.getTenantId().toString()).list(); } @Override public void createProcessInstanceExt(ProcessInstance instance) { // 获得流程定义 ProcessDefinition definition = processDefinitionService.getProcessDefinition2(instance.getProcessDefinitionId()); // 插入 BpmProcessInstanceExtDO 对象 BpmProcessInstanceExtDO instanceExtDO = new BpmProcessInstanceExtDO() .setProcessInstanceId(instance.getId()) .setProcessDefinitionId(definition.getId()) .setName(instance.getProcessDefinitionName()) .setStartUserId(Long.valueOf(instance.getStartUserId())) .setCategory(definition.getCategory()) .setStatus(BpmProcessInstanceStatusEnum.RUNNING.getStatus()) .setResult(BpmProcessInstanceResultEnum.PROCESS.getResult()); processInstanceExtMapper.insert(instanceExtDO); } @Override public void updateProcessInstanceExtCancel(FlowableCancelledEvent event) { // 判断是否为 Reject 不通过。如果是,则不进行更新. // 因为,updateProcessInstanceExtReject 方法,已经进行更新了 if (BpmProcessInstanceDeleteReasonEnum.isRejectReason((String) event.getCause())) { return; } // 需要主动查询,因为 instance 只有 id 属性 // 另外,此时如果去查询 ProcessInstance 的话,字段是不全的,所以去查询了 HistoricProcessInstance HistoricProcessInstance processInstance = getHistoricProcessInstance(event.getProcessInstanceId()); // 更新拓展表 BpmProcessInstanceExtDO instanceExtDO = new BpmProcessInstanceExtDO() .setProcessInstanceId(event.getProcessInstanceId()) // 由于 ProcessInstance 里没有办法拿到 endTime,所以这里设置 .setEndTime(new Date()) .setStatus(BpmProcessInstanceStatusEnum.FINISH.getStatus()) .setResult(BpmProcessInstanceResultEnum.CANCEL.getResult()); processInstanceExtMapper.updateByProcessInstanceId(instanceExtDO); // 发送流程实例的状态事件 processInstanceResultEventPublisher.sendProcessInstanceResultEvent( BpmProcessInstanceConvert.INSTANCE.convert(this, processInstance, instanceExtDO.getResult())); } @Override public void updateProcessInstanceExtComplete(ProcessInstance instance) { // 需要主动查询,因为 instance 只有 id 属性 // 另外,此时如果去查询 ProcessInstance 的话,字段是不全的,所以去查询了 HistoricProcessInstance HistoricProcessInstance processInstance = getHistoricProcessInstance(instance.getId()); // 更新拓展表 BpmProcessInstanceExtDO instanceExtDO = new BpmProcessInstanceExtDO() .setProcessInstanceId(instance.getProcessInstanceId()) // 由于 ProcessInstance 里没有办法拿到 endTime,所以这里设置 .setEndTime(new Date()) .setStatus(BpmProcessInstanceStatusEnum.FINISH.getStatus()) // 如果正常完全,说明审批通过 .setResult(BpmProcessInstanceResultEnum.APPROVE.getResult()); processInstanceExtMapper.updateByProcessInstanceId(instanceExtDO); // 发送流程被通过的消息 messageService.sendMessageWhenProcessInstanceApprove(BpmProcessInstanceConvert.INSTANCE.convert2ApprovedReq(instance)); // 发送流程实例的状态事件 processInstanceResultEventPublisher.sendProcessInstanceResultEvent( BpmProcessInstanceConvert.INSTANCE.convert(this, processInstance, instanceExtDO.getResult())); } @Override @Transactional(rollbackFor = Exception.class) public void updateProcessInstanceExtReject(String id, String reason) { // 需要主动查询,因为 instance 只有 id 属性 ProcessInstance processInstance = getProcessInstance(id); // 删除流程实例,以实现驳回任务时,取消整个审批流程 deleteProcessInstance(id, StrUtil.format(BpmProcessInstanceDeleteReasonEnum.REJECT_TASK.format(reason))); // 更新 status + result // 注意,不能和上面的逻辑更换位置。因为 deleteProcessInstance 会触发流程的取消,进而调用 updateProcessInstanceExtCancel 方法, // 设置 result 为 BpmProcessInstanceStatusEnum.CANCEL,显然和 result 不一定是一致的 BpmProcessInstanceExtDO instanceExtDO = new BpmProcessInstanceExtDO().setProcessInstanceId(id) .setStatus(BpmProcessInstanceStatusEnum.FINISH.getStatus()) .setResult(BpmProcessInstanceResultEnum.REJECT.getResult()); processInstanceExtMapper.updateByProcessInstanceId(instanceExtDO); // 发送流程被不通过的消息 messageService.sendMessageWhenProcessInstanceReject(BpmProcessInstanceConvert.INSTANCE.convert2RejectReq(processInstance, reason)); // 发送流程实例的状态事件 processInstanceResultEventPublisher.sendProcessInstanceResultEvent( BpmProcessInstanceConvert.INSTANCE.convert(this, processInstance, instanceExtDO.getResult())); } private void deleteProcessInstance(String id, String reason) { runtimeService.deleteProcessInstance(id, reason); } private String createProcessInstance0(Long userId, ProcessDefinition definition, Map variables, String businessKey) { // 校验流程定义 if (definition == null) { throw exception(PROCESS_DEFINITION_NOT_EXISTS); } if (definition.isSuspended()) { throw exception(PROCESS_DEFINITION_IS_SUSPENDED); } // 创建流程实例 ProcessInstanceBuilder processInstanceBuilder = runtimeService.createProcessInstanceBuilder() .processDefinitionId(definition.getId()) .businessKey(businessKey) .variables(variables) .tenantId(TenantContextHolder.getTenantId().toString()); ProcessInstance instance = processInstanceBuilder.start(); // ProcessInstance instance = runtimeService.startProcessInstanceById(definition.getId(), businessKey, variables); // 设置流程名字 runtimeService.setProcessInstanceName(instance.getId(), definition.getName()); // 补全流程实例的拓展表 processInstanceExtMapper.updateByProcessInstanceId(new BpmProcessInstanceExtDO().setProcessInstanceId(instance.getId()) .setFormVariables(variables)); return instance.getId(); } } \ No newline at end of file diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/impl/BpmTaskServiceImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/impl/BpmTaskServiceImpl.java new file mode 100644 index 00000000..cfb72a83 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/task/impl/BpmTaskServiceImpl.java @@ -0,0 +1,511 @@ +package com.ruoyi.flowable.service.task.impl; + +import cn.hutool.core.collection.CollUtil; +import cn.hutool.core.util.StrUtil; +import com.baomidou.mybatisplus.core.toolkit.CollectionUtils; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.utils.NumberUtils; +import com.ruoyi.common.utils.PageUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.flowable.convert.task.BpmTaskConvert; +import com.ruoyi.flowable.core.enums.definition.BpmTaskRuleScriptEnum; +import com.ruoyi.flowable.core.enums.task.BpmProcessInstanceDeleteReasonEnum; +import com.ruoyi.flowable.core.enums.task.BpmProcessInstanceResultEnum; +import com.ruoyi.flowable.domain.dto.user.AdminUserRespDTO; +import com.ruoyi.flowable.domain.dto.user.DeptRespDTO; +import com.ruoyi.flowable.domain.entity.task.BpmTaskExtDO; +import com.ruoyi.flowable.domain.vo.task.*; +import com.ruoyi.flowable.mapper.task.BpmTaskExtMapper; +import com.ruoyi.flowable.service.definition.BpmModelService; +import com.ruoyi.flowable.service.message.BpmMessageService; +import com.ruoyi.flowable.service.task.BpmProcessInstanceService; +import com.ruoyi.flowable.service.task.BpmTaskService; +import com.ruoyi.flowable.service.user.AdminUserApi; +import com.ruoyi.flowable.service.user.DeptApi; +import com.ruoyi.plugin.tenant.core.context.TenantContextHolder; +import lombok.extern.slf4j.Slf4j; +import org.apache.commons.collections4.MapUtils; +import org.flowable.bpmn.constants.BpmnXMLConstants; +import org.flowable.bpmn.model.FlowNode; +import org.flowable.engine.HistoryService; +import org.flowable.engine.IdentityService; +import org.flowable.engine.RuntimeService; +import org.flowable.engine.TaskService; +import org.flowable.engine.history.HistoricProcessInstance; +import org.flowable.engine.runtime.ActivityInstance; +import org.flowable.engine.runtime.ProcessInstance; +import org.flowable.idm.api.User; +import org.flowable.task.api.Task; +import org.flowable.task.api.TaskQuery; +import org.flowable.task.api.history.HistoricTaskInstance; +import org.flowable.task.api.history.HistoricTaskInstanceQuery; +import org.flowable.task.service.impl.persistence.entity.TaskEntity; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; +import org.springframework.transaction.support.TransactionSynchronization; +import org.springframework.transaction.support.TransactionSynchronizationManager; + +import javax.annotation.Resource; +import javax.validation.Valid; +import java.util.*; +import java.util.stream.Collectors; + +import static com.ruoyi.common.exception.util.ServiceExceptionUtil.exception; +import static com.ruoyi.common.utils.collection.CollectionUtils.convertMap; +import static com.ruoyi.common.utils.collection.CollectionUtils.convertSet; +import static com.ruoyi.flowable.core.enums.ErrorCodeConstants.*; + + +/** + * 流程任务实例 Service 实现类 + *

    + * hasPermi + * + * @author jason + */ +@Slf4j +@Service +public class BpmTaskServiceImpl implements BpmTaskService { + + @Resource + private RuntimeService runtimeService; + @Resource + private TaskService taskService; + @Resource + private HistoryService historyService; + + @Resource + private BpmProcessInstanceService processInstanceService; + + @Resource + private BpmModelService modelService; + + @Resource + protected IdentityService identityService; + + @Resource + private AdminUserApi adminUserApi; + + @Resource + private DeptApi deptApi; + + @Resource + private BpmTaskExtMapper taskExtMapper; + @Resource + private BpmMessageService messageService; + + @Override + public PageResult getTodoTaskPage(Long userId, BpmTaskTodoPageReqVO pageVO) { + // 查询待办任务 + TaskQuery taskQuery = taskService.createTaskQuery().taskAssignee(String.valueOf(userId)).taskTenantId(TenantContextHolder.getTenantId().toString()) // 分配给自己 + .orderByTaskCreateTime().desc(); // 创建时间倒序 + if (StrUtil.isNotBlank(pageVO.getName())) { + taskQuery.taskNameLike("%" + pageVO.getName() + "%"); + } + if (pageVO.getBeginCreateTime() != null) { + taskQuery.taskCreatedAfter(pageVO.getBeginCreateTime()); + } + if (pageVO.getEndCreateTime() != null) { + taskQuery.taskCreatedBefore(pageVO.getEndCreateTime()); + } + // 执行查询 + List tasks = taskQuery.listPage(PageUtils.getStart(pageVO), pageVO.getPageSize()); + if (CollUtil.isEmpty(tasks)) { + return PageResult.empty(taskQuery.count()); + } + + // 获得 ProcessInstance Map + Map processInstanceMap = + processInstanceService.getProcessInstanceMap(convertSet(tasks, Task::getProcessInstanceId)); + // 获得 User Map + Map userMap = adminUserApi.getUserMap(convertSet(processInstanceMap.values(), instance -> Long.valueOf(instance.getStartUserId()))); + // 拼接结果 + return new PageResult<>(BpmTaskConvert.INSTANCE.convertList1(tasks, processInstanceMap, userMap), taskQuery.count()); + } + + @Override + public PageResult getDoneTaskPage(Long userId, BpmTaskDonePageReqVO pageVO) { + // 查询已办任务 + // 已完成 + HistoricTaskInstanceQuery taskQuery = historyService.createHistoricTaskInstanceQuery().finished() + // 分配给自己 + .taskAssignee(String.valueOf(userId)) + // 审批时间倒序 + .orderByHistoricTaskInstanceEndTime().desc(); + if (StrUtil.isNotBlank(pageVO.getName())) { + taskQuery.taskNameLike("%" + pageVO.getName() + "%"); + } + if (pageVO.getBeginCreateTime() != null) { + taskQuery.taskCreatedAfter(pageVO.getBeginCreateTime()); + } + if (pageVO.getEndCreateTime() != null) { + taskQuery.taskCreatedBefore(pageVO.getEndCreateTime()); + } + // 执行查询 + List tasks = taskQuery.listPage(PageUtils.getStart(pageVO), pageVO.getPageSize()); + if (CollUtil.isEmpty(tasks)) { + return PageResult.empty(taskQuery.count()); + } + + // 获得 TaskExtDO Map + List bpmTaskExtDOs = + taskExtMapper.selectListByTaskIds(convertSet(tasks, HistoricTaskInstance::getId)); + Map bpmTaskExtDOMap = convertMap(bpmTaskExtDOs, BpmTaskExtDO::getTaskId); + // 获得 ProcessInstance Map + Map historicProcessInstanceMap = + processInstanceService.getHistoricProcessInstanceMap( + convertSet(tasks, HistoricTaskInstance::getProcessInstanceId)); + // 获得 User Map + Map userMap = adminUserApi.getUserMap(convertSet(historicProcessInstanceMap.values(), instance -> Long.valueOf(instance.getStartUserId()))); + // 拼接结果 + return new PageResult<>(BpmTaskConvert.INSTANCE.convertList2(tasks, bpmTaskExtDOMap, historicProcessInstanceMap, userMap), taskQuery.count()); + } + + @Override + public List getTasksByProcessInstanceIds(List processInstanceIds) { + if (CollUtil.isEmpty(processInstanceIds)) { + return Collections.emptyList(); + } + return taskService.createTaskQuery().processInstanceIdIn(processInstanceIds).taskTenantId(TenantContextHolder.getTenantId().toString()).list(); + } + + @Override + public List getTaskListByProcessInstanceId(String processInstanceId) { + // 获得任务列表 + List tasks = historyService.createHistoricTaskInstanceQuery() + .processInstanceId(processInstanceId) + .orderByHistoricTaskInstanceStartTime().desc() // 创建时间倒序 + .list(); + if (CollUtil.isEmpty(tasks)) { + return Collections.emptyList(); + } + + // 获得 TaskExtDO Map + List bpmTaskExtDOs = taskExtMapper.selectListByTaskIds(convertSet(tasks, HistoricTaskInstance::getId)); + Map bpmTaskExtDOMap = convertMap(bpmTaskExtDOs, BpmTaskExtDO::getTaskId); + // 获得 ProcessInstance Map + HistoricProcessInstance processInstance = processInstanceService.getHistoricProcessInstance(processInstanceId); + // 获得 User Map + Set userIds = convertSet(tasks, task -> NumberUtils.parseLong(task.getAssignee())); + userIds.add(NumberUtils.parseLong(processInstance.getStartUserId())); + Map userMap = adminUserApi.getUserMap(userIds); + // 获得 Dept Map + Map deptMap = deptApi.getDeptMap(convertSet(userMap.values(), AdminUserRespDTO::getDeptId)); + + // 拼接数据 + return BpmTaskConvert.INSTANCE.convertList3(tasks, bpmTaskExtDOMap, processInstance, userMap, deptMap); + } + + @Override + @Transactional(rollbackFor = Exception.class) + public void approveTask(Long userId, @Valid BpmTaskApproveReqVO reqVO) { + // 校验任务存在 + Task task = checkTask(userId, reqVO.getId()); + // 校验流程实例存在 + ProcessInstance instance = processInstanceService.getProcessInstance(task.getProcessInstanceId()); + if (instance == null) { + throw exception(PROCESS_INSTANCE_NOT_EXISTS); + } + + // 完成任务,审批通过 + taskService.complete(task.getId(), instance.getProcessVariables()); + + // 更新任务拓展表为通过 + taskExtMapper.updateByTaskId( + new BpmTaskExtDO().setTaskId(task.getId()).setResult(BpmProcessInstanceResultEnum.APPROVE.getResult()) + .setReason(reqVO.getReason())); + } + + @Override + @Transactional(rollbackFor = Exception.class) + public void rejectTask(Long userId, @Valid BpmTaskRejectReqVO reqVO) { + Task task = checkTask(userId, reqVO.getId()); + // 校验流程实例存在 + ProcessInstance instance = processInstanceService.getProcessInstance(task.getProcessInstanceId()); + if (instance == null) { + throw exception(PROCESS_INSTANCE_NOT_EXISTS); + } + + // 更新流程实例为不通过 + processInstanceService.updateProcessInstanceExtReject(instance.getProcessInstanceId(), reqVO.getReason()); + + // 更新任务拓展表为不通过 + taskExtMapper.updateByTaskId( + new BpmTaskExtDO().setTaskId(task.getId()).setResult(BpmProcessInstanceResultEnum.REJECT.getResult()) + .setEndTime(new Date()).setReason(reqVO.getReason())); + } + + @Override + public void updateTaskAssignee(Long userId, BpmTaskUpdateAssigneeReqVO reqVO) { + // 校验任务存在 + Task task = checkTask(userId, reqVO.getId()); + // 更新负责人 + updateTaskAssignee(task.getId(), reqVO.getAssigneeUserId()); + } + + @Override + public void updateTaskAssignee(String id, Long userId) { + taskService.setAssignee(id, String.valueOf(userId)); + } + + /** + * 校验任务是否存在, 并且是否是分配给自己的任务 + * + * @param userId 用户 id + * @param taskId task id + */ + private Task checkTask(Long userId, String taskId) { + Task task = getTask(taskId); + if (task == null) { + throw exception(TASK_COMPLETE_FAIL_NOT_EXISTS); + } + if (!Objects.equals(userId, NumberUtils.parseLong(task.getAssignee()))) { + throw exception(TASK_COMPLETE_FAIL_ASSIGN_NOT_SELF); + } + return task; + } + + @Override + public void createTaskExt(Task task) { + BpmTaskExtDO taskExtDO = + BpmTaskConvert.INSTANCE.convert2TaskExt(task).setResult(BpmProcessInstanceResultEnum.PROCESS.getResult()); + taskExtMapper.insert(taskExtDO); + } + + @Override + public void updateTaskExtComplete(Task task) { + BpmTaskExtDO taskExtDO = BpmTaskConvert.INSTANCE.convert2TaskExt(task) + // 不设置也问题不大,因为 Complete 一般是审核通过,已经设置 + .setResult(BpmProcessInstanceResultEnum.APPROVE.getResult()) + .setEndTime(new Date()); + taskExtMapper.updateByTaskId(taskExtDO); + } + + @Override + public void updateTaskExtCancel(String taskId) { + // 需要在事务提交后,才进行查询。不然查询不到历史的原因 + TransactionSynchronizationManager.registerSynchronization(new TransactionSynchronization() { + + @Override + public void afterCommit() { + // 可能只是活动,不是任务,所以查询不到 + HistoricTaskInstance task = getHistoricTask(taskId); + if (task == null) { + return; + } + // 如果任务拓展表已经是完成的状态,则跳过 + BpmTaskExtDO taskExt = taskExtMapper.selectByTaskId(taskId); + if (taskExt == null) { + log.error("[updateTaskExtCancel][taskId({}) 查找不到对应的记录,可能存在问题]", taskId); + return; + } + // 如果已经是最终的结果,则跳过 + if (BpmProcessInstanceResultEnum.isEndResult(taskExt.getResult())) { + log.error("[updateTaskExtCancel][taskId({}) 处于结果({}),无需进行更新]", taskId, taskExt.getResult()); + return; + } + + // 更新任务 + taskExtMapper.updateById(new BpmTaskExtDO().setId(taskExt.getId()).setResult(BpmProcessInstanceResultEnum.CANCEL.getResult()) + .setEndTime(new Date()).setReason(BpmProcessInstanceDeleteReasonEnum.translateReason(task.getDeleteReason()))); + } + + }); + } + + @Override + public void updateTaskExtAssign(Task task) { + BpmTaskExtDO taskExtDO = + new BpmTaskExtDO().setAssigneeUserId(NumberUtils.parseLong(task.getAssignee())).setTaskId(task.getId()); + taskExtMapper.updateByTaskId(taskExtDO); + // 发送通知。在事务提交时,批量执行操作,所以直接查询会无法查询到 ProcessInstance,所以这里是通过监听事务的提交来实现。 + TransactionSynchronizationManager.registerSynchronization(new TransactionSynchronization() { + @Override + public void afterCommit() { + ProcessInstance processInstance = + processInstanceService.getProcessInstance(task.getProcessInstanceId()); + // TODO: 2022/8/1 + AdminUserRespDTO startUser = adminUserApi.getUser(Long.valueOf(processInstance.getStartUserId())); + messageService.sendMessageWhenTaskAssigned(BpmTaskConvert.INSTANCE.convert(processInstance, startUser, task)); + } + }); + } + + @Override + public List getBackNodesByProcessInstanceId(String processInstanceId, String taskId) { + List backNods = new ArrayList<>(); + TaskEntity taskEntity = (TaskEntity) taskService.createTaskQuery().taskId(taskId).taskTenantId(TenantContextHolder.getTenantId().toString()).singleResult(); + String currActId = taskEntity.getTaskDefinitionKey(); + //获取运行节点表中usertask + String sql = "select t.* from act_ru_actinst t where t.ACT_TYPE_ = 'userTask' " + + " and t.PROC_INST_ID_=#{processInstanceId} and t.END_TIME_ is not null "; + List activityInstances = runtimeService.createNativeActivityInstanceQuery().sql(sql) + .parameter("processInstanceId", processInstanceId) + .list(); + //获取运行节点表的parallelGateway节点并出重 + sql = "SELECT t.ID_, t.REV_,t.PROC_DEF_ID_,t.PROC_INST_ID_,t.EXECUTION_ID_,t.ACT_ID_, t.TASK_ID_, t.CALL_PROC_INST_ID_, t.ACT_NAME_, t.ACT_TYPE_, " + + " t.ASSIGNEE_, t.START_TIME_, max(t.END_TIME_) as END_TIME_, t.DURATION_, t.DELETE_REASON_, t.TENANT_ID_" + + " FROM act_ru_actinst t WHERE t.ACT_TYPE_ = 'parallelGateway' AND t.PROC_INST_ID_ = #{processInstanceId} and t.END_TIME_ is not null" + + " and t.ACT_ID_ <> #{actId} GROUP BY t.act_id_"; + List parallelGatewaies = runtimeService.createNativeActivityInstanceQuery().sql(sql) + .parameter("processInstanceId", processInstanceId) + .parameter("actId", currActId) + .list(); + //排序 + if (CollectionUtils.isNotEmpty(parallelGatewaies)) { + activityInstances.addAll(parallelGatewaies); + activityInstances.sort(Comparator.comparing(ActivityInstance::getEndTime)); + } + //分组节点 + int count = 0; + Map> parallelGatewayUserTasks = new HashMap<>(); + List userTasks = new ArrayList<>(); + ActivityInstance currActivityInstance = null; + for (ActivityInstance activityInstance : activityInstances) { + if (BpmnXMLConstants.ELEMENT_GATEWAY_PARALLEL.equals(activityInstance.getActivityType())) { + count++; + if (count % 2 != 0) { + List datas = new ArrayList<>(); + currActivityInstance = activityInstance; + parallelGatewayUserTasks.put(currActivityInstance, datas); + } + } + if (BpmnXMLConstants.ELEMENT_TASK_USER.equals(activityInstance.getActivityType())) { + if (count % 2 == 0) { + userTasks.add(activityInstance); + } else { + if (parallelGatewayUserTasks.containsKey(currActivityInstance)) { + parallelGatewayUserTasks.get(currActivityInstance).add(activityInstance); + } + } + } + } + //组装人员名称 + List historicTaskInstances = historyService.createHistoricTaskInstanceQuery() + .processInstanceId(processInstanceId).finished().list(); + Map> taskInstanceMap = new HashMap<>(); + List userCodes = new ArrayList<>(); + historicTaskInstances.forEach(historicTaskInstance -> { + userCodes.add(historicTaskInstance.getAssignee()); + String taskDefinitionKey = historicTaskInstance.getTaskDefinitionKey(); + if (taskInstanceMap.containsKey(historicTaskInstance.getTaskDefinitionKey())) { + taskInstanceMap.get(taskDefinitionKey).add(historicTaskInstance); + } else { + List tasks = new ArrayList<>(); + tasks.add(historicTaskInstance); + taskInstanceMap.put(taskDefinitionKey, tasks); + } + }); + //组装usertask的数据 + List userList = identityService.createUserQuery().userIds(userCodes).list(); + Map activityIdUserNames = this.getApplyers(processInstanceId, userList, taskInstanceMap); + if (CollectionUtils.isNotEmpty(userTasks)) { + userTasks.forEach(activityInstance -> { + FlowNodeVo node = new FlowNodeVo(); + node.setNodeId(activityInstance.getActivityId()); + node.setNodeName(activityInstance.getActivityName()); + node.setEndTime(activityInstance.getEndTime()); + node.setUserName(activityIdUserNames.get(activityInstance.getActivityId())); + backNods.add(node); + }); + } + //组装会签节点数据 + if (MapUtils.isNotEmpty(taskInstanceMap)) { + parallelGatewayUserTasks.forEach((activity, activities) -> { + FlowNodeVo node = new FlowNodeVo(); + node.setNodeId(activity.getActivityId()); + node.setEndTime(activity.getEndTime()); + StringBuffer nodeNames = new StringBuffer("会签:"); + StringBuffer userNames = new StringBuffer("审批人员:"); + if (CollectionUtils.isNotEmpty(activities)) { + activities.forEach(activityInstance -> { + nodeNames.append(activityInstance.getActivityName()).append(","); + userNames.append(activityIdUserNames.get(activityInstance.getActivityId())).append(","); + }); + node.setNodeName(nodeNames.toString()); + node.setUserName(userNames.toString()); + backNods.add(node); + } + }); + } + //去重合并 + List datas = backNods.stream().collect( + Collectors.collectingAndThen(Collectors.toCollection(() -> + new TreeSet<>(Comparator.comparing(nodeVo -> nodeVo.getNodeId()))), ArrayList::new)); + + //排序 + datas.sort(Comparator.comparing(FlowNodeVo::getEndTime)); + return datas; + } + + @Override + public Boolean backToStepTask(BackTaskVo backTaskVo) { + TaskEntity taskEntity = (TaskEntity) taskService.createTaskQuery().taskId(backTaskVo.getTaskId()).taskTenantId(TenantContextHolder.getTenantId().toString()).singleResult(); + //TODO 添加驳回功能 + //1.把当前的节点设置为空 + if (taskEntity != null) { + //2.设置审批人 + taskEntity.setAssignee(backTaskVo.getUserCode()); + taskService.saveTask(taskEntity); + //3.添加驳回意见 + //4.处理提交人节点 + FlowNode distActivity = modelService.findFlowNodeByActivityId(taskEntity.getProcessDefinitionId(), backTaskVo.getDistFlowElementId()); + if (distActivity != null) { + if ("提交人节点名称".equals(distActivity.getName())) { + ProcessInstance processInstance = runtimeService.createProcessInstanceQuery().processInstanceId(taskEntity.getProcessInstanceId()).processInstanceTenantId(TenantContextHolder.getTenantId().toString()).singleResult(); + runtimeService.setVariable(backTaskVo.getProcessInstanceId(), "提交人的变量名称", processInstance.getStartUserId()); + } + } + //5.删除节点 + //6.判断节点是不是子流程内部的节点 + if (true) { + //6.1 子流程内部驳回 + } else { + //6.2 普通驳回 + } + return Boolean.TRUE; + } else { + return Boolean.FALSE; + } + } + + private Task getTask(String id) { + return taskService.createTaskQuery().taskId(id).taskTenantId(TenantContextHolder.getTenantId().toString()).singleResult(); + } + + private HistoricTaskInstance getHistoricTask(String id) { + return historyService.createHistoricTaskInstanceQuery().taskId(id).singleResult(); + } + + private Map getApplyers(String processInstanceId, List userList, Map> taskInstanceMap) { + Map userMap = userList.stream().collect(Collectors.toMap(User::getId, user -> user)); + Map applyMap = new HashMap<>(); + ProcessInstance processInstance = runtimeService.createProcessInstanceQuery().processInstanceId(processInstanceId).processInstanceTenantId(TenantContextHolder.getTenantId().toString()).singleResult(); + taskInstanceMap.forEach((activityId, taskInstances) -> { + StringBuffer applyers = new StringBuffer(); + StringBuffer finalApplyers = applyers; + taskInstances.forEach(taskInstance -> { + if (!taskInstance.getName().equals(BpmTaskRuleScriptEnum.START_USER.getDesc())) { + User user = userMap.get(taskInstance.getAssignee()); + if (user != null) { + if (StringUtils.indexOf(finalApplyers.toString(), user.getDisplayName()) == -1) { + finalApplyers.append(user.getDisplayName()).append(","); + } + } + } else { + String startUserId = processInstance.getStartUserId(); + User user = identityService.createUserQuery().userId(startUserId).singleResult(); + if (user != null) { + finalApplyers.append(user.getDisplayName()).append(","); + } + } + }); + if (applyers.length() > 0) { + applyers = applyers.deleteCharAt(applyers.length() - 1); + } + applyMap.put(activityId, applyers.toString()); + }); + return applyMap; + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/AdminUserApi.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/AdminUserApi.java new file mode 100644 index 00000000..f7756240 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/AdminUserApi.java @@ -0,0 +1,71 @@ +package com.ruoyi.flowable.service.user; + + +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.domain.dto.user.AdminUserRespDTO; + +import java.util.Collection; +import java.util.List; +import java.util.Map; +import java.util.Set; + +/** + * Admin 用户 API 接口 + * + * hasPermi + */ +public interface AdminUserApi { + + /** + * 通过用户 ID 查询用户 + * + * @param id 用户ID + * @return 用户对象信息 + */ + AdminUserRespDTO getUser(Long id); + + /** + * 通过用户 ID 查询用户们 + * + * @param ids 用户 ID 们 + * @return 用户对象信息 + */ + List getUsers(Collection ids); + + /** + * 获得指定部门的用户数组 + * + * @param deptIds 部门数组 + * @return 用户数组 + */ + List getUsersByDeptIds(Collection deptIds); + + /** + * 获得指定岗位的用户数组 + * + * @param postIds 岗位数组 + * @return 用户数组 + */ + List getUsersByPostIds(Collection postIds); + + /** + * 获得用户 Map + * + * @param ids 用户编号数组 + * @return 用户 Map + */ + default Map getUserMap(Collection ids) { + List users = getUsers(ids); + return CollectionUtils.convertMap(users, AdminUserRespDTO::getId); + } + + /** + * 校验用户们是否有效。如下情况,视为无效: + * 1. 用户编号不存在 + * 2. 用户被禁用 + * + * @param ids 用户编号数组 + */ + void validUsers(Set ids); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/DeptApi.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/DeptApi.java new file mode 100644 index 00000000..e512c4f5 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/DeptApi.java @@ -0,0 +1,54 @@ +package com.ruoyi.flowable.service.user; + +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.domain.dto.user.DeptRespDTO; + +import java.util.Collection; +import java.util.List; +import java.util.Map; +import java.util.Set; + +/** + * 部门 API 接口 + * + * hasPermi + */ +public interface DeptApi { + + /** + * 获得部门信息 + * + * @param id 部门编号 + * @return 部门信息 + */ + DeptRespDTO getDept(Long id); + + /** + * 获得部门信息数组 + * + * @param ids 部门编号数组 + * @return 部门信息数组 + */ + List getDepts(Collection ids); + + /** + * 校验部门们是否有效。如下情况,视为无效: + * 1. 部门编号不存在 + * 2. 部门被禁用 + * + * @param ids 角色编号数组 + */ + void validDepts(Collection ids); + + /** + * 获得指定编号的部门 Map + * + * @param ids 部门编号数组 + * @return 部门 Map + */ + default Map getDeptMap(Set ids) { + List list = getDepts(ids); + return CollectionUtils.convertMap(list, DeptRespDTO::getId); + } + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/DictDataApi.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/DictDataApi.java new file mode 100644 index 00000000..f4ffcf6e --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/DictDataApi.java @@ -0,0 +1,23 @@ +package com.ruoyi.flowable.service.user; + +import java.util.Collection; + +/** + * 字典数据 API 接口 + * + * hasPermi + */ +public interface DictDataApi { + + /** + * 校验字典数据们是否有效。如下情况,视为无效: + * 1. 字典数据不存在 + * 2. 字典数据被禁用 + * + * @param dictType 字典类型 + * @param values 字典数据值的数组 + */ + void validDictDatas(String dictType, Collection values); + + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/PermissionApi.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/PermissionApi.java new file mode 100644 index 00000000..a964dc18 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/PermissionApi.java @@ -0,0 +1,21 @@ +package com.ruoyi.flowable.service.user; + +import java.util.Collection; +import java.util.Set; + +/** + * 权限 API 接口 + * + * hasPermi + */ +public interface PermissionApi { + + /** + * 获得拥有多个角色的用户编号集合 + * + * @param roleIds 角色编号集合 + * @return 用户编号集合 + */ + Set getUserRoleIdListByRoleIds(Collection roleIds); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/PostApi.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/PostApi.java new file mode 100644 index 00000000..454079d0 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/PostApi.java @@ -0,0 +1,21 @@ +package com.ruoyi.flowable.service.user; + +import java.util.Collection; + +/** + * 岗位 API 接口 + * + * hasPermi + */ +public interface PostApi { + + /** + * 校验岗位们是否有效。如下情况,视为无效: + * 1. 岗位编号不存在 + * 2. 岗位被禁用 + * + * @param ids 岗位编号数组 + */ + void validPosts(Collection ids); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/RoleApi.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/RoleApi.java new file mode 100644 index 00000000..b053ab93 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/RoleApi.java @@ -0,0 +1,21 @@ +package com.ruoyi.flowable.service.user; + +import java.util.Collection; + +/** + * 角色 API 接口 + * + * hasPermi + */ +public interface RoleApi { + + /** + * 校验角色们是否有效。如下情况,视为无效: + * 1. 角色编号不存在 + * 2. 角色被禁用 + * + * @param ids 角色编号数组 + */ + void validRoles(Collection ids); + +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/AdminUserApiImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/AdminUserApiImpl.java new file mode 100644 index 00000000..101efd70 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/AdminUserApiImpl.java @@ -0,0 +1,82 @@ +package com.ruoyi.flowable.service.user.impl; + +import cn.hutool.core.collection.CollUtil; +import com.baomidou.mybatisplus.core.toolkit.Wrappers; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.enums.CommonStatusEnum; +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.convert.user.UserConvert; +import com.ruoyi.flowable.domain.dto.user.AdminUserRespDTO; +import com.ruoyi.flowable.service.user.AdminUserApi; +import com.ruoyi.system.service.ISysUserService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import java.util.*; + +import static com.ruoyi.common.exception.util.ServiceExceptionUtil.exception; +import static com.ruoyi.flowable.core.enums.user.ErrorCodeConstants.USER_NOT_EXISTS; +import static com.ruoyi.flowable.core.enums.user.SysErrorCodeConstants.USER_IS_DISABLE; + +/** + * Admin 用户 API 实现类 + *

    + * hasPermi + */ +@Service +public class AdminUserApiImpl implements AdminUserApi { + + @Autowired + private ISysUserService userService; + + @Override + public AdminUserRespDTO getUser(Long id) { + SysUser sysUser = userService.selectUserById(id); + return UserConvert.INSTANCE.convert(sysUser); + } + + @Override + public List getUsers(Collection ids) { + if (CollUtil.isEmpty(ids)) { + return Collections.emptyList(); + } + List sysUsers = userService.listByIds(ids); + return UserConvert.INSTANCE.convertList(sysUsers); + } + + @Override + public List getUsersByDeptIds(Collection deptIds) { + if (CollUtil.isEmpty(deptIds)) { + return Collections.emptyList(); + } + List sysUsers = userService.list(Wrappers.query().lambda().in(SysUser::getDeptId, deptIds)); + return UserConvert.INSTANCE.convertList(sysUsers); + } + + @Override + public List getUsersByPostIds(Collection postIds) { + List sysUsers = userService.getUsersByPostIds(postIds); + return UserConvert.INSTANCE.convertList(sysUsers); + } + + @Override + public void validUsers(Set ids) { + if (CollUtil.isEmpty(ids)) { + return; + } + // 获得岗位信息 + List users = userService.listByIds(ids); + Map userMap = CollectionUtils.convertMap(users, SysUser::getUserId); + // 校验 + ids.forEach(id -> { + SysUser user = userMap.get(id); + if (user == null) { + throw exception(USER_NOT_EXISTS); + } + if (!CommonStatusEnum.ENABLE.getStatus().equals(Integer.parseInt(user.getStatus()))) { + throw exception(USER_IS_DISABLE, user.getNickName()); + } + }); + + } +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/DeptApiImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/DeptApiImpl.java new file mode 100644 index 00000000..187050a8 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/DeptApiImpl.java @@ -0,0 +1,70 @@ +package com.ruoyi.flowable.service.user.impl; + +import cn.hutool.core.collection.CollUtil; +import com.ruoyi.common.core.domain.entity.SysDept; +import com.ruoyi.common.enums.CommonStatusEnum; +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.convert.user.DeptConvert; +import com.ruoyi.flowable.domain.dto.user.DeptRespDTO; +import com.ruoyi.flowable.service.user.DeptApi; +import com.ruoyi.system.service.ISysDeptService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import java.util.Collection; +import java.util.Collections; +import java.util.List; +import java.util.Map; + +import static com.ruoyi.common.exception.util.ServiceExceptionUtil.exception; +import static com.ruoyi.flowable.core.enums.user.SysErrorCodeConstants.DEPT_NOT_ENABLE; +import static com.ruoyi.flowable.core.enums.user.SysErrorCodeConstants.DEPT_NOT_FOUND; + +/** + * 部门 API 实现类 + *

    + * hasPermi + * + * @author wangzongrun + */ +@Service +public class DeptApiImpl implements DeptApi { + + @Autowired + private ISysDeptService deptService; + + @Override + public DeptRespDTO getDept(Long id) { + SysDept sysDept = deptService.selectDeptById(id); + return DeptConvert.INSTANCE.convert(sysDept); + } + + @Override + public List getDepts(Collection ids) { + if (CollUtil.isEmpty(ids)) { + return Collections.emptyList(); + } + List sysDepts = deptService.listByIds(ids); + return DeptConvert.INSTANCE.convertList(sysDepts); + } + + @Override + public void validDepts(Collection ids) { + if (CollUtil.isEmpty(ids)) { + return; + } + // 获得科室信息 + List depts = deptService.listByIds(ids); + Map deptMap = CollectionUtils.convertMap(depts, SysDept::getDeptId); + // 校验 + ids.forEach(id -> { + SysDept dept = deptMap.get(id); + if (dept == null) { + throw exception(DEPT_NOT_FOUND); + } + if (!CommonStatusEnum.ENABLE.getStatus().equals(Integer.parseInt(dept.getStatus()))) { + throw exception(DEPT_NOT_ENABLE, dept.getDeptName()); + } + }); + } +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/DictDataApiImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/DictDataApiImpl.java new file mode 100644 index 00000000..b87bdced --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/DictDataApiImpl.java @@ -0,0 +1,53 @@ +package com.ruoyi.flowable.service.user.impl; + +import cn.hutool.core.collection.CollUtil; +import com.baomidou.mybatisplus.core.toolkit.Wrappers; +import com.ruoyi.common.core.domain.entity.SysDictData; +import com.ruoyi.common.enums.CommonStatusEnum; +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.service.user.DictDataApi; +import com.ruoyi.system.service.ISysDictDataService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import java.util.Collection; +import java.util.List; +import java.util.Map; + +import static com.ruoyi.common.exception.util.ServiceExceptionUtil.exception; +import static com.ruoyi.flowable.core.enums.user.SysErrorCodeConstants.DICT_DATA_NOT_ENABLE; +import static com.ruoyi.flowable.core.enums.user.SysErrorCodeConstants.DICT_DATA_NOT_EXISTS; + + +/** + * 字典数据 API 实现类 + *

    + * hasPermi + */ +@Service +public class DictDataApiImpl implements DictDataApi { + + @Autowired + private ISysDictDataService dictDataService; + + @Override + public void validDictDatas(String dictType, Collection values) { + if (CollUtil.isEmpty(values)) { + return; + } + List list = dictDataService.list( + Wrappers.query().lambda().eq(SysDictData::getDictType, dictType).in(SysDictData::getDictValue, values)); + Map dictDataMap = CollectionUtils.convertMap( + list, SysDictData::getDictValue); + // 校验 + values.forEach(value -> { + SysDictData dictData = dictDataMap.get(value); + if (dictData == null) { + throw exception(DICT_DATA_NOT_EXISTS); + } + if (!CommonStatusEnum.ENABLE.getStatus().equals(Integer.parseInt(dictData.getStatus()))) { + throw exception(DICT_DATA_NOT_ENABLE, dictData.getDictLabel()); + } + }); + } +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/PermissionApiImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/PermissionApiImpl.java new file mode 100644 index 00000000..36a5f9a7 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/PermissionApiImpl.java @@ -0,0 +1,26 @@ +package com.ruoyi.flowable.service.user.impl; + +import com.ruoyi.flowable.service.user.PermissionApi; +import com.ruoyi.system.service.ISysUserService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import java.util.Collection; +import java.util.Set; + +/** + * 权限 API 实现类 + * + * hasPermi + */ +@Service +public class PermissionApiImpl implements PermissionApi { + + @Autowired + private ISysUserService userService; + + @Override + public Set getUserRoleIdListByRoleIds(Collection roleIds) { + return userService.getUserRoleIdListByRoleIds(roleIds); + } +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/PostApiImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/PostApiImpl.java new file mode 100644 index 00000000..39b6982f --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/PostApiImpl.java @@ -0,0 +1,50 @@ +package com.ruoyi.flowable.service.user.impl; + +import cn.hutool.core.collection.CollUtil; +import com.ruoyi.common.enums.CommonStatusEnum; +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.service.user.PostApi; +import com.ruoyi.system.domain.SysPost; +import com.ruoyi.system.service.ISysPostService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import java.util.Collection; +import java.util.List; +import java.util.Map; + +import static com.ruoyi.common.exception.util.ServiceExceptionUtil.exception; +import static com.ruoyi.flowable.core.enums.user.SysErrorCodeConstants.POST_NOT_ENABLE; +import static com.ruoyi.flowable.core.enums.user.SysErrorCodeConstants.POST_NOT_FOUND; + +/** + * 岗位 API 实现类 + *

    + * hasPermi + */ +@Service +public class PostApiImpl implements PostApi { + + @Autowired + private ISysPostService postService; + + @Override + public void validPosts(Collection ids) { + if (CollUtil.isEmpty(ids)) { + return; + } + // 获得岗位信息 + List posts = postService.listByIds(ids); + Map postMap = CollectionUtils.convertMap(posts, SysPost::getPostId); + // 校验 + ids.forEach(id -> { + SysPost post = postMap.get(id); + if (post == null) { + throw exception(POST_NOT_FOUND); + } + if (!CommonStatusEnum.ENABLE.getStatus().equals(Integer.parseInt(post.getStatus()))) { + throw exception(POST_NOT_ENABLE, post.getPostName()); + } + }); + } +} diff --git a/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/RoleApiImpl.java b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/RoleApiImpl.java new file mode 100644 index 00000000..80c3bd75 --- /dev/null +++ b/ruoyi-flowable/src/main/java/com/ruoyi/flowable/service/user/impl/RoleApiImpl.java @@ -0,0 +1,51 @@ +package com.ruoyi.flowable.service.user.impl; + +import cn.hutool.core.collection.CollUtil; +import com.ruoyi.common.core.domain.entity.SysRole; +import com.ruoyi.common.enums.CommonStatusEnum; +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.flowable.service.user.RoleApi; +import com.ruoyi.system.service.ISysRoleService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import java.util.Collection; +import java.util.List; +import java.util.Map; + +import static com.ruoyi.common.exception.util.ServiceExceptionUtil.exception; +import static com.ruoyi.flowable.core.enums.user.SysErrorCodeConstants.ROLE_IS_DISABLE; +import static com.ruoyi.flowable.core.enums.user.SysErrorCodeConstants.ROLE_NOT_EXISTS; + + +/** + * 角色 API 实现类 + *

    + * hasPermi + */ +@Service +public class RoleApiImpl implements RoleApi { + + @Autowired + private ISysRoleService roleService; + + @Override + public void validRoles(Collection ids) { + if (CollUtil.isEmpty(ids)) { + return; + } + // 获得角色信息 + List roles = roleService.listByIds(ids); + Map roleMap = CollectionUtils.convertMap(roles, SysRole::getRoleId); + // 校验 + ids.forEach(id -> { + SysRole role = roleMap.get(id); + if (role == null) { + throw exception(ROLE_NOT_EXISTS); + } + if (!CommonStatusEnum.ENABLE.getStatus().equals(Integer.parseInt(role.getStatus()))) { + throw exception(ROLE_IS_DISABLE, role.getRoleName()); + } + }); + } +} diff --git a/ruoyi-flowable/src/main/resources/mapper/flowable/BpmOALeaveMapper.xml b/ruoyi-flowable/src/main/resources/mapper/flowable/BpmOALeaveMapper.xml new file mode 100644 index 00000000..8d88b183 --- /dev/null +++ b/ruoyi-flowable/src/main/resources/mapper/flowable/BpmOALeaveMapper.xml @@ -0,0 +1,64 @@ + + + + + + + + + + + + + + + + + + + + + + + + select id, user_id, type, reason, start_time, end_time, day, result, process_instance_id, creator, create_time, updater, update_time, deleted, tenant_id from bpm_oa_leave + + + + + diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/config/FlowableConfiguration.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/config/FlowableConfiguration.class new file mode 100644 index 0000000000000000000000000000000000000000..67d9cdbd62e03003679d972920257aea8b31c92f GIT binary patch literal 2020 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 NHpk~+S2H*d{1-s4_YVL7 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/controller/definition/BpmModelController.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/controller/definition/BpmModelController.class new file mode 100644 index 0000000000000000000000000000000000000000..4631053f17b4f9acf9f43bf1508532fb1dc5c4b0 GIT binary patch literal 5978 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/controller/task/BpmActivityController.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/controller/task/BpmActivityController.class new file mode 100644 index 0000000000000000000000000000000000000000..31bbf8e82f7c1d246f8a4d3cd5977d51a57bc933 GIT binary patch literal 1989 zcmb7F>u*#=6hE_lER>dl@{otPtzaKWM-dQPN^EO1O;%}I(hvG+_U^Eqa`)cL%-ojx z2?!CDLTkdql&Gl&(?~2RDCLpz-G9T~yX`0c1LK){YqzbNNgy zEC3vW<5f@zZ>)z3cyj|Bgtw|tcQ_75;_xRbi!ARKN?=5X;W~=IuH+yaVrMB^RMK9q zTwhuoBP)*yN9f}O${L&c36yuaX@pUm6b}E$&!)IMX&?}IA?aGo?q^b%^MbNm4+y-W zwG?Q}b+mMCo6G;;?pibr3=vbCIT_lgrEoHBO-rsohO(|hEmv}y zcC$=4v}2H+#iD#$W5&&%bSqO@$`c7)+{E+7v#w8SNXA@BIDe%TXU16 zxtX8QJ@@2E{>#Zf&t1sPT$y`zY3|4K!TqV=;@DqLN9XR|4zAwLkNpr#TnKLd9{f64 zxHg6Q`EwKKng8NS;m%#mkm!j>>KA;N%a$RZCkod`=fC;h*ak#qz{a0`Pp}LUlL&4+ z3hrNz@DpfSrQq(YXA4W{9)@vPbh(4P64<(;-on!x`LTxt)=7TG7m}yTa9Rj~nl#hw zftgg8XHnx zw$+SK>lvTRk%lG$^~=X5#LP99%jW0Eb9YHE*22|d=~aEXqgc>ToMJ2jlk!{t+ZrHXn&;T>9%Uv0M*qHhjno{S`F_&YaCE{4B8P~iM9A# z8-~E9;^d!54RQ-N%)t`b{%6Tzv#eNDog)h)v~Bl7WDRQ(>hLe9(hM2I48K`~2Rm}Oh!xsU~8 zU<+);YzA68l;d3lwtoUd!Ar0MeYY#v3A+^RhL_R0N1;2aUV+5NSYN$`v9Fp41$$v1 orl>Y$ycTJ;V0c3aqA^Tgj|rO4)EuVY53i%`fN>4aEyw}<1ENBibN~PV literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/controller/task/BpmProcessInstanceController.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/controller/task/BpmProcessInstanceController.class new file mode 100644 index 0000000000000000000000000000000000000000..51212a199e5c0406a1fa5e550af9aab0ff4387b9 GIT binary patch literal 4058 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmFormConvert.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmFormConvert.class new file mode 100644 index 0000000000000000000000000000000000000000..68eacf51ff52a57685d00ea373f9fc817dae27cb GIT binary patch literal 1407 zcmb_cO>Yx15PeQdz8Yw0p->7GDv;6(mQckdRiqNAR4F7VDY-ivCvvoFFT38L{15&J z;(|CJ#E~C`nB6XI8mc8Aa)@{AnR)Y`XZ-!g=Pv;Ev0K2MGPZHI1crMB+-E2rx4I{X zt;S1+XHCa=Jc!Ix@u6$piM}hjWAsD@mJj4mX=Rns{JHNPnZRqDp41o$sS|^#UY28E zJfSq77(S%9v-m&?D?9RSuU*^A2$R|>JH4cU7j44+8^kyM;6EUavp8lIn?-U?n~n^9 zl7*p?O50SSeaf&~ZH~o6@W?8c$9tsQ+y+Cbt43PbD3A>A=K~J@QR(_`NLiWY=ya9m zyL8nFsXEbGNol_*X%8}mJH(m%K|d|Nzm7MrX~ zjdfYHqAZL##b&fB9yo41dUb}S>RyjwxnTyR+GIyyL?fzIgmY$%Ca6{o(6HHTBs;4xP_Ou3u$7v@g2^y{l!(Saa zld?uQBoIAz3|lcQedU+wg+dWX)>68blXV3Z^1Mo3Ib5UjRji%a*C}Ql8)V<0^GCGC kzsfrMgxpuKk68LZHp5N&&nKswlx35w9B$!uy1NDZ1Tl`h@c;k- literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmFormConvertImpl.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmFormConvertImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..0779fd3b591fdd3282e79a382e8bdcf571c6c73c GIT binary patch literal 6279 zcmd5=Ygimt8Ga8eOjw2_LIO?NQkv38fUxF9nm{82#BAl#ay4mC2N=kZy^!4{38|(v zZLHc_Z}n2^{Z{X$UY4fOs#RN!_jR_R4m^22AJXXcx8zVn^; zJ>Pr2IkT7l^W;+icH{3sv|=`hMw|@d6ix?m8|K71Bi3DFy-}<;iFH=P-9ZF#kF>m5 zTF!}euUzg6;ym6W*8O5J(0Pe*UWT8?S-G6k@YW!zoayLLg}2GX^s2(!#bUH7JRlYW zRpCLgIJ^q)5Q{OY@Q_&V6zg53csJf7aNaA{!=-p1-Y=IAX!u|dEqFwlKcwN&AVM-{ z2p^W_kBIeAxqM76AJ_1SAezOYS!Qe&P%ZeRhEHkuw1&?p6t`M2D|JAjux{J1LZBl) zVJfWawqj=QY;?>_3>sq*wv>0r$BoFak+9_1Z49K2SxJTc-Q)3SC@~wKvqF=R_$l!S zjmKjr%|t3RVNS{nRy-CuI1}xPC!!rrPdGXgX;IKzM}>IZQgJ5YQNxObPR2u%9Gi`i zFpZSiZ=M+LYuV;A%#5Y1)Z7vvI{P?xV9YdP3e7$NeCAas8FLtQ_G$R6!g`lrXWzkD zD>7jw6!dT`W+pl!Mlxw~r4ISbi_gRPoK}uAe=+<$p?H9ueNp7; z%By}IufmAJEeim$JIXEjOPnY@dFHXqPMLkO>KmIj$Eo?$L ztf_li<-0Lv_a1dCf%ho8is(xSgNNeOERXOTbAW_nDRat9kYRDsbr|(x$@H%x7+Di) zZIa|EM&1Nnmt{sKh^{<6nKs&F+c4Gw&($*aZbfA+@itmpuUfXCtj{tu_ zxZzn8sS8h_pr^4RjlwhnBVrV#QEVGp8f!+_P|~~o8LV%-gy7-r^C(S2x4X;ISUb{~ z#<~j(7O(*{p_FzNpROU8Yx!@{^{7BKuEPzu88xUUrzUJ6uUf*|Y6Gozz_>sO3$Ml@ zLJg8_7(Jv`&(Xc;Lm_TwbRp)n+`*Q9yQQG`-`K5TK*OL-b;zaq55^X%{kc@vr|~kI zLV1RBMTYYRm-EJ@Iae;`ypxE42F}w+&LQI7LC!nLc^5hFCf_}@nz5I^PV6J277y?K zEbo5u9=3TOa(VAs#M^1f<$bG%_lV2;pDynqhxa8^9iGQEcA7V(v01WwZ5r3f{rWVj z<$gmNHFCc(jV-+0B>dNTa$YN)H>a_+aUtE!C?oYYPS#H9omfkaRgy+E6S$Qmn)v@@ zGhRc&-ArZ=tv(Fk4iEjIEd3#ldoAIw<^3>@FzqeaoR@Z|C6|7uhC7|KAH^s~wGq*C z6h;y6?HT&P!wnbla)o)+_0Za;(7XL2>J=VAHH`+v^J(M$Kx6%;KpJdlD!|BN|9!zj zwHg`&4wLE#sUE=w+{s)Y<^CKcwby4Ug}5YP5)sw4XUVnOM0jSvcD#;@P+%*$l6Udy z<@ikB8`vhAUWY3+U_*=c&Kb90WeB{af)$_D?ZJYKF)0!wQ1BiF|*)nF^MTVGnMq7;@q+g9q`g; zW&5UG@P$ytF5u&wh6iZ6v~kc~bP&Kb8Nkg1yw}zx^>>Sar9SuBU<>Vv;($HDC<%g{ zCD>D(`ZOhW#sjt{3%15yeUVW)Vu*VI1>`)=1_AQY6te%y3OP%pcN6J3=K5YL=F;x26D-YfYZO;Yc0 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmModelConvert.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmModelConvert.class new file mode 100644 index 0000000000000000000000000000000000000000..21d208a44edd062de1403746156901e0817eab99 GIT binary patch literal 10048 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmProcessDefinitionConvert.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmProcessDefinitionConvert.class new file mode 100644 index 0000000000000000000000000000000000000000..17b77f62b0c08e3bebf5261dd1e7dda7fe4e05b4 GIT binary patch literal 7108 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmTaskAssignRuleConvertImpl.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmTaskAssignRuleConvertImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..34835cbf4461da95a32aa3aa22d43b0cdd4c7cac GIT binary patch literal 5272 zcmc&&X>%Oa6+Le>a@*tfDp|%DjAd+)wRnti;shzSUBIVD@qF2!B=Q4AaL zW08Kc6nEnudE6UA9QVoCPh~?7lLv=LlDJ>SJR}|w8v2=rpKJJq!lEwAw%pwcbuC** z6`~0zWh$)bvutxv48R!*U}Z^D_5XJ?#QD}FTX zoDz-rgkzsHvu-?P9+e#|$BysKOph43<2|{YHE9pcq|Jn{lbp_^I~BBGiNf72)hwo* zX~VMPrtMnpY)!kpgPmJzSUBkrEFo6l88UO3(Lq9qj+=(9&{M<6{9q`o9Y3G7zl%a= z4eN75tjn3M1nQV^(umKvRysaxx`f_T9;n$&Az01js$KqP+OQ|%eU3fZxtoiZ4ZFs~ z@qQ!Y1(+3};Q@vFYlY*2NE2DpaLpm}1U*_S8uN8z_r6DpO(37@pW)o>6mou5|x7WZiFD{!eyjiD*=HmQ)OD6UI_jecsJr`7TpOtfW zC~Rs8rJ#&4FG}g_4Y2gmoKRRUeubfYRVZhJ#T`g<#g{{zDQ6j#lMw}{HJ4dIU%lZF{t`Hk`X0j9J zK1&X%<_e6V6K0#FnvNZ~Ug5h{4y@pKnFF+jhjcuQM|3=j$8FFM}FJM3bm?Rv6VV^#`>`KIGE-j$92s^f2XkLjo5edKkV!3R2S!|giG z;zJ$h@Vdg@Dh>#WwdTnh@+WK51LB<^gX71{36|~dh1ul^R3b+|Djcd}@S9u&YBXDg z<)zKUiM-{M29mCsHC%`Nf6oFT@MbeqY_m_l|PBS9Z?(`PZesS^Yup+(eY1GqvGDQ~@ESGKu z3iVtrlMP{T-mRg4te^MBw5T+etsi!?7Vj2BPYXqzHDNP>iWq7lhMTj_gqh3r7O#zB zw27;$PBe;Vd8aI?A24SLg8Eg4weiB4ewZj17}gq@MZVPvVXh7nx~woGrVAd;RP)xQ zMZe0e^l8-fGWbLoO(B6Vu`@NNkoVKU?!~mf#5>n}8KaO`c)na@QiN_)3K5INV22~% z7vZJNhG@>Zi@l4#5R<&0wIrof7hA>t*i@)^FDQk+DNSw4$PJixvafRKAuhGOkMa)T zpB&V#Gg2g1)*6C2S=*4Q>8iu0}bu5Ud9^)7`7z7;3| z4fr-?-B-Xae1~7W6W|6uW$~AY@(<{5`v>aP>2ru2JcGLa_O?8tdDQc@XrSZoSkZnC z+QE)9Se(ZanT+MJRNi_X%jCU0PaVEj=CP{%Gz~^Q5IeXvu$Et!5%qd3$CYTp21-}) zZzI~Ui9D`GCpII2EjWN{J;1F#r~okGQKO5HmtzmUhux&Oo*vkPZq%WNI|xC4WG_b& z-biH8zpzG1_aV)ef@U&aWMG{PnnkF5W^vw35>{a_OYoB^*Ax?jpNx&fG+FG0yHs zAKA6iOXckRk#crBHS`zQ4Fv2y4cP4}XLm&&S9+v21f;H-i_|q(La2LauA8J11a~7z z?IWq1n2h^LEJ>PuAws)~goNs#N2ntp)LcQxA1No)RwQ&Yen2-!hJMa~*7My{$k2@k z+s>j{@vgV2pVHL|1083vS@F)ffx;GrPY}5w+TOY@n#VO~acu-+Z~W(rP*!f>KefOO zq8K8zThNG6R?!$U>{b#!RHPKgGB4XCTY8G*dbnzs91nA+BP2e~n5Gz0lQBKYcuq1&r%2PHBaSiZ$60oE(HS5$zki4%MJNfoKZtKv eMST5{^7!rx;v2yoo(B%I;6W>jxn$t`Sn^*MHR4bJ literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmUserGroupConvert.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmUserGroupConvert.class new file mode 100644 index 0000000000000000000000000000000000000000..13aaf1cec759410417ebda79da6e36552c67fcf8 GIT binary patch literal 1550 zcmb_cZBG+H5Pr6ReW41XR1^_KNXi#CF@7;6Bv@ZUQj6iG-?!K0vd!J@dACM=Vs>n&o5sAJi%HXOC{XHauE#o^LW5e*lzCa zZ#C;L7-|jAggj17U-84hyc6v}a?j|Vj4kiTL#35fM)PM;c#z2WRcz9zeq^pOX=X{&3lIX?&9evu@qMB%D1hZ+Iq%isjag8zvc3Lhtjw__(9b9U*Kt0$8~#m(iRyZbVX|&IbUxEiS~k zXE6I)zfSrb*>lHs+#tVcvfre4J|SZP72G0afefo8-BX@ppOO0p_6ZXoNq32W5^>fF Q1-nBshs8lKcX}821v|&+bpQYW literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmUserGroupConvertImpl.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/definition/BpmUserGroupConvertImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..cbc1104258a481b986cf481d915796b58852cc84 GIT binary patch literal 5849 zcmcgw`Bxj)75*LwCWrxJTc)w=5^qT`2&*=B+!z}NLlUb+;2I3Y={86MjIGV15d?Qh z_cdLTu4%e&>0YOch~qZhlQwPIw7>WCr~VT?r^|O|w2W*lc?mr_XXeg*_uY5D``!2E zj-LO|)6W3dkG}@d4=adnoC)GAl0gh1B~n_Xj7V9L&WU7e$ORFEBbvNi7K2#AgCZ@9 z#OX^ib4eyGiT(K?LY{w)h426{(vihXWL_p$d$LW#F;AN3rp5<$*O+Xk9m`q1T0A4;3?ndW8fk^YHN4af zL7{!thjV0H!>1KC`LU0TAIn?GglQ}2(RA9hhm%GwXEKu~Yph%^{#~msK~#-7(-BI~ z7>hoGKiwQAf3+;W4Gq zTHK7IR*rSEquW;vq9W8vCAkW2oTO{papg#pjOT=|2@|;Cn$5g#S|8X9@p@B9bdo~b$kh5*6|gQ zzADn!@O2&Ez&CX~fp00SuN)c2joboXK;q-uI=+MN>i8bMui*zeeuxV?p2Uw74pa)Y z+)>nO0=$Dy$B*$79Y2*z0V_J5!bKfVxu9#6dj)o~bOI{tv?6}l^KYVSWCf5e}3Okh&S3;45+ zzes9c#3h9z6(8l)6p6fAbK~Prp~J=RzRkyH&zNzZ@ps>_ONA4w%q=oiFEgumEbJ{R z46#tyP*t9VEWYz9il}4ShLd3{rOlR_s>yEu{wXKvrlR5-r$!B|y3>X$jVYs>H=Qvf z#hn#mbQKCF9NS_`#p5>Zj?NWdJT5=Q!8JkC&VWYDT->&@(wy_PlJGyZaF{KglV?}D zsVd@V+A-%%n;|#niU#xkpiARx5_Jc|ikb3VB3}~G1fyGDiq6;CZuY;vhcn3}%bXww zD(F8Y8DQA_YUZKI_NWxF0+wHGO zI<#DAJ8beVSSonZ)J}`4`s|?MGU_*6K^-Ty(5_Uuoqp%#!lY3XjU*8p{RSRfDG{d0 z+r5*PY}sSI(Z|!CAGO=wvAZ@rJ7)xT9{k zoIaUxj3gtcsVibT|5oUS;vXk~#ycAVUJvpq!n@+Vsb>Z1NeT^k6|c>%GK9M+>z)A) zVSrEk1i}$sW%7#%@CV$}`y86og{ROE3-=b#SU_M}gr)+TT|q10mT4+l#`>N`NB9zg zGkq&)EkK_R7tnTrJ_7E#J^Y&nxRpL|fnUxK-z2y%K-AkDJ2(zCs_Y>j)N_R2I2MOYk z4{EdwHA<*47wXX>)cz}=dYWpeM|@DnictSqgc|dpUP5SQ1)E$lI}3Q3aCvJ1n|a$J z(AN1Fe7Ov6E#MV$?<(Lnxo<0Ad$_{tKIR?dCBQ(NF^qOH(n(Ib$jMF|$DL%NpV=S4 zC`V(I#(j`uWymo?9%sny^mPKSCA&MxW;MH>rW$fs!%2@_wrHFswfi^n-^BY+3G$Aa z-iz3&u!3DNO1JYXxbGtF;MXO(DBP*oa&{jIgnKpz3fO%SJq?(C;=lj!!R+GR0v8D2 z6k)!O{{qgC+tb`I4=~_2lp#gX#zc(Mj?0(94RO{K!G+L=X-W-l!FKX4R;TE%G&Vz> zlEgIc9tTTVQebopoVs^L&|m$=%aqT1MUWV7mwLC42|U zCCVYnnU6xLW=1So4oQ5l>y=%8&#!!W_4ONo4Xiftq=RKVZG*7V#4|!`ub1|>d)*I& zcS&YT;VOHk#Zh5T<)Ba^vu3PZ5T){7=1T00%Kb^$o$SPf=JXKZWGy+CV<`%0a*{S#rfYlp0OKG8?{D7$6!9#HB*tYtP9lPFF-(YIQpQvg zapc4zFQXu1T873DUodUcFEd0ZCf6CHMW<#k^i)jSc-yiz4R?)iHpr8$I91+Q=dLN% zp;Pj=Opjr{QgtlNZ8>|U_Oao776`5C*t>@7YnI`8yl!YOo7PI(Tx{{zRR@+x9eKPvFoLdweGv-nWyEO4Q+)tOOxF)*BpzRwpQ~U%`@zprdys)vrgxU zx%0s@-1o-LtF=|i9<_K=TsKI37M9A=BGht?p|^uWz==We4WiC{qg|_P2O?z2kUqt{ z<1V+TMymHW77XR<-lG3vbM~624Vt@nv4zW2J3OnK+eXzdm8mQ#-RIR$RNRXo@C?0Jc#(3aN zp_go>4L}B2S`lg7rXb?*FGz$0mU1UZ_b0df?3E~SOaukTy+U5MTrE~5aFVn g9L5-;EK$k=Dxj49iL8u~go=mch~NZMG66=N+v-`mHot>Sd-c_#vTXd7x){* z2C=AE^HGTF)IgdN(|Y0YeV%*ooa-MyznuYii6=$uRB#`=WiUJ_VvnJ8)b1TO+pRYY zFC5ndJW9+&@u6=%ih(b=YxGz~mK(ueg+Zt3NHLbJb7h^OnAsUlYyS#AF@aE;kIh_y zND3>v@#_nlln(!4s=| zo~o$&IR!(xr#vle5=n+n^8ts~@3wIXBrnN=P8u`4ycZso)}I|A-F#p)4<-X)Rm?*( zGCXyoIV67CWT9jS#XQ#G_G1qUizFll@48BX=NJ>OSOYO!*a`<(8Sgq zrDZz_1~NKMmxH0wGfCviw<;C3FD=OPlp74Y-Gq1?$Rj;gv7*4H*2dDHjgBeIF_Fgu z5ym!3T+82yFq9Ep7wz7)*0G3(RXoDuDps({urVd3FSN%ygOPNJzIzvuhQhTbnq-Jn zO4*b49G>A>#@uOwmM`M?Hw|Y})(MBkLq8XWofMY7R+j0vfD$l0%hR)hHS(yEwSXI> zzlrrf@&@_d!Y0YL$-Yl#`cyaBR}{X3eZkUak{Pz>y*jmSQIBC25_+uh{T$)cv^GOX)YWS3fPb;(@ zv>n?YRA}zqF|Cjoc4sYxHDk789WLf(EN{x3$#P`fn42-P)23(3cWE%;pRx-I`^PeF zF6kBB1v_~%>pmcevj-Hkl8(Ze-g+ix-JEGV z$$58)5zjJxYr=YHI(=YAgLRhU+x|lB+L1Kc*W=WuZ(l;|_#?5@V4giwrfkvy5{dF@@Eoq#3gdzQWetvD4T4CvT5-R@_$sh zo%zsy-%8f#vL8wQKzzd=XzV@MV0(z*q6Cfv<`5b$r9XH}RZ- zZ;AA6k)FqQ6n4kCRjE$Pl$PjnGVon|&%pPkc?QpF_<@0QctPC$(7=!IV*@`yVBkC! z4P3xQ125vNfuG96&&2m7S^0B8_6r0o4)KjNANh_+Dy4z-`iWA_Z8AB_RJe#|MCo$o{Hi4IYTgWmOsVvqp*Qs;#j5}-LDW&n? zr0?0z92d=-*`k#`sj#h74yxi`E{Ne!w1QR=e(2zo(s)~2_Xf)>?)GmIUb#Y zDCq69QvKb<`6Hg2u?mHh!z6GrmN@C6TAkrU!G4q}#NRWFMuO2G=v8**xOdm2R{0tr zcI}i|IBY#aY+QPrOP%neHglspjoQuTEFshcsWc~RisHk zxqQm?%`8Q#h9ZCT?cvW;-pl;0+QOcNk?n2#O0rin+IJr61qw~Loo#D4*ooIsHX;SQ z9(V9du~ZynD~q3}b~F1s`(HwfI=6_X@%{kKm$7C*`b8uHv>X+=HGmchs{*uzf*xS? z(SZQQIT~mVoo+)Lu?;s8>lOmq8iLvqIWB=B1pW;YmQId$ zVVJ;1XsZoPt^Y!|hPy-Dqb1ybFX28B;eHYA$Ihc;T!5|#uvYrc0PEPTzl=`#T)&8} zMQjMrEnlw*@LK6N2G}J1bpfuI{)PaXrN1%2mVubpZTtle{uEE3m%RGXft_Tyi>#8k z1-prOFN1n3QSW0l-H%BI_W`n$;72@9RCt~s&wEIt18LlgR1MDo4fk=11)0OLKd&%I zE$oNNJa0PIe*s$+UmrJ*Q|eJTyz>HXQFt1gDQr`GbZoyPG0?Y>JJ)*wJDPCxx&Qtn z0qo;N5sxc)f~RFLJS$>{q15dUhV6dm zEyI4#349*K!9?*>FZe7*p5#uT6B$`u`A13>L=*l#v&xzE(bdKZ?a1cA zjY6$?^Na&uD9w|Ai|Hg+r#g}*JU*mc48`eK44>+^>!@+-`V3n~v9`*WU7aYS=pfo! z2Ub`WXv46PHROxX*eG@^{~*FpY1dHOi;3`JT5D~72t^c35JwJ$RAw9Kj)}D*awQ#L z{#;HNsx#kk5>n`b%}lA~cuFgKKwH`>$mqgXAi&r^}|m9uA&sfFlIDd zi{m;nQ&B~6kuv?zb`}V%=R~KvYba}Tls}Qf7 zW$cQtFqX({oBO7*=i4PiM6=1`0)=7b^j*)+HzKB;if=l3lAO-CUh#;6C@^H`Zn)TKBqd=66=i-Qch^yszQDLf5)obWDHGW4pdN-tM-!A3vtlNjKt!nQQe+ zq3}hby>2Gk&d2{eOgwt-E_(-nwhcYdU-KgSaHOU&qTht1xi% z66QV6tjcrXA?jF1R!lz7v4ICXQym{-Q^!Mmq;R#J`>*3oT-Nb1K2exy^Tz4;6rbt% z9HgHUlowQhrQ;EbI<^Gag0@Aw1Cy`ygj_E$H5B@S)|5=AXsqS7tvn~kmH!?~q35W@ z(p?U@BSG4?Jd*a5_@%?FAW@Xj-iec&ckgiU! z8En|M@TYayEEE(@*I#6-PJFwImfswAw0kzly6@RekwKyr%U`yt$Kur0!n?O6PL?x_ zEa_51I&~Zwo}abLRP;A^2E+Ka>03;o)4IiGpml%*CXyt^KzE!%Eli9a%}p0IS$+@F zETKyh2N>u0NgIa38GZ%BTzUWb`wDjkSH-ns`~d0+K?tw%HyQ|gFhZ=?4DcF8xl;@_ zMk{gtjlnv&&QJV;4)yp)gf=G*5mq=rWQC|hVRh;dQH5_YLZB)9jLsX8$?-Fhr|3FF zEQAN&{r&soV+M%?iASIj_j5Nu5`b;dUj zps!4_#BPPB=n>Q_s87%-LH&XT1Puy0EodkxG#nIq5nYTp$^y=@&d*BjrgX nG0ie>5lO+Lwd~C@7jT6hd|v#Q<}Lo!O5wrEvAs=)5T^eD7{O%8 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/task/BpmProcessInstanceConvert.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/task/BpmProcessInstanceConvert.class new file mode 100644 index 0000000000000000000000000000000000000000..22a7fbc59e3e392731f6e48b5132221fe079f111 GIT binary patch literal 9704 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/task/BpmProcessInstanceConvertImpl.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/task/BpmProcessInstanceConvertImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..720eee3ec3ab73d972fad126cb502499420c93c8 GIT binary patch literal 7745 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/task/BpmTaskConvert.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/task/BpmTaskConvert.class new file mode 100644 index 0000000000000000000000000000000000000000..a9620fa1175e858d7515d8ee9ad0df3f9f23b420 GIT binary patch literal 13736 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 NdYk^ogqXSFe*yi+4H^Id literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/task/BpmTaskConvertImpl.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/task/BpmTaskConvertImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..ae80ac57ff8a92580c1757113e5800200823c9e2 GIT binary patch literal 5783 zcmb_gd3;pW89fgYhD=@}Nmzse5=0xwGLQhJAXXqCfg~WrP^k-($s-QTyccKQps|Zq zTB(SLm|}}vthIJ;NtXenwXGtycCV%TzVF-L{`%`VZzeO7$z&kaA3yFp_rCjm_niBE z-+edp!hfH78o*Ng$B%8;@59^tn1KD|csuU(<2oEr_Z{lqCv;HgexXA`4+tF=dQj+y z&_hB;eR$Z9DmfaIWtA)kWmzT5L0MMGa!{64vK*9U6^;tMQ|Oq`BSP=;;oW}J%G+^y ztChFo@>VNv$ML8i@4Re+^^k9TnI8xp2go2UoU@WYDda%U3H5@mvv_0fR1F?iN5Dx5)IJav?AmrHlte6{cgYmtA zwf)g9KG&tj+M@lDRR$)frz|n>hPrkq)|WN*jDr2)fUH(6Eabn}i3Y=VV4pK$g)XPp z*&6J#+FUET&5HMTZ{b2EJyy^*FuT|q24?o8NRD=4VD-ynP_f+&hW2&@`_mgYEWO!4 zd@SkQTGxpU1U7ola7bIxX2;#29kRx=C9RHajoOl>W9&uIMYDS}+UL0TXtX(_C65JbucL^o z3s%=Cp*%#hmEB^XFE3P?$}XB|?_fTZ+|3-m(P>*ZXShjycgW zz60rVdd$TQOmNBkbY7mMt06y2^6RR|j=dsrx~S4#R?hENr%ar|7fgH+Pn-A>zHH(vc*exDIBViLp>smdt$&K$zeM%l2AT`3bb;Fz4}RrR zAQH6u0$Y0aSRvOy)A;-tm^Jnt%|LlLzC97|x9nc4*Sk&bG*FwSm3h(Sq+mSk*t`~dF z>9;cX(Y!MlB_AJWrjW@oUOB?iE?z2l10MFAaKn*6E6LNZ$=Nw`YSQkrQ{=otj$#%q zQl2Vgl+&Ne_qyG+TDx@_=iY3#1}KHo3JfHNx_!Y&!rHRiKuz7I+%!n@uO$Bwt4!yz=`Tul@3s7N&Mf#sMAs(b zyluwgUXbK-#K4R~KE0_k3OV)Ov=pes^pP{>SBw?Ml{&+ry|$OP%9f0{?C=y0XSI2& zm2)UNp^RQyt$wdM%Dl-bTW>@*l9Z{&|!dC(dBP4JR?NqhV3~X_OfJbGkH%Ne%4y)R`<) zCgc|?7czw^gerwD5t`C)f<7mDJQl%Ed?`vWlbmLw5_8#JhAS}-iz#a*=A#vtqZ3zn zT<4|uq`8tAZoEkmvY9Lb`{AR8dHEhbYyE(o~^oLeqt2 z2wf^PQ)rgZtAu6?%@LX_beYh+963!W=PHXRr=D_}NNX|0Eun~JH1G$G<@_P&8Z5&m zqAsk+k<^`$)J;jfo}`ViuzRGWKI}`&iZrcY$UzQMQ?%7 z6+%}MyxQP>r$$3Wr%-na3zJx+e!ZqN2sH{d2?c}}H=ILxV?z>4l4x$shHfR*fxpR2 z#?=hs8dPH)rZZUcu%0`xfoiYiNEi70bbX{TKx{fur0dl8{HjYVNo zcQtpyK|h5msO2c}zl0S&+%~72(_)CHcjCo#aCZ#{w}rc~)C;Z`v}FpF7v|+cUYK7a zKkQe6H3V9YQg}gAY67s@)RZf_jSu;!~3fRFg@8m|@$S~i;FyG8D-@@STVwi&z z-or45IBMmDdRIpBu1u)6!9`+ZsAIT2E$~2EVBN65)f9L&1t#&jB(Bkuv00<%+9cNH z5Sm2TWQini=6T(iqOD38s`Zm&Grg;X9LnereSQ68O|;+4l1 zA+J2P3KcJpd#UmO%i|!E`4G4M0hY&Mrv8I0k0UIPqb!PtnMH%lmSfD4N0|D@$1IP# gJ>3^k9uv5MI_mF1n9pjWTK>NkWvtg0>_gfA0QKl=;s5{u literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/user/DeptConvert.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/user/DeptConvert.class new file mode 100644 index 0000000000000000000000000000000000000000..e4af826103548d464246b6599226e483db542e66 GIT binary patch literal 955 zcmb7DO>Yx15PjaX`D&n{KtCwuqqHay0$;eLil|bMB860HA}$<|vvDI=yLM!IgL38< zAufmm;>?di%r2oUr4q2xuKmU{^E_{6zkd7l8NgFKEa6TScX6)*hWjP#GL*ah!SJx( zdCu^-7n+#c#LN{xjm%pyiX;z>o=NL@;-uwAa_T!j3@wJz;*8->&7Ai+(ZY4?%`dJK>83}+W8&IANhn7wyLzUaqKh4BZM^WxXveVG zGj_sbF?HT1q318fbgJ~kQQ-1SM2TeBxZH(2iq0f9B$witF4rB?#tS=Di{T%LiekwDN6+LZkVLitCs$-EOs~wCpGGNZMgK0SwhaCeRC&f_iQSo}^_mtY0Fn zv_$dfKjx`~2Q}*CvGc6g($WxAsAXy6rDfP( zNSr0;M8dhBqUBPyh=XJyJ2WD6DJcCZ+!}% zz_q+;Z$i%ikeZ>HAT%TYEDsa$1#uVYFSXcMGfyHa0Bng(T(>Ml}X@ze4v)BhNU=8 zV_C6RG~Cqip@v%m9W$0~d9woH)X17ZWWgy&fu5{o%jH^mQ@X3>W{I5stdloOYo=?d zbHI#vTUJ$IGMjhGhFf!XE#q#<`ATURdB@(7u4mM$(lr)k#an1FGUZBXT0jfr0(Vkv z=5$ncY#O`LC^%))vJGi_mbYu1M#HNls3OXe23JkP{VBU4jT{Rjko{l266c9_t zHtw;OI!Y+(u;B=#ThNZ}sE(I#UPlFA3h=T$)8!TD_2HFFyJ)OzZp%C;xh?oX(f{uS z5a@2@r&8nV(qd#h>6)J7a!iWSQ?1ApsE`W!>_^Qyc9A0)tx8`w5X?|Zrk=7yV7_dY zRZ|Z+$a${iyU^&Op>V3*Ng!{!xoNt5g%pki^;;)}kaygx@e%B66(wVPvM8BBXT@}@ zl1G6jy3g=-@0g{UTw&9bsZ4v0ffJ?~W6WlhWIM{HivMs|t#6nnUvWB(&=o7tX&uxu z*-YPmYMM2sb8XA4F3Ycpkfs|nwf)24ay7b4?XX#35f0@N$VN6eQm6U-5*XsgEX&;z_r6?c4L^d-cZ1cILDJv^T|~i|3K|9_iO225Eb`-LTG&t z;ia*39g#YsYVW8+Q+sC}F>djn(Wj2_he$kxUPss1eL4vHK`$Xjx*rh?5I~Zj$}?Pt z2;wX*5%D;KP7{U#b*>RO07VFR85anvn;BiCb_lOfD~3?VpP15+qG$x8WGmdi(KyO| zzKOegJ-v?}fj#stk?Ipz9^Xg5z;_rXF(B|5gI6PCqeIN=)IO3S-2VQb-+k;6ANw#g zI=n#a7l}Q^zW^gdGKv|bF-PPI^C+bTy2w#_41a)r9%IzuRp2_oUx<2O?jc`m43`gN flR;)Z2zUI2K@G2Jc(4KK16m61PId literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/user/UserConvert.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/convert/user/UserConvert.class new file mode 100644 index 0000000000000000000000000000000000000000..a5e1e5087270c1e7852aeb7b4ba37d8dc7136da9 GIT binary patch literal 1007 zcmb7DO>Yx15PeSCd_zKkeo#uGw2%scFC36kMMMFlN+Fe+z=e~uaf8dQ9j(1V`3L+R z#07CcocU3R@is)ER03AoUBB6RZ#>Vl-+z4h3g9v7dE74H4t9!QxSPi=L*cO7>$ke? z=L`=!fr)sWn5p6?p?NO`q2z(l6B#?6SQ+zI^k|<28Vvc_H-=$-fl4Bg(L{_TADT!g z&82qAO?hu>y+UJuA$wkEmO6CitX69nDecpbr5!)(A5p>0|`6St2e^M3_j1wTn|IBWFz#2b7ZYjUEnZ;Y2)=jv=E6 z6|@T8bwxxO=~Ac5^7yUMQYX9xsK!uc1o*C0omiRmM&Z(FPo;Ve9AEN%x1sIAw zHPXT*G0{JqCmbyJ^>=rhf9Nmr|K0B<*-|(NLZy{!GGy!fuNjuwW=LDCc9fRgwC}zj z6NXYRHSk4hKy6`A9{FAwRxhxgMuFkMe+*q7dllTn{R)=F$>3KS_4 zdP46K-Q_pPK1Br?T(Y=~szq%MuXwmJhgUtkMk?2_O3XD<$zYunH*kaSCgELr`>V3T gJ|puD>=Uvd2{YWJXDxJbhy9TVxeNGC#g5+gF46x}J&Jtfl9B0UqrW;`p? z(;AF&jAAT=S|mg=%W($dGE8W&#PF=B&uK{d_v{5iNWl(agIGJFp49N1hO|Pk+p;aU zN1-g-d|bib=Oj#ps(#Bh2eYYBGd*mKCaJ0EcgBq5aU*TXJg@Y-6IMoHTmP7oil(#9 zv=u#*bk2)Kbj-1*%(NTLX3TW-7>B-`B9@v=b}4B2b%jK@#GQdsj!oyZ8BI7T!?L5M z?ON`1G(Me?C|%7Z)`iXTrU}={?bn+~S+M2RFz~n7jpALP4{9Dl1(1${Ti|gJHtyNm!^dg@8YiziaXi#n0dgG zLtnS(+_pbM^dDD1e)nP2W0>X^cL9TzaI<078d@d94d@sda{i}VVf*YK*2*YLWI zOL#-aWxT25ia5O`PH*F?j=kvB@ebZqs4bojuSRla*ibs&!}|jB105gYBlee$kMW6) zPw|;TYZ3Uet*)aL5gni73s$PQ*&!n{!31i!rsGR|<>i}q){5>xc7_+wYmA}Mv*s9U zUV^RAr~h`5DpVE`@Gj92B9k<6MyXA~O3u~MS>AOtEq z&hOGt3Ul5>a0--t=A@StahR7wg5SwdPmHzU<|PM?$BE_zlbX_dEzVoQ`{HJ-i%cXx zzeMaOPHwo$DpapwKEd8cVZN$Fu1y%3LGuFX)Ab}>ZEv!$Ai2Jxb(xW1MHeJ0zb3x9 zM)LAIV7L=BPZ<|@g@t3W<`{n_HuFD5;XeM<`1$7X=eU)#DBp_jK+7!Dw-kJMkYmtO zRwGPV&l#W@Eu68#5#cD#w^7CiIMVtf0_w*1@Quu(Y#`D)1OE&H@(#{GllQ6_tmZ3p z8*5}5x{30e&}UE~v&tD%MOr9S&tT0AY9conw9Et2j&jO%L|so%Yf*`HXux_j@lR+g zHlmGE7dD}f42Q^10B_2H$^#}mTD22$C9R@(h!nTcYBhYpKe1E8b}sGUzn;(I^fR3U zT<)FcR6Ei-hdPB>)DKWqK){)IXX-5MVKgGuS3sgGOv<>I9zwfluLfz>z9bHhmMt;;nV?W_LN)zvrA zG>dx&S_j*1b@?fNI%-R7QBv4^0?U&OWqDCsWp8W2x?m-PP4UcPh Vf}>&=|IVcx*M2$;(A0;)e*mmo2x9;M literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/DictTypeConstants.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/DictTypeConstants.class new file mode 100644 index 0000000000000000000000000000000000000000..b5f040e31c4e5693353eba732868ae328fb12579 GIT binary patch literal 318 zcmZXQ!Ab)$6h-f=+8L|WA}I7H%*LIoQmYh;EzKyn3Yj4eshP}>WR(7z3qQb*5@SIu zx;*#3`|jg?e7?N{xWI9Og9L|!(`>|V#fbCi&7D}@&n6;!TucZjvz2_2%t%`@9-Ox2 zC80mAtq;L>5dzqb=uQCG~h z^;)whQ@_gGs32D`-?QfVN)>@H%$rIC>7Rx4UYAz5)+iBTU2OjA`NHWY dbP0q`{6q&s>|i(2-iEr7lPyl8O4#3cy)VKTRm=bY literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/ErrorCodeConstants.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/ErrorCodeConstants.class new file mode 100644 index 0000000000000000000000000000000000000000..0a0cf6e214415a31742f8fcfc158db5b46fcbb18 GIT binary patch literal 4581 zcmbW4d0bT2701t0D=&5$6t^mBfVhAW5w~bGro-@nSBIH#W}u9Z6uOK zXeZG@VuJ`@CvlNPq6n8rbduO8LKlgbNhFEz3W;tMYv+s5L*g=t1tPpkqL)OJ2vpbKZ4!4#{6mC65PfMFiv8E#Ggc%B=HRt$*+s>O%mTC@rDTBCh;8<+pdf7T@v3T zaYKaflX#uPfCz7pxJ%-u2yc>ji^MGven8@fC?uN*KO*sC68l8>35lPgkavjiGZH^1 zu~UR!koaW;{3-%|Ex^JIIX%OSf1NHlGuN3loXekil1@s!tPe1%PbkQC6~qObV*Ok4yQwa#YkO&t+AHM*vkf5q1R|%svW;drRe)HH zVD@Hfjw{V5n_X#EdnQlMWVNKp=>jB%`Z$e_y)L7}A*Wkh_AIluOb`6y-p60Iq*-0k zzAU2|*D}Utx0)n}0~an$w#ZJ|YT@iD-Atp?l!3X0aj_UPwZQ^DE5rHdw(OBG?Xtt= z$a2^uOR9t`5Ff@P#g=JtX|vETU-Sc4+2U{-EvA{(u`Z<5L_WBnw5s|zYyPXJLo%lc zu#%VAWX-ghC8rb`CoW8TU{h!~Ey&W4y?Rj2(IGARB$sn*mMzt&ADS0MfWXLhN!RGY z>88r;p5>76v}x%F5^|XAvTf$Ei40cSgNzn1Ns}dWs>?3fFghMO3g$|;TeEC47E#Y} zM^j~oF~uwiAnwj9$S=w-+atj2_=Ic$=FpiIAkv&)!#f7ZheJ3ofB&FTS>ro<6Fuamh`|1J}YJoJb@PUfn6G-Q)3sg*A z3o`5>c|1!Efq5w1SJZRYmC55%U88%ROg6xmsd0lcIDnJT=B6!%+F9f8ul5d&+^c9l za`axsDV%}UVt^7381U=uQZAmkDey+bFJ+S9&~NoA};>ArjpPxI&R zKwtb;ntFW4hVPG68la34$;0LMjSpz)DBU;ww|Xdc<0YlN^ZrpBf3-@fz3{&eeVB&c8JM4^%{yGDr4c$;?_`%+*^k32O{aWkFDXr} zO8t2QJVyaC^#N$7dH2xS&ci zo}QNfKj;e-bg@2DUD&W@lMo`V#cj#-`0LQ3fY+!a=`1*F;=Dnn4*>qLysMFlFJqZq^?}vq`)-@={EB&4! z|I6333N%0!#n`T%Y4DDBgf5a=H$mgoUe^-W&y+8V?Km+SHEMcpY>z zsHXmll@q;KX4-Y4YpmhaG>oqR_f<~U$7n&uC-G|Vj#T^Fhm_`e$_x*Q=Qf@eYBxYF z^^Vf658r5&E^ZyQBxuKK=1m>+#?8Ss@?EY(Vt^CWVbk088TK1apW7 z{$iiIJMieo$A@XxAxBwBe$l~Q3E6W6{13wy@jraAu#BE${8~n@FI#+K+2SM07GFxX z_!hFor;jbZa%}PCVvCOyTYMwf;swqYZ&S8-DYC_Tj4fV0Z1I+1i}wgyyeZh?rN9=? zIa@rpZ1MQ9#UsQPPXt>mc(z!yY_a^BkJ%h&x zY+#VcU?YJf2Adcp6WGjP3xll$wlUbw;Bf*w80=*51c6-)b~AXAz#axqG57=lBZCwM zCIYDpBnD{&(ivnhkO_Q}!CnSt0+|df46Fof4E8aw6L2taGRPv3&EQiEo+gmP;L{8~ zL*TOvKF8qm1Y8Whz#x~veg=6Ao+037aNq{lQX~Fbji(FXVOW5F;vfpPz(PocMUVxH z;Q%bb9|=qG9$F3;AR4a13K)fz@D{9scOVA-0*}JKVYM(H)(Ek%R@e;d1QVj literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/definition/BpmModelFormTypeEnum.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/definition/BpmModelFormTypeEnum.class new file mode 100644 index 0000000000000000000000000000000000000000..62fdd3f4a6f51039284d78dc09cdc193e922efd2 GIT binary patch literal 1706 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 u8oJFzX$={3F|~$%a~*<{-D{9eDNtw8M@i<qsx zui8Kx1_mad`B4n_%9@Eo^2z$JyLb2Ox#ynU)t9dyJ_A_9!%-+O3iuf}@|erxmjZPB z%FQCL+*C28;#Pr%w>iF}Vp_#r9^O+iqhg7N%X!=vn0@|YqgFS~_S(AHtghGU?do%y zYSjg9HSEAM!d~#VW4v~QzpO1+8g>v$L;5{0GCUbYR!15SyWX?@uuk(!0;i5}zNkN~ z*O~$+npN}ncGWc396|Q;j}i;0%eI?}$&`5WlK~W8isj)9SfR5om_aYJu@${(TTw*D%U2p^cwzkDs);4DQK zspqe-3-Lcj_L2FAB7t+%=90BB!k@>56u*d(WVWCymG{W*Lt*^LA^Zl4fxrDVx%%sNh<<`3;%E-{X%^Hgz?lkM5!H7Gp`MDf)>d98bbxR-Z`1$=oMG uXLd1WZYsMNH#f7pm^9x)a5a4g^`z1wikKjWCUG5OR8wF7PZY5;3*P|1`EjEF literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceDeleteReasonEnum.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceDeleteReasonEnum.class new file mode 100644 index 0000000000000000000000000000000000000000..f406d74ebf295d894009b00e6a1c042b6e96de12 GIT binary patch literal 2598 zcmb_e-*XdH6#j1WJ8Vk{r9fK{5N(nQu>$^T3oU6VY10;)wpgp;Hf-C4%`Wb4a2UrK zd_oAgOfp|`P$3B`ohSdKZpUo4sy(?FnztmO@@MK z3?j?-!~D%i5W9jH<&UR>*d4^n0gN$h8cOx1lEY_)yRv-@J(--APv`~hlAM@Sw70|w zMM~r}T}nu5A#Ws1(Kw$tej$HK*K(3!q*cQd)tuBVDUvA-Nur^tDFU{U-(=T7GL>=s z*7T2LhSS#OOlqK;!LM6jhDhd|cu`C!qB@nxn!2n`k#S)EXjf(=m1Q_NzF~~~7e%EY z84SJA*v5JFCbhE?!{&^vN&|)bgrpC1(I^Ki+~6d`c=X{MV;h!MB%|d-WmME<9@h!( zRmD7U@K0)bUNjk+qvI|PgA?bZoY@w09j!_4HK%3L$Sdkov}~)Pm~*txqR`8Z{MK%JWACc&T0icC!LUa88!cZ zw;kjiBw#O}7W)K@<5h;#hV-$3X1pfgG3*K8bpb6n!_c_A^wFKmGpm1ISuT~zAI#pp z`C<9v>*ddH+`Ty?;0+Ly3FPR)1e^u0!TvOFc>$A{60jRh0-nWl0-nb-!><2u_Vzm* z^n!p!G%@U1UZQBIB7s*vxpC)<%Qj&3$FFTXRsH=z#c!OU_Wpt0J`BE`qN$Yh z`to%8jp2y9DwB$67;UaPa(d8R*v>F*%&Cd#f~jeWJ&X&ctng_$LMy|ubzY}MJuAIg zkZ3HoRp?%~_qC=E(HPgw)3Qlda_EuXTvU_pC^72IfNZ4l7tBjkACI)>xZt#COxq)q z8YxZXL0m4cKc}MpYS|b5mmR~nWG&|DEjtW?*zg(BkJG}qslrBN4)VR_-?_= z@m1^aF**@!6yiKK-le8_%JI-a=qfr3#eassWeI@?9qa{nP;Y~9z)FJPIEVzf2;ph8 zI<8s?!|4a&k>C+=>8&4AEbj zZX7`$b^U#L`W;e-30DnY=x;OxaD;{Hxbid}9G*OF9>>QOdr++%E`L5pttvh-C^>$? zoy5jVs}OH#T10L9z%TIsM0>YT)h@+_HN~xoN7gS|=r$^xc62zD&Re+wJtcU@B3nF> z$Rf5bpw8NE^G52e?RIOs!x!0UZEO6$6WTM6y6l*D9^12HK2j~a0LBT~H5akS)?zcZ Tkp6Whn literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceResultEnum.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceResultEnum.class new file mode 100644 index 0000000000000000000000000000000000000000..3243ad216e61387b0fed8639172584ad952096b9 GIT binary patch literal 2197 zcmb_cOH&+G6#g1|rhA%(z(9zg;gzT}6QPYJzCgk-Oq65_U+3!1}MM7|;;Mq6(9*Eb+;* ziXjy%Duz{DQ8A+8H5I2+6cxNKL0MRw&*e>1LbgzFsMx|I*DHP<#um8xyn zUajmKf#ttvOut`Vs8&k0@6UREV0k5b(e`U@kSCRKvX)&SdnB-Mh?xO>UbM${q#z$Zer1;?}JYUMS%y2Lf4xP_zns}hz|_vV#;0I4Jk zm6GMISXGBThHhogDwFp%$IpA~4VonMMk8pJJxz^}*sz19((zO}5@%D3$O+zcs9p-G z&Ip=%4k-RdRD8nmoM2MIc+^{XJbQ%c2o!3iS12UY+3>8OR;4UbQA^E;j=rW&l-$sN zYl`id5j}3yiTkejBz2=^R%+FfebHfxhyQOd#<_ zeL=?n25DIDZhrOmZ=bU8-PYdM->}|Jeed%x|J?peLi^p#O#+SPS;u}bTxJQel3Vep zItSZie(joF3aEnTqnzek%lF44Of?%J0)?;4$bRieDpx$C7C5eNGz>4ZOo7@T)GXJ} zdN&R=Pw0L%VmE|%3=Ygkvtzl5dQ-DU4aL-1dSXw~6RuA6am@hQi2<=w{l!ich@Ds< z%_I{bcH#i&?B~v)e+3C-a0U{dpuZ8?`P1r&cAU=j~sEi zg?pKbzy)1ED T6Z9VFRGvm67cfQoEy(^0SAY#r literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceStatusEnum.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/core/enums/task/BpmProcessInstanceStatusEnum.class new file mode 100644 index 0000000000000000000000000000000000000000..67ab31df04bee3c44723b3e94bad3c520278d6fd GIT binary patch literal 1723 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/core/utils/FlowableUtils.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/core/utils/FlowableUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..c06df7fa49b32a7d10f5f63a9fbb830a4ff826b4 GIT binary patch literal 4033 zcmbVP>v!D59lc{G)>;v`uC0*jA#I7OmD{ zdB5N9K53zazTg8Npgkvqq^DndPEY@)p5BqPUhT$)(Bxz_n)%JWcYd!?{{7#J{{(Ow zKG(4e59)XbKhm*)MGePwY*8~O@NkYMKk30waazYt{7i?1B@MQY9%ayjk_M@xSH1LN zxtlgCYP6bxqvH(D<{;F5Pwn3YUq=OxDYzhms)kTQO`zMVR-F}3%Bn!Fc*a_@OxN;O z%!M#;yp@7L*QDb);gmq<$mj`yEi-;e3T!Jno_wHIUXsD0wd8UpSM+VmJz)iodQT>| zgsTpL_Z4lwYz8&I?wHH2e@rQ=l@ z94!g-kEA3PeF{{Xx@yUCwtZD#&~~L2YzUTLk>PZu?9KB;?yM`zq@&omk)j{0G<7b~ z#4IZtbKSa-BARjO!op}gqt0U%oodKaY>hSvTz>_4w6VpIPiivp zTWgu>7I&^WVO^EUo_4?_T20(*U8}rQvW7P*f#I74X4AZJhPP!+VS`i0Rq|lq+ftPT zOH-@FF9Ji~X-~%_s=S)(4j+gTpxKeWr;2ovDkVLt@g#l_xM5Q@>3z(VN!v~OpT6MN z0$a{GQMXfOs*Oc0Yrj%yU=Q|cSTk@Ak87wKxDl-9{^$fz#U2R)tFD@Zh9?a?g{N82 z)4m^8!@#P{%W%~%RkwBFnJk{=p)bNslm+^{$N!+91RRee7HGzrC6Myrp zGw?d@5!lgmXPivaHODQv@qvb)8~6o& zso_@!evRK4_z=Gp(AVu*M&sY3+tUS2O~;nbumD=7Vwp^&HOsBZW6NzN5$|Yp-`0}2 zv&BKwRQ5endMmuSOs8CNO{b(Tn6Pf9t_M~hJ7hZh8&$^~8frn@=wc91EhVXd=uRDJ zZ*v75zr#muB#m>NP&fU@0>hhkqI7w6S+*@J^w{CF;*BdMtz)_7MUBQ}TdLN|GGJHg zZ6>P9QM#O6WY)4*1@f@>&L}AA3ntyZ<493dvj!5otW!M_eVM5yGZ*+}fqdFq@lNI3POC&$ zwxnz$n@(!@gMm-*X9KgCGjJ3?5V)?%<#=oUS!oudyMPVITIMa{2X%JKa`R2YUm3Ha z@1Lzz7|WxcCxeJy<@bNmq7uBgmECW7C0AC5HLr@_!+_NI8xP0xHU<(8<5RHJ7mE(XX*}e(XHD#wRY4 z3Ev>ofr&4<+8JT=MwTPUkaQ3SA|ZR@-Pniyv?C5ZaDWmr$l+$(LY7-OI}piNYqzbh z-Ojl?xYmI?spu};%@w8SFlW?Z4LYHFoc}IB|01$e=ZGD7Fyf9;4IE6k4HP(1k>Qn^ zpq)<82FCsct@9t5FE62o=QN7rJm^);v*qX+V4A-()IN(W=29F3$@#M6pTZ$xAEtFD zM22KXxGk1`h-4~$+b^M)B-==l{TurDdmcH(l}?8FR3NvLRpnAQ?p+6)O2G0duw4Jf zSP!MJ@-bFcSKT2Iwm_{a%6${(kslksi0vx610873urVIojKJc+_C?|3PV`0LRHy61 zeYhVTBw*)B2yBW~1bGtd;IRikM9=udH@Lb34=avz0*$j|CeY7NscMLM^6NbLb)Gza zh$3+hNBc!T<1$b4*wwrFRDZg1*ZduQiXzVQI6wMH_yFfRa14LqQ|IcZ_zV7stMLfm SRpvZOo0B+&A15Oh<^KT^z#gIi literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/core/web/FlowableWebFilter.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/core/web/FlowableWebFilter.class new file mode 100644 index 0000000000000000000000000000000000000000..ca064505f8fcd3ace3fccc311673210a2589fb01 GIT binary patch literal 1477 zcmb7E*-{fh6g^EQ=@5coP$VEA2qCPkf*V29qNJoMie&(whoMPECNr@!!}3Kw^CbG9 zZ&v9yS?Zl7gvAm~ReHHO_ny;t$@ib1J_DG;;}keXQ^;W~g*?XFFo8+`G{rH^F~g8p zFm2OaWQgU*HyE@fXG=146ir*c^s1Xut?8QGl-($RE6@K z15<2U&Yq7DhNGm|lbd2Wvb>R-%ckW@Rbc4ca>Db9O$J-H7^?Z=TYXpG7j>z2E$NDi z>(<0`T1uhewS4Evx?8vqR}2Aps;kiEQ*d@CV5Uv7lE@G@d_fF-XLwGkWoR$Cy0P;@uLXJM5JB~Baopy(!!gT{ zdbV%KnrqU_Ip#R-lBwi)%8(R^%GZ6a3NwDT(@3I~<6au~@qpuD8uKVn%rqA8D2+w5 zGEDzhaT#VERTlM{GVSuV(yMaMQ9D5+YzNgBtF|FuNfnhdd?>>uUrw_K45RDPXkY6n4zo5zE)9T2Qv(u_K5oMIMWR`(-AAHC#)mQgwd($*ury7t1e2?@RaEu ztot{5aap=Wr)=8mQ6pyZSTYSdJIeI>JRi7LW2H>{7(yLQqW3kif)2-MtgkX1K45k za~rLRz?ear&gl?8M`)SXvf5W9v)U12)00ylp~Y~7_TzZ{Yma1H&m_G2*EWANFANf3}l9K95`j~E6BH$=B#LJeUQHv=Vegoq=9%kk>B?ClGFeI literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/definition/BpmFormFieldRespDTO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/definition/BpmFormFieldRespDTO.class new file mode 100644 index 0000000000000000000000000000000000000000..9bd9fdf2b5aea6981a3fc6baa4410cf165424221 GIT binary patch literal 2077 zcmbtVT~ixX7=BK&o85$^rIfE!RJ5r1fNs_L(FD*^palu-K!+K=H4cfor z50D!!bVlin&NzBgXZ#KR3@@$qIlCKLAkH{CkaM2%p7-;4-;@0I*AG7dxPgrvE@3VU z1@m%Q(C|Ty`-gJ5CGQp`x-HR?hUFY`nA5Q$_d7c7>R6RerSb`?{x&s^}63Ocdbrn2Vb^a zbKk0e*6}^lI#Dv$c)lO_Z951b(ye-6xm}y?=N!)o7X;FU;^CHgiVLh7ozAv*MI1kyRQF*)G!-<%W^!k5&CAs-jzX^s=3&1 zt@uG}#j)MSrrl{TJ-EkY^gTOVk2x4A6#KD|rN(wbgx)#P-YlF~F8<$avh?4Jm9anX zS}toGeLm8?oqf9=R*Rnq2>*27k}DGIhfO;W$VrZ-lMW9=!jn^U)$6Yj-}R1zTOpfR zv)U2Y9It3p^t$CO%all3)9N(icCmM>jCg-@wYJQZh%AE37FPJvn^Q?z{JY7s~ z%kKttd&QA0p6Jt@YjPqDOdxMy7?%x<;IhD+oVNekZv$`QbptPB%D`(9y&}8UnM=cQko! z=Vvd;pvo*2zGWcAKQa*^k^zne@c{QRe#SdwC`&-bxL)Cv$z5<)OOGJFjqG0Fnu&yi zm?So01-yutcoLEuuBVN^L+YC|#v^@=NoOD9^jYJ4U*lJCH8HM6l>%C+{5?{ov*vN) zV^pF_lDI^Y8d>i&9i^fYPB}{dYe)aUt&lU5r1%FTXpG-1m%fIo9F>28@)gwdQTb;m zwVBc})aXc%*_h!{QD%R|Kwf!_ONTIK^U5)_o`zmgsiIdhc~$CU9%J|rgLyTd=_x5k zjFraO=G~2RIN_j)H&9@Pqu4@`4NZ}YC6p*jb6r50NWmQr)C`e|8a>VuWpIuDr;ye3 zKXE~W`2=GcuJd0PPcRX~EWe`V`$-tM$z66pm8Hw{Qs>f(Tcn<(r!py-o{aq~Ov zR9aVts&mSGMAzg{wMnw{q4FWL*<)lsm2$nJ$cwy^+~|@*UQPC==0@x_iJiuK{Na3u zF;%Yb@{{6E;B`tRHCHi?_lcBfaCrfRO+I literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/definition/BpmModelMetaInfoRespDTO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/definition/BpmModelMetaInfoRespDTO.class new file mode 100644 index 0000000000000000000000000000000000000000..726735aa1a76c4846203dd58d5e7ff9371576cca GIT binary patch literal 3587 zcmbtWYjYE26n-|@TaryEDQO851ufbu@qXD-gqDld6tJyTzzf@C+XRwqNH!oJopBsT zzUU`s`~mU>XLQssqmF}qz#0DlKR|zmi0ABXvYTWXJL9xH?|IL8&b#OOzWw8$AAbSx z96pF+FD}HO;2kFKGI@{5`%Eq>FygdsFexy!$ehV!lgSp75-Z9IF2yl~3o0)2T2WEu zwWh*SQCD$AMS}%R1-5`h(QFnP<+@$AY68;vOU6|rS21c^xmCMSu5HZ;P&chcb?tiH z6fhjnFV$>w%WM!pX24QWz)(P)w;bbLtXpg~ZL7N2Fb&&0YuF|FkS|^@o7Y^Se7IaI z+eZXMCnnbgh>KQ{zPTe`u9>G>)eW<;W^7bQlFnNNqq1%^%Dnf4qFpLC1uWzXRyEgX zS=YYEhY0bpN8|e_n^#&!g-X)n+nJ3^X2G7DJTE}7dK=DD z36LziWHzXsxTNDf4Z71&^~#bCj2p^r(v*ww1&JQ;qtOkce<9Q+8IK%VAM`+2F`|x69y#;{`3O^mc20D==Wt)}IWOX+uye%ip2K~`=e!&qm$cn; zxG(t}J^UPbyXSCU^Eq^h5T(Wx?obYsl;?ZXQ(r+6Z%zLSThn)-EYD1}p*mY) zVy9<_=;G|}NMyyk*m)D$Y*uU|-a#aEi69AG&Pt4t?;?E@$*h!>10YRT2&Cz1R$-9p zft0Ko044Ppfs%SW8)H!11I4oO04Sv=2$a&bY=S|W2O_Qm5^qsxQS{I0M{da7#<_=i z9HE3{@EcyC*^SWHe}|(ai_-3E93xJ|DSU>~ewT5S< zII6MmLaw-Odcj2T3C`jT3PQriSRpP#ss9kG6kwE|;u6+~6X|amZxWSoj-JkS;$+wu z#aqNFPFem$`~e9>DvKacDIf?B=m>uwVL=8`2hhQYy&?}L2Vg$3U+RICUa*f3QFtcm zfQfGvt|LQA6Ftdp#c`EV#gAM5Ck&mtjh*Rq8^c_9TGG2*KG8-dJ>v2-VVU%(%O@GL b+vQWtXNk12r~k?izL9yaWCE3dyY&3OzSXoX literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/definition/BpmProcessDefinitionCreateReqDTO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/definition/BpmProcessDefinitionCreateReqDTO.class new file mode 100644 index 0000000000000000000000000000000000000000..f6572b100c716176f37b48472b0b055c1c2e2c31 GIT binary patch literal 8848 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenProcessInstanceRejectReqDTO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenProcessInstanceRejectReqDTO.class new file mode 100644 index 0000000000000000000000000000000000000000..04ff0ec2685f5f932cdc6ace1ee40ea2d7ad32a2 GIT binary patch literal 3746 zcmcIlYi|@~6n@^_nVs3)X=#_Xbhv1%6?$b66?IFoTnbd%f^8`mZ%lWG?v&k~Wp@@O z@qsELTJXJAObkd$q`Q8;^b`!?}X_2E+_AC)KSi9oJ?{O=dy$ZJ%m~ul5vbbr(`5$ z7@VD!aa=}D1|LLEYAGXO3!`yTSUJteSo!FPX$_>a)+stgO4D81 zl+Mt4@7BkQ*Um0{|JD4Rvx`?gUHtirh4ZuXx6j@G`u`1Sq>xHcM9mB5FW$d3J3lwa zgSd3<-kDh^6ox=f*YYMMc@16u*GE_G|N7bDh0}|dZj}2H*lJ`9Ya4^Vy<>tw*l8vx ziWNgfMjt7pC-vO8MhWL-G2>clLdzNa-4zN}(x3_)7>b+eXs%$MGNQ*)=KCBGO_*sa zKr~^QQ8(?;9oh7-Q-Lu(lQ@{vGyBWyKdMjbacfjRv1@!EMb|K;TNMbNu!i=Il9Vm= zNK-Z1DM>9U#!53DDLcT;d5}<}Zps?+o);VK)%SKhq1cN$oc-aKs;L&WR5$_998n!QY% zw5Xm&j;_u&D;v6c2VN!%+pyx;C=R0dT0=RHA0O7TcI(q7v=^8h*D?b<5e#xt%O{=b zp-~#7IQNeZ4i8Wwh^Jx97g82w#4D_ZN}=Pn(pKi4)K>IGwoIq%8A4-bAs5$o8+=AL zKh}V4R5hRd4kvFp;I!>u#yg@r?H|)ONR&^*aJrAo{O&{f5ABcjpWw`;{1Zq?<@g==lR-Rk(}#t5AAWVFtk>A`}w|5{d;> zkz)dP5xI&QRa65dpm0ngP&g*55(mjHNK)kzkP-_LNQs5iAP0q9P*4pOLBe(N+FvyR zStpvMa^6HAwo$B&_!6(tRQS+_&(TjZKgE9r+ldpf2gk94I1$6pv6Hv}QrL}M#7W2z zetLV?WXu!9Ma- zq=fG$&PNG6hBqk+KlS)9Mu`*XzZqjhMI6Q|j1w2I)BG*uha?av&5TiM!PrAGTz>a) zK_kQ>pokI5MJ_B*0$WB_N?ov24qnE4D9D6jr0qSh4zF-|6OkFL^4?(1((iY3%rIg1&cS~HVaQ1a literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenTaskCreatedReqDTO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/message/BpmMessageSendWhenTaskCreatedReqDTO.class new file mode 100644 index 0000000000000000000000000000000000000000..a27d861835cc3a261185f3ceb508f9320a4d0576 GIT binary patch literal 5330 zcmb_f-E$jP75}X)X;;$f(~grR3AKsa`ZIzM3apd1Nt(8%wi9AEp@q_Jq>ZguAC9z6 z8@@_Op>5JMDU_!4p)-AH3KL3Uuw9&zp)X~IXP)4R89VNOzzdXmp&)xjx-z(Pv?8Yym=*8&>B6ymYXLvco%XfJBE-&BX<@+*z5T)N|dHEqb zJ;&0Icsa|<^Su0+m!I(R0z+Pu@lq6BIIZ9u|DIRyvVsdNzM|lwf=dcsRq#^xQJ%?8S=B;$W>kW9WT9*oO6Ji*t#G_( z?kkromSGht<(h=MZHH&Fr;TDEZ!5DoGFVlkP_}B>$%=KbG;f`uC~}G3Wz3iodT%T~ zxBl+=wfEjyy?%cE&F9xYe0}ZW^6Cc{J~{ix)elzIf4L%Id+R%=&&;>Jx6`xvNu%Tw zdF|4LwO?OujU1zFtJb46v&splTnUd?>~!z?pD9f)6pNJ6mbHtoezLf{y0XGay!P&m z7nemURNKZ&CJVW_vc0*C8-C-B#q}%iO9)#=ZH_nM6lL|=wY76^-=ZiAgIh9f!2L2! z7`0kqrfiz^Wc$|Md;7*qe`w!uYlXsl3gv=zuY`{NfujiWX2l#?giX`aK2NE;YQ8&uvcG-91GlwQ-R&=Sk|#qWEl%PI z_g?JWZ|_dH1dSKE>PYHF9Ua-JBRzTR7&cEW7)2t~CiiX+9Y1O2tkHqTB}kRlspXg? zgeumoS(VVugzWQ-3(kMD4e+yU@pBP-)O?+UEoHma+yqMtc#BSKsG(|B#@UdMC1ywF z)6W}MMbP?VykT@2-82hDrw#B9gDwH{s^e3Zk(-+^=I!%Lmyvx}C^@5ikUJqknKf#& zVkFRf7^6ZTJ~B3OkUBwfxN6oGiWarRudZEQrP!a^*GQSu%QPRGwUPVk=|*fxcqaS2 z+gGfLm;(}`Q{anPF4J)t?(x{T)O#f)hI?FXvf*#gWOixm-6C*gxJUGhHbXohs0s(SolJjJD}p$J0ktg#G{-0d796ee@KA zggfc)t{Uz}KPlpnI?+@Gm?Q1j!FKE*p@>Q@?1+fF(}iuQ0?arU*4e}7UD$>yz{G6F z?rO*GA{6nr9lN_7yPHr%a~Iax+&wOAL)A&S(7B=Yt5l>c{Yvy38eE3-TM`1$X${*- z6{94p0)Rc(OHUGWpVn6A-y`t8$GOYbnU0H(bBEVi^L3_U=HndnI&bxLrsL`39P&DE z_jRUY?c*HwI`8mxrg7loEPI{%eVu8H_&6(G=bgUJ58+|Qd6XI(pbp7{LyHIudfi2* zy9LDQG#@ZNUQCcN|3DoZ4#a1V1APsT(8}a#*->!Af%sr@pmF~lg*F_BPb~+U@DCK; za3DU?9B9%%ki6kQeBL?GA^#dF8xF+7fCJG5MJp9M@Gx~YO^3I4Xz(3`f>(z=LgY<^ zI<5}=36Y86!6iiPM-pO_!?ftZkv}7*1wTgm3e*uTxCFJ1i0dJO#PzTiVodmBWUe5t zg|x5>l+a}YC3HoT8KgKMSyNn~q#hwqQjcm821OlEM2osWDLqD@l&)$q2B{8+Bo`>H z#|f0y6Iz@>2?rF{5-w0iPZB7jr?e!4QVuAorRt#IRbpmG+X%cWidR1vQZJ5Rib}gl zQZY?qk#C@zc$9P<7{otul%yaY!ryR=q!4lJulNQ@VHB~5$4HV<#c%L9N%Y@+T*5a= zir_NN;aem{v4lB1K~fAK!o;^pQt@~Crg)O1IQ~JO9Z!*zpwZn6gQTPs##S6BDTN&6 za12{yEFEqfMal0rn6SyV_0!y+QhlO*FJ zcHtCB3VwynL?tEd2W_W literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/send/SmsSendSingleToUserReqDTO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/send/SmsSendSingleToUserReqDTO.class new file mode 100644 index 0000000000000000000000000000000000000000..4dc3aff3e684a75dc0412b9b890d445f359ce490 GIT binary patch literal 3449 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/user/AdminUserRespDTO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/user/AdminUserRespDTO.class new file mode 100644 index 0000000000000000000000000000000000000000..df3fa29d05ee74eeafe81059ccdc4032a3930754 GIT binary patch literal 3988 zcmbVOS#KLv6#mAU&Cb|K-6q9Qpi3xOFN;Y1U z=S#A@Y~vLhCvCiH<24(v+jv8c-?Z?ShJ;_#&_8{w@KhmRDwG%U)0OhVqz1d}&mS)r zmOKsFHtB4=>IZVORxi{WH4UrVq*LX(x8PN&ZWg`e`V>uD%as}diG7OLsQab-tXC%~ zIqNT!X{E|P+0ZH7+GaKIXuB9)s?7N%lFhAt*{^Tc&@()8Sc85~rO4C1X}|34Z7j`s z)q{mO;bo^Q^M%skLe-ajc&OJGeY&${dcLxhuQn2(W|m`rBQ}tKJ4Hmfa&0{H5?fwY8~_jglIY(C62aA7|li~Y;FV1=Dehp zYICo2qY(&oqivd%m_I&KSPq(;w?;7B_I#ndQ_{c?77MjSH76XJDMq>fz|_o6rh(Y3 zdbLKW&Up%AOb)fev4$ewt)a+Kd#LG@H57SZ*DItL&#J&MOPtZiL<%KN(;eyJ$cU(@aPuwUHf;BB0es6u@R;eOR zjjs=rr35Nn#M+1hA6t0&wp>sx$a)l(m%TC@Z=!v!qSJvYsH>dxERcGK#UYy5^}Cr|T1c6Yl+I-$l72R}J|C&PH2e2F?!z>*Hh@1d!+R*f?)eRSdDg>g?N{t0 z(y<%ou%E~v=X2aoWTJ%kae&A|6{j#uWa9%I$3dbbK7@xuL@BIb7Y-Bk;yc#)0U`&# z^J+apl*S*p0!N85c#!^1W5BY{BV*yA3mCNUFn_jo0hcM5{7;a6^Zlnfg6}$h-}4^~ zX$475;v%hUG*ctNN@_HDJt{MaJ=9B5`*i>0R()I0KI{4?7uh^^|JWH=6DwHtxSTt# zE`ppMN^EHbo%L}pXO9#dLVFdB)zuPmxzqB+_}AEV8810!NvbCx}ec`N~=# zvhW_(QZ6mPGHM&){LKTMWz;m54x08ijkWd(IT(Nu5h9M>c@)yjHd>3>eHxOjPI8MM zBM4ROf2p*z;$@*?GET9jdr>jfnQwmasIJ0E=3KrNj(m&k(XX*OyJiIg(p)PT%&t|K x(?5{COyRVg=M?6^4`we{xKHpE3itDoUB_hwS9ZS$+dn^!gngjlSTGgG{{e5H)_ec} literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/user/DeptRespDTO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/dto/user/DeptRespDTO.class new file mode 100644 index 0000000000000000000000000000000000000000..3d03e3eb9daf32ee81e933082f93aab4dc8b0e48 GIT binary patch literal 3310 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 DjKwaC literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/definition/BpmProcessDefinitionExtDO$BpmProcessDefinitionExtDOBuilder.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/definition/BpmProcessDefinitionExtDO$BpmProcessDefinitionExtDOBuilder.class new file mode 100644 index 0000000000000000000000000000000000000000..9e947f2fbd10f9b8edd31dd088f8919055a521b5 GIT binary patch literal 3522 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/definition/BpmUserGroupDO$BpmUserGroupDOBuilder.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/definition/BpmUserGroupDO$BpmUserGroupDOBuilder.class new file mode 100644 index 0000000000000000000000000000000000000000..7dc486ada9ec8d0260a129ca989054b8eaf8324e GIT binary patch literal 2464 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 Sfp+qbL{nfUD4 zd%kn--E+Tl?p0oX?V0C^=t}xag4WQ(aWd#TEWXR)5f&#|e2>NVS^R*-4_TaI@u)^W zO5phzyFbp_6AAjU(tg6PPqKK5#UhK-EPl%3X%^40c$R&iW4Grt`dNaO(Zf1D&(EK; z<%~`*@XIfB`lU|4(&^Vaoz>|@onF%EH#)tn({FY99YfA&^m{>F#Th~A9Y^hBcA;dK zj}&%P%171-GWXTXZgI{zP^=ZFO3tQox#HSxu~M!HTD@bsGFO?W=-X~S=P2C*XU!n`ABtRuJanMt78Yt#P8-GY)e*RHzzYfg0>D~*)xIfTmi z7Chlri*m_*^VQ0%@a;ZmN)+=BYOv8~SGV!6oc=TUvycAX=Y>Vlvcux=e6)$a>A zUWVDJViB>bQ>&L;L7G#ZnM6LC$*Q}>Qei9N0F^pism>j+t401~#OhVCBxI*O4@7*T zc%+Ppst9mZyLG*9V((jgBfF=LI@9jDjfk}$5 z)AI&ZrtMNACEj4foh>3e*UDv7>y?FK;c%&PJ14C$Q<<}i<$?p!xeEo?uH9PLJU_Rm zeMh$)cen1wDxyamm-jKVa+NPt9&>wOobYo-!@w>{n`|4tNt`SIftfgZzBSxVpCk-h znhc8L4f9D7<=Q4_(TFkoJSHWeZ)=dL`2~JDrGePq&ZxddU>~Qh366Elq-ag$Z|}-r z`c_@^-t-`Ld-^e;(m&y!&ddJAPnbpgu-$772X+qz-|pe9v3ryL^))u{;ujdl6wS?t z-rW2~`dfQ{@sp)7L4Z4Zyw7S<=i+C{x0g3sgjeFn0&{gp$%|i6%(<;@m(X2u{!R3{ z=cZ=U5X+$7rJcHFn^U_jCEZ#FTaUuF29gqhWy?-wD0Aag85@V4XKcPo&*lM`Qm)PS zj5G(1z<>eXpy zd$EKwaM=>a)f&E92`Vgo_6F}v+${N!J7$;a2s}77KX=IcwnIhlZgU6+HtrpE-L0tB zf}D2aZYm7Ap2iI-&{~78W^oO@-Jo}{uvn~Pv7R;cZDv@ z*sfiXH@?DVyXI`g{WsmpiE3H6gxT}+m;@Jbf3IsEs4a)KB!W_H_(pH^T>;whbQykc ztO2PK)Ww7%L2toZ3eUIV?`;l#30(yfXole~W}1A0Z1M@d$tNf$pJ1JQf_U-??#UWtzJan^&*5J(s#NiyhsbBMEI z?)xzJA-cs|!8!!%0>QMA(Wj_uBp5~!HLoBZQ33XYaWiDeMiwb0Uq?nyQ;*UXsdv{H zgcVu+A|=hpS?W7U#%eRNNa+Tm&x!)lXT{7YV`671bCObK)QtH+8A}5wW9g>GAl(CL zrtSmvTXBH;t%Mn8P{ISn&4dq>wUPj3EyGMQ$nZeWe4w0_0w`zom?;MJc%YQo;{)Zb zUV!pe+U#Xe+5`2PX&-37>H}!N%9wo&%6OnYGvfmdTKxbGT3NH7L0J#fZ)SZU)5-y4 zT6r_apu7jlnfV4NatizC-}OsnH~WX$wMH8fC#Xs_>}gKiOD@jPE?Of>RL591T`3OJ zF=!FmE5_+|Xi>URY@y@OVsuOl(*iV&?hwoA{m^vW$1MQH z>bGLj6l%u8#x$k1m&GyL=vDf}k7pA#x{`W#Z-O;^yr z==0DbXnU{F7obJ)bLDAz09p+6e~`WiO#|j``Vusq9>p2(WoU7F2{U{JT7v$D`|?+z zCFwPk+6ia|Q?ctHWzW$i6af{50Q*RYbC|UC-o*nsiZ%cZOr(9tgT;KX7BV>MfwgvU z3m;5@W@=oI-15JU7mTP()(B+ zRyvNM6`40Fy`S|fl#Y{UMdr;)&#}Hz={RgI&#Y4V0PBd&@dsHSRl3PK&Z-rQv^sb* lUgo=CKj7797kyne$ZyCWhiQDvgZQFv*63UG&_Vk4e*xy347LCO literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/task/BpmTaskExtDO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/entity/task/BpmTaskExtDO.class new file mode 100644 index 0000000000000000000000000000000000000000..305562deb8730e4bc8a06c3b8ba0f2ff6d6441c1 GIT binary patch literal 5884 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/form/BpmFormBaseVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/form/BpmFormBaseVO.class new file mode 100644 index 0000000000000000000000000000000000000000..2a89d88afe0a299b4d1eeb78c96f649d39525638 GIT binary patch literal 2876 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/form/BpmFormRespVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/form/BpmFormRespVO.class new file mode 100644 index 0000000000000000000000000000000000000000..58b97e30b2d48976e513852d0be806df9525f848 GIT binary patch literal 4000 zcmb7HZF3XX6@FIIYFF|KjCqR!AtH6buh~FpQe>b30tr;MDR$k4zGact_7X{}XjcZ( zv`q>LBrPv(CuN3w;!8+7G($V7p+GwAmvpAT1LN=?nx@a)yOJd#CdrI__uO-ybMAS& z=i2}I->rWD*oNOHu^w{?#Bp3KC&Y46EU#+#S(4w^1mNd_`b84Il+^2D`-WJ4C8Vc> z;FN~bNpxdQ$D88&mXMs$aaL^J*71&xU+egdj&p)>M#Fgp5vQP_J9o^Qv&^DZo-uQ7 zd1gdG`f#o6J0*L{@tmVYdw03)`j+pwWluqW&T&oexHU6lS50e4V(zXuqi(@29;&(( zyXv2$9d*ts)@%jFr&m5$IQQ=2oj)vGxS>F=+OO1{s$GaO1&RXAK4FzAMFL}6p4z%i z!S-C~b5J4G|+cs|caOgJ-T%db^=O0Nv8 zA>Q#|pu4EjBb~W_A?sm_O?<5nF;jsMg$*jyYX)j|+rZa7(O!Ip>{HwTK)2ksx4yScZm$My6?gQ(dpYd`S9J;;5#KwOR{vrh3xZ zm3{5LikxYbN^e&;%R^@cB0)sm-MWp2VLz4}7$2f$ElyIXdl9q#l zKm;nzxnLDpv=(2-j~=u0{z(531&Z5xCqoe{=K8aCmEnq__lmKm2b_RR1Wb9Olaehr zU4fu}Weh2*ZHjK` z?6Ew1YMcXX$w%3Utf*C~*k!igaHreWbU|B!Vjf%Y^^Sii9vk`gujlI-LqwDX3LfWf z19urjvC+d5_}UTl;_DtZVKc>jdGu4t>Tzwu4oU2=hbLhY`;><**h+gTC@ZnViio)p@4qHm=|1F~S{%Uv=OHl)nV$ zKFc*8FuSpvv>^f5gT34-qN}*JDu0B?bqXVRma9}=+fkXV*g-kkuDq_JGFzBOw4WSn zSAM*MvP|hgp3N)J!2`A3$P*omC>X_9sGOxI5fSXb;NKAGoVhg;5oS&#Mz)fIaUAMo zC;DaWh&B_2`z{WL8Ob&K6BBl#O~sIve9#}!Zwe-emQ{|k1D_^GrfZVa_b|m(!;y0R)EUD z$cr-XXYpfFG1l;9{DhRs9NG9OsW|31vtJ^m;Wbv#BBkR!4yB`{61akO$dgK6#6w6P&8H$P?sk4fJFB7^BD&_QETfa%qc);b8Iy=k z5f6$qqr6Q$joM!K$PP}FWjk5T#Mt`oky#c_?;gAgZFnBbUlRPTY)lB!u~4Fma!Qer zRRa+#87S##wJu6!v!`ib=E8hOPz_qi3!btr`rz}IZ&B_4xDbV{ literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/form/BpmFormSimpleRespVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/form/BpmFormSimpleRespVO.class new file mode 100644 index 0000000000000000000000000000000000000000..040952ff9424a1e55a19fb66d9b31359d6fccb24 GIT binary patch literal 2272 zcmbVN-E$LF6#w01H`#_QErHTTsur!1HVpwRB54b@*g~;iwpayq#-&}F#UvXyn<8)W z;)4v*8Al&{Fzxz8S@N zfsi#T&^U70m^U(cqd1otv5Rv(0*T2=(X|TZv{kk)=FR?M(RK~jvWsPbjuFewl&=_b zbEcCqHYJ(|Vvj`@DYa?IH< z%Mt=n^8=$$%2QD}eCS9wx1v$tDZVAMDc7;Q+yY;#3pdslmjogYKfeCkwNFX(T1Cq} zArMY>Ow+|F<|5EAVinD?O5vjEoHJO8bZXBS`Dw$k7TG8p*+B@Md;~-vW)1uhL%;SqGclqRq8|@AijwDgqnz~D=BvvD~#=(I^*ZC zQE?7O@B%Yz!W6%32bg0!$N4CEVP2LKc#)KX(;SCmq*RPzA6_CA!EyTMs6@3turrG6 zBkYZ$hd)g`LW@t<@%@)KFhJlqcZs(uOPA@%VAX@e?Pi0avJs~e`;N}7S$M!uSrnp zBxA~(m;4T~^DEd9*OyT*3tq;~c!N)C+~SQ~ma%L571??@lR;e}40^~9d2hrH`ajKo Iz7c2s0yy);djJ3c literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/form/BpmFormUpdateReqVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/form/BpmFormUpdateReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..1f2ff1ae1e1b602905d41b9466f1633763a6ef18 GIT binary patch literal 3398 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/group/BpmUserGroupCreateReqVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/group/BpmUserGroupCreateReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..89c094c3944417b7c4fc41905ae2184ab1c3a410 GIT binary patch literal 1247 zcmbVKOHUI~6#nkC!;~_BRvsdV1z&vtBPu2eng9`FQX~OwqPt7Emci*vnFmDI#;{Tr zu5{(bxF7~MG=aqE*8hS16UKOMJCR0$i8kpu_nbNR`_B2!{q}SBGk~kO7Dp@68ZvPx z$jV?ynqdtiaWo+vgCYGTnH-JbvJ6HwTw!RMFm2Pn#SlrQ^9<2xXGt)$=S*AN3rdT^ zz0Vgd$|Q15fm?a*nlcWPQGeO=7$$NBr)0Q+^VBpRTh6+yFqWJWH*I6hF^aAel#Qu! z>47KQJ2IGd1^30gSk2FkQww4>;1=O@rgAHMjT;uXi^kmIiYWNw=|>FAk^F}~F{qBe zEL?`x1)mq5%<^)rFOq>FR^aw+iA%?2?k(5nq9N}Rp)0(=@(GxNnC~q3u4xw;I#THd zmU@=@i!V5VTM#p*)S>T)IYwlA3_bIK?VBZ$H$9V7n6z!j=f3IKG~PhYbPR8u7mLC* z_`w%rvTV+h>=tpUYupNmCSCpbuDbbRfAi)3tL@})vikbN_l-(*@A>}5+x@-WBxx52 z&rjy(bhM$J;l}^3Kt~_CbaYB{8vP922U6EXmtAYIH0670Mf^k*(|(D6>PE9Yktj zCR;(Yc8YFMd8C1&rACpg!-NzBh;Ez=sd|W3!6~}ef~Xezfi?}j&4|j?gne8muTX(1 z8;}f*L)00%>IJ_URP@vsnA~d!GakZdq=h=s`!^sh1X2n143cz_9x(<(pc%?aQTz7L u^soXg5!*pCfyT)tTFC4`-~OXby+P}~5M9C9TGct6C*MMt7wCV{L+Tf}w-Mz4 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/group/BpmUserGroupPageReqVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/group/BpmUserGroupPageReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..0d6f9259f2742ee8bad60ea81fa956e0717c3b7a GIT binary patch literal 3045 zcmbVNYi|@)7=C7VXLq}u!cy8oq3VJt-Ch_`1l$4&<>A*B9W9c?>X<~yytnJ_uW7K zx%Ufz9k>uf6HZk@#yetpS1j*E@P3Tn(?X{SX*7m0PZ}3nJ%$OSg?3UXxJd^19wjUpF?Dfam&&I&s#3R~>65jYY;IW4P8sO_23rZqLA<O+ zEmPnn;raWY&0YIKLPS5WW%C(v^yk&jA6@*Kg3Ztc23}j1Nq5fC8J{7-^H+-xFIf@5UVgFL47&K}cBt9D+zW5VJC&zaU4b;7i=np2jnYc8T3 zb~SX2Sss#MZRg>>2nKM*PJW^ZlQ!#G#K)c ztzTlYh;0pRI?E7U+nlU`X~ZsFRo}ewCh`0#T9oc;D7PtHC5&olRDM}q^NPEKSy(?? z&@zN)$z}bcQ~H?GkvvHrm`h3Pt4KD}nba+2BjDZQu{-9T`I~B2QKFmTsY=@#K49!j zua?+y6E3%~&IO>>Q zQApn*vtW(shYdlE4HZkfO?0D#P5;$oq(E~1#~bt4F3(;2a_+0!iT1?&H$ObQGAlam z%H>Bl?j}f;f^IvBk$wd#UQ+N3o>lMyUR01oi-g_(mwpQR(5c{nSaxC;fl{y=uPI2O zBZ6rK=Wt#@KL!-c-~$QkmQR=i1tT*qO4U>mgJ!$H0kT)Yhxkar2zHU~Z3$7(y{v9q z!b8^8e@tGrB^IXw4?n@DtVvSB*2VWx%9=SfJ2T2DW~+H~%2Y)+4QQMZ65@*qdTc5k zxua#jWoa`cWNcj5^DY^TV+t`L`3S=zT?ToMTEAcD0%tHE!LI5vw z4ZDY{@Cs$cGvJTvtK9Kt0K2#@b-shZ?Pbn&6`e`t3eLgh&g&{Vvjr+Rhn72UtmwQ4 zdwu5=9~lsjY-zoVK;=g_cozPn1t4BcB<#cMmEr`SHjY<432$J3r8uFdjpNl>5|hG< zBgL6>gzvi4+VVZ*(5=?rAb$%vc&qhS$h~bXMTFgLOY1Bmo>WA%uZ>G8)c!lF8$%DU z<|dT(#!wMeC5u=}rb#RnZj^;h_yKBfqNY)943`;I)1>-mD6UiJ7PAPJ5oxtskuER0 z1xzpDAPzD2dSr2!k^v4A8!q0!<#AP(vw zmkdfap`}MO<9+;v@L!(gonlmU-behzEY{T47O|EP)bUeWFJvMi|sP2Uu@Wq_^EUD5lm4{^tlg`7bNllOq5C literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/group/BpmUserGroupRespVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/group/BpmUserGroupRespVO.class new file mode 100644 index 0000000000000000000000000000000000000000..8c3e6d7d2be65ecfa1ca6bb5e6b4c64d611a141f GIT binary patch literal 2414 zcmbVNU2_vv7=BK&n{3hzlt6(9YS5}})3Sg?q@^h3vzWHnLXaxRHd)ifG#fV?kbB4B zO1N-jMz6dPFBk`BFr5L%(HorcPk2>2{s$4Cvzs-3{uHA^*1AhPu17mGI+1T@$BsAjuX zA#6r4SolMSNvGU$(j+spDo_JmmWwjeg5s^8w*PhKS@in+4ojDH>sl= zFX`BZ?K*bh1%dtlt8sL^fk7Rw${~Y(BBo;iyLG&YBdnvzQ&8w=%`O!rOR2RL_Ec-U zY9l&K%<4FaQ^dDFZ5Mn^o(NHEX7%nt~ypZ8YU$V!5|r4%41uowH*^ArBJegb%f zY~Y5Ur;HV4G&j@t4OHcB<`*bmLJi-|{0wEhzpoC}U-o4d5DBC@q7(fbhLnL{(UDRf zVbdMx11Y5rZP_9=q|zid6iKPlCh`c$J4mF|RAkvmxyz_wG&$3K9gPbMwszgEN7h02Fz>T_+pUSLcenLJ04L85@oXXQ9ox_LNo$(Q( z7@gSJIXp+ew9d>OL<%LdCfO1p|?jK}w?QS2qg zl=pD(A@mCiNF+DZv5{4Ga!PKJGK;SxncN)6F)44MTu0aH%adriJcErILm2akIPE_i N9gOuB|M@Cp{|02A?Zf~8 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/group/BpmUserGroupSimpleRespVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/group/BpmUserGroupSimpleRespVO.class new file mode 100644 index 0000000000000000000000000000000000000000..f9e3bd9798812234056183b964dfb5296f49e22e GIT binary patch literal 2434 zcmbtWU2_vv7=BK&n@!U|OQ1lcXi(53ZP|kOLE55xQ*2snpbCDBOS?9UNj7daAa^q3 zIF5|MjE>HD$BlS_h>o-~;-KE(jK9H4OZDDM5udZWY0^?gXPma@yyu+v>v`Vyob4}v ze|HnW4!qiiFb>P%ISnH*&UrbE#*o06l#WR0c@5(+w4o5iQ8^!z!$cG>MDd~&k4vS3 zh7$rIYf7MV%8m?tmY66)N%hqdW zjOl69(T(PbzPoDW?J2W##IdWUDokWTKr`Pm%GDBem2LgocW|m0Wv&t}$rW73^3rR%o4feI+?SsUM1Ft&{2%8& zV#I0F9VVzmI@4kqz*6=yKV~yxRm{U*l?V>Qz`C>I4w0vR`J)rm23pfRIS56hZG zIW3@3YktbB(@QhXOwgRCzIj*cd0;IyB4pAiG1El~V<%6UlkPx|_Lpho>_t&G`%a^-?L(r*pjENlUDQZRcw1UwdxdPFsfC7E!LB=# z<^f9Q;qu@Mgn4E&F#i6aL+j!k$tj;&HkBNN8}1_jnNuc{AyPu(g_nNA#g zu}`4;ez4Ql)KDBR;bl3z!ix8%B=1A430?0&hR;CLwcjADgi)=U6}ClRYYiHl-&2Ml z7gD!KztM^J)hE(0pCll}A5sXmGFM%k*Yadt4eOCSfd{amhHk2m;CodbT2M(L#$&R_ z@e%sbISbBe_FIT+UYN}sBi?2wwos1y3V0NcaU~?-9G4o0mKv{K);Novz&Ph2VyJA- zb%?K*gpPUEYpElX@k2|=XF#AAeara?{Z~IeO+vO%;O9@q(ipASll>g3a;4`dgg=8C zzS8p}!uj6p3>0rkK^^VokW>18Mtf4Zg;kdk?@KB(h%_{`oJtif7fGs8Cvppk%UF?A zlaYp!a)q(NXp-$CKIG@HfVhZ$Y-5I9C}2Aa5TeCCJkGr^ve?D%-Zm&W$cjEeNktwT z@Fb-Oo}&L0+O+6jSgGOZJ6Nk>Cx4^j4%Yi(8^0v-It2{u8VUgrpG&t!NAS0v{Px9>l_$VhCI5uI|qs+E85v}8Cu^AqOZw)EvLwXq!OG&C51fd zUY?pAzSkgjm;i@J03T37e}+EdKswHt^75o_pzZi9Vu|({#3kSvtVpc%WsOs!gTo9umtGN{ zQ~!RY16)ErR)PQK<($Q;rJdcr?ETt{{ef|w$;tm8II_|k3anDW4S6!%U eM{v<-?!74T@i{QY6hzFWI|k#tB$_tjK71n{se z70d{DjGRDw@|w1&sRgZ+SCeKbKP1pST`4(6QO_8*F<;O}OC{6M9K$Tx0n zpVuu_+c{B3%SOt~>4hoFEbErD#5n#%tx(Yg0$bnR*|@bT5VZ8`6~oeVUa}Gbihffo zmJ7V%JALZRfWV0wn42mcja)6Inq^wY(u|U0tCOZPSt%44S}E$bt>x(%af3g6@X>GY zegE*m+SXTV0*9Iz%r2FiSp0(+B2NuR6~=XGmVBXI^#npv%o z(JVu*>zh7j!Jrbu$*fsat%|v1sP7ca8}f#lGmC_(E}ClIGAm_utX!P4b?dTR%#}$= zpVqHu(o&qf?o3esNHq4(3ixgJup_$rOu`LNMXf*xosGxo`D=RC8H!yM5T=B_Q`h-Y{Y?zrT?!DIF)LP9zhp?Ed!8WFNl8NB=)ZLx6e=yu5PWrzxmny%{wc-6uzR{PH!e1#xXn-#(s1P4F3Nj zhVdd!hS4vF7~)hbjF)gu;J}XZF59t+QOHT%4?lrs)UNQ+jD#_VOc>|!GPUjNHFC9` zHz;sS;Ou`yI;PorCd~)FgU0Ox)=Ddv^%AS!-`GI)$~3lP6MSmibpP*$YiP9Q`#8#< zrw;kl;gM={?BeP$=L0->%0@R1Ud7Ysv2lpo0!M80(hF#VpyT$I!ZX&4<8dCUoCRlp zd==sw8Xi2$F;LrV#R=MB_X3_nlq)s|;vAcu@58gQ%Q^B`=LGud&Vx+lk*VT|Z{c}- z>Mr+2J+*`-sY}G{lzNW zhI?LqXu9z{EgvngR*Dk*9eWgNdYDh>=cD(r_Mu@GH8?^A0+T)MLklDuY){;S(qBc(+ao=XCmDCW>p8q6 zG>9R_4&oq&X?fUCofxT;1Ut#|h(|tB6N0SxF?vM)J8KWWPC`D!I>2d$kS!NoN9e6p zv_`^Jv=QkZPLaLRCg3XCBl~Dq5!u~XZ4HeTt;6)-m6~9qwO7%~_+$L%tGMtNX;{Fp literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceCancelReqVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceCancelReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..fdcc56e1e34f044e810470228fd72d107644b1bc GIT binary patch literal 2437 zcmb_e-E$LF6#w01H`$~ql$HWf)D}?EFBT9#XekP%K(Qa#ViokkC0U!rWH)3td^|Eb z=nNz9!~@Rw@gh3IAkIk3RAlr8XZ#y{Y^ZO(_$cDPJ?GqWe!p|? z_Se5>?*iC?cj8!uLoq0LO%B6RjKn#Q%3)0I9hTDTQW}q9B91r?X*eS1H#AIYcvFgR zMR8OhWMu@Jhff%%3_WMqQ~HEgwCt%qf#i6}_N=@)X}Q+%oVmxg9nbJA$94rehAl^T zPa9KHW>Gg5cl134YsATzxx+=LU>3czjH8}1awSus_5Q7M^Iu;m-}r3q?uGfQAI{&o zQob}JpcT!NC97y=!Yob*M9niszL2AbvTf`39Ri#E?9S+nV`cmd^fa^ZiiTx-u0HB` zgZYAgR5Wk8t}#W|_PU47-TvhF4}P4xJv;x+tiV&X1s*wDs4X&5G!56W1rp^;SME=L zT)y~u`O_~1R?0Bn&Xmu6G&g(xk2~iC);(%adq`xLWn12EfpDs0k|hlg9flaTY;&}f zKW-L}7z9^R;G~V*q*1iwx}sFPtVOg3htp18FP5CMmi}JOIV~;pjFV^m`YA`Zh=pON zO?_`6Pu$bQV5p)Rkow%Xd2(`0GBIU(L!^6ks-qs^F>a4nN><&#v^BNdQq~dUe$_TY z{)`;+tT7%te!@(9eI3UHgj3sMl`SUZcv-VZ%4MCKWG8CA#upoWRM=;tz4* zxSFe{IkzH=H7?d--7zHbgo`J!9=zu+cb}n6BSZ@pRUwJ!a`*NH_YGCsjmzCPBF4}p zhvRdMt#cNf)y^4+?@N1Ayhx_HGxGXZ&{OsU6#AGew^@sjU(}waoB_$Wj=rMtSIMf zSLatymFrzUL%9kye7)-@C?nmS(@^~-g~(_(hhC-U7c?Z5dsuZ1iJqh~jcCs4u@ z^+uAaw29n9^EEUk)nue2?u*+gm?Jul|qVECLM6rv%ns|V9ff(agL0-rT0y{ZNMpRk4 zOmBH31%#*z>8VUgrYGbbu7vrGek#3toBDPs{XSgloBFZ@OKa-7hG@?;V(&`5)~m>k zq*B?@B!t{rUmqG-o@|?-iNh@YRqWxJGKamCLj2ESD+ZWP7#n#N_fb-O7!E;sNQFWe zgkb6X)<2}t;yNT1tx&55RU4&NMKhkqSEW+vIM0+fFZBcBM`zH`oR~(VESOXCia^$+ jypqE-nwDOXM=xipQdbE3edG`LZ%YH?ALJk3mZ5(DzDyN6 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceCreateReqVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceCreateReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..a1f1e87e639c86691f6cc7acc0f0043d4f95c2fa GIT binary patch literal 2860 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 IJ}G|mKanCWzyJUM literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstancePageItemRespVO$Task.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstancePageItemRespVO$Task.class new file mode 100644 index 0000000000000000000000000000000000000000..3c5d4f265e5ac307b6a488df58a85865eda17496 GIT binary patch literal 2317 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceRespVO$ProcessDefinition.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceRespVO$ProcessDefinition.class new file mode 100644 index 0000000000000000000000000000000000000000..f12ec1c5f57d06ca0d6784ddae3412c260a9d2cf GIT binary patch literal 6289 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 XI*;HNFYKSuGf=qEanmoUhg|;!G@sN% literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceRespVO$User.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceRespVO$User.class new file mode 100644 index 0000000000000000000000000000000000000000..7890a7a3a42e0282dd07a43786311de68c1d67ce GIT binary patch literal 3349 zcmb_eZEqA+6nOt(Wj?e5I7F9nn@ z#>AjTk%WjbG0`uU7!r|4o2U`}f{DLFDEtQzpF1=A(smn3G->agd(LytoR{a`*+2gN z@fQHk;%o#Pa6F6<46%%fB`KD16(=HmPKaevpe!LxiDgtxJq{tV+NGe)XlO3)WBUcP;9e&OfO zZk?Z(pk>XI1uJWg2U)NrK{Zbs=}d}1X)CR^m7KPXG`IB?sg2~bmYbX(?Dpr^7iO(h^2*YLtu7jK=vKsdd3Teg+& zmJn=i8I>USvJ?rc1}xhgDx}BE>|ukQ5GgrHBQccYvdz2Gt2k)+qRkQ zO&PhINp0Ui(n-g&1?P+vKap}y3BUNblQt|nKI6nKJC`@?q#56vNwalHGnec4Rl{a3 zGkUN|L>0}PFx~R%n_DW{;!i9}<4~z@YZpOEoeLAR&u@}&5VF@8CFjLn)4VjaXhxLQ za`1*qj2~N|!bFv#`?423#Bm!bwqlX%!LcbbneS{l#<4q#XYHFn*>`m zSL2E`uCmOQTF+EgL|uRwze@|xT-plY<0&Dvph?AP3BCUV zS#`XOy*jpIhmJi$dJa2v>=N6AkcKfLVSV}5@#@}Nuu|h<*o`YzEKWkRHQuG;4Bi&Y zJ0xfO=3;BRNW%8bUUMnaQk*7tQj3n)u#@*)`Nba3By|6`qB(6NlQC`5v!mMKs5n!L zO66o0;fp6?MLZgL%WvSNDI*{T2ni2!*TCmS1hFZHNATz|Y(`@aj}gN0E{8l(f;>qG zhrb-sT!M%wq{ebcYYEbZ8k)%|u3MN*oKJ~QrF9Zo8I^(bfLRioZy4T@i-#k`Re(|4;sfKl7yb>`6v$>$77o$ zE)K6Q=`3rHvOm|_TEBrJUupXdp)a5WueAM&&|rJ(98`BpK^to4l8`%oM>Hni#Ohh- z9Wi+h;UXfEPzZ@6LNP^PLN`%Ai<+1c3sr!k36-E|LW`*ar1>B{u}BdlUt!k4W$e#*)s$v=4ZYaMS{v{s`gkG&Ja3<1KXpO0;v8O}By*j{0ZIx6 zVWOWBfAA6YA|(}B_WdPF8ZPjX7@!o!C0-7Llp+{n9y_o~)&4@Qii7vCPQ@YqY0^Dx z@Wk*6BHzr*+-KM>J_S=1F#=Is<>~Z}wntQ@RGh++L~)Y1^}R6C`}9-kovG{GEqA$N z+gR5*$)Ra=ZL?52=1_B7=pzYP7-F(-(Znb{Q#7HtCQi|H(qc-{q$fhIDO56Xw)EOq zeH;h8T>OlKNaHY$FjIvMAEgvv15e-;RuSZdI*3;($t3wU9HpdiU5{gwLT;PCg?66` znQfLNwwAwd@6)lo4+upAlp>&rkr%6cSf~P4MlRL*V0AILjIUy#3v1*D(VGdTSf{w4 zdwxRg(Rr+@ubabaF`7Bl*RSzp5_@(1T2GD&eS;_KLcfm79M&(r{3Grrce(S1 a`}KIfLU$3|D3=I)(Lge^1g#Q7(eVT#* literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceRespVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/instance/BpmProcessInstanceRespVO.class new file mode 100644 index 0000000000000000000000000000000000000000..4753f8a5458747b339cc47b70b7973c53bb0e007 GIT binary patch literal 8016 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 Q=ObU+k=qyIIh_Lk12iC4TL1t6 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModeImportReqVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModeImportReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..9ca89cad3a06908bdf2c10ce2f262e2d7c3c975b GIT binary patch literal 2248 zcmbVNZEsU$7`~pqtX>sCYf6-PUa+D#`Xd_w#b!_jO3x z_smZKy76WJ>u|IlZXENWKfw8Ax$}y&1_F3hTI+CJTE~5e1rWedKjLyu$df@ohUDTk zKZfNz=);JBe=MKP_3If;U{}mA(~6Zh^<4Utsb;ka!#tx*Xk$vYkg@f=YT8PC!ICaL z0`0?voULcIQQgwVGTPx>&ahQmH*%IhM@%;qYeG$@HB(WSPL#uWJ#M75%#dm1HPb%N zINoz=ra%l0eM9j?sC?z4x!EGYncCTcZfYqv84vN^TA>nSIVl4L5|RP~%~ zDG9?)6fzlx^JO*5Qqv5$p(fzm^d}GA|8Z`5X8yYwfz8XIpE#di4!`c8p406^0`73f zsDS4PAq%XD={YS?$c|~|36&KKw8V_0ni*A1U9PK}9(!CT1^Z%2BdeGNC0y&9Ru0CVcNsm+0j8((2&+_9#mViJKXUY&1tY|V|0iAcL-T` z9b2GgSW9czVPDCpxwJAkc3Ml?Jsl?ngdsVQjahcYo*MSJW(qV)F19b}tjXE0*=QWG z)#RDDns>;fxP3Um<|ozMF`1j@xN418_-8W)7UsH61p4Q{oGwp&zQV11<#uUy zru^0Q2UBAaN&dpqyn|^n0`c`Ox(&~E&B7`kLtVgTBp2yN%5PQ%W zL{tvXBFr`fu^0RJidGWXSI{#l+1&Qk%!RE2pLK5#uVXZbgXm?wor_?d{WsXU4uQS@ z9cAW-sj0T+JP(29B|H@+QU+>1ujNt#U9~!`wtKY#ss*rn%kD4edX-`p?qdr-x9jBR z2rfxA$2P9oId97am&Xu_KM?H9x9v#XK&;l#1AxF z*v7HW*=)pi+QG^NJcAuv3E3Qu%boAQb$f;L`bRo`5;$2%iMc${S(N_z3fwLsUD=UxyYFkkOP~pS72*As< z5&+eVebq3Z(a%Hgt<60L*;Ico8K_HaYVH|lvHs@h4fwi>sDHb6)1!-wRJ^M>+$?B6 zBs4!Z;s7ld1=Wfkr$|}aHn<-0$ivcvU{(CBe8~Ej)-E9d@+npU4l{(5TzD3Nx9*{# zC0Ilwkv4H^StD%%E~2?*E$t#&RtjrPjj+O1m@XW02z$|a5Y3GL60`6@97dmoBmV$w CQz&}? literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelBaseVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelBaseVO.class new file mode 100644 index 0000000000000000000000000000000000000000..75cb544a544ff3a36d78417a1be9ad1f3eda1488 GIT binary patch literal 5862 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelCreateReqVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelCreateReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..d3402f82d2e789e4645c7d83a3af8e3ac75b3ce2 GIT binary patch literal 2863 zcmbVOZEqA+6h3!%-gY}JOQEGJpo>;$yDg(w5w{fOr6|%DY@rGwFx?%t1G_uR%q-B4 zM59Df!T7~sV&sd(ga{^5NQrM>F!6WL7ypBZ&z+feciT-QCheJX&pqedbD!tjll}9b znO}&go6baOEggxFNTXc5$Hn_H9gX69jEfJr!*Q;C$i)f1JSo$uD5-QrA)SB66iO?U zQOMw{(=uflh0hujj2ipT=;w7Urx&u?h-;gLY!9ROaH-&$d1KUc%(0xYyHK!P-8HR( z!>GOAv^3{}p3NGzrk8iL-9>Z2${4vrwpBE2cLI6D^LnmiFjD94UYfsoW$xQg7jA#T zNU@D`CDS%CVcd{0l8uXczLwu#|S11=HOH2HM(3 z83}uV8lzSHX2BRN<;M*Buny2UHC9^Bjq0|^*MU)R$4#)Xqd#rsHM?X@nA+)_b%A?m z87mK5wDXph=R|0Gig}Nby|$sd#;|d2bO@wKS;IZZsbAaHUWaqhV8jQb>ku>&aR z(C~qQK2!`{vJIz{b3vnDjo9iRuC`VkqPUj-x)?=AtdgBJ_M4pJmKq#w>Y+-&~o${_*VPuV)|3B*1*haNNY`kV+ktRH=!Y zReFJoE!50t`~T}DmENM1N-xtkm0suCD_ra5;%x#n>y}TJJtZ@j;Y`-LbsLE7RcV|| zv{7e*qu9%+p)*l|BMhuvr9-rhQLLQZ?-X#M7mG$A!>FrPd#>o%pqw7sgt566qbN)v z4h`fwte(ZU1-7_D&(Qi~w1GA{Fg{0&5)NXXhfh7RHB>f*IH)VbTbIILT=d<%GJG=v zK?ltxJT?qC?V>ClssdUwnA5Z2e`h372`xf zRtNt?p*zchSJey#w>5&p%Y)a{3`S4X2o{zHH`fe)m9_`LDO@tdFWHg2OQG7A=J0bE z{7Z9}gt#+R9)8^sS@;p0g=MkPZkIymVPX^gQ2it*~BdB^Z+3_8T!mZ@*Bwi;m zd@K1Ii36P-(*#=1xlBB*&FSC*c+f3a{v?<9?w_@J><>?Zh>j=pyx^&qCBnChdYP zjAl*K8_)#IiT(5@G?51I-3?8m9G=h~XfoNfp7ughXdli@k}4~IQ-e%>kEvOv{rFSZ zV`}l$NTuqf4p0Q1%;R3<@bT(bz6ifL#HH#*)Xb}Ad|*LxGPWX4LiCYX&rYG&gM34* zXB>blvE&rVUDH%|irXV8fp5fxU`OFJ_*>DG=xu=kFGz`t-qwcNpziVy49W`AFug+u zQI3d7*AFd3QOE(58peRxMuX4;IQFrl6DXArrS1o$9=%8PjSbVZibIG`W6akhTwm?$QLZ;a gOw*d>!=Q48bqCOd=v|N3VQ)n8fjEMHjL5_P0(c!$Gynhq literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelPageItemRespVO$ProcessDefinition.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelPageItemRespVO$ProcessDefinition.class new file mode 100644 index 0000000000000000000000000000000000000000..0abe38fdd45b08fdc21f7834ee9bef44d919c1f3 GIT binary patch literal 3565 zcmb_fYi|@)7=F&~%hv#j1E6xr*Ec@^ z*o1R2tVX&C5two~E{7>OOh<7d#`&ZiG7^=QQce!G9P)DSR1}36mLaXdk+Z9zDCZds zr!}0>a8|CKkKzS^kd+pQkDf4QjAX{hO(nnU^Tk{L&&eLUvYBJ0<*WZ2hx7qVuP4-~V<%)$|anUndoCymU6QLyAX*i_tUi+8j$RgEtI4 zcGsOQD>7xeBP{!hp5BW5mJ1)FdZJW;jw;Mj4cZABuE~77gssedoWPp=(F4c#prfbW z0h!4d^<@`UR)R_%NK`vfwLUBlz)LPjWSq!4CnBVwkzszTyf}F5ggNOB^&Ta??CPfx zsOY}!PMZZ1RhF<@z97xp@kHI*EK|6}*Ob@GJ#{TW*A5?Dx8FiKPo5kz z@*c(5E4(MJO&YnqGE}-bZ8+0DOG()gMsn!z$k<+nN;y(6onpo%oBfcw%RaYu*E-hL zU3aPW zOj?p-;2+|ydyzd>db^I7@G=uO&=n-z+j)f?ALy#i{dRJ4psObP+!Vb!p2ZgSza@`+ z!U}Bt&siZWj68cKDLz>F41;D--#>!zS+y#B=~dM|_Yxj!`Sn`O4}yY_EG|eKt~xk( zB8)W-Zo}h#&JEpk+a~e_FaYeoI(g2I7Yn9E^MT%`vBaF&0Gn|9*)(il32<*n5M&~FudyY>ef#Jsfs5g6#PdOo*iSLhik+C;3KUezuK8C8y z_5Xy(MX2Gq{vQz;8|a%y)LT-}4h(QeDTBYDIidWF_F3qI31uElB}6Qx5)w;A5~{>R zenxy2EeSOdsQ@*nq69Ukv_w>bv;Y)MXcZtm)kKhDU460s6UnWNX?I`-%N zXv(#`f(Nmmv3B4?jPON-_~_ojL);7V0ltBUDJjI9#3&^dV=(!rc z$y8OAKn7R;w0)wg<=|9?B!d(3OW+?PUZ-7@*7>%fZOV49Y@KaG(=3|S)<28r;5=HM zll!rhA{7ZG&}g!hzN$H;dMZ}YGtv@jNu{SEo+?sSv9`48Y+MeRKKc@P97Yz8;|Y4I zGT}!lg_ywOc#=_s*-*#u6eR^a_^6+zq!O_T&rphZY5oM-4Q?n*vk*)z|8~DY!=-gd zZgfB`0ZJI9dMkiMDqu^<+T8#gtp+dQ8z^YP82K&qYl0JuQ+Dm0-(uMlSJ4`8n@77W z4X1eAmr3m9@fE(@EaeVg)}_3X@;p}6HowY-zq6d<=Xg^HhWGVq=Do-G%P-3$?xP~c LMFbZA#cTQ-;_&K` literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelPageItemRespVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelPageItemRespVO.class new file mode 100644 index 0000000000000000000000000000000000000000..dde1987df51d37badc646daa9fbbc608ff87f664 GIT binary patch literal 3779 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 C1u*6S literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelRespVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/model/BpmModelRespVO.class new file mode 100644 index 0000000000000000000000000000000000000000..a8f5376e4f7de5105d3676866150d2a64c3689c4 GIT binary patch literal 2838 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveBaseVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/oa/BpmOALeaveBaseVO.class new file mode 100644 index 0000000000000000000000000000000000000000..0164929fdefe8017d83af79dffad280401d9ea22 GIT binary patch literal 3544 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 RQrY`XiH}T$Km5p4{|irSJAMEF literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionListReqVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionListReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..7883e1998c60c6e4737d59fa6d5527a41276f153 GIT binary patch literal 2052 zcmbtVTW=dh6#mAS*m0aTaq5(m6x=jWU$Ybflw3-aCWU~VgrtfDPi&m6v%%hVz3Y^` z5grg1iB>|QQXY`{Li&(aMWRljQUxy{@jG(qj{wRywuzm@Di28Z%$b=p-#OoRj>mue z{qrLL{kZHy2QDSETlQ6U6D~qJ*ivvjCOTyio60D%|4K(KrkF>C}zNF;M^H* z;45i)ZFANx=tf51*gn9@E`v8|SBq+vWPaFt+WV>rRMoQEC?j(d#l=d_3gb+A8NkO|KDD$>%sm0 zIK^kGgAbj6LrTxNlPftN=R@|4g)W@1@GQD5oP>wlEE2~MofOW3vpc#8@dFJ9(SWA% zW(#^~`zsDOjb5&V)SY99cJ1n+YN{wr(-JTBLN6dtF2&wxp$H0cnjYGNNlrXD0eO?r zK^+I-#bJiGz6cKp=s+quI{Jb${s~_Qwf{lHZ|5YmP2=D=PMW>fUTf`j`PN!H{q3*s ze*(~jWC$%73_`(>T!#G^32`5l?ER7&3t`-)4oKcXc|7RHLm+E)5d^fq{ob0c4CC#)=eAE?nh{^X^YjuyNOEuRjG$7s!Nlv%PqCm z+_ut~OuC6$)#g^0br-EkO&`J2Q%rf8sgcQ(M$+kt9T5=L@)jwJSX#@OF>HZ)3E9TQ zerhQ3moQeB6silm##O6k*N;1T^1UHFTPg@U-;Z}$o5cMa7w&!c;lk{lsO%Yz zm=T>A(XbKEX;_PO8lFZ}V9)>5ks7w4OT)`@iKBz?YUspf4R2t(z}o7!U`O6eP02FH z9*<31F3-8|({LQRhCS$D^LIomozcg(?1(NeMM-y(wTlUC`_BowqsF&lF?hF2*${d* zYhZ6sfatq%d4?nCK_=57($o&a&*5|MRPv-CxX%=&( zv=z@B!LtZb!m@LHp0^X+1$VW54&s}V@l9NPC8mZp(weJ)7}|LfvY=d-U%my;&8o}c znwQxHH7(bkxPwc#()lauBFbGf zU4hmaQ3?oDG=kkKRf65Vh$?k_cM-mV`iL6wRg{!#^y}pGj!S;9Taki@QkDcHj1+CC4NsSAxa{7`q<3H^-3AC#KPeCD z*=oBOl9!FX7Q0C)B`EemdB|~72!#;Lp8pIyq|)NqBN@$5D+(1IWjTv#^vCC`oa{0B zlylhi1GK|)s1G+3u!>nXatk*}n#oH#?9#ykR)^P+E}*%3cp8=rPggld5B8P{F;IFl N>ga19zkD;^{2P?XLOK8d literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionPageReqVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/process/BpmProcessDefinitionPageReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..824dba623636749def690d5718f017dd0eb8bf70 GIT binary patch literal 1880 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleBaseVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleBaseVO.class new file mode 100644 index 0000000000000000000000000000000000000000..d6365774a64b2b70967d24f1d986faf133461e83 GIT binary patch literal 2656 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 P_v)^hR4jHd literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleRespVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/rule/BpmTaskAssignRuleRespVO.class new file mode 100644 index 0000000000000000000000000000000000000000..e78049d57cd19331c8f7b98e773f0704724f4451 GIT binary patch literal 4145 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 TQ0cV)UF%>Bv;5D`(%gRlU1Jzr literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskDonePageReqVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskDonePageReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..cd72431a526d4ffceeb690b0edffc46e413e0aaa GIT binary patch literal 3065 zcmbVNZEqA+6h3!%W@ozH!m{lGE1-*1=-V=ah`OZ+Ek)42*_QGmD%0(BJMHewGBexS z#1Co=F%XpmlO`B7e6fBY1`TWzYK(rt#NXjk`41vKcV^3Ow`(Le?Y-xo^V~D%Jm;L- z-~Yb*GZ7u2i$RLf={geW3}4Rj<(x!s1+ky!wzs+TPLSU9qznA0a>F>+jZ2gY(gr#$ z(*)l&Zkd!REz=Y~>oQHtG{eP7i871?L(OW8B7M{9tg2*GBdv@$mTsgw8O4V4hNEY- zG2Pb3GulzZFdfy=O~Yo?+NYa}J*TGAnx&{^i*hul_nQ-1X2>#gn&r&Hk2tGl^7zQc z`*%KE`R>Xe%ge>fH;Pw3Wh80uso7ixw!oj4uRQqh8bIS(S~q$uO?9+Ue4J6(i!JZy z8RZzX@E9%r^g(g)%F6YRi=TX1T)un%i^7A2+l&tOnO0h{b3B1b3z^NC){HV~T3OYp zh|+b#oga5Ca46+e$I&cNkC}!xq^7lD?c&%VPa~~418zZ^T3TzB zGy*_(HNlZpY$e5EAbZy)E!nHyqb=*5^L!&ftaIO30++pWXc*8c{{ih{Ud@1kHSPz; zr?r&R*?NuPm+u*RZa&q8?_QKH|cwYToF9tF#Rx zj+&b3S95L=F{u(|8ObTt=;hf!oKo#6uMf}^ClTz>@X7vOWCJ;1X?8y2Kv@bj+`KNB z4b4@blpWKXQAWWLGjFA|<2rAuSgo$u%Lkj$&i|TMbkE_H?{BPJy;QvVS@H8*@kG3G zT+!)q-fi*X(vQWA{UVakO4uvz_?ZV2IkNJw#8@wh%o_yF%1LyD*afubD%1 zk`9HalP~*dKS&MH3v?hvFY@JedV|rH>L;r^uV*HB8*HmZM2nrrgVYrwgUk>O(|(Lg zN1XRc7rLq=Ud6i$%%M*^;%n#!n6)z6Q%(y~)-;st{5T#*TgjQzroxAENW}{T2jwum zenr*QTnlE_n6nm0EuMPwL03Afse$Kf;sx@#@U@iRSycXKq zz9mt(+5Rhu*Gcr>Z2yJC{*JZ+1>9p>`w~f>R3Lev14~j!{6_UL;T~;Tq);Lz6sT?$ z5lo7J1e1Z7$T5L?6j`K(m>3IGfa;SHK=nyECUKBl0!cA>6(rn5X#R@s3cF=G?}hxB z=mqMc!$>GfAJMDmI3G|t9f8h|>Zj-^B!S+fnVKnl<-2ILqdiQdNu^+J+? zB6MnlB>zPlBzmpxA#IlE1b$`qkhXeaU8M-$c+W}&=7r{sB622q_?5%qh41x=_&9tK zLG$n#fBHN(Y#;6gaQjTS^AN@?)P+jO1KY!$Qy^Uqw=a^EC{V)%t`8;!Ziopbi_A;m znd+0GYeE&=OL9zfP1|b|%PGM(OOg5M6!lR*B9du{1|a#U0TB)&T0iZ^NDo00K<^fi z%ZcUG%~9YH8eaf)jDam;>=EGQvyUsHB(4HhF~V9^2^Oe;m66q2fDtC%eUCnxMyULW zI;;@lgmw>94t^Yve}dCxx58(fwopZ6PqfYi6bSNd^CbD S`DoOAfcaB-3V(QF$NmA!B$(s? literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskRejectReqVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskRejectReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..0c0dada0baeb963853aecfb7a9980550e95fc618 GIT binary patch literal 2370 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 NLxh_UIlc+ue*ifveOmwk literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskRespVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskRespVO.class new file mode 100644 index 0000000000000000000000000000000000000000..dde45a9a3bcd3b2ad90c2da39710c5ab45b064ea GIT binary patch literal 2670 zcmbVO-*XdH6#j0q`PnTEffS;sL94b&(*>;vEk#QUQs^&QC<20PlOBAp5#S*U=U{{j56cs3(ojY_(G|FgoxkGrqB&_1SZYB=iYnHJ?DJqJ2(B~pPzpL z(2vtmbYe0B1ygdE4&h*w^Nie{mC~Uo4!hEv+&dzdM?!c%3Kf%K9F_Aid2&3A55hPh z*C)gHP|n9gI2D2}pd79kj(|F5S%x!G)GHOEBG8^UPMellHErvFaX}zHenvm1Yen5E zXtPzvvGG%dzWMC0?=K339pj^#=@@xGaS;L` z<72&4F47`gWANE6TedO*Ms5}?!*KK2HlDLfnp3kcnA+*0eO_mPTHY?{rlp;;wW?k@ ztL-kA4so6_D&@JUE@{VLf;(m2y9E5*J#&O-gux1QjGLA*Su4#O&LMe~M8wYN#W~$E z<+{14R2NJpzx8RI({HR`R7bs_ukY?z4fYOd4ttW_PQ_^UYNY%lGaN2H5gU67QGOR&3Ih`O$E*t)y6q3UHw z+|~4Mw$E1e+}R1e+$bnJE`)i~ET>z0WVjR;^vZ(E2@+(C(M->bP3&Q~l!J~@sTHdN zYgz+n@ACQyS8cbA1fsKc&B+;i%?7JHoa~cLC$RbddWrN&FI~T~bm`*arLPvhzLCnN z?%n)k>ANr8hGHwFB+1#BMrv+K#cOz7#dCOGMHgNb(4LYzDu%I5#anX7qK^=%(6B|t zZj1WMAG%9RwCC>XZML7fx#yZ zd(^g!X}w^KRgIF{?E?KzY`x-rV2F>ha|y|+>*caxoCE1w5I<1x zp@(Czu^C00vg#@zgI=x}B>FgdjemsiMvHN5Rb#SZ6=Q#^abi{Dt$3qpJkZFQai%jj z;affKPU^@^+_Ig4uLi3|jj0jU$b zgZNdnCj-ggvXXL*Ui~X-*4997pBv4?+#S4)?Tj^n6WGBz`dGtJ3~2?nXo5zp*BScOv)E8N$2#3yb^M=!y~kD#{n23C0f268eBdLB_X| zQ*L~2Z2}MD3ot4fpO7y`^O`?t7ohb}Y;Y&p866^up~+3L!36>qj%BVQl&z!fl-!RD zDDohwG*7}3MjmZi9iD#o;--5)ff&Uex(&0Wdnx(QPM*HUko@eG4cJFXX+W_b$^-I9 zArwL|d;a(FfJ%?nKDm)VU|C^VNAYe^jsEz?HIqF_pRy18Z$mwL3+?eWb#yT67^nDJ zDKmK~$6Yy6$GZ4>%5@}Kn@3&IJpB)I+}|j~fyNWjMqlIn@`;#0k_S;PHZaAFi2Mur C@>Ze% literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageItemRespVO$ProcessInstance.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageItemRespVO$ProcessInstance.class new file mode 100644 index 0000000000000000000000000000000000000000..50cdae0097090628343196c461620e1c74e2effb GIT binary patch literal 5137 zcmb_g>vt4q8UM}RW_Gi;EV-~PR$^1lwM&sIoj?^R*hmr(XlkjKPIiZ6$YyuD7bxCZ zY;6nJ0v6jE6g!;OI?7+J2trt^T-ICf>Fvjy4J$ta2FX+YcDf;oB(6SYs54pK;diAYyE5EqBeB<2eyRWXC zUy`5{^(QMvQO|f8A4!6&Kc&qTvV{BY+p={VFF9?lnY?o0rPV8MO9Z6jP$JaG#B{I z*B5HfymjVx5;hcy`LtdtJ)|Eqa)yYDcy?#?kn_vGIU}BBpVPN>%eL*@=v1X-Dux;&)j_Dg*q9N|>MPGTwU^ZjSmN?32j;PE`MFfFl0Rjnj%D*Fg^N@sKc^YF)QNnmtd(X{y9;wics-KO3_H6WPE%-v1kx@DtDDY!LQ(jvLqtPM4^n7yHfwWPauGlqAp;odYxnBVm5xK=RR zha=jYph{ZH?G=${)Mm8OjMaf`$596M;Gxm+y^K8dKv6GMvSoIv8CHL<6~%CGL$nh0 z+j=}YdR@;O?rk_%XDiCwwo=YpttBCJIA1BI^?imI33sj8viFMDfrOs_H09aK;p)H> zQ$36w*crx0aCaC(LZq-IjIBbuUx@8OsJL6k8xr>aA7@G!bNGB1d$BK!M}_top*@Vz zFg`1^{X&chF)l<7?1b*-ci!%bkc$C2VQ)%gpZ-`DEcu_ z!XP26tyaiz4RS9bth`ppeKm;i!n$mQe7pwPMhL6674nH1%7%4f(=|`mry>)Bc^zjc=Zn~LHZ>X4?aa1Fc*9AY3gAMfL(ZqcakV_%0}|9;dz@z zC%NC1%)W3Tdz;Bau4ML;3)$C9PPvlVhc0A)GkL2knf>cR4m6YRcO|p$UC44XdAlo_ z^TLI!G?P_V@&pdpWR_m$quv$UFL3cmi_4={;MZ)KB$ZbEq z_4=`zO2T3H{rK1G$Lc%@N8J4c*6YV=PYI8>`;ph{$Le7TlkR?$_4=_|UJ|b$E6$#W znbUT5ZO`Dq&*1l69efYLdHB6o2Y-*?_|U*2Lgp=%wuvDM)i?YHv?YD-qx}lP!%5#F z!c{~>^%D|N14+NY1l~vD3L;5=GT;D3Rhgiusw8CrQf!c%R2-m~8YC#DhLS-63fZ7w zGUNco)i#3SYB<>@Kw%q1%>imxBLuap(PTt`qBba!j5nC= zrWWPQ?!gSTm=wUBFsQ|GoM(6x8)fAmh{~9~jV>8k{#T^iTs76;f2;CX{KT=l_W@qT zH1Uga5|MKbe(SAa4HJJga*V~}+@pBe@Z-mFX6G)r((d}M|NsKf`a zKptMih9|^&MD+Nq^N8Qv;?cP&2VkEmmBWO%qGhX314k zyQ-KbrDkH`TC9W`G)=5on-w%oo7{`%m~}Wi1YX=g0Z;Piex|laO&op+C5FYzSM_C- zsrmR~eHImJewNw{PEZTbGLDnfWSr+$!&B50{2bjlMJ*`OE=?o)0T+BsyCgBy{9DEc zbX&joh=or0tAHxT*SKWE0uERm*}Q6l00%RmzhHG@s#;3Iv!g>XQFEn-J*iTy3;ixjPYC_P6pPs0{K9vBn`LSx_EobNzh-{DiqFBX^T*G@Z(t7%A>IV> I0(K+#FWl5nssI20 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageItemRespVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageItemRespVO.class new file mode 100644 index 0000000000000000000000000000000000000000..e31f9483a83a14ffdd4cc4758a966088ec14e983 GIT binary patch literal 5337 zcmb_gTXPfV8Gcrlv@3aau&^N_kcz}@ZEQy&rKdHd<&@%rfdX|Jddea%_JS-qS_wjr zkWfM(0dk}@EsfhIq#1H(2TT$&q`66Fdflt`sx!!jU(lP<^!avI)>>LxPsxD4efNEz z=R3ToZ%2Rm_j^|XtiBzR*DcGiBm~T138~F{_5FG_uNOwLJIW=qFw!p} zy}eQ>n`6e1X_>q8#+pK*Sk}vCu|QV$X0w>J4(KBzMk%Y;E!j2W=3sHy$aC3oqf|b~ zc|!a3d_^pL?aGzv@yVI@U#gzEEI}z5k5@f?b@Zcunpo9*`AJsP} zuD$!z%*2uDm!G`;)=`FRyfI{hn{&mnY^hQ_XlD21iwAUuBRgCi)6GJ5e=#fO&aNFF z+sWt7;&5@BK4J`%jj`>9H9oYpvmpa`knk$8j|U{QbafBW(d!s7uDjVR7+Wf1yN%LL zv4Tj8VouKw=_OOVyUJjB)Z~d*-_D*yR*V?s0j6GiSGUL6v&kpZyv2z>C1z_a)TtWY z8|;|CcmkZ_dqXJb?;6ui^Lkj@2^Owv2#=Vf!F3Y0a*j_n79q>=Lq}J$8$@tq+FeIb z@;G9FOQ`!nz+(dB^#tFWJh9NbR?QKD-A=ThE2wUA#kQ_b)DvR)Fgt>tXL8MP#@5|? zja<3EdlyTtIJZ~1CT3HyJZhBaEs>m^qNORe#ug*C(&;8qNA0}CZG)$eBpj<(<3Kzf zO}ZG5s@(=Pb&(DS*`!}+!$s9Lq}(Eh;A3%CXle}Z3QhVL`&e$@pgwNbEbkj$2XskJ zFKiGVrZ}owqfXhg2Mo}G+qMr3ZlH%Lhf9W4$(Pw4Z1;saUDtJb{Lce9=)SjDdmg@P0@Sqx@AMVSKL(+mi@0Qh)rhW%$t)lr;k=opRAsG zJJXk$zI1rzwG-|=G;t)u8?M01E_3~Zv)3n|zV^-=Y{S)KM{i7Cx-t1~M)XhKc1C7s ztBN&PtK!r6jEbxn?#Jg;d|nJ+#Ftd87P_zCs}lNeqdll7d)C$7?K@-i(OuYF?PVnc4W- z*59+6P&b?B#H4yN4S}Yzy*JYoY6sg8Y$~O06^FqbOV+pT6tRR&x8b?QOO8pbkMmAv zBk1!sJNJHRzH(fzF=yuBYrUtbH`nm|G>h;LkP#5+0STYwTL+(AXhFAy<>=XkUi4YG zmk=gTGvvNm$O=N3S}r7M&k_)`taKm&k#o(~FVw6|M%TL7<^~1#%d^vK=(n(nYcOA3 zT&s=a7oCL%XkZR=`wOUK!gE~5w=AC$pP}WKA^m|u0P8u1ZDkZ2D60+t58`WlW8q7ndLMF2Be~6&%xdu=2OG)lzGPOJ4>{CGUgS$=b^4IQjpW6?WLC8g zS#BgR^(C_#_>h%G^4-4V9oXrTH3l{y0=vBD5(55#Ug#j)K#L@Bl3v0${pV?!?>tUr zNEpIHe)|c|cOIvFBz((%p3r>faSBbs!`S7wpYVL=aY|CcxBdMi&vzcDWF>qD-}T#% zGT(Vbg^POT1>?*U*vOc+v1*s~EPoZD;Dw$K5t%@!!_VJ&NR=m$S@|0Bk zo$HTLis2s&aDh^sJ$o67loC=HcVe7U5|8o>58)13`4{3`=Oz})u=rn*Zc=NDk^fiZ zvG@VwWEF26{&pLSEv?>_UrEOC7n5kJMd`oD%v^QJzAk@xu3hn+?yV*yDCBK4E{Sim$&T V{NvZ(&)BBGY?E*|^W`2y{sX3_m%9J} literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageReqVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskTodoPageReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..bf388691b9bad37d9730c7ad1200c063e83c8ebf GIT binary patch literal 3065 zcmbVNZF3V<6h1fE&2H1QgtP%7pg}71Evz7-rW8v{5wvf%rM!sBHrb}zCc9y?Y3+<3 z#8Jk9QJKM+&fuuS7wZSc!5IiM)M4}s&iFeFmH!~(b9Ymcrg3EKwD+ES&U4S6^PF>T zfB*aLJtEpq=lv9+lXWE0DZZTM%NdE@@nb*BZSQjFJwLtgO6T}d<%UVFo0KTwr$#y{ zQPj^oV6(%ge<}*Nay^XC!GKsF`dUHs7C@E}2vOIw)3z^NE)~qsRS{c=@ zh|+PxogZ~9a42O}+tw_DQPTo`VuORRWHNf}SUi)7=a6oLW+W?P3a>0*eelh9aQl2Wjzb$sj9frOb+ zth~9PD^qE6o+Ff`nNf8^nKKnz&CM#k*~~chaWiR-s3~n!J3leZ(@1IdkWk%xTS;*c$o{oSOSx6=;nwxedA=ba*12ygfy>@GGze&wf1h?fucpDk8u!DK zGg`v#YCFS-nQJ?>WI~6T_Oxa(s^bi{xGn8>bfrVHV_4T(VHdMbA9CPTHE(jkRoaFz zTTRRks#&Lqm{f@}jO2uB^z&>WPOG_Tw-3-1#}Vwv=<&gRWCPh}X}NsbhO!iBuw`8^ zjV)E5lx@?UQAYkTGjAodqdIS@NUg5e!v~wuj{llhbkCucAFr=mxmdjNMe)m<(O9%{ zT+!*#@!=!G(c-;Ni$jt5y}XvQqZ7jcYNw6>Jwsaq^gQhhP%G`kNdCWO z4$yHr7@#h`?4^AmH9#-X{s6tqmpADxMw_djtlqqyPVzR`T8oI*TpkZncYq8s12jte zFfN@@-Yebcs?KN??+!4FKJARIp&ww@#%OmrEl3&DP%;aXcpP&|)|@dFK9nOWULZIq zhw1h!s;*|UaLOoFYmU~9Rw*x3XbpaKF#Z)!JMh0Df`2Ps^6;_{qir|^v2Taw*&IDb z(Hu}7Uayk18K>=#VQrxoAneM~Zs<_V^@z4Iq8?V%A8RMRp+;~_YVy;78 ziyiIXlPKKi_?5(KBzkXj{6gYjXM2Hs&au7Y4oR+5AbF@0OI(QkM)eWlK5bm0KrA8@ zsBRVEkBfl#f;hX^>H~OagbaBNfCJ!B-}t~-iq!DI%PWR zhWv!+Me3$QNGMDn)9dIs4^TSwK<7pE6Lc7oKyTAg>V+iIN!a=z`DhLUas-k@A7X_1 zA;~}yI<-NP|Dr~T-l%&-nV1p?RZ-oJk&j<#4#+yL}=) z3139eJbcEVKGzM~i+euYJ{9abh%pOvqZ0DawqVyZNSA{hOC-e#)NqdL{c(XCB0|X` z^HR8``nc$rPzC3b91$JUw%WvUO7P86WL`Q!12l+;WE!C%NFHiHgu{r|OS>@ABaj5p zyBXwiVmWp5<97eF0jV2c=g40!qM;fgSctAJIEuvS%q`6^&#WVIGxgo$_GrH^J1 zDu1GSmZ|ad9oi5I7HA`Q3}Fj3aT#=Sx!INd1qz2YxpF<1H$yJamg;9>W5qMkQ=-X3 UWB8JX#+?V4Kb0r&hbMO8A6#6Rt^fc4 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskUpdateAssigneeReqVO.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/domain/vo/task/BpmTaskUpdateAssigneeReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..2998d055fa5a67c9c9b8d11898cc5aa3654d8d6d GIT binary patch literal 2647 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/bpm/config/BpmCommonConfiguration.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/bpm/config/BpmCommonConfiguration.class new file mode 100644 index 0000000000000000000000000000000000000000..10200707d8db6512aebe1fa9dafd46c77f33fcea GIT binary patch literal 1014 zcmbVLT~8B16g|_HF0DmMRgjMt;tObGUI-+jL})NEX@C^cgvZ%sS_XG#HnX!ue}w-< zpGY+E2l%6mcNRiJVz96eckaD=?zv~~%>4fI;}?K+EY&fA*#L8Ol<+*je1HXp>RY9i z+h!=Qto9fxZF4FaraMZ@{X7Y!Jr-e1nO4U{BJK&Re0-g(xHFY8yzfLN;Wjs46z|98 z!XtR!ibP%*`-z8X!Xu;mYQT5Xq-~PK=ynn3Rybw!7Q>6wnn-5ZN1ZvLBiWT%9=pAv z)b213W0jpr%kXujW9)!usa1L~YCz;pes+8(O=A@ml(#FlR>x6jC6~UTS9$n>2! zf?;D^>f2}4KotQ)cZ?&y=RiI4BE$9=%y(QShNj;ikA)uagYaBNj^WLNFHR53sL;dl zUkUVH5Rjduh||hwop^Z(_PzLBBU>$!4b;hRTmkTiJdi(y$3)_b-=Rz%Q2U9}$*W70 lzfn-ah@e>z%wB=62?&m9G>K@2(7sWtsCrV^r#IPWz+W{lE!F@4 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEvent.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEvent.class new file mode 100644 index 0000000000000000000000000000000000000000..b0df093dcdf840899709585c386a002e3e501baa GIT binary patch literal 3427 zcmb_dYi|@)7=C7VZo50ZLR*%Lwpg&2EsJ>HQiMX0N?WkSDtKeMown2Njon$GiC>Jy zm=HC_L=&QX6g9*skwQR;e!;}w0n{HM;&WzZySwc|_+ZkWdC&Vk?|a_MbI$(t@AaDi zo`Dm_226&a;&rjSA(rD}IicarFu!k!u@%H+DzHgt}>pa!O5F;#wo{lRayuU%jM{TZ$29tka76%mh2&09&Qzk5=Gygxh5vKvO|GgeGbS@~lo+bEiyO`54mGjEH%uauoR zn|(bzlFX*#`C@j_j?bjBX9OZXlQ+}WnQVR`K9x(e&v`3uowhPgd@z?j>NRATy*D#S zYuv4JNLX2wTF4gjN%zR|eEk$9D6^I`%;B%^>aM6@W&CY4`_WgmJ4B0jkf^@Wx!llt zVg@Vwr4+o=s@mlrCfe0T`vZ=nrF2|tWwT4C3HN$}s?JlbQ$;hymR22Dn$HOdQkE^q zI&)T@EEV1F6gNS4HLe(Sm9-hS(Sy}CMqI}d;S3L2+BUkzDk?R3h$>y(gkvTbM$Med z58iqW%vVpEnITaZ#W}Mu=TR)!)jiBwj*btH4zVuELAQXG&f58Douy=q&Z<=xc%bp%Ea0;YFTdciS2L~jCV(nfJ zT_{;rA8W5@B@t4$feGxC5XjPSxs;uBA5izwN@y(S?va(# z>=F!D&0NmPkfME+cc65G*4$vfWWEAb_|gS>{i@QD42l!QcdapwInNk4V zT*I!m3kr8V+_!ot^diO`@#w=2#*hSu{8({!4=6pqAn+4GUMQ7n5kQ17Tv_qMNP=@U zoP(>KH`R0|S8F)StDW0xI+NoyoR!ti+iE&Li|2gj1dH_hcLjlR|Gc6bT<^8L6KoK@ zd%kv@;G@P7Oc8Njz>8ePJgf4NA2p6(k%*J1osaUUaRkFe9KI)9nUC}yyAxqvmwS7@ zg(6?-{R8TGD8Wm;ze633^(;Yiw-ofT7?*_H_a`Dz`4*Zk!sv_2O9+(^;er@?MG_i8k%S)A1W5NmT2!w98Ho@W^PVLv!O2n#5@MX>Rsp o61%B+ohL_@u)ev)lMSJ7<+6mf)sLUY3h&L{yYglqbGh@{e`+Yo&;S4c literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEventListener.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEventListener.class new file mode 100644 index 0000000000000000000000000000000000000000..5c9170f0857dd2e78cf528bde1a83a115173da33 GIT binary patch literal 1370 zcmb_cOHUL*5dL}@cHCu=H>mi+=k8+c7`!bZLDYn>;$xMA2b12hSsG@$*Pb4bf8p7G z!3B*6%}S(P8pq@GOSno7pvhD9?X7ApNy98Zq#~w?T&SL^`pTBDEjp1jYK?!{Yj1-&dKyZg^3W?XNRr5=b(nUXB=gq# z7qA0T3OG+|F*6Qfgmiz^yNzOu{A3ioq4j6An*+VkfdH9gk_3-DUZF7$1c_9f1r4X V@P9vF%3~BTOV|Zm#+6K7{SBkTfdv2n literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEventPublisher.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/bpm/core/event/BpmProcessInstanceResultEventPublisher.class new file mode 100644 index 0000000000000000000000000000000000000000..b5edba612dcbc0e19f309d318d6fa252e371e65b GIT binary patch literal 1228 zcmb_bQEL-H5dJnv6LX2S);6|kt5zS<1Y87*V5Crsf*5K@OWrq^ZMyZcckJz@?avYv z(FcEkKT4dv#H0ZgEP2?SnZ23$zL~lC@$=hv0MGEah#c;ea2NMHEMTL6%>wQ-6r-dY zsQ5%0hUd-D^m!Z^rTfQ54CGm8K5#$OR-RkF7e#^cg;k+`J(SwEeg`%fHe;!KEfe}O zj^FCo3hm3bjFZ5ot_%m&rXLQtNx}=okAv_`bOXt!Al+y{Kt^)uQu9~Q;P2TQ+Z~47 ze%O-?i%q5FK{Dt{bL3JoEH^`61RY_Nv&WA)d!k~7qxq4zT>U~yn&I92(o;`2+KIMm zAUi5nu9AyL?r9y`bOKb&TJuy4#W^2}K=sl`-f=30Zq0_agd3)nW+p!WQfXy(8D3Pc zICnP(?=Vb^FW)HF`}#lUHfh-tN1S1OmgNk&oj+2#RH~S?VpO%Tvezghk3skXWX3i%x;1kPdhSt8_cC-7%O~`+{t({uykFl_wui-YJeZ!SPb+jG`Sx2DhYU|?#_;^h|p4r`y1OXG|$L{RRobNg3owI-b{q_$45Ah<70i-X|H;-W|j&7vw z>*E1Dz$bAc_%x13_{_dOkKu6)UoiAeD_5BrhVE2)i=k&;??{HBta9Z_wVapUhR8d# zaxtq5!r2m@vh!%M$LuPf;c2#@%iOE#eZ`B8eq||m(Gz9)N_%^arGj>gs>J6i<;93< zK3J)G!YJ)#81CjPr1kvwEA<@M@7Vh3*053Tl(oxcaNb;06yK4K zED0mItV!4Kda^9oSvHDWX2wP_r%J9cRZlXO={zCM%8;{;_G$C#kUvpA=7JJAzN&&q zURkfYMwR83@)cF@tm|qM6eZ>7y(3SUirL5?3>OEae2$8bPUWZ6Q33I#S5pN^hg-CW zL-lS;YeV>Zv%XKqxL$Q+E~KHtY8O!2tW-iYLzo6MH=j(y0@KM}W=n=yakO;yGRQ3UX1|yX7h+LcPfA7#}3x{gRLiO8&H2l0=6(`5=dZ>VR()J**Bi3EK9Sicl@DVWQH+K!`7fj7#7N zW@DI3U>;w`u#mtap2qMjfhByCz_-XUEVQfg#9Nabv{B-Ya7)pJ7soPI7?P>%;hju+ zD~>o;86Jmuc=+=APq46A&|l92I=iGVO;pD%)iwYBL4Y((b$Z7$^eH2$pU~7JjHJ?G z1gb_kd?{Kdgt}F5`EJ!{?eKyo2q|dCp+>=ClD%IxiJUqkqwR!YB8p?HlxESho+tJ# zG0A>tP3|PSlEf*XC?`pH+K>VSbA-XeFxk8QTewX})ktCG*7Tm_*#L8?mPOR&F$|O& zvlzDN?6O+#RVxgMC6}74c}MuZq=^32ToRGI5EY&YcgK= zppnsdZu||{FZ9ucA$s=)i72*rE*u7EO`+!!u@FA|!Xl=PhBO~hTv}!eP(r7gg zy+iklH|SZOoOq3Mlh%OHf%Cz3nv{)`uM_FUgc}qWf)vCeLz>hL$8Cbru2uJa zq%Ju*{u_E3enspx`nIjo{_RH8LZvr}5j*m(*7 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/BpmActivityBehaviorFactory.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/BpmActivityBehaviorFactory.class new file mode 100644 index 0000000000000000000000000000000000000000..b11bca67de1d7c7147603ff1b8447f4cec3c66c0 GIT binary patch literal 3335 zcmb_eYf}?f7=BLfi3=iEQ7eeHhJ?7SwHJ+8!Anae5kaZgo13t}>av?|HaPx{e(MjY zUz}+>(}tNc(`mo7GyP3%?Q?cF1VX}Cs*&BZ=e(EaecsFNpa1;)3xEqK$MI1L0y9Y@ z@v&Upieol~C<^j&D}^BxT1dVGCw%e;WZY{i~X0vYQ6dvyZe5O*J)ZRlQw$q?g_N z_yYl9%P!n=BoMXTHJvkJNOt2H(zOTRQBR`nQP+a2l^++hir3N{ka2v@A}DLt4T*=# zHLbelPX)3tLw9rYGld((!*$Hjt1NthzP82_uD#$ohP5itPmY?{{asl-)%KKLOaEU+ zWx`SmcFig4Hw`(#Mwq(RG7SdKIf+c*cAGi=GGnaMlu(ev7YdGISiyCWvbzU#uy378 zFJH>bIf3E<9dzJDK_2H>5DMPHc?BxoRB#!SEVb5DPS*@`Sr*}`&Mc>^wTkXcDtHJ@ z!8EQh{S&giTk6%5D6zoC`4Zpt*4r~1E1JVDQ&%+ay`f&$S2UJmhiC+Pny~%G5_rAy zfNrEUFH7lBXSs&ES2W1!ET&Z|y0t8j+c^t@rrFy^0r_mFUGk0Dv8V4E@;Jp$pdo%f z34~;oa+T!z7>@ILf=^2b^9S^8K`doA5h_e%enhyA$i(;D4&gOkyFHN+s*d6aO`hc5 zsIP_*uH?Ty?vK%E5-FU<8ETSwyb-urMub10e+$v2?8GKw#hmXao@@1UmKM_J#kl7u zBm`~3TcZH8Tv`_m&F9}kR6;0JgU4)sc2>si66M|Ly38t1DukDKf$j8CZlVT3@F7DJ^3{To9M|tLp)bU*TcyZ?Su3yU@Y(v z=KTsjpzQ?9_$pT+bTQy5FGv~MX@p+zD8kT$U{ny|1>CNk!AtHgPenjw7nFB83W;s5{u literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/BpmParallelMultiInstanceBehavior.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/BpmParallelMultiInstanceBehavior.class new file mode 100644 index 0000000000000000000000000000000000000000..389b6c8fa79fe5e313265f4a534ff7541be05c2e GIT binary patch literal 2427 zcmbtV`%@cL6#i~PlMohDvC!6PUrHMi%2EOQ2qF+3Mg!FH@b!bU+1#W{H@nR4CN2I= z{^BTO>x}*Z{!xzK-Mk^zL1E_R-gD3S&f|XP?4N)A_6LA@e3!&U%noz8KZ1`im!xb# z9Y0pP2Ws~)fk*1@@i0b_8^Izz8Nm`hjbk~96|5?8K8`hksO?q-#)__2)qT4%f2bGT zYE^nU3QI?|J;Q(gv~J5?=`}1<3M>~*x2Ah__t?@aw)?^;*;4nLw|ZGtEXN8g*U?v- zwky7GRh=FL;>H%0(vrYoI&|BCp+85jxn*hVD`sFdtl&7;uOXeP|#WQO1h3h@<7Ql2psdr*w3{MQtux)8?)$PD4IDTL_rtHKckYK1`f&#buQyf$T zfpN=mq<1o28QSMQ#i zDhLeA_EBJVV1RD`5Qvx!^(u!(!_aNRsp`9dXE{|eCuM|ml5(GnE0h^vs;Nfbw!SL^ z>P!^f-se0%YCq|%e$~Yu2Su0gOEez$rDD(-*5}iMh;Sa-D+(qSO}j)426Q^RK{#PPX?FHqF5g-H$DYPW+)fl<@7?eFW1vS|5% zh9~$^!#lVX$5Rcv*o$Ld!vVe$*c;H(|C{A;e68Ude5>JYj0vo%3Darye#V;#B5Qr_1m^o`#(ZiVLM<=Gk;T!}qHNcS?>#dpsE+CxcGG=T2Kl<@NhfGG zW1&oOda8#`9lmA<)HNIxim$%D(VCz8>LhTb;(9eB$afNK&m@66>9gk903c1+#F~u!@a4+)f9l^V*9l{-3k-tOW*uezX%RChjgg+%yq2-ZdptlnCNwTd6#zt|~<= zlRZJQTc8;4L0_sk6HSedx0o7xyiZ^n_gXlrH0p17Z1ff0`WYz&#U|w1gi=p3!&3zM E-z9yrxBvhE literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/BpmUserTaskActivityBehavior.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/BpmUserTaskActivityBehavior.class new file mode 100644 index 0000000000000000000000000000000000000000..1e2cd715029feb610f9413e73cee2594fd82df12 GIT binary patch literal 4152 zcmcIn*-{i&82%0eG&Jpoiop$chb6R$Mu{K_2s&{PbzHJ1O><^w=;@yH^stycnp~tR zRe6O}oPYoR<@BHbto#Px1U^@> z4Hs2xL97L@D7d5|j!6}lF{R=PuC~(gH5Cb5S8$^ZH*u?#XkTr^Yk0j4Z{ST8Z{ckf z@8DephKgyJf?!4gXNcHtilII3dMQ0?Cr`}jaW|FX-XLvGXVTY=?EJNy%@f>PuuP&q z7dPFs?&aKNOHbPFqA_iAJsX_rGdyWImT$R^J{s_jWV2SvDU(2Qn}-C^FvH=fR9h57 z`W!v&&Tw15K_Z3j!S0(3jbpUOuqkdi{BkZmEsjX7G}|=nn}%nJb-3B+&stfAtF;VG zdPbTry54-rcGLB^KFw#11*#*@jrf)4+{_<#eMwYkaLbDhStZzEQmRZA@v3U@L zGm;pI&NBpKj2w{Z(I)%~qsDNw zQh$Zd!nZXKe9*EX11X>oKu1h84+M zN48XQ*P9lVT(PS;kH7rp@wa!M zeEr3vm6b>Nul~OC@KJu{$#*MMXQkdZH9f~oUqMns3bP6<4Re@RU~5Rj(cmJZ;5`i< zD4-ild?G{&d<{7)XxM@06f6qeOA3}X+{XI~KG5(XK2q?phEMP*!&HrX#D5vj3O>_t z2X{4WM+d{%ipdw|kW5l}%P0plzz|*2c{#I$zZbbp z!=J8DCf!j|vJiu7iWrB9suFzWK2*w#&?b?;d(t`Ew_;A#H|V~xI^Yv)H@UVZNy zvuR2WmMbo%&Tx%_dWN3EFj9qj-Hlf!RnSs5NaPC}bPLK_w<)2hfl-g$TUAk#>>9Oy zH+_Syur!`MxrjMbVU5w4;5ksmV-^9?1vUX+ghI&B8-`AbEde(es~N5x0;lpXo(aSB zU2mDJAZfuzko-g`<#HkT)trOaPv2A9=wqz`J4O4EZ`1|p*#ssUG5zkM)i6;s(6hOx z_a4}fl5RKss&e-rQ6It{T6Y9g*o!V&5k2;i4hjkm;2@m|S$`pdHhOmTbhg|>!)dis zZI47AqA`!ATb<3FY97sbC_+gDLPmnl#lQI&GBRDEe z=|Ys=B66=AJyKE^$?74K1zjJWCnLpMKS@>O+cCUAy2Rd#IF43&-$7rqFOjn+X*Y1O z3*9(Xly-`w4bWKwUZ#H zV6yK%HZtU~X}E7s9-GA}@ixjBbVz%TlZ7V;lmYVd4B0y@;}Cc?P>jO>ga?!Z*oq;d zYaq&2qCHF22oJR4**Q8Jp`8erKzbBoq2Z_LAjJC@XeGRx>0AV+Hr66@X#V3o#zQ#* ed-2!Wz9oScns(RgUKj!Uh&GfwI~d7 zv`Sm#B^p%3;@8_@_KY$%+SzoR&BaP^0*2?AtK!>RJ3#i`u(A~FmMYW>)7j~E=5f1lK4C1gMWb#@j0`%wk#Ke+3d_YXWsL^&wG2$`SYLKzXI5g zui|LN@j9G{<0RgwM-xU8coU}*NaAz?qc{`CS-chJhp{+P7?&dF;+VkO3A}^za&$q# zWE}6}J^AhZ1U|sUICM-YFchQ}m;$Pv&RO<3&vXa!mcZVm;p8;8;4E0$bk@17Pi0Mw zALlH?)CwGE11seF1X{)lwr}Ok3Cpu6J7C+6ultr`djhS=OZuEXuj!RVZOrtXf@_%E zsie)k9~kX;qS4`^sI)hg&z*H0!}Pon+w*nXFjGZ%0)3CuGYs zUCYi)yL!&N?6|X<;n==8@2~VBsb5*3eC&v2TYkSlxV>XSATs2n8PD}e%Qi;~xhc~f zm%$TgOge_1ozPuN&P$6Cf5u{jj9Q4R#w!^x9M{yQ%o%;oa$L1ubo6 z%?zcq;nLvvylE6609xPmmwNS z3v4JloYn1&mUQe41tPM11=gi}-IzVC=YzxsjWw+x!^TrEqkwMJr;MaC2$qnuw9@jNa(h^zg_?P+n?`#zxdbfuebFmm{pNQPQ^>uqF`Eu4M#y<#fNYi zd=;LY`zi{UQ_+v33NEXd$AXGhY-VCqT)|ZZAF22lM^$`+s{#$H@ zHM}6HpsBsYenDrIyr@ zDCgJ#hkIJ_z&b}h=Ydd9F854Ha{@bSqNvtoi42>oSsQY)S(a+?22f|_^8Nyk2!V~| zxRrBDd?A; z0V~h3lK)ZN3lQ&YpIlKH#SZLy_~`4Z>aLBjQl1r<5|+Tpm9Y&1`BW5LQCH?zW~}7; zNIJ5eH@#-CtPw^?QWwW9yimsYev$8xWWa~k$fKpTI3=XqR{q0WBMbA{#+k;q;5*uR z3*tu(LfFBlR17NJ$){Qrz{}XhSp>V$PAP5{Go00N?Z)PBphOlC2{(q85bpd5kv+?Z z3M?ZQ!f5w06oDnw^+rT*G<-1D9NYXI`dXrQ(A3!yxrO@PsGRIrLi`7APP9aCLS4qX z5PFpsjP9Qg(7&xeCqY6%+_5tI7~CmguYkM$hyLIdG$ zK?i?b{B`31eeOjM2JtHI;QMe22Qi65xQN3r(1#3Vay(w7>tl3pk^$&IClv1M#~$j4 zNta9EkfUzw4MKE-5?u_<7VhuCE5y-(DO%abFm>^ozMop6v@fGxUw2GApbkaBK?Sb~ z1&0(IR?y41x8BDlj_x6R57CxEg`kdbQwXo~&vTE)Wa_SP7UsK+A=}7!H->P!>n5JN z!^j6=l#?Ke)u1#I3g~(Q!)U~Dmet9?(QQRd6{p$)*z9E%+c6B#l7w9caUV?z_X`Yh H9>VayJ;|>W literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderX1Script.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderX1Script.class new file mode 100644 index 0000000000000000000000000000000000000000..ee374c74d45023fdbe7c7f6bdadb897f71e6e667 GIT binary patch literal 1315 zcmcIkO>Yx15FIDmED0%r7D`L`C~!%FR8xsts)#fV64Hu8Qld%-A!p-kO?K_b_O|q2 z0SAyc@B{c!i18*>DyWd6s(bK^y)$o~-;Dh0_m7_d@B*HeUjO53@nj`XA&dJ=3%8#gdiF>d0`G%RB~+*C zRM1hZPFw^XX_oL4rN{H5P-#vF{D>V3rKt(Em{=;3Nlf=9$pJItmN6od0cS(54|jT5 zK!d<0Ru0pcS*&ZuZ6=3eh?}>DYpmX?21-ZsDqKcLo(_4;BaEW^GpF;3hp81xHfq5~ zcFZVlV%p;tD-`+JPXf;ug~Rk7(q=Z*C{*vVbatogVkRRRC>b@HNYIbC?Z`ACuve{J zBzLAQ#|VQrAGx05P0w`grZJzY=`awk)KjSr`77aCyY(OSx1CZ1HoB>_BH?{u1U9!N zrLxRbK>h2PUS^_&jOI-@mTRu;>4ej^N+wEjX&YrIfM14Xc;v%Y86LwE0`IPHb_lR5 zKyM9<)hx7AMO{Z0otW4EPGIu_q6_pCXE5lr_B-7VhdTsbUUovTH6Fef?#lx{65!){ z4R;hRLAzA{4&)0uIVj><$POz|Lccr>fa~Z1`pa+wqg-+W!E$K*`ZMnv)^(lHn+a1&u}Wzj3>t-@_w9pE-v$Hw~#`M=TD&yn$FXgRo(p{&7Ow2t&Tz#rAw Ben|iT literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderX2Script.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/behavior/script/impl/BpmTaskAssignLeaderX2Script.class new file mode 100644 index 0000000000000000000000000000000000000000..e8483c394e468dea014df5949b6bf0747cbd8c0f GIT binary patch literal 1315 zcmcIkO>Yx15FO`hNk|E_P+H1IflCsk8dcmx> zly-y^woM>kZyXRPbk&FxsQE(jH)%ZNdY=s=JX!Wtz@h`Dh1<^#3-(AD0`L7m#Z;&2 zRM2sxPFw^XYZmhpr6=>FKxs~g{D>V3rKt(DNGug`64Bix-e+dgF-C;a=WN9F;r1X4 zXc5@J%0U`2i*?PU%j8ImaP!u1jnzAKU+HjOh074h(-Dt&h*5NJ=JY=CAhkltR>S|u zjv2*GL>6xzmG~!b=9R|Wx1}fD7eN>LM#Jv7@0^1i5U7)8pgMP2G*Xw^c+$QkyvJ-->Dd2nIzC7R~0UoZ` za7WP+w8iFkAYahQK^fOlc36W7`qgOwTt^SkUxFJL<&s+nmP6|`pB2AB?lZdQ(^s9MW9^L7 zgs%dGM1uq$z(*nO>^gSp*oZ2X<(-+GJLlZjnf?2pKmG!68($VsL}?lYl#8ffv51>k zDq#huhP5A0IgQ(7{Iz?l8RC>_p!r%9mPi#@dM3m!3=x0$(yz>%NM3 z(p5(R-wV|v(F`Tm*)BIi@AI{BhpU9#=*A|96hG#Fun(sjiYX*PSGY+k1(K?q9-7l<>f)y(vY%QB;=rsjOq9yI4Ec4{ zBJVHMgII2Lqoz!@t>+9E>&h45u1EsA?!C;L2LXBX!>CvXiK8=jq z?eFrsc^d!>y-N?%^WC7N3>GQi7tg+}@o;x)3ehYygaA8@1c8*+=sm|!?YK8LSv z9%GW>F{ZZif5Q28V##}q=YA&341I>20Nx~*-Xg5aDN7|o>ah?e=J7hQ&C>20WSb*P ym+UXo*@YD83hiFSHF{cKmT9zp=6}WXfGLaP`l&ebJ%${vrzkhl1X;VcfqwzHCW?jt literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/listener/BpmProcessInstanceEventListener.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/listener/BpmProcessInstanceEventListener.class new file mode 100644 index 0000000000000000000000000000000000000000..4c9351f33fca1d8fc1d3d7abc366ff93396b251f GIT binary patch literal 2669 zcmb_eZC4vb6n=)cK(ZEEDlN27S`m4%Ol`4RgD8XrVz+=viq$VY%Vrw4-R$PzDOdCCAIoek!7|0DuJ&PSYa5cgn=WXDC6H!nN9YKFC_5}1VvIf zXc5#_wY(?7f(j1n6%AIGD~&`@yCjim5J*dZm_GAn*h zwq1(zoe2dwQ)IAaF6KF`onY!Sy zo#|Hl&L!VuQBbaULdUUx$?@W}C-b_P*Ikw5sgrHel9L713=DJHp7ye*w1y&(R9HMi zssf+NENx%}*9?qd)xc$3Ng!ij4JO09>YPSp#4d%kM&xa+n!vh&4Qv|7V%oqr*i7KD z0Sns+>=^hKxdieC3V3Q@7dZppDe?^8GvqrAhKp7v!+Iy<+w!@x7s=3eWQXRfmpFCR zOlxQBvfOAvKW}#**Er3Pl8~l8V7NLnThry)eu;1Iy%dfbGmkn1=xmdd;ku(6vn^ZZ zfn>PXx#mwvuy%W*S}9VSpG2#51jFR13~^;MOe=3nK~SRk9|z6}s0$s*v*mJCs@X;3 zl_9ZLb-khpX|-Bk8Cjp^QpU5RNYGTfPXRj{+*Axf;WRIlyZU6N0bIW+GC6alV6HK2 zv<`vTySQ=AG4vT0&gT_f4R(>n*jikx&GqviB6PXonYsWiOZj$uw*6$sGPS?GIzu#U z&YXrcZqp~m5Pf%a(T}DL7=~$=q}^3|zEA&AHIL~0y5<^1FRjOLgXm2Q}GVZ{fh4Sx9DN`3%zgA*M)Yt zSu$}>!;RELVmK1U2c(RyV^TLA17~93R1BOr3AjLZ@+QDk129E^A8EjucUszi68HgG zy4VCf+yJgy+K<^r7&o;yz|G$=aD?O=q>f-{`X13YxbzE{iUorT?HKxqjuV}ru-+m~ jla#94gf@)iT8 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/listener/BpmTaskEventListener.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/framework/flowable/core/listener/BpmTaskEventListener.class new file mode 100644 index 0000000000000000000000000000000000000000..a0c178e1ac5108034a4f207bd851f890fb6c40ed GIT binary patch literal 4832 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/mapper/definition/BpmTaskAssignRuleMapper.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/mapper/definition/BpmTaskAssignRuleMapper.class new file mode 100644 index 0000000000000000000000000000000000000000..60e4af8c8e543a16187383ef722f5cddbd8162db GIT binary patch literal 2154 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 E0oz_Nxc~qF literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/BpmFormService.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/BpmFormService.class new file mode 100644 index 0000000000000000000000000000000000000000..43caeae90059905e072f8c17f5c2ea559a99a44e GIT binary patch literal 2892 zcmcIm+gck{7+n(%BOz^Qv9z_e)!G9T?O3JN3Zf+x!G?nb;_1fCWG8Hx%#JgYD7U^v z-=de;3m?GGXX@vgL&#B}?dL@@`|uyv|Nm>|@0TzB0x*N0)A$JY(y;LHD28$WJWoGM z<8wSR@r8+518LinB9Keks~T9I%-gzZd3AlDtg@qj6FZKyO1dhPYc;f0rj)f%tA^6K zNN-dAT3nx<%I}GWupHr5th{zBw0m{4?gpwVi^^9tvUO0Ck6l*>B2e1(4P1&;ezzLJ zQKjh6Duzp{-sn`_5T2;YKzbee>3WT7N!*WKA@1!+FNwQI07}xK^%#Kh$?i<1C}LJ* z5Z|x#-XDXQBm%*KRgz`psxa>+c8lvwb2PD63^pen8i>Yg<4 zv`1j>HC@ls@??@oqct6NBKK~`pZSN=Q_gL?jbTaD4HE!5F&DPxz z6+-*VD@IqG%&WH zb>Ih{sIAIiSC{l#ji}*V z2Q7vqkj7Kw4cu-oLL6I@FQWHUa+Jm@)(nu?`v7o?ap?2v8xpme!_2OCgjTM*5ye_d z$%1WtzvVL-fiPFU8?u<2L*STu+SYDEdO?)^;(vhG#z?(>yNF@d#CirB*fO!5K@s1x zgsrM7x1sl?l^4~Wl3*i=vakc~F}+c~l$$#fKN=X$YrS75U1wHYS9&=|_`YPs|Fl;k zu@|@GvT#d|^l!$Bv%OQ(cG963eonNDpZ+p-dAx3WO%uyG;Wz~qNEYout7u1)a&n^> z7~&gkrPTJOZ+$MAV6)%q0lgnc2Cf}1H_Sr7+} E7fOY5%m4rY literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/BpmModelService.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/BpmModelService.class new file mode 100644 index 0000000000000000000000000000000000000000..635fc96bdd290b047671f097a97cfc884d2c5edc GIT binary patch literal 1234 zcmcIkO>fgc5S~<-{}COa}gqgS%2*7 zn43%&l3zsniTup!_6>SDXoco}LW{H5w&|wc>Fx6IP}?(R0mR_6eq0 zWQPnoo(z?+nW2K;{(HffHEI9U(a2&pQg2zf}OI~V~GU%10uRIr#49c5)>Mlv?)63Lz zVML4;%_aWU@`W>a)4wX9{v!YCR^<|TlqIBP>Cy079(^pS)fi1FSAa>Ri^}hVCXEYJ zM7>q0ZmqsYR?%Sf8;G@bN6KP|W;mc+ZU<{SPs~^eisA|86yAqh1k!>?p?4^~L8~2GFV%unNWh}uBB8O?AFh+hAw6_v&UEG^HZR~i z_!j=ux|R>%@|j$IXEG!cG7_=skL29;eSdrJfXkKVPtwYXvpckk-DVzT;&_SQX`R=~;DU`As%! z)pt`R?1~qq?Zt(J5QuY?XSt0s*D|zxb)YPxYgdFdM=vrOF3PIMbuEDUU;bzHO9?~& zfwvo4xbVY=%ggX2HLe92YDT5kaLh6&gzO*R;7i@RH)@5!3auz&!8=eio?CCRn&p>s));_-I}MRD@rnyfbd?!7b~8p zG(f1l5V?w&URrhT$Vzl+mCP2bbDgX^Jm9X-BIw}1So6Ww?OPEV0Oa}kZle-68=e+b z5iBm07+pxKM%0)0WV68NMv9&8O09y1)fFwsKa(c^9w3o+zuL$P z-%$-0K5SGN-Ap4gW!g!Id|Zj`jS9Ia^dW?kg)NWJEG6I*qkOrGZ>ex%r%mHJ56f3< zE$iKQW;Bk+wxgTeXEb}nwhlW$0g(eaRY4t$6%h5Unp+mZo*5;KCJM^tPKgK7oa4&@ z{Y)Y*xwLj1il8oS458y&La|8()%~&pS!2z3wL;8js#7yMLahC2CuwS=8HGojP9=*= zGt!Dcq|o%rP&?GoOgY^VefU*AsTl@=J#leF>!}W{j2^~HlRb9f1fDR;LZWGEn}CJG z=zed3Wl&qLAvYtS$%yQZ6jd#-#kw?Ln#D_39mhzxW2Qou?$8V{ysDHAb-?{ip`WQr zI5tEp-aaDL2+o=%5h-7 zq0Y6okIfU3VJojL0v&aQd=i;$(#1q|b?Z%9WjaTOwz9NM#SHCb=`mqG&b1>cy}EiX ztO9q-6^=~ueQujx&S(@9`tX#YCya&)Ni-dU{d67QL!c-6o^IfKs!2;X zX&xuz#sb~K`)%yA5uLet8@pyK-hocEtJc}V`CYmj*?N!O16M!()tRV$K)YB+K)FL- S(UjWMyY(#CDe_EBF(g| zlv~-2sCj8$X;!kzZAD^K9;^!FpVi&6`a`;J@FRYCm<-bEU(9iE5~&Cmqm>C#@l?D- z=47+vzA-jX5NkZbSrd9wYbqsghH6^N7`@z%4#`o)f!$WVH?}LM`BBPP>-_(P{iWR0 zRT0u14L!TH=-8C!Vx&H4N6CmlIR5$9V76c#ItU1@_-+7fBkVyty8Y<7fTsx1i885e Ny@QQKIEcCh-EUm`f%pIb literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/BpmTaskAssignRuleService.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/BpmTaskAssignRuleService.class new file mode 100644 index 0000000000000000000000000000000000000000..ada3fbb8587e03b3dbdd3c54f98318f0b2bac668 GIT binary patch literal 1606 zcmb_cOK%e~5FUrpkn$=tX$yVvl1Mq=3*wThQkoPADbOlubKvG~CW{-#PS?9>_pfo_ z2k@g1<4qa}*``t9u(HQH^ZA?k#`ELnx9BDQ4~tW9axD$u0h};i~?j3 z-w0%<=9w-JA`?sAR_0usDC8~E6;gB5p$uc*8gb=&z3zBY)q~P5@pw^>$hV74X(a(1 zia>NR#GtIT35843GI&`tfzNHvrPp>2=O*Y3a{e4C{C z3~v7?`9dmV1~!Gk^{P(e)HkFQ=`$fi9c`~UkRtB|C_)^Ui9x+MMd&MWD+yK|&j@^2 z1ygJJTn%Y&0s^ul=t3^Fu= z0djK;ZowQ`x5>)DJY?Oe`^n%A0R7Iv0__LC?~;9w+%N6>WIuq16u9WZfGyMVh(3ps nb=QzVK96DPTAw@=65nN5N!C?(lB`c*ZFJu{?KTKH1Dn49S6%n& literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/BpmUserGroupService.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/BpmUserGroupService.class new file mode 100644 index 0000000000000000000000000000000000000000..e916d02e29b8cc63c49170985d97e89ff15bb313 GIT binary patch literal 1459 zcmcgs+invv5FMA&w55>JLJ5~r34xTn;1^y{sv=qhLW)S$(7ZdF$;y$}UcBDW{U5%9 z2R?w0LX4Mmv)PtQ6^VzHJ>GN1o->#4KfZngfETb=gWVcDW>60-3Wui&+IQBZF@q1y zPGBN#Q`J%kyFn2L?}7T4|(5jiTY%odMvDn(4n2<*VCA2P#WaLI$udMm7Zo8g*lt!MYY}cEL{|clc9rm3e-IN%7MJlMj--}ej5$@`bV-2C zvj2`u9E0Zcj?_uTkBm|nc<#CK`LRb5d#xu!EnI4m!P(`s{py1cE;!m>B!mNcMxz?U zVb-JUrr=$zWeA|%;HS1u*bJMV>ABhfCR1V`C`Dm>w_ z85{l!nWW0+>dSw*Xkm%!d6yO?Pbu6g7gi5l+xNxO%Hd^YMzHmsNo{~{r6Q>t<(bk$ zJCOlC_wB^s!hS%z IhdJ}nPnz_~(*OVf literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/impl/BpmFormServiceImpl.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/impl/BpmFormServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..5b93bedcb0124edc6de2cc4fd21125350ca208fb GIT binary patch literal 6525 zcmcIoX?Ppe6+I&_6H9|loH*1h0ZJMiJ7Fjdt3!fg%OE1#A$A-L1v8Py_9T)<(MTa? zEwq&G`v%=tx))NKI8Y$86k6!M(0$)Gy8h}%q36y#Nn=|Maq{(t-psuB?z#JY_v!Ix z?t2ixTKqeTcHG#2n{acC!%G^l8n?u7D_$B!6J8d@Z8EIJ%VqU;S$KsEuax0c4R|$P z6UA%sIvuZ^5 zv~DS83r@x!c07j^UD>Q#uww91p_7Ta@+`IJq{`gMZ2UPDdWs$mVa z-R_vJp(*KP?fznByPY2r5i~R>-ISFcw(^dAtI68Jq(crHstN7c`Q1*+4omH1a_N%j zgTZowQG0*VbG-t!QBV@gv$Zbjy zpIf(^PTMI-wo(L!g{X1HXp?1u?Pl+ng&Aozl`ExIbq?!h`#_}e4twuxBOQ$mN)Fg; zP0+sAEFh+ZqLi_fYR9(>D_dr2I}4X*=G+~w8SM0mY2Tf>65TvyvU-K6aMN(d|5?hT zv1e0i(oXG^MCo?3N@T}f z4I2-^Wvt-Ns&aiiGSYpDm`6*lXK0|42P#il4UN89o9wCn?tUx7NNK626>Ch*PJ2oU zwr>T;PD|R9s!=7Oq#7+6mt`{1uzo>;ig9~|wj?C6-1Jz|+HR*c)cLSH5&b8*tAbZy zUSiGKCP$=iLw}VM%)=F-87V>GWK~v{bUiz?jf7Qp&GpWD+VgWd_64nl-#P5paC$iL z=jA9Siw^qdeXk=&`ElxG3LU{`h%T&aNjE=XdN~QT@jTsit()IzdIdXgy9L%JQ<3;u zRNJb~18Gr0?Z&V=CB`}poP`(Zm^N?#_Zhe!pEqy@OdSsxcn}X6IEaUJe8Ipah|BU9 z4Ll;#U&5mX9>bS)e8s@yctXcl4SWq>r!^}Tq}(s5LU4#9*}yl@XW*OIY~Wkirs1Ob zVGb|gb$r{vcko>U-@}szzArNVz`zgjBLhFilRAE4;HUVRfuG}H1D9fxj$at~C4Ob# z*LXt1;z~4KVtJDc3?2Io{06_J+YS5Su>|&PnKpZOAcfu4ABv?snfIWRKX+m%R;mVMMc(BGT5xO!m>IeO+b z-fy0y<_O<7BZ2wXPp@|+=ebuOJ`%;1yH}u4=zIxOrs?YKT}4Z5SZXA~771BR*=W@j zj1P6~lAKvNk8{*<&nBScBFDQtlRjhvNa&`(w4351qr+XvL=UU4Q!ePsO8yZ4o25c8 z8cteXzrBamihp1br8et-EG_L1tEA!hRMwm<7F;)N`gZhJyS$*>>HO2AznEqgtPDp{ zN&C453CuCK&O|D2e&#MAH7dsq&^t5>cMoj8^Mezilx8vY{Ew()r;|CX!g^M!mGe6qdcxRzhCs_|LZ zz8~5>e2L&}js7LVOK(*YJ7#UPL+$peBO8_Ipvg`T*)8 z*tU8f>hI=ifY|8o?120^*Fv!b`ILZKArl9kSV!zAj>UyoPfBJNE>hek6}RQ>9s8i~ z$D)91LxpRUTq`;>qn13@Vi9>YlgCMvzruBSnd@?NVIz4oQ`2t6wFhy#Bwe&lH9F-LTA@b1U*k^MNikGXUTeYWNRPUZhTxQZ{&*Lb}= z?Sk6Y+U6C$T~}OC*IGA?7i?SITDK1?WjOr++G_l5^{w@fq5iIC5AMaP`UlWHQqx*L zI8y6x8XO@ZDF&y};pdZ>3kXbnJ?p{^tZ+B71m48?o0%}TFg$N%T-`?9ZuhlY&q`>r zDooPsb==X09n`L#bOUt4)qVhYW!*ibk)iI3xR!-Wr6oRtT$Qz4d2JWh#6qW|=D%pt vk=NlpgN-^0I*J^h#mT7Av0KNrT;#?etoM*?ieC|2=hyTo->&EP4Z!~ZYVJDv literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/impl/BpmModelServiceImpl.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/impl/BpmModelServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..ce8945888794baea851c2db67540ab0943b649aa GIT binary patch literal 16115 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 N`C8s&k9F|Y{|B2We(?YR literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/impl/BpmProcessDefinitionServiceImpl.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/impl/BpmProcessDefinitionServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..e6cea37b0133cf52a718711ef12955abb77d5adc GIT binary patch literal 15896 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 J_HKU9{{iT{o9zGq literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/impl/BpmProcessInstanceApiImpl.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/impl/BpmProcessInstanceApiImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..66e435aa1370c0a862d521dc0c6bdb0494450899 GIT binary patch literal 1267 zcmb7ETTc@~6#k|y)|M*BrGm&sycIOgi$0hd6NHdpHbQ{(eb}AWf$q*aJ1z3BG%+#y z;1BRe8P7}`ZM8P|u;=XAbIy0}`~An)ZvdX+VHp!xEunz5N!-DD5qFEYSHyjWYHD;S zvurQPtVlw6Bu!U^lA#`iI_9RMFBEUK^|?4{OP=+#+=}dsze?kSkv6k&Hwe>H5|!yn3bJIDL=jRDLfYZReUJ37?!y@+9YKL1}ee>o`X0jmteExpq<-NO{ zd-mntn|$-_1CJBY0^isGP2vjzd?8;{Ob_sSd0bq=A>JU5uynKsIKmqP=;)B%O(h)V z&GNZLK08aei#J;w3visf1AGZ5iaE)t5K+)dA~d!3ve&@N#B7IKFD8| zwy((N<0X8Ezbcn{S-&Oa*RUkVv73(L02 z*jME7D|!4{9Suvq0Nk!hhkv z3N3#tBR}sEn*Uiw1@iV3|3|3$uf=Z#=xV;8j0*W3OHqkp-W8zhlqw@j70Bmj%cw{d zL6KB3uvP)+rYeEsF49R3N!F{IiO0_Xv|P$fUm}sQL#EsJ||eJyjV?ClLB}g zAE4Xi(V!-m&|PW@bW=^W)HI;3rh^fxETCqn6J%b6jG8GOCqnq@B+#U0$z!%WDlJtN zP}ORVrRFmEI^!KoWlix!M=;sBalz(bQ@o=ilBi)CnF=Mhtm^5EtO#{?M-ohDHihF| z!9-7dM>M#xGrldfp)(Tf(w)Kf$i`?anu^9_!6n^YZJ1e`Oh!9mnfZWNzN#mdigrcT zM3Yg>t&PRvsgTS`GEHjQ9NHS%9t^pS!BvrDyeAQkV5Z#_Z;y1gMiN`2;Rw^I`7lYE z-hQa8@Js}tIfl~iL^Rg1F%jyDY>Owh1jF%IDzZK0UA!rD=?;va(4B~fBgtgFOP`BG z-SClEIRFqWP}&2$mnGso-7eso;R9aXZ?do|)ZP`1Nk?sWlxgl@z+~-Ut!F&uTZuSC zBU5J&Idzp8Dzn-n-Kk;Dsy9PrR(Cu(+*vElP|PZ6kA_qAp;Sn(H)lvJ>odbKdjxd3 zD+=X|$MpQ*kn>l1CjdfmGMtEZr(hOLqqN!vds5NPAR45%6>1+!^(3IDmt^-YHlLlL zSO?53){$4x8Lf0}EF4cnf*T^6LR+KpM9{3BZDy?+qnb0Ze&J#<#4};qvu3Yh^4Gxt z1kY$J(%jRvA(ChlyVW)y4t1^xC8F}}H2PDUqAOD-!JR-?e&i+yzjF7Ey zgkl8XoN$Y<*}cZ7*4S_&5=wzh-f+ycYSti%J0RRT4Qo~85&(A)&3G~qx%4odeXOt{ z^vHHtwYUbg;B`HnI>mvAZ^g0tSn@LmdWD+wIpvLIjxQWgr5sg-#I@xcNCEd<80z?S?_%N#@-*vsyduYF^W?0< zL`$OSMILxqz#GBS;ptcwu++(xnup2_Yz@b|ck~m^u*CxHDA|~ZcWJjNAz)l8o;Sc2 z#Uk5;&EVnj;Z2e7mh2U4J3H&*v5isG1mp53gd`Ly3h<*+SuO?8CTf$(hSt*UB=Am4eQIn^ zT53MiuK(|tBJ$&=v@w`fUrQng9J-Hp!7`5M?p+pfeRWMtW#H=FepN%Vg>@HoCX3LhJ(Pd zR_8MZMJip5GhtVnxRI%7t5cw77qcxJHb=q|_=-0tv5%2RgiypzYzoJMn|f05cxOSA5BW>FMI>}Wlnf>qmo2EEgqd%t?_WDUm`T% zR)&*j4X*a->NdS@9=p^rxHHtXp*>WQi+aUEB8RaVZvp&J=jKh1!(*5A{^?nQYJ$^+?X! zYi#-kebZ74ZFPz|)uyk}Gd6vmzF?`-Y;`)lV5>#+BU{y|#g;n5R%fcSZ2BQuXdivi zQfJ$$RxPpVyBJ81(tbt5uJsuCdj%g6DPW`cicR5ZGC& zZnV`+>SkMgP<_Z&A6B=Z_102ulh%(|>NZ<_RPD0qAbr_V|7WX@savpxoZ-AvpW5nn zj8b=~U6$Hyt3B%DHhqO2x73}s`h>d0R(A=IL-bWk-EFH+O3SCDrH>9+>Ky4KOdH0v92W<7A`mCiMveoC*=WX=`wF^Hk}YY*?-{rV#Z58rk3tjjK+ZK;QC^@!RBlY-Ll1TMFE3{SSy7j5+= zwHlRemSun-snMJ|GSnRnE_DtSJ*zEz?a`}VzyIb}4<0;x)2FaVQ2#%B+G{sH{%Y@s z4)1*U^@l!l_|~1k#tbnyiLzq7`2$bJ%r_3*aQNOm-f2RFD9WQYPv;pnSIFZec~r`y zN*>kncn?pudA>Z(fRyMV`W(~Rb>V?`2`*eW04n{L&k%O_%H3~XdE=`G4qRxd{kD2c zK`r`hbwC}o)t6yr^b|d9sjt}TadpU6UsX?xDo{_Bs;6xAw0{3usd~m%&#EVE^>y_` zT4!~ZbxtU`2>}qA@QqUSO;Pl3VbetmU-#>rQGzc@Q3z8#H_B4qE>+LV$nRj*cS{u< z<4x*&w)(#MfvsLtKeW}4#6SI5on@(?*y^Y1XUH@y^^&cAu6|*wU#gdF^@{pcsd^dn ze_g6xwbfzunyp?}Z$M6YbbF}W(Nb^P>No1Qw)&ko%PuScWgBa&->W|$*5`qE3|W$` z{;1Yj>QA=%Gn_J(XcW5sVynN3zx$g|{&!)|Kh!^M^)K~Ll!bW^fcux@YwwxRkOyd% z`nRqAqyCHVKRk#7#Z;WNi%>$;XN$kOeQ$qFt z2ajXCBivooVvUvIR#_HW(gwUHWhm9o*c36T#A=)eqtcBgo8zelp$tR4=~dmPP@*+* zNlyg(=QUEBEX+e-9({piG;&VXgVTQO(Rsv`wa4k?aI>>ZjC^tiCAGswAZK6MMW!kb z43{lQ*S!mK5TC|+QE%vsNSs0Q(tbN%dGYlW1$CVWZ&_Qaj%;T1lTFywx1gw=5s?*5 zI`S#^mIoQ#)Y8dkd$=>QHIhhqtr>#I&Y=*EWem#r#^%u;|;In2pN(NM=Co&!Ma49MoN-mM>(HVmDKX7=j?7~!4(WE-_qUoa00 za7}G&78FNyjgkmY`<>UHl*e3aW9@lH<~s}bXpd_nc~sZo$M6arWH)(0crvtxL?WJ$ z%NS6Kbc%LdTW#y|^|h_7jmw(XuUg&Iu)g7ajje61*iG|P;()6YrF0G!bU}fM3o@(b zj5FTIL>)J$GJ1~#2REbJ@)gQcm)##{lER|dLaA+tk~KUNP#n&+Sv zreXAZ$qref(lPL8T&mqD&Z94teY>>J^Av#K@T44-YHn$B)oQhu>p23;h}z5BoMCz7 zIv(7}%STfwX+<>5gNH9f;LAO8c6ggd#}FPTYN}SRtgmfrSiiKkv1z?fwW6iIq3PYK zl84{_6IBZI=;L}LGJi69DRww;1*aW*B6uMa|;rh@5 zBubsgK&GZ$tQN<{o|sO)f=itjD9_0mi*lqcLzAdI)2W&cohEA&M*?XE9Fx|$TfH|H zd4i}Rbbxie;mM?XMNv1m9G``BI_DHuvMH2o#-&ZoWvGZhra!Xdh{I@%fr2tk>6a2{ zgv=PB23D7Ck8Sll)wVZh`_)A!DRy5(7YNgnz= z;n8!eFei2YN)N3uNfJTc$2U#I+k=c=tRi?px8LT+8B&-mRFFDy5I_}3_GYoldy6;?LN+`BA|N2vFI&~J zdS(7yyfmO}9dNhXTa8F?>qJYzZ3pPW*1A=VE87A-{8B-`i<0Kp$KzpLf0*`Ix>h}A zk7aV%e5FsCEUikOOUNk2H6iIH&{5^p0tZo#&-8dwvCX12wW}Iymxv`)^z&%?8oRf; zG=8lRrr#BU-#E*=kat{7KBddYy}4Pt21DKgF)4gqPa**?SBr?V71v@Vs~I<|(Q*9Q z3Cy&Nre#^sxj4~u?8i>U<;(DC=+Np4zEn{6{+5F{d`ZjunV~ZwA4kCb89EoKTh-vn zMb_8VHrF@m?8Mm1oD3ED?Ir6{IT9SiCsAy1F^%mKV$5|4eKelFnZN%rnWukDX`H_DBx2*AMPs*3$n~e zUr@*c)I*}UC{1FiL$*Ht7g`W?eSWR&dE_uyUFSL%=j>{(H&P0Zy~#>$5Cd>n^q4OmBEq-7Cn)cjs4=A3ri<`?s%j%zEHdN zuO9F1kRm&cvg;yRuJiTbcos>?MRlya3N4v;3K!~dLgm!}nUvR5Rwg3t(J+o)foPGr z%+Uzarh6y5s+$=sn0xXcCo-6`uR5J*wc=OFdXgqhENo~t;W1wwtIx~&&jB{YV2lBR zyaFu5U0Ch1GD&qt_Z-X9T-P;;rylR`XL0M?Ta%lNMc1$C&7MVh>~WY*3dgEI+gwT$ zscLsPK0CB#w&dpMig`Wq*QS=WrjGWargxyEu&!%qUfQ?}F&eP5_tctk=^arsX*NGb z;HTLYY{RehW%NU{0KcJyJqkEH8D*+%GwQhhVR&gMjB^Ae!y*Fn`-iv|;dP$rP2T=9 zYb;-p?JM?;uz5AFvALDoEZ;~uoawcEqikQPirC!3D{XG#6*iy4jW#b~9MXG^I-{|z z@hy>HlfGYsqv_DbG&&`?;Gh1od}UCJrg(gdT=}pYW3fm=VAb6#8>c;=A%99$Mg z(NJL~dez3UUyqSyO_Z9)JAC+^X34?ldcC6qxh~Y%*^0Uwmf*QqkoJ3X2UbxxCGZ4p zp=d|HEKu#OZ`*YfG;-Axo^ZcAP9OmY<;Vuf= zp4Iq%ohIWQS5@%*7Frg88r;tyyYeyO%3e~BlFv~599o4|qEpFFZS-v!WyaF;^c{S$ z+*iQ}fpadQbdnL3Um|~1bsrV@GzS$49N{C;`5MMFGtPk_FuzCNci{d2Z_2^+MYQ3D zj068H{1jYDH2gj)R5{?+Yw+XEFc*9Y{aAxH z6sM(>pcYEfd6c4cv`rIvra{{!vcn+~q-bs`Kn~N-=p_(n(Z%$0`UP5AXafBbZ3Xl) zRu@K@QLkW>>H8J>evQ5{x5@x z@o(H?G+v8nKOKk934K)VqZQT9(DD0dV!>Bv(pn#yk854)m)_Q1I{pwi1U^qz)I6tb z>VBFg)4{Omiq4y}pJs?1`ZU*0q++~Z4vo2jM$?rvg|4C*bT!q`_0&K+X*u0UYv?A; zp-O|6%b|7jCjADePXS+lOTPo{4dB4<@#fPM7ZJZjH5UDW{)jb-iO1u|pYZR`_=k*z z^#{{m=&ugxz4$2_-y)=+fPWPk(r5P3i3%1wWkvO~6xc^6_0p_EWmBv7)9hZV+#^lU zwONORR6~BX!WEu5AX$vhkUN8l@O~@Ey$weDQ8=qzG>dMh1#|~UzLT2i6CQ(VaY=0f zsec2)lWCqqs))3h)LS$Q1eOg!;os>Wm<-|a^*}^8ep6)?lzwh=C4+#ui$pj+_fv6_ywF(R`-e)Ff{U80=WM$SN-=xAxO1eRQhuXW^ov z@}g%o1hk%3UW7+4oxYpK$f!l~y{MOJ_Q;66?>y{(2-p}4T0*6GzX#g=X&OcM(p0)n zd-GZJAkBe4Uk-160sMIg-aHIf(F31-IeuP)3wPJSckjeCJMn0%z~l>PHvNa{-649;{QNXnHBqtOkjjQ%q5!d$YhT8H2z1RMPU45 zQB+LiqbCodopfVF_W(0<(H)70@6hbwV90)Zz2@pOM8(UF-}Zrk|AmZ zd@bCk{}Iq}wnK-Qg+YG2M(jB(@ogagJdl3@i~P`I3X@!0D?+;K8a$c z!qrFRJ)&+z;t(0DL7gO*I%z_FFjoam)5Y>w~v;hA5Lu_ zEt8jCI>(G`#K?1H2@JuT>2DLos}^ImVyD)h8UPZN6D(mfM_ z(dx4Ez^wO)39jv<^L-dI3uD%$#}w|R2?yzdwPhFjjSmKj7d=Mn4`uke6h4*cCo}>9 zav}sf2mCr2KTk#Us7L&GFJi|<;NKQ{g|^eLU`4-%V|tbDro+gAUW4kqiP69H$l)bd z4lnTt9tp04mpqC~J#sKDqa8T}sA%Hs^78T%MvXP$hUL%9;5wF(2%?lxlo8@{FJ4hu z1&0)p0AbP$ldk*;!ST-?G)A~+jNozb>qx&LSaJPMs=>!W*tlhA76E!j?AKKyEN#G^ zY9ECeB-)N%1`ksE+OkNYVPqen>XdM%$F{C5FvAUhI}Sl&@Rk01$;{d`%vruZis}F& z#2LMxhd@B^%?iG11b&W&pPB$~H4~ob4E(Hz_i2XnSVwOoTR1|Og56@sCf~ZwWx{n1 z6Gl^tCuZ4%X?X=Jh)w8#Q4dpH$dmAi6poL_Pm-&c5Q7-VOK3&(=aJlOECk}%tgXbZ ztbUBPH18!iCNYidj!tykVN4~vqYE8Z7(>bKh@oSnv6JkMI677sGs*7gM#lnUC2j}Q z254P^o{7db+@65$NijJc?2|QByHtUbOm(p79LG^K<)46F|-eg|0IvY zHP~``9#VJ_68H^IMmBOh7w{Ced72inPIdspq~#(Hqvv>Y^ju-=6hQa64$!oWMvDj= ze=F!Mnq=|R3G!e0`2QmHd=~ABr)5cBF)p+kJHZH(l8=P{*#UdJw2v;!$;!DxV`#E+ z7ehF1u|KCM;S;b9?#McDJMmN8?gYSXF>v3nkV76o02luz{52Vpws)RIBYCz*)hDF+ z0PdSrMpA%hW~sVv;S;fvp;9ow1>HOtG#U&(U@*8MF9vgf!Q3G+n3aRUEa2_MpfWE8 z4iByzj0f|8!TcdH7@7yTnhcC2vp*fuk%Fs0=+C%VUCHsO9-$Vv>@LU=YPAb)7l28j zPywN?GAz9c1RbQS*FHwqG*{0#K-W5745$XgEvE5&2F>6z^~xr@uW(nc za0NY{C2lDsArOhH@S=Vr0bvdY-9AzMjPP+28$CjJAKE3usyRs4t%ZYtT3^q&O@BkP zd{hF_oigY~h2%*ikA%3~Uv&Tpqe;2S{TU6_$qLueXs)N}+(ebUf==URO@zs?YFr{} z(pHP?-bAO-7=JRz@u4Ujt;z9g`xSC@AJ239MuSz3#X^^Hr`(W(*d~c6QMk$kre39H3W2 zGkF0oL^uJWE%p$pa4~Qar(Nb|e(@~x23{<?Srws3FSNs36n`jjc=5`(Li4J@!9AT zaKwY{OpO@^K_tb{T3k$^iQK?T9gIrI_ZCgHc$x2P1{T-}b+A|tu)@0$mDNDwP6b!} z30=*4wt%4hnUrLN)@WHNotMemxxAe4ckE=J)d74f1Szhg62kBypVAcG)ja1}D%(eQ z`*nb>);kGRD1kniRYM#5YXHBOA_6ai4K~tjK38L5oJgfB9b;>fq*iLm#bq3?@$qaC zOt-ihWMeDX!A>M4e$tkd^izFwkF&m)>NE6uM$WcNwaxE^i3k(1+w4H=0Ih;Sb)~L+ zp@S88JFRqaFMV1Iaoj$-*N}|%`!v;=9fOft0N6g^e+qF&W(*;T`U{sp*>PE}!1 z>~lMdu60NlR|S~Uq}Kn8Jod`tet~XiF@@I?!0<_I3_gXz>uGFSea%DLI9(ZH0*!MV zauJL^z128fQ-;K{?15f-@QaiZ(Ae|F)9eqPf4 zFW?*PD?zylOT^w;fIo%{8@KZvyj#ER<@@;o=k0U+d8hZ={Jit_cm9X-_AmaoGwuk# VZQhhm`JJ~yWyzZwtH!DE{|_4>$u|H1 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/impl/BpmUserGroupServiceImpl.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/definition/impl/BpmUserGroupServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..2dccd1c2e9e3a013711a77f6f0d33248bbc3211d GIT binary patch literal 6534 zcmcgx`F9i775+vx7M2Fv7|fR7KuZ!YBtu$K0wz#w1QC^OYS~Ugn!00YEDx4u#AswN zUD7S-zOSKso37~+XkwN&OZPTS_kI0KdV2c3nbBxujg1KB^as!K=HBn__uVHy|KBUm z0N9Ox$IuO}6*I`id6{j+cIa`O#9R#R$j49+m+dgbS5tf_ipwc+v071r9YY!OQJfYl zXT)VeT+YVudc2{NGTs`)+wk^INO)%)@4~y|cn{v&iud9DFLs0QhYZ8CRY6{oa6{^(IV`6JDmmf58pI(7VoWyrB}@vBY_G0uDIt+>}=f(^+-L66f#OCE+DO1mP)HJwDdSO`lkq8hTE4nYJrgP9Sq8N0>@;W$7mE6WPBG- zGD`xhQ@h z_Wi(&L9lh7uA6!YZXP9A;V6qm*#=n%iXU z>=@#S2GfLU>J z#ds{h&OJZG=Egl1a|g`91z}svoiN>(&7LE4Q*;xZt>LUVf;i5Il{kuKY?rWaIl@4r z^BgH*SBSP5)6x{pC>Kgf!m>_>f=Wa2La6icy%52#-&Aj3`F5?f&&fM(k9@R^ULeXB} z5jD8|1PALAG5j8Xpc}&yJ}h)Ao(iwl!z;6Cb=zY}u`=^i@uo^iIl-7I8_wvj9Pu9L zraS5)C`e0mK;@tiI5P+j(33H*C2`B;QF$3J1g3|2xxsl}Y~0H2)QreJ z3AYD@3C19hFGgf|xxlHN=Pzf2ajfSo#%?-eS_w5XD`8s@s-9I$asg(?)Fw763AfgH z%q_4Xr%>EeM1mzpwG$(U223@>don$BIGJ{pi&EeS2OSCPYs95(F#}Pf*#;~0f%?=AGrZCYDwCAo8wR5I zlZ-#(?=oJCDVAf^O5K<@=QJhdjDnm<)EPBnn-()XUZ}oB@lX08WtwwkCZ?P;49yzK ztEG}gx4aN6Mi0-OZSGNxY+ftva2bO`r00TwT{Vp3oCkyCjem4SuFMN`<_Je&dXR8!M!^d z(eebpx`=HKWj|uX=^}=*7o&N zq%e<`#%lz*rrnKII7r1`=?K~2GWkN*VgUOI84(8o5Bh`$-Q}Um5f0X|3s`drZO;lj znw(g;@IzRG!&r|a0r)N-zRQC@;BZOnDd8YBaUD14$!V*~c@FK>(73@q765MZf!jP_ z35OtXL?B-6;l(|?jtGw&6>VLx(EwQ72a98fHb_X2+RZAla)c{Kxe~!Wd`jY&C+6T1 zF>75h!33RHM$D^_@p^oQ7949ph6P;~wZ!yo+4x>%W9`kD~c} z)iJ|6uP|slUv-W2-*yEXfXmnz;pq(I5S6xtwFXQf%4%AT4AvpbOr4>lO-GMD7uok= zA3Zif>Fcl&lem|Xw=y&C!~M(_;a9h&1l0+j>H(kXHKg)#WVPq?2RmY-L6ca-_8hZV zl=R*{run7_m?*!n(p-NCJXp=m23|G@@&XFDCCav_3gr=P|6l;L}p le}Cg&j28R_|G;0d8E@pfsK7VTmWS|Wyv2EX7?0o`{{tINZ{Gj_ literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/message/BpmMessageService.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/message/BpmMessageService.class new file mode 100644 index 0000000000000000000000000000000000000000..5cd873eb8fe59c5d78eeb9eb605e88bf251c44cd GIT binary patch literal 727 zcmb`F%}xR_5XYy0p!mg!m>4fc6TR3AH%_QAF+m|fP>*GI#Nuwdwq=!9^WX#cP{!%v z0SsbdJWSf@{O32FPTxOX-vFQnl@e@~V4K0AMHOD5wPJ`j52)Hk2jpEU>x2rhkt9Z^ z$l&g%8R(dsv7ShNAL(Z?h>(Xm7E%D|hwS1b87Mnp|~uA|{mqp7`fv zP+Krii_HQ1W~ literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/message/impl/BpmMessageServiceImpl.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/message/impl/BpmMessageServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..ea898380b414995d34e0c8ec6041b816d5a29518 GIT binary patch literal 2969 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 MJmD~L)a^CkFN&3(0{{R3 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/task/BpmActivityService.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/task/BpmActivityService.class new file mode 100644 index 0000000000000000000000000000000000000000..ccbc19b94ca3ad6036a0388b37e2dbb22848b5aa GIT binary patch literal 501 zcma)(!Arw16vkiXT-{Xg=3REsivjUg5gaHE3KI*S1j({zeX~tQlGgQK^WY!gA0?*U z%8Gc{@ynN&@AtiY|9pM{z$F|8uphtygOdc^jdWCyD7sZa@zogtNkXRD*!QOc{qkU>ADL;e(!^|A=%ig2DeL;o=dBW{dp@8ZITTFs)Q8sxBBst z{|#w2Q&S9Log0aFl={-gEz%fGK@!ia#WA?}r>R}t;7-{0cE|N3gQLyTC(}7f*JHq- b<3A9f(`B#&U9Wd51wF5WmfrJDzjg+{6`Y=K literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/task/BpmProcessInstanceService.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/task/BpmProcessInstanceService.class new file mode 100644 index 0000000000000000000000000000000000000000..df8d17657193ed5b1c0860d1a78c08db8632e6e8 GIT binary patch literal 3929 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 literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/task/impl/BpmTaskServiceImpl$1.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/task/impl/BpmTaskServiceImpl$1.class new file mode 100644 index 0000000000000000000000000000000000000000..f2d10002d525b388ca12ab6b21fd502059b36fa7 GIT binary patch literal 3088 zcmb_eTT>KA6#jZy7+FR@6wE~dq!dA;d9*6O!$o|`AFxt6J;TBpr$XGyR_%26>F#sB^YuBW z`;WhF+y>BwlciXJE&Q`Jgq3IyVKcVz&x>Vv3EO$^9R2-^*WYj47D?qIK!G<*TS$Ru9!*5N}Cx?8r02kc|cbsTeZeC zMU}V;NlT`5sUwx_qd~7X5hD>-vQ0BI=Ou>jhJv*`3NE~#As8_eDnmtFGt};MazM5E zI56&vsmS_%+0yuURv6^YFo-e3P_2kA+qO!>_Chh$wUTdT@}NV?%w$q?7>XKPRGbfn zwzz2x&6DMlB&RfqKsctAk#@VSrf|YyhV96PqLM~|FN_``ZEb$Y5@u4?3`sQ{&B=Jo z&yr4iI`@!*#l4QKjCRQ>*KGj@DQ4)bd(E_^s5>=I&+> znGkSD#9_Q4;#sT~u?n>!)}UELKi(Aa7Ty*hix_|+A_3Ceb2yz!$PQ8NIuYSc>HUYf zPq9S9vE$*|>2H6U{`mIY$urrHE@v-IWp8|b@9R@D7cXaTeevgSr?a14n)%>X_Lph{^+hEw>6W|A=~T+J9I0BzWshYJ60Kc4-dERYU_Cdb4E&cw2Yu*;{d54Md+adT+MMO^6yXr7FrShXzEVp!*c zBrgWM5O3!twsc5!dQ>~DQ^Z%3o%uY78II~1+U`S*Vp=NI&UDhAV|1@&@^H#kU{7n5 zmQZom?5L5**{R7#mopPSL>ZhvF?)69A)1B#sDwLGOpds>??X<+P51UKw!O)n`3BrM z-%wfjd`coax+Bpz!w$dmeH?P=uuBPDj+lH+FCq=%a_NcPWiyV*7`M8;T@q*;5u5Ikl*lp(J+ zc4T;zJX8{M=@3ILJtkLz;TZ%fEBPG_P)ip!JvamOUxPZDT}!%Mf%Go@u;F`mQ1{iD10x z4uV&>U=qb#HieP^x|;8xY7#=#B2QFWge%Px2u-5wE7b4_kxy{Z6cz`OQ*aaI2d|@o zKULc!l|cc7qSxVwMzEE&O!IuiYVmP^{cd~)* z?=oDdJ6>Dc?l_P1tq#|uP}5yYYf)#3VKUgU{OlaVwYuThn%8w6TLW#MdwZ7Ax3X-v zqb<9(v}}eGH4;fD*DYZMHyG~c#`ZD_W`2_)UU5v$kg8jPH@fx~_gZp845#Xjp?5ZQ z&ywT5FfO;npw@)oUZtb^KBwW*xahL;w9BNv9gs7}wyl6+A}3L$T+cDM@7IJM=)xd4 zmxedOMVkN66(D!r#TNOsf@h#C_Uk>zAU5JcRkv~wx`?tk>l-Zfu{>!Hv-EMn%{L@#|yO9 zK-PRLj2+Jr)^EC0VYoUbCWg~p*VF@wGJm)FYf$!mt4;o9a>Iu{oMbXi61pM7Bzt9! zQf+ViEXhYj2*tu6OX>6v1z9tkrZ_qG>QcaPrp<#xmNk>axqN+a49!3?Pp*zV)>q=7 zj6J#%LpgVJnfx%V$L7)j%L|fa>ou`FoP=f9A~uRc#E!guWvJCBWW4dCjg}p5q$(O`6mRjgfOhW>|ZXv6d7XwrwH{!d{L7B@HLTdQxfTI zF0Y}rusewE*$55{N9=K`2EB~z7sg5`qg%4pXtv6YN|j+QDz8Qu6Xi;On&HKv^dEix zk(bc1{0?Wrp#l>H{w0r5qzUuh4i~bHRXk1=&D`J*!^>nLg2sai6XmH_dl=69T$uec zIdC1;M@|vmh>V9ZG3GaMkseEFFuaLaIxTN5fLgl7=s^{ue+Fl1_HF7_Xgon}s!)6a zRwz9|tnd^QfAkdJq3Wp8K(A;h+!?1CjW$5EQJPd#K~Yfh5?WaIR>jD&Q_3Nw@_j= ytgzcXU>CzB`@k}|ObonFfJ!9!03Y`Dm?gk5e1wmwjnn%uhYQq8kXDLX`uzu_Uo_PK literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/task/impl/BpmTaskServiceImpl.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/task/impl/BpmTaskServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..ba30f07569d092ea42764fa8da2837d1dced621c GIT binary patch literal 30900 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 F>IdqAW|05@ literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/DeptApi.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/DeptApi.class new file mode 100644 index 0000000000000000000000000000000000000000..727469ccec11cd4d8b5cc5b4f5daf51975f60de2 GIT binary patch literal 1941 zcmbVNTT|0O7(EMxK#3Mn@vewB5GCGkMO4Z-GE;QMqVG$y;A)!9B-=XtD<9Pde}Lm} zay)4R4U%D$r|ftCa=!2E{`~#@2Y`E69zzLJ6Fgrn;~Hih%sRL(;A|?ptRgG0G*ka5 zJJJhexarkRxLKX8H%-fn+h$998-e*O*8}DGrX_Xg`PO*tM8#ej{8lB=^4goKz(|%Z z5tzyHwXF`knh65cwAzHs+AY&n*L7kUHQLadp|tH-2?WJq3x7AaxF^xV|F*lx(&df} zw4eS2TRv4>yE^+`V6NC!)EB+W84jy55}3+|Hk94fIX&1$sb+lse5I~K^}5|!SMgek zw!lQ)G-dE!#yXWVWyyZfY`1VeoJmJFRev}yBFzGW+Gj*1hke=z3kMv5zk6XeAn^9XHiWo#OANt3e3o)1`GW zf>DCCWQmNy% zGKqU%;A)0Lmzh1w_6p@yWjw&cE{_W;?nm1U#6Ix_hGY~4TLKqm@(6FV!!C!tm)U^} zbh}@Sr}JhaK}td3D)Z_a#RrH)PYAn7=pH>Prrm1cvLzd}@=k?n)S{xE9DHQSA6v zegs**-tb?^&<|8{xffXNb=`%sVPnl(G^g1WxJE`D%-Acre{6oHS=pKlf`+z=L!0jp z?O!f-?gQ_$6`$A9_CEBN#X(Kr${w_f&=EMlzuzz~Bn3dU(ULVLvTAiopW=e|J2nE}apez0LqBW) literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/DictDataApi.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/DictDataApi.class new file mode 100644 index 0000000000000000000000000000000000000000..e679765094ceec94c5710ab4699ce61db56d6677 GIT binary patch literal 292 zcma)%K}!QM6oe=3+I3eDe}<)CF1>AODHczHhXwCxjLTcnq-2v7|C$GXKz~&6wM8oQ z;%(-^%=>1y|M~_n#Z8VPM=5aID@$)H(s8At>I5eD&#mf}w8}NIj)7cLoh~Vom5<(9 zqa%4&jb8v^p tYind@LQmSr#6tdwaKU-U%erOgk$}K}_X8pm7$ReT#cqHRtNf_0e*j6QRC@pb literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/PermissionApi.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/PermissionApi.class new file mode 100644 index 0000000000000000000000000000000000000000..da0459072525ba41f8fd009a39aaf0247bdad3cc GIT binary patch literal 316 zcmZvYO-chn5QSgGm@#o9F1>(pp}lc8BB&sU1Bycq&>732LT6g~Cz7kV@Bki4EGHY4 z?7mmjE8eTmulElC=h!bWE-)b+wVY;=!;^P>_gqO#*Xs>O!tu0P*p)Roi8D9eIc}2p zt~~iQt~r&2aV>3UQw~hH-|o8Dn4IleQ~9nfFaO`!vO$>CK8J>H#p&t8M-EGgQA1q4 z%Gpmi!fE4|CS<=B^WyxQojW$#XeAAstYH3_2nT(qKF7m+p_N7i!a)B5(2%f$q3T^# L1B{f4p7y>0g0WrE literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/PostApi.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/PostApi.class new file mode 100644 index 0000000000000000000000000000000000000000..0c6e477e06f8c70a4f47c26c78721910214ea353 GIT binary patch literal 242 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 S2-Wy79>rt9+sAMLAO9bx@GV9F literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/impl/DictDataApiImpl.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/impl/DictDataApiImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..1eadf238eca1350ffcec7f028b0775dbcd907728 GIT binary patch literal 5763 zcmcIo33wc38Ga|%Y&O$2-5%IO=#jR$+DS`=!uBvtQdvn;VbefCt&`nJGR^MHc6PQU zcpxIh0}Ix(c;SJFc(fJMGzhInMG^5n@jejmyNZC1?>{rUJLzuHRv+^`ng9C!@BhB< zf4}43hyOkPAb_=aCX6=h4C5uxYH=a@!$=?*hK_*<*Ud0Ouw-N+ki3B{BU=XtxdMf=7O4#;>@2yYH!3EonNx8iLwZm-4LaWITK@Qzx%6YmQ1^X@Q~<2_;AiT8@n z`^0CTpnJcJ4~Vt91pNnP+%4ln5@sZggwv@xT3omH7ztg%s_ul9R_&ZMVyM}XtfQyZ zthc7d;v?D4iS$tk^RCR9j*-^;jI7b0(%Vhbax}-V%&dfU-IhJ5W;3>74i4B_THkBg z!)m{-nOSu}OE{K2qH2W=YJ1MH_8PXHB-@(xhG{q(B~&&x^+~Aeu#%K+Mz>+=J-Kwh zZue?DO+uvGN@%G*%{Ii?pR96*3?6?`K$n4(wO3FSw8>E|)ktSjYF?-Ij1l9Qhfm+5 zrHo{Lo}5}5yLV}OG&QA}gKFFn`b52%oMWWa4l9+?6GG{z5H=*N+B=eQ#VgrI-N-(t zlkME{Jg9EU$T|}01MA(Kp=fo2x45ybT1MEzW$y~X39iqbXb%eE+oU-_T9kLaHyhMz zBe`@0h{eI{1C}H{(w3egr$Mi$vDWkN5XmDj)sXKY-p3v;7d`s?*L05l>&d&F1~ym z3am(2ea@Ch7G*kmFs~6{3F? zPs->v*^enWhH(jPXU)vtE8~6zTkt7L`anG%6oMU>aYDhT@tJyjmeZfB$4LdB$3rr{ zpx}#mSjLwWJR)NHJqo^zM@x`S#hJDRTxo&?o^fcKgsBcO9a^b8hVvQ5O*Y$^btK3}wkw@+ zMwo26 zuA2J7d|v!n!h}U%Ig!b0w1kL+bmE{8f?@E0!toufx*0%isy(}$%nTU=mffW#h9oQs zaxCQ_PG4M%jFS{B5@s?>CJZ}4&pgH%qb72> z+%S_87LarP9<@}=WI@3Y1(T`LvuM( z6cA8iZ5e-2@K^jp!7ZY0xmkSvi6<5G;i?H$mhZ$~k|ew}{|(Luuaik0-7{gIgV+- z7^YuZ)jF?g95cYzSbYVA09CF4OZcN3uoN;H*jr8^R-l<}8yb0Z3ofI6Lb$b_0KRY( z1)I^~3LHZxx+rKH=JCDuJpMoz#dPkA@r37d=OWGs`c7P4pz1DAZ6#F?*DE1~u#KGx zyr4deD{v*5Q+H_{mjr`T&7+vvU9o|>M(*W!ji5a_^u%3k=j|T5aUa_iEsRgUJf}<8&vRPIexB1}_Vb+Pu%G8t>)z*~ z`>|w4Wa&{f9H)ep0i-MlAmtR__Ptye=8K@_88pb)E=>#1e+scD5u+6usXQz4=gMl= zikq5Sj$v5^PGI@g)`t*2f)!(EJbnaCqi7~{TE;+(tsKRwJ22-6)T#&2wxf~@tK(x> zb6nuY6KB4_9a>R`l|ziC0kdTnR+Qmy9k+cKX@EU~JwWxR+ zb7fq+f-ra?Co1rwf)l-%<5T<(Q-{CdU-&x#HpuZB(hgA{1G_Mc6w>}^w?EoL5e@9d MUWndH@iM&pKewjgMgRZ+ literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/impl/PermissionApiImpl.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/impl/PermissionApiImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..9bc3477a22bfd814ffbd4fc1939cad0204119d54 GIT binary patch literal 1057 zcmbVLTW`}a6#m=`p%>T~P_}U$7ts}gUl2m-G^tENnuxe8(DyWTgGUlqj1 z5Bvar6yi7?Z3h(~9_-_D`_B1}?Qh?|d%I!((m{{_({q~eqfZ&xj#$bz09bI)M7}mr*>o_%`QW^z1?T1?5QDj zw&Kb}JkR2R(64=}#<1q9z>oUAmR3JDR?J&jl%Mk$MQUOrW=t$&F5_{;UkV+|G^Lh% zW4TWjD!&;DV|gB^NbC@l@Jb zw|<+{13L$X_R@Q%hoglu#E5&-=hJPXfo2m9PCK~W#2U^sJo>*&VpuIMp_s&r!6DrN z!~MTuonSzX&*T66(6L3|rV4tOv`Ur$$r^nttJg`END?>N&ydDRGLqHKC$JBN`z-A? z7&wO&+HEKWV3l;xnPx#CHu)9F63IsABTDx^qg=v!Qp=b_tQQciY1lCYCEx-s5>$(# Yr|{Nti7e}6DPe=wW^uYa>-`Gw1B|{N(*OVf literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/impl/PostApiImpl.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/impl/PostApiImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..a35c7bbfa95d089e92b5344966a3ed662027cf22 GIT binary patch literal 3594 zcmb7H`FGS*6#g<@(wRx^&`m*Qvp_cpn~D~xZ7DK3ZILNeK*e+>Z9_B3kYoyV$6efa z!3~#x0FGke9L3)}$K#*m@%X(=I!S3Q^@rw7^6vfa{qFW&|NQ6LZvb}V#{`=3Kmz;F z8^^79Fo{0&H*)J>0%<(l2noYUjNnibkK(Y5Y#gIFlE4@qi{o(|P4ICnfdq~x@C2R| zkEg`rgphk$h9>sLg?vs%UWP6qS+rd*t2;ABj&GfroLx|zl09pv?yT$S1=TI@sp;&j zdnlCck+Ax3$?}YXK4!SaxT*J9mhEYtVOy?*JsI1XRNbOuSd$ZuR?tt|&a^tNYnH1{ zXgSYzW>u}Cp!StK`?TTcdBWE2H7vv1C!wmnV@yKzfSspj4H?7Ihf9TV-5J$rO+sVF z&S~bD<{08V_*m^t88p8$!ehd;PYa0(lM*SZMxkh`f=^%3Nb@%hCuTI$$ct}8SlymE zsm*9=$umrKz&1@iC#dxZm#U0BC0dsjkWeE+N`X{FAY-^5O~_(vNGl3HAv`~DuXQ1x z+2TB>r9IhO1~4^iQq9=bWY0cg?24cYlY1j^2=bY?3z}i6K{kD@{)iTRS%kbIkov5r z<)*0~WFnq5CN0e?IZRd8iYbWjHkn#sJg>Ed>1f+7VPh0SMC?ZXLN#XOD{&7a!y*!K zm$2ccOxRN_gsfe1a{2*7lvzp?L{KE#M9w6%FIn8|7Du-|=1mRysYk(`SSw>f!6c>> zti~D{hJurrCT_oNd#>kb#Ub6Bvh!|3EzA@O4K=V*uoV<>N`|Arg_lB!yE7@AR&WNh zGM-WJES{6`yn+`{kg#r1+b&GAj29KWgtIbUR`3d5mGPQ_*YSpeH}RHuoRjgkf_Lz) zg7@&gf)DVaf{$>Pe)@Il2XkcnB)dSt$2ccYK9R6Jw7ar4ld$DxDnY!N@tmbjaWL4X znzJ39kNGb28}bM{lO1KagN33uyG+L(W5zA?fCDtcD3=nM~Nupq87Gus({o7(rOC z4>4$x^L|pND|^ClR%+d^OWi6JT-85SI8_H7$9BY-C7kj!OPmC^92&`vo){h(J#k>< z$nbs^N+k0^ZuB#PF^?Fvfd7)D<yzOF~BPyWd}7YZ|ZJa zSuuK02hom@Tc9|~!!s(r^3Afz<#nSe4Uh*k)65#49@Lx2Q<^u3#szidPbC>YNZ7O} zfJ+gUuy%PlY~bu`=4Jji{>8+#fa~qN=GSt)gKHI6-W>Q`$2XO$#I>gL7f6@*BZl>S z3Sm%n1D{G+02^@^-z2nQ6QzXm&s?jyc6QAnb{^H2I)6e{*Hu)FwXGtha(6rTNxqH-1kIhPgs5B!3n~}stpMolr2p|4en(=c`wt{8BRL1< z`eoE#!7bfa&@hM899CUH;{~)_MpN~#Xg*rSjh5^jTCWR!RS|w2NYcj!dfAGt^m7~k z{o3yHEBo8)GoTWBaX0RvGfBF1FZR$u;cpzVztJk=zAX&L{rnI^PbHgs`TjNk5>)#V b-{C7}dw}n)l-f@>25|rn`M(}P21EY>(+CpM literal 0 HcmV?d00001 diff --git a/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/impl/RoleApiImpl.class b/ruoyi-flowable/target/classes/com/ruoyi/flowable/service/user/impl/RoleApiImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..2c2a9993c70c843103441c1691ee95ed1557a577 GIT binary patch literal 3655 zcmb7H`EwLS6#iy&OtKkFNFWL-mw{Z60YpU;iAh*wb(1I?B8M6$yOU(d?94JVo3P$^ zi}x*f;qebZDF&((f3wQ+PqHk(o}KJW$Qqzh>7MR>?|b*_-oO66@;iV%_$iJxcrcEA z=Ec&Gsb7;NUz!8nq5q!|)M8Ze5-8t^y{$w=2@42RJUW26dp|+4F{@X9-)|XIO@}PeN5k=eUIGAv;UW8dHX)j}&r~x-+IxnuO+* zozcv3%`wC}*sS(u42s_!v17`#=LJRCk^)Uqja=STg*^j#Bgt|!=?$hx0qUPGOM+PVQSkcVM`Q4c=B4m z2-cX9EypAD7?VrHL&A=mvWj)bWJ%ivC!-%Q1b>rAnjn#SL|i3wR4f2il%v}oW2*)= z*Q?-8Y>+XfU>Y+D)?vL2L%|8m5_izHJ=b%z{IKrL*jcx+7G@K2jWw{EU@OSuqzp%a z3$F=co**}cpo1q_z)i{_!y_Dv!B<#lSd~{uxb>1f-?f;Qwcjm9d4{FCA43U zCGb5nA!DgCY$Uc>O7cv(YW)H}%t~ej8g3$&_lhf2_%TK=6uNR7lu&Q7&IXG~(GohM zxfpF4k)rZDK^&jqa|zo5`iQrBJ0tm9a$v9M+F!7^J8mjXqd2tL?Ic7+b$m*^XU)TLt=J7J_L3eVE|c*eHobjM>6O4$EDu84SIx!kB$ zqLd}9t>mHcI4{R9*$|?KzKNpNYqd;9&wCPjqACcuD12G?5bvu(PIoA5P1usd;4x*U zY$u^*W+ZHiA}&V|(pQBTz8|Wlb*i#AbZ0s5{YI=?g`BJUHxj#W!f|Xz+;>7LPqRFF zweiqsDlsuKIyR9wl1z`W^=w<72Kp&sGVdp~3_*MLgi?bbhID@9gC3MvQF3&Xsh!Rt!~(RP~~2z9)wKWaE|L72E`O7|z7Vz+fsNp^}J-JcO^O*GnKG zg`$;^v^;&f6zwJZX-_;=$htc0UjeOzy1eGNI+xcZWWdy!WRmb!D(ItAyvI9|l?B%7 zCkvV=l!{5su2TwiGCi?BnJ(GfKs!W#8{rE|!psc@|u8e{bE>i_N=a^{Og!ElNP+__?paxD;Ym9dMVqUEwCRbNz2lmAyac* z@izK9TDk#V>5e<7Sy?`x+DeSk+dS}g#JDAlW4u+PEerxzk;5Pop#Cw7B?fi{U%lQt@Rh;>x z;JA@1m9xaTrt4Qo7x)pwCXRv_WZTS9DG^``?&3;98@7^4F#p22nsZn80%B)TeWB}T zRCQlQbqvGXub>84K>RI8JJyM3u$CFDRV zWss(w(pzX=2dQGDUd7#R8nm5An>d%Kc9f}l%2X;(ck-OZ*I^ewV%Xgj7nw_9Vaj=G zD**btsDFJ7e;_`*{ZGU%qG18b)r(kl3Ab#&gvJFlEnxK}G@rxTi)g9-4Qq~8@nCIw z0j*bseN_?ry3jxk8>wL{dN`{5?Y7IeucU9kZvmOmk9%-0m1&?#_hBy;6#CX9_77TR u+}}=jJiv_@ddt!5 + + + + + + + + + + + + + + + + + + + + + + + select id, user_id, type, reason, start_time, end_time, day, result, process_instance_id, creator, create_time, updater, update_time, deleted, tenant_id from bpm_oa_leave + + + + + diff --git a/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmFormConvertImpl.java b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmFormConvertImpl.java new file mode 100644 index 00000000..ff68c62c --- /dev/null +++ b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmFormConvertImpl.java @@ -0,0 +1,137 @@ +package com.ruoyi.flowable.convert.definition; + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.flowable.domain.entity.definition.BpmFormDO; +import com.ruoyi.flowable.domain.entity.definition.BpmFormDO.BpmFormDOBuilder; +import com.ruoyi.flowable.domain.vo.form.BpmFormCreateReqVO; +import com.ruoyi.flowable.domain.vo.form.BpmFormRespVO; +import com.ruoyi.flowable.domain.vo.form.BpmFormSimpleRespVO; +import com.ruoyi.flowable.domain.vo.form.BpmFormUpdateReqVO; +import java.util.ArrayList; +import java.util.List; +import javax.annotation.Generated; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:16+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class BpmFormConvertImpl implements BpmFormConvert { + + @Override + public BpmFormDO convert(BpmFormCreateReqVO bean) { + if ( bean == null ) { + return null; + } + + BpmFormDOBuilder bpmFormDO = BpmFormDO.builder(); + + bpmFormDO.name( bean.getName() ); + bpmFormDO.status( bean.getStatus() ); + bpmFormDO.conf( bean.getConf() ); + List list = bean.getFields(); + if ( list != null ) { + bpmFormDO.fields( new ArrayList( list ) ); + } + bpmFormDO.remark( bean.getRemark() ); + + return bpmFormDO.build(); + } + + @Override + public BpmFormDO convert(BpmFormUpdateReqVO bean) { + if ( bean == null ) { + return null; + } + + BpmFormDOBuilder bpmFormDO = BpmFormDO.builder(); + + bpmFormDO.id( bean.getId() ); + bpmFormDO.name( bean.getName() ); + bpmFormDO.status( bean.getStatus() ); + bpmFormDO.conf( bean.getConf() ); + List list = bean.getFields(); + if ( list != null ) { + bpmFormDO.fields( new ArrayList( list ) ); + } + bpmFormDO.remark( bean.getRemark() ); + + return bpmFormDO.build(); + } + + @Override + public BpmFormRespVO convert(BpmFormDO bean) { + if ( bean == null ) { + return null; + } + + BpmFormRespVO bpmFormRespVO = new BpmFormRespVO(); + + bpmFormRespVO.setName( bean.getName() ); + bpmFormRespVO.setStatus( bean.getStatus() ); + bpmFormRespVO.setRemark( bean.getRemark() ); + bpmFormRespVO.setId( bean.getId() ); + bpmFormRespVO.setConf( bean.getConf() ); + List list = bean.getFields(); + if ( list != null ) { + bpmFormRespVO.setFields( new ArrayList( list ) ); + } + bpmFormRespVO.setCreateTime( bean.getCreateTime() ); + + return bpmFormRespVO; + } + + @Override + public List convertList2(List list) { + if ( list == null ) { + return null; + } + + List list1 = new ArrayList( list.size() ); + for ( BpmFormDO bpmFormDO : list ) { + list1.add( bpmFormDOToBpmFormSimpleRespVO( bpmFormDO ) ); + } + + return list1; + } + + @Override + public PageResult convertPage(PageResult page) { + if ( page == null ) { + return null; + } + + PageResult pageResult = new PageResult(); + + pageResult.setList( bpmFormDOListToBpmFormRespVOList( page.getList() ) ); + pageResult.setTotal( page.getTotal() ); + + return pageResult; + } + + protected BpmFormSimpleRespVO bpmFormDOToBpmFormSimpleRespVO(BpmFormDO bpmFormDO) { + if ( bpmFormDO == null ) { + return null; + } + + BpmFormSimpleRespVO bpmFormSimpleRespVO = new BpmFormSimpleRespVO(); + + bpmFormSimpleRespVO.setId( bpmFormDO.getId() ); + bpmFormSimpleRespVO.setName( bpmFormDO.getName() ); + + return bpmFormSimpleRespVO; + } + + protected List bpmFormDOListToBpmFormRespVOList(List list) { + if ( list == null ) { + return null; + } + + List list1 = new ArrayList( list.size() ); + for ( BpmFormDO bpmFormDO : list ) { + list1.add( convert( bpmFormDO ) ); + } + + return list1; + } +} diff --git a/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmModelConvertImpl.java b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmModelConvertImpl.java new file mode 100644 index 00000000..efb4177b --- /dev/null +++ b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmModelConvertImpl.java @@ -0,0 +1,72 @@ +package com.ruoyi.flowable.convert.definition; + +import com.ruoyi.flowable.domain.dto.definition.BpmModelMetaInfoRespDTO; +import com.ruoyi.flowable.domain.dto.definition.BpmProcessDefinitionCreateReqDTO; +import com.ruoyi.flowable.domain.vo.model.BpmModeImportReqVO; +import com.ruoyi.flowable.domain.vo.model.BpmModelBaseVO; +import com.ruoyi.flowable.domain.vo.model.BpmModelCreateReqVO; +import javax.annotation.Generated; +import org.flowable.engine.repository.ProcessDefinition; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:16+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class BpmModelConvertImpl implements BpmModelConvert { + + @Override + public BpmModelCreateReqVO convert(BpmModeImportReqVO bean) { + if ( bean == null ) { + return null; + } + + BpmModelCreateReqVO bpmModelCreateReqVO = new BpmModelCreateReqVO(); + + bpmModelCreateReqVO.setKey( bean.getKey() ); + bpmModelCreateReqVO.setName( bean.getName() ); + bpmModelCreateReqVO.setDescription( bean.getDescription() ); + + return bpmModelCreateReqVO; + } + + @Override + public void copyTo(BpmModelMetaInfoRespDTO from, BpmProcessDefinitionCreateReqDTO to) { + if ( from == null ) { + return; + } + + to.setDescription( from.getDescription() ); + to.setFormType( from.getFormType() ); + to.setFormId( from.getFormId() ); + to.setFormCustomCreatePath( from.getFormCustomCreatePath() ); + to.setFormCustomViewPath( from.getFormCustomViewPath() ); + } + + @Override + public void copyTo(BpmModelMetaInfoRespDTO from, BpmModelBaseVO to) { + if ( from == null ) { + return; + } + + to.setDescription( from.getDescription() ); + to.setFormType( from.getFormType() ); + to.setFormId( from.getFormId() ); + to.setFormCustomCreatePath( from.getFormCustomCreatePath() ); + to.setFormCustomViewPath( from.getFormCustomViewPath() ); + } + + @Override + public com.ruoyi.flowable.domain.vo.model.BpmModelPageItemRespVO.ProcessDefinition convert(ProcessDefinition bean) { + if ( bean == null ) { + return null; + } + + com.ruoyi.flowable.domain.vo.model.BpmModelPageItemRespVO.ProcessDefinition processDefinition = new com.ruoyi.flowable.domain.vo.model.BpmModelPageItemRespVO.ProcessDefinition(); + + processDefinition.setId( bean.getId() ); + processDefinition.setVersion( bean.getVersion() ); + + return processDefinition; + } +} diff --git a/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmProcessDefinitionConvertImpl.java b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmProcessDefinitionConvertImpl.java new file mode 100644 index 00000000..ec9320a2 --- /dev/null +++ b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmProcessDefinitionConvertImpl.java @@ -0,0 +1,107 @@ +package com.ruoyi.flowable.convert.definition; + +import com.ruoyi.flowable.domain.dto.definition.BpmProcessDefinitionCreateReqDTO; +import com.ruoyi.flowable.domain.entity.definition.BpmProcessDefinitionExtDO; +import com.ruoyi.flowable.domain.entity.definition.BpmProcessDefinitionExtDO.BpmProcessDefinitionExtDOBuilder; +import com.ruoyi.flowable.domain.vo.process.BpmProcessDefinitionPageItemRespVO; +import com.ruoyi.flowable.domain.vo.process.BpmProcessDefinitionRespVO; +import java.util.ArrayList; +import java.util.List; +import javax.annotation.Generated; +import org.flowable.engine.repository.ProcessDefinition; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:16+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class BpmProcessDefinitionConvertImpl implements BpmProcessDefinitionConvert { + + @Override + public BpmProcessDefinitionPageItemRespVO convert(ProcessDefinition bean) { + if ( bean == null ) { + return null; + } + + BpmProcessDefinitionPageItemRespVO bpmProcessDefinitionPageItemRespVO = new BpmProcessDefinitionPageItemRespVO(); + + bpmProcessDefinitionPageItemRespVO.setId( bean.getId() ); + bpmProcessDefinitionPageItemRespVO.setVersion( bean.getVersion() ); + bpmProcessDefinitionPageItemRespVO.setName( bean.getName() ); + bpmProcessDefinitionPageItemRespVO.setDescription( bean.getDescription() ); + bpmProcessDefinitionPageItemRespVO.setCategory( bean.getCategory() ); + + return bpmProcessDefinitionPageItemRespVO; + } + + @Override + public BpmProcessDefinitionExtDO convert2(BpmProcessDefinitionCreateReqDTO bean) { + if ( bean == null ) { + return null; + } + + BpmProcessDefinitionExtDOBuilder bpmProcessDefinitionExtDO = BpmProcessDefinitionExtDO.builder(); + + bpmProcessDefinitionExtDO.modelId( bean.getModelId() ); + bpmProcessDefinitionExtDO.description( bean.getDescription() ); + bpmProcessDefinitionExtDO.formType( bean.getFormType() ); + bpmProcessDefinitionExtDO.formId( bean.getFormId() ); + bpmProcessDefinitionExtDO.formConf( bean.getFormConf() ); + List list = bean.getFormFields(); + if ( list != null ) { + bpmProcessDefinitionExtDO.formFields( new ArrayList( list ) ); + } + bpmProcessDefinitionExtDO.formCustomCreatePath( bean.getFormCustomCreatePath() ); + bpmProcessDefinitionExtDO.formCustomViewPath( bean.getFormCustomViewPath() ); + + return bpmProcessDefinitionExtDO.build(); + } + + @Override + public BpmProcessDefinitionRespVO convert3(ProcessDefinition bean) { + if ( bean == null ) { + return null; + } + + BpmProcessDefinitionRespVO bpmProcessDefinitionRespVO = new BpmProcessDefinitionRespVO(); + + bpmProcessDefinitionRespVO.setSuspensionState( convertSuspendedToSuspensionState( bean.isSuspended() ) ); + bpmProcessDefinitionRespVO.setId( bean.getId() ); + bpmProcessDefinitionRespVO.setVersion( bean.getVersion() ); + bpmProcessDefinitionRespVO.setName( bean.getName() ); + bpmProcessDefinitionRespVO.setDescription( bean.getDescription() ); + bpmProcessDefinitionRespVO.setCategory( bean.getCategory() ); + + return bpmProcessDefinitionRespVO; + } + + @Override + public void copyTo(BpmProcessDefinitionExtDO from, BpmProcessDefinitionRespVO to) { + if ( from == null ) { + return; + } + + to.setDescription( from.getDescription() ); + to.setFormType( from.getFormType() ); + to.setFormId( from.getFormId() ); + to.setFormConf( from.getFormConf() ); + if ( to.getFormFields() != null ) { + List list = from.getFormFields(); + if ( list != null ) { + to.getFormFields().clear(); + to.getFormFields().addAll( list ); + } + else { + to.setFormFields( null ); + } + } + else { + List list = from.getFormFields(); + if ( list != null ) { + to.setFormFields( new ArrayList( list ) ); + } + } + to.setFormCustomCreatePath( from.getFormCustomCreatePath() ); + to.setFormCustomViewPath( from.getFormCustomViewPath() ); + } +} diff --git a/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmTaskAssignRuleConvertImpl.java b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmTaskAssignRuleConvertImpl.java new file mode 100644 index 00000000..87eea98a --- /dev/null +++ b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmTaskAssignRuleConvertImpl.java @@ -0,0 +1,112 @@ +package com.ruoyi.flowable.convert.definition; + +import com.ruoyi.flowable.domain.entity.definition.BpmTaskAssignRuleDO; +import com.ruoyi.flowable.domain.entity.definition.BpmTaskAssignRuleDO.BpmTaskAssignRuleDOBuilder; +import com.ruoyi.flowable.domain.vo.rule.BpmTaskAssignRuleCreateReqVO; +import com.ruoyi.flowable.domain.vo.rule.BpmTaskAssignRuleRespVO; +import com.ruoyi.flowable.domain.vo.rule.BpmTaskAssignRuleUpdateReqVO; +import java.util.ArrayList; +import java.util.HashSet; +import java.util.List; +import java.util.Set; +import javax.annotation.Generated; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:16+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class BpmTaskAssignRuleConvertImpl implements BpmTaskAssignRuleConvert { + + @Override + public BpmTaskAssignRuleRespVO convert(BpmTaskAssignRuleDO bean) { + if ( bean == null ) { + return null; + } + + BpmTaskAssignRuleRespVO bpmTaskAssignRuleRespVO = new BpmTaskAssignRuleRespVO(); + + bpmTaskAssignRuleRespVO.setType( bean.getType() ); + Set set = bean.getOptions(); + if ( set != null ) { + bpmTaskAssignRuleRespVO.setOptions( new HashSet( set ) ); + } + bpmTaskAssignRuleRespVO.setId( bean.getId() ); + bpmTaskAssignRuleRespVO.setModelId( bean.getModelId() ); + bpmTaskAssignRuleRespVO.setProcessDefinitionId( bean.getProcessDefinitionId() ); + bpmTaskAssignRuleRespVO.setTaskDefinitionKey( bean.getTaskDefinitionKey() ); + + return bpmTaskAssignRuleRespVO; + } + + @Override + public BpmTaskAssignRuleDO convert(BpmTaskAssignRuleCreateReqVO bean) { + if ( bean == null ) { + return null; + } + + BpmTaskAssignRuleDOBuilder bpmTaskAssignRuleDO = BpmTaskAssignRuleDO.builder(); + + bpmTaskAssignRuleDO.modelId( bean.getModelId() ); + bpmTaskAssignRuleDO.taskDefinitionKey( bean.getTaskDefinitionKey() ); + bpmTaskAssignRuleDO.type( bean.getType() ); + Set set = bean.getOptions(); + if ( set != null ) { + bpmTaskAssignRuleDO.options( new HashSet( set ) ); + } + + return bpmTaskAssignRuleDO.build(); + } + + @Override + public BpmTaskAssignRuleDO convert(BpmTaskAssignRuleUpdateReqVO bean) { + if ( bean == null ) { + return null; + } + + BpmTaskAssignRuleDOBuilder bpmTaskAssignRuleDO = BpmTaskAssignRuleDO.builder(); + + bpmTaskAssignRuleDO.id( bean.getId() ); + bpmTaskAssignRuleDO.type( bean.getType() ); + Set set = bean.getOptions(); + if ( set != null ) { + bpmTaskAssignRuleDO.options( new HashSet( set ) ); + } + + return bpmTaskAssignRuleDO.build(); + } + + @Override + public List convertList2(List list) { + if ( list == null ) { + return null; + } + + List list1 = new ArrayList( list.size() ); + for ( BpmTaskAssignRuleRespVO bpmTaskAssignRuleRespVO : list ) { + list1.add( bpmTaskAssignRuleRespVOToBpmTaskAssignRuleDO( bpmTaskAssignRuleRespVO ) ); + } + + return list1; + } + + protected BpmTaskAssignRuleDO bpmTaskAssignRuleRespVOToBpmTaskAssignRuleDO(BpmTaskAssignRuleRespVO bpmTaskAssignRuleRespVO) { + if ( bpmTaskAssignRuleRespVO == null ) { + return null; + } + + BpmTaskAssignRuleDOBuilder bpmTaskAssignRuleDO = BpmTaskAssignRuleDO.builder(); + + bpmTaskAssignRuleDO.id( bpmTaskAssignRuleRespVO.getId() ); + bpmTaskAssignRuleDO.modelId( bpmTaskAssignRuleRespVO.getModelId() ); + bpmTaskAssignRuleDO.processDefinitionId( bpmTaskAssignRuleRespVO.getProcessDefinitionId() ); + bpmTaskAssignRuleDO.taskDefinitionKey( bpmTaskAssignRuleRespVO.getTaskDefinitionKey() ); + bpmTaskAssignRuleDO.type( bpmTaskAssignRuleRespVO.getType() ); + Set set = bpmTaskAssignRuleRespVO.getOptions(); + if ( set != null ) { + bpmTaskAssignRuleDO.options( new HashSet( set ) ); + } + + return bpmTaskAssignRuleDO.build(); + } +} diff --git a/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmUserGroupConvertImpl.java b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmUserGroupConvertImpl.java new file mode 100644 index 00000000..11db940a --- /dev/null +++ b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/definition/BpmUserGroupConvertImpl.java @@ -0,0 +1,123 @@ +package com.ruoyi.flowable.convert.definition; + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.flowable.domain.entity.definition.BpmUserGroupDO; +import com.ruoyi.flowable.domain.entity.definition.BpmUserGroupDO.BpmUserGroupDOBuilder; +import com.ruoyi.flowable.domain.vo.group.BpmUserGroupCreateReqVO; +import com.ruoyi.flowable.domain.vo.group.BpmUserGroupRespVO; +import com.ruoyi.flowable.domain.vo.group.BpmUserGroupUpdateReqVO; +import java.util.ArrayList; +import java.util.HashSet; +import java.util.List; +import java.util.Set; +import javax.annotation.Generated; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:16+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class BpmUserGroupConvertImpl implements BpmUserGroupConvert { + + @Override + public BpmUserGroupDO convert(BpmUserGroupCreateReqVO bean) { + if ( bean == null ) { + return null; + } + + BpmUserGroupDOBuilder bpmUserGroupDO = BpmUserGroupDO.builder(); + + bpmUserGroupDO.name( bean.getName() ); + bpmUserGroupDO.description( bean.getDescription() ); + bpmUserGroupDO.status( bean.getStatus() ); + Set set = bean.getMemberUserIds(); + if ( set != null ) { + bpmUserGroupDO.memberUserIds( new HashSet( set ) ); + } + + return bpmUserGroupDO.build(); + } + + @Override + public BpmUserGroupDO convert(BpmUserGroupUpdateReqVO bean) { + if ( bean == null ) { + return null; + } + + BpmUserGroupDOBuilder bpmUserGroupDO = BpmUserGroupDO.builder(); + + bpmUserGroupDO.id( bean.getId() ); + bpmUserGroupDO.name( bean.getName() ); + bpmUserGroupDO.description( bean.getDescription() ); + bpmUserGroupDO.status( bean.getStatus() ); + Set set = bean.getMemberUserIds(); + if ( set != null ) { + bpmUserGroupDO.memberUserIds( new HashSet( set ) ); + } + + return bpmUserGroupDO.build(); + } + + @Override + public BpmUserGroupRespVO convert(BpmUserGroupDO bean) { + if ( bean == null ) { + return null; + } + + BpmUserGroupRespVO bpmUserGroupRespVO = new BpmUserGroupRespVO(); + + bpmUserGroupRespVO.setName( bean.getName() ); + bpmUserGroupRespVO.setDescription( bean.getDescription() ); + Set set = bean.getMemberUserIds(); + if ( set != null ) { + bpmUserGroupRespVO.setMemberUserIds( new HashSet( set ) ); + } + bpmUserGroupRespVO.setStatus( bean.getStatus() ); + bpmUserGroupRespVO.setId( bean.getId() ); + bpmUserGroupRespVO.setCreateTime( bean.getCreateTime() ); + + return bpmUserGroupRespVO; + } + + @Override + public List convertList(List list) { + if ( list == null ) { + return null; + } + + List list1 = new ArrayList( list.size() ); + for ( BpmUserGroupDO bpmUserGroupDO : list ) { + list1.add( convert( bpmUserGroupDO ) ); + } + + return list1; + } + + @Override + public PageResult convertPage(PageResult page) { + if ( page == null ) { + return null; + } + + PageResult pageResult = new PageResult(); + + pageResult.setList( convertList( page.getList() ) ); + pageResult.setTotal( page.getTotal() ); + + return pageResult; + } + + @Override + public List convertList2(List list) { + if ( list == null ) { + return null; + } + + List list1 = new ArrayList( list.size() ); + for ( BpmUserGroupDO bpmUserGroupDO : list ) { + list1.add( convert( bpmUserGroupDO ) ); + } + + return list1; + } +} diff --git a/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/message/BpmMessageConvertImpl.java b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/message/BpmMessageConvertImpl.java new file mode 100644 index 00000000..e4cc21a5 --- /dev/null +++ b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/message/BpmMessageConvertImpl.java @@ -0,0 +1,38 @@ +package com.ruoyi.flowable.convert.message; + +import com.ruoyi.flowable.domain.dto.send.SmsSendSingleToUserReqDTO; +import java.util.HashMap; +import java.util.Map; +import javax.annotation.Generated; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:16+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class BpmMessageConvertImpl implements BpmMessageConvert { + + @Override + public SmsSendSingleToUserReqDTO convert(Long userId, String templateCode, Map templateParams) { + if ( userId == null && templateCode == null && templateParams == null ) { + return null; + } + + SmsSendSingleToUserReqDTO smsSendSingleToUserReqDTO = new SmsSendSingleToUserReqDTO(); + + if ( userId != null ) { + smsSendSingleToUserReqDTO.setUserId( userId ); + } + if ( templateCode != null ) { + smsSendSingleToUserReqDTO.setTemplateCode( templateCode ); + } + if ( templateParams != null ) { + Map map = templateParams; + if ( map != null ) { + smsSendSingleToUserReqDTO.setTemplateParams( new HashMap( map ) ); + } + } + + return smsSendSingleToUserReqDTO; + } +} diff --git a/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/oa/BpmOALeaveConvertImpl.java b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/oa/BpmOALeaveConvertImpl.java new file mode 100644 index 00000000..e5b1887d --- /dev/null +++ b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/oa/BpmOALeaveConvertImpl.java @@ -0,0 +1,86 @@ +package com.ruoyi.flowable.convert.oa; + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.flowable.domain.entity.oa.BpmOALeaveDO; +import com.ruoyi.flowable.domain.entity.oa.BpmOALeaveDO.BpmOALeaveDOBuilder; +import com.ruoyi.flowable.domain.vo.oa.BpmOALeaveCreateReqVO; +import com.ruoyi.flowable.domain.vo.oa.BpmOALeaveRespVO; +import java.util.ArrayList; +import java.util.List; +import javax.annotation.Generated; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:16+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class BpmOALeaveConvertImpl implements BpmOALeaveConvert { + + @Override + public BpmOALeaveDO convert(BpmOALeaveCreateReqVO bean) { + if ( bean == null ) { + return null; + } + + BpmOALeaveDOBuilder bpmOALeaveDO = BpmOALeaveDO.builder(); + + if ( bean.getType() != null ) { + bpmOALeaveDO.type( String.valueOf( bean.getType() ) ); + } + bpmOALeaveDO.reason( bean.getReason() ); + bpmOALeaveDO.startTime( bean.getStartTime() ); + bpmOALeaveDO.endTime( bean.getEndTime() ); + + return bpmOALeaveDO.build(); + } + + @Override + public BpmOALeaveRespVO convert(BpmOALeaveDO bean) { + if ( bean == null ) { + return null; + } + + BpmOALeaveRespVO bpmOALeaveRespVO = new BpmOALeaveRespVO(); + + bpmOALeaveRespVO.setStartTime( bean.getStartTime() ); + bpmOALeaveRespVO.setEndTime( bean.getEndTime() ); + if ( bean.getType() != null ) { + bpmOALeaveRespVO.setType( Integer.parseInt( bean.getType() ) ); + } + bpmOALeaveRespVO.setReason( bean.getReason() ); + bpmOALeaveRespVO.setId( bean.getId() ); + bpmOALeaveRespVO.setResult( bean.getResult() ); + bpmOALeaveRespVO.setCreateTime( bean.getCreateTime() ); + bpmOALeaveRespVO.setProcessInstanceId( bean.getProcessInstanceId() ); + + return bpmOALeaveRespVO; + } + + @Override + public List convertList(List list) { + if ( list == null ) { + return null; + } + + List list1 = new ArrayList( list.size() ); + for ( BpmOALeaveDO bpmOALeaveDO : list ) { + list1.add( convert( bpmOALeaveDO ) ); + } + + return list1; + } + + @Override + public PageResult convertPage(PageResult page) { + if ( page == null ) { + return null; + } + + PageResult pageResult = new PageResult(); + + pageResult.setList( convertList( page.getList() ) ); + pageResult.setTotal( page.getTotal() ); + + return pageResult; + } +} diff --git a/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/task/BpmActivityConvertImpl.java b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/task/BpmActivityConvertImpl.java new file mode 100644 index 00000000..b440919d --- /dev/null +++ b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/task/BpmActivityConvertImpl.java @@ -0,0 +1,46 @@ +package com.ruoyi.flowable.convert.task; + +import com.ruoyi.flowable.domain.vo.activity.BpmActivityRespVO; +import java.util.ArrayList; +import java.util.List; +import javax.annotation.Generated; +import org.flowable.engine.history.HistoricActivityInstance; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:16+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class BpmActivityConvertImpl implements BpmActivityConvert { + + @Override + public List convertList(List list) { + if ( list == null ) { + return null; + } + + List list1 = new ArrayList( list.size() ); + for ( HistoricActivityInstance historicActivityInstance : list ) { + list1.add( convert( historicActivityInstance ) ); + } + + return list1; + } + + @Override + public BpmActivityRespVO convert(HistoricActivityInstance bean) { + if ( bean == null ) { + return null; + } + + BpmActivityRespVO bpmActivityRespVO = new BpmActivityRespVO(); + + bpmActivityRespVO.setKey( bean.getActivityId() ); + bpmActivityRespVO.setType( bean.getActivityType() ); + bpmActivityRespVO.setStartTime( bean.getStartTime() ); + bpmActivityRespVO.setEndTime( bean.getEndTime() ); + bpmActivityRespVO.setTaskId( bean.getTaskId() ); + + return bpmActivityRespVO; + } +} diff --git a/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/task/BpmProcessInstanceConvertImpl.java b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/task/BpmProcessInstanceConvertImpl.java new file mode 100644 index 00000000..1433b707 --- /dev/null +++ b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/task/BpmProcessInstanceConvertImpl.java @@ -0,0 +1,188 @@ +package com.ruoyi.flowable.convert.task; + +import com.ruoyi.flowable.domain.dto.user.AdminUserRespDTO; +import com.ruoyi.flowable.domain.entity.definition.BpmProcessDefinitionExtDO; +import com.ruoyi.flowable.domain.entity.task.BpmProcessInstanceExtDO; +import com.ruoyi.flowable.domain.vo.instance.BpmProcessInstancePageItemRespVO; +import com.ruoyi.flowable.domain.vo.instance.BpmProcessInstanceRespVO; +import com.ruoyi.flowable.domain.vo.instance.BpmProcessInstanceRespVO.User; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import javax.annotation.Generated; +import org.flowable.engine.history.HistoricProcessInstance; +import org.flowable.engine.repository.ProcessDefinition; +import org.flowable.task.api.Task; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:16+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class BpmProcessInstanceConvertImpl implements BpmProcessInstanceConvert { + + @Override + public List convertList(List list) { + if ( list == null ) { + return null; + } + + List list1 = new ArrayList( list.size() ); + for ( BpmProcessInstanceExtDO bpmProcessInstanceExtDO : list ) { + list1.add( convert( bpmProcessInstanceExtDO ) ); + } + + return list1; + } + + @Override + public BpmProcessInstancePageItemRespVO convert(BpmProcessInstanceExtDO bean) { + if ( bean == null ) { + return null; + } + + BpmProcessInstancePageItemRespVO bpmProcessInstancePageItemRespVO = new BpmProcessInstancePageItemRespVO(); + + bpmProcessInstancePageItemRespVO.setId( bean.getProcessInstanceId() ); + bpmProcessInstancePageItemRespVO.setName( bean.getName() ); + bpmProcessInstancePageItemRespVO.setProcessDefinitionId( bean.getProcessDefinitionId() ); + bpmProcessInstancePageItemRespVO.setCategory( bean.getCategory() ); + bpmProcessInstancePageItemRespVO.setStatus( bean.getStatus() ); + bpmProcessInstancePageItemRespVO.setResult( bean.getResult() ); + bpmProcessInstancePageItemRespVO.setCreateTime( bean.getCreateTime() ); + bpmProcessInstancePageItemRespVO.setEndTime( bean.getEndTime() ); + + return bpmProcessInstancePageItemRespVO; + } + + @Override + public List convertList2(List tasks) { + if ( tasks == null ) { + return null; + } + + List list = new ArrayList( tasks.size() ); + for ( Task task : tasks ) { + list.add( taskToTask( task ) ); + } + + return list; + } + + @Override + public BpmProcessInstanceRespVO convert2(HistoricProcessInstance bean) { + if ( bean == null ) { + return null; + } + + BpmProcessInstanceRespVO bpmProcessInstanceRespVO = new BpmProcessInstanceRespVO(); + + bpmProcessInstanceRespVO.setId( bean.getId() ); + bpmProcessInstanceRespVO.setName( bean.getName() ); + bpmProcessInstanceRespVO.setEndTime( bean.getEndTime() ); + bpmProcessInstanceRespVO.setBusinessKey( bean.getBusinessKey() ); + + return bpmProcessInstanceRespVO; + } + + @Override + public void copyTo(BpmProcessInstanceExtDO from, BpmProcessInstanceRespVO to) { + if ( from == null ) { + return; + } + + to.setName( from.getName() ); + to.setCategory( from.getCategory() ); + to.setStatus( from.getStatus() ); + to.setResult( from.getResult() ); + to.setCreateTime( from.getCreateTime() ); + to.setEndTime( from.getEndTime() ); + if ( to.getFormVariables() != null ) { + Map map = from.getFormVariables(); + if ( map != null ) { + to.getFormVariables().clear(); + to.getFormVariables().putAll( map ); + } + else { + to.setFormVariables( null ); + } + } + else { + Map map = from.getFormVariables(); + if ( map != null ) { + to.setFormVariables( new HashMap( map ) ); + } + } + } + + @Override + public com.ruoyi.flowable.domain.vo.instance.BpmProcessInstanceRespVO.ProcessDefinition convert2(ProcessDefinition bean) { + if ( bean == null ) { + return null; + } + + com.ruoyi.flowable.domain.vo.instance.BpmProcessInstanceRespVO.ProcessDefinition processDefinition = new com.ruoyi.flowable.domain.vo.instance.BpmProcessInstanceRespVO.ProcessDefinition(); + + processDefinition.setId( bean.getId() ); + + return processDefinition; + } + + @Override + public void copyTo(BpmProcessDefinitionExtDO from, com.ruoyi.flowable.domain.vo.instance.BpmProcessInstanceRespVO.ProcessDefinition to) { + if ( from == null ) { + return; + } + + to.setFormType( from.getFormType() ); + to.setFormId( from.getFormId() ); + to.setFormConf( from.getFormConf() ); + if ( to.getFormFields() != null ) { + List list = from.getFormFields(); + if ( list != null ) { + to.getFormFields().clear(); + to.getFormFields().addAll( list ); + } + else { + to.setFormFields( null ); + } + } + else { + List list = from.getFormFields(); + if ( list != null ) { + to.setFormFields( new ArrayList( list ) ); + } + } + to.setFormCustomCreatePath( from.getFormCustomCreatePath() ); + to.setFormCustomViewPath( from.getFormCustomViewPath() ); + } + + @Override + public User convert2(AdminUserRespDTO bean) { + if ( bean == null ) { + return null; + } + + User user = new User(); + + user.setId( bean.getId() ); + user.setNickname( bean.getNickname() ); + user.setDeptId( bean.getDeptId() ); + + return user; + } + + protected com.ruoyi.flowable.domain.vo.instance.BpmProcessInstancePageItemRespVO.Task taskToTask(Task task) { + if ( task == null ) { + return null; + } + + com.ruoyi.flowable.domain.vo.instance.BpmProcessInstancePageItemRespVO.Task task1 = new com.ruoyi.flowable.domain.vo.instance.BpmProcessInstancePageItemRespVO.Task(); + + task1.setId( task.getId() ); + task1.setName( task.getName() ); + + return task1; + } +} diff --git a/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/task/BpmTaskConvertImpl.java b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/task/BpmTaskConvertImpl.java new file mode 100644 index 00000000..f661e30e --- /dev/null +++ b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/task/BpmTaskConvertImpl.java @@ -0,0 +1,156 @@ +package com.ruoyi.flowable.convert.task; + +import com.ruoyi.flowable.domain.dto.user.AdminUserRespDTO; +import com.ruoyi.flowable.domain.entity.task.BpmTaskExtDO; +import com.ruoyi.flowable.domain.vo.task.BpmTaskDonePageItemRespVO; +import com.ruoyi.flowable.domain.vo.task.BpmTaskRespVO; +import com.ruoyi.flowable.domain.vo.task.BpmTaskRespVO.User; +import com.ruoyi.flowable.domain.vo.task.BpmTaskTodoPageItemRespVO; +import com.ruoyi.flowable.domain.vo.task.BpmTaskTodoPageItemRespVO.ProcessInstance; +import javax.annotation.Generated; +import org.flowable.engine.history.HistoricProcessInstance; +import org.flowable.task.api.Task; +import org.flowable.task.api.history.HistoricTaskInstance; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:16+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class BpmTaskConvertImpl implements BpmTaskConvert { + + @Override + public BpmTaskTodoPageItemRespVO convert1(Task bean) { + if ( bean == null ) { + return null; + } + + BpmTaskTodoPageItemRespVO bpmTaskTodoPageItemRespVO = new BpmTaskTodoPageItemRespVO(); + + bpmTaskTodoPageItemRespVO.setSuspensionState( convertSuspendedToSuspensionState( bean.isSuspended() ) ); + bpmTaskTodoPageItemRespVO.setId( bean.getId() ); + bpmTaskTodoPageItemRespVO.setName( bean.getName() ); + bpmTaskTodoPageItemRespVO.setClaimTime( bean.getClaimTime() ); + bpmTaskTodoPageItemRespVO.setCreateTime( bean.getCreateTime() ); + bpmTaskTodoPageItemRespVO.setTaskDefinitionKey( bean.getTaskDefinitionKey() ); + + return bpmTaskTodoPageItemRespVO; + } + + @Override + public BpmTaskDonePageItemRespVO convert2(HistoricTaskInstance bean) { + if ( bean == null ) { + return null; + } + + BpmTaskDonePageItemRespVO bpmTaskDonePageItemRespVO = new BpmTaskDonePageItemRespVO(); + + bpmTaskDonePageItemRespVO.setId( bean.getId() ); + bpmTaskDonePageItemRespVO.setName( bean.getName() ); + bpmTaskDonePageItemRespVO.setClaimTime( bean.getClaimTime() ); + bpmTaskDonePageItemRespVO.setCreateTime( bean.getCreateTime() ); + bpmTaskDonePageItemRespVO.setTaskDefinitionKey( bean.getTaskDefinitionKey() ); + bpmTaskDonePageItemRespVO.setEndTime( bean.getEndTime() ); + bpmTaskDonePageItemRespVO.setDurationInMillis( bean.getDurationInMillis() ); + + return bpmTaskDonePageItemRespVO; + } + + @Override + public ProcessInstance convert(org.flowable.engine.runtime.ProcessInstance processInstance, AdminUserRespDTO startUser) { + if ( processInstance == null && startUser == null ) { + return null; + } + + ProcessInstance processInstance1 = new ProcessInstance(); + + if ( processInstance != null ) { + processInstance1.setId( processInstance.getId() ); + processInstance1.setName( processInstance.getName() ); + if ( processInstance.getStartUserId() != null ) { + processInstance1.setStartUserId( Long.parseLong( processInstance.getStartUserId() ) ); + } + processInstance1.setProcessDefinitionId( processInstance.getProcessDefinitionId() ); + processInstance1.setBusinessKey( processInstance.getBusinessKey() ); + processInstance1.setProcessDefinitionKey( processInstance.getProcessDefinitionKey() ); + } + if ( startUser != null ) { + processInstance1.setStartUserNickname( startUser.getNickname() ); + } + + return processInstance1; + } + + @Override + public BpmTaskRespVO convert3(HistoricTaskInstance bean) { + if ( bean == null ) { + return null; + } + + BpmTaskRespVO bpmTaskRespVO = new BpmTaskRespVO(); + + bpmTaskRespVO.setDefinitionKey( bean.getTaskDefinitionKey() ); + bpmTaskRespVO.setId( bean.getId() ); + bpmTaskRespVO.setName( bean.getName() ); + bpmTaskRespVO.setClaimTime( bean.getClaimTime() ); + bpmTaskRespVO.setCreateTime( bean.getCreateTime() ); + bpmTaskRespVO.setTaskDefinitionKey( bean.getTaskDefinitionKey() ); + bpmTaskRespVO.setEndTime( bean.getEndTime() ); + bpmTaskRespVO.setDurationInMillis( bean.getDurationInMillis() ); + + return bpmTaskRespVO; + } + + @Override + public User convert3(AdminUserRespDTO bean) { + if ( bean == null ) { + return null; + } + + User user = new User(); + + user.setId( bean.getId() ); + user.setNickname( bean.getNickname() ); + user.setDeptId( bean.getDeptId() ); + + return user; + } + + @Override + public void copyTo(BpmTaskExtDO from, BpmTaskDonePageItemRespVO to) { + if ( from == null ) { + return; + } + + to.setName( from.getName() ); + to.setCreateTime( from.getCreateTime() ); + to.setEndTime( from.getEndTime() ); + to.setResult( from.getResult() ); + to.setReason( from.getReason() ); + } + + @Override + public ProcessInstance convert(HistoricProcessInstance processInstance, AdminUserRespDTO startUser) { + if ( processInstance == null && startUser == null ) { + return null; + } + + ProcessInstance processInstance1 = new ProcessInstance(); + + if ( processInstance != null ) { + processInstance1.setId( processInstance.getId() ); + processInstance1.setName( processInstance.getName() ); + if ( processInstance.getStartUserId() != null ) { + processInstance1.setStartUserId( Long.parseLong( processInstance.getStartUserId() ) ); + } + processInstance1.setProcessDefinitionId( processInstance.getProcessDefinitionId() ); + processInstance1.setBusinessKey( processInstance.getBusinessKey() ); + processInstance1.setProcessDefinitionKey( processInstance.getProcessDefinitionKey() ); + } + if ( startUser != null ) { + processInstance1.setStartUserNickname( startUser.getNickname() ); + } + + return processInstance1; + } +} diff --git a/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/user/DeptConvertImpl.java b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/user/DeptConvertImpl.java new file mode 100644 index 00000000..032ccc3b --- /dev/null +++ b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/user/DeptConvertImpl.java @@ -0,0 +1,47 @@ +package com.ruoyi.flowable.convert.user; + +import com.ruoyi.common.core.domain.entity.SysDept; +import com.ruoyi.flowable.domain.dto.user.DeptRespDTO; +import java.util.ArrayList; +import java.util.List; +import javax.annotation.Generated; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:16+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class DeptConvertImpl implements DeptConvert { + + @Override + public DeptRespDTO convert(SysDept bean) { + if ( bean == null ) { + return null; + } + + DeptRespDTO deptRespDTO = new DeptRespDTO(); + + deptRespDTO.setId( bean.getDeptId() ); + deptRespDTO.setName( bean.getDeptName() ); + deptRespDTO.setParentId( bean.getParentId() ); + if ( bean.getStatus() != null ) { + deptRespDTO.setStatus( Integer.parseInt( bean.getStatus() ) ); + } + + return deptRespDTO; + } + + @Override + public List convertList(List list) { + if ( list == null ) { + return null; + } + + List list1 = new ArrayList( list.size() ); + for ( SysDept sysDept : list ) { + list1.add( convert( sysDept ) ); + } + + return list1; + } +} diff --git a/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/user/UserConvertImpl.java b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/user/UserConvertImpl.java new file mode 100644 index 00000000..23c8e343 --- /dev/null +++ b/ruoyi-flowable/target/generated-sources/annotations/com/ruoyi/flowable/convert/user/UserConvertImpl.java @@ -0,0 +1,64 @@ +package com.ruoyi.flowable.convert.user; + +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.flowable.domain.dto.user.AdminUserRespDTO; +import java.util.ArrayList; +import java.util.HashSet; +import java.util.List; +import java.util.Set; +import javax.annotation.Generated; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:16+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class UserConvertImpl implements UserConvert { + + @Override + public AdminUserRespDTO convert(SysUser bean) { + if ( bean == null ) { + return null; + } + + AdminUserRespDTO adminUserRespDTO = new AdminUserRespDTO(); + + adminUserRespDTO.setId( bean.getUserId() ); + adminUserRespDTO.setNickname( bean.getNickName() ); + adminUserRespDTO.setMobile( bean.getPhonenumber() ); + if ( bean.getStatus() != null ) { + adminUserRespDTO.setStatus( Integer.parseInt( bean.getStatus() ) ); + } + adminUserRespDTO.setDeptId( bean.getDeptId() ); + adminUserRespDTO.setPostIds( longArrayToLongSet( bean.getPostIds() ) ); + + return adminUserRespDTO; + } + + @Override + public List convertList(List users) { + if ( users == null ) { + return null; + } + + List list = new ArrayList( users.size() ); + for ( SysUser sysUser : users ) { + list.add( convert( sysUser ) ); + } + + return list; + } + + protected Set longArrayToLongSet(Long[] longArray) { + if ( longArray == null ) { + return null; + } + + Set set = new HashSet( Math.max( (int) ( longArray.length / .75f ) + 1, 16 ) ); + for ( Long long1 : longArray ) { + set.add( long1 ); + } + + return set; + } +} diff --git a/ruoyi-framework/pom.xml b/ruoyi-framework/pom.xml new file mode 100644 index 00000000..a0c7ed4d --- /dev/null +++ b/ruoyi-framework/pom.xml @@ -0,0 +1,82 @@ + + + + ruoyi + com.ruoyi + 3.8.4 + + 4.0.0 + + ruoyi-framework + + + framework框架核心 + + + + + + + org.springframework.boot + spring-boot-starter-web + + + + + org.springframework.boot + spring-boot-starter-aop + + + + + com.alibaba + druid-spring-boot-starter + + + + + com.github.penggle + kaptcha + + + javax.servlet-api + javax.servlet + + + + + + + com.github.oshi + oshi-core + + + + + com.ruoyi + ruoyi-system + + + + + org.springframework.boot + spring-boot-starter-websocket + + + + + org.projectlombok + lombok + true + + + + + + + + + + diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/aspectj/DataScopeAspect.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/aspectj/DataScopeAspect.java new file mode 100644 index 00000000..760dd130 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/aspectj/DataScopeAspect.java @@ -0,0 +1,167 @@ +package com.ruoyi.framework.aspectj; + +import java.util.ArrayList; +import java.util.List; + +import com.ruoyi.common.core.text.Convert; +import com.ruoyi.framework.security.context.PermissionContextHolder; +import org.aspectj.lang.JoinPoint; +import org.aspectj.lang.annotation.Aspect; +import org.aspectj.lang.annotation.Before; +import org.springframework.stereotype.Component; +import com.ruoyi.common.annotation.DataScope; +import com.ruoyi.common.core.domain.BaseEntity; +import com.ruoyi.common.core.domain.entity.SysRole; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.SecurityUtils; + +/** + * 数据过滤处理 + * + * @author ruoyi + */ +@Aspect +@Component +public class DataScopeAspect +{ + /** + * 全部数据权限 + */ + public static final String DATA_SCOPE_ALL = "1"; + + /** + * 自定数据权限 + */ + public static final String DATA_SCOPE_CUSTOM = "2"; + + /** + * 部门数据权限 + */ + public static final String DATA_SCOPE_DEPT = "3"; + + /** + * 部门及以下数据权限 + */ + public static final String DATA_SCOPE_DEPT_AND_CHILD = "4"; + + /** + * 仅本人数据权限 + */ + public static final String DATA_SCOPE_SELF = "5"; + + /** + * 数据权限过滤关键字 + */ + public static final String DATA_SCOPE = "dataScope"; + + @Before("@annotation(controllerDataScope)") + public void doBefore(JoinPoint point, DataScope controllerDataScope) throws Throwable + { + clearDataScope(point); + handleDataScope(point, controllerDataScope); + } + + protected void handleDataScope(final JoinPoint joinPoint, DataScope controllerDataScope) + { + // 获取当前的用户 + LoginUser loginUser = SecurityUtils.getLoginUser(); + if (StringUtils.isNotNull(loginUser)) + { + String permission = StringUtils.defaultIfEmpty(controllerDataScope.permission(), PermissionContextHolder.getContext()); + SysUser currentUser = loginUser.getUser(); + // 如果是超级管理员,则不过滤数据 + if (StringUtils.isNotNull(currentUser) && !currentUser.isAdmin()) + { + dataScopeFilter(joinPoint, currentUser, controllerDataScope.deptAlias(), + controllerDataScope.userAlias(), permission); + } + } + } + + /** + * 数据范围过滤 + * + * @param joinPoint 切点 + * @param user 用户 + * @param deptAlias 部门别名 + * @param userAlias 用户别名 + */ + public static void dataScopeFilter(JoinPoint joinPoint, SysUser user, String deptAlias, String userAlias, String permission) + { + StringBuilder sqlString = new StringBuilder(); + List conditions = new ArrayList(); + + for (SysRole role : user.getRoles()) + { + String dataScope = role.getDataScope(); + if (!DATA_SCOPE_CUSTOM.equals(dataScope) && conditions.contains(dataScope)) + { + continue; + } + if (StringUtils.isNotEmpty(permission) && StringUtils.isNotEmpty(role.getPermissions()) + && !StringUtils.containsAny(role.getPermissions(), Convert.toStrArray(permission))) + { + continue; + } + if (DATA_SCOPE_ALL.equals(dataScope)) + { + sqlString = new StringBuilder(); + break; + } + else if (DATA_SCOPE_CUSTOM.equals(dataScope)) + { + sqlString.append(StringUtils.format( + " OR {}.dept_id IN ( SELECT dept_id FROM sys_role_dept WHERE role_id = {} ) ", deptAlias, + role.getRoleId())); + } + else if (DATA_SCOPE_DEPT.equals(dataScope)) + { + sqlString.append(StringUtils.format(" OR {}.dept_id = {} ", deptAlias, user.getDeptId())); + } + else if (DATA_SCOPE_DEPT_AND_CHILD.equals(dataScope)) + { + sqlString.append(StringUtils.format( + " OR {}.dept_id IN ( SELECT dept_id FROM sys_dept WHERE dept_id = {} or find_in_set( {} , ancestors ) )", + deptAlias, user.getDeptId(), user.getDeptId())); + } + else if (DATA_SCOPE_SELF.equals(dataScope)) + { + if (StringUtils.isNotBlank(userAlias)) + { + sqlString.append(StringUtils.format(" OR {}.user_id = {} ", userAlias, user.getUserId())); + } + else + { + // 数据权限为仅本人且没有userAlias别名不查询任何数据 + sqlString.append(StringUtils.format(" OR {}.dept_id = 0 ", deptAlias)); + } + } + conditions.add(dataScope); + } + + if (StringUtils.isNotBlank(sqlString.toString())) + { + Object params = joinPoint.getArgs()[0]; + if (StringUtils.isNotNull(params) && params instanceof BaseEntity) + { + BaseEntity baseEntity = (BaseEntity) params; + baseEntity.getParams().put(DATA_SCOPE, " AND (" + sqlString.substring(4) + ")"); + } + } + } + + /** + * 拼接权限sql前先清空params.dataScope参数防止注入 + */ + private void clearDataScope(final JoinPoint joinPoint) + { + Object params = joinPoint.getArgs()[0]; + if (StringUtils.isNotNull(params) && params instanceof BaseEntity) + { + BaseEntity baseEntity = (BaseEntity) params; + baseEntity.getParams().put(DATA_SCOPE, ""); + } + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/aspectj/DataSourceAspect.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/aspectj/DataSourceAspect.java new file mode 100644 index 00000000..8c2c9f43 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/aspectj/DataSourceAspect.java @@ -0,0 +1,72 @@ +package com.ruoyi.framework.aspectj; + +import java.util.Objects; +import org.aspectj.lang.ProceedingJoinPoint; +import org.aspectj.lang.annotation.Around; +import org.aspectj.lang.annotation.Aspect; +import org.aspectj.lang.annotation.Pointcut; +import org.aspectj.lang.reflect.MethodSignature; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.core.annotation.AnnotationUtils; +import org.springframework.core.annotation.Order; +import org.springframework.stereotype.Component; +import com.ruoyi.common.annotation.DataSource; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.framework.datasource.DynamicDataSourceContextHolder; + +/** + * 多数据源处理 + * + * @author ruoyi + */ +@Aspect +@Order(1) +@Component +public class DataSourceAspect +{ + protected Logger logger = LoggerFactory.getLogger(getClass()); + + @Pointcut("@annotation(com.ruoyi.common.annotation.DataSource)" + + "|| @within(com.ruoyi.common.annotation.DataSource)") + public void dsPointCut() + { + + } + + @Around("dsPointCut()") + public Object around(ProceedingJoinPoint point) throws Throwable + { + DataSource dataSource = getDataSource(point); + + if (StringUtils.isNotNull(dataSource)) + { + DynamicDataSourceContextHolder.setDataSourceType(dataSource.value().name()); + } + + try + { + return point.proceed(); + } + finally + { + // 销毁数据源 在执行方法之后 + DynamicDataSourceContextHolder.clearDataSourceType(); + } + } + + /** + * 获取需要切换的数据源 + */ + public DataSource getDataSource(ProceedingJoinPoint point) + { + MethodSignature signature = (MethodSignature) point.getSignature(); + DataSource dataSource = AnnotationUtils.findAnnotation(signature.getMethod(), DataSource.class); + if (Objects.nonNull(dataSource)) + { + return dataSource; + } + + return AnnotationUtils.findAnnotation(signature.getDeclaringType(), DataSource.class); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/aspectj/LogAspect.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/aspectj/LogAspect.java new file mode 100644 index 00000000..fef51fd3 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/aspectj/LogAspect.java @@ -0,0 +1,200 @@ +package com.ruoyi.framework.aspectj; + +import com.alibaba.fastjson2.JSON; +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.common.enums.BusinessStatus; +import com.ruoyi.common.enums.HttpMethod; +import com.ruoyi.common.filter.PropertyPreExcludeFilter; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.common.utils.ServletUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.ip.IpUtils; +import com.ruoyi.framework.manager.AsyncManager; +import com.ruoyi.framework.manager.factory.AsyncFactory; +import com.ruoyi.system.domain.SysOperLog; +import org.aspectj.lang.JoinPoint; +import org.aspectj.lang.annotation.AfterReturning; +import org.aspectj.lang.annotation.AfterThrowing; +import org.aspectj.lang.annotation.Aspect; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.stereotype.Component; +import org.springframework.validation.BindingResult; +import org.springframework.web.multipart.MultipartFile; +import org.springframework.web.servlet.HandlerMapping; + +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import java.util.Collection; +import java.util.Map; + +/** + * 操作日志记录处理 + * + * @author ruoyi + */ +@Aspect +@Component +public class LogAspect { + private static final Logger log = LoggerFactory.getLogger(LogAspect.class); + + /** + * 排除敏感属性字段 + */ + public static final String[] EXCLUDE_PROPERTIES = {"password", "oldPassword", "newPassword", "confirmPassword"}; + + /** + * 处理完请求后执行 + * + * @param joinPoint 切点 + */ + @AfterReturning(pointcut = "@annotation(controllerLog)", returning = "jsonResult") + public void doAfterReturning(JoinPoint joinPoint, Log controllerLog, Object jsonResult) { + handleLog(joinPoint, controllerLog, null, jsonResult); + } + + /** + * 拦截异常操作 + * + * @param joinPoint 切点 + * @param e 异常 + */ + @AfterThrowing(value = "@annotation(controllerLog)", throwing = "e") + public void doAfterThrowing(JoinPoint joinPoint, Log controllerLog, Exception e) { + handleLog(joinPoint, controllerLog, e, null); + } + + protected void handleLog(final JoinPoint joinPoint, Log controllerLog, final Exception e, Object jsonResult) { + try { + // 获取当前的用户 + LoginUser loginUser = SecurityUtils.getLoginUser(); + + // *========数据库日志=========*// + SysOperLog operLog = new SysOperLog(); + operLog.setStatus(BusinessStatus.SUCCESS.ordinal()); + // 请求的地址 + String ip = IpUtils.getIpAddr(ServletUtils.getRequest()); + operLog.setOperIp(ip); + operLog.setOperUrl(StringUtils.substring(ServletUtils.getRequest().getRequestURI(), 0, 255)); + if (loginUser != null) { + operLog.setOperName(loginUser.getUsername()); + } + + if (e != null) { + operLog.setStatus(BusinessStatus.FAIL.ordinal()); + operLog.setErrorMsg(StringUtils.substring(e.getMessage(), 0, 2000)); + } + // 设置方法名称 + String className = joinPoint.getTarget().getClass().getName(); + String methodName = joinPoint.getSignature().getName(); + operLog.setMethod(className + "." + methodName + "()"); + // 设置请求方式 + operLog.setRequestMethod(ServletUtils.getRequest().getMethod()); + // 处理设置注解上的参数 + getControllerMethodDescription(joinPoint, controllerLog, operLog, jsonResult); + // 保存数据库 + AsyncManager.me().execute(AsyncFactory.recordOper(operLog)); + } catch (Exception exp) { + // 记录本地异常日志 + log.error("==前置通知异常=="); + log.error("异常信息:{}", exp.getMessage()); + exp.printStackTrace(); + } + } + + /** + * 获取注解中对方法的描述信息 用于Controller层注解 + * + * @param log 日志 + * @param operLog 操作日志 + * @throws Exception + */ + public void getControllerMethodDescription(JoinPoint joinPoint, Log log, SysOperLog operLog, Object jsonResult) throws Exception { + // 设置action动作 + operLog.setBusinessType(log.businessType().ordinal()); + // 设置标题 + operLog.setTitle(log.title()); + // 设置操作人类别 + operLog.setOperatorType(log.operatorType().ordinal()); + // 是否需要保存request,参数和值 + if (log.isSaveRequestData()) { + // 获取参数的信息,传入到数据库中。 + setRequestValue(joinPoint, operLog); + } + // 是否需要保存response,参数和值 + if (log.isSaveResponseData() && StringUtils.isNotNull(jsonResult)) { + operLog.setJsonResult(StringUtils.substring(JSON.toJSONString(jsonResult), 0, 2000)); + } + } + + /** + * 获取请求的参数,放到log中 + * + * @param operLog 操作日志 + * @throws Exception 异常 + */ + private void setRequestValue(JoinPoint joinPoint, SysOperLog operLog) throws Exception { + String requestMethod = operLog.getRequestMethod(); + if (HttpMethod.PUT.name().equals(requestMethod) || HttpMethod.POST.name().equals(requestMethod)) { + String params = argsArrayToString(joinPoint.getArgs()); + operLog.setOperParam(StringUtils.substring(params, 0, 2000)); + } else { + Map paramsMap = (Map) ServletUtils.getRequest().getAttribute(HandlerMapping.URI_TEMPLATE_VARIABLES_ATTRIBUTE); + operLog.setOperParam(StringUtils.substring(paramsMap.toString(), 0, 2000)); + } + } + + /** + * 参数拼装 + */ + private String argsArrayToString(Object[] paramsArray) { + String params = ""; + if (paramsArray != null && paramsArray.length > 0) { + for (Object o : paramsArray) { + if (StringUtils.isNotNull(o) && !isFilterObject(o)) { + try { + String jsonObj = JSON.toJSONString(o, excludePropertyPreFilter()); + params += jsonObj.toString() + " "; + } catch (Exception e) { + } + } + } + } + return params.trim(); + } + + /** + * 忽略敏感属性 + */ + public PropertyPreExcludeFilter excludePropertyPreFilter() { + return new PropertyPreExcludeFilter().addExcludes(EXCLUDE_PROPERTIES); + } + + /** + * 判断是否需要过滤的对象。 + * + * @param o 对象信息。 + * @return 如果是需要过滤的对象,则返回true;否则返回false。 + */ + @SuppressWarnings("rawtypes") + public boolean isFilterObject(final Object o) { + Class clazz = o.getClass(); + if (clazz.isArray()) { + return clazz.getComponentType().isAssignableFrom(MultipartFile.class); + } else if (Collection.class.isAssignableFrom(clazz)) { + Collection collection = (Collection) o; + for (Object value : collection) { + return value instanceof MultipartFile; + } + } else if (Map.class.isAssignableFrom(clazz)) { + Map map = (Map) o; + for (Object value : map.entrySet()) { + Map.Entry entry = (Map.Entry) value; + return entry.getValue() instanceof MultipartFile; + } + } + return o instanceof MultipartFile || o instanceof HttpServletRequest || o instanceof HttpServletResponse + || o instanceof BindingResult; + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/aspectj/RateLimiterAspect.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/aspectj/RateLimiterAspect.java new file mode 100644 index 00000000..92ff9c86 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/aspectj/RateLimiterAspect.java @@ -0,0 +1,80 @@ +//package com.ruoyi.framework.aspectj; +// +//import com.ruoyi.common.annotation.RateLimiter; +//import com.ruoyi.common.enums.LimitType; +//import com.ruoyi.common.exception.ServiceException; +//import com.ruoyi.common.utils.ServletUtils; +//import com.ruoyi.common.utils.StringUtils; +//import com.ruoyi.common.utils.ip.IpUtils; +//import org.aspectj.lang.JoinPoint; +//import org.aspectj.lang.annotation.Aspect; +//import org.aspectj.lang.annotation.Before; +//import org.aspectj.lang.reflect.MethodSignature; +//import org.slf4j.Logger; +//import org.slf4j.LoggerFactory; +//import org.springframework.beans.factory.annotation.Autowired; +//import org.springframework.data.redis.core.RedisTemplate; +//import org.springframework.data.redis.core.script.RedisScript; +//import org.springframework.stereotype.Component; +// +//import java.lang.reflect.Method; +//import java.util.Collections; +//import java.util.List; +// +///** +// * 限流处理 +// * +// * @author ruoyi +// */ +//@Aspect +//@Component +//public class RateLimiterAspect { +// private static final Logger log = LoggerFactory.getLogger(RateLimiterAspect.class); +// +// private RedisTemplate redisTemplate; +// +// private RedisScript limitScript; +// +// @Autowired +// public void setRedisTemplate1(RedisTemplate redisTemplate) { +// this.redisTemplate = redisTemplate; +// } +// +// @Autowired +// public void setLimitScript(RedisScript limitScript) { +// this.limitScript = limitScript; +// } +// +// @Before("@annotation(rateLimiter)") +// public void doBefore(JoinPoint point, RateLimiter rateLimiter) throws Throwable { +// String key = rateLimiter.key(); +// int time = rateLimiter.time(); +// int count = rateLimiter.count(); +// +// String combineKey = getCombineKey(rateLimiter, point); +// List keys = Collections.singletonList(combineKey); +// try { +// Long number = redisTemplate.execute(limitScript, keys, count, time); +// if (StringUtils.isNull(number) || number.intValue() > count) { +// throw new ServiceException("访问过于频繁,请稍候再试"); +// } +// log.info("限制请求'{}',当前请求'{}',缓存key'{}'", count, number.intValue(), combineKey); +// } catch (ServiceException e) { +// throw e; +// } catch (Exception e) { +// throw new RuntimeException("服务器限流异常,请稍候再试"); +// } +// } +// +// public String getCombineKey(RateLimiter rateLimiter, JoinPoint point) { +// StringBuffer stringBuffer = new StringBuffer(rateLimiter.key()); +// if (rateLimiter.limitType() == LimitType.IP) { +// stringBuffer.append(IpUtils.getIpAddr(ServletUtils.getRequest())).append("-"); +// } +// MethodSignature signature = (MethodSignature) point.getSignature(); +// Method method = signature.getMethod(); +// Class targetClass = method.getDeclaringClass(); +// stringBuffer.append(targetClass.getName()).append("-").append(method.getName()); +// return stringBuffer.toString(); +// } +//} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/config/ApplicationConfig.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/ApplicationConfig.java new file mode 100644 index 00000000..1d4dc1f7 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/ApplicationConfig.java @@ -0,0 +1,30 @@ +package com.ruoyi.framework.config; + +import java.util.TimeZone; +import org.mybatis.spring.annotation.MapperScan; +import org.springframework.boot.autoconfigure.jackson.Jackson2ObjectMapperBuilderCustomizer; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.context.annotation.EnableAspectJAutoProxy; + +/** + * 程序注解配置 + * + * @author ruoyi + */ +@Configuration +// 表示通过aop框架暴露该代理对象,AopContext能够访问 +@EnableAspectJAutoProxy(exposeProxy = true) +// 指定要扫描的Mapper类的包的路径 +@MapperScan("com.ruoyi.**.mapper") +public class ApplicationConfig +{ + /** + * 时区配置 + */ + @Bean + public Jackson2ObjectMapperBuilderCustomizer jacksonObjectMapperCustomization() + { + return jacksonObjectMapperBuilder -> jacksonObjectMapperBuilder.timeZone(TimeZone.getDefault()); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/config/CaptchaConfig.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/CaptchaConfig.java new file mode 100644 index 00000000..43e78aeb --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/CaptchaConfig.java @@ -0,0 +1,83 @@ +package com.ruoyi.framework.config; + +import java.util.Properties; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import com.google.code.kaptcha.impl.DefaultKaptcha; +import com.google.code.kaptcha.util.Config; +import static com.google.code.kaptcha.Constants.*; + +/** + * 验证码配置 + * + * @author ruoyi + */ +@Configuration +public class CaptchaConfig +{ + @Bean(name = "captchaProducer") + public DefaultKaptcha getKaptchaBean() + { + DefaultKaptcha defaultKaptcha = new DefaultKaptcha(); + Properties properties = new Properties(); + // 是否有边框 默认为true 我们可以自己设置yes,no + properties.setProperty(KAPTCHA_BORDER, "yes"); + // 验证码文本字符颜色 默认为Color.BLACK + properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_COLOR, "black"); + // 验证码图片宽度 默认为200 + properties.setProperty(KAPTCHA_IMAGE_WIDTH, "160"); + // 验证码图片高度 默认为50 + properties.setProperty(KAPTCHA_IMAGE_HEIGHT, "60"); + // 验证码文本字符大小 默认为40 + properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_SIZE, "38"); + // KAPTCHA_SESSION_KEY + properties.setProperty(KAPTCHA_SESSION_CONFIG_KEY, "kaptchaCode"); + // 验证码文本字符长度 默认为5 + properties.setProperty(KAPTCHA_TEXTPRODUCER_CHAR_LENGTH, "4"); + // 验证码文本字体样式 默认为new Font("Arial", 1, fontSize), new Font("Courier", 1, fontSize) + properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_NAMES, "Arial,Courier"); + // 图片样式 水纹com.google.code.kaptcha.impl.WaterRipple 鱼眼com.google.code.kaptcha.impl.FishEyeGimpy 阴影com.google.code.kaptcha.impl.ShadowGimpy + properties.setProperty(KAPTCHA_OBSCURIFICATOR_IMPL, "com.google.code.kaptcha.impl.ShadowGimpy"); + Config config = new Config(properties); + defaultKaptcha.setConfig(config); + return defaultKaptcha; + } + + @Bean(name = "captchaProducerMath") + public DefaultKaptcha getKaptchaBeanMath() + { + DefaultKaptcha defaultKaptcha = new DefaultKaptcha(); + Properties properties = new Properties(); + // 是否有边框 默认为true 我们可以自己设置yes,no + properties.setProperty(KAPTCHA_BORDER, "yes"); + // 边框颜色 默认为Color.BLACK + properties.setProperty(KAPTCHA_BORDER_COLOR, "105,179,90"); + // 验证码文本字符颜色 默认为Color.BLACK + properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_COLOR, "blue"); + // 验证码图片宽度 默认为200 + properties.setProperty(KAPTCHA_IMAGE_WIDTH, "160"); + // 验证码图片高度 默认为50 + properties.setProperty(KAPTCHA_IMAGE_HEIGHT, "60"); + // 验证码文本字符大小 默认为40 + properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_SIZE, "35"); + // KAPTCHA_SESSION_KEY + properties.setProperty(KAPTCHA_SESSION_CONFIG_KEY, "kaptchaCodeMath"); + // 验证码文本生成器 + properties.setProperty(KAPTCHA_TEXTPRODUCER_IMPL, "com.ruoyi.framework.config.KaptchaTextCreator"); + // 验证码文本字符间距 默认为2 + properties.setProperty(KAPTCHA_TEXTPRODUCER_CHAR_SPACE, "3"); + // 验证码文本字符长度 默认为5 + properties.setProperty(KAPTCHA_TEXTPRODUCER_CHAR_LENGTH, "6"); + // 验证码文本字体样式 默认为new Font("Arial", 1, fontSize), new Font("Courier", 1, fontSize) + properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_NAMES, "Arial,Courier"); + // 验证码噪点颜色 默认为Color.BLACK + properties.setProperty(KAPTCHA_NOISE_COLOR, "white"); + // 干扰实现类 + properties.setProperty(KAPTCHA_NOISE_IMPL, "com.google.code.kaptcha.impl.NoNoise"); + // 图片样式 水纹com.google.code.kaptcha.impl.WaterRipple 鱼眼com.google.code.kaptcha.impl.FishEyeGimpy 阴影com.google.code.kaptcha.impl.ShadowGimpy + properties.setProperty(KAPTCHA_OBSCURIFICATOR_IMPL, "com.google.code.kaptcha.impl.ShadowGimpy"); + Config config = new Config(properties); + defaultKaptcha.setConfig(config); + return defaultKaptcha; + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/config/DruidConfig.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/DruidConfig.java new file mode 100644 index 00000000..f6abac13 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/DruidConfig.java @@ -0,0 +1,126 @@ +package com.ruoyi.framework.config; + +import java.io.IOException; +import java.util.HashMap; +import java.util.Map; +import javax.servlet.Filter; +import javax.servlet.FilterChain; +import javax.servlet.ServletException; +import javax.servlet.ServletRequest; +import javax.servlet.ServletResponse; +import javax.sql.DataSource; +import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; +import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.boot.web.servlet.FilterRegistrationBean; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.context.annotation.Primary; +import com.alibaba.druid.pool.DruidDataSource; +import com.alibaba.druid.spring.boot.autoconfigure.DruidDataSourceBuilder; +import com.alibaba.druid.spring.boot.autoconfigure.properties.DruidStatProperties; +import com.alibaba.druid.util.Utils; +import com.ruoyi.common.enums.DataSourceType; +import com.ruoyi.common.utils.spring.SpringUtils; +import com.ruoyi.framework.config.properties.DruidProperties; +import com.ruoyi.framework.datasource.DynamicDataSource; + +/** + * druid 配置多数据源 + * + * @author ruoyi + */ +@Configuration +public class DruidConfig +{ + @Bean + @ConfigurationProperties("spring.datasource.druid.master") + public DataSource masterDataSource(DruidProperties druidProperties) + { + DruidDataSource dataSource = DruidDataSourceBuilder.create().build(); + return druidProperties.dataSource(dataSource); + } + + @Bean + @ConfigurationProperties("spring.datasource.druid.slave") + @ConditionalOnProperty(prefix = "spring.datasource.druid.slave", name = "enabled", havingValue = "true") + public DataSource slaveDataSource(DruidProperties druidProperties) + { + DruidDataSource dataSource = DruidDataSourceBuilder.create().build(); + return druidProperties.dataSource(dataSource); + } + + @Bean(name = "dynamicDataSource") + @Primary + public DynamicDataSource dataSource(DataSource masterDataSource) + { + Map targetDataSources = new HashMap<>(); + targetDataSources.put(DataSourceType.MASTER.name(), masterDataSource); + setDataSource(targetDataSources, DataSourceType.SLAVE.name(), "slaveDataSource"); + return new DynamicDataSource(masterDataSource, targetDataSources); + } + + /** + * 设置数据源 + * + * @param targetDataSources 备选数据源集合 + * @param sourceName 数据源名称 + * @param beanName bean名称 + */ + public void setDataSource(Map targetDataSources, String sourceName, String beanName) + { + try + { + DataSource dataSource = SpringUtils.getBean(beanName); + targetDataSources.put(sourceName, dataSource); + } + catch (Exception e) + { + } + } + + /** + * 去除监控页面底部的广告 + */ + @SuppressWarnings({ "rawtypes", "unchecked" }) + @Bean + @ConditionalOnProperty(name = "spring.datasource.druid.statViewServlet.enabled", havingValue = "true") + public FilterRegistrationBean removeDruidFilterRegistrationBean(DruidStatProperties properties) + { + // 获取web监控页面的参数 + DruidStatProperties.StatViewServlet config = properties.getStatViewServlet(); + // 提取common.js的配置路径 + String pattern = config.getUrlPattern() != null ? config.getUrlPattern() : "/druid/*"; + String commonJsPattern = pattern.replaceAll("\\*", "js/common.js"); + final String filePath = "support/http/resources/js/common.js"; + // 创建filter进行过滤 + Filter filter = new Filter() + { + @Override + public void init(javax.servlet.FilterConfig filterConfig) throws ServletException + { + } + @Override + public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) + throws IOException, ServletException + { + chain.doFilter(request, response); + // 重置缓冲区,响应头不会被重置 + response.resetBuffer(); + // 获取common.js + String text = Utils.readFromResource(filePath); + // 正则替换banner, 除去底部的广告信息 + text = text.replaceAll("
    ", ""); + text = text.replaceAll("powered.*?shrek.wang", ""); + response.getWriter().write(text); + } + @Override + public void destroy() + { + } + }; + FilterRegistrationBean registrationBean = new FilterRegistrationBean(); + registrationBean.setFilter(filter); + registrationBean.addUrlPatterns(commonJsPattern); + return registrationBean; + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/config/FastJson2JsonRedisSerializer.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/FastJson2JsonRedisSerializer.java new file mode 100644 index 00000000..20bd8706 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/FastJson2JsonRedisSerializer.java @@ -0,0 +1,48 @@ +//package com.ruoyi.framework.config; +// +//import java.nio.charset.Charset; +//import org.springframework.data.redis.serializer.RedisSerializer; +//import org.springframework.data.redis.serializer.SerializationException; +//import com.alibaba.fastjson2.JSON; +//import com.alibaba.fastjson2.JSONReader; +//import com.alibaba.fastjson2.JSONWriter; +// +///** +// * Redis使用FastJson序列化 +// * +// * @author ruoyi +// */ +//public class FastJson2JsonRedisSerializer implements RedisSerializer +//{ +// public static final Charset DEFAULT_CHARSET = Charset.forName("UTF-8"); +// +// private Class clazz; +// +// public FastJson2JsonRedisSerializer(Class clazz) +// { +// super(); +// this.clazz = clazz; +// } +// +// @Override +// public byte[] serialize(T t) throws SerializationException +// { +// if (t == null) +// { +// return new byte[0]; +// } +// return JSON.toJSONString(t, JSONWriter.Feature.WriteClassName).getBytes(DEFAULT_CHARSET); +// } +// +// @Override +// public T deserialize(byte[] bytes) throws SerializationException +// { +// if (bytes == null || bytes.length <= 0) +// { +// return null; +// } +// String str = new String(bytes, DEFAULT_CHARSET); +// +// return JSON.parseObject(str, clazz, JSONReader.Feature.SupportAutoType); +// } +//} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/config/FilterConfig.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/FilterConfig.java new file mode 100644 index 00000000..c5e6b15b --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/FilterConfig.java @@ -0,0 +1,58 @@ +package com.ruoyi.framework.config; + +import java.util.HashMap; +import java.util.Map; +import javax.servlet.DispatcherType; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; +import org.springframework.boot.web.servlet.FilterRegistrationBean; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import com.ruoyi.common.filter.RepeatableFilter; +import com.ruoyi.common.filter.XssFilter; +import com.ruoyi.common.utils.StringUtils; + +/** + * Filter配置 + * + * @author ruoyi + */ +@Configuration +public class FilterConfig +{ + @Value("${xss.excludes}") + private String excludes; + + @Value("${xss.urlPatterns}") + private String urlPatterns; + + @SuppressWarnings({ "rawtypes", "unchecked" }) + @Bean + @ConditionalOnProperty(value = "xss.enabled", havingValue = "true") + public FilterRegistrationBean xssFilterRegistration() + { + FilterRegistrationBean registration = new FilterRegistrationBean(); + registration.setDispatcherTypes(DispatcherType.REQUEST); + registration.setFilter(new XssFilter()); + registration.addUrlPatterns(StringUtils.split(urlPatterns, ",")); + registration.setName("xssFilter"); + registration.setOrder(FilterRegistrationBean.HIGHEST_PRECEDENCE); + Map initParameters = new HashMap(); + initParameters.put("excludes", excludes); + registration.setInitParameters(initParameters); + return registration; + } + +// @SuppressWarnings({ "rawtypes", "unchecked" }) +// @Bean +// public FilterRegistrationBean someFilterRegistration() +// { +// FilterRegistrationBean registration = new FilterRegistrationBean(); +// registration.setFilter(new RepeatableFilter()); +// registration.addUrlPatterns("/*"); +// registration.setName("repeatableFilter"); +// registration.setOrder(FilterRegistrationBean.LOWEST_PRECEDENCE); +// return registration; +// } + +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/config/KaptchaTextCreator.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/KaptchaTextCreator.java new file mode 100644 index 00000000..7f8e1d50 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/KaptchaTextCreator.java @@ -0,0 +1,68 @@ +package com.ruoyi.framework.config; + +import java.util.Random; +import com.google.code.kaptcha.text.impl.DefaultTextCreator; + +/** + * 验证码文本生成器 + * + * @author ruoyi + */ +public class KaptchaTextCreator extends DefaultTextCreator +{ + private static final String[] CNUMBERS = "0,1,2,3,4,5,6,7,8,9,10".split(","); + + @Override + public String getText() + { + Integer result = 0; + Random random = new Random(); + int x = random.nextInt(10); + int y = random.nextInt(10); + StringBuilder suChinese = new StringBuilder(); + int randomoperands = random.nextInt(3); + if (randomoperands == 0) + { + result = x * y; + suChinese.append(CNUMBERS[x]); + suChinese.append("*"); + suChinese.append(CNUMBERS[y]); + } + else if (randomoperands == 1) + { + if ((x != 0) && y % x == 0) + { + result = y / x; + suChinese.append(CNUMBERS[y]); + suChinese.append("/"); + suChinese.append(CNUMBERS[x]); + } + else + { + result = x + y; + suChinese.append(CNUMBERS[x]); + suChinese.append("+"); + suChinese.append(CNUMBERS[y]); + } + } + else + { + if (x >= y) + { + result = x - y; + suChinese.append(CNUMBERS[x]); + suChinese.append("-"); + suChinese.append(CNUMBERS[y]); + } + else + { + result = y - x; + suChinese.append(CNUMBERS[y]); + suChinese.append("-"); + suChinese.append(CNUMBERS[x]); + } + } + suChinese.append("=?@" + result); + return suChinese.toString(); + } +} \ No newline at end of file diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/config/MybatisPlusConfig.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/MybatisPlusConfig.java new file mode 100644 index 00000000..997a1013 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/MybatisPlusConfig.java @@ -0,0 +1,70 @@ +package com.ruoyi.framework.config; + +import com.baomidou.mybatisplus.annotation.DbType; +import com.baomidou.mybatisplus.core.handlers.MetaObjectHandler; +import com.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor; +import com.baomidou.mybatisplus.extension.plugins.inner.BlockAttackInnerInterceptor; +import com.baomidou.mybatisplus.extension.plugins.inner.OptimisticLockerInnerInterceptor; +import com.baomidou.mybatisplus.extension.plugins.inner.PaginationInnerInterceptor; +import com.ruoyi.common.mybatis.handler.DefaultDBFieldHandler; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.transaction.annotation.EnableTransactionManagement; + +/** + * Mybatis Plus 配置 + * + * @author ruoyi + */ +@EnableTransactionManagement(proxyTargetClass = true) +@Configuration +public class MybatisPlusConfig +{ + @Bean + public MybatisPlusInterceptor mybatisPlusInterceptor() + { + MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor(); + // 分页插件 + interceptor.addInnerInterceptor(paginationInnerInterceptor()); + // 乐观锁插件 + interceptor.addInnerInterceptor(optimisticLockerInnerInterceptor()); + // 阻断插件 + interceptor.addInnerInterceptor(blockAttackInnerInterceptor()); + return interceptor; + } + + /** + * 分页插件,自动识别数据库类型 https://baomidou.com/guide/interceptor-pagination.html + */ + public PaginationInnerInterceptor paginationInnerInterceptor() + { + PaginationInnerInterceptor paginationInnerInterceptor = new PaginationInnerInterceptor(); + // 设置数据库类型为mysql + paginationInnerInterceptor.setDbType(DbType.MYSQL); + // 设置最大单页限制数量,默认 500 条,-1 不受限制 + paginationInnerInterceptor.setMaxLimit(-1L); + return paginationInnerInterceptor; + } + + /** + * 乐观锁插件 https://baomidou.com/guide/interceptor-optimistic-locker.html + */ + public OptimisticLockerInnerInterceptor optimisticLockerInnerInterceptor() + { + return new OptimisticLockerInnerInterceptor(); + } + + /** + * 如果是对全表的删除或更新操作,就会终止该操作 https://baomidou.com/guide/interceptor-block-attack.html + */ + public BlockAttackInnerInterceptor blockAttackInnerInterceptor() + { + return new BlockAttackInnerInterceptor(); + } + + @Bean + public MetaObjectHandler defaultMetaObjectHandler(){ + // 自动填充参数类 + return new DefaultDBFieldHandler(); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/config/RedisConfig.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/RedisConfig.java new file mode 100644 index 00000000..3235c42a --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/RedisConfig.java @@ -0,0 +1,69 @@ +//package com.ruoyi.framework.config; +// +//import org.springframework.cache.annotation.CachingConfigurerSupport; +//import org.springframework.cache.annotation.EnableCaching; +//import org.springframework.context.annotation.Bean; +//import org.springframework.context.annotation.Configuration; +//import org.springframework.data.redis.connection.RedisConnectionFactory; +//import org.springframework.data.redis.core.RedisTemplate; +//import org.springframework.data.redis.core.script.DefaultRedisScript; +//import org.springframework.data.redis.serializer.StringRedisSerializer; +// +///** +// * redis配置 +// * +// * @author ruoyi +// */ +//@Configuration +//@EnableCaching +//public class RedisConfig extends CachingConfigurerSupport +//{ +// @Bean +// @SuppressWarnings(value = { "unchecked", "rawtypes" }) +// public RedisTemplate redisTemplate(RedisConnectionFactory connectionFactory) +// { +// RedisTemplate template = new RedisTemplate<>(); +// template.setConnectionFactory(connectionFactory); +// +// FastJson2JsonRedisSerializer serializer = new FastJson2JsonRedisSerializer(Object.class); +// +// // 使用StringRedisSerializer来序列化和反序列化redis的key值 +// template.setKeySerializer(new StringRedisSerializer()); +// template.setValueSerializer(serializer); +// +// // Hash的key也采用StringRedisSerializer的序列化方式 +// template.setHashKeySerializer(new StringRedisSerializer()); +// template.setHashValueSerializer(serializer); +// +// template.afterPropertiesSet(); +// return template; +// } +// +// @Bean +// public DefaultRedisScript limitScript() +// { +// DefaultRedisScript redisScript = new DefaultRedisScript<>(); +// redisScript.setScriptText(limitScriptText()); +// redisScript.setResultType(Long.class); +// return redisScript; +// } +// +// /** +// * 限流脚本 +// */ +// private String limitScriptText() +// { +// return "local key = KEYS[1]\n" + +// "local count = tonumber(ARGV[1])\n" + +// "local time = tonumber(ARGV[2])\n" + +// "local current = redis.call('get', key);\n" + +// "if current and tonumber(current) > count then\n" + +// " return tonumber(current);\n" + +// "end\n" + +// "current = redis.call('incr', key)\n" + +// "if tonumber(current) == 1 then\n" + +// " redis.call('expire', key, time)\n" + +// "end\n" + +// "return tonumber(current);"; +// } +//} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/config/ResourcesConfig.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/ResourcesConfig.java new file mode 100644 index 00000000..26a7d4da --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/ResourcesConfig.java @@ -0,0 +1,71 @@ +package com.ruoyi.framework.config; + +import com.ruoyi.common.config.RuoYiConfig; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.framework.interceptor.RepeatSubmitInterceptor; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.http.CacheControl; +import org.springframework.web.cors.CorsConfiguration; +import org.springframework.web.cors.UrlBasedCorsConfigurationSource; +import org.springframework.web.filter.CorsFilter; +import org.springframework.web.servlet.config.annotation.InterceptorRegistry; +import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry; +import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; + +import java.util.concurrent.TimeUnit; + +/** + * 通用配置 + * + * @author ruoyi + */ +@Configuration +public class ResourcesConfig implements WebMvcConfigurer { + @Autowired + private RepeatSubmitInterceptor repeatSubmitInterceptor; + + @Override + public void addResourceHandlers(ResourceHandlerRegistry registry) { + /** 本地文件上传路径 */ + registry.addResourceHandler(Constants.RESOURCE_PREFIX + "/**") + .addResourceLocations("file:" + RuoYiConfig.getProfile() + "/"); + + /** swagger配置 */ + registry.addResourceHandler("/swagger-ui/**") + .addResourceLocations("classpath:/META-INF/resources/webjars/springfox-swagger-ui/") + .setCacheControl(CacheControl.maxAge(5, TimeUnit.HOURS).cachePublic()); + ; + } + + /** + * 自定义拦截规则 + */ + @Override + public void addInterceptors(InterceptorRegistry registry) { + registry.addInterceptor(repeatSubmitInterceptor).addPathPatterns("/**"); + } + + /** + * 跨域配置 + */ + @Bean + public CorsFilter corsFilter() { + CorsConfiguration config = new CorsConfiguration(); + config.setAllowCredentials(true); + // 设置访问源地址 + config.addAllowedOriginPattern("*"); + // 设置访问源请求头 + config.addAllowedHeader("*"); + // 设置访问源请求方法 + config.addAllowedMethod("*"); + // 有效期 1800秒 + config.setMaxAge(1800L); + // 添加映射路径,拦截一切请求 + UrlBasedCorsConfigurationSource source = new UrlBasedCorsConfigurationSource(); + source.registerCorsConfiguration("/**", config); + // 返回新的CorsFilter + return new CorsFilter(source); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/config/SecurityConfig.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/SecurityConfig.java new file mode 100644 index 00000000..ee972246 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/SecurityConfig.java @@ -0,0 +1,204 @@ +package com.ruoyi.framework.config; + +import com.google.common.collect.HashMultimap; +import com.google.common.collect.Multimap; +import com.ruoyi.framework.config.properties.PermitAllUrlProperties; +import com.ruoyi.framework.security.filter.JwtAuthenticationTokenFilter; +import com.ruoyi.framework.security.handle.AuthenticationEntryPointImpl; +import com.ruoyi.framework.security.handle.LogoutSuccessHandlerImpl; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.context.ApplicationContext; +import org.springframework.context.annotation.Bean; +import org.springframework.http.HttpMethod; +import org.springframework.security.authentication.AuthenticationManager; +import org.springframework.security.config.annotation.authentication.builders.AuthenticationManagerBuilder; +import org.springframework.security.config.annotation.method.configuration.EnableGlobalMethodSecurity; +import org.springframework.security.config.annotation.web.builders.HttpSecurity; +import org.springframework.security.config.annotation.web.configuration.WebSecurityConfigurerAdapter; +import org.springframework.security.config.annotation.web.configurers.ExpressionUrlAuthorizationConfigurer; +import org.springframework.security.config.http.SessionCreationPolicy; +import org.springframework.security.core.userdetails.UserDetailsService; +import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder; +import org.springframework.security.web.authentication.UsernamePasswordAuthenticationFilter; +import org.springframework.security.web.authentication.logout.LogoutFilter; +import org.springframework.web.filter.CorsFilter; +import org.springframework.web.method.HandlerMethod; +import org.springframework.web.servlet.mvc.method.RequestMappingInfo; +import org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping; + +import javax.annotation.Resource; +import javax.annotation.security.PermitAll; +import java.util.Map; +import java.util.Set; + +/** + * spring security配置 + * + * @author ruoyi + */ +@EnableGlobalMethodSecurity(prePostEnabled = true, securedEnabled = true) +public class SecurityConfig extends WebSecurityConfigurerAdapter { + /** + * 自定义用户认证逻辑 + */ + @Autowired + private UserDetailsService userDetailsService; + + /** + * 认证失败处理类 + */ + @Autowired + private AuthenticationEntryPointImpl unauthorizedHandler; + + /** + * 退出处理类 + */ + @Autowired + private LogoutSuccessHandlerImpl logoutSuccessHandler; + + /** + * token认证过滤器 + */ + @Autowired + private JwtAuthenticationTokenFilter authenticationTokenFilter; + + /** + * 跨域过滤器 + */ + @Autowired + private CorsFilter corsFilter; + + /** + * 允许匿名访问的地址 + */ + @Autowired + private PermitAllUrlProperties permitAllUrl; + @Resource + private ApplicationContext applicationContext; + /** + * 解决 无法直接注入 AuthenticationManager + * + * @return + * @throws Exception + */ + @Bean + @Override + public AuthenticationManager authenticationManagerBean() throws Exception { + return super.authenticationManagerBean(); + } + + /** + * anyRequest | 匹配所有请求路径 + * access | SpringEl表达式结果为true时可以访问 + * anonymous | 匿名可以访问 + * denyAll | 用户不能访问 + * fullyAuthenticated | 用户完全认证可以访问(非remember-me下自动登录) + * hasAnyAuthority | 如果有参数,参数表示权限,则其中任何一个权限可以访问 + * hasAnyRole | 如果有参数,参数表示角色,则其中任何一个角色可以访问 + * hasAuthority | 如果有参数,参数表示权限,则其权限可以访问 + * hasIpAddress | 如果有参数,参数表示IP地址,如果用户IP和参数匹配,则可以访问 + * hasRole | 如果有参数,参数表示角色,则其角色可以访问 + * permitAll | 用户可以任意访问 + * rememberMe | 允许通过remember-me登录的用户访问 + * authenticated | 用户登录后可访问 + */ + @Override + protected void configure(HttpSecurity httpSecurity) throws Exception { + // 注解标记允许匿名访问的url + ExpressionUrlAuthorizationConfigurer.ExpressionInterceptUrlRegistry registry = httpSecurity.authorizeRequests(); + permitAllUrl.getUrls().forEach(url -> registry.antMatchers(url).permitAll()); + // 获得 @PermitAll 带来的 URL 列表,免登录 + Multimap permitAllUrls = getPermitAllUrlsFromAnnotations(); + httpSecurity + // CSRF禁用,因为不使用session + .csrf().disable() + // 禁用HTTP响应标头 + .headers().cacheControl().disable().and() + // 认证失败处理类 + .exceptionHandling().authenticationEntryPoint(unauthorizedHandler).and() + // 基于token,所以不需要session + .sessionManagement().sessionCreationPolicy(SessionCreationPolicy.STATELESS).and() + // 过滤请求 + .authorizeRequests() + // 对于登录login 注册register 验证码captchaImage 允许匿名访问 + .antMatchers("/login", "/register", "/captchaImage").permitAll() + // 静态资源,可匿名访问 + .antMatchers(HttpMethod.GET, "/", "/*.html", "/**/*.html", "/**/*.css", "/**/*.js", "/profile/**").permitAll() + .antMatchers("/swagger-ui.html", "/swagger-resources/**", "/webjars/**", "/*/api-docs", "/druid/**","/common/chatgpt").permitAll() + // 1.2 设置 @PermitAll 无需认证 + .antMatchers(HttpMethod.GET, permitAllUrls.get(HttpMethod.GET).toArray(new String[0])).permitAll() + .antMatchers(HttpMethod.POST, permitAllUrls.get(HttpMethod.POST).toArray(new String[0])).permitAll() + .antMatchers(HttpMethod.PUT, permitAllUrls.get(HttpMethod.PUT).toArray(new String[0])).permitAll() + .antMatchers(HttpMethod.DELETE, permitAllUrls.get(HttpMethod.DELETE).toArray(new String[0])).permitAll() + // websocket如果需要不登录也可以访问,需要在`SecurityConfig.java`中设置匿名访问 + .antMatchers("/websocket/**").permitAll() + //积木报表 + .antMatchers("/jmreport/**").permitAll() + // 任务回退接口 + .antMatchers("/bpm/task/back").permitAll() + // 除上面外的所有请求全部需要鉴权认证 + .anyRequest().authenticated() + .and() + .headers().frameOptions().disable(); + // 添加Logout filter + httpSecurity.logout().logoutUrl("/logout").logoutSuccessHandler(logoutSuccessHandler); + // 添加JWT filter + httpSecurity.addFilterBefore(authenticationTokenFilter, UsernamePasswordAuthenticationFilter.class); + // 添加CORS filter + httpSecurity.addFilterBefore(corsFilter, JwtAuthenticationTokenFilter.class); + httpSecurity.addFilterBefore(corsFilter, LogoutFilter.class); + } + + /** + * 强散列哈希加密实现 + */ + @Bean + public BCryptPasswordEncoder bCryptPasswordEncoder() { + return new BCryptPasswordEncoder(); + } + + /** + * 身份认证接口 + */ + @Override + protected void configure(AuthenticationManagerBuilder auth) throws Exception { + auth.userDetailsService(userDetailsService).passwordEncoder(bCryptPasswordEncoder()); + } + + private Multimap getPermitAllUrlsFromAnnotations() { + Multimap result = HashMultimap.create(); + // 获得接口对应的 HandlerMethod 集合 + RequestMappingHandlerMapping requestMappingHandlerMapping = (RequestMappingHandlerMapping) + applicationContext.getBean("requestMappingHandlerMapping"); + Map handlerMethodMap = requestMappingHandlerMapping.getHandlerMethods(); + // 获得有 @PermitAll 注解的接口 + for (Map.Entry entry : handlerMethodMap.entrySet()) { + HandlerMethod handlerMethod = entry.getValue(); + if (!handlerMethod.hasMethodAnnotation(PermitAll.class)) { + continue; + } + if (entry.getKey().getPatternsCondition() == null) { + continue; + } + Set urls = entry.getKey().getPatternsCondition().getPatterns(); + // 根据请求方法,添加到 result 结果 + entry.getKey().getMethodsCondition().getMethods().forEach(requestMethod -> { + switch (requestMethod) { + case GET: + result.putAll(HttpMethod.GET, urls); + break; + case POST: + result.putAll(HttpMethod.POST, urls); + break; + case PUT: + result.putAll(HttpMethod.PUT, urls); + break; + case DELETE: + result.putAll(HttpMethod.DELETE, urls); + break; + } + }); + } + return result; + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/config/ServerConfig.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/ServerConfig.java new file mode 100644 index 00000000..b5b7de31 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/ServerConfig.java @@ -0,0 +1,32 @@ +package com.ruoyi.framework.config; + +import javax.servlet.http.HttpServletRequest; +import org.springframework.stereotype.Component; +import com.ruoyi.common.utils.ServletUtils; + +/** + * 服务相关配置 + * + * @author ruoyi + */ +@Component +public class ServerConfig +{ + /** + * 获取完整的请求路径,包括:域名,端口,上下文访问路径 + * + * @return 服务地址 + */ + public String getUrl() + { + HttpServletRequest request = ServletUtils.getRequest(); + return getDomain(request); + } + + public static String getDomain(HttpServletRequest request) + { + StringBuffer url = request.getRequestURL(); + String contextPath = request.getServletContext().getContextPath(); + return url.delete(url.length() - request.getRequestURI().length(), url.length()).append(contextPath).toString(); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/config/ThreadPoolConfig.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/ThreadPoolConfig.java new file mode 100644 index 00000000..7840141e --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/ThreadPoolConfig.java @@ -0,0 +1,63 @@ +package com.ruoyi.framework.config; + +import com.ruoyi.common.utils.Threads; +import org.apache.commons.lang3.concurrent.BasicThreadFactory; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor; +import java.util.concurrent.ScheduledExecutorService; +import java.util.concurrent.ScheduledThreadPoolExecutor; +import java.util.concurrent.ThreadPoolExecutor; + +/** + * 线程池配置 + * + * @author ruoyi + **/ +@Configuration +public class ThreadPoolConfig +{ + // 核心线程池大小 + private int corePoolSize = 50; + + // 最大可创建的线程数 + private int maxPoolSize = 200; + + // 队列最大长度 + private int queueCapacity = 1000; + + // 线程池维护线程所允许的空闲时间 + private int keepAliveSeconds = 300; + + @Bean(name = "threadPoolTaskExecutor") + public ThreadPoolTaskExecutor threadPoolTaskExecutor() + { + ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor(); + executor.setMaxPoolSize(maxPoolSize); + executor.setCorePoolSize(corePoolSize); + executor.setQueueCapacity(queueCapacity); + executor.setKeepAliveSeconds(keepAliveSeconds); + // 线程池对拒绝任务(无线程可用)的处理策略 + executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy()); + return executor; + } + + /** + * 执行周期性或定时任务 + */ + @Bean(name = "scheduledExecutorService") + protected ScheduledExecutorService scheduledExecutorService() + { + return new ScheduledThreadPoolExecutor(corePoolSize, + new BasicThreadFactory.Builder().namingPattern("schedule-pool-%d").daemon(true).build(), + new ThreadPoolExecutor.CallerRunsPolicy()) + { + @Override + protected void afterExecute(Runnable r, Throwable t) + { + super.afterExecute(r, t); + Threads.printException(r, t); + } + }; + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/config/properties/DruidProperties.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/properties/DruidProperties.java new file mode 100644 index 00000000..84f7e009 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/properties/DruidProperties.java @@ -0,0 +1,77 @@ +package com.ruoyi.framework.config.properties; + +import org.springframework.beans.factory.annotation.Value; +import org.springframework.context.annotation.Configuration; +import com.alibaba.druid.pool.DruidDataSource; + +/** + * druid 配置属性 + * + * @author ruoyi + */ +@Configuration +public class DruidProperties +{ + @Value("${spring.datasource.druid.initialSize}") + private int initialSize; + + @Value("${spring.datasource.druid.minIdle}") + private int minIdle; + + @Value("${spring.datasource.druid.maxActive}") + private int maxActive; + + @Value("${spring.datasource.druid.maxWait}") + private int maxWait; + + @Value("${spring.datasource.druid.timeBetweenEvictionRunsMillis}") + private int timeBetweenEvictionRunsMillis; + + @Value("${spring.datasource.druid.minEvictableIdleTimeMillis}") + private int minEvictableIdleTimeMillis; + + @Value("${spring.datasource.druid.maxEvictableIdleTimeMillis}") + private int maxEvictableIdleTimeMillis; + + @Value("${spring.datasource.druid.validationQuery}") + private String validationQuery; + + @Value("${spring.datasource.druid.testWhileIdle}") + private boolean testWhileIdle; + + @Value("${spring.datasource.druid.testOnBorrow}") + private boolean testOnBorrow; + + @Value("${spring.datasource.druid.testOnReturn}") + private boolean testOnReturn; + + public DruidDataSource dataSource(DruidDataSource datasource) + { + /** 配置初始化大小、最小、最大 */ + datasource.setInitialSize(initialSize); + datasource.setMaxActive(maxActive); + datasource.setMinIdle(minIdle); + + /** 配置获取连接等待超时的时间 */ + datasource.setMaxWait(maxWait); + + /** 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 */ + datasource.setTimeBetweenEvictionRunsMillis(timeBetweenEvictionRunsMillis); + + /** 配置一个连接在池中最小、最大生存的时间,单位是毫秒 */ + datasource.setMinEvictableIdleTimeMillis(minEvictableIdleTimeMillis); + datasource.setMaxEvictableIdleTimeMillis(maxEvictableIdleTimeMillis); + + /** + * 用来检测连接是否有效的sql,要求是一个查询语句,常用select 'x'。如果validationQuery为null,testOnBorrow、testOnReturn、testWhileIdle都不会起作用。 + */ + datasource.setValidationQuery(validationQuery); + /** 建议配置为true,不影响性能,并且保证安全性。申请连接的时候检测,如果空闲时间大于timeBetweenEvictionRunsMillis,执行validationQuery检测连接是否有效。 */ + datasource.setTestWhileIdle(testWhileIdle); + /** 申请连接时执行validationQuery检测连接是否有效,做了这个配置会降低性能。 */ + datasource.setTestOnBorrow(testOnBorrow); + /** 归还连接时执行validationQuery检测连接是否有效,做了这个配置会降低性能。 */ + datasource.setTestOnReturn(testOnReturn); + return datasource; + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/config/properties/PermitAllUrlProperties.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/properties/PermitAllUrlProperties.java new file mode 100644 index 00000000..9632eb1f --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/properties/PermitAllUrlProperties.java @@ -0,0 +1,72 @@ +package com.ruoyi.framework.config.properties; + +import java.util.ArrayList; +import java.util.List; +import java.util.Map; +import java.util.Optional; +import java.util.regex.Pattern; +import org.apache.commons.lang3.RegExUtils; +import org.springframework.beans.BeansException; +import org.springframework.beans.factory.InitializingBean; +import org.springframework.context.ApplicationContext; +import org.springframework.context.ApplicationContextAware; +import org.springframework.context.annotation.Configuration; +import org.springframework.core.annotation.AnnotationUtils; +import org.springframework.web.method.HandlerMethod; +import org.springframework.web.servlet.mvc.method.RequestMappingInfo; +import org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping; +import com.ruoyi.common.annotation.Anonymous; + +/** + * 设置Anonymous注解允许匿名访问的url + * + * @author ruoyi + */ +@Configuration +public class PermitAllUrlProperties implements InitializingBean, ApplicationContextAware +{ + private static final Pattern PATTERN = Pattern.compile("\\{(.*?)\\}"); + + private ApplicationContext applicationContext; + + private List urls = new ArrayList<>(); + + public String ASTERISK = "*"; + + @Override + public void afterPropertiesSet() + { + RequestMappingHandlerMapping mapping = applicationContext.getBean(RequestMappingHandlerMapping.class); + Map map = mapping.getHandlerMethods(); + + map.keySet().forEach(info -> { + HandlerMethod handlerMethod = map.get(info); + + // 获取方法上边的注解 替代path variable 为 * + Anonymous method = AnnotationUtils.findAnnotation(handlerMethod.getMethod(), Anonymous.class); + Optional.ofNullable(method).ifPresent(anonymous -> info.getPatternsCondition().getPatterns() + .forEach(url -> urls.add(RegExUtils.replaceAll(url, PATTERN, ASTERISK)))); + + // 获取类上边的注解, 替代path variable 为 * + Anonymous controller = AnnotationUtils.findAnnotation(handlerMethod.getBeanType(), Anonymous.class); + Optional.ofNullable(controller).ifPresent(anonymous -> info.getPatternsCondition().getPatterns() + .forEach(url -> urls.add(RegExUtils.replaceAll(url, PATTERN, ASTERISK)))); + }); + } + + @Override + public void setApplicationContext(ApplicationContext context) throws BeansException + { + this.applicationContext = context; + } + + public List getUrls() + { + return urls; + } + + public void setUrls(List urls) + { + this.urls = urls; + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/datasource/DynamicDataSource.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/datasource/DynamicDataSource.java new file mode 100644 index 00000000..e70b8cfa --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/datasource/DynamicDataSource.java @@ -0,0 +1,26 @@ +package com.ruoyi.framework.datasource; + +import java.util.Map; +import javax.sql.DataSource; +import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource; + +/** + * 动态数据源 + * + * @author ruoyi + */ +public class DynamicDataSource extends AbstractRoutingDataSource +{ + public DynamicDataSource(DataSource defaultTargetDataSource, Map targetDataSources) + { + super.setDefaultTargetDataSource(defaultTargetDataSource); + super.setTargetDataSources(targetDataSources); + super.afterPropertiesSet(); + } + + @Override + protected Object determineCurrentLookupKey() + { + return DynamicDataSourceContextHolder.getDataSourceType(); + } +} \ No newline at end of file diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/datasource/DynamicDataSourceContextHolder.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/datasource/DynamicDataSourceContextHolder.java new file mode 100644 index 00000000..3572db91 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/datasource/DynamicDataSourceContextHolder.java @@ -0,0 +1,45 @@ +package com.ruoyi.framework.datasource; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * 数据源切换处理 + * + * @author ruoyi + */ +public class DynamicDataSourceContextHolder +{ + public static final Logger log = LoggerFactory.getLogger(DynamicDataSourceContextHolder.class); + + /** + * 使用ThreadLocal维护变量,ThreadLocal为每个使用该变量的线程提供独立的变量副本, + * 所以每一个线程都可以独立地改变自己的副本,而不会影响其它线程所对应的副本。 + */ + private static final ThreadLocal CONTEXT_HOLDER = new ThreadLocal<>(); + + /** + * 设置数据源的变量 + */ + public static void setDataSourceType(String dsType) + { + log.info("切换到{}数据源", dsType); + CONTEXT_HOLDER.set(dsType); + } + + /** + * 获得数据源的变量 + */ + public static String getDataSourceType() + { + return CONTEXT_HOLDER.get(); + } + + /** + * 清空数据源变量 + */ + public static void clearDataSourceType() + { + CONTEXT_HOLDER.remove(); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/interceptor/RepeatSubmitInterceptor.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/interceptor/RepeatSubmitInterceptor.java new file mode 100644 index 00000000..4ddacaed --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/interceptor/RepeatSubmitInterceptor.java @@ -0,0 +1,55 @@ +package com.ruoyi.framework.interceptor; + +import java.lang.reflect.Method; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import org.springframework.stereotype.Component; +import org.springframework.web.method.HandlerMethod; +import org.springframework.web.servlet.HandlerInterceptor; +import com.alibaba.fastjson2.JSON; +import com.ruoyi.common.annotation.RepeatSubmit; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.utils.ServletUtils; + +/** + * 防止重复提交拦截器 + * + * @author ruoyi + */ +@Component +public abstract class RepeatSubmitInterceptor implements HandlerInterceptor +{ + @Override + public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception + { + if (handler instanceof HandlerMethod) + { + HandlerMethod handlerMethod = (HandlerMethod) handler; + Method method = handlerMethod.getMethod(); + RepeatSubmit annotation = method.getAnnotation(RepeatSubmit.class); + if (annotation != null) + { + if (this.isRepeatSubmit(request, annotation)) + { + AjaxResult ajaxResult = AjaxResult.error(annotation.message()); + ServletUtils.renderString(response, JSON.toJSONString(ajaxResult)); + return false; + } + } + return true; + } + else + { + return true; + } + } + + /** + * 验证是否重复提交由子类实现具体的防重复提交的规则 + * + * @param request + * @return + * @throws Exception + */ + public abstract boolean isRepeatSubmit(HttpServletRequest request, RepeatSubmit annotation); +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/interceptor/impl/SameUrlDataInterceptor.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/interceptor/impl/SameUrlDataInterceptor.java new file mode 100644 index 00000000..3160c054 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/interceptor/impl/SameUrlDataInterceptor.java @@ -0,0 +1,112 @@ +package com.ruoyi.framework.interceptor.impl; + +import java.util.HashMap; +import java.util.Map; +import java.util.concurrent.TimeUnit; +import javax.servlet.http.HttpServletRequest; + +import com.ruoyi.common.utils.EhcacheUtil; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.stereotype.Component; +import com.alibaba.fastjson2.JSON; +import com.ruoyi.common.annotation.RepeatSubmit; +import com.ruoyi.common.constant.CacheConstants; +import com.ruoyi.common.filter.RepeatedlyRequestWrapper; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.http.HttpHelper; +import com.ruoyi.framework.interceptor.RepeatSubmitInterceptor; + +/** + * 判断请求url和数据是否和上一次相同, + * 如果和上次相同,则是重复提交表单。 有效时间为10秒内。 + * + * @author ruoyi + */ +@Component +public class SameUrlDataInterceptor extends RepeatSubmitInterceptor +{ + public final String REPEAT_PARAMS = "repeatParams"; + + public final String REPEAT_TIME = "repeatTime"; + + // 令牌自定义标识 + @Value("${token.header}") + private String header; + +// @Autowired +// private RedisCache redisCache; + + @SuppressWarnings("unchecked") + @Override + public boolean isRepeatSubmit(HttpServletRequest request, RepeatSubmit annotation) + { + String nowParams = ""; + if (request instanceof RepeatedlyRequestWrapper) + { + RepeatedlyRequestWrapper repeatedlyRequest = (RepeatedlyRequestWrapper) request; + nowParams = HttpHelper.getBodyString(repeatedlyRequest); + } + + // body参数为空,获取Parameter的数据 + if (StringUtils.isEmpty(nowParams)) + { + nowParams = JSON.toJSONString(request.getParameterMap()); + } + Map nowDataMap = new HashMap(); + nowDataMap.put(REPEAT_PARAMS, nowParams); + nowDataMap.put(REPEAT_TIME, System.currentTimeMillis()); + + // 请求地址(作为存放cache的key值) + String url = request.getRequestURI(); + + // 唯一值(没有消息头则使用请求地址) + String submitKey = StringUtils.trimToEmpty(request.getHeader(header)); + + // 唯一标识(指定key + url + 消息头) + String cacheRepeatKey = CacheConstants.REPEAT_SUBMIT_KEY + url + submitKey; +// Object sessionObj = redisCache.getCacheObject(cacheRepeatKey); + Object sessionObj = EhcacheUtil.get("repeatsubmit",cacheRepeatKey); + if (sessionObj != null) + { + Map sessionMap = (Map) sessionObj; + if (sessionMap.containsKey(url)) + { + Map preDataMap = (Map) sessionMap.get(url); + if (compareParams(nowDataMap, preDataMap) && compareTime(nowDataMap, preDataMap, annotation.interval())) + { + return true; + } + } + } + Map cacheMap = new HashMap(); + cacheMap.put(url, nowDataMap); + EhcacheUtil.put("repeatsubmit",cacheRepeatKey,cacheMap); +// redisCache.setCacheObject(cacheRepeatKey, cacheMap, annotation.interval(), TimeUnit.MILLISECONDS); + return false; + } + + /** + * 判断参数是否相同 + */ + private boolean compareParams(Map nowMap, Map preMap) + { + String nowParams = (String) nowMap.get(REPEAT_PARAMS); + String preParams = (String) preMap.get(REPEAT_PARAMS); + return nowParams.equals(preParams); + } + + /** + * 判断两次间隔时间 + */ + private boolean compareTime(Map nowMap, Map preMap, int interval) + { + long time1 = (Long) nowMap.get(REPEAT_TIME); + long time2 = (Long) preMap.get(REPEAT_TIME); + if ((time1 - time2) < interval) + { + return true; + } + return false; + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/manager/AsyncManager.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/manager/AsyncManager.java new file mode 100644 index 00000000..7387a02c --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/manager/AsyncManager.java @@ -0,0 +1,55 @@ +package com.ruoyi.framework.manager; + +import java.util.TimerTask; +import java.util.concurrent.ScheduledExecutorService; +import java.util.concurrent.TimeUnit; +import com.ruoyi.common.utils.Threads; +import com.ruoyi.common.utils.spring.SpringUtils; + +/** + * 异步任务管理器 + * + * @author ruoyi + */ +public class AsyncManager +{ + /** + * 操作延迟10毫秒 + */ + private final int OPERATE_DELAY_TIME = 10; + + /** + * 异步操作任务调度线程池 + */ + private ScheduledExecutorService executor = SpringUtils.getBean("scheduledExecutorService"); + + /** + * 单例模式 + */ + private AsyncManager(){} + + private static AsyncManager me = new AsyncManager(); + + public static AsyncManager me() + { + return me; + } + + /** + * 执行任务 + * + * @param task 任务 + */ + public void execute(TimerTask task) + { + executor.schedule(task, OPERATE_DELAY_TIME, TimeUnit.MILLISECONDS); + } + + /** + * 停止任务线程池 + */ + public void shutdown() + { + Threads.shutdownAndAwaitTermination(executor); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/manager/ShutdownManager.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/manager/ShutdownManager.java new file mode 100644 index 00000000..ec695cc9 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/manager/ShutdownManager.java @@ -0,0 +1,41 @@ +package com.ruoyi.framework.manager; + +import com.ruoyi.common.utils.EhcacheUtil; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.stereotype.Component; +import javax.annotation.PreDestroy; + +/** + * 确保应用退出时能关闭后台线程 + * + * @author ruoyi + */ +@Component +public class ShutdownManager +{ + private static final Logger logger = LoggerFactory.getLogger("sys-user"); + + @PreDestroy + public void destroy() + { + shutdownAsyncManager(); + } + + /** + * 停止异步执行任务 + */ + private void shutdownAsyncManager() + { + try + { + logger.info("====关闭后台任务任务线程池===="); + EhcacheUtil.cacheManager.shutdown(); + AsyncManager.me().shutdown(); + } + catch (Exception e) + { + logger.error(e.getMessage(), e); + } + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/manager/factory/AsyncFactory.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/manager/factory/AsyncFactory.java new file mode 100644 index 00000000..23d02308 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/manager/factory/AsyncFactory.java @@ -0,0 +1,102 @@ +package com.ruoyi.framework.manager.factory; + +import java.util.TimerTask; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.utils.LogUtils; +import com.ruoyi.common.utils.ServletUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.ip.AddressUtils; +import com.ruoyi.common.utils.ip.IpUtils; +import com.ruoyi.common.utils.spring.SpringUtils; +import com.ruoyi.system.domain.SysLogininfor; +import com.ruoyi.system.domain.SysOperLog; +import com.ruoyi.system.service.ISysLogininforService; +import com.ruoyi.system.service.ISysOperLogService; +import eu.bitwalker.useragentutils.UserAgent; + +/** + * 异步工厂(产生任务用) + * + * @author ruoyi + */ +public class AsyncFactory +{ + private static final Logger sys_user_logger = LoggerFactory.getLogger("sys-user"); + + /** + * 记录登录信息 + * + * @param username 用户名 + * @param status 状态 + * @param message 消息 + * @param args 列表 + * @return 任务task + */ + public static TimerTask recordLogininfor(final String username, final String status, final String message, + final Object... args) + { + final UserAgent userAgent = UserAgent.parseUserAgentString(ServletUtils.getRequest().getHeader("User-Agent")); + final String ip = IpUtils.getIpAddr(ServletUtils.getRequest()); + return new TimerTask() + { + @Override + public void run() + { + String address = AddressUtils.getRealAddressByIP(ip); + StringBuilder s = new StringBuilder(); + s.append(LogUtils.getBlock(ip)); + s.append(address); + s.append(LogUtils.getBlock(username)); + s.append(LogUtils.getBlock(status)); + s.append(LogUtils.getBlock(message)); + // 打印信息到日志 + sys_user_logger.info(s.toString(), args); + // 获取客户端操作系统 + String os = userAgent.getOperatingSystem().getName(); + // 获取客户端浏览器 + String browser = userAgent.getBrowser().getName(); + // 封装对象 + SysLogininfor logininfor = new SysLogininfor(); + logininfor.setUserName(username); + logininfor.setIpaddr(ip); + logininfor.setLoginLocation(address); + logininfor.setBrowser(browser); + logininfor.setOs(os); + logininfor.setMsg(message); + // 日志状态 + if (StringUtils.equalsAny(status, Constants.LOGIN_SUCCESS, Constants.LOGOUT, Constants.REGISTER)) + { + logininfor.setStatus(Constants.SUCCESS); + } + else if (Constants.LOGIN_FAIL.equals(status)) + { + logininfor.setStatus(Constants.FAIL); + } + // 插入数据 + SpringUtils.getBean(ISysLogininforService.class).insertLogininfor(logininfor); + } + }; + } + + /** + * 操作日志记录 + * + * @param operLog 操作日志信息 + * @return 任务task + */ + public static TimerTask recordOper(final SysOperLog operLog) + { + return new TimerTask() + { + @Override + public void run() + { + // 远程查询操作地点 + operLog.setOperLocation(AddressUtils.getRealAddressByIP(operLog.getOperIp())); + SpringUtils.getBean(ISysOperLogService.class).insertOperlog(operLog); + } + }; + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/security/context/AuthenticationContextHolder.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/security/context/AuthenticationContextHolder.java new file mode 100644 index 00000000..2daa9e63 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/security/context/AuthenticationContextHolder.java @@ -0,0 +1,28 @@ +package com.ruoyi.framework.security.context; + +import org.springframework.security.core.Authentication; + +/** + * 身份验证信息 + * + * @author ruoyi + */ +public class AuthenticationContextHolder +{ + private static final ThreadLocal contextHolder = new ThreadLocal<>(); + + public static Authentication getContext() + { + return contextHolder.get(); + } + + public static void setContext(Authentication context) + { + contextHolder.set(context); + } + + public static void clearContext() + { + contextHolder.remove(); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/security/context/PermissionContextHolder.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/security/context/PermissionContextHolder.java new file mode 100644 index 00000000..f9766e52 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/security/context/PermissionContextHolder.java @@ -0,0 +1,27 @@ +package com.ruoyi.framework.security.context; + +import com.ruoyi.common.core.text.Convert; +import org.springframework.web.context.request.RequestAttributes; +import org.springframework.web.context.request.RequestContextHolder; + +/** + * 权限信息 + * + * @author ruoyi + */ +public class PermissionContextHolder +{ + private static final String PERMISSION_CONTEXT_ATTRIBUTES = "PERMISSION_CONTEXT"; + + public static void setContext(String permission) + { + RequestContextHolder.currentRequestAttributes().setAttribute(PERMISSION_CONTEXT_ATTRIBUTES, permission, + RequestAttributes.SCOPE_REQUEST); + } + + public static String getContext() + { + return Convert.toStr(RequestContextHolder.currentRequestAttributes().getAttribute(PERMISSION_CONTEXT_ATTRIBUTES, + RequestAttributes.SCOPE_REQUEST)); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/security/filter/JwtAuthenticationTokenFilter.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/security/filter/JwtAuthenticationTokenFilter.java new file mode 100644 index 00000000..3eb24954 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/security/filter/JwtAuthenticationTokenFilter.java @@ -0,0 +1,44 @@ +package com.ruoyi.framework.security.filter; + +import java.io.IOException; +import javax.servlet.FilterChain; +import javax.servlet.ServletException; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.authentication.UsernamePasswordAuthenticationToken; +import org.springframework.security.core.context.SecurityContextHolder; +import org.springframework.security.web.authentication.WebAuthenticationDetailsSource; +import org.springframework.stereotype.Component; +import org.springframework.web.filter.OncePerRequestFilter; +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.framework.web.service.TokenService; + +/** + * token过滤器 验证token有效性 + * + * @author ruoyi + */ +@Component +public class JwtAuthenticationTokenFilter extends OncePerRequestFilter +{ + @Autowired + private TokenService tokenService; + + @Override + protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain chain) + throws ServletException, IOException + { + LoginUser loginUser = tokenService.getLoginUser(request); + if (StringUtils.isNotNull(loginUser) && StringUtils.isNull(SecurityUtils.getAuthentication())) + { + tokenService.verifyToken(loginUser); + UsernamePasswordAuthenticationToken authenticationToken = new UsernamePasswordAuthenticationToken(loginUser, null, loginUser.getAuthorities()); + authenticationToken.setDetails(new WebAuthenticationDetailsSource().buildDetails(request)); + SecurityContextHolder.getContext().setAuthentication(authenticationToken); + } + chain.doFilter(request, response); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/security/handle/AuthenticationEntryPointImpl.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/security/handle/AuthenticationEntryPointImpl.java new file mode 100644 index 00000000..93b70325 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/security/handle/AuthenticationEntryPointImpl.java @@ -0,0 +1,34 @@ +package com.ruoyi.framework.security.handle; + +import java.io.IOException; +import java.io.Serializable; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import org.springframework.security.core.AuthenticationException; +import org.springframework.security.web.AuthenticationEntryPoint; +import org.springframework.stereotype.Component; +import com.alibaba.fastjson2.JSON; +import com.ruoyi.common.constant.HttpStatus; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.utils.ServletUtils; +import com.ruoyi.common.utils.StringUtils; + +/** + * 认证失败处理类 返回未授权 + * + * @author ruoyi + */ +@Component +public class AuthenticationEntryPointImpl implements AuthenticationEntryPoint, Serializable +{ + private static final long serialVersionUID = -8970718410437077606L; + + @Override + public void commence(HttpServletRequest request, HttpServletResponse response, AuthenticationException e) + throws IOException + { + int code = HttpStatus.UNAUTHORIZED; + String msg = StringUtils.format("请求访问:{},认证失败,无法访问系统资源", request.getRequestURI()); + ServletUtils.renderString(response, JSON.toJSONString(AjaxResult.error(code, msg))); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/security/handle/LogoutSuccessHandlerImpl.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/security/handle/LogoutSuccessHandlerImpl.java new file mode 100644 index 00000000..6c2d4057 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/security/handle/LogoutSuccessHandlerImpl.java @@ -0,0 +1,50 @@ +package com.ruoyi.framework.security.handle; + +import com.alibaba.fastjson2.JSON; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.common.utils.ServletUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.framework.manager.AsyncManager; +import com.ruoyi.framework.manager.factory.AsyncFactory; +import com.ruoyi.framework.web.service.TokenService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.context.annotation.Configuration; +import org.springframework.security.core.Authentication; +import org.springframework.security.web.authentication.logout.LogoutSuccessHandler; + +import javax.servlet.ServletException; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import java.io.IOException; + +/** + * 自定义退出处理类 返回成功 + * + * @author ruoyi + */ +@Configuration +public class LogoutSuccessHandlerImpl implements LogoutSuccessHandler { + @Autowired + private TokenService tokenService; + + /** + * 退出处理 + * + * @return + */ + @Override + public void onLogoutSuccess(HttpServletRequest request, HttpServletResponse response, Authentication authentication) + throws IOException, ServletException { + LoginUser loginUser = tokenService.getLoginUser(request); + if (StringUtils.isNotNull(loginUser)) { + String userName = loginUser.getUsername(); + // 删除用户缓存记录 + tokenService.delLoginUser(loginUser.getToken()); + // 记录用户退出日志 + AsyncManager.me().execute(AsyncFactory.recordLogininfor(userName, Constants.LOGOUT, "退出成功")); + } + ServletUtils.renderString(response, JSON.toJSONString(AjaxResult.success("退出成功"))); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/Server.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/Server.java new file mode 100644 index 00000000..63b03da7 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/Server.java @@ -0,0 +1,240 @@ +package com.ruoyi.framework.web.domain; + +import java.net.UnknownHostException; +import java.util.LinkedList; +import java.util.List; +import java.util.Properties; +import com.ruoyi.common.utils.Arith; +import com.ruoyi.common.utils.ip.IpUtils; +import com.ruoyi.framework.web.domain.server.Cpu; +import com.ruoyi.framework.web.domain.server.Jvm; +import com.ruoyi.framework.web.domain.server.Mem; +import com.ruoyi.framework.web.domain.server.Sys; +import com.ruoyi.framework.web.domain.server.SysFile; +import oshi.SystemInfo; +import oshi.hardware.CentralProcessor; +import oshi.hardware.CentralProcessor.TickType; +import oshi.hardware.GlobalMemory; +import oshi.hardware.HardwareAbstractionLayer; +import oshi.software.os.FileSystem; +import oshi.software.os.OSFileStore; +import oshi.software.os.OperatingSystem; +import oshi.util.Util; + +/** + * 服务器相关信息 + * + * @author ruoyi + */ +public class Server +{ + private static final int OSHI_WAIT_SECOND = 1000; + + /** + * CPU相关信息 + */ + private Cpu cpu = new Cpu(); + + /** + * 內存相关信息 + */ + private Mem mem = new Mem(); + + /** + * JVM相关信息 + */ + private Jvm jvm = new Jvm(); + + /** + * 服务器相关信息 + */ + private Sys sys = new Sys(); + + /** + * 磁盘相关信息 + */ + private List sysFiles = new LinkedList(); + + public Cpu getCpu() + { + return cpu; + } + + public void setCpu(Cpu cpu) + { + this.cpu = cpu; + } + + public Mem getMem() + { + return mem; + } + + public void setMem(Mem mem) + { + this.mem = mem; + } + + public Jvm getJvm() + { + return jvm; + } + + public void setJvm(Jvm jvm) + { + this.jvm = jvm; + } + + public Sys getSys() + { + return sys; + } + + public void setSys(Sys sys) + { + this.sys = sys; + } + + public List getSysFiles() + { + return sysFiles; + } + + public void setSysFiles(List sysFiles) + { + this.sysFiles = sysFiles; + } + + public void copyTo() throws Exception + { + SystemInfo si = new SystemInfo(); + HardwareAbstractionLayer hal = si.getHardware(); + + setCpuInfo(hal.getProcessor()); + + setMemInfo(hal.getMemory()); + + setSysInfo(); + + setJvmInfo(); + + setSysFiles(si.getOperatingSystem()); + } + + /** + * 设置CPU信息 + */ + private void setCpuInfo(CentralProcessor processor) + { + // CPU信息 + long[] prevTicks = processor.getSystemCpuLoadTicks(); + Util.sleep(OSHI_WAIT_SECOND); + long[] ticks = processor.getSystemCpuLoadTicks(); + long nice = ticks[TickType.NICE.getIndex()] - prevTicks[TickType.NICE.getIndex()]; + long irq = ticks[TickType.IRQ.getIndex()] - prevTicks[TickType.IRQ.getIndex()]; + long softirq = ticks[TickType.SOFTIRQ.getIndex()] - prevTicks[TickType.SOFTIRQ.getIndex()]; + long steal = ticks[TickType.STEAL.getIndex()] - prevTicks[TickType.STEAL.getIndex()]; + long cSys = ticks[TickType.SYSTEM.getIndex()] - prevTicks[TickType.SYSTEM.getIndex()]; + long user = ticks[TickType.USER.getIndex()] - prevTicks[TickType.USER.getIndex()]; + long iowait = ticks[TickType.IOWAIT.getIndex()] - prevTicks[TickType.IOWAIT.getIndex()]; + long idle = ticks[TickType.IDLE.getIndex()] - prevTicks[TickType.IDLE.getIndex()]; + long totalCpu = user + nice + cSys + idle + iowait + irq + softirq + steal; + cpu.setCpuNum(processor.getLogicalProcessorCount()); + cpu.setTotal(totalCpu); + cpu.setSys(cSys); + cpu.setUsed(user); + cpu.setWait(iowait); + cpu.setFree(idle); + } + + /** + * 设置内存信息 + */ + private void setMemInfo(GlobalMemory memory) + { + mem.setTotal(memory.getTotal()); + mem.setUsed(memory.getTotal() - memory.getAvailable()); + mem.setFree(memory.getAvailable()); + } + + /** + * 设置服务器信息 + */ + private void setSysInfo() + { + Properties props = System.getProperties(); + sys.setComputerName(IpUtils.getHostName()); + sys.setComputerIp(IpUtils.getHostIp()); + sys.setOsName(props.getProperty("os.name")); + sys.setOsArch(props.getProperty("os.arch")); + sys.setUserDir(props.getProperty("user.dir")); + } + + /** + * 设置Java虚拟机 + */ + private void setJvmInfo() throws UnknownHostException + { + Properties props = System.getProperties(); + jvm.setTotal(Runtime.getRuntime().totalMemory()); + jvm.setMax(Runtime.getRuntime().maxMemory()); + jvm.setFree(Runtime.getRuntime().freeMemory()); + jvm.setVersion(props.getProperty("java.version")); + jvm.setHome(props.getProperty("java.home")); + } + + /** + * 设置磁盘信息 + */ + private void setSysFiles(OperatingSystem os) + { + FileSystem fileSystem = os.getFileSystem(); + List fsArray = fileSystem.getFileStores(); + for (OSFileStore fs : fsArray) + { + long free = fs.getUsableSpace(); + long total = fs.getTotalSpace(); + long used = total - free; + SysFile sysFile = new SysFile(); + sysFile.setDirName(fs.getMount()); + sysFile.setSysTypeName(fs.getType()); + sysFile.setTypeName(fs.getName()); + sysFile.setTotal(convertFileSize(total)); + sysFile.setFree(convertFileSize(free)); + sysFile.setUsed(convertFileSize(used)); + sysFile.setUsage(Arith.mul(Arith.div(used, total, 4), 100)); + sysFiles.add(sysFile); + } + } + + /** + * 字节转换 + * + * @param size 字节大小 + * @return 转换后值 + */ + public String convertFileSize(long size) + { + long kb = 1024; + long mb = kb * 1024; + long gb = mb * 1024; + if (size >= gb) + { + return String.format("%.1f GB", (float) size / gb); + } + else if (size >= mb) + { + float f = (float) size / mb; + return String.format(f > 100 ? "%.0f MB" : "%.1f MB", f); + } + else if (size >= kb) + { + float f = (float) size / kb; + return String.format(f > 100 ? "%.0f KB" : "%.1f KB", f); + } + else + { + return String.format("%d B", size); + } + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/Cpu.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/Cpu.java new file mode 100644 index 00000000..a13a66cf --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/Cpu.java @@ -0,0 +1,101 @@ +package com.ruoyi.framework.web.domain.server; + +import com.ruoyi.common.utils.Arith; + +/** + * CPU相关信息 + * + * @author ruoyi + */ +public class Cpu +{ + /** + * 核心数 + */ + private int cpuNum; + + /** + * CPU总的使用率 + */ + private double total; + + /** + * CPU系统使用率 + */ + private double sys; + + /** + * CPU用户使用率 + */ + private double used; + + /** + * CPU当前等待率 + */ + private double wait; + + /** + * CPU当前空闲率 + */ + private double free; + + public int getCpuNum() + { + return cpuNum; + } + + public void setCpuNum(int cpuNum) + { + this.cpuNum = cpuNum; + } + + public double getTotal() + { + return Arith.round(Arith.mul(total, 100), 2); + } + + public void setTotal(double total) + { + this.total = total; + } + + public double getSys() + { + return Arith.round(Arith.mul(sys / total, 100), 2); + } + + public void setSys(double sys) + { + this.sys = sys; + } + + public double getUsed() + { + return Arith.round(Arith.mul(used / total, 100), 2); + } + + public void setUsed(double used) + { + this.used = used; + } + + public double getWait() + { + return Arith.round(Arith.mul(wait / total, 100), 2); + } + + public void setWait(double wait) + { + this.wait = wait; + } + + public double getFree() + { + return Arith.round(Arith.mul(free / total, 100), 2); + } + + public void setFree(double free) + { + this.free = free; + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/Jvm.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/Jvm.java new file mode 100644 index 00000000..444b280d --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/Jvm.java @@ -0,0 +1,130 @@ +package com.ruoyi.framework.web.domain.server; + +import java.lang.management.ManagementFactory; +import com.ruoyi.common.utils.Arith; +import com.ruoyi.common.utils.DateUtils; + +/** + * JVM相关信息 + * + * @author ruoyi + */ +public class Jvm +{ + /** + * 当前JVM占用的内存总数(M) + */ + private double total; + + /** + * JVM最大可用内存总数(M) + */ + private double max; + + /** + * JVM空闲内存(M) + */ + private double free; + + /** + * JDK版本 + */ + private String version; + + /** + * JDK路径 + */ + private String home; + + public double getTotal() + { + return Arith.div(total, (1024 * 1024), 2); + } + + public void setTotal(double total) + { + this.total = total; + } + + public double getMax() + { + return Arith.div(max, (1024 * 1024), 2); + } + + public void setMax(double max) + { + this.max = max; + } + + public double getFree() + { + return Arith.div(free, (1024 * 1024), 2); + } + + public void setFree(double free) + { + this.free = free; + } + + public double getUsed() + { + return Arith.div(total - free, (1024 * 1024), 2); + } + + public double getUsage() + { + return Arith.mul(Arith.div(total - free, total, 4), 100); + } + + /** + * 获取JDK名称 + */ + public String getName() + { + return ManagementFactory.getRuntimeMXBean().getVmName(); + } + + public String getVersion() + { + return version; + } + + public void setVersion(String version) + { + this.version = version; + } + + public String getHome() + { + return home; + } + + public void setHome(String home) + { + this.home = home; + } + + /** + * JDK启动时间 + */ + public String getStartTime() + { + return DateUtils.parseDateToStr(DateUtils.YYYY_MM_DD_HH_MM_SS, DateUtils.getServerStartDate()); + } + + /** + * JDK运行时间 + */ + public String getRunTime() + { + return DateUtils.getDatePoor(DateUtils.getNowDate(), DateUtils.getServerStartDate()); + } + + /** + * 运行参数 + */ + public String getInputArgs() + { + return ManagementFactory.getRuntimeMXBean().getInputArguments().toString(); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/Mem.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/Mem.java new file mode 100644 index 00000000..13eec521 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/Mem.java @@ -0,0 +1,61 @@ +package com.ruoyi.framework.web.domain.server; + +import com.ruoyi.common.utils.Arith; + +/** + * 內存相关信息 + * + * @author ruoyi + */ +public class Mem +{ + /** + * 内存总量 + */ + private double total; + + /** + * 已用内存 + */ + private double used; + + /** + * 剩余内存 + */ + private double free; + + public double getTotal() + { + return Arith.div(total, (1024 * 1024 * 1024), 2); + } + + public void setTotal(long total) + { + this.total = total; + } + + public double getUsed() + { + return Arith.div(used, (1024 * 1024 * 1024), 2); + } + + public void setUsed(long used) + { + this.used = used; + } + + public double getFree() + { + return Arith.div(free, (1024 * 1024 * 1024), 2); + } + + public void setFree(long free) + { + this.free = free; + } + + public double getUsage() + { + return Arith.mul(Arith.div(used, total, 4), 100); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/Sys.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/Sys.java new file mode 100644 index 00000000..45d64d9c --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/Sys.java @@ -0,0 +1,84 @@ +package com.ruoyi.framework.web.domain.server; + +/** + * 系统相关信息 + * + * @author ruoyi + */ +public class Sys +{ + /** + * 服务器名称 + */ + private String computerName; + + /** + * 服务器Ip + */ + private String computerIp; + + /** + * 项目路径 + */ + private String userDir; + + /** + * 操作系统 + */ + private String osName; + + /** + * 系统架构 + */ + private String osArch; + + public String getComputerName() + { + return computerName; + } + + public void setComputerName(String computerName) + { + this.computerName = computerName; + } + + public String getComputerIp() + { + return computerIp; + } + + public void setComputerIp(String computerIp) + { + this.computerIp = computerIp; + } + + public String getUserDir() + { + return userDir; + } + + public void setUserDir(String userDir) + { + this.userDir = userDir; + } + + public String getOsName() + { + return osName; + } + + public void setOsName(String osName) + { + this.osName = osName; + } + + public String getOsArch() + { + return osArch; + } + + public void setOsArch(String osArch) + { + this.osArch = osArch; + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/SysFile.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/SysFile.java new file mode 100644 index 00000000..1320cde6 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/domain/server/SysFile.java @@ -0,0 +1,114 @@ +package com.ruoyi.framework.web.domain.server; + +/** + * 系统文件相关信息 + * + * @author ruoyi + */ +public class SysFile +{ + /** + * 盘符路径 + */ + private String dirName; + + /** + * 盘符类型 + */ + private String sysTypeName; + + /** + * 文件类型 + */ + private String typeName; + + /** + * 总大小 + */ + private String total; + + /** + * 剩余大小 + */ + private String free; + + /** + * 已经使用量 + */ + private String used; + + /** + * 资源的使用率 + */ + private double usage; + + public String getDirName() + { + return dirName; + } + + public void setDirName(String dirName) + { + this.dirName = dirName; + } + + public String getSysTypeName() + { + return sysTypeName; + } + + public void setSysTypeName(String sysTypeName) + { + this.sysTypeName = sysTypeName; + } + + public String getTypeName() + { + return typeName; + } + + public void setTypeName(String typeName) + { + this.typeName = typeName; + } + + public String getTotal() + { + return total; + } + + public void setTotal(String total) + { + this.total = total; + } + + public String getFree() + { + return free; + } + + public void setFree(String free) + { + this.free = free; + } + + public String getUsed() + { + return used; + } + + public void setUsed(String used) + { + this.used = used; + } + + public double getUsage() + { + return usage; + } + + public void setUsage(double usage) + { + this.usage = usage; + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/exception/GlobalExceptionHandler.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/exception/GlobalExceptionHandler.java new file mode 100644 index 00000000..51dd8c54 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/exception/GlobalExceptionHandler.java @@ -0,0 +1,114 @@ +package com.ruoyi.framework.web.exception; + +import javax.servlet.http.HttpServletRequest; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.security.access.AccessDeniedException; +import org.springframework.validation.BindException; +import org.springframework.web.HttpRequestMethodNotSupportedException; +import org.springframework.web.bind.MethodArgumentNotValidException; +import org.springframework.web.bind.annotation.ExceptionHandler; +import org.springframework.web.bind.annotation.RestControllerAdvice; +import com.ruoyi.common.constant.HttpStatus; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.exception.DemoModeException; +import com.ruoyi.common.exception.ServiceException; +import com.ruoyi.common.utils.StringUtils; + +/** + * 全局异常处理器 + * + * @author ruoyi + */ +@RestControllerAdvice +public class GlobalExceptionHandler +{ + private static final Logger log = LoggerFactory.getLogger(GlobalExceptionHandler.class); + + /** + * 权限校验异常 + */ + @ExceptionHandler(AccessDeniedException.class) + public AjaxResult handleAccessDeniedException(AccessDeniedException e, HttpServletRequest request) + { + String requestURI = request.getRequestURI(); + log.error("请求地址'{}',权限校验失败'{}'", requestURI, e.getMessage()); + return AjaxResult.error(HttpStatus.FORBIDDEN, "没有权限,请联系管理员授权"); + } + + /** + * 请求方式不支持 + */ + @ExceptionHandler(HttpRequestMethodNotSupportedException.class) + public AjaxResult handleHttpRequestMethodNotSupported(HttpRequestMethodNotSupportedException e, + HttpServletRequest request) + { + String requestURI = request.getRequestURI(); + log.error("请求地址'{}',不支持'{}'请求", requestURI, e.getMethod()); + return AjaxResult.error(e.getMessage()); + } + + /** + * 业务异常 + */ + @ExceptionHandler(ServiceException.class) + public AjaxResult handleServiceException(ServiceException e, HttpServletRequest request) + { + log.error(e.getMessage(), e); + Integer code = e.getCode(); + return StringUtils.isNotNull(code) ? AjaxResult.error(code, e.getMessage()) : AjaxResult.error(e.getMessage()); + } + + /** + * 拦截未知的运行时异常 + */ + @ExceptionHandler(RuntimeException.class) + public AjaxResult handleRuntimeException(RuntimeException e, HttpServletRequest request) + { + String requestURI = request.getRequestURI(); + log.error("请求地址'{}',发生未知异常.", requestURI, e); + return AjaxResult.error(e.getMessage()); + } + + /** + * 系统异常 + */ + @ExceptionHandler(Exception.class) + public AjaxResult handleException(Exception e, HttpServletRequest request) + { + String requestURI = request.getRequestURI(); + log.error("请求地址'{}',发生系统异常.", requestURI, e); + return AjaxResult.error(e.getMessage()); + } + + /** + * 自定义验证异常 + */ + @ExceptionHandler(BindException.class) + public AjaxResult handleBindException(BindException e) + { + log.error(e.getMessage(), e); + String message = e.getAllErrors().get(0).getDefaultMessage(); + return AjaxResult.error(message); + } + + /** + * 自定义验证异常 + */ + @ExceptionHandler(MethodArgumentNotValidException.class) + public Object handleMethodArgumentNotValidException(MethodArgumentNotValidException e) + { + log.error(e.getMessage(), e); + String message = e.getBindingResult().getFieldError().getDefaultMessage(); + return AjaxResult.error(message); + } + + /** + * 演示模式异常 + */ + @ExceptionHandler(DemoModeException.class) + public AjaxResult handleDemoModeException(DemoModeException e) + { + return AjaxResult.error("演示模式,不允许操作"); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/DataScopeService.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/DataScopeService.java new file mode 100644 index 00000000..88b9050d --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/DataScopeService.java @@ -0,0 +1,163 @@ +package com.ruoyi.framework.web.service; + +import cn.hutool.core.collection.CollectionUtil; +import cn.hutool.core.util.ObjectUtil; +import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; +import com.ruoyi.common.core.domain.entity.SysDept; +import com.ruoyi.common.core.domain.entity.SysRole; +import com.ruoyi.common.enums.CommonStatusEnum; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.framework.aspectj.DataScopeAspect; +import com.ruoyi.system.domain.SysRoleDept; +import com.ruoyi.system.domain.SysUserRole; +import com.ruoyi.system.mapper.SysRoleDeptMapper; +import com.ruoyi.system.mapper.SysUserRoleMapper; +import com.ruoyi.system.service.ISysDeptService; +import com.ruoyi.system.service.ISysRoleService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Component; + +import java.util.List; +import java.util.Set; +import java.util.stream.Collectors; + +/** + * 数据权限范围操作类 + * @author wangzongrun + */ + +@Component +public class DataScopeService { + + @Autowired + private SysUserRoleMapper userRoleMapper; + + @Autowired + private SysRoleDeptMapper sysRoleDeptMapper; + + @Autowired + private ISysRoleService sysRoleService; + + @Autowired + private ISysDeptService sysDeptService; + + /** + * 根据用户id集合获取数据范围id集合 + * + * @param userId 用户id集合 + * @param orgId 机构id + * @return 数据范围id集合 + * @author wangzongrun + */ + public List getUserDataScopeIdList(Long userId, Long orgId){ + List roleIdList = CollectionUtil.newArrayList(); + // 获取用户所有角色 + LambdaQueryWrapper queryWrapper = new LambdaQueryWrapper<>(); + queryWrapper.eq(SysUserRole::getUserId, userId); + userRoleMapper.selectList(queryWrapper).forEach(sysUserRole -> roleIdList.add(sysUserRole.getRoleId())); + // 获取这些角色对应的数据范围 + if (ObjectUtil.isNotEmpty(roleIdList)) { + return getUserDataScopeIdList(roleIdList, orgId); + } + return CollectionUtil.newArrayList(); + } + + /** + * 根据角色id集合获取数据范围id集合 + * + * @param roleIdList 角色id集合 + * @param orgId 机构id + * @return 数据范围id集合 + */ + public List getUserDataScopeIdList(List roleIdList, Long orgId) { + Set resultList = CollectionUtil.newHashSet(); + + //定义角色中最大数据范围的类型,目前系统按最大范围策略来,如果你同时拥有ALL和SELF的权限,系统最后按ALL返回 + int strongerDataScopeType = Integer.parseInt(DataScopeAspect.DATA_SCOPE_SELF); + + //获取用户自定义数据范围的角色集合 + List customDataScopeRoleIdList = CollectionUtil.newArrayList(); + if (ObjectUtil.isNotEmpty(roleIdList)) { + List sysRoleList = sysRoleService.listByIds(roleIdList); + for (SysRole sysRole : sysRoleList) { + if (DataScopeAspect.DATA_SCOPE_CUSTOM.equals(sysRole.getDataScope())) { + customDataScopeRoleIdList.add(sysRole.getRoleId()); + } else { + if (Integer.parseInt(sysRole.getDataScope()) <= strongerDataScopeType) { + strongerDataScopeType = Integer.parseInt(sysRole.getDataScope()); + } + } + } + } + + //自定义数据范围的角色对应的数据范围 + List roleDataScopeIdList = getRoleDataScopeIdList(customDataScopeRoleIdList); + + //角色中拥有最大数据范围类型的数据范围 + List dataScopeIdList = getDataScopeListByDataScopeType(strongerDataScopeType, orgId); + + resultList.addAll(dataScopeIdList); + resultList.addAll(roleDataScopeIdList); + return CollectionUtil.newArrayList(resultList); + } + + /** + * 根据角色id获取角色数据范围集合 + * + * @param roleIdList 角色id集合 + * @return 数据范围id集合 + */ + List getRoleDataScopeIdList(List roleIdList){ + List resultList = CollectionUtil.newArrayList(); + if (ObjectUtil.isNotEmpty(roleIdList)) { + LambdaQueryWrapper queryWrapper = new LambdaQueryWrapper<>(); + queryWrapper.in(SysRoleDept::getRoleId, roleIdList); + sysRoleDeptMapper.selectList(queryWrapper).forEach(sysRoleDataScope -> resultList.add(sysRoleDataScope.getDeptId())); + } + return resultList; + } + + /** + * 根据数据范围类型获取当前登录用户的数据范围id集合 + * + * @param dataScopeType 数据范围类型(1全部数据 2本部门及以下数据 3本部门数据 4仅本人数据) + * @param orgId 组织机构id + * @return 数据范围id集合 + */ + public List getDataScopeListByDataScopeType(Integer dataScopeType, Long orgId) { + List resultList = CollectionUtil.newArrayList(); + + if (ObjectUtil.isEmpty(orgId)) { + return CollectionUtil.newArrayList(); + } + + // 如果是范围类型是全部数据,则获取当前系统所有的组织架构id + switch (dataScopeType.toString()) { + case DataScopeAspect.DATA_SCOPE_ALL: { + LambdaQueryWrapper queryWrapper = new LambdaQueryWrapper<>(); + queryWrapper.eq(SysDept::getStatus, CommonStatusEnum.ENABLE.getStatus()); + List list = sysDeptService.list(queryWrapper); + resultList = list.stream().map(SysDept::getDeptId).collect(Collectors.toList()); + break; + } + // 如果范围类型是本部门及以下部门,则查询本节点和子节点集合,包含本节点 + case DataScopeAspect.DATA_SCOPE_DEPT_AND_CHILD: { + LambdaQueryWrapper queryWrapper = new LambdaQueryWrapper<>(); + queryWrapper.eq(SysDept::getStatus, CommonStatusEnum.ENABLE.getStatus()); + String format = StringUtils.format(" AND dept_id IN ( SELECT dept_id FROM sys_dept WHERE dept_id = {} or find_in_set( {} , ancestors ) )", orgId, orgId); + queryWrapper.last(format); + List list = sysDeptService.list(queryWrapper); + resultList = list.stream().map(SysDept::getDeptId).collect(Collectors.toList()); + break; + } + // 如果数据范围是本部门,不含子节点,则直接返回本部门 + case DataScopeAspect.DATA_SCOPE_SELF: + resultList.add(orgId); + break; + default: + break; + } + return resultList; + } + +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/JimuReportTokenService.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/JimuReportTokenService.java new file mode 100644 index 00000000..0f2a57ad --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/JimuReportTokenService.java @@ -0,0 +1,47 @@ +package com.ruoyi.framework.web.service; + + +import cn.hutool.core.bean.BeanUtil; +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.common.utils.StringUtils; +import org.jeecg.modules.jmreport.api.JmReportTokenServiceI; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Component; + +import javax.servlet.http.HttpServletRequest; +import java.util.Map; + +/** + * 积木报表权限认证 + */ +@Component +public class JimuReportTokenService implements JmReportTokenServiceI { + @Autowired + TokenService tokenService; + + @Override + public String getUsername(String token) { + LoginUser loginUser = tokenService.getLoginUser(token); + return loginUser.getUsername(); + } + + @Override + public Boolean verifyToken(String token) { + LoginUser loginUser = tokenService.getLoginUser(token); + if (StringUtils.isNotNull(loginUser) && (loginUser.getTenantId() == 1 && "admin".equals(loginUser.getUsername()))) { + tokenService.verifyToken(loginUser); + return true; + } + return false; + } + + @Override + public String getToken(HttpServletRequest request) { + return tokenService.getToken(request); + } + + @Override + public Map getUserInfo(String token) { + return BeanUtil.beanToMap(tokenService.getLoginUser(token)); + } +} \ No newline at end of file diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/PermissionService.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/PermissionService.java new file mode 100644 index 00000000..dcd3e469 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/PermissionService.java @@ -0,0 +1,169 @@ +package com.ruoyi.framework.web.service; + +import java.util.Set; + +import com.ruoyi.framework.security.context.PermissionContextHolder; +import org.springframework.stereotype.Service; +import org.springframework.util.CollectionUtils; +import com.ruoyi.common.core.domain.entity.SysRole; +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.common.utils.StringUtils; + +/** + * RuoYi首创 自定义权限实现,ss取自SpringSecurity首字母 + * + * @author ruoyi + */ +@Service("ss") +public class PermissionService +{ + /** 所有权限标识 */ + private static final String ALL_PERMISSION = "*:*:*"; + + /** 管理员角色权限标识 */ + private static final String SUPER_ADMIN = "admin"; + + private static final String ROLE_DELIMETER = ","; + + private static final String PERMISSION_DELIMETER = ","; + + /** + * 验证用户是否具备某权限 + * + * @param permission 权限字符串 + * @return 用户是否具备某权限 + */ + public boolean hasPermi(String permission) + { + if (StringUtils.isEmpty(permission)) + { + return false; + } + LoginUser loginUser = SecurityUtils.getLoginUser(); + if (StringUtils.isNull(loginUser) || CollectionUtils.isEmpty(loginUser.getPermissions())) + { + return false; + } + PermissionContextHolder.setContext(permission); + return hasPermissions(loginUser.getPermissions(), permission); + } + + /** + * 验证用户是否不具备某权限,与 hasPermi逻辑相反 + * + * @param permission 权限字符串 + * @return 用户是否不具备某权限 + */ + public boolean lacksPermi(String permission) + { + return hasPermi(permission) != true; + } + + /** + * 验证用户是否具有以下任意一个权限 + * + * @param permissions 以 PERMISSION_NAMES_DELIMETER 为分隔符的权限列表 + * @return 用户是否具有以下任意一个权限 + */ + public boolean hasAnyPermi(String permissions) + { + if (StringUtils.isEmpty(permissions)) + { + return false; + } + LoginUser loginUser = SecurityUtils.getLoginUser(); + if (StringUtils.isNull(loginUser) || CollectionUtils.isEmpty(loginUser.getPermissions())) + { + return false; + } + PermissionContextHolder.setContext(permissions); + Set authorities = loginUser.getPermissions(); + for (String permission : permissions.split(PERMISSION_DELIMETER)) + { + if (permission != null && hasPermissions(authorities, permission)) + { + return true; + } + } + return false; + } + + /** + * 判断用户是否拥有某个角色 + * + * @param role 角色字符串 + * @return 用户是否具备某角色 + */ + public boolean hasRole(String role) + { + if (StringUtils.isEmpty(role)) + { + return false; + } + LoginUser loginUser = SecurityUtils.getLoginUser(); + if (StringUtils.isNull(loginUser) || CollectionUtils.isEmpty(loginUser.getUser().getRoles())) + { + return false; + } + for (SysRole sysRole : loginUser.getUser().getRoles()) + { + String roleKey = sysRole.getRoleKey(); + if (SUPER_ADMIN.equals(roleKey) || roleKey.equals(StringUtils.trim(role))) + { + return true; + } + } + return false; + } + + /** + * 验证用户是否不具备某角色,与 isRole逻辑相反。 + * + * @param role 角色名称 + * @return 用户是否不具备某角色 + */ + public boolean lacksRole(String role) + { + return hasRole(role) != true; + } + + /** + * 验证用户是否具有以下任意一个角色 + * + * @param roles 以 ROLE_NAMES_DELIMETER 为分隔符的角色列表 + * @return 用户是否具有以下任意一个角色 + */ + public boolean hasAnyRoles(String roles) + { + if (StringUtils.isEmpty(roles)) + { + return false; + } + LoginUser loginUser = SecurityUtils.getLoginUser(); + if (StringUtils.isNull(loginUser) || CollectionUtils.isEmpty(loginUser.getUser().getRoles())) + { + return false; + } + for (String role : roles.split(ROLE_DELIMETER)) + { + if (hasRole(role)) + { + return true; + } + } + return false; + } + + /** + * 判断是否包含权限 + * + * @param permissions 权限列表 + * @param permission 权限字符串 + * @return 用户是否具备某权限 + */ + private boolean hasPermissions(Set permissions, String permission) + { + return permissions.contains(ALL_PERMISSION) || permissions.contains(StringUtils.trim(permission)); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/SysLoginService.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/SysLoginService.java new file mode 100644 index 00000000..07a523ee --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/SysLoginService.java @@ -0,0 +1,136 @@ +package com.ruoyi.framework.web.service; + +import javax.annotation.Resource; + +import com.ruoyi.common.utils.*; +import com.ruoyi.framework.security.context.AuthenticationContextHolder; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.authentication.AuthenticationManager; +import org.springframework.security.authentication.BadCredentialsException; +import org.springframework.security.authentication.UsernamePasswordAuthenticationToken; +import org.springframework.security.core.Authentication; +import org.springframework.stereotype.Component; +import com.ruoyi.common.constant.CacheConstants; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.common.exception.ServiceException; +import com.ruoyi.common.exception.user.CaptchaException; +import com.ruoyi.common.exception.user.CaptchaExpireException; +import com.ruoyi.common.exception.user.UserPasswordNotMatchException; +import com.ruoyi.common.utils.ip.IpUtils; +import com.ruoyi.framework.manager.AsyncManager; +import com.ruoyi.framework.manager.factory.AsyncFactory; +import com.ruoyi.system.service.ISysConfigService; +import com.ruoyi.system.service.ISysUserService; + +/** + * 登录校验方法 + * + * @author ruoyi + */ +@Component +public class SysLoginService +{ + @Autowired + private TokenService tokenService; + + @Resource + private AuthenticationManager authenticationManager; + + @Autowired + private ISysUserService userService; + + @Autowired + private ISysConfigService configService; + + /** + * 登录验证 + * + * @param username 用户名 + * @param password 密码 + * @param code 验证码 + * @param uuid 唯一标识 + * @return 结果 + */ + public String login(String username, String password, String code, String uuid) + { + boolean captchaEnabled = configService.selectCaptchaEnabled(); + // 验证码开关 + if (captchaEnabled) + { + validateCaptcha(username, code, uuid); + } + // 用户验证 + Authentication authentication = null; + try + { + UsernamePasswordAuthenticationToken authenticationToken = new UsernamePasswordAuthenticationToken(username, password); + AuthenticationContextHolder.setContext(authenticationToken); + // 该方法会去调用UserDetailsServiceImpl.loadUserByUsername + authentication = authenticationManager.authenticate(authenticationToken); + } + catch (Exception e) + { + if (e instanceof BadCredentialsException) + { + AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_FAIL, MessageUtils.message("user.password.not.match"))); + throw new UserPasswordNotMatchException(); + } + else + { + AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_FAIL, e.getMessage())); + throw new ServiceException(e.getMessage()); + } + } + finally + { + AuthenticationContextHolder.clearContext(); + } + AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_SUCCESS, MessageUtils.message("user.login.success"))); + LoginUser loginUser = (LoginUser) authentication.getPrincipal(); + recordLoginInfo(loginUser.getUserId()); + // 生成token + return tokenService.createToken(loginUser); + } + + /** + * 校验验证码 + * + * @param username 用户名 + * @param code 验证码 + * @param uuid 唯一标识 + * @return 结果 + */ + public void validateCaptcha(String username, String code, String uuid) + { + String verifyKey = CacheConstants.CAPTCHA_CODE_KEY + StringUtils.nvl(uuid, ""); + Object captcha = EhcacheUtil.get("verify", verifyKey); +// redisCache.deleteObject(verifyKey); + if (captcha == null) + { + AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_FAIL, MessageUtils.message("user.jcaptcha.expire"))); + throw new CaptchaExpireException(); + } + EhcacheUtil.remove("verify", verifyKey); + if (!code.equalsIgnoreCase(captcha.toString())) + { + AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_FAIL, MessageUtils.message("user.jcaptcha.error"))); + throw new CaptchaException(); + } + } + + /** + * 记录登录信息 + * + * @param userId 用户ID + */ + public void recordLoginInfo(Long userId) + { + SysUser sysUser = new SysUser(); + sysUser.setUserId(userId); + sysUser.setLoginIp(IpUtils.getIpAddr(ServletUtils.getRequest())); + sysUser.setLoginDate(DateUtils.getNowDate()); + userService.updateUserProfile(sysUser); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/SysPasswordService.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/SysPasswordService.java new file mode 100644 index 00000000..c01f6e6a --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/SysPasswordService.java @@ -0,0 +1,98 @@ +package com.ruoyi.framework.web.service; + +import java.util.concurrent.TimeUnit; + +import com.ruoyi.common.utils.EhcacheUtil; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.security.core.Authentication; +import org.springframework.stereotype.Component; +import com.ruoyi.common.constant.CacheConstants; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.exception.user.UserPasswordNotMatchException; +import com.ruoyi.common.exception.user.UserPasswordRetryLimitExceedException; +import com.ruoyi.common.utils.MessageUtils; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.framework.manager.AsyncManager; +import com.ruoyi.framework.manager.factory.AsyncFactory; +import com.ruoyi.framework.security.context.AuthenticationContextHolder; + +/** + * 登录密码方法 + * + * @author ruoyi + */ +@Component +public class SysPasswordService +{ +// @Autowired +// private RedisCache redisCache; + + @Value(value = "${user.password.maxRetryCount}") + private int maxRetryCount; + + @Value(value = "${user.password.lockTime}") + private int lockTime; + + /** + * 登录账户密码错误次数缓存键名 + * + * @param username 用户名 + * @return 缓存键key + */ + private String getCacheKey(String username) + { + return CacheConstants.PWD_ERR_CNT_KEY + username; + } + + public void validate(SysUser user) + { + Authentication usernamePasswordAuthenticationToken = AuthenticationContextHolder.getContext(); + String username = usernamePasswordAuthenticationToken.getName(); + String password = usernamePasswordAuthenticationToken.getCredentials().toString(); + + Object obj = EhcacheUtil.get("lock", getCacheKey(username)); + Integer retryCount = 0; + if (obj != null) + { + retryCount = Integer.valueOf(obj.toString()); + } + + if (retryCount >= Integer.valueOf(maxRetryCount).intValue()) + { + AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_FAIL, + MessageUtils.message("user.password.retry.limit.exceed", maxRetryCount, lockTime))); + throw new UserPasswordRetryLimitExceedException(maxRetryCount, lockTime); + } + + if (!matches(user, password)) + { + retryCount = retryCount + 1; + AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_FAIL, + MessageUtils.message("user.password.retry.limit.count", retryCount))); + EhcacheUtil.put("lock", getCacheKey(username), retryCount); +// redisCache.setCacheObject(getCacheKey(username), retryCount, lockTime, TimeUnit.MINUTES); + throw new UserPasswordNotMatchException(); + } + else + { + clearLoginRecordCache(username); + } + } + + public boolean matches(SysUser user, String rawPassword) + { + return SecurityUtils.matchesPassword(rawPassword, user.getPassword()); + } + + public void clearLoginRecordCache(String loginName) + { + Object lock = EhcacheUtil.get("lock", getCacheKey(loginName)); + if (lock!= null) + { + EhcacheUtil.remove("lock", getCacheKey(loginName)); +// redisCache.deleteObject(getCacheKey(loginName)); + } + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/SysPermissionService.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/SysPermissionService.java new file mode 100644 index 00000000..8fb4866f --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/SysPermissionService.java @@ -0,0 +1,83 @@ +package com.ruoyi.framework.web.service; + +import java.util.HashSet; +import java.util.List; +import java.util.Set; + +import com.ruoyi.common.core.domain.entity.SysRole; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Component; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.system.service.ISysMenuService; +import com.ruoyi.system.service.ISysRoleService; + +/** + * 用户权限处理 + * + * @author ruoyi + */ +@Component +public class SysPermissionService +{ + @Autowired + private ISysRoleService roleService; + + @Autowired + private ISysMenuService menuService; + + /** + * 获取角色数据权限 + * + * @param user 用户信息 + * @return 角色权限信息 + */ + public Set getRolePermission(SysUser user) + { + Set roles = new HashSet(); + // 管理员拥有所有权限 + if (user.isAdmin()) + { + roles.add("admin"); + } + else + { + roles.addAll(roleService.selectRolePermissionByUserId(user.getUserId())); + } + return roles; + } + + /** + * 获取菜单数据权限 + * + * @param user 用户信息 + * @return 菜单权限信息 + */ + public Set getMenuPermission(SysUser user) + { + Set perms = new HashSet(); + // 管理员拥有所有权限 + if (user.isAdmin()) + { + perms.add("*:*:*"); + } + else + { + List roles = user.getRoles(); + if (!roles.isEmpty() && roles.size() > 1) + { + // 多角色设置permissions属性,以便数据权限匹配权限 + for (SysRole role : roles) + { + Set rolePerms = menuService.selectMenuPermsByRoleId(role.getRoleId()); + role.setPermissions(rolePerms); + perms.addAll(rolePerms); + } + } + else + { + perms.addAll(menuService.selectMenuPermsByUserId(user.getUserId())); + } + } + return perms; + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/SysRegisterService.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/SysRegisterService.java new file mode 100644 index 00000000..0736204d --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/SysRegisterService.java @@ -0,0 +1,95 @@ +package com.ruoyi.framework.web.service; + +import com.ruoyi.common.constant.CacheConstants; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.core.domain.model.RegisterBody; +import com.ruoyi.common.exception.user.CaptchaException; +import com.ruoyi.common.exception.user.CaptchaExpireException; +import com.ruoyi.common.utils.EhcacheUtil; +import com.ruoyi.common.utils.MessageUtils; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.framework.manager.AsyncManager; +import com.ruoyi.framework.manager.factory.AsyncFactory; +import com.ruoyi.system.service.ISysConfigService; +import com.ruoyi.system.service.ISysUserService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Component; + +/** + * 注册校验方法 + * + * @author ruoyi + */ +@Component +public class SysRegisterService { + @Autowired + private ISysUserService userService; + + @Autowired + private ISysConfigService configService; + +// @Autowired +// private RedisCache redisCache; + + /** + * 注册 + */ + public String register(RegisterBody registerBody) { + String msg = "", username = registerBody.getUsername(), password = registerBody.getPassword(); + SysUser sysUser = new SysUser(); + sysUser.setUserName(username); + boolean captchaEnabled = configService.selectCaptchaEnabled(); + // 验证码开关 + if (captchaEnabled) { + validateCaptcha(username, registerBody.getCode(), registerBody.getUuid()); + } + + if (StringUtils.isEmpty(username)) { + msg = "用户名不能为空"; + } else if (StringUtils.isEmpty(password)) { + msg = "用户密码不能为空"; + } else if (username.length() < UserConstants.USERNAME_MIN_LENGTH + || username.length() > UserConstants.USERNAME_MAX_LENGTH) { + msg = "账户长度必须在2到20个字符之间"; + } else if (password.length() < UserConstants.PASSWORD_MIN_LENGTH + || password.length() > UserConstants.PASSWORD_MAX_LENGTH) { + msg = "密码长度必须在5到20个字符之间"; + } else if (UserConstants.NOT_UNIQUE.equals(userService.checkUserNameUnique(sysUser))) { + msg = "保存用户'" + username + "'失败,注册账号已存在"; + } else { + sysUser.setNickName(username); + sysUser.setPassword(SecurityUtils.encryptPassword(password)); + boolean regFlag = userService.registerUser(sysUser); + if (!regFlag) { + msg = "注册失败,请联系系统管理人员"; + } else { + AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.REGISTER, MessageUtils.message("user.register.success"))); + } + } + return msg; + } + + /** + * 校验验证码 + * + * @param username 用户名 + * @param code 验证码 + * @param uuid 唯一标识 + * @return 结果 + */ + public void validateCaptcha(String username, String code, String uuid) { + String verifyKey = CacheConstants.CAPTCHA_CODE_KEY + StringUtils.nvl(uuid, ""); + Object obj = EhcacheUtil.get("verify", verifyKey); +// String captcha = redisCache.getCacheObject(verifyKey); +// redisCache.deleteObject(verifyKey); + if (obj == null) { + throw new CaptchaExpireException(); + } + if (!code.equalsIgnoreCase(obj.toString())) { + throw new CaptchaException(); + } + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/TokenService.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/TokenService.java new file mode 100644 index 00000000..1bedfbf3 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/TokenService.java @@ -0,0 +1,264 @@ +package com.ruoyi.framework.web.service; + +import java.util.HashMap; +import java.util.Map; +import java.util.concurrent.TimeUnit; +import javax.servlet.http.HttpServletRequest; + +import cn.hutool.core.util.StrUtil; +import cn.hutool.json.JSONUtil; +import com.ruoyi.common.utils.EhcacheUtil; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.stereotype.Component; +import com.ruoyi.common.constant.CacheConstants; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.common.utils.ServletUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.ip.AddressUtils; +import com.ruoyi.common.utils.ip.IpUtils; +import com.ruoyi.common.utils.uuid.IdUtils; +import eu.bitwalker.useragentutils.UserAgent; +import io.jsonwebtoken.Claims; +import io.jsonwebtoken.Jwts; +import io.jsonwebtoken.SignatureAlgorithm; + +/** + * token验证处理 + * + * @author ruoyi + */ +@Component +public class TokenService +{ + // 令牌自定义标识 + @Value("${token.header}") + private String header; + + // 令牌秘钥 + @Value("${token.secret}") + private String secret; + + // 令牌有效期(默认30分钟) + @Value("${token.expireTime}") + private int expireTime; + + protected static final long MILLIS_SECOND = 1000; + + protected static final long MILLIS_MINUTE = 60 * MILLIS_SECOND; + + private static final Long MILLIS_MINUTE_TEN = 20 * 60 * 1000L; + +// @Autowired +// private RedisCache redisCache; + + /** + * 获取用户身份信息 + * + * @return 用户信息 + */ + public LoginUser getLoginUser(HttpServletRequest request) + { + // 获取请求携带的令牌 + String token = getToken(request); + if (StringUtils.isNotEmpty(token)) + { + try + { + Claims claims = parseToken(token); + // 解析对应的权限以及用户信息 + String uuid = (String) claims.get(Constants.LOGIN_USER_KEY); + String userKey = getTokenKey(uuid); + Object obj = EhcacheUtil.get("login", userKey); + if (obj == null) { + return null; + } + LoginUser user = JSONUtil.toBean(JSONUtil.toJsonStr(obj), LoginUser.class); +// LoginUser user = redisCache.getCacheObject(userKey); + return user; + } + catch (Exception e) + { + } + } + return null; + } + public LoginUser getLoginUser(String token) + { + // 获取请求携带的令牌 + if (StringUtils.isNotEmpty(token)) + { + try + { + Claims claims = parseToken(token); + // 解析对应的权限以及用户信息 + String uuid = (String) claims.get(Constants.LOGIN_USER_KEY); + String userKey = getTokenKey(uuid); + Object obj = EhcacheUtil.get("login", userKey); + if (obj == null) { + return null; + } + LoginUser user = JSONUtil.toBean(JSONUtil.toJsonStr(obj), LoginUser.class); +// LoginUser user = redisCache.getCacheObject(userKey); + return user; + } + catch (Exception e) + { + } + } + return null; + } + /** + * 设置用户身份信息 + */ + public void setLoginUser(LoginUser loginUser) + { + if (StringUtils.isNotNull(loginUser) && StringUtils.isNotEmpty(loginUser.getToken())) + { + refreshToken(loginUser); + } + } + + /** + * 删除用户身份信息 + */ + public void delLoginUser(String token) + { + if (StringUtils.isNotEmpty(token)) + { + String userKey = getTokenKey(token); + EhcacheUtil.remove("login", userKey); +// redisCache.deleteObject(userKey); + } + } + + /** + * 创建令牌 + * + * @param loginUser 用户信息 + * @return 令牌 + */ + public String createToken(LoginUser loginUser) + { + String token = IdUtils.fastUUID(); + loginUser.setToken(token); + setUserAgent(loginUser); + refreshToken(loginUser); + + Map claims = new HashMap<>(); + claims.put(Constants.LOGIN_USER_KEY, token); + return createToken(claims); + } + + /** + * 验证令牌有效期,相差不足20分钟,自动刷新缓存 + * + * @param loginUser + * @return 令牌 + */ + public void verifyToken(LoginUser loginUser) + { + long expireTime = loginUser.getExpireTime(); + long currentTime = System.currentTimeMillis(); + if (expireTime - currentTime <= MILLIS_MINUTE_TEN) + { + refreshToken(loginUser); + } + } + + /** + * 刷新令牌有效期 + * + * @param loginUser 登录信息 + */ + public void refreshToken(LoginUser loginUser) + { + loginUser.setLoginTime(System.currentTimeMillis()); + loginUser.setExpireTime(loginUser.getLoginTime() + expireTime * MILLIS_MINUTE); + // 根据uuid将loginUser缓存 + String userKey = getTokenKey(loginUser.getToken()); + EhcacheUtil.put("login", userKey,loginUser); +// redisCache.setCacheObject(userKey, loginUser, expireTime, TimeUnit.MINUTES); + } + + /** + * 设置用户代理信息 + * + * @param loginUser 登录信息 + */ + public void setUserAgent(LoginUser loginUser) + { + UserAgent userAgent = UserAgent.parseUserAgentString(ServletUtils.getRequest().getHeader("User-Agent")); + String ip = IpUtils.getIpAddr(ServletUtils.getRequest()); + loginUser.setIpaddr(ip); + loginUser.setLoginLocation(AddressUtils.getRealAddressByIP(ip)); + loginUser.setBrowser(userAgent.getBrowser().getName()); + loginUser.setOs(userAgent.getOperatingSystem().getName()); + } + + /** + * 从数据声明生成令牌 + * + * @param claims 数据声明 + * @return 令牌 + */ + private String createToken(Map claims) + { + String token = Jwts.builder() + .setClaims(claims) + .signWith(SignatureAlgorithm.HS512, secret).compact(); + return token; + } + + /** + * 从令牌中获取数据声明 + * + * @param token 令牌 + * @return 数据声明 + */ + private Claims parseToken(String token) + { + return Jwts.parser() + .setSigningKey(secret) + .parseClaimsJws(token) + .getBody(); + } + + /** + * 从令牌中获取用户名 + * + * @param token 令牌 + * @return 用户名 + */ + public String getUsernameFromToken(String token) + { + Claims claims = parseToken(token); + return claims.getSubject(); + } + + /** + * 获取请求token + * + * @param request + * @return token + */ + public String getToken(HttpServletRequest request) { + String token = request.getParameter("token"); + if (StrUtil.isBlank(token)) { + token = request.getHeader("token"); + } + if (StrUtil.isBlank(token)) { + token = request.getHeader(header); + } + if (StringUtils.isNotEmpty(token) && token.startsWith(Constants.TOKEN_PREFIX)) { + token = token.replace(Constants.TOKEN_PREFIX, ""); + } + return token; + } + + private String getTokenKey(String uuid) + { + return CacheConstants.LOGIN_TOKEN_KEY + uuid; + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/UserDetailsServiceImpl.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/UserDetailsServiceImpl.java new file mode 100644 index 00000000..c9d8b1a9 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/UserDetailsServiceImpl.java @@ -0,0 +1,71 @@ +package com.ruoyi.framework.web.service; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.core.userdetails.UserDetails; +import org.springframework.security.core.userdetails.UserDetailsService; +import org.springframework.security.core.userdetails.UsernameNotFoundException; +import org.springframework.stereotype.Service; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.common.enums.UserStatus; +import com.ruoyi.common.exception.ServiceException; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.system.service.ISysUserService; + +import java.util.List; + +/** + * 用户验证处理 + * + * @author ruoyi + */ +@Service +public class UserDetailsServiceImpl implements UserDetailsService +{ + private static final Logger log = LoggerFactory.getLogger(UserDetailsServiceImpl.class); + + @Autowired + private ISysUserService userService; + + @Autowired + private SysPermissionService permissionService; + + @Autowired + private SysPasswordService passwordService; + + @Autowired + private DataScopeService dataScopeService; + + @Override + public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException + { + SysUser user = userService.selectUserByUserName(username); + if (StringUtils.isNull(user)) + { + log.info("登录用户:{} 不存在.", username); + throw new ServiceException("登录用户:" + username + " 不存在"); + } + else if (UserStatus.DELETED.getCode().equals(user.getDelFlag())) + { + log.info("登录用户:{} 已被删除.", username); + throw new ServiceException("对不起,您的账号:" + username + " 已被删除"); + } + else if (UserStatus.DISABLE.getCode().equals(user.getStatus())) + { + log.info("登录用户:{} 已被停用.", username); + throw new ServiceException("对不起,您的账号:" + username + " 已停用"); + } + passwordService.validate(user); + + return createLoginUser(user); + } + + public UserDetails createLoginUser(SysUser user) + { + //查询数据权限范围 + List userDataScopeIdList = dataScopeService.getUserDataScopeIdList(user.getUserId(), user.getDeptId()); + return new LoginUser(user.getUserId(), user.getDeptId(), user, permissionService.getMenuPermission(user),userDataScopeIdList,user.getTenantId()); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/websocket/SemaphoreUtils.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/websocket/SemaphoreUtils.java new file mode 100644 index 00000000..57567f30 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/websocket/SemaphoreUtils.java @@ -0,0 +1,58 @@ +package com.ruoyi.framework.websocket; + +import java.util.concurrent.Semaphore; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * 信号量相关处理 + * + * @author ruoyi + */ +public class SemaphoreUtils +{ + /** + * SemaphoreUtils 日志控制器 + */ + private static final Logger LOGGER = LoggerFactory.getLogger(SemaphoreUtils.class); + + /** + * 获取信号量 + * + * @param semaphore + * @return + */ + public static boolean tryAcquire(Semaphore semaphore) + { + boolean flag = false; + + try + { + flag = semaphore.tryAcquire(); + } + catch (Exception e) + { + LOGGER.error("获取信号量异常", e); + } + + return flag; + } + + /** + * 释放信号量 + * + * @param semaphore + */ + public static void release(Semaphore semaphore) + { + + try + { + semaphore.release(); + } + catch (Exception e) + { + LOGGER.error("释放信号量异常", e); + } + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/websocket/WebSocketConfig.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/websocket/WebSocketConfig.java new file mode 100644 index 00000000..be63f35f --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/websocket/WebSocketConfig.java @@ -0,0 +1,20 @@ +package com.ruoyi.framework.websocket; + +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.web.socket.server.standard.ServerEndpointExporter; + +/** + * websocket 配置 + * + * @author ruoyi + */ +@Configuration +public class WebSocketConfig +{ + @Bean + public ServerEndpointExporter serverEndpointExporter() + { + return new ServerEndpointExporter(); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/websocket/WebSocketServer.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/websocket/WebSocketServer.java new file mode 100644 index 00000000..49c0d165 --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/websocket/WebSocketServer.java @@ -0,0 +1,103 @@ +package com.ruoyi.framework.websocket; + +import java.util.concurrent.Semaphore; +import javax.websocket.OnClose; +import javax.websocket.OnError; +import javax.websocket.OnMessage; +import javax.websocket.OnOpen; +import javax.websocket.Session; +import javax.websocket.server.ServerEndpoint; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.stereotype.Component; + +/** + * websocket 消息处理 + * + * @author ruoyi + */ +@Component +@ServerEndpoint("/websocket/message") +public class WebSocketServer +{ + /** + * WebSocketServer 日志控制器 + */ + private static final Logger LOGGER = LoggerFactory.getLogger(WebSocketServer.class); + + /** + * 默认最多允许同时在线人数100 + */ + public static int socketMaxOnlineCount = 100; + + private static Semaphore socketSemaphore = new Semaphore(socketMaxOnlineCount); + + /** + * 连接建立成功调用的方法 + */ + @OnOpen + public void onOpen(Session session) throws Exception + { + boolean semaphoreFlag = false; + // 尝试获取信号量 + semaphoreFlag = SemaphoreUtils.tryAcquire(socketSemaphore); + if (!semaphoreFlag) + { + // 未获取到信号量 + LOGGER.error("\n 当前在线人数超过限制数- {}", socketMaxOnlineCount); + WebSocketUsers.sendMessageToUserByText(session, "当前在线人数超过限制数:" + socketMaxOnlineCount); + session.close(); + } + else + { + // 添加用户 + WebSocketUsers.put(session.getId(), session); + LOGGER.info("\n 建立连接 - {}", session); + LOGGER.info("\n 当前人数 - {}", WebSocketUsers.getUsers().size()); + WebSocketUsers.sendMessageToUserByText(session, "连接成功"); + } + } + + /** + * 连接关闭时处理 + */ + @OnClose + public void onClose(Session session) + { + LOGGER.info("\n 关闭连接 - {}", session); + // 移除用户 + WebSocketUsers.remove(session.getId()); + // 获取到信号量则需释放 + SemaphoreUtils.release(socketSemaphore); + } + + /** + * 抛出异常时处理 + */ + @OnError + public void onError(Session session, Throwable exception) throws Exception + { + if (session.isOpen()) + { + // 关闭连接 + session.close(); + } + String sessionId = session.getId(); + LOGGER.info("\n 连接异常 - {}", sessionId); + LOGGER.info("\n 异常信息 - {}", exception); + // 移出用户 + WebSocketUsers.remove(sessionId); + // 获取到信号量则需释放 + SemaphoreUtils.release(socketSemaphore); + } + + /** + * 服务器接收到客户端消息时调用的方法 + */ + @OnMessage + public void onMessage(String message, Session session) + { + String msg = message.replace("你", "我").replace("吗", ""); + WebSocketUsers.sendMessageToUserByText(session, msg); + } +} diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/websocket/WebSocketUsers.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/websocket/WebSocketUsers.java new file mode 100644 index 00000000..81e549ec --- /dev/null +++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/websocket/WebSocketUsers.java @@ -0,0 +1,140 @@ +package com.ruoyi.framework.websocket; + +import java.io.IOException; +import java.util.Collection; +import java.util.Map; +import java.util.Set; +import java.util.concurrent.ConcurrentHashMap; +import javax.websocket.Session; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * websocket 客户端用户集 + * + * @author ruoyi + */ +public class WebSocketUsers +{ + /** + * WebSocketUsers 日志控制器 + */ + private static final Logger LOGGER = LoggerFactory.getLogger(WebSocketUsers.class); + + /** + * 用户集 + */ + private static Map USERS = new ConcurrentHashMap(); + + /** + * 存储用户 + * + * @param key 唯一键 + * @param session 用户信息 + */ + public static void put(String key, Session session) + { + USERS.put(key, session); + } + + /** + * 移除用户 + * + * @param session 用户信息 + * + * @return 移除结果 + */ + public static boolean remove(Session session) + { + String key = null; + boolean flag = USERS.containsValue(session); + if (flag) + { + Set> entries = USERS.entrySet(); + for (Map.Entry entry : entries) + { + Session value = entry.getValue(); + if (value.equals(session)) + { + key = entry.getKey(); + break; + } + } + } + else + { + return true; + } + return remove(key); + } + + /** + * 移出用户 + * + * @param key 键 + */ + public static boolean remove(String key) + { + LOGGER.info("\n 正在移出用户 - {}", key); + Session remove = USERS.remove(key); + if (remove != null) + { + boolean containsValue = USERS.containsValue(remove); + LOGGER.info("\n 移出结果 - {}", containsValue ? "失败" : "成功"); + return containsValue; + } + else + { + return true; + } + } + + /** + * 获取在线用户列表 + * + * @return 返回用户集合 + */ + public static Map getUsers() + { + return USERS; + } + + /** + * 群发消息文本消息 + * + * @param message 消息内容 + */ + public static void sendMessageToUsersByText(String message) + { + Collection values = USERS.values(); + for (Session value : values) + { + sendMessageToUserByText(value, message); + } + } + + /** + * 发送文本消息 + * + * @param userName 自己的用户名 + * @param message 消息内容 + */ + public static void sendMessageToUserByText(Session session, String message) + { + if (session != null) + { + try + { + session.getBasicRemote().sendText(message); + } + catch (IOException e) + { + LOGGER.error("\n[发送消息异常]", e); + } + } + else + { + LOGGER.info("\n[你已离线]"); + } + } +} diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/aspectj/DataScopeAspect.class b/ruoyi-framework/target/classes/com/ruoyi/framework/aspectj/DataScopeAspect.class new file mode 100644 index 0000000000000000000000000000000000000000..f3b9babc0b3c13f34ed44b681ddb376842db505b GIT binary patch literal 5393 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 literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/aspectj/DataSourceAspect.class b/ruoyi-framework/target/classes/com/ruoyi/framework/aspectj/DataSourceAspect.class new file mode 100644 index 0000000000000000000000000000000000000000..4b69b4bf4f8f931ebd24775866af653da72c8c5e GIT binary patch literal 2759 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 literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/config/DruidConfig$1.class b/ruoyi-framework/target/classes/com/ruoyi/framework/config/DruidConfig$1.class new file mode 100644 index 0000000000000000000000000000000000000000..4ad12cdc5c72ee7de51141bb00f61692f145dd18 GIT binary patch literal 2107 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 PkmD%-3j9aXP4xZ`;%H_m literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/config/FilterConfig.class b/ruoyi-framework/target/classes/com/ruoyi/framework/config/FilterConfig.class new file mode 100644 index 0000000000000000000000000000000000000000..72ca705c1dd49e77754296e1fcb8c20a3ad32e70 GIT binary patch literal 2109 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 literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/config/ResourcesConfig.class b/ruoyi-framework/target/classes/com/ruoyi/framework/config/ResourcesConfig.class new file mode 100644 index 0000000000000000000000000000000000000000..5d831212db7291c0c225780e8a77a7a29b8850a3 GIT binary patch literal 4065 zcmb_f30Krs6#gCp8Dgpw5XEX;s&v?lwbl)51w;XdMb@IWH8b-Vf|+ER$*^?Q?)$#l z-TDLCo}Q6YPWuD;qk4K@l3@a5wB-nA-pg|DcfWhzz4yKR{m&nN0XU3XGFs6m;k<-c z1pP7wFeqV2MkV?pxFF_X86y~tU`*VM%b38E5llvKv4Rl9Uquxz;j)aU@U$475h7Qr z&;m_H9Ek{$GB~Crq-1PIpNwhf;%X*>t7109kQRgKLrM`D7Bb?|vl46>jY6#vSqYAe zCgdc{N|=){&#=|z8Ll}axp-Q4dQFGh37&B*o8fpYVWm|&XD#UJl&z)toMm5C^`eM6 z9Io2V&@h}c9X-v*^{gH@c!z0Pj^^l=nPoT@v+R_b&Dgq`D&oeuW@gnXEkU*yRBc&7 z?Z`RSoNn_Z!B(EsP2D-gup!zs&QRWEB{{?9m~QfcTsqF}QH`uI)WocWW{hjLF7Ew@ z<<7KD=37JEN?7KUo+6L3R*oW&?Q-wQWUZD=7M}KKX42p`feb`LotWcsHOuW;gFC8^ z81Q5j%rH;sS;t;z7p_EX|Bm7OJ%py<5dXlk68FCKa1TJN;C*f#WW%r+K{_2FX$pWsu517Y1q6mEEeNcc>_=lDXwm-vcYYF6+yZYcOh4Bz60gzv=r?-l%j z9~n;G5&bSHq~IsqWN0ddu_p7yD*w#TTwKy(L#0&>>1ZobVlY0#6AnXXDTAeZ*#wXG z&n5~vCo-V-|AQ1V-tVi0#zOsjZPR5)MlZ!y=U;p2Hc^^;xY^})pk*@LOo|>9W^OeD zzXGV5Lkr7Yf}Ay}d!eG5PjP3+D=n&dYICar2`S=Og=+1;S5bA{Ba4B^icQkrlXto- za*l3@Zji{?Hm80tN;~)%ZMan4J%eMzBMiGdx}^k(`~EP((~gr-yR^hKr!Y9SWl+~m zYx5l`>U+J>zPo^iukjj8GQKe(Yz*b%hMu50DzS-#X>QJPXC=Y;@NIIfayY&990dv~ z*c}$bI|GL7snl*)?)?nc!+i=u`?VOrTUU(*^!>6fIJb^IQ3A|6+zip?K|eY+h{NTT zd(|XL*rX)8zgV%ImSxaZ&LGWdMvf0oG1Nyd25Z`vAhlY`unxnp=DO%COVy=o2Jx*A zHkpF3I1p~7wR6RJa?sXOy6OAKuqCMC^$GU~SzXlV;hMWE2OjlvXWF8!wyDTIW{LAa zZL}{)SUe=QRm8U6uh80{LHE5N*x;R5xxMB*#Lyko@F|9#V?^eK@4K!`$4lpHxR!qu zYZ}KM`haQ$b#eNYiSq!>2kC38lIBNfE~lA}NHiX%JMor@Z?WdvV87Bu86Kli2m=i` zLZjjd;3$sK9ep0*1dT%Z7Ri;-e7U9kPn1n=s3{-Gqr$shLgnObNHvi>sum&BShJCa zJQR|-Z=#^Fsi;x2S-h!3^>3(IL~R~*d2E>|a@tPTfZftl9amr^rfC^WtzNPth)aS9)C%%nLHw0|Su9S)`6gm+D54(ODyPmM0 zaIsILgJeZCZej!d180`7om_t))>6KR9eM0*sd+GuUGBKs9rv`r7A9$K>hjw5)gVQYyf5#V} z)x;QL_yPVX^`_`*9}s-gC3({`_@x4B!#`JmzpYhaBc}lwGm#s)K75T(|Cq zbqfw|I#^_Q=`}|(3VL;?vCMoG16ww8;Q2pma5}%rF6t2sg?Iud>D#B zR2NZCcNh}e>kQfFK`51sUN9t!r7eb3S$71(WKAluG3>X6Y4LVQFH!OH~i@R86n71u{$W_{8K9Uq=Ii%@W(3nr&P!s z@Q_qtcv>uN*UmKL8EN;r(qedW=6*+;j#NC1uYk@Xwb^LWz~tsT@|m?tt@%^!z@83 zhM$olz8TCSO^3fo$C1Vo9%7Z^W4yu|#Uu*Emx)=AaDnm!*%a9&WZyfOM*XCtj+waX UB2|DGX35_o%hQY;MHI5X0ghVbXaE2J literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/config/SecurityConfig.class b/ruoyi-framework/target/classes/com/ruoyi/framework/config/SecurityConfig.class new file mode 100644 index 0000000000000000000000000000000000000000..366392135e1205c1a90447ec7ad9e9f7f348fd46 GIT binary patch literal 13563 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 literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/config/ServerConfig.class b/ruoyi-framework/target/classes/com/ruoyi/framework/config/ServerConfig.class new file mode 100644 index 0000000000000000000000000000000000000000..7814bfa9f0c1c6fdbe7b2eef60a873c3b5fd5607 GIT binary patch literal 1423 zcma)6Sy$6Q6#k|y4NXH?#IjfsM5GH6QCvZkMWmj(91z|^J1xOBQ_~5{ztWcqKKKLt zQ6BFkJ(LPZdAWDKx%)En>-UeJ0Os*lM*??rOyRDM9!$rPMn;ir968MBxQAH4-|Q*VP3-{hUlX0*m8*>oX+epL{>$KGxQd0hrjn~yWIU`?N;e#pdgA?b;ojT zgqxo0;RGKHe3nXV^}ZL{oJHGU}E&t_3LWxHZ-arc0`t3FtucoixhG}a+#sSn(#2+=NH#I@xh86s?Jz8) zuV{8@^suK=8@`a(idQakmqNv^MSN(iRT2GFuRC#cb47H`=XiNbYhWkL%oA6}7r>xFZ(~^rBD0V*^j{ z)W9fiXn1DeIY_dI7aEofyu^@(R|b}`VqjH)H3P4)Zr}|z4CFCHeZJ!G82Z}Pe%RgT zMH69}h$ZN}8upG&Dw}R~s5}+qg4;Ck0e3A3lIi<1Lz~# zPsk+Bh<9bbgMFhz2m?f;{$&rY(b)(TFo^4f7?Kzws*IP3hls~c5Gv$O5dKOaaG?8w zBqF4&oFoIqmXip&FpLp0kE#7`xjUrX!lXNwJw=3}K&*@56j6qExm*((U34=v5i2Rw zgg(+VjZ6c^zHVaIktsm`0pIbD9loshfjGwx8m6z^=|`dl?D2Mk6_x}0IY z;|cyg8Xa=AZy);PS?EZ|_7803>HWlMXxa?st`J9Ocn%1^k%t| zd{tNQ*GgzCK9>L2n6FN0ku{Pceo4XP{m=~OkCi9F_L;+v^%KDwJ^nQP53oW(-v(JJ z`eu;>!!5Et!JrHvU#)!utFAPbjSHl{#(5^KY03j$<2J2Z<4ojHpdHo39d$g9J6MQ2 z=W!Q{n2UWy+@tP>zWWJUmm)f5R%`VO%+>1OVN9{|aqJCYHtB5JG3MGh{|eK@G9JWd eGVu^AggJ*t)G>usJf>fR7(c;0S(>3{MxQ^YbzzhM literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/config/ThreadPoolConfig.class b/ruoyi-framework/target/classes/com/ruoyi/framework/config/ThreadPoolConfig.class new file mode 100644 index 0000000000000000000000000000000000000000..a8bfca7aaa02e035a0f70b893f703b5c6f6201a2 GIT binary patch literal 2310 zcmb_dZBrXn6n<_9ETjtrDoBe~YN>=!y7E%2O~pc5Y;4+6LQ&|a%jOnt+3e!JAo$H0 ze~`{dn2{MjJN^=XMW^R%0)!GB=`=GrXYaXtpYxoTd-m_wfBXqx8ZS~9z+?wL#1xBZ z7PnZ;82BQEHcYa(&EiWIcUa^Me3gP(+xyzUJp=P8q%ql!`>Y?ZSYYu@3f*{UpkUxz z0khz`%FjD82vk5z+w;|;=Q$<)lM)aGfeRJ6dt&MQAySdb%WY}va8Dq;rPTJEqjywE z*`8Y_jJCVl)#0o_OJ;OMAer~d1l3*Cu6hzx)|J03*B!F-7Cl=!E7I4zt{ao#rY4N> zqU}{IKl1jp_0pFW^`qx+S>)iQ-msQ8eI?7xDIXhh0{!8!ZCM6e4|bIug`O{PKQmhN z{0%GE_O-il0u|VssvJ3FV^llxedUG?$c-q3->!}k=idk3rKQLXbw#b{KvVE@uIq&| z)Ses1-i$D@4ApLE$>Ue8c_m%qmUJmU0t5AsRQV|SlJa-7OE^YSrJ}wN$Y<8x16=Dmw~oM_`XoF*H{$e5 z=3E20(7Y_CS~}PKo&P8uuF^+vfWDDQBzSnxQ-bG5tZ9VAnx+Wp76_g(B(KtHjouTa zhq4C{+4ST-64@U7wU3r;Ps>01NM?J)jeWHK79)I0vLCG^d&yHD_;cJtr~@>iu94NO zVPP28X(jL(MhKt%{7ilmq;DOfZS?>~aV+~gTB>Mgov5ONb+U>S>((kRj2*(H7dxxy z8vB(FCcw_Gz#w@I?c^d&P6o)u5IMHQP>Ll7R9`6d;1-gjT82xX&fC%Vp zl6G&_DLyA`PQ}O}x>vJI>kw(8(9_#n#YN)wK^1*?K1Q-IwBuZj5iAWaE9ZevhM3hG`kR>F5i(-uS?l#b{J44OR^3d|5 z|3|A*Yn31J1N>2z=gzP^GA_o{R!yJobMNhY`#$!szyEvzU>IHpdNI?Ew{gD%v-nET z13`0w<^??zlohlf=#ijBL5~GJ5ws*|SV>v!a<*dgB{S)HeyBt1dliM@tRHN+m7TzP8|#5Cn>|0+cJoGi z6?a`1LO(cg^^t;`)1|5zS7_N4pwQpQGnwYEUqlVo#3PW z!iDUX-qmhNdmHX@C{@914>tDsB?GkC6`D%8x@qIkS%GT`>Ba_5NmJdCgGF!34+4Kr z;nvxjb-`oBB@Q6t;tOM553r{S!Pn<;!B=qq)Zg<{_FT) z+@aYIPY#ah=qai}RBPi7?qbxz6K8oIoO1C;NoKkTiAARTTn^Ru%I%w6-HhwOA0;5O|h8YG6fEqelhjmiBdRU zG2z_Fk(^4;Nu4=7U#QP7W=Xd785wQr<1Ce)7y0WHkU?_B%(@oaXgDS!zTrGzg;Sps z{`5q+lunKW@ux@pIo;~1^qMf;hfKXWPoBhGha3Fc>g8W$Cz_-Rpzuz#<6I_ww}SUb zunv;M4H7JsB=J58)=QH3fCS4YN!%jAYDy9VBv@QYVu%E5ElFfZu=J9|CnQ*jN#YI( z7G{zdA;HGND8FNT6ql>w+A{bD)b9jM_>|w)=&%c)5j(K}#&M53h0ig;uPDDjGv$G2 zLxa!JR710@sTx{jZLOhA)^oqs&@P~Zpi_WD&?VqJLAQXO8s3ogLJe=qda;H}vR#`ZvShEV6ij70lx)9!3#dj=emLfc{eU SOUBTI$;kFpR38%aH1H2~4}Ks3 literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/config/properties/PermitAllUrlProperties.class b/ruoyi-framework/target/classes/com/ruoyi/framework/config/properties/PermitAllUrlProperties.class new file mode 100644 index 0000000000000000000000000000000000000000..fdf29a67d5ba4454ed42b9bf8d78cf433e083326 GIT binary patch literal 5792 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 literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/datasource/DynamicDataSourceContextHolder.class b/ruoyi-framework/target/classes/com/ruoyi/framework/datasource/DynamicDataSourceContextHolder.class new file mode 100644 index 0000000000000000000000000000000000000000..0ce5fe6ff96935f9a16d24ab2ade9a10c1ac6931 GIT binary patch literal 1290 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 literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/manager/AsyncManager.class b/ruoyi-framework/target/classes/com/ruoyi/framework/manager/AsyncManager.class new file mode 100644 index 0000000000000000000000000000000000000000..0fa509e966fa5ec441d42cff22591b6353391682 GIT binary patch literal 1438 zcmb7D`A-v35dL1TZObB1PUR9&Xv?u4fJgDDP?C}s3@tSzCS=(@=|a0}w!0wzm7)aH z#Q3v+lyTm+z=lASCY|?Y-t0HmH^2XU{|R6Q=_uMT9zhuKFcMKDF(F}6=u<*WOHhL3 zD1Olh?&CoVf_NCkBg{nUeO$sULu~a;W-VRJY%XSU>5a`|HlJZ&S%%gH)2Q33VVBgZ z!~Ft;Aj#FGZz%LdXkIsT`vrqPo+vQ{7R)k1wB>Yz zuQ;_WZWYz7Dp_JVQ&X!Y)zXFDGzRR7UT2t$C$1AxYE(#|eQ{5PMZG4X)pt{FA~xL^ zx-WI0F4ZfJT{ia(y3T9WW)0DT=~x7N(#1xu`w9W0;T&3QOH5=?g6>Q;}N;F#PK_?mvURtXd$vV-4LC$A$ zxojb`u)4BXU>Np{#ly@~e*z4j_Lt$EH;FaJFa(2A%m09Arg@RoyOftEL;Ot((-MgqMS!S}rbEqJ$Az={DL@KVpJJpsgUyhfey_ z@WuX$`O_IB3vrD+ZjqKJjnUbgJO)d)M~~o3#{4G;{6g>uq0jW>LqEwF0wf1Xb`n${ zfem7SEOO&>(gb*f>_biH0rv$6976g;Z&WSGkZTJA}tDdRE+=s{>(ydxF-BXtSV=@>CISqJ?qy e3}e)d1GM}=c;gs1zCaQ%-eQJHyGJ%3#{L3!Mp`xi literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/manager/ShutdownManager.class b/ruoyi-framework/target/classes/com/ruoyi/framework/manager/ShutdownManager.class new file mode 100644 index 0000000000000000000000000000000000000000..d2087c92b8f613ea74d62ff2310846853bd1321b GIT binary patch literal 1481 zcmb7ETW=Fb6#m9`vi7oRE;!-Vlq8g3=fboQT4=&0goI0-f(S(7akH7&8@xN#?3mym z(1*4X^sTS$Q~Oe>0V#w;g7^N82E?C0oLM_@2neZuID5{S^PSs#H%_M$e(38949Q%6o*^|Q7X?Fm!4Jg`)nGxWJMKc6JOuXK^1Q2j zV>g^BUGgJ_(Sj!fuBx);^Cjg5VnwR^Ja9vo>hif#RWHhwaJFt!O>0p^TFDy2q1)9^ z`+=DEBcIYvhN0B1_GL&}dkf3%1NR|!cbxo|5?AU_%94#5i6^64=ryu4Flx3thVrr- z7Wwo;PgG1wg0#(P*SkOKR^t9?YYgcLuiV&xJtwQm6EnVM}#HUPXlRQU0act&pkb)uCEZ)4Sph3#_ zIg<$J#z%7bPx5#-gt%lEg`O2rL>wo!5VpCn#s(N#gi?}T-q1((vn&2iNy!!Stm35| z*t_N#*V9tbYspZ_`om=y{7B0DhYuY6^xx4ze<%!x4GT$2kxtAc*-aLwFQaeE&@-@A zQWEGP*%~|Bae#EE9)N>5M3zaEr6>bFP1ZIz(FEIc3`&zC_=gFO(3wg9Ieio6d5~_B z{e)zU%RND|EA<>L2|PjT8mx7s*O2)Wodr4>Ucu literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/manager/factory/AsyncFactory$1.class b/ruoyi-framework/target/classes/com/ruoyi/framework/manager/factory/AsyncFactory$1.class new file mode 100644 index 0000000000000000000000000000000000000000..7556caa3f29b4da031e22c31c0551c8330895559 GIT binary patch literal 3197 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 H3~PS?kS;b$ literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/security/filter/JwtAuthenticationTokenFilter.class b/ruoyi-framework/target/classes/com/ruoyi/framework/security/filter/JwtAuthenticationTokenFilter.class new file mode 100644 index 0000000000000000000000000000000000000000..b49866b35d8ef2b60b6efb4bbabe11a95e410f23 GIT binary patch literal 2876 zcmbtW=~mlD6#fQm1VUK|O&d3DT9P!FB}`MGNpKbjA%)bz1haOR<-rJKiArOZN6BOK zoE{*_IemaWM4zOmr*|Y_Ffll^e`eF2d%y2)qd)(7^9O(-tn28+U>h!BNXIa~)^Qj2 zblk^?T4mJP0}T&#Ji=ogtr*oXrlTE$YIvef#&tZ!L_410n|4fMO2cyv-!d4|TNiF# z_?xyR7zVPISLS}j+p+nIZq%4E zo*OXS%X)r^2OGZamJXE_gy{x+#k8d7?{Kqvfsa+Bw`KdHNNQUi*sd)fF*K)p78nxa zUXjpU$l9)$u9OSHpEJoC!=TLKrR-&Xg%eUv}-yH<#cw<+%&gCmpDT5RVg?4Bn`@qbTBJ^t%yKo z8j1xQgq)$f zA@+zOL#r<$8Vrs75))=fT8bXS)nj&!a%X6BypruI?6sLmIa>ByZh5}oMXyYnc$pH! z;n@RuB6BV;O>6yyxe;cChAe|Vv2BTsFgrE8&@fF$d9UJIg6vW@b)A+B11jhYz4iT0 z$ss(svmyAnSEjI1K4lDCLCU}fxU3;(Up1yRQ0ODkEP6_DAZ=jQz1>;2_lCX_z6E7_ywkhf&mLf4Z^?*N*Y!T z*jQs2t=CJcj8+=gAH!fh)Cz2TJLg(rM)+~XMcKp9e`vUIsxVxuE$6t3tGPlIs06s} z@u}RA(<4?6yW2^vk~(ck$ZCEvEHxxja79JvD!>w7)tlluakrB8`f^^ErQ-c zs6t5Xo5HtOcEVoFa3_5>AE>T*K1StHv$q8?zCw_~I4W;Yl{yabq=$;kKCI(_NcN~p z>fctjj6%hBigA4~1n(c(dg)!QkTVn|kLcL`Ob^__9{@G9 z!LNRepS_+1bkQ$dC;iefG^uxi<_~GtLi0y7H_=Q_28|!nj?>I&PWJAB{T6<{ zN~01634KDN5eeW^T%%nApW$;-qAvf1BuN5Gy}zTWZx_uYiBw`AiB}liM{*x6uW(m= zN+ouoy+!MCD#6scZ4dfLGL?Lb_T@bobo1OU&hO(wD)|~0n{d0Y|1~x*Kw1$|ScrrWWaAdGvSzC+PqUNGrsK}^+S3Dx zmIu_MR^h>-3J-X|Yp4`jgkO61AMh_QtS9~h%Qrg{H&)h!Di1r|^Y-=k-uK@3dVc%s z-Y)=-p_oM)hb$cK!mAdJbc5ryEMCXaEZ)GeZoG-(S-gd}Eu64$(n6kLS0J=>YGt7V zsr;G3IR-Y#uwz{LfpL6Oc50!>e0k;jf$;a4>kOTzq%X}VLq~47%#a>evw~q~QTpOy z=v9RNz^T;8(o<9}!8!y^?};&O=4HTew&*I4>rgF8KBpZ|TvqyH9tbzo(k$_L$DgeU zJ{FpJ;T!2XhBBM*jb3_RN#7K_dW{nAA`DOXF2(vLS6px|Ig13LFV%$M^TyQq1$vZX z!?gG$6oJWa>lW0Bw#XNiuJWL+rC)86pR4qy>=s>7kJ8LXMX_`ok6~!rh;d>>M~gT( z!}e^FG^tg@{{Li44Ro6gWCiM|7gVFhMYZHL$FK5|iAq4l&Q7co3@n_Y_mT>=E6&TP z)%&*U)R72{p?^B0R(hf=14()q^L=HSeI+@Jv@f?2S}3zr7kpfK)ECl1-bN348TPmI z6RAM?+)d;fcb%X^0?s#X-C0}tqOttzpVyWjKe+zc=Z_zJ-B`ZaxOL@^n=6f6J`!^atf3E$qP4wC<#jND8~?N7n#)>DfmXPAj7|Gja!PrRlPpo}EqO9_*og1VRpY z7SGX2(@v17=HaVk?Ic_78TRe*bWcb6A<`dq9J-4PLunPACo_k4r~5OjuvXEvilC^xr^%J>4>?g|s W`aezn2g!4Y_9^lnrZwU-0{jif8a$c+ literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/security/handle/LogoutSuccessHandlerImpl.class b/ruoyi-framework/target/classes/com/ruoyi/framework/security/handle/LogoutSuccessHandlerImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..4e5d956460b654c7603007d8c7b115422f8549e9 GIT binary patch literal 2625 zcmb7G-%}e^6#g!RY#O$uVyjKH)wWs)MYhFSkx&#V2yFl*K(+NpH=9dX*xVa;?-qFL zgEKll`RbE@gEI~?ozYik{6Ff9|AXUr?rz8?B^Diqd+(mTd(L;h^DDpq{rFb^*HO#D z#GA*^kLy{yg&SGCjj=4=!MI)uS~iizO-veiH-q;wm@@Ew7Sp(;r88N~;)7nyVLpr7 zxMQGbV1dC@a*K-+_aC|rXBa6uvS#_A+;y#W->&g(>2F!vyle&W6>FvCet{vk917*u z_?jEIWsgq@A(gFMDFTMEqV%g)Q1@L?O_Y_nEdp!Zc9isYExY-FH5Dqk?fSezYSR<0 zaMdJ3_h5dFAvG;4gs!jX3ceWD%G_VE$r?j{Q98D_X8W#wZU|Fq!zJ^R9eoM76Z)>& zwKi-~@wio#RT-*M=r}wG<|3ItU#oirqECvpw+vqm7B}sO_D;lx$Cb6ARNb1RMKef( zW&UNz164TkT2Lp7c%dUhO^A;4xu#`<6UR8;6lMVC?C+s$K45RAoNCWN%Eg)#RzyQZ)@)Z; zHA)0e2Q9u&&K@r*+u2&M>rwO#EHY$gb{t-hve3Yife%SklA-VLSyyMxxwf#j1bsx4 z$#A2i6OI&$@2J+4ot9$Vt%iOa8WU%bGjR&14SZx`86^WNCRVXVs7&0&$0k0}%RSsT z@TrN<@HxY!q~e=z>&Jh5^WBqge_%M7gny~L$sJ|j z3lk4un<&FEP%*)=ZlG#n1FnHh6I<{YCK9VjX*Lt@h}vMd)2XtXpX+|no){KIn98hU zwJ{7QV44(A=64P`!}F2PuC(Ttnt^6Gucb+4HVU%kLW`Be<$r5!M0Ay_W}Y$JJ-RC5 zQp&fqouj>>1ipC4`%t-F&?x6seWns|gGH$pLyuy2I))M#eH`^d{vJawA=X7CqRwz? zFyE;+Wav`V!^uM`r4|p{)HZsF?lyx7zJ#}$E$k}yt*KyFI1BLuLziwz!({lsenReF zeb(KBF%g-Wjb{cWz0av9R3hxISeMi`9<*}O0Dsv1!;pkq&~jLZgjlO|YfA zD&L{mp~%BC4P#GT(PRxI-q7jk?lrY(?=15m^c1NJ;+ey6vjg^43ZJQnfSuJgZO<*+ zWt#?Tpf&?3My=bWCF+GV;PoMs^J3txG_NmPXArHkJA1E{rE}K{vz#Nv{Q?v^(;MEv@_a!hJFV7F?#(H zt@<+XGW{V^Mi+1n=jn;z6}(DH^vA!EG)?==&`;>PvWM>RR4%oT)c45eQhVqb@6j*x zF}(+4ADLWFf40B3`xlt|IKGDy!~MPc=-bE1eLSa)cS9RL4};c0WRWLdh6w90UdIT= za22<3Edn)3hI0hv0xl8=g3y4cPnYmo1b>FUx=c{DY=HE2lg0w+96}0LNSXor3u6Xp Xe}kUVbaRCG=prkcrE5*rMuC3-TFf2f literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/web/domain/Server.class b/ruoyi-framework/target/classes/com/ruoyi/framework/web/domain/Server.class new file mode 100644 index 0000000000000000000000000000000000000000..ecc282e66d0b712a9d2b188177e00f5e2ab9c3f7 GIT binary patch literal 8200 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 N=5O(b^Zi@u{|~|iFD(E7 literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/web/domain/server/Cpu.class b/ruoyi-framework/target/classes/com/ruoyi/framework/web/domain/server/Cpu.class new file mode 100644 index 0000000000000000000000000000000000000000..a1d1316d9558bae95b35f9cee65bab5fdf51deea GIT binary patch literal 1605 zcma)*dr#9~6vm(K(p@MHaN-5L;Ein{6-5-mfnkzhh?syF{KKT%3KX}MbS;GMWMWJp z694c6_@Ru?=^DVM3vS6d=Y4w4`~J>*`TghTF91_m$f6f|1{xlVnN$}xsq>e`6Fe0& zCHgZl(_)^Bd6B|Q3NIC?wLR~h*HoY^D@eMwYcv>F6vW$yZ3PLhZPw`y49lh4a!ivo zb5_f8%giYhH>fV!b*i})t7USks_Cp7)drP;6}x6MHVns-aUdq#9g7o>t<>zM?s)d0 zrEfV#(>$=9FZzL5)$4ZCuv&WCboNb0U)=L%6=b$emyr+3MTweepDBsLvT&rCwBBie zIcy=#oIpqvDw3iyWlj20`NaJPisSKZ$Giur`-E*x*TYv3X?hX&B>G4RDg-S*CB-xJmMr1d?4eNSfkUMSKi?KO^cdWAk&`T^y;pLU&A^~Ef1FxEN=7{V||3U1;S z?Z3rGh}=_*uTCP#7a+?IcCv!y1x@-D>Ji)x=9)o* z5s)h#A)ZI#1ZwgGsc#{q>2pZA(|+#ZelSfLasCBq^ce9l(kzkYx**Bc3(|vAq;i0i zCsJCl?Cx9(=eY8xTo0qU(qXPQG8ogLOrS$>EjnA6?1wCrP~u3!NmJ1O_Ko)A zsO$X053molv*#uyDJ4C}a(LhS-ro1Q&%O6aPXGPy&07G|ur*x3oj#n$-R%5Ly7@!-H=;@+wilq1f}N!Go!*wK^A3yUm6ic(6JUxkKB z&r(ztT}_L=wVJgZL9W%zs#HzqDk&%!&W)zktubKzs52VD;={Sz$#`KSkjr7SRE9^E zPG2c7{h(}1E&+X9ccX)Rx8m;CgL%Ko?r0^i?pLfOTiV#m9!$tIbbN>_I!16&MM*~) zYdYS;MIG+td#`59`*yHPy=>bD98fNwUy&hpJ9UaM z!=s$8CvcAGaxjkN21YS9m&{7w`7f9^YFxlB7N0CwCe?cV*xXvKv%yImg>)@7$+0K} zeN3?KM2fi5p`yGQSZh+7g0ma^Y!-`~`TXYcvKXZj#ZQn@sNrzt5>Mw5Pic!GK5`n^ zH~pF=hqA{lE$)A?lE`P?F(+a35w z+l9}kseIH9eD0?5`7E7}+RkTzd=@+Kk?{+kiI~r9b6$foFaE354y5k4kQ$7KX2d{} zRt7&31Bsz>{KM#NKx-F4p#^Uw#=D70qKJ^DLZU!l1O5L%SMUd&ttu+F0QAKGpT_`+ zj9=nZ`{j+6%bED{6n_dt+2@!_#FiZ!e}z-+Eq=;cH$#Mz5|k%PTKq*Sk%4w1l~f`# z=|l$FiR_Yy-GPWa+Ct>Zn8-9sk`ZZt;dBFM8W?P6v){r-i`#PRBtip2A;Q^qgy*RU gw_}8vkQ3hz;aG|Jh633D+sdxom z1|fl{SnvR>cp}6Z$4aUwmhj@tIlpIqf8+o6-=Dt#JjNpjF792zW!yJ+|BytFg9lhu zQdO0j`SQb9uUlBNu+Ct_LG1MySc9REM505>aVP|X`Be60{G36r){aPh6?8}~Hf3MD zP5hP!54={7l-Xv`_IgKNDAk;ah4@593~SAH;PWsEK1qHYdcHUd!jJq+w0I}*J=y1x z2nQnMZ-l?YV0T4)FvOv65|AB@ghF*+QLudmi?$u6q!zu;I@0!1Y^4o5Wp@~PU1D_J z2T9l#dr|@0^tH`XZ{WGOidh#|Fk@lE#U{9mB4%8aRC$OkhT4DgGnAAk?|J<$f7dz{ z?U-SHe4IKz=<_6&y@>CIGCrX{+mQqM(bY!dwYskECzOOiW37QD%FLy}bZCl@I!(c) zEAi24gQ*KpEz`tlWwaXgA7Ee8Z566!Dmu7Ey*u21Ib5fS;RbF}Rfo4>r0?ANH)s{; z#yN~Hq-z*^EL`w#M;^CuJM(x!s5+go>WiPhLmMMFWCJV{%1UZ6m(EsZK?mr5^JK!X zFcrwm0a~33bSDSMoCI1Ypp6`$Vg|H070Ai~+L{VXO*i+HTdi0ttk~ z1rNYOA!gR;$!sKUW_D+1_Wx#ffB*UU3&1*_*r?)R4Hc}hdBo;1n--gP1)U1I0*)6= z57NX}o9@&XSRCxThpwEs;aCn66@=p#0`^JH#(_X3jeYedPy*#B&WdR9T6udEc^QO3 z(i14O+B*WpezZ$>HU>fHZ>H0cuRge=iB^d`ce3NEfcLXtG1&`Zf$o6Rq)MaXKz>pL zbrh-3^2i^_-Dv6tp(Oc3U&-NdOlXa$r?PLj9!ZPgyR&bT<#o}+UzT}G6jK$Y*eOBBfQj4WJ-S4o@ zq-`DTP)A*T!%CJllV>#Ulc{eQK?->gX#}}x1}Wr0)(Pa<91tEfgWNKL6!Rd@jUczp zAjLdLk3e3{0pXWskUK_@QXZsl1i5PlDdj=l5y-|I5dLZma?c1-&V#%+f~=ZB%6X7& R0vXN$;lIux_w}!Q@D~s(yiEWA literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/web/domain/server/SysFile.class b/ruoyi-framework/target/classes/com/ruoyi/framework/web/domain/server/SysFile.class new file mode 100644 index 0000000000000000000000000000000000000000..2b70dfc045aa25118ade4c16bf699dfe5c15dd8e GIT binary patch literal 1729 zcmb7@+iuf95Qb+RCw3i^l+u>-p`4newC0ck6)0RZsz3@wB5k>W3n$qGw@F-Vr&XQ` zkXj*e!2|G6h?%t^bXi|imj2nDo!R+zy!-Rl_a6XIg*6K-aHe1i?vp$qc}Vhzq)M_% z^4NeU20Ue8bi82G9dHISja~P^b^7kG=QN|h8}^g3VDY?b`-ijoU%HD+;eZ}MKuPQQh5g>*8L8y6dT@hY-l z=SZ`qvg%M)_)pAS>HiSoE#OAD?d0$U%sS!7W5i@RkIR#pNo#S3Oj?MC26{I{zYKIH zBlK;H=8-QRQTBG+No*eWlJ-a(dZ|u$R?{B^ZBD0VU?=Bap+9KDEX>)E2it}-Bn6Tp z$u!AXk{LM1VEI(dWiUOtAa7c`ydA*;UL^}Jo&g#a#F0idg~hVMVr5~mxUg7XSVTZu z1V+55^Eh+RGPK#sH(*~y*m=~PaGG!d$95b57vU1l7+i)cs3((;K>H#kJIds%a4ku$ zA-P7SDoe+pO-ftBTZJsFiXGq?NDY7kupd5IveQb6d>6UZ$kNLB`UtpvHP2Fc1GZxN)K0zy}hK<+3(ax%y} zCCFVhNKOWMk02jXK7{_s<0FtVR8Zqrqv432+!2_c mrB}131X+T6NpcM`_|dCbJO)z+*+-C1DIoOv31nGRT=@;8>fNgV literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/web/exception/GlobalExceptionHandler.class b/ruoyi-framework/target/classes/com/ruoyi/framework/web/exception/GlobalExceptionHandler.class new file mode 100644 index 0000000000000000000000000000000000000000..ea0be7f0714429257f530c7d6242120970114edf GIT binary patch literal 5307 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 literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/web/service/PermissionService.class b/ruoyi-framework/target/classes/com/ruoyi/framework/web/service/PermissionService.class new file mode 100644 index 0000000000000000000000000000000000000000..fb3e9b3836fb4693df09a6390c66d22954e25de6 GIT binary patch literal 3644 zcmb7HTXz%J756VfJKRpeJ|HylfgJ(^)YJe#LfH4hwW1a~;n&9e>d*RV{&Z;SA55n?61_%OhP2 zCv!9Bj?d&~=VngNEC}>A6gG=-2Eg7Cwxoj6LCCjV9{H1Q*kyQWv+A$972CHh&bmWatG-<_ z3YJgZ{jJcfuhrmrvUTg!)*z1@CT%o$8hi~^0gY}fxDs!z!p8|~op0WHBpZEj)RyDh z{<={Jx6?62>dHlrm@!!b0-1#>!z^2iwr7)w(~jf%rcYwg-@%+) zSuwmdSs)D_dcIY$T%T!Sz5+TD8ufBmG%?T9@i?B)a7D+f7}Bt=<2C$P$4_vTG|}-> z{7l2^a`ERHexc)+ctgi=%;*@zkih=`oB9G`gelkYE4-=WExfJa9UbrDJq^Fs@jiaT zjOq9QAL{rmJ`%__EVQ9|YtnW64%Zl+xm>m#9lys_9UtQlIwqyB>hL6gGHFvoQADZ@X2jvRc=p1Nw^P*IR+W_~6ik%V<-OBum2f^3_slQ=bYUJ1i@p zizXU`rIXufq!D4*CI;J)Gr;MXTPhKKk~nfX(_BT3)xg$NBkCrOMA|-yc|iEVK|=r!5Qz;dC;HEceGSN*yW0JHAA(hA|nAQTH&<^%&Q?hqD{# zQ2qo>xd|oSa3KtIDlpX_ZmpZ@j!a1dDq@{SrS9>XZt*BROLFP_!$dEG&u4qm8_*|} zjB&6R%m~ zb%J>1h*zF?o#OgDJzL=FX?~C{@_TfFUzINffgPgz&trQKjzs)?J;d`wS(2#}g$T_d zJdFc5$k+~Xr_6mQd5RJV#9kz7NZ%v-Gz}9Po+j`bp3(4aK{5^ei}r-PW;l{3LL;yV zo?UH&)HWj1xq+@4Rifz~bPME1?_it2b&T_6yTD)3Bf;Cij!CtrH=}N#?*_UvD%JM$ zJpk?Iy?->NcMNITT_$1|h}bIQw8@)G^s3Bz+a-o;{4}rd%iQOVY8|Z;4DB$HQMqe3 zrtlpW9{sQ5wu&PWZfBcVxEmU=*(q80C3Fqn{R-n6juM}LL;Qyz2u#;NVcMFgx%0l% zeEmMu4DGj2^Vz!YA`dj25*4@sFHy*r`AC)7T*Px%y&jP|JqCKJP%#0GYTTvW-$ ze>NdI%$ft<2{lLn8YTk-bK2J|^EjAxp3EpPD}g5eyT-Bg}^MTXJE(j^#X# zMOYqh!m>Z$S%Br&=nR;(Q}QY%zqdHb{_`52vi}TZdv@JHCWaf>mSy|t4<^@}iao5K zu-I<~{)ByHpzg&0E8!p$B)gLAFtIxSrDQ*z;+jM|f%vZYQGw_99%Ga-yb!$2;(L6i Mx$6Y~avtXX3kktV^#A|> literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/web/service/SysLoginService.class b/ruoyi-framework/target/classes/com/ruoyi/framework/web/service/SysLoginService.class new file mode 100644 index 0000000000000000000000000000000000000000..010536a823cb6637563130cfe9ac287ea1d79799 GIT binary patch literal 5781 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 QmMgedk+ehkQZfwu4*(C>mjD0& literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/web/service/SysPasswordService.class b/ruoyi-framework/target/classes/com/ruoyi/framework/web/service/SysPasswordService.class new file mode 100644 index 0000000000000000000000000000000000000000..08f2e6acd67031f5810c11bb34ed4c20ca2dcb3a GIT binary patch literal 3837 zcmb7H`Ewk_75;h;&1yA15ZFc*0b?Az)-?o3011d?MaIICY`nG{*$(z-wk3_VJF}UY zm9!y*TtGrDawmjvB|r{Sm84wq23O!WS0z8BQWgFiQc1p^oz<+oixO99yQjNfzxTcG zy?)*L+rQs^55WC+CyQM;t>Lp-Bve0}!5Q^=RyF6c_#B?>!g*X!jj3eMWsuKeKjyNq zP{=}JUNsAyje=V;ZO4#|PEb|oPEqGbNPyBVFnB{71@3BqYPdOC2^N)lD~$I^ zcS$+|_qL6>FRhBFaD>n{ZX`{aMu-_BckbdwuaC`LlvY5gtk;x#H#hMz;RJGl&M`yt zfoUyGndLBgOj-?J)9`f-&+;H>cuqhonE~U$kRG}*=9=s^LU)D%_RN(A9|Ak9qBOlp zcfoc}NV-}GW6$|^foCdq7M19980eJBSUQ<^E1o3}+bSb>wIv$gD(D6;Ypy69OzK!pYY8~Ij3p&1s7j-;<2Q_?O#}DvBaPWistrXI)4$d+g=v91J{qB= zs;7=O@s_|NH)43&4W?A~HW@sQCz#?o-c~{QjgH^qcRGHLOFW3N=Kr@~3G~OC__cX0 zVeM&G-Zwky8=_J_MFKa)fr<_?OSV}qONUV$YrQwzc(O>|)GI z6KK|w)xg+TqId$->=CzqV6v*J*Qa>N(aeqVK%g3R)DQkvrOlHCRZ7hw57|cG>*a56 z3_OZWC9sRWF}`T28|D-TET2>+@L(I(nn;UOHq*0iMGG(AWAg$#a}&)Hagp8MMtgh7 zjO_&BtsdUJd@VGUnwb2O>6k3v#;9L)tf~4yAfax*0g@ZLz4;cB+FC^F6irFrrwK(7 z`PEAGp}<3JJUP2LQBmz@@^6N#@wy3JI$VG1F!{akQV8z{+nHB|>_Qs^O#mWmtur5u zQh98sM-iVj{*uZxxysvDy%lbaj9n+9-UR8~#DwaTd z_Qfo1NZNe#Ac#V;ExMKCjqQ@G$zVjx0bao!o-Db`bnky`X=MEwzH#^SDGffc6Y3HI zuTze?*!S}{;A4;ryewK545T^9`;_gy++(m8?9+W$ApRJtypL^1IN8ewKA_hH@JW0s zT<4|88MXOH2mrX{DiX6-knB#cp`)L_HE4ebWn*1Kp{||WLHS6R`g#b%)Fv>3hwEC! zWn`dH>Vq0GQv-iSr@%D3tiT$&hEv^o-{2K&sbTAII$6WE;f|j4HEijj?XDWOb4c?y z^<73c7y0!%4%TqN~UFd8aOT?lPPD(^^zDtkIM>WDD*bkwltD()mVd*AOHT*F<8;Up9*GuXlQ zKBUM?7x~!9$HrbVndFI{vEDfX8$f8$2Cmi z?>LTs;zY>ukvf0tEPsGUsEz+P;2-!jbtP!=uOahE+I^1}hPk#Ef1-^sCP)sK@EO|b zz+qzChva_&!-33>^14lo(^)$s9493(=w z>xnfJ!rW6AT8G&g!Q@69Bcue?@p%v_^(n2C_LAKGdIC%hT*KD&;oYZWcDv~h$XyrC vVH=*rZkSCbd)H0&k};KkUD%F1eNTiQXCVHC-5QQ^d@_W7Dm-8c{0#74^PV&% literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/web/service/SysPermissionService.class b/ruoyi-framework/target/classes/com/ruoyi/framework/web/service/SysPermissionService.class new file mode 100644 index 0000000000000000000000000000000000000000..c598dbf64351668782c2bee528857f56b2f64d5c GIT binary patch literal 2620 zcmbtWU2_vv7=BKgWYcD$P@oV|0YU8-T}4D_N^4sS6w?9?6pNpmCFreJ zPJaMgdEpGVI^)1pXB@BjNsM35*-f&6B2&i6%-M6^ec$)I&+~q49{>I8?*LBX+aBaF zso}#OBrw^7DNJWDlf_4v?ZyF|Q=?UCiKPB`IVvj|H_{Qp@Emu3%9uMGaS# z-L)R{VJVC2_(biN)$WFdPc?idkoDZEEK2XTRh9xng|cfKUfpe2hTrf5X&Zj@&X_AU z{KZ%|FOXZTJAq}(CCj%;RXOE2ZeRwM>-Ykv3$C|f_%+XRR&IKxE!SOd%_vFJ@r|2i zIdHv(VYUs7se0h9Tb``YZPu1feY+9O$GS94k6VrvObB!h3@!;IXWR-w^%g8gF4XOk z^sbr=RG_cmmd)yt=~?RB6efdJi;6KQ4GxT+Pql{Nz?GTKzGqvam=9Z zk=Vumh4(-1RK%)0cM{vJ04f1NCe{8WI8sR>lwBQw3zo0gPSqlhlh3>w-!A>3>xsOE zFEo59&{K5lURj>ARD8MT)y=3%xazy-wF~>tlWxEq88fb3a~+mTUdLe^VYS7Ta>4Xh zSuGmA(qW=R7-q$`941Fc85JF`;GhnPn;KShtisZ93u{yx8skq#6*dFb;h?0$MNP-o z@EE@iUr7Sg1%~6f@WL`y1n77jhjiRl7q08Lqb_XVu8wbD37m{QZNI&ktq^Zmj+aWe zWI13h#uN~~z|Q!r^2s)j1#_g*a$! zUQ;<7EHiFZIp*vNW6m_k78|J@a;Vqr2{JHi*MbK7%(w2cjyYTB1Tt2e(f$_F(5%HY z(Q2!vzaZ~06G}_dw^%ydA-b=>^F~XQe!;23`&YKtW8Vq8H0c7|brd|Ta zX{*AEB%_oVlyoEUCk|?OXAeyjH7BU@UMp&`Ot%{48(X5zwxaeKu)c776FUU%;|e!B zSr|X!Ja@eUW64}{6T1ZN;e^`l7We~wV<|%2lS@6s-c9UFU@V`uzVdMl60cVIQPe#s?ktybDngC|`N9jDr<^>MMIS#&ySjPgsr&OhBf=M&6 zi~r3=xVu7|_i>8s9>yBk`XJHi6@BZPGCGF{%pG%CxCwelkKXJ literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/web/service/SysRegisterService.class b/ruoyi-framework/target/classes/com/ruoyi/framework/web/service/SysRegisterService.class new file mode 100644 index 0000000000000000000000000000000000000000..152d871fdb65705b682ad10762a60eecbb1e5368 GIT binary patch literal 4175 zcmb7HYjhOV9sf;|%`BS<SY#h@qh-(Go(dwCH3q*$mm8S$1Xv z+n1#lN);4aERSk2iilPOB;ABWjBRag?d$lZpZcL6vIai+r9Gb0)7S6r>?WIJK|I;q zx%bZhb$|cYo%jAb@)m%#_)`Ft=<-7kpc{r5dSrwG@S#f#y>sw6m?hYSKJhLrhDd;W zRsc4le(d%m7C;>w8Ll63Kl=UHBbfGp0R&1A0 z)MR(_VDnl{%V>{Bx{TO1jmAm{w%WQD-mb+=vCeE3xxFT>Uz?+H%KAM<*CZMQW2+G| ziBJ|is`ke+TN0Y9azqf*h;6C59W&H!JEEDE8lhETH7jD1-JRI5qIH+nuc=|p3aK5g z0FEMMirgI5LK38hBq&)0fvkW=F{aDprDR>^IB&?juvpmr; z9ZCfQctOG4_`Hl46`aOP3eE^gFN@(T_^OPv3SPmh63Qozp8eq9Q0mCx@!`W)AG zJTmdhh=fI%{qqMVp5K@IO2NShr2hUK$wP(gsBWF@$ z`>zd*rk*`ppE`Jbnv|7QPxJ$-41+U}0 zf(y7vC(KNWO>r~aEyA&MeC)Z@x#Ql+DiyqeOA?k=rcPhHI{4CuZyx&K(%IC3Lo_$_ z)KF^Z4U*DO1%m>#U*H89Nd-d~R&ZIMJAzRORUVHgqUP%Pp{x6jPFxxlztK|@uMbQd zIWRsll6sm&^-Tp=@Ro$zua8^~YW{4d(?uf7>avMk=frj0a0qPK)Iq(7lrg5@ZTwQk zuN3?m?ZZ9fG9}bcvD|FImB5OcZC0Fpe2HbZ3mL8m zXs^mGsLES&1S7SlZKtu{=*v5(TGv$`sbinSFCwkCL#!F~Tf4Sq!t~kqwJA zUZ!@hXNg&7x_~X4lAbwe+%>~arcHE;IT=l-ITCeQ3w2Lz|d)#yW=d7jUUtGrkAn&4X)VO9oK~U(~Qp4WPwMq7L6Jft9Wfb z=w~&AOnDI9*tN6qEqvH+>V2LM@35G3ZTu3cu3HNfvD={{6yob||_*LWQ-bcZL zwv|Z~wsRn+D!66hk_$qw0NyycM@L$FpHlyP5;u9#Vt|eK3|0PK%wZkYqfjPms zZ7<_uMNwuokE<8STjm=e37=&PPLn9e?L!qsT>A%5nx1$s=JSOUTtH3b6~4>(Q~{D$ zcmhjDv51yb1V5d`Vv1QDyoEy&OGtii+PGUQd_w8cBtDa6zO4Z3<#PEX^(`aaEyK9I zHu%}#^1>^q97dJ*Sdm0^%>^F45CX0qRB&8@xv0iMti%%3po*VpwZyjp^`u#YFY;sb zetxcXVm)@!Yd!Q=m>%0hibv_K{k#|+r?&>_n-RkM4!Pc^1O7_j|3=XNiI%iGwi5F9 zaR=qK@G?s9Gma6nE%*{$Q;2`yKD06-{P+-UoE6ftYiQ@Ji1z)Rc054WRpBrAGNlw_ zE9F!p@IRU&%jl4?O-5(-$Mk?wTILJ->i&&N>0kV^mBek;3j+L^(u>L8k_Ed)fE_Ft zM(x|E^YCsc3aSFz4c=v2t8D9O`mTY(d@)> z>RX8iiQ^99^EIA+XBv%-p6*EuTGHng!P`P=fq|Gkl*V8QCG1FJupG-MxPgcP@_o3l-?9KDYmJ<~Ml-KJ?AyWg7`&90;c6!Y`$ z{IB=E_r2r)cJ!&2KJ;M#cgm{_{A>~@@pKY`_ayON{G2|$&%n@RvILD;@rI5}yx-Ur6GM_)-J@27gCetfk7U&Fr`_*Z4)-x8Qp#s8kff8am$+;{@^LqjSuOpPAopEiiVkwy5Qqq3Bkag_| zLJ`Q2VB^y1@&dOtjtmVC4~?H4ALt(&J;t3UIM+|Ko|X4}0HGgizFgisGBkQ>VnDDV zJRFSAP7I6+)>n`+?D&+_9nI#m-o1jF_Kwqnc)v5H)(mI!_GoE#(sm~-GNpVv87p_% za5 zR8*JAS=m`a+%lYX(&vj#o~V33rThIA?unO5*(rfhB9;g3IWpl)o?jk7W0U9YjOXw5 z0aqHlCLD$vSRmNFMpGGCK2ADz+_N$lMy!HQItz<|msvcRIgEG~3PE4M-?Cu;wy83U z%Y3r0{oCVOgQq@M_Nhr%M4k||RD;MP4NMWsv;Fw+4Jk*`8i?(vi03HD*9DU$FPlq~ zHF|B+^4Og#EXD7E)LrQNN3Zv&$)JSCR|in zHOhOZxq}pGl!ec`hwV~&GV5Kkau;kjtt*!9Qsl*Sbcp(5*@EEKkiy8Ys@B(!XQ%U) zS8|!NLn}_K(Y~s3lLgByF8Qgunm4bVVa-f;H7C!idfatpLuziXDy2%lzQL)gBci=&X7>6g}Iu9ghK(?ssMj zPM+SQySmFkeB*Lx;vA+;`~==@urP{9%*2o3UG#TZN|Xvg>8G;bfrYKf&$+W2JZCT1 zbEectoP~a+@g{x}mswN%-G$g! zy7!1FN97(v?lt8;x!;gpQ~IRe#KV{{<(LeZa$FBSAcKYsnY^waG-X&u3>h_LOdet- zs(uTZ@(Mk7QbCTJG9jl-jAMeAQlD+Pw!7Vw(-IRfKF-K11)Iy%2W79Xl+F2VPy2O( z#>&jt6_L{zM2@FdbMde#56dH_JSt7gq9-VNg4@X08gB)rhbHSJtO+>e`A2W-@BGY8KpGrQ)ov@Kxn+&g9cGCC_nk zejAZKF+MgL2(9;=6KpJK;f>)&msFkS(6xEN!Rtd}mr^XU?khHoQ1&-C`DLph5dNy&+Dtf0*4Y65G@Zt-Lc4Il) zndx+Gp+wg%b4I9XwVjptBhwoh8v_+BbZVKIVQ1S^=1gb&_9Q>;x6Opm-oa+NvP{mg zn9s5@Rb1vH*<4QhTf^`1$Ih|lwV$X26l#iEr=SDaSV^kCMVq6j=M6SBlAcNv4pkls z9krT`)iWykTE5XMjNNnSnKt);s7ypT@S;G?d%zdV2oMNyz=fkammx5`J>lI%y*FKZ zHE>q&n@rFXCPQhdhrfZ~oht=LO>l2++Hteq3^TiSaQwi&I|RE{fn2?tU=u|)(r0*7 zFwwr2BNe$B=I;$q8w|+;-z@NFs-G>4qpj5`d`Nq5m!Mi#8cH7oi~OBPKdB`Q4Zpoj zSE(pf5k=CQd>DZfPF$jIJJu+oNN5{ZL(}NcEODlI&o940lCMi#uNb(JKg0_; zE5pP(cFjW-j$AIRVj=V}U1t>vcAoDKmSVQ@jZ@`a5u7L|L&9@{DeU0uu9JKlSBIE( zeH@+QsDa<7`Tv-@-^mcY%MeS%$%2-4qh8IwOINsSBzl!AH+Ehbp?E=ihb**)uM6FzDty@52s%Zi1 zUO>}fqt&=FDpx495eNCEZ$0kDkpEo+1$76ZfbS0L(90bI*oX(P9fN4+u6^8h7art^ z5zdZr^a_qn`39UM$c>cfpnw{l=)`NtZJg(J!sSyfPq(57kCGqYjuI}yP_sgl_308G zr_s7T#}hGpr;n%RWhyibyw<>X8F-z6@Ag0W@IAoG*npa74PxN+0ex=>>1&}J)icoL z6L(){7l|9`E)zG}*`u_@Nn2cLt8J|fXd{Vb+DPV=5uY}L?(8FNdr6Rf2ej>BWC4#* z)vHO{V>Hzw36tnxfb3yV>}5FIM}PK_tbY1deLl$DjX2;_7UPKypE4ycpe)XrfU;Vi z4JfNqQzFU+e4@lBsv9I~FY0_Us<0Fb79xt?h$jMy80j(o12&VQ4VRH(oTYrS5*l(F zHDCiHoA?cSZUa5FiK^9zO~KI!(q?=wJsUu(fo$UAoA6|)X@;9sL`&d|_3Dh|NUDKB z-F&93C&jn{9ckc=Bbrnfl(0?+dj)#bYXvN?S&{ zy63S;?b_@c_dK>dd=;&lN!^RMsRm~>9NMmMdtmc6>I2?R0YA+5y^8g&f4OJB;Ku28x64r_3LqG)?wk<}I8Nyft#gla!<`-a&vj&tq%v z0=6BDZ{HNRpT#X4zw;@qbjl{C7VtFF?>!{vy%hXD67~KHU^^mUJ3?Ry#?_-NTv`O0 z@isoi{TMPJ-@^8IoS68YA*AmOWI&y^pZslG!1j5fBfOo2)A<~3eN!`bWi5aHcBM2G zx|F+jmup>s@);!YETj85bm4<6HCH3IQhbDRe~7Y+HU+L0eZLXHxlGjR%|i=lPjxJ! zGsZvPbd7MdfbP+*o_X|i^(Nk2~~n|2*!T$AL1az_+Z5{98^7KEmq# zQG)vzdhl_UoKI9xcqjsQCE6Vis@SxernC1<5*eSx`4a77I82} z!|x_vhZb?T28-a`KGIzdvmoR@P0K#RQ28vq__+#hwnpl=`eCHu6AMFYDWIr|Yi}V- z@1)uoLo}vuyh7F< z!34S97J+FCvrh2ixEv}PqYSmXF)Xx;=oOSjCw&xOWsrZZ!oJ2xWMe3@j>`2aiBT+H z;)TMmgrpwtO7$(GKSl}{aO^Tnf6Pb-$4y_fTKZue2Z!kN)$sBr-E`A9|c*h(QsX%)?LVG;Cvg&b9=ncKHlBjtRt-Pc*Y$1NW zAq>$Ea-=?JejX=SbRJBN%wv=pGdhp4GDCh>AaxalH%I!K{TqT^G@go@i=DnqYHh|t f%)$@g77cLm9T{{O`?=$%I2XfD`#q4({tWQ{o2H~1 literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/web/service/UserDetailsServiceImpl.class b/ruoyi-framework/target/classes/com/ruoyi/framework/web/service/UserDetailsServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..644d33139e2d6e2cc17b7a9f14ec970d26494e3d GIT binary patch literal 3912 zcmbtX`*#%O8GeR@%w{qgSQSK20gEIgW@t>25r(7EshyC%a$bgxQ&OW)?`T zH$=r2)E2Qosio8l-W4@LW82eT{X2}%pZX8<^z`{=_A=Q`i8;+V`^|ip_x3#RyziH9 z{`1w}0CeGF4XxPKh_y&JVK??@*o*t+^nnESHQ_-#q~c)}KiANRT?za`J|9WoQTa4A zBrx2F46+&rpR?t-N zg_joI0V^XEtV?CwoZ%JRam(<>{XpalZp%GmAU*Eyshp+gvRwryuySIrZnZWhN4YMp@^b`X3pyi1y)h^BpFK78a z6_x$k5c8{|?w}YhyGQSKR?AUYbKdlQ+Rm0$b$u^YGo`3#QPvF1bjHn#vZl^CHT71> zP*1~V%dvv3`JE+ieodkgSYp?=?M3&Wnrn!DChkZiS`U z(U3&-iompNzv%8jE^pI((ss?P+}$=VM~BQP*wK<2GY^=CZ91byI*^=ox6VqSFERzs z3dW6$>j|Sk4cW+w5wl1$3Aq(1PuA*Y*m7UN9WewMQqFZE#aTCJT8<%@UbIzAJgU7W z9hjN@gJwR&L&ft7w7x?bkq@&@#S26-;|Vhm%x}w)J}DS$nU|ISFYxPRU}f3|vMI|C zm^={%g}|~6xx!GEq}({a<75iXIIiQYwDd=uxSHl? z&k`tV)}-uoIMZ<_x>UTO<4-uJ<30@N_z^bf_%qJy_zTXfcvHt)cw0ex4T#~#SNukD ztRGCOQo8H-Av#sOqvHbJ)$yLxb{F1Pu%V)5J|#oN2Rc5)N6g_0J3EHQM22a7PuNeY zYW(*q5yeECEkokL%`5;4?w!9uS%ik%px-Sx*;uZ%RN%OJpHpzXFKmV@zpSWe4uwUt zXV#)wQOmbE>)Gm>D!h3vrX|zWi5Yfo#oOGx*-X=lXDsmA__X3W#8DaomZ6)Y9v#4DY~hR);%+)cVWk`OsN>o{)_slz zZC|2p0S4PIq5d>FE@EM_flF!~rjXb-g~ol8(2`9P(A)S+Uc+Gm*G{7O?-#LX0*l)x zketAEMbYbPMQ8eSj8Rr*O**;2`!&t Zq0A8_geuCB6~ literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/websocket/SemaphoreUtils.class b/ruoyi-framework/target/classes/com/ruoyi/framework/websocket/SemaphoreUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..0d2b37f92897d211e192f2e8475bf75cbc0b1351 GIT binary patch literal 1231 zcmb7DOHb5L7(I6w=sb$R@K8Zz5JX;56%cTUi6O#fkVMplyHYPhp)>6ki?k;x^dP(PaJXd z#?Y5UKL#{-E`~t^Rs!d7!N5ggl~L5S{ezN`bnpTaedF`FW{s+VdU}20ONQ+g7gLSHNhn=oSSAvyPv)RLNg+tp#Q0 z<*KiityMW+^qplHSaULOuPpjX-U?i=$U_r>TD$7pE4hlwojutl`@U_J_|bCwf>TmT z7WPGDdhZAbGMjgao?XaU*B&@>C2;)$<>JMk2MDD9CdM2WJUd54s@Bd0wzE8AuY?I1 z7-B^Q3?)5j7yr#0ix!)3yh;@lbACxVa>~^;Ydo0vpia}o5!CS@6U`VVB~1NbMkY+u zGTX0Dw#pmNK7V{(-q_k%-zdL-Sl-??Fk<2oMopwqN7Byv<1bG?93lw}{7**=jF}k6 zWr6x%c4p_7q!S37`g3!_!C@uq56_lBwN%Pi0)3SRhgr@A$}QwF`#O1ZQTeNSs4{v0 zj~>>UvYo(JYwU7P2D@V`aAv>cq-Ph4ncm;08y);PlKg1|T6ImiigL{c@LA8D#kb(Q zx^D|&lZyx%_^b&*6Dh9ET>=`>#9f?SYayb8AEAmjaI9*g?+qdoFHkr03ei;cz+2Q* zq3tDk5g15>JgwVGWuT2{JMk`l>7cz%Dt3owrO8b4K8j=HP#MP=qg}Ny>IC&pa@T;! zcZNm`oT7UJ?Md27Q)<^%o+3K9iKYyC%4$!s{?OAVIl2bMo~0I46B!~lMsx%*j55{n y&`t|=Vih|&uBl>2n=lZW4v&uE^j|HMN_RY}U(P>64(~2l!Fq z?6zLZxPE%Nw?)zZgovWCE*8>xAo0uEL&#Z{#m9ZE6O8T)0E~Mip z(r;wZuF+?zwZSkGOM5A8O^02hw5wfrjdjv8>@Qa9#VJq4qCrQ)hqr3x@Huc-|4Z#Si_q3>+4ar%6dy4m2Pr zk}c7WlV{`ym+rytvg0ALg$yjiqj;p3fMJS2aT%i|l9E?5F`)E_-03|AZVAibgJ1c? WFZKCjl#{BDXSoxZt>6uv1YQ9c-^TR- literal 0 HcmV?d00001 diff --git a/ruoyi-framework/target/classes/com/ruoyi/framework/websocket/WebSocketServer.class b/ruoyi-framework/target/classes/com/ruoyi/framework/websocket/WebSocketServer.class new file mode 100644 index 0000000000000000000000000000000000000000..2fbc53c76d7023d0f4ad11db08b1ce25a1968322 GIT binary patch literal 3400 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 literal 0 HcmV?d00001 diff --git a/ruoyi-generator/pom.xml b/ruoyi-generator/pom.xml new file mode 100644 index 00000000..f570d810 --- /dev/null +++ b/ruoyi-generator/pom.xml @@ -0,0 +1,46 @@ + + + + ruoyi + com.ruoyi + 3.8.4 + + 4.0.0 + + ruoyi-generator + + + generator代码生成 + + + + + + + org.apache.velocity + velocity-engine-core + + + + + commons-collections + commons-collections + + + + + com.ruoyi + ruoyi-common + + + + + com.baomidou + mybatis-plus-generator + + + + + diff --git a/ruoyi-generator/src/main/java/com/ruoyi/generator/config/GenConfig.java b/ruoyi-generator/src/main/java/com/ruoyi/generator/config/GenConfig.java new file mode 100644 index 00000000..cc4cd14c --- /dev/null +++ b/ruoyi-generator/src/main/java/com/ruoyi/generator/config/GenConfig.java @@ -0,0 +1,73 @@ +package com.ruoyi.generator.config; + +import org.springframework.beans.factory.annotation.Value; +import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.context.annotation.PropertySource; +import org.springframework.stereotype.Component; + +/** + * 读取代码生成相关配置 + * + * @author ruoyi + */ +@Component +@ConfigurationProperties(prefix = "gen") +@PropertySource(value = { "classpath:generator.yml" }) +public class GenConfig +{ + /** 作者 */ + public static String author; + + /** 生成包路径 */ + public static String packageName; + + /** 自动去除表前缀,默认是false */ + public static boolean autoRemovePre; + + /** 表前缀(类名不会包含表前缀) */ + public static String tablePrefix; + + public static String getAuthor() + { + return author; + } + + @Value("${author}") + public void setAuthor(String author) + { + GenConfig.author = author; + } + + public static String getPackageName() + { + return packageName; + } + + @Value("${packageName}") + public void setPackageName(String packageName) + { + GenConfig.packageName = packageName; + } + + public static boolean getAutoRemovePre() + { + return autoRemovePre; + } + + @Value("${autoRemovePre}") + public void setAutoRemovePre(boolean autoRemovePre) + { + GenConfig.autoRemovePre = autoRemovePre; + } + + public static String getTablePrefix() + { + return tablePrefix; + } + + @Value("${tablePrefix}") + public void setTablePrefix(String tablePrefix) + { + GenConfig.tablePrefix = tablePrefix; + } +} diff --git a/ruoyi-generator/src/main/java/com/ruoyi/generator/controller/GenController.java b/ruoyi-generator/src/main/java/com/ruoyi/generator/controller/GenController.java new file mode 100644 index 00000000..1354eb1a --- /dev/null +++ b/ruoyi-generator/src/main/java/com/ruoyi/generator/controller/GenController.java @@ -0,0 +1,194 @@ +package com.ruoyi.generator.controller; + +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.page.TableDataInfo; +import com.ruoyi.common.core.text.Convert; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.generator.domain.GenTable; +import com.ruoyi.generator.domain.GenTableColumn; +import com.ruoyi.generator.service.IGenTableColumnService; +import com.ruoyi.generator.service.IGenTableService; +import org.apache.commons.io.IOUtils; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.validation.annotation.Validated; +import org.springframework.web.bind.annotation.*; + +import javax.servlet.http.HttpServletResponse; +import java.io.IOException; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +/** + * 代码生成 操作处理 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/tool/gen") +public class GenController extends BaseController { + @Autowired + private IGenTableService genTableService; + + @Autowired + private IGenTableColumnService genTableColumnService; + + /** + * 查询代码生成列表 + */ + @PreAuthorize("@ss.hasPermi('tool:gen:list')") + @GetMapping("/list") + public TableDataInfo genList(GenTable genTable) { + startPage(); + List list = genTableService.selectGenTableList(genTable); + return getDataTable(list); + } + + /** + * 修改代码生成业务 + */ + @PreAuthorize("@ss.hasPermi('tool:gen:query')") + @GetMapping(value = "/{tableId}") + public AjaxResult getInfo(@PathVariable Long tableId) { + GenTable table = genTableService.selectGenTableById(tableId); + List tables = genTableService.selectGenTableAll(); + List list = genTableColumnService.selectGenTableColumnListByTableId(tableId); + Map map = new HashMap(); + map.put("info", table); + map.put("rows", list); + map.put("tables", tables); + return success(map); + } + + /** + * 查询数据库列表 + */ + @PreAuthorize("@ss.hasPermi('tool:gen:list')") + @GetMapping("/db/list") + public TableDataInfo dataList(GenTable genTable) { + startPage(); + List list = genTableService.selectDbTableList(genTable); + return getDataTable(list); + } + + /** + * 查询数据表字段列表 + */ + @PreAuthorize("@ss.hasPermi('tool:gen:list')") + @GetMapping(value = "/column/{tableId}") + public TableDataInfo columnList(Long tableId) { + TableDataInfo dataInfo = new TableDataInfo(); + List list = genTableColumnService.selectGenTableColumnListByTableId(tableId); + dataInfo.setRows(list); + dataInfo.setTotal(list.size()); + return dataInfo; + } + + /** + * 导入表结构(保存) + */ + @PreAuthorize("@ss.hasPermi('tool:gen:import')") + @Log(title = "代码生成", businessType = BusinessType.IMPORT) + @PostMapping("/importTable") + public AjaxResult importTableSave(String tables) { + String[] tableNames = Convert.toStrArray(tables); + // 查询表信息 + List tableList = genTableService.selectDbTableListByNames(tableNames); + genTableService.importGenTable(tableList); + return success(); + } + + /** + * 修改保存代码生成业务 + */ + @PreAuthorize("@ss.hasPermi('tool:gen:edit')") + @Log(title = "代码生成", businessType = BusinessType.UPDATE) + @PutMapping + public AjaxResult editSave(@Validated @RequestBody GenTable genTable) { + genTableService.validateEdit(genTable); + genTableService.updateGenTable(genTable); + return success(); + } + + /** + * 删除代码生成 + */ + @PreAuthorize("@ss.hasPermi('tool:gen:remove')") + @Log(title = "代码生成", businessType = BusinessType.DELETE) + @DeleteMapping("/{tableIds}") + public AjaxResult remove(@PathVariable Long[] tableIds) { + genTableService.deleteGenTableByIds(tableIds); + return success(); + } + + /** + * 预览代码 + */ + @PreAuthorize("@ss.hasPermi('tool:gen:preview')") + @GetMapping("/preview/{tableId}") + public AjaxResult preview(@PathVariable("tableId") Long tableId) throws IOException { + Map dataMap = genTableService.previewCode(tableId); + return success(dataMap); + } + + /** + * 生成代码(下载方式) + */ + @PreAuthorize("@ss.hasPermi('tool:gen:code')") + @Log(title = "代码生成", businessType = BusinessType.GENCODE) + @GetMapping("/download/{tableName}") + public void download(HttpServletResponse response, @PathVariable("tableName") String tableName) throws IOException { + byte[] data = genTableService.downloadCode(tableName); + genCode(response, data); + } + + /** + * 生成代码(自定义路径) + */ + @PreAuthorize("@ss.hasPermi('tool:gen:code')") + @Log(title = "代码生成", businessType = BusinessType.GENCODE) + @GetMapping("/genCode/{tableName}") + public AjaxResult genCode(@PathVariable("tableName") String tableName) { + genTableService.generatorCode(tableName); + return success(); + } + + /** + * 同步数据库 + */ + @PreAuthorize("@ss.hasPermi('tool:gen:edit')") + @Log(title = "代码生成", businessType = BusinessType.UPDATE) + @GetMapping("/synchDb/{tableName}") + public AjaxResult synchDb(@PathVariable("tableName") String tableName) { + genTableService.synchDb(tableName); + return success(); + } + + /** + * 批量生成代码 + */ + @PreAuthorize("@ss.hasPermi('tool:gen:code')") + @Log(title = "代码生成", businessType = BusinessType.GENCODE) + @GetMapping("/batchGenCode") + public void batchGenCode(HttpServletResponse response, String tables) throws IOException { + String[] tableNames = Convert.toStrArray(tables); + byte[] data = genTableService.downloadCode(tableNames); + genCode(response, data); + } + + /** + * 生成zip文件 + */ + private void genCode(HttpServletResponse response, byte[] data) throws IOException { + response.reset(); + response.addHeader("Access-Control-Allow-Origin", "*"); + response.addHeader("Access-Control-Expose-Headers", "Content-Disposition"); + response.setHeader("Content-Disposition", "attachment; filename=\"ruoyi.zip\""); + response.addHeader("Content-Length", "" + data.length); + response.setContentType("application/octet-stream; charset=UTF-8"); + IOUtils.write(data, response.getOutputStream()); + } +} diff --git a/ruoyi-generator/src/main/java/com/ruoyi/generator/domain/GenTable.java b/ruoyi-generator/src/main/java/com/ruoyi/generator/domain/GenTable.java new file mode 100644 index 00000000..269779cf --- /dev/null +++ b/ruoyi-generator/src/main/java/com/ruoyi/generator/domain/GenTable.java @@ -0,0 +1,372 @@ +package com.ruoyi.generator.domain; + +import java.util.List; +import javax.validation.Valid; +import javax.validation.constraints.NotBlank; +import org.apache.commons.lang3.ArrayUtils; +import com.ruoyi.common.constant.GenConstants; +import com.ruoyi.common.core.domain.BaseEntity; +import com.ruoyi.common.utils.StringUtils; + +/** + * 业务表 gen_table + * + * @author ruoyi + */ +public class GenTable extends BaseEntity +{ + private static final long serialVersionUID = 1L; + + /** 编号 */ + private Long tableId; + + /** 表名称 */ + @NotBlank(message = "表名称不能为空") + private String tableName; + + /** 表描述 */ + @NotBlank(message = "表描述不能为空") + private String tableComment; + + /** 关联父表的表名 */ + private String subTableName; + + /** 本表关联父表的外键名 */ + private String subTableFkName; + + /** 实体类名称(首字母大写) */ + @NotBlank(message = "实体类名称不能为空") + private String className; + + /** 使用的模板(crud单表操作 tree树表操作 sub主子表操作) */ + private String tplCategory; + + /** 生成包路径 */ + @NotBlank(message = "生成包路径不能为空") + private String packageName; + + /** 生成模块名 */ + @NotBlank(message = "生成模块名不能为空") + private String moduleName; + + /** 生成业务名 */ + @NotBlank(message = "生成业务名不能为空") + private String businessName; + + /** 生成功能名 */ + @NotBlank(message = "生成功能名不能为空") + private String functionName; + + /** 生成作者 */ + @NotBlank(message = "作者不能为空") + private String functionAuthor; + + /** 生成代码方式(0zip压缩包 1自定义路径) */ + private String genType; + + /** 生成路径(不填默认项目路径) */ + private String genPath; + + /** 主键信息 */ + private GenTableColumn pkColumn; + + /** 子表信息 */ + private GenTable subTable; + + /** 表列信息 */ + @Valid + private List columns; + + /** 其它生成选项 */ + private String options; + + /** 树编码字段 */ + private String treeCode; + + /** 树父编码字段 */ + private String treeParentCode; + + /** 树名称字段 */ + private String treeName; + + /** 上级菜单ID字段 */ + private String parentMenuId; + + /** 上级菜单名称字段 */ + private String parentMenuName; + + public Long getTableId() + { + return tableId; + } + + public void setTableId(Long tableId) + { + this.tableId = tableId; + } + + public String getTableName() + { + return tableName; + } + + public void setTableName(String tableName) + { + this.tableName = tableName; + } + + public String getTableComment() + { + return tableComment; + } + + public void setTableComment(String tableComment) + { + this.tableComment = tableComment; + } + + public String getSubTableName() + { + return subTableName; + } + + public void setSubTableName(String subTableName) + { + this.subTableName = subTableName; + } + + public String getSubTableFkName() + { + return subTableFkName; + } + + public void setSubTableFkName(String subTableFkName) + { + this.subTableFkName = subTableFkName; + } + + public String getClassName() + { + return className; + } + + public void setClassName(String className) + { + this.className = className; + } + + public String getTplCategory() + { + return tplCategory; + } + + public void setTplCategory(String tplCategory) + { + this.tplCategory = tplCategory; + } + + public String getPackageName() + { + return packageName; + } + + public void setPackageName(String packageName) + { + this.packageName = packageName; + } + + public String getModuleName() + { + return moduleName; + } + + public void setModuleName(String moduleName) + { + this.moduleName = moduleName; + } + + public String getBusinessName() + { + return businessName; + } + + public void setBusinessName(String businessName) + { + this.businessName = businessName; + } + + public String getFunctionName() + { + return functionName; + } + + public void setFunctionName(String functionName) + { + this.functionName = functionName; + } + + public String getFunctionAuthor() + { + return functionAuthor; + } + + public void setFunctionAuthor(String functionAuthor) + { + this.functionAuthor = functionAuthor; + } + + public String getGenType() + { + return genType; + } + + public void setGenType(String genType) + { + this.genType = genType; + } + + public String getGenPath() + { + return genPath; + } + + public void setGenPath(String genPath) + { + this.genPath = genPath; + } + + public GenTableColumn getPkColumn() + { + return pkColumn; + } + + public void setPkColumn(GenTableColumn pkColumn) + { + this.pkColumn = pkColumn; + } + + public GenTable getSubTable() + { + return subTable; + } + + public void setSubTable(GenTable subTable) + { + this.subTable = subTable; + } + + public List getColumns() + { + return columns; + } + + public void setColumns(List columns) + { + this.columns = columns; + } + + public String getOptions() + { + return options; + } + + public void setOptions(String options) + { + this.options = options; + } + + public String getTreeCode() + { + return treeCode; + } + + public void setTreeCode(String treeCode) + { + this.treeCode = treeCode; + } + + public String getTreeParentCode() + { + return treeParentCode; + } + + public void setTreeParentCode(String treeParentCode) + { + this.treeParentCode = treeParentCode; + } + + public String getTreeName() + { + return treeName; + } + + public void setTreeName(String treeName) + { + this.treeName = treeName; + } + + public String getParentMenuId() + { + return parentMenuId; + } + + public void setParentMenuId(String parentMenuId) + { + this.parentMenuId = parentMenuId; + } + + public String getParentMenuName() + { + return parentMenuName; + } + + public void setParentMenuName(String parentMenuName) + { + this.parentMenuName = parentMenuName; + } + + public boolean isSub() + { + return isSub(this.tplCategory); + } + + public static boolean isSub(String tplCategory) + { + return tplCategory != null && StringUtils.equals(GenConstants.TPL_SUB, tplCategory); + } + + public boolean isTree() + { + return isTree(this.tplCategory); + } + + public static boolean isTree(String tplCategory) + { + return tplCategory != null && StringUtils.equals(GenConstants.TPL_TREE, tplCategory); + } + + public boolean isCrud() + { + return isCrud(this.tplCategory); + } + + public static boolean isCrud(String tplCategory) + { + return tplCategory != null && StringUtils.equals(GenConstants.TPL_CRUD, tplCategory); + } + + public boolean isSuperColumn(String javaField) + { + return isSuperColumn(this.tplCategory, javaField); + } + + public static boolean isSuperColumn(String tplCategory, String javaField) + { + if (isTree(tplCategory)) + { + return StringUtils.equalsAnyIgnoreCase(javaField, + ArrayUtils.addAll(GenConstants.TREE_ENTITY, GenConstants.BASE_ENTITY)); + } + return StringUtils.equalsAnyIgnoreCase(javaField, GenConstants.BASE_ENTITY); + } +} \ No newline at end of file diff --git a/ruoyi-generator/src/main/java/com/ruoyi/generator/domain/GenTableColumn.java b/ruoyi-generator/src/main/java/com/ruoyi/generator/domain/GenTableColumn.java new file mode 100644 index 00000000..d1733b64 --- /dev/null +++ b/ruoyi-generator/src/main/java/com/ruoyi/generator/domain/GenTableColumn.java @@ -0,0 +1,373 @@ +package com.ruoyi.generator.domain; + +import javax.validation.constraints.NotBlank; +import com.ruoyi.common.core.domain.BaseEntity; +import com.ruoyi.common.utils.StringUtils; + +/** + * 代码生成业务字段表 gen_table_column + * + * @author ruoyi + */ +public class GenTableColumn extends BaseEntity +{ + private static final long serialVersionUID = 1L; + + /** 编号 */ + private Long columnId; + + /** 归属表编号 */ + private Long tableId; + + /** 列名称 */ + private String columnName; + + /** 列描述 */ + private String columnComment; + + /** 列类型 */ + private String columnType; + + /** JAVA类型 */ + private String javaType; + + /** JAVA字段名 */ + @NotBlank(message = "Java属性不能为空") + private String javaField; + + /** 是否主键(1是) */ + private String isPk; + + /** 是否自增(1是) */ + private String isIncrement; + + /** 是否必填(1是) */ + private String isRequired; + + /** 是否为插入字段(1是) */ + private String isInsert; + + /** 是否编辑字段(1是) */ + private String isEdit; + + /** 是否列表字段(1是) */ + private String isList; + + /** 是否查询字段(1是) */ + private String isQuery; + + /** 查询方式(EQ等于、NE不等于、GT大于、LT小于、LIKE模糊、BETWEEN范围) */ + private String queryType; + + /** 显示类型(input文本框、textarea文本域、select下拉框、checkbox复选框、radio单选框、datetime日期控件、image图片上传控件、upload文件上传控件、editor富文本控件) */ + private String htmlType; + + /** 字典类型 */ + private String dictType; + + /** 排序 */ + private Integer sort; + + public void setColumnId(Long columnId) + { + this.columnId = columnId; + } + + public Long getColumnId() + { + return columnId; + } + + public void setTableId(Long tableId) + { + this.tableId = tableId; + } + + public Long getTableId() + { + return tableId; + } + + public void setColumnName(String columnName) + { + this.columnName = columnName; + } + + public String getColumnName() + { + return columnName; + } + + public void setColumnComment(String columnComment) + { + this.columnComment = columnComment; + } + + public String getColumnComment() + { + return columnComment; + } + + public void setColumnType(String columnType) + { + this.columnType = columnType; + } + + public String getColumnType() + { + return columnType; + } + + public void setJavaType(String javaType) + { + this.javaType = javaType; + } + + public String getJavaType() + { + return javaType; + } + + public void setJavaField(String javaField) + { + this.javaField = javaField; + } + + public String getJavaField() + { + return javaField; + } + + public String getCapJavaField() + { + return StringUtils.capitalize(javaField); + } + + public void setIsPk(String isPk) + { + this.isPk = isPk; + } + + public String getIsPk() + { + return isPk; + } + + public boolean isPk() + { + return isPk(this.isPk); + } + + public boolean isPk(String isPk) + { + return isPk != null && StringUtils.equals("1", isPk); + } + + public String getIsIncrement() + { + return isIncrement; + } + + public void setIsIncrement(String isIncrement) + { + this.isIncrement = isIncrement; + } + + public boolean isIncrement() + { + return isIncrement(this.isIncrement); + } + + public boolean isIncrement(String isIncrement) + { + return isIncrement != null && StringUtils.equals("1", isIncrement); + } + + public void setIsRequired(String isRequired) + { + this.isRequired = isRequired; + } + + public String getIsRequired() + { + return isRequired; + } + + public boolean isRequired() + { + return isRequired(this.isRequired); + } + + public boolean isRequired(String isRequired) + { + return isRequired != null && StringUtils.equals("1", isRequired); + } + + public void setIsInsert(String isInsert) + { + this.isInsert = isInsert; + } + + public String getIsInsert() + { + return isInsert; + } + + public boolean isInsert() + { + return isInsert(this.isInsert); + } + + public boolean isInsert(String isInsert) + { + return isInsert != null && StringUtils.equals("1", isInsert); + } + + public void setIsEdit(String isEdit) + { + this.isEdit = isEdit; + } + + public String getIsEdit() + { + return isEdit; + } + + public boolean isEdit() + { + return isInsert(this.isEdit); + } + + public boolean isEdit(String isEdit) + { + return isEdit != null && StringUtils.equals("1", isEdit); + } + + public void setIsList(String isList) + { + this.isList = isList; + } + + public String getIsList() + { + return isList; + } + + public boolean isList() + { + return isList(this.isList); + } + + public boolean isList(String isList) + { + return isList != null && StringUtils.equals("1", isList); + } + + public void setIsQuery(String isQuery) + { + this.isQuery = isQuery; + } + + public String getIsQuery() + { + return isQuery; + } + + public boolean isQuery() + { + return isQuery(this.isQuery); + } + + public boolean isQuery(String isQuery) + { + return isQuery != null && StringUtils.equals("1", isQuery); + } + + public void setQueryType(String queryType) + { + this.queryType = queryType; + } + + public String getQueryType() + { + return queryType; + } + + public String getHtmlType() + { + return htmlType; + } + + public void setHtmlType(String htmlType) + { + this.htmlType = htmlType; + } + + public void setDictType(String dictType) + { + this.dictType = dictType; + } + + public String getDictType() + { + return dictType; + } + + public void setSort(Integer sort) + { + this.sort = sort; + } + + public Integer getSort() + { + return sort; + } + + public boolean isSuperColumn() + { + return isSuperColumn(this.javaField); + } + + public static boolean isSuperColumn(String javaField) + { + return StringUtils.equalsAnyIgnoreCase(javaField, + // BaseEntity + "createBy", "createTime", "updateBy", "updateTime", "remark", + // TreeEntity + "parentName", "parentId", "orderNum", "ancestors"); + } + + public boolean isUsableColumn() + { + return isUsableColumn(javaField); + } + + public static boolean isUsableColumn(String javaField) + { + // isSuperColumn()中的名单用于避免生成多余Domain属性,若某些属性在生成页面时需要用到不能忽略,则放在此处白名单 + return StringUtils.equalsAnyIgnoreCase(javaField, "parentId", "orderNum", "remark"); + } + + public String readConverterExp() + { + String remarks = StringUtils.substringBetween(this.columnComment, "(", ")"); + StringBuffer sb = new StringBuffer(); + if (StringUtils.isNotEmpty(remarks)) + { + for (String value : remarks.split(" ")) + { + if (StringUtils.isNotEmpty(value)) + { + Object startStr = value.subSequence(0, 1); + String endStr = value.substring(1); + sb.append("").append(startStr).append("=").append(endStr).append(","); + } + } + return sb.deleteCharAt(sb.length() - 1).toString(); + } + else + { + return this.columnComment; + } + } +} diff --git a/ruoyi-generator/src/main/java/com/ruoyi/generator/mapper/GenTableColumnMapper.java b/ruoyi-generator/src/main/java/com/ruoyi/generator/mapper/GenTableColumnMapper.java new file mode 100644 index 00000000..951e1667 --- /dev/null +++ b/ruoyi-generator/src/main/java/com/ruoyi/generator/mapper/GenTableColumnMapper.java @@ -0,0 +1,60 @@ +package com.ruoyi.generator.mapper; + +import java.util.List; +import com.ruoyi.generator.domain.GenTableColumn; + +/** + * 业务字段 数据层 + * + * @author ruoyi + */ +public interface GenTableColumnMapper +{ + /** + * 根据表名称查询列信息 + * + * @param tableName 表名称 + * @return 列信息 + */ + public List selectDbTableColumnsByName(String tableName); + + /** + * 查询业务字段列表 + * + * @param tableId 业务字段编号 + * @return 业务字段集合 + */ + public List selectGenTableColumnListByTableId(Long tableId); + + /** + * 新增业务字段 + * + * @param genTableColumn 业务字段信息 + * @return 结果 + */ + public int insertGenTableColumn(GenTableColumn genTableColumn); + + /** + * 修改业务字段 + * + * @param genTableColumn 业务字段信息 + * @return 结果 + */ + public int updateGenTableColumn(GenTableColumn genTableColumn); + + /** + * 删除业务字段 + * + * @param genTableColumns 列数据 + * @return 结果 + */ + public int deleteGenTableColumns(List genTableColumns); + + /** + * 批量删除业务字段 + * + * @param ids 需要删除的数据ID + * @return 结果 + */ + public int deleteGenTableColumnByIds(Long[] ids); +} diff --git a/ruoyi-generator/src/main/java/com/ruoyi/generator/mapper/GenTableMapper.java b/ruoyi-generator/src/main/java/com/ruoyi/generator/mapper/GenTableMapper.java new file mode 100644 index 00000000..9b330df8 --- /dev/null +++ b/ruoyi-generator/src/main/java/com/ruoyi/generator/mapper/GenTableMapper.java @@ -0,0 +1,83 @@ +package com.ruoyi.generator.mapper; + +import java.util.List; +import com.ruoyi.generator.domain.GenTable; + +/** + * 业务 数据层 + * + * @author ruoyi + */ +public interface GenTableMapper +{ + /** + * 查询业务列表 + * + * @param genTable 业务信息 + * @return 业务集合 + */ + public List selectGenTableList(GenTable genTable); + + /** + * 查询据库列表 + * + * @param genTable 业务信息 + * @return 数据库表集合 + */ + public List selectDbTableList(GenTable genTable); + + /** + * 查询据库列表 + * + * @param tableNames 表名称组 + * @return 数据库表集合 + */ + public List selectDbTableListByNames(String[] tableNames); + + /** + * 查询所有表信息 + * + * @return 表信息集合 + */ + public List selectGenTableAll(); + + /** + * 查询表ID业务信息 + * + * @param id 业务ID + * @return 业务信息 + */ + public GenTable selectGenTableById(Long id); + + /** + * 查询表名称业务信息 + * + * @param tableName 表名称 + * @return 业务信息 + */ + public GenTable selectGenTableByName(String tableName); + + /** + * 新增业务 + * + * @param genTable 业务信息 + * @return 结果 + */ + public int insertGenTable(GenTable genTable); + + /** + * 修改业务 + * + * @param genTable 业务信息 + * @return 结果 + */ + public int updateGenTable(GenTable genTable); + + /** + * 批量删除业务 + * + * @param ids 需要删除的数据ID + * @return 结果 + */ + public int deleteGenTableByIds(Long[] ids); +} diff --git a/ruoyi-generator/src/main/java/com/ruoyi/generator/service/GenTableColumnServiceImpl.java b/ruoyi-generator/src/main/java/com/ruoyi/generator/service/GenTableColumnServiceImpl.java new file mode 100644 index 00000000..0679689d --- /dev/null +++ b/ruoyi-generator/src/main/java/com/ruoyi/generator/service/GenTableColumnServiceImpl.java @@ -0,0 +1,68 @@ +package com.ruoyi.generator.service; + +import java.util.List; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +import com.ruoyi.common.core.text.Convert; +import com.ruoyi.generator.domain.GenTableColumn; +import com.ruoyi.generator.mapper.GenTableColumnMapper; + +/** + * 业务字段 服务层实现 + * + * @author ruoyi + */ +@Service +public class GenTableColumnServiceImpl implements IGenTableColumnService +{ + @Autowired + private GenTableColumnMapper genTableColumnMapper; + + /** + * 查询业务字段列表 + * + * @param tableId 业务字段编号 + * @return 业务字段集合 + */ + @Override + public List selectGenTableColumnListByTableId(Long tableId) + { + return genTableColumnMapper.selectGenTableColumnListByTableId(tableId); + } + + /** + * 新增业务字段 + * + * @param genTableColumn 业务字段信息 + * @return 结果 + */ + @Override + public int insertGenTableColumn(GenTableColumn genTableColumn) + { + return genTableColumnMapper.insertGenTableColumn(genTableColumn); + } + + /** + * 修改业务字段 + * + * @param genTableColumn 业务字段信息 + * @return 结果 + */ + @Override + public int updateGenTableColumn(GenTableColumn genTableColumn) + { + return genTableColumnMapper.updateGenTableColumn(genTableColumn); + } + + /** + * 删除业务字段对象 + * + * @param ids 需要删除的数据ID + * @return 结果 + */ + @Override + public int deleteGenTableColumnByIds(String ids) + { + return genTableColumnMapper.deleteGenTableColumnByIds(Convert.toLongArray(ids)); + } +} diff --git a/ruoyi-generator/src/main/java/com/ruoyi/generator/service/GenTableServiceImpl.java b/ruoyi-generator/src/main/java/com/ruoyi/generator/service/GenTableServiceImpl.java new file mode 100644 index 00000000..f4b12e9c --- /dev/null +++ b/ruoyi-generator/src/main/java/com/ruoyi/generator/service/GenTableServiceImpl.java @@ -0,0 +1,498 @@ +package com.ruoyi.generator.service; + +import com.alibaba.fastjson2.JSON; +import com.alibaba.fastjson2.JSONObject; +import com.baomidou.mybatisplus.generator.config.po.TableField; +import com.baomidou.mybatisplus.generator.config.po.TableInfo; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.constant.GenConstants; +import com.ruoyi.common.core.text.CharsetKit; +import com.ruoyi.common.exception.ServiceException; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.generator.domain.GenTable; +import com.ruoyi.generator.domain.GenTableColumn; +import com.ruoyi.generator.mapper.GenTableColumnMapper; +import com.ruoyi.generator.mapper.GenTableMapper; +import com.ruoyi.generator.util.GenTableUtils; +import com.ruoyi.generator.util.GenUtils; +import com.ruoyi.generator.util.VelocityInitializer; +import com.ruoyi.generator.util.VelocityUtils; +import org.apache.commons.io.FileUtils; +import org.apache.commons.io.IOUtils; +import org.apache.velocity.Template; +import org.apache.velocity.VelocityContext; +import org.apache.velocity.app.Velocity; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; + +import java.io.ByteArrayOutputStream; +import java.io.File; +import java.io.IOException; +import java.io.StringWriter; +import java.util.*; +import java.util.concurrent.atomic.AtomicInteger; +import java.util.function.Function; +import java.util.stream.Collectors; +import java.util.zip.ZipEntry; +import java.util.zip.ZipOutputStream; + +/** + * 业务 服务层实现 + * + * @author ruoyi + */ +@Service +public class GenTableServiceImpl implements IGenTableService { + private static final Logger log = LoggerFactory.getLogger(GenTableServiceImpl.class); + + @Autowired + private GenTableMapper genTableMapper; + + @Autowired + private GenTableColumnMapper genTableColumnMapper; + + /** + * 查询业务信息 + * + * @param id 业务ID + * @return 业务信息 + */ + @Override + public GenTable selectGenTableById(Long id) { + GenTable genTable = genTableMapper.selectGenTableById(id); + setTableFromOptions(genTable); + return genTable; + } + + /** + * 查询业务列表 + * + * @param genTable 业务信息 + * @return 业务集合 + */ + @Override + public List selectGenTableList(GenTable genTable) { + return genTableMapper.selectGenTableList(genTable); + } + + /** + * 查询据库列表 + * + * @param genTable 业务信息 + * @return 数据库表集合 + */ + @Override + public List selectDbTableList(GenTable genTable) { + List tableList = GenTableUtils.getTableList(genTable.getTableName(), genTable.getTableComment()); + // 移除置顶前缀的表名 + tableList.removeIf(table -> table.getName().toUpperCase().startsWith("QRTZ_")); + tableList.removeIf(table -> table.getName().toUpperCase().startsWith("ACT_")); + tableList.removeIf(table -> table.getName().toUpperCase().startsWith("FLW_")); + // 移除已经生成的表 + // 移除在 Codegen 中,已经存在的 + List genTables = genTableMapper.selectGenTableList(new GenTable()); + Set existsTables = CollectionUtils.convertSet(genTables, GenTable::getTableName); + tableList.removeIf(table -> existsTables.contains(table.getName())); + List result = tableList.stream().map(tmp -> { + GenTable data = new GenTable(); + data.setTableName(tmp.getName()); + data.setTableComment(tmp.getComment()); + return data; + }).collect(Collectors.toList()); + return result; + //return genTableMapper.selectDbTableList(genTable); + } + + /** + * 查询据库列表 + * + * @param tableNames 表名称组 + * @return 数据库表集合 + */ + @Override + public List selectDbTableListByNames(String[] tableNames) { + List result = new ArrayList<>(tableNames.length); + for (String tableName : tableNames) { + TableInfo table = GenTableUtils.getTable(tableName); + GenTable data = new GenTable(); + data.setTableName(table.getName()); + data.setTableComment(table.getComment()); + result.add(data); + } + return result; + //return genTableMapper.selectDbTableListByNames(tableNames); + } + + /** + * 查询所有表信息 + * + * @return 表信息集合 + */ + @Override + public List selectGenTableAll() { + return genTableMapper.selectGenTableAll(); + } + + /** + * 修改业务 + * + * @param genTable 业务信息 + * @return 结果 + */ + @Override + @Transactional + public void updateGenTable(GenTable genTable) { + String options = JSON.toJSONString(genTable.getParams()); + genTable.setOptions(options); + int row = genTableMapper.updateGenTable(genTable); + if (row > 0) { + for (GenTableColumn cenTableColumn : genTable.getColumns()) { + genTableColumnMapper.updateGenTableColumn(cenTableColumn); + } + } + } + + /** + * 删除业务对象 + * + * @param tableIds 需要删除的数据ID + * @return 结果 + */ + @Override + @Transactional(rollbackFor = Exception.class) + public void deleteGenTableByIds(Long[] tableIds) { + genTableMapper.deleteGenTableByIds(tableIds); + genTableColumnMapper.deleteGenTableColumnByIds(tableIds); + } + + /** + * 导入表结构 + * + * @param tableList 导入表列表 + */ + @Override + @Transactional(rollbackFor = Exception.class) + public void importGenTable(List tableList) { + String operName = SecurityUtils.getUsername(); + try { + for (GenTable table : tableList) { + String tableName = table.getTableName(); + GenUtils.initTable(table, operName); + int row = genTableMapper.insertGenTable(table); + if (row > 0) { + // 保存列信息 + TableInfo tableInfo = GenTableUtils.getTable(tableName); + List fields = tableInfo.getFields(); + AtomicInteger index = new AtomicInteger(1); + List genTableColumns = fields.stream().map(tmp -> { + GenTableColumn data = new GenTableColumn(); + data.setColumnName(tmp.getColumnName()); + data.setColumnComment(tmp.getComment()); + data.setIsRequired(tmp.getMetaInfo().isNullable() ? "1" : "0"); + data.setIsPk(tmp.isKeyFlag() ? "1" : "0"); + data.setSort(index.getAndIncrement()); + data.setIsIncrement(tmp.isKeyIdentityFlag() ? "1" : "0"); + data.setColumnType(tmp.getType()); + return data; + }).collect(Collectors.toList()); + //List genTableColumns = genTableColumnMapper.selectDbTableColumnsByName(tableName); + for (GenTableColumn column : genTableColumns) { + GenUtils.initColumnField(column, table); + genTableColumnMapper.insertGenTableColumn(column); + } + } + } + } catch (Exception e) { + throw new ServiceException("导入失败:" + e.getMessage()); + } + } + + /** + * 预览代码 + * + * @param tableId 表编号 + * @return 预览数据列表 + */ + @Override + public Map previewCode(Long tableId) { + Map dataMap = new LinkedHashMap<>(); + // 查询表信息 + GenTable table = genTableMapper.selectGenTableById(tableId); + // 设置主子表信息 + setSubTable(table); + // 设置主键列信息 + setPkColumn(table); + VelocityInitializer.initVelocity(); + + VelocityContext context = VelocityUtils.prepareContext(table); + + // 获取模板列表 + List templates = VelocityUtils.getTemplateList(table.getTplCategory()); + for (String template : templates) { + // 渲染模板 + StringWriter sw = new StringWriter(); + Template tpl = Velocity.getTemplate(template, Constants.UTF8); + tpl.merge(context, sw); + dataMap.put(template, sw.toString()); + } + return dataMap; + } + + /** + * 生成代码(下载方式) + * + * @param tableName 表名称 + * @return 数据 + */ + @Override + public byte[] downloadCode(String tableName) { + ByteArrayOutputStream outputStream = new ByteArrayOutputStream(); + ZipOutputStream zip = new ZipOutputStream(outputStream); + generatorCode(tableName, zip); + IOUtils.closeQuietly(zip); + return outputStream.toByteArray(); + } + + /** + * 生成代码(自定义路径) + * + * @param tableName 表名称 + */ + @Override + public void generatorCode(String tableName) { + // 查询表信息 + GenTable table = genTableMapper.selectGenTableByName(tableName); + // 设置主子表信息 + setSubTable(table); + // 设置主键列信息 + setPkColumn(table); + + VelocityInitializer.initVelocity(); + + VelocityContext context = VelocityUtils.prepareContext(table); + + // 获取模板列表 + List templates = VelocityUtils.getTemplateList(table.getTplCategory()); + for (String template : templates) { + if (!StringUtils.containsAny(template, "sql.vm", "api.js.vm", "index.vue.vm", "index-tree.vue.vm")) { + // 渲染模板 + StringWriter sw = new StringWriter(); + Template tpl = Velocity.getTemplate(template, Constants.UTF8); + tpl.merge(context, sw); + try { + String path = getGenPath(table, template); + FileUtils.writeStringToFile(new File(path), sw.toString(), CharsetKit.UTF_8); + } catch (IOException e) { + throw new ServiceException("渲染模板失败,表名:" + table.getTableName()); + } + } + } + } + + /** + * 同步数据库 + * + * @param tableName 表名称 + */ + @Override + @Transactional + public void synchDb(String tableName) { + GenTable table = genTableMapper.selectGenTableByName(tableName); + List tableColumns = table.getColumns(); + Map tableColumnMap = tableColumns.stream().collect(Collectors.toMap(GenTableColumn::getColumnName, Function.identity())); + + List dbTableColumns = genTableColumnMapper.selectDbTableColumnsByName(tableName); + if (StringUtils.isEmpty(dbTableColumns)) { + throw new ServiceException("同步数据失败,原表结构不存在"); + } + List dbTableColumnNames = dbTableColumns.stream().map(GenTableColumn::getColumnName).collect(Collectors.toList()); + + dbTableColumns.forEach(column -> { + GenUtils.initColumnField(column, table); + if (tableColumnMap.containsKey(column.getColumnName())) { + GenTableColumn prevColumn = tableColumnMap.get(column.getColumnName()); + column.setColumnId(prevColumn.getColumnId()); + if (column.isList()) { + // 如果是列表,继续保留查询方式/字典类型选项 + column.setDictType(prevColumn.getDictType()); + column.setQueryType(prevColumn.getQueryType()); + } + if (StringUtils.isNotEmpty(prevColumn.getIsRequired()) && !column.isPk() + && (column.isInsert() || column.isEdit()) + && ((column.isUsableColumn()) || (!column.isSuperColumn()))) { + // 如果是(新增/修改&非主键/非忽略及父属性),继续保留必填/显示类型选项 + column.setIsRequired(prevColumn.getIsRequired()); + column.setHtmlType(prevColumn.getHtmlType()); + } + genTableColumnMapper.updateGenTableColumn(column); + } else { + genTableColumnMapper.insertGenTableColumn(column); + } + }); + + List delColumns = tableColumns.stream().filter(column -> !dbTableColumnNames.contains(column.getColumnName())).collect(Collectors.toList()); + if (StringUtils.isNotEmpty(delColumns)) { + genTableColumnMapper.deleteGenTableColumns(delColumns); + } + } + + /** + * 批量生成代码(下载方式) + * + * @param tableNames 表数组 + * @return 数据 + */ + @Override + public byte[] downloadCode(String[] tableNames) { + ByteArrayOutputStream outputStream = new ByteArrayOutputStream(); + ZipOutputStream zip = new ZipOutputStream(outputStream); + for (String tableName : tableNames) { + generatorCode(tableName, zip); + } + IOUtils.closeQuietly(zip); + return outputStream.toByteArray(); + } + + /** + * 查询表信息并生成代码 + */ + private void generatorCode(String tableName, ZipOutputStream zip) { + // 查询表信息 + GenTable table = genTableMapper.selectGenTableByName(tableName); + // 设置主子表信息 + setSubTable(table); + // 设置主键列信息 + setPkColumn(table); + + VelocityInitializer.initVelocity(); + + VelocityContext context = VelocityUtils.prepareContext(table); + + // 获取模板列表 + List templates = VelocityUtils.getTemplateList(table.getTplCategory()); + for (String template : templates) { + // 渲染模板 + StringWriter sw = new StringWriter(); + Template tpl = Velocity.getTemplate(template, Constants.UTF8); + tpl.merge(context, sw); + try { + // 添加到zip + zip.putNextEntry(new ZipEntry(VelocityUtils.getFileName(template, table))); + IOUtils.write(sw.toString(), zip, Constants.UTF8); + IOUtils.closeQuietly(sw); + zip.flush(); + zip.closeEntry(); + } catch (IOException e) { + log.error("渲染模板失败,表名:" + table.getTableName(), e); + } + } + } + + /** + * 修改保存参数校验 + * + * @param genTable 业务信息 + */ + @Override + public void validateEdit(GenTable genTable) { + if (GenConstants.TPL_TREE.equals(genTable.getTplCategory())) { + String options = JSON.toJSONString(genTable.getParams()); + JSONObject paramsObj = JSON.parseObject(options); + if (StringUtils.isEmpty(paramsObj.getString(GenConstants.TREE_CODE))) { + throw new ServiceException("树编码字段不能为空"); + } else if (StringUtils.isEmpty(paramsObj.getString(GenConstants.TREE_PARENT_CODE))) { + throw new ServiceException("树父编码字段不能为空"); + } else if (StringUtils.isEmpty(paramsObj.getString(GenConstants.TREE_NAME))) { + throw new ServiceException("树名称字段不能为空"); + } else if (GenConstants.TPL_SUB.equals(genTable.getTplCategory())) { + if (StringUtils.isEmpty(genTable.getSubTableName())) { + throw new ServiceException("关联子表的表名不能为空"); + } else if (StringUtils.isEmpty(genTable.getSubTableFkName())) { + throw new ServiceException("子表关联的外键名不能为空"); + } + } + } + } + + /** + * 设置主键列信息 + * + * @param table 业务表信息 + */ + public void setPkColumn(GenTable table) { + for (GenTableColumn column : table.getColumns()) { + if (column.isPk()) { + table.setPkColumn(column); + break; + } + } + if (StringUtils.isNull(table.getPkColumn())) { + table.setPkColumn(table.getColumns().get(0)); + } + if (GenConstants.TPL_SUB.equals(table.getTplCategory())) { + for (GenTableColumn column : table.getSubTable().getColumns()) { + if (column.isPk()) { + table.getSubTable().setPkColumn(column); + break; + } + } + if (StringUtils.isNull(table.getSubTable().getPkColumn())) { + table.getSubTable().setPkColumn(table.getSubTable().getColumns().get(0)); + } + } + } + + /** + * 设置主子表信息 + * + * @param table 业务表信息 + */ + public void setSubTable(GenTable table) { + String subTableName = table.getSubTableName(); + if (StringUtils.isNotEmpty(subTableName)) { + table.setSubTable(genTableMapper.selectGenTableByName(subTableName)); + } + } + + /** + * 设置代码生成其他选项值 + * + * @param genTable 设置后的生成对象 + */ + public void setTableFromOptions(GenTable genTable) { + JSONObject paramsObj = JSON.parseObject(genTable.getOptions()); + if (StringUtils.isNotNull(paramsObj)) { + String treeCode = paramsObj.getString(GenConstants.TREE_CODE); + String treeParentCode = paramsObj.getString(GenConstants.TREE_PARENT_CODE); + String treeName = paramsObj.getString(GenConstants.TREE_NAME); + String parentMenuId = paramsObj.getString(GenConstants.PARENT_MENU_ID); + String parentMenuName = paramsObj.getString(GenConstants.PARENT_MENU_NAME); + + genTable.setTreeCode(treeCode); + genTable.setTreeParentCode(treeParentCode); + genTable.setTreeName(treeName); + genTable.setParentMenuId(parentMenuId); + genTable.setParentMenuName(parentMenuName); + } + } + + /** + * 获取代码生成地址 + * + * @param table 业务表信息 + * @param template 模板文件路径 + * @return 生成地址 + */ + public static String getGenPath(GenTable table, String template) { + String genPath = table.getGenPath(); + if (StringUtils.equals(genPath, "/")) { + return System.getProperty("user.dir") + File.separator + "src" + File.separator + VelocityUtils.getFileName(template, table); + } + return genPath + File.separator + VelocityUtils.getFileName(template, table); + } +} diff --git a/ruoyi-generator/src/main/java/com/ruoyi/generator/service/IGenTableColumnService.java b/ruoyi-generator/src/main/java/com/ruoyi/generator/service/IGenTableColumnService.java new file mode 100644 index 00000000..3037f707 --- /dev/null +++ b/ruoyi-generator/src/main/java/com/ruoyi/generator/service/IGenTableColumnService.java @@ -0,0 +1,44 @@ +package com.ruoyi.generator.service; + +import java.util.List; +import com.ruoyi.generator.domain.GenTableColumn; + +/** + * 业务字段 服务层 + * + * @author ruoyi + */ +public interface IGenTableColumnService +{ + /** + * 查询业务字段列表 + * + * @param tableId 业务字段编号 + * @return 业务字段集合 + */ + public List selectGenTableColumnListByTableId(Long tableId); + + /** + * 新增业务字段 + * + * @param genTableColumn 业务字段信息 + * @return 结果 + */ + public int insertGenTableColumn(GenTableColumn genTableColumn); + + /** + * 修改业务字段 + * + * @param genTableColumn 业务字段信息 + * @return 结果 + */ + public int updateGenTableColumn(GenTableColumn genTableColumn); + + /** + * 删除业务字段信息 + * + * @param ids 需要删除的数据ID + * @return 结果 + */ + public int deleteGenTableColumnByIds(String ids); +} diff --git a/ruoyi-generator/src/main/java/com/ruoyi/generator/service/IGenTableService.java b/ruoyi-generator/src/main/java/com/ruoyi/generator/service/IGenTableService.java new file mode 100644 index 00000000..9d53f95f --- /dev/null +++ b/ruoyi-generator/src/main/java/com/ruoyi/generator/service/IGenTableService.java @@ -0,0 +1,121 @@ +package com.ruoyi.generator.service; + +import java.util.List; +import java.util.Map; +import com.ruoyi.generator.domain.GenTable; + +/** + * 业务 服务层 + * + * @author ruoyi + */ +public interface IGenTableService +{ + /** + * 查询业务列表 + * + * @param genTable 业务信息 + * @return 业务集合 + */ + public List selectGenTableList(GenTable genTable); + + /** + * 查询据库列表 + * + * @param genTable 业务信息 + * @return 数据库表集合 + */ + public List selectDbTableList(GenTable genTable); + + /** + * 查询据库列表 + * + * @param tableNames 表名称组 + * @return 数据库表集合 + */ + public List selectDbTableListByNames(String[] tableNames); + + /** + * 查询所有表信息 + * + * @return 表信息集合 + */ + public List selectGenTableAll(); + + /** + * 查询业务信息 + * + * @param id 业务ID + * @return 业务信息 + */ + public GenTable selectGenTableById(Long id); + + /** + * 修改业务 + * + * @param genTable 业务信息 + * @return 结果 + */ + public void updateGenTable(GenTable genTable); + + /** + * 删除业务信息 + * + * @param tableIds 需要删除的表数据ID + * @return 结果 + */ + public void deleteGenTableByIds(Long[] tableIds); + + /** + * 导入表结构 + * + * @param tableList 导入表列表 + */ + public void importGenTable(List tableList); + + /** + * 预览代码 + * + * @param tableId 表编号 + * @return 预览数据列表 + */ + public Map previewCode(Long tableId); + + /** + * 生成代码(下载方式) + * + * @param tableName 表名称 + * @return 数据 + */ + public byte[] downloadCode(String tableName); + + /** + * 生成代码(自定义路径) + * + * @param tableName 表名称 + * @return 数据 + */ + public void generatorCode(String tableName); + + /** + * 同步数据库 + * + * @param tableName 表名称 + */ + public void synchDb(String tableName); + + /** + * 批量生成代码(下载方式) + * + * @param tableNames 表数组 + * @return 数据 + */ + public byte[] downloadCode(String[] tableNames); + + /** + * 修改保存参数校验 + * + * @param genTable 业务信息 + */ + public void validateEdit(GenTable genTable); +} diff --git a/ruoyi-generator/src/main/java/com/ruoyi/generator/util/GenTableUtils.java b/ruoyi-generator/src/main/java/com/ruoyi/generator/util/GenTableUtils.java new file mode 100644 index 00000000..a6a8609a --- /dev/null +++ b/ruoyi-generator/src/main/java/com/ruoyi/generator/util/GenTableUtils.java @@ -0,0 +1,56 @@ +package com.ruoyi.generator.util; + +import cn.hutool.core.collection.CollUtil; +import cn.hutool.core.util.StrUtil; +import com.baomidou.mybatisplus.generator.config.DataSourceConfig; +import com.baomidou.mybatisplus.generator.config.GlobalConfig; +import com.baomidou.mybatisplus.generator.config.StrategyConfig; +import com.baomidou.mybatisplus.generator.config.builder.ConfigBuilder; +import com.baomidou.mybatisplus.generator.config.po.TableInfo; +import com.baomidou.mybatisplus.generator.config.rules.DateType; +import com.ruoyi.common.utils.spring.SpringUtils; +import org.springframework.core.env.Environment; + +import java.util.Comparator; +import java.util.List; +import java.util.stream.Collectors; + +public class GenTableUtils { + + private static final String DATASOURCE_DYNAMIC_KEY = "spring.datasource.druid.master"; + + public static Environment environment = SpringUtils.getBean(Environment.class); + + public static List getTableList(String nameLike, String commentLike) { + List tables = getTableList0(null); + return tables.stream().filter(tableInfo -> (StrUtil.isEmpty(nameLike) || tableInfo.getName().contains(nameLike)) + && (StrUtil.isEmpty(commentLike) || tableInfo.getComment().contains(commentLike))) + .collect(Collectors.toList()); + } + + public static TableInfo getTable(String name) { + return CollUtil.getFirst(getTableList0(name)); + } + + public static List getTableList0(String name) { + String url = environment.getProperty(DATASOURCE_DYNAMIC_KEY + ".url"); + String username = environment.getProperty(DATASOURCE_DYNAMIC_KEY + ".username"); + String password = environment.getProperty(DATASOURCE_DYNAMIC_KEY + ".password"); + // 使用 MyBatis Plus Generator 解析表结构 + DataSourceConfig dataSourceConfig = new DataSourceConfig.Builder(url, username, + password).build(); + StrategyConfig.Builder strategyConfig = new StrategyConfig.Builder(); + if (StrUtil.isNotEmpty(name)) { + strategyConfig.addInclude(name); + } + // 只使用 Date 类型,不使用 LocalDate + GlobalConfig globalConfig = new GlobalConfig.Builder().dateType(DateType.ONLY_DATE).build(); + ConfigBuilder builder = new ConfigBuilder(null, dataSourceConfig, strategyConfig.build(), + null, globalConfig, null); + // 按照名字排序 + List tables = builder.getTableInfoList(); + tables.sort(Comparator.comparing(TableInfo::getName)); + return tables; + } + +} diff --git a/ruoyi-generator/src/main/java/com/ruoyi/generator/util/GenUtils.java b/ruoyi-generator/src/main/java/com/ruoyi/generator/util/GenUtils.java new file mode 100644 index 00000000..4f281621 --- /dev/null +++ b/ruoyi-generator/src/main/java/com/ruoyi/generator/util/GenUtils.java @@ -0,0 +1,257 @@ +package com.ruoyi.generator.util; + +import java.util.Arrays; +import org.apache.commons.lang3.RegExUtils; +import com.ruoyi.common.constant.GenConstants; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.generator.config.GenConfig; +import com.ruoyi.generator.domain.GenTable; +import com.ruoyi.generator.domain.GenTableColumn; + +/** + * 代码生成器 工具类 + * + * @author ruoyi + */ +public class GenUtils +{ + /** + * 初始化表信息 + */ + public static void initTable(GenTable genTable, String operName) + { + genTable.setClassName(convertClassName(genTable.getTableName())); + genTable.setPackageName(GenConfig.getPackageName()); + genTable.setModuleName(getModuleName(GenConfig.getPackageName())); + genTable.setBusinessName(getBusinessName(genTable.getTableName())); + genTable.setFunctionName(replaceText(genTable.getTableComment())); + genTable.setFunctionAuthor(GenConfig.getAuthor()); + genTable.setCreateBy(operName); + } + + /** + * 初始化列属性字段 + */ + public static void initColumnField(GenTableColumn column, GenTable table) + { + String dataType = getDbType(column.getColumnType()); + String columnName = column.getColumnName(); + column.setTableId(table.getTableId()); + column.setCreateBy(table.getCreateBy()); + // 设置java字段名 + column.setJavaField(StringUtils.toCamelCase(columnName)); + // 设置默认类型 + column.setJavaType(GenConstants.TYPE_STRING); + column.setQueryType(GenConstants.QUERY_EQ); + + if (arraysContains(GenConstants.COLUMNTYPE_STR, dataType) || arraysContains(GenConstants.COLUMNTYPE_TEXT, dataType)) + { + // 字符串长度超过500设置为文本域 + Integer columnLength = getColumnLength(column.getColumnType()); + String htmlType = columnLength >= 500 || arraysContains(GenConstants.COLUMNTYPE_TEXT, dataType) ? GenConstants.HTML_TEXTAREA : GenConstants.HTML_INPUT; + column.setHtmlType(htmlType); + } + else if (arraysContains(GenConstants.COLUMNTYPE_TIME, dataType)) + { + column.setJavaType(GenConstants.TYPE_DATE); + column.setHtmlType(GenConstants.HTML_DATETIME); + } + else if (arraysContains(GenConstants.COLUMNTYPE_NUMBER, dataType)) + { + column.setHtmlType(GenConstants.HTML_INPUT); + + // 如果是浮点型 统一用BigDecimal + String[] str = StringUtils.split(StringUtils.substringBetween(column.getColumnType(), "(", ")"), ","); + if (str != null && str.length == 2 && Integer.parseInt(str[1]) > 0) + { + column.setJavaType(GenConstants.TYPE_BIGDECIMAL); + } + // 如果是整形 + else if (str != null && str.length == 1 && Integer.parseInt(str[0]) <= 10) + { + column.setJavaType(GenConstants.TYPE_INTEGER); + } + // 长整形 + else + { + column.setJavaType(GenConstants.TYPE_LONG); + } + } + + // 插入字段(默认所有字段都需要插入) + column.setIsInsert(GenConstants.REQUIRE); + + // 编辑字段 + if (!arraysContains(GenConstants.COLUMNNAME_NOT_EDIT, columnName) && !column.isPk()) + { + column.setIsEdit(GenConstants.REQUIRE); + } + // 列表字段 + if (!arraysContains(GenConstants.COLUMNNAME_NOT_LIST, columnName) && !column.isPk()) + { + column.setIsList(GenConstants.REQUIRE); + } + // 查询字段 + if (!arraysContains(GenConstants.COLUMNNAME_NOT_QUERY, columnName) && !column.isPk()) + { + column.setIsQuery(GenConstants.REQUIRE); + } + + // 查询字段类型 + if (StringUtils.endsWithIgnoreCase(columnName, "name")) + { + column.setQueryType(GenConstants.QUERY_LIKE); + } + // 状态字段设置单选框 + if (StringUtils.endsWithIgnoreCase(columnName, "status")) + { + column.setHtmlType(GenConstants.HTML_RADIO); + } + // 类型&性别字段设置下拉框 + else if (StringUtils.endsWithIgnoreCase(columnName, "type") + || StringUtils.endsWithIgnoreCase(columnName, "sex")) + { + column.setHtmlType(GenConstants.HTML_SELECT); + } + // 图片字段设置图片上传控件 + else if (StringUtils.endsWithIgnoreCase(columnName, "image")) + { + column.setHtmlType(GenConstants.HTML_IMAGE_UPLOAD); + } + // 文件字段设置文件上传控件 + else if (StringUtils.endsWithIgnoreCase(columnName, "file")) + { + column.setHtmlType(GenConstants.HTML_FILE_UPLOAD); + } + // 内容字段设置富文本控件 + else if (StringUtils.endsWithIgnoreCase(columnName, "content")) + { + column.setHtmlType(GenConstants.HTML_EDITOR); + } + } + + /** + * 校验数组是否包含指定值 + * + * @param arr 数组 + * @param targetValue 值 + * @return 是否包含 + */ + public static boolean arraysContains(String[] arr, String targetValue) + { + return Arrays.asList(arr).contains(targetValue); + } + + /** + * 获取模块名 + * + * @param packageName 包名 + * @return 模块名 + */ + public static String getModuleName(String packageName) + { + int lastIndex = packageName.lastIndexOf("."); + int nameLength = packageName.length(); + return StringUtils.substring(packageName, lastIndex + 1, nameLength); + } + + /** + * 获取业务名 + * + * @param tableName 表名 + * @return 业务名 + */ + public static String getBusinessName(String tableName) + { + int lastIndex = tableName.lastIndexOf("_"); + int nameLength = tableName.length(); + return StringUtils.substring(tableName, lastIndex + 1, nameLength); + } + + /** + * 表名转换成Java类名 + * + * @param tableName 表名称 + * @return 类名 + */ + public static String convertClassName(String tableName) + { + boolean autoRemovePre = GenConfig.getAutoRemovePre(); + String tablePrefix = GenConfig.getTablePrefix(); + if (autoRemovePre && StringUtils.isNotEmpty(tablePrefix)) + { + String[] searchList = StringUtils.split(tablePrefix, ","); + tableName = replaceFirst(tableName, searchList); + } + return StringUtils.convertToCamelCase(tableName); + } + + /** + * 批量替换前缀 + * + * @param replacementm 替换值 + * @param searchList 替换列表 + * @return + */ + public static String replaceFirst(String replacementm, String[] searchList) + { + String text = replacementm; + for (String searchString : searchList) + { + if (replacementm.startsWith(searchString)) + { + text = replacementm.replaceFirst(searchString, ""); + break; + } + } + return text; + } + + /** + * 关键字替换 + * + * @param text 需要被替换的名字 + * @return 替换后的名字 + */ + public static String replaceText(String text) + { + return RegExUtils.replaceAll(text, "(?:表|泽火)", ""); + } + + /** + * 获取数据库类型字段 + * + * @param columnType 列类型 + * @return 截取后的列类型 + */ + public static String getDbType(String columnType) + { + if (StringUtils.indexOf(columnType, "(") > 0) + { + return StringUtils.substringBefore(columnType, "("); + } + else + { + return columnType; + } + } + + /** + * 获取字段长度 + * + * @param columnType 列类型 + * @return 截取后的列类型 + */ + public static Integer getColumnLength(String columnType) + { + if (StringUtils.indexOf(columnType, "(") > 0) + { + String length = StringUtils.substringBetween(columnType, "(", ")"); + return Integer.valueOf(length); + } + else + { + return 0; + } + } +} diff --git a/ruoyi-generator/src/main/java/com/ruoyi/generator/util/VelocityInitializer.java b/ruoyi-generator/src/main/java/com/ruoyi/generator/util/VelocityInitializer.java new file mode 100644 index 00000000..9f694038 --- /dev/null +++ b/ruoyi-generator/src/main/java/com/ruoyi/generator/util/VelocityInitializer.java @@ -0,0 +1,34 @@ +package com.ruoyi.generator.util; + +import java.util.Properties; +import org.apache.velocity.app.Velocity; +import com.ruoyi.common.constant.Constants; + +/** + * VelocityEngine工厂 + * + * @author ruoyi + */ +public class VelocityInitializer +{ + /** + * 初始化vm方法 + */ + public static void initVelocity() + { + Properties p = new Properties(); + try + { + // 加载classpath目录下的vm文件 + p.setProperty("resource.loader.file.class", "org.apache.velocity.runtime.resource.loader.ClasspathResourceLoader"); + // 定义字符集 + p.setProperty(Velocity.INPUT_ENCODING, Constants.UTF8); + // 初始化Velocity引擎,指定配置Properties + Velocity.init(p); + } + catch (Exception e) + { + throw new RuntimeException(e); + } + } +} diff --git a/ruoyi-generator/src/main/java/com/ruoyi/generator/util/VelocityUtils.java b/ruoyi-generator/src/main/java/com/ruoyi/generator/util/VelocityUtils.java new file mode 100644 index 00000000..a00aa0d6 --- /dev/null +++ b/ruoyi-generator/src/main/java/com/ruoyi/generator/util/VelocityUtils.java @@ -0,0 +1,405 @@ +package com.ruoyi.generator.util; + +import java.util.ArrayList; +import java.util.HashSet; +import java.util.List; +import java.util.Set; +import org.apache.velocity.VelocityContext; +import com.alibaba.fastjson2.JSON; +import com.alibaba.fastjson2.JSONObject; +import com.ruoyi.common.constant.GenConstants; +import com.ruoyi.common.utils.DateUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.generator.domain.GenTable; +import com.ruoyi.generator.domain.GenTableColumn; + +/** + * 模板处理工具类 + * + * @author ruoyi + */ +public class VelocityUtils +{ + /** 项目空间路径 */ + private static final String PROJECT_PATH = "main/java"; + + /** mybatis空间路径 */ + private static final String MYBATIS_PATH = "main/resources/mapper"; + + /** 默认上级菜单,系统工具 */ + private static final String DEFAULT_PARENT_MENU_ID = "3"; + + /** 模板切换 mybatis:vm;mybatis-plus:mybatisplusvm */ + private static final String VM_PATH = "mybatisplusvm"; + + /** + * 设置模板变量信息 + * + * @return 模板列表 + */ + public static VelocityContext prepareContext(GenTable genTable) + { + String moduleName = genTable.getModuleName(); + String businessName = genTable.getBusinessName(); + String packageName = genTable.getPackageName(); + String tplCategory = genTable.getTplCategory(); + String functionName = genTable.getFunctionName(); + + VelocityContext velocityContext = new VelocityContext(); + velocityContext.put("tplCategory", genTable.getTplCategory()); + velocityContext.put("tableName", genTable.getTableName()); + velocityContext.put("functionName", StringUtils.isNotEmpty(functionName) ? functionName : "【请填写功能名称】"); + velocityContext.put("ClassName", genTable.getClassName()); + velocityContext.put("className", StringUtils.uncapitalize(genTable.getClassName())); + velocityContext.put("moduleName", genTable.getModuleName()); + velocityContext.put("BusinessName", StringUtils.capitalize(genTable.getBusinessName())); + velocityContext.put("businessName", genTable.getBusinessName()); + velocityContext.put("basePackage", getPackagePrefix(packageName)); + velocityContext.put("packageName", packageName); + velocityContext.put("author", genTable.getFunctionAuthor()); + velocityContext.put("datetime", DateUtils.getDate()); + velocityContext.put("pkColumn", genTable.getPkColumn()); + velocityContext.put("importList", getImportList(genTable)); + velocityContext.put("permissionPrefix", getPermissionPrefix(moduleName, businessName)); + velocityContext.put("columns", genTable.getColumns()); + velocityContext.put("table", genTable); + velocityContext.put("dicts", getDicts(genTable)); + setMenuVelocityContext(velocityContext, genTable); + if (GenConstants.TPL_TREE.equals(tplCategory)) + { + setTreeVelocityContext(velocityContext, genTable); + } + if (GenConstants.TPL_SUB.equals(tplCategory)) + { + setSubVelocityContext(velocityContext, genTable); + } + return velocityContext; + } + + public static void setMenuVelocityContext(VelocityContext context, GenTable genTable) + { + String options = genTable.getOptions(); + JSONObject paramsObj = JSON.parseObject(options); + String parentMenuId = getParentMenuId(paramsObj); + context.put("parentMenuId", parentMenuId); + } + + public static void setTreeVelocityContext(VelocityContext context, GenTable genTable) + { + String options = genTable.getOptions(); + JSONObject paramsObj = JSON.parseObject(options); + String treeCode = getTreecode(paramsObj); + String treeParentCode = getTreeParentCode(paramsObj); + String treeName = getTreeName(paramsObj); + + context.put("treeCode", treeCode); + context.put("treeParentCode", treeParentCode); + context.put("treeName", treeName); + context.put("expandColumn", getExpandColumn(genTable)); + if (paramsObj.containsKey(GenConstants.TREE_PARENT_CODE)) + { + context.put("tree_parent_code", paramsObj.getString(GenConstants.TREE_PARENT_CODE)); + } + if (paramsObj.containsKey(GenConstants.TREE_NAME)) + { + context.put("tree_name", paramsObj.getString(GenConstants.TREE_NAME)); + } + } + + public static void setSubVelocityContext(VelocityContext context, GenTable genTable) + { + GenTable subTable = genTable.getSubTable(); + String subTableName = genTable.getSubTableName(); + String subTableFkName = genTable.getSubTableFkName(); + String subClassName = genTable.getSubTable().getClassName(); + String subTableFkClassName = StringUtils.convertToCamelCase(subTableFkName); + + context.put("subTable", subTable); + context.put("subTableName", subTableName); + context.put("subTableFkName", subTableFkName); + context.put("subTableFkClassName", subTableFkClassName); + context.put("subTableFkclassName", StringUtils.uncapitalize(subTableFkClassName)); + context.put("subClassName", subClassName); + context.put("subclassName", StringUtils.uncapitalize(subClassName)); + context.put("subImportList", getImportList(genTable.getSubTable())); + } + + /** + * 获取模板信息 + * + * @return 模板列表 + */ + public static List getTemplateList(String tplCategory) + { + List templates = new ArrayList(); + templates.add(VM_PATH+"/java/domain.java.vm"); + templates.add(VM_PATH+"/java/mapper.java.vm"); + templates.add(VM_PATH+"/java/service.java.vm"); + templates.add(VM_PATH+"/java/serviceImpl.java.vm"); + templates.add(VM_PATH+"/java/controller.java.vm"); + templates.add(VM_PATH+"/xml/mapper.xml.vm"); + templates.add(VM_PATH+"/sql/sql.vm"); + templates.add(VM_PATH+"/js/api.js.vm"); + if (GenConstants.TPL_CRUD.equals(tplCategory)) + { + templates.add(VM_PATH+"/vue/index.vue.vm"); + } + else if (GenConstants.TPL_TREE.equals(tplCategory)) + { + templates.add(VM_PATH+"/vue/index-tree.vue.vm"); + } + else if (GenConstants.TPL_SUB.equals(tplCategory)) + { + templates.add(VM_PATH+"/vue/index.vue.vm"); + templates.add(VM_PATH+"/java/sub-domain.java.vm"); + } + return templates; + } + + /** + * 获取文件名 + */ + public static String getFileName(String template, GenTable genTable) + { + // 文件名称 + String fileName = ""; + // 包路径 + String packageName = genTable.getPackageName(); + // 模块名 + String moduleName = genTable.getModuleName(); + // 大写类名 + String className = genTable.getClassName(); + // 业务名称 + String businessName = genTable.getBusinessName(); + + String javaPath = PROJECT_PATH + "/" + StringUtils.replace(packageName, ".", "/"); + String mybatisPath = MYBATIS_PATH + "/" + moduleName; + String vuePath = "vue"; + + if (template.contains("domain.java.vm")) + { + fileName = StringUtils.format("{}/domain/{}.java", javaPath, className); + } + if (template.contains("sub-domain.java.vm") && StringUtils.equals(GenConstants.TPL_SUB, genTable.getTplCategory())) + { + fileName = StringUtils.format("{}/domain/{}.java", javaPath, genTable.getSubTable().getClassName()); + } + else if (template.contains("mapper.java.vm")) + { + fileName = StringUtils.format("{}/mapper/{}Mapper.java", javaPath, className); + } + else if (template.contains("service.java.vm")) + { + fileName = StringUtils.format("{}/service/I{}Service.java", javaPath, className); + } + else if (template.contains("serviceImpl.java.vm")) + { + fileName = StringUtils.format("{}/service/impl/{}ServiceImpl.java", javaPath, className); + } + else if (template.contains("controller.java.vm")) + { + fileName = StringUtils.format("{}/controller/{}Controller.java", javaPath, className); + } + else if (template.contains("mapper.xml.vm")) + { + fileName = StringUtils.format("{}/{}Mapper.xml", mybatisPath, className); + } + else if (template.contains("sql.vm")) + { + fileName = businessName + "Menu.sql"; + } + else if (template.contains("api.js.vm")) + { + fileName = StringUtils.format("{}/api/{}/{}.js", vuePath, moduleName, businessName); + } + else if (template.contains("index.vue.vm")) + { + fileName = StringUtils.format("{}/views/{}/{}/index.vue", vuePath, moduleName, businessName); + } + else if (template.contains("index-tree.vue.vm")) + { + fileName = StringUtils.format("{}/views/{}/{}/index.vue", vuePath, moduleName, businessName); + } + return fileName; + } + + /** + * 获取包前缀 + * + * @param packageName 包名称 + * @return 包前缀名称 + */ + public static String getPackagePrefix(String packageName) + { + int lastIndex = packageName.lastIndexOf("."); + return StringUtils.substring(packageName, 0, lastIndex); + } + + /** + * 根据列类型获取导入包 + * + * @param genTable 业务表对象 + * @return 返回需要导入的包列表 + */ + public static HashSet getImportList(GenTable genTable) + { + List columns = genTable.getColumns(); + GenTable subGenTable = genTable.getSubTable(); + HashSet importList = new HashSet(); + if (StringUtils.isNotNull(subGenTable)) + { + importList.add("java.util.List"); + } + for (GenTableColumn column : columns) + { + if (!column.isSuperColumn() && GenConstants.TYPE_DATE.equals(column.getJavaType())) + { + importList.add("java.util.Date"); + importList.add("com.fasterxml.jackson.annotation.JsonFormat"); + } + else if (!column.isSuperColumn() && GenConstants.TYPE_BIGDECIMAL.equals(column.getJavaType())) + { + importList.add("java.math.BigDecimal"); + } + } + return importList; + } + + /** + * 根据列类型获取字典组 + * + * @param genTable 业务表对象 + * @return 返回字典组 + */ + public static String getDicts(GenTable genTable) + { + List columns = genTable.getColumns(); + Set dicts = new HashSet(); + addDicts(dicts, columns); + if (StringUtils.isNotNull(genTable.getSubTable())) + { + List subColumns = genTable.getSubTable().getColumns(); + addDicts(dicts, subColumns); + } + return StringUtils.join(dicts, ", "); + } + + /** + * 添加字典列表 + * + * @param dicts 字典列表 + * @param columns 列集合 + */ + public static void addDicts(Set dicts, List columns) + { + for (GenTableColumn column : columns) + { + if (!column.isSuperColumn() && StringUtils.isNotEmpty(column.getDictType()) && StringUtils.equalsAny( + column.getHtmlType(), + new String[] { GenConstants.HTML_SELECT, GenConstants.HTML_RADIO, GenConstants.HTML_CHECKBOX })) + { + dicts.add("'" + column.getDictType() + "'"); + } + } + } + + /** + * 获取权限前缀 + * + * @param moduleName 模块名称 + * @param businessName 业务名称 + * @return 返回权限前缀 + */ + public static String getPermissionPrefix(String moduleName, String businessName) + { + return StringUtils.format("{}:{}", moduleName, businessName); + } + + /** + * 获取上级菜单ID字段 + * + * @param paramsObj 生成其他选项 + * @return 上级菜单ID字段 + */ + public static String getParentMenuId(JSONObject paramsObj) + { + if (StringUtils.isNotEmpty(paramsObj) && paramsObj.containsKey(GenConstants.PARENT_MENU_ID) + && StringUtils.isNotEmpty(paramsObj.getString(GenConstants.PARENT_MENU_ID))) + { + return paramsObj.getString(GenConstants.PARENT_MENU_ID); + } + return DEFAULT_PARENT_MENU_ID; + } + + /** + * 获取树编码 + * + * @param paramsObj 生成其他选项 + * @return 树编码 + */ + public static String getTreecode(JSONObject paramsObj) + { + if (paramsObj.containsKey(GenConstants.TREE_CODE)) + { + return StringUtils.toCamelCase(paramsObj.getString(GenConstants.TREE_CODE)); + } + return StringUtils.EMPTY; + } + + /** + * 获取树父编码 + * + * @param paramsObj 生成其他选项 + * @return 树父编码 + */ + public static String getTreeParentCode(JSONObject paramsObj) + { + if (paramsObj.containsKey(GenConstants.TREE_PARENT_CODE)) + { + return StringUtils.toCamelCase(paramsObj.getString(GenConstants.TREE_PARENT_CODE)); + } + return StringUtils.EMPTY; + } + + /** + * 获取树名称 + * + * @param paramsObj 生成其他选项 + * @return 树名称 + */ + public static String getTreeName(JSONObject paramsObj) + { + if (paramsObj.containsKey(GenConstants.TREE_NAME)) + { + return StringUtils.toCamelCase(paramsObj.getString(GenConstants.TREE_NAME)); + } + return StringUtils.EMPTY; + } + + /** + * 获取需要在哪一列上面显示展开按钮 + * + * @param genTable 业务表对象 + * @return 展开按钮列序号 + */ + public static int getExpandColumn(GenTable genTable) + { + String options = genTable.getOptions(); + JSONObject paramsObj = JSON.parseObject(options); + String treeName = paramsObj.getString(GenConstants.TREE_NAME); + int num = 0; + for (GenTableColumn column : genTable.getColumns()) + { + if (column.isList()) + { + num++; + String columnName = column.getColumnName(); + if (columnName.equals(treeName)) + { + break; + } + } + } + return num; + } +} diff --git a/ruoyi-generator/src/main/resources/generator.yml b/ruoyi-generator/src/main/resources/generator.yml new file mode 100644 index 00000000..0bc255a7 --- /dev/null +++ b/ruoyi-generator/src/main/resources/generator.yml @@ -0,0 +1,10 @@ +# 代码生成 +gen: + # 作者 + author: chenqp + # 默认生成包路径 system 需改成自己的模块名称 如 system monitor tool + packageName: com.ruoyi.system + # 自动去除表前缀,默认是false + autoRemovePre: false + # 表前缀(生成类名不会包含表前缀,多个用逗号分隔) + tablePrefix: sys_ \ No newline at end of file diff --git a/ruoyi-generator/src/main/resources/mapper/generator/GenTableColumnMapper.xml b/ruoyi-generator/src/main/resources/mapper/generator/GenTableColumnMapper.xml new file mode 100644 index 00000000..3c24013c --- /dev/null +++ b/ruoyi-generator/src/main/resources/mapper/generator/GenTableColumnMapper.xml @@ -0,0 +1,127 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + select column_id, table_id, column_name, column_comment, column_type, java_type, java_field, is_pk, is_increment, is_required, is_insert, is_edit, is_list, is_query, query_type, html_type, dict_type, sort, create_by, create_time, update_by, update_time from gen_table_column + + + + + + + + insert into gen_table_column ( + table_id, + column_name, + column_comment, + column_type, + java_type, + java_field, + is_pk, + is_increment, + is_required, + is_insert, + is_edit, + is_list, + is_query, + query_type, + html_type, + dict_type, + sort, + create_by, + create_time + )values( + #{tableId}, + #{columnName}, + #{columnComment}, + #{columnType}, + #{javaType}, + #{javaField}, + #{isPk}, + #{isIncrement}, + #{isRequired}, + #{isInsert}, + #{isEdit}, + #{isList}, + #{isQuery}, + #{queryType}, + #{htmlType}, + #{dictType}, + #{sort}, + #{createBy}, + sysdate() + ) + + + + update gen_table_column + + column_comment = #{columnComment}, + java_type = #{javaType}, + java_field = #{javaField}, + is_insert = #{isInsert}, + is_edit = #{isEdit}, + is_list = #{isList}, + is_query = #{isQuery}, + is_required = #{isRequired}, + query_type = #{queryType}, + html_type = #{htmlType}, + dict_type = #{dictType}, + sort = #{sort}, + update_by = #{updateBy}, + update_time = sysdate() + + where column_id = #{columnId} + + + + delete from gen_table_column where table_id in + + #{tableId} + + + + + delete from gen_table_column where column_id in + + #{item.columnId} + + + + diff --git a/ruoyi-generator/src/main/resources/mapper/generator/GenTableMapper.xml b/ruoyi-generator/src/main/resources/mapper/generator/GenTableMapper.xml new file mode 100644 index 00000000..b605e906 --- /dev/null +++ b/ruoyi-generator/src/main/resources/mapper/generator/GenTableMapper.xml @@ -0,0 +1,202 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + select table_id, table_name, table_comment, sub_table_name, sub_table_fk_name, class_name, tpl_category, package_name, module_name, business_name, function_name, function_author, gen_type, gen_path, options, create_by, create_time, update_by, update_time, remark from gen_table + + + + + + + + + + + + + + + + + + insert into gen_table ( + table_name, + table_comment, + class_name, + tpl_category, + package_name, + module_name, + business_name, + function_name, + function_author, + gen_type, + gen_path, + remark, + create_by, + create_time + )values( + #{tableName}, + #{tableComment}, + #{className}, + #{tplCategory}, + #{packageName}, + #{moduleName}, + #{businessName}, + #{functionName}, + #{functionAuthor}, + #{genType}, + #{genPath}, + #{remark}, + #{createBy}, + sysdate() + ) + + + + update gen_table + + table_name = #{tableName}, + table_comment = #{tableComment}, + sub_table_name = #{subTableName}, + sub_table_fk_name = #{subTableFkName}, + class_name = #{className}, + function_author = #{functionAuthor}, + gen_type = #{genType}, + gen_path = #{genPath}, + tpl_category = #{tplCategory}, + package_name = #{packageName}, + module_name = #{moduleName}, + business_name = #{businessName}, + function_name = #{functionName}, + options = #{options}, + update_by = #{updateBy}, + remark = #{remark}, + update_time = sysdate() + + where table_id = #{tableId} + + + + delete from gen_table where table_id in + + #{tableId} + + + + \ No newline at end of file diff --git a/ruoyi-generator/src/main/resources/mybatisplusvm/java/controller.java.vm b/ruoyi-generator/src/main/resources/mybatisplusvm/java/controller.java.vm new file mode 100644 index 00000000..d2894379 --- /dev/null +++ b/ruoyi-generator/src/main/resources/mybatisplusvm/java/controller.java.vm @@ -0,0 +1,129 @@ +package ${packageName}.controller; + +import java.util.List; + +import io.swagger.annotations.*; +import com.baomidou.mybatisplus.core.metadata.IPage; + +import javax.servlet.http.HttpServletResponse; + +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.PutMapping; +import org.springframework.web.bind.annotation.DeleteMapping; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.enums.BusinessType; +import ${packageName}.domain.${ClassName}; +import ${packageName}.service.I${ClassName}Service; +import com.ruoyi.common.utils.poi.ExcelUtil; +#if($table.crud || $table.sub) +import com.ruoyi.common.core.page.TableDataInfo; +#elseif($table.tree) +#end + +/** + * ${functionName}Controller + * + * @author ${author} + * @date ${datetime} + */ +@RestController +@RequestMapping("/${moduleName}/${businessName}") +@Api(tags = {"【${functionName}】Controller"}) +public class ${ClassName}Controller extends BaseController { + @Autowired + private I${ClassName}Service ${className}Service; + +/** + * 分页查询${functionName}列表 + */ +@PreAuthorize("@ss.hasPermi('${permissionPrefix}:list')") +@GetMapping("/list") +@ApiOperation("分页查询${functionName}列表") + #if($table.crud || $table.sub) + public TableDataInfo list(${ClassName} ${className}) { + IPage<${ClassName}> list = ${className}Service.selectList(getPage(), ${className}); + return getDataTable(list); + } + #elseif($table.tree) + public AjaxResult list(${ClassName} ${className}) { + List<${ClassName}> list = ${className}Service.select${ClassName}List(${className}); + return AjaxResult.success(list); + } + #end + + /** + * 查询${functionName}列表 + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:list')") + @GetMapping("/list/all") + @ApiOperation("查询${functionName}列表") + public AjaxResult listAll(${ClassName} ${className}) { + List<${ClassName}> list = ${className}Service.selectListAll(${className}); + return success(list); + } + + /** + * 导出${functionName}列表 + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:export')") + @Log(title = "${functionName}", businessType = BusinessType.EXPORT) + @PostMapping("/export") + @ApiOperation("导出${functionName}列表Excel") + public void export(HttpServletResponse response, ${ClassName} ${className}) { + List<${ClassName}> list = ${className}Service.selectListAll(${className}); + ExcelUtil<${ClassName}> util = new ExcelUtil<${ClassName}>(${ClassName}. class); + util.exportExcel(response, list, "${functionName}数据"); + } + + /** + * 获取${functionName}详细信息 + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:query')") + @GetMapping(value = "/{${pkColumn.javaField}}") + @ApiOperation("获取${functionName}详细信息") + public AjaxResult getInfo(@PathVariable("${pkColumn.javaField}") ${pkColumn.javaType} ${pkColumn.javaField}) { + return success(${className}Service.selectById(${pkColumn.javaField})); + } + + /** + * 新增${functionName} + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:add')") + @Log(title = "${functionName}", businessType = BusinessType.INSERT) + @PostMapping + @ApiOperation("新增${functionName}") + public AjaxResult add(@RequestBody ${ClassName} ${className}) { + return toAjax(${className}Service.insert(${className})); + } + + /** + * 修改${functionName} + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:edit')") + @Log(title = "${functionName}", businessType = BusinessType.UPDATE) + @PutMapping + @ApiOperation("修改${functionName}") + public AjaxResult edit(@RequestBody ${ClassName} ${className}) { + return toAjax(${className}Service.update(${className})); + } + + /** + * 删除${functionName} + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:remove')") + @Log(title = "${functionName}", businessType = BusinessType.DELETE) + @DeleteMapping("/{${pkColumn.javaField}s}") + @ApiOperation("删除${functionName}") + public AjaxResult remove(@PathVariable ${pkColumn.javaType}[] ${pkColumn.javaField}s) { + return toAjax(${className}Service.deleteByIds(${pkColumn.javaField}s)); + } +} diff --git a/ruoyi-generator/src/main/resources/mybatisplusvm/java/domain.java.vm b/ruoyi-generator/src/main/resources/mybatisplusvm/java/domain.java.vm new file mode 100644 index 00000000..23249f21 --- /dev/null +++ b/ruoyi-generator/src/main/resources/mybatisplusvm/java/domain.java.vm @@ -0,0 +1,72 @@ +package ${packageName}.domain; + +#foreach ($import in $importList) +import ${import}; +#end +import com.ruoyi.common.annotation.Excel; +#if($table.crud || $table.sub) +import com.ruoyi.common.core.domain.BaseEntity; +#elseif($table.tree) +import com.ruoyi.common.core.domain.TreeEntity; +#end +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import com.baomidou.mybatisplus.annotation.TableName; +/** + * ${functionName}对象 ${tableName} + * + * @author ${author} + * @date ${datetime} + */ +#if($table.crud || $table.sub) + #set($Entity="BaseEntity") +#elseif($table.tree) + #set($Entity="TreeEntity") +#end + +@ApiModel(value = "${ClassName}", description = "${functionName}对象 ${tableName}") +@Data +@TableName(value = "${tableName}", autoResultMap = true) +public class ${ClassName} extends ${Entity} +{ + private static final long serialVersionUID=1L; + +#foreach ($column in $columns) +#if(!$table.isSuperColumn($column.javaField)) + /** $column.columnComment */ +#if($column.list) +#set($parentheseIndex=$column.columnComment.indexOf("(")) +#if($parentheseIndex != -1) +#set($comment=$column.columnComment.substring(0, $parentheseIndex)) +#else +#set($comment=$column.columnComment) +#end +#if($parentheseIndex != -1) + @Excel(name = "${comment}", readConverterExp = "$column.readConverterExp()") +#elseif($column.javaType == 'Date') + @JsonFormat(pattern = "yyyy-MM-dd") + @Excel(name = "${comment}", width = 30, dateFormat = "yyyy-MM-dd") +#else + @Excel(name = "${comment}") +#end +#end + @ApiModelProperty(value = "$column.columnComment") + private $column.javaType $column.javaField; + +#end +#end +#if($table.sub) +/** $table.subTable.functionName信息 */ + private List<${subClassName}> ${subclassName}List; + +#end +#foreach ($column in $columns) + #if(!$table.isSuperColumn($column.javaField)) + #if($column.javaField.length() > 2 && $column.javaField.substring(1,2).matches("[A-Z]")) + #set($AttrName=$column.javaField) + #else + #set($AttrName=$column.javaField.substring(0,1).toUpperCase() + ${column.javaField.substring(1)}) + #end + #end#end +} diff --git a/ruoyi-generator/src/main/resources/mybatisplusvm/java/mapper.java.vm b/ruoyi-generator/src/main/resources/mybatisplusvm/java/mapper.java.vm new file mode 100644 index 00000000..a46b08e2 --- /dev/null +++ b/ruoyi-generator/src/main/resources/mybatisplusvm/java/mapper.java.vm @@ -0,0 +1,36 @@ +package ${packageName}.mapper; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import com.baomidou.mybatisplus.core.metadata.IPage; +import java.util.List; +import org.apache.ibatis.annotations.Param; +import ${packageName}.domain.${ClassName}; +#if($table.sub) +import ${packageName}.domain.${subClassName}; +#end + +/** + * ${functionName}Mapper接口 + * + * @author ${author} + * @date ${datetime} + */ +public interface ${ClassName}Mapper extends BaseMapper<${ClassName}> +{ + /** + * 分页查询${functionName}列表 + * + * param page 分页信息 + * @param ${className} ${functionName}信息 + * @return ${functionName}集合 + */ + public IPage<${ClassName}> select${ClassName}Page(IPage<${ClassName}> page,@Param("entity") ${ClassName} ${className}); + + /** + * 查询所有${functionName}列表 + * + * @param ${className} ${functionName}信息 + * @return ${functionName}集合 + */ + public List<${ClassName}> select${ClassName}List(${ClassName} ${className}); +} diff --git a/ruoyi-generator/src/main/resources/mybatisplusvm/java/service.java.vm b/ruoyi-generator/src/main/resources/mybatisplusvm/java/service.java.vm new file mode 100644 index 00000000..5cda17db --- /dev/null +++ b/ruoyi-generator/src/main/resources/mybatisplusvm/java/service.java.vm @@ -0,0 +1,71 @@ +package ${packageName}.service; + +import java.util.List; +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.baomidou.mybatisplus.extension.service.IService; +import ${packageName}.domain.${ClassName}; + +/** + * ${functionName}Service接口 + * + * @author ${author} + * @date ${datetime} + */ +public interface I${ClassName}Service extends IService<${ClassName}> +{ + /** + * 查询${functionName} + * + * @param ${pkColumn.javaField} ${functionName}ID + * @return ${functionName} + */ + public ${ClassName} selectById(${pkColumn.javaType} ${pkColumn.javaField}); + + /** + * 分页查询${functionName}列表 + * + * @param ${className} ${functionName} + * @return ${functionName}集合 + */ + public IPage<${ClassName}> selectList(IPage<${ClassName}> page, ${ClassName} ${className}); + + /** + * 查询所有${functionName}列表 + * + * @param ${className} ${functionName} + * @return ${functionName}集合 + */ + public List<${ClassName}> selectListAll(${ClassName} ${className}); + + /** + * 新增${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ + public int insert(${ClassName} ${className}); + + /** + * 修改${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ + public int update(${ClassName} ${className}); + + /** + * 批量删除${functionName} + * + * @param ${pkColumn.javaField}s 需要删除的${functionName}ID + * @return 结果 + */ + public int deleteByIds(${pkColumn.javaType}[] ${pkColumn.javaField}s); + + /** + * 删除${functionName}信息 + * + * @param ${pkColumn.javaField} ${functionName}ID + * @return 结果 + */ + public int deleteById(${pkColumn.javaType} ${pkColumn.javaField}); +} diff --git a/ruoyi-generator/src/main/resources/mybatisplusvm/java/serviceImpl.java.vm b/ruoyi-generator/src/main/resources/mybatisplusvm/java/serviceImpl.java.vm new file mode 100644 index 00000000..77411bd9 --- /dev/null +++ b/ruoyi-generator/src/main/resources/mybatisplusvm/java/serviceImpl.java.vm @@ -0,0 +1,174 @@ +package ${packageName}.service.impl; + +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import java.util.Arrays; +import java.util.List; +#foreach ($column in $columns) +#if($column.javaField == 'createTime' || $column.javaField == 'updateTime') +import com.ruoyi.common.utils.DateUtils; +#break +#end +#end +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +#if($table.sub) +import java.util.ArrayList; +import com.ruoyi.common.utils.StringUtils; +import org.springframework.transaction.annotation.Transactional; +import ${packageName}.domain.${subClassName}; +#end +import ${packageName}.mapper.${ClassName}Mapper; +import ${packageName}.domain.${ClassName}; +import ${packageName}.service.I${ClassName}Service; + +/** + * ${functionName}Service业务层处理 + * + * @author ${author} + * @date ${datetime} + */ +@Service +public class ${ClassName}ServiceImpl extends ServiceImpl<${ClassName}Mapper, ${ClassName}> implements I${ClassName}Service +{ + @Autowired + private ${ClassName}Mapper ${className}Mapper; + + /** + * 查询${functionName} + * + * @param ${pkColumn.javaField} ${functionName}ID + * @return ${functionName} + */ + @Override + public ${ClassName} selectById(${pkColumn.javaType} ${pkColumn.javaField}) { + return ${className}Mapper.selectById(${pkColumn.javaField}); + } + + /** + * 分页查询${functionName}列表 + * + * @param ${className} ${functionName} + * @return ${functionName} + */ + @Override + public IPage<${ClassName}> selectList(IPage<${ClassName}> page, ${ClassName} ${className}) { + return ${className}Mapper.select${ClassName}Page(page, ${className}); + } + + /** + * 查询所有${functionName}列表 + * + * @param ${className} ${functionName} + * @return ${functionName} + */ + @Override + public List<${ClassName}> selectListAll(${ClassName} ${className}) { + return ${className}Mapper.select${ClassName}List(${className}); + } + + /** + * 新增${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ +#if($table.sub) + @Transactional +#end + @Override + public int insert(${ClassName} ${className}) { +#foreach ($column in $columns) + #if($column.javaField == 'createTime') + ${className}.setCreateTime(DateUtils.getNowDate()); + #end +#end +#if($table.sub) + int rows = ${className}Mapper.insert(${className}); + insert${subClassName}(${className}); + return rows; +#else + return ${className}Mapper.insert(${className}); +#end + } + + /** + * 修改${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ +#if($table.sub) + @Transactional +#end + @Override + public int update(${ClassName} ${className}) { +#foreach ($column in $columns) + #if($column.javaField == 'updateTime') + ${className}.setUpdateTime(DateUtils.getNowDate()); + #end +#end +#if($table.sub) + ${className}Mapper.delete${subClassName}By${subTableFkClassName}(${className}.get${pkColumn.capJavaField}()); + insert${subClassName}(${className}); +#end + return ${className}Mapper.updateById(${className}); + } + + /** + * 批量删除${functionName} + * + * @param ${pkColumn.javaField}s 需要删除的${functionName}ID + * @return 结果 + */ +#if($table.sub) + @Transactional +#end + @Override + public int deleteByIds(${pkColumn.javaType}[] ${pkColumn.javaField}s) { +#if($table.sub) + ${className}Mapper.delete${subClassName}By${subTableFkClassName}s(${pkColumn.javaField}s); +#end + return ${className}Mapper.deleteBatchIds(Arrays.asList(${pkColumn.javaField}s)); + } + + /** + * 删除${functionName}信息 + * + * @param ${pkColumn.javaField} ${functionName}ID + * @return 结果 + */ +#if($table.sub) + @Transactional +#end + @Override + public int deleteById(${pkColumn.javaType} ${pkColumn.javaField}) { +#if($table.sub) + ${className}Mapper.delete${subClassName}By${subTableFkClassName}(${pkColumn.javaField}); +#end + return ${className}Mapper.deleteById(${pkColumn.javaField}); + } +#if($table.sub) + + /** + * 新增${subTable.functionName}信息 + * + * @param ${className} ${functionName}对象 + */ + public void insert${subClassName}(${ClassName} ${className}) { + List<${subClassName}> ${subclassName}List = ${className}.get${subClassName}List(); + Long ${pkColumn.javaField} = ${className}.get${pkColumn.capJavaField}(); + if (StringUtils.isNotNull(${subclassName}List)) { + List<${subClassName}> list = new ArrayList<${subClassName}>(); + for (${subClassName} ${subclassName} :${subclassName}List) + { + ${subclassName}.set${subTableFkClassName}(${pkColumn.javaField}); + list.add(${subclassName}); + } + if (list.size() > 0) { + ${className}Mapper.batch${subClassName}(list); + } + } + } +#end +} diff --git a/ruoyi-generator/src/main/resources/mybatisplusvm/java/sub-domain.java.vm b/ruoyi-generator/src/main/resources/mybatisplusvm/java/sub-domain.java.vm new file mode 100644 index 00000000..ad3890a0 --- /dev/null +++ b/ruoyi-generator/src/main/resources/mybatisplusvm/java/sub-domain.java.vm @@ -0,0 +1,76 @@ +package ${packageName}.domain; + + #foreach ($import in $subImportList) + import ${import}; + #end +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import com.ruoyi.common.annotation.Excel; +import com.ruoyi.common.core.domain.BaseEntity; + +/** + * ${subTable.functionName}对象 ${subTableName} + * + * @author ${author} + * @date ${datetime} + */ +public class ${subClassName} extends BaseEntity + { +private static final long serialVersionUID=1L; + +#foreach ($column in $subTable.columns) + #if(!$table.isSuperColumn($column.javaField)) + /** $column.columnComment */ + #if($column.list) + #set($parentheseIndex=$column.columnComment.indexOf("(")) + #if($parentheseIndex != -1) + #set($comment=$column.columnComment.substring(0, $parentheseIndex)) + #else + #set($comment=$column.columnComment) + #end + #if($parentheseIndex != -1) + @Excel(name = "${comment}", readConverterExp = "$column.readConverterExp()") + #elseif($column.javaType == 'Date') + @JsonFormat(pattern = "yyyy-MM-dd") + @Excel(name = "${comment}", width = 30, dateFormat = "yyyy-MM-dd") + #else + @Excel(name = "${comment}") + #end + #end + private $column.javaType $column.javaField; + + #end +#end +#foreach ($column in $subTable.columns) + #if(!$table.isSuperColumn($column.javaField)) + #if($column.javaField.length() > 2 && $column.javaField.substring(1,2).matches("[A-Z]")) + #set($AttrName=$column.javaField) + #else + #set($AttrName=$column.javaField.substring(0,1).toUpperCase() + ${column.javaField.substring(1)}) + #end + public void set${AttrName}($column.javaType $column.javaField) + { + this.$column.javaField = $column.javaField; + } + + public $column.javaType get${AttrName}() + { + return $column.javaField; + } + #end +#end + +@Override +public String toString(){ + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) + #foreach ($column in $subTable.columns) + #if($column.javaField.length() > 2 && $column.javaField.substring(1,2).matches("[A-Z]")) + #set($AttrName=$column.javaField) + #else + #set($AttrName=$column.javaField.substring(0,1).toUpperCase() + ${column.javaField.substring(1)}) + #end + .append("${column.javaField}",get${AttrName}()) + #end + .toString(); + } + } diff --git a/ruoyi-generator/src/main/resources/mybatisplusvm/js/api.js.vm b/ruoyi-generator/src/main/resources/mybatisplusvm/js/api.js.vm new file mode 100644 index 00000000..9295524a --- /dev/null +++ b/ruoyi-generator/src/main/resources/mybatisplusvm/js/api.js.vm @@ -0,0 +1,44 @@ +import request from '@/utils/request' + +// 查询${functionName}列表 +export function list${BusinessName}(query) { + return request({ + url: '/${moduleName}/${businessName}/list', + method: 'get', + params: query + }) +} + +// 查询${functionName}详细 +export function get${BusinessName}(${pkColumn.javaField}) { + return request({ + url: '/${moduleName}/${businessName}/' + ${pkColumn.javaField}, + method: 'get' + }) +} + +// 新增${functionName} +export function add${BusinessName}(data) { + return request({ + url: '/${moduleName}/${businessName}', + method: 'post', + data: data + }) +} + +// 修改${functionName} +export function update${BusinessName}(data) { + return request({ + url: '/${moduleName}/${businessName}', + method: 'put', + data: data + }) +} + +// 删除${functionName} +export function del${BusinessName}(${pkColumn.javaField}) { + return request({ + url: '/${moduleName}/${businessName}/' + ${pkColumn.javaField}, + method: 'delete' + }) +} diff --git a/ruoyi-generator/src/main/resources/mybatisplusvm/sql/sql.vm b/ruoyi-generator/src/main/resources/mybatisplusvm/sql/sql.vm new file mode 100644 index 00000000..05755835 --- /dev/null +++ b/ruoyi-generator/src/main/resources/mybatisplusvm/sql/sql.vm @@ -0,0 +1,22 @@ +-- 菜单 SQL +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}', '${parentMenuId}', '1', '${businessName}', '${moduleName}/${businessName}/index', 1, 0, 'C', '0', '0', '${permissionPrefix}:list', '#', 'admin', sysdate(), '', null, '${functionName}菜单'); + +-- 按钮父菜单ID +SELECT @parentId := LAST_INSERT_ID(); + +-- 按钮 SQL +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}查询', @parentId, '1', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:query', '#', 'admin', sysdate(), '', null, ''); + +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}新增', @parentId, '2', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:add', '#', 'admin', sysdate(), '', null, ''); + +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}修改', @parentId, '3', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:edit', '#', 'admin', sysdate(), '', null, ''); + +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}删除', @parentId, '4', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:remove', '#', 'admin', sysdate(), '', null, ''); + +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}导出', @parentId, '5', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:export', '#', 'admin', sysdate(), '', null, ''); \ No newline at end of file diff --git a/ruoyi-generator/src/main/resources/mybatisplusvm/vue/index-tree.vue.vm b/ruoyi-generator/src/main/resources/mybatisplusvm/vue/index-tree.vue.vm new file mode 100644 index 00000000..3def7045 --- /dev/null +++ b/ruoyi-generator/src/main/resources/mybatisplusvm/vue/index-tree.vue.vm @@ -0,0 +1,502 @@ + + + diff --git a/ruoyi-generator/src/main/resources/mybatisplusvm/vue/index.vue.vm b/ruoyi-generator/src/main/resources/mybatisplusvm/vue/index.vue.vm new file mode 100644 index 00000000..e9a1fae1 --- /dev/null +++ b/ruoyi-generator/src/main/resources/mybatisplusvm/vue/index.vue.vm @@ -0,0 +1,598 @@ + + + diff --git a/ruoyi-generator/src/main/resources/mybatisplusvm/vue/v3/index-tree.vue.vm b/ruoyi-generator/src/main/resources/mybatisplusvm/vue/v3/index-tree.vue.vm new file mode 100644 index 00000000..862297c7 --- /dev/null +++ b/ruoyi-generator/src/main/resources/mybatisplusvm/vue/v3/index-tree.vue.vm @@ -0,0 +1,486 @@ + + + diff --git a/ruoyi-generator/src/main/resources/mybatisplusvm/vue/v3/index.vue.vm b/ruoyi-generator/src/main/resources/mybatisplusvm/vue/v3/index.vue.vm new file mode 100644 index 00000000..f66cc3b8 --- /dev/null +++ b/ruoyi-generator/src/main/resources/mybatisplusvm/vue/v3/index.vue.vm @@ -0,0 +1,596 @@ + + + diff --git a/ruoyi-generator/src/main/resources/mybatisplusvm/vue/v3/readme.txt b/ruoyi-generator/src/main/resources/mybatisplusvm/vue/v3/readme.txt new file mode 100644 index 00000000..99239bb5 --- /dev/null +++ b/ruoyi-generator/src/main/resources/mybatisplusvm/vue/v3/readme.txt @@ -0,0 +1 @@ +ʹõRuoYi-Vue3ǰˣôҪһ´Ŀ¼ģindex.vue.vmindex-tree.vue.vmļϼvueĿ¼ \ No newline at end of file diff --git a/ruoyi-generator/src/main/resources/mybatisplusvm/xml/mapper.xml.vm b/ruoyi-generator/src/main/resources/mybatisplusvm/xml/mapper.xml.vm new file mode 100644 index 00000000..4fc7d573 --- /dev/null +++ b/ruoyi-generator/src/main/resources/mybatisplusvm/xml/mapper.xml.vm @@ -0,0 +1,109 @@ + + + + + +#foreach ($column in $columns) + +#end + +#if($table.sub) + + + + + + +#foreach ($column in $subTable.columns) + +#end + +#end + + + select#foreach($column in $columns) $column.columnName#if($foreach.count != $columns.size()),#end#end from ${tableName} + + + + + + diff --git a/ruoyi-generator/src/main/resources/vm/java/controller.java.vm b/ruoyi-generator/src/main/resources/vm/java/controller.java.vm new file mode 100644 index 00000000..5955c6f6 --- /dev/null +++ b/ruoyi-generator/src/main/resources/vm/java/controller.java.vm @@ -0,0 +1,108 @@ +package ${packageName}.controller; + +import java.util.List; +import javax.servlet.http.HttpServletResponse; + +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.PutMapping; +import org.springframework.web.bind.annotation.DeleteMapping; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.enums.BusinessType; +import ${packageName}.domain.${ClassName}; +import ${packageName}.service.I${ClassName}Service; +import com.ruoyi.common.utils.poi.ExcelUtil; +#if($table.crud || $table.sub) +import com.ruoyi.common.core.page.TableDataInfo; +#elseif($table.tree) +#end + +/** + * ${functionName}Controller + * + * @author ${author} + * @date ${datetime} + */ +@RestController +@RequestMapping("/${moduleName}/${businessName}") +public class ${ClassName}Controller extends BaseController { + @Autowired + private I${ClassName}Service ${className}Service; + +/** + * 查询${functionName}列表 + */ +@PreAuthorize("@ss.hasPermi('${permissionPrefix}:list')") +@GetMapping("/list") + #if($table.crud || $table.sub) + public TableDataInfo list(${ClassName} ${className}) { + startPage(); + List<${ClassName}> list = ${className}Service.select${ClassName}List(${className}); + return getDataTable(list); + } + #elseif($table.tree) + public AjaxResult list(${ClassName} ${className}) { + List<${ClassName}> list = ${className}Service.select${ClassName}List(${className}); + return success(list); + } + #end + + /** + * 导出${functionName}列表 + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:export')") + @Log(title = "${functionName}", businessType = BusinessType.EXPORT) + @PostMapping("/export") + public void export(HttpServletResponse response, ${ClassName} ${className}) { + List<${ClassName}> list = ${className}Service.select${ClassName}List(${className}); + ExcelUtil<${ClassName}> util = new ExcelUtil<${ClassName}>(${ClassName}. class); + util.exportExcel(response, list, "${functionName}数据"); + } + + /** + * 获取${functionName}详细信息 + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:query')") + @GetMapping(value = "/{${pkColumn.javaField}}") + public AjaxResult getInfo(@PathVariable("${pkColumn.javaField}") ${pkColumn.javaType} ${pkColumn.javaField}) { + return success(${className}Service.select${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaField})); + } + + /** + * 新增${functionName} + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:add')") + @Log(title = "${functionName}", businessType = BusinessType.INSERT) + @PostMapping + public AjaxResult add(@RequestBody ${ClassName} ${className}) { + return toAjax(${className}Service.insert${ClassName}(${className})); + } + + /** + * 修改${functionName} + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:edit')") + @Log(title = "${functionName}", businessType = BusinessType.UPDATE) + @PutMapping + public AjaxResult edit(@RequestBody ${ClassName} ${className}) { + return toAjax(${className}Service.update${ClassName}(${className})); + } + + /** + * 删除${functionName} + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:remove')") + @Log(title = "${functionName}", businessType = BusinessType.DELETE) + @DeleteMapping("/{${pkColumn.javaField}s}") + public AjaxResult remove(@PathVariable ${pkColumn.javaType}[] ${pkColumn.javaField}s) { + return toAjax(${className}Service.delete${ClassName}By${pkColumn.capJavaField}s(${pkColumn.javaField}s)); + } +} diff --git a/ruoyi-generator/src/main/resources/vm/java/domain.java.vm b/ruoyi-generator/src/main/resources/vm/java/domain.java.vm new file mode 100644 index 00000000..bd51c177 --- /dev/null +++ b/ruoyi-generator/src/main/resources/vm/java/domain.java.vm @@ -0,0 +1,105 @@ +package ${packageName}.domain; + +#foreach ($import in $importList) +import ${import}; +#end +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import com.ruoyi.common.annotation.Excel; +#if($table.crud || $table.sub) +import com.ruoyi.common.core.domain.BaseEntity; +#elseif($table.tree) +import com.ruoyi.common.core.domain.TreeEntity; +#end + +/** + * ${functionName}对象 ${tableName} + * + * @author ${author} + * @date ${datetime} + */ +#if($table.crud || $table.sub) +#set($Entity="BaseEntity") +#elseif($table.tree) +#set($Entity="TreeEntity") +#end +public class ${ClassName} extends ${Entity} +{ + private static final long serialVersionUID = 1L; + +#foreach ($column in $columns) +#if(!$table.isSuperColumn($column.javaField)) + /** $column.columnComment */ +#if($column.list) +#set($parentheseIndex=$column.columnComment.indexOf("(")) +#if($parentheseIndex != -1) +#set($comment=$column.columnComment.substring(0, $parentheseIndex)) +#else +#set($comment=$column.columnComment) +#end +#if($parentheseIndex != -1) + @Excel(name = "${comment}", readConverterExp = "$column.readConverterExp()") +#elseif($column.javaType == 'Date') + @JsonFormat(pattern = "yyyy-MM-dd") + @Excel(name = "${comment}", width = 30, dateFormat = "yyyy-MM-dd") +#else + @Excel(name = "${comment}") +#end +#end + private $column.javaType $column.javaField; + +#end +#end +#if($table.sub) + /** $table.subTable.functionName信息 */ + private List<${subClassName}> ${subclassName}List; + +#end +#foreach ($column in $columns) +#if(!$table.isSuperColumn($column.javaField)) +#if($column.javaField.length() > 2 && $column.javaField.substring(1,2).matches("[A-Z]")) +#set($AttrName=$column.javaField) +#else +#set($AttrName=$column.javaField.substring(0,1).toUpperCase() + ${column.javaField.substring(1)}) +#end + public void set${AttrName}($column.javaType $column.javaField) + { + this.$column.javaField = $column.javaField; + } + + public $column.javaType get${AttrName}() + { + return $column.javaField; + } +#end +#end + +#if($table.sub) + public List<${subClassName}> get${subClassName}List() + { + return ${subclassName}List; + } + + public void set${subClassName}List(List<${subClassName}> ${subclassName}List) + { + this.${subclassName}List = ${subclassName}List; + } + +#end + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) +#foreach ($column in $columns) +#if($column.javaField.length() > 2 && $column.javaField.substring(1,2).matches("[A-Z]")) +#set($AttrName=$column.javaField) +#else +#set($AttrName=$column.javaField.substring(0,1).toUpperCase() + ${column.javaField.substring(1)}) +#end + .append("${column.javaField}", get${AttrName}()) +#end +#if($table.sub) + .append("${subclassName}List", get${subClassName}List()) +#end + .toString(); + } +} diff --git a/ruoyi-generator/src/main/resources/vm/java/mapper.java.vm b/ruoyi-generator/src/main/resources/vm/java/mapper.java.vm new file mode 100644 index 00000000..7e7d7c26 --- /dev/null +++ b/ruoyi-generator/src/main/resources/vm/java/mapper.java.vm @@ -0,0 +1,91 @@ +package ${packageName}.mapper; + +import java.util.List; +import ${packageName}.domain.${ClassName}; +#if($table.sub) +import ${packageName}.domain.${subClassName}; +#end + +/** + * ${functionName}Mapper接口 + * + * @author ${author} + * @date ${datetime} + */ +public interface ${ClassName}Mapper +{ + /** + * 查询${functionName} + * + * @param ${pkColumn.javaField} ${functionName}主键 + * @return ${functionName} + */ + public ${ClassName} select${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaType} ${pkColumn.javaField}); + + /** + * 查询${functionName}列表 + * + * @param ${className} ${functionName} + * @return ${functionName}集合 + */ + public List<${ClassName}> select${ClassName}List(${ClassName} ${className}); + + /** + * 新增${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ + public int insert${ClassName}(${ClassName} ${className}); + + /** + * 修改${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ + public int update${ClassName}(${ClassName} ${className}); + + /** + * 删除${functionName} + * + * @param ${pkColumn.javaField} ${functionName}主键 + * @return 结果 + */ + public int delete${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaType} ${pkColumn.javaField}); + + /** + * 批量删除${functionName} + * + * @param ${pkColumn.javaField}s 需要删除的数据主键集合 + * @return 结果 + */ + public int delete${ClassName}By${pkColumn.capJavaField}s(${pkColumn.javaType}[] ${pkColumn.javaField}s); +#if($table.sub) + + /** + * 批量删除${subTable.functionName} + * + * @param ${pkColumn.javaField}s 需要删除的数据主键集合 + * @return 结果 + */ + public int delete${subClassName}By${subTableFkClassName}s(${pkColumn.javaType}[] ${pkColumn.javaField}s); + + /** + * 批量新增${subTable.functionName} + * + * @param ${subclassName}List ${subTable.functionName}列表 + * @return 结果 + */ + public int batch${subClassName}(List<${subClassName}> ${subclassName}List); + + + /** + * 通过${functionName}主键删除${subTable.functionName}信息 + * + * @param ${pkColumn.javaField} ${functionName}ID + * @return 结果 + */ + public int delete${subClassName}By${subTableFkClassName}(${pkColumn.javaType} ${pkColumn.javaField}); +#end +} diff --git a/ruoyi-generator/src/main/resources/vm/java/service.java.vm b/ruoyi-generator/src/main/resources/vm/java/service.java.vm new file mode 100644 index 00000000..264882b2 --- /dev/null +++ b/ruoyi-generator/src/main/resources/vm/java/service.java.vm @@ -0,0 +1,61 @@ +package ${packageName}.service; + +import java.util.List; +import ${packageName}.domain.${ClassName}; + +/** + * ${functionName}Service接口 + * + * @author ${author} + * @date ${datetime} + */ +public interface I${ClassName}Service +{ + /** + * 查询${functionName} + * + * @param ${pkColumn.javaField} ${functionName}主键 + * @return ${functionName} + */ + public ${ClassName} select${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaType} ${pkColumn.javaField}); + + /** + * 查询${functionName}列表 + * + * @param ${className} ${functionName} + * @return ${functionName}集合 + */ + public List<${ClassName}> select${ClassName}List(${ClassName} ${className}); + + /** + * 新增${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ + public int insert${ClassName}(${ClassName} ${className}); + + /** + * 修改${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ + public int update${ClassName}(${ClassName} ${className}); + + /** + * 批量删除${functionName} + * + * @param ${pkColumn.javaField}s 需要删除的${functionName}主键集合 + * @return 结果 + */ + public int delete${ClassName}By${pkColumn.capJavaField}s(${pkColumn.javaType}[] ${pkColumn.javaField}s); + + /** + * 删除${functionName}信息 + * + * @param ${pkColumn.javaField} ${functionName}主键 + * @return 结果 + */ + public int delete${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaType} ${pkColumn.javaField}); +} diff --git a/ruoyi-generator/src/main/resources/vm/java/serviceImpl.java.vm b/ruoyi-generator/src/main/resources/vm/java/serviceImpl.java.vm new file mode 100644 index 00000000..14746e1a --- /dev/null +++ b/ruoyi-generator/src/main/resources/vm/java/serviceImpl.java.vm @@ -0,0 +1,169 @@ +package ${packageName}.service.impl; + +import java.util.List; +#foreach ($column in $columns) +#if($column.javaField == 'createTime' || $column.javaField == 'updateTime') +import com.ruoyi.common.utils.DateUtils; +#break +#end +#end +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +#if($table.sub) +import java.util.ArrayList; +import com.ruoyi.common.utils.StringUtils; +import org.springframework.transaction.annotation.Transactional; +import ${packageName}.domain.${subClassName}; +#end +import ${packageName}.mapper.${ClassName}Mapper; +import ${packageName}.domain.${ClassName}; +import ${packageName}.service.I${ClassName}Service; + +/** + * ${functionName}Service业务层处理 + * + * @author ${author} + * @date ${datetime} + */ +@Service +public class ${ClassName}ServiceImpl implements I${ClassName}Service +{ + @Autowired + private ${ClassName}Mapper ${className}Mapper; + + /** + * 查询${functionName} + * + * @param ${pkColumn.javaField} ${functionName}主键 + * @return ${functionName} + */ + @Override + public ${ClassName} select${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaType} ${pkColumn.javaField}) + { + return ${className}Mapper.select${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaField}); + } + + /** + * 查询${functionName}列表 + * + * @param ${className} ${functionName} + * @return ${functionName} + */ + @Override + public List<${ClassName}> select${ClassName}List(${ClassName} ${className}) + { + return ${className}Mapper.select${ClassName}List(${className}); + } + + /** + * 新增${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ +#if($table.sub) + @Transactional +#end + @Override + public int insert${ClassName}(${ClassName} ${className}) + { +#foreach ($column in $columns) +#if($column.javaField == 'createTime') + ${className}.setCreateTime(DateUtils.getNowDate()); +#end +#end +#if($table.sub) + int rows = ${className}Mapper.insert${ClassName}(${className}); + insert${subClassName}(${className}); + return rows; +#else + return ${className}Mapper.insert${ClassName}(${className}); +#end + } + + /** + * 修改${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ +#if($table.sub) + @Transactional +#end + @Override + public int update${ClassName}(${ClassName} ${className}) + { +#foreach ($column in $columns) +#if($column.javaField == 'updateTime') + ${className}.setUpdateTime(DateUtils.getNowDate()); +#end +#end +#if($table.sub) + ${className}Mapper.delete${subClassName}By${subTableFkClassName}(${className}.get${pkColumn.capJavaField}()); + insert${subClassName}(${className}); +#end + return ${className}Mapper.update${ClassName}(${className}); + } + + /** + * 批量删除${functionName} + * + * @param ${pkColumn.javaField}s 需要删除的${functionName}主键 + * @return 结果 + */ +#if($table.sub) + @Transactional +#end + @Override + public int delete${ClassName}By${pkColumn.capJavaField}s(${pkColumn.javaType}[] ${pkColumn.javaField}s) + { +#if($table.sub) + ${className}Mapper.delete${subClassName}By${subTableFkClassName}s(${pkColumn.javaField}s); +#end + return ${className}Mapper.delete${ClassName}By${pkColumn.capJavaField}s(${pkColumn.javaField}s); + } + + /** + * 删除${functionName}信息 + * + * @param ${pkColumn.javaField} ${functionName}主键 + * @return 结果 + */ +#if($table.sub) + @Transactional +#end + @Override + public int delete${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaType} ${pkColumn.javaField}) + { +#if($table.sub) + ${className}Mapper.delete${subClassName}By${subTableFkClassName}(${pkColumn.javaField}); +#end + return ${className}Mapper.delete${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaField}); + } +#if($table.sub) + + /** + * 新增${subTable.functionName}信息 + * + * @param ${className} ${functionName}对象 + */ + public void insert${subClassName}(${ClassName} ${className}) + { + List<${subClassName}> ${subclassName}List = ${className}.get${subClassName}List(); + ${pkColumn.javaType} ${pkColumn.javaField} = ${className}.get${pkColumn.capJavaField}(); + if (StringUtils.isNotNull(${subclassName}List)) + { + List<${subClassName}> list = new ArrayList<${subClassName}>(); + for (${subClassName} ${subclassName} : ${subclassName}List) + { + ${subclassName}.set${subTableFkClassName}(${pkColumn.javaField}); + list.add(${subclassName}); + } + if (list.size() > 0) + { + ${className}Mapper.batch${subClassName}(list); + } + } + } +#end +} diff --git a/ruoyi-generator/src/main/resources/vm/java/sub-domain.java.vm b/ruoyi-generator/src/main/resources/vm/java/sub-domain.java.vm new file mode 100644 index 00000000..a3f53eba --- /dev/null +++ b/ruoyi-generator/src/main/resources/vm/java/sub-domain.java.vm @@ -0,0 +1,76 @@ +package ${packageName}.domain; + +#foreach ($import in $subImportList) +import ${import}; +#end +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import com.ruoyi.common.annotation.Excel; +import com.ruoyi.common.core.domain.BaseEntity; + +/** + * ${subTable.functionName}对象 ${subTableName} + * + * @author ${author} + * @date ${datetime} + */ +public class ${subClassName} extends BaseEntity +{ + private static final long serialVersionUID = 1L; + +#foreach ($column in $subTable.columns) +#if(!$table.isSuperColumn($column.javaField)) + /** $column.columnComment */ +#if($column.list) +#set($parentheseIndex=$column.columnComment.indexOf("(")) +#if($parentheseIndex != -1) +#set($comment=$column.columnComment.substring(0, $parentheseIndex)) +#else +#set($comment=$column.columnComment) +#end +#if($parentheseIndex != -1) + @Excel(name = "${comment}", readConverterExp = "$column.readConverterExp()") +#elseif($column.javaType == 'Date') + @JsonFormat(pattern = "yyyy-MM-dd") + @Excel(name = "${comment}", width = 30, dateFormat = "yyyy-MM-dd") +#else + @Excel(name = "${comment}") +#end +#end + private $column.javaType $column.javaField; + +#end +#end +#foreach ($column in $subTable.columns) +#if(!$table.isSuperColumn($column.javaField)) +#if($column.javaField.length() > 2 && $column.javaField.substring(1,2).matches("[A-Z]")) +#set($AttrName=$column.javaField) +#else +#set($AttrName=$column.javaField.substring(0,1).toUpperCase() + ${column.javaField.substring(1)}) +#end + public void set${AttrName}($column.javaType $column.javaField) + { + this.$column.javaField = $column.javaField; + } + + public $column.javaType get${AttrName}() + { + return $column.javaField; + } +#end +#end + + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) +#foreach ($column in $subTable.columns) +#if($column.javaField.length() > 2 && $column.javaField.substring(1,2).matches("[A-Z]")) +#set($AttrName=$column.javaField) +#else +#set($AttrName=$column.javaField.substring(0,1).toUpperCase() + ${column.javaField.substring(1)}) +#end + .append("${column.javaField}", get${AttrName}()) +#end + .toString(); + } +} diff --git a/ruoyi-generator/src/main/resources/vm/js/api.js.vm b/ruoyi-generator/src/main/resources/vm/js/api.js.vm new file mode 100644 index 00000000..9295524a --- /dev/null +++ b/ruoyi-generator/src/main/resources/vm/js/api.js.vm @@ -0,0 +1,44 @@ +import request from '@/utils/request' + +// 查询${functionName}列表 +export function list${BusinessName}(query) { + return request({ + url: '/${moduleName}/${businessName}/list', + method: 'get', + params: query + }) +} + +// 查询${functionName}详细 +export function get${BusinessName}(${pkColumn.javaField}) { + return request({ + url: '/${moduleName}/${businessName}/' + ${pkColumn.javaField}, + method: 'get' + }) +} + +// 新增${functionName} +export function add${BusinessName}(data) { + return request({ + url: '/${moduleName}/${businessName}', + method: 'post', + data: data + }) +} + +// 修改${functionName} +export function update${BusinessName}(data) { + return request({ + url: '/${moduleName}/${businessName}', + method: 'put', + data: data + }) +} + +// 删除${functionName} +export function del${BusinessName}(${pkColumn.javaField}) { + return request({ + url: '/${moduleName}/${businessName}/' + ${pkColumn.javaField}, + method: 'delete' + }) +} diff --git a/ruoyi-generator/src/main/resources/vm/sql/sql.vm b/ruoyi-generator/src/main/resources/vm/sql/sql.vm new file mode 100644 index 00000000..05755835 --- /dev/null +++ b/ruoyi-generator/src/main/resources/vm/sql/sql.vm @@ -0,0 +1,22 @@ +-- 菜单 SQL +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}', '${parentMenuId}', '1', '${businessName}', '${moduleName}/${businessName}/index', 1, 0, 'C', '0', '0', '${permissionPrefix}:list', '#', 'admin', sysdate(), '', null, '${functionName}菜单'); + +-- 按钮父菜单ID +SELECT @parentId := LAST_INSERT_ID(); + +-- 按钮 SQL +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}查询', @parentId, '1', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:query', '#', 'admin', sysdate(), '', null, ''); + +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}新增', @parentId, '2', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:add', '#', 'admin', sysdate(), '', null, ''); + +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}修改', @parentId, '3', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:edit', '#', 'admin', sysdate(), '', null, ''); + +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}删除', @parentId, '4', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:remove', '#', 'admin', sysdate(), '', null, ''); + +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}导出', @parentId, '5', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:export', '#', 'admin', sysdate(), '', null, ''); \ No newline at end of file diff --git a/ruoyi-generator/src/main/resources/vm/vue/index-tree.vue.vm b/ruoyi-generator/src/main/resources/vm/vue/index-tree.vue.vm new file mode 100644 index 00000000..3def7045 --- /dev/null +++ b/ruoyi-generator/src/main/resources/vm/vue/index-tree.vue.vm @@ -0,0 +1,502 @@ + + + diff --git a/ruoyi-generator/src/main/resources/vm/vue/index.vue.vm b/ruoyi-generator/src/main/resources/vm/vue/index.vue.vm new file mode 100644 index 00000000..3f9a4b4a --- /dev/null +++ b/ruoyi-generator/src/main/resources/vm/vue/index.vue.vm @@ -0,0 +1,609 @@ + + + diff --git a/ruoyi-generator/src/main/resources/vm/vue/v3/index-tree.vue.vm b/ruoyi-generator/src/main/resources/vm/vue/v3/index-tree.vue.vm new file mode 100644 index 00000000..862297c7 --- /dev/null +++ b/ruoyi-generator/src/main/resources/vm/vue/v3/index-tree.vue.vm @@ -0,0 +1,486 @@ + + + diff --git a/ruoyi-generator/src/main/resources/vm/vue/v3/index.vue.vm b/ruoyi-generator/src/main/resources/vm/vue/v3/index.vue.vm new file mode 100644 index 00000000..f66cc3b8 --- /dev/null +++ b/ruoyi-generator/src/main/resources/vm/vue/v3/index.vue.vm @@ -0,0 +1,596 @@ + + + diff --git a/ruoyi-generator/src/main/resources/vm/vue/v3/readme.txt b/ruoyi-generator/src/main/resources/vm/vue/v3/readme.txt new file mode 100644 index 00000000..99239bb5 --- /dev/null +++ b/ruoyi-generator/src/main/resources/vm/vue/v3/readme.txt @@ -0,0 +1 @@ +ʹõRuoYi-Vue3ǰˣôҪһ´Ŀ¼ģindex.vue.vmindex-tree.vue.vmļϼvueĿ¼ \ No newline at end of file diff --git a/ruoyi-generator/src/main/resources/vm/xml/mapper.xml.vm b/ruoyi-generator/src/main/resources/vm/xml/mapper.xml.vm new file mode 100644 index 00000000..0ceb3d85 --- /dev/null +++ b/ruoyi-generator/src/main/resources/vm/xml/mapper.xml.vm @@ -0,0 +1,135 @@ + + + + + +#foreach ($column in $columns) + +#end + +#if($table.sub) + + + + + + +#foreach ($column in $subTable.columns) + +#end + +#end + + + select#foreach($column in $columns) $column.columnName#if($foreach.count != $columns.size()),#end#end from ${tableName} + + + + + + + + insert into ${tableName} + +#foreach($column in $columns) +#if($column.columnName != $pkColumn.columnName || !$pkColumn.increment) + $column.columnName, +#end +#end + + +#foreach($column in $columns) +#if($column.columnName != $pkColumn.columnName || !$pkColumn.increment) + #{$column.javaField}, +#end +#end + + + + + update ${tableName} + +#foreach($column in $columns) +#if($column.columnName != $pkColumn.columnName) + $column.columnName = #{$column.javaField}, +#end +#end + + where ${pkColumn.columnName} = #{${pkColumn.javaField}} + + + + delete from ${tableName} where ${pkColumn.columnName} = #{${pkColumn.javaField}} + + + + delete from ${tableName} where ${pkColumn.columnName} in + + #{${pkColumn.javaField}} + + +#if($table.sub) + + + delete from ${subTableName} where ${subTableFkName} in + + #{${subTableFkclassName}} + + + + + delete from ${subTableName} where ${subTableFkName} = #{${subTableFkclassName}} + + + + insert into ${subTableName}(#foreach($column in $subTable.columns) $column.columnName#if($foreach.count != $subTable.columns.size()),#end#end) values + + (#foreach($column in $subTable.columns) #{item.$column.javaField}#if($foreach.count != $subTable.columns.size()),#end#end) + + +#end + \ No newline at end of file diff --git a/ruoyi-generator/target/classes/com/ruoyi/generator/config/GenConfig.class b/ruoyi-generator/target/classes/com/ruoyi/generator/config/GenConfig.class new file mode 100644 index 0000000000000000000000000000000000000000..90bf2a62a2b7739ded62cc22c293e1b6ab7cac29 GIT binary patch literal 1587 zcma)*TTc@~6vxl>>Xy=0tXh{#tdU&pfl_b4L z(ki0_Z>mG-F&Zl!@e^*=xl=Rum1jG(SByqnZXNTQ*x?PqXcWG3Pc-C-*!5tu19&K2 zscW-ppCa43?bzxyqiAui%qX@cKfy0svK_J0Y*d8zL3`_B(&F_p_iSx40_8HBT zEZH!@!gxc)Tj~C~{_zl13GEp4%(2%$GnvSv? zqHOy%U^X2`Dz0qlfMum5y_)H}T8^p*5k5=r*sKWd_-2(`0Bf1N8(@}sy(vHzKhX;i z7S7u8eqodW?QXxC8L(a6D4g}ljgVaXfc=s~SBRc#AFKaItazXglM5>Lesl^43STry z7rwkvI8(hYA|$mho1P`!+PYVnuI7vShA^5B6~$M=6H>KYVQ$HWD;?n|Nbx4rxgw>~ zEmh)FnXV^Y;VE1Ah}2zfqUF$kx4Vx#3Ab~_qPP=`)WbA86CrPV)A|#r!>fHBT zt`1*zZ`We0QBPBbvT3?TMw(K37@>5UM#*4Q_@4_2?RyXCedS14iXPylCGikrWaz|L z89H|`BYiNVLNG&U$`Z1`GBGpE@%f*`zGD%g>lhQkW{$?NPPZNC293iqXo4oOql51R ze^B%XMZUpieX-N9rsyW}Y0nK<5zO(0bBgrJq=L;EP|pUXCfiP(Qg9E>f`^095u~_{ zFYlwSL3hI_qJtC*2)GwY5gnvh0>$zb6k5q4iu>4{hVVmL#J&z_u?nAu^k`6v{5i!2 zDb_);aRr6$;}Asw6pzESh!1M9iQt(~E#iX|+n{)N1%C0qwl--(8%&$fqy>xzlig`DWp`(p znN1oF6}5mjD2fOQih>u)WlAU%@D}g;z7WA%5$^;2e>1Z?vlpcL^VeVAynXZD_x|7i zJKp!E4?lJ9F(TR^PK?qfx;aW4={*f--W#J^=vMyuKJLArn-B1d560+2bX%A{9Hl7T z9HEcI=ytjzLU%^!T%LXx&%8TKAB|F+<;7_>Mn~u#?#)H%UOLJX?&Icuo_~yc4{-A! zHxF_1F>XH2Z$H7kPjd4qZay8O&(LQh^l*ee$KB6I=nE10VuZfLzrGxyuY~EVQA#kd zukrKi+jkgTvf8Bd_YM;@{Ft$7sx$%OP|&ZH3hMY3&?ip>3k+z#J6F7_)1nmT>Ta; zt6AFw1=~CJ3kvPfQ&6^PP|K=&@|khfuvrikAJmgddcR_5%!)f0vL-cXQ=OWmp0x}; zomLGQV|faip6zPN;(en9a+ST!#cvhoB8rHW@;lc~)!MD(a_~eJFFQ=YLKf30sJq=!G_7PO zGK(@bC3Vs{xx339ph#9%Y~C4A6+!jXa*CCAR^Spmo(Wvg$5 z?J_T}9Z&CO#!vUHhr}&DjYcev!EWEk(D*?$Y1!*eszLIFdDWO2v01 zjJIaQBBvrh7fo6wryJI&okEoPp3Zd3aE_W5m*|G&GVPJ$m!wH(mDlH8(zX-1UqBQlm z6sp;L#*}+~(_q*7f!)JHdq>$xtDOneB6b*g!3$YQ_Hhm8P*a*UVpBf+za`gd@mwqI z@TSaP{l4M8p3(l}$v?w+pEA`+K+eQjP@T&syVxM3DXY+t*Q}6?v|`ioDVSGzX%GLA}M`3o7NQ^ z2W~nzWLHqoL!Pv(oV*JSS2P&F5t2i#r212HA#I>gi5TvCK~7?sha^ExGT?lHGoNV@ z6d3EReVQgw4Y4-io&9@u4E4c0i{*k2u3{{V>Biyf)J`p29&)#Af;p2-PWFupYLVS; z=>%z9v67QJ3t|??rN0yNR&%VEE4k>1o;Q-}nHs06xGlC&kFDqPwN?>YJ!VX+q=2hl zzgAyrhnwf64Kmj%%#lRTqwNy?h@0oq<}m$OqMy(s68)4OmFQ>mm_(1$&n4PQ+rsn< ziGE4HLUehx!7jy|wB0(Sp;8IcuO<2o{T7Bb^l4L~-_h?SdV(H-gLswu(;sFM{ehkg z(;p@J6a87DztCU9^f!tAPXCbTpY$(@o}wpN@V_NNLP#ROO;Cg+Q77sp5f%}NuBCTK zqCrF@x(*G-VqqakVgYRzw7kcz?Gi4RiJo*?pH2)J+Ju&c#7;>xiY9nt*|h#cs07u- zE>%gX1|M|1Ud>vGJ`FKqY8;pdO2x92p^5*|YG>t!?&=W&K53u1yjPS$bn( z^`M%auqFj9wJEg=Gbg}$^I94^2_i`}i#RunB+()k3tEdxF|8$?T&yQ8)k>I_p(>f~ zRmn-kFjZ^ozR@!inwO=Zhg+;3*P7uolb$Moh_1#)LS-cmN-Vy|%swmvI{8sZV z=v7RWm2qwv;Kf!%)8&DoV!}eqo0ei&!?-EbfKclWhp;h`Vv`nv$1Y@?B4XowPgT!T zc_GrMgrd3+Nyiq(M~+jkc-M_u^-$Gvtqff63|H9z=UKDj6!U1WyaF|s>I&C{{_P#* z5Cxs)*~ODr!C}cg6LnRobmuM8(mP|b!PwiXV6WhCfj45uelV9uMZC$s)GgqBM7Sfa zGPb9?LrC0a-VVy*ktF-y*Rn{p+Q?oq0SQ9(+jZ!|ikXdz$(t%CUR!-d6yVu&~s z7gw7I1WQYR`6ZmnTR(GggWEJJA{DGPD5;clR8%*hauSkD@dw8*9XGlr?C3Fhj-@$0*fx9wBjuy`&c{3m{rW zJJ3qb1nQ%H{1S8~?ZjrhID=;p&&8dy6zII0g6rle6rkOmU9(hoJ6;5dg*Dof5)`8K z)Px;b=qhKSD-3ZWpcetqh5&629i`}bM<}+ruC*>MbQse4SwZbd-ZVV0{oQv!>e|C`C<2Xdmqd+!!=I z2Ra9^dK;Y!or92Znyq~Zk~-)-dI8nhJTP0%tlh;~W8l-7^+L=ljB&EOx8}NM8N7W%qT4a&C5Xa$sn1{(CG*)nqiG=h6t@7l@4IzMfljyt`{u2 zo-;CrkurMyt~UYXlcc%G597(mSFXEwRpOrk|iArawc6!hDs(`l!quM zbf(aorbC#_%esUxd*=17vQh6Ve*ZS&S&f<1PgjSy}`vg4y5dpamEUF zT*>aZE^&lbc?sXi(N_WN)znOH^0{EV2snX>c; zrdy8(Q)N(i=iA$>V!9qCc6ge`lXG8d7t&54*PP5JT<5yhP>H3TgmVX`-w8|Kg(P{m zjdFt{plI#ZqP1J?)Dyt_7CZF>u`XiUJ@nt%{RX%WzCxholye)LlS63)>E;v^IRPI78w^5ygP?n{oIhycL08k?pS|@OA{=k)YaAO-)G;sivhF`lgjd=Id$>5DGBBW;|xxW2V=<$wA09BRNZ#mT-dW`_7%>J zY-NWBslQH_D!pp=n4^d#(~4y|&+}$pI5>Uo(Tk^E76FypALgPEKdXk1dU_7()5DsI3aiwS6p0!5}6K8k&+=1D9b_C-`QmRfvjWdK#{fa#* zL3Dhq(@1ACx&%vllSU?CW{n9gf$vpQu?;gZVq|cF6Kk<4jkIHolN|L1TY_sYr6H}0 zB#VS=$z7XSTT3T;HOr>8IiRI<#m<_Dx6)agxz}x%62K)}g}xdKmn!CoHbM->xVxUC zwy9!wMlfC_J@zXmE;dKO-Z=v=k2C#xRnJmZRpzVC`?5Ao*X&gi6gM>wNl@ZAsOrUu z?#pI|Rg(=H<(rYjJXSE9aTr}?i9M>}aKuxp%`_CDwyD{dnr`OCEuTeQ6K_C~WzUJj zRUdGZ%W&Cqmoeo}8Z4cc@E(voa~8tSp9@v@P6 z%oXo>N}#PC>()i{a%}&KtvM2qrGhmz%t6 z+}z2TUtVypiyOQRiCA(K8tuW{21R3-uh9~tnt#WlCC2MKA5N-!KAajYcc;C#v(Zux zYA8-#~tnGpp)JwN-`>;7<}P%wwd&j(HsLyj2d? zRK4p!hhnK+cvZCVwzQzYk}|?dICHXzo^g^9LZOG{{)r6Wg<&`2q?QkE?CQF`t8cJp zaI*w;{x0T@w(^elfxOK^!%QU*@$(TnkbtZsuj_8!AjgQ zSc#hkD{cL)gOC4p!{kWwDx zULnXJAhf3v$eluv7!UHi5X2ON#CVVs z2y!w6gf?{ov4kLT9^{M=#1@0Zd61V7WGV!NP7DIc3PH+wkn=*2ZDNpe9^^Fyc|8P# zjw1s3vJgb(LEaRCY!`#bJjmM!@=gc{opl5@*sZ^f_zO3Qptn-6+!+M0z#)Wf!rkosp3H{2tn=^gH-V#e@Bpign-aBfk3_v z_n_oY-)bJ@pF)stz`btc!0B7fgZv9Y{v85B_Y?xT5AGL$)bJqx5rXW1Z;C)_c#!`= zDPBbA7E*W(poUh*j0jdOh#H*-|8#BDJmrC^^h;&t? zG21Hyspml&g&+@$LF##s`3Q1l2ngL(3FHwW$Q&MIfe_?TF~}SqWHEv?g@DjynLr*B zf;8|T*9t)%7lSnLAlD(tvJeoutrN%-LXazXkn4pYPl`dV;6Yjuq%8!5J{Jh&J3^2~ z9%PLWWSh%sq~}B9@)(gGE~BV~M(7 z3^Dq_#BF~wTGNM3UCSF#T|2L35QA|*$mq?lYeiSD@C;B!^yevm$mHlKX!jF9Te52M=bb0wrY6h`$ip6WKAt2`C{C?ur1>cgm#K377j!#S@j z-Q6`tGq_|DuI8~unJ;IzkL5o4VsL~lh8X=)e#vQAa2gg)!pC2Q zYbIgQaa3S!QGR9{aP~mTKtwX2QW}SPEE^@$7x5~0rqQb#k>0avT5_!M~z#3GAzai{Wl6K10Q4srVcf`>D8xihHB*dH4cZ z_eHU|UxEjy=Zn;HfO-y+^&nYaBI^)Y50O=*;xIKQ$oeu_N631ZilbBbn2IMQ_z7M9DHT7H;O7!NMdmN4_$3v;65-b(JS{+-sT!J+8B`5Z%jpA&6#@ur z1*qxC>87RV)}WFps9}_i{saM}R4!A<>WMT??HyCLD)Ee>kHmX(`p8lNL`xaYsB|&v zjO$afssJ%&WYRJ;du)v}q9>Qls=DRP=--jY84|g6%qlWhX=(<&Hg7EGmX=iqHB-Zd zmg{=XQY-{+3NXiBWP5z8lF`z3XFNqD3`Nr|Gv1f8x)JI)q0FkLsf^$5Ejse^l; z-L?PZ#O=@TK6Y~A$aD7|5#X%Sg$PN>Vj<1EY+Qh9%}nSiLvU9t42TF5V`}o6y9Y8?=`mF&X6^#p>XZJsa!U0 z6mmPX_=u{jhGON6csiFw;o_@R-5%Xz-*#NF+Egv4LASYnE(6OB0)kOJ!j6f}Z4C~> zyNzya$9Aybc&}08E#L%CBfEqfJ+w%Tw}>;Yp=22^%rL-Ryk4Z$c{9B(e^Ezt-?`H! z))Xsl3my0^*}dk*sO(;Jn^R~pB~QcM9udbU?C=rsnq%SZyFvgiRqU@xT1sksosxHN zT&+EiiM(3dCDTEC5==A_AGS?=5?+AP{qkE?_BAJBtL!VTZTC9??DRV}-O9vdJV)&b zZg7(_w&}=5$}mk@b&VJCgs8N&!rKBg(r=M|NqbK;v7s@!DJV8bSZ7FMNmNXZK}j_; zS;(t~6UqYAYUY4RdtFxb>wQ zaQ+y^kqlSEH8S+T3K>>Xu?kkpu!f2R6>F(@3#=C5w=(<=PRg)>dN#mLOzs!Ta0;H4 zVIy^Igq<=Z$x4#dPgXx!17r<41vWR4wTUiVPSy|=S5R>!TqVPu7N zB2@JhGLSGoDwf)f$u8=c{g}bAzmUf??esgQ(;sm}!_j%gz$b&VyCtVV*+R-0Y1P2= zhS^iWB}`0LrVP))^D=yZpj5te^cK6kRfZSf_a)d^-DUWLEy*kqo{`~?@Fy8wgqLLa zGyFvWkm0ZJG8KPA)h@wbhYWv*f5`ApctwU+0he$6zY`}up8T1%;+BWM{B9b}56Xq@NO5y)%8ez9TWBBh4P9C%~ezzF2yvIjb`Z!=`;|=vJ+5 zs;UdHmi6RQ9XyVLMsc?ld2lw`O!XeZX&_1Qo;rU z*5g9izGS55Frwu1m`Mca;9)q8rkg_o!`-CBLU9)GhDS20K4Oj9M~8#}a$3!(mP&Y+ zBmWj2IB%|G7(>)PPVAepdx$G>3l8U1JWjYs_GF zjrq&2F@@PRW-+_QM27VeToZFEmT$#Y9BWJj5NYT4$wQMf%Jo3rNj6$|GmN;td?1IQ;s8#nfNRM$ zUIysn0vY84sSE;1B9Q(vK{yE|4)8cUhDu5()yj8i8C>1_%WX0m(qt z^~EX@3IZAC0@3+E!a*P-2xPPj5QHf>88E(5c!owdB=s36b(lfLD&?RmQR)Uh*}5Rvqe%7` zIudzs}!9dNU3*lm5K$GdXgdPE7i|eDprL0pi)mU4t%BX9qi1NdYYlcD&YC<=!8W|?dB@g5LD^~hN!R90AHyFj#4i&4t%8sUXM~QGn81R z97-ih-N;v}F{solsMM>BTUIGLgOXA=ag}NeD)k?RsISx@U#Z3-GzDTxs9+rUN)5ga zr38_o#46=bDpBfYzEaZyN(nNAg&M{!s}!BmNvU^nm6{e%N|?$J^_AMhS87@jrU#X3 zVjTEN;fD^yaAp}@XkjR^N;#BDl)8nlRC7?NS*X-(#x1K9y~mJJw{n$g4k~pXL)2I5 zHtysZ!O3mh$?xW~m=RL>K`TNe-7~HX z+h7m&3DAJAs~?4r;US)~(Af}r4O$^0!pBAUga~^@_$2;c5B?3y@EL~RS>RJRsscX! EKOD6;B>(^b literal 0 HcmV?d00001 diff --git a/ruoyi-generator/target/classes/com/ruoyi/generator/mapper/GenTableColumnMapper.class b/ruoyi-generator/target/classes/com/ruoyi/generator/mapper/GenTableColumnMapper.class new file mode 100644 index 0000000000000000000000000000000000000000..1ac6c99574703cff1792fdf34ef64fb8e571ac8a GIT binary patch literal 798 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 c@vXYa~KKL_rU%vQoRcCAIBjB?UjCvsa!ZA5pO zXv*s6+SJNztx1@S{-HCDS*Ll)rB~e_2$wluVVj<<KJ^@J%n7#l2 literal 0 HcmV?d00001 diff --git a/ruoyi-generator/target/classes/com/ruoyi/generator/service/IGenTableService.class b/ruoyi-generator/target/classes/com/ruoyi/generator/service/IGenTableService.class new file mode 100644 index 0000000000000000000000000000000000000000..24acd779164989b011642ef95e366cc87c37200f GIT binary patch literal 1295 zcmbVMTTc@~7(IhX1xn>sK)fIz$&wiN#l$GY#EKG&UKQRySD8=kgHkkgXUgo}P8?QG zQd(nccox_-+6CEuJ#D+KvR8&i(hqnn@|qc(io5*5uojS_M!sn&H2n2DL(PgIazpm| z+7$95Z>KD(L!!MapH`hr>b%%DVj#uUM;(wVcW*~Nk)wj+i{U37-3^Vk#0F7COdzdAB zpU@2E2-iyZ0p^SQ1-f5Ey}(O^8+ZtiUB-&Ft5~yk9nFHbLHC>3D)1x1+jwkwPw>>* TXSa46I~G4rVJmP@lyRnofUb;E5?%@&C?|0?iF;+tDp-rx zh@bmpyjH`k;&tDflqX$#_&k+u?&J4vro>dB?!u#NqLggTq4u6SohJ zOGsx=>t}V%((P$&)N@UHx?jSo0mm+Ry6uhWR@q46BN9@Eeb#gxyI|O!gs!aPPHUy2 zfS+>pf^o0o&S*KuH8jF$gG;ca*1XrWO>dushK|lL35^3z-jJ{+Yud(0xiD$CC-g~+ zJL#;G)2%VxHN|_l+31}zOA3Juv)PX9iN)-q~N+qzeFDO>TX z^!vZqCo2|~kgdVJ^J-ONYzUh({wfTFD{hw1UXzO-xKWQalwFH{E0+v6ppz`>r4mCk zPhaJAPak#4Zq8tE)2b5U*x>LoY&v0e$zv26)3X(N@}aV6gZgt|w_^OJp;R%*XQDftmBo!KH_8s+A zXbD?`zR@c(vgBuEZ{97N`QCzF@(j0^S*+qooR@Ip^Fc0V`tw0n3IT(Y`B=Z6%k#dH zkis8NsrWj+A!AO$f%+-dPH-9XDlT9_#YJ3_@l6T)>L(e^?}~sjzNO-6d|Sn3d`AfS zT?zZ^Cs{Mw1)1*&&A+eW2l!ztezc|u+gtHt6+giRGknvj;zr*ht>Pr*KvB}d$j!k{@l>?&+hX&`ABB8H-?U#Fz!g z^{=dOQ9-B5cFr>$TRZL=c{9fr!$z=SsSs2tLUpH1i>WAKPsqt{vo(lBc~sFL`AsXx z4>*=Z|2Zz3VzcK21KQ?V7F~oC{VErrNw_&GO~`HuGL*$1Wka*G*yL>ORM~SJD;Q5h zTEY|pQw)tP#B3o)Ojnq;BWl_q+Ca>D{WVFah;`g`iiYdWviwC6L!U&^!mTBfMn~?I z&Re-7X6Gh9FIgODv#Sqh+H`mQ*c3c6kb!$g&f|QPp%Pe3CsfqA!5e9SP>T`LN)frLaq+Deo5j$ zSAQTJ=QOHMheLjU48-L{KMw9Pkz;T@VHAoMog9qSad=kQI0nZdTl7#Cz>uY=E3K6o zTFkUTSx#4#_)2!ODoY?i{(C)1)M z{$X}qMdoP1XKLph+tW>(txzE>2?bXxos=}Z=-{frJ|!e6g_kYbsDI}oYAq23lS77X zOV}PQpZ>wKR?m|0XBB_J-&Kr*1$gVyU}f58of$*R`X?%$DD){^^l6$rRaknH@lQ7H ztmDj-ixQrV@(c+Chml)#JI{*P9zfO@O(G#vg(%8r6ukbA0n01nMnfy4JfK_FsL5$a z#=mIA6$YuL5690d!{G&-^}2XHn83B@Mi1}mZt~xH(IU=DTsL!%Cp!MVm~R@N5}!?7 z^N^nOL2n6xfHZb;tp)(tg_rQH1-r3_13l?kl&Q8H+vgd1?3&tgkJ27Q!R{B1+OFMAU_gg`s_6xyV^D9{D8T!tbTCw#_2abQbCrxXtR3|qwMmQW-i<$0;u zLb!*{Bb8pYfYk|P#d3|m92Uz~e>rjqYsb5LZkk7W4%hTGUc$QZd93eiBJZ|yP%=$j zbJ%d_C2Zt&<~**Q!=^c0*VlXr*N;n?#!S;3HcOdiX&yK9$r*VLTl$ijO_Gr#RwIO?O;C=Vf#}g_KAPDH23&WP3OF5}2TkfUUIFI$Cv@c3z8D`_GMd fga7<1{(-;IYB|ES5hG7rgVvt*S8v1H@s9ri3PV!T literal 0 HcmV?d00001 diff --git a/ruoyi-generator/target/classes/com/ruoyi/generator/util/GenUtils.class b/ruoyi-generator/target/classes/com/ruoyi/generator/util/GenUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..058580d0f0537bee5f42bfc8c311897532a7fb62 GIT binary patch literal 5739 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 OT4wsmW0v$O%>4oKuoF)J literal 0 HcmV?d00001 diff --git a/ruoyi-generator/target/classes/com/ruoyi/generator/util/VelocityUtils.class b/ruoyi-generator/target/classes/com/ruoyi/generator/util/VelocityUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..7e2677a7a2f2eb338b803ac6fc5bcb4daa8d081d GIT binary patch literal 10931 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 l?qVK<)KH+!<8aD9fW%ysS^FckenK=${D65d2 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + select column_id, table_id, column_name, column_comment, column_type, java_type, java_field, is_pk, is_increment, is_required, is_insert, is_edit, is_list, is_query, query_type, html_type, dict_type, sort, create_by, create_time, update_by, update_time from gen_table_column + + + + + + + + insert into gen_table_column ( + table_id, + column_name, + column_comment, + column_type, + java_type, + java_field, + is_pk, + is_increment, + is_required, + is_insert, + is_edit, + is_list, + is_query, + query_type, + html_type, + dict_type, + sort, + create_by, + create_time + )values( + #{tableId}, + #{columnName}, + #{columnComment}, + #{columnType}, + #{javaType}, + #{javaField}, + #{isPk}, + #{isIncrement}, + #{isRequired}, + #{isInsert}, + #{isEdit}, + #{isList}, + #{isQuery}, + #{queryType}, + #{htmlType}, + #{dictType}, + #{sort}, + #{createBy}, + sysdate() + ) + + + + update gen_table_column + + column_comment = #{columnComment}, + java_type = #{javaType}, + java_field = #{javaField}, + is_insert = #{isInsert}, + is_edit = #{isEdit}, + is_list = #{isList}, + is_query = #{isQuery}, + is_required = #{isRequired}, + query_type = #{queryType}, + html_type = #{htmlType}, + dict_type = #{dictType}, + sort = #{sort}, + update_by = #{updateBy}, + update_time = sysdate() + + where column_id = #{columnId} + + + + delete from gen_table_column where table_id in + + #{tableId} + + + + + delete from gen_table_column where column_id in + + #{item.columnId} + + + + diff --git a/ruoyi-generator/target/classes/mapper/generator/GenTableMapper.xml b/ruoyi-generator/target/classes/mapper/generator/GenTableMapper.xml new file mode 100644 index 00000000..b605e906 --- /dev/null +++ b/ruoyi-generator/target/classes/mapper/generator/GenTableMapper.xml @@ -0,0 +1,202 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + select table_id, table_name, table_comment, sub_table_name, sub_table_fk_name, class_name, tpl_category, package_name, module_name, business_name, function_name, function_author, gen_type, gen_path, options, create_by, create_time, update_by, update_time, remark from gen_table + + + + + + + + + + + + + + + + + + insert into gen_table ( + table_name, + table_comment, + class_name, + tpl_category, + package_name, + module_name, + business_name, + function_name, + function_author, + gen_type, + gen_path, + remark, + create_by, + create_time + )values( + #{tableName}, + #{tableComment}, + #{className}, + #{tplCategory}, + #{packageName}, + #{moduleName}, + #{businessName}, + #{functionName}, + #{functionAuthor}, + #{genType}, + #{genPath}, + #{remark}, + #{createBy}, + sysdate() + ) + + + + update gen_table + + table_name = #{tableName}, + table_comment = #{tableComment}, + sub_table_name = #{subTableName}, + sub_table_fk_name = #{subTableFkName}, + class_name = #{className}, + function_author = #{functionAuthor}, + gen_type = #{genType}, + gen_path = #{genPath}, + tpl_category = #{tplCategory}, + package_name = #{packageName}, + module_name = #{moduleName}, + business_name = #{businessName}, + function_name = #{functionName}, + options = #{options}, + update_by = #{updateBy}, + remark = #{remark}, + update_time = sysdate() + + where table_id = #{tableId} + + + + delete from gen_table where table_id in + + #{tableId} + + + + \ No newline at end of file diff --git a/ruoyi-generator/target/classes/mybatisplusvm/java/controller.java.vm b/ruoyi-generator/target/classes/mybatisplusvm/java/controller.java.vm new file mode 100644 index 00000000..d2894379 --- /dev/null +++ b/ruoyi-generator/target/classes/mybatisplusvm/java/controller.java.vm @@ -0,0 +1,129 @@ +package ${packageName}.controller; + +import java.util.List; + +import io.swagger.annotations.*; +import com.baomidou.mybatisplus.core.metadata.IPage; + +import javax.servlet.http.HttpServletResponse; + +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.PutMapping; +import org.springframework.web.bind.annotation.DeleteMapping; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.enums.BusinessType; +import ${packageName}.domain.${ClassName}; +import ${packageName}.service.I${ClassName}Service; +import com.ruoyi.common.utils.poi.ExcelUtil; +#if($table.crud || $table.sub) +import com.ruoyi.common.core.page.TableDataInfo; +#elseif($table.tree) +#end + +/** + * ${functionName}Controller + * + * @author ${author} + * @date ${datetime} + */ +@RestController +@RequestMapping("/${moduleName}/${businessName}") +@Api(tags = {"【${functionName}】Controller"}) +public class ${ClassName}Controller extends BaseController { + @Autowired + private I${ClassName}Service ${className}Service; + +/** + * 分页查询${functionName}列表 + */ +@PreAuthorize("@ss.hasPermi('${permissionPrefix}:list')") +@GetMapping("/list") +@ApiOperation("分页查询${functionName}列表") + #if($table.crud || $table.sub) + public TableDataInfo list(${ClassName} ${className}) { + IPage<${ClassName}> list = ${className}Service.selectList(getPage(), ${className}); + return getDataTable(list); + } + #elseif($table.tree) + public AjaxResult list(${ClassName} ${className}) { + List<${ClassName}> list = ${className}Service.select${ClassName}List(${className}); + return AjaxResult.success(list); + } + #end + + /** + * 查询${functionName}列表 + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:list')") + @GetMapping("/list/all") + @ApiOperation("查询${functionName}列表") + public AjaxResult listAll(${ClassName} ${className}) { + List<${ClassName}> list = ${className}Service.selectListAll(${className}); + return success(list); + } + + /** + * 导出${functionName}列表 + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:export')") + @Log(title = "${functionName}", businessType = BusinessType.EXPORT) + @PostMapping("/export") + @ApiOperation("导出${functionName}列表Excel") + public void export(HttpServletResponse response, ${ClassName} ${className}) { + List<${ClassName}> list = ${className}Service.selectListAll(${className}); + ExcelUtil<${ClassName}> util = new ExcelUtil<${ClassName}>(${ClassName}. class); + util.exportExcel(response, list, "${functionName}数据"); + } + + /** + * 获取${functionName}详细信息 + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:query')") + @GetMapping(value = "/{${pkColumn.javaField}}") + @ApiOperation("获取${functionName}详细信息") + public AjaxResult getInfo(@PathVariable("${pkColumn.javaField}") ${pkColumn.javaType} ${pkColumn.javaField}) { + return success(${className}Service.selectById(${pkColumn.javaField})); + } + + /** + * 新增${functionName} + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:add')") + @Log(title = "${functionName}", businessType = BusinessType.INSERT) + @PostMapping + @ApiOperation("新增${functionName}") + public AjaxResult add(@RequestBody ${ClassName} ${className}) { + return toAjax(${className}Service.insert(${className})); + } + + /** + * 修改${functionName} + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:edit')") + @Log(title = "${functionName}", businessType = BusinessType.UPDATE) + @PutMapping + @ApiOperation("修改${functionName}") + public AjaxResult edit(@RequestBody ${ClassName} ${className}) { + return toAjax(${className}Service.update(${className})); + } + + /** + * 删除${functionName} + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:remove')") + @Log(title = "${functionName}", businessType = BusinessType.DELETE) + @DeleteMapping("/{${pkColumn.javaField}s}") + @ApiOperation("删除${functionName}") + public AjaxResult remove(@PathVariable ${pkColumn.javaType}[] ${pkColumn.javaField}s) { + return toAjax(${className}Service.deleteByIds(${pkColumn.javaField}s)); + } +} diff --git a/ruoyi-generator/target/classes/mybatisplusvm/java/domain.java.vm b/ruoyi-generator/target/classes/mybatisplusvm/java/domain.java.vm new file mode 100644 index 00000000..23249f21 --- /dev/null +++ b/ruoyi-generator/target/classes/mybatisplusvm/java/domain.java.vm @@ -0,0 +1,72 @@ +package ${packageName}.domain; + +#foreach ($import in $importList) +import ${import}; +#end +import com.ruoyi.common.annotation.Excel; +#if($table.crud || $table.sub) +import com.ruoyi.common.core.domain.BaseEntity; +#elseif($table.tree) +import com.ruoyi.common.core.domain.TreeEntity; +#end +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import com.baomidou.mybatisplus.annotation.TableName; +/** + * ${functionName}对象 ${tableName} + * + * @author ${author} + * @date ${datetime} + */ +#if($table.crud || $table.sub) + #set($Entity="BaseEntity") +#elseif($table.tree) + #set($Entity="TreeEntity") +#end + +@ApiModel(value = "${ClassName}", description = "${functionName}对象 ${tableName}") +@Data +@TableName(value = "${tableName}", autoResultMap = true) +public class ${ClassName} extends ${Entity} +{ + private static final long serialVersionUID=1L; + +#foreach ($column in $columns) +#if(!$table.isSuperColumn($column.javaField)) + /** $column.columnComment */ +#if($column.list) +#set($parentheseIndex=$column.columnComment.indexOf("(")) +#if($parentheseIndex != -1) +#set($comment=$column.columnComment.substring(0, $parentheseIndex)) +#else +#set($comment=$column.columnComment) +#end +#if($parentheseIndex != -1) + @Excel(name = "${comment}", readConverterExp = "$column.readConverterExp()") +#elseif($column.javaType == 'Date') + @JsonFormat(pattern = "yyyy-MM-dd") + @Excel(name = "${comment}", width = 30, dateFormat = "yyyy-MM-dd") +#else + @Excel(name = "${comment}") +#end +#end + @ApiModelProperty(value = "$column.columnComment") + private $column.javaType $column.javaField; + +#end +#end +#if($table.sub) +/** $table.subTable.functionName信息 */ + private List<${subClassName}> ${subclassName}List; + +#end +#foreach ($column in $columns) + #if(!$table.isSuperColumn($column.javaField)) + #if($column.javaField.length() > 2 && $column.javaField.substring(1,2).matches("[A-Z]")) + #set($AttrName=$column.javaField) + #else + #set($AttrName=$column.javaField.substring(0,1).toUpperCase() + ${column.javaField.substring(1)}) + #end + #end#end +} diff --git a/ruoyi-generator/target/classes/mybatisplusvm/java/mapper.java.vm b/ruoyi-generator/target/classes/mybatisplusvm/java/mapper.java.vm new file mode 100644 index 00000000..a46b08e2 --- /dev/null +++ b/ruoyi-generator/target/classes/mybatisplusvm/java/mapper.java.vm @@ -0,0 +1,36 @@ +package ${packageName}.mapper; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import com.baomidou.mybatisplus.core.metadata.IPage; +import java.util.List; +import org.apache.ibatis.annotations.Param; +import ${packageName}.domain.${ClassName}; +#if($table.sub) +import ${packageName}.domain.${subClassName}; +#end + +/** + * ${functionName}Mapper接口 + * + * @author ${author} + * @date ${datetime} + */ +public interface ${ClassName}Mapper extends BaseMapper<${ClassName}> +{ + /** + * 分页查询${functionName}列表 + * + * param page 分页信息 + * @param ${className} ${functionName}信息 + * @return ${functionName}集合 + */ + public IPage<${ClassName}> select${ClassName}Page(IPage<${ClassName}> page,@Param("entity") ${ClassName} ${className}); + + /** + * 查询所有${functionName}列表 + * + * @param ${className} ${functionName}信息 + * @return ${functionName}集合 + */ + public List<${ClassName}> select${ClassName}List(${ClassName} ${className}); +} diff --git a/ruoyi-generator/target/classes/mybatisplusvm/java/service.java.vm b/ruoyi-generator/target/classes/mybatisplusvm/java/service.java.vm new file mode 100644 index 00000000..5cda17db --- /dev/null +++ b/ruoyi-generator/target/classes/mybatisplusvm/java/service.java.vm @@ -0,0 +1,71 @@ +package ${packageName}.service; + +import java.util.List; +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.baomidou.mybatisplus.extension.service.IService; +import ${packageName}.domain.${ClassName}; + +/** + * ${functionName}Service接口 + * + * @author ${author} + * @date ${datetime} + */ +public interface I${ClassName}Service extends IService<${ClassName}> +{ + /** + * 查询${functionName} + * + * @param ${pkColumn.javaField} ${functionName}ID + * @return ${functionName} + */ + public ${ClassName} selectById(${pkColumn.javaType} ${pkColumn.javaField}); + + /** + * 分页查询${functionName}列表 + * + * @param ${className} ${functionName} + * @return ${functionName}集合 + */ + public IPage<${ClassName}> selectList(IPage<${ClassName}> page, ${ClassName} ${className}); + + /** + * 查询所有${functionName}列表 + * + * @param ${className} ${functionName} + * @return ${functionName}集合 + */ + public List<${ClassName}> selectListAll(${ClassName} ${className}); + + /** + * 新增${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ + public int insert(${ClassName} ${className}); + + /** + * 修改${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ + public int update(${ClassName} ${className}); + + /** + * 批量删除${functionName} + * + * @param ${pkColumn.javaField}s 需要删除的${functionName}ID + * @return 结果 + */ + public int deleteByIds(${pkColumn.javaType}[] ${pkColumn.javaField}s); + + /** + * 删除${functionName}信息 + * + * @param ${pkColumn.javaField} ${functionName}ID + * @return 结果 + */ + public int deleteById(${pkColumn.javaType} ${pkColumn.javaField}); +} diff --git a/ruoyi-generator/target/classes/mybatisplusvm/java/serviceImpl.java.vm b/ruoyi-generator/target/classes/mybatisplusvm/java/serviceImpl.java.vm new file mode 100644 index 00000000..77411bd9 --- /dev/null +++ b/ruoyi-generator/target/classes/mybatisplusvm/java/serviceImpl.java.vm @@ -0,0 +1,174 @@ +package ${packageName}.service.impl; + +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import java.util.Arrays; +import java.util.List; +#foreach ($column in $columns) +#if($column.javaField == 'createTime' || $column.javaField == 'updateTime') +import com.ruoyi.common.utils.DateUtils; +#break +#end +#end +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +#if($table.sub) +import java.util.ArrayList; +import com.ruoyi.common.utils.StringUtils; +import org.springframework.transaction.annotation.Transactional; +import ${packageName}.domain.${subClassName}; +#end +import ${packageName}.mapper.${ClassName}Mapper; +import ${packageName}.domain.${ClassName}; +import ${packageName}.service.I${ClassName}Service; + +/** + * ${functionName}Service业务层处理 + * + * @author ${author} + * @date ${datetime} + */ +@Service +public class ${ClassName}ServiceImpl extends ServiceImpl<${ClassName}Mapper, ${ClassName}> implements I${ClassName}Service +{ + @Autowired + private ${ClassName}Mapper ${className}Mapper; + + /** + * 查询${functionName} + * + * @param ${pkColumn.javaField} ${functionName}ID + * @return ${functionName} + */ + @Override + public ${ClassName} selectById(${pkColumn.javaType} ${pkColumn.javaField}) { + return ${className}Mapper.selectById(${pkColumn.javaField}); + } + + /** + * 分页查询${functionName}列表 + * + * @param ${className} ${functionName} + * @return ${functionName} + */ + @Override + public IPage<${ClassName}> selectList(IPage<${ClassName}> page, ${ClassName} ${className}) { + return ${className}Mapper.select${ClassName}Page(page, ${className}); + } + + /** + * 查询所有${functionName}列表 + * + * @param ${className} ${functionName} + * @return ${functionName} + */ + @Override + public List<${ClassName}> selectListAll(${ClassName} ${className}) { + return ${className}Mapper.select${ClassName}List(${className}); + } + + /** + * 新增${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ +#if($table.sub) + @Transactional +#end + @Override + public int insert(${ClassName} ${className}) { +#foreach ($column in $columns) + #if($column.javaField == 'createTime') + ${className}.setCreateTime(DateUtils.getNowDate()); + #end +#end +#if($table.sub) + int rows = ${className}Mapper.insert(${className}); + insert${subClassName}(${className}); + return rows; +#else + return ${className}Mapper.insert(${className}); +#end + } + + /** + * 修改${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ +#if($table.sub) + @Transactional +#end + @Override + public int update(${ClassName} ${className}) { +#foreach ($column in $columns) + #if($column.javaField == 'updateTime') + ${className}.setUpdateTime(DateUtils.getNowDate()); + #end +#end +#if($table.sub) + ${className}Mapper.delete${subClassName}By${subTableFkClassName}(${className}.get${pkColumn.capJavaField}()); + insert${subClassName}(${className}); +#end + return ${className}Mapper.updateById(${className}); + } + + /** + * 批量删除${functionName} + * + * @param ${pkColumn.javaField}s 需要删除的${functionName}ID + * @return 结果 + */ +#if($table.sub) + @Transactional +#end + @Override + public int deleteByIds(${pkColumn.javaType}[] ${pkColumn.javaField}s) { +#if($table.sub) + ${className}Mapper.delete${subClassName}By${subTableFkClassName}s(${pkColumn.javaField}s); +#end + return ${className}Mapper.deleteBatchIds(Arrays.asList(${pkColumn.javaField}s)); + } + + /** + * 删除${functionName}信息 + * + * @param ${pkColumn.javaField} ${functionName}ID + * @return 结果 + */ +#if($table.sub) + @Transactional +#end + @Override + public int deleteById(${pkColumn.javaType} ${pkColumn.javaField}) { +#if($table.sub) + ${className}Mapper.delete${subClassName}By${subTableFkClassName}(${pkColumn.javaField}); +#end + return ${className}Mapper.deleteById(${pkColumn.javaField}); + } +#if($table.sub) + + /** + * 新增${subTable.functionName}信息 + * + * @param ${className} ${functionName}对象 + */ + public void insert${subClassName}(${ClassName} ${className}) { + List<${subClassName}> ${subclassName}List = ${className}.get${subClassName}List(); + Long ${pkColumn.javaField} = ${className}.get${pkColumn.capJavaField}(); + if (StringUtils.isNotNull(${subclassName}List)) { + List<${subClassName}> list = new ArrayList<${subClassName}>(); + for (${subClassName} ${subclassName} :${subclassName}List) + { + ${subclassName}.set${subTableFkClassName}(${pkColumn.javaField}); + list.add(${subclassName}); + } + if (list.size() > 0) { + ${className}Mapper.batch${subClassName}(list); + } + } + } +#end +} diff --git a/ruoyi-generator/target/classes/mybatisplusvm/java/sub-domain.java.vm b/ruoyi-generator/target/classes/mybatisplusvm/java/sub-domain.java.vm new file mode 100644 index 00000000..ad3890a0 --- /dev/null +++ b/ruoyi-generator/target/classes/mybatisplusvm/java/sub-domain.java.vm @@ -0,0 +1,76 @@ +package ${packageName}.domain; + + #foreach ($import in $subImportList) + import ${import}; + #end +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import com.ruoyi.common.annotation.Excel; +import com.ruoyi.common.core.domain.BaseEntity; + +/** + * ${subTable.functionName}对象 ${subTableName} + * + * @author ${author} + * @date ${datetime} + */ +public class ${subClassName} extends BaseEntity + { +private static final long serialVersionUID=1L; + +#foreach ($column in $subTable.columns) + #if(!$table.isSuperColumn($column.javaField)) + /** $column.columnComment */ + #if($column.list) + #set($parentheseIndex=$column.columnComment.indexOf("(")) + #if($parentheseIndex != -1) + #set($comment=$column.columnComment.substring(0, $parentheseIndex)) + #else + #set($comment=$column.columnComment) + #end + #if($parentheseIndex != -1) + @Excel(name = "${comment}", readConverterExp = "$column.readConverterExp()") + #elseif($column.javaType == 'Date') + @JsonFormat(pattern = "yyyy-MM-dd") + @Excel(name = "${comment}", width = 30, dateFormat = "yyyy-MM-dd") + #else + @Excel(name = "${comment}") + #end + #end + private $column.javaType $column.javaField; + + #end +#end +#foreach ($column in $subTable.columns) + #if(!$table.isSuperColumn($column.javaField)) + #if($column.javaField.length() > 2 && $column.javaField.substring(1,2).matches("[A-Z]")) + #set($AttrName=$column.javaField) + #else + #set($AttrName=$column.javaField.substring(0,1).toUpperCase() + ${column.javaField.substring(1)}) + #end + public void set${AttrName}($column.javaType $column.javaField) + { + this.$column.javaField = $column.javaField; + } + + public $column.javaType get${AttrName}() + { + return $column.javaField; + } + #end +#end + +@Override +public String toString(){ + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) + #foreach ($column in $subTable.columns) + #if($column.javaField.length() > 2 && $column.javaField.substring(1,2).matches("[A-Z]")) + #set($AttrName=$column.javaField) + #else + #set($AttrName=$column.javaField.substring(0,1).toUpperCase() + ${column.javaField.substring(1)}) + #end + .append("${column.javaField}",get${AttrName}()) + #end + .toString(); + } + } diff --git a/ruoyi-generator/target/classes/mybatisplusvm/js/api.js.vm b/ruoyi-generator/target/classes/mybatisplusvm/js/api.js.vm new file mode 100644 index 00000000..9295524a --- /dev/null +++ b/ruoyi-generator/target/classes/mybatisplusvm/js/api.js.vm @@ -0,0 +1,44 @@ +import request from '@/utils/request' + +// 查询${functionName}列表 +export function list${BusinessName}(query) { + return request({ + url: '/${moduleName}/${businessName}/list', + method: 'get', + params: query + }) +} + +// 查询${functionName}详细 +export function get${BusinessName}(${pkColumn.javaField}) { + return request({ + url: '/${moduleName}/${businessName}/' + ${pkColumn.javaField}, + method: 'get' + }) +} + +// 新增${functionName} +export function add${BusinessName}(data) { + return request({ + url: '/${moduleName}/${businessName}', + method: 'post', + data: data + }) +} + +// 修改${functionName} +export function update${BusinessName}(data) { + return request({ + url: '/${moduleName}/${businessName}', + method: 'put', + data: data + }) +} + +// 删除${functionName} +export function del${BusinessName}(${pkColumn.javaField}) { + return request({ + url: '/${moduleName}/${businessName}/' + ${pkColumn.javaField}, + method: 'delete' + }) +} diff --git a/ruoyi-generator/target/classes/mybatisplusvm/sql/sql.vm b/ruoyi-generator/target/classes/mybatisplusvm/sql/sql.vm new file mode 100644 index 00000000..05755835 --- /dev/null +++ b/ruoyi-generator/target/classes/mybatisplusvm/sql/sql.vm @@ -0,0 +1,22 @@ +-- 菜单 SQL +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}', '${parentMenuId}', '1', '${businessName}', '${moduleName}/${businessName}/index', 1, 0, 'C', '0', '0', '${permissionPrefix}:list', '#', 'admin', sysdate(), '', null, '${functionName}菜单'); + +-- 按钮父菜单ID +SELECT @parentId := LAST_INSERT_ID(); + +-- 按钮 SQL +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}查询', @parentId, '1', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:query', '#', 'admin', sysdate(), '', null, ''); + +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}新增', @parentId, '2', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:add', '#', 'admin', sysdate(), '', null, ''); + +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}修改', @parentId, '3', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:edit', '#', 'admin', sysdate(), '', null, ''); + +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}删除', @parentId, '4', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:remove', '#', 'admin', sysdate(), '', null, ''); + +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}导出', @parentId, '5', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:export', '#', 'admin', sysdate(), '', null, ''); \ No newline at end of file diff --git a/ruoyi-generator/target/classes/mybatisplusvm/vue/index-tree.vue.vm b/ruoyi-generator/target/classes/mybatisplusvm/vue/index-tree.vue.vm new file mode 100644 index 00000000..3def7045 --- /dev/null +++ b/ruoyi-generator/target/classes/mybatisplusvm/vue/index-tree.vue.vm @@ -0,0 +1,502 @@ + + + diff --git a/ruoyi-generator/target/classes/mybatisplusvm/vue/index.vue.vm b/ruoyi-generator/target/classes/mybatisplusvm/vue/index.vue.vm new file mode 100644 index 00000000..e9a1fae1 --- /dev/null +++ b/ruoyi-generator/target/classes/mybatisplusvm/vue/index.vue.vm @@ -0,0 +1,598 @@ + + + diff --git a/ruoyi-generator/target/classes/mybatisplusvm/vue/v3/index-tree.vue.vm b/ruoyi-generator/target/classes/mybatisplusvm/vue/v3/index-tree.vue.vm new file mode 100644 index 00000000..862297c7 --- /dev/null +++ b/ruoyi-generator/target/classes/mybatisplusvm/vue/v3/index-tree.vue.vm @@ -0,0 +1,486 @@ + + + diff --git a/ruoyi-generator/target/classes/mybatisplusvm/vue/v3/index.vue.vm b/ruoyi-generator/target/classes/mybatisplusvm/vue/v3/index.vue.vm new file mode 100644 index 00000000..f66cc3b8 --- /dev/null +++ b/ruoyi-generator/target/classes/mybatisplusvm/vue/v3/index.vue.vm @@ -0,0 +1,596 @@ + + + diff --git a/ruoyi-generator/target/classes/mybatisplusvm/vue/v3/readme.txt b/ruoyi-generator/target/classes/mybatisplusvm/vue/v3/readme.txt new file mode 100644 index 00000000..99239bb5 --- /dev/null +++ b/ruoyi-generator/target/classes/mybatisplusvm/vue/v3/readme.txt @@ -0,0 +1 @@ +ʹõRuoYi-Vue3ǰˣôҪһ´Ŀ¼ģindex.vue.vmindex-tree.vue.vmļϼvueĿ¼ \ No newline at end of file diff --git a/ruoyi-generator/target/classes/mybatisplusvm/xml/mapper.xml.vm b/ruoyi-generator/target/classes/mybatisplusvm/xml/mapper.xml.vm new file mode 100644 index 00000000..4fc7d573 --- /dev/null +++ b/ruoyi-generator/target/classes/mybatisplusvm/xml/mapper.xml.vm @@ -0,0 +1,109 @@ + + + + + +#foreach ($column in $columns) + +#end + +#if($table.sub) + + + + + + +#foreach ($column in $subTable.columns) + +#end + +#end + + + select#foreach($column in $columns) $column.columnName#if($foreach.count != $columns.size()),#end#end from ${tableName} + + + + + + diff --git a/ruoyi-generator/target/classes/vm/java/controller.java.vm b/ruoyi-generator/target/classes/vm/java/controller.java.vm new file mode 100644 index 00000000..5955c6f6 --- /dev/null +++ b/ruoyi-generator/target/classes/vm/java/controller.java.vm @@ -0,0 +1,108 @@ +package ${packageName}.controller; + +import java.util.List; +import javax.servlet.http.HttpServletResponse; + +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.PutMapping; +import org.springframework.web.bind.annotation.DeleteMapping; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.enums.BusinessType; +import ${packageName}.domain.${ClassName}; +import ${packageName}.service.I${ClassName}Service; +import com.ruoyi.common.utils.poi.ExcelUtil; +#if($table.crud || $table.sub) +import com.ruoyi.common.core.page.TableDataInfo; +#elseif($table.tree) +#end + +/** + * ${functionName}Controller + * + * @author ${author} + * @date ${datetime} + */ +@RestController +@RequestMapping("/${moduleName}/${businessName}") +public class ${ClassName}Controller extends BaseController { + @Autowired + private I${ClassName}Service ${className}Service; + +/** + * 查询${functionName}列表 + */ +@PreAuthorize("@ss.hasPermi('${permissionPrefix}:list')") +@GetMapping("/list") + #if($table.crud || $table.sub) + public TableDataInfo list(${ClassName} ${className}) { + startPage(); + List<${ClassName}> list = ${className}Service.select${ClassName}List(${className}); + return getDataTable(list); + } + #elseif($table.tree) + public AjaxResult list(${ClassName} ${className}) { + List<${ClassName}> list = ${className}Service.select${ClassName}List(${className}); + return success(list); + } + #end + + /** + * 导出${functionName}列表 + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:export')") + @Log(title = "${functionName}", businessType = BusinessType.EXPORT) + @PostMapping("/export") + public void export(HttpServletResponse response, ${ClassName} ${className}) { + List<${ClassName}> list = ${className}Service.select${ClassName}List(${className}); + ExcelUtil<${ClassName}> util = new ExcelUtil<${ClassName}>(${ClassName}. class); + util.exportExcel(response, list, "${functionName}数据"); + } + + /** + * 获取${functionName}详细信息 + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:query')") + @GetMapping(value = "/{${pkColumn.javaField}}") + public AjaxResult getInfo(@PathVariable("${pkColumn.javaField}") ${pkColumn.javaType} ${pkColumn.javaField}) { + return success(${className}Service.select${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaField})); + } + + /** + * 新增${functionName} + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:add')") + @Log(title = "${functionName}", businessType = BusinessType.INSERT) + @PostMapping + public AjaxResult add(@RequestBody ${ClassName} ${className}) { + return toAjax(${className}Service.insert${ClassName}(${className})); + } + + /** + * 修改${functionName} + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:edit')") + @Log(title = "${functionName}", businessType = BusinessType.UPDATE) + @PutMapping + public AjaxResult edit(@RequestBody ${ClassName} ${className}) { + return toAjax(${className}Service.update${ClassName}(${className})); + } + + /** + * 删除${functionName} + */ + @PreAuthorize("@ss.hasPermi('${permissionPrefix}:remove')") + @Log(title = "${functionName}", businessType = BusinessType.DELETE) + @DeleteMapping("/{${pkColumn.javaField}s}") + public AjaxResult remove(@PathVariable ${pkColumn.javaType}[] ${pkColumn.javaField}s) { + return toAjax(${className}Service.delete${ClassName}By${pkColumn.capJavaField}s(${pkColumn.javaField}s)); + } +} diff --git a/ruoyi-generator/target/classes/vm/java/domain.java.vm b/ruoyi-generator/target/classes/vm/java/domain.java.vm new file mode 100644 index 00000000..bd51c177 --- /dev/null +++ b/ruoyi-generator/target/classes/vm/java/domain.java.vm @@ -0,0 +1,105 @@ +package ${packageName}.domain; + +#foreach ($import in $importList) +import ${import}; +#end +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import com.ruoyi.common.annotation.Excel; +#if($table.crud || $table.sub) +import com.ruoyi.common.core.domain.BaseEntity; +#elseif($table.tree) +import com.ruoyi.common.core.domain.TreeEntity; +#end + +/** + * ${functionName}对象 ${tableName} + * + * @author ${author} + * @date ${datetime} + */ +#if($table.crud || $table.sub) +#set($Entity="BaseEntity") +#elseif($table.tree) +#set($Entity="TreeEntity") +#end +public class ${ClassName} extends ${Entity} +{ + private static final long serialVersionUID = 1L; + +#foreach ($column in $columns) +#if(!$table.isSuperColumn($column.javaField)) + /** $column.columnComment */ +#if($column.list) +#set($parentheseIndex=$column.columnComment.indexOf("(")) +#if($parentheseIndex != -1) +#set($comment=$column.columnComment.substring(0, $parentheseIndex)) +#else +#set($comment=$column.columnComment) +#end +#if($parentheseIndex != -1) + @Excel(name = "${comment}", readConverterExp = "$column.readConverterExp()") +#elseif($column.javaType == 'Date') + @JsonFormat(pattern = "yyyy-MM-dd") + @Excel(name = "${comment}", width = 30, dateFormat = "yyyy-MM-dd") +#else + @Excel(name = "${comment}") +#end +#end + private $column.javaType $column.javaField; + +#end +#end +#if($table.sub) + /** $table.subTable.functionName信息 */ + private List<${subClassName}> ${subclassName}List; + +#end +#foreach ($column in $columns) +#if(!$table.isSuperColumn($column.javaField)) +#if($column.javaField.length() > 2 && $column.javaField.substring(1,2).matches("[A-Z]")) +#set($AttrName=$column.javaField) +#else +#set($AttrName=$column.javaField.substring(0,1).toUpperCase() + ${column.javaField.substring(1)}) +#end + public void set${AttrName}($column.javaType $column.javaField) + { + this.$column.javaField = $column.javaField; + } + + public $column.javaType get${AttrName}() + { + return $column.javaField; + } +#end +#end + +#if($table.sub) + public List<${subClassName}> get${subClassName}List() + { + return ${subclassName}List; + } + + public void set${subClassName}List(List<${subClassName}> ${subclassName}List) + { + this.${subclassName}List = ${subclassName}List; + } + +#end + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) +#foreach ($column in $columns) +#if($column.javaField.length() > 2 && $column.javaField.substring(1,2).matches("[A-Z]")) +#set($AttrName=$column.javaField) +#else +#set($AttrName=$column.javaField.substring(0,1).toUpperCase() + ${column.javaField.substring(1)}) +#end + .append("${column.javaField}", get${AttrName}()) +#end +#if($table.sub) + .append("${subclassName}List", get${subClassName}List()) +#end + .toString(); + } +} diff --git a/ruoyi-generator/target/classes/vm/java/mapper.java.vm b/ruoyi-generator/target/classes/vm/java/mapper.java.vm new file mode 100644 index 00000000..7e7d7c26 --- /dev/null +++ b/ruoyi-generator/target/classes/vm/java/mapper.java.vm @@ -0,0 +1,91 @@ +package ${packageName}.mapper; + +import java.util.List; +import ${packageName}.domain.${ClassName}; +#if($table.sub) +import ${packageName}.domain.${subClassName}; +#end + +/** + * ${functionName}Mapper接口 + * + * @author ${author} + * @date ${datetime} + */ +public interface ${ClassName}Mapper +{ + /** + * 查询${functionName} + * + * @param ${pkColumn.javaField} ${functionName}主键 + * @return ${functionName} + */ + public ${ClassName} select${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaType} ${pkColumn.javaField}); + + /** + * 查询${functionName}列表 + * + * @param ${className} ${functionName} + * @return ${functionName}集合 + */ + public List<${ClassName}> select${ClassName}List(${ClassName} ${className}); + + /** + * 新增${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ + public int insert${ClassName}(${ClassName} ${className}); + + /** + * 修改${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ + public int update${ClassName}(${ClassName} ${className}); + + /** + * 删除${functionName} + * + * @param ${pkColumn.javaField} ${functionName}主键 + * @return 结果 + */ + public int delete${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaType} ${pkColumn.javaField}); + + /** + * 批量删除${functionName} + * + * @param ${pkColumn.javaField}s 需要删除的数据主键集合 + * @return 结果 + */ + public int delete${ClassName}By${pkColumn.capJavaField}s(${pkColumn.javaType}[] ${pkColumn.javaField}s); +#if($table.sub) + + /** + * 批量删除${subTable.functionName} + * + * @param ${pkColumn.javaField}s 需要删除的数据主键集合 + * @return 结果 + */ + public int delete${subClassName}By${subTableFkClassName}s(${pkColumn.javaType}[] ${pkColumn.javaField}s); + + /** + * 批量新增${subTable.functionName} + * + * @param ${subclassName}List ${subTable.functionName}列表 + * @return 结果 + */ + public int batch${subClassName}(List<${subClassName}> ${subclassName}List); + + + /** + * 通过${functionName}主键删除${subTable.functionName}信息 + * + * @param ${pkColumn.javaField} ${functionName}ID + * @return 结果 + */ + public int delete${subClassName}By${subTableFkClassName}(${pkColumn.javaType} ${pkColumn.javaField}); +#end +} diff --git a/ruoyi-generator/target/classes/vm/java/service.java.vm b/ruoyi-generator/target/classes/vm/java/service.java.vm new file mode 100644 index 00000000..264882b2 --- /dev/null +++ b/ruoyi-generator/target/classes/vm/java/service.java.vm @@ -0,0 +1,61 @@ +package ${packageName}.service; + +import java.util.List; +import ${packageName}.domain.${ClassName}; + +/** + * ${functionName}Service接口 + * + * @author ${author} + * @date ${datetime} + */ +public interface I${ClassName}Service +{ + /** + * 查询${functionName} + * + * @param ${pkColumn.javaField} ${functionName}主键 + * @return ${functionName} + */ + public ${ClassName} select${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaType} ${pkColumn.javaField}); + + /** + * 查询${functionName}列表 + * + * @param ${className} ${functionName} + * @return ${functionName}集合 + */ + public List<${ClassName}> select${ClassName}List(${ClassName} ${className}); + + /** + * 新增${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ + public int insert${ClassName}(${ClassName} ${className}); + + /** + * 修改${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ + public int update${ClassName}(${ClassName} ${className}); + + /** + * 批量删除${functionName} + * + * @param ${pkColumn.javaField}s 需要删除的${functionName}主键集合 + * @return 结果 + */ + public int delete${ClassName}By${pkColumn.capJavaField}s(${pkColumn.javaType}[] ${pkColumn.javaField}s); + + /** + * 删除${functionName}信息 + * + * @param ${pkColumn.javaField} ${functionName}主键 + * @return 结果 + */ + public int delete${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaType} ${pkColumn.javaField}); +} diff --git a/ruoyi-generator/target/classes/vm/java/serviceImpl.java.vm b/ruoyi-generator/target/classes/vm/java/serviceImpl.java.vm new file mode 100644 index 00000000..14746e1a --- /dev/null +++ b/ruoyi-generator/target/classes/vm/java/serviceImpl.java.vm @@ -0,0 +1,169 @@ +package ${packageName}.service.impl; + +import java.util.List; +#foreach ($column in $columns) +#if($column.javaField == 'createTime' || $column.javaField == 'updateTime') +import com.ruoyi.common.utils.DateUtils; +#break +#end +#end +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +#if($table.sub) +import java.util.ArrayList; +import com.ruoyi.common.utils.StringUtils; +import org.springframework.transaction.annotation.Transactional; +import ${packageName}.domain.${subClassName}; +#end +import ${packageName}.mapper.${ClassName}Mapper; +import ${packageName}.domain.${ClassName}; +import ${packageName}.service.I${ClassName}Service; + +/** + * ${functionName}Service业务层处理 + * + * @author ${author} + * @date ${datetime} + */ +@Service +public class ${ClassName}ServiceImpl implements I${ClassName}Service +{ + @Autowired + private ${ClassName}Mapper ${className}Mapper; + + /** + * 查询${functionName} + * + * @param ${pkColumn.javaField} ${functionName}主键 + * @return ${functionName} + */ + @Override + public ${ClassName} select${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaType} ${pkColumn.javaField}) + { + return ${className}Mapper.select${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaField}); + } + + /** + * 查询${functionName}列表 + * + * @param ${className} ${functionName} + * @return ${functionName} + */ + @Override + public List<${ClassName}> select${ClassName}List(${ClassName} ${className}) + { + return ${className}Mapper.select${ClassName}List(${className}); + } + + /** + * 新增${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ +#if($table.sub) + @Transactional +#end + @Override + public int insert${ClassName}(${ClassName} ${className}) + { +#foreach ($column in $columns) +#if($column.javaField == 'createTime') + ${className}.setCreateTime(DateUtils.getNowDate()); +#end +#end +#if($table.sub) + int rows = ${className}Mapper.insert${ClassName}(${className}); + insert${subClassName}(${className}); + return rows; +#else + return ${className}Mapper.insert${ClassName}(${className}); +#end + } + + /** + * 修改${functionName} + * + * @param ${className} ${functionName} + * @return 结果 + */ +#if($table.sub) + @Transactional +#end + @Override + public int update${ClassName}(${ClassName} ${className}) + { +#foreach ($column in $columns) +#if($column.javaField == 'updateTime') + ${className}.setUpdateTime(DateUtils.getNowDate()); +#end +#end +#if($table.sub) + ${className}Mapper.delete${subClassName}By${subTableFkClassName}(${className}.get${pkColumn.capJavaField}()); + insert${subClassName}(${className}); +#end + return ${className}Mapper.update${ClassName}(${className}); + } + + /** + * 批量删除${functionName} + * + * @param ${pkColumn.javaField}s 需要删除的${functionName}主键 + * @return 结果 + */ +#if($table.sub) + @Transactional +#end + @Override + public int delete${ClassName}By${pkColumn.capJavaField}s(${pkColumn.javaType}[] ${pkColumn.javaField}s) + { +#if($table.sub) + ${className}Mapper.delete${subClassName}By${subTableFkClassName}s(${pkColumn.javaField}s); +#end + return ${className}Mapper.delete${ClassName}By${pkColumn.capJavaField}s(${pkColumn.javaField}s); + } + + /** + * 删除${functionName}信息 + * + * @param ${pkColumn.javaField} ${functionName}主键 + * @return 结果 + */ +#if($table.sub) + @Transactional +#end + @Override + public int delete${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaType} ${pkColumn.javaField}) + { +#if($table.sub) + ${className}Mapper.delete${subClassName}By${subTableFkClassName}(${pkColumn.javaField}); +#end + return ${className}Mapper.delete${ClassName}By${pkColumn.capJavaField}(${pkColumn.javaField}); + } +#if($table.sub) + + /** + * 新增${subTable.functionName}信息 + * + * @param ${className} ${functionName}对象 + */ + public void insert${subClassName}(${ClassName} ${className}) + { + List<${subClassName}> ${subclassName}List = ${className}.get${subClassName}List(); + ${pkColumn.javaType} ${pkColumn.javaField} = ${className}.get${pkColumn.capJavaField}(); + if (StringUtils.isNotNull(${subclassName}List)) + { + List<${subClassName}> list = new ArrayList<${subClassName}>(); + for (${subClassName} ${subclassName} : ${subclassName}List) + { + ${subclassName}.set${subTableFkClassName}(${pkColumn.javaField}); + list.add(${subclassName}); + } + if (list.size() > 0) + { + ${className}Mapper.batch${subClassName}(list); + } + } + } +#end +} diff --git a/ruoyi-generator/target/classes/vm/java/sub-domain.java.vm b/ruoyi-generator/target/classes/vm/java/sub-domain.java.vm new file mode 100644 index 00000000..a3f53eba --- /dev/null +++ b/ruoyi-generator/target/classes/vm/java/sub-domain.java.vm @@ -0,0 +1,76 @@ +package ${packageName}.domain; + +#foreach ($import in $subImportList) +import ${import}; +#end +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import com.ruoyi.common.annotation.Excel; +import com.ruoyi.common.core.domain.BaseEntity; + +/** + * ${subTable.functionName}对象 ${subTableName} + * + * @author ${author} + * @date ${datetime} + */ +public class ${subClassName} extends BaseEntity +{ + private static final long serialVersionUID = 1L; + +#foreach ($column in $subTable.columns) +#if(!$table.isSuperColumn($column.javaField)) + /** $column.columnComment */ +#if($column.list) +#set($parentheseIndex=$column.columnComment.indexOf("(")) +#if($parentheseIndex != -1) +#set($comment=$column.columnComment.substring(0, $parentheseIndex)) +#else +#set($comment=$column.columnComment) +#end +#if($parentheseIndex != -1) + @Excel(name = "${comment}", readConverterExp = "$column.readConverterExp()") +#elseif($column.javaType == 'Date') + @JsonFormat(pattern = "yyyy-MM-dd") + @Excel(name = "${comment}", width = 30, dateFormat = "yyyy-MM-dd") +#else + @Excel(name = "${comment}") +#end +#end + private $column.javaType $column.javaField; + +#end +#end +#foreach ($column in $subTable.columns) +#if(!$table.isSuperColumn($column.javaField)) +#if($column.javaField.length() > 2 && $column.javaField.substring(1,2).matches("[A-Z]")) +#set($AttrName=$column.javaField) +#else +#set($AttrName=$column.javaField.substring(0,1).toUpperCase() + ${column.javaField.substring(1)}) +#end + public void set${AttrName}($column.javaType $column.javaField) + { + this.$column.javaField = $column.javaField; + } + + public $column.javaType get${AttrName}() + { + return $column.javaField; + } +#end +#end + + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) +#foreach ($column in $subTable.columns) +#if($column.javaField.length() > 2 && $column.javaField.substring(1,2).matches("[A-Z]")) +#set($AttrName=$column.javaField) +#else +#set($AttrName=$column.javaField.substring(0,1).toUpperCase() + ${column.javaField.substring(1)}) +#end + .append("${column.javaField}", get${AttrName}()) +#end + .toString(); + } +} diff --git a/ruoyi-generator/target/classes/vm/js/api.js.vm b/ruoyi-generator/target/classes/vm/js/api.js.vm new file mode 100644 index 00000000..9295524a --- /dev/null +++ b/ruoyi-generator/target/classes/vm/js/api.js.vm @@ -0,0 +1,44 @@ +import request from '@/utils/request' + +// 查询${functionName}列表 +export function list${BusinessName}(query) { + return request({ + url: '/${moduleName}/${businessName}/list', + method: 'get', + params: query + }) +} + +// 查询${functionName}详细 +export function get${BusinessName}(${pkColumn.javaField}) { + return request({ + url: '/${moduleName}/${businessName}/' + ${pkColumn.javaField}, + method: 'get' + }) +} + +// 新增${functionName} +export function add${BusinessName}(data) { + return request({ + url: '/${moduleName}/${businessName}', + method: 'post', + data: data + }) +} + +// 修改${functionName} +export function update${BusinessName}(data) { + return request({ + url: '/${moduleName}/${businessName}', + method: 'put', + data: data + }) +} + +// 删除${functionName} +export function del${BusinessName}(${pkColumn.javaField}) { + return request({ + url: '/${moduleName}/${businessName}/' + ${pkColumn.javaField}, + method: 'delete' + }) +} diff --git a/ruoyi-generator/target/classes/vm/sql/sql.vm b/ruoyi-generator/target/classes/vm/sql/sql.vm new file mode 100644 index 00000000..05755835 --- /dev/null +++ b/ruoyi-generator/target/classes/vm/sql/sql.vm @@ -0,0 +1,22 @@ +-- 菜单 SQL +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}', '${parentMenuId}', '1', '${businessName}', '${moduleName}/${businessName}/index', 1, 0, 'C', '0', '0', '${permissionPrefix}:list', '#', 'admin', sysdate(), '', null, '${functionName}菜单'); + +-- 按钮父菜单ID +SELECT @parentId := LAST_INSERT_ID(); + +-- 按钮 SQL +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}查询', @parentId, '1', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:query', '#', 'admin', sysdate(), '', null, ''); + +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}新增', @parentId, '2', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:add', '#', 'admin', sysdate(), '', null, ''); + +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}修改', @parentId, '3', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:edit', '#', 'admin', sysdate(), '', null, ''); + +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}删除', @parentId, '4', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:remove', '#', 'admin', sysdate(), '', null, ''); + +insert into sys_menu (menu_name, parent_id, order_num, path, component, is_frame, is_cache, menu_type, visible, status, perms, icon, create_by, create_time, update_by, update_time, remark) +values('${functionName}导出', @parentId, '5', '#', '', 1, 0, 'F', '0', '0', '${permissionPrefix}:export', '#', 'admin', sysdate(), '', null, ''); \ No newline at end of file diff --git a/ruoyi-generator/target/classes/vm/vue/index-tree.vue.vm b/ruoyi-generator/target/classes/vm/vue/index-tree.vue.vm new file mode 100644 index 00000000..3def7045 --- /dev/null +++ b/ruoyi-generator/target/classes/vm/vue/index-tree.vue.vm @@ -0,0 +1,502 @@ + + + diff --git a/ruoyi-generator/target/classes/vm/vue/index.vue.vm b/ruoyi-generator/target/classes/vm/vue/index.vue.vm new file mode 100644 index 00000000..3f9a4b4a --- /dev/null +++ b/ruoyi-generator/target/classes/vm/vue/index.vue.vm @@ -0,0 +1,609 @@ + + + diff --git a/ruoyi-generator/target/classes/vm/vue/v3/index-tree.vue.vm b/ruoyi-generator/target/classes/vm/vue/v3/index-tree.vue.vm new file mode 100644 index 00000000..862297c7 --- /dev/null +++ b/ruoyi-generator/target/classes/vm/vue/v3/index-tree.vue.vm @@ -0,0 +1,486 @@ + + + diff --git a/ruoyi-generator/target/classes/vm/vue/v3/index.vue.vm b/ruoyi-generator/target/classes/vm/vue/v3/index.vue.vm new file mode 100644 index 00000000..f66cc3b8 --- /dev/null +++ b/ruoyi-generator/target/classes/vm/vue/v3/index.vue.vm @@ -0,0 +1,596 @@ + + + diff --git a/ruoyi-generator/target/classes/vm/vue/v3/readme.txt b/ruoyi-generator/target/classes/vm/vue/v3/readme.txt new file mode 100644 index 00000000..99239bb5 --- /dev/null +++ b/ruoyi-generator/target/classes/vm/vue/v3/readme.txt @@ -0,0 +1 @@ +ʹõRuoYi-Vue3ǰˣôҪһ´Ŀ¼ģindex.vue.vmindex-tree.vue.vmļϼvueĿ¼ \ No newline at end of file diff --git a/ruoyi-generator/target/classes/vm/xml/mapper.xml.vm b/ruoyi-generator/target/classes/vm/xml/mapper.xml.vm new file mode 100644 index 00000000..0ceb3d85 --- /dev/null +++ b/ruoyi-generator/target/classes/vm/xml/mapper.xml.vm @@ -0,0 +1,135 @@ + + + + + +#foreach ($column in $columns) + +#end + +#if($table.sub) + + + + + + +#foreach ($column in $subTable.columns) + +#end + +#end + + + select#foreach($column in $columns) $column.columnName#if($foreach.count != $columns.size()),#end#end from ${tableName} + + + + + + + + insert into ${tableName} + +#foreach($column in $columns) +#if($column.columnName != $pkColumn.columnName || !$pkColumn.increment) + $column.columnName, +#end +#end + + +#foreach($column in $columns) +#if($column.columnName != $pkColumn.columnName || !$pkColumn.increment) + #{$column.javaField}, +#end +#end + + + + + update ${tableName} + +#foreach($column in $columns) +#if($column.columnName != $pkColumn.columnName) + $column.columnName = #{$column.javaField}, +#end +#end + + where ${pkColumn.columnName} = #{${pkColumn.javaField}} + + + + delete from ${tableName} where ${pkColumn.columnName} = #{${pkColumn.javaField}} + + + + delete from ${tableName} where ${pkColumn.columnName} in + + #{${pkColumn.javaField}} + + +#if($table.sub) + + + delete from ${subTableName} where ${subTableFkName} in + + #{${subTableFkclassName}} + + + + + delete from ${subTableName} where ${subTableFkName} = #{${subTableFkclassName}} + + + + insert into ${subTableName}(#foreach($column in $subTable.columns) $column.columnName#if($foreach.count != $subTable.columns.size()),#end#end) values + + (#foreach($column in $subTable.columns) #{item.$column.javaField}#if($foreach.count != $subTable.columns.size()),#end#end) + + +#end + \ No newline at end of file diff --git a/ruoyi-plugin/pom.xml b/ruoyi-plugin/pom.xml new file mode 100644 index 00000000..478cc7fb --- /dev/null +++ b/ruoyi-plugin/pom.xml @@ -0,0 +1,20 @@ + + + + ruoyi + com.ruoyi + 3.8.4 + + 4.0.0 + + ruoyi-plugin + pom + + ruoyi-springboot-starter-tenant + ruoyi-springboot-starter-oss + + + + diff --git a/ruoyi-plugin/ruoyi-springboot-starter-oss/pom.xml b/ruoyi-plugin/ruoyi-springboot-starter-oss/pom.xml new file mode 100644 index 00000000..57a0dfc4 --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-oss/pom.xml @@ -0,0 +1,34 @@ + + + + ruoyi-plugin + com.ruoyi + 3.8.4 + + 4.0.0 + + ruoyi-springboot-starter-oss + + + + org.springframework.boot + spring-boot-starter-web + provided + + + + + org.projectlombok + lombok + true + + + + com.amazonaws + aws-java-sdk-s3 + + + + diff --git a/ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/java/com/ruoyi/plugin/oss/config/OssConfiguration.java b/ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/java/com/ruoyi/plugin/oss/config/OssConfiguration.java new file mode 100644 index 00000000..ccab10af --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/java/com/ruoyi/plugin/oss/config/OssConfiguration.java @@ -0,0 +1,69 @@ +/* + * Copyright (c) 2018-2025, lengleng 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 pig4cloud.com 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: lengleng (wangiegie@gmail.com) + */ +package com.ruoyi.plugin.oss.config; + +import com.ruoyi.plugin.oss.controller.OssEndpoint; +import com.ruoyi.plugin.oss.core.OssTemplate; +import com.ruoyi.plugin.oss.props.OssProperties; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; +import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; +import org.springframework.boot.context.properties.EnableConfigurationProperties; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; + +/** + * AWS自动配置类 + * + * @author lengleng + * @author 858695266 + * @link https://github.com/pig-mesh/oss-spring-boot-starter + * @since 1.0.0 + */ +@Configuration(proxyBeanMethods = false) +@EnableConfigurationProperties({OssProperties.class}) +public class OssConfiguration { + + @Autowired + private OssProperties properties; + + /** + * OSS操作模板 + * + * @return OSS操作模板 + */ + @Bean + @ConditionalOnMissingBean(OssTemplate.class) + @ConditionalOnProperty(prefix = OssProperties.PREFIX, name = "enable", havingValue = "true", matchIfMissing = true) + public OssTemplate ossTemplate(OssProperties properties) { + return new OssTemplate(properties); + } + + /** + * OSS端点信息 + * + * @param template oss操作模版 + * @return oss远程服务端点 + */ + @Bean + @ConditionalOnWebApplication + @ConditionalOnProperty(prefix = OssProperties.PREFIX, name = "http.enable", havingValue = "true") + public OssEndpoint ossEndpoint(OssTemplate template) { + return new OssEndpoint(template); + } +} diff --git a/ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/java/com/ruoyi/plugin/oss/controller/OssEndpoint.java b/ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/java/com/ruoyi/plugin/oss/controller/OssEndpoint.java new file mode 100644 index 00000000..0d67206b --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/java/com/ruoyi/plugin/oss/controller/OssEndpoint.java @@ -0,0 +1,128 @@ +/* + * Copyright (c) 2018-2025, lengleng 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 pig4cloud.com 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: lengleng (wangiegie@gmail.com) + */ +package com.ruoyi.plugin.oss.controller; + +import com.amazonaws.services.s3.model.Bucket; +import com.amazonaws.services.s3.model.S3Object; +import com.amazonaws.services.s3.model.S3ObjectSummary; +import com.ruoyi.plugin.oss.core.OssTemplate; +import lombok.AllArgsConstructor; +import lombok.SneakyThrows; +import org.springframework.http.HttpStatus; +import org.springframework.web.bind.annotation.*; +import org.springframework.web.multipart.MultipartFile; + +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +/** + * AWS对外提供服务接口 + * + * @author lengleng + * @author 858695266 + * @since 1.0.0 + */ +@RestController +@AllArgsConstructor +@RequestMapping("/oss") +public class OssEndpoint { + + private final OssTemplate template; + + /** + * Bucket Endpoints + */ + @SneakyThrows + @PostMapping("/bucket/{bucketName}") + public Bucket createBucket(@PathVariable String bucketName) { + + template.createBucket(bucketName); + return template.getBucket(bucketName).get(); + + } + + @SneakyThrows + @GetMapping("/bucket") + public List getBuckets() { + return template.getAllBuckets(); + } + + @SneakyThrows + @GetMapping("/bucket/{bucketName}") + public Bucket getBucket(@PathVariable String bucketName) { + return template.getBucket(bucketName).orElseThrow(() -> new IllegalArgumentException("Bucket Name not found!")); + } + + @SneakyThrows + @DeleteMapping("/bucket/{bucketName}") + @ResponseStatus(HttpStatus.ACCEPTED) + public void deleteBucket(@PathVariable String bucketName) { + template.removeBucket(bucketName); + } + + /** + * Object Endpoints + */ + @SneakyThrows + @PostMapping("/object/{bucketName}") + public S3Object createObject(@RequestBody MultipartFile object, @PathVariable String bucketName) { + String name = object.getOriginalFilename(); + template.putObject(bucketName, name, object.getInputStream(), object.getSize(), object.getContentType()); + return template.getObjectInfo(bucketName, name); + + } + + @SneakyThrows + @PostMapping("/object/{bucketName}/{objectName}") + public S3Object createObject(@RequestBody MultipartFile object, @PathVariable String bucketName, + @PathVariable String objectName) { + template.putObject(bucketName, objectName, object.getInputStream(), object.getSize(), object.getContentType()); + return template.getObjectInfo(bucketName, objectName); + + } + + @SneakyThrows + @GetMapping("/object/{bucketName}/{objectName}") + public List filterObject(@PathVariable String bucketName, @PathVariable String objectName) { + + return template.getAllObjectsByPrefix(bucketName, objectName, true); + + } + + @SneakyThrows + @GetMapping("/object/{bucketName}/{objectName}/{expires}") + public Map getObject(@PathVariable String bucketName, @PathVariable String objectName, + @PathVariable Integer expires) { + Map responseBody = new HashMap<>(8); + // Put Object info + responseBody.put("bucket", bucketName); + responseBody.put("object", objectName); + responseBody.put("url", template.getObjectURL(bucketName, objectName, expires)); + responseBody.put("expires", expires); + return responseBody; + } + + @SneakyThrows + @ResponseStatus(HttpStatus.ACCEPTED) + @DeleteMapping("/object/{bucketName}/{objectName}/") + public void deleteObject(@PathVariable String bucketName, @PathVariable String objectName) { + + template.removeObject(bucketName, objectName); + } + +} diff --git a/ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/java/com/ruoyi/plugin/oss/core/OssTemplate.java b/ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/java/com/ruoyi/plugin/oss/core/OssTemplate.java new file mode 100644 index 00000000..74b265af --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/java/com/ruoyi/plugin/oss/core/OssTemplate.java @@ -0,0 +1,242 @@ +/* + * Copyright (c) 2018-2025, lengleng 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 pig4cloud.com 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: lengleng (wangiegie@gmail.com) + */ +package com.ruoyi.plugin.oss.core; + +import com.amazonaws.ClientConfiguration; +import com.amazonaws.auth.AWSCredentials; +import com.amazonaws.auth.AWSCredentialsProvider; +import com.amazonaws.auth.AWSStaticCredentialsProvider; +import com.amazonaws.auth.BasicAWSCredentials; +import com.amazonaws.client.builder.AwsClientBuilder; +import com.amazonaws.services.s3.AmazonS3; +import com.amazonaws.services.s3.AmazonS3Client; +import com.amazonaws.services.s3.model.*; +import com.amazonaws.util.IOUtils; +import com.ruoyi.plugin.oss.props.OssProperties; +import lombok.Cleanup; +import lombok.RequiredArgsConstructor; +import lombok.SneakyThrows; +import org.springframework.beans.factory.InitializingBean; + +import java.io.ByteArrayInputStream; +import java.io.InputStream; +import java.net.URL; +import java.util.*; + +/** + * aws-s3 通用存储操作 支持所有兼容s3协议的云存储: {阿里云OSS,腾讯云COS,七牛云,京东云,minio 等} + * + * @author lengleng + * @author 858695266 + * @since 1.0 + */ +@RequiredArgsConstructor +public class OssTemplate implements InitializingBean { + + private final OssProperties ossProperties; + + private AmazonS3 amazonS3; + + /** + * 创建bucket + * + * @param bucketName bucket名称 + */ + @SneakyThrows + public void createBucket(String bucketName) { + if (!amazonS3.doesBucketExistV2(bucketName)) { + amazonS3.createBucket((bucketName)); + } + } + + /** + * 获取全部bucket + *

    + * If the object identified by the given bucket and key has public read permissions + * (ex: {@link CannedAccessControlList#PublicRead}), then this URL can be directly + * accessed to retrieve the object's data. + * + * @param bucketName bucket名称 + * @param objectName 文件名称 + * @return url + */ + public String getObjectURL(String bucketName, String objectName) { + URL url = amazonS3.getUrl(bucketName, objectName); + return url.toString(); + } + + /** + * 获取文件 + * + * @param bucketName bucket名称 + * @param objectName 文件名称 + * @return 二进制流 + * @see AWS + * API Documentation + */ + @SneakyThrows + public InputStream getObject(String bucketName, String objectName) { + return amazonS3.getObject(bucketName, objectName).getObjectContent(); + } + + /** + * 上传文件 + * + * @param bucketName bucket名称 + * @param objectName 文件名称 + * @param stream 文件流 + * @throws Exception + */ + public void putObject(String bucketName, String objectName, InputStream stream) throws Exception { + putObject(bucketName, objectName, stream, (long) stream.available(), "application/octet-stream"); + } + + /** + * 上传文件 + * + * @param bucketName bucket名称 + * @param objectName 文件名称 + * @param stream 文件流 + * @param size 大小 + * @param contextType 类型 + * @throws Exception + * @see AWS + * API Documentation + */ + public PutObjectResult putObject(String bucketName, String objectName, InputStream stream, long size, + String contextType) throws Exception { + // String fileName = getFileName(objectName); + byte[] bytes = IOUtils.toByteArray(stream); + ObjectMetadata objectMetadata = new ObjectMetadata(); + objectMetadata.setContentLength(size); + objectMetadata.setContentType(contextType); + ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(bytes); + // 上传 + return amazonS3.putObject(bucketName, objectName, byteArrayInputStream, objectMetadata); + + } + + /** + * 获取文件信息 + * + * @param bucketName bucket名称 + * @param objectName 文件名称 + * @throws Exception + * @see AWS + * API Documentation + */ + public S3Object getObjectInfo(String bucketName, String objectName) throws Exception { + @Cleanup S3Object object = amazonS3.getObject(bucketName, objectName); + return object; + } + + /** + * 删除文件 + * + * @param bucketName bucket名称 + * @param objectName 文件名称 + * @throws Exception + * @see AWS API + * Documentation + */ + public void removeObject(String bucketName, String objectName) throws Exception { + amazonS3.deleteObject(bucketName, objectName); + } + + @Override + public void afterPropertiesSet() throws Exception { + ClientConfiguration clientConfiguration = new ClientConfiguration(); + AwsClientBuilder.EndpointConfiguration endpointConfiguration = new AwsClientBuilder.EndpointConfiguration( + ossProperties.getEndpoint(), ossProperties.getRegion()); + AWSCredentials awsCredentials = new BasicAWSCredentials(ossProperties.getAccessKey(), + ossProperties.getSecretKey()); + AWSCredentialsProvider awsCredentialsProvider = new AWSStaticCredentialsProvider(awsCredentials); + this.amazonS3 = AmazonS3Client.builder().withEndpointConfiguration(endpointConfiguration) + .withClientConfiguration(clientConfiguration).withCredentials(awsCredentialsProvider) + .disableChunkedEncoding().withPathStyleAccessEnabled(ossProperties.getPathStyleAccess()).build(); + } + +} diff --git a/ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/java/com/ruoyi/plugin/oss/props/OssProperties.java b/ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/java/com/ruoyi/plugin/oss/props/OssProperties.java new file mode 100644 index 00000000..34775df1 --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/java/com/ruoyi/plugin/oss/props/OssProperties.java @@ -0,0 +1,94 @@ +/* + * Copyright (c) 2018-2025, lengleng 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 pig4cloud.com 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: lengleng (wangiegie@gmail.com) + */ +package com.ruoyi.plugin.oss.props; + +import lombok.Data; +import org.springframework.boot.context.properties.ConfigurationProperties; + +import java.io.Serializable; + +/** + * AWS配置信息 + * + * @author lengleng + * @author 858695266 + *

    + * oss: enable: true + * endpoint: http://127.0.0.1:9000 # + * pathStyleAccess 采用nginx反向代理或者AWS S3 配置成true,支持第三方云存储配置成false + * pathStyleAccess: false + * access-key: mate + * secret-key: mate + * bucket-name: mate + * region: custom-domain: oss.mate.vip + *

    + */ +@Data +@ConfigurationProperties(prefix = OssProperties.PREFIX) +public class OssProperties implements Serializable { + + private static final long serialVersionUID = 7111600914195056135L; + + + /** + * 配置前缀 + */ + public static final String PREFIX = "oss"; + + /** + * 是否启用 oss,默认为:true + */ + private boolean enable = true; + + /** + * 对象存储服务的URL + */ + private String endpoint; + + /** + * 自定义域名 + */ + private String customDomain; + + /** + * true path-style nginx 反向代理和S3默认支持 pathStyle {http://endpoint/bucketname} false + * supports virtual-hosted-style 阿里云等需要配置为 virtual-hosted-style + * 模式{http://bucketname.endpoint} + */ + private Boolean pathStyleAccess = true; + + /** + * 区域 + */ + private String region; + + /** + * Access key就像用户ID,可以唯一标识你的账户 + */ + private String accessKey; + + /** + * Secret key是你账户的密码 + */ + private String secretKey; + + /** + * 默认的存储桶名称 + */ + private String bucketName; + +} diff --git a/ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/resources/META-INF/spring.factories b/ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/resources/META-INF/spring.factories new file mode 100644 index 00000000..6d49fd9c --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-oss/src/main/resources/META-INF/spring.factories @@ -0,0 +1,2 @@ +org.springframework.boot.autoconfigure.EnableAutoConfiguration=\ + com.ruoyi.plugin.oss.config.OssConfiguration diff --git a/ruoyi-plugin/ruoyi-springboot-starter-oss/target/classes/META-INF/spring.factories b/ruoyi-plugin/ruoyi-springboot-starter-oss/target/classes/META-INF/spring.factories new file mode 100644 index 00000000..6d49fd9c --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-oss/target/classes/META-INF/spring.factories @@ -0,0 +1,2 @@ +org.springframework.boot.autoconfigure.EnableAutoConfiguration=\ + com.ruoyi.plugin.oss.config.OssConfiguration diff --git a/ruoyi-plugin/ruoyi-springboot-starter-oss/target/classes/com/ruoyi/plugin/oss/config/OssConfiguration.class b/ruoyi-plugin/ruoyi-springboot-starter-oss/target/classes/com/ruoyi/plugin/oss/config/OssConfiguration.class new file mode 100644 index 0000000000000000000000000000000000000000..9a927c6c82fb4a7eff52de592b735e059ed4013b GIT binary patch literal 1760 zcmbtVYflqF6ur|H3IzdCKz!l@DX*>%Bqpc{5{$+IMo5kLX}g`$!R^dub{6Qr(oZCs z_yhb=#xvd0K-n4-zU<6w?!D*UbI#8G{`2D(5iQYjjUu`$hU2yqoYlyvL!o&EWqV%;DaHKVoQD#k5IHTdBkbE<$m$}|?fyQXGs65v%xmwtB>t)8&1mFwFb9hQt#hx8xt+|VI zvtNJ?V9Xs}4}904^maZuKZ%Fg-x~v3BNW-YqMW}P^cU4Imt$d$r ztJO(eG~8#}X4t$S4WD>7U*@scOw0-wV!qQ>QbvTjJu@SQ(^2y2NNY z@#^SQ!Uiufn(q>vB!^9T#%Q5iO428d$t`n^?@D~{<$dnKEsI^;Vkhmq-t5&g*CoDl zrj^+e-KAdKvXq@?@r9U2 z*Ju*|gzF*QpqqFR(k;3j(jA)G#bAu?(f#x28H?W2iN4ka{Q)pyPNrJu)97Fl`%|b7 bW7HJQG-t{;Ih+9WEOO^|X`U9^E%*NgCb9Q< literal 0 HcmV?d00001 diff --git a/ruoyi-plugin/ruoyi-springboot-starter-oss/target/classes/com/ruoyi/plugin/oss/controller/OssEndpoint.class b/ruoyi-plugin/ruoyi-springboot-starter-oss/target/classes/com/ruoyi/plugin/oss/controller/OssEndpoint.class new file mode 100644 index 0000000000000000000000000000000000000000..2987f43ed68886d608cf5c0f6538d03dba949613 GIT binary patch literal 6175 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 DXJkeb literal 0 HcmV?d00001 diff --git a/ruoyi-plugin/ruoyi-springboot-starter-oss/target/classes/com/ruoyi/plugin/oss/core/OssTemplate.class b/ruoyi-plugin/ruoyi-springboot-starter-oss/target/classes/com/ruoyi/plugin/oss/core/OssTemplate.class new file mode 100644 index 0000000000000000000000000000000000000000..cde35eaae6bde47950c830eb12a3d416abe3d268 GIT binary patch literal 9038 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 literal 0 HcmV?d00001 diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/pom.xml b/ruoyi-plugin/ruoyi-springboot-starter-tenant/pom.xml new file mode 100644 index 00000000..688717a3 --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-tenant/pom.xml @@ -0,0 +1,39 @@ + + + + ruoyi-plugin + com.ruoyi + 3.8.4 + + 4.0.0 + + ruoyi-springboot-starter-tenant + + + 多租户插件 + + + + + + com.ruoyi + ruoyi-common + + + + + org.projectlombok + lombok + true + + + + + org.springframework.boot + spring-boot-starter-aop + + + + diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/config/TenantConfiguration.java b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/config/TenantConfiguration.java new file mode 100644 index 00000000..50dccd81 --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/config/TenantConfiguration.java @@ -0,0 +1,63 @@ +package com.ruoyi.plugin.tenant.config; + +import com.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor; +import com.baomidou.mybatisplus.extension.plugins.inner.TenantLineInnerInterceptor; +import com.ruoyi.common.mybatis.util.MyBatisUtils; +import com.ruoyi.plugin.tenant.core.aop.TenantIgnoreAspect; +import com.ruoyi.plugin.tenant.core.db.TenantDatabaseInterceptor; +import com.ruoyi.plugin.tenant.core.security.TenantSecurityWebFilter; +import com.ruoyi.plugin.tenant.core.web.TenantContextWebFilter; +import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; +import org.springframework.boot.context.properties.EnableConfigurationProperties; +import org.springframework.boot.web.servlet.FilterRegistrationBean; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; + +@Configuration +@ConditionalOnProperty(prefix = "ruoyi.tenant", value = "enable", matchIfMissing = true) // 允许使用 yudao.tenant.enable=false 禁用多租户 +@EnableConfigurationProperties(TenantProperties.class) +public class TenantConfiguration { +// +// @Bean +// public TenantFrameworkService tenantFrameworkService(TenantApi tenantApi) { +// return new TenantFrameworkServiceImpl(tenantApi); +// } + + // ========== AOP ========== + + @Bean + public TenantIgnoreAspect tenantIgnoreAspect() { + return new TenantIgnoreAspect(); + } + + // ========== DB ========== + + @Bean + public TenantLineInnerInterceptor tenantLineInnerInterceptor(TenantProperties properties, MybatisPlusInterceptor interceptor) { + TenantLineInnerInterceptor inner = new TenantLineInnerInterceptor(new TenantDatabaseInterceptor(properties)); + // 添加到 interceptor 中 + // 需要加在首个,主要是为了在分页插件前面。这个是 MyBatis Plus 的规定 + MyBatisUtils.addInterceptor(interceptor, inner, 0); + return inner; + } + +// ========== WEB ========== + + @Bean + public FilterRegistrationBean tenantContextWebFilter() { + FilterRegistrationBean registrationBean = new FilterRegistrationBean<>(); + registrationBean.setFilter(new TenantContextWebFilter()); + registrationBean.setOrder(-104); + return registrationBean; + } + + // ========== Security ========== + + @Bean + public FilterRegistrationBean tenantSecurityWebFilter(TenantProperties tenantProperties) { + FilterRegistrationBean registrationBean = new FilterRegistrationBean<>(); + registrationBean.setFilter(new TenantSecurityWebFilter(tenantProperties)); + registrationBean.setOrder(-99); + return registrationBean; + } +} diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/config/TenantProperties.java b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/config/TenantProperties.java new file mode 100644 index 00000000..f01af7fd --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/config/TenantProperties.java @@ -0,0 +1,40 @@ +package com.ruoyi.plugin.tenant.config; + +import lombok.Data; +import org.springframework.boot.context.properties.ConfigurationProperties; + +import java.util.Collections; +import java.util.Set; + +/** + * 多租户配置 + * + * @author 芋道源码 + */ +@ConfigurationProperties(prefix = "ruoyi.tenant") +@Data +public class TenantProperties { + /** + * 租户是否开启 + */ + private static final Boolean ENABLE_DEFAULT = true; + + /** + * 是否开启 + */ + private Boolean enable = ENABLE_DEFAULT; + /** + * 需要忽略多租户的请求 + * + * 默认情况下,每个请求需要带上 tenant-id 的请求头。但是,部分请求是无需带上的,例如说短信回调、支付回调等 Open API! + */ + private Set ignoreUrls = Collections.emptySet(); + +// /** +// * 多租户的表 +// * +// * 即默认所有表都开启多租户的功能,所以记得添加对应的 tenant_id 字段哟 +// */ +// private Set noignoreTables = Collections.emptySet(); + +} diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/aop/TenantIgnore.java b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/aop/TenantIgnore.java new file mode 100644 index 00000000..f6370c5b --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/aop/TenantIgnore.java @@ -0,0 +1,18 @@ +package com.ruoyi.plugin.tenant.core.aop; + +import java.lang.annotation.*; + +/** + * 忽略租户,标记指定方法不进行租户的自动过滤 + * + * 注意,只有 DB 的场景会过滤,其它场景暂时不过滤: + * 1、Redis 场景:因为是基于 Key 实现多租户的能力,所以忽略没有意义,不像 DB 是一个 column 实现的 + * 2、MQ 场景:有点难以抉择,目前可以通过 Consumer 手动在消费的方法上,添加 @TenantIgnore 进行忽略 + * + * @author 芋道源码 + */ +@Target({ElementType.METHOD}) +@Retention(RetentionPolicy.RUNTIME) +@Inherited +public @interface TenantIgnore { +} diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/aop/TenantIgnoreAspect.java b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/aop/TenantIgnoreAspect.java new file mode 100644 index 00000000..b58d7f93 --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/aop/TenantIgnoreAspect.java @@ -0,0 +1,32 @@ +package com.ruoyi.plugin.tenant.core.aop; + +import com.ruoyi.plugin.tenant.core.context.TenantContextHolder; +import lombok.extern.slf4j.Slf4j; +import org.aspectj.lang.ProceedingJoinPoint; +import org.aspectj.lang.annotation.Around; +import org.aspectj.lang.annotation.Aspect; + +/** + * 忽略多租户的 Aspect,基于 {@link TenantIgnore} 注解实现,用于一些全局的逻辑。 + * 例如说,一个定时任务,读取所有数据,进行处理。 + * 又例如说,读取所有数据,进行缓存。 + * + * @author 芋道源码 + */ +@Aspect +@Slf4j +public class TenantIgnoreAspect { + + @Around("@annotation(tenantIgnore)") + public Object around(ProceedingJoinPoint joinPoint, TenantIgnore tenantIgnore) throws Throwable { + Boolean oldIgnore = TenantContextHolder.isIgnore(); + try { + TenantContextHolder.setIgnore(true); + // 执行逻辑 + return joinPoint.proceed(); + } finally { + TenantContextHolder.setIgnore(oldIgnore); + } + } + +} diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/context/TenantContextHolder.java b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/context/TenantContextHolder.java new file mode 100644 index 00000000..37e2997a --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/context/TenantContextHolder.java @@ -0,0 +1,66 @@ +package com.ruoyi.plugin.tenant.core.context; + +import com.alibaba.ttl.TransmittableThreadLocal; + +/** + * 多租户上下文 Holder + * + * @author 芋道源码 + */ +public class TenantContextHolder { + + /** + * 当前租户编号 + */ + private static final ThreadLocal TENANT_ID = new TransmittableThreadLocal<>(); + + /** + * 是否忽略租户 + */ + private static final ThreadLocal IGNORE = new TransmittableThreadLocal<>(); + + /** + * 获得租户编号。 + * + * @return 租户编号 + */ + public static Long getTenantId() { + return TENANT_ID.get(); + } + + /** + * 获得租户编号。如果不存在,则抛出 NullPointerException 异常 + * + * @return 租户编号 + */ + public static Long getRequiredTenantId() { + Long tenantId = getTenantId(); + if (tenantId == null) { + throw new NullPointerException("TenantContextHolder 不存在租户编号"); // TODO 芋艿:增加文档链接 + } + return tenantId; + } + + public static void setTenantId(Long tenantId) { + TENANT_ID.set(tenantId); + } + + public static void setIgnore(Boolean ignore) { + IGNORE.set(ignore); + } + + /** + * 当前是否忽略租户 + * + * @return 是否忽略 + */ + public static boolean isIgnore() { + return Boolean.TRUE.equals(IGNORE.get()); + } + + public static void clear() { + TENANT_ID.remove(); + IGNORE.remove(); + } + +} diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/db/TenantBaseDO.java b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/db/TenantBaseDO.java new file mode 100644 index 00000000..59fb0023 --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/db/TenantBaseDO.java @@ -0,0 +1,21 @@ +package com.ruoyi.plugin.tenant.core.db; + +import com.ruoyi.common.mybatis.dataobject.BaseDO; +import lombok.Data; +import lombok.EqualsAndHashCode; + +/** + * 拓展多租户的 BaseDO 基类 + * + * @author 芋道源码 + */ +@Data +@EqualsAndHashCode(callSuper = true) +public abstract class TenantBaseDO extends BaseDO { + + /** + * 多租户编号 + */ + private Long tenantId; + +} diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/db/TenantDatabaseInterceptor.java b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/db/TenantDatabaseInterceptor.java new file mode 100644 index 00000000..d9de74c3 --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/db/TenantDatabaseInterceptor.java @@ -0,0 +1,66 @@ +package com.ruoyi.plugin.tenant.core.db; + +import cn.hutool.core.collection.CollUtil; +import com.baomidou.mybatisplus.extension.plugins.handler.TenantLineHandler; +import com.ruoyi.plugin.tenant.config.TenantProperties; +import com.ruoyi.plugin.tenant.core.context.TenantContextHolder; +import net.sf.jsqlparser.expression.Expression; +import net.sf.jsqlparser.expression.LongValue; + +import java.util.Arrays; +import java.util.HashSet; +import java.util.List; +import java.util.Set; + +/** + * 基于 MyBatis Plus 多租户的功能,实现 DB 层面的多租户的功能 + * + * @author 芋道源码 + */ +public class TenantDatabaseInterceptor implements TenantLineHandler { + + private final Set noIgnoreTables = new HashSet<>(); + /** + *多租户的系统类相关表 + */ + public static List tables = + Arrays.asList( "sys_role_menu", "sys_logininfor", "sys_user", "sys_post","sys_user_post", "sys_role_dept", "sys_role_menu", "sys_role", "sys_dept", "sys_user_role", "sys_oper_log","sys_notice"); + + /** + * 业务/工作流非多租户的相关表 + */ + public static List bstables = + Arrays.asList("bs_product"); + public TenantDatabaseInterceptor(TenantProperties properties) { + // 不同 DB 下,大小写的习惯不同,所以需要都添加进去 + tables.forEach(table -> { + noIgnoreTables.add(table.toLowerCase()); + noIgnoreTables.add(table.toUpperCase()); + }); + // 在 OracleKeyGenerator 中,生成主键时,会查询这个表,查询这个表后,会自动拼接 TENANT_ID 导致报错 + noIgnoreTables.add("DUAL"); + } + + @Override + public Expression getTenantId() { + return new LongValue(TenantContextHolder.getRequiredTenantId()); + } + + @Override + public boolean ignoreTable(String tableName) { + if (TenantContextHolder.isIgnore()) { + // 情况一,全局忽略多租户 + return true; + } + if (tableName.indexOf("bs_") ==0 || tableName.indexOf("bpm_") ==0) { + //业务相关表 + if (bstables.contains(tableName)) { + return true; + }else { + return false; + } + } + return !CollUtil.contains(noIgnoreTables, tableName); // 情况二,忽略非多租户的系统基础功能表 + } + +} diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/security/TenantSecurityWebFilter.java b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/security/TenantSecurityWebFilter.java new file mode 100644 index 00000000..2f56d2ae --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/security/TenantSecurityWebFilter.java @@ -0,0 +1,102 @@ +package com.ruoyi.plugin.tenant.core.security; + +import cn.hutool.core.collection.CollUtil; +import com.ruoyi.common.core.domain.R; +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.common.filter.ApiRequestFilter; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.common.utils.ServletUtils; +import com.ruoyi.plugin.tenant.config.TenantProperties; +import com.ruoyi.plugin.tenant.core.context.TenantContextHolder; +import lombok.extern.slf4j.Slf4j; +import org.springframework.util.AntPathMatcher; + +import javax.servlet.FilterChain; +import javax.servlet.ServletException; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import java.io.IOException; +import java.util.Objects; + +/** + * 多租户 Security Web 过滤器 + * 1. 如果是登陆的用户,校验是否有权限访问该租户,避免越权问题。 + * 2. 如果请求未带租户的编号,检查是否是忽略的 URL,否则也不允许访问。 + * 3. 校验租户是合法,例如说被禁用、到期 + *

    + * 校验用户访问的租户,是否是其所在的租户, + * + * @author 芋道源码 + */ +@Slf4j +public class TenantSecurityWebFilter extends ApiRequestFilter { + + private final TenantProperties tenantProperties; + + private final AntPathMatcher pathMatcher; + + + public TenantSecurityWebFilter(TenantProperties tenantProperties) { + this.tenantProperties = tenantProperties; + this.pathMatcher = new AntPathMatcher(); + } + + @Override + protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain chain) throws ServletException, IOException { + Long tenantId = TenantContextHolder.getTenantId(); + // 1. 登陆的用户,校验是否有权限访问该租户,避免越权问题。 + LoginUser user = SecurityUtils.getLoginUserNoExcePtion(); + if (user != null) { + // 如果获取不到租户编号,则尝试使用登陆用户的租户编号 + if (tenantId == null) { + tenantId = user.getTenantId(); + TenantContextHolder.setTenantId(tenantId); + // 如果传递了租户编号,则进行比对租户编号,避免越权问题 + } else if (!Objects.equals(user.getTenantId(), TenantContextHolder.getTenantId())) { + log.error("[doFilterInternal][租户({}) User({}/{}) 越权访问租户({}) URL({}/{})]", user.getTenantId(), user.getUserId(), "admin", TenantContextHolder.getTenantId(), request.getRequestURI(), request.getMethod()); + ServletUtils.writeJSON(response, R.fail("您无权访问该租户的数据")); + return; + } + } + + // 如果非允许忽略租户的 URL,则校验租户是否合法 + if (!isIgnoreUrl(request)) { + // 2. 如果请求未带租户的编号,不允许访问。 + if (tenantId == null) { + log.error("[doFilterInternal][URL({}/{}) 未传递租户编号]", request.getRequestURI(), request.getMethod()); + ServletUtils.writeJSON(response, R.fail("租户的请求未传递,请进行排查")); + return; + } + // 3. 校验租户是合法,例如说被禁用、到期 cqptodo +// try { +// tenantFrameworkService.validTenant(tenantId); +// } catch (Throwable ex) { +// R result = globalExceptionHandler.allExceptionHandler(request, ex); +// ServletUtils.writeJSON(response, result); +// return; +// } + } else { // 如果是允许忽略租户的 URL,若未传递租户编号,则默认忽略租户编号,避免报错 + if (tenantId == null) { + TenantContextHolder.setIgnore(true); + } + } + + // 继续过滤 + chain.doFilter(request, response); + } + + private boolean isIgnoreUrl(HttpServletRequest request) { + // 快速匹配,保证性能 + if (CollUtil.contains(tenantProperties.getIgnoreUrls(), request.getRequestURI())) { + return true; + } + // 逐个 Ant 路径匹配 + for (String url : tenantProperties.getIgnoreUrls()) { + if (pathMatcher.match(url, request.getRequestURI())) { + return true; + } + } + return false; + } + +} diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/util/TenantUtils.java b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/util/TenantUtils.java new file mode 100644 index 00000000..2febb100 --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/util/TenantUtils.java @@ -0,0 +1,49 @@ +package com.ruoyi.plugin.tenant.core.util; + + +import cn.hutool.core.util.StrUtil; +import com.ruoyi.plugin.tenant.core.context.TenantContextHolder; + +import javax.servlet.http.HttpServletRequest; + +/** + * 多租户 Util + * + * @author 芋道源码 + */ +public class TenantUtils { + public static final String HEADER_TENANT_ID = "tenant-id"; + /** + * 使用指定租户,执行对应的逻辑 + * + * 注意,如果当前是忽略租户的情况下,会被强制设置成不忽略租户 + * 当然,执行完成后,还是会恢复回去 + * + * @param tenantId 租户编号 + * @param runnable 逻辑 + */ + public static void execute(Long tenantId, Runnable runnable) { + Long oldTenantId = TenantContextHolder.getTenantId(); + Boolean oldIgnore = TenantContextHolder.isIgnore(); + try { + TenantContextHolder.setTenantId(tenantId); + TenantContextHolder.setIgnore(false); + // 执行逻辑 + runnable.run(); + } finally { + TenantContextHolder.setTenantId(oldTenantId); + TenantContextHolder.setIgnore(oldIgnore); + } + } + /** + * 获得租户编号,从 header 中 + * 考虑到其它 framework 组件也会使用到租户编号,所以不得不放在 WebFrameworkUtils 统一提供 + * + * @param request 请求 + * @return 租户编号 + */ + public static Long getTenantId(HttpServletRequest request) { + String tenantId = request.getHeader(HEADER_TENANT_ID); + return StrUtil.isNotEmpty(tenantId) ? Long.valueOf(tenantId) : null; + } +} diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/web/TenantContextWebFilter.java b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/web/TenantContextWebFilter.java new file mode 100644 index 00000000..f9e2648b --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/java/com/ruoyi/plugin/tenant/core/web/TenantContextWebFilter.java @@ -0,0 +1,44 @@ +package com.ruoyi.plugin.tenant.core.web; + +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.plugin.tenant.core.context.TenantContextHolder; +import com.ruoyi.plugin.tenant.core.util.TenantUtils; +import org.springframework.security.core.context.SecurityContext; +import org.springframework.security.core.context.SecurityContextHolder; +import org.springframework.web.filter.OncePerRequestFilter; + +import javax.servlet.FilterChain; +import javax.servlet.ServletException; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import java.io.IOException; + +/** + * 多租户 Context Web 过滤器 + * 将请求 Header 中的 tenant-id 解析出来,添加到 {@link TenantContextHolder} 中,这样后续的 DB 等操作,可以获得到租户编号。 + * + * @author 芋道源码 + */ +public class TenantContextWebFilter extends OncePerRequestFilter { + + @Override + protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain chain) throws ServletException, IOException { + // 设置 + Long tenantId = TenantUtils.getTenantId(request); + SecurityContext securityContext = SecurityContextHolder.getContext(); + if (securityContext != null && securityContext.getAuthentication() != null && securityContext.getAuthentication().isAuthenticated()) { + //已登录直接取用户的租户ID。这样判断是因为未登录的接口就需要从请求头获取了 + tenantId = SecurityUtils.getLoginUser().getTenantId(); + } + if (tenantId != null) { + TenantContextHolder.setTenantId(tenantId); + } + try { + chain.doFilter(request, response); + } finally { + // 清理 + TenantContextHolder.clear(); + } + } + +} diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/resources/META-INF/spring.factories b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/resources/META-INF/spring.factories new file mode 100644 index 00000000..5b27b46c --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-tenant/src/main/resources/META-INF/spring.factories @@ -0,0 +1,2 @@ +org.springframework.boot.autoconfigure.EnableAutoConfiguration=\ + com.ruoyi.plugin.tenant.config.TenantConfiguration diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/META-INF/spring.factories b/ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/META-INF/spring.factories new file mode 100644 index 00000000..5b27b46c --- /dev/null +++ b/ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/META-INF/spring.factories @@ -0,0 +1,2 @@ +org.springframework.boot.autoconfigure.EnableAutoConfiguration=\ + com.ruoyi.plugin.tenant.config.TenantConfiguration diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/config/TenantConfiguration.class b/ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/config/TenantConfiguration.class new file mode 100644 index 0000000000000000000000000000000000000000..47145a6ab07ca690c90dbc7c84874da446e2ab31 GIT binary patch literal 3572 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 N2m7lV;Q)s?0+ueHgopqD literal 0 HcmV?d00001 diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/aop/TenantIgnoreAspect.class b/ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/aop/TenantIgnoreAspect.class new file mode 100644 index 0000000000000000000000000000000000000000..7d80da96f2949a66eebdd14f3ff7c8c8519fd930 GIT binary patch literal 1577 zcmbVMTT|0O6#kZ8LQ4=RSSW&sLTy2a3Mf)UE`sP(V1(g;wcOsX zw)n0^5;E4gwap#V=WD{Kw{N-(Q$<}KXiilgn_9)H8m6sz!sfQ8>CzDzmlbU*;9nXx z2^QRn&^_NbcVyKrF_^g^G#)6*n#FBH+i;{VM9H*`SJJdMXnBRJ`6=X!Wq!y*xb@v~ z=rmD2g=5J6n_C7od`}oZgLcJ07!uMdomzJ`gtaK8CAdw=Xx;R<{`NJm1i7UeDwtOA zM8S*#QYV*>bW!n4X}bZRc~iAL^FVByu1N)1ux;t_K+Q08@mc&#uPp>gC&<_#x2jay z%zRy&YkX*)p?OxfY7LUylvPI;E2iHL-4_cw?kD?F=V9$tsAyF%tKuo<=!q(xVP3@o z7U|V0mhhZm;y;>7!Lo`Kyik!u3q$+)`cruR3|!V|3}aVv&d^c!R4XY%XI-|n@5p!I zAWi=jS(mmaj=b8uEQQ=EHIHzp7^ZvLLo}^B3`$VJ^*x4kZl|%^>ur^Ag?DO{&i$eD z{Ha$e;jl4ub&8F$aJF&KaYiB$Xq$|p4Y8KEDV_AnzIF)FC77X18LiqIRF3v literal 0 HcmV?d00001 diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/context/TenantContextHolder.class b/ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/context/TenantContextHolder.class new file mode 100644 index 0000000000000000000000000000000000000000..4c860985382b6a571d3ed50b8fb17b4eaaaa7ec9 GIT binary patch literal 1669 zcmbVMU2oe|7=BKZ#F}A^2nvI$k7^5xSN0)73S&G_h+PFcEbCh|wc5$6) z`~z+fcZdrlF5vP}w(n+Q@pE zD8EQj`Rl~gIzGb|IG9E3)J%a2Y*CDu$Vlt}1d8s)js%PPk-)ISa)-QFaJ+uCjn$wV zI`W#UYI}H(%1o;PF>x42Oq4L8;i`!$t_h5$lcQ2K@d{1|98ax{uIGIixU6|tedNgX z$PIkX(jGoe{Pp;Y?H|A1{^tAV-+lY{=e_68zS{n>r<`6laRW6I2bJd!Ox(m-6Ls7& z(ZCUbvg)_(xh=b8TT$d$&CvEY*4!wHha(+mfeZh4G}zTNr{&fI=|lpPsVOtt99~X{ zz=2)ElW;g;li5D5Mxo1BMzt}=UcYeL=cwE9)K$N=d^?&ex2xgrlm7V2JT)l4q87a6W`0_v|&-qeNJI3O%mvNjYT}=TkihL@*0gO!yZ6WtN@;}ikkSZJ_{^S6E7rB^m;4y}8R{p@qQxxu% zv>uATRTMK9GfT*Gf12y-4E7AunHt2AXPT$*D(_$8pQ-$Vl18jf?3sR4=UD`mQjt+r zwxFk38N31BfQIRTqEbl|b(HBT$tb7joDoq6`ZBEj&sycG)~9ffRRn?+l0Xa zZNdD3plpHWNSPjVLs8X_6c&^#StBa`gfhws0gLR;MRw&9Bew?BnZCElrx?B0!`Lqv V8tjZzA}PTb49E%zrRBad7|u literal 0 HcmV?d00001 diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/db/TenantBaseDO.class b/ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/db/TenantBaseDO.class new file mode 100644 index 0000000000000000000000000000000000000000..29cef21e02dd44870fcbb46a6ced6c650a980d33 GIT binary patch literal 1612 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 literal 0 HcmV?d00001 diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/util/TenantUtils.class b/ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/util/TenantUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..5b74ba79f36bd328b38e73da6e89053aa2302259 GIT binary patch literal 1696 zcmb7EU31$+6g_J@QEaQJ>y(s|QXtKTf_1BBv{{XO#?+ZACS^;aY^zvZ= zAK_yyeWI83BCg?61D{RddJ#8pvw#iUD&TY6Ht>Z&sZrm&Ti<)wskb&;orl|Z1H4x29UaK9 zV;^`#%FV#By?r}$wLS!Ms^|6vtfmtjSz#1BcCBMC>bkzAq|Y3d6NJ)=lUeVuMYmo}Be+&*NpDVx}br{U06;C$-zWdwm@;2kDa z=n#2o+g+da3CyG%?gW7+ZC~G1TiMQ!JN9whtbs2LY#O*@V2iQ3QjKHXNP0N6`Z7H6 zq_TQS9a{}9?F879KSi>yHda!JhIJcZa>kp858oWb0B2ih2T|zAd#;{MDRsGPdgdn1 zp=_XTVhQ(5e1!%JH?fVcO?-oDnzna}21ZFD_(hg(V6#Hpv%>0Mmn#y_F zi~`T5Ek;Sn|b$|&i z3^2L+42Hl}0ySz+P`a5LDHn5pK4mpUT*5K~EMq0!v(U-chy(8+hdD~`vii$R{yuHz zxn98nzh%lb%IjE+i>%U`h0tq`yOu7|lRsJh*)J2*xvF@N{#o7)*h1zX2FV$iH*h6h z7a$BQn(T0iCrO769W&azTFyU1A%o`X0L5QW;ATESPW;Nlr0PO!xJGv?u4p!KM1{TS cYb&ufJqMlLMCLgz7^w0`H&1MpSO!=B1xu2hbN~PV literal 0 HcmV?d00001 diff --git a/ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/web/TenantContextWebFilter.class b/ruoyi-plugin/ruoyi-springboot-starter-tenant/target/classes/com/ruoyi/plugin/tenant/core/web/TenantContextWebFilter.class new file mode 100644 index 0000000000000000000000000000000000000000..579e373b4df702cc35052c6d261a6d8ec1444749 GIT binary patch literal 2165 zcmbtVOMBBs6g?9^CfFh&q@^_}w9rs&@(?K$hz$Xn0&(jQh?CHEr^th?B1^6$lkije z3%X|0CSA06?~g1q%X+d|^W%x#-rEz-YyC)Wddj zQ+aE~rcKGoifbD7hT&P^ITj{^hUE*)S4_95du{iHrMK*M-E#CmIffJHrt2wvOKs|F zo%5pW1Zq2YqBieacA&hHz=@h0?aRFK9K#l9<}1G#&y8)}SKf151$rY0TKW<{)kv_a zer+p1C=FEeTdw1)!+Ozsiw(naN@32@i~I>(9OyB6SYX&wk#m8;(Z*p2q|ES9fr&%! z^hQMAQSr((R;Qz=+lEuuE3Q*#!A5;$wmmC&5f@XSRB^q!?zcS4sXz6MrrL77XDnW% z7WKf4Dz8`jjp^@rH89L)%SNlyQ;Ay=3lg^lGWWJk)e0<@P~winU5cu1+cOozB_bw{ z?&)mUI}OJ%BJr(;MSLf5Ps4pYpkob7C`){=;RjSCmNh)Yio~N(@3F+HhAP%1)-`P4 ziH09BB5?gZ!6k6zsJet`eA+>uzT%kbk@8|VMNJSG2@~j+obW!%&0CdMV7%*Wt>L*_ z(J-Cu$9D|tgL?0qS*~7Q>5_gvPIsfaJP5wS%EzatDcGDwo>BWG=Ofq`)3d$Gcsm1b#zy2kAc`i^*Ja55qZH z49`qW@8VbjJIMTwY%$F>m;UD^Unr6E%Jg0)8HQ^DVPvyR1ejoipK*5v1$>Soxzd6H>?7I{48KYp0O0<{@soI$udg-<$Jt>RPU_*N)8O|J=RR_OT*^UUG_&eD4d jU+{hm$yb<@I45!b*s{cz5|b3CX?y`+(b+|whLHasEG2wq literal 0 HcmV?d00001 diff --git a/ruoyi-quartz/pom.xml b/ruoyi-quartz/pom.xml new file mode 100644 index 00000000..863fcabd --- /dev/null +++ b/ruoyi-quartz/pom.xml @@ -0,0 +1,40 @@ + + + + ruoyi + com.ruoyi + 3.8.4 + + 4.0.0 + + ruoyi-quartz + + + quartz定时任务 + + + + + + + org.quartz-scheduler + quartz + + + com.mchange + c3p0 + + + + + + + com.ruoyi + ruoyi-common + + + + + diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/config/ScheduleConfig.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/config/ScheduleConfig.java new file mode 100644 index 00000000..a558170c --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/config/ScheduleConfig.java @@ -0,0 +1,57 @@ +//package com.ruoyi.quartz.config; +// +//import org.springframework.context.annotation.Bean; +//import org.springframework.context.annotation.Configuration; +//import org.springframework.scheduling.quartz.SchedulerFactoryBean; +//import javax.sql.DataSource; +//import java.util.Properties; +// +///** +// * 定时任务配置(单机部署建议删除此类和qrtz数据库表,默认走内存会最高效) +// * +// * @author ruoyi +// */ +//@Configuration +//public class ScheduleConfig +//{ +// @Bean +// public SchedulerFactoryBean schedulerFactoryBean(DataSource dataSource) +// { +// SchedulerFactoryBean factory = new SchedulerFactoryBean(); +// factory.setDataSource(dataSource); +// +// // quartz参数 +// Properties prop = new Properties(); +// prop.put("org.quartz.scheduler.instanceName", "RuoyiScheduler"); +// prop.put("org.quartz.scheduler.instanceId", "AUTO"); +// // 线程池配置 +// prop.put("org.quartz.threadPool.class", "org.quartz.simpl.SimpleThreadPool"); +// prop.put("org.quartz.threadPool.threadCount", "20"); +// prop.put("org.quartz.threadPool.threadPriority", "5"); +// // JobStore配置 +// prop.put("org.quartz.jobStore.class", "org.springframework.scheduling.quartz.LocalDataSourceJobStore"); +// // 集群配置 +// prop.put("org.quartz.jobStore.isClustered", "true"); +// prop.put("org.quartz.jobStore.clusterCheckinInterval", "15000"); +// prop.put("org.quartz.jobStore.maxMisfiresToHandleAtATime", "1"); +// prop.put("org.quartz.jobStore.txIsolationLevelSerializable", "true"); +// +// // sqlserver 启用 +// // prop.put("org.quartz.jobStore.selectWithLockSQL", "SELECT * FROM {0}LOCKS UPDLOCK WHERE LOCK_NAME = ?"); +// prop.put("org.quartz.jobStore.misfireThreshold", "12000"); +// prop.put("org.quartz.jobStore.tablePrefix", "QRTZ_"); +// factory.setQuartzProperties(prop); +// +// factory.setSchedulerName("RuoyiScheduler"); +// // 延时启动 +// factory.setStartupDelay(1); +// factory.setApplicationContextSchedulerContextKey("applicationContextKey"); +// // 可选,QuartzScheduler +// // 启动时更新己存在的Job,这样就不用每次修改targetObject后删除qrtz_job_details表对应记录了 +// factory.setOverwriteExistingJobs(true); +// // 设置自动启动,默认为true +// factory.setAutoStartup(true); +// +// return factory; +// } +//} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/controller/SysJobController.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/controller/SysJobController.java new file mode 100644 index 00000000..a3f3e1c6 --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/controller/SysJobController.java @@ -0,0 +1,148 @@ +package com.ruoyi.quartz.controller; + +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.page.TableDataInfo; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.exception.job.TaskException; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.poi.ExcelUtil; +import com.ruoyi.quartz.domain.SysJob; +import com.ruoyi.quartz.service.ISysJobService; +import com.ruoyi.quartz.util.CronUtils; +import com.ruoyi.quartz.util.ScheduleUtils; +import org.quartz.SchedulerException; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.web.bind.annotation.*; + +import javax.servlet.http.HttpServletResponse; +import java.util.List; + +/** + * 调度任务信息操作处理 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/monitor/job") +public class SysJobController extends BaseController { + @Autowired + private ISysJobService jobService; + + /** + * 查询定时任务列表 + */ + @PreAuthorize("@ss.hasPermi('monitor:job:list')") + @GetMapping("/list") + public TableDataInfo list(SysJob sysJob) { + startPage(); + List list = jobService.selectJobList(sysJob); + return getDataTable(list); + } + + /** + * 导出定时任务列表 + */ + @PreAuthorize("@ss.hasPermi('monitor:job:export')") + @Log(title = "定时任务", businessType = BusinessType.EXPORT) + @PostMapping("/export") + public void export(HttpServletResponse response, SysJob sysJob) { + List list = jobService.selectJobList(sysJob); + ExcelUtil util = new ExcelUtil(SysJob.class); + util.exportExcel(response, list, "定时任务"); + } + + /** + * 获取定时任务详细信息 + */ + @PreAuthorize("@ss.hasPermi('monitor:job:query')") + @GetMapping(value = "/{jobId}") + public AjaxResult getInfo(@PathVariable("jobId") Long jobId) { + return success(jobService.selectJobById(jobId)); + } + + /** + * 新增定时任务 + */ + @PreAuthorize("@ss.hasPermi('monitor:job:add')") + @Log(title = "定时任务", businessType = BusinessType.INSERT) + @PostMapping + public AjaxResult add(@RequestBody SysJob job) throws SchedulerException, TaskException { + if (!CronUtils.isValid(job.getCronExpression())) { + return error("新增任务'" + job.getJobName() + "'失败,Cron表达式不正确"); + } else if (StringUtils.containsIgnoreCase(job.getInvokeTarget(), Constants.LOOKUP_RMI)) { + return error("新增任务'" + job.getJobName() + "'失败,目标字符串不允许'rmi'调用"); + } else if (StringUtils.containsAnyIgnoreCase(job.getInvokeTarget(), new String[]{Constants.LOOKUP_LDAP, Constants.LOOKUP_LDAPS})) { + return error("新增任务'" + job.getJobName() + "'失败,目标字符串不允许'ldap(s)'调用"); + } else if (StringUtils.containsAnyIgnoreCase(job.getInvokeTarget(), new String[]{Constants.HTTP, Constants.HTTPS})) { + return error("新增任务'" + job.getJobName() + "'失败,目标字符串不允许'http(s)'调用"); + } else if (StringUtils.containsAnyIgnoreCase(job.getInvokeTarget(), Constants.JOB_ERROR_STR)) { + return error("新增任务'" + job.getJobName() + "'失败,目标字符串存在违规"); + } else if (!ScheduleUtils.whiteList(job.getInvokeTarget())) { + return error("新增任务'" + job.getJobName() + "'失败,目标字符串不在白名单内"); + } + job.setCreateBy(getUsername()); + return toAjax(jobService.insertJob(job)); + } + + /** + * 修改定时任务 + */ + @PreAuthorize("@ss.hasPermi('monitor:job:edit')") + @Log(title = "定时任务", businessType = BusinessType.UPDATE) + @PutMapping + public AjaxResult edit(@RequestBody SysJob job) throws SchedulerException, TaskException { + if (!CronUtils.isValid(job.getCronExpression())) { + return error("修改任务'" + job.getJobName() + "'失败,Cron表达式不正确"); + } else if (StringUtils.containsIgnoreCase(job.getInvokeTarget(), Constants.LOOKUP_RMI)) { + return error("修改任务'" + job.getJobName() + "'失败,目标字符串不允许'rmi'调用"); + } else if (StringUtils.containsAnyIgnoreCase(job.getInvokeTarget(), new String[]{Constants.LOOKUP_LDAP, Constants.LOOKUP_LDAPS})) { + return error("修改任务'" + job.getJobName() + "'失败,目标字符串不允许'ldap(s)'调用"); + } else if (StringUtils.containsAnyIgnoreCase(job.getInvokeTarget(), new String[]{Constants.HTTP, Constants.HTTPS})) { + return error("修改任务'" + job.getJobName() + "'失败,目标字符串不允许'http(s)'调用"); + } else if (StringUtils.containsAnyIgnoreCase(job.getInvokeTarget(), Constants.JOB_ERROR_STR)) { + return error("修改任务'" + job.getJobName() + "'失败,目标字符串存在违规"); + } else if (!ScheduleUtils.whiteList(job.getInvokeTarget())) { + return error("修改任务'" + job.getJobName() + "'失败,目标字符串不在白名单内"); + } + job.setUpdateBy(getUsername()); + return toAjax(jobService.updateJob(job)); + } + + /** + * 定时任务状态修改 + */ + @PreAuthorize("@ss.hasPermi('monitor:job:changeStatus')") + @Log(title = "定时任务", businessType = BusinessType.UPDATE) + @PutMapping("/changeStatus") + public AjaxResult changeStatus(@RequestBody SysJob job) throws SchedulerException { + SysJob newJob = jobService.selectJobById(job.getJobId()); + newJob.setStatus(job.getStatus()); + return toAjax(jobService.changeStatus(newJob)); + } + + /** + * 定时任务立即执行一次 + */ + @PreAuthorize("@ss.hasPermi('monitor:job:changeStatus')") + @Log(title = "定时任务", businessType = BusinessType.UPDATE) + @PutMapping("/run") + public AjaxResult run(@RequestBody SysJob job) throws SchedulerException { + boolean result = jobService.run(job); + return result ? success() : error("任务不存在或已过期!"); + } + + /** + * 删除定时任务 + */ + @PreAuthorize("@ss.hasPermi('monitor:job:remove')") + @Log(title = "定时任务", businessType = BusinessType.DELETE) + @DeleteMapping("/{jobIds}") + public AjaxResult remove(@PathVariable Long[] jobIds) throws SchedulerException, TaskException { + jobService.deleteJobByIds(jobIds); + return success(); + } +} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/controller/SysJobLogController.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/controller/SysJobLogController.java new file mode 100644 index 00000000..24acf418 --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/controller/SysJobLogController.java @@ -0,0 +1,82 @@ +package com.ruoyi.quartz.controller; + +import com.ruoyi.common.annotation.Log; +import com.ruoyi.common.core.controller.BaseController; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.core.page.TableDataInfo; +import com.ruoyi.common.enums.BusinessType; +import com.ruoyi.common.utils.poi.ExcelUtil; +import com.ruoyi.quartz.domain.SysJobLog; +import com.ruoyi.quartz.service.ISysJobLogService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.web.bind.annotation.*; + +import javax.servlet.http.HttpServletResponse; +import java.util.List; + +/** + * 调度日志操作处理 + * + * @author ruoyi + */ +@RestController +@RequestMapping("/monitor/jobLog") +public class SysJobLogController extends BaseController { + @Autowired + private ISysJobLogService jobLogService; + + /** + * 查询定时任务调度日志列表 + */ + @PreAuthorize("@ss.hasPermi('monitor:job:list')") + @GetMapping("/list") + public TableDataInfo list(SysJobLog sysJobLog) { + startPage(); + List list = jobLogService.selectJobLogList(sysJobLog); + return getDataTable(list); + } + + /** + * 导出定时任务调度日志列表 + */ + @PreAuthorize("@ss.hasPermi('monitor:job:export')") + @Log(title = "任务调度日志", businessType = BusinessType.EXPORT) + @PostMapping("/export") + public void export(HttpServletResponse response, SysJobLog sysJobLog) { + List list = jobLogService.selectJobLogList(sysJobLog); + ExcelUtil util = new ExcelUtil(SysJobLog.class); + util.exportExcel(response, list, "调度日志"); + } + + /** + * 根据调度编号获取详细信息 + */ + @PreAuthorize("@ss.hasPermi('monitor:job:query')") + @GetMapping(value = "/{configId}") + public AjaxResult getInfo(@PathVariable Long jobLogId) { + return success(jobLogService.selectJobLogById(jobLogId)); + } + + + /** + * 删除定时任务调度日志 + */ + @PreAuthorize("@ss.hasPermi('monitor:job:remove')") + @Log(title = "定时任务调度日志", businessType = BusinessType.DELETE) + @DeleteMapping("/{jobLogIds}") + public AjaxResult remove(@PathVariable Long[] jobLogIds) { + return toAjax(jobLogService.deleteJobLogByIds(jobLogIds)); + } + + /** + * 清空定时任务调度日志 + */ + @PreAuthorize("@ss.hasPermi('monitor:job:remove')") + @Log(title = "调度日志", businessType = BusinessType.CLEAN) + @DeleteMapping("/clean") + public AjaxResult clean() { + jobLogService.cleanJobLog(); + return success(); + } +} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/domain/SysJob.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/domain/SysJob.java new file mode 100644 index 00000000..1f496959 --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/domain/SysJob.java @@ -0,0 +1,171 @@ +package com.ruoyi.quartz.domain; + +import java.util.Date; +import javax.validation.constraints.NotBlank; +import javax.validation.constraints.Size; +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import com.fasterxml.jackson.annotation.JsonFormat; +import com.ruoyi.common.annotation.Excel; +import com.ruoyi.common.annotation.Excel.ColumnType; +import com.ruoyi.common.constant.ScheduleConstants; +import com.ruoyi.common.core.domain.BaseEntity; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.quartz.util.CronUtils; + +/** + * 定时任务调度表 sys_job + * + * @author ruoyi + */ +public class SysJob extends BaseEntity +{ + private static final long serialVersionUID = 1L; + + /** 任务ID */ + @Excel(name = "任务序号", cellType = ColumnType.NUMERIC) + private Long jobId; + + /** 任务名称 */ + @Excel(name = "任务名称") + private String jobName; + + /** 任务组名 */ + @Excel(name = "任务组名") + private String jobGroup; + + /** 调用目标字符串 */ + @Excel(name = "调用目标字符串") + private String invokeTarget; + + /** cron执行表达式 */ + @Excel(name = "执行表达式 ") + private String cronExpression; + + /** cron计划策略 */ + @Excel(name = "计划策略 ", readConverterExp = "0=默认,1=立即触发执行,2=触发一次执行,3=不触发立即执行") + private String misfirePolicy = ScheduleConstants.MISFIRE_DEFAULT; + + /** 是否并发执行(0允许 1禁止) */ + @Excel(name = "并发执行", readConverterExp = "0=允许,1=禁止") + private String concurrent; + + /** 任务状态(0正常 1暂停) */ + @Excel(name = "任务状态", readConverterExp = "0=正常,1=暂停") + private String status; + + public Long getJobId() + { + return jobId; + } + + public void setJobId(Long jobId) + { + this.jobId = jobId; + } + + @NotBlank(message = "任务名称不能为空") + @Size(min = 0, max = 64, message = "任务名称不能超过64个字符") + public String getJobName() + { + return jobName; + } + + public void setJobName(String jobName) + { + this.jobName = jobName; + } + + public String getJobGroup() + { + return jobGroup; + } + + public void setJobGroup(String jobGroup) + { + this.jobGroup = jobGroup; + } + + @NotBlank(message = "调用目标字符串不能为空") + @Size(min = 0, max = 500, message = "调用目标字符串长度不能超过500个字符") + public String getInvokeTarget() + { + return invokeTarget; + } + + public void setInvokeTarget(String invokeTarget) + { + this.invokeTarget = invokeTarget; + } + + @NotBlank(message = "Cron执行表达式不能为空") + @Size(min = 0, max = 255, message = "Cron执行表达式不能超过255个字符") + public String getCronExpression() + { + return cronExpression; + } + + public void setCronExpression(String cronExpression) + { + this.cronExpression = cronExpression; + } + + @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") + public Date getNextValidTime() + { + if (StringUtils.isNotEmpty(cronExpression)) + { + return CronUtils.getNextExecution(cronExpression); + } + return null; + } + + public String getMisfirePolicy() + { + return misfirePolicy; + } + + public void setMisfirePolicy(String misfirePolicy) + { + this.misfirePolicy = misfirePolicy; + } + + public String getConcurrent() + { + return concurrent; + } + + public void setConcurrent(String concurrent) + { + this.concurrent = concurrent; + } + + public String getStatus() + { + return status; + } + + public void setStatus(String status) + { + this.status = status; + } + + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) + .append("jobId", getJobId()) + .append("jobName", getJobName()) + .append("jobGroup", getJobGroup()) + .append("cronExpression", getCronExpression()) + .append("nextValidTime", getNextValidTime()) + .append("misfirePolicy", getMisfirePolicy()) + .append("concurrent", getConcurrent()) + .append("status", getStatus()) + .append("createBy", getCreateBy()) + .append("createTime", getCreateTime()) + .append("updateBy", getUpdateBy()) + .append("updateTime", getUpdateTime()) + .append("remark", getRemark()) + .toString(); + } +} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/domain/SysJobLog.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/domain/SysJobLog.java new file mode 100644 index 00000000..121c0359 --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/domain/SysJobLog.java @@ -0,0 +1,155 @@ +package com.ruoyi.quartz.domain; + +import java.util.Date; +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import com.ruoyi.common.annotation.Excel; +import com.ruoyi.common.core.domain.BaseEntity; + +/** + * 定时任务调度日志表 sys_job_log + * + * @author ruoyi + */ +public class SysJobLog extends BaseEntity +{ + private static final long serialVersionUID = 1L; + + /** ID */ + @Excel(name = "日志序号") + private Long jobLogId; + + /** 任务名称 */ + @Excel(name = "任务名称") + private String jobName; + + /** 任务组名 */ + @Excel(name = "任务组名") + private String jobGroup; + + /** 调用目标字符串 */ + @Excel(name = "调用目标字符串") + private String invokeTarget; + + /** 日志信息 */ + @Excel(name = "日志信息") + private String jobMessage; + + /** 执行状态(0正常 1失败) */ + @Excel(name = "执行状态", readConverterExp = "0=正常,1=失败") + private String status; + + /** 异常信息 */ + @Excel(name = "异常信息") + private String exceptionInfo; + + /** 开始时间 */ + private Date startTime; + + /** 停止时间 */ + private Date stopTime; + + public Long getJobLogId() + { + return jobLogId; + } + + public void setJobLogId(Long jobLogId) + { + this.jobLogId = jobLogId; + } + + public String getJobName() + { + return jobName; + } + + public void setJobName(String jobName) + { + this.jobName = jobName; + } + + public String getJobGroup() + { + return jobGroup; + } + + public void setJobGroup(String jobGroup) + { + this.jobGroup = jobGroup; + } + + public String getInvokeTarget() + { + return invokeTarget; + } + + public void setInvokeTarget(String invokeTarget) + { + this.invokeTarget = invokeTarget; + } + + public String getJobMessage() + { + return jobMessage; + } + + public void setJobMessage(String jobMessage) + { + this.jobMessage = jobMessage; + } + + public String getStatus() + { + return status; + } + + public void setStatus(String status) + { + this.status = status; + } + + public String getExceptionInfo() + { + return exceptionInfo; + } + + public void setExceptionInfo(String exceptionInfo) + { + this.exceptionInfo = exceptionInfo; + } + + public Date getStartTime() + { + return startTime; + } + + public void setStartTime(Date startTime) + { + this.startTime = startTime; + } + + public Date getStopTime() + { + return stopTime; + } + + public void setStopTime(Date stopTime) + { + this.stopTime = stopTime; + } + + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) + .append("jobLogId", getJobLogId()) + .append("jobName", getJobName()) + .append("jobGroup", getJobGroup()) + .append("jobMessage", getJobMessage()) + .append("status", getStatus()) + .append("exceptionInfo", getExceptionInfo()) + .append("startTime", getStartTime()) + .append("stopTime", getStopTime()) + .toString(); + } +} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/mapper/SysJobLogMapper.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/mapper/SysJobLogMapper.java new file mode 100644 index 00000000..727d9169 --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/mapper/SysJobLogMapper.java @@ -0,0 +1,64 @@ +package com.ruoyi.quartz.mapper; + +import java.util.List; +import com.ruoyi.quartz.domain.SysJobLog; + +/** + * 调度任务日志信息 数据层 + * + * @author ruoyi + */ +public interface SysJobLogMapper +{ + /** + * 获取quartz调度器日志的计划任务 + * + * @param jobLog 调度日志信息 + * @return 调度任务日志集合 + */ + public List selectJobLogList(SysJobLog jobLog); + + /** + * 查询所有调度任务日志 + * + * @return 调度任务日志列表 + */ + public List selectJobLogAll(); + + /** + * 通过调度任务日志ID查询调度信息 + * + * @param jobLogId 调度任务日志ID + * @return 调度任务日志对象信息 + */ + public SysJobLog selectJobLogById(Long jobLogId); + + /** + * 新增任务日志 + * + * @param jobLog 调度日志信息 + * @return 结果 + */ + public int insertJobLog(SysJobLog jobLog); + + /** + * 批量删除调度日志信息 + * + * @param logIds 需要删除的数据ID + * @return 结果 + */ + public int deleteJobLogByIds(Long[] logIds); + + /** + * 删除任务日志 + * + * @param jobId 调度日志ID + * @return 结果 + */ + public int deleteJobLogById(Long jobId); + + /** + * 清空任务日志 + */ + public void cleanJobLog(); +} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/mapper/SysJobMapper.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/mapper/SysJobMapper.java new file mode 100644 index 00000000..20f45dbb --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/mapper/SysJobMapper.java @@ -0,0 +1,67 @@ +package com.ruoyi.quartz.mapper; + +import java.util.List; +import com.ruoyi.quartz.domain.SysJob; + +/** + * 调度任务信息 数据层 + * + * @author ruoyi + */ +public interface SysJobMapper +{ + /** + * 查询调度任务日志集合 + * + * @param job 调度信息 + * @return 操作日志集合 + */ + public List selectJobList(SysJob job); + + /** + * 查询所有调度任务 + * + * @return 调度任务列表 + */ + public List selectJobAll(); + + /** + * 通过调度ID查询调度任务信息 + * + * @param jobId 调度ID + * @return 角色对象信息 + */ + public SysJob selectJobById(Long jobId); + + /** + * 通过调度ID删除调度任务信息 + * + * @param jobId 调度ID + * @return 结果 + */ + public int deleteJobById(Long jobId); + + /** + * 批量删除调度任务信息 + * + * @param ids 需要删除的数据ID + * @return 结果 + */ + public int deleteJobByIds(Long[] ids); + + /** + * 修改调度任务信息 + * + * @param job 调度任务信息 + * @return 结果 + */ + public int updateJob(SysJob job); + + /** + * 新增调度任务信息 + * + * @param job 调度任务信息 + * @return 结果 + */ + public int insertJob(SysJob job); +} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/service/ISysJobLogService.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/service/ISysJobLogService.java new file mode 100644 index 00000000..8546792a --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/service/ISysJobLogService.java @@ -0,0 +1,56 @@ +package com.ruoyi.quartz.service; + +import java.util.List; +import com.ruoyi.quartz.domain.SysJobLog; + +/** + * 定时任务调度日志信息信息 服务层 + * + * @author ruoyi + */ +public interface ISysJobLogService +{ + /** + * 获取quartz调度器日志的计划任务 + * + * @param jobLog 调度日志信息 + * @return 调度任务日志集合 + */ + public List selectJobLogList(SysJobLog jobLog); + + /** + * 通过调度任务日志ID查询调度信息 + * + * @param jobLogId 调度任务日志ID + * @return 调度任务日志对象信息 + */ + public SysJobLog selectJobLogById(Long jobLogId); + + /** + * 新增任务日志 + * + * @param jobLog 调度日志信息 + */ + public void addJobLog(SysJobLog jobLog); + + /** + * 批量删除调度日志信息 + * + * @param logIds 需要删除的日志ID + * @return 结果 + */ + public int deleteJobLogByIds(Long[] logIds); + + /** + * 删除任务日志 + * + * @param jobId 调度日志ID + * @return 结果 + */ + public int deleteJobLogById(Long jobId); + + /** + * 清空任务日志 + */ + public void cleanJobLog(); +} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/service/ISysJobService.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/service/ISysJobService.java new file mode 100644 index 00000000..437ade82 --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/service/ISysJobService.java @@ -0,0 +1,102 @@ +package com.ruoyi.quartz.service; + +import java.util.List; +import org.quartz.SchedulerException; +import com.ruoyi.common.exception.job.TaskException; +import com.ruoyi.quartz.domain.SysJob; + +/** + * 定时任务调度信息信息 服务层 + * + * @author ruoyi + */ +public interface ISysJobService +{ + /** + * 获取quartz调度器的计划任务 + * + * @param job 调度信息 + * @return 调度任务集合 + */ + public List selectJobList(SysJob job); + + /** + * 通过调度任务ID查询调度信息 + * + * @param jobId 调度任务ID + * @return 调度任务对象信息 + */ + public SysJob selectJobById(Long jobId); + + /** + * 暂停任务 + * + * @param job 调度信息 + * @return 结果 + */ + public int pauseJob(SysJob job) throws SchedulerException; + + /** + * 恢复任务 + * + * @param job 调度信息 + * @return 结果 + */ + public int resumeJob(SysJob job) throws SchedulerException; + + /** + * 删除任务后,所对应的trigger也将被删除 + * + * @param job 调度信息 + * @return 结果 + */ + public int deleteJob(SysJob job) throws SchedulerException; + + /** + * 批量删除调度信息 + * + * @param jobIds 需要删除的任务ID + * @return 结果 + */ + public void deleteJobByIds(Long[] jobIds) throws SchedulerException; + + /** + * 任务调度状态修改 + * + * @param job 调度信息 + * @return 结果 + */ + public int changeStatus(SysJob job) throws SchedulerException; + + /** + * 立即运行任务 + * + * @param job 调度信息 + * @return 结果 + */ + public boolean run(SysJob job) throws SchedulerException; + + /** + * 新增任务 + * + * @param job 调度信息 + * @return 结果 + */ + public int insertJob(SysJob job) throws SchedulerException, TaskException; + + /** + * 更新任务 + * + * @param job 调度信息 + * @return 结果 + */ + public int updateJob(SysJob job) throws SchedulerException, TaskException; + + /** + * 校验cron表达式是否有效 + * + * @param cronExpression 表达式 + * @return 结果 + */ + public boolean checkCronExpressionIsValid(String cronExpression); +} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/service/impl/SysJobLogServiceImpl.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/service/impl/SysJobLogServiceImpl.java new file mode 100644 index 00000000..812eed77 --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/service/impl/SysJobLogServiceImpl.java @@ -0,0 +1,87 @@ +package com.ruoyi.quartz.service.impl; + +import java.util.List; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +import com.ruoyi.quartz.domain.SysJobLog; +import com.ruoyi.quartz.mapper.SysJobLogMapper; +import com.ruoyi.quartz.service.ISysJobLogService; + +/** + * 定时任务调度日志信息 服务层 + * + * @author ruoyi + */ +@Service +public class SysJobLogServiceImpl implements ISysJobLogService +{ + @Autowired + private SysJobLogMapper jobLogMapper; + + /** + * 获取quartz调度器日志的计划任务 + * + * @param jobLog 调度日志信息 + * @return 调度任务日志集合 + */ + @Override + public List selectJobLogList(SysJobLog jobLog) + { + return jobLogMapper.selectJobLogList(jobLog); + } + + /** + * 通过调度任务日志ID查询调度信息 + * + * @param jobLogId 调度任务日志ID + * @return 调度任务日志对象信息 + */ + @Override + public SysJobLog selectJobLogById(Long jobLogId) + { + return jobLogMapper.selectJobLogById(jobLogId); + } + + /** + * 新增任务日志 + * + * @param jobLog 调度日志信息 + */ + @Override + public void addJobLog(SysJobLog jobLog) + { + jobLogMapper.insertJobLog(jobLog); + } + + /** + * 批量删除调度日志信息 + * + * @param logIds 需要删除的数据ID + * @return 结果 + */ + @Override + public int deleteJobLogByIds(Long[] logIds) + { + return jobLogMapper.deleteJobLogByIds(logIds); + } + + /** + * 删除任务日志 + * + * @param jobId 调度日志ID + */ + @Override + public int deleteJobLogById(Long jobId) + { + return jobLogMapper.deleteJobLogById(jobId); + } + + /** + * 清空任务日志 + */ + @Override + public void cleanJobLog() + { + jobLogMapper.cleanJobLog(); + } +} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/service/impl/SysJobServiceImpl.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/service/impl/SysJobServiceImpl.java new file mode 100644 index 00000000..77fdbb57 --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/service/impl/SysJobServiceImpl.java @@ -0,0 +1,261 @@ +package com.ruoyi.quartz.service.impl; + +import java.util.List; +import javax.annotation.PostConstruct; +import org.quartz.JobDataMap; +import org.quartz.JobKey; +import org.quartz.Scheduler; +import org.quartz.SchedulerException; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; +import com.ruoyi.common.constant.ScheduleConstants; +import com.ruoyi.common.exception.job.TaskException; +import com.ruoyi.quartz.domain.SysJob; +import com.ruoyi.quartz.mapper.SysJobMapper; +import com.ruoyi.quartz.service.ISysJobService; +import com.ruoyi.quartz.util.CronUtils; +import com.ruoyi.quartz.util.ScheduleUtils; + +/** + * 定时任务调度信息 服务层 + * + * @author ruoyi + */ +@Service +public class SysJobServiceImpl implements ISysJobService +{ + @Autowired + private Scheduler scheduler; + + @Autowired + private SysJobMapper jobMapper; + + /** + * 项目启动时,初始化定时器 主要是防止手动修改数据库导致未同步到定时任务处理(注:不能手动修改数据库ID和任务组名,否则会导致脏数据) + */ + @PostConstruct + public void init() throws SchedulerException, TaskException + { + scheduler.clear(); + List jobList = jobMapper.selectJobAll(); + for (SysJob job : jobList) + { + ScheduleUtils.createScheduleJob(scheduler, job); + } + } + + /** + * 获取quartz调度器的计划任务列表 + * + * @param job 调度信息 + * @return + */ + @Override + public List selectJobList(SysJob job) + { + return jobMapper.selectJobList(job); + } + + /** + * 通过调度任务ID查询调度信息 + * + * @param jobId 调度任务ID + * @return 调度任务对象信息 + */ + @Override + public SysJob selectJobById(Long jobId) + { + return jobMapper.selectJobById(jobId); + } + + /** + * 暂停任务 + * + * @param job 调度信息 + */ + @Override + @Transactional(rollbackFor = Exception.class) + public int pauseJob(SysJob job) throws SchedulerException + { + Long jobId = job.getJobId(); + String jobGroup = job.getJobGroup(); + job.setStatus(ScheduleConstants.Status.PAUSE.getValue()); + int rows = jobMapper.updateJob(job); + if (rows > 0) + { + scheduler.pauseJob(ScheduleUtils.getJobKey(jobId, jobGroup)); + } + return rows; + } + + /** + * 恢复任务 + * + * @param job 调度信息 + */ + @Override + @Transactional(rollbackFor = Exception.class) + public int resumeJob(SysJob job) throws SchedulerException + { + Long jobId = job.getJobId(); + String jobGroup = job.getJobGroup(); + job.setStatus(ScheduleConstants.Status.NORMAL.getValue()); + int rows = jobMapper.updateJob(job); + if (rows > 0) + { + scheduler.resumeJob(ScheduleUtils.getJobKey(jobId, jobGroup)); + } + return rows; + } + + /** + * 删除任务后,所对应的trigger也将被删除 + * + * @param job 调度信息 + */ + @Override + @Transactional(rollbackFor = Exception.class) + public int deleteJob(SysJob job) throws SchedulerException + { + Long jobId = job.getJobId(); + String jobGroup = job.getJobGroup(); + int rows = jobMapper.deleteJobById(jobId); + if (rows > 0) + { + scheduler.deleteJob(ScheduleUtils.getJobKey(jobId, jobGroup)); + } + return rows; + } + + /** + * 批量删除调度信息 + * + * @param jobIds 需要删除的任务ID + * @return 结果 + */ + @Override + @Transactional(rollbackFor = Exception.class) + public void deleteJobByIds(Long[] jobIds) throws SchedulerException + { + for (Long jobId : jobIds) + { + SysJob job = jobMapper.selectJobById(jobId); + deleteJob(job); + } + } + + /** + * 任务调度状态修改 + * + * @param job 调度信息 + */ + @Override + @Transactional(rollbackFor = Exception.class) + public int changeStatus(SysJob job) throws SchedulerException + { + int rows = 0; + String status = job.getStatus(); + if (ScheduleConstants.Status.NORMAL.getValue().equals(status)) + { + rows = resumeJob(job); + } + else if (ScheduleConstants.Status.PAUSE.getValue().equals(status)) + { + rows = pauseJob(job); + } + return rows; + } + + /** + * 立即运行任务 + * + * @param job 调度信息 + */ + @Override + @Transactional(rollbackFor = Exception.class) + public boolean run(SysJob job) throws SchedulerException + { + boolean result = false; + Long jobId = job.getJobId(); + String jobGroup = job.getJobGroup(); + SysJob properties = selectJobById(job.getJobId()); + // 参数 + JobDataMap dataMap = new JobDataMap(); + dataMap.put(ScheduleConstants.TASK_PROPERTIES, properties); + JobKey jobKey = ScheduleUtils.getJobKey(jobId, jobGroup); + if (scheduler.checkExists(jobKey)) + { + result = true; + scheduler.triggerJob(jobKey, dataMap); + } + return result; + } + + /** + * 新增任务 + * + * @param job 调度信息 调度信息 + */ + @Override + @Transactional(rollbackFor = Exception.class) + public int insertJob(SysJob job) throws SchedulerException, TaskException + { + job.setStatus(ScheduleConstants.Status.PAUSE.getValue()); + int rows = jobMapper.insertJob(job); + if (rows > 0) + { + ScheduleUtils.createScheduleJob(scheduler, job); + } + return rows; + } + + /** + * 更新任务的时间表达式 + * + * @param job 调度信息 + */ + @Override + @Transactional(rollbackFor = Exception.class) + public int updateJob(SysJob job) throws SchedulerException, TaskException + { + SysJob properties = selectJobById(job.getJobId()); + int rows = jobMapper.updateJob(job); + if (rows > 0) + { + updateSchedulerJob(job, properties.getJobGroup()); + } + return rows; + } + + /** + * 更新任务 + * + * @param job 任务对象 + * @param jobGroup 任务组名 + */ + public void updateSchedulerJob(SysJob job, String jobGroup) throws SchedulerException, TaskException + { + Long jobId = job.getJobId(); + // 判断是否存在 + JobKey jobKey = ScheduleUtils.getJobKey(jobId, jobGroup); + if (scheduler.checkExists(jobKey)) + { + // 防止创建时存在数据问题 先移除,然后在执行创建操作 + scheduler.deleteJob(jobKey); + } + ScheduleUtils.createScheduleJob(scheduler, job); + } + + /** + * 校验cron表达式是否有效 + * + * @param cronExpression 表达式 + * @return 结果 + */ + @Override + public boolean checkCronExpressionIsValid(String cronExpression) + { + return CronUtils.isValid(cronExpression); + } +} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/task/CommonTask.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/task/CommonTask.java new file mode 100644 index 00000000..f47aa604 --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/task/CommonTask.java @@ -0,0 +1,30 @@ +package com.ruoyi.quartz.task; + +import com.ruoyi.common.config.RuoYiConfig; +import org.apache.commons.io.FileUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.stereotype.Component; + +import java.io.File; +import java.io.IOException; + +/** + * 定时任务调度测试 + * + * @author ruoyi + */ +@Component("commonTask") +public class CommonTask { + protected final Logger logger = LoggerFactory.getLogger(this.getClass()); + + public void deleteTempFile() { + logger.info("开始删除文件"); + String downloadPath = RuoYiConfig.getTempDownloadPath(); + try { + FileUtils.deleteDirectory(new File(downloadPath)); + } catch (IOException e) { + logger.error("删除download tempfile文件夹失败" + e.getMessage(), e); + } + } +} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/AbstractQuartzJob.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/AbstractQuartzJob.java new file mode 100644 index 00000000..731a5eb5 --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/AbstractQuartzJob.java @@ -0,0 +1,107 @@ +package com.ruoyi.quartz.util; + +import java.util.Date; +import org.quartz.Job; +import org.quartz.JobExecutionContext; +import org.quartz.JobExecutionException; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.constant.ScheduleConstants; +import com.ruoyi.common.utils.ExceptionUtil; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.bean.BeanUtils; +import com.ruoyi.common.utils.spring.SpringUtils; +import com.ruoyi.quartz.domain.SysJob; +import com.ruoyi.quartz.domain.SysJobLog; +import com.ruoyi.quartz.service.ISysJobLogService; + +/** + * 抽象quartz调用 + * + * @author ruoyi + */ +public abstract class AbstractQuartzJob implements Job +{ + private static final Logger log = LoggerFactory.getLogger(AbstractQuartzJob.class); + + /** + * 线程本地变量 + */ + private static ThreadLocal threadLocal = new ThreadLocal<>(); + + @Override + public void execute(JobExecutionContext context) throws JobExecutionException + { + SysJob sysJob = new SysJob(); + BeanUtils.copyBeanProp(sysJob, context.getMergedJobDataMap().get(ScheduleConstants.TASK_PROPERTIES)); + try + { + before(context, sysJob); + if (sysJob != null) + { + doExecute(context, sysJob); + } + after(context, sysJob, null); + } + catch (Exception e) + { + log.error("任务执行异常 - :", e); + after(context, sysJob, e); + } + } + + /** + * 执行前 + * + * @param context 工作执行上下文对象 + * @param sysJob 系统计划任务 + */ + protected void before(JobExecutionContext context, SysJob sysJob) + { + threadLocal.set(new Date()); + } + + /** + * 执行后 + * + * @param context 工作执行上下文对象 + * @param sysJob 系统计划任务 + */ + protected void after(JobExecutionContext context, SysJob sysJob, Exception e) + { + Date startTime = threadLocal.get(); + threadLocal.remove(); + + final SysJobLog sysJobLog = new SysJobLog(); + sysJobLog.setJobName(sysJob.getJobName()); + sysJobLog.setJobGroup(sysJob.getJobGroup()); + sysJobLog.setInvokeTarget(sysJob.getInvokeTarget()); + sysJobLog.setStartTime(startTime); + sysJobLog.setStopTime(new Date()); + long runMs = sysJobLog.getStopTime().getTime() - sysJobLog.getStartTime().getTime(); + sysJobLog.setJobMessage(sysJobLog.getJobName() + " 总共耗时:" + runMs + "毫秒"); + if (e != null) + { + sysJobLog.setStatus(Constants.FAIL); + String errorMsg = StringUtils.substring(ExceptionUtil.getExceptionMessage(e), 0, 2000); + sysJobLog.setExceptionInfo(errorMsg); + } + else + { + sysJobLog.setStatus(Constants.SUCCESS); + } + + // 写入数据库当中 + SpringUtils.getBean(ISysJobLogService.class).addJobLog(sysJobLog); + } + + /** + * 执行方法,由子类重载 + * + * @param context 工作执行上下文对象 + * @param sysJob 系统计划任务 + * @throws Exception 执行过程中的异常 + */ + protected abstract void doExecute(JobExecutionContext context, SysJob sysJob) throws Exception; +} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/CronUtils.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/CronUtils.java new file mode 100644 index 00000000..dd538397 --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/CronUtils.java @@ -0,0 +1,63 @@ +package com.ruoyi.quartz.util; + +import java.text.ParseException; +import java.util.Date; +import org.quartz.CronExpression; + +/** + * cron表达式工具类 + * + * @author ruoyi + * + */ +public class CronUtils +{ + /** + * 返回一个布尔值代表一个给定的Cron表达式的有效性 + * + * @param cronExpression Cron表达式 + * @return boolean 表达式是否有效 + */ + public static boolean isValid(String cronExpression) + { + return CronExpression.isValidExpression(cronExpression); + } + + /** + * 返回一个字符串值,表示该消息无效Cron表达式给出有效性 + * + * @param cronExpression Cron表达式 + * @return String 无效时返回表达式错误描述,如果有效返回null + */ + public static String getInvalidMessage(String cronExpression) + { + try + { + new CronExpression(cronExpression); + return null; + } + catch (ParseException pe) + { + return pe.getMessage(); + } + } + + /** + * 返回下一个执行时间根据给定的Cron表达式 + * + * @param cronExpression Cron表达式 + * @return Date 下次Cron表达式执行时间 + */ + public static Date getNextExecution(String cronExpression) + { + try + { + CronExpression cron = new CronExpression(cronExpression); + return cron.getNextValidTimeAfter(new Date(System.currentTimeMillis())); + } + catch (ParseException e) + { + throw new IllegalArgumentException(e.getMessage()); + } + } +} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/JobInvokeUtil.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/JobInvokeUtil.java new file mode 100644 index 00000000..4f7de72b --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/JobInvokeUtil.java @@ -0,0 +1,182 @@ +package com.ruoyi.quartz.util; + +import java.lang.reflect.InvocationTargetException; +import java.lang.reflect.Method; +import java.util.LinkedList; +import java.util.List; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.spring.SpringUtils; +import com.ruoyi.quartz.domain.SysJob; + +/** + * 任务执行工具 + * + * @author ruoyi + */ +public class JobInvokeUtil +{ + /** + * 执行方法 + * + * @param sysJob 系统任务 + */ + public static void invokeMethod(SysJob sysJob) throws Exception + { + String invokeTarget = sysJob.getInvokeTarget(); + String beanName = getBeanName(invokeTarget); + String methodName = getMethodName(invokeTarget); + List methodParams = getMethodParams(invokeTarget); + + if (!isValidClassName(beanName)) + { + Object bean = SpringUtils.getBean(beanName); + invokeMethod(bean, methodName, methodParams); + } + else + { + Object bean = Class.forName(beanName).newInstance(); + invokeMethod(bean, methodName, methodParams); + } + } + + /** + * 调用任务方法 + * + * @param bean 目标对象 + * @param methodName 方法名称 + * @param methodParams 方法参数 + */ + private static void invokeMethod(Object bean, String methodName, List methodParams) + throws NoSuchMethodException, SecurityException, IllegalAccessException, IllegalArgumentException, + InvocationTargetException + { + if (StringUtils.isNotNull(methodParams) && methodParams.size() > 0) + { + Method method = bean.getClass().getMethod(methodName, getMethodParamsType(methodParams)); + method.invoke(bean, getMethodParamsValue(methodParams)); + } + else + { + Method method = bean.getClass().getMethod(methodName); + method.invoke(bean); + } + } + + /** + * 校验是否为为class包名 + * + * @param invokeTarget 名称 + * @return true是 false否 + */ + public static boolean isValidClassName(String invokeTarget) + { + return StringUtils.countMatches(invokeTarget, ".") > 1; + } + + /** + * 获取bean名称 + * + * @param invokeTarget 目标字符串 + * @return bean名称 + */ + public static String getBeanName(String invokeTarget) + { + String beanName = StringUtils.substringBefore(invokeTarget, "("); + return StringUtils.substringBeforeLast(beanName, "."); + } + + /** + * 获取bean方法 + * + * @param invokeTarget 目标字符串 + * @return method方法 + */ + public static String getMethodName(String invokeTarget) + { + String methodName = StringUtils.substringBefore(invokeTarget, "("); + return StringUtils.substringAfterLast(methodName, "."); + } + + /** + * 获取method方法参数相关列表 + * + * @param invokeTarget 目标字符串 + * @return method方法相关参数列表 + */ + public static List getMethodParams(String invokeTarget) + { + String methodStr = StringUtils.substringBetween(invokeTarget, "(", ")"); + if (StringUtils.isEmpty(methodStr)) + { + return null; + } + String[] methodParams = methodStr.split(",(?=([^\"']*[\"'][^\"']*[\"'])*[^\"']*$)"); + List classs = new LinkedList<>(); + for (int i = 0; i < methodParams.length; i++) + { + String str = StringUtils.trimToEmpty(methodParams[i]); + // String字符串类型,以'或"开头 + if (StringUtils.startsWithAny(str, "'", "\"")) + { + classs.add(new Object[] { StringUtils.substring(str, 1, str.length() - 1), String.class }); + } + // boolean布尔类型,等于true或者false + else if ("true".equalsIgnoreCase(str) || "false".equalsIgnoreCase(str)) + { + classs.add(new Object[] { Boolean.valueOf(str), Boolean.class }); + } + // long长整形,以L结尾 + else if (StringUtils.endsWith(str, "L")) + { + classs.add(new Object[] { Long.valueOf(StringUtils.substring(str, 0, str.length() - 1)), Long.class }); + } + // double浮点类型,以D结尾 + else if (StringUtils.endsWith(str, "D")) + { + classs.add(new Object[] { Double.valueOf(StringUtils.substring(str, 0, str.length() - 1)), Double.class }); + } + // 其他类型归类为整形 + else + { + classs.add(new Object[] { Integer.valueOf(str), Integer.class }); + } + } + return classs; + } + + /** + * 获取参数类型 + * + * @param methodParams 参数相关列表 + * @return 参数类型列表 + */ + public static Class[] getMethodParamsType(List methodParams) + { + Class[] classs = new Class[methodParams.size()]; + int index = 0; + for (Object[] os : methodParams) + { + classs[index] = (Class) os[1]; + index++; + } + return classs; + } + + /** + * 获取参数值 + * + * @param methodParams 参数相关列表 + * @return 参数值列表 + */ + public static Object[] getMethodParamsValue(List methodParams) + { + Object[] classs = new Object[methodParams.size()]; + int index = 0; + for (Object[] os : methodParams) + { + classs[index] = (Object) os[0]; + index++; + } + return classs; + } +} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/QuartzDisallowConcurrentExecution.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/QuartzDisallowConcurrentExecution.java new file mode 100644 index 00000000..5e135580 --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/QuartzDisallowConcurrentExecution.java @@ -0,0 +1,21 @@ +package com.ruoyi.quartz.util; + +import org.quartz.DisallowConcurrentExecution; +import org.quartz.JobExecutionContext; +import com.ruoyi.quartz.domain.SysJob; + +/** + * 定时任务处理(禁止并发执行) + * + * @author ruoyi + * + */ +@DisallowConcurrentExecution +public class QuartzDisallowConcurrentExecution extends AbstractQuartzJob +{ + @Override + protected void doExecute(JobExecutionContext context, SysJob sysJob) throws Exception + { + JobInvokeUtil.invokeMethod(sysJob); + } +} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/QuartzJobExecution.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/QuartzJobExecution.java new file mode 100644 index 00000000..e9753261 --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/QuartzJobExecution.java @@ -0,0 +1,19 @@ +package com.ruoyi.quartz.util; + +import org.quartz.JobExecutionContext; +import com.ruoyi.quartz.domain.SysJob; + +/** + * 定时任务处理(允许并发执行) + * + * @author ruoyi + * + */ +public class QuartzJobExecution extends AbstractQuartzJob +{ + @Override + protected void doExecute(JobExecutionContext context, SysJob sysJob) throws Exception + { + JobInvokeUtil.invokeMethod(sysJob); + } +} diff --git a/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/ScheduleUtils.java b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/ScheduleUtils.java new file mode 100644 index 00000000..f885d422 --- /dev/null +++ b/ruoyi-quartz/src/main/java/com/ruoyi/quartz/util/ScheduleUtils.java @@ -0,0 +1,139 @@ +package com.ruoyi.quartz.util; + +import org.quartz.CronScheduleBuilder; +import org.quartz.CronTrigger; +import org.quartz.Job; +import org.quartz.JobBuilder; +import org.quartz.JobDetail; +import org.quartz.JobKey; +import org.quartz.Scheduler; +import org.quartz.SchedulerException; +import org.quartz.TriggerBuilder; +import org.quartz.TriggerKey; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.constant.ScheduleConstants; +import com.ruoyi.common.exception.job.TaskException; +import com.ruoyi.common.exception.job.TaskException.Code; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.spring.SpringUtils; +import com.ruoyi.quartz.domain.SysJob; + +/** + * 定时任务工具类 + * + * @author ruoyi + * + */ +public class ScheduleUtils +{ + /** + * 得到quartz任务类 + * + * @param sysJob 执行计划 + * @return 具体执行任务类 + */ + private static Class getQuartzJobClass(SysJob sysJob) + { + boolean isConcurrent = "0".equals(sysJob.getConcurrent()); + return isConcurrent ? QuartzJobExecution.class : QuartzDisallowConcurrentExecution.class; + } + + /** + * 构建任务触发对象 + */ + public static TriggerKey getTriggerKey(Long jobId, String jobGroup) + { + return TriggerKey.triggerKey(ScheduleConstants.TASK_CLASS_NAME + jobId, jobGroup); + } + + /** + * 构建任务键对象 + */ + public static JobKey getJobKey(Long jobId, String jobGroup) + { + return JobKey.jobKey(ScheduleConstants.TASK_CLASS_NAME + jobId, jobGroup); + } + + /** + * 创建定时任务 + */ + public static void createScheduleJob(Scheduler scheduler, SysJob job) throws SchedulerException, TaskException + { + Class jobClass = getQuartzJobClass(job); + // 构建job信息 + Long jobId = job.getJobId(); + String jobGroup = job.getJobGroup(); + JobDetail jobDetail = JobBuilder.newJob(jobClass).withIdentity(getJobKey(jobId, jobGroup)).build(); + + // 表达式调度构建器 + CronScheduleBuilder cronScheduleBuilder = CronScheduleBuilder.cronSchedule(job.getCronExpression()); + cronScheduleBuilder = handleCronScheduleMisfirePolicy(job, cronScheduleBuilder); + + // 按新的cronExpression表达式构建一个新的trigger + CronTrigger trigger = TriggerBuilder.newTrigger().withIdentity(getTriggerKey(jobId, jobGroup)) + .withSchedule(cronScheduleBuilder).build(); + + // 放入参数,运行时的方法可以获取 + jobDetail.getJobDataMap().put(ScheduleConstants.TASK_PROPERTIES, job); + + // 判断是否存在 + if (scheduler.checkExists(getJobKey(jobId, jobGroup))) + { + // 防止创建时存在数据问题 先移除,然后在执行创建操作 + scheduler.deleteJob(getJobKey(jobId, jobGroup)); + } + + // 判断任务是否过期 + if (StringUtils.isNotNull(CronUtils.getNextExecution(job.getCronExpression()))) + { + // 执行调度任务 + scheduler.scheduleJob(jobDetail, trigger); + } + + // 暂停任务 + if (job.getStatus().equals(ScheduleConstants.Status.PAUSE.getValue())) + { + scheduler.pauseJob(ScheduleUtils.getJobKey(jobId, jobGroup)); + } + } + + /** + * 设置定时任务策略 + */ + public static CronScheduleBuilder handleCronScheduleMisfirePolicy(SysJob job, CronScheduleBuilder cb) + throws TaskException + { + switch (job.getMisfirePolicy()) + { + case ScheduleConstants.MISFIRE_DEFAULT: + return cb; + case ScheduleConstants.MISFIRE_IGNORE_MISFIRES: + return cb.withMisfireHandlingInstructionIgnoreMisfires(); + case ScheduleConstants.MISFIRE_FIRE_AND_PROCEED: + return cb.withMisfireHandlingInstructionFireAndProceed(); + case ScheduleConstants.MISFIRE_DO_NOTHING: + return cb.withMisfireHandlingInstructionDoNothing(); + default: + throw new TaskException("The task misfire policy '" + job.getMisfirePolicy() + + "' cannot be used in cron schedule tasks", Code.CONFIG_ERROR); + } + } + + /** + * 检查包名是否为白名单配置 + * + * @param invokeTarget 目标字符串 + * @return 结果 + */ + public static boolean whiteList(String invokeTarget) + { + String packageName = StringUtils.substringBefore(invokeTarget, "("); + int count = StringUtils.countMatches(packageName, "."); + if (count > 1) + { + return StringUtils.containsAnyIgnoreCase(invokeTarget, Constants.JOB_WHITELIST_STR); + } + Object obj = SpringUtils.getBean(StringUtils.split(invokeTarget, ".")[0]); + return StringUtils.containsAnyIgnoreCase(obj.getClass().getPackage().getName(), Constants.JOB_WHITELIST_STR); + } +} diff --git a/ruoyi-quartz/src/main/resources/mapper/quartz/SysJobLogMapper.xml b/ruoyi-quartz/src/main/resources/mapper/quartz/SysJobLogMapper.xml new file mode 100644 index 00000000..e608e428 --- /dev/null +++ b/ruoyi-quartz/src/main/resources/mapper/quartz/SysJobLogMapper.xml @@ -0,0 +1,93 @@ + + + + + + + + + + + + + + + + + select job_log_id, job_name, job_group, invoke_target, job_message, status, exception_info, create_time + from sys_job_log + + + + + + + + + + delete from sys_job_log where job_log_id = #{jobLogId} + + + + delete from sys_job_log where job_log_id in + + #{jobLogId} + + + + + truncate table sys_job_log + + + + insert into sys_job_log( + job_log_id, + job_name, + job_group, + invoke_target, + job_message, + status, + exception_info, + create_time + )values( + #{jobLogId}, + #{jobName}, + #{jobGroup}, + #{invokeTarget}, + #{jobMessage}, + #{status}, + #{exceptionInfo}, + sysdate() + ) + + + \ No newline at end of file diff --git a/ruoyi-quartz/src/main/resources/mapper/quartz/SysJobMapper.xml b/ruoyi-quartz/src/main/resources/mapper/quartz/SysJobMapper.xml new file mode 100644 index 00000000..5605c444 --- /dev/null +++ b/ruoyi-quartz/src/main/resources/mapper/quartz/SysJobMapper.xml @@ -0,0 +1,111 @@ + + + + + + + + + + + + + + + + + + + + + + select job_id, job_name, job_group, invoke_target, cron_expression, misfire_policy, concurrent, status, create_by, create_time, remark + from sys_job + + + + + + + + + + delete from sys_job where job_id = #{jobId} + + + + delete from sys_job where job_id in + + #{jobId} + + + + + update sys_job + + job_name = #{jobName}, + job_group = #{jobGroup}, + invoke_target = #{invokeTarget}, + cron_expression = #{cronExpression}, + misfire_policy = #{misfirePolicy}, + concurrent = #{concurrent}, + status = #{status}, + remark = #{remark}, + update_by = #{updateBy}, + update_time = sysdate() + + where job_id = #{jobId} + + + + insert into sys_job( + job_id, + job_name, + job_group, + invoke_target, + cron_expression, + misfire_policy, + concurrent, + status, + remark, + create_by, + create_time + )values( + #{jobId}, + #{jobName}, + #{jobGroup}, + #{invokeTarget}, + #{cronExpression}, + #{misfirePolicy}, + #{concurrent}, + #{status}, + #{remark}, + #{createBy}, + sysdate() + ) + + + \ No newline at end of file diff --git a/ruoyi-quartz/target/classes/com/ruoyi/quartz/controller/SysJobController.class b/ruoyi-quartz/target/classes/com/ruoyi/quartz/controller/SysJobController.class new file mode 100644 index 0000000000000000000000000000000000000000..5fd2ef7da244af2c8d69be4f600158816536dba1 GIT binary patch literal 6940 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 literal 0 HcmV?d00001 diff --git a/ruoyi-quartz/target/classes/com/ruoyi/quartz/controller/SysJobLogController.class b/ruoyi-quartz/target/classes/com/ruoyi/quartz/controller/SysJobLogController.class new file mode 100644 index 0000000000000000000000000000000000000000..90a44df99cb1fd10970b918a86c652b8bda8e354 GIT binary patch literal 3731 zcmb7H$#N4%82($9EiZ!&4r^8e2FtRcAz_PU2wnsjWLvR~10mZ;YRhA!88Or21uCc9 z@&Z(G%ON?YNL9)dP?g;C95I!rNagF9u{5%zl(?w7r~mc)x7J_(zWRfRF3?st_0!#M zIzY=^`1zuT?$Ld9{4z!lV)Ruvber>s0|>q{G^m~%^6VK3>>#;8AS<@DmRZX4(Y~!&2bNjkpbPyC4FRd+JNE*bG5;9wxMW5}y9h33#O zqYDl7Y=|t+8fHG61UH48)yrjMjMBjUaCC^xvSlMKb8*knO)n*jLh@B9%lsC8RQ}?_ zf^f>nJ&|ky0F_vscEG@>WzonI6%#te_NjDgVh%9!Kw9^r z<=Q!M!%&%iv_aQDtJ-X{Vb!QbA+co5&}cm88nYs&!r9)(gLcV%Bs4lgM>Q&_pLghB zj8-)=XicMsgoZO%9nH=A4N|&F)?~aWH6@KqvNS5wBaLiwG?L_Mv`!luZ7TgXr5RzQ z?RgFS{|v3+2#oZmVu)>0cSIFHH5(<~85xXCMu|FoO^=&*s<1@|BZq(mGfnB}F4Nn;6}u_UE{0 z^Mv(!L?OU zZ5|MFXc)T$jo?z*^$e=ou?{GkMm?M6UQ@^Y{@4rZ9E%={j-2W5+NN&huWw>MI20H? zbPgb$hld|~W?t~+RLmTpQ91)%knI^90mmWF;5P7e(^<%fdZr(SE&kmgboh7YaHpIY z!k55_^H3C_PaOQQxhUuoU8XA#?>qG=c4L$Tv2tn(YXoawB0#z)LffP%n8Ps}^XyjI zL0Yl9kH+aL#O;CbS_s0==sH+pDjg=UVKnKfY^ZvlQr){(b*hQ#X-WmELyq6rb^NC9 z_$`_Nk?;8Jy{6ouG#pZX1)Nqq?WHtnxk((O4uPhFJptnrh#HvnP}_!S#w ceH-RNHq1j%28tr|xrbUmb_+N!R;*n54<&pQ!vFvP literal 0 HcmV?d00001 diff --git a/ruoyi-quartz/target/classes/com/ruoyi/quartz/domain/SysJob.class b/ruoyi-quartz/target/classes/com/ruoyi/quartz/domain/SysJob.class new file mode 100644 index 0000000000000000000000000000000000000000..3c5544ba96479fa9a459a78b55816595ca11f6fd GIT binary patch literal 4875 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 AWdHyG literal 0 HcmV?d00001 diff --git a/ruoyi-quartz/target/classes/com/ruoyi/quartz/service/ISysJobLogService.class b/ruoyi-quartz/target/classes/com/ruoyi/quartz/service/ISysJobLogService.class new file mode 100644 index 0000000000000000000000000000000000000000..0cc728c451bef7a69bedbaaaa228db82ba295fe2 GIT binary patch literal 608 zcmbV~%}T>S6ot=iYhzn~s;xePq%NF|t|~5E2$6t`fD2tslOaq=Gnk)3d^8t6fDa|! zOz^jY#nt6z?#cbmnR$7AdIoTc0|z4qW5O_3i3*ErGxJUC>%1VGxqfICA}dX$#X~8x z;!#9qA$2N(DsLvI6Mrt3Qj~>GL@k>VdV!8pS(KR~-2LYt=WFb};6>kt;Kvx*o!!`LN&U2e@+;?_e~;S2s}ppIZ7D0zYPw;M9K6W zrR`35Pd_ka7OE?qaH5CaNBDutmO4}?wPV8ZiqE_`7b^k literal 0 HcmV?d00001 diff --git a/ruoyi-quartz/target/classes/com/ruoyi/quartz/service/ISysJobService.class b/ruoyi-quartz/target/classes/com/ruoyi/quartz/service/ISysJobService.class new file mode 100644 index 0000000000000000000000000000000000000000..e016a8f346befd9e52f433ef4e7c1137ad803fbc GIT binary patch literal 898 zcmb7?%T8M{5QcwKnnOakm207t1q*_NuvB#iVo?NAsz@vjV$rUgbFAQSVh3NMq>sje z2k1i~<|HKnf)u*4XZ(BS^FRLkaCZ;jJ9Z*$Mfk!n=VT@m|I^fK%6W!wm0Dt2+y>KA z{5A;d|M1kbgwj0jIg%ai*P5auc7iD-&>B=IrsSL-?B|F3 zoX#<_1RaCDOn_v@pt}+_ha&<7=SlW(C)0S3hR=Q4%|~cK>*D|b literal 0 HcmV?d00001 diff --git a/ruoyi-quartz/target/classes/com/ruoyi/quartz/service/impl/SysJobLogServiceImpl.class b/ruoyi-quartz/target/classes/com/ruoyi/quartz/service/impl/SysJobLogServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..617df9dee929edb1798daf54590be543b25ba300 GIT binary patch literal 1775 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 literal 0 HcmV?d00001 diff --git a/ruoyi-quartz/target/classes/com/ruoyi/quartz/util/AbstractQuartzJob.class b/ruoyi-quartz/target/classes/com/ruoyi/quartz/util/AbstractQuartzJob.class new file mode 100644 index 0000000000000000000000000000000000000000..e9355385377349dc8d4b05e3564f915f0322d8b9 GIT binary patch literal 3878 zcmbtX`F9&v75>JSJkn?!+e+iqri)W2^%h6X(goWMB#zVCwG-EJL(3M%(pXj^YuwSu zjmy%c5TGS&rEMD4h5`l(ge0JN2_+N=P`0pdzmm}3_y;&We0N5cJ(4&kthQ~=`|(y%m4;86{C!ImK-gQFrF(<(lup;5Ni%JxQ_2;k%L{X_ttl<%ix<QAQ4ecAC5(;hZPQsmG%dT_w7lQPdUea=`iBlcp}c*M?H zQ_0AqS;KZ7kGMv5jbt3#h&d0sygqA$_EhtT8Ou7<*x2mGo8Q3I9(N^arMs=PV;*rj z#UX`~>q_G3ITAA`914l9m?x;9;d-hVXl5!yBO<%Z+qgAuB-4?oD?!DTQOAfK9xx`{ zI9Z)6It5z6c1Fclsd2;{wFK|(>qc@tlM<=l7y91E4bx zZ4tmiVd&_>y$b67IX{2(>C@MqntuKCiK`c%ygD<}(9quS?~7~)XLNiQXLanvTNJEu zbrepdj&nFKR==m?`x5JhVhH(jvZz!EvW^@s=$OH*jya*`g}Nxz4}^LJKh$wah9BWo z9Y4lTbo>;rvXvCw_wv6xn@q(`TgPSmjOlNUMwWuc_Y{s-F zlQA>W>zz!78}&FU{-EQJ_>+!5<1Y%DmhD`*|ETz@j=$mWY$?U)OZq6oF7o$}95Q1L zb0l54*rnJ&7BdQMt55~3(N#o@hVMwVaVQChcH zWnoV_H7U5gx#ZA$o_CgB19_H$O+~2k$r7kCkrC5KM|N{~NTv$(m^Cpa^FiC9XH6^U zU%?`KOttt~dWo?MxNV0CE>)6mIweY&2-DgNETP$4Yb{Z`dWEL69i%821d5Z3{tt?>(d0XblOtW+2i z6J|QDpuH5ng{Vq^^H}+lPyB9b?kmN0>2koaJQlA=4~YRYlQ9y+IOxUcWLY~KSE;eW z0g)Q>&Om`xP}EZDa!E-n^P2A7wu&LoNwGq2f|;zmLwLg}Xf9RmNp<%2E)SWys~J$i zyWaGu#dpps<}wqaJrZ?4d8?8kIX*mbr8c`$MkcfJ$qX8CX$J1?%Hv$LwMyrAD`)-E z)3%rIEz6!_*mQup_=;S`g{3w`_HOz4j zIT3ype*LW%pqzC@dN_&@a3l6|)I9;*hx<8G(2G7QlFe7(;}7VWN7*A6P(HA^Z5BR0 zH7jyZXAo#@+nhsi^JN5GO5>>H5Sm8qzH_JyuW6q}RXNCQC-*E>ug4Z_rNMHD6I&4C zxDf$t;z~1aMhiBe4NYk0+IC7tu#Mxr9QW|Ah&bCyhqfV%2hdL~bS>|;Py^UU8v)Ai z#~?#^kn5Ew{|}l}4E3srhA~W*k3tV|Ov!ILhSUXwRX4SCZ(d4YJ^m1DFOojJToDoScur_?x4DLRU%6Z&FEZ=+{)e^bl7OPhS_1t(X KIiuv4A@N@igC?~A literal 0 HcmV?d00001 diff --git a/ruoyi-quartz/target/classes/com/ruoyi/quartz/util/CronUtils.class b/ruoyi-quartz/target/classes/com/ruoyi/quartz/util/CronUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..de14f5c17f9e01de9375f48148fb7e2ecdcc8f9e GIT binary patch literal 1284 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 literal 0 HcmV?d00001 diff --git a/ruoyi-quartz/target/classes/com/ruoyi/quartz/util/QuartzDisallowConcurrentExecution.class b/ruoyi-quartz/target/classes/com/ruoyi/quartz/util/QuartzDisallowConcurrentExecution.class new file mode 100644 index 0000000000000000000000000000000000000000..a749705c0df14fbbf17a527e8a7fda97ef6eed33 GIT binary patch literal 882 zcma)4T~8B16g{^eEUZ!QV04o()iXuPZ?qb1V-NmH?3wN*khx9{TyylHp6Lsv zqy4j#@l8KM5Eo!VeViPRpXNl59hu>(G$Gs>#PZnpod$Ts`M$|)EZ->a&-&P5pB;%4 zL0I~jX{Y+Kqf*6AZfk9v$VCZP=D*3e8mgEG@VJH*)Ctc=djCKF2^-flx4WscB6h`4 z>?~o|hmT{;YnLKz>}_ z1I{9j#IgG1JLo(wPx9`|z#UBSUMm0yaf`DG=eJqJlXtmN=2(6D1EqiDkjuavL+5jW jdI1^=Fr0;_hoUpM%jnr$m|fxXJfBNgz#>QA#y#K<;_UDJ literal 0 HcmV?d00001 diff --git a/ruoyi-quartz/target/classes/com/ruoyi/quartz/util/QuartzJobExecution.class b/ruoyi-quartz/target/classes/com/ruoyi/quartz/util/QuartzJobExecution.class new file mode 100644 index 0000000000000000000000000000000000000000..1f4bf9cd7321e418dc9386ecda0420ae6508ac71 GIT binary patch literal 754 zcmah{*-pYh6g|UMt++3^i}67u#=iK5F);*<22Bjg`%oq_3LR~y1pOhdd8W$B<;^PhrNN)hgl@=yAzyY?%ke{XFP*!P`})BNwQM@)(fvf#N*&>n zM|lN?;kxQV5-sM+%D?)``AsY;Pm7K&^eWd?i%ZX`+y}%jn4T@SOJ>LplB0QpDA90x zn*J-zV3#_sxHWwT7M2KCROq|nP@08gdZ*jD=53xv5-AJIX{=zCA=gvopVcsI_NN}y z0_}5Gf7fG}Hieuf_pY4p78y4B=hK>wy|%g$m-I~-(lP?4LN`>Ml5&IguvNu6;`HN~ zCkAHbqthbfkY!|(`6sZ)C_F@`2?J>iM`uKUjS)f#@{W>-Nj@T0oNRLY8L^*ao5H{( hQKuq-Y?t&YKy-{@yjv9$#ETKn2-8;RouPXSvmY%*!ao22 literal 0 HcmV?d00001 diff --git a/ruoyi-quartz/target/classes/com/ruoyi/quartz/util/ScheduleUtils.class b/ruoyi-quartz/target/classes/com/ruoyi/quartz/util/ScheduleUtils.class new file mode 100644 index 0000000000000000000000000000000000000000..3e507cc320f9a7b18112bb375b5184d5d8492bd3 GIT binary patch literal 6271 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 literal 0 HcmV?d00001 diff --git a/ruoyi-quartz/target/classes/mapper/quartz/SysJobLogMapper.xml b/ruoyi-quartz/target/classes/mapper/quartz/SysJobLogMapper.xml new file mode 100644 index 00000000..e608e428 --- /dev/null +++ b/ruoyi-quartz/target/classes/mapper/quartz/SysJobLogMapper.xml @@ -0,0 +1,93 @@ + + + + + + + + + + + + + + + + + select job_log_id, job_name, job_group, invoke_target, job_message, status, exception_info, create_time + from sys_job_log + + + + + + + + + + delete from sys_job_log where job_log_id = #{jobLogId} + + + + delete from sys_job_log where job_log_id in + + #{jobLogId} + + + + + truncate table sys_job_log + + + + insert into sys_job_log( + job_log_id, + job_name, + job_group, + invoke_target, + job_message, + status, + exception_info, + create_time + )values( + #{jobLogId}, + #{jobName}, + #{jobGroup}, + #{invokeTarget}, + #{jobMessage}, + #{status}, + #{exceptionInfo}, + sysdate() + ) + + + \ No newline at end of file diff --git a/ruoyi-quartz/target/classes/mapper/quartz/SysJobMapper.xml b/ruoyi-quartz/target/classes/mapper/quartz/SysJobMapper.xml new file mode 100644 index 00000000..5605c444 --- /dev/null +++ b/ruoyi-quartz/target/classes/mapper/quartz/SysJobMapper.xml @@ -0,0 +1,111 @@ + + + + + + + + + + + + + + + + + + + + + + select job_id, job_name, job_group, invoke_target, cron_expression, misfire_policy, concurrent, status, create_by, create_time, remark + from sys_job + + + + + + + + + + delete from sys_job where job_id = #{jobId} + + + + delete from sys_job where job_id in + + #{jobId} + + + + + update sys_job + + job_name = #{jobName}, + job_group = #{jobGroup}, + invoke_target = #{invokeTarget}, + cron_expression = #{cronExpression}, + misfire_policy = #{misfirePolicy}, + concurrent = #{concurrent}, + status = #{status}, + remark = #{remark}, + update_by = #{updateBy}, + update_time = sysdate() + + where job_id = #{jobId} + + + + insert into sys_job( + job_id, + job_name, + job_group, + invoke_target, + cron_expression, + misfire_policy, + concurrent, + status, + remark, + create_by, + create_time + )values( + #{jobId}, + #{jobName}, + #{jobGroup}, + #{invokeTarget}, + #{cronExpression}, + #{misfirePolicy}, + #{concurrent}, + #{status}, + #{remark}, + #{createBy}, + sysdate() + ) + + + \ No newline at end of file diff --git a/ruoyi-system/pom.xml b/ruoyi-system/pom.xml new file mode 100644 index 00000000..d59edb83 --- /dev/null +++ b/ruoyi-system/pom.xml @@ -0,0 +1,41 @@ + + + + ruoyi + com.ruoyi + 3.8.4 + + 4.0.0 + + ruoyi-system + + + system系统模块 + + + + + + org.jeecgframework.jimureport + jimureport-spring-boot-starter + + + + com.ruoyi + ruoyi-common + + + + org.projectlombok + lombok + true + + + com.ruoyi + ruoyi-springboot-starter-tenant + + + + diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/convert/Device/DeviceConvert.java b/ruoyi-system/src/main/java/com/ruoyi/system/convert/Device/DeviceConvert.java new file mode 100644 index 00000000..dec38dde --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/convert/Device/DeviceConvert.java @@ -0,0 +1,17 @@ +package com.ruoyi.system.convert.Device; + +import org.mapstruct.Mapper; +import org.mapstruct.factory.Mappers; + +@Mapper +public interface DeviceConvert { + DeviceConvert INSTANCE = Mappers.getMapper(DeviceConvert.class); +// @Mappings({ +// @Mapping(source = "deviceId", target = "id"), +// @Mapping(source = "deviceName", target = "name") +// }) +// NameIdVo bsDeviceToNameIdVo(BsDevice bsDevice); + +// List bsDeviceListToNameIdVoList(List bsDeviceList); +} + diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/convert/task/TaskConvert.java b/ruoyi-system/src/main/java/com/ruoyi/system/convert/task/TaskConvert.java new file mode 100644 index 00000000..fd82d952 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/convert/task/TaskConvert.java @@ -0,0 +1,27 @@ +package com.ruoyi.system.convert.task; + +//import com.ruoyi.system.domain.BsTask; + +import org.mapstruct.Mapper; +import org.mapstruct.factory.Mappers; + +@Mapper +public interface TaskConvert { + + TaskConvert INSTANCE = Mappers.getMapper(TaskConvert.class); + + +// BsTask pointResqToTask(PointResqVo vo); + +// BsTask settingUserInfoResqToTask(SettingUserInfoResqVo vo); + +// BsTask batchSettingResqVoToTask(BatchSettingResqVo vo); + +// @Mappings({ +// @Mapping(source = "taskId", target = "id"), +// @Mapping(source = "taskName", target = "name") +// }) +// NameIdVo taskToNameIdVo(BsTask task); +// +// List taskListToNameIdVoList(List taskList); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/convert/tenant/TenantConvert.java b/ruoyi-system/src/main/java/com/ruoyi/system/convert/tenant/TenantConvert.java new file mode 100644 index 00000000..b40196b3 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/convert/tenant/TenantConvert.java @@ -0,0 +1,47 @@ +package com.ruoyi.system.convert.tenant; + + +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.system.domain.TenantDO; +import com.ruoyi.system.domain.vo.tenant.TenantCreateReqVO; +import com.ruoyi.system.domain.vo.tenant.TenantExcelVO; +import com.ruoyi.system.domain.vo.tenant.TenantRespVO; +import com.ruoyi.system.domain.vo.tenant.TenantUpdateReqVO; +import org.mapstruct.Mapper; +import org.mapstruct.factory.Mappers; + +import java.util.List; + +/** + * 租户 Convert + * + * @author 芋道源码 + */ +@Mapper +public interface TenantConvert { + + TenantConvert INSTANCE = Mappers.getMapper(TenantConvert.class); + + TenantDO convert(TenantCreateReqVO bean); + + TenantDO convert(TenantUpdateReqVO bean); + + TenantRespVO convert(TenantDO bean); + + List convertList(List list); + + PageResult convertPage(PageResult page); + + List convertList02(List list); + + default SysUser convert02(TenantCreateReqVO bean) { + SysUser reqVO = new SysUser(); + reqVO.setUserName(bean.getUsername()); + reqVO.setPassword(bean.getPassword()); + reqVO.setNickName(bean.getContactName()); + reqVO.setPhonenumber(bean.getContactMobile()); + return reqVO; + } + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/convert/tenant/TenantPackageConvert.java b/ruoyi-system/src/main/java/com/ruoyi/system/convert/tenant/TenantPackageConvert.java new file mode 100644 index 00000000..167f721c --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/convert/tenant/TenantPackageConvert.java @@ -0,0 +1,37 @@ +package com.ruoyi.system.convert.tenant; + + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.system.domain.TenantPackageDO; +import com.ruoyi.system.domain.vo.packages.TenantPackageCreateReqVO; +import com.ruoyi.system.domain.vo.packages.TenantPackageRespVO; +import com.ruoyi.system.domain.vo.packages.TenantPackageSimpleRespVO; +import com.ruoyi.system.domain.vo.packages.TenantPackageUpdateReqVO; +import org.mapstruct.Mapper; +import org.mapstruct.factory.Mappers; + +import java.util.List; + +/** + * 租户套餐 Convert + * + * @author 芋道源码 + */ +@Mapper +public interface TenantPackageConvert { + + TenantPackageConvert INSTANCE = Mappers.getMapper(TenantPackageConvert.class); + + TenantPackageDO convert(TenantPackageCreateReqVO bean); + + TenantPackageDO convert(TenantPackageUpdateReqVO bean); + + TenantPackageRespVO convert(TenantPackageDO bean); + + List convertList(List list); + + PageResult convertPage(PageResult page); + + List convertList02(List list); + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/convert/user/UserConvert.java b/ruoyi-system/src/main/java/com/ruoyi/system/convert/user/UserConvert.java new file mode 100644 index 00000000..bf2213b6 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/convert/user/UserConvert.java @@ -0,0 +1,19 @@ +package com.ruoyi.system.convert.user; + +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.system.domain.vo.user.UserSimpleRespVO; +import org.mapstruct.Mapper; +import org.mapstruct.factory.Mappers; + +import java.util.List; + +@Mapper +public interface UserConvert { + + UserConvert INSTANCE = Mappers.getMapper(UserConvert.class); + + + List convertList04(List list); + + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysCache.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysCache.java new file mode 100644 index 00000000..83f0703a --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysCache.java @@ -0,0 +1,81 @@ +package com.ruoyi.system.domain; + +import com.ruoyi.common.utils.StringUtils; + +/** + * 缓存信息 + * + * @author ruoyi + */ +public class SysCache +{ + /** 缓存名称 */ + private String cacheName = ""; + + /** 缓存键名 */ + private String cacheKey = ""; + + /** 缓存内容 */ + private String cacheValue = ""; + + /** 备注 */ + private String remark = ""; + + public SysCache() + { + + } + + public SysCache(String cacheName, String remark) + { + this.cacheName = cacheName; + this.remark = remark; + } + + public SysCache(String cacheName, String cacheKey, String cacheValue) + { + this.cacheName = StringUtils.replace(cacheName, ":", ""); + this.cacheKey = StringUtils.replace(cacheKey, cacheName, ""); + this.cacheValue = cacheValue; + } + + public String getCacheName() + { + return cacheName; + } + + public void setCacheName(String cacheName) + { + this.cacheName = cacheName; + } + + public String getCacheKey() + { + return cacheKey; + } + + public void setCacheKey(String cacheKey) + { + this.cacheKey = cacheKey; + } + + public String getCacheValue() + { + return cacheValue; + } + + public void setCacheValue(String cacheValue) + { + this.cacheValue = cacheValue; + } + + public String getRemark() + { + return remark; + } + + public void setRemark(String remark) + { + this.remark = remark; + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysConfig.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysConfig.java new file mode 100644 index 00000000..b87e9022 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysConfig.java @@ -0,0 +1,116 @@ +package com.ruoyi.system.domain; + +import javax.validation.constraints.NotBlank; +import javax.validation.constraints.Size; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableId; +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import com.ruoyi.common.annotation.Excel; +import com.ruoyi.common.annotation.Excel.ColumnType; +import com.ruoyi.common.core.domain.BaseEntity; + +/** + * 参数配置表 sys_config + * + * @author ruoyi + */ +public class SysConfig extends BaseEntity +{ + private static final long serialVersionUID = 1L; + + /** 参数主键 */ + @Excel(name = "参数主键", cellType = ColumnType.NUMERIC) + @TableId(type = IdType.AUTO) + private Long configId; + + /** 参数名称 */ + @Excel(name = "参数名称") + private String configName; + + /** 参数键名 */ + @Excel(name = "参数键名") + private String configKey; + + /** 参数键值 */ + @Excel(name = "参数键值") + private String configValue; + + /** 系统内置(Y是 N否) */ + @Excel(name = "系统内置", readConverterExp = "Y=是,N=否") + private String configType; + + public Long getConfigId() + { + return configId; + } + + public void setConfigId(Long configId) + { + this.configId = configId; + } + + @NotBlank(message = "参数名称不能为空") + @Size(min = 0, max = 100, message = "参数名称不能超过100个字符") + public String getConfigName() + { + return configName; + } + + public void setConfigName(String configName) + { + this.configName = configName; + } + + @NotBlank(message = "参数键名长度不能为空") + @Size(min = 0, max = 100, message = "参数键名长度不能超过100个字符") + public String getConfigKey() + { + return configKey; + } + + public void setConfigKey(String configKey) + { + this.configKey = configKey; + } + + @NotBlank(message = "参数键值不能为空") + @Size(min = 0, max = 500, message = "参数键值长度不能超过500个字符") + public String getConfigValue() + { + return configValue; + } + + public void setConfigValue(String configValue) + { + this.configValue = configValue; + } + + public String getConfigType() + { + return configType; + } + + public void setConfigType(String configType) + { + this.configType = configType; + } + + + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) + .append("configId", getConfigId()) + .append("configName", getConfigName()) + .append("configKey", getConfigKey()) + .append("configValue", getConfigValue()) + .append("configType", getConfigType()) + .append("createBy", getCreateBy()) + .append("createTime", getCreateTime()) + .append("updateBy", getUpdateBy()) + .append("updateTime", getUpdateTime()) + .append("remark", getRemark()) + .toString(); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysLogininfor.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysLogininfor.java new file mode 100644 index 00000000..be46a28b --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysLogininfor.java @@ -0,0 +1,159 @@ +package com.ruoyi.system.domain; + +import java.util.Date; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableId; +import com.fasterxml.jackson.annotation.JsonFormat; +import com.ruoyi.common.annotation.Excel; +import com.ruoyi.common.annotation.Excel.ColumnType; +import com.ruoyi.common.core.domain.BaseEntity; + +/** + * 系统访问记录表 sys_logininfor + * + * @author ruoyi + */ +public class SysLogininfor extends BaseEntity +{ + private static final long serialVersionUID = 1L; + + /** ID */ + @Excel(name = "序号", cellType = ColumnType.NUMERIC) + @TableId(type = IdType.AUTO) + private Long infoId; + + /** 用户账号 */ + @Excel(name = "用户账号") + private String userName; + + /** 登录状态 0成功 1失败 */ + @Excel(name = "登录状态", readConverterExp = "0=成功,1=失败") + private String status; + + /** 登录IP地址 */ + @Excel(name = "登录地址") + private String ipaddr; + + /** 登录地点 */ + @Excel(name = "登录地点") + private String loginLocation; + + /** 浏览器类型 */ + @Excel(name = "浏览器") + private String browser; + + /** 操作系统 */ + @Excel(name = "操作系统") + private String os; + + /** 提示消息 */ + @Excel(name = "提示消息") + private String msg; + /** + * 多租户编号 + */ + private Long tenantId; + /** 访问时间 */ + @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") + @Excel(name = "访问时间", width = 30, dateFormat = "yyyy-MM-dd HH:mm:ss") + private Date loginTime; + + public Long getInfoId() + { + return infoId; + } + + public void setInfoId(Long infoId) + { + this.infoId = infoId; + } + + public String getUserName() + { + return userName; + } + + public void setUserName(String userName) + { + this.userName = userName; + } + + public String getStatus() + { + return status; + } + + public void setStatus(String status) + { + this.status = status; + } + + public String getIpaddr() + { + return ipaddr; + } + + public void setIpaddr(String ipaddr) + { + this.ipaddr = ipaddr; + } + + public String getLoginLocation() + { + return loginLocation; + } + + public void setLoginLocation(String loginLocation) + { + this.loginLocation = loginLocation; + } + + public String getBrowser() + { + return browser; + } + + public void setBrowser(String browser) + { + this.browser = browser; + } + + public String getOs() + { + return os; + } + + public void setOs(String os) + { + this.os = os; + } + + public String getMsg() + { + return msg; + } + + public void setMsg(String msg) + { + this.msg = msg; + } + + public Date getLoginTime() + { + return loginTime; + } + + public void setLoginTime(Date loginTime) + { + this.loginTime = loginTime; + } + + public Long getTenantId() { + return this.tenantId; + } + + public void setTenantId(Long tenantId) { + this.tenantId = tenantId; + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysNotice.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysNotice.java new file mode 100644 index 00000000..0007cb4a --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysNotice.java @@ -0,0 +1,106 @@ +package com.ruoyi.system.domain; + +import javax.validation.constraints.NotBlank; +import javax.validation.constraints.Size; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableId; +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import com.ruoyi.common.core.domain.BaseEntity; +import com.ruoyi.common.xss.Xss; + +/** + * 通知公告表 sys_notice + * + * @author ruoyi + */ +public class SysNotice extends BaseEntity +{ + private static final long serialVersionUID = 1L; + + /** 公告ID */ + @TableId(type = IdType.AUTO) + private Long noticeId; + + /** 公告标题 */ + private String noticeTitle; + + /** 公告类型(1通知 2公告) */ + private String noticeType; + + /** 公告内容 */ + private String noticeContent; + + /** 公告状态(0正常 1关闭) */ + private String status; + + public Long getNoticeId() + { + return noticeId; + } + + public void setNoticeId(Long noticeId) + { + this.noticeId = noticeId; + } + + public void setNoticeTitle(String noticeTitle) + { + this.noticeTitle = noticeTitle; + } + + @Xss(message = "公告标题不能包含脚本字符") + @NotBlank(message = "公告标题不能为空") + @Size(min = 0, max = 50, message = "公告标题不能超过50个字符") + public String getNoticeTitle() + { + return noticeTitle; + } + + public void setNoticeType(String noticeType) + { + this.noticeType = noticeType; + } + + public String getNoticeType() + { + return noticeType; + } + + public void setNoticeContent(String noticeContent) + { + this.noticeContent = noticeContent; + } + + public String getNoticeContent() + { + return noticeContent; + } + + public void setStatus(String status) + { + this.status = status; + } + + public String getStatus() + { + return status; + } + + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) + .append("noticeId", getNoticeId()) + .append("noticeTitle", getNoticeTitle()) + .append("noticeType", getNoticeType()) + .append("noticeContent", getNoticeContent()) + .append("status", getStatus()) + .append("createBy", getCreateBy()) + .append("createTime", getCreateTime()) + .append("updateBy", getUpdateBy()) + .append("updateTime", getUpdateTime()) + .append("remark", getRemark()) + .toString(); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysOperLog.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysOperLog.java new file mode 100644 index 00000000..2a6f9d5b --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysOperLog.java @@ -0,0 +1,259 @@ +package com.ruoyi.system.domain; + +import java.util.Date; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableId; +import com.fasterxml.jackson.annotation.JsonFormat; +import com.ruoyi.common.annotation.Excel; +import com.ruoyi.common.annotation.Excel.ColumnType; +import com.ruoyi.common.core.domain.BaseEntity; + +/** + * 操作日志记录表 oper_log + * + * @author ruoyi + */ +public class SysOperLog extends BaseEntity +{ + private static final long serialVersionUID = 1L; + + /** 日志主键 */ + @Excel(name = "操作序号", cellType = ColumnType.NUMERIC) + @TableId(type = IdType.AUTO) + private Long operId; + + /** 操作模块 */ + @Excel(name = "操作模块") + private String title; + + /** 业务类型(0其它 1新增 2修改 3删除) */ + @Excel(name = "业务类型", readConverterExp = "0=其它,1=新增,2=修改,3=删除,4=授权,5=导出,6=导入,7=强退,8=生成代码,9=清空数据") + private Integer businessType; + + /** 业务类型数组 */ + private Integer[] businessTypes; + + /** 请求方法 */ + @Excel(name = "请求方法") + private String method; + + /** 请求方式 */ + @Excel(name = "请求方式") + private String requestMethod; + + /** 操作类别(0其它 1后台用户 2手机端用户) */ + @Excel(name = "操作类别", readConverterExp = "0=其它,1=后台用户,2=手机端用户") + private Integer operatorType; + + /** 操作人员 */ + @Excel(name = "操作人员") + private String operName; + + /** 部门名称 */ + @Excel(name = "部门名称") + private String deptName; + + /** 请求url */ + @Excel(name = "请求地址") + private String operUrl; + + /** 操作地址 */ + @Excel(name = "操作地址") + private String operIp; + + /** 操作地点 */ + @Excel(name = "操作地点") + private String operLocation; + + /** 请求参数 */ + @Excel(name = "请求参数") + private String operParam; + + /** 返回参数 */ + @Excel(name = "返回参数") + private String jsonResult; + + /** 操作状态(0正常 1异常) */ + @Excel(name = "状态", readConverterExp = "0=正常,1=异常") + private Integer status; + + /** 错误消息 */ + @Excel(name = "错误消息") + private String errorMsg; + + /** 操作时间 */ + @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") + @Excel(name = "操作时间", width = 30, dateFormat = "yyyy-MM-dd HH:mm:ss") + private Date operTime; + + public Long getOperId() + { + return operId; + } + + public void setOperId(Long operId) + { + this.operId = operId; + } + + public String getTitle() + { + return title; + } + + public void setTitle(String title) + { + this.title = title; + } + + public Integer getBusinessType() + { + return businessType; + } + + public void setBusinessType(Integer businessType) + { + this.businessType = businessType; + } + + public Integer[] getBusinessTypes() + { + return businessTypes; + } + + public void setBusinessTypes(Integer[] businessTypes) + { + this.businessTypes = businessTypes; + } + + public String getMethod() + { + return method; + } + + public void setMethod(String method) + { + this.method = method; + } + + public String getRequestMethod() + { + return requestMethod; + } + + public void setRequestMethod(String requestMethod) + { + this.requestMethod = requestMethod; + } + + public Integer getOperatorType() + { + return operatorType; + } + + public void setOperatorType(Integer operatorType) + { + this.operatorType = operatorType; + } + + public String getOperName() + { + return operName; + } + + public void setOperName(String operName) + { + this.operName = operName; + } + + public String getDeptName() + { + return deptName; + } + + public void setDeptName(String deptName) + { + this.deptName = deptName; + } + + public String getOperUrl() + { + return operUrl; + } + + public void setOperUrl(String operUrl) + { + this.operUrl = operUrl; + } + + public String getOperIp() + { + return operIp; + } + + public void setOperIp(String operIp) + { + this.operIp = operIp; + } + + public String getOperLocation() + { + return operLocation; + } + + public void setOperLocation(String operLocation) + { + this.operLocation = operLocation; + } + + public String getOperParam() + { + return operParam; + } + + public void setOperParam(String operParam) + { + this.operParam = operParam; + } + + public String getJsonResult() + { + return jsonResult; + } + + public void setJsonResult(String jsonResult) + { + this.jsonResult = jsonResult; + } + + public Integer getStatus() + { + return status; + } + + public void setStatus(Integer status) + { + this.status = status; + } + + public String getErrorMsg() + { + return errorMsg; + } + + public void setErrorMsg(String errorMsg) + { + this.errorMsg = errorMsg; + } + + public Date getOperTime() + { + return operTime; + } + + public void setOperTime(Date operTime) + { + this.operTime = operTime; + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysPost.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysPost.java new file mode 100644 index 00000000..52d49c85 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysPost.java @@ -0,0 +1,129 @@ +package com.ruoyi.system.domain; + +import javax.validation.constraints.NotBlank; +import javax.validation.constraints.Size; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableField; +import com.baomidou.mybatisplus.annotation.TableId; +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; +import com.ruoyi.common.annotation.Excel; +import com.ruoyi.common.annotation.Excel.ColumnType; +import com.ruoyi.common.core.domain.BaseEntity; + +/** + * 岗位表 sys_post + * + * @author ruoyi + */ +public class SysPost extends BaseEntity +{ + private static final long serialVersionUID = 1L; + + /** 岗位序号 */ + @Excel(name = "岗位序号", cellType = ColumnType.NUMERIC) + @TableId(type = IdType.AUTO) + private Long postId; + + /** 岗位编码 */ + @Excel(name = "岗位编码") + private String postCode; + + /** 岗位名称 */ + @Excel(name = "岗位名称") + private String postName; + + /** 岗位排序 */ + @Excel(name = "岗位排序") + private String postSort; + + /** 状态(0正常 1停用) */ + @Excel(name = "状态", readConverterExp = "0=正常,1=停用") + private String status; + + /** 用户是否存在此岗位标识 默认不存在 */ + @TableField(exist = false) + private boolean flag = false; + + public Long getPostId() + { + return postId; + } + + public void setPostId(Long postId) + { + this.postId = postId; + } + + @NotBlank(message = "岗位编码不能为空") + @Size(min = 0, max = 64, message = "岗位编码长度不能超过64个字符") + public String getPostCode() + { + return postCode; + } + + public void setPostCode(String postCode) + { + this.postCode = postCode; + } + + @NotBlank(message = "岗位名称不能为空") + @Size(min = 0, max = 50, message = "岗位名称长度不能超过50个字符") + public String getPostName() + { + return postName; + } + + public void setPostName(String postName) + { + this.postName = postName; + } + + @NotBlank(message = "显示顺序不能为空") + public String getPostSort() + { + return postSort; + } + + public void setPostSort(String postSort) + { + this.postSort = postSort; + } + + public String getStatus() + { + return status; + } + + public void setStatus(String status) + { + this.status = status; + } + + public boolean isFlag() + { + return flag; + } + + public void setFlag(boolean flag) + { + this.flag = flag; + } + + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) + .append("postId", getPostId()) + .append("postCode", getPostCode()) + .append("postName", getPostName()) + .append("postSort", getPostSort()) + .append("status", getStatus()) + .append("createBy", getCreateBy()) + .append("createTime", getCreateTime()) + .append("updateBy", getUpdateBy()) + .append("updateTime", getUpdateTime()) + .append("remark", getRemark()) + .toString(); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysRoleDept.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysRoleDept.java new file mode 100644 index 00000000..47b21bf7 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysRoleDept.java @@ -0,0 +1,46 @@ +package com.ruoyi.system.domain; + +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; + +/** + * 角色和部门关联 sys_role_dept + * + * @author ruoyi + */ +public class SysRoleDept +{ + /** 角色ID */ + private Long roleId; + + /** 部门ID */ + private Long deptId; + + public Long getRoleId() + { + return roleId; + } + + public void setRoleId(Long roleId) + { + this.roleId = roleId; + } + + public Long getDeptId() + { + return deptId; + } + + public void setDeptId(Long deptId) + { + this.deptId = deptId; + } + + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) + .append("roleId", getRoleId()) + .append("deptId", getDeptId()) + .toString(); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysRoleMenu.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysRoleMenu.java new file mode 100644 index 00000000..de10a747 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysRoleMenu.java @@ -0,0 +1,46 @@ +package com.ruoyi.system.domain; + +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; + +/** + * 角色和菜单关联 sys_role_menu + * + * @author ruoyi + */ +public class SysRoleMenu +{ + /** 角色ID */ + private Long roleId; + + /** 菜单ID */ + private Long menuId; + + public Long getRoleId() + { + return roleId; + } + + public void setRoleId(Long roleId) + { + this.roleId = roleId; + } + + public Long getMenuId() + { + return menuId; + } + + public void setMenuId(Long menuId) + { + this.menuId = menuId; + } + + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) + .append("roleId", getRoleId()) + .append("menuId", getMenuId()) + .toString(); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysTreeDict.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysTreeDict.java new file mode 100644 index 00000000..011f05ce --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysTreeDict.java @@ -0,0 +1,88 @@ +package com.ruoyi.system.domain; + +import com.baomidou.mybatisplus.annotation.*; +import com.fasterxml.jackson.annotation.JsonFormat; +import com.ruoyi.common.annotation.Excel; +import lombok.Data; + +import java.util.Date; + +@Data +public class SysTreeDict { + + private static final long serialVersionUID = 1L; + /** + * 主键 + */ + @Excel(name = "主键") + @TableId(value = "ID", type = IdType.ASSIGN_ID) + private String id; + + /** + * 重写:分类名称 + */ + @Excel(name = "分类名称") + private String name; + + /** + * 编码 + */ + @Excel(name = "编码") + private String code; + + /** + * 类别(0:树型结构;1:平铺结构) + */ + @Excel(name = "类别") + private String struType; + + /** + * 备注 + */ + private String remark; + + /** + * 租户ID + */ + @Excel(name = "租户ID") + private String tenantId; + + /** + * 创建者 + */ + @TableField(fill = FieldFill.INSERT) + private String createBy; + + /** + * 创建时间 + */ + @TableField(fill = FieldFill.INSERT) + @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") + private Date createTime; + + /** + * 更新者 + */ + @TableField(fill = FieldFill.INSERT_UPDATE) + private String updateBy; + + /** + * 更新时间 + */ + @TableField(fill = FieldFill.INSERT_UPDATE) + @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") + private Date updateTime; + + /** + * 删除标志 + */ + @TableLogic + private String delFlag; + + /** + * 是否系统参数 + */ + @Excel(name = "图标地址") + private String isSysParam; + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysTreeDictData.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysTreeDictData.java new file mode 100644 index 00000000..5d6b0d33 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysTreeDictData.java @@ -0,0 +1,37 @@ +package com.ruoyi.system.domain; + +import com.ruoyi.common.annotation.Excel; +import com.ruoyi.common.core.domain.DictTreeEntity; +import lombok.Data; + +import java.io.Serializable; + +/** + * @author wangzongrun + */ +@Data +public class SysTreeDictData extends DictTreeEntity implements Serializable { + + private static final long serialVersionUID = 1L; + + + /** + * 树形类别 + */ + @Excel(name = "树形类别") + private String treeDict; + + /** + * 路径 + */ + @Excel(name = "路径") + private String path; + + /** + * 图标地址 + */ + @Excel(name = "图标地址") + private String icon; + + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysUserOnline.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysUserOnline.java new file mode 100644 index 00000000..2bbd3188 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysUserOnline.java @@ -0,0 +1,113 @@ +package com.ruoyi.system.domain; + +/** + * 当前在线会话 + * + * @author ruoyi + */ +public class SysUserOnline +{ + /** 会话编号 */ + private String tokenId; + + /** 部门名称 */ + private String deptName; + + /** 用户名称 */ + private String userName; + + /** 登录IP地址 */ + private String ipaddr; + + /** 登录地址 */ + private String loginLocation; + + /** 浏览器类型 */ + private String browser; + + /** 操作系统 */ + private String os; + + /** 登录时间 */ + private Long loginTime; + + public String getTokenId() + { + return tokenId; + } + + public void setTokenId(String tokenId) + { + this.tokenId = tokenId; + } + + public String getDeptName() + { + return deptName; + } + + public void setDeptName(String deptName) + { + this.deptName = deptName; + } + + public String getUserName() + { + return userName; + } + + public void setUserName(String userName) + { + this.userName = userName; + } + + public String getIpaddr() + { + return ipaddr; + } + + public void setIpaddr(String ipaddr) + { + this.ipaddr = ipaddr; + } + + public String getLoginLocation() + { + return loginLocation; + } + + public void setLoginLocation(String loginLocation) + { + this.loginLocation = loginLocation; + } + + public String getBrowser() + { + return browser; + } + + public void setBrowser(String browser) + { + this.browser = browser; + } + + public String getOs() + { + return os; + } + + public void setOs(String os) + { + this.os = os; + } + + public Long getLoginTime() + { + return loginTime; + } + + public void setLoginTime(Long loginTime) + { + this.loginTime = loginTime; + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysUserPost.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysUserPost.java new file mode 100644 index 00000000..56662dd0 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysUserPost.java @@ -0,0 +1,46 @@ +package com.ruoyi.system.domain; + +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; + +/** + * 用户和岗位关联 sys_user_post + * + * @author ruoyi + */ +public class SysUserPost +{ + /** 用户ID */ + private Long userId; + + /** 岗位ID */ + private Long postId; + + public Long getUserId() + { + return userId; + } + + public void setUserId(Long userId) + { + this.userId = userId; + } + + public Long getPostId() + { + return postId; + } + + public void setPostId(Long postId) + { + this.postId = postId; + } + + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) + .append("userId", getUserId()) + .append("postId", getPostId()) + .toString(); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysUserRole.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysUserRole.java new file mode 100644 index 00000000..4d158101 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/SysUserRole.java @@ -0,0 +1,46 @@ +package com.ruoyi.system.domain; + +import org.apache.commons.lang3.builder.ToStringBuilder; +import org.apache.commons.lang3.builder.ToStringStyle; + +/** + * 用户和角色关联 sys_user_role + * + * @author ruoyi + */ +public class SysUserRole +{ + /** 用户ID */ + private Long userId; + + /** 角色ID */ + private Long roleId; + + public Long getUserId() + { + return userId; + } + + public void setUserId(Long userId) + { + this.userId = userId; + } + + public Long getRoleId() + { + return roleId; + } + + public void setRoleId(Long roleId) + { + this.roleId = roleId; + } + + @Override + public String toString() { + return new ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) + .append("userId", getUserId()) + .append("roleId", getRoleId()) + .toString(); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/TenantDO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/TenantDO.java new file mode 100644 index 00000000..6a973427 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/TenantDO.java @@ -0,0 +1,84 @@ +package com.ruoyi.system.domain; + + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.KeySequence; +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import com.ruoyi.common.mybatis.dataobject.BaseDO; +import lombok.*; + +import java.util.Date; + +/** + * 租户 DO + * + * @author 芋道源码 + */ +@TableName(value = "system_tenant", autoResultMap = true) +//@KeySequence("system_tenant_seq") // 用于 Oracle、PostgreSQL、Kingbase、DB2、H2 数据库的主键自增。如果是 MySQL 等数据库,可不写。 +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +@Builder +@AllArgsConstructor +@NoArgsConstructor +public class TenantDO extends BaseDO { + + /** + * 套餐编号 - 系统 + */ + public static final Long PACKAGE_ID_SYSTEM = 0L; + + /** + * 租户编号,自增 + */ + @TableId(type = IdType.AUTO) + private Long id; + /** + * 租户名,唯一 + */ + private String name; + /** + * 联系人的用户编号 + * + * 关联 {@link AdminUserDO#getId()} + */ + private Long contactUserId; + /** + * 联系人 + */ + private String contactName; + /** + * 联系手机 + */ + private String contactMobile; + /** + * 租户状态 + * + * 枚举 {@link CommonStatusEnum} + */ + private Integer status; + /** + * 绑定域名 + * + * TODO 芋艿:目前是预留字段,未来会支持根据域名,自动查询到对应的租户。等等 + */ + private String domain; + /** + * 租户套餐编号 + * + * 关联 {@link TenantPackageDO#getId()} + * 特殊逻辑:系统内置租户,不使用套餐,暂时使用 {@link #PACKAGE_ID_SYSTEM} 标识 + */ + private Long packageId; + /** + * 过期时间 + */ + private Date expireTime; + /** + * 账号数量 + */ + private Integer accountCount; + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/TenantPackageDO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/TenantPackageDO.java new file mode 100644 index 00000000..6ef510d5 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/TenantPackageDO.java @@ -0,0 +1,50 @@ +package com.ruoyi.system.domain; + +import com.baomidou.mybatisplus.annotation.*; +import com.ruoyi.common.handler.JsonLongSetTypeHandler; +import com.ruoyi.common.mybatis.dataobject.BaseDO; +import lombok.*; + +import java.util.Set; + +/** + * 租户套餐 DO + * + * @author 芋道源码 + */ +@TableName(value = "system_tenant_package", autoResultMap = true) +//@KeySequence("system_tenant_package_seq") // 用于 Oracle、PostgreSQL、Kingbase、DB2、H2 数据库的主键自增。如果是 MySQL 等数据库,可不写。 +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +@Builder +@AllArgsConstructor +@NoArgsConstructor +public class TenantPackageDO extends BaseDO { + + /** + * 套餐编号,自增 + */ + @TableId(type = IdType.AUTO) + private Long id; + /** + * 套餐名,唯一 + */ + private String name; + /** + * 租户套餐状态 + * + * 枚举 {@link CommonStatusEnum} + */ + private Integer status; + /** + * 备注 + */ + private String remark; + /** + * 关联的菜单编号 + */ + @TableField(typeHandler = JsonLongSetTypeHandler.class) + private Set menuIds; + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/ImageVo.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/ImageVo.java new file mode 100644 index 00000000..24cca81a --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/ImageVo.java @@ -0,0 +1,18 @@ +package com.ruoyi.system.domain.vo; + +import com.ruoyi.common.utils.StringUtils; +import lombok.Data; + +import java.io.Serializable; + +/** + * 路由显示信息 + * + * @author ruoyi + */ +@Data +public class ImageVo implements Serializable { + private static final long serialVersionUID=1L; + String name; + String url; +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/MetaVo.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/MetaVo.java new file mode 100644 index 00000000..a5d5fdcc --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/MetaVo.java @@ -0,0 +1,106 @@ +package com.ruoyi.system.domain.vo; + +import com.ruoyi.common.utils.StringUtils; + +/** + * 路由显示信息 + * + * @author ruoyi + */ +public class MetaVo +{ + /** + * 设置该路由在侧边栏和面包屑中展示的名字 + */ + private String title; + + /** + * 设置该路由的图标,对应路径src/assets/icons/svg + */ + private String icon; + + /** + * 设置为true,则不会被 缓存 + */ + private boolean noCache; + + /** + * 内链地址(http(s)://开头) + */ + private String link; + + public MetaVo() + { + } + + public MetaVo(String title, String icon) + { + this.title = title; + this.icon = icon; + } + + public MetaVo(String title, String icon, boolean noCache) + { + this.title = title; + this.icon = icon; + this.noCache = noCache; + } + + public MetaVo(String title, String icon, String link) + { + this.title = title; + this.icon = icon; + this.link = link; + } + + public MetaVo(String title, String icon, boolean noCache, String link) + { + this.title = title; + this.icon = icon; + this.noCache = noCache; + if (StringUtils.ishttp(link)) + { + this.link = link; + } + } + + public boolean isNoCache() + { + return noCache; + } + + public void setNoCache(boolean noCache) + { + this.noCache = noCache; + } + + public String getTitle() + { + return title; + } + + public void setTitle(String title) + { + this.title = title; + } + + public String getIcon() + { + return icon; + } + + public void setIcon(String icon) + { + this.icon = icon; + } + + public String getLink() + { + return link; + } + + public void setLink(String link) + { + this.link = link; + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/NameIdVo.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/NameIdVo.java new file mode 100644 index 00000000..f990614d --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/NameIdVo.java @@ -0,0 +1,17 @@ +package com.ruoyi.system.domain.vo; + +import lombok.Data; + +import java.io.Serializable; + +/** + * 路由显示信息 + * + * @author ruoyi + */ +@Data +public class NameIdVo implements Serializable { + private static final long serialVersionUID=1L; + String name; + Long id; +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/RouterVo.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/RouterVo.java new file mode 100644 index 00000000..afff8c9c --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/RouterVo.java @@ -0,0 +1,148 @@ +package com.ruoyi.system.domain.vo; + +import com.fasterxml.jackson.annotation.JsonInclude; +import java.util.List; + +/** + * 路由配置信息 + * + * @author ruoyi + */ +@JsonInclude(JsonInclude.Include.NON_EMPTY) +public class RouterVo +{ + /** + * 路由名字 + */ + private String name; + + /** + * 路由地址 + */ + private String path; + + /** + * 是否隐藏路由,当设置 true 的时候该路由不会再侧边栏出现 + */ + private boolean hidden; + + /** + * 重定向地址,当设置 noRedirect 的时候该路由在面包屑导航中不可被点击 + */ + private String redirect; + + /** + * 组件地址 + */ + private String component; + + /** + * 路由参数:如 {"id": 1, "name": "ry"} + */ + private String query; + + /** + * 当你一个路由下面的 children 声明的路由大于1个时,自动会变成嵌套的模式--如组件页面 + */ + private Boolean alwaysShow; + + /** + * 其他元素 + */ + private MetaVo meta; + + /** + * 子路由 + */ + private List children; + + public String getName() + { + return name; + } + + public void setName(String name) + { + this.name = name; + } + + public String getPath() + { + return path; + } + + public void setPath(String path) + { + this.path = path; + } + + public boolean getHidden() + { + return hidden; + } + + public void setHidden(boolean hidden) + { + this.hidden = hidden; + } + + public String getRedirect() + { + return redirect; + } + + public void setRedirect(String redirect) + { + this.redirect = redirect; + } + + public String getComponent() + { + return component; + } + + public void setComponent(String component) + { + this.component = component; + } + + public String getQuery() + { + return query; + } + + public void setQuery(String query) + { + this.query = query; + } + + public Boolean getAlwaysShow() + { + return alwaysShow; + } + + public void setAlwaysShow(Boolean alwaysShow) + { + this.alwaysShow = alwaysShow; + } + + public MetaVo getMeta() + { + return meta; + } + + public void setMeta(MetaVo meta) + { + this.meta = meta; + } + + public List getChildren() + { + return children; + } + + public void setChildren(List children) + { + this.children = children; + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/UserInfoVo.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/UserInfoVo.java new file mode 100644 index 00000000..af8b828d --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/UserInfoVo.java @@ -0,0 +1,18 @@ +package com.ruoyi.system.domain.vo; + +import lombok.Data; + +import java.io.Serializable; + +/** + * 路由显示信息 + * + * @author ruoyi + */ +@Data +public class UserInfoVo implements Serializable { + private static final long serialVersionUID=1L; + private Long userId; + private String userName; + private String nickName; +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/menu/MenuSimpleRespVO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/menu/MenuSimpleRespVO.java new file mode 100644 index 00000000..a685110e --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/menu/MenuSimpleRespVO.java @@ -0,0 +1,27 @@ +package com.ruoyi.system.domain.vo.menu; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@ApiModel("管理后台 - 菜单精简信息 Response VO") +@Data +@NoArgsConstructor +@AllArgsConstructor +public class MenuSimpleRespVO { + + @ApiModelProperty(value = "菜单编号", required = true, example = "1024") + private Long id; + + @ApiModelProperty(value = "菜单名称", required = true, example = "芋道") + private String name; + + @ApiModelProperty(value = "父菜单 ID", required = true, example = "1024") + private Long parentId; + + @ApiModelProperty(value = "类型", required = true, example = "1", notes = "参见 MenuTypeEnum 枚举类") + private Integer type; + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageBaseVO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageBaseVO.java new file mode 100644 index 00000000..f6fec063 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageBaseVO.java @@ -0,0 +1,31 @@ +package com.ruoyi.system.domain.vo.packages; + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotNull; +import java.util.Set; + +/** +* 租户套餐 Base VO,提供给添加、修改、详细的子 VO 使用 +* 如果子 VO 存在差异的字段,请不要添加到这里,影响 Swagger 文档生成 +*/ +@Data +public class TenantPackageBaseVO { + + @ApiModelProperty(value = "套餐名", required = true, example = "VIP") + @NotNull(message = "套餐名不能为空") + private String name; + + @ApiModelProperty(value = "状态", required = true, example = "1", notes = "参见 CommonStatusEnum 枚举") + @NotNull(message = "状态不能为空") + private Integer status; + + @ApiModelProperty(value = "备注", example = "好") + private String remark; + + @ApiModelProperty(value = "关联的菜单编号", required = true) + @NotNull(message = "关联的菜单编号不能为空") + private Set menuIds; + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageCreateReqVO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageCreateReqVO.java new file mode 100644 index 00000000..2a9079b6 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageCreateReqVO.java @@ -0,0 +1,14 @@ +package com.ruoyi.system.domain.vo.packages; + +import io.swagger.annotations.ApiModel; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +@ApiModel("管理后台 - 租户套餐创建 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class TenantPackageCreateReqVO extends TenantPackageBaseVO { + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackagePageReqVO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackagePageReqVO.java new file mode 100644 index 00000000..d3e2613e --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackagePageReqVO.java @@ -0,0 +1,32 @@ +package com.ruoyi.system.domain.vo.packages; + +import com.ruoyi.common.mybatis.pojo.PageParam; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; +import org.springframework.format.annotation.DateTimeFormat; + +import java.util.Date; + + +@ApiModel("管理后台 - 租户套餐分页 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class TenantPackagePageReqVO extends PageParam { + + @ApiModelProperty(value = "套餐名", example = "VIP") + private String name; + + @ApiModelProperty(value = "状态", example = "1") + private Integer status; + + @ApiModelProperty(value = "备注", example = "好") + private String remark; + + @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") + @ApiModelProperty(value = "创建时间") + private Date[] createTime; +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageRespVO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageRespVO.java new file mode 100644 index 00000000..8e557c44 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageRespVO.java @@ -0,0 +1,23 @@ +package com.ruoyi.system.domain.vo.packages; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.util.Date; + +@ApiModel("管理后台 - 租户套餐 Response VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class TenantPackageRespVO extends TenantPackageBaseVO { + + @ApiModelProperty(value = "套餐编号", required = true, example = "1024") + private Long id; + + @ApiModelProperty(value = "创建时间", required = true) + private Date createTime; + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageSimpleRespVO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageSimpleRespVO.java new file mode 100644 index 00000000..1aca32c4 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageSimpleRespVO.java @@ -0,0 +1,21 @@ +package com.ruoyi.system.domain.vo.packages; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotNull; + +@ApiModel("管理后台 - 租户套餐精简 Response VO") +@Data +public class TenantPackageSimpleRespVO { + + @ApiModelProperty(value = "套餐编号", required = true, example = "1024") + @NotNull(message = "套餐编号不能为空") + private Long id; + + @ApiModelProperty(value = "套餐名", required = true, example = "VIP") + @NotNull(message = "套餐名不能为空") + private String name; + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageUpdateReqVO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageUpdateReqVO.java new file mode 100644 index 00000000..eea2ff17 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/packages/TenantPackageUpdateReqVO.java @@ -0,0 +1,21 @@ +package com.ruoyi.system.domain.vo.packages; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.validation.constraints.NotNull; + +@ApiModel("管理后台 - 租户套餐更新 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class TenantPackageUpdateReqVO extends TenantPackageBaseVO { + + @ApiModelProperty(value = "套餐编号", required = true, example = "1024") + @NotNull(message = "套餐编号不能为空") + private Long id; + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/report/ReportDataRequestVo.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/report/ReportDataRequestVo.java new file mode 100644 index 00000000..7512585a --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/report/ReportDataRequestVo.java @@ -0,0 +1,16 @@ +package com.ruoyi.system.domain.vo.report; + +import com.fasterxml.jackson.annotation.JsonFormat; +import com.fasterxml.jackson.annotation.OptBoolean; +import lombok.Data; + +import java.io.Serializable; +import java.util.Date; + +@Data +public class ReportDataRequestVo implements Serializable { + + String reportType; + private Date reportTime; + private String dateType ="0"; //0按日,1按月 +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/report/ReportDataResponseVo.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/report/ReportDataResponseVo.java new file mode 100644 index 00000000..aacde204 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/report/ReportDataResponseVo.java @@ -0,0 +1,19 @@ +package com.ruoyi.system.domain.vo.report; + +import com.fasterxml.jackson.annotation.JsonFormat; +import com.ruoyi.common.annotation.Excel; +import lombok.Data; + +import java.io.Serializable; +import java.util.Date; + +@Data +public class ReportDataResponseVo implements Serializable { + + Long id; + String name; + /** 创建时间 */ + @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") + @Excel(name = "日期",sort = 10,dateFormat="yyyy-MM-dd") + private Date createTime; +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/report/ReportDownLoadVo.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/report/ReportDownLoadVo.java new file mode 100644 index 00000000..53a39585 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/report/ReportDownLoadVo.java @@ -0,0 +1,18 @@ +package com.ruoyi.system.domain.vo.report; + +import com.fasterxml.jackson.annotation.JsonFormat; +import com.fasterxml.jackson.annotation.OptBoolean; +import lombok.Data; + +import java.io.Serializable; +import java.util.Date; + +@Data +public class ReportDownLoadVo implements Serializable { + + String reportType; + @JsonFormat(shape = JsonFormat.Shape.STRING, pattern = "yyyy-M-d", lenient = OptBoolean.TRUE) + private Date reportTime; + private String dateType = "0"; //0按日,1按月 + private Long id; +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/task/BatchSettingResqVo.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/task/BatchSettingResqVo.java new file mode 100644 index 00000000..985b6ff8 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/task/BatchSettingResqVo.java @@ -0,0 +1,63 @@ +package com.ruoyi.system.domain.vo.task; + + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.util.Date; +import java.util.Set; + +/** + * 批量设置 + */ +@Data +public class BatchSettingResqVo { + /** + * 任务ID + */ + private Long taskId; + + /** + * 管理人 + */ + @ApiModelProperty(value = "管理人") + private String manager; + + /** + * 周期 + */ + @ApiModelProperty(value = "周期") + private String cycle; + + /** + * 提单人员 + */ + @ApiModelProperty(value = "提单人员") + private Set submitUserInfo; + + /** + * 派单人员 + */ + @ApiModelProperty(value = "派单人员") + private Set deliveryUserInfo; + + /** + * 审核人员 + */ + @ApiModelProperty(value = "审核人员") + private Set auditUserInfo; + + /** + * 抄送人员 + */ + @ApiModelProperty(value = "抄送人员") + private Set sendUserInfo; + + /** 任务开始时间 */ + @ApiModelProperty(value = "任务开始时间") + private Date startTime; + + /** 任务结束时间 */ + @ApiModelProperty(value = "任务结束时间") + private Date endTime; +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/task/PointResqVo.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/task/PointResqVo.java new file mode 100644 index 00000000..60e079d1 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/task/PointResqVo.java @@ -0,0 +1,68 @@ +package com.ruoyi.system.domain.vo.task; + + +import com.ruoyi.system.domain.vo.ImageVo; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.util.List; +import java.util.Set; + +/** + * 点位巡查新增/编辑Vo + */ +@Data +public class PointResqVo { + /** + * 任务ID + */ + private Long taskId; + /** 点位编码 */ + @ApiModelProperty(value = "点位编码") + private String taskNo; + /** + * 点位名称 + */ + private String taskName; + + /** + * 位置 + */ + @ApiModelProperty(value = "位置") + private String position; + + /** + * 管理人 + */ + @ApiModelProperty(value = "管理人") + private String manager; + + /** + * 周期 + */ + @ApiModelProperty(value = "周期") + private String cycle; + + /** + * 巡查设备 + */ + @ApiModelProperty(value = "巡查设备") + private Set deviceInfo; + + /** + * 备注 + */ + private String remark; + + /** + * 巡检要求 + */ + @ApiModelProperty(value = "巡检要求") + private String requireDesc; + + /** + * 点位实拍 + */ + @ApiModelProperty(value = "点位实拍") + private List taskImg; +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/task/SettingUserInfoResqVo.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/task/SettingUserInfoResqVo.java new file mode 100644 index 00000000..14d69f1c --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/task/SettingUserInfoResqVo.java @@ -0,0 +1,51 @@ +package com.ruoyi.system.domain.vo.task; + + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.util.Date; +import java.util.Set; + +/** + * 任务设置人员信息、开始结束时间的请求参数 + */ +@Data +public class SettingUserInfoResqVo { + /** + * 任务ID + */ + private Long taskId; + + /** + * 提单人员 + */ + @ApiModelProperty(value = "提单人员") + private Set submitUserInfo; + + /** + * 派单人员 + */ + @ApiModelProperty(value = "派单人员") + private Set deliveryUserInfo; + + /** + * 审核人员 + */ + @ApiModelProperty(value = "审核人员") + private Set auditUserInfo; + + /** + * 抄送人员 + */ + @ApiModelProperty(value = "抄送人员") + private Set sendUserInfo; + + /** 任务开始时间 */ + @ApiModelProperty(value = "任务开始时间") + private Date startTime; + + /** 任务结束时间 */ + @ApiModelProperty(value = "任务结束时间") + private Date endTime; +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantBaseVO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantBaseVO.java new file mode 100644 index 00000000..4cd80920 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantBaseVO.java @@ -0,0 +1,48 @@ +package com.ruoyi.system.domain.vo.tenant; + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import org.hibernate.validator.constraints.URL; + +import javax.validation.constraints.NotNull; +import java.util.Date; + +/** +* 租户 Base VO,提供给添加、修改、详细的子 VO 使用 +* 如果子 VO 存在差异的字段,请不要添加到这里,影响 Swagger 文档生成 +*/ +@Data +public class TenantBaseVO { + + @ApiModelProperty(value = "租户名", required = true, example = "芋道") + @NotNull(message = "租户名不能为空") + private String name; + + @ApiModelProperty(value = "联系人", required = true, example = "芋艿") + @NotNull(message = "联系人不能为空") + private String contactName; + + @ApiModelProperty(value = "联系手机", example = "15601691300") + private String contactMobile; + + @ApiModelProperty(value = "租户状态", required = true, example = "1") + @NotNull(message = "租户状态") + private Integer status; + + @ApiModelProperty(value = "绑定域名", example = "https://www.iocoder.cn") + @URL(message = "绑定域名的地址非 URL 格式") + private String domain; + + @ApiModelProperty(value = "租户套餐编号", required = true, example = "1024") + @NotNull(message = "租户套餐编号不能为空") + private Long packageId; + + @ApiModelProperty(value = "过期时间", required = true) + @NotNull(message = "过期时间不能为空") + private Date expireTime; + + @ApiModelProperty(value = "账号数量", required = true, example = "1024") + @NotNull(message = "账号数量不能为空") + private Integer accountCount; + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantCreateReqVO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantCreateReqVO.java new file mode 100644 index 00000000..8830d8c9 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantCreateReqVO.java @@ -0,0 +1,32 @@ +package com.ruoyi.system.domain.vo.tenant; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; +import org.hibernate.validator.constraints.Length; + +import javax.validation.constraints.NotBlank; +import javax.validation.constraints.NotEmpty; +import javax.validation.constraints.Pattern; +import javax.validation.constraints.Size; + +@ApiModel("管理后台 - 租户创建 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class TenantCreateReqVO extends TenantBaseVO { + + @ApiModelProperty(value = "用户账号", required = true, example = "yudao") + @NotBlank(message = "用户账号不能为空") + @Pattern(regexp = "^[a-zA-Z0-9]{4,30}$", message = "用户账号由 数字、字母 组成") + @Size(min = 4, max = 30, message = "用户账号长度为 4-30 个字符") + private String username; + + @ApiModelProperty(value = "密码", required = true, example = "123456") + @NotEmpty(message = "密码不能为空") + @Length(min = 4, max = 16, message = "密码长度为 4-16 位") + private String password; + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantExcelVO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantExcelVO.java new file mode 100644 index 00000000..e36ade18 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantExcelVO.java @@ -0,0 +1,38 @@ +package com.ruoyi.system.domain.vo.tenant; + +import com.alibaba.excel.annotation.ExcelProperty; +import com.ruoyi.common.annotation.DictFormat; +import com.ruoyi.common.convert.DictConvert; +import lombok.Data; + +import java.util.Date; + + +/** + * 租户 Excel VO + * + * @author 芋道源码 + */ +@Data +public class TenantExcelVO { + + @ExcelProperty("租户编号") + private Long id; + + @ExcelProperty("租户名") + private String name; + + @ExcelProperty("联系人") + private String contactName; + + @ExcelProperty("联系手机") + private String contactMobile; + + @ExcelProperty(value = "状态", converter = DictConvert.class) + @DictFormat("sys_common_status") + private Integer status; + + @ExcelProperty("创建时间") + private Date createTime; + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantExportReqVO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantExportReqVO.java new file mode 100644 index 00000000..a1340cdd --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantExportReqVO.java @@ -0,0 +1,31 @@ +package com.ruoyi.system.domain.vo.tenant; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import org.springframework.format.annotation.DateTimeFormat; + +import java.util.Date; + + +@ApiModel(value = "管理后台 - 租户 Excel 导出 Request VO", description = "参数和 TenantPageReqVO 是一致的") +@Data +public class TenantExportReqVO { + + @ApiModelProperty(value = "租户名", example = "芋道") + private String name; + + @ApiModelProperty(value = "联系人", example = "芋艿") + private String contactName; + + @ApiModelProperty(value = "联系手机", example = "15601691300") + private String contactMobile; + + @ApiModelProperty(value = "租户状态(0正常 1停用)", example = "1") + private Integer status; + + @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") + @ApiModelProperty(value = "创建时间") + private Date[] createTime; + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantPageReqVO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantPageReqVO.java new file mode 100644 index 00000000..6d800158 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantPageReqVO.java @@ -0,0 +1,36 @@ +package com.ruoyi.system.domain.vo.tenant; + +import com.ruoyi.common.mybatis.pojo.PageParam; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; +import org.springframework.format.annotation.DateTimeFormat; + +import java.util.Date; + + +@ApiModel("管理后台 - 租户分页 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class TenantPageReqVO extends PageParam { + + @ApiModelProperty(value = "租户名", example = "芋道") + private String name; + + @ApiModelProperty(value = "联系人", example = "芋艿") + private String contactName; + + @ApiModelProperty(value = "联系手机", example = "15601691300") + private String contactMobile; + + @ApiModelProperty(value = "租户状态(0正常 1停用)", example = "1") + private Integer status; + + @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") + @ApiModelProperty(value = "创建时间") + private Date[] createTime; + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantRespVO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantRespVO.java new file mode 100644 index 00000000..0e39a9fa --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantRespVO.java @@ -0,0 +1,23 @@ +package com.ruoyi.system.domain.vo.tenant; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import java.util.Date; + +@ApiModel("管理后台 - 租户 Response VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class TenantRespVO extends TenantBaseVO { + + @ApiModelProperty(value = "租户编号", required = true, example = "1024") + private Long id; + + @ApiModelProperty(value = "创建时间", required = true) + private Date createTime; + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantUpdateReqVO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantUpdateReqVO.java new file mode 100644 index 00000000..622ee29c --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/tenant/TenantUpdateReqVO.java @@ -0,0 +1,21 @@ +package com.ruoyi.system.domain.vo.tenant; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.ToString; + +import javax.validation.constraints.NotNull; + +@ApiModel("管理后台 - 租户更新 Request VO") +@Data +@EqualsAndHashCode(callSuper = true) +@ToString(callSuper = true) +public class TenantUpdateReqVO extends TenantBaseVO { + + @ApiModelProperty(value = "租户编号", required = true, example = "1024") + @NotNull(message = "租户编号不能为空") + private Long id; + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/user/UserSimpleRespVO.java b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/user/UserSimpleRespVO.java new file mode 100644 index 00000000..b14d2f7b --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/user/UserSimpleRespVO.java @@ -0,0 +1,23 @@ +package com.ruoyi.system.domain.vo.user; + +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Schema(description = "管理后台 - 用户精简信息 Response VO") +@Data +@NoArgsConstructor +@AllArgsConstructor +public class UserSimpleRespVO { + + @Schema(description = "用户编号", required = true, example = "1024") + private Long userId; + + @Schema(description = "用户昵称", required = true, example = "芋道") + private String nickName; + + /** 用户账号 */ + private String userName; + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/enums/ReportTypeEnum.java b/ruoyi-system/src/main/java/com/ruoyi/system/enums/ReportTypeEnum.java new file mode 100644 index 00000000..ac5eaa05 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/enums/ReportTypeEnum.java @@ -0,0 +1,64 @@ +package com.ruoyi.system.enums; + +import cn.hutool.core.util.StrUtil; +import com.ruoyi.system.service.ISysConfigService; +import com.ruoyi.system.service.ISysDictTypeService; +import lombok.AllArgsConstructor; +import lombok.Getter; + +/** + * + */ +@Getter +@AllArgsConstructor +public enum ReportTypeEnum { + + KEYAREAS("情况登记表", ISysConfigService.class, ISysDictTypeService.class,"area_id","create_time"), + ; + + + /** + * 表单名称,注意也是找模板文件的使用名称 + */ + private final String title; + /** + * service class + */ + private final Class serviceClass; + private final Class entityClass; + /** + * 数据库中的ID字段名 + */ + private final String idField; + /** + * 数据库中的时间统计字段名 + */ + private final String timeField; + /** + * 根据名称查找枚举值 + * @param name 枚举名称 + * @return 枚举值,如果找不到返回 null + */ + public static ReportTypeEnum getEnumByName(String name) { + for (ReportTypeEnum value : values()) { + if (value.name().equals(name)) { + return value; + } + } + return null; + } + + /** + * 实体中的ID字段名 + */ + public String getIdFieldToCamerlCase(){ + return StrUtil.toCamelCase(this.getIdField()); + } + /** + * 实体中的timeField字段名 + */ + public String getTimeFieldToCamerlCase(){ + return StrUtil.toCamelCase(this.getTimeField()); + } + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysConfigMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysConfigMapper.java new file mode 100644 index 00000000..f6d161ff --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysConfigMapper.java @@ -0,0 +1,70 @@ +package com.ruoyi.system.mapper; + +import java.util.List; +import org.apache.ibatis.annotations.Mapper; +import com.ruoyi.system.domain.SysConfig; + +/** + * 参数配置 数据层 + * + * @author ruoyi +*/ +@Mapper +public interface SysConfigMapper +{ + /** + * 查询参数配置信息 + * + * @param config 参数配置信息 + * @return 参数配置信息 + */ + public SysConfig selectConfig(SysConfig config); + + /** + * 查询参数配置列表 + * + * @param config 参数配置信息 + * @return 参数配置集合 + */ + public List selectConfigList(SysConfig config); + + /** + * 根据键名查询参数配置信息 + * + * @param configKey 参数键名 + * @return 参数配置信息 + */ + public SysConfig checkConfigKeyUnique(String configKey); + + /** + * 新增参数配置 + * + * @param config 参数配置信息 + * @return 结果 + */ + public int insertConfig(SysConfig config); + + /** + * 修改参数配置 + * + * @param config 参数配置信息 + * @return 结果 + */ + public int updateConfig(SysConfig config); + + /** + * 删除参数配置 + * + * @param configId 参数ID + * @return 结果 + */ + public int deleteConfigById(Long configId); + + /** + * 批量删除参数信息 + * + * @param configIds 需要删除的参数ID + * @return 结果 + */ + public int deleteConfigByIds(Long[] configIds); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysDeptMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysDeptMapper.java new file mode 100644 index 00000000..e70e1f8e --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysDeptMapper.java @@ -0,0 +1,122 @@ +package com.ruoyi.system.mapper; + +import java.util.List; +import org.apache.ibatis.annotations.Mapper; + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import org.apache.ibatis.annotations.Param; +import com.ruoyi.common.core.domain.entity.SysDept; + +/** + * 部门管理 数据层 + * + * @author ruoyi +*/ +@Mapper +public interface SysDeptMapper extends BaseMapperX +{ + /** + * 查询部门管理数据 + * + * @param dept 部门信息 + * @return 部门信息集合 + */ + public List selectDeptList(SysDept dept); + + /** + * 根据角色ID查询部门树信息 + * + * @param roleId 角色ID + * @param deptCheckStrictly 部门树选择项是否关联显示 + * @return 选中部门列表 + */ + public List selectDeptListByRoleId(@Param("roleId") Long roleId, @Param("deptCheckStrictly") boolean deptCheckStrictly); + + /** + * 根据部门ID查询信息 + * + * @param deptId 部门ID + * @return 部门信息 + */ + public SysDept selectDeptById(Long deptId); + + /** + * 根据ID查询所有子部门 + * + * @param deptId 部门ID + * @return 部门列表 + */ + public List selectChildrenDeptById(Long deptId); + + /** + * 根据ID查询所有子部门(正常状态) + * + * @param deptId 部门ID + * @return 子部门数 + */ + public int selectNormalChildrenDeptById(Long deptId); + + /** + * 是否存在子节点 + * + * @param deptId 部门ID + * @return 结果 + */ + public int hasChildByDeptId(Long deptId); + + /** + * 查询部门是否存在用户 + * + * @param deptId 部门ID + * @return 结果 + */ + public int checkDeptExistUser(Long deptId); + + /** + * 校验部门名称是否唯一 + * + * @param deptName 部门名称 + * @param parentId 父部门ID + * @return 结果 + */ + public SysDept checkDeptNameUnique(@Param("deptName") String deptName, @Param("parentId") Long parentId); + + /** + * 新增部门信息 + * + * @param dept 部门信息 + * @return 结果 + */ + public int insertDept(SysDept dept); + + /** + * 修改部门信息 + * + * @param dept 部门信息 + * @return 结果 + */ + public int updateDept(SysDept dept); + + /** + * 修改所在部门正常状态 + * + * @param deptIds 部门ID组 + */ + public void updateDeptStatusNormal(Long[] deptIds); + + /** + * 修改子元素关系 + * + * @param depts 子元素 + * @return 结果 + */ + public int updateDeptChildren(@Param("depts") List depts); + + /** + * 删除部门管理信息 + * + * @param deptId 部门ID + * @return 结果 + */ + public int deleteDeptById(Long deptId); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysDictDataMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysDictDataMapper.java new file mode 100644 index 00000000..67736272 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysDictDataMapper.java @@ -0,0 +1,99 @@ +package com.ruoyi.system.mapper; + +import java.util.List; +import org.apache.ibatis.annotations.Mapper; + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import org.apache.ibatis.annotations.Param; +import com.ruoyi.common.core.domain.entity.SysDictData; + +/** + * 字典表 数据层 + * + * @author ruoyi +*/ +@Mapper +public interface SysDictDataMapper extends BaseMapperX +{ + /** + * 根据条件分页查询字典数据 + * + * @param dictData 字典数据信息 + * @return 字典数据集合信息 + */ + public List selectDictDataList(SysDictData dictData); + + /** + * 根据字典类型查询字典数据 + * + * @param dictType 字典类型 + * @return 字典数据集合信息 + */ + public List selectDictDataByType(String dictType); + + /** + * 根据字典类型和字典键值查询字典数据信息 + * + * @param dictType 字典类型 + * @param dictValue 字典键值 + * @return 字典标签 + */ + public String selectDictLabel(@Param("dictType") String dictType, @Param("dictValue") String dictValue); + + /** + * 根据字典数据ID查询信息 + * + * @param dictCode 字典数据ID + * @return 字典数据 + */ + public SysDictData selectDictDataById(Long dictCode); + + /** + * 查询字典数据 + * + * @param dictType 字典类型 + * @return 字典数据 + */ + public int countDictDataByType(String dictType); + + /** + * 通过字典ID删除字典数据信息 + * + * @param dictCode 字典数据ID + * @return 结果 + */ + public int deleteDictDataById(Long dictCode); + + /** + * 批量删除字典数据信息 + * + * @param dictCodes 需要删除的字典数据ID + * @return 结果 + */ + public int deleteDictDataByIds(Long[] dictCodes); + + /** + * 新增字典数据信息 + * + * @param dictData 字典数据信息 + * @return 结果 + */ + public int insertDictData(SysDictData dictData); + + /** + * 修改字典数据信息 + * + * @param dictData 字典数据信息 + * @return 结果 + */ + public int updateDictData(SysDictData dictData); + + /** + * 同步修改字典类型 + * + * @param oldDictType 旧字典类型 + * @param newDictType 新旧字典类型 + * @return 结果 + */ + public int updateDictDataType(@Param("oldDictType") String oldDictType, @Param("newDictType") String newDictType); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysDictTypeMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysDictTypeMapper.java new file mode 100644 index 00000000..25faedd6 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysDictTypeMapper.java @@ -0,0 +1,87 @@ +package com.ruoyi.system.mapper; + +import java.util.List; +import org.apache.ibatis.annotations.Mapper; + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.ruoyi.common.core.domain.entity.SysDictType; + +/** + * 字典表 数据层 + * + * @author ruoyi +*/ +@Mapper +public interface SysDictTypeMapper extends BaseMapperX +{ + /** + * 根据条件分页查询字典类型 + * + * @param dictType 字典类型信息 + * @return 字典类型集合信息 + */ + public List selectDictTypeList(SysDictType dictType); + + /** + * 根据所有字典类型 + * + * @return 字典类型集合信息 + */ + public List selectDictTypeAll(); + + /** + * 根据字典类型ID查询信息 + * + * @param dictId 字典类型ID + * @return 字典类型 + */ + public SysDictType selectDictTypeById(Long dictId); + + /** + * 根据字典类型查询信息 + * + * @param dictType 字典类型 + * @return 字典类型 + */ + public SysDictType selectDictTypeByType(String dictType); + + /** + * 通过字典ID删除字典信息 + * + * @param dictId 字典ID + * @return 结果 + */ + public int deleteDictTypeById(Long dictId); + + /** + * 批量删除字典类型信息 + * + * @param dictIds 需要删除的字典ID + * @return 结果 + */ + public int deleteDictTypeByIds(Long[] dictIds); + + /** + * 新增字典类型信息 + * + * @param dictType 字典类型信息 + * @return 结果 + */ + public int insertDictType(SysDictType dictType); + + /** + * 修改字典类型信息 + * + * @param dictType 字典类型信息 + * @return 结果 + */ + public int updateDictType(SysDictType dictType); + + /** + * 校验字典类型称是否唯一 + * + * @param dictType 字典类型 + * @return 结果 + */ + public SysDictType checkDictTypeUnique(String dictType); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysLogininforMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysLogininforMapper.java new file mode 100644 index 00000000..c521eaad --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysLogininforMapper.java @@ -0,0 +1,44 @@ +package com.ruoyi.system.mapper; + +import java.util.List; +import org.apache.ibatis.annotations.Mapper; +import com.ruoyi.system.domain.SysLogininfor; + +/** + * 系统访问日志情况信息 数据层 + * + * @author ruoyi +*/ +@Mapper +public interface SysLogininforMapper +{ + /** + * 新增系统登录日志 + * + * @param logininfor 访问日志对象 + */ + public void insertLogininfor(SysLogininfor logininfor); + + /** + * 查询系统登录日志集合 + * + * @param logininfor 访问日志对象 + * @return 登录记录集合 + */ + public List selectLogininforList(SysLogininfor logininfor); + + /** + * 批量删除系统登录日志 + * + * @param infoIds 需要删除的登录日志ID + * @return 结果 + */ + public int deleteLogininforByIds(Long[] infoIds); + + /** + * 清空系统登录日志 + * + * @return 结果 + */ + public int cleanLogininfor(); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysMenuMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysMenuMapper.java new file mode 100644 index 00000000..838387e1 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysMenuMapper.java @@ -0,0 +1,136 @@ +package com.ruoyi.system.mapper; + +import java.util.List; + +import com.ruoyi.common.mybatis.query.LambdaQueryWrapperX; +import org.apache.ibatis.annotations.Mapper; + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import org.apache.ibatis.annotations.Param; +import com.ruoyi.common.core.domain.entity.SysMenu; + +/** + * 菜单表 数据层 + * + * @author ruoyi +*/ +@Mapper +public interface SysMenuMapper extends BaseMapperX +{ + /** + * 查询系统菜单列表 + * + * @param menu 菜单信息 + * @return 菜单列表 + */ + public List selectMenuList(SysMenu menu); + + /** + * 根据用户所有权限 + * + * @return 权限列表 + */ + public List selectMenuPerms(); + + /** + * 根据用户查询系统菜单列表 + * + * @param menu 菜单信息 + * @return 菜单列表 + */ + public List selectMenuListByUserId(SysMenu menu); + + /** + * 根据角色ID查询权限 + * + * @param roleId 角色ID + * @return 权限列表 + */ + public List selectMenuPermsByRoleId(Long roleId); + + /** + * 根据用户ID查询权限 + * + * @param userId 用户ID + * @return 权限列表 + */ + public List selectMenuPermsByUserId(Long userId); + + /** + * 根据用户ID查询菜单 + * + * @return 菜单列表 + */ + public List selectMenuTreeAll(); + + /** + * 根据用户ID查询菜单 + * + * @param userId 用户ID + * @return 菜单列表 + */ + public List selectMenuTreeByUserId(Long userId); + + /** + * 根据角色ID查询菜单树信息 + * + * @param roleId 角色ID + * @param menuCheckStrictly 菜单树选择项是否关联显示 + * @return 选中菜单列表 + */ + public List selectMenuListByRoleId(@Param("roleId") Long roleId, @Param("menuCheckStrictly") boolean menuCheckStrictly); + + /** + * 根据菜单ID查询信息 + * + * @param menuId 菜单ID + * @return 菜单信息 + */ + public SysMenu selectMenuById(Long menuId); + + /** + * 是否存在菜单子节点 + * + * @param menuId 菜单ID + * @return 结果 + */ + public int hasChildByMenuId(Long menuId); + + /** + * 新增菜单信息 + * + * @param menu 菜单信息 + * @return 结果 + */ + public int insertMenu(SysMenu menu); + + /** + * 修改菜单信息 + * + * @param menu 菜单信息 + * @return 结果 + */ + public int updateMenu(SysMenu menu); + + /** + * 删除菜单管理信息 + * + * @param menuId 菜单ID + * @return 结果 + */ + public int deleteMenuById(Long menuId); + + /** + * 校验菜单名称是否唯一 + * + * @param menuName 菜单名称 + * @param parentId 父菜单ID + * @return 结果 + */ + public SysMenu checkMenuNameUnique(@Param("menuName") String menuName, @Param("parentId") Long parentId); + + default List selectList(SysMenu reqVO) { + return selectList(new LambdaQueryWrapperX().likeIfPresent(SysMenu::getMenuName, reqVO.getMenuName()) + .eqIfPresent(SysMenu::getStatus, reqVO.getStatus()).eqIfPresent(SysMenu::getVisible, reqVO.getVisible())); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysNoticeMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysNoticeMapper.java new file mode 100644 index 00000000..1551e1d8 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysNoticeMapper.java @@ -0,0 +1,62 @@ +package com.ruoyi.system.mapper; + +import java.util.List; +import org.apache.ibatis.annotations.Mapper; +import com.ruoyi.system.domain.SysNotice; + +/** + * 通知公告表 数据层 + * + * @author ruoyi +*/ +@Mapper +public interface SysNoticeMapper +{ + /** + * 查询公告信息 + * + * @param noticeId 公告ID + * @return 公告信息 + */ + public SysNotice selectNoticeById(Long noticeId); + + /** + * 查询公告列表 + * + * @param notice 公告信息 + * @return 公告集合 + */ + public List selectNoticeList(SysNotice notice); + + /** + * 新增公告 + * + * @param notice 公告信息 + * @return 结果 + */ + public int insertNotice(SysNotice notice); + + /** + * 修改公告 + * + * @param notice 公告信息 + * @return 结果 + */ + public int updateNotice(SysNotice notice); + + /** + * 批量删除公告 + * + * @param noticeId 公告ID + * @return 结果 + */ + public int deleteNoticeById(Long noticeId); + + /** + * 批量删除公告信息 + * + * @param noticeIds 需要删除的公告ID + * @return 结果 + */ + public int deleteNoticeByIds(Long[] noticeIds); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysOperLogMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysOperLogMapper.java new file mode 100644 index 00000000..47ac30a7 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysOperLogMapper.java @@ -0,0 +1,50 @@ +package com.ruoyi.system.mapper; + +import java.util.List; +import org.apache.ibatis.annotations.Mapper; +import com.ruoyi.system.domain.SysOperLog; + +/** + * 操作日志 数据层 + * + * @author ruoyi +*/ +@Mapper +public interface SysOperLogMapper +{ + /** + * 新增操作日志 + * + * @param operLog 操作日志对象 + */ + public void insertOperlog(SysOperLog operLog); + + /** + * 查询系统操作日志集合 + * + * @param operLog 操作日志对象 + * @return 操作日志集合 + */ + public List selectOperLogList(SysOperLog operLog); + + /** + * 批量删除系统操作日志 + * + * @param operIds 需要删除的操作日志ID + * @return 结果 + */ + public int deleteOperLogByIds(Long[] operIds); + + /** + * 查询操作日志详细 + * + * @param operId 操作ID + * @return 操作日志对象 + */ + public SysOperLog selectOperLogById(Long operId); + + /** + * 清空操作日志 + */ + public void cleanOperLog(); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysPostMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysPostMapper.java new file mode 100644 index 00000000..5de093e2 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysPostMapper.java @@ -0,0 +1,104 @@ +package com.ruoyi.system.mapper; + +import java.util.List; +import org.apache.ibatis.annotations.Mapper; + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.system.domain.SysPost; + +/** + * 岗位信息 数据层 + * + * @author ruoyi +*/ +@Mapper +public interface SysPostMapper extends BaseMapperX +{ + /** + * 查询岗位数据集合 + * + * @param post 岗位信息 + * @return 岗位数据集合 + */ + public List selectPostList(SysPost post); + + /** + * 查询所有岗位 + * + * @return 岗位列表 + */ + public List selectPostAll(); + + /** + * 通过岗位ID查询岗位信息 + * + * @param postId 岗位ID + * @return 角色对象信息 + */ + public SysPost selectPostById(Long postId); + + /** + * 根据用户ID获取岗位选择框列表 + * + * @param userId 用户ID + * @return 选中岗位ID列表 + */ + public List selectPostListByUserId(Long userId); + + /** + * 查询用户所属岗位组 + * + * @param userName 用户名 + * @return 结果 + */ + public List selectPostsByUserName(String userName); + + /** + * 删除岗位信息 + * + * @param postId 岗位ID + * @return 结果 + */ + public int deletePostById(Long postId); + + /** + * 批量删除岗位信息 + * + * @param postIds 需要删除的岗位ID + * @return 结果 + */ + public int deletePostByIds(Long[] postIds); + + /** + * 修改岗位信息 + * + * @param post 岗位信息 + * @return 结果 + */ + public int updatePost(SysPost post); + + /** + * 新增岗位信息 + * + * @param post 岗位信息 + * @return 结果 + */ + public int insertPost(SysPost post); + + /** + * 校验岗位名称 + * + * @param postName 岗位名称 + * @return 结果 + */ + public SysPost checkPostNameUnique(String postName); + + /** + * 校验岗位编码 + * + * @param postCode 岗位编码 + * @return 结果 + */ + public SysPost checkPostCodeUnique(String postCode); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysRoleDeptMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysRoleDeptMapper.java new file mode 100644 index 00000000..3661a594 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysRoleDeptMapper.java @@ -0,0 +1,48 @@ +package com.ruoyi.system.mapper; + +import java.util.List; +import org.apache.ibatis.annotations.Mapper; + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.ruoyi.system.domain.SysRoleDept; + +/** + * 角色与部门关联表 数据层 + * + * @author ruoyi +*/ +@Mapper +public interface SysRoleDeptMapper extends BaseMapperX +{ + /** + * 通过角色ID删除角色和部门关联 + * + * @param roleId 角色ID + * @return 结果 + */ + public int deleteRoleDeptByRoleId(Long roleId); + + /** + * 批量删除角色部门关联信息 + * + * @param ids 需要删除的数据ID + * @return 结果 + */ + public int deleteRoleDept(Long[] ids); + + /** + * 查询部门使用数量 + * + * @param deptId 部门ID + * @return 结果 + */ + public int selectCountRoleDeptByDeptId(Long deptId); + + /** + * 批量新增角色部门信息 + * + * @param roleDeptList 角色部门列表 + * @return 结果 + */ + public int batchRoleDept(List roleDeptList); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysRoleMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysRoleMapper.java new file mode 100644 index 00000000..36b1dd65 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysRoleMapper.java @@ -0,0 +1,111 @@ +package com.ruoyi.system.mapper; + +import java.util.List; +import org.apache.ibatis.annotations.Mapper; + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.ruoyi.common.core.domain.entity.SysRole; + +/** + * 角色表 数据层 + * + * @author ruoyi +*/ +@Mapper +public interface SysRoleMapper extends BaseMapperX +{ + /** + * 根据条件分页查询角色数据 + * + * @param role 角色信息 + * @return 角色数据集合信息 + */ + public List selectRoleList(SysRole role); + + /** + * 根据用户ID查询角色 + * + * @param userId 用户ID + * @return 角色列表 + */ + public List selectRolePermissionByUserId(Long userId); + + /** + * 查询所有角色 + * + * @return 角色列表 + */ + public List selectRoleAll(); + + /** + * 根据用户ID获取角色选择框列表 + * + * @param userId 用户ID + * @return 选中角色ID列表 + */ + public List selectRoleListByUserId(Long userId); + + /** + * 通过角色ID查询角色 + * + * @param roleId 角色ID + * @return 角色对象信息 + */ + public SysRole selectRoleById(Long roleId); + + /** + * 根据用户ID查询角色 + * + * @param userName 用户名 + * @return 角色列表 + */ + public List selectRolesByUserName(String userName); + + /** + * 校验角色名称是否唯一 + * + * @param roleName 角色名称 + * @return 角色信息 + */ + public SysRole checkRoleNameUnique(String roleName); + + /** + * 校验角色权限是否唯一 + * + * @param roleKey 角色权限 + * @return 角色信息 + */ + public SysRole checkRoleKeyUnique(String roleKey); + + /** + * 修改角色信息 + * + * @param role 角色信息 + * @return 结果 + */ + public int updateRole(SysRole role); + + /** + * 新增角色信息 + * + * @param role 角色信息 + * @return 结果 + */ + public int insertRole(SysRole role); + + /** + * 通过角色ID删除角色 + * + * @param roleId 角色ID + * @return 结果 + */ + public int deleteRoleById(Long roleId); + + /** + * 批量删除角色信息 + * + * @param roleIds 需要删除的角色ID + * @return 结果 + */ + public int deleteRoleByIds(Long[] roleIds); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysRoleMenuMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysRoleMenuMapper.java new file mode 100644 index 00000000..63e9d415 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysRoleMenuMapper.java @@ -0,0 +1,48 @@ +package com.ruoyi.system.mapper; + +import java.util.List; +import org.apache.ibatis.annotations.Mapper; + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.ruoyi.system.domain.SysRoleMenu; + +/** + * 角色与菜单关联表 数据层 + * + * @author ruoyi +*/ +@Mapper +public interface SysRoleMenuMapper extends BaseMapperX +{ + /** + * 查询菜单使用数量 + * + * @param menuId 菜单ID + * @return 结果 + */ + public int checkMenuExistRole(Long menuId); + + /** + * 通过角色ID删除角色和菜单关联 + * + * @param roleId 角色ID + * @return 结果 + */ + public int deleteRoleMenuByRoleId(Long roleId); + + /** + * 批量删除角色菜单关联信息 + * + * @param ids 需要删除的数据ID + * @return 结果 + */ + public int deleteRoleMenu(Long[] ids); + + /** + * 批量新增角色菜单信息 + * + * @param roleMenuList 角色菜单列表 + * @return 结果 + */ + public int batchRoleMenu(List roleMenuList); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysTreeDictDataMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysTreeDictDataMapper.java new file mode 100644 index 00000000..01aa66bc --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysTreeDictDataMapper.java @@ -0,0 +1,33 @@ +package com.ruoyi.system.mapper; + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.ruoyi.system.domain.SysTreeDictData; +import org.apache.ibatis.annotations.Param; + +import java.util.List; +import org.apache.ibatis.annotations.Mapper; + +/** + * 树形字典数据Mapper接口 +*/ +@Mapper +public interface SysTreeDictDataMapper extends BaseMapperX { + /** + * 分页查询树形字典数据列表 + *

    + * param page 分页信息 + * + * @param entity 树形字典数据信息 + * @return 树形字典数据集合 + */ + public IPage selectTreeDictDataPage(IPage page, @Param("entity") SysTreeDictData entity); + + /** + * 查询所有树形字典数据列表 + * + * @param entity 树形字典数据信息 + * @return 树形字典数据集合 + */ + public List selectTreeDictDataList(@Param("entity") SysTreeDictData entity); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysTreeDictMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysTreeDictMapper.java new file mode 100644 index 00000000..2a04574d --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysTreeDictMapper.java @@ -0,0 +1,34 @@ +package com.ruoyi.system.mapper; + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.ruoyi.system.domain.SysTreeDict; +import org.apache.ibatis.annotations.Param; + +import java.util.List; +import org.apache.ibatis.annotations.Mapper; + +/** + * 系统树形Mapper接口 +*/ +@Mapper +public interface SysTreeDictMapper extends BaseMapperX { + /** + * 分页查询系统树形列表 + *

    + * param page 分页信息 + * param entity 实体 + * + * @param sysTreeDict 系统树形信息 + * @return 系统树形集合 + */ + public IPage selectTreeDictPage(IPage page, @Param("entity") SysTreeDict sysTreeDict); + + /** + * 查询所有系统树形列表 + * + * @param entity 系统树形信息 + * @return 系统树形集合 + */ + public List selectTreeDictList(@Param("entity") SysTreeDict entity); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysUserMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysUserMapper.java new file mode 100644 index 00000000..d4884688 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysUserMapper.java @@ -0,0 +1,133 @@ +package com.ruoyi.system.mapper; + +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import org.apache.ibatis.annotations.Mapper; +import org.apache.ibatis.annotations.Param; + +import java.util.List; + +/** + * 用户表 数据层 + * + * @author ruoyi + */ +@Mapper +public interface SysUserMapper extends BaseMapperX { + /** + * 根据条件分页查询用户列表 + * + * @param sysUser 用户信息 + * @return 用户信息集合信息 + */ + public List selectUserList(SysUser sysUser); + + /** + * 根据条件分页查询已配用户角色列表 + * + * @param user 用户信息 + * @return 用户信息集合信息 + */ + public List selectAllocatedList(SysUser user); + + /** + * 根据条件分页查询未分配用户角色列表 + * + * @param user 用户信息 + * @return 用户信息集合信息 + */ + public List selectUnallocatedList(SysUser user); + + /** + * 通过用户名查询用户 + * + * @param userName 用户名 + * @return 用户对象信息 + */ + public SysUser selectUserByUserName(String userName); + /** + * 通过用户ID查询用户 + * + * @param userId 用户ID + * @return 用户对象信息 + */ + public SysUser selectUserById(Long userId); + + /** + * 新增用户信息 + * + * @param user 用户信息 + * @return 结果 + */ + public int insertUser(SysUser user); + + /** + * 修改用户信息 + * + * @param user 用户信息 + * @return 结果 + */ + public int updateUser(SysUser user); + + /** + * 修改用户头像 + * + * @param userName 用户名 + * @param avatar 头像地址 + * @return 结果 + */ + public int updateUserAvatar(@Param("userName") String userName, @Param("avatar") String avatar); + + /** + * 重置用户密码 + * + * @param userName 用户名 + * @param password 密码 + * @return 结果 + */ + public int resetUserPwd(@Param("userName") String userName, @Param("password") String password); + + /** + * 通过用户ID删除用户 + * + * @param userId 用户ID + * @return 结果 + */ + public int deleteUserById(Long userId); + + /** + * 批量删除用户信息 + * + * @param userIds 需要删除的用户ID + * @return 结果 + */ + public int deleteUserByIds(Long[] userIds); + + /** + * 校验用户名称是否唯一 + * + * @param userName 用户名称 + * @return 结果 + */ + public SysUser checkUserNameUnique(String userName); + + /** + * 校验手机号码是否唯一 + * + * @param phonenumber 手机号码 + * @return 结果 + */ + public SysUser checkPhoneUnique(String phonenumber); + + /** + * 校验email是否唯一 + * + * @param email 用户邮箱 + * @return 结果 + */ + public SysUser checkEmailUnique(String email); + + default List selectListByStatus(Integer status) { + return selectList(SysUser::getStatus, status); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysUserPostMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysUserPostMapper.java new file mode 100644 index 00000000..78bb6e8f --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysUserPostMapper.java @@ -0,0 +1,48 @@ +package com.ruoyi.system.mapper; + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.ruoyi.system.domain.SysUserPost; + +import java.util.List; +import org.apache.ibatis.annotations.Mapper; + +/** + * 用户与岗位关联表 数据层 + * + * @author ruoyi +*/ +@Mapper +public interface SysUserPostMapper extends BaseMapperX +{ + /** + * 通过用户ID删除用户和岗位关联 + * + * @param userId 用户ID + * @return 结果 + */ + public int deleteUserPostByUserId(Long userId); + + /** + * 通过岗位ID查询岗位使用数量 + * + * @param postId 岗位ID + * @return 结果 + */ + public int countUserPostById(Long postId); + + /** + * 批量删除用户和岗位关联 + * + * @param ids 需要删除的数据ID + * @return 结果 + */ + public int deleteUserPost(Long[] ids); + + /** + * 批量新增用户岗位信息 + * + * @param userPostList 用户角色列表 + * @return 结果 + */ + public int batchUserPost(List userPostList); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysUserRoleMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysUserRoleMapper.java new file mode 100644 index 00000000..980790fe --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysUserRoleMapper.java @@ -0,0 +1,66 @@ +package com.ruoyi.system.mapper; + +import java.util.List; +import org.apache.ibatis.annotations.Mapper; + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import org.apache.ibatis.annotations.Param; +import com.ruoyi.system.domain.SysUserRole; + +/** + * 用户与角色关联表 数据层 + * + * @author ruoyi +*/ +@Mapper +public interface SysUserRoleMapper extends BaseMapperX +{ + /** + * 通过用户ID删除用户和角色关联 + * + * @param userId 用户ID + * @return 结果 + */ + public int deleteUserRoleByUserId(Long userId); + + /** + * 批量删除用户和角色关联 + * + * @param ids 需要删除的数据ID + * @return 结果 + */ + public int deleteUserRole(Long[] ids); + + /** + * 通过角色ID查询角色使用数量 + * + * @param roleId 角色ID + * @return 结果 + */ + public int countUserRoleByRoleId(Long roleId); + + /** + * 批量新增用户角色信息 + * + * @param userRoleList 用户角色列表 + * @return 结果 + */ + public int batchUserRole(List userRoleList); + + /** + * 删除用户和角色关联信息 + * + * @param userRole 用户和角色关联信息 + * @return 结果 + */ + public int deleteUserRoleInfo(SysUserRole userRole); + + /** + * 批量取消授权用户角色 + * + * @param roleId 角色ID + * @param userIds 需要删除的用户数据ID + * @return 结果 + */ + public int deleteUserRoleInfos(@Param("roleId") Long roleId, @Param("userIds") Long[] userIds); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/TenantMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/TenantMapper.java new file mode 100644 index 00000000..94f8e27b --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/TenantMapper.java @@ -0,0 +1,47 @@ +package com.ruoyi.system.mapper; + + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.mybatis.query.LambdaQueryWrapperX; +import com.ruoyi.system.domain.TenantDO; +import com.ruoyi.system.domain.vo.tenant.TenantExportReqVO; +import com.ruoyi.system.domain.vo.tenant.TenantPageReqVO; +import org.apache.ibatis.annotations.Mapper; +import org.apache.ibatis.annotations.Select; + +import java.util.Date; +import java.util.List; + +/** + * 租户 Mapper + * + * @author 芋道源码 + */ +@Mapper +public interface TenantMapper extends BaseMapperX { + + default PageResult selectPage(TenantPageReqVO reqVO) { + return selectPage(reqVO, new LambdaQueryWrapperX().likeIfPresent(TenantDO::getName, reqVO.getName()).likeIfPresent(TenantDO::getContactName, reqVO.getContactName()).likeIfPresent(TenantDO::getContactMobile, reqVO.getContactMobile()).eqIfPresent(TenantDO::getStatus, reqVO.getStatus()).betweenIfPresent(TenantDO::getCreateTime, reqVO.getCreateTime()).orderByDesc(TenantDO::getId)); + } + + default List selectList(TenantExportReqVO reqVO) { + return selectList(new LambdaQueryWrapperX().likeIfPresent(TenantDO::getName, reqVO.getName()).likeIfPresent(TenantDO::getContactName, reqVO.getContactName()).likeIfPresent(TenantDO::getContactMobile, reqVO.getContactMobile()).eqIfPresent(TenantDO::getStatus, reqVO.getStatus()).betweenIfPresent(TenantDO::getCreateTime, reqVO.getCreateTime()).orderByDesc(TenantDO::getId)); + } + + default TenantDO selectByName(String name) { + return selectOne(TenantDO::getName, name); + } + + default Long selectCountByPackageId(Long packageId) { + return selectCount(TenantDO::getPackageId, packageId); + } + + default List selectListByPackageId(Long packageId) { + return selectList(TenantDO::getPackageId, packageId); + } + + @Select("SELECT COUNT(*) FROM system_tenant WHERE update_time > #{maxUpdateTime}") + Long selectCountByUpdateTimeGt(Date maxUpdateTime); + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/TenantPackageMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/TenantPackageMapper.java new file mode 100644 index 00000000..141fe76f --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/TenantPackageMapper.java @@ -0,0 +1,28 @@ +package com.ruoyi.system.mapper; + + +import com.ruoyi.common.mybatis.mapper.BaseMapperX; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.mybatis.query.LambdaQueryWrapperX; +import com.ruoyi.system.domain.TenantPackageDO; +import com.ruoyi.system.domain.vo.packages.TenantPackagePageReqVO; +import org.apache.ibatis.annotations.Mapper; + +import java.util.List; + +/** + * 租户套餐 Mapper + * + * @author 芋道源码 + */ +@Mapper +public interface TenantPackageMapper extends BaseMapperX { + + default PageResult selectPage(TenantPackagePageReqVO reqVO) { + return selectPage(reqVO, new LambdaQueryWrapperX().likeIfPresent(TenantPackageDO::getName, reqVO.getName()).eqIfPresent(TenantPackageDO::getStatus, reqVO.getStatus()).likeIfPresent(TenantPackageDO::getRemark, reqVO.getRemark()).betweenIfPresent(TenantPackageDO::getCreateTime, reqVO.getCreateTime()).orderByDesc(TenantPackageDO::getId)); + } + + default List selectListByStatus(Integer status) { + return selectList(TenantPackageDO::getStatus, status); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysConfigService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysConfigService.java new file mode 100644 index 00000000..58bdc292 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysConfigService.java @@ -0,0 +1,89 @@ +package com.ruoyi.system.service; + +import java.util.List; +import com.ruoyi.system.domain.SysConfig; + +/** + * 参数配置 服务层 + * + * @author ruoyi + */ +public interface ISysConfigService +{ + /** + * 查询参数配置信息 + * + * @param configId 参数配置ID + * @return 参数配置信息 + */ + public SysConfig selectConfigById(Long configId); + + /** + * 根据键名查询参数配置信息 + * + * @param configKey 参数键名 + * @return 参数键值 + */ + public String selectConfigByKey(String configKey); + + /** + * 获取验证码开关 + * + * @return true开启,false关闭 + */ + public boolean selectCaptchaEnabled(); + + /** + * 查询参数配置列表 + * + * @param config 参数配置信息 + * @return 参数配置集合 + */ + public List selectConfigList(SysConfig config); + + /** + * 新增参数配置 + * + * @param config 参数配置信息 + * @return 结果 + */ + public int insertConfig(SysConfig config); + + /** + * 修改参数配置 + * + * @param config 参数配置信息 + * @return 结果 + */ + public int updateConfig(SysConfig config); + + /** + * 批量删除参数信息 + * + * @param configIds 需要删除的参数ID + */ + public void deleteConfigByIds(Long[] configIds); + + /** + * 加载参数缓存数据 + */ + public void loadingConfigCache(); + + /** + * 清空参数缓存数据 + */ + public void clearConfigCache(); + + /** + * 重置参数缓存数据 + */ + public void resetConfigCache(); + + /** + * 校验参数键名是否唯一 + * + * @param config 参数信息 + * @return 结果 + */ + public String checkConfigKeyUnique(SysConfig config); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysDeptService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysDeptService.java new file mode 100644 index 00000000..c4e68eb8 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysDeptService.java @@ -0,0 +1,126 @@ +package com.ruoyi.system.service; + +import java.util.List; + +import com.baomidou.mybatisplus.extension.service.IService; +import com.ruoyi.common.core.domain.TreeSelect; +import com.ruoyi.common.core.domain.entity.SysDept; + +/** + * 部门管理 服务层 + * + * @author ruoyi + */ +public interface ISysDeptService extends IService +{ + /** + * 查询部门管理数据 + * + * @param dept 部门信息 + * @return 部门信息集合 + */ + public List selectDeptList(SysDept dept); + + /** + * 查询部门树结构信息 + * + * @param dept 部门信息 + * @return 部门树信息集合 + */ + public List selectDeptTreeList(SysDept dept); + + /** + * 构建前端所需要树结构 + * + * @param depts 部门列表 + * @return 树结构列表 + */ + public List buildDeptTree(List depts); + + /** + * 构建前端所需要下拉树结构 + * + * @param depts 部门列表 + * @return 下拉树结构列表 + */ + public List buildDeptTreeSelect(List depts); + + /** + * 根据角色ID查询部门树信息 + * + * @param roleId 角色ID + * @return 选中部门列表 + */ + public List selectDeptListByRoleId(Long roleId); + + /** + * 根据部门ID查询信息 + * + * @param deptId 部门ID + * @return 部门信息 + */ + public SysDept selectDeptById(Long deptId); + + /** + * 根据ID查询所有子部门(正常状态) + * + * @param deptId 部门ID + * @return 子部门数 + */ + public int selectNormalChildrenDeptById(Long deptId); + + /** + * 是否存在部门子节点 + * + * @param deptId 部门ID + * @return 结果 + */ + public boolean hasChildByDeptId(Long deptId); + + /** + * 查询部门是否存在用户 + * + * @param deptId 部门ID + * @return 结果 true 存在 false 不存在 + */ + public boolean checkDeptExistUser(Long deptId); + + /** + * 校验部门名称是否唯一 + * + * @param dept 部门信息 + * @return 结果 + */ + public String checkDeptNameUnique(SysDept dept); + + /** + * 校验部门是否有数据权限 + * + * @param deptId 部门id + */ + public void checkDeptDataScope(Long deptId); + + /** + * 新增保存部门信息 + * + * @param dept 部门信息 + * @return 结果 + */ + public int insertDept(SysDept dept); + + /** + * 修改保存部门信息 + * + * @param dept 部门信息 + * @return 结果 + */ + public int updateDept(SysDept dept); + + /** + * 删除部门管理信息 + * + * @param deptId 部门ID + * @return 结果 + */ + public int deleteDeptById(Long deptId); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysDictDataService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysDictDataService.java new file mode 100644 index 00000000..23d18826 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysDictDataService.java @@ -0,0 +1,62 @@ +package com.ruoyi.system.service; + +import java.util.List; + +import com.baomidou.mybatisplus.extension.service.IService; +import com.ruoyi.common.core.domain.entity.SysDictData; + +/** + * 字典 业务层 + * + * @author ruoyi + */ +public interface ISysDictDataService extends IService +{ + /** + * 根据条件分页查询字典数据 + * + * @param dictData 字典数据信息 + * @return 字典数据集合信息 + */ + public List selectDictDataList(SysDictData dictData); + + /** + * 根据字典类型和字典键值查询字典数据信息 + * + * @param dictType 字典类型 + * @param dictValue 字典键值 + * @return 字典标签 + */ + public String selectDictLabel(String dictType, String dictValue); + + /** + * 根据字典数据ID查询信息 + * + * @param dictCode 字典数据ID + * @return 字典数据 + */ + public SysDictData selectDictDataById(Long dictCode); + + /** + * 批量删除字典数据信息 + * + * @param dictCodes 需要删除的字典数据ID + */ + public void deleteDictDataByIds(Long[] dictCodes); + + /** + * 新增保存字典数据信息 + * + * @param dictData 字典数据信息 + * @return 结果 + */ + public int insertDictData(SysDictData dictData); + + /** + * 修改保存字典数据信息 + * + * @param dictData 字典数据信息 + * @return 结果 + */ + public int updateDictData(SysDictData dictData); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysDictTypeService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysDictTypeService.java new file mode 100644 index 00000000..31c60f76 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysDictTypeService.java @@ -0,0 +1,100 @@ +package com.ruoyi.system.service; + +import java.util.List; + +import com.baomidou.mybatisplus.extension.service.IService; +import com.ruoyi.common.core.domain.entity.SysDictData; +import com.ruoyi.common.core.domain.entity.SysDictType; + +/** + * 字典 业务层 + * + * @author ruoyi + */ +public interface ISysDictTypeService extends IService +{ + /** + * 根据条件分页查询字典类型 + * + * @param dictType 字典类型信息 + * @return 字典类型集合信息 + */ + public List selectDictTypeList(SysDictType dictType); + + /** + * 根据所有字典类型 + * + * @return 字典类型集合信息 + */ + public List selectDictTypeAll(); + + /** + * 根据字典类型查询字典数据 + * + * @param dictType 字典类型 + * @return 字典数据集合信息 + */ + public List selectDictDataByType(String dictType); + + /** + * 根据字典类型ID查询信息 + * + * @param dictId 字典类型ID + * @return 字典类型 + */ + public SysDictType selectDictTypeById(Long dictId); + + /** + * 根据字典类型查询信息 + * + * @param dictType 字典类型 + * @return 字典类型 + */ + public SysDictType selectDictTypeByType(String dictType); + + /** + * 批量删除字典信息 + * + * @param dictIds 需要删除的字典ID + */ + public void deleteDictTypeByIds(Long[] dictIds); + + /** + * 加载字典缓存数据 + */ + public void loadingDictCache(); + + /** + * 清空字典缓存数据 + */ + public void clearDictCache(); + + /** + * 重置字典缓存数据 + */ + public void resetDictCache(); + + /** + * 新增保存字典类型信息 + * + * @param dictType 字典类型信息 + * @return 结果 + */ + public int insertDictType(SysDictType dictType); + + /** + * 修改保存字典类型信息 + * + * @param dictType 字典类型信息 + * @return 结果 + */ + public int updateDictType(SysDictType dictType); + + /** + * 校验字典类型称是否唯一 + * + * @param dictType 字典类型 + * @return 结果 + */ + public String checkDictTypeUnique(SysDictType dictType); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysLogininforService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysLogininforService.java new file mode 100644 index 00000000..ce3151d3 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysLogininforService.java @@ -0,0 +1,40 @@ +package com.ruoyi.system.service; + +import java.util.List; +import com.ruoyi.system.domain.SysLogininfor; + +/** + * 系统访问日志情况信息 服务层 + * + * @author ruoyi + */ +public interface ISysLogininforService +{ + /** + * 新增系统登录日志 + * + * @param logininfor 访问日志对象 + */ + public void insertLogininfor(SysLogininfor logininfor); + + /** + * 查询系统登录日志集合 + * + * @param logininfor 访问日志对象 + * @return 登录记录集合 + */ + public List selectLogininforList(SysLogininfor logininfor); + + /** + * 批量删除系统登录日志 + * + * @param infoIds 需要删除的登录日志ID + * @return 结果 + */ + public int deleteLogininforByIds(Long[] infoIds); + + /** + * 清空系统登录日志 + */ + public void cleanLogininfor(); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysMenuService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysMenuService.java new file mode 100644 index 00000000..e9d3479f --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysMenuService.java @@ -0,0 +1,148 @@ +package com.ruoyi.system.service; + +import java.util.List; +import java.util.Set; + +import com.baomidou.mybatisplus.extension.service.IService; +import com.ruoyi.common.core.domain.TreeSelect; +import com.ruoyi.common.core.domain.entity.SysMenu; +import com.ruoyi.system.domain.vo.RouterVo; + +/** + * 菜单 业务层 + * + * @author ruoyi + */ +public interface ISysMenuService extends IService +{ + /** + * 根据用户查询系统菜单列表 + * + * @param userId 用户ID + * @return 菜单列表 + */ + public List selectMenuList(Long userId); + + /** + * 根据用户查询系统菜单列表 + * + * @param menu 菜单信息 + * @param userId 用户ID + * @return 菜单列表 + */ + public List selectMenuList(SysMenu menu, Long userId); + + /** + * 根据用户ID查询权限 + * + * @param userId 用户ID + * @return 权限列表 + */ + public Set selectMenuPermsByUserId(Long userId); + + /** + * 根据角色ID查询权限 + * + * @param roleId 角色ID + * @return 权限列表 + */ + public Set selectMenuPermsByRoleId(Long roleId); + + /** + * 根据用户ID查询菜单树信息 + * + * @param userId 用户ID + * @return 菜单列表 + */ + public List selectMenuTreeByUserId(Long userId); + + /** + * 根据角色ID查询菜单树信息 + * + * @param roleId 角色ID + * @return 选中菜单列表 + */ + public List selectMenuListByRoleId(Long roleId); + + /** + * 构建前端路由所需要的菜单 + * + * @param menus 菜单列表 + * @return 路由列表 + */ + public List buildMenus(List menus); + + /** + * 构建前端所需要树结构 + * + * @param menus 菜单列表 + * @return 树结构列表 + */ + public List buildMenuTree(List menus); + + /** + * 构建前端所需要下拉树结构 + * + * @param menus 菜单列表 + * @return 下拉树结构列表 + */ + public List buildMenuTreeSelect(List menus); + + /** + * 根据菜单ID查询信息 + * + * @param menuId 菜单ID + * @return 菜单信息 + */ + public SysMenu selectMenuById(Long menuId); + + /** + * 是否存在菜单子节点 + * + * @param menuId 菜单ID + * @return 结果 true 存在 false 不存在 + */ + public boolean hasChildByMenuId(Long menuId); + + /** + * 查询菜单是否存在角色 + * + * @param menuId 菜单ID + * @return 结果 true 存在 false 不存在 + */ + public boolean checkMenuExistRole(Long menuId); + + /** + * 新增保存菜单信息 + * + * @param menu 菜单信息 + * @return 结果 + */ + public int insertMenu(SysMenu menu); + + /** + * 修改保存菜单信息 + * + * @param menu 菜单信息 + * @return 结果 + */ + public int updateMenu(SysMenu menu); + + /** + * 删除菜单管理信息 + * + * @param menuId 菜单ID + * @return 结果 + */ + public int deleteMenuById(Long menuId); + + /** + * 校验菜单名称是否唯一 + * + * @param menu 菜单信息 + * @return 结果 + */ + public String checkMenuNameUnique(SysMenu menu); + + List getTenantMenus(SysMenu sysMenu); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysNoticeService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysNoticeService.java new file mode 100644 index 00000000..47ce1b71 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysNoticeService.java @@ -0,0 +1,60 @@ +package com.ruoyi.system.service; + +import java.util.List; +import com.ruoyi.system.domain.SysNotice; + +/** + * 公告 服务层 + * + * @author ruoyi + */ +public interface ISysNoticeService +{ + /** + * 查询公告信息 + * + * @param noticeId 公告ID + * @return 公告信息 + */ + public SysNotice selectNoticeById(Long noticeId); + + /** + * 查询公告列表 + * + * @param notice 公告信息 + * @return 公告集合 + */ + public List selectNoticeList(SysNotice notice); + + /** + * 新增公告 + * + * @param notice 公告信息 + * @return 结果 + */ + public int insertNotice(SysNotice notice); + + /** + * 修改公告 + * + * @param notice 公告信息 + * @return 结果 + */ + public int updateNotice(SysNotice notice); + + /** + * 删除公告信息 + * + * @param noticeId 公告ID + * @return 结果 + */ + public int deleteNoticeById(Long noticeId); + + /** + * 批量删除公告信息 + * + * @param noticeIds 需要删除的公告ID + * @return 结果 + */ + public int deleteNoticeByIds(Long[] noticeIds); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysOperLogService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysOperLogService.java new file mode 100644 index 00000000..4fd8e5a8 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysOperLogService.java @@ -0,0 +1,48 @@ +package com.ruoyi.system.service; + +import java.util.List; +import com.ruoyi.system.domain.SysOperLog; + +/** + * 操作日志 服务层 + * + * @author ruoyi + */ +public interface ISysOperLogService +{ + /** + * 新增操作日志 + * + * @param operLog 操作日志对象 + */ + public void insertOperlog(SysOperLog operLog); + + /** + * 查询系统操作日志集合 + * + * @param operLog 操作日志对象 + * @return 操作日志集合 + */ + public List selectOperLogList(SysOperLog operLog); + + /** + * 批量删除系统操作日志 + * + * @param operIds 需要删除的操作日志ID + * @return 结果 + */ + public int deleteOperLogByIds(Long[] operIds); + + /** + * 查询操作日志详细 + * + * @param operId 操作ID + * @return 操作日志对象 + */ + public SysOperLog selectOperLogById(Long operId); + + /** + * 清空操作日志 + */ + public void cleanOperLog(); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysPostService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysPostService.java new file mode 100644 index 00000000..7bfe2ecf --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysPostService.java @@ -0,0 +1,101 @@ +package com.ruoyi.system.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import com.ruoyi.system.domain.SysPost; + +import java.util.List; + +/** + * 岗位信息 服务层 + * + * @author ruoyi + */ +public interface ISysPostService extends IService +{ + /** + * 查询岗位信息集合 + * + * @param post 岗位信息 + * @return 岗位列表 + */ + public List selectPostList(SysPost post); + + /** + * 查询所有岗位 + * + * @return 岗位列表 + */ + public List selectPostAll(); + + /** + * 通过岗位ID查询岗位信息 + * + * @param postId 岗位ID + * @return 角色对象信息 + */ + public SysPost selectPostById(Long postId); + + /** + * 根据用户ID获取岗位选择框列表 + * + * @param userId 用户ID + * @return 选中岗位ID列表 + */ + public List selectPostListByUserId(Long userId); + + /** + * 校验岗位名称 + * + * @param post 岗位信息 + * @return 结果 + */ + public String checkPostNameUnique(SysPost post); + + /** + * 校验岗位编码 + * + * @param post 岗位信息 + * @return 结果 + */ + public String checkPostCodeUnique(SysPost post); + + /** + * 通过岗位ID查询岗位使用数量 + * + * @param postId 岗位ID + * @return 结果 + */ + public int countUserPostById(Long postId); + + /** + * 删除岗位信息 + * + * @param postId 岗位ID + * @return 结果 + */ + public int deletePostById(Long postId); + + /** + * 批量删除岗位信息 + * + * @param postIds 需要删除的岗位ID + * @return 结果 + */ + public int deletePostByIds(Long[] postIds); + + /** + * 新增保存岗位信息 + * + * @param post 岗位信息 + * @return 结果 + */ + public int insertPost(SysPost post); + + /** + * 修改保存岗位信息 + * + * @param post 岗位信息 + * @return 结果 + */ + public int updatePost(SysPost post); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysRoleService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysRoleService.java new file mode 100644 index 00000000..d0e63b23 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysRoleService.java @@ -0,0 +1,179 @@ +package com.ruoyi.system.service; + +import java.util.List; +import java.util.Set; + +import com.baomidou.mybatisplus.extension.service.IService; +import com.ruoyi.common.core.domain.entity.SysRole; +import com.ruoyi.system.domain.SysUserRole; + +/** + * 角色业务层 + * + * @author ruoyi + */ +public interface ISysRoleService extends IService +{ + /** + * 根据条件分页查询角色数据 + * + * @param role 角色信息 + * @return 角色数据集合信息 + */ + public List selectRoleList(SysRole role); + + /** + * 根据用户ID查询角色列表 + * + * @param userId 用户ID + * @return 角色列表 + */ + public List selectRolesByUserId(Long userId); + + /** + * 根据用户ID查询角色权限 + * + * @param userId 用户ID + * @return 权限列表 + */ + public Set selectRolePermissionByUserId(Long userId); + + /** + * 查询所有角色 + * + * @return 角色列表 + */ + public List selectRoleAll(); + + /** + * 根据用户ID获取角色选择框列表 + * + * @param userId 用户ID + * @return 选中角色ID列表 + */ + public List selectRoleListByUserId(Long userId); + + /** + * 通过角色ID查询角色 + * + * @param roleId 角色ID + * @return 角色对象信息 + */ + public SysRole selectRoleById(Long roleId); + + /** + * 校验角色名称是否唯一 + * + * @param role 角色信息 + * @return 结果 + */ + public String checkRoleNameUnique(SysRole role); + + /** + * 校验角色权限是否唯一 + * + * @param role 角色信息 + * @return 结果 + */ + public String checkRoleKeyUnique(SysRole role); + + /** + * 校验角色是否允许操作 + * + * @param role 角色信息 + */ + public void checkRoleAllowed(SysRole role); + + /** + * 校验角色是否有数据权限 + * + * @param roleId 角色id + */ + public void checkRoleDataScope(Long roleId); + + /** + * 通过角色ID查询角色使用数量 + * + * @param roleId 角色ID + * @return 结果 + */ + public int countUserRoleByRoleId(Long roleId); + + /** + * 新增保存角色信息 + * + * @param role 角色信息 + * @return 结果 + */ + public int insertRole(SysRole role); + + /** + * 修改保存角色信息 + * + * @param role 角色信息 + * @return 结果 + */ + public int updateRole(SysRole role); + + /** + * 修改角色状态 + * + * @param role 角色信息 + * @return 结果 + */ + public int updateRoleStatus(SysRole role); + + /** + * 修改数据权限信息 + * + * @param role 角色信息 + * @return 结果 + */ + public int authDataScope(SysRole role); + + /** + * 通过角色ID删除角色 + * + * @param roleId 角色ID + * @return 结果 + */ + public int deleteRoleById(Long roleId); + + /** + * 批量删除角色信息 + * + * @param roleIds 需要删除的角色ID + * @return 结果 + */ + public int deleteRoleByIds(Long[] roleIds); + + /** + * 取消授权用户角色 + * + * @param userRole 用户和角色关联信息 + * @return 结果 + */ + public int deleteAuthUser(SysUserRole userRole); + + /** + * 批量取消授权用户角色 + * + * @param roleId 角色ID + * @param userIds 需要取消授权的用户数据ID + * @return 结果 + */ + public int deleteAuthUsers(Long roleId, Long[] userIds); + + /** + * 批量选择授权用户角色 + * + * @param roleId 角色ID + * @param userIds 需要删除的用户数据ID + * @return 结果 + */ + public int insertAuthUsers(Long roleId, Long[] userIds); + + SysRole selectByRoleKey(String roleKey); + + Set getRoleMenuIds(Long roleId); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysTreeDictDataService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysTreeDictDataService.java new file mode 100644 index 00000000..ad017f4d --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysTreeDictDataService.java @@ -0,0 +1,107 @@ +package com.ruoyi.system.service; + +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.system.domain.SysTreeDict; +import com.ruoyi.system.domain.SysTreeDictData; + +import java.util.List; + +/** + * 树形字典数据Service接口 + * + * @author wangzongrun + * @date 2021-05-31 + */ +public interface ISysTreeDictDataService { + /** + * 查询树形字典数据 + * + * @param id 树形字典数据ID + * @return 树形字典数据 + */ + SysTreeDictData selectById(String id); + + /** + * ` + * 分页查询树形字典数据列表 + * + * @param page 分页 + * @param sysTreeDictData 树形字典数据 + * @return 树形字典数据集合 + */ + IPage selectList(IPage page, SysTreeDictData sysTreeDictData); + + /** + * 查询所有树形字典数据列表 + * + * @param sysTreeDictData 树形字典数据 + * @return 树形字典数据集合 + */ + List selectListAll(SysTreeDictData sysTreeDictData); + + /** + * 查询所有树形字典数据树结构 + * + * @param sysTreeDictData 树形字典数据 + * @return 树结构 + */ + List buildTree(SysTreeDictData sysTreeDictData); + + /** + * 新增树形字典数据 + * + * @param sysTreeDictData 树形字典数据 + * @return 结果 + */ + int insert(SysTreeDictData sysTreeDictData); + + /** + * 修改树形字典数据 + * + * @param sysTreeDictData 树形字典数据 + * @return 结果 + */ + int update(SysTreeDictData sysTreeDictData); + + /** + * 批量删除树形字典数据 + * + * @param sids 需要删除的树形字典数据ID + * @return 结果 + */ + int deleteByIds(String[] sids); + + /** + * 删除树形字典数据信息 + * + * @param id 树形字典数据ID + * @return 结果 + */ + int deleteById(String id); + + /** + * 检查名称唯一性 + * + * @param entity 查询条件 + * @return 结果 + */ + AjaxResult checkUniqueByLabel(SysTreeDictData entity); + + + /** + * 检查编码唯一性 + * + * @param treeDict 查询条件 + * @return 结果 + */ + List selectTreeDictByType(String treeDict); + + /** + * 校验编码 + * + * @param sysTreeDictData 查询条件 + * @return 结果 + */ + AjaxResult checkCode(SysTreeDictData sysTreeDictData); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysTreeDictService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysTreeDictService.java new file mode 100644 index 00000000..d705601e --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysTreeDictService.java @@ -0,0 +1,70 @@ +package com.ruoyi.system.service; + +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.ruoyi.system.domain.SysTreeDict; + +import java.util.List; + +/** + * 系统树形Service接口 + * + * @author wangzongrun + * @date 2021-06-04 + */ +public interface ISysTreeDictService { + /** + * 查询系统树形 + * + * @param id 系统树形ID + * @return 系统树形 + */ + public SysTreeDict selectById(String id); + + /** + * 分页查询系统树形列表 + * + * @param sysTreeDict 系统树形 + * @return 系统树形集合 + */ + public IPage selectList(IPage page, SysTreeDict sysTreeDict); + + /** + * 查询所有系统树形列表 + * + * @param sysTreeDict 系统树形 + * @return 系统树形集合 + */ + public List selectListAll(SysTreeDict sysTreeDict); + + /** + * 新增系统树形 + * + * @param sysTreeDict 系统树形 + * @return 结果 + */ + public int insert(SysTreeDict sysTreeDict); + + /** + * 修改系统树形 + * + * @param sysTreeDict 系统树形 + * @return 结果 + */ + public int update(SysTreeDict sysTreeDict); + + /** + * 批量删除系统树形 + * + * @param sids 需要删除的系统树形ID + * @return 结果 + */ + public int deleteByIds(String[] sids); + + /** + * 删除系统树形信息 + * + * @param id 系统树形ID + * @return 结果 + */ + public int deleteById(String id); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysUserOnlineService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysUserOnlineService.java new file mode 100644 index 00000000..8eb5448c --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysUserOnlineService.java @@ -0,0 +1,48 @@ +package com.ruoyi.system.service; + +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.system.domain.SysUserOnline; + +/** + * 在线用户 服务层 + * + * @author ruoyi + */ +public interface ISysUserOnlineService +{ + /** + * 通过登录地址查询信息 + * + * @param ipaddr 登录地址 + * @param user 用户信息 + * @return 在线用户信息 + */ + public SysUserOnline selectOnlineByIpaddr(String ipaddr, LoginUser user); + + /** + * 通过用户名称查询信息 + * + * @param userName 用户名称 + * @param user 用户信息 + * @return 在线用户信息 + */ + public SysUserOnline selectOnlineByUserName(String userName, LoginUser user); + + /** + * 通过登录地址/用户名称查询信息 + * + * @param ipaddr 登录地址 + * @param userName 用户名称 + * @param user 用户信息 + * @return 在线用户信息 + */ + public SysUserOnline selectOnlineByInfo(String ipaddr, String userName, LoginUser user); + + /** + * 设置在线用户信息 + * + * @param user 用户信息 + * @return 在线用户 + */ + public SysUserOnline loginUserToUserOnline(LoginUser user); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysUserService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysUserService.java new file mode 100644 index 00000000..2f9d390d --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/ISysUserService.java @@ -0,0 +1,227 @@ +package com.ruoyi.system.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import com.ruoyi.common.core.domain.entity.SysUser; + +import java.util.Collection; +import java.util.List; +import java.util.Set; + +/** + * 用户 业务层 + * + * @author ruoyi + */ +public interface ISysUserService extends IService { + /** + * 根据条件分页查询用户列表 + * + * @param user 用户信息 + * @return 用户信息集合信息 + */ + public List selectUserList(SysUser user); + + /** + * 根据条件分页查询已分配用户角色列表 + * + * @param user 用户信息 + * @return 用户信息集合信息 + */ + public List selectAllocatedList(SysUser user); + + /** + * 根据条件分页查询未分配用户角色列表 + * + * @param user 用户信息 + * @return 用户信息集合信息 + */ + public List selectUnallocatedList(SysUser user); + + /** + * 通过用户名查询用户 + * + * @param userName 用户名 + * @return 用户对象信息 + */ + public SysUser selectUserByUserName(String userName); + + /** + * 通过用户ID查询用户 + * + * @param userId 用户ID + * @return 用户对象信息 + */ + public SysUser selectUserById(Long userId); + + /** + * 根据用户ID查询用户所属角色组 + * + * @param userName 用户名 + * @return 结果 + */ + public String selectUserRoleGroup(String userName); + + /** + * 根据用户ID查询用户所属岗位组 + * + * @param userName 用户名 + * @return 结果 + */ + public String selectUserPostGroup(String userName); + + /** + * 校验用户名称是否唯一 + * + * @param user 用户信息 + * @return 结果 + */ + public String checkUserNameUnique(SysUser user); + + /** + * 校验手机号码是否唯一 + * + * @param user 用户信息 + * @return 结果 + */ + public String checkPhoneUnique(SysUser user); + + /** + * 校验email是否唯一 + * + * @param user 用户信息 + * @return 结果 + */ + public String checkEmailUnique(SysUser user); + + /** + * 校验用户是否允许操作 + * + * @param user 用户信息 + */ + public void checkUserAllowed(SysUser user); + + /** + * 校验用户是否有数据权限 + * + * @param userId 用户id + */ + public void checkUserDataScope(Long userId); + + /** + * 新增用户信息 + * + * @param user 用户信息 + * @return 结果 + */ + public int insertUser(SysUser user); + + /** + * 注册用户信息 + * + * @param user 用户信息 + * @return 结果 + */ + public boolean registerUser(SysUser user); + + /** + * 修改用户信息 + * + * @param user 用户信息 + * @return 结果 + */ + public int updateUser(SysUser user); + + /** + * 用户授权角色 + * + * @param userId 用户ID + * @param roleIds 角色组 + */ + public void insertUserAuth(Long userId, Long[] roleIds); + + /** + * 修改用户状态 + * + * @param user 用户信息 + * @return 结果 + */ + public int updateUserStatus(SysUser user); + + /** + * 修改用户基本信息 + * + * @param user 用户信息 + * @return 结果 + */ + public int updateUserProfile(SysUser user); + + /** + * 修改用户头像 + * + * @param userName 用户名 + * @param avatar 头像地址 + * @return 结果 + */ + public boolean updateUserAvatar(String userName, String avatar); + + /** + * 重置用户密码 + * + * @param user 用户信息 + * @return 结果 + */ + public int resetPwd(SysUser user); + + /** + * 重置用户密码 + * + * @param userName 用户名 + * @param password 密码 + * @return 结果 + */ + public int resetUserPwd(String userName, String password); + + /** + * 通过用户ID删除用户 + * + * @param userId 用户ID + * @return 结果 + */ + public int deleteUserById(Long userId); + + /** + * 批量删除用户信息 + * + * @param userIds 需要删除的用户ID + * @return 结果 + */ + public int deleteUserByIds(Long[] userIds); + + /** + * 导入用户数据 + * + * @param userList 用户数据列表 + * @param isUpdateSupport 是否更新支持,如果已存在,则进行更新数据 + * @param operName 操作用户 + * @return 结果 + */ + public String importUser(List userList, Boolean isUpdateSupport, String operName); + + /** + * 获得指定岗位的用户数组 + * + * @param postIds 岗位数组 + * @return 用户数组 + */ + List getUsersByPostIds(Collection postIds); + + /** + * 获得拥有多个角色的用户编号集合 + * + * @param roleIds 角色编号集合 + * @return 用户编号集合 + */ + Set getUserRoleIdListByRoleIds(Collection roleIds); + + List getUserListByStatus(Integer status); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/TenantPackageService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/TenantPackageService.java new file mode 100644 index 00000000..9b7820f7 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/TenantPackageService.java @@ -0,0 +1,74 @@ +package com.ruoyi.system.service; + + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.system.domain.TenantPackageDO; +import com.ruoyi.system.domain.vo.packages.TenantPackageCreateReqVO; +import com.ruoyi.system.domain.vo.packages.TenantPackagePageReqVO; +import com.ruoyi.system.domain.vo.packages.TenantPackageUpdateReqVO; + +import javax.validation.Valid; +import java.util.List; + +/** + * 租户套餐 Service 接口 + * + * @author 芋道源码 + */ +public interface TenantPackageService { + + /** + * 创建租户套餐 + * + * @param createReqVO 创建信息 + * @return 编号 + */ + Long createTenantPackage(@Valid TenantPackageCreateReqVO createReqVO); + + /** + * 更新租户套餐 + * + * @param updateReqVO 更新信息 + */ + void updateTenantPackage(@Valid TenantPackageUpdateReqVO updateReqVO); + + /** + * 删除租户套餐 + * + * @param id 编号 + */ + void deleteTenantPackage(Long id); + + /** + * 获得租户套餐 + * + * @param id 编号 + * @return 租户套餐 + */ + TenantPackageDO getTenantPackage(Long id); + + /** + * 获得租户套餐分页 + * + * @param pageReqVO 分页查询 + * @return 租户套餐分页 + */ + PageResult getTenantPackagePage(TenantPackagePageReqVO pageReqVO); + + /** + * 校验租户套餐 + * + * @param id 编号 + * @return 租户套餐 + */ + TenantPackageDO validTenantPackage(Long id); + + /** + * 获得指定状态的租户套餐列表 + * + * @param status 状态 + * @return 租户套餐 + */ + List getTenantPackageListByStatus(Integer status); + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/TenantService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/TenantService.java new file mode 100644 index 00000000..377b399b --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/TenantService.java @@ -0,0 +1,131 @@ +package com.ruoyi.system.service; + + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.system.domain.TenantDO; +import com.ruoyi.system.domain.vo.tenant.TenantCreateReqVO; +import com.ruoyi.system.domain.vo.tenant.TenantExportReqVO; +import com.ruoyi.system.domain.vo.tenant.TenantPageReqVO; +import com.ruoyi.system.domain.vo.tenant.TenantUpdateReqVO; +import com.ruoyi.system.service.handler.TenantInfoHandler; +import com.ruoyi.system.service.handler.TenantMenuHandler; + +import javax.validation.Valid; +import java.util.List; +import java.util.Set; + +/** + * 租户 Service 接口 + * + * @author 芋道源码 + */ +public interface TenantService { + + /** + * 创建租户 + * + * @param createReqVO 创建信息 + * @return 编号 + */ + Long createTenant(@Valid TenantCreateReqVO createReqVO); + + /** + * 更新租户 + * + * @param updateReqVO 更新信息 + */ + void updateTenant(@Valid TenantUpdateReqVO updateReqVO); + + /** + * 更新租户的角色菜单 + * + * @param tenantId 租户编号 + * @param menuIds 菜单编号数组 + */ + void updateTenantRoleMenu(Long tenantId, Set menuIds); + + /** + * 删除租户 + * + * @param id 编号 + */ + void deleteTenant(Long id); + + /** + * 获得租户 + * + * @param id 编号 + * @return 租户 + */ + TenantDO getTenant(Long id); + + /** + * 获得租户分页 + * + * @param pageReqVO 分页查询 + * @return 租户分页 + */ + PageResult getTenantPage(TenantPageReqVO pageReqVO); + + /** + * 获得租户列表, 用于 Excel 导出 + * + * @param exportReqVO 查询条件 + * @return 租户列表 + */ + List getTenantList(TenantExportReqVO exportReqVO); + + /** + * 获得名字对应的租户 + * + * @param name 组户名 + * @return 租户 + */ + TenantDO getTenantByName(String name); + + /** + * 获得使用指定套餐的租户数量 + * + * @param packageId 租户套餐编号 + * @return 租户数量 + */ + Long getTenantCountByPackageId(Long packageId); + + /** + * 获得使用指定套餐的租户数组 + * + * @param packageId 租户套餐编号 + * @return 租户数组 + */ + List getTenantListByPackageId(Long packageId); + + /** + * 进行租户的信息处理逻辑 + * 其中,租户编号从 {@link TenantContextHolder} 上下文中获取 + * + * @param handler 处理器 + */ + void handleTenantInfo(TenantInfoHandler handler); + + /** + * 进行租户的菜单处理逻辑 + * 其中,租户编号从 {@link TenantContextHolder} 上下文中获取 + * + * @param handler 处理器 + */ + void handleTenantMenu(TenantMenuHandler handler); + + /** + * 获得所有租户 + * + * @return 租户编号数组 + */ + List getTenantIds(); + + /** + * 校验租户是否合法 + * + * @param id 租户编号 + */ + void validTenant(Long id); +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/handler/TenantInfoHandler.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/handler/TenantInfoHandler.java new file mode 100644 index 00000000..78574c59 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/handler/TenantInfoHandler.java @@ -0,0 +1,22 @@ +package com.ruoyi.system.service.handler; + + +import com.ruoyi.system.domain.TenantDO; + +/** + * 租户信息处理 + * 目的:尽量减少租户逻辑耦合到系统中 + * + * @author 芋道源码 + */ +public interface TenantInfoHandler { + + /** + * 基于传入的租户信息,进行相关逻辑的执行 + * 例如说,创建用户时,超过最大账户配额 + * + * @param tenant 租户信息 + */ + void handle(TenantDO tenant); + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/handler/TenantMenuHandler.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/handler/TenantMenuHandler.java new file mode 100644 index 00000000..e1831c95 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/handler/TenantMenuHandler.java @@ -0,0 +1,21 @@ +package com.ruoyi.system.service.handler; + +import java.util.Set; + +/** + * 租户菜单处理 + * 目的:尽量减少租户逻辑耦合到系统中 + * + * @author 芋道源码 + */ +public interface TenantMenuHandler { + + /** + * 基于传入的租户菜单【全】列表,进行相关逻辑的执行 + * 例如说,返回可分配菜单的时候,可以移除多余的 + * + * @param menuIds 菜单列表 + */ + void handle(Set menuIds); + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysConfigServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysConfigServiceImpl.java new file mode 100644 index 00000000..62549f90 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysConfigServiceImpl.java @@ -0,0 +1,235 @@ +package com.ruoyi.system.service.impl; + +import java.util.Collection; +import java.util.List; +import javax.annotation.PostConstruct; + +import com.ruoyi.common.utils.EhcacheUtil; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +import com.ruoyi.common.annotation.DataSource; +import com.ruoyi.common.constant.CacheConstants; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.core.text.Convert; +import com.ruoyi.common.enums.DataSourceType; +import com.ruoyi.common.exception.ServiceException; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.system.domain.SysConfig; +import com.ruoyi.system.mapper.SysConfigMapper; +import com.ruoyi.system.service.ISysConfigService; + +/** + * 参数配置 服务层实现 + * + * @author ruoyi + */ +@Service +public class SysConfigServiceImpl implements ISysConfigService +{ + @Autowired + private SysConfigMapper configMapper; + +// @Autowired +// private RedisCache redisCache; + + /** + * 项目启动时,初始化参数到缓存 + */ + @PostConstruct + public void init() + { + loadingConfigCache(); + } + + /** + * 查询参数配置信息 + * + * @param configId 参数配置ID + * @return 参数配置信息 + */ + @Override + @DataSource(DataSourceType.MASTER) + public SysConfig selectConfigById(Long configId) + { + SysConfig config = new SysConfig(); + config.setConfigId(configId); + return configMapper.selectConfig(config); + } + + /** + * 根据键名查询参数配置信息 + * + * @param configKey 参数key + * @return 参数键值 + */ + @Override + public String selectConfigByKey(String configKey) + { +// String configValue = Convert.toStr(redisCache.getCacheObject(getCacheKey(configKey))); + Object obj = EhcacheUtil.get("sysconfig", getCacheKey(configKey)); + String configValue = Convert.toStr(obj); + if (StringUtils.isNotEmpty(configValue)) + { + return configValue; + } + SysConfig config = new SysConfig(); + config.setConfigKey(configKey); + SysConfig retConfig = configMapper.selectConfig(config); + if (StringUtils.isNotNull(retConfig)) + { + EhcacheUtil.put("sysconfig", getCacheKey(configKey),retConfig.getConfigValue()); +// redisCache.setCacheObject(getCacheKey(configKey), retConfig.getConfigValue()); + return retConfig.getConfigValue(); + } + return StringUtils.EMPTY; + } + + /** + * 获取验证码开关 + * + * @return true开启,false关闭 + */ + @Override + public boolean selectCaptchaEnabled() + { + String captchaEnabled = selectConfigByKey("sys.account.captchaEnabled"); + if (StringUtils.isEmpty(captchaEnabled)) + { + return true; + } + return Convert.toBool(captchaEnabled); + } + + /** + * 查询参数配置列表 + * + * @param config 参数配置信息 + * @return 参数配置集合 + */ + @Override + public List selectConfigList(SysConfig config) + { + return configMapper.selectConfigList(config); + } + + /** + * 新增参数配置 + * + * @param config 参数配置信息 + * @return 结果 + */ + @Override + public int insertConfig(SysConfig config) + { + int row = configMapper.insertConfig(config); + if (row > 0) + { + EhcacheUtil.put("sysconfig", getCacheKey(config.getConfigKey()), config.getConfigValue()); +// redisCache.setCacheObject(getCacheKey(config.getConfigKey()), config.getConfigValue()); + } + return row; + } + + /** + * 修改参数配置 + * + * @param config 参数配置信息 + * @return 结果 + */ + @Override + public int updateConfig(SysConfig config) + { + int row = configMapper.updateConfig(config); + if (row > 0) + { + EhcacheUtil.put("sysconfig", getCacheKey(config.getConfigKey()), config.getConfigValue()); +// redisCache.setCacheObject(getCacheKey(config.getConfigKey()), config.getConfigValue()); + } + return row; + } + + /** + * 批量删除参数信息 + * + * @param configIds 需要删除的参数ID + */ + @Override + public void deleteConfigByIds(Long[] configIds) + { + for (Long configId : configIds) + { + SysConfig config = selectConfigById(configId); + if (StringUtils.equals(UserConstants.YES, config.getConfigType())) + { + throw new ServiceException(String.format("内置参数【%1$s】不能删除 ", config.getConfigKey())); + } + configMapper.deleteConfigById(configId); + EhcacheUtil.remove("sysconfig", getCacheKey(config.getConfigKey())); +// redisCache.deleteObject(getCacheKey(config.getConfigKey())); + } + } + + /** + * 加载参数缓存数据 + */ + @Override + public void loadingConfigCache() + { + List configsList = configMapper.selectConfigList(new SysConfig()); + for (SysConfig config : configsList) + { + EhcacheUtil.put("sysconfig", getCacheKey(config.getConfigKey()),config.getConfigValue()); +// redisCache.setCacheObject(getCacheKey(config.getConfigKey()), config.getConfigValue()); + } + } + + /** + * 清空参数缓存数据 + */ + @Override + public void clearConfigCache() + { + EhcacheUtil.removeAll("sysconfig"); +// Collection keys = redisCache.keys(CacheConstants.SYS_CONFIG_KEY + "*"); +// redisCache.deleteObject(keys); + } + + /** + * 重置参数缓存数据 + */ + @Override + public void resetConfigCache() + { + clearConfigCache(); + loadingConfigCache(); + } + + /** + * 校验参数键名是否唯一 + * + * @param config 参数配置信息 + * @return 结果 + */ + @Override + public String checkConfigKeyUnique(SysConfig config) + { + Long configId = StringUtils.isNull(config.getConfigId()) ? -1L : config.getConfigId(); + SysConfig info = configMapper.checkConfigKeyUnique(config.getConfigKey()); + if (StringUtils.isNotNull(info) && info.getConfigId().longValue() != configId.longValue()) + { + return UserConstants.NOT_UNIQUE; + } + return UserConstants.UNIQUE; + } + + /** + * 设置cache key + * + * @param configKey 参数键 + * @return 缓存键key + */ + private String getCacheKey(String configKey) + { + return CacheConstants.SYS_CONFIG_KEY + configKey; + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysDeptServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysDeptServiceImpl.java new file mode 100644 index 00000000..a9383287 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysDeptServiceImpl.java @@ -0,0 +1,344 @@ +package com.ruoyi.system.service.impl; + +import java.util.ArrayList; +import java.util.Iterator; +import java.util.List; +import java.util.stream.Collectors; + +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +import com.ruoyi.common.annotation.DataScope; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.core.domain.TreeSelect; +import com.ruoyi.common.core.domain.entity.SysDept; +import com.ruoyi.common.core.domain.entity.SysRole; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.core.text.Convert; +import com.ruoyi.common.exception.ServiceException; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.spring.SpringUtils; +import com.ruoyi.system.mapper.SysDeptMapper; +import com.ruoyi.system.mapper.SysRoleMapper; +import com.ruoyi.system.service.ISysDeptService; + +/** + * 部门管理 服务实现 + * + * @author ruoyi + */ +@Service +public class SysDeptServiceImpl extends ServiceImpl implements ISysDeptService +{ + @Autowired + private SysDeptMapper deptMapper; + + @Autowired + private SysRoleMapper roleMapper; + + /** + * 查询部门管理数据 + * + * @param dept 部门信息 + * @return 部门信息集合 + */ + @Override + @DataScope(deptAlias = "d") + public List selectDeptList(SysDept dept) + { + return deptMapper.selectDeptList(dept); + } + + /** + * 查询部门树结构信息 + * + * @param dept 部门信息 + * @return 部门树信息集合 + */ + @Override + public List selectDeptTreeList(SysDept dept) + { + List depts = SpringUtils.getAopProxy(this).selectDeptList(dept); + return buildDeptTreeSelect(depts); + } + + /** + * 构建前端所需要树结构 + * + * @param depts 部门列表 + * @return 树结构列表 + */ + @Override + public List buildDeptTree(List depts) + { + List returnList = new ArrayList(); + List tempList = new ArrayList(); + for (SysDept dept : depts) + { + tempList.add(dept.getDeptId()); + } + for (SysDept dept : depts) + { + // 如果是顶级节点, 遍历该父节点的所有子节点 + if (!tempList.contains(dept.getParentId())) + { + recursionFn(depts, dept); + returnList.add(dept); + } + } + if (returnList.isEmpty()) + { + returnList = depts; + } + return returnList; + } + + /** + * 构建前端所需要下拉树结构 + * + * @param depts 部门列表 + * @return 下拉树结构列表 + */ + @Override + public List buildDeptTreeSelect(List depts) + { + List deptTrees = buildDeptTree(depts); + return deptTrees.stream().map(TreeSelect::new).collect(Collectors.toList()); + } + + /** + * 根据角色ID查询部门树信息 + * + * @param roleId 角色ID + * @return 选中部门列表 + */ + @Override + public List selectDeptListByRoleId(Long roleId) + { + SysRole role = roleMapper.selectRoleById(roleId); + return deptMapper.selectDeptListByRoleId(roleId, role.isDeptCheckStrictly()); + } + + /** + * 根据部门ID查询信息 + * + * @param deptId 部门ID + * @return 部门信息 + */ + @Override + public SysDept selectDeptById(Long deptId) + { + return deptMapper.selectDeptById(deptId); + } + + /** + * 根据ID查询所有子部门(正常状态) + * + * @param deptId 部门ID + * @return 子部门数 + */ + @Override + public int selectNormalChildrenDeptById(Long deptId) + { + return deptMapper.selectNormalChildrenDeptById(deptId); + } + + /** + * 是否存在子节点 + * + * @param deptId 部门ID + * @return 结果 + */ + @Override + public boolean hasChildByDeptId(Long deptId) + { + int result = deptMapper.hasChildByDeptId(deptId); + return result > 0; + } + + /** + * 查询部门是否存在用户 + * + * @param deptId 部门ID + * @return 结果 true 存在 false 不存在 + */ + @Override + public boolean checkDeptExistUser(Long deptId) + { + int result = deptMapper.checkDeptExistUser(deptId); + return result > 0; + } + + /** + * 校验部门名称是否唯一 + * + * @param dept 部门信息 + * @return 结果 + */ + @Override + public String checkDeptNameUnique(SysDept dept) + { + Long deptId = StringUtils.isNull(dept.getDeptId()) ? -1L : dept.getDeptId(); + SysDept info = deptMapper.checkDeptNameUnique(dept.getDeptName(), dept.getParentId()); + if (StringUtils.isNotNull(info) && info.getDeptId().longValue() != deptId.longValue()) + { + return UserConstants.NOT_UNIQUE; + } + return UserConstants.UNIQUE; + } + + /** + * 校验部门是否有数据权限 + * + * @param deptId 部门id + */ + @Override + public void checkDeptDataScope(Long deptId) + { + if (!SysUser.isAdmin(SecurityUtils.getUserId())) + { + SysDept dept = new SysDept(); + dept.setDeptId(deptId); + List depts = SpringUtils.getAopProxy(this).selectDeptList(dept); + if (StringUtils.isEmpty(depts)) + { + throw new ServiceException("没有权限访问部门数据!"); + } + } + } + + /** + * 新增保存部门信息 + * + * @param dept 部门信息 + * @return 结果 + */ + @Override + public int insertDept(SysDept dept) + { + SysDept info = deptMapper.selectDeptById(dept.getParentId()); + // 如果父节点不为正常状态,则不允许新增子节点 + if (!UserConstants.DEPT_NORMAL.equals(info.getStatus())) + { + throw new ServiceException("部门停用,不允许新增"); + } + dept.setAncestors(info.getAncestors() + "," + dept.getParentId()); + return deptMapper.insertDept(dept); + } + + /** + * 修改保存部门信息 + * + * @param dept 部门信息 + * @return 结果 + */ + @Override + public int updateDept(SysDept dept) + { + SysDept newParentDept = deptMapper.selectDeptById(dept.getParentId()); + SysDept oldDept = deptMapper.selectDeptById(dept.getDeptId()); + if (StringUtils.isNotNull(newParentDept) && StringUtils.isNotNull(oldDept)) + { + String newAncestors = newParentDept.getAncestors() + "," + newParentDept.getDeptId(); + String oldAncestors = oldDept.getAncestors(); + dept.setAncestors(newAncestors); + updateDeptChildren(dept.getDeptId(), newAncestors, oldAncestors); + } + int result = deptMapper.updateDept(dept); + if (UserConstants.DEPT_NORMAL.equals(dept.getStatus()) && StringUtils.isNotEmpty(dept.getAncestors()) + && !StringUtils.equals("0", dept.getAncestors())) + { + // 如果该部门是启用状态,则启用该部门的所有上级部门 + updateParentDeptStatusNormal(dept); + } + return result; + } + + /** + * 修改该部门的父级部门状态 + * + * @param dept 当前部门 + */ + private void updateParentDeptStatusNormal(SysDept dept) + { + String ancestors = dept.getAncestors(); + Long[] deptIds = Convert.toLongArray(ancestors); + deptMapper.updateDeptStatusNormal(deptIds); + } + + /** + * 修改子元素关系 + * + * @param deptId 被修改的部门ID + * @param newAncestors 新的父ID集合 + * @param oldAncestors 旧的父ID集合 + */ + public void updateDeptChildren(Long deptId, String newAncestors, String oldAncestors) + { + List children = deptMapper.selectChildrenDeptById(deptId); + for (SysDept child : children) + { + child.setAncestors(child.getAncestors().replaceFirst(oldAncestors, newAncestors)); + } + if (children.size() > 0) + { + deptMapper.updateDeptChildren(children); + } + } + + /** + * 删除部门管理信息 + * + * @param deptId 部门ID + * @return 结果 + */ + @Override + public int deleteDeptById(Long deptId) + { + return deptMapper.deleteDeptById(deptId); + } + + /** + * 递归列表 + */ + private void recursionFn(List list, SysDept t) + { + // 得到子节点列表 + List childList = getChildList(list, t); + t.setChildren(childList); + for (SysDept tChild : childList) + { + if (hasChild(list, tChild)) + { + recursionFn(list, tChild); + } + } + } + + /** + * 得到子节点列表 + */ + private List getChildList(List list, SysDept t) + { + List tlist = new ArrayList(); + Iterator it = list.iterator(); + while (it.hasNext()) + { + SysDept n = (SysDept) it.next(); + if (StringUtils.isNotNull(n.getParentId()) && n.getParentId().longValue() == t.getDeptId().longValue()) + { + tlist.add(n); + } + } + return tlist; + } + + /** + * 判断是否有子节点 + */ + private boolean hasChild(List list, SysDept t) + { + return getChildList(list, t).size() > 0; + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysDictDataServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysDictDataServiceImpl.java new file mode 100644 index 00000000..40c1d673 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysDictDataServiceImpl.java @@ -0,0 +1,113 @@ +package com.ruoyi.system.service.impl; + +import java.util.List; + +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +import com.ruoyi.common.core.domain.entity.SysDictData; +import com.ruoyi.common.utils.DictUtils; +import com.ruoyi.system.mapper.SysDictDataMapper; +import com.ruoyi.system.service.ISysDictDataService; + +/** + * 字典 业务层处理 + * + * @author ruoyi + */ +@Service +public class SysDictDataServiceImpl extends ServiceImpl implements ISysDictDataService +{ + @Autowired + private SysDictDataMapper dictDataMapper; + + /** + * 根据条件分页查询字典数据 + * + * @param dictData 字典数据信息 + * @return 字典数据集合信息 + */ + @Override + public List selectDictDataList(SysDictData dictData) + { + return dictDataMapper.selectDictDataList(dictData); + } + + /** + * 根据字典类型和字典键值查询字典数据信息 + * + * @param dictType 字典类型 + * @param dictValue 字典键值 + * @return 字典标签 + */ + @Override + public String selectDictLabel(String dictType, String dictValue) + { + return dictDataMapper.selectDictLabel(dictType, dictValue); + } + + /** + * 根据字典数据ID查询信息 + * + * @param dictCode 字典数据ID + * @return 字典数据 + */ + @Override + public SysDictData selectDictDataById(Long dictCode) + { + return dictDataMapper.selectDictDataById(dictCode); + } + + /** + * 批量删除字典数据信息 + * + * @param dictCodes 需要删除的字典数据ID + */ + @Override + public void deleteDictDataByIds(Long[] dictCodes) + { + for (Long dictCode : dictCodes) + { + SysDictData data = selectDictDataById(dictCode); + dictDataMapper.deleteDictDataById(dictCode); + List dictDatas = dictDataMapper.selectDictDataByType(data.getDictType()); + DictUtils.setDictCache(data.getDictType(), dictDatas); + } + } + + /** + * 新增保存字典数据信息 + * + * @param data 字典数据信息 + * @return 结果 + */ + @Override + public int insertDictData(SysDictData data) + { + int row = dictDataMapper.insertDictData(data); + if (row > 0) + { + List dictDatas = dictDataMapper.selectDictDataByType(data.getDictType()); + DictUtils.setDictCache(data.getDictType(), dictDatas); + } + return row; + } + + /** + * 修改保存字典数据信息 + * + * @param data 字典数据信息 + * @return 结果 + */ + @Override + public int updateDictData(SysDictData data) + { + int row = dictDataMapper.updateDictData(data); + if (row > 0) + { + List dictDatas = dictDataMapper.selectDictDataByType(data.getDictType()); + DictUtils.setDictCache(data.getDictType(), dictDatas); + } + return row; + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysDictTypeServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysDictTypeServiceImpl.java new file mode 100644 index 00000000..e8f0b1b9 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysDictTypeServiceImpl.java @@ -0,0 +1,225 @@ +package com.ruoyi.system.service.impl; + +import java.util.Comparator; +import java.util.List; +import java.util.Map; +import java.util.stream.Collectors; +import javax.annotation.PostConstruct; + +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.core.domain.entity.SysDictData; +import com.ruoyi.common.core.domain.entity.SysDictType; +import com.ruoyi.common.exception.ServiceException; +import com.ruoyi.common.utils.DictUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.system.mapper.SysDictDataMapper; +import com.ruoyi.system.mapper.SysDictTypeMapper; +import com.ruoyi.system.service.ISysDictTypeService; + +/** + * 字典 业务层处理 + * + * @author ruoyi + */ +@Service +public class SysDictTypeServiceImpl extends ServiceImpl implements ISysDictTypeService +{ + @Autowired + private SysDictTypeMapper dictTypeMapper; + + @Autowired + private SysDictDataMapper dictDataMapper; + + /** + * 项目启动时,初始化字典到缓存 + */ + @PostConstruct + public void init() + { + loadingDictCache(); + } + + /** + * 根据条件分页查询字典类型 + * + * @param dictType 字典类型信息 + * @return 字典类型集合信息 + */ + @Override + public List selectDictTypeList(SysDictType dictType) + { + return dictTypeMapper.selectDictTypeList(dictType); + } + + /** + * 根据所有字典类型 + * + * @return 字典类型集合信息 + */ + @Override + public List selectDictTypeAll() + { + return dictTypeMapper.selectDictTypeAll(); + } + + /** + * 根据字典类型查询字典数据 + * + * @param dictType 字典类型 + * @return 字典数据集合信息 + */ + @Override + public List selectDictDataByType(String dictType) + { + List dictDatas = DictUtils.getDictCache(dictType); + if (StringUtils.isNotEmpty(dictDatas)) + { + return dictDatas; + } + dictDatas = dictDataMapper.selectDictDataByType(dictType); + if (StringUtils.isNotEmpty(dictDatas)) + { + DictUtils.setDictCache(dictType, dictDatas); + return dictDatas; + } + return null; + } + + /** + * 根据字典类型ID查询信息 + * + * @param dictId 字典类型ID + * @return 字典类型 + */ + @Override + public SysDictType selectDictTypeById(Long dictId) + { + return dictTypeMapper.selectDictTypeById(dictId); + } + + /** + * 根据字典类型查询信息 + * + * @param dictType 字典类型 + * @return 字典类型 + */ + @Override + public SysDictType selectDictTypeByType(String dictType) + { + return dictTypeMapper.selectDictTypeByType(dictType); + } + + /** + * 批量删除字典类型信息 + * + * @param dictIds 需要删除的字典ID + */ + @Override + public void deleteDictTypeByIds(Long[] dictIds) + { + for (Long dictId : dictIds) + { + SysDictType dictType = selectDictTypeById(dictId); + if (dictDataMapper.countDictDataByType(dictType.getDictType()) > 0) + { + throw new ServiceException(String.format("%1$s已分配,不能删除", dictType.getDictName())); + } + dictTypeMapper.deleteDictTypeById(dictId); + DictUtils.removeDictCache(dictType.getDictType()); + } + } + + /** + * 加载字典缓存数据 + */ + @Override + public void loadingDictCache() + { + SysDictData dictData = new SysDictData(); + dictData.setStatus("0"); + Map> dictDataMap = dictDataMapper.selectDictDataList(dictData).stream().collect(Collectors.groupingBy(SysDictData::getDictType)); + for (Map.Entry> entry : dictDataMap.entrySet()) + { + DictUtils.setDictCache(entry.getKey(), entry.getValue().stream().sorted(Comparator.comparing(SysDictData::getDictSort)).collect(Collectors.toList())); + } + } + + /** + * 清空字典缓存数据 + */ + @Override + public void clearDictCache() + { + DictUtils.clearDictCache(); + } + + /** + * 重置字典缓存数据 + */ + @Override + public void resetDictCache() + { + clearDictCache(); + loadingDictCache(); + } + + /** + * 新增保存字典类型信息 + * + * @param dict 字典类型信息 + * @return 结果 + */ + @Override + public int insertDictType(SysDictType dict) + { + int row = dictTypeMapper.insertDictType(dict); + if (row > 0) + { + DictUtils.setDictCache(dict.getDictType(), null); + } + return row; + } + + /** + * 修改保存字典类型信息 + * + * @param dict 字典类型信息 + * @return 结果 + */ + @Override + @Transactional + public int updateDictType(SysDictType dict) + { + SysDictType oldDict = dictTypeMapper.selectDictTypeById(dict.getDictId()); + dictDataMapper.updateDictDataType(oldDict.getDictType(), dict.getDictType()); + int row = dictTypeMapper.updateDictType(dict); + if (row > 0) + { + List dictDatas = dictDataMapper.selectDictDataByType(dict.getDictType()); + DictUtils.setDictCache(dict.getDictType(), dictDatas); + } + return row; + } + + /** + * 校验字典类型称是否唯一 + * + * @param dict 字典类型 + * @return 结果 + */ + @Override + public String checkDictTypeUnique(SysDictType dict) + { + Long dictId = StringUtils.isNull(dict.getDictId()) ? -1L : dict.getDictId(); + SysDictType dictType = dictTypeMapper.checkDictTypeUnique(dict.getDictType()); + if (StringUtils.isNotNull(dictType) && dictType.getDictId().longValue() != dictId.longValue()) + { + return UserConstants.NOT_UNIQUE; + } + return UserConstants.UNIQUE; + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysLogininforServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysLogininforServiceImpl.java new file mode 100644 index 00000000..f013d18a --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysLogininforServiceImpl.java @@ -0,0 +1,68 @@ +package com.ruoyi.system.service.impl; + +import java.util.List; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +import com.ruoyi.system.domain.SysLogininfor; +import com.ruoyi.system.mapper.SysLogininforMapper; +import com.ruoyi.system.service.ISysLogininforService; + +/** + * 系统访问日志情况信息 服务层处理 + * + * @author ruoyi + */ +@Service +public class SysLogininforServiceImpl implements ISysLogininforService +{ + + @Autowired + private SysLogininforMapper logininforMapper; + + /** + * 新增系统登录日志 + * + * @param logininfor 访问日志对象 + */ + @Override + public void insertLogininfor(SysLogininfor logininfor) + { + if (logininfor.getMsg() != null && logininfor.getMsg().length() >255) { + logininfor.setMsg(logininfor.getMsg().substring(0,250)); + } + logininforMapper.insertLogininfor(logininfor); + } + + /** + * 查询系统登录日志集合 + * + * @param logininfor 访问日志对象 + * @return 登录记录集合 + */ + @Override + public List selectLogininforList(SysLogininfor logininfor) + { + return logininforMapper.selectLogininforList(logininfor); + } + + /** + * 批量删除系统登录日志 + * + * @param infoIds 需要删除的登录日志ID + * @return 结果 + */ + @Override + public int deleteLogininforByIds(Long[] infoIds) + { + return logininforMapper.deleteLogininforByIds(infoIds); + } + + /** + * 清空系统登录日志 + */ + @Override + public void cleanLogininfor() + { + logininforMapper.cleanLogininfor(); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysMenuServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysMenuServiceImpl.java new file mode 100644 index 00000000..58d2b0c2 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysMenuServiceImpl.java @@ -0,0 +1,562 @@ +package com.ruoyi.system.service.impl; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashSet; +import java.util.Iterator; +import java.util.LinkedList; +import java.util.List; +import java.util.Set; +import java.util.stream.Collectors; + +import cn.hutool.core.collection.CollUtil; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import com.ruoyi.system.service.TenantService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.context.annotation.Lazy; +import org.springframework.stereotype.Service; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.core.domain.TreeSelect; +import com.ruoyi.common.core.domain.entity.SysMenu; +import com.ruoyi.common.core.domain.entity.SysRole; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.system.domain.vo.MetaVo; +import com.ruoyi.system.domain.vo.RouterVo; +import com.ruoyi.system.mapper.SysMenuMapper; +import com.ruoyi.system.mapper.SysRoleMapper; +import com.ruoyi.system.mapper.SysRoleMenuMapper; +import com.ruoyi.system.service.ISysMenuService; + +import javax.annotation.Resource; + +/** + * 菜单 业务层处理 + * + * @author ruoyi + */ +@Service +public class SysMenuServiceImpl extends ServiceImpl implements ISysMenuService +{ + public static final String PREMISSION_STRING = "perms[\"{0}\"]"; + + @Autowired + private SysMenuMapper menuMapper; + + @Autowired + private SysRoleMapper roleMapper; + + @Autowired + private SysRoleMenuMapper roleMenuMapper; + @Resource + @Lazy // 延迟,避免循环依赖报错 + private TenantService tenantService; + /** + * 根据用户查询系统菜单列表 + * + * @param userId 用户ID + * @return 菜单列表 + */ + @Override + public List selectMenuList(Long userId) + { + return selectMenuList(new SysMenu(), userId); + } + + /** + * 查询系统菜单列表 + * + * @param menu 菜单信息 + * @return 菜单列表 + */ + @Override + public List selectMenuList(SysMenu menu, Long userId) + { + List menuList = null; + // 管理员显示所有菜单信息 + if (SysUser.isAdmin(userId)) + { + menuList = menuMapper.selectMenuList(menu); + } + else + { + menu.getParams().put("userId", userId); + menuList = menuMapper.selectMenuListByUserId(menu); + } + return menuList; + } + + /** + * 根据用户ID查询权限 + * + * @param userId 用户ID + * @return 权限列表 + */ + @Override + public Set selectMenuPermsByUserId(Long userId) + { + List perms = menuMapper.selectMenuPermsByUserId(userId); + Set permsSet = new HashSet<>(); + for (String perm : perms) + { + if (StringUtils.isNotEmpty(perm)) + { + permsSet.addAll(Arrays.asList(perm.trim().split(","))); + } + } + return permsSet; + } + + /** + * 根据角色ID查询权限 + * + * @param roleId 角色ID + * @return 权限列表 + */ + @Override + public Set selectMenuPermsByRoleId(Long roleId) + { + List perms = menuMapper.selectMenuPermsByRoleId(roleId); + Set permsSet = new HashSet<>(); + for (String perm : perms) + { + if (StringUtils.isNotEmpty(perm)) + { + permsSet.addAll(Arrays.asList(perm.trim().split(","))); + } + } + return permsSet; + } + + /** + * 根据用户ID查询菜单 + * + * @param userId 用户名称 + * @return 菜单列表 + */ + @Override + public List selectMenuTreeByUserId(Long userId) + { + List menus = null; + if (SecurityUtils.isAdmin(userId)) + { + menus = menuMapper.selectMenuTreeAll(); + } + else + { + menus = menuMapper.selectMenuTreeByUserId(userId); + } + return getChildPerms(menus, 0); + } + + /** + * 根据角色ID查询菜单树信息 + * + * @param roleId 角色ID + * @return 选中菜单列表 + */ + @Override + public List selectMenuListByRoleId(Long roleId) + { + SysRole role = roleMapper.selectRoleById(roleId); + return menuMapper.selectMenuListByRoleId(roleId, role.isMenuCheckStrictly()); + } + + /** + * 构建前端路由所需要的菜单 + * + * @param menus 菜单列表 + * @return 路由列表 + */ + @Override + public List buildMenus(List menus) + { + List routers = new LinkedList(); + for (SysMenu menu : menus) + { + RouterVo router = new RouterVo(); + router.setHidden("1".equals(menu.getVisible())); + router.setName(getRouteName(menu)); + router.setPath(getRouterPath(menu)); + router.setComponent(getComponent(menu)); + router.setQuery(menu.getQuery()); + router.setMeta(new MetaVo(menu.getMenuName(), menu.getIcon(), StringUtils.equals("1", menu.getIsCache()), menu.getPath())); + List cMenus = menu.getChildren(); + if (!cMenus.isEmpty() && cMenus.size() > 0 && UserConstants.TYPE_DIR.equals(menu.getMenuType())) + { + router.setAlwaysShow(true); + router.setRedirect("noRedirect"); + router.setChildren(buildMenus(cMenus)); + } + else if (isMenuFrame(menu)) + { + router.setMeta(null); + List childrenList = new ArrayList(); + RouterVo children = new RouterVo(); + children.setPath(menu.getPath()); + children.setComponent(menu.getComponent()); + children.setName(StringUtils.capitalize(menu.getPath())); + children.setMeta(new MetaVo(menu.getMenuName(), menu.getIcon(), StringUtils.equals("1", menu.getIsCache()), menu.getPath())); + children.setQuery(menu.getQuery()); + childrenList.add(children); + router.setChildren(childrenList); + } + else if (menu.getParentId().intValue() == 0 && isInnerLink(menu)) + { + router.setMeta(new MetaVo(menu.getMenuName(), menu.getIcon())); + router.setPath("/"); + List childrenList = new ArrayList(); + RouterVo children = new RouterVo(); + String routerPath = innerLinkReplaceEach(menu.getPath()); + children.setPath(routerPath); + children.setComponent(UserConstants.INNER_LINK); + children.setName(StringUtils.capitalize(routerPath)); + children.setMeta(new MetaVo(menu.getMenuName(), menu.getIcon(), menu.getPath())); + childrenList.add(children); + router.setChildren(childrenList); + } + routers.add(router); + } + return routers; + } + + /** + * 构建前端所需要树结构 + * + * @param menus 菜单列表 + * @return 树结构列表 + */ + @Override + public List buildMenuTree(List menus) + { + List returnList = new ArrayList(); + List tempList = new ArrayList(); + for (SysMenu dept : menus) + { + tempList.add(dept.getMenuId()); + } + for (Iterator iterator = menus.iterator(); iterator.hasNext();) + { + SysMenu menu = (SysMenu) iterator.next(); + // 如果是顶级节点, 遍历该父节点的所有子节点 + if (!tempList.contains(menu.getParentId())) + { + recursionFn(menus, menu); + returnList.add(menu); + } + } + if (returnList.isEmpty()) + { + returnList = menus; + } + return returnList; + } + + /** + * 构建前端所需要下拉树结构 + * + * @param menus 菜单列表 + * @return 下拉树结构列表 + */ + @Override + public List buildMenuTreeSelect(List menus) + { + List menuTrees = buildMenuTree(menus); + return menuTrees.stream().map(TreeSelect::new).collect(Collectors.toList()); + } + + /** + * 根据菜单ID查询信息 + * + * @param menuId 菜单ID + * @return 菜单信息 + */ + @Override + public SysMenu selectMenuById(Long menuId) + { + return menuMapper.selectMenuById(menuId); + } + + /** + * 是否存在菜单子节点 + * + * @param menuId 菜单ID + * @return 结果 + */ + @Override + public boolean hasChildByMenuId(Long menuId) + { + int result = menuMapper.hasChildByMenuId(menuId); + return result > 0; + } + + /** + * 查询菜单使用数量 + * + * @param menuId 菜单ID + * @return 结果 + */ + @Override + public boolean checkMenuExistRole(Long menuId) + { + int result = roleMenuMapper.checkMenuExistRole(menuId); + return result > 0; + } + + /** + * 新增保存菜单信息 + * + * @param menu 菜单信息 + * @return 结果 + */ + @Override + public int insertMenu(SysMenu menu) + { + return menuMapper.insertMenu(menu); + } + + /** + * 修改保存菜单信息 + * + * @param menu 菜单信息 + * @return 结果 + */ + @Override + public int updateMenu(SysMenu menu) + { + return menuMapper.updateMenu(menu); + } + + /** + * 删除菜单管理信息 + * + * @param menuId 菜单ID + * @return 结果 + */ + @Override + public int deleteMenuById(Long menuId) + { + return menuMapper.deleteMenuById(menuId); + } + + /** + * 校验菜单名称是否唯一 + * + * @param menu 菜单信息 + * @return 结果 + */ + @Override + public String checkMenuNameUnique(SysMenu menu) + { + Long menuId = StringUtils.isNull(menu.getMenuId()) ? -1L : menu.getMenuId(); + SysMenu info = menuMapper.checkMenuNameUnique(menu.getMenuName(), menu.getParentId()); + if (StringUtils.isNotNull(info) && info.getMenuId().longValue() != menuId.longValue()) + { + return UserConstants.NOT_UNIQUE; + } + return UserConstants.UNIQUE; + } + + /** + * 获取路由名称 + * + * @param menu 菜单信息 + * @return 路由名称 + */ + public String getRouteName(SysMenu menu) + { + String routerName = StringUtils.capitalize(menu.getPath()); + // 非外链并且是一级目录(类型为目录) + if (isMenuFrame(menu)) + { + routerName = StringUtils.EMPTY; + } + return routerName; + } + + /** + * 获取路由地址 + * + * @param menu 菜单信息 + * @return 路由地址 + */ + public String getRouterPath(SysMenu menu) + { + String routerPath = menu.getPath(); + // 内链打开外网方式 + if (menu.getParentId().intValue() != 0 && isInnerLink(menu)) + { + routerPath = innerLinkReplaceEach(routerPath); + } + // 非外链并且是一级目录(类型为目录) + if (0 == menu.getParentId().intValue() && UserConstants.TYPE_DIR.equals(menu.getMenuType()) + && UserConstants.NO_FRAME.equals(menu.getIsFrame())) + { + routerPath = "/" + menu.getPath(); + } + // 非外链并且是一级目录(类型为菜单) + else if (isMenuFrame(menu)) + { + routerPath = "/"; + } + return routerPath; + } + + /** + * 获取组件信息 + * + * @param menu 菜单信息 + * @return 组件信息 + */ + public String getComponent(SysMenu menu) + { + String component = UserConstants.LAYOUT; + if (StringUtils.isNotEmpty(menu.getComponent()) && !isMenuFrame(menu)) + { + component = menu.getComponent(); + } + else if (StringUtils.isEmpty(menu.getComponent()) && menu.getParentId().intValue() != 0 && isInnerLink(menu)) + { + component = UserConstants.INNER_LINK; + } + else if (StringUtils.isEmpty(menu.getComponent()) && isParentView(menu)) + { + component = UserConstants.PARENT_VIEW; + } + return component; + } + + /** + * 是否为菜单内部跳转 + * + * @param menu 菜单信息 + * @return 结果 + */ + public boolean isMenuFrame(SysMenu menu) + { + return menu.getParentId().intValue() == 0 && UserConstants.TYPE_MENU.equals(menu.getMenuType()) + && menu.getIsFrame().equals(UserConstants.NO_FRAME); + } + + /** + * 是否为内链组件 + * + * @param menu 菜单信息 + * @return 结果 + */ + public boolean isInnerLink(SysMenu menu) + { + return menu.getIsFrame().equals(UserConstants.NO_FRAME) && StringUtils.ishttp(menu.getPath()); + } + + /** + * 是否为parent_view组件 + * + * @param menu 菜单信息 + * @return 结果 + */ + public boolean isParentView(SysMenu menu) + { + return menu.getParentId().intValue() != 0 && UserConstants.TYPE_DIR.equals(menu.getMenuType()); + } + + /** + * 根据父节点的ID获取所有子节点 + * + * @param list 分类表 + * @param parentId 传入的父节点ID + * @return String + */ + public List getChildPerms(List list, int parentId) + { + List returnList = new ArrayList(); + for (Iterator iterator = list.iterator(); iterator.hasNext();) + { + SysMenu t = (SysMenu) iterator.next(); + // 一、根据传入的某个父节点ID,遍历该父节点的所有子节点 + if (t.getParentId() == parentId) + { + recursionFn(list, t); + returnList.add(t); + } + } + return returnList; + } + + /** + * 递归列表 + * + * @param list + * @param t + */ + private void recursionFn(List list, SysMenu t) + { + // 得到子节点列表 + List childList = getChildList(list, t); + t.setChildren(childList); + for (SysMenu tChild : childList) + { + if (hasChild(list, tChild)) + { + recursionFn(list, tChild); + } + } + } + + /** + * 得到子节点列表 + */ + private List getChildList(List list, SysMenu t) + { + List tlist = new ArrayList(); + Iterator it = list.iterator(); + while (it.hasNext()) + { + SysMenu n = (SysMenu) it.next(); + if (n.getParentId().longValue() == t.getMenuId().longValue()) + { + tlist.add(n); + } + } + return tlist; + } + + /** + * 判断是否有子节点 + */ + private boolean hasChild(List list, SysMenu t) + { + return getChildList(list, t).size() > 0; + } + + /** + * 内链域名特殊字符替换 + * + * @return + */ + public String innerLinkReplaceEach(String path) + { + return StringUtils.replaceEach(path, new String[] { Constants.HTTP, Constants.HTTPS, Constants.WWW, "." }, + new String[] { "", "", "", "/" }); + } + + @Override + public List getTenantMenus(SysMenu sysMenu) { + List menus = menuMapper.selectList(sysMenu); + Iterator iterator = menus.iterator(); +// while (iterator.hasNext()) { +// SysMenu next = iterator.next(); +// if ("C".equalsIgnoreCase(next.getMenuType()) || "F".equalsIgnoreCase(next.getMenuType())) { +// SysMenu pm = CollUtil.findOneByField(menus, "menuId", next.getParentId()); +// if (pm == null) { +// iterator.remove(); +// } +// } +// } + // 开启多租户的情况下,需要过滤掉未开通的菜单 + tenantService.handleTenantMenu(menuIds -> menus.removeIf(menu -> !CollUtil.contains(menuIds, menu.getMenuId()))); + return menus; + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysNoticeServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysNoticeServiceImpl.java new file mode 100644 index 00000000..765438b8 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysNoticeServiceImpl.java @@ -0,0 +1,92 @@ +package com.ruoyi.system.service.impl; + +import java.util.List; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +import com.ruoyi.system.domain.SysNotice; +import com.ruoyi.system.mapper.SysNoticeMapper; +import com.ruoyi.system.service.ISysNoticeService; + +/** + * 公告 服务层实现 + * + * @author ruoyi + */ +@Service +public class SysNoticeServiceImpl implements ISysNoticeService +{ + @Autowired + private SysNoticeMapper noticeMapper; + + /** + * 查询公告信息 + * + * @param noticeId 公告ID + * @return 公告信息 + */ + @Override + public SysNotice selectNoticeById(Long noticeId) + { + return noticeMapper.selectNoticeById(noticeId); + } + + /** + * 查询公告列表 + * + * @param notice 公告信息 + * @return 公告集合 + */ + @Override + public List selectNoticeList(SysNotice notice) + { + return noticeMapper.selectNoticeList(notice); + } + + /** + * 新增公告 + * + * @param notice 公告信息 + * @return 结果 + */ + @Override + public int insertNotice(SysNotice notice) + { + return noticeMapper.insertNotice(notice); + } + + /** + * 修改公告 + * + * @param notice 公告信息 + * @return 结果 + */ + @Override + public int updateNotice(SysNotice notice) + { + return noticeMapper.updateNotice(notice); + } + + /** + * 删除公告对象 + * + * @param noticeId 公告ID + * @return 结果 + */ + @Override + public int deleteNoticeById(Long noticeId) + { + return noticeMapper.deleteNoticeById(noticeId); + } + + /** + * 批量删除公告信息 + * + * @param noticeIds 需要删除的公告ID + * @return 结果 + */ + @Override + public int deleteNoticeByIds(Long[] noticeIds) + { + return noticeMapper.deleteNoticeByIds(noticeIds); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysOperLogServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysOperLogServiceImpl.java new file mode 100644 index 00000000..54898152 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysOperLogServiceImpl.java @@ -0,0 +1,76 @@ +package com.ruoyi.system.service.impl; + +import java.util.List; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +import com.ruoyi.system.domain.SysOperLog; +import com.ruoyi.system.mapper.SysOperLogMapper; +import com.ruoyi.system.service.ISysOperLogService; + +/** + * 操作日志 服务层处理 + * + * @author ruoyi + */ +@Service +public class SysOperLogServiceImpl implements ISysOperLogService +{ + @Autowired + private SysOperLogMapper operLogMapper; + + /** + * 新增操作日志 + * + * @param operLog 操作日志对象 + */ + @Override + public void insertOperlog(SysOperLog operLog) + { + operLogMapper.insertOperlog(operLog); + } + + /** + * 查询系统操作日志集合 + * + * @param operLog 操作日志对象 + * @return 操作日志集合 + */ + @Override + public List selectOperLogList(SysOperLog operLog) + { + return operLogMapper.selectOperLogList(operLog); + } + + /** + * 批量删除系统操作日志 + * + * @param operIds 需要删除的操作日志ID + * @return 结果 + */ + @Override + public int deleteOperLogByIds(Long[] operIds) + { + return operLogMapper.deleteOperLogByIds(operIds); + } + + /** + * 查询操作日志详细 + * + * @param operId 操作ID + * @return 操作日志对象 + */ + @Override + public SysOperLog selectOperLogById(Long operId) + { + return operLogMapper.selectOperLogById(operId); + } + + /** + * 清空操作日志 + */ + @Override + public void cleanOperLog() + { + operLogMapper.cleanOperLog(); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysPostServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysPostServiceImpl.java new file mode 100644 index 00000000..0e7ff371 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysPostServiceImpl.java @@ -0,0 +1,180 @@ +package com.ruoyi.system.service.impl; + +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.exception.ServiceException; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.system.domain.SysPost; +import com.ruoyi.system.mapper.SysPostMapper; +import com.ruoyi.system.mapper.SysUserPostMapper; +import com.ruoyi.system.service.ISysPostService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import java.util.List; + +/** + * 岗位信息 服务层处理 + * + * @author ruoyi + */ +@Service +public class SysPostServiceImpl extends ServiceImpl implements ISysPostService +{ + @Autowired + private SysPostMapper postMapper; + + @Autowired + private SysUserPostMapper userPostMapper; + + /** + * 查询岗位信息集合 + * + * @param post 岗位信息 + * @return 岗位信息集合 + */ + @Override + public List selectPostList(SysPost post) + { + return postMapper.selectPostList(post); + } + + /** + * 查询所有岗位 + * + * @return 岗位列表 + */ + @Override + public List selectPostAll() + { + return postMapper.selectPostAll(); + } + + /** + * 通过岗位ID查询岗位信息 + * + * @param postId 岗位ID + * @return 角色对象信息 + */ + @Override + public SysPost selectPostById(Long postId) + { + return postMapper.selectPostById(postId); + } + + /** + * 根据用户ID获取岗位选择框列表 + * + * @param userId 用户ID + * @return 选中岗位ID列表 + */ + @Override + public List selectPostListByUserId(Long userId) + { + return postMapper.selectPostListByUserId(userId); + } + + /** + * 校验岗位名称是否唯一 + * + * @param post 岗位信息 + * @return 结果 + */ + @Override + public String checkPostNameUnique(SysPost post) + { + Long postId = StringUtils.isNull(post.getPostId()) ? -1L : post.getPostId(); + SysPost info = postMapper.checkPostNameUnique(post.getPostName()); + if (StringUtils.isNotNull(info) && info.getPostId().longValue() != postId.longValue()) + { + return UserConstants.NOT_UNIQUE; + } + return UserConstants.UNIQUE; + } + + /** + * 校验岗位编码是否唯一 + * + * @param post 岗位信息 + * @return 结果 + */ + @Override + public String checkPostCodeUnique(SysPost post) + { + Long postId = StringUtils.isNull(post.getPostId()) ? -1L : post.getPostId(); + SysPost info = postMapper.checkPostCodeUnique(post.getPostCode()); + if (StringUtils.isNotNull(info) && info.getPostId().longValue() != postId.longValue()) + { + return UserConstants.NOT_UNIQUE; + } + return UserConstants.UNIQUE; + } + + /** + * 通过岗位ID查询岗位使用数量 + * + * @param postId 岗位ID + * @return 结果 + */ + @Override + public int countUserPostById(Long postId) + { + return userPostMapper.countUserPostById(postId); + } + + /** + * 删除岗位信息 + * + * @param postId 岗位ID + * @return 结果 + */ + @Override + public int deletePostById(Long postId) + { + return postMapper.deletePostById(postId); + } + + /** + * 批量删除岗位信息 + * + * @param postIds 需要删除的岗位ID + * @return 结果 + */ + @Override + public int deletePostByIds(Long[] postIds) + { + for (Long postId : postIds) + { + SysPost post = selectPostById(postId); + if (countUserPostById(postId) > 0) + { + throw new ServiceException(String.format("%1$s已分配,不能删除", post.getPostName())); + } + } + return postMapper.deletePostByIds(postIds); + } + + /** + * 新增保存岗位信息 + * + * @param post 岗位信息 + * @return 结果 + */ + @Override + public int insertPost(SysPost post) + { + return postMapper.insertPost(post); + } + + /** + * 修改保存岗位信息 + * + * @param post 岗位信息 + * @return 结果 + */ + @Override + public int updatePost(SysPost post) + { + return postMapper.updatePost(post); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysRoleServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysRoleServiceImpl.java new file mode 100644 index 00000000..230d723b --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysRoleServiceImpl.java @@ -0,0 +1,409 @@ +package com.ruoyi.system.service.impl; + +import cn.hutool.core.collection.CollectionUtil; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import com.ruoyi.common.annotation.DataScope; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.core.domain.entity.SysMenu; +import com.ruoyi.common.core.domain.entity.SysRole; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.enums.RoleCodeEnum; +import com.ruoyi.common.exception.ServiceException; +import com.ruoyi.common.mybatis.query.LambdaQueryWrapperX; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.spring.SpringUtils; +import com.ruoyi.system.domain.SysRoleDept; +import com.ruoyi.system.domain.SysRoleMenu; +import com.ruoyi.system.domain.SysUserRole; +import com.ruoyi.system.mapper.*; +import com.ruoyi.system.service.ISysRoleService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; + +import java.util.*; + +import static com.ruoyi.common.utils.collection.CollectionUtils.convertSet; + +/** + * 角色 业务层处理 + * + * @author ruoyi + */ +@Service +public class SysRoleServiceImpl extends ServiceImpl implements ISysRoleService { + @Autowired + private SysRoleMapper roleMapper; + + @Autowired + private SysRoleMenuMapper roleMenuMapper; + + @Autowired + private SysUserRoleMapper userRoleMapper; + + @Autowired + private SysRoleDeptMapper roleDeptMapper; + @Autowired + private SysMenuMapper menuMapper; + /** + * 根据条件分页查询角色数据 + * + * @param role 角色信息 + * @return 角色数据集合信息 + */ + @Override + @DataScope(deptAlias = "d") + public List selectRoleList(SysRole role) { + return roleMapper.selectRoleList(role); + } + + /** + * 根据用户ID查询角色 + * + * @param userId 用户ID + * @return 角色列表 + */ + @Override + public List selectRolesByUserId(Long userId) { + List userRoles = roleMapper.selectRolePermissionByUserId(userId); + List roles = selectRoleAll(); + for (SysRole role : roles) { + for (SysRole userRole : userRoles) { + if (role.getRoleId().longValue() == userRole.getRoleId().longValue()) { + role.setFlag(true); + break; + } + } + } + return roles; + } + + /** + * 根据用户ID查询权限 + * + * @param userId 用户ID + * @return 权限列表 + */ + @Override + public Set selectRolePermissionByUserId(Long userId) { + List perms = roleMapper.selectRolePermissionByUserId(userId); + Set permsSet = new HashSet<>(); + for (SysRole perm : perms) { + if (StringUtils.isNotNull(perm)) { + permsSet.addAll(Arrays.asList(perm.getRoleKey().trim().split(","))); + } + } + return permsSet; + } + + /** + * 查询所有角色 + * + * @return 角色列表 + */ + @Override + public List selectRoleAll() { + return SpringUtils.getAopProxy(this).selectRoleList(new SysRole()); + } + + /** + * 根据用户ID获取角色选择框列表 + * + * @param userId 用户ID + * @return 选中角色ID列表 + */ + @Override + public List selectRoleListByUserId(Long userId) { + return roleMapper.selectRoleListByUserId(userId); + } + + /** + * 通过角色ID查询角色 + * + * @param roleId 角色ID + * @return 角色对象信息 + */ + @Override + public SysRole selectRoleById(Long roleId) { + return roleMapper.selectRoleById(roleId); + } + + /** + * 校验角色名称是否唯一 + * + * @param role 角色信息 + * @return 结果 + */ + @Override + public String checkRoleNameUnique(SysRole role) { + Long roleId = StringUtils.isNull(role.getRoleId()) ? -1L : role.getRoleId(); + SysRole info = roleMapper.checkRoleNameUnique(role.getRoleName()); + if (StringUtils.isNotNull(info) && info.getRoleId().longValue() != roleId.longValue()) { + return UserConstants.NOT_UNIQUE; + } + return UserConstants.UNIQUE; + } + + /** + * 校验角色权限是否唯一 + * + * @param role 角色信息 + * @return 结果 + */ + @Override + public String checkRoleKeyUnique(SysRole role) { + Long roleId = StringUtils.isNull(role.getRoleId()) ? -1L : role.getRoleId(); + SysRole info = roleMapper.checkRoleKeyUnique(role.getRoleKey()); + if (StringUtils.isNotNull(info) && info.getRoleId().longValue() != roleId.longValue()) { + return UserConstants.NOT_UNIQUE; + } + return UserConstants.UNIQUE; + } + + /** + * 校验角色是否允许操作 + * + * @param role 角色信息 + */ + @Override + public void checkRoleAllowed(SysRole role) { + if (StringUtils.isNotNull(role.getRoleId()) && role.isAdmin()) { + throw new ServiceException("不允许操作超级管理员角色"); + } + } + + /** + * 校验角色是否有数据权限 + * + * @param roleId 角色id + */ + @Override + public void checkRoleDataScope(Long roleId) { + if (!SysUser.isAdmin(SecurityUtils.getUserId())) { + SysRole role = new SysRole(); + role.setRoleId(roleId); + List roles = SpringUtils.getAopProxy(this).selectRoleList(role); + if (StringUtils.isEmpty(roles)) { + throw new ServiceException("没有权限访问角色数据!"); + } + } + } + + /** + * 通过角色ID查询角色使用数量 + * + * @param roleId 角色ID + * @return 结果 + */ + @Override + public int countUserRoleByRoleId(Long roleId) { + return userRoleMapper.countUserRoleByRoleId(roleId); + } + + /** + * 新增保存角色信息 + * + * @param role 角色信息 + * @return 结果 + */ + @Override + @Transactional + public int insertRole(SysRole role) { + // 新增角色信息 + roleMapper.insertRole(role); + return insertRoleMenu(role); + } + + /** + * 修改保存角色信息 + * + * @param role 角色信息 + * @return 结果 + */ + @Override + @Transactional + public int updateRole(SysRole role) { + // 修改角色信息 + roleMapper.updateRole(role); + // 删除角色与菜单关联 + roleMenuMapper.deleteRoleMenuByRoleId(role.getRoleId()); + return insertRoleMenu(role); + } + + /** + * 修改角色状态 + * + * @param role 角色信息 + * @return 结果 + */ + @Override + public int updateRoleStatus(SysRole role) { + return roleMapper.updateRole(role); + } + + /** + * 修改数据权限信息 + * + * @param role 角色信息 + * @return 结果 + */ + @Override + @Transactional + public int authDataScope(SysRole role) { + // 修改角色信息 + roleMapper.updateRole(role); + // 删除角色与部门关联 + roleDeptMapper.deleteRoleDeptByRoleId(role.getRoleId()); + // 新增角色和部门信息(数据权限) + return insertRoleDept(role); + } + + /** + * 新增角色菜单信息 + * + * @param role 角色对象 + */ + public int insertRoleMenu(SysRole role) { + int rows = 1; + // 新增用户与角色管理 + List list = new ArrayList(); + for (Long menuId : role.getMenuIds()) { + SysRoleMenu rm = new SysRoleMenu(); + rm.setRoleId(role.getRoleId()); + rm.setMenuId(menuId); + list.add(rm); + } + if (list.size() > 0) { + rows = roleMenuMapper.batchRoleMenu(list); + } + return rows; + } + + /** + * 新增角色部门信息(数据权限) + * + * @param role 角色对象 + */ + public int insertRoleDept(SysRole role) { + int rows = 1; + // 新增角色与部门(数据权限)管理 + List list = new ArrayList(); + for (Long deptId : role.getDeptIds()) { + SysRoleDept rd = new SysRoleDept(); + rd.setRoleId(role.getRoleId()); + rd.setDeptId(deptId); + list.add(rd); + } + if (list.size() > 0) { + rows = roleDeptMapper.batchRoleDept(list); + } + return rows; + } + + /** + * 通过角色ID删除角色 + * + * @param roleId 角色ID + * @return 结果 + */ + @Override + @Transactional + public int deleteRoleById(Long roleId) { + // 删除角色与菜单关联 + roleMenuMapper.deleteRoleMenuByRoleId(roleId); + // 删除角色与部门关联 + roleDeptMapper.deleteRoleDeptByRoleId(roleId); + return roleMapper.deleteRoleById(roleId); + } + + /** + * 批量删除角色信息 + * + * @param roleIds 需要删除的角色ID + * @return 结果 + */ + @Override + @Transactional + public int deleteRoleByIds(Long[] roleIds) { + for (Long roleId : roleIds) { + checkRoleAllowed(new SysRole(roleId)); + checkRoleDataScope(roleId); + SysRole role = selectRoleById(roleId); + if (countUserRoleByRoleId(roleId) > 0) { + throw new ServiceException(String.format("%1$s已分配,不能删除", role.getRoleName())); + } + } + // 删除角色与菜单关联 + roleMenuMapper.deleteRoleMenu(roleIds); + // 删除角色与部门关联 + roleDeptMapper.deleteRoleDept(roleIds); + return roleMapper.deleteRoleByIds(roleIds); + } + + /** + * 取消授权用户角色 + * + * @param userRole 用户和角色关联信息 + * @return 结果 + */ + @Override + public int deleteAuthUser(SysUserRole userRole) { + return userRoleMapper.deleteUserRoleInfo(userRole); + } + + /** + * 批量取消授权用户角色 + * + * @param roleId 角色ID + * @param userIds 需要取消授权的用户数据ID + * @return 结果 + */ + @Override + public int deleteAuthUsers(Long roleId, Long[] userIds) { + return userRoleMapper.deleteUserRoleInfos(roleId, userIds); + } + + /** + * 批量选择授权用户角色 + * + * @param roleId 角色ID + * @param userIds 需要授权的用户数据ID + * @return 结果 + */ + @Override + public int insertAuthUsers(Long roleId, Long[] userIds) { + // 新增用户与角色管理 + List list = new ArrayList(); + for (Long userId : userIds) { + SysUserRole ur = new SysUserRole(); + ur.setUserId(userId); + ur.setRoleId(roleId); + list.add(ur); + } + return userRoleMapper.batchUserRole(list); + } + + @Override + public SysRole selectByRoleKey(String roleKey) { + return roleMapper.selectOne(SysRole::getRoleKey, roleKey); + } + + @Override + public Set getRoleMenuIds(Long roleId) { + // 如果是管理员的情况下,获取全部菜单编号 + SysRole role = selectRoleById(roleId); + if (hasAnySuperAdmin(Collections.singleton(role))) { + return convertSet(menuMapper.selectList(), SysMenu::getMenuId); + } + // 如果是非管理员的情况下,获得拥有的菜单编号 + return convertSet(roleMenuMapper.selectList(new LambdaQueryWrapperX().eq(SysRoleMenu::getRoleId, roleId)), SysRoleMenu::getMenuId); + } + + public boolean hasAnySuperAdmin(Collection roleList) { + if (CollectionUtil.isEmpty(roleList)) { + return false; + } + return roleList.stream().anyMatch(role -> RoleCodeEnum.isSuperAdmin(role.getRoleKey())); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysUserOnlineServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysUserOnlineServiceImpl.java new file mode 100644 index 00000000..f80a8771 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysUserOnlineServiceImpl.java @@ -0,0 +1,96 @@ +package com.ruoyi.system.service.impl; + +import org.springframework.stereotype.Service; +import com.ruoyi.common.core.domain.model.LoginUser; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.system.domain.SysUserOnline; +import com.ruoyi.system.service.ISysUserOnlineService; + +/** + * 在线用户 服务层处理 + * + * @author ruoyi + */ +@Service +public class SysUserOnlineServiceImpl implements ISysUserOnlineService +{ + /** + * 通过登录地址查询信息 + * + * @param ipaddr 登录地址 + * @param user 用户信息 + * @return 在线用户信息 + */ + @Override + public SysUserOnline selectOnlineByIpaddr(String ipaddr, LoginUser user) + { + if (StringUtils.equals(ipaddr, user.getIpaddr())) + { + return loginUserToUserOnline(user); + } + return null; + } + + /** + * 通过用户名称查询信息 + * + * @param userName 用户名称 + * @param user 用户信息 + * @return 在线用户信息 + */ + @Override + public SysUserOnline selectOnlineByUserName(String userName, LoginUser user) + { + if (StringUtils.equals(userName, user.getUsername())) + { + return loginUserToUserOnline(user); + } + return null; + } + + /** + * 通过登录地址/用户名称查询信息 + * + * @param ipaddr 登录地址 + * @param userName 用户名称 + * @param user 用户信息 + * @return 在线用户信息 + */ + @Override + public SysUserOnline selectOnlineByInfo(String ipaddr, String userName, LoginUser user) + { + if (StringUtils.equals(ipaddr, user.getIpaddr()) && StringUtils.equals(userName, user.getUsername())) + { + return loginUserToUserOnline(user); + } + return null; + } + + /** + * 设置在线用户信息 + * + * @param user 用户信息 + * @return 在线用户 + */ + @Override + public SysUserOnline loginUserToUserOnline(LoginUser user) + { + if (StringUtils.isNull(user) || StringUtils.isNull(user.getUser())) + { + return null; + } + SysUserOnline sysUserOnline = new SysUserOnline(); + sysUserOnline.setTokenId(user.getToken()); + sysUserOnline.setUserName(user.getUsername()); + sysUserOnline.setIpaddr(user.getIpaddr()); + sysUserOnline.setLoginLocation(user.getLoginLocation()); + sysUserOnline.setBrowser(user.getBrowser()); + sysUserOnline.setOs(user.getOs()); + sysUserOnline.setLoginTime(user.getLoginTime()); + if (StringUtils.isNotNull(user.getUser().getDept())) + { + sysUserOnline.setDeptName(user.getUser().getDept().getDeptName()); + } + return sysUserOnline; + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysUserServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysUserServiceImpl.java new file mode 100644 index 00000000..65d467ce --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysUserServiceImpl.java @@ -0,0 +1,518 @@ +package com.ruoyi.system.service.impl; + +import cn.hutool.core.collection.CollUtil; +import cn.hutool.core.util.ArrayUtil; +import com.baomidou.mybatisplus.core.toolkit.Wrappers; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import com.ruoyi.common.annotation.DataScope; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.core.domain.entity.SysRole; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.exception.ServiceException; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.bean.BeanValidators; +import com.ruoyi.common.utils.spring.SpringUtils; +import com.ruoyi.system.domain.SysPost; +import com.ruoyi.system.domain.SysUserPost; +import com.ruoyi.system.domain.SysUserRole; +import com.ruoyi.system.mapper.*; +import com.ruoyi.system.service.ISysConfigService; +import com.ruoyi.system.service.ISysUserService; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; +import org.springframework.util.CollectionUtils; + +import javax.validation.Validator; +import java.util.*; +import java.util.stream.Collectors; + +/** + * 用户 业务层处理 + * + * @author ruoyi + */ +@Service +public class SysUserServiceImpl extends ServiceImpl implements ISysUserService { + private static final Logger log = LoggerFactory.getLogger(SysUserServiceImpl.class); + + @Autowired + private SysUserMapper userMapper; + + @Autowired + private SysRoleMapper roleMapper; + + @Autowired + private SysPostMapper postMapper; + + @Autowired + private SysUserRoleMapper userRoleMapper; + + @Autowired + private SysUserPostMapper userPostMapper; + + @Autowired + private ISysConfigService configService; + + @Autowired + protected Validator validator; + + /** + * 根据条件分页查询用户列表 + * + * @param user 用户信息 + * @return 用户信息集合信息 + */ + @Override + @DataScope(deptAlias = "d", userAlias = "u") + public List selectUserList(SysUser user) { + return userMapper.selectUserList(user); + } + + /** + * 根据条件分页查询已分配用户角色列表 + * + * @param user 用户信息 + * @return 用户信息集合信息 + */ + @Override + @DataScope(deptAlias = "d", userAlias = "u") + public List selectAllocatedList(SysUser user) { + return userMapper.selectAllocatedList(user); + } + + /** + * 根据条件分页查询未分配用户角色列表 + * + * @param user 用户信息 + * @return 用户信息集合信息 + */ + @Override + @DataScope(deptAlias = "d", userAlias = "u") + public List selectUnallocatedList(SysUser user) { + return userMapper.selectUnallocatedList(user); + } + + /** + * 通过用户名查询用户 + * + * @param userName 用户名 + * @return 用户对象信息 + */ + @Override + public SysUser selectUserByUserName(String userName) { + return userMapper.selectUserByUserName(userName); + } + + /** + * 通过用户ID查询用户 + * + * @param userId 用户ID + * @return 用户对象信息 + */ + @Override + public SysUser selectUserById(Long userId) { + return userMapper.selectUserById(userId); + } + + /** + * 查询用户所属角色组 + * + * @param userName 用户名 + * @return 结果 + */ + @Override + public String selectUserRoleGroup(String userName) { + List list = roleMapper.selectRolesByUserName(userName); + if (CollectionUtils.isEmpty(list)) { + return StringUtils.EMPTY; + } + return list.stream().map(SysRole::getRoleName).collect(Collectors.joining(",")); + } + + /** + * 查询用户所属岗位组 + * + * @param userName 用户名 + * @return 结果 + */ + @Override + public String selectUserPostGroup(String userName) { + List list = postMapper.selectPostsByUserName(userName); + if (CollectionUtils.isEmpty(list)) { + return StringUtils.EMPTY; + } + return list.stream().map(SysPost::getPostName).collect(Collectors.joining(",")); + } + + /** + * 校验用户名称是否唯一 + * + * @param user 用户信息 + * @return 结果 + */ + @Override + public String checkUserNameUnique(SysUser user) { + Long userId = StringUtils.isNull(user.getUserId()) ? -1L : user.getUserId(); + SysUser info = userMapper.checkUserNameUnique(user.getUserName()); + if (StringUtils.isNotNull(info) && info.getUserId().longValue() != userId.longValue()) { + return UserConstants.NOT_UNIQUE; + } + return UserConstants.UNIQUE; + } + + /** + * 校验手机号码是否唯一 + * + * @param user 用户信息 + * @return + */ + @Override + public String checkPhoneUnique(SysUser user) { + Long userId = StringUtils.isNull(user.getUserId()) ? -1L : user.getUserId(); + SysUser info = userMapper.checkPhoneUnique(user.getPhonenumber()); + if (StringUtils.isNotNull(info) && info.getUserId().longValue() != userId.longValue()) { + return UserConstants.NOT_UNIQUE; + } + return UserConstants.UNIQUE; + } + + /** + * 校验email是否唯一 + * + * @param user 用户信息 + * @return + */ + @Override + public String checkEmailUnique(SysUser user) { + Long userId = StringUtils.isNull(user.getUserId()) ? -1L : user.getUserId(); + SysUser info = userMapper.checkEmailUnique(user.getEmail()); + if (StringUtils.isNotNull(info) && info.getUserId().longValue() != userId.longValue()) { + return UserConstants.NOT_UNIQUE; + } + return UserConstants.UNIQUE; + } + + /** + * 校验用户是否允许操作 + * + * @param user 用户信息 + */ + @Override + public void checkUserAllowed(SysUser user) { + if (StringUtils.isNotNull(user.getUserId()) && user.isAdmin()) { + throw new ServiceException("不允许操作超级管理员用户"); + } + } + + /** + * 校验用户是否有数据权限 + * + * @param userId 用户id + */ + @Override + public void checkUserDataScope(Long userId) { + if (!SysUser.isAdmin(SecurityUtils.getUserId())) { + SysUser user = new SysUser(); + user.setUserId(userId); + List users = SpringUtils.getAopProxy(this).selectUserList(user); + if (StringUtils.isEmpty(users)) { + throw new ServiceException("没有权限访问用户数据!"); + } + } + } + + /** + * 新增保存用户信息 + * + * @param user 用户信息 + * @return 结果 + */ + @Override + @Transactional + public int insertUser(SysUser user) { + // 新增用户信息 + int rows = userMapper.insertUser(user); + // 新增用户岗位关联 + insertUserPost(user); + // 新增用户与角色管理 + insertUserRole(user); + return rows; + } + + /** + * 注册用户信息 + * + * @param user 用户信息 + * @return 结果 + */ + @Override + public boolean registerUser(SysUser user) { + return userMapper.insertUser(user) > 0; + } + + /** + * 修改保存用户信息 + * + * @param user 用户信息 + * @return 结果 + */ + @Override + @Transactional + public int updateUser(SysUser user) { + Long userId = user.getUserId(); + // 删除用户与角色关联 + userRoleMapper.deleteUserRoleByUserId(userId); + // 新增用户与角色管理 + insertUserRole(user); + // 删除用户与岗位关联 + userPostMapper.deleteUserPostByUserId(userId); + // 新增用户与岗位管理 + insertUserPost(user); + return userMapper.updateUser(user); + } + + /** + * 用户授权角色 + * + * @param userId 用户ID + * @param roleIds 角色组 + */ + @Override + @Transactional + public void insertUserAuth(Long userId, Long[] roleIds) { + userRoleMapper.deleteUserRoleByUserId(userId); + insertUserRole(userId, roleIds); + } + + /** + * 修改用户状态 + * + * @param user 用户信息 + * @return 结果 + */ + @Override + public int updateUserStatus(SysUser user) { + return userMapper.updateUser(user); + } + + /** + * 修改用户基本信息 + * + * @param user 用户信息 + * @return 结果 + */ + @Override + public int updateUserProfile(SysUser user) { + return userMapper.updateUser(user); + } + + /** + * 修改用户头像 + * + * @param userName 用户名 + * @param avatar 头像地址 + * @return 结果 + */ + @Override + public boolean updateUserAvatar(String userName, String avatar) { + return userMapper.updateUserAvatar(userName, avatar) > 0; + } + + /** + * 重置用户密码 + * + * @param user 用户信息 + * @return 结果 + */ + @Override + public int resetPwd(SysUser user) { + return userMapper.updateUser(user); + } + + /** + * 重置用户密码 + * + * @param userName 用户名 + * @param password 密码 + * @return 结果 + */ + @Override + public int resetUserPwd(String userName, String password) { + return userMapper.resetUserPwd(userName, password); + } + + /** + * 新增用户角色信息 + * + * @param user 用户对象 + */ + public void insertUserRole(SysUser user) { + this.insertUserRole(user.getUserId(), user.getRoleIds()); + } + + /** + * 新增用户岗位信息 + * + * @param user 用户对象 + */ + public void insertUserPost(SysUser user) { + Long[] posts = user.getPostIds(); + if (StringUtils.isNotEmpty(posts)) { + // 新增用户与岗位管理 + List list = new ArrayList(posts.length); + for (Long postId : posts) { + SysUserPost up = new SysUserPost(); + up.setUserId(user.getUserId()); + up.setPostId(postId); + list.add(up); + } + userPostMapper.batchUserPost(list); + } + } + + /** + * 新增用户角色信息 + * + * @param userId 用户ID + * @param roleIds 角色组 + */ + public void insertUserRole(Long userId, Long[] temps) { + Long[] roleIds = ArrayUtil.removeNull(temps); + if (StringUtils.isNotEmpty(roleIds)) { + // 新增用户与角色管理 + List list = new ArrayList(roleIds.length); + for (Long roleId : roleIds) { + SysUserRole ur = new SysUserRole(); + ur.setUserId(userId); + ur.setRoleId(roleId); + list.add(ur); + } + userRoleMapper.batchUserRole(list); + } + } + + /** + * 通过用户ID删除用户 + * + * @param userId 用户ID + * @return 结果 + */ + @Override + @Transactional + public int deleteUserById(Long userId) { + // 删除用户与角色关联 + userRoleMapper.deleteUserRoleByUserId(userId); + // 删除用户与岗位表 + userPostMapper.deleteUserPostByUserId(userId); + return userMapper.deleteUserById(userId); + } + + /** + * 批量删除用户信息 + * + * @param userIds 需要删除的用户ID + * @return 结果 + */ + @Override + @Transactional + public int deleteUserByIds(Long[] userIds) { + for (Long userId : userIds) { + checkUserAllowed(new SysUser(userId)); + checkUserDataScope(userId); + } + // 删除用户与角色关联 + userRoleMapper.deleteUserRole(userIds); + // 删除用户与岗位关联 + userPostMapper.deleteUserPost(userIds); + return userMapper.deleteUserByIds(userIds); + } + + /** + * 导入用户数据 + * + * @param userList 用户数据列表 + * @param isUpdateSupport 是否更新支持,如果已存在,则进行更新数据 + * @param operName 操作用户 + * @return 结果 + */ + @Override + public String importUser(List userList, Boolean isUpdateSupport, String operName) { + if (StringUtils.isNull(userList) || userList.size() == 0) { + throw new ServiceException("导入用户数据不能为空!"); + } + int successNum = 0; + int failureNum = 0; + StringBuilder successMsg = new StringBuilder(); + StringBuilder failureMsg = new StringBuilder(); + String password = configService.selectConfigByKey("sys.user.initPassword"); + for (SysUser user : userList) { + try { + // 验证是否存在这个用户 + SysUser u = userMapper.selectUserByUserName(user.getUserName()); + if (StringUtils.isNull(u)) { + BeanValidators.validateWithException(validator, user); + user.setPassword(SecurityUtils.encryptPassword(password)); + user.setCreateBy(operName); + this.insertUser(user); + successNum++; + successMsg.append("
    " + successNum + "、账号 " + user.getUserName() + " 导入成功"); + } else if (isUpdateSupport) { + BeanValidators.validateWithException(validator, user); + checkUserAllowed(user); + checkUserDataScope(user.getUserId()); + user.setUpdateBy(operName); + this.updateUser(user); + successNum++; + successMsg.append("
    " + successNum + "、账号 " + user.getUserName() + " 更新成功"); + } else { + failureNum++; + failureMsg.append("
    " + failureNum + "、账号 " + user.getUserName() + " 已存在"); + } + } catch (Exception e) { + failureNum++; + String msg = "
    " + failureNum + "、账号 " + user.getUserName() + " 导入失败:"; + failureMsg.append(msg + e.getMessage()); + log.error(msg, e); + } + } + if (failureNum > 0) { + failureMsg.insert(0, "很抱歉,导入失败!共 " + failureNum + " 条数据格式不正确,错误如下:"); + throw new ServiceException(failureMsg.toString()); + } else { + successMsg.insert(0, "恭喜您,数据已全部导入成功!共 " + successNum + " 条,数据如下:"); + } + return successMsg.toString(); + } + + @Override + public List getUsersByPostIds(Collection postIds) { + if (CollUtil.isEmpty(postIds)) { + return Collections.emptyList(); + } + List sysUserPosts = userPostMapper.selectList(Wrappers.query().lambda().in(SysUserPost::getPostId, postIds)); + Set userIds = sysUserPosts.stream().map(sysUserPost -> sysUserPost.getUserId()).collect(Collectors.toSet()); + if (CollUtil.isEmpty(userIds)) { + return Collections.emptyList(); + } + return userMapper.selectBatchIds(userIds); + } + + @Override + public Set getUserRoleIdListByRoleIds(Collection roleIds) { + List sysUserRoles = userRoleMapper.selectList(Wrappers.query().lambda().in(SysUserRole::getRoleId, roleIds)); + Set userIds = sysUserRoles.stream().map(sysUserRole -> sysUserRole.getUserId()).collect(Collectors.toSet()); + if (CollUtil.isEmpty(userIds)) { + return Collections.emptySet(); + } else { + return userIds; + } + } + @Override + public List getUserListByStatus(Integer status) { + return userMapper.selectListByStatus(status); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TenantPackageServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TenantPackageServiceImpl.java new file mode 100644 index 00000000..e03e9370 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TenantPackageServiceImpl.java @@ -0,0 +1,115 @@ +package com.ruoyi.system.service.impl; + +import cn.hutool.core.collection.CollUtil; +import com.ruoyi.common.enums.CommonStatusEnum; +import com.ruoyi.common.exception.ServiceException; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.system.convert.tenant.TenantPackageConvert; +import com.ruoyi.system.domain.TenantDO; +import com.ruoyi.system.domain.TenantPackageDO; +import com.ruoyi.system.domain.vo.packages.TenantPackageCreateReqVO; +import com.ruoyi.system.domain.vo.packages.TenantPackagePageReqVO; +import com.ruoyi.system.domain.vo.packages.TenantPackageUpdateReqVO; +import com.ruoyi.system.mapper.TenantPackageMapper; +import com.ruoyi.system.service.TenantPackageService; +import com.ruoyi.system.service.TenantService; +import org.springframework.context.annotation.Lazy; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; +import org.springframework.validation.annotation.Validated; + +import javax.annotation.Resource; +import java.util.List; +import static com.ruoyi.common.constant.ErrorCodeConstants.*; +/** + * 租户套餐 Service 实现类 + * + * @author 芋道源码 + */ +@Service +@Validated +public class TenantPackageServiceImpl implements TenantPackageService { + + @Resource + private TenantPackageMapper tenantPackageMapper; + + @Resource + @Lazy // 避免循环依赖的报错 + private TenantService tenantService; + + @Override + public Long createTenantPackage(TenantPackageCreateReqVO createReqVO) { + // 插入 + TenantPackageDO tenantPackage = TenantPackageConvert.INSTANCE.convert(createReqVO); + tenantPackageMapper.insert(tenantPackage); + // 返回 + return tenantPackage.getId(); + } + + @Override + @Transactional(rollbackFor = Exception.class) + public void updateTenantPackage(TenantPackageUpdateReqVO updateReqVO) { + // 校验存在 + TenantPackageDO tenantPackage = validateTenantPackageExists(updateReqVO.getId()); + // 更新 + TenantPackageDO updateObj = TenantPackageConvert.INSTANCE.convert(updateReqVO); + tenantPackageMapper.updateById(updateObj); + // 如果菜单发生变化,则修改每个租户的菜单 + if (!CollUtil.isEqualList(tenantPackage.getMenuIds(), updateReqVO.getMenuIds())) { + List tenants = tenantService.getTenantListByPackageId(tenantPackage.getId()); + tenants.forEach(tenant -> tenantService.updateTenantRoleMenu(tenant.getId(), updateReqVO.getMenuIds())); + } + } + + @Override + public void deleteTenantPackage(Long id) { + // 校验存在 + this.validateTenantPackageExists(id); + // 校验正在使用 + this.validateTenantUsed(id); + // 删除 + tenantPackageMapper.deleteById(id); + } + + private TenantPackageDO validateTenantPackageExists(Long id) { + TenantPackageDO tenantPackage = tenantPackageMapper.selectById(id); + if (tenantPackage == null) { + throw new ServiceException(TENANT_PACKAGE_NOT_EXISTS.getMsg()); + } + return tenantPackage; + } + + private void validateTenantUsed(Long id) { + if (tenantService.getTenantCountByPackageId(id) > 0) { + throw new ServiceException(TENANT_PACKAGE_USED.getMsg()); + } + } + + @Override + public TenantPackageDO getTenantPackage(Long id) { + return tenantPackageMapper.selectById(id); + } + + @Override + public PageResult getTenantPackagePage(TenantPackagePageReqVO pageReqVO) { + return tenantPackageMapper.selectPage(pageReqVO); + } + + @Override + public TenantPackageDO validTenantPackage(Long id) { + TenantPackageDO tenantPackage = tenantPackageMapper.selectById(id); + if (tenantPackage == null) { + throw new ServiceException(TENANT_PACKAGE_NOT_EXISTS.getMsg()); + } + if (tenantPackage.getStatus().equals(CommonStatusEnum.DISABLE.getStatus())) { + throw new ServiceException("名字为【"+tenantPackage.getName()+"】的租户套餐已被禁用"); + } + return tenantPackage; + } + + @Override + public List getTenantPackageListByStatus(Integer status) { + return tenantPackageMapper.selectListByStatus(status); + } + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TenantServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TenantServiceImpl.java new file mode 100644 index 00000000..819cc10a --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TenantServiceImpl.java @@ -0,0 +1,317 @@ +package com.ruoyi.system.service.impl; + +import cn.hutool.core.collection.CollUtil; +import cn.hutool.core.lang.Assert; +import cn.hutool.core.util.ObjectUtil; +import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; +import com.ruoyi.common.constant.CacheConstants; +import com.ruoyi.common.constant.HttpStatus; +import com.ruoyi.common.core.domain.entity.SysDictData; +import com.ruoyi.common.core.domain.entity.SysMenu; +import com.ruoyi.common.core.domain.entity.SysRole; +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.common.enums.CommonStatusEnum; +import com.ruoyi.common.enums.RoleCodeEnum; +import com.ruoyi.common.exception.ServiceException; +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.common.utils.DateUtils; +import com.ruoyi.common.utils.DictUtils; +import com.ruoyi.common.utils.EhcacheUtil; +import com.ruoyi.common.utils.SecurityUtils; +import com.ruoyi.common.utils.collection.CollectionUtils; +import com.ruoyi.plugin.tenant.config.TenantProperties; +import com.ruoyi.plugin.tenant.core.context.TenantContextHolder; +import com.ruoyi.plugin.tenant.core.util.TenantUtils; +import com.ruoyi.system.convert.tenant.TenantConvert; +import com.ruoyi.system.domain.TenantDO; +import com.ruoyi.system.domain.TenantPackageDO; +import com.ruoyi.system.domain.vo.tenant.TenantCreateReqVO; +import com.ruoyi.system.domain.vo.tenant.TenantExportReqVO; +import com.ruoyi.system.domain.vo.tenant.TenantPageReqVO; +import com.ruoyi.system.domain.vo.tenant.TenantUpdateReqVO; +import com.ruoyi.system.mapper.TenantMapper; +import com.ruoyi.system.service.*; +import com.ruoyi.system.service.handler.TenantInfoHandler; +import com.ruoyi.system.service.handler.TenantMenuHandler; +import lombok.extern.slf4j.Slf4j; +import org.apache.poi.ss.formula.functions.T; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.context.annotation.Lazy; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; +import org.springframework.validation.annotation.Validated; + +import javax.annotation.PostConstruct; +import javax.annotation.Resource; +import java.util.*; +import java.util.stream.Collectors; + +import static com.ruoyi.common.constant.ErrorCodeConstants.*; + +/** + * 租户 Service 实现类 + * + * @author 芋道源码 + */ +@Service +@Validated +@Slf4j +public class TenantServiceImpl implements TenantService { + + @SuppressWarnings("SpringJavaAutowiredFieldsWarningInspection") + @Autowired(required = false) // 由于 yudao.tenant.enable 配置项,可以关闭多租户的功能,所以这里只能不强制注入 + private TenantProperties tenantProperties; + + @Resource + private TenantMapper tenantMapper; + + @Resource + private TenantPackageService tenantPackageService; + @Resource + @Lazy // 延迟,避免循环依赖报错 + private ISysUserService userService; + @Resource + private ISysRoleService roleService; + @Resource + private ISysMenuService menuService; +// @Resource +// private PermissionService permissionService; + + /** + * 项目启动时,初始化租户到缓存 + */ + @PostConstruct + public void init() + { + loadingTenantCache(); + } + + private void loadingTenantCache() { + List tenants = tenantMapper.selectList(new LambdaQueryWrapper().eq(TenantDO::getStatus, 0).eq(TenantDO::getDeleted, 0)); + for (TenantDO tenant : tenants) { + EhcacheUtil.put("tenant", getCacheKey(tenant.getId().toString()), tenant); + } + } + /** + * 设置cache key + * + * @param tenantId 参数键 + * @return 缓存键key + */ + public static String getCacheKey(String tenantId) + { + return CacheConstants.SYS_TENANT_KEY + tenantId; + } + @Override + public List getTenantIds() { + List tenants = tenantMapper.selectList(); + return CollectionUtils.convertList(tenants, TenantDO::getId); + } + + @Override + public void validTenant(Long id) { + TenantDO tenant = getTenant(id); + if (tenant == null) { + throw new ServiceException(TENANT_NOT_EXISTS.getMsg()); + } + if (tenant.getStatus().equals(CommonStatusEnum.DISABLE.getStatus())) { + throw new ServiceException(String.format(TENANT_DISABLE.getMsg(), tenant.getName())); + } + + if (DateUtils.isExpired(tenant.getExpireTime())) { + throw new ServiceException(String.format(TENANT_EXPIRE.getMsg(), tenant.getName())); + } + } + + @Override + @Transactional(rollbackFor = Exception.class) + public Long createTenant(TenantCreateReqVO createReqVO) { + // 校验套餐被禁用 + TenantPackageDO tenantPackage = tenantPackageService.validTenantPackage(createReqVO.getPackageId()); + + // 创建租户 + TenantDO tenant = TenantConvert.INSTANCE.convert(createReqVO); + tenantMapper.insert(tenant); + EhcacheUtil.put("tenant", getCacheKey(tenant.getId().toString()), tenant); + TenantUtils.execute(tenant.getId(), () -> { + // 创建角色 + Long roleId = createRole(tenantPackage); + // 创建用户,并分配角色 + Long userId = createUser(roleId, createReqVO); + // 修改租户的管理员 + TenantDO tenantDO = new TenantDO(); + tenantDO.setId(tenant.getId()); + tenantDO.setContactUserId(userId); + tenantMapper.updateById(tenantDO); + }); + return tenant.getId(); + } + + private Long createUser(Long roleId, TenantCreateReqVO createReqVO) { + SysUser user = userService.selectUserByUserName(createReqVO.getUsername()); + if (user != null) { + throw new ServiceException("租户已存在", HttpStatus.ERROR); + } + user = new SysUser(); + user.setUserName(createReqVO.getUsername()); + user.setStatus("0"); + user.setPassword(SecurityUtils.encryptPassword(createReqVO.getPassword())); + user.setNickName(createReqVO.getContactName()); + user.setPhonenumber(createReqVO.getContactMobile()); + user.setRoleIds(new Long[]{roleId}); + userService.insertUser(user); + return user.getUserId(); + } + + private Long createRole(TenantPackageDO tenantPackage) { + // 创建角色 + SysRole role = roleService.selectByRoleKey("tenant_admin"); + if (role == null) { + role = new SysRole(); + role.setRoleName("租户管理员"); + role.setRoleKey("tenant_admin"); + role.setRoleSort("0"); + role.setDataScope("1"); + role.setStatus("0"); + role.setMenuIds(tenantPackage.getMenuIds().toArray(new Long[tenantPackage.getMenuIds().size()])); + roleService.insertRole(role); + }else{ + role.setMenuIds(tenantPackage.getMenuIds().toArray(new Long[tenantPackage.getMenuIds().size()])); + roleService.updateRole(role); + } + return role.getRoleId(); + } + + @Override + @Transactional(rollbackFor = Exception.class) + public void updateTenant(TenantUpdateReqVO updateReqVO) { + // 校验存在 + TenantDO tenant = checkUpdateTenant(updateReqVO.getId()); + // 校验套餐被禁用 + TenantPackageDO tenantPackage = tenantPackageService.validTenantPackage(updateReqVO.getPackageId()); + EhcacheUtil.put("tenant", getCacheKey(tenant.getId().toString()), tenant); + // 更新租户 + TenantDO updateObj = TenantConvert.INSTANCE.convert(updateReqVO); + tenantMapper.updateById(updateObj); + // 如果套餐发生变化,则修改其角色的权限 + if (ObjectUtil.notEqual(tenant.getPackageId(), updateReqVO.getPackageId())) { + updateTenantRoleMenu(tenant.getId(), tenantPackage.getMenuIds()); + } + } + + @Override + @Transactional(rollbackFor = Exception.class) + public void updateTenantRoleMenu(Long tenantId, Set menuIds) { + TenantUtils.execute(tenantId, () -> { + // 获得所有角色 + List roles = roleService.list(null); + roles.forEach(role -> Assert.isTrue(tenantId.equals(role.getTenantId()), "角色({}/{}) 租户不匹配", role.getRoleId(), role.getTenantId(), tenantId)); // 兜底校验 + // 重新分配每个角色的权限 + roles.forEach(role -> { + // 如果是租户管理员,重新分配其权限为租户套餐的权限 + if (Objects.equals(role.getRoleKey(), RoleCodeEnum.TENANT_ADMIN.getCode())) { + role.setMenuIds(menuIds.toArray(new Long[menuIds.size()])); + roleService.updateRole(role); + log.info("[updateTenantRoleMenu][租户管理员({}/{}) 的权限修改为({})]", role.getRoleId(), role.getTenantId(), menuIds); + return; + } + // 如果是其他角色,则去掉超过套餐的权限 + Set roleMenuIds = roleService.getRoleMenuIds(role.getRoleId()); + roleMenuIds = CollUtil.intersectionDistinct(roleMenuIds, menuIds); + role.setMenuIds(roleMenuIds.toArray(new Long[roleMenuIds.size()])); + roleService.updateRole(role); + log.info("[updateTenantRoleMenu][角色({}/{}) 的权限修改为({})]", role.getRoleId(), role.getTenantId(), roleMenuIds); + }); + }); + } + + @Override + public void deleteTenant(Long id) { + // 校验存在 + checkUpdateTenant(id); + // 删除 + tenantMapper.deleteById(id); + EhcacheUtil.remove("tenant", getCacheKey(id.toString())); + } + + private TenantDO checkUpdateTenant(Long id) { + TenantDO tenant = tenantMapper.selectById(id); + if (tenant == null) { + throw new ServiceException(TENANT_NOT_EXISTS.getMsg()); + } + // 内置租户,不允许删除 + if (isSystemTenant(tenant)) { + throw new ServiceException(TENANT_CAN_NOT_UPDATE_SYSTEM.getMsg()); + } + return tenant; + } + + @Override + public TenantDO getTenant(Long id) { + return tenantMapper.selectById(id); + } + + @Override + public PageResult getTenantPage(TenantPageReqVO pageReqVO) { + return tenantMapper.selectPage(pageReqVO); + } + + @Override + public List getTenantList(TenantExportReqVO exportReqVO) { + return tenantMapper.selectList(exportReqVO); + } + + @Override + public TenantDO getTenantByName(String name) { + return tenantMapper.selectByName(name); + } + + @Override + public Long getTenantCountByPackageId(Long packageId) { + return tenantMapper.selectCountByPackageId(packageId); + } + + @Override + public List getTenantListByPackageId(Long packageId) { + return tenantMapper.selectListByPackageId(packageId); + } + + @Override + public void handleTenantInfo(TenantInfoHandler handler) { + // 如果禁用,则不执行逻辑 + if (isTenantDisable()) { + return; + } + // 获得租户 + TenantDO tenant = getTenant(TenantContextHolder.getRequiredTenantId()); + // 执行处理器 + handler.handle(tenant); + } + + @Override + public void handleTenantMenu(TenantMenuHandler handler) { + // 如果禁用,则不执行逻辑 + if (isTenantDisable()) { + return; + } + // 获得租户,然后获得菜单 + TenantDO tenant = getTenant(TenantContextHolder.getRequiredTenantId()); + Set menuIds; + if (isSystemTenant(tenant)) { // 系统租户,菜单是全量的 + menuIds = CollectionUtils.convertSet(menuService.list(), SysMenu::getMenuId); + } else { + menuIds = tenantPackageService.getTenantPackage(tenant.getPackageId()).getMenuIds(); + } + // 执行处理器 + handler.handle(menuIds); + } + + private static boolean isSystemTenant(TenantDO tenant) { + return Objects.equals(tenant.getPackageId(), TenantDO.PACKAGE_ID_SYSTEM); + } + + private boolean isTenantDisable() { + return tenantProperties == null || Boolean.FALSE.equals(tenantProperties.getEnable()); + } + +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TreeDictDataServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TreeDictDataServiceImpl.java new file mode 100644 index 00000000..d06fb4f6 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TreeDictDataServiceImpl.java @@ -0,0 +1,231 @@ +package com.ruoyi.system.service.impl; + +import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.ruoyi.common.constant.Constants; +import com.ruoyi.common.constant.TreeConstants; +import com.ruoyi.common.core.domain.AjaxResult; +import com.ruoyi.common.utils.StringUtils; +import com.ruoyi.common.utils.TreeUtils; +import com.ruoyi.system.domain.SysTreeDict; +import com.ruoyi.system.domain.SysTreeDictData; +import com.ruoyi.system.mapper.SysTreeDictDataMapper; +import com.ruoyi.system.service.ISysTreeDictDataService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; + +import java.util.Arrays; +import java.util.List; + +/** + * 树形字典数据Service业务层处理 + * + * @author wangzongrun + * @date 2021-05-31 + */ +@Service +public class TreeDictDataServiceImpl implements ISysTreeDictDataService { + + @Autowired + private SysTreeDictDataMapper sysTreeDictDataMapper; + + /** + * 查询树形字典数据 + * + * @param id 树形字典数据ID + * @return 树形字典数据 + */ + @Override + public SysTreeDictData selectById(String id) { + return sysTreeDictDataMapper.selectById(id); + } + + /** + * 分页查询树形字典数据列表 + * + * @param sysTreeDictData 树形字典数据 + * @return 树形字典数据 + */ + @Override + public IPage selectList(IPage page, SysTreeDictData sysTreeDictData) { + return sysTreeDictDataMapper.selectTreeDictDataPage(page, sysTreeDictData); + } + + /** + * 查询所有树形字典数据列表 + * + * @param sysTreeDictData 树形字典数据 + * @return 树形字典数据 + */ + @Override + public List selectListAll(SysTreeDictData sysTreeDictData) { + return sysTreeDictDataMapper.selectTreeDictDataList(sysTreeDictData); + } + + /** + * 查询所有树形字典数据树结构 + * fsd + * + * @param sysTreeDictData 树形字典数据 + * @return 树结构 + */ + @Override + public List buildTree(SysTreeDictData sysTreeDictData) { + List list = sysTreeDictDataMapper.selectTreeDictDataList(sysTreeDictData); + TreeUtils treeUtils = new TreeUtils(); + List result = treeUtils.buildTree(list); + return result; + } + + /** + * 新增树形字典数据 + * + * @param sysTreeDictData 树形字典数据 + * @return 结果 + */ + @Override + @Transactional(rollbackFor = Exception.class) + public int insert(SysTreeDictData sysTreeDictData) { + // 更新父类节点 + SysTreeDictData parent = new SysTreeDictData(); + parent.setId(sysTreeDictData.getPid()); + parent.setIsLeaf(Constants.FALSE); + sysTreeDictDataMapper.updateById(parent); + + // 插入实体 + return sysTreeDictDataMapper.insert(sysTreeDictData); + } + + /** + * 修改树形字典数据 + * + * @param sysTreeDictData 树形字典数据 + * @return 结果 + */ + @Override + public int update(SysTreeDictData sysTreeDictData) { + return sysTreeDictDataMapper.updateById(sysTreeDictData); + } + + /** + * 批量删除树形字典数据 + * + * @param sids 需要删除的树形字典数据ID + * @return 结果 + */ + @Override + public int deleteByIds(String[] sids) { + return sysTreeDictDataMapper.deleteBatchIds(Arrays.asList(sids)); + } + + /** + * 删除树形字典数据信息 + * + * @param id 树形字典数据ID + * @return 结果 + */ + @Override + public int deleteById(String id) { + return sysTreeDictDataMapper.deleteById(id); + } + + /** + * 检查名称唯一性 + * + * @param entity 查询条件 + * @return 结果 + */ + @Override + public AjaxResult checkUniqueByLabel(SysTreeDictData entity) { + LambdaQueryWrapper queryWrapper = new LambdaQueryWrapper(); + String parentId = TreeConstants.PARENT_ROOT_VALUE; + // 如果传入数据名称为空的情况下,返回null + if (StringUtils.isEmpty(entity.getLabel())) { + return null; + } + + // 增加查询条件:名称, 分组KEY + queryWrapper.eq(SysTreeDictData::getLabel, entity.getLabel()); + queryWrapper.eq(SysTreeDictData::getTreeDict, entity.getTreeDict()); + + // 同级层次下不能重名 + if (StringUtils.isNotNull(entity.getPid())) { + parentId = entity.getPid(); + } + queryWrapper.eq(SysTreeDictData::getPid, parentId); + + + // 如果存在ID,排除本身ID的实体存在 + if (StringUtils.isNotNull(entity.getId())) { + queryWrapper.ne(SysTreeDictData::getId, entity.getId()); + } + + SysTreeDictData result = sysTreeDictDataMapper.selectOne(queryWrapper); + + if (StringUtils.isNotNull(result)) { + return AjaxResult.error(501, "名称:" + entity.getLabel() + " 重复"); + } else { + return AjaxResult.success(); + } + } + + @Override + public List selectTreeDictByType(String treeDict) { + LambdaQueryWrapper wrapper = new LambdaQueryWrapper<>(); + wrapper.eq(SysTreeDictData::getTreeDict, treeDict); + sysTreeDictDataMapper.selectList(wrapper); + return null; + } + + @Override + public AjaxResult checkCode(SysTreeDictData sysTreeDictData) { + String code = sysTreeDictData.getCode(); + if (code.length() <= 0) { + return AjaxResult.error(501, "不能为空"); + } + if (checkUniqueCode(sysTreeDictData)) { + return AjaxResult.error(502, "编码重复!" ); + } + return AjaxResult.success(sysTreeDictData); + } + + private String getPLevelCode(String pid) { + String pLevelCode = ""; + if (!TreeConstants.PARENT_ROOT_VALUE.equals(pid)) { + SysTreeDictData parent = sysTreeDictDataMapper.selectById(pid); + pLevelCode = parent.getLevelCode(); + } + return pLevelCode; + } + + private String getLevelCode(SysTreeDictData sysTreeDictData, String pid) { + String levelCode = ""; + LambdaQueryWrapper queryWrapper = new LambdaQueryWrapper(); + queryWrapper.eq(SysTreeDictData::getTreeDict, sysTreeDictData.getTreeDict()); + queryWrapper.eq(SysTreeDictData::getPid, pid); + queryWrapper.orderByDesc(SysTreeDictData::getLevelCode); + List sibling = sysTreeDictDataMapper.selectList(queryWrapper); + if (sibling.size() > 0) { + levelCode = sibling.get(0).getLevelCode(); + } + return levelCode; + } + + private boolean checkUniqueCode(SysTreeDictData entity) { + LambdaQueryWrapper queryWrapper = new LambdaQueryWrapper(); + + // 增加查询条件:编码, 分组KEY + queryWrapper.eq(SysTreeDictData::getCode, entity.getCode()); + queryWrapper.eq(SysTreeDictData::getTreeDict, entity.getTreeDict()); + + // 如果存在ID,排除本身ID的实体存在 + if (StringUtils.isNotNull(entity.getId())) { + queryWrapper.ne(SysTreeDictData::getId, entity.getId()); + } + + SysTreeDictData result = sysTreeDictDataMapper.selectOne(queryWrapper); + + return StringUtils.isNotNull(result); + } +} diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TreeDictServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TreeDictServiceImpl.java new file mode 100644 index 00000000..659575f8 --- /dev/null +++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TreeDictServiceImpl.java @@ -0,0 +1,124 @@ +package com.ruoyi.system.service.impl; + +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.ruoyi.common.constant.UserConstants; +import com.ruoyi.common.utils.TreeUtils; +import com.ruoyi.system.domain.SysTreeDict; +import com.ruoyi.system.domain.SysTreeDictData; +import com.ruoyi.system.mapper.SysTreeDictDataMapper; +import com.ruoyi.system.mapper.SysTreeDictMapper; +import com.ruoyi.system.service.ISysTreeDictService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import javax.annotation.PostConstruct; +import java.util.Arrays; +import java.util.List; + +/** + * 系统树形Service业务层处理 + * + * @author wangzongrun + * @date 2021-06-04 + */ +@Service +public class TreeDictServiceImpl implements ISysTreeDictService { + + @Autowired + private SysTreeDictMapper sysTreeDictMapper; + + @Autowired + private SysTreeDictDataMapper sysTreeDictDataMapper; + + /** + * 项目启动时,初始化分类树到缓存 + */ + @PostConstruct + public void init() { + List list = sysTreeDictMapper.selectList(null); + list.stream().forEach(dictData -> { + List dictDatas =sysTreeDictDataMapper.selectList(new QueryWrapper().lambda().eq(SysTreeDictData::getDelFlag, UserConstants.NORMAL) + .eq(SysTreeDictData::getTreeDict, dictData.getCode()).orderByAsc(SysTreeDictData::getOrderNum).orderByAsc(SysTreeDictData::getLevelCode)); + List buildTree = new TreeUtils().buildTree(dictDatas); + TreeUtils.initCacheTreeData(dictData.getCode(),buildTree); + }); + } + + /** + * 查询系统树形 + * + * @param id 系统树形ID + * @return 系统树形 + */ + @Override + public SysTreeDict selectById(String id) { + return sysTreeDictMapper.selectById(id); + } + + /** + * 分页查询系统树形列表 + * + * @param sysTreeDict 系统树形 + * @return 系统树形 + */ + @Override + public IPage selectList(IPage page, SysTreeDict sysTreeDict) { + return sysTreeDictMapper.selectTreeDictPage(page, sysTreeDict); + } + + /** + * 查询所有系统树形列表 + * + * @param sysTreeDict 系统树形 + * @return 系统树形 + */ + @Override + public List selectListAll(SysTreeDict sysTreeDict) { + return sysTreeDictMapper.selectTreeDictList(sysTreeDict); + } + + /** + * 新增系统树形 + * + * @param sysTreeDict 系统树形 + * @return 结果 + */ + @Override + public int insert(SysTreeDict sysTreeDict) { + return sysTreeDictMapper.insert(sysTreeDict); + } + + /** + * 修改系统树形 + * + * @param sysTreeDict 系统树形 + * @return 结果 + */ + @Override + public int update(SysTreeDict sysTreeDict) { + return sysTreeDictMapper.updateById(sysTreeDict); + } + + /** + * 批量删除系统树形 + * + * @param sids 需要删除的系统树形ID + * @return 结果 + */ + @Override + public int deleteByIds(String[] sids) { + return sysTreeDictMapper.deleteBatchIds(Arrays.asList(sids)); + } + + /** + * 删除系统树形信息 + * + * @param id 系统树形ID + * @return 结果 + */ + @Override + public int deleteById(String id) { + return sysTreeDictMapper.deleteById(id); + } +} diff --git a/ruoyi-system/src/main/resources/mapper/system/SysConfigMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysConfigMapper.xml new file mode 100644 index 00000000..8b979061 --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysConfigMapper.xml @@ -0,0 +1,112 @@ + + + + + + + + + + + + + + + + + + select config_id, config_name, config_key, config_value, config_type, create_by, create_time, update_by, update_time, remark + from sys_config + + + + + + + and config_id = #{configId} + + + and config_key = #{configKey} + + + + + + + + + + + + insert into sys_config ( + config_name, + config_key, + config_value, + config_type, + create_by, + remark, + create_time + )values( + #{configName}, + #{configKey}, + #{configValue}, + #{configType}, + #{createBy}, + #{remark}, + sysdate() + ) + + + + update sys_config + + config_name = #{configName}, + config_key = #{configKey}, + config_value = #{configValue}, + config_type = #{configType}, + update_by = #{updateBy}, + remark = #{remark}, + update_time = sysdate() + + where config_id = #{configId} + + + + delete from sys_config where config_id = #{configId} + + + + delete from sys_config where config_id in + + #{configId} + + + + \ No newline at end of file diff --git a/ruoyi-system/src/main/resources/mapper/system/SysDeptMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysDeptMapper.xml new file mode 100644 index 00000000..1786c907 --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysDeptMapper.xml @@ -0,0 +1,161 @@ + + + + + + + + + + + + + + + + + + + + + + + + + select d.dept_id, d.parent_id, d.ancestors, d.dept_name, d.order_num, d.leader, d.phone, d.email, d.status, d.del_flag, d.create_by, d.create_time + from sys_dept d + + + + + + + + + + + + + + + + + + + + insert into sys_dept( + dept_id, + parent_id, + dept_name, + ancestors, + order_num, + leader, + phone, + email, + status, + create_by, + create_time + )values( + #{deptId}, + #{parentId}, + #{deptName}, + #{ancestors}, + #{orderNum}, + #{leader}, + #{phone}, + #{email}, + #{status}, + #{createBy}, + sysdate() + ) + + + + update sys_dept + + parent_id = #{parentId}, + dept_name = #{deptName}, + ancestors = #{ancestors}, + order_num = #{orderNum}, + leader = #{leader}, + phone = #{phone}, + email = #{email}, + status = #{status}, + update_by = #{updateBy}, + update_time = sysdate() + + where dept_id = #{deptId} + + + + update sys_dept set ancestors = + + when #{item.deptId} then #{item.ancestors} + + where dept_id in + + #{item.deptId} + + + + + update sys_dept set status = '0' where dept_id in + + #{deptId} + + + + + update sys_dept set del_flag = '2' where dept_id = #{deptId} + + + diff --git a/ruoyi-system/src/main/resources/mapper/system/SysDictDataMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysDictDataMapper.xml new file mode 100644 index 00000000..8da9030b --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysDictDataMapper.xml @@ -0,0 +1,124 @@ + + + + + + + + + + + + + + + + + + + + + + select dict_code, dict_sort, dict_label, dict_value, dict_type, css_class, list_class, is_default, status, create_by, create_time, remark + from sys_dict_data + + + + + + + + + + + + + + delete from sys_dict_data where dict_code = #{dictCode} + + + + delete from sys_dict_data where dict_code in + + #{dictCode} + + + + + update sys_dict_data + + dict_sort = #{dictSort}, + dict_label = #{dictLabel}, + dict_value = #{dictValue}, + dict_type = #{dictType}, + css_class = #{cssClass}, + list_class = #{listClass}, + is_default = #{isDefault}, + status = #{status}, + remark = #{remark}, + update_by = #{updateBy}, + update_time = sysdate() + + where dict_code = #{dictCode} + + + + update sys_dict_data set dict_type = #{newDictType} where dict_type = #{oldDictType} + + + + insert into sys_dict_data( + dict_sort, + dict_label, + dict_value, + dict_type, + css_class, + list_class, + is_default, + status, + remark, + create_by, + create_time + )values( + #{dictSort}, + #{dictLabel}, + #{dictValue}, + #{dictType}, + #{cssClass}, + #{listClass}, + #{isDefault}, + #{status}, + #{remark}, + #{createBy}, + sysdate() + ) + + + \ No newline at end of file diff --git a/ruoyi-system/src/main/resources/mapper/system/SysDictTypeMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysDictTypeMapper.xml new file mode 100644 index 00000000..55b4075f --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysDictTypeMapper.xml @@ -0,0 +1,105 @@ + + + + + + + + + + + + + + + + + select dict_id, dict_name, dict_type, status, create_by, create_time, remark + from sys_dict_type + + + + + + + + + + + + + + delete from sys_dict_type where dict_id = #{dictId} + + + + delete from sys_dict_type where dict_id in + + #{dictId} + + + + + update sys_dict_type + + dict_name = #{dictName}, + dict_type = #{dictType}, + status = #{status}, + remark = #{remark}, + update_by = #{updateBy}, + update_time = sysdate() + + where dict_id = #{dictId} + + + + insert into sys_dict_type( + dict_name, + dict_type, + status, + remark, + create_by, + create_time + )values( + #{dictName}, + #{dictType}, + #{status}, + #{remark}, + #{createBy}, + sysdate() + ) + + + \ No newline at end of file diff --git a/ruoyi-system/src/main/resources/mapper/system/SysLogininforMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysLogininforMapper.xml new file mode 100644 index 00000000..b8178fa3 --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysLogininforMapper.xml @@ -0,0 +1,57 @@ + + + + + + + + + + + + + + + + + + insert into sys_logininfor (user_name, status, ipaddr, login_location, browser, os, msg, login_time) + values (#{userName}, #{status}, #{ipaddr}, #{loginLocation}, #{browser}, #{os}, #{msg}, sysdate()) + + + + + + delete from sys_logininfor where info_id in + + #{infoId} + + + + + truncate table sys_logininfor + + + \ No newline at end of file diff --git a/ruoyi-system/src/main/resources/mapper/system/SysMenuMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysMenuMapper.xml new file mode 100644 index 00000000..eb995204 --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysMenuMapper.xml @@ -0,0 +1,201 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + select menu_id, menu_name, parent_id, order_num, path, component, `query`, is_frame, is_cache, menu_type, visible, status, ifnull(perms,'') as perms, icon, create_time + from sys_menu + + + + + + + + + + + + + + + + + + + + + + + + + + update sys_menu + + menu_name = #{menuName}, + parent_id = #{parentId}, + order_num = #{orderNum}, + path = #{path}, + component = #{component}, + `query` = #{query}, + is_frame = #{isFrame}, + is_cache = #{isCache}, + menu_type = #{menuType}, + visible = #{visible}, + status = #{status}, + perms = #{perms}, + icon = #{icon}, + remark = #{remark}, + update_by = #{updateBy}, + update_time = sysdate() + + where menu_id = #{menuId} + + + + insert into sys_menu( + menu_id, + parent_id, + menu_name, + order_num, + path, + component, + `query`, + is_frame, + is_cache, + menu_type, + visible, + status, + perms, + icon, + remark, + create_by, + create_time + )values( + #{menuId}, + #{parentId}, + #{menuName}, + #{orderNum}, + #{path}, + #{component}, + #{query}, + #{isFrame}, + #{isCache}, + #{menuType}, + #{visible}, + #{status}, + #{perms}, + #{icon}, + #{remark}, + #{createBy}, + sysdate() + ) + + + + delete from sys_menu where menu_id = #{menuId} + + + diff --git a/ruoyi-system/src/main/resources/mapper/system/SysNoticeMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysNoticeMapper.xml new file mode 100644 index 00000000..65d30794 --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysNoticeMapper.xml @@ -0,0 +1,89 @@ + + + + + + + + + + + + + + + + + + + select notice_id, notice_title, notice_type, cast(notice_content as char) as notice_content, status, create_by, create_time, update_by, update_time, remark + from sys_notice + + + + + + + + insert into sys_notice ( + notice_title, + notice_type, + notice_content, + status, + remark, + create_by, + create_time + )values( + #{noticeTitle}, + #{noticeType}, + #{noticeContent}, + #{status}, + #{remark}, + #{createBy}, + sysdate() + ) + + + + update sys_notice + + notice_title = #{noticeTitle}, + notice_type = #{noticeType}, + notice_content = #{noticeContent}, + status = #{status}, + update_by = #{updateBy}, + update_time = sysdate() + + where notice_id = #{noticeId} + + + + delete from sys_notice where notice_id = #{noticeId} + + + + delete from sys_notice where notice_id in + + #{noticeId} + + + + \ No newline at end of file diff --git a/ruoyi-system/src/main/resources/mapper/system/SysOperLogMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysOperLogMapper.xml new file mode 100644 index 00000000..018a7471 --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysOperLogMapper.xml @@ -0,0 +1,83 @@ + + + + + + + + + + + + + + + + + + + + + + + + + select oper_id, title, business_type, method, request_method, operator_type, oper_name, dept_name, oper_url, oper_ip, oper_location, oper_param, json_result, status, error_msg, oper_time + from sys_oper_log + + + + insert into sys_oper_log(title, business_type, method, request_method, operator_type, oper_name, dept_name, oper_url, oper_ip, oper_location, oper_param, json_result, status, error_msg, oper_time) + values (#{title}, #{businessType}, #{method}, #{requestMethod}, #{operatorType}, #{operName}, #{deptName}, #{operUrl}, #{operIp}, #{operLocation}, #{operParam}, #{jsonResult}, #{status}, #{errorMsg}, sysdate()) + + + + + + delete from sys_oper_log where oper_id in + + #{operId} + + + + + + + truncate table sys_oper_log + + + \ No newline at end of file diff --git a/ruoyi-system/src/main/resources/mapper/system/SysPostMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysPostMapper.xml new file mode 100644 index 00000000..fc37f49d --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysPostMapper.xml @@ -0,0 +1,122 @@ + + + + + + + + + + + + + + + + + + + select post_id, post_code, post_name, post_sort, status, create_by, create_time, remark + from sys_post + + + + + + + + + + + + + + + + + + update sys_post + + post_code = #{postCode}, + post_name = #{postName}, + post_sort = #{postSort}, + status = #{status}, + remark = #{remark}, + update_by = #{updateBy}, + update_time = sysdate() + + where post_id = #{postId} + + + + insert into sys_post( + post_id, + post_code, + post_name, + post_sort, + status, + remark, + create_by, + create_time + )values( + #{postId}, + #{postCode}, + #{postName}, + #{postSort}, + #{status}, + #{remark}, + #{createBy}, + sysdate() + ) + + + + delete from sys_post where post_id = #{postId} + + + + delete from sys_post where post_id in + + #{postId} + + + + \ No newline at end of file diff --git a/ruoyi-system/src/main/resources/mapper/system/SysRoleDeptMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysRoleDeptMapper.xml new file mode 100644 index 00000000..7c4139bc --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysRoleDeptMapper.xml @@ -0,0 +1,34 @@ + + + + + + + + + + + delete from sys_role_dept where role_id=#{roleId} + + + + + + delete from sys_role_dept where role_id in + + #{roleId} + + + + + insert into sys_role_dept(role_id, dept_id) values + + (#{item.roleId},#{item.deptId}) + + + + \ No newline at end of file diff --git a/ruoyi-system/src/main/resources/mapper/system/SysRoleMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysRoleMapper.xml new file mode 100644 index 00000000..58743901 --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysRoleMapper.xml @@ -0,0 +1,153 @@ + + + + + + + + + + + + + + + + + + + + + + + + select distinct r.role_id, r.role_name, r.role_key, r.role_sort, r.data_scope, r.menu_check_strictly, r.dept_check_strictly, + r.status, r.del_flag, r.create_time, r.remark + from sys_role r + left join sys_user_role urs on urs.role_id = r.role_id + left join sys_user u on u.user_id = urs.user_id + left join sys_dept d on u.dept_id = d.dept_id + + + + + + + + + + + + + + + + + + + + insert into sys_role( + role_id, + role_name, + role_key, + role_sort, + data_scope, + menu_check_strictly, + dept_check_strictly, + status, + remark, + create_by, + create_time + )values( + #{roleId}, + #{roleName}, + #{roleKey}, + #{roleSort}, + #{dataScope}, + #{menuCheckStrictly}, + #{deptCheckStrictly}, + #{status}, + #{remark}, + #{createBy}, + sysdate() + ) + + + + update sys_role + + role_name = #{roleName}, + role_key = #{roleKey}, + role_sort = #{roleSort}, + data_scope = #{dataScope}, + menu_check_strictly = #{menuCheckStrictly}, + dept_check_strictly = #{deptCheckStrictly}, + status = #{status}, + remark = #{remark}, + update_by = #{updateBy}, + update_time = sysdate() + + where role_id = #{roleId} + + + + update sys_role set del_flag = '2' where role_id = #{roleId} + + + + update sys_role set del_flag = '2' where role_id in + + #{roleId} + + + + diff --git a/ruoyi-system/src/main/resources/mapper/system/SysRoleMenuMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysRoleMenuMapper.xml new file mode 100644 index 00000000..cb60a852 --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysRoleMenuMapper.xml @@ -0,0 +1,34 @@ + + + + + + + + + + + + + delete from sys_role_menu where role_id=#{roleId} + + + + delete from sys_role_menu where role_id in + + #{roleId} + + + + + insert into sys_role_menu(role_id, menu_id) values + + (#{item.roleId},#{item.menuId}) + + + + \ No newline at end of file diff --git a/ruoyi-system/src/main/resources/mapper/system/SysTreeDictDataMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysTreeDictDataMapper.xml new file mode 100644 index 00000000..69421d5b --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysTreeDictDataMapper.xml @@ -0,0 +1,77 @@ + + + + + + + + + + + + + + + + + + + + + + + + SELECT + o.id as id, + o.label as label, + o.tree_dict as tree_dict, + o.code as code, + o.remark as remark, + o.pid as pid, + o.order_num as order_num, + o.level_code as level_code, + o.level_depth as level_depth, + o.is_leaf as is_leaf, + o.path as path, + o.icon as icon + FROM sys_tree_dict_data o + LEFT JOIN sys_user u ON u.user_id = o.create_by + LEFT JOIN sys_dept d ON u.dept_id = d.dept_id + + + + + + diff --git a/ruoyi-system/src/main/resources/mapper/system/SysTreeDictMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysTreeDictMapper.xml new file mode 100644 index 00000000..f274a9a0 --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysTreeDictMapper.xml @@ -0,0 +1,53 @@ + + + + + + + + + + + + + + + + SELECT + o.id as id, + o.code as code, + o.name as name, + o.stru_type as stru_type, + o.create_by as create_by, + o.remark as remark, + o.is_sys_param as is_sys_param + FROM sys_tree_dict o + + + + + + diff --git a/ruoyi-system/src/main/resources/mapper/system/SysUserMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysUserMapper.xml new file mode 100644 index 00000000..1c170fd5 --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysUserMapper.xml @@ -0,0 +1,232 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + select u.user_id, u.dept_id, u.user_name, u.nick_name, u.email, u.avatar, u.phonenumber, u.password, u.sex, u.status, u.del_flag, u.login_ip, u.login_date, u.create_by, u.create_time, u.remark,u.tenant_id, + d.dept_id, d.parent_id, d.ancestors, d.dept_name, d.order_num, d.leader, d.status as dept_status, + r.role_id, r.role_name, r.role_key, r.role_sort, r.data_scope, r.status as role_status + from sys_user u + left join sys_dept d on u.dept_id = d.dept_id + left join sys_user_role urs on u.user_id = urs.user_id + left join sys_role r on r.role_id = urs.role_id + + + + + + + + + + + + + + + + + + + + insert into sys_user( + user_id, + dept_id, + user_name, + nick_name, + email, + avatar, + phonenumber, + sex, + password, + status, + create_by, + remark, + create_time + )values( + #{userId}, + #{deptId}, + #{userName}, + #{nickName}, + #{email}, + #{avatar}, + #{phonenumber}, + #{sex}, + #{password}, + #{status}, + #{createBy}, + #{remark}, + sysdate() + ) + + + + update sys_user + + dept_id = #{deptId}, + user_name = #{userName}, + nick_name = #{nickName}, + email = #{email}, + phonenumber = #{phonenumber}, + sex = #{sex}, + avatar = #{avatar}, + password = #{password}, + status = #{status}, + login_ip = #{loginIp}, + login_date = #{loginDate}, + update_by = #{updateBy}, + remark = #{remark}, + update_time = sysdate() + + where user_id = #{userId} + + + + update sys_user set status = #{status} where user_id = #{userId} + + + + update sys_user set avatar = #{avatar} where user_name = #{userName} + + + + update sys_user set password = #{password} where user_name = #{userName} + + + + update sys_user set del_flag = '2' where user_id = #{userId} + + + + update sys_user set del_flag = '2' where user_id in + + #{userId} + + + + diff --git a/ruoyi-system/src/main/resources/mapper/system/SysUserPostMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysUserPostMapper.xml new file mode 100644 index 00000000..912f9dfb --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysUserPostMapper.xml @@ -0,0 +1,34 @@ + + + + + + + + + + + delete from sys_user_post where user_id=#{userId} + + + + + + delete from sys_user_post where user_id in + + #{userId} + + + + + insert into sys_user_post(user_id, post_id) values + + (#{item.userId},#{item.postId}) + + + + diff --git a/ruoyi-system/src/main/resources/mapper/system/SysUserRoleMapper.xml b/ruoyi-system/src/main/resources/mapper/system/SysUserRoleMapper.xml new file mode 100644 index 00000000..dd726891 --- /dev/null +++ b/ruoyi-system/src/main/resources/mapper/system/SysUserRoleMapper.xml @@ -0,0 +1,44 @@ + + + + + + + + + + + delete from sys_user_role where user_id=#{userId} + + + + + + delete from sys_user_role where user_id in + + #{userId} + + + + + insert into sys_user_role(user_id, role_id) values + + (#{item.userId},#{item.roleId}) + + + + + delete from sys_user_role where user_id=#{userId} and role_id=#{roleId} + + + + delete from sys_user_role where role_id=#{roleId} and user_id in + + #{userId} + + + \ No newline at end of file diff --git a/ruoyi-system/target/classes/com/ruoyi/system/convert/Device/DeviceConvert.class b/ruoyi-system/target/classes/com/ruoyi/system/convert/Device/DeviceConvert.class new file mode 100644 index 0000000000000000000000000000000000000000..14e6af1c891d2249407de0a4d94fcc1c315dd0a6 GIT binary patch literal 474 zcmb7AK}*9h7=77v?Phgz>O{nw!gkPHyiJ5r6$W*J8{V@RM@E~BHZAODi5Kx8p8Zi` zx)#)Atw(B5hJJ=y`hokXiF!HYnXux$u<5VvtO_qrf z5#?HCA~y6&WRi<=_yyUw%NIP9N}5Z8YPa`5u;%MWLC_4O5~DPlig?1Np=BCloyJ_; z$Q-l_xl-qB&N4=Dc%LdGBQaE&Or%{JD5VWEQmcfZ6XPhh65Aa8c zFKrKAOJI_3@-l?^`ThC^aED=nHcn#nV+>-P61oqjH2#s$KEGHHI%E5)3H{WRdR7;i zb}ur^*%_u*$$TN53A{5q{@qlB+f>;?xZ19as8*HNg;2I^wDaOgH%95*Hr|@k;v>H& zBy(Fkr6(rDkG6yVCa#yVkv&A{5^na+Ck(=Sk<0Q;JZDR-JmG4u7@VQSWpO9m9P#3> ju*OVc!$5TT30kw!!U1cz1`-^y_BQ8;>4=G#!_@H~y5?mH literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/convert/task/TaskConvert.class b/ruoyi-system/target/classes/com/ruoyi/system/convert/task/TaskConvert.class new file mode 100644 index 0000000000000000000000000000000000000000..0ecbe4e71981e99ff4a35f0b90532efa1ffeb7f9 GIT binary patch literal 464 zcma)2&r1S97=5GV`eRG8G9tPaC_!`SW(2Yn2%AAl_i>wK%$<>)S;0RmxjbKT%Y4I|eD6QNTL%!L1VAu%EqhzRtO3Pjv4`nhC!wE6vzDbiv z-lz=J)>O{NVkQ`l?o;hlEW3K9tfG~U*2W2^jJ6Do&?F-si>Y-<8aaL|rc;?vfY!g$ zdf4-^kA{y;6evPw%O^sQ_}y?UBgb(3*W*b<&Ls1d7lkN|q+1#@oHoN>fnXx6ZMW8# zrDU6Ys1jWP#EgM(tRcdO`VxYCR6BVxe{FX0e+PD zQeC-H0+*co@-E?ie7?N{xWG6=7Y8YZDMl#{3H>Wu+i*?jo=ld6-qhV2!Z5eBnYUG8 z{6ZC_IOE)DRW6mck$=oyc(e_HbL}ejt=m{`Hcc=UYgez04;)nU#0!z>w=t`p%S%Fb z=UT7Ljg8SqzvkyFwN?Wp=o8NVl}s4LT`pDqz_-Q9=s=kKIReM%NIlYmbVYLH3!-_sF#rGn literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/convert/tenant/TenantConvert.class b/ruoyi-system/target/classes/com/ruoyi/system/convert/tenant/TenantConvert.class new file mode 100644 index 0000000000000000000000000000000000000000..71307d17de0a7889aaf000e2c8e512577957e06d GIT binary patch literal 2206 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 literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/convert/tenant/TenantPackageConvert.class b/ruoyi-system/target/classes/com/ruoyi/system/convert/tenant/TenantPackageConvert.class new file mode 100644 index 0000000000000000000000000000000000000000..04d094753ebdb7ce838203aa4366e3eead52ec6e GIT binary patch literal 1553 zcmc&!-)|B@5dN0d@}tz&s;yeS&$~Sm{sW)< zUyLut7~`{llyMGD3pAIKzQ_Y}vorI}%r~?9>-V=G01mOA#r*}*2W~6vH3a#|GWAso4mRqTXw!A$Sr^0zJ`m#E)l^L>$8-prc#ON7M zD9wk4`;)+M4kwVp%BK9#X_OCA!)H>zYEagz_#j^ZrtPf7GC1Y@O^RM^nT_i7;~{`4e6+hu(UQc@8>TV zKKGIu~ku;JKR$QOcqjNNO-n;j{``z!} zH*e;-e?R#&fKL1^fCI1tXhAA~1>6z9jW{dPqDbdNx>KY#iFB8Sy8{T|9vOMF{JkZB zCA?Lndqtx8l2|T@(UR!j7eKx1pL6xNUsj?*JsuE=TJ?BPq`O3Vn@IFlkGG5T4v`+x z@Xi`MjCTdF1Me2;JvDeQ-Y0+W*YJS=_Tz&x{2>h=4xm*QYsDin{1K5JmA{Y5-^XO^ z;~G8@K!<2`$SNJUU&ALgd`iQo6{@=dM;zUa%V#QO<~oJ0f~oGJF&5@h zs^{jCtyem zkAF+j%^NK_HEqp{ z-gofXE!(n=gXq%n7`~w4i#oo9FYEXUzN+JE__~hscwEOfWbB(V_APu{VN-55zsX3R z;<0>G&Cbu27c?=4c{eEXwnR8@7&L;Fl8CuXH?*UvvM~@dAFMq5u@6Djm|DfoNMt|Jk z;(HTNsAN9EGpy-MDsrzn3S|~&hW+$7??7vSRZC{-bBo`7rUIGkYHDOksKKw3tq@Ns zLverxS!OpcUn)DWz7es?Y+&ek#$ZV!!G(m2_SIK|P&_cUg^mM;8C8CN+S=PNCWr>S7B{0D{q!tY z+>rr}3_2raRsTY#27cl=?w~v2p?i_O1zk8F-TE{(IG{FWU~bC5-0Z=uTM6cF)X;Go z-M16w9)`P@FgpmdlQ6F*#C^Qm@37WKutI|iHJpVSCe&dE>R}IR$1r5aHr#}3$xk`I~ePPpvy1Vt;U-51DzC9g*_katYBUc}XkC(h;pO4le1wqL}x3XfnLg`J8= z$CiT?ZNbKhGHBdh-lsFvTl literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/convert/user/UserConvert.class b/ruoyi-system/target/classes/com/ruoyi/system/convert/user/UserConvert.class new file mode 100644 index 0000000000000000000000000000000000000000..3ec93aca77ebd3c283c3cc83319ee377234c065f GIT binary patch literal 681 zcma)4O-~y!6r2|*A!z~>=qKWWKq9JCbKn+)grZhL+6IZG+?~ZlM!WXP_D0$t&>w_) zsW>3c{3yiRuvJrf!3Wzj)_XH=cJCi4VBdV$DRkAPTRX_`%!YiRcr$^;@KU*xM&;?M7E}B1I8&Dk`$ydrlBq zu^t;43a1D^pBEg==(Cf|8rFA8T-Z!%Ba{g`lp-#DWu$-jU;o+q+J9X-v41Csb(W{< zRQde;gqv^z2T7`p4u^!f-QGE2eqb-y-*Tjl8W!0|xidLRnQ6ziaEbb+E3h&X$Cqky z!p3RA9y4`pCfaK*IWoou8MHN?^Ny?=i%jM|xFQMSyUcT*gV1|%>`i>=;3GbD@CplT zVH$WU%~+g_u2d2ToBwKD$|P7<)>>?$HCEwCn6OiSfN6t4D!u;`=}gN$>u@Ce`w%O$ zEZ-XQgeDf5X;lbqzQ0C?ahLHnpVjKD(r+~GK)*0|&6u#n`@;0K%T>#a8d$-bnq37R E0pKma5C8xG literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/convert/user/UserConvertImpl.class b/ruoyi-system/target/classes/com/ruoyi/system/convert/user/UserConvertImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..3166f5d9e5938e0a4262430b3ee8489ccadefa2b GIT binary patch literal 2046 zcmbVNYf~Fl7=BK&VIf(dLQIPnO797QF2#B)2qLv@Y6#j;Dz)k?VTYbJ*=2S&jPbkU zZ}1oRwSEbM&NzN^#*h91|A#YPp0i2FCY`C^mvi3pUY_^9``h0?{tVzczDy#Gc^z-0 z_;@pk7~WFU+bO(*chzx2$3hb0SXAuybSx#2!h8~gSXR{gids?B2Z~x%)J+{9>bRxj zw!pxmbY!q35YOf|1+-On#}*hZO2;nMYumQBVQp8*87sOKtGa131oWH zYwUGWT<)`3Jdx`qI=}e>gm$&saqqj`dX*(BW#6;?z0LIn1v6X@tjg0hYcH(5jypOE z0z+lF>sUeEV~f7&FXBmK?XP*%{sfUuwEMK-s%}LS5wNl+s$VXmear~hW;JxZZs-3G zPO|LQy^4KHs`*d%4e?bquLd$WEim4l-4)NX8ft*qAOm;tky0rdSVM{ZGO&)14Sa%o z0{QOF{O1T8n8B=pGByO}x^~B_h=KdqH1GhQ8rZ@^1E1lMfzK871s-vko&$W*^0K^P zwpGj7HP^SF*cC2T?${6lBi&-DI`PZWWfTJ2vjUeJuKDt7R-GRu4!M>ssBhDtJQ8QJ z(OK=7aZB{amS3{JA}mcG($onb>5hIRY--1?ogIOf+j|q0Dchl4J5XIOP-mY%70=xWEClHBt$I((EAy1ir&030>e94BpUYrYE!}h7OU8;lcO+{4w*4AX?~n z5<2NIzQ!3uhEbnpjAyV&=`ur50ZmhCVi;#JMT+j*p(*Md&NIX`W^sX33>PWS@NUU2 z9my_}cZJ+h#0UODT1Qq#?oS#`lNm#vz}zURt31CYC(b+D$uqTe0Apn@Nloy|tWdn=7nC;f2kA4)t zfhH1-4}JhYlyT)UX;JD{h zE$c`mQRP0hb_8v=8O zcD-tP&3el<8?A=#RL%2x)pBd*NvpBTghWy(4)P7|FhlvjBGq?|3&)ok15HMYOM`I& zM&qxjF-1ZKNw*)9`lZ{KNZr|7Kh>nHW5XFGc?i<)Wcv&OA6 zkj1b745VQg$gsJ?W(bCWc$UB;1MAoj$p1@kfoyjej?b3go`6D!u82Nl_>vICH>PNb_|ov(dsS5 zks_EXeS;|FBG-sYt4SM^HvUCkA0az|INAG@CPV46^o?Mgz6Y4ZD4iI=V;G}7$V7L; zL?0{-V8nLb>izGcTuK-*?k?=MN;3;_wSqhSNAj=)dvTRp` zk$5Jl7(qHugt#XCxs+JxE25vtiIDK1X7J`pkMyyQ^pn(-pvOBD%@4<=yj^fz?ye(- z4f(=&5Y>T;hj0Q@N)W9d~g$SA+DlGEn3N qf$ZG?!mVMDStUrkA7oz%GN%UV8TtU^m_T0M0Ky~9AP?m*%>MzAL=$=d literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/SysConfig.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/SysConfig.class new file mode 100644 index 0000000000000000000000000000000000000000..21e01451d024814a0485f26f745f030972d25b63 GIT binary patch literal 3589 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 XN}r?A)salk;hVt97P9Es?#=%KAESKy literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/SysNotice.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/SysNotice.class new file mode 100644 index 0000000000000000000000000000000000000000..b19e6afda896cb1750a2cbd59df8a4b645899008 GIT binary patch literal 2913 zcmb7GZF3V<6n-{slQd0BE$=8wL8mW3w-oRN5a=748d@c3Eh?5xvZM>iZrt5qgKx|z z!;GUd`bo#}gX0HhRGd*q3xlJ4_jgdJf5P!OyGxpe)Pc_Qo^$q|bMAA_ea>zF{`clz z01n`C4BdFO6+191%V}9&ljU_;-iY8#E8fIh3=y1-q9?|Wni!cV2E@olF(@(@#XgaG z6hk7mTThiu*C^^UhGS$3 z`mjlF&1GVqv^SZxifTr)ibl>VtHnx&J5H%kc2uqEr=~T=C34Kw>{d#;f;|rumdK^) zEm6bM=_v(mKFqY?7MP=}25HK*4G(K9aPdxQ_YGF<>ZYq8%;McLQHPHhrr{n{(A3*E z!}Oz8jtSQ#4O5>i7c;u;0hS}IEZeBDnJggq*zC?5%(2TW!7f`BLv<>Ss~6RrRn!bq zO;wypub)KTme*b1kmh#v_SJw6+S}&Tx~%W5_Yi~~RiB{vjkRt;^`2w(yufk8sw1m? z+E#;}H(0bxb-{7eS;ryhNKtnjEzg!bxpd{)(&r!F`R3|B-+p=f#;404{<-wYm8H*q zT>j{*J6~U0`ss_i*S=HG<6&MprPk8&r&AZ>< zl!glP)=ZKzE(sNyiiX+b(Uz*VAnFif?@znDcxCz4)q{h#Z~Sl%=JwS@7ILj3bNfn{ zeg3R8)3Z5N!DUt{)me3^erlyxDj-@bx9{mu*Yex1AeOSqc2+-O2#-6ed>N1i#IYX- z;^;wd90O3}7?fq7EJIKg3|e+x)k<1+UT2f#<{WQY_p6z*QON1Gnhwqx@$CrQILf#v ziQkUn5Dv$&fJz+4F&xK~EU(Ct!Z2TgtgUOVKEmxwV#j$dA#R& z__puyzIcRfy{Oq2;`l()rp2jSUWxym6_RMhCtzgOvh{mIHKIBCIG-W6qF~^GTaa=q z(&EmO)5&yVE}57dpG&1rC&$^wnHjB(?SSyuulh46yZNgwQ zYoIC^CB2dka758`Ur|l< zMICBTl-xlNgGZQV)WI0WedPpR;MW%uNMIs?mt-%AlXTvR&<0e!s5bCZFxj}e=KAW6 zvAW}HR3{9O>iG6>t-rG7`s&6SRHxwO#z0!?fg}iIVhtcdK>@<|uK|#jdLWYxft+d# zBwP<9MIh-lfXEC85Wc|;fQ0LT%r*qVmo^B#g=T7i@dqNO`+r6BI$FD8{flT@M7*n= zN{1-xsC0_*2$e2T)>GLa%0?=iMA=Mbizr*EY!l^CDvyb>oyy}>7V*Riz#$GMa0VUx rcXu=9umflL9BBO8mEnM9$-^Akreg-@xSOXoUsH@x9p`Cg31Q|x{g#)c literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/SysOperLog.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/SysOperLog.class new file mode 100644 index 0000000000000000000000000000000000000000..2b327b08f57198379db0b5dca61a1ba1eded6d79 GIT binary patch literal 5304 zcma)=OLP>+6^3sK^coG85Rb%3>=>KG5*`@*wy_5Tk_f^QcJ#`_&ZDJi#K26q+}(qb z^RN&m77!3N0)dTW9?r3HoFI`n0)!H0lPtVR7TF|=Y@%+VMY6~yS>)v8ztcUMQBBQp zbdIj7`rf+r)vbH0>gbdI&isam9;5d|w3<#<(keQ|VwlBg7H_e5o5dLxBP`CcILBg? z#d#JNSiHmHT^1KvjIkImr}xU~lAsz}w-Q>iOSkNVX>>L>3KF{ot!OX}+tCcCOG~Ep zQqb^^5L9ON=~iZ47Edh{ebEH|{2E-37sIXU<1i|)*@JA6Y>1(eB-{yx2& zt_TV>n8|d?;65yy4MVpYlA3Mnm}S$a&$2!@921uB>1=IkYi*KfPyC$Kq|uEnv12X)KQt)@eL zg7nQX_vo~H?MUR|n0Ic`3JCm^(S^ zzBv~@`iVP&^vd_`{h{-xAa%iy6vhxLdvVGg8^~4#qMXg#;ClwQ#q7ycLB4i=@Rp!TcK4!Y zX{nsU;XwovNxt7UjW*p*Cmq&(JmS7P?gqLt6sD6#Xv^l0PkRH01+Cm1^L~ESy?GOR z>&_0MfmP_1Wm>IvFUBv7T(~ni=}ixLhbM!Tcc4xei#Y8hlF>%Z;SBZbV6SE)aSo-D z(fwM_0kGs3!EW?-npR44U|-&+IWRQ@t?I|m#@5!2@%Vkac5O+ew%CBoX7lLu!stiH z{8tlkXJ6U2t-*;xp*ZICyu&PehGhPK;bohn5dWna{Qm%lHz+z3li@<=LRKv-RB^ADAiLYFZSHsy-%eDDF zvxe-nMskX#?MrOAB>2VS#P8u$&m zE1>0K$QJ4fsPp?gb9+#%Jae|qYiwF9PhLk}FPPCU(8bKmFVSpJe3NW$<}7%lLhWYS z>d|*5_?oHCW&S3q(h6e%#4 zUk-V9x}bHRYEJ9$0$Pm&s1_kuLu&~ck6sCV7QYAz`d>i*9`yM4UYov1_vy5rzC>S! z?m?aEs2&Q0E-x=~Hh_nGA9aAjgY<_@{`VSCk;-GRP?9{}>lA|U>SD(H13$XznXpOhd2 zYLL5RkUseCupcEo0r4Lwf{rR9vQh^5rxN5x>WHkALH-4he-{Dq-$H`kq#?znS1W`3M+tHa sH;8{@E_sjCN+9?bU#Tc50^+~DO6Wx91@vRQu!utNx{fZ>{oC&TA5wNTV*mgE literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/SysPost.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/SysPost.class new file mode 100644 index 0000000000000000000000000000000000000000..7b5f81912511613dafb68b93adc5533ab066174a GIT binary patch literal 3743 zcmb7G`EyfM6#j0PBrQQoSp*eX)wV3Jf=d+VXsD$RU*p5;5w!5(jR@BB{$>Olcz-K` zjl3U?;BDTIMeq*q$0K-`_Y)Dk$NT#coaFrjcJ^Tyr@}ZbQ0GY7(6gP=b_~nxh;I=P z+XQB|S*GLarrW7!3$mKf^Pdn1@*ys zD=Rmfrse7`xpf4VC(>3H~31 zy0+m(QUze-kBcK;A64e<6GC(L)X4YOlqPA}?pV-nojyl#qeM~q+yccUG;(s_)?4JI z&X#(H^4ceDSK6`uyuj?{4Y#g-J3KhJ>fsH;N52?3{{unn&gwl>1^Wb=r$x?-$yP&V zIV{z(-*DVYuUZg#$}kQ0X@Say#!m9=GoF<&Oxa$@bxGU1$}zCgRD(KIB869dSmhov z2>#MQN}ZzP%AA(5a=Kw^$)dB9!bp$S^hlRQuFLv{#y~_>W!5<326Gxp^^9$Xa^XXq zJHg3VS|0Ll_G^85*2s93O|?vYViO$K(b_GymHOc@AqwZDDJ+c3>E?Koah=vxFtQnGYbo_qtA7sTqbSbdWA=VNiXGS)#RXi9B8gNKFYzLI zIfNA9OjGB&vb7k+B|g{s=M;_FC_cfbWKqadJ$e?Oc^3ZJxA>ebLblB5_TebLV7De4 zEgql9WgAQ2p>Yc}v(uI>NBgf;cVvtPzFQ3AOMz9Bj7tZ6VD!?co_5HRZjrN6x4k2g ziXTkG+hYflslAC9&!_+Agt{pfEb$#(y)x~t|G!?ic0zhSFHM>^dx8#C&jbdHdZKaq zZ>S0(tlSQ5D)sc0Ja*!=1_oWhHL}_kO24|&p=!au;-CC0+$g)fI-t3F$8^zeg{Aa+ z1qkCFEXN86JV2)kJV^i43UqFw^CCK}r2kb8R%4CCTCAg=Is%VMJch?fK@Fo!XVAWq z{>k)7Qcx33Ry;)tYOKkMr%6H0rWAFwt5-K+vx8RBQjZ3%#Yo<|56|E(5~zo1-9~U* z$WYK;+jJeGsq!~e{Opl>jz1$EFc(!=g9J%Y-xAx=PA9=_s=R2lvzO|oUr~9DgbJ{O zkoV!bz~$<(%e$~!U1lcyP^f9;Z>TJNz6I69lAoXNTPY9h-H*E9DS=9lP7p{{DUgk2 zfN;w0C&wKGrqjXm*h8N7#wkww;@Hn09l#4@vkVn;F=j<#89RX&aj+bzs!~#0h*WF} zQht31o_b|SRhN=_?#`t231GXJRC+Qgy@Zsq6UdY&Rb5JICz0AU1u4I_1yAuZq(Y^n zI;SAT9iuYU?J<}MF`kxg()N@G8!836pTHiN0<7O}g1>E4HiKmqE(QBi8L$FIc_86Z zAd-M|PXWa5Z-HJME(4_E3e{z>(R#}Q$s(sNk2l)AW%}GAAo(eP_@hMd7p3xbgQfy^ zyzx6Cdz*ei&2`k)&uAJzbO1B!?;>Fq3w0#SW}%*hIV{X2VIB+fNw}MZ1tctF;T{t1 zW#K*&7O`+Y35!V>z>@K(ucbi*xHQNLxEp;~f>&tz_S2*-(hNO9Q}0!p5wD>WuTu`* epqhI#5Ew(vj|x~``0{v|_F5vN;S8ELE&LC{wS`px literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/SysRoleDept.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/SysRoleDept.class new file mode 100644 index 0000000000000000000000000000000000000000..0479a66525ce46e3bde689536d6f889ae0612cea GIT binary patch literal 1293 zcmb7^ZEw<06vzLUhw`vGSlQ;o)QKAkWzy-?H<-OpF(K(@A~@Zvr9i{FKubz7;ajOu zmT2M!@Ix8TDODI^!a{QIIrrh5-#PzVe*XIQ1HfZEO=1~a30%Q854jlbC2$}4BqAup zkxJ5{%1$wk6?S&mc@RS>22DV6osL;;38eKiTPUd!xxyeB=gY|Gmh z2(9-(D zzGrr|meVyXTdNKGZwTTgC8MN?w&}6Ud6j(OU#diZav+nR-Df@y!O=JbpZ;UVJ*Vcm zmfaRuy%2f4O39NoXW%x?R~A>3o&qazb7UyU$VeeAV+Cn}UB_)}M$c%Tni^Htb?m<1 zz&))ousSW%)#^dU=VL#HT^SGYh$HRExQ-h#N%%y@Ew(msM__A#HWAxItk;b*v*`(x z7KW*LL%+U*w|c#LqE`yf17)UE z4pI~r&k^*#j4ZwWW%_7B^dk{{i*%$}0zEclXj~;vqgl`#RnH+l`N!6RL%FhZ@8V=?}!|$pAkKWq{P$_;v*!K iBpFL&j37@S6}kcxNG+9WEK*fFQ^483ZO{7sz55#_aSAv9 literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/SysRoleMenu.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/SysRoleMenu.class new file mode 100644 index 0000000000000000000000000000000000000000..1c0c26bf4b7e80d7800eba31ba936692b9b3d6c1 GIT binary patch literal 1293 zcmb7^ZEw<06vzLUhXP$4tZegP>ckD@VbbZ;H<-Q9U_#PuiQsgvmI4jy0xc=Ugm0xr zS)z#_zz=0Sr&M8x2@A=+=iG;Le&_se`T6VH4**Z`T*flC61awK9TPUe~k-yeGY|Y|Gmh z2&}(9-(D zzGrr|meVyXTdNKGZwTUml2Ot`+w|DwylO80FIA#HIgrlH?lT{U;Ak9zPyeywo>TK& z%Wey-UWzroc+v90e-U3X(`ESV2l)*KymL(KDK7rbg9u9lP&0 za8GLttWL{xwR({8<=Bs5SHWXE;YfQ5ZsL|g5!@*VHa&sT z!Z0;&=+}4fRzOtw_I^5Gpv;uY zL5lq1IfCAok)hYWOdm~%ek7uAk&ZM=pvQ(ZjjQBoGz*%eg$syJ{;{>-5U_-GvX$`x zT*nRa1S?77<;!0X`Z#lWZSLg_+zu}9`)tHhD1Jp~l3J#9P>Luv{0Qs5F9qqDpX*_2 zdPs9X!m}Wob3yLT2MN!DY!gWKKOo#t2BF6sfV`vs8*XRjJ0d5A&xl??Qe%Y?;v*zf inT#beMo=b@DqR8cq*j4yEK*fFQ^1+OZO{1qz5g5FuL@`Y literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/SysTreeDict.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/SysTreeDict.class new file mode 100644 index 0000000000000000000000000000000000000000..2610d78fc31604939e5294ec7c4af17d88e201c5 GIT binary patch literal 7229 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 literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/SysUserOnline.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/SysUserOnline.class new file mode 100644 index 0000000000000000000000000000000000000000..70aef709a018cd6e92efcd4f442f79597c77e09f GIT binary patch literal 1982 zcma)+S#Q%o5XWcJIEUlXG(EVFLQ9hrYM=oMZSznNP|0cpEq#FpoWz7JiMO&HP`?#O zAS5360DLIK%;sL!5-saw-HwODZs-NQVTzx)&1S%0 zx-DAVcSOUDxbX3`dLTZd8-s)h8D!=5A*b;C*zt~#ho`o=&!Z}XsbcAfLGp=cqT`Ii zeQ&?pse8epTW{k^ffBTj+6=VlN zsxcsR0|;bA3u34syIPQ&dJsbed5$12VnFB$63D6+B&C8lT9Be1B&CA9M37f8AoK2&HF%wA|PA?c2_A zqKO~C4`n>3RAGn-3(39b+=p|1=lpN^`Rm&c08jB;!ZNm!xQ1;W@^L&!;votWA}A)1 zl4wz3rqGa5Al_0kN}B8#9=n`3Qz-sRm2@WuvW3}w=Hn0?jYIJ1KX$^i>z-p; z9f8$Lk;ki)T&mjxr)9i0xtjD8Sect6BZI7r6w)$QkQUgrosO#a_12l8QguDsa{UJG zsm+1eZ5xi-2r|AL`*G~bc#J0;X-~#Y+>%McXEN@vwTXKITMM*_*d}7VX`UM`PoTUo zOx+v$^&K2(joPVJJFK488z)+o3ev6?gj!aemSD_MxoQ*^EvEPThDD0KpAHx(Go^Bn zqPTdDp!XHz==CqtM-!qSiRfFTBh4|;V?&n4Rq|At1fab<+W2+52@ h#u6DL$P>s6T>*-umO?d_sH&YQ;N0J~=luTO{|%*|3mgCd literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/SysUserRole.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/SysUserRole.class new file mode 100644 index 0000000000000000000000000000000000000000..2d0e324589b9f90953d1084db130e22beb9b8902 GIT binary patch literal 1293 zcmb7^ZEw<06vzLUXILGqZ1Z92#0`Zq>2&HF%wA|PA?ao!Y`RxVfrfQ~mbAr$Z>2_A zqKO~C4`n>3RAGn-3(39b+=p|1=lpN^`Rm&c08jBeg=K6daShu%#Z|GrRsXN<@ycW zQ=0>`+cq4v5oCNh_T$)<@EA`x(w>ByxFwN<&m`PoYZLbbwiaj;u}#E!(>ym?opM8o8nsicc33^FH%_!F6{KA)2(_#_Ey0+ja@8m-T1@Zv4T}_eKOHboW=iEC zMRD;QLGLTb(d%EPk0wMv64AFvN19`x$A&D8tK_LP3!0tSko zNOM5Kvml#uLGI263D1IT6G;9)Aly&}p~oD6yrcgcZfEX0A}7jcL@yAN hj3qKgkS35aT>*-umO?d_sH&YQ;N0J~=luTO{|(aD3Zwu4 literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/TenantDO$TenantDOBuilder.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/TenantDO$TenantDOBuilder.class new file mode 100644 index 0000000000000000000000000000000000000000..1b556bf020d0bafe73a0f6c7f0e58bb23ecd7d9f GIT binary patch literal 2754 zcmbVNYjfLF5Iu7IkQF&8ZfSUw)`7IO9j6YZJkvnigp`1r07(NWugZ!hsMu0QmKpjF z`g15#k{LR~58y|2IIAn$YHf%oAMWnaYWHaGy=U!zp8fSVfNQV~T)@pd&f}Jt6)~$~ zJ{7Yj=C+v6#H@??T+A0aGz^%ylf#CADcsbtDgH|_U+MT-$Ci$59d~u?==es*w>oxp z+|%)$q|m(A?5CAIMsMy{c6=)mbYJT_}>1KhO}q39Sz0Ao2@W#qjv?{_d?4K z?{=JEo#m79^F~zCdwa)kx-C*?hdD7)vkE!WeKKd|gytbIoq z4d=H5H*mIH=^^f}8@jFfsuemcGA-NoyI!~=h8)*j&kaA;Fj21Tu=f>zk9E@x*K;^1{{4TV1!c=LG!7y5~8;O3UhW941_C z*nYbnbp3~Jz4NdWI_w%bpjjomxn~sjQauS-gBu#t|CnCC?B8CMtre@?st}6ALJ7C>?k6!u*qlnHT%%Yw29SksgR&mEFB%6ZgU9 z?xU$8S1ZZnEmQV2el{^@3{CYzGHTD|sA*#;Un{ABQ&>rq*sPWM1IEWftu&Z1K2O(5 z$(dqAtC&~@hidV--M^@N_C7d>&E^Bg=8Qg0DleLZ;oD0@j?aj1JW=b>jI{zn9)!zw+%g?HmxSYu(m?eS6{d_QrXku(4IS?V58{JWQY$7>P zA-Xuaqs-VuawS4^X*5xGY$CZ+A-aqYX@?FwH#U)oLWnA(JJQD{5>*LNb@XB9$0ib4 z3eh}1q8-|rF*cFtT8I`#pV8#lL^2pcw1n%l($T$G^@~qo?pB{r+)psUmFCKDWw~-( zI#-^{;F>%_Ve|;730mw=5Pw7C@BG2xknwx;6jQrLm@b|=#7pE#9imt~t?&e0@r=T0 ziNCCHM&f4`&I+DUI43!;D6C8TRfY2se@$US@aqas3O+}Ch&TSP|3`8EDSQ$Q`VG8G NZ?OOKJgsZU{|hn$G2j3I literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/TenantDO.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/TenantDO.class new file mode 100644 index 0000000000000000000000000000000000000000..d4f57d8d8c819d9d7b44ae1555468390e86fc36c GIT binary patch literal 6856 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 I=|+(J4Nesl>Hq)$ literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/TenantPackageDO.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/TenantPackageDO.class new file mode 100644 index 0000000000000000000000000000000000000000..1b23fb2c39750ba647b3c15a586bbccfed01aa10 GIT binary patch literal 4644 zcmcIo?QvwO~8|NX~50eBq0H1O37G+a-^#Mfl`x-8$2<#}0NF!0R`pD$)`7~c}=+Znv1 zsF!8?9a&zHaCeqgod^dw(TsLt;p5GJ6O%vaj%@0ic(8P~S{Mf`#O#D=MHwAFr zz|S;{t~j3U`RDDmO?%yW;@oW0t(2X*25ZjqocigC-Do%s4LxpI!{Gc?`*xq_^ zzUHmZXvi%!J>RW57u<%sT5(Q!Ud^|Cx8_l2dVZ}|Ew0+Ns#~r#i`DH_N*Y_0W}|4g zaK#mydUIvkO8VPdj)tjysm+yFh!(9=Pp_Qg!?UYY&c<9^_UmroWU4`*n)GNmCY|$q zXWgmOLaOdm?fRw$qw09gxiWzR3fT1BN^#lo2}&=!>mIGt`I5u!(&I6!!Lghq{^-7< z2!cQBIu(h(SK@!t_R1AWz@veIdb76e7I}`u# zfuGZ*HQPHQ5t7`n8yl*u6XO$e4B-6I+`<_~NSckH(X99y5`p{rnDW*bi(E6tB0tQr z);mqVrm{-|$6e3$PinXn&uh>VVzm<1LW~n-KE`STwIzG=|6}_-N`t4w0?91bn)Nm3 ztXp9}IMC5B9+4KHp}7B!Cl`b_T)$>lnlw0~`qLFZm_=8%)G*S}XE*)YQjim-G{F(7 zX*elEMYYBMrL9?b7QTfse9*!qrYub35er43j^a@bNA_vnf`b(c%d*ViIP1*9CveKb zEKXZ^LZ~x1YvGGn*KlyJNvgcI#%$;O9d9(Zocak1uj361TR6p2n##N7fb$F9v#Gok z0lYncP$2(YWC!iOF9;M~|e3pGgmM*T?4({*F7CK!GAA zjqD!DTkvG+f+w>_?qdOcEKu#G+T;BF&>B9BA*AsUe3aXd5t1iV{+Vw=yL1OVza_1q z#5KhyaDXeb%2m)sK833n{(yu)dt%TBae^F60dNu@=T4JS;M%F2>{LG7RhcE!MOp7u zKGId0rPoC{)u}w$RhcEr;5ywlI+c%hQC27aNxu6@%!MA5XiN?|RhZmC;&&Y(CRo zC*Oj(Fjd$^I@lH_?;xY7UGy$aaVaII|AO9J@-BvM!J5t`cVR_{zLHKzUn!N-1(Ujq z>@D==^jsMt1t^_R??As{mZ896fs8Yra+8YpFQX#r(IP&${1fd)&x1Pzv~T(5wv z5JV~hC2w~eUN$JM7gg}B;(8lT;{uOn1b@e8cvX9t+TY_Mc?q_P-{2CdB#-tcE|b!U zc?F*(mBKY_;&Y_<7aTkP=Si7(gSX`iq|*2mFZ>l!8N9_F_6(_B+{FX1Nm*E>Z?hOS z%zvTZz*_npj2bBO$JE}zeTq!~caolQ=BUInx%{Jyc6G_91Rw87DtL9X^j7c^js&mC zP*P^Je^Sq+_Km@rCzQT|6{6v%%z>gOj&lOz$H5jPUzPNfG3dp~RGuP@?{l z9*9^Cfs&ciBhf%96^K$T5o<(g4VH{R#PSHWj6ig#doq`lr(rw^yowFDjDvq9<0>hs z^yir-N#?_pRY)b-4Hr-)r89kx!6TKzb2x|^DI-YDKahDB_an*FXd1J@uZ(wTVsGCg zj1lM&Ai^ZuSs^SHgY6+ZxgpqS2k+s#2tpO}9+f?d}Cp3#-7rE^HitZEs1B&h!{vkyV2>(Gv4+{T%q<8WD&T%mu8y6SD z^lAJKtJ4peH8=Q7@tLO&`CwEOrgo4lOz&{y=8`*{A31#o_cX|n!0t(<+bZ*XI6!DG z_*jrzk`F!?0?egNtJ09A_0-T(jq literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/ImageVo.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/ImageVo.class new file mode 100644 index 0000000000000000000000000000000000000000..5849ad9df6a4000eded56773dd11081e8ec7893d GIT binary patch literal 1882 zcma)7U2hvz5Ixu4FT3m32}xQ*fY1hT?6@vaK58c^ZIZSLNm>#b37*(Et9XmO8|-zZ z@>BQ$^J#-who<+;&<$S7&DVQwo~zv|WYKwR_Her`mG-z3Nuj@%+8ILb}&! zDHwB}?}ZE87H75@uo&zz;LMumyBodsj@$Xn*=a>r1Pvy3n4bH@sD(|ht1!LR2-?+7 zFL>ZpyAQgd+pg{gZO8Mg_k-$6+u3us112>0+;AhpGgh1#hRC49T>)iw2cF}_la7MZ zphx6txL^027C{{GyuEYJZG`oiFUU`DwEpBsK@Y;F+fm3!5z{eI*~pe`tmq(ODHGcx zJ+~D)jjz_7{fGoMl+97F8jim#P^dPYZZqbQq^>a2oz0c?WkR8Dblh&Q71Eew(WawY ztuTl?qmbPSdYy)Q!xP@^gujc@K^w1N(#AOyY`lzu!ukIu-sTj}*f@_1Hj1K6i}pS~ zP&j*hDz5ju)~;}pOR^S)mU$a@@F}6sP7&%n>)iFX;%Ca#;A{Two{m(BoQ*`!Ubi9XXJ z6<59V5b6giDZEB$M8+Imr*6j%m|_|}7vYHVXz-6neSa)CKQx%M4hc>l4;~pB{3hN? zf@{Q?5}c*-Pe={NJrR2(xTPi`Ldt!6_&w?Wdyfz)_i#`sg^9nI%jDWDm%f9p9hM(K z`xbiou>2de_1RJ%dbFitY|K*Bw94-|RnQ(|PF(-k literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/MetaVo.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/MetaVo.class new file mode 100644 index 0000000000000000000000000000000000000000..43975f53ef93b13254602bee8f01df51682e5be1 GIT binary patch literal 1984 zcmb7@TTc@~7>3_zw_PYj5DSWkLJ=tCuqt@qAjS|)R1Oy$61dPTWdjTCE@>ANev1FV z3okU0XuR+T_@mT!X4V$kC13(GvtPe?-)H9g=GX5ZKLJePNeWFEZH9?4IgHC;LJn36 z_i*3D0~3=1iNFntjzGuazWv^|inh0Ftp;V++np6K+=A~3n4Z627xtJXwiqnB-hn`J z&h^}2ULc+u*7}VZUG(H|(-2zr&yr?70ax(}k_WtTf%EtY z67|jx=rKv%DLOx->!bg0metYD-J$bGo>`tJ8*%1t<2#Zc(46l!LXJ3O;cl}JfN65f zkYkn{b7Xjm9z0_?pDUgzty;Z_Z1pCx)tktMH$iu5q^Nh3x-9x1$|Owmn7GA1!GFdD zdg-hi!s4tu>MM*-{4u7$k}CK@0n~TX9rT5p=b0@#FPZN-Q&`=PuL=G}DRk+q>JgF( zR~q*dxDh1wxejC{5@Y~(8$e`B*Xpc6)(EnG4v2hx5@awEBwi1)83{5J4RY#PX^^)B pd3O$oeEt$-I1(gL4^oH($wh;l>U<5dM-cZM5ZONxWJJ{<{};E?CMW;^ literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/NameIdVo.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/NameIdVo.class new file mode 100644 index 0000000000000000000000000000000000000000..f1dd488c03de7e60d7dca5e6ed25ad17232cae5b GIT binary patch literal 1999 zcma)7T~ixX7=BK&o86=fq_I#_QBgnxfu<^2n?Th9wLzejwD@u5lI)PJA&beTGxewV z1LTGaozZkgXB@q$GyVpDhL_g*yt^BkKrYHK=e+0rI?wyO@7etI-!H!bSip{mlR@d zjS2fKn_i$#EW1tDzpfC^&FwP9s#jx*u}!z>Y+(e6c$=gtNTc%2>+{zAV!h!$(Y`KDtx2zMa-_TGJ`>X+ueB)8rmcae#CJ-_a> z6p~WzOh|Gn&_xF$FbKnDs1|Xg;@j1)x9r1!EY1VlN{6c5To)h|>vp>y@=fx~yywnN zd25|OC?{G@yVLNgjP8if1o?e0WJ@7c@j9)lbJLafjPz)}EWNkzGG4Ke#U%?bVNzlG zIT~c)eJom-!K{S^k>0?Y7OtVFaPjm&-RQWDnou&@yEiAatXQ}QX~e>mTMG`Kv6?vx zA7Yl%_bf|DE$1Lmxn7|XOuGARIBW^~@X%?J!^QqiMq_ezaw0UGumS&GYtIZz&mXVx zNoK%-K^&KG0ppY>_?BY0$d&ZT!x9F$CTSz9Twmp>z+G|I=O01+NFjzPu8BZtF)?6; z4tO2YJSicT>tOIti2XogfAHAQVD@52aQt*|dT4MCb5U@SSH%S9eEw&|hU1(vEpo{7k2Ag&0mM zuH$mh;Bi!Nh24v>cWZcyx;Qzwj<+fC(@w76p`>GrBeFy(fp?jI3Q5EK7o!H=dx8lA zSNU(MCwMUwll;iaON$t|#$7t1OX&iye-^?)3LB!Iz|#pToIgI!D9zuDlV5^jy0omV z1awWNOLdZErt?QI7Q0A(C3>@{i6N^+7E>ssX>vF>jxI~6O< r{fg8Vk8mzybul6ZbIXi|vf0I0CQZ4E^QWIk|KrR>bt!xlFn{BJqU2h< literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/RouterVo.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/RouterVo.class new file mode 100644 index 0000000000000000000000000000000000000000..190e8f93f70de3b8f601c364c76f15fbdd907545 GIT binary patch literal 2924 zcmb7_OLH4V5XXCDy}csAjtwRO0(l@i!OO!WBymh^k%xmUKV%e=Pz7UYu+~}anteFv z0tYT|;6lCz2dE-N6$d^5ABy7NyOLMltWwHBJ+m|2J-_bhY5n`(-~SNN0)1H|jn)b@ zPBj+mEH+qdve;r#XR*!Va~6#p-OJGzg3^}kC_xj|J$WE?Q(C)v-E$0U_Zs&0rPmge zZ5yqYvIL0-g7S`P8IEdtf(lK$vu|6ubi%+qBI}Nu3K;0j|EK*I9;)AQ%Q@x zc9bUtovFfHcYM2R=x*2bR7Y>w9cftlfvvAWrC~!k-)tLZ%i+_Hhu(b8F!ie8de|-0 zja^H6zJmxZ$F>)r=YPxgJ>}qpC3wj$8kXTL2^uTUHo#Z0Tkvaj zVC_!tmJkK*serxyO1k`j4acFhb$EN@P>8-X>{X)c)UY}5naA#Cc)@w>^rj`OE)QJy z)RBmy3opTBP;YuLu)bP9GQN4F6Rt)n56(bnalSl)K|c$X!=jq#GY^*4m1h=5FD6cF zbKgk3LFc=Usqe|=BiFWcd8nk{#rLY!H2s#sgUK97(^ok{ndN$` zg0xl3QclH`uB+hm(hE5~8#yo3)@xtgT-(^buhHxDhDIfNNuy&dj?>E;P4H}z#R(Qu zG_BDq^r}WP^cuR#bE>AF610-2*{$#Fp_%3Ap`Zn}9W@PoVnG*Q#EjlT^VQI_3zXu> zFiUZ8!HD>R5g`U6q6|g^8jOfF7!htTB4QGBGMJx2P6Q_m@=dJh7zM`6++QUA2vkl( zW`jiy7zB(!f!?Mw_!4xM&OshE{*zL_V>3mkAw%QqiN@#You2U$v{S4-ci}0eqN9t! z>I!A35_p&nwR*?1cg_b>mlHw8;vlyZL1vRd#^NAVfUFGx;R|7qIl2HD&q(^WSVlGz zK`zp}J>w`NC!bO}4$=U~y&)i65ezb)n2}68BM%cn^yG|W;vioGL=FMrl4Owg5<#+Y z5S0k>elkcl4zdT3M?*mPO)$s@i6FT+h@A-XVKPWA4&njC9|FRUl0g>eBglA-YvJhZ33JOS)O;{hOxOu3AfPo|+AyE*K$z~)2o1M516#oN% zz#m|~oJC6(&Fef!)yr_a6J$>0C}`a6KL*hpg-iz%2` zl3`hft2VBsIe#j{bpfo1bwh?#S+3dmEDaZn4nCLprVO_neBt1>gD+)u$HrGS?kWsc z{Yv1K*8EB}2+OMr7ZlWGh5os)T&;QK+L~9Y`zE=_k5aJe3|QD;;NpGnp_eat<$L+X zuzYVy!I8aXZ_8K6wC%3cDuLcF2gL_^-a`TtS#$iEd@^y}kvOx4rF`*}Rjzny9V!I#{ytJMfJz z`d>ViH2Hie9EO3Ha0EwbXL;`vTqQ)}$fnBCOrbgV5yJ{_Ji%3-v*K)yKZW`!f}P}O zMNSu|XuBGKH*lIOCF*nR4E`C3CtbkK}(;K823EOd`xqdR!0XYeG>#K8q> zlaND>=YBz=_n{AKM0{uwMAQ_m@_4gXp5%V>NHrwSTfOrb`^_WOlsvo#99a%^j+|3G z*W=vyk1&nh+@COifSKIQ{Q+}nVtfx)w4`A#PjDz0lYgNvYdpiy4&2GCu?MGtNEJ*% zQUxn(3dVYd%ntgqX4Yy0r3*Ge>4KBB1?0paJL@z+#x4mZ(cM$GHPNQLkNKzbT^?f? z-{C#>S%MJ(-ls3gJ6^;&S_UrRB0iwSKdyWT3bZVg*vwP3Y*g?vrfE5t;Wu+|ZRZ~x zurd1_hi!bwUq?O1QEjH$O{V6%qE+MvR8)IY!3YN?swIMx9}t z#kkeMscB;-l6+)vYLkLHgSj2plY8j9EB;i$kd3Sn?>JHhy_GJQ(H1KZ1vyzW+8XIC z>a6}?Oe=}&n8O9qF>wv^v=T_;3MCX#Pq1$;(K0a0cjFUUW+dCUuwKw%P&TC~75{~J zL3nGO5XUgg2B3j4I$bf$YJ;_q-Clx8lW(ln4tGdZ9;LIt!M*tu{h0%M7!V2fFqj$A fc8XIbqwTcV2RZEFQ1@fd{)U~6rBS#X)%D7MoF};L literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/menu/MenuSimpleRespVO.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/menu/MenuSimpleRespVO.class new file mode 100644 index 0000000000000000000000000000000000000000..6cfef50a80b146e07e6127a8c8b8ba8a9950d2bc GIT binary patch literal 3477 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 KC7vdFXZb6L&H;D; literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/packages/TenantPackageCreateReqVO.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/packages/TenantPackageCreateReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..ecff55657b0f9e8634a89ce3b15c7bc96030c3f9 GIT binary patch literal 1258 zcmbtSTTc^F5dKcvZA)7~D;JSl!CNoDir1JT2@pw4iZwu*=-Z(^X&2kwvKI{g0K=2= zW}*+q#LEjY_(Bs%jK2CmD)B!Uc(6lQ$}==@=WDYHe7ysHg>_!Zlr8H`FT6KjcMW zsmns=x_z%1&Y6PSVo|K+vT15UtU0_)NZs+wDqrVnnd?P0yRs_sb~^EZff=%cBcB)) z!!8Mvp?%3F;00c7bVfoj1oK?KEph2s;#R5oGJ576A~c2Nlx+g0AZQy)wyEhwhOT&` zg{7IL{z6NJW9G%YCY2aG?jaMhKZd?VN4K?#$Z3{FQqAbPVRKtEbb8-NMl)3F5ib^n zsq(`w>P%HzAmz(MrmXX_Lp);#A6_49J*jU!tv}m|jmHk&zxuXWtG|8z{moW=^JRU1 zFGl(~!m?wzYy=(XWSIJ&K}0Zso(Q_78N?7n@1fvLDP|q5T##}PAE!awa;n0-6``HH z7QqNcN#X>Rpi9XvQm*563BAxh^+vjC^iMG-9}ddSMlkEz;AK9x$ogW}sxeSzl%6z_KGGdv4p z$z6CGVKP;NuW|BiQ@OW=BGjOWHDO!|0z@xPyHtHd>%kei7ltno{DF=D`rF`>({cN_ zN!~*RiflkKv_7KA(9A=qKW=prpMMqQwJ z%1TiO_Tj%@Lm(R5MH_*J$VJ0sb`jb6qs?%O)&nlO2WJ~q=Ww2Um@qHU|DuKXFO7B< A#Q*>R literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/packages/TenantPackagePageReqVO.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/packages/TenantPackagePageReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..d7927cdc1642fe2717743b513f082de6112f265e GIT binary patch literal 3501 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 Th)=|M|KaFhtatd&S0VQ=0rmFw literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/packages/TenantPackageSimpleRespVO.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/packages/TenantPackageSimpleRespVO.class new file mode 100644 index 0000000000000000000000000000000000000000..7a1c4bfc67808d1d97e102b55c3bdd761abbeb56 GIT binary patch literal 2484 zcmbtW+jA3D9R4=h&2B;pC8a>*Vv&OBC2T>w&{DA6q}Zmltx@pCHrZ(wliirzAboSj z8OIl-Pdej^F%`anI%2%ln7eC?C519_a>pb`h;m4wl^9q$1TC!nCX_YBZBcM(;H+{ z6RD0mIJLH_+|2k`w=`@oqgC9%yCt>SSwK>`8@RXD@7KJAMTC7diUiVNFfw_`%z1qs z7bHkdBhR%h*6nx&v%(yLh<0%-HDBY4QC}H!DHf=MMbo%P7KV+oPb05`f2y^dVV@R- zFi6303qdE?mOCHy{9iqG}cMq9eN=3Oi-x z`{Q^67sWvQYImDpu%GO9#4(5-zAYg|`#)I;sb zm&|t%2=R{ua%|;lH|MPgW1EZZNM67r*x_O)c7gqM@hIA8(*&snE2@y7ey#i7W%u?a z+kI=@_aVyA>>|e$W9ytH&PwMzq^~H1(7`d{D`V)S91jf8g>J5h8wWTxI)8)E%~j5u z)^%ow*KrQ7c5YeM`5+F}oO@YkNYvTc^({i{*S$5csMRen3EBnRC-~hJAZUma{;zQa z{~`|0Y%R{8%%!rPLRaS(P~_`fKSBN&O89!$kC2DEJLjPIOA3)pH-}z1{WCTs<-6EC zi+DOI&p}@JsI-&^ZqP(>SDQJkYT%N+BM0t|9^0hF((odmxZ*$*Kx!h+Y(WtydNsNx8P82@1KjbA4(K2T^Oe!zAVuPBW*%CZ3@b;%|^1QX#0>i)}bd zN%mJ2Q*q)hlHXP%96p7johfB8T0WKsv#X?PI2>6-ywE>9vc$z vIW!5tbJ&>J6v!HMv)ihAV_~KJ(A}Z_5V8AL2jXmf?Q@=WGz8 literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/packages/TenantPackageUpdateReqVO.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/packages/TenantPackageUpdateReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..3eadcd3ce9601d39527350a06f96dc77924c096e GIT binary patch literal 2063 zcmbtV-%lJ>6#nl1V3uW}6bdaBEUmRJyU=MXTFb8rEmiC;AkbC&3yX%5aSNoIYxrmaDmjWjX>q3ByvI zx3z3mw^eP|qmCAgq?OTg)3#O6ZFh-zf{R+Ns0&0kKmX{r&z85py|a0HT_9xZH;aa? zXZ++Q1Qh+2mM`RZFmU$Fxqg9DWk|PFh8vl3VKvR-uB{oS>!_2KJ6X)-SU}0^j-zE6 zd8E3;!~5_5`p$O`?{92>xgpS1%Vut=P|K)p$S@6eM8F^INfE;(vJg0sFid^2n4j0} zIgKI+v?Q#wmP={2A=j17fV*H&m61f+%ByzKS~65;$#M0(nz8bPt1enJ<5sh9kI!R0Jcq zNQry9jZAsxHR{|WaQ;7o9n%~=HN|(o3&?8-UYb@Y=qB5suey^eS5)1XH4LbA&H2AO zud3r}&*M11TCMW!!zU%@*v{3HoDb7;#6bs+UdI#YbkN0ZO5~uMQ9vUE19!F*`gn01 zPtvJ!7Mz2zb%?KN`0zBxy7FcOr)WpK19%2euJ|=WjAL!^J^0r41h?)R97k^@cz~sR zvQ#Yo4Sf5TZud@9N=sOhx-_j`d8yWz@O}ml@KT| zWASwaJ*$Me$zBcvfxe&7&=L3v2Uig3>j;#flura!nb2RQL}QsuzgK7nPLlGSA&fS* z>shMfBdB9I$2~uPe>(6itpF|Xt`sHoJDLx<^(#a!aF&7uB?Aed z`YJpCo*uy+06~_O0EB!ml{6nRF2Lv$&4WX{)X*>$sGIC;9$X;VP;-0*N?!@}Zw`0v zPcr63S8Di4XaIxE9l}uz(ekmO+Av%p33ih25nTaN6N0Sx>-32Hch^4tT=8pe)`kTwCA(A;v6b_p$e8>^|Rv7!~2KD<~KY_$9;8km2K J|9lmf{sP?y&LjW; literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/report/ReportDataRequestVo.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/report/ReportDataRequestVo.class new file mode 100644 index 0000000000000000000000000000000000000000..522532ce519950b0124154363581ff2de34aa61b GIT binary patch literal 2532 zcmbVNZBrXn6n>WM8_9;!v{0gGQK>?JrmfW009D)4iUo>5(blTnh8@@%vY2eD^*{6v z_ygn%A9Tjj8J%(TOP%pIS?hE5Zpem4$8jce?z!ha=iGCibN1$szkd1!z$Dx>`tiPv zxo(*FK*Ti>^A;A;To*-L7tE4qABtF(?TUrfG_vA)L)JALAIbW$jhi-ZiTa6!+ZI06 zkgj>#K`mT;uW3FJq((;9H0YOtb^08d_p9DwW9zP0TXpYNqI-g}TUm2!zU*V89&Y+|4d>>|!B(-> z2p;&w`h$AtZ57vpE!VFW?*~PdY;iftnfJTP-j@xp9O^I8bY!$sJqcJ; zQmzS07L{QvsZT}9*4rcZjXWP%3Qpjn&_`liE3%+6jMj>6vI4il!rOvgzkAOshf|}s zG-yG4W8xzXMi6d#HS#OO4JZ598=bV7+CZw6ZJ@}MjUu)B*`pvWLv|%}%U>+G+fmOr zha4kLK-sO%N)=Q$-TJ0#J7-{yNiHqVEzGhC>SoQWH!2|+XiIH>7v<1B+`7&V12u-6 z!j+&=D|=UbnV*3U18`DC*TGA8nU|e%Fo?W^0TD0YMGa^F*Hk%}M$y4APB=Iz+Ut13 z!H8^Uz&Ss5@O!z~@GI-ml;a&z9BK7v#=&R!oD(xX6dPH5WV}Oq#)q05%&?eI2eUZA zZ?AQkngI<*BabZK>Fc;}j)*;&){7`V$=9fiscU z_t4!$q)P@N>5`c@1YvQ2 z>3xV(IL$Kp@HNhG&bl~dKHj7)MViX|7e38-yoxJ0OU=Ln-o;zgOjOvebJQ%sKF`(4$q{`5IU=)jV)}r1ULZ4)x9O~q7G)If`&zvQD zO`anaMmtGAlioLar>6Cpi1NYSsZA1Y_m1trn%IMLN9^g6E(du%KCz_@>L^n(q9b-7 zaOX#U0;d^*l}fjO@+F=*b=0 qL$6TCC3jfq={@9fN0gos`%#KL^c{Q*RL9gCHW^c+agL;~SN{e)rPJ^L literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/report/ReportDataResponseVo.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/report/ReportDataResponseVo.class new file mode 100644 index 0000000000000000000000000000000000000000..227ff57a8574a468d2fc07d1f3ea65318f2ab2bf GIT binary patch literal 2726 zcmb7EZF3V<6n-|@&2G{ylt6(fVwI|G)3BlyF{LOkEiJYM+foG)*(OW7CE2an4TArH z&(1i`_(}P~2c6L}qce_v!5M!Dze2?4?5548#u?0H?w)(jb6)Oq&b@#B{qwH?#!$-aPW6H5ke$k&!-#X!YC)qo?rn>to? ztSLyjRRw*s>-KHis@dMEHS2q;;|f&It~(0y3Cm*Ga3fo;(s1n1S#kx@@@vrW<;Cq(64YDq9Kjk;- zcF0)zh8+^>DcF0LztY@XsahSFo-Wtx<$&>yMDZF;|E_B>a=kOuVz|FOMYlLL4T_;I*s#DZ}xSVx4Y9zz}ZBDwxc9lqy=$`y+b=Q$bofQ;~f1-dn<(? zdWlFT*^1N7A*hV=S<|kOjE*oDSJs_MI6iz;f#P?LSj$8T{BX@_uo5A2P-fJKG?AE% zl<|P(Vs*UP9T_^NGd3qxy^A4dZqD9_`o+7%tHAIT+nbbXDX!VUTHJE>e}=_eT9}!e zWVMtt4JT;ULXJjMiZ&Q$?ioy`FhW1(igUH-Hyaga%9VL3bZhugxqBvF!;pysIB24P z=S)10Lkf<|U5RJ=|BR4{X^fjV3d=-Uq}OrG#2a|i#0)MdIMBZBahYeDZmlYGAM94( zkXxNF;bL8uTkOf`fE;>)Q!qNvJ~sEtu!#?FjE_P)L)=mY`y!R=TZ;~_OU?ZvzCsy$ zW5e;--%__pOAKd6rMx5^_^pxpPdr}YC~cR2TUb!H>GIHz|y@9OHz0qEka?sU#~bsoc8E$1>BO35Kdia#OM{m}bkMC;H( zh%hc(9mm_<;-sH7j&w}ooWMKX;;7FWC+@f+_aly@oM6pap6gz5-$nYtDix=rsroWul=(T_Vg#W_oHT3x(Lo%mMp z9wim$F@@8VG|W+RhEfJK&iGkMIvTtL=O`JNWX>W?-S`W;bWAn7Y z$k2RmTHTPY$aH8zpH!dvSM~95Xxr6mtXZm8Zv$ktE@B@@IGc)j)qGl zY?cM(sF`P}X&m8edXbXK(htxsqzY}HBlDCBm9!}ese+@GWqVt+~GP)l}Bmp7noNbqBp;56T5}NP4wmW#B!FOd_I{P R2FW*UtVNANYNW4={{Zm44LSe- literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/report/ReportDownLoadVo.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/report/ReportDownLoadVo.class new file mode 100644 index 0000000000000000000000000000000000000000..126776257be3a390324641c5ef92296f7cde7e54 GIT binary patch literal 3377 zcmb7F+jA3T6#s3qo82TES_%cCRf|;Zm2eXzrCRP3+k$PWf;YD9(r!t1L$U#iczJSs za>gfTobgp=93GUBGNUt&zTk|1lZd~wyJ?aV8QMv|?|kR{&iT&u+rR()^>+XV@I?w) zSO(^j&@nH|f-JTyS9C0-{ zH+0<8v7#VVwo6{wpIIr{3OdG@t?O2r|Y%f_>C=J>T*j&#fpJ8lU%yxhq!1x63z+h1{|=f3@PdIcp7+JInJauUxcz z+R`dZR>{`!3CU?86_{hLYnP7~tV+eE#r+$x*b~^G-Nf|F)Y#-{1$xQy>5HqtSmEE$ z#Lxnn1>1FO*MBB|&z1aRo>#Cfmv+^esS77*Yk{c;U-$tA0@6!s7sYK)$rpBkHBD=MTDWxG-> zu$|LQ{;r(@$C5pDAL2d1{k|6tJ!gK}tCr{OQ;xJ`=lY)7Ew6-$9QN?OJthXxW#V~R zUciechOk?~J{kOtwW^63oHTI|hfJIh>J1z=k(c8zj+nRr)^_{)7wTBmDJ)3My4Jhd zTd&`UiQD*;c^&MIn9M=I{ES%|?5?F^gozyNb{4oReI`D_Ar5o>#;{KmYzMh%&Lyh73|%>dl$Y=ESFU(*)FR-)M`SW7@8g@5^_$thEtC`Ml{dG%c?54p%6SGU`APH-+=p7vS{QUH z(B2jzqM#)qQaj1jTNrI+M|)N~VeJ(hZS7A(6`$^pG>7=iq&Xv!qo1U&Y&8HB7+9 zIH?2*)SMutqs%UtBxT?m&iFY}Nqoy5IZrBucj(7KwCTn_*ra3X5w_`==Fds!0hXxW?&hY}DF}5Xz7wI7J@%(2VjW^>k{N+J}#*BSG1^+J~1| zG^2gs4)mc_Y`QG`bY2xfR*fWvlrj{Vc`XpJih+`m)oP+lJ`so#Ya-T`T3xNnaVGR7 zDn}eeyo>kfsm6p~Bo$)P9$WLKd6D%`Md3Z;EL)*m% zXz$p(iVi6aw~j3#onFP(j?R!ag}*JNGs53adKEj?KYn3Xgb)5`l;apa4tnP@8(lsN M3SP!7>>ADf2dzM+5&!@I literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/task/BatchSettingResqVo.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/task/BatchSettingResqVo.class new file mode 100644 index 0000000000000000000000000000000000000000..099617e453a9c8d2f316698b8262171d16ce36bf GIT binary patch literal 6018 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 HNp$?b>OZpQ literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/task/PointResqVo.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/task/PointResqVo.class new file mode 100644 index 0000000000000000000000000000000000000000..81b9e25d1483e80f78821822728651ab22b58abd GIT binary patch literal 6337 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 literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/task/SettingUserInfoResqVo.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/task/SettingUserInfoResqVo.class new file mode 100644 index 0000000000000000000000000000000000000000..f67f5a5c6fecc2c12250ecd506c5f16e0b3e171b GIT binary patch literal 5044 zcmbVQ+ix6a75|OBm)Z5~)v@Dk(_+(w`kF1hP&!TmX_HceorXAdNx5}wPwYw7yUy;c zi=ohz7Ftr;CX|v=XiHy+gd%QA9h6iNeJK*pJi!xEZPdSj7ee`+nc3TT99fmTXU?4S zJLmh(?VC^j+yAay18_Iqiemsz#t_5zgnVDfQ$kJ)d0NO1ggg_)vvEFuDC9?C^qg?d z3;D5-GeTYv@)IF13go3IUXEi2o>XyGp67)8RK+VQex~A86+c&TUd3xFej%FIqxhwQ zh;39)PRuJvPo6MVj6%^UEfglprG;??$!cwO$+91*nw5#t@v?%Hi>cXGu`q4ggv6$; zg_2>{DyD*gcI_U&!+pf(j?_xFwPem%Rcp3r?k|l#5M`~HmDT2+wT(;H zS`Z1NHs2bOw*I@fH-7!+x&y6hmgZZH&pi9#>8I=3Sk*Qv_EGM?kuN6~2MpWfrm`Pg zy}Ewp?e%w0uU~j!QZE4o595t9H`kpM$8O0f+V#&KF4B3ko6L;_ATzRQbsg+l)LUpxjn@ffH@{(bd z3M=J;)K`jSTcw41U`NdAshP5rZ^5*WdJT{r9P(F8L56^7Ul(NRz?MF^%`XeAemA`x z*i@l3rTg2fl9FZHD4_kKB9=1rDVb=p`MlJs11U2H0n5Sw+kzLW#8uY`nC(Jo} zeCRO+3I{pck`GVCnJU|hW<^1lbXC57E@+Ol@!IVu{RhJi7vvZBIN;4Q+3UhLE3%G( zZ$!IYv`?CGH`11^thbFeVbTRQlbG^QO^N0}Xyp!S!UZ(nS%JBeQ^vA0c08J$lT)2D zN(Uv^3e-iTy68?W$9RINJbYwg>L7DXK2kBOwW7_FppnE>-k+&VeseOCd0)theEaDX z&sUom&f;61OjD~&qyoN`F`vUfBctFSjAqRy#h%ah* zNJtS&3U)TXQ*J!>*R0~aH0~{dfegBxyjR1o@P?2#Ir^jh?G@O|DIM+i*MQRWXn$K3 zC`!?|rNb!ED5+P1ukq1-uNHh!hBSN|tMt^|w>xrd4P$wkNCji9v+7l;^)zk^l=W-F z*JrzBrrYtn8^rhnqUexXlIxIil78CZZ^s~?UEnvEzu_uIsA8$N28>a~HrN_apo%TB zHQ+N;v3Lv_=C?l8)(elbHW7125JR5#G#ZM>v4Yk>i%#ngAqTObTaI$Wv#4Y(Dat{Zu5ZdkvGKlRf zjQUnM9K1q!yDP{fx393v5h*oXNiKN z!G6^3`f-mQMLtLFemWlF`Wc=c10%z~K{#}A}X$%O?Ic^}yeNaVt~NDC;bM+r*mYAz}u)dNLy zY6~c(#|TR4@mx$maSs&B#alpWy^El*a@vHYog{NxHF>z6Ip*RR_b7)JFG8-Hjz7^ZPV4v5<9kATofh1c z9$#bws_Bsnh>oqH`*D#c^pGfWAy1=9Te_;G9(GjhNXJOch3l%69&uEWhKhaaTBY@< zqhgDCR?)gDqpOZeZK&9|u2oi#IV$$7XBBf)yMmkNxEq|8f=*mR31x;JW@(qnN${6( zifQTOdHyCFLz-tFo&#O{-v?XBZ1IY}#(A z_RQm3m^VG#2tg+-A1}mi|KcTyIh{9P+#^Im(K`)tIKDF{{;Cp^!qOst#`zEuf-I6$C<_N{vY=2?BxIe literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantBaseVO.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantBaseVO.class new file mode 100644 index 0000000000000000000000000000000000000000..18ff1f907937a84f594ec1b635ca73b711ffe9e8 GIT binary patch literal 5814 zcmb_gX>%0U6}_$5dS-f7A;Fj>GRQ<58f0S$Ev;Vcj50`y7qMY=4~ z6_Lh8y2`$50eVj$Z(bQy1xohqRCX)roRS|-Z`DfK{O}5aQd=r{Ejy}i&z7@;IdyeD zU(ggSTgaCMYVXSy(&atM@UU7+E47vM>SDIPFr?--mkLF-r0qi--)<#W!6w0}3+E=E zyE<{=m_TAleY%n@sY4!ANf0QY?o~#MIiUQvj~u;o;01ve>u1`V20J^X@0-qGZ>^+c z^IADQP|yY{xg6pLM%8jz83wnhy2Gt&$8H~d|JJqZQ?FkaXudO(ZTpJOY{CevDH&}* zFGle8ffuLVy>aV0n0%Vshd)5*s?!-dVp97HgIPYlWZ)(bADui6ez@o1J>_k;yu2l5Bn{$$DBI#?8NCy z6KD3{IdirJ>n)S7+?=?19FMK2WOkvv8;1mn+s@rr&{dK&YyR5UomWpx-F$iC_*KZN zXX%5>1X@tP=B!>w-CKkTwn1gp;1w;KORvM(xgxheI5c_s+~nB!ow3UT&9P6Nkt8LP zDL}7ld0^t@^Ek=mOPB5(LJo86Qzy~CGMmq8j|${zYu}EOuSF>ZiuGml>Of@_W!$Df z{Cq_U86~$}DP?(XI=$LR7LRjjU#2jcE>#Nqvgz`^vZjuvjpvr$T}W$cUde0eZTim| zrL1n>#G)NmO}Wi#YqwRkdi$aPthLKBUUOJt3&OS4lvI-=OifgbS!am|_U78tOgbp$ z^Nawq-u6_gAp3KP0kqmt@Icwhr>U(z9jQe1tc05FajbkeP!OPMMzC!e zHQG6PLQIGNWKYRL)iPpK3@B$#e)K3Mhv(}^Y180NHKVO)-+>!iaJEi!2`#Unji@CY zmG#kT^w^N@GbeF^Xg1|u05YicZ3dx^1VO`|%~cFSYN zy4|Bg9GBI=*j))7v4z=2{3%ATBwvVYOmhu6ZRY3&F@c%HqmB@9-ze%J;cMOc#vs@M31tymexu1Ioc>uAE5#9B zfOhREw6RQANpziV;IekL*owLe&AI9@$_iJotHmg+PP(Z(jQV0x>T0Q07}H2sORdJ3 z$GTeVRfb1sm*_=$0z1`?YuqP1a-~>Q^LYO(vEOgARn~V;Ge4BrfwwMhU3;?dbU!|) zTJb?5QUgCljGx480e;)aL+xeipw1oCMN7)mO=&{oFH;Xa2uy^8Myk0Q`1#ksNUA>g%4+acR`8Ye!B>I^F`D!ge$C*Y;ozU8 z)kaopY|N^wffZc8@%rko@+qK{sKK)iWOh2N$M( zOEzk2WL$3-vUfW9K36i@f(zLaPu!6W}u(*cY<9KEohnok- zQQhNsXBx-oA_9G#h7pbR>&VMD(>UC3_&g)-=kd=p4mTc-lck*)>&_FHX&i1t9A_8h z+~SBcjl=zj$o?~>n9IOY9_5*J8XlJbsI#6m=6A0ScLpYk!r{}ClGP$cC``E8(> z8~`XLi>UyEL=zN9i8fGN4gwUHL#ZHxLMAAf3fVvjISf!jmQrB`NhS!E4V084043#U zD#Do1j=KZUfDd69CPUlc@xQk|rpTN?IWAMJU%(*QGx+ zDs0_9p_hvEH0s|Z^iT<%yMbDTg;a*mLmk3A(qMUM1Kp(xEFbmLKWH~BKUDM=+5;;9 z-XCc%ERjyq71{?YNMrOCeFIjA#_0ro6IPgR(h>R=EQ$UK6@D95g#M2E`W;wNyj^pqU!fJp}{y@*8VjfyV@6j<>UVL+%r60iZ;fwts9f#$IAV(l5mfdlP@gyvf z-oOXU4`BuAw=|nxfE8ky2=7q*E*xISL=Yei{HNJnY*3pwu%n557Qn)Ioh}pRx4~-2 z>0T2YaDr=imj#m|)=_FK_cfCkbR*Wd_gpsVp?lD45&jdx3=+O~kXoGo0zxQg0)NmL z_k~VD7yQ*(yg|)RUZvT|IpdV#YlENUeTI$JnM%$z>=@hg3>)2YesY0fC)oc9!$yOg kpKLMgMz-%a>{)Ctggs7+rhoX@KX8kth6Q>_?*lLY8{{VBMF0Q* literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantCreateReqVO.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantCreateReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..ad3dd49e7eaaab1f02c2280c6812b9c197c34159 GIT binary patch literal 2940 zcmbVO+fx%)82@duAqfOU6g1u%l`6NmC}=B)7H<{HO+Z@lQdii9)nqsBZdB}ark!>i zU#vdpbf$f1`vPr;R-Lin;MmKHo#{W(hXS$xL#_RtY=9sc>vRa`eCIpg<@eoA^7}uN zKLKdKg+i3!WDqh=iRH8(?-p`DBiO2-&K6?8r3M906GzRD_X-ih$pB(v9}+iu0OtZY zA3$8NbUzXj0x4Uw3^k!iC~Z5ZUR0I1YQ&Tt$I^{hlZ0@0%5e0A)~DP0*|^qX7^b5- zx@p)Fs@il@u`j8ym}V*J{EgC*)Z5KLE#76BNzHPGX~%a_ji=}&l)m-N%(aOp_eZ90 zOwhfhT}bJcHt1!nk_5kYSxqG46!Z(u)Vwe@u$RM1YYi}bB~k(t|fr@#N~&ksN1I&nJ{nQJ#CYvY{2j9_CdTd0(x`j3B9ZoK+Qy{6@ z_9fFAWK`34uBN~Gh{U$-^$k0AzD#}(CX$&Fmb$tN^I6+wS}|ovKP$S((PqnJS_`Vw zrWrA3s7ZuX!t%I=as%47Gcxt)I$`bB4c*zpEL2wYNs#xmjU+5>(+#a7m7tSem0c)$ z)f`aceX6C4bCxMPLps}XN85mzP^^?WtSk1g?Pv*Q&`hvhl#8a~XohMyN^fS{Z)wEb ztzGErWDSa9nsaP!8dg+R<(mf~)HRpRB0E>5ta7o2h(bus>lCo#)i`l35br#DP8)EV zs`@2J?E2hU%wBOMOlL^5BovD{E9M7xNrs=pa0TY8o)ws96!$pl!1;DHnaLbq6h90O zsrHbYd{&~BzPq|x+Yb^PX^w`SiaQd5IX;6G^X3DN={_J53VY0yHJ}~R zMb*j|rG2Y-awKf{ua|`7*_{62>-5d5(>FhzzHv7ajig7vmBo}D$oJY?iN|*()jb^hTlsOP}*ye)Rj`|$8 z4Z`h?jN6Uy(Dtkp$vkpF+F;jkeVsrR?h=QBtk5M{`;;=7VFe@aF(W^J7X3H{} zJM5yFi`~LO93s{-oWh%|qv*8*I82_GzqSu cl;#XiLpDYa-pLf=c;<~OqOX4b^Nl<4FEOC9>i_@% literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantExcelVO.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantExcelVO.class new file mode 100644 index 0000000000000000000000000000000000000000..89565b1187d94dd32493a9432781a357d30a8dca GIT binary patch literal 4273 zcmb7G>yH~(75|Ob;~9I$Uayl)>NYKDvXp(+6w0HsNobo#yRe&tBrNFz(pm4YJE=Xh zwZ}`tt4$w}rjG<#H3%vpgw%YA1c=s=k_y3>BJq(=NPI#fyYN4xl;52@V~?Gwv`D*m z&OPUM&YXMB@7(cU|9kC^0Pe?IIot(53mq-7tcm5jVtH09XTd>ua)&!33poQ_v?oY(QHj@NX&uHy}%zM$cvf|RqQ zpg4EhTC*xO%U!O_dG7Lzg7UGJ>pOLO!D%{+HTxmg^?b{BJh!P}YOdKL&nn0=%o=mG8w&0Tj@D>-XPgQTS@&E%0&C@fQ}y>t zncic!0%Ii**h5~UZux``H_tRr$zZ3VlycREZTa>IR=89CmhaRm2dJfwapT-CZ(P5& zeyRPROYbq)J&x=6`xFd}jW4it`@JRl4b3^OeYjO$v>PWZHc1rIt6H@MtKo=!s7(7S z4y$lqEMsQm+x5zlSGOFuvgTEM+qGQ3aw2#H4O}?FG}L9=pCv&`W8<+Jh7{3glug;% zWA1dZKmaS&ZCUh|vk)AN!PUVGLtX`dL;uTYyGSw|({h5l2X|&2*N8lF(VJ{n(Mgu7DA{fCI8m-5P+s`OvW*hzCM!@G`lRhVr3|XL1L;rYgv7I-dYWa%BwE8k@TwN9u&pl!B(v1isU+pKg+WpJvKXk zkj0^#Y1qwH&F4%6Cs#-1L5})jkwlDy5#8 zxK~`12EK?d8Tb@-8u&EsF)${U39(FJ+Q8>=zXEZ6ROCgnscAD{vYM;=U zwcq*9L}yMyf^(+d`R+t#PEdlg*6%!;===yC4V_K)CM9|^G5J1H$-UYp5n-=HA0!+y%=J@Zca|T3pIUZ^3TwJ0d?TYFK{1XNYIdJlm`XK2tkx$prV;4sAv|-c>yYfpnSOyfznra{()O=l%LDG zbXa;5$8myH8^H#?!CNN93-s?;pl*QI;otC0N@+ZT_i>Vv%AWrgk5S5?hKqQdl7-%kqZcPHLz`}6_$i?`%(O6+)1gHdl#-^rsRkypDR9!u zYGf*ynZT6kn8-2dHDqdmiS!D+w8&I6^}wWeOe7t>b|h2aNqK_soCEj`R^ae_Dt?XA zlu~TkJ8Yz&z;3+AMhgn;BdShGW&7vhQp&*R_s*lF;R3c}m69&nsO*CAF%@aHQBl}B zD9XnS*4?LsVgzagh%o71RS3((U|nRtHUw+E;4VJF5W48MxOBaeZJ}c}$+7GEiDtU9 z-uyKsS%nu^bMZ^yI?El=we_Wgld7yBN#Z)x^2M2mbQ2+n{ literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantExportReqVO.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantExportReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..4aee75385de2a1f2abf79ed52b53763bdd8c559a GIT binary patch literal 4025 zcmbVP`*R!B75>(iv@3b-D0b|i7N{sL_=)U>gho!FAvid#9jA%w7U+Yuw6Qmlw94)( zv7xjvrQ}6OAb~KHX{Xbfrqc-}ou*F4$@IYwbf$j?vcpgO1Euu4yDM3;wr!>!&)sv+ z`Oewzp2uDLAOBkZGk`~NIfnaTMj@jpmZM^^#A1uZkx`2Ac}y(F1$16W3u1XeEH8@X zM=`vF9}DYGgmgm2Vhrst6}&8-KUMIGf}bgPRqReGI3-Z06`T?Kvoc1d8&7HkQ<R!*N?d5^f3;jP<-hPQ1W z-ZC^qFPm&j+;Q;$>H+id#jgz`|0h| z5~N`Xv5ci_jy}aiNZ9zC7pvqLxztlsGtjni)0$1$MUk~b7Retsts|*Jrj^&6n$);( z5oM2Sp`%>X97nebWVpbe!HJ2%Y&JPImd@wXHhHV%Q$H`?T3)%l^vUJxJkJiJU^tIU zXzuBqX61G>%@Q_@8wGt|DL16jH~`l%p54f|HtZAEWa{({c{zN2ezR)mh!I%ch>*nsiosrHz2ywe^dy zDJs>9QgzU;NoQS{wW_oVx0-}^^z;U#T(&E8I(r(~i0rfxQO{>hQRPEbg-=&)oC+Ng zs8#sY5Zct!z(>F$`q7e>V}|PpP0k$FGtNlwvuq`^e#(3kr^9sSbc@p=D%9;a1OIAkT z;|_UOLqk3yE-Oy(|GIyHz}eQ5ok{ zypHoKeu(E)T)-O=Hm|?OpDY=$1cyD}Uj#X6R z{Z}9lE+PKB(6^*R!jK607Dd$7GpXsYYhqblCnXWCm|D{j*A%ImSZsO;C4;g)8@UNC z&X`Bcr{9bhpkaoz!u=oJPt>HJ X@+KH=UmdP~eUFd~KYlsnZ4~|uvZ80n literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantPageReqVO.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantPageReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..c96dc77e75f82af88a5faf77bb565a81c52cdff0 GIT binary patch literal 4050 zcmb7G{c{v|8UJi@ce{7l%Nt3`wY44?OI~}4v{)|CN>eas5+cNC!Iy3>OLB0xd+hE7 zVp}C@X#m@z#e#*=VVn`iL8p#w!a&EDU+Rq0Kf@&OQ~v>L{d{-#axYm&%p~7^zR&Y{ zo_*do|M>TvKLdCY*JId@Q&GsU#9@oW5l2xR^D<7y_&g&}3xfJy3@>@q%i{WqIKD5A zAH;ALivssU0pW2LKa%m|7}{`3!K>nVPKaMqu%zI;f(r^R3hqx7ye`mhD0oxcUz2f3 zLdekax`fuz6WY9%%4x=I>WE`yjoBdyiNi(1$>#N`teu_C>H7@BbTlVx8n%S)(X5%W zPiwQYx|Pz(lGMIJcFdg7a}$`}*Khu6ZE0oo((4jr{fw3`Tm%40^c0pZ>A2{+XHN$VfV$PuuicU0u4idiT!SmF4?a z-eWv_vPRZTZ>AjG&{ok<5)w zX(V8{vU<&BMyXa&vX1x_X|FO|Rit}MS9d_dWxM1~d)H=fL~a_1sP?DK*zkv} zl0O}lc1rFDLvf*3;ZR%GCVvDhqMs^iIp(*@p+nOr^o%pqeUy!4R?nC(;%u1CoNjS8 zM14BFVT`#v|I+LVsz#&5lUMOtD`BM&w^RX3+y)m~8V=RNl&*26By)01E4Y2by(HtO z5|oT)>=z-DoYU+%ul3l62N}V{;e%uQ86xS3rQ5}vL#FS&+F9jgW9eG*MeZ!USaD3R zoj7qv%%YXi54a=VR@078h(*Qe{a^csZTtB8A8xE)I=6c1jny~bP4*}K^|N&D{*CvO z?Cqj%JISd-Du%E}#plqeqE{SGh+`06Q1KLYNf`XUM@Gdoj;J^+j^{AU6smY0!z%WR z;{d*_Vgv_e{7l7LxU3?B85KXr+Y%nCUBk~6v$+}3ARRS%=(3C4uX|No!BrJG40BBT zlA?k3vj6*%6?rG4d4u6)o~&qJ(#yST;}`68Vm4Zx%y)2&Rqd9rqnrRzdDBSc7pD0k zuu}!|gqaeXVnXBM<7>+9dmUb4x%+aTWoZi>lGu!{7ul9F-E@Ejw~I_tNciyGRHGOIb5s!SfDb;IzTy6 ztGqo>`78LUubgIj>qL5cdf!J~aJoA@gr9Cv3a^tSj0W4Oe^5JK&q){ywi9|#J6?B6 z7{{TYxWW%=$LnJW6Zl$?oyddQ@j6}-d)c!iU1bg%nCFjrdwzp(=vMFFAislf{jJ`= zLLTeuS%%_XdwN$8^{8dU#`}1rL;Zh8Ln8DcT5dw^PlT4yP(rBbFd=F>k_Zb-v`kPutt4atQhbn{P%1!;=_o;s=~yBvKrtT_O~fidO@uVvMzj0+&@IMO zUoo7`Zm$)*I5)cGBTVA!%tAX>FvWotyX2Sn26^?|4p;F_QX%I1CA>f?Oi2+(NkuSE z{4r88US>tUMM}Xn?wW6tisA-0)^|w7@Ll@xG+JckUucx^V)PSq$k6z!NS|PdJ~PMeqV0thw6}IF;~^GnD<7>1L9=Fpe%PZ`K|kWraX~-o(T&UaOzUGF o&9r}3h>|>(v8{F&w^Z!nV}APUkaye6aKFn9bjjo&KZ}L`0H4=rrvLx| literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantRespVO.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantRespVO.class new file mode 100644 index 0000000000000000000000000000000000000000..07d404acfa688694420525f214aee2644758614f GIT binary patch literal 2384 zcmb7FU2_vv7=BK&o3C!51PVk@Q$=l?mIbUIv=rr2is^@K4N?W!COhq7x*K;lD1X4= zO1P34y>Ps-H;jWbn9hLX=nc;JC%h^h|ASbcvzs5MW>&w0;#p7VY@?|aVv@z4EV z01Tp>!fuQupwZ@#XS($X5s_FXIX-BA+Wt*+2(j{an5vS3}zvaDL5sgQZ!skt{Y0lU$B_g zp+d=7)ZLnM-O|15o^LMdWoOZ_Z2g*}`=)K!zCII7Q>IrfPRQ)$O@EXGW_o&`q%h6R zQ!QC+fLilVz0ktWp3Mei9%=5jnpj6QEl5<1TpAT-(A8>WV(zM0@`rn81%x9rZ`Qc6 z#rPe6!E{N#WZ-~og+!nU#O;Bytq$R4t13P18>LUijB3Co8#In9v{EwcGcqWO3x>B4 zmV>n!rO(N!(XlfOigL^~y;{X**#eIp2U<-hd}rEsEqh*IR~x^fmh`8lotj%R&swsy zJ6daVP$nzT{a>9*Bi*Y%e7Ab*i z1P(r3qiA>wgBo6yLmvHPNW%aQYIqwb1okxR5+-)CW>w0vDmz;<>hWs4QzIG-%xQQB z=g4G#&MF6*5?iG|*CcO*^!7eQoF@&>EKcyEHDd{Tge5YnRnul?474^^qXey8-xfT1 zZNslOV@ngZ<4LT-cM6E`k4%G45LaECpFtG+JnYA_vv>~Ad&m*OIy4}i0VEX9$w)X3zQmqC~F}Az3Agg$RcrUKYS07yPFPYwmi(rZgDue z`Eb{khjkok93BdoV4QvVpAgwP?%l-5Ov1RO6f!4p7)Q3U6aBw-LVAVl*@m4p`ij$= zoA3J`s&Xg)8mcP0y++ZjSvp|uTo}zp?k--#>x{Jvm+%Jb7-1Ps z<4x+K%;7kWQsSqRjWa|^#RcrcFr^rd(SA2l@x2o7Do>Ae4j*H8rbftOe0+cB@B#@- zbmo^3AE=|_iqt2E6uFUA8aoL|Be(W%P0wIxxZys^q)*}$-6lxWX-W~K*=lDPQk0Fp z2WKfM0Tt(=tntn%q(TU0&(HH3LCtkUD!QPq1J*H0yNWRS<3rU*_A-6SdpP(2+Qnt0 zGu!Ie&a68*Wp+rJ$=8v|>Fd7u%eDH8|(AT^C=c`co E7rjd3A^-pY literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantUpdateReqVO.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/tenant/TenantUpdateReqVO.class new file mode 100644 index 0000000000000000000000000000000000000000..44bf12259f54746698591390cc7f9f8b1c1f4eb1 GIT binary patch literal 2016 zcmbVN+in|G6kW%+aUCa3lO|0{O5IY5W5=0ZQlPmMdPyPFPD+v*=o1rr+=+?jD` zo{{># zR{9o)|8Yg}~mdVd~Sh%7Sjs zYZO7CD{B?Ba$d6yxo&KR+(m<`oXHlfifY%aWkYqA9apcYMXN%%>XN0px~ZA2I`2)_ zs$`_k>NoQmVM)x} z#cdohspT%}Hr12ZkI5d0dOM!kKIdwM_i|d*Bgke`FwYAG&773^XfA5bqOTDJn_{@D zvs1ZA=A#|4b*EN#sihy+wdP#gFiQgaTNwJg6d*Qd)$D?P*^mnN?vQ9$!WTI5Z%xNY z$G5)ua_i2A8+Sh0xP32?N%#%1`T70LyY~{*yQVvCB0m$yQ9KvNe)I^O{vSn)<0TBo zk&(k7QdB37mvL6$z_!vpy9+g=T$DQZ?Vy!(YJ4LTaa=<_j&pd0Dh?%#qPO!Fr5qF( z{;%LJXpWwr;X~dA;Fp@^(5hA4WPvlyJ<<@rx#L<8Q_Pn8|MXN-FSXvs5q^fc zB`ir@B4%?m;ymP?FyrLXsjm@T4pVBWx@1vb#<1A`)f(k^Y2Rg?VNjBP6|bY6 uNIN)n?UgnG*U{Ovk9HkhPd8RaQ)49?Favni6YRYAD%u$T0{{6cF8&Sj<+@}5 literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/user/UserSimpleRespVO.class b/ruoyi-system/target/classes/com/ruoyi/system/domain/vo/user/UserSimpleRespVO.class new file mode 100644 index 0000000000000000000000000000000000000000..7998f39175d697f03facffb8dd3aa421cd92272e GIT binary patch literal 2873 zcmb7FZF3V<6n<{9NjB-Wl+XZC)FP;D(y+W#!IUB|f>zTOTd0D7Y?HNJNU}9AynNFy zj-!JxqvJS!{oynXnKDnU{>hT%OS>JGSo}B%o+z&yN`eYNAbHPF6~m6R79! z-j@r@^S7AOUmt(+_XnTRPA2Etg=fslaWB6pQ! z<&AdM$WIw1OYS|TQkk{L!p=m&+Bipy%v@VHNcYwk|K%<(-q3%x1zY z&rOX>E*Q#KJ)xcz!PHZdAZhDGUv5OduCF!t{wX>oqTw{wjPZQdPz|p^(r)CL#tQiO z^f@zI8R|bNAncWddq)BRyE1E*c&nsgt4vh3kmhLX+fL`OQMxYoxS?&<i%hx`jBEG9JgKlXwD8mhluJWP3H_={jT!A*=;G zXoQhiuFr9&^DFobBz}OnMZt$Yu0cn+2K|(!0Rc!L$uDOrQe0b&eXYjPw#KYs8)JW~ zajdPej_satnsNGMoQdQD#CMdM;|@F4>s^Ewx6H||xAdW^*N(_!VZOWwFXjdFd%kvcyyS)?f`cm+o=L@9t#?8jb8LF7rpFr^Sm zZ0dcKRJ_DHlh8uyKj;c!|9z|v;Q;?saUUC9G2ATDHAjZaA}149CfJkA;Q7*Dns>S3%hTGl0wQh;5AAC zhqj9dKA=J&Z9E@?gOxSOeLrC75AyX1d(r47rnIWAG<* zoLGVu?Oa5cq;L^yqTQ~{4(p1pb>*;>qg)oTu5}o={pOCLL5~_CC%t!9V}>oEr}gbc z+D?)T#ysOw3{+vDJqa2+(+lF=vQ{RrA{D?-97FSjF@#tiR1h~Pua@-D}aIbP;@S>WX(UOr~*VhHUa+>vot z#wQ_s8p3A`T4IIIWvBw7Ta$|uvy&4!fxfJk-%J-u`5hx&+$q}nW?HvOo5l33zLhW7 zb30r5B%Ox^0=8kBxC^bQ?=I8IlEw3YxGhn>V<7X(=(=8EDD_R=r|%$3FnM8 zOSMY{;y7LXek3(UDs{R=Ms{2h*;u)0=;jJRZDUh+_j1p|M0S3XTusJ~;kk($jpT{lie~egYA&b+gLz$Y z$x=7TM*W&@bL6k@Ogra*W0T}LTh;E}R<~UB80k%w12vb=kZQrqs6~!(B6-v|0m)uB zC=jg#n>&bVB{1Z?a&p~B+gwFvn=4okavM7;EoZcW2qwKrjgY`~m#W6GbeP)Uh+!G_ zn1EFys#8ZVLCX8ET7BRkf3XhbvJi~AEv z^5&|CjZ0wUgaySd6v^dF1&x}8>$JJDi7#^FDtMizOrL^fXac<_YJUX-SW%EbQb8I+ zGIRy2SW_^94Q8`SaVozwbYN@zd^$ z2R~9r9&x}OCF!Z$PBD|WR*f~c1O*0;RlZ?p&cHfUm(_xNMKf`VtQkcI6!f z?dT9_cs0v7(lq5XGjAQ~o|)whU9%~?m%a6Hr05PRsw#hkYkFm;tRxWBzbL6Dwc;5s zr)w#BN1$D^((5HVpEuK5zM!W|wqf$bou?<=$acEjZT0~#M`m3uUQ>Gm~N;c5c#$`rl1z|25 z(oE5*o1XlOJcQVX?>YRtbQ0*NHR!x!{Q+EX^@qq7BO*~L^#pBcUMU34fO%OL*j1g}$8gYrDBr(5hdyw`G@V`V@hF^%*e~3fiP5RE4BQQu$ zzEOzmhMyo(eV;21d-*;gTy}CG@K#NHT@CzbP5hNZeA=}iNc{q(+IWix95^)oHm=fT zIelJmeKokf2KYRIe7Ht7hij5FRFiZZOG49zsH#0_tR^{EgGBIjssV1Q2KUqe`*>ow zG5C&(BShEgN%eO;#;MeWJt(QTw1@gsJg|p`R6Mwc##CJ1L(|WsS1zS;ao0QCV^r#d zcjL=DZZW?j<0j*B>T%kO3%7^^KiV8U9X%6`MOz-D^%3IIZzT6WL_GHhf?Y26W5hAhgcj;9pG$lLll0z) GoBsh?#!z(t literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysConfigMapper.class b/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysConfigMapper.class new file mode 100644 index 0000000000000000000000000000000000000000..7553bcc4b2064d45034b977659ff40c937b58dca GIT binary patch literal 786 zcmbVK%TB{E5L}ni&{CkK<@E!lT<`_Np$G|(5EVg4L~wwUo0#B|*dcb1@@pLU06q$_ zN$5jxiMV*RtMTmY?tFf|e*nM%Y+JBt!4`q0<^gxjxsp%9Cve%fT@}(eQK_Ih)rN=E zQy~+QI;k#DgS`?M0@(U5*gx%?NuUQoFct4w$z|L(VjR^U@ z&|(zuQz?~UMktB2JzK>-Wf8-jQ8B`@rfdeF1raWF_m88(yvNK@1r#ka+w#q5jJB*1 zAW%Wa0$5c73s42XxCk|@TUf2Y66SSi-(x~jE{!lDZHL;vY1CV1?1It22$x!b27!gP=t*V*#Rf-TJp+(2# zADLnvH(b4u(l^WqUupu?malr04VcsCRCMrKQEPvv=OO-kjNt%Jp#{X92Cs65DdD^aGMp@E;ZWNaDk3BBd}WgIzj&w6-oX7 z>WYdadZyD{$2iUL+ga;qK{8eXoPlB&|C^X}+*|S=b{{%t_!l)M+tv%?Y4O-#z!hyW z%@3pF=UKU+Y;8YK4u3CPEJrE%$6~Zy+tdU$Mu&N1hPS88a14Y)Tupyb9csfpnvfrf zh~1hFZJa)Vo%Ct)0slY$ECB*@xUS*XzBEr@9x`ZU(VBxC+Id*OQ*0ODI^M5a_eESw z8j9)h20Aap3IOgUC=Wqa(Y^_5(fxX|-%9okxQ%|}pm$QxO?-S8?nRI9C;J0<7}*ur k8Zvtnfh^i&5@uC=xoyK>W-Zsv!xP-I=(+>&2S)D(^b literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysDictDataMapper.class b/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysDictDataMapper.class new file mode 100644 index 0000000000000000000000000000000000000000..51886cdfa85df9bfbb58db6261e095eb01c4421c GIT binary patch literal 1516 zcmbtU+invv5FM8`X&OjNleUyQrI$oV_@xg>K}ZCIRA~{2pn}AcleL1YT|2V9LHlbw z@Bw@jV%AQw83EoJ|&xH}ah`(o=#oTiJQc7i+6-pWcM}g7<%7(1}l~d8fWkXpWKwTCxwl-o> zf)uTXs0q(jC7Yi6`pT7|i zs%5wE8ThJCGer>CpVj7!B#-|hNnZ#x*~-OaQe$~c$4$(gN?B3Tx<<&|85r`s+}c`q_RJuA_eg_xG~28jei^Tlw@R25mzf0Br+y toZW>zXK%s2v$x@nvv#9x9>Iab_81P)mN4W9;5T?<&A|Ww literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysDictTypeMapper.class b/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysDictTypeMapper.class new file mode 100644 index 0000000000000000000000000000000000000000..1ba177f8499278075006a835c81ca227fa9d1c22 GIT binary patch literal 1206 zcmbtUO-~y!5Pb%k5NK0KD20CBs-Q~sgUmb<=QnT0zPkSP8^AtZ*YTo`mvy{iXgQh4*nd#5|5Ri$QqD8H z??kalxy?Ss=N*clDyg9{q&?!4D&F%Vuko{B7!miL5{XJPquv97Q6 zO>Sd(q7qWDJ##Ti)_3FXGwgoM>E@*Ji*jm^$Rn+dCmBYQd2b?PM_gorisGtNC#nQq z646BORfFRS=Vdx6V(wIijcV#jD;VBZ+i!06zrCn2Ff7rxOz%cxg<%B`D5_DkggV8m wXiVcZir4WFAbf;oz{l7KxQSN4Hnsvjxxwvm90&Hyikl?&aPH&WldtcOPXJdqY-7KT1A#snr!ybem<*X$n+co`g2<*a%WXl@70&Ba z&TOj4$VuU9=4dz-IC47C(Xae~oENxWyGAfq?@H#L5?Q8>1UeInjq-V>1;YQNYK%l{20cbW^0Gj*|*hGu*Hlqz}aoj<-!nd(gVY}72haTT0 Hvk&|LxpuZ8 literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysMenuMapper.class b/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysMenuMapper.class new file mode 100644 index 0000000000000000000000000000000000000000..5dd74ac565ead393cef9525d9ee9bf5a55f58a06 GIT binary patch literal 4546 zcmb_gS#um$75-Y=jK?iY8pkqDgs~jkku5nLOcsJ92T8H9$ao!FmK`8m&0I@u^>mMV zaWWMbo1hSy4=td(Js``R>{8 zmH+hLTfYbJAl}a4$s9C%Do?STgTy5z&gc0Pil!1T=g}64)@BT%l7kH=hboq`a255k z@UsZi_!af@bQaI#up6EdYFS)W^mAD}tLW#mcuvtTWN|Hn=d<`?4m7N65drVP`Gs$oPG~B+`(ayzZ0+C+jQ8)7H)lL{Z(ndhS5zH!NobD-@wmcD3=A?U8Fu-gJ1*ase8R42rjt4vZ z1(SikFV%Lun&-=kQ4!Uubd$DsJd{+ZUAZ)KQp3U2j`f=+{OPWcTCMV}Evw2gGE-S>)aBs?|YawXpF8YEY64lV|3|?ZV zPHl`~n{BGpqO6a-F)2J5xng$XojO-^0=FcOnqf5J`y@iUUloFe9gRiVP>cK4sl^D0 zo;lQ1Lx~YgOSfi3UOlb&^KKZ{Ssh=-S6EJyj^lei2aijC!6|z^UHED*p6%&GM=!qC z&TG6={XoZ6d_9BbbojWY;~V%Umt=GVN_rXJqLkC&D(M)$oxyi>d{^DaFX{LmUSYz{ zOFwQrI=+t|XxPk=q=0y#QewdTz|-+VyrSaN9?>N?? z=^I{9tvar66psdWNi{MZKT_K}X{%M!@na?IODA3&{kXfP^GwH2)av0Tm#+Ol(DAC; za#wN>rSDg^-FnBHwk$a>tO<8MU{(7^o+-(yQdh&yX`2nzEboePd{@14!p5wk^%|RV zU8&oYETdXKGBUK4JxPi*=XnD_)W%qSW2SBThc&bh?Vrivb-bZ&-sbXI+fA*jdX!#! zpHXJ!cU$J7oW69*Wv8+EvU@(#=zXNjXsS2Ftvgbz;gy&;ZOyu(PMhV@|6QY%e642y zIrVORFB(0*o>xkaT{goPfDt!{Bv|Q3?_fQx)@ZvL`dG))m8uoBllb1^oxKpcSU`eK!Te4j;o$X!eSR(8OY$*bWg!?7K5q= zecHCA8|oox~>TPy8}6e#g#9b>0m{Rx=0|XFW&VE7 z8>{38Qu{y5`9rDsk5GOP5Amevyod1!&)wcez>o5K)WgT>>?jwHVH_YmjK@OyaZH4C zGNx1U_~SUjjjO?q*1c5d3yJ6xaSzZcH?jV_&e$N`M+KzfpOvWkulQ2;ry0m^xVK;`}ms} z7|!>G($)@2TZZ#}q14$qKJd2M1^g=_p;iOrFY&v{^YYm@@E6jp`MZ{}?FRl#zCXV` z1px_(jxf;=_&FFQZe*= zf^>v43DTqF6QqOW6Qn`%3DP$53DTx8Za3e>&I|buy@OqUpsnptk?Nwyfxa+8y5hB2 zoI#OeXZiIb`yY&Ca8B#)&JFw<)BN{OMyNB6+6>NfRU2k;fmC5#B>g2n^*<{=$D8;C ccHq;LM|j?6xbG=^76Nl9#nM89v=~YM12zUnEdT%j literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysNoticeMapper.class b/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysNoticeMapper.class new file mode 100644 index 0000000000000000000000000000000000000000..8d098b34cd76455c776677df8d29060517fc4ebf GIT binary patch literal 682 zcma)4%}&BV5dIbs1W`fI#1}xgxEDELO!QzvQZ6O?PY3SM%Tl_)x~7 zMM>Zwm)*?d`^kJWU*8{}04{M_N3V_(hPIWF^xcDT%9pqKBw!c~yt#N6JQ6zOp3&iW z==mn*X=ZZ8ZEl^6d0=9pG@s_Sq>k65JY`)W^*?i1+GS2fd_g0r##Dt`xGa?nFM}=I z*EO!T;9QeYOKB_9<@gN!t-L10%aTAiSZiKtSBx#R?97+H0J6q9%pkpn;HW{g)R!HRT*|r1?b;Jjrw)!RHg(e%T6! z-N*MGN%a@x+sRFohhn_241?fhZ7LBu=AqG)ar`@D$@Lk%{T=2$(hTh=kwR~D84iMS zimj>1vq(Owgpv=pBYqMqn)Oww#BlbUYo}5Ue{cP?l$0LzN97B85k<`a)4TutfLB0t)WhMgU|{N>A#7V W$B)qVwS!}ScH+NX^r)5{ec%UMNW*FX literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysPostMapper.class b/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysPostMapper.class new file mode 100644 index 0000000000000000000000000000000000000000..083c6168dd9252f9dfc573ed796220e885aa0896 GIT binary patch literal 1327 zcmb7^+fLg+5Qb+dO-d-I1Ze5G6sn*~-HXCiP$g2S6_R^UBT%JYZR}N8YVW4rbtS(V z7d${83NgDI*@jeddJ#>=n*INo&$Iq}b#Vy*`>@-D?Iyfx!VZCD&0`*#Z%Ug$XhYy# zHwaZi(@f=p>RcP1&`2drNIJ;1t?TUt=j;!onGrFyhk67S24W^#2tOYdzni z$x_+tQ5+L!cmL7J!PLXq#48~B`ElexxG`fUM>J4!gmc%CA+TBYXwmfZ6U|dAx83#B zs&Z2p_iKv|Z>v}^8>OlCj$c@UZ(^q^H!!JiLrl4TnxSk)h!Jj+;dVYrpF^u_}W@SU)-fO;X^q}?e9lZJ0N z5?C#-h98;0oAPMw%nbqr=J0n1KS+!71m@u`W(~~dpo#fCSSa!q=8JG20OJxY7tdEP wUxl?IZ)4tpbyv3mn=U?phc0eS;v;zM@+UT;yLWsF&z$pfc!9BjiZ21K0m=xJ1poj5 literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysRoleDeptMapper.class b/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysRoleDeptMapper.class new file mode 100644 index 0000000000000000000000000000000000000000..e9fed737c98da364b534303b3e5bb0974ef0a082 GIT binary patch literal 652 zcma)4QA@)x5WdX0y6JSfIS}7vLohEsSrMEf3@X9~BKVYbvCC+Z(xgKFnh*W}f0TG_ zH#?c258*Dyci;EjB_E$}?*MQPhYlP#&~~6hpgZLuw|t~Se#;{}NV9A>CD879bN0$; z$W%Z*tpc|{B(VDvPN36!{tb6*kk+?1I#IUd&J;54=7d>)wyc+0i4`IBgt56+eJldS zY!Y(<7YjGvh)d7clE#Tn1vRO$T+*qQOei`|&63%@%86c996n6u=*wN{MjojZl_{DM zhDsJiJf;I?__H1-4H5eQU##!LIsgCw literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysRoleMapper.class b/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysRoleMapper.class new file mode 100644 index 0000000000000000000000000000000000000000..7eae59a5cc734a859e6a742b89eb8a0062d33038 GIT binary patch literal 1565 zcmbtU+iuf95S?vHLPKvPZJ|)0^hPSAE)RVR0woVf73Nl}s1i>bXO(WTcY}9b$-l+} zAHYW;W^Fg?k|1jF5RK1fcIM2S+4=eF`wsv(f@gKut;1d&_6aO$9`eAPsE~U?8v?J} zUZ5fxCn^;bzak~k#GDRP#Dt_=8evlEr`q1>9C#P(lF`J7klFwp0`tBYN@kLn6OjKG z?PVn{uQ2(htLGz+BcZiWvX`D}9v=?~Y`1g1A(KPuDLI^0z^8vYRp+(ncy+^&K(jri zbKQx<>WS7}BCCfh{pbc&Wm>x4*oKw{q?g(j;(+|&c2G$r0&9+<&iTD(5f004IkInJ zFzD3 z#ADiHn!n3F&#E`S;j1JL_#0un(JXdL9_`Q$8-b0J1Q$=lKMO7TA@54545n8StJwEc zJfv*Iu+dcHteA74c_5tSJtv$_wc*hf`D+;othfQs%O~*MT`iX4-|MdtATWc!SzNaZ za|Gt#7NQ!W8K@(^4fA8%Kzs)l#&{9&QtAI?#4B(Y0BsXmW4~3zYp|Z3-z)5W*vRmM e!ajt}3_pU$n0vNh@C3FpCQo4-Z4D#t0Q?5tJml2? literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysRoleMenuMapper.class b/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysRoleMenuMapper.class new file mode 100644 index 0000000000000000000000000000000000000000..9fa86135731740d68e32d131a8bcb2f03c7c224b GIT binary patch literal 643 zcma)4(MrQG6uo!O)lH{!n*)D9HU#tHlNG^%!k|OgKm?z%E_RtsGMZHEule8y_)%io zZgw(3AHuy!&OPVcn|yq}y#v5G968W*pyfatq3O?<|2$+ezIo-^j8w=FT3v5WUnmKw z42Y*>;PwUxohb{MVJRfV_meamOcC~fULdr)kG~PrC)D_}Red278y*tRs%J9QF%Kj) zab%5LERB34EG zO$1#mSVa0%vtgD#{8tL0Hdb-uvs<3duCrz&XF5w~iEuoMrQw3zbIm6qyOvTJE3KqP zIPp{z5W1lDH3-ig(E>nnA{U{z(f-o-hAKj%c*DGXgwvw@&;Bb22qn9=?58Q05z4S_ k6NJh#I(Qy<6|-qo63F^Ws}YP>LW_6f5HEbQ^2RW>b=EVZXqC^1%=AAN&<@ zcI&OBpey*08=N^aIcJjR*T*LSIDpj>ER|rn1S;jk(iZ@;sEUBGE`>3RR+P-?lc8qUl=A{AJz0NeUz@swu~#9c!JOyr@6q|x zY1gvcAw?~`XtcR+zq`ims!jCKsHmyakrq$L$@6S`n=jd9-NZJKrz&I#*9W?|$GD%C zBVeuHv_7x4e(F~VO$VN}2(!msCreWq2Gn-+iQ<@J1bM@lqpq%l{!ve2GhAaZS0b)jXv;Y7A literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysTreeDictMapper.class b/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysTreeDictMapper.class new file mode 100644 index 0000000000000000000000000000000000000000..5a50246e33f87a43cae3b6699c542dcea11e4a1f GIT binary patch literal 1103 zcmcIjO-lnY5S`T8YW@DktDsl}=OTEh2#N?o#bQM~3Da$?DVt46vcjJIS04NU-uy%2 z?AB_lhgQKuHXC^J=HwPQYf>*cMWU zf_sT7ivj5ifLT>V!1x1&F&kBo%;}RK!B&;?0`|K$4v(r$4}`H-A*O}QfBUz}(cQSK zW<5iST6ocD^Gx=q!{ew;)Eh+9)agiz$K>RB_IopWqf*_(Hjt+(WV=@TCbY}6o5~Qd zJ{ZnGCR;-?=@PMN<@6N|nEfIAM^^#!->)xV=c|0^{R;vF(JnF-EeJYIy!d(Gid0g zhA;Nyw-+z71*V6AY3TwWuV7WltU4?RJ+8p{e`x!cWy?W2d_ej$VOeg41>BHM4aHtM ztX)6Y@_w=*{5j5SvU*}ndSx-Es+R)KbgJa&a%gEIcCt}0B4Q(|BG96{0!Y<#$i(Qb z6zpCmvjqieST~3h;irOZYcjFG0!FHB9o+6f1R-rY8B!S7e9rXEvc+eZN4y2xn{XUA zV1em6zJe`9*Q@HR&MGyon`Kh_I%_S^qg`}my$j3=iS5SxeAZfkimn`jXUq)sPj-5+5b#LJSSy`EH#=t4ZydrGMM%tO0 z>+p8fG$n)8ROMtCiCCL#l!$;(xSzDA-~_zdIbC|DT!T*PFPaWN9@=H@%`)OtL9R$4 zY|Syx6!Tcw6Vyxs-I99G84Mz>S zII`R_uk#|a%LW@&us3;_=`6VOTrV|O8FE>(qC{zOqI{W(lkmcdFcfS~c<~?v1#9RM zD)Yoa#9A8jREZKBW7q_TaiOytGE8GGCz;PBk*AkWfm(7yufh+T(Ot&%Qljk=g+RfR z?OD{8JX;@;JQ9eQ$xga%;0GS7ALBvIHT?b*tSm0}_rlJ?>82jiEB>J20$dGsTuH;z z;ERqhuOUE4CA-Q48Mhi1a8<)Kgc=&4JliYEUYWVJX}F;t6I8cCUv`DSb**_b(EYGp zcfCL_9S)s}C;%F+i|xG>hgL(wdt$ORdH?Yx#0?~Q-5TB(tH+m7a5a1&whWFX;o{0o z<*HS93zo&J%$o43p-ru9`dWq8g}DlzDL7P!Ok+fb+DI#aqk^X4LwrP8-7@MSc}LgM zI#N2?$)5*@TN{A%L&eAVM8&7L5jCim9V9(olTLE1AWvKgkQLl5g`L_ec?!T+o>C*_?3`ZR6mD0MdI0|C`V0) zf@%XHH=~+aqI7zfScOkn)Qc31bbgyOW6Q&qBtDk3*o7E*T|AXgiFK*{reg*Nmw3_q zPMJ&NGYy~Pt2D|QPK&fi#DC16thI#9$d$nL8q`pk6(kgCd_#N{U3WgLE6@s#!@bBb zPEOxU(icNYGmkLGu(&@eH|7)98{3!Av~8Si)OkMHnLE^nc*7KZ#g)u}=hOI>GV;FQ z=wu1y_ruqv6#CF!4+T$9t<3gNb;eUP+d#7}4AcDSCG%&N%%8;wKaOzKewI^z864-V&dq%{w2pXcsNgu{%P39xp`Wi}A^O;_)Rh67I{`OSZp4 z@&vj`0@m!fOL~m|gsyJfAw?JV(SJ%PaRaZ?|2nx4dF-bdZNUK?r1z4DLs=EC5u6gi z{!Xx>x7^yXi0)$7f$>H3{6O-biwPE*r^1B(KfoB6(Hh)t#R)a3(kSBZht@ z-0p;D{gG!{bmlEgVv63sPEQ|t{=sk>(~3IyHwx&Dy*!LJ=xrB{kWRAmCXHXxqY}nV Ye2p(CJdV;lOYB@9{`{WhyVZp literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysUserPostMapper.class b/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysUserPostMapper.class new file mode 100644 index 0000000000000000000000000000000000000000..006dd034433e98c69214eb19a2f75ddc21c6bfcf GIT binary patch literal 642 zcma)4-AcnS6#g>j>ZY5Un*(1!HUx9yl@-BV2!o2^Pz0~CE_NAhGMZHAt9juA_)ucj zZgw(3FTyzq-``2{@%i=+;0%Wj4ji-{bO^c=9&*DUG>`9;HiLB04JQQco;PE!jD}1G z)KfBW`$K}3uM%kr^a9!VDM8TbJ^#k7kD2kOD{kSC7!guWXp=cqM22P@QVSBRWwL6OxWny;62BGohEY47cMM_l>*8wmOlLG)iqKp=reCa~{(H z)BGlDpZ;q_P#dWv_W894ZRGA+kX-3>ag78=_x9pM#20@C8L?z<>Y% literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysUserRoleMapper.class b/ruoyi-system/target/classes/com/ruoyi/system/mapper/SysUserRoleMapper.class new file mode 100644 index 0000000000000000000000000000000000000000..e2d7086889d2d775ea7a25031b99256a1e617028 GIT binary patch literal 920 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 ekIO literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/ISysConfigService.class b/ruoyi-system/target/classes/com/ruoyi/system/service/ISysConfigService.class new file mode 100644 index 0000000000000000000000000000000000000000..81252ab74f8403bcb7763f1c37733bba64cc1041 GIT binary patch literal 841 zcmb7CO;5r=5S>K?`A`cYex6P7;9fKyFdihD5Sw^t^njK!89Fj~bwo$9W+SfVd6?g+8LJZixG20UoMLjvoX$2>HLd}8`S8v@VU z{ZJ(|%~UQZekDqxNI8vE!i1z;8ewu8!1mxd& zJE+Cw83x}h^EpmA_c@LKDGM7Xa8>MB25W;%#E~tIeIT&go-fgYnth&Es+;+|&${*m zT0imj2@&}8pNY*CbGvM(ZDluqt76`Z%Bsdp4ryP>;i5_0FUbF1>A?4_j)W z!Eh(sl$UCIwFZ;Gj?d#rr3s5)j&MLyF71r?0IgpG^ayN>n0C6|+^Sk83l!hu(1#-) zey}>PPO(ZSny08b`}2q;{6vb68BXJiKUxKscwkb2b*_dxWQGNynxNwa%-#`b3W;Rg zc_HwmW^61$Gn+)raGM(TCBiN9z?W@WY0I0z81Goe^<}_RA|jR1Bp)y%^d!zS<)?;A zE%2Nnqp1iv?FIg${;o6wm8BtnEn>`kt5+Pu!rfy#D+Jo*8P&NqJgLO>$FtDKMwip& z(;)EFTGz~Dg#dvi{GGuwzrIXh8O|bFL9_%7#OEq_6>$^J&)zQ}UaMehe-Soj@0Sp7 sLJI)p7F>368?HEcwUF1~y2CeM$H|*;%gNh?ykqTGJdV6jB{a=c zE-2m-B@t3iBb6{ADVIi=ocg(5=S|1noGls6jEHGLG9}RV#Y{4jrJTU>|BQEAYr01f z-Ksy21w1BjJ1V8dOwOopQXyy4%`dBd>W?;t^^m)f9hPm~KJ^r`jki`xU=ZOR3}4T) zpwz9|^R@L!UmWKRU#AujHx8|-D?R7C{0#}tcb*>FhYV|8e zu5t+Us_m355x6NBw(O}zfItJ!9{xkECV?jGW7NW^0S?9o&;~&4z@fz>IJS5KU5mW} N@okxL3Vj<5egVaJBlrLS literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/ISysDictTypeService.class b/ruoyi-system/target/classes/com/ruoyi/system/service/ISysDictTypeService.class new file mode 100644 index 0000000000000000000000000000000000000000..b8ff632c6927651345beba8e8adc96f278ec8fda GIT binary patch literal 1418 zcmbtU$!^;)5FI9IoTRC9-5nj0NDx{Fwld#9D1T9!eL60u1FbGJ}rlS zKtHO;kdiI6^`S}+!e}UJ=DjzM{Q2wp2LK$vs}?+O!HX8WB(Q0?}aceeE$dojidWWk+^G`Ftjq=H)vvTGaB1)f{qzO@l8r zMGNeo2&_xZViecyK4sAbN(dG~(}sL`J*4+Ko__edGK^V85bja$|W?>tEd> zNLNHc^lRM oFs{RfkDJi(aSOJ6+$rPzGVa0yzy7d{kKnQIRdjp;Pkr|61~w(X2mk;8 literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/ISysLogininforService.class b/ruoyi-system/target/classes/com/ruoyi/system/service/ISysLogininforService.class new file mode 100644 index 0000000000000000000000000000000000000000..1005acda90e9b5932d2220623749e9ce1ff32b11 GIT binary patch literal 519 zcmbVJyH3L}6g{_uhL%T*1Pfn~GSmwT3S#JhR4EceBo?MLF_EjpmTad|evN?-;G+;X zVF(C@ip@Uvbsp~@uWtZua2{e7VosPV?WFaIDV0|G#aP01l;oxswlS>|u6174qA+!) zv`AYwbjFJ(!iAHS%)iYi%6Y==-aC?Qw#!80RVBL7m@r9IsWabLN!a{%&)uFH?)l(l z!5X|Al&#uz;aFSr{Kc-ybSV;}%XqOS%<@WRdgvw`M4c7W)HF7i52|9~SL+{MrPA(H zE|;BH!t$5DoIzI?e@<~qy-7Wd$IX`g9uo*7juHq0CV+q+ghPxO4;dZdi2oBzd;AzD Kj1f-z%GoC!44&Tr literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/ISysMenuService.class b/ruoyi-system/target/classes/com/ruoyi/system/service/ISysMenuService.class new file mode 100644 index 0000000000000000000000000000000000000000..45f5528b6bae8fbd28b8c2aa4306aaea40309e04 GIT binary patch literal 2166 zcmcgt+iuf95S=Zhp@EjRDJ^$sxh32diSm>RL=Z@k0)@Kefj7rqC0p!u@vfs-LWu9; zfe+xL5VLF7Hfhwx;_?t}PBND>vuAdG{QUME05;%31@2bhUIp$Gn3v3Bj(W{R)ZtPQ zSg&{X?14=^Tl8or5WQw2`5{u=quQoPVAkS2VXG))1l}F%u-S0}pN3HobBf>mK;RKF z>IS~e1!Y2U6;msgnnm*|#^0)!a+!efm%kgad`YW$qu;X7m#uhLvaszESUa+T7Sj!X za-7X(p$C=#m7LE`$XHIN@Q1$bR(#;GaU=(ATO?tmqbXFo5%*ku zwK7*%E~YW+8m5Udqq5x|jal@Uu~f`I(B2@R+d-sQ_&zZGn#;)QrV#l0FR4s63$@=U zS!P@beE!eGbUTvVrcZ@uqZMm8JsTsAGE+}rp>N9L$eqkpvT+KTC@_$a#A+zle? z$6Z@-x$i}ivY}!^a@-7fiv#X3+P0D}&GbRY z^Co*JwMSj%>o6&Uz+>%QvW^)71g7wF0(bD@G=XV2iD(AV6jTg8mEl>$RXCmDGl=I# z@cfAX*#cfb`$bsF{A-Ap3-}!3l>%Nxd>$?U;CT@)ne%dbUV*CyUrW#Hux9W&+%V@& SxRvF(jkP2d-GREPtnmwqPp6^) literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/ISysNoticeService.class b/ruoyi-system/target/classes/com/ruoyi/system/service/ISysNoticeService.class new file mode 100644 index 0000000000000000000000000000000000000000..09a84e978d4801f9f9eea868d73477acb2ab76a8 GIT binary patch literal 606 zcma)4Jx{|h6g-!b&{951D}Dk>8SnyA5d#BKqzHy=Fqy=PTqQAzol5y@4Ez9o6yhZ* zASDBttoQuxp6|||U+*6PuCa7+=3>EcYGf)S`>3soPm4_87ng$>rAMehm~oe>kd_*jBTj?N3Pm+rBx}fVMH|vRgw!^7LwuF8?e30 zaXSEWM~h}EH?r8tXSf)&%O_m<77Hs|?Ib-W7k|pu#YNvm!p*ybhGFTw^bDubX{gI0 zk`F2+6VCl#r$bq6DXXhGBZkYKzoe(W=RE{H^Doz@*A2z)FffehM*z;4VGm<~&V4x4 dcd0wV0o4;soBrYMoZ+ack8wgX#M?~HzX8T)w1ofw literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/ISysOperLogService.class b/ruoyi-system/target/classes/com/ruoyi/system/service/ISysOperLogService.class new file mode 100644 index 0000000000000000000000000000000000000000..112bf28f780f01eb01dd682136bb5171f68fdd24 GIT binary patch literal 574 zcmbVJ!AiqG5PehI#@1HbD)AtX6Nb)FvZb_+eslQP!qW!Zx;&xkybO zt3pJLZDFCwmX}Y2xs_TbZj)P~vK8UB+jh8)H?gQIrA529B#a`J6>(LUlJN4cPwu+t zzGtdM%DOALwervesU@8G&p)DcTx23N1$zwsdTg=Iqm%y&*@c=gNwkcMEid7~Z?8Cx zOkF1OQE7I59_(%$$#SC-dEIgmuJ%IV)Y`e&Um?tP0#~nV4rxdr^f-B7&}VA&0j>iK bx$kkeheLjkFn0YTOkADfm}i6&hnRf>0a&aT literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/ISysPostService.class b/ruoyi-system/target/classes/com/ruoyi/system/service/ISysPostService.class new file mode 100644 index 0000000000000000000000000000000000000000..d2782ae33ad34237f3233822ca1145c9b36aaa80 GIT binary patch literal 1142 zcmb7DOHbQC5T2z#Ae3iP9_7_mJ(vr7D4c>S0TNQAh*q-T!f9i#!lw2v_3onjul2wW z;71{5*H#<|Qo_L+f1{aiX11BK_{p~*GKSl!ICura0*t?h&)d=*5Sbuf%T-9po+RG>r%DI#30^1_qjtkNy^Zm`z@{~8Z` z03U^zU2kHeM6C=D$#|@poik_7j(`3B@e=@Aa8Q6}1=ue@%Ysr!J?hGj!lQi_N(){z z`mXRv5Q&%(eDVcH6Hqb~zQZ`7TrwFGI}TM&yV*Z+CJu=t^N2d2ZNaL|M%WQb5{W>r~__HVa*kMFG!9 zvKy=Ep(M4)veV0>QB;L{Ox-U^ux}lo9&z?HLUDHfNt_ym;Tc(-o&1`{(}OFsm&m;M zMzM#zoV7k>kACS$$9BaSF<0lgz|aj>L|iJhs~)?t`d7>?i0)ZXWE>;)ozD+UKvzAUONh5}4u4RSIji{c%8_J{J z5b^EL;mh7w+y)D;%OpeGyTJ)A^R@;uaD>l>A|ie~a3l-IUKA4gT~e-g5J5U8%%!Af zClC4WebE+C;L=x2$)MIVUj%d9uA&F$mEKb`t+a0P1u^j`q?FL44J1Tm=j>y}0t=S# zcMiXG^<@i|;XHaP=q*73{R^-<^NZ+TgiAAj4gC_7XMP3!%bECf^s7(i!wnxH>VA^aTjb-q)S0W>! zQ<-ST(;^a9nu&6T$2v!zq7?}NdHic#&p3_774Wx~1~8T@R<>Q!x4d;e>I zCKMPfS9*F0_LX7~RP)x+Z?_WPS*f_klb|Pmk8Vc06FI38Zzl%F|MKXie~+Uv89Ugc zF51IVz zsNq0FNHPr@5W8YYTFq2v)jFMT<^EWFd5^{^uitv*hKn^lD2By^#@U%g?WX!~Yn0WJ zxfG#O&V8s8BG3)B%VT^a6(w8!Yd=Yd`I(IIx!01xiyQwBY3c9zA8-s-ue+edfWaL7 zd;m)e4CY}0fZ{D^(RrIr=U|cY4s@rvue*IEU#9#HtdzLb^LQ85%KRRzmvI9&%Xl9i SO#3_}+#`QZCy&dQwtfQ>J;G7| literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/ISysTreeDictService.class b/ruoyi-system/target/classes/com/ruoyi/system/service/ISysTreeDictService.class new file mode 100644 index 0000000000000000000000000000000000000000..54da39f0ed4ec42ba75f64075daa10d8afac8c0e GIT binary patch literal 968 zcmcIjT}#6-6g`=#FCJDT8bnb z4aa^OQ<RxhwEqpy%G#U$~37(u1HDPHif;6LXK4Ds@S(GX2 zr;^i{YZfrgsC&mkKB_gNVp6*&LCX_i!gMA%!Q1Z`o!4S?@gKzNjwa|<+7(}Z8ASxA z)f<1(YFLS!X%W!^Ys7o}8?ruCOC*X*;|!jio7(FKEDx>X?*B@@vz6`T=@LB*Z_i~Z>~4|i+j2PL(p69&Fy4r7Rn+Z yXqa6C>?Xk)nt+Mxu*|t_P8--Tc?<1D-ZA+mx@FzgXWT}w%y+PBFa&!ga{mL%3pB3) literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/ISysUserOnlineService.class b/ruoyi-system/target/classes/com/ruoyi/system/service/ISysUserOnlineService.class new file mode 100644 index 0000000000000000000000000000000000000000..160f48d4751216f7ea0d9309e853e49c03e5d675 GIT binary patch literal 597 zcmcgqyH3ME5S$GrhVT$VK*1M;CfCrVqd>A`C51uMhrL)A-2JLycjT0C)tAs> zGijJ`-Ukx<;s|fMl>M95OT97p;5rk8r}38EKTT(pxuwtfUg?m?b)qggr!|A?&Z~5b@9b0EdA=7=?3yqi>1- B!wCQY literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/ISysUserService.class b/ruoyi-system/target/classes/com/ruoyi/system/service/ISysUserService.class new file mode 100644 index 0000000000000000000000000000000000000000..ab377d679cfd6ef46b25e401325f81ba1f9ba595 GIT binary patch literal 2478 zcmbtWS##4c5MGr++ECIXDM!mqkCdY>w45nr60Y%(fefZRz?18URK>P}EvNNgT`ec9T-$FMmI#>NLZVq@ z84$J>n@uIGl1e45^An`h+$7GA+@)^=9<>Q94F%gTrSbQqk=q+Q6ncBn*`?0+F86Jp z{fH6;ve4CCkk*U9l*-)Lf~@gVLRRiE zbD&&>CJvZ{Y;q5VoC=0b4u&C{Qu!!v49qjH&AZ2EYAXHllOj^%cu8!(l3MC=JGw;R z;Bc<2!zYzAW9jKuB5hcU<;^skNWYwh^OC!AN;B>c8@OzE=w=F#YMZSio3m0RFsGr) z0c5J_T8oo!#N6!)bsYb|Y*G0}ygLzO+&hF6UUUsg4*Oc%y$*}IhMG#IFKBD{d-dr! zaoLsXq~!mwvxmo5QBTN@%{|uSk>SNVwqRk~jY5MS2Ajcs++v*IvyuT|0zjm5LlPHGx9h?fWQd;PT_x$ z`B4I+a2mZc=#4-Q{jo#-_#r{R4C_!|y<03U^z zq%9>u2d zwa}w*sba~xm#uN1u!}C*46yta_^r4p>P3}mFJ&Fpk_O$4P*=e5xhh5^mR(?5kh{*i@{)QLn}QV!^t;o!f=$o!~HGY>ADUB2QQ^Zif*LRu4B-Z4j#rK85|6|)ZanV zdrvQ3pG3nq%(U8Uqhs?7802W-0Se;`#$cRo6Lgybwo1Qz{1AUJjxL0mq_>|i(=d}@ hXJIb+&%*-o7s*i$mdM9)-(G~$hkb=kamXC3y#v@jX4?P& literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/TenantService.class b/ruoyi-system/target/classes/com/ruoyi/system/service/TenantService.class new file mode 100644 index 0000000000000000000000000000000000000000..a2be97384410a17dca9df20bb2463262018c555c GIT binary patch literal 1798 zcmcIlTW=CU6h2dHu`O+p)=JfSskMS8&P$^vG)-)^N`gY5@@j@<#G$)0xI0U_&;BbP z`~m(biQ}W)C)Y#9okY36O#56)doc)@Tnoz_}50ISSWW`k5QY+HZ3dJE)v+S`BDpqH-sme zHa`!#{FRhSGcA<#2;>aqlJ=M_TIP^8!kd#8%N(w`JH}7@U8Kd1zfA7s8 z-XO3ws#H@pf6t|lUgRf(iuS%1Hm!4AEFgEeF4~f5-{k}j=Mau2^^T21y2WjbZjgb^ zDHU^1o9h8$`kulclNNDaSMXKWVu>niqr0DyzgS2ZBzjmL_{&j}IzbcD=h3d}C>kL> z-`05JxBq*?(fp{7$Koyv@83%N&1F})dJwrOag+T}2$e6A$M-j-gmG-d`8x?dU_%gi zK3Xbu?FwwXM5VA3OPwn0(vzUdtP9L>sWq$YXEb*%76_jr8GrQU&PF{q1EYUQ3`+PY znJ-MXaJrbmRFa<+ffqBf;&HbpEKVb|+n-<|{+;2wrW41+q9Kr6EX0qBrQ;_mwQ$^J z^OF%`0qsv6&X^GdR?UnYwEs$7`L4y!gpDPTjZW)Pe9yuXC2%n8(f$Y@#6N~6kQq+x zO0&Z)y+nY(0{%k)tS=I{0*ko2io0bX+qf*@mkPfSha1FPL)$#e3albNjl>05L(V#E z7;Xlxn>7oYv3(2eH()#7-^6_mb`0(o+&1eS*p2OX(S8r^oAU>-XV!X^% z+{Da0{gBkW#Jmz0e{0P!Mh33n{L-T2RJTl!CNYSVXI@&q2iTkAeJYsHA8m4ZmBLw7h=lEYd?2(Fm1$|6@qyt_VDZftQfFdQ z>TJFYl}=JRlk!S^jt=&$rM6qvZXGyg@_`VhX?YRGl{Y1-TV?qoDmGV@viR;X;;%{Q gg@FOMJLa6)h(M_MhX862>S%c1^sa`;t2n6E8&`uv5&!@I literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysConfigServiceImpl.class b/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysConfigServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..15ca3a3ba6c0910bb0f64017ae527f1a18724bb3 GIT binary patch literal 5668 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 literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysDeptServiceImpl.class b/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysDeptServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..d6b43f7e652cec5fac6812d778061f8c87c1f6c8 GIT binary patch literal 10385 zcmcgy349dib^pKBN+YcX5JJKT2sW5-=wJz8j1iX*V1y-s5fGnAhSdNId&qkLO58To zaU2@EaS|sKirpqXaN2~>G!eo$vD>swX)!@Bhy4+kNhpr@jDSnG8m7JI>dj5>MCS^Y}sp&)``Le-PmeZz}PJ7QPrk zHO|*!I{v5@e~drT%a<(tX#^ASWu1ObM_FBFE`g0xqMLk#gyITA`{vm>| z;p;m3hJ|nHqW@TnZ{eRJ_-Fi!F8^)4d`Bb)L@q-Bd3qOqDzwslbIIou8o^S$q?7*}!DYN$jaMvO ziQpEGQoxZ&wS@GuLN8&`N+s8kDyk%wj;hH|YV;D(x!Q=>GC?0q)XO9?mO82~ll8V< zFH@ARL0L8GXsRXGSu%~w^}7SP9%o>{%?Rdq$I{7YCZ8URN3(<3oSTd$y+_f$!R&@2 zVY{Gddp?zmC*57~YN<@OX(dbMaX~DsPf)&@6(T-n<{r|71sdjyyv$QK>6L#n0 ziGKAPeIS^#piKU;TqBZkY2cIrshZI_V1{$b-d$zx6HaPxv^$;JYgX*bIk5wbEzj+l z{w!${yt{ne@#0qQ!Q*{jFl7W31|#256dU@%{Xb(N%WO5Z~Qz`(jp1r7? zMyj%!X)x+dXOd2$b02*>jPKQE+*m%7WpA>HkJtHYl}hfos+}%cmGeZW zYb3myFEmh%CxA)E>GCDUn-j+^k-gV7G8qB3mJ856K8^W1Rfw;@2& zm2)!=9h1S*ImYyOwHJJkO0cth%rcQJZjA&F}-f&V{OzIX_XT!7;|63T~5YL=h#U`3mk3SiMwp{VTUC%Y`g=GB{OY#lgwgv zEEUKyxvsKHuyGgzEE1(gzax`z1~pP`xn6Ey8)nOFnPbb1GFQ-2;{U1z*p@fTJX_|= z0$Ud9sF@@D7Jl277BWN{1GcnE8!PSlbUK&KWt@Q?H@7d{pPf`Ci|S<7q-u%Q$zogH zB1xkXkopOysoWX4-@hb8T{tdiB1tg&USF7+;1XX6*~fQ<+7 zUKE;tVrs@zI`_qCIquDeCA^pP9DAR z_yZS@Klg;TXs@&*(GnK8m&6G!MkyV592W#zmAXD_y|XGx7?!(-fQDmanh2v+46RIhb@lmwk3x9ZSJ33TlUCaz9jo> z{HB8bNqowdxa_y_S$s~=uF$^D$Y#Wp3{&RpWkJZoUUHK{Jt+82vAq<$c@)F3&KCp= zOLQFRCR(de)<0B;_IYOuJ9SD>%i`CO9@v&iA08CU^J|B<=+@o)xjQdxrGisT+Ht>Z z38pRx_Rj_JnOH0)_Fkqy?_hz6GMw^m8Vd9yIo41R?Xa|utsg<+H$N3lKOZ%YB_Wxs z9;PQ#2Gy~2D#sH8Drm*C87c8W~8RXq2zV` zqt0|fPh`9{xhj`74biL-GL}=Sp=0JXulv0^&#^t}T28Gyw!(RqzyH#yH@yGycY?3A zS|~YdP~1Dj>(u=doh2U2sR3^c`lIKmcLgC=1c}g`t3;&oN$vGRu_~VJ%_kCUhZ8(@ z*ySX26UH;SEkyXB-fi8Z+odf&|6*bwfAxM19^JmNy(}2aBQa}`==Pwf{%l}l;0eF* z+CE%%07p?Nb2?|lo_jSVbqB$7J6}x=bS+F8r?1rlW}C`T?&wd(8JRWNQnF|Y27vFt ztXJ&|nH5H43twg3gs2Q=-^S`*UT&Bv-23v5W=Gu`1a^g0dYqf0+-+k(88nMo%CB35 zegyp(d9|FrI+r$66oIM+M2L8R`x$La^{DpJpm?48oJ=1Ta=FWF57>Leyn;2_Nxiez z9J6+mUBp4|$LVRz3iM8|3u8qJE>ysGjE(Em(#Cbq z;b^y-(F5Z`@(Vh8K*fSYqo-JnvhjDC_~LOv;o!tevtGu-G+ez;3ci1^P{_#B-Crfvh9HPT@r_%@W%;t7iD#e~H;bgOV zcK3V`U!R}pK{q?8{)C&IYhDc6)mJ*8^N1IBuvZU~zxIb_$!otG3<95Xr;|wZv0Y#o zjMb`^PWMX%(vrjUtx-shr8)b_Ytvyizo@z$g6#bcy&k7ROHxDEH zEGk=uP}S4iIs{7>tMC?!cnhqdtZTgmZ!m8Z7p(Uew5i#C93XXq@2X~ICdnBiOp0^d z9`T;=$i3uKMarY+QGMrG)b#kDz0@`$SQ%~#pF^$SIA-a{7CegyD=XWYDu*!f9465K zr>=Z!6!GFw#C7K|IfN6K(%jNic^37pO_hb$pKm>fDY`^uOHZq+9QIUR&Myjq+xP`& zH|o%XMc9g!*oM{gh`Z?taeBZ2?LSCs-%oo#$hRIMAZpSzRB91bS1VMKdX_(;qD9D# zdO)$?#1Y)*K~_gz8ICGE@C?LCl_X?_N?|lqSyJV=(o9snf@v1=-oJw{VeVyE77i|b z3H&0{Kz@ktb1Q5h$GiEd=AP3`LpStL`9{H0m87!LLc%?p54a~EWBfOJy?;>)>3GbmyD%Etl`V880W(Yl=*V7N*bPlsB z0-kT2@esZIa6p>rMQNtv=kW97UW-Zi1y5gt!x7GLr_8HHW_fU3KRUSHZ$k4aVe2s`+>ThQ=Zc zjWBZ7@|*3W_$4w`hp9&BGTnWhrlt2^b;&u`TJjTI@^@b6k`Ln%U(3%i&cYO^i*7h^ z7&jhAO#|khLPYO}FmD+1jRy>20jC(i3oFpuHjL)uXlyeybdGtC9YzVqtlDkf2 z&;di3fwO3%M|aY{H{oedCj5fyct!Y^Zyg%4PG zl)`@yAM#cBG-*_YWy4rBjOYopoyP>GvBf>Dl=v-vKBbUVD)*A(m~b9T@2p>T7B@Yk zwZJ5rS+F?%5^C^eOvZDJ#IIlup2tFbl|NYc%c9hiIjadJ#LvlIz-hoND{8XLw|61) z8mgqPM+Ial#pX$eFZ?I@KpAZFyvzfbA<$FbMtPSH;pP**v@2BFmh-rUiml|c_93jQ zU(JgeWe97AuyzRRd{^x-es3JqnBkk4$X|L)=MO#R;@jAW?*u$&TT%FJz8NNBy$625 zS>-b38rXR_v5}JdcPtCkqqo&_ln&!BE5{I@s?g0;{FPz4Q~S!wmZq?3 zqW;$u)(W%U*Qzsw<>C4^K2Ts-i#Tc@#)ji)@MbyQcmmbUEoZT*FiCSJH?Ndl5egp_h4(ibwhHGL{HQd93AoL7aE$!i}m@ z4?(-NSCwu%hs`0B`jvVm68>&SBCN}`Xb`*TmwdS>RiiIeCCk*~ls3fNQ;R3?Yjicy z6E=U@OXR0szr#nkE9BRDE&7#Oo4ukxTStD;uWI673aS3Ng?2i1Dznfhru))T-NHigaU!QqwWFxZ4nLlBPA#D=)(+1S z6}0PS{00kMIMCrZ-!1?`C2(>%V5b(1a3K+Wd(mN zwi4~K3TtGwC)hGNV>6S`DT1bMxwZ(}T0BWuw7Ff2Pk3%=VFjy2p{p?}DV6pF=2;;{ zd7w2I(B`p!@!OZ*<%ox?Mqkm4tNL!gYNvQhP*)A(IxiJm?+sZnOyJLZ>;!D~k%cbs zi(dUO8F>z?BxD1sWh3fj6MxRN8S|tI8{{_hNDsF0?7PtB&BJ`0_F>pwgkgIThV4EK z)!2ehdoa{7#hqbn8Q(E&p3<8(PpK3TL-iTopC+h3)fs436@dADLc+6V0}PvLlg}HB zBlwX~Y8s4c-o`_AdE%R;8;U|V6oqc^g)Wp5K%c3A>fDtG)vqgTE{1s9&&d#L?XdTG iuDC^diYR6|7lFuqa!@j8Vm&kag1@4;e~rJv-~J!f)a0cA literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysDictDataServiceImpl.class b/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysDictDataServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..e9113c5154867f61fbbc17b27a23c8e541bb4aa7 GIT binary patch literal 3002 zcmb_eO;;OL7=G@M5C{WpDWI`Xsg{xeWm-h61gH>fNlj533bj2hCdrgeo6O|QOu+2@ z8@h1g#)Y2asp6s=SNaS5DK0$K=gv$*AYc#Z;c)Leci#KF&-*#~{pBw&0L+AmIM$Gr(jB>c6hkhCyD_W_^p(t_x2k*kUAKTCFq;a?P`MO~)t^wsXO>Om9h`BbnM1 zh-B=NA#f&VTE=>&3! zO@a`*s}XI1>nFw&=+2wlmhROYy1w(@2`!wM!V-Nrvmc(EUNFi`bTaU?tXtb!-jn%G zx1WUJXjKXWvQe-4x%9(z@O`%o*rr~tNripGvS3&0*^-}^{cAa!mQyG8Dx?t<+1VE6 zKAyn&5(~mJ8U-N@v%UtBj}EK)wOS%KrmB%?YKD-lh2X!I2KoA28;-!x!NO*M!s5;Z zHk(+d%VBTc(~Hm8@;+(|*Ek4Fi|x^@dV$$f6j+uW>)5*jf>K|t%2+hRCh(h|nSyRt z%#vNxD)jLLp#```;FP^=&R<^i7Q|!-Et*FZxG40%)M|>unXdO zyXF*)Tc#XC7ur?iESZ*-DUdvJ4sx z;uCxt!#x#wY^b=8%^0>+Jiuov-o>1X_c6zuwaxB^7ymNy1k&=_Df1#Qb*#}NLMM=F zQ$Oze!j^CEJho%&xCvUG)y8Su_-s*n^>`NO**5%g55~_x^2h--aCqnh+Lv27gd^{W zc|Y1lN;a;m$x(7&N&>1Iu*m4eCoHA0BY+RJd^)wsvgqaapTn5{imVv!{rq9j$^AU{ z7+y(dO#7_`EP*Ct1SE&rDzs^++6p+9hd}2vsm|F7kP3{W!o{1l!OzxpW z!S{S@g}CHH3Yum-R_XSLw*`cT^30 z81flxG0zWJhYy+UN6g6QKKBXgj4*hTvCQ*$ljpea54(@Py+@WxlX0LwLw(7Z#rYS; pWI{OEj-jfajH*eaPf4{*Y86to|7WTlVX4m}O$QV#G+S~J_y^Xy_%Z+h literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysDictTypeServiceImpl.class b/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysDictTypeServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..dd5034c7306ebbd365a6f3a6488ec95473d275db GIT binary patch literal 7513 zcmcIp349dQ9sb^ClSy{RBrM?w1O&>F3sw-V62(Fc8Vx}WhiF@!YzA1_-HE%iKw7J< z2epS+wFhmb)>^UEwx~2DT8|#y+S49ttv#gG+S>cF3jO{wv%9lwK!U%LA2V;>```CV z9^ZS<{QwrKc_EyHYeHCzYpeLUE{u=idU@Os!pHH65I%`d$-%ZTK8?@F-e=|XMtN)( zv>QYC9Bz`+&uh535?|18i)?){ght#NLM?U(>}`VjB@JH=VIr=n#O?CFL(skwMl-$| zp)|Wf_&Nr|*o`3#!(mLvJz?C7Z-j9l?w7{{8XgQ|HomFhTk`(4hVMk6uq%v5@Ted@ zCL|sgUQdW{PYCNLMVzOE)pzCdd!p0#%lCQzLihoGD32e>;nQ;XV+}v4#7{N+Ov5u8 zo>i#si^uGh1O4W5qrcxwE3|aQtYkEuwFctR%s|FAlhLGe6zv(voLmIlrci%YHf6_? z=IVGR-kUI&q*9h`*l{bBQ8=#4O0SP*`qS~$`gLg|X>PXC8>79Zk;+8Z88O>R4@8YT zLUc*iwl>GpW*^BqQk`tr#vw~p0493H3Gr0iUaU~o*tA+9*lzWi3N>Bvl-Zq4_L}LH z28C6KbXhSYvD!$-<=fi~+8g4O{@7AGWX$xYc+8B(ll=+5HG16rPHvNVP$fnzC>m1yAjB9^wWV$vUmSpB z=LKQ=ez2obNcgph1&`>ljo3!!kVBOvMPbb$D06ts_KW7F1D$L`7Blxa-Inb)D(P96L;mD)-j@pW{ogU!3!+q9craM8R5EhYTO`1|79d9|6gc~HEDBf||p5Rm7O9&sBl%T^2A&J;^J~|ZO z|HX*intfQq^K3P-glVLGeyC2H8Ph4so@1)xDOUb`nO0bM7!|RTjX_M!o|LvWD=2PM zXZ!n%LToX9tVEwkqOfSh-f5@VJJ~teU;TY$WntTp4m6hW6x%3-BqLU(;umHaU(ATn zhQC*9{e@5AF^5dAVI|{zRyLX(=w-vt^e3{JsJX>9QyG!2xK$SVqdzX*Y`Pe=xQp9- zC?55kAf3MH#p~C!RH(j9912s;8YzK z%3}$ZYWRhYGjOJk)mWq91s%V{IU0VYZQx>+jHej zFK@er#ivk`SM*-M2}H*$!sie8qX_@1jz8hgI{t!J6*zn3ra`U_K8(0kNFAl)uR`Q+ z_`8mO;Gbl;)Us@jJx2d>)81h9Wojz$uUh=OrX2sN#cMkLi`O;0p<^%hX$rbhD!}el zREu;~rh=L(*Hwkmn7O*DR8_hPsW1b+2-2-ly3*yyr>k43bajO6R~JyEUe%PV>RMH! zsajo)Q4w8@Rdu==r^f5JL>&16T&k-HYND>{)g&ET<#V!XU`5x}6g5>>N2+N$K7-zI#VD&Sl88j7L)??eqno%QB$E!&<-;Z6_L7v%aWy&o}^v8w(yz zD4yxI?2cr=&GuXG4GvBZY_}3Jp|hg3HJ$V3vU+{gxku^6O=Q*x^N6q4Ib*xrRl=V9 zuEpwdlWlPc9vicG?GmkpV{YUbHv)?$wwWBsZeGDSORclgNrNrs_z}SUQ;S;~a+7Yc z03))Z^pVnOiP`CAOEC>-VIP(s5$OL%7EHPxAUdrUfaOp&tOB3EAra9}Fq`LZ#-2s~F#h@)*VGTj_R zoXbCRHkfigiZdxMKn2t7bd%<(Nas}eux z8HHo~Ry(*~3M5FnB4ed(lXs%|egZy_5j^VpOptBMO{u9xW-M_UFR)%clG|7Q^6ghK zQW#<-qTRYR$CJq)*QIv`DNYrvy;-^07nPJ;s?mQ{3EmnV&o~}EjUL`4ep<0kD$8m9 zuySe(F`RNPR5dkASF=@x}pIlmOTi5zf{kLH()JcvL6ImBb2 z1wpiqKp-dI3WpfSb&uFdd<$Nxl@I}>!{XXO!REfOAy{|_!JxLZ`g3DVa)>=*2Fpz zQF2E*05QFk*o6lm<|?BQ#|0j-SD^6+Fj|(k-oLo)*x$MxmI7$3kI2V)2mE)S|5 z`)-%mEc5x|BGlnxH1NVPm-mfjxQv#*oUXoz_T1`--%Q^%FuNqQDhRy{VvBMSHt!z; z8ME}4DUHzDdM_rf-HrOLmdK<* zO!k7QA%I~_QMdr!x>W+OwVjVY9r#kMqn(=xKqx4$L!*z_?>v1V=z!m&^ zh(xJarkjbCMwTn;;Mt>+7Ttu+)Nmfo#uf~C@ha)+!oP>`?%8|2$Q^^gUM$jZk%sqa zxcD`?NXaeGH4_$)ywsQA5Tg5+mm6uq?T$cheF}PFD7-(9-$rb~uWBB|QT*?#QPoba z-AajeP}|#lgsSp{coX#mu_&)3f4zhl(|vvbCpK(HxP@V+b8AAX4+*%RR~8r%73YrPE;GX><)zFdl1JB;doKK%vF04V{7gZ+O}cbT=$$N zIqrsSsEC}nGcrkPuZoko93kKHjIzpqofnSRrwx#HNS2Ck8pQ3yLG0bO3hI%s*#o9YuuOI R*hvZR#9g@CTRnt_{|D7BDoy|Z literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysLogininforServiceImpl.class b/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysLogininforServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..5d91ad1c53e3479ff15c6a8a355909711c0b97bc GIT binary patch literal 1753 zcmbVM&rcIU6#k|lYuiZSTjC z#DgRr;*k=wN<3C#PQw!o^BNWy`W(*|u5dR!f8DCpxX&D)u@I? zMzIl?$ISB#LvQP@6ji<<0#S1KitBpPlEQNXhI!NTZ6m1p!nHSjtIGF0f5#|s%MFZ8 zt1LagVOWO_#!6j!d&1`xg6&zPyyOx?S0=l`(7onWI76=~T>iRVEph*yMbQ{irdPI{ z4a*m5AN}l>TOthKQA)u5T~X$SsMegL1r?ho1v;SceZnPE*{YXeD%07SidVISdlV?I z@=dfF%P@IHM}|v9X_a?q6j34RfIGb0mWC+;$*^$dRLpH_*D~r-IEHei%Hl;~yOyl` z)V$JpaTd>5(Gmq5s89`(w_3T{C{$=V>CA^_c8=xRhUvL>K3ibW6dOZ>{*H4ZK-nSo zw6LhtMX&Cc`75E=Fmi;WZ6lpgePGC(NJt{N&pp|waU*ic>qy~>hNn96Sk&-L#}bxx zJjaTTRjlc_j9v{dbPQow$4!jtxQ%fgcd^VccRtz?>%wV;{diN_rol7JoQhRAsaA%G z4u_LSN*B!Lay_us9I|b5C`xr%)eUIn()>p*4tH(272ZjjR6MAcg3z2{FjFX;2-6cp zVFogtQPfSzb>w^cbAvH@({#{1p)aOpK)R3K!5-56q%V-B!I4bU&LGXy->Gk4UuY4- z07+#GavLP6z?9onTq8}d8HPzJ=RI_jJus8oN9-p?mD)#FKmLA1DyMd|{=JXxeZv2W zp4F6yucmG!aBa9BiUS5eXmHaf4!xgr01imP98qQb{b6R!JnI z2p41cOxyN4rrS`Hhp5R2^;T#-MxaE`sIudr@fbcG16?=;^bRHngb~gE_jn=PM;RU< f!SKG-=|q_PDiT-=0h`_apJfb_Vc5*!2(!RHgYmf% literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysMenuServiceImpl.class b/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysMenuServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..e7ae8cec94efdb8f3f6161d3531be8f278e5665c GIT binary patch literal 15679 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 literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysNoticeServiceImpl.class b/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysNoticeServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..efeeb09258b7c4250281db82434c3b169fb122d7 GIT binary patch literal 1720 zcma)+TTj$L6vzKlF3Uodt0>+Th22FNML-mS0TYu=DkiSs0iR}fhjnn7NV1NfngXST~F?G}0HIi2>L-oZ$ZsniwnHsl?si`BN;}*+6S~eY^FH#l za^L^d{L~d0E9|Tonm~i}>Xzq$YnRWMJzy74C{BpXD zPzHz8C|7Q)dD=`Rep;r7RC+9wOc{gR<#u&zJ*tnF# z3maE)&BhJfv~dsjZOj_;01s`H!TgLn)I}PWHqwp9=2% z-P4Zq^p{~6q02HIk-tp8pF#4YGQH~s>rcZ<8L(Tp zO{hDa(s${O4vlOt=zxVUi5;$<3#HFC6E>KJZT|;GLzJ?>_)+V5@*U%6VME>KGp2p&r)s!1eG* z507(rlEZoq8w}&BEqqJ0UfFG$FqEx^a=9O=j^sfn2!+et_=wj!!9m)$!Z7zH@F7cgvF%5U+Y?Mtt9Qwro zNwA|k(v#sX!%(qw$S}OGnu1}%lAd@Sxpm>cwdpp7X-hS1=g{`0UMHKw;Ri|AZ)Wod zgnuF%f=jpUq_U}Xx2v>EcaKYt{6d|nqgo6r#Vk-$xwiCD5EV@`mng}wJV*$`R3IGD zh_9joSTYD1wg$V-I<`-29);53I*xefYqI6pVdRr8@Ba&AXAsuAbk%f|c!wfshG(5> zlk}b`zUz{5Y_G*F<+UoMY9dPPFqJv)RX`)f3dG5!a0VepU8D8a2$Q`;7|e!4jh8TH zDAZKsH^g(PyJ-HRu~V(Erb#dq|JcR!EPSEDPFwIKq+((MlP1Ps=CEmE7S~M7W5L7{ zZko7*yC&{ql3_jbyr|v&(Uz3bg(L^{V`>_P)xJR0OhS}L#=Fm}#%=m@sSD$5IS4+jUJ~Cu_wz`BCh+J+@SU^iJeS#<5aDzz7T zu{s_4q0{Mf>X&{{r-RUr`mrOj4N`|(xNgiA{aj>L%{k)*9s^MYs0!?B@?ur{wH_Vyr_#7;@nJIb`U=6e3SO&I=;toMXnzS32>`tQJs~;m;9nOL4`p z?S`uP#yK_?;9_L!hDEKL=fiRIR!2tE$Qai|4>dubuANR&vKp71EPE$mi`$56L(_*0 zN=evKB4gCBcGZe_U&II5uDImWu2w}1>e6(VI6+?R-AgCc?;zE$r7~B?Y{lFO>Wh-9 z6Z1D&O)Etw{A94o^IpV35*SUVFxsslVAB#dPoTkMd{A-d2)Jn%!GL1Kv_T^&$5I0% zG-f=SG-dUyrRpZBj!Z7HS~Sy}IZ837qAglzElpmNw9`&k9m9@aN>(szBoh&Jzs72_ zX6|d2jBQ-#0;+NmZDoSS4+G5KuHyL z(EoM`Uot=^15~3%#8(o!a6m!`2S}qmXzJOB8t2m>y@8JOgn)|8J2sj3Z{NCqY2v}f z%XNQEU;g|2o%@%5crf`B^>9W2dIv^HQAK<$;R?Rt09=*OjUEYy5SDNlVFBASp55ar zostElT0j#odCFp&5zeO%F);m^=Y?E0bBLU2W#+_Y?&=(S4!*VF69XSVlb!NM68D!#?RzFfq`x7Ut z!W`ceh#?~pQ|R2?lY?2dsHJ1$0R|>J$;aE1PLGV=tuoa_gm=WFhJKTy5?~| z&6|_xncAJktz;q7RiJBr&20!b?Iqi2WB{OyUIV;<46q$L=t`e3*h#>=c%SVSD607t zd9^d}<>3ZR=OF{J#fB)OIWlu0_^tts1hvbyev$4-Zd}j?dSZt`2AzRl@L<+w(F(el z4$@v)Nt#7tE_(T1dMyjkdkHNz4bncB*8IH&ykK1mfVJ9S4#PHrZO1Dv!<+Ea59p|! zf;f$WNfef#@H&G0JcXiZNZod9r?6}W%k$8)bs8%sQM%Q6Uq?1qClM%Kb)&eLL+`Ug zUq;yi`mq*Ar~r;qo{pmpCvAE)j>FkhbhxQlONGL_=8~yI2c^e{{e-d{zLop^|FSH< zD&jRF-uX22SI?m2$>@&|eT~SB68$LA*Z(K_bh5b7&rWG6m2X)dexV&_m+!d+Qkt$0 zI-zoaaWAa#f=w&}M#ql}b`K)`fZp2fy1OmDUH(%4vp-@_U8%o*Z+=aA{uG|s8?dkC z-=p$(R2(Z_>%W7_W4`hLl}dOT&)&v!_b&K_>yQ3$n{%B*$f=MD>TuA>SAwgQ?Q0|* z-%{?cP-efI4Ra-l3%6;2teo25s_LDsoL;&=NQB)gd58iLuM0xsKU94JAEOII=6P>q keqJClj!Fo894`R2-V1i?v0y$|bo!`2IM+vJ1@kEIAK@9$YXATM literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysRoleServiceImpl.class b/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysRoleServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..f906677696e77436e1dc6990cc73d1a09e5d71ab GIT binary patch literal 13631 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 F{{l;eiA(?h literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysUserOnlineServiceImpl.class b/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysUserOnlineServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..cdf53b04f92090d43c796417e744bb1549bf52ee GIT binary patch literal 2682 zcmcgu?N%FA6y28qfiRSBw9;a&)&`O?)hbp~X$!TA4Nyq{!D=0n0S1zpbY=o(`6T!h z_$OuQT0Vde;}f`C?vt57BGAGg|J=Lpx#ymJ?)kVgfBy5^9{@hWPce+(ZVaE|vlvpi z7sFflT$WR^yf5fM6c1zQ2(+dJ&B*XbhA%|nv7j#neI;mC(AR>V2ucf@i(wuMV)AJW z-(XQ>nJBVREJd-b(0jg^s(@xGPEE}#R z>rmM0Z&{wg-E__=>TcPoSh`p7e6y%~rn_t9Ox-G$3VNpEEm5#w7icvT?96f_r7-B3 z1vBRdTho==jPF|ZW~zaNbVbM3bB=53d8cStwqB&)1wHL- zTDEYd#v0T^pF+(hZ|Fk16_WO!(%@-9P=Zzu* z+aUx2>~GeYwKp7vpHBNMRVEmEY|cx+je}zw;Y=#9Zx&%rtX!b@-|#1 zYR_?Zbe6VjI)0^O>Y=%mhN~EjVnxHZSdC&$!*_V5LC0+k-(yw74|uNON09Hvi7q*< zA~mebu^jRmOc^$0*c928AWP7;pdCR4K}Faa9F(GXp}~cx!N>C`${Kb>XAgOW$rJ2M z=j(e^q4&7h7S^}ftn7v}ZOxt*bA{yb0ixIRu0rRg>DQlv!T8u|eR=(a#jaTxYvKwYLV2gVs3fih4@W0g^?3Et zQtcU(2V1I{8RYhVL@{bjyUrd>1dK&Lnfrtdbd}NkbY!?Wo9H2X$%#`B?$vQ7co5J61zg9wTq(SV8yw~EI<+I52NKCDS`HD} zN9Qw!9;I8rlpsirc5F0%IVn3%j491Bg-0N%&V8qEOW7^B(o+IiJSVd8YYwpa2c z$u~j1+h_14&*rTHaiNC@imlB6){{`KNDxyUDXj2Ejqhs|YI_HuU#;WL&xvdIK z=9VhDWxiBJkIcPQ^vT>`#emF%RSa>8Rxx~tH}-=~Syl076_=C88Jt8not$ShpE6U6 ztfUM}Ar3_r#sXDP%_fPVozzq(og literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysUserServiceImpl.class b/ruoyi-system/target/classes/com/ruoyi/system/service/impl/SysUserServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..fd5ae49e437743843dde021c24e80f9f5d518ab1 GIT binary patch literal 17139 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 AkpKVy literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/impl/TenantPackageServiceImpl.class b/ruoyi-system/target/classes/com/ruoyi/system/service/impl/TenantPackageServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..d6e15e56f52fde9dc7705cdbc20859717ef456ee GIT binary patch literal 6552 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 literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/impl/TenantServiceImpl.class b/ruoyi-system/target/classes/com/ruoyi/system/service/impl/TenantServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..2c603018526a7aaec8435c3595c25bfa144881bb GIT binary patch literal 17733 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 literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/impl/TreeDictDataServiceImpl.class b/ruoyi-system/target/classes/com/ruoyi/system/service/impl/TreeDictDataServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..f00565db89f8d5653b8569175ccd036ccbafb7a8 GIT binary patch literal 9520 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 literal 0 HcmV?d00001 diff --git a/ruoyi-system/target/classes/com/ruoyi/system/service/impl/TreeDictServiceImpl.class b/ruoyi-system/target/classes/com/ruoyi/system/service/impl/TreeDictServiceImpl.class new file mode 100644 index 0000000000000000000000000000000000000000..0934491892365628142ec6ae10816e379642450f GIT binary patch literal 6907 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 + + + + + + + + + + + + + + + + + select config_id, config_name, config_key, config_value, config_type, create_by, create_time, update_by, update_time, remark + from sys_config + + + + + + + and config_id = #{configId} + + + and config_key = #{configKey} + + + + + + + + + + + + insert into sys_config ( + config_name, + config_key, + config_value, + config_type, + create_by, + remark, + create_time + )values( + #{configName}, + #{configKey}, + #{configValue}, + #{configType}, + #{createBy}, + #{remark}, + sysdate() + ) + + + + update sys_config + + config_name = #{configName}, + config_key = #{configKey}, + config_value = #{configValue}, + config_type = #{configType}, + update_by = #{updateBy}, + remark = #{remark}, + update_time = sysdate() + + where config_id = #{configId} + + + + delete from sys_config where config_id = #{configId} + + + + delete from sys_config where config_id in + + #{configId} + + + + \ No newline at end of file diff --git a/ruoyi-system/target/classes/mapper/system/SysDeptMapper.xml b/ruoyi-system/target/classes/mapper/system/SysDeptMapper.xml new file mode 100644 index 00000000..1786c907 --- /dev/null +++ b/ruoyi-system/target/classes/mapper/system/SysDeptMapper.xml @@ -0,0 +1,161 @@ + + + + + + + + + + + + + + + + + + + + + + + + + select d.dept_id, d.parent_id, d.ancestors, d.dept_name, d.order_num, d.leader, d.phone, d.email, d.status, d.del_flag, d.create_by, d.create_time + from sys_dept d + + + + + + + + + + + + + + + + + + + + insert into sys_dept( + dept_id, + parent_id, + dept_name, + ancestors, + order_num, + leader, + phone, + email, + status, + create_by, + create_time + )values( + #{deptId}, + #{parentId}, + #{deptName}, + #{ancestors}, + #{orderNum}, + #{leader}, + #{phone}, + #{email}, + #{status}, + #{createBy}, + sysdate() + ) + + + + update sys_dept + + parent_id = #{parentId}, + dept_name = #{deptName}, + ancestors = #{ancestors}, + order_num = #{orderNum}, + leader = #{leader}, + phone = #{phone}, + email = #{email}, + status = #{status}, + update_by = #{updateBy}, + update_time = sysdate() + + where dept_id = #{deptId} + + + + update sys_dept set ancestors = + + when #{item.deptId} then #{item.ancestors} + + where dept_id in + + #{item.deptId} + + + + + update sys_dept set status = '0' where dept_id in + + #{deptId} + + + + + update sys_dept set del_flag = '2' where dept_id = #{deptId} + + + diff --git a/ruoyi-system/target/classes/mapper/system/SysDictDataMapper.xml b/ruoyi-system/target/classes/mapper/system/SysDictDataMapper.xml new file mode 100644 index 00000000..8da9030b --- /dev/null +++ b/ruoyi-system/target/classes/mapper/system/SysDictDataMapper.xml @@ -0,0 +1,124 @@ + + + + + + + + + + + + + + + + + + + + + + select dict_code, dict_sort, dict_label, dict_value, dict_type, css_class, list_class, is_default, status, create_by, create_time, remark + from sys_dict_data + + + + + + + + + + + + + + delete from sys_dict_data where dict_code = #{dictCode} + + + + delete from sys_dict_data where dict_code in + + #{dictCode} + + + + + update sys_dict_data + + dict_sort = #{dictSort}, + dict_label = #{dictLabel}, + dict_value = #{dictValue}, + dict_type = #{dictType}, + css_class = #{cssClass}, + list_class = #{listClass}, + is_default = #{isDefault}, + status = #{status}, + remark = #{remark}, + update_by = #{updateBy}, + update_time = sysdate() + + where dict_code = #{dictCode} + + + + update sys_dict_data set dict_type = #{newDictType} where dict_type = #{oldDictType} + + + + insert into sys_dict_data( + dict_sort, + dict_label, + dict_value, + dict_type, + css_class, + list_class, + is_default, + status, + remark, + create_by, + create_time + )values( + #{dictSort}, + #{dictLabel}, + #{dictValue}, + #{dictType}, + #{cssClass}, + #{listClass}, + #{isDefault}, + #{status}, + #{remark}, + #{createBy}, + sysdate() + ) + + + \ No newline at end of file diff --git a/ruoyi-system/target/classes/mapper/system/SysDictTypeMapper.xml b/ruoyi-system/target/classes/mapper/system/SysDictTypeMapper.xml new file mode 100644 index 00000000..55b4075f --- /dev/null +++ b/ruoyi-system/target/classes/mapper/system/SysDictTypeMapper.xml @@ -0,0 +1,105 @@ + + + + + + + + + + + + + + + + + select dict_id, dict_name, dict_type, status, create_by, create_time, remark + from sys_dict_type + + + + + + + + + + + + + + delete from sys_dict_type where dict_id = #{dictId} + + + + delete from sys_dict_type where dict_id in + + #{dictId} + + + + + update sys_dict_type + + dict_name = #{dictName}, + dict_type = #{dictType}, + status = #{status}, + remark = #{remark}, + update_by = #{updateBy}, + update_time = sysdate() + + where dict_id = #{dictId} + + + + insert into sys_dict_type( + dict_name, + dict_type, + status, + remark, + create_by, + create_time + )values( + #{dictName}, + #{dictType}, + #{status}, + #{remark}, + #{createBy}, + sysdate() + ) + + + \ No newline at end of file diff --git a/ruoyi-system/target/classes/mapper/system/SysLogininforMapper.xml b/ruoyi-system/target/classes/mapper/system/SysLogininforMapper.xml new file mode 100644 index 00000000..b8178fa3 --- /dev/null +++ b/ruoyi-system/target/classes/mapper/system/SysLogininforMapper.xml @@ -0,0 +1,57 @@ + + + + + + + + + + + + + + + + + + insert into sys_logininfor (user_name, status, ipaddr, login_location, browser, os, msg, login_time) + values (#{userName}, #{status}, #{ipaddr}, #{loginLocation}, #{browser}, #{os}, #{msg}, sysdate()) + + + + + + delete from sys_logininfor where info_id in + + #{infoId} + + + + + truncate table sys_logininfor + + + \ No newline at end of file diff --git a/ruoyi-system/target/classes/mapper/system/SysMenuMapper.xml b/ruoyi-system/target/classes/mapper/system/SysMenuMapper.xml new file mode 100644 index 00000000..eb995204 --- /dev/null +++ b/ruoyi-system/target/classes/mapper/system/SysMenuMapper.xml @@ -0,0 +1,201 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + select menu_id, menu_name, parent_id, order_num, path, component, `query`, is_frame, is_cache, menu_type, visible, status, ifnull(perms,'') as perms, icon, create_time + from sys_menu + + + + + + + + + + + + + + + + + + + + + + + + + + update sys_menu + + menu_name = #{menuName}, + parent_id = #{parentId}, + order_num = #{orderNum}, + path = #{path}, + component = #{component}, + `query` = #{query}, + is_frame = #{isFrame}, + is_cache = #{isCache}, + menu_type = #{menuType}, + visible = #{visible}, + status = #{status}, + perms = #{perms}, + icon = #{icon}, + remark = #{remark}, + update_by = #{updateBy}, + update_time = sysdate() + + where menu_id = #{menuId} + + + + insert into sys_menu( + menu_id, + parent_id, + menu_name, + order_num, + path, + component, + `query`, + is_frame, + is_cache, + menu_type, + visible, + status, + perms, + icon, + remark, + create_by, + create_time + )values( + #{menuId}, + #{parentId}, + #{menuName}, + #{orderNum}, + #{path}, + #{component}, + #{query}, + #{isFrame}, + #{isCache}, + #{menuType}, + #{visible}, + #{status}, + #{perms}, + #{icon}, + #{remark}, + #{createBy}, + sysdate() + ) + + + + delete from sys_menu where menu_id = #{menuId} + + + diff --git a/ruoyi-system/target/classes/mapper/system/SysNoticeMapper.xml b/ruoyi-system/target/classes/mapper/system/SysNoticeMapper.xml new file mode 100644 index 00000000..65d30794 --- /dev/null +++ b/ruoyi-system/target/classes/mapper/system/SysNoticeMapper.xml @@ -0,0 +1,89 @@ + + + + + + + + + + + + + + + + + + + select notice_id, notice_title, notice_type, cast(notice_content as char) as notice_content, status, create_by, create_time, update_by, update_time, remark + from sys_notice + + + + + + + + insert into sys_notice ( + notice_title, + notice_type, + notice_content, + status, + remark, + create_by, + create_time + )values( + #{noticeTitle}, + #{noticeType}, + #{noticeContent}, + #{status}, + #{remark}, + #{createBy}, + sysdate() + ) + + + + update sys_notice + + notice_title = #{noticeTitle}, + notice_type = #{noticeType}, + notice_content = #{noticeContent}, + status = #{status}, + update_by = #{updateBy}, + update_time = sysdate() + + where notice_id = #{noticeId} + + + + delete from sys_notice where notice_id = #{noticeId} + + + + delete from sys_notice where notice_id in + + #{noticeId} + + + + \ No newline at end of file diff --git a/ruoyi-system/target/classes/mapper/system/SysOperLogMapper.xml b/ruoyi-system/target/classes/mapper/system/SysOperLogMapper.xml new file mode 100644 index 00000000..018a7471 --- /dev/null +++ b/ruoyi-system/target/classes/mapper/system/SysOperLogMapper.xml @@ -0,0 +1,83 @@ + + + + + + + + + + + + + + + + + + + + + + + + + select oper_id, title, business_type, method, request_method, operator_type, oper_name, dept_name, oper_url, oper_ip, oper_location, oper_param, json_result, status, error_msg, oper_time + from sys_oper_log + + + + insert into sys_oper_log(title, business_type, method, request_method, operator_type, oper_name, dept_name, oper_url, oper_ip, oper_location, oper_param, json_result, status, error_msg, oper_time) + values (#{title}, #{businessType}, #{method}, #{requestMethod}, #{operatorType}, #{operName}, #{deptName}, #{operUrl}, #{operIp}, #{operLocation}, #{operParam}, #{jsonResult}, #{status}, #{errorMsg}, sysdate()) + + + + + + delete from sys_oper_log where oper_id in + + #{operId} + + + + + + + truncate table sys_oper_log + + + \ No newline at end of file diff --git a/ruoyi-system/target/classes/mapper/system/SysPostMapper.xml b/ruoyi-system/target/classes/mapper/system/SysPostMapper.xml new file mode 100644 index 00000000..fc37f49d --- /dev/null +++ b/ruoyi-system/target/classes/mapper/system/SysPostMapper.xml @@ -0,0 +1,122 @@ + + + + + + + + + + + + + + + + + + + select post_id, post_code, post_name, post_sort, status, create_by, create_time, remark + from sys_post + + + + + + + + + + + + + + + + + + update sys_post + + post_code = #{postCode}, + post_name = #{postName}, + post_sort = #{postSort}, + status = #{status}, + remark = #{remark}, + update_by = #{updateBy}, + update_time = sysdate() + + where post_id = #{postId} + + + + insert into sys_post( + post_id, + post_code, + post_name, + post_sort, + status, + remark, + create_by, + create_time + )values( + #{postId}, + #{postCode}, + #{postName}, + #{postSort}, + #{status}, + #{remark}, + #{createBy}, + sysdate() + ) + + + + delete from sys_post where post_id = #{postId} + + + + delete from sys_post where post_id in + + #{postId} + + + + \ No newline at end of file diff --git a/ruoyi-system/target/classes/mapper/system/SysRoleDeptMapper.xml b/ruoyi-system/target/classes/mapper/system/SysRoleDeptMapper.xml new file mode 100644 index 00000000..7c4139bc --- /dev/null +++ b/ruoyi-system/target/classes/mapper/system/SysRoleDeptMapper.xml @@ -0,0 +1,34 @@ + + + + + + + + + + + delete from sys_role_dept where role_id=#{roleId} + + + + + + delete from sys_role_dept where role_id in + + #{roleId} + + + + + insert into sys_role_dept(role_id, dept_id) values + + (#{item.roleId},#{item.deptId}) + + + + \ No newline at end of file diff --git a/ruoyi-system/target/classes/mapper/system/SysRoleMapper.xml b/ruoyi-system/target/classes/mapper/system/SysRoleMapper.xml new file mode 100644 index 00000000..58743901 --- /dev/null +++ b/ruoyi-system/target/classes/mapper/system/SysRoleMapper.xml @@ -0,0 +1,153 @@ + + + + + + + + + + + + + + + + + + + + + + + + select distinct r.role_id, r.role_name, r.role_key, r.role_sort, r.data_scope, r.menu_check_strictly, r.dept_check_strictly, + r.status, r.del_flag, r.create_time, r.remark + from sys_role r + left join sys_user_role urs on urs.role_id = r.role_id + left join sys_user u on u.user_id = urs.user_id + left join sys_dept d on u.dept_id = d.dept_id + + + + + + + + + + + + + + + + + + + + insert into sys_role( + role_id, + role_name, + role_key, + role_sort, + data_scope, + menu_check_strictly, + dept_check_strictly, + status, + remark, + create_by, + create_time + )values( + #{roleId}, + #{roleName}, + #{roleKey}, + #{roleSort}, + #{dataScope}, + #{menuCheckStrictly}, + #{deptCheckStrictly}, + #{status}, + #{remark}, + #{createBy}, + sysdate() + ) + + + + update sys_role + + role_name = #{roleName}, + role_key = #{roleKey}, + role_sort = #{roleSort}, + data_scope = #{dataScope}, + menu_check_strictly = #{menuCheckStrictly}, + dept_check_strictly = #{deptCheckStrictly}, + status = #{status}, + remark = #{remark}, + update_by = #{updateBy}, + update_time = sysdate() + + where role_id = #{roleId} + + + + update sys_role set del_flag = '2' where role_id = #{roleId} + + + + update sys_role set del_flag = '2' where role_id in + + #{roleId} + + + + diff --git a/ruoyi-system/target/classes/mapper/system/SysRoleMenuMapper.xml b/ruoyi-system/target/classes/mapper/system/SysRoleMenuMapper.xml new file mode 100644 index 00000000..cb60a852 --- /dev/null +++ b/ruoyi-system/target/classes/mapper/system/SysRoleMenuMapper.xml @@ -0,0 +1,34 @@ + + + + + + + + + + + + + delete from sys_role_menu where role_id=#{roleId} + + + + delete from sys_role_menu where role_id in + + #{roleId} + + + + + insert into sys_role_menu(role_id, menu_id) values + + (#{item.roleId},#{item.menuId}) + + + + \ No newline at end of file diff --git a/ruoyi-system/target/classes/mapper/system/SysTreeDictDataMapper.xml b/ruoyi-system/target/classes/mapper/system/SysTreeDictDataMapper.xml new file mode 100644 index 00000000..69421d5b --- /dev/null +++ b/ruoyi-system/target/classes/mapper/system/SysTreeDictDataMapper.xml @@ -0,0 +1,77 @@ + + + + + + + + + + + + + + + + + + + + + + + + SELECT + o.id as id, + o.label as label, + o.tree_dict as tree_dict, + o.code as code, + o.remark as remark, + o.pid as pid, + o.order_num as order_num, + o.level_code as level_code, + o.level_depth as level_depth, + o.is_leaf as is_leaf, + o.path as path, + o.icon as icon + FROM sys_tree_dict_data o + LEFT JOIN sys_user u ON u.user_id = o.create_by + LEFT JOIN sys_dept d ON u.dept_id = d.dept_id + + + + + + diff --git a/ruoyi-system/target/classes/mapper/system/SysTreeDictMapper.xml b/ruoyi-system/target/classes/mapper/system/SysTreeDictMapper.xml new file mode 100644 index 00000000..f274a9a0 --- /dev/null +++ b/ruoyi-system/target/classes/mapper/system/SysTreeDictMapper.xml @@ -0,0 +1,53 @@ + + + + + + + + + + + + + + + + SELECT + o.id as id, + o.code as code, + o.name as name, + o.stru_type as stru_type, + o.create_by as create_by, + o.remark as remark, + o.is_sys_param as is_sys_param + FROM sys_tree_dict o + + + + + + diff --git a/ruoyi-system/target/classes/mapper/system/SysUserMapper.xml b/ruoyi-system/target/classes/mapper/system/SysUserMapper.xml new file mode 100644 index 00000000..1c170fd5 --- /dev/null +++ b/ruoyi-system/target/classes/mapper/system/SysUserMapper.xml @@ -0,0 +1,232 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + select u.user_id, u.dept_id, u.user_name, u.nick_name, u.email, u.avatar, u.phonenumber, u.password, u.sex, u.status, u.del_flag, u.login_ip, u.login_date, u.create_by, u.create_time, u.remark,u.tenant_id, + d.dept_id, d.parent_id, d.ancestors, d.dept_name, d.order_num, d.leader, d.status as dept_status, + r.role_id, r.role_name, r.role_key, r.role_sort, r.data_scope, r.status as role_status + from sys_user u + left join sys_dept d on u.dept_id = d.dept_id + left join sys_user_role urs on u.user_id = urs.user_id + left join sys_role r on r.role_id = urs.role_id + + + + + + + + + + + + + + + + + + + + insert into sys_user( + user_id, + dept_id, + user_name, + nick_name, + email, + avatar, + phonenumber, + sex, + password, + status, + create_by, + remark, + create_time + )values( + #{userId}, + #{deptId}, + #{userName}, + #{nickName}, + #{email}, + #{avatar}, + #{phonenumber}, + #{sex}, + #{password}, + #{status}, + #{createBy}, + #{remark}, + sysdate() + ) + + + + update sys_user + + dept_id = #{deptId}, + user_name = #{userName}, + nick_name = #{nickName}, + email = #{email}, + phonenumber = #{phonenumber}, + sex = #{sex}, + avatar = #{avatar}, + password = #{password}, + status = #{status}, + login_ip = #{loginIp}, + login_date = #{loginDate}, + update_by = #{updateBy}, + remark = #{remark}, + update_time = sysdate() + + where user_id = #{userId} + + + + update sys_user set status = #{status} where user_id = #{userId} + + + + update sys_user set avatar = #{avatar} where user_name = #{userName} + + + + update sys_user set password = #{password} where user_name = #{userName} + + + + update sys_user set del_flag = '2' where user_id = #{userId} + + + + update sys_user set del_flag = '2' where user_id in + + #{userId} + + + + diff --git a/ruoyi-system/target/classes/mapper/system/SysUserPostMapper.xml b/ruoyi-system/target/classes/mapper/system/SysUserPostMapper.xml new file mode 100644 index 00000000..912f9dfb --- /dev/null +++ b/ruoyi-system/target/classes/mapper/system/SysUserPostMapper.xml @@ -0,0 +1,34 @@ + + + + + + + + + + + delete from sys_user_post where user_id=#{userId} + + + + + + delete from sys_user_post where user_id in + + #{userId} + + + + + insert into sys_user_post(user_id, post_id) values + + (#{item.userId},#{item.postId}) + + + + diff --git a/ruoyi-system/target/classes/mapper/system/SysUserRoleMapper.xml b/ruoyi-system/target/classes/mapper/system/SysUserRoleMapper.xml new file mode 100644 index 00000000..dd726891 --- /dev/null +++ b/ruoyi-system/target/classes/mapper/system/SysUserRoleMapper.xml @@ -0,0 +1,44 @@ + + + + + + + + + + + delete from sys_user_role where user_id=#{userId} + + + + + + delete from sys_user_role where user_id in + + #{userId} + + + + + insert into sys_user_role(user_id, role_id) values + + (#{item.userId},#{item.roleId}) + + + + + delete from sys_user_role where user_id=#{userId} and role_id=#{roleId} + + + + delete from sys_user_role where role_id=#{roleId} and user_id in + + #{userId} + + + \ No newline at end of file diff --git a/ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/Device/DeviceConvertImpl.java b/ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/Device/DeviceConvertImpl.java new file mode 100644 index 00000000..307d3874 --- /dev/null +++ b/ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/Device/DeviceConvertImpl.java @@ -0,0 +1,11 @@ +package com.ruoyi.system.convert.Device; + +import javax.annotation.Generated; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:10+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class DeviceConvertImpl implements DeviceConvert { +} diff --git a/ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/task/TaskConvertImpl.java b/ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/task/TaskConvertImpl.java new file mode 100644 index 00000000..90f75eb5 --- /dev/null +++ b/ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/task/TaskConvertImpl.java @@ -0,0 +1,11 @@ +package com.ruoyi.system.convert.task; + +import javax.annotation.Generated; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:10+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class TaskConvertImpl implements TaskConvert { +} diff --git a/ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/tenant/TenantConvertImpl.java b/ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/tenant/TenantConvertImpl.java new file mode 100644 index 00000000..dda3c7fe --- /dev/null +++ b/ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/tenant/TenantConvertImpl.java @@ -0,0 +1,142 @@ +package com.ruoyi.system.convert.tenant; + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.system.domain.TenantDO; +import com.ruoyi.system.domain.TenantDO.TenantDOBuilder; +import com.ruoyi.system.domain.vo.tenant.TenantCreateReqVO; +import com.ruoyi.system.domain.vo.tenant.TenantExcelVO; +import com.ruoyi.system.domain.vo.tenant.TenantRespVO; +import com.ruoyi.system.domain.vo.tenant.TenantUpdateReqVO; +import java.util.ArrayList; +import java.util.List; +import javax.annotation.Generated; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:10+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class TenantConvertImpl implements TenantConvert { + + @Override + public TenantDO convert(TenantCreateReqVO bean) { + if ( bean == null ) { + return null; + } + + TenantDOBuilder tenantDO = TenantDO.builder(); + + tenantDO.name( bean.getName() ); + tenantDO.contactName( bean.getContactName() ); + tenantDO.contactMobile( bean.getContactMobile() ); + tenantDO.status( bean.getStatus() ); + tenantDO.domain( bean.getDomain() ); + tenantDO.packageId( bean.getPackageId() ); + tenantDO.expireTime( bean.getExpireTime() ); + tenantDO.accountCount( bean.getAccountCount() ); + + return tenantDO.build(); + } + + @Override + public TenantDO convert(TenantUpdateReqVO bean) { + if ( bean == null ) { + return null; + } + + TenantDOBuilder tenantDO = TenantDO.builder(); + + tenantDO.id( bean.getId() ); + tenantDO.name( bean.getName() ); + tenantDO.contactName( bean.getContactName() ); + tenantDO.contactMobile( bean.getContactMobile() ); + tenantDO.status( bean.getStatus() ); + tenantDO.domain( bean.getDomain() ); + tenantDO.packageId( bean.getPackageId() ); + tenantDO.expireTime( bean.getExpireTime() ); + tenantDO.accountCount( bean.getAccountCount() ); + + return tenantDO.build(); + } + + @Override + public TenantRespVO convert(TenantDO bean) { + if ( bean == null ) { + return null; + } + + TenantRespVO tenantRespVO = new TenantRespVO(); + + tenantRespVO.setName( bean.getName() ); + tenantRespVO.setContactName( bean.getContactName() ); + tenantRespVO.setContactMobile( bean.getContactMobile() ); + tenantRespVO.setStatus( bean.getStatus() ); + tenantRespVO.setDomain( bean.getDomain() ); + tenantRespVO.setPackageId( bean.getPackageId() ); + tenantRespVO.setExpireTime( bean.getExpireTime() ); + tenantRespVO.setAccountCount( bean.getAccountCount() ); + tenantRespVO.setId( bean.getId() ); + tenantRespVO.setCreateTime( bean.getCreateTime() ); + + return tenantRespVO; + } + + @Override + public List convertList(List list) { + if ( list == null ) { + return null; + } + + List list1 = new ArrayList( list.size() ); + for ( TenantDO tenantDO : list ) { + list1.add( convert( tenantDO ) ); + } + + return list1; + } + + @Override + public PageResult convertPage(PageResult page) { + if ( page == null ) { + return null; + } + + PageResult pageResult = new PageResult(); + + pageResult.setList( convertList( page.getList() ) ); + pageResult.setTotal( page.getTotal() ); + + return pageResult; + } + + @Override + public List convertList02(List list) { + if ( list == null ) { + return null; + } + + List list1 = new ArrayList( list.size() ); + for ( TenantDO tenantDO : list ) { + list1.add( tenantDOToTenantExcelVO( tenantDO ) ); + } + + return list1; + } + + protected TenantExcelVO tenantDOToTenantExcelVO(TenantDO tenantDO) { + if ( tenantDO == null ) { + return null; + } + + TenantExcelVO tenantExcelVO = new TenantExcelVO(); + + tenantExcelVO.setId( tenantDO.getId() ); + tenantExcelVO.setName( tenantDO.getName() ); + tenantExcelVO.setContactName( tenantDO.getContactName() ); + tenantExcelVO.setContactMobile( tenantDO.getContactMobile() ); + tenantExcelVO.setStatus( tenantDO.getStatus() ); + tenantExcelVO.setCreateTime( tenantDO.getCreateTime() ); + + return tenantExcelVO; + } +} diff --git a/ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/tenant/TenantPackageConvertImpl.java b/ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/tenant/TenantPackageConvertImpl.java new file mode 100644 index 00000000..a945299d --- /dev/null +++ b/ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/tenant/TenantPackageConvertImpl.java @@ -0,0 +1,137 @@ +package com.ruoyi.system.convert.tenant; + +import com.ruoyi.common.mybatis.pojo.PageResult; +import com.ruoyi.system.domain.TenantPackageDO; +import com.ruoyi.system.domain.TenantPackageDO.TenantPackageDOBuilder; +import com.ruoyi.system.domain.vo.packages.TenantPackageCreateReqVO; +import com.ruoyi.system.domain.vo.packages.TenantPackageRespVO; +import com.ruoyi.system.domain.vo.packages.TenantPackageSimpleRespVO; +import com.ruoyi.system.domain.vo.packages.TenantPackageUpdateReqVO; +import java.util.ArrayList; +import java.util.HashSet; +import java.util.List; +import java.util.Set; +import javax.annotation.Generated; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:10+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class TenantPackageConvertImpl implements TenantPackageConvert { + + @Override + public TenantPackageDO convert(TenantPackageCreateReqVO bean) { + if ( bean == null ) { + return null; + } + + TenantPackageDOBuilder tenantPackageDO = TenantPackageDO.builder(); + + tenantPackageDO.name( bean.getName() ); + tenantPackageDO.status( bean.getStatus() ); + tenantPackageDO.remark( bean.getRemark() ); + Set set = bean.getMenuIds(); + if ( set != null ) { + tenantPackageDO.menuIds( new HashSet( set ) ); + } + + return tenantPackageDO.build(); + } + + @Override + public TenantPackageDO convert(TenantPackageUpdateReqVO bean) { + if ( bean == null ) { + return null; + } + + TenantPackageDOBuilder tenantPackageDO = TenantPackageDO.builder(); + + tenantPackageDO.id( bean.getId() ); + tenantPackageDO.name( bean.getName() ); + tenantPackageDO.status( bean.getStatus() ); + tenantPackageDO.remark( bean.getRemark() ); + Set set = bean.getMenuIds(); + if ( set != null ) { + tenantPackageDO.menuIds( new HashSet( set ) ); + } + + return tenantPackageDO.build(); + } + + @Override + public TenantPackageRespVO convert(TenantPackageDO bean) { + if ( bean == null ) { + return null; + } + + TenantPackageRespVO tenantPackageRespVO = new TenantPackageRespVO(); + + tenantPackageRespVO.setName( bean.getName() ); + tenantPackageRespVO.setStatus( bean.getStatus() ); + tenantPackageRespVO.setRemark( bean.getRemark() ); + Set set = bean.getMenuIds(); + if ( set != null ) { + tenantPackageRespVO.setMenuIds( new HashSet( set ) ); + } + tenantPackageRespVO.setId( bean.getId() ); + tenantPackageRespVO.setCreateTime( bean.getCreateTime() ); + + return tenantPackageRespVO; + } + + @Override + public List convertList(List list) { + if ( list == null ) { + return null; + } + + List list1 = new ArrayList( list.size() ); + for ( TenantPackageDO tenantPackageDO : list ) { + list1.add( convert( tenantPackageDO ) ); + } + + return list1; + } + + @Override + public PageResult convertPage(PageResult page) { + if ( page == null ) { + return null; + } + + PageResult pageResult = new PageResult(); + + pageResult.setList( convertList( page.getList() ) ); + pageResult.setTotal( page.getTotal() ); + + return pageResult; + } + + @Override + public List convertList02(List list) { + if ( list == null ) { + return null; + } + + List list1 = new ArrayList( list.size() ); + for ( TenantPackageDO tenantPackageDO : list ) { + list1.add( tenantPackageDOToTenantPackageSimpleRespVO( tenantPackageDO ) ); + } + + return list1; + } + + protected TenantPackageSimpleRespVO tenantPackageDOToTenantPackageSimpleRespVO(TenantPackageDO tenantPackageDO) { + if ( tenantPackageDO == null ) { + return null; + } + + TenantPackageSimpleRespVO tenantPackageSimpleRespVO = new TenantPackageSimpleRespVO(); + + tenantPackageSimpleRespVO.setId( tenantPackageDO.getId() ); + tenantPackageSimpleRespVO.setName( tenantPackageDO.getName() ); + + return tenantPackageSimpleRespVO; + } +} diff --git a/ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/user/UserConvertImpl.java b/ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/user/UserConvertImpl.java new file mode 100644 index 00000000..a44ef099 --- /dev/null +++ b/ruoyi-system/target/generated-sources/annotations/com/ruoyi/system/convert/user/UserConvertImpl.java @@ -0,0 +1,43 @@ +package com.ruoyi.system.convert.user; + +import com.ruoyi.common.core.domain.entity.SysUser; +import com.ruoyi.system.domain.vo.user.UserSimpleRespVO; +import java.util.ArrayList; +import java.util.List; +import javax.annotation.Generated; + +@Generated( + value = "org.mapstruct.ap.MappingProcessor", + date = "2023-07-11T22:37:10+0800", + comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_91 (Oracle Corporation)" +) +public class UserConvertImpl implements UserConvert { + + @Override + public List convertList04(List list) { + if ( list == null ) { + return null; + } + + List list1 = new ArrayList( list.size() ); + for ( SysUser sysUser : list ) { + list1.add( sysUserToUserSimpleRespVO( sysUser ) ); + } + + return list1; + } + + protected UserSimpleRespVO sysUserToUserSimpleRespVO(SysUser sysUser) { + if ( sysUser == null ) { + return null; + } + + UserSimpleRespVO userSimpleRespVO = new UserSimpleRespVO(); + + userSimpleRespVO.setUserId( sysUser.getUserId() ); + userSimpleRespVO.setNickName( sysUser.getNickName() ); + userSimpleRespVO.setUserName( sysUser.getUserName() ); + + return userSimpleRespVO; + } +} diff --git a/注意事项.txt b/注意事项.txt new file mode 100644 index 00000000..b9289584 --- /dev/null +++ b/注意事项.txt @@ -0,0 +1,32 @@ +关于代码生成器代码生成后需修改的点: + 1、表实体ID自增的要加注解:@TableId(type = IdType.AUTO) +关于工作流: + 工作流flw_和act_开头的表可以不用建 +本地开发添加Jvm启动参数:-Dspring.profiles.active=dev + +多租户表: + 1、系统表如果是多租户的或者bs_,bpm_开头的非多租户表都要在TenantDatabaseInterceptor里面进行配置 + +部署注意事项: + 1、如果docker部署,需挂载日志与附件的目录 + +数据库存json 对象: + 1、实体字段加注解 @TableField(typeHandler = JacksonTypeHandler.class) + 2、实体加@TableName(value = "bs_xxx", autoResultMap = true),目前代码生成会自动加上这个 + 2、mapperxml 加 typeHandler="com.baomidou.mybatisplus.extension.handlers.JacksonTypeHandler" + +权限注意事项: + 1、部门等权限控制使用@DataScope注解,然后mapperxml里面要去关联部门表。勿用mybatis plus + + +开发规范: +图片统一格式: [{"name": "xx","url": "xx"}] + +表设计固定需要字段: +`files` varchar(255) NOT NULL DEFAULT '[]' COMMENT '附件', +`create_by` varchar(64) DEFAULT '' COMMENT '创建者', +`create_time` datetime DEFAULT NULL COMMENT '创建时间', +`update_by` varchar(64) DEFAULT '' COMMENT '更新者', +`update_time` datetime DEFAULT NULL COMMENT '更新时间', +`tenant_id` bigint(20) NOT NULL COMMENT '租户编号', +tenant_id的话看是不是需要分为多租户表 \ No newline at end of file

    + * + * @see AWS + * API Documentation + */ + @SneakyThrows + public List getAllBuckets() { + return amazonS3.listBuckets(); + } + + /** + * @param bucketName bucket名称 + * @see AWS + * API Documentation + */ + @SneakyThrows + public Optional getBucket(String bucketName) { + return amazonS3.listBuckets().stream().filter(b -> b.getName().equals(bucketName)).findFirst(); + } + + /** + * @param bucketName bucket名称 + * @see AWS API + * Documentation + */ + @SneakyThrows + public void removeBucket(String bucketName) { + amazonS3.deleteBucket(bucketName); + } + + /** + * 根据文件前置查询文件 + * + * @param bucketName bucket名称 + * @param prefix 前缀 + * @param recursive 是否递归查询 + * @return S3ObjectSummary 列表 + * @see AWS + * API Documentation + */ + @SneakyThrows + public List getAllObjectsByPrefix(String bucketName, String prefix, boolean recursive) { + ObjectListing objectListing = amazonS3.listObjects(bucketName, prefix); + return new ArrayList<>(objectListing.getObjectSummaries()); + } + + /** + * 获取文件外链 + * + * @param bucketName bucket名称 + * @param objectName 文件名称 + * @param expires 过期时间 <=7 + * @return url + * @see AmazonS3#generatePresignedUrl(String bucketName, String key, Date expiration) + */ + @SneakyThrows + public String getObjectURL(String bucketName, String objectName, Integer expires) { + Date date = new Date(); + Calendar calendar = new GregorianCalendar(); + calendar.setTime(date); + calendar.add(Calendar.DAY_OF_MONTH, expires); + URL url = amazonS3.generatePresignedUrl(bucketName, objectName, calendar.getTime()); + return url.toString(); + } + + /** + * 获取文件URL + *