初始化
This commit is contained in:
		
							
								
								
									
										21
									
								
								.editorconfig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								.editorconfig
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | ||||
| # http://editorconfig.org | ||||
| root = true | ||||
|  | ||||
| # 空格替代Tab缩进在各种编辑工具下效果一致 | ||||
| [*] | ||||
| indent_style = space | ||||
| indent_size = 4 | ||||
| charset = utf-8 | ||||
| end_of_line = lf | ||||
| trim_trailing_whitespace = true | ||||
| insert_final_newline = true | ||||
|  | ||||
| [*.java] | ||||
| indent_style = tab | ||||
|  | ||||
| [*.{json,yml}] | ||||
| indent_size = 2 | ||||
|  | ||||
| [*.md] | ||||
| insert_final_newline = false | ||||
| trim_trailing_whitespace = false | ||||
							
								
								
									
										44
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,44 @@ | ||||
| ###################################################################### | ||||
| # Build Tools | ||||
|  | ||||
| .gradle | ||||
| /build/ | ||||
| !gradle/wrapper/gradle-wrapper.jar | ||||
|  | ||||
| target/ | ||||
| !.mvn/wrapper/maven-wrapper.jar | ||||
|  | ||||
| ###################################################################### | ||||
| # IDE | ||||
|  | ||||
| ### STS ### | ||||
| .apt_generated | ||||
| .classpath | ||||
| .factorypath | ||||
| .project | ||||
| .settings | ||||
| .springBeans | ||||
|  | ||||
| ### IntelliJ IDEA ### | ||||
| .idea | ||||
| *.iws | ||||
| *.iml | ||||
| *.ipr | ||||
|  | ||||
| ### NetBeans ### | ||||
| nbproject/private/ | ||||
| build/* | ||||
| nbbuild/ | ||||
| dist/ | ||||
| nbdist/ | ||||
| .nb-gradle/ | ||||
|  | ||||
| ###################################################################### | ||||
| # Others | ||||
| *.log | ||||
| *.xml.versionsBackup | ||||
| *.swp | ||||
|  | ||||
| !*/build/*.java | ||||
| !*/build/*.html | ||||
| !*/build/*.xml | ||||
							
								
								
									
										36
									
								
								README.en.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								README.en.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,36 @@ | ||||
| # 若依mp版本 | ||||
|  | ||||
| #### Description | ||||
| {**When you're done, you can delete the content in this README and update the file with details for others getting started with your repository**} | ||||
|  | ||||
| #### Software Architecture | ||||
| Software architecture description | ||||
|  | ||||
| #### Installation | ||||
|  | ||||
| 1.  xxxx | ||||
| 2.  xxxx | ||||
| 3.  xxxx | ||||
|  | ||||
| #### Instructions | ||||
|  | ||||
| 1.  xxxx | ||||
| 2.  xxxx | ||||
| 3.  xxxx | ||||
|  | ||||
| #### Contribution | ||||
|  | ||||
| 1.  Fork the repository | ||||
| 2.  Create Feat_xxx branch | ||||
| 3.  Commit your code | ||||
| 4.  Create Pull Request | ||||
|  | ||||
|  | ||||
| #### Gitee Feature | ||||
|  | ||||
| 1.  You can use Readme\_XXX.md to support different languages, such as Readme\_en.md, Readme\_zh.md | ||||
| 2.  Gitee blog [blog.gitee.com](https://blog.gitee.com) | ||||
| 3.  Explore open source project [https://gitee.com/explore](https://gitee.com/explore) | ||||
| 4.  The most valuable open source project [GVP](https://gitee.com/gvp) | ||||
| 5.  The manual of Gitee [https://gitee.com/help](https://gitee.com/help) | ||||
| 6.  The most popular members  [https://gitee.com/gitee-stars/](https://gitee.com/gitee-stars/) | ||||
							
								
								
									
										174
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										174
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,174 @@ | ||||
| ## 平台简介 | ||||
| [](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus) | ||||
| [](https://github.com/JavaLionLi/RuoYi-Vue-Plus) | ||||
| [](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/blob/master/LICENSE) | ||||
| [](https://www.jetbrains.com/?from=RuoYi-Vue-Plus) | ||||
| <br> | ||||
| [](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus) | ||||
| []() | ||||
| []() | ||||
| []() | ||||
|  | ||||
| 基于 RuoYi-Vue 集成 Mybatis-Plus Lombok Hutool 等便捷开发工具 适配重写相关业务 便于开发 定期与 RuoYi-Vue 同步 | ||||
| * 前端开发框架 Vue、Element UI | ||||
| * 后端开发框架 Spring Boot、Redis | ||||
| * 容器框架 Undertow 基于 Netty 的高性能容器 | ||||
| * 权限认证框架 Spring Security、Jwt,支持多终端认证系统 | ||||
| * 关系数据库 MySQL 适配 8.X  | ||||
| * 缓存数据库 Redis 适配 6.X | ||||
| * 数据库开发框架 Mybatis-Plus 快速 CRUD 增加开发效率 插件化支持各类需求 | ||||
| * 网络框架 Feign、OkHttp3 接口化管理 HTTP 请求 | ||||
| * 工具类框架 Hutool、Lombok 减少代码冗余 增加安全性 | ||||
| * 监控框架 spring-boot-admin 全方位服务监控 | ||||
| * 校验框架 validation 增强接口安全性 严谨性 | ||||
| * 文档框架 knife4j 美化接口文档 | ||||
| * 序列化框架 统一使用 jackson 高效可靠 | ||||
| * 代码生成器 一键生成前后端代码 | ||||
| * 多数据源框架 dynamic-datasource 支持主从与多种类数据库异构 | ||||
| * Redis客户端 采用 Redisson 性能更强 | ||||
| * 分布式锁 Lock4j 注解锁、工具锁 多种多样 | ||||
|  | ||||
| ## 参考文档 | ||||
|  | ||||
| 使用框架前请仔细阅读文档重点注意事项 | ||||
| <br> | ||||
| >[初始化项目 必看](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/关于初始化项目?sort_id=4164117) | ||||
| >  | ||||
| >[参考文档 Wiki](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages) | ||||
|  | ||||
| ## 提问四部曲 | ||||
| ### 一、查阅wiki | ||||
| 优先在`wiki->重点事项`,查找是否有相关问题及解决方案,尤其是框架更新后产生的问题,多会在wiki中提及 | ||||
|  | ||||
| > [参考文档 Wiki](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages) | ||||
|  | ||||
| ### 二、借助issues | ||||
| 尝试issues中搜索问题关键字(记得选择已完成),看看是否有其他人提出相同问题 | ||||
| - `如果有`那么依据评论中的解决方案自行尝试解决 | ||||
| - `如果没有`那么提交一个新的issues描述清楚你的问题,需要包含以下内容(优质的issues,能够帮助作者更高效的帮你解决问题): | ||||
|     - 出现问题的模块或功能或类,总之你要说清楚在哪出的问题 | ||||
|     - 描述产生问题的相关操作流程,以便复现快速解决 | ||||
|     - 报错的日志截图,一定是截图,不要复制一堆报错的文本 | ||||
| > [issues](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/issues) | ||||
|  | ||||
| ### 三、百度 | ||||
| 大家都懂,不多描述,将关键的报错信息CC->CV到百度中看看大佬们怎么解决的 | ||||
| > [百度](http://www.baidu.com) | ||||
|  | ||||
| ### 四、加群 | ||||
| 以上三点已经能解决大家绝大部分问题了,如果还有问题没能通过这几种方式解决,那么加群,大家一起在群里探讨一下 | ||||
|  | ||||
| ## 修改RuoYi功能 | ||||
| ### 依赖改动 | ||||
|  | ||||
| * ORM框架 使用 Mybatis-Plus 简化CRUD (不支持主子表) | ||||
| * Bean简化 使用 Lombok 简化 get set toString 等等 | ||||
| * 容器改动 Tomcat 改为 并发性能更好的 undertow | ||||
| * 分页移除 pagehelper 改为 Mybatis-Plus 分页 | ||||
| * 升级 swagger 为 knife4j | ||||
| * 集成 Hutool 5.X 并重写RuoYi部分功能 | ||||
| * 集成 Feign 接口化管理 Http 请求(如三方请求 支付,短信,推送等) | ||||
| * 移除 自带服务监控 改为 spring-boot-admin 全方位监控 | ||||
| * 增加 demo 模块示例(给不会增加模块的小伙伴做参考) | ||||
| * 增加 redisson 高性能 Redis 客户端 | ||||
| * 移除 fastjson 统一使用 jackson 序列化 | ||||
| * 集成 dynamic-datasource 多数据源(默认支持MySQL,其他种类需自行适配) | ||||
| * 集成 Lock4j 实现分布式 注解锁、工具锁 多种多样 | ||||
|  | ||||
| ### 代码改动 | ||||
|  | ||||
| * 所有原生功能使用 Mybatis-Plus 与 Lombok 重写 | ||||
| * 增加 IServicePlus 与 BaseMapperPlus 可自定义通用方法 | ||||
| * 代码生成模板 改为适配 Mybatis-Plus 的代码 | ||||
| * 代码生成模板 拆分出 Vo,QueryBo,AddBo,EditBo 等领域对象 | ||||
| * 代码生成模板 增加 文档注解 与 校验注解 简化通用操作 | ||||
| * 项目修改为 maven多环境配置 | ||||
| * 项目配置修改为 application.yml 统一管理 | ||||
| * 数据权限修改为 适配支持单表、多表 | ||||
| * 使用 redisson 实现 spring-cache 整合 | ||||
| * 增加 mybatis-plus 二级缓存 redis 存储 | ||||
|  | ||||
| ### 其他 | ||||
|  | ||||
| * 同步升级 RuoYi-Vue 3.5.0 | ||||
| * GitHub 地址 [RuoYi-Vue-Plus-github](https://github.com/JavaLionLi/RuoYi-Vue-Plus) | ||||
| * 单模块 fast 分支 [RuoYi-Vue-Plus-fast](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/tree/fast/) | ||||
| * Oracle 模块 oracle 分支 [RuoYi-Vue-Plus-oracle](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/tree/oracle/) | ||||
|  | ||||
| ## 扫码加群 一起交流 | ||||
|  | ||||
|  | ||||
| ## 捐献作者 | ||||
| 作者为兼职做开源,平时还需要工作,如果帮到了您可以请作者吃个盒饭   | ||||
| <img src="https://images.gitee.com/uploads/images/2021/0525/101654_451e4523_1766278.jpeg" width="300px" height="450px" /> | ||||
| <img src="https://images.gitee.com/uploads/images/2021/0525/101713_3d18b119_1766278.jpeg" width="300px" height="450px" /> | ||||
|  | ||||
| ## 内置功能 | ||||
|  | ||||
| 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找出系统性能瓶颈。 | ||||
|  | ||||
| ## 演示图例 | ||||
|  | ||||
| <table border="1" cellpadding="1" cellspacing="1" style="width:500px"> | ||||
| 	<tbody> | ||||
| 		<tr> | ||||
| 			<td><img src="https://oscimg.oschina.net/oscnet/up-972235bcbe3518dedd351ff0e2ee7d1031c.png" width="1920" /></td> | ||||
| 			<td><img src="https://oscimg.oschina.net/oscnet/up-5e0097702fa91e2e36391de8127676a7fa1.png" width="1920" /></td> | ||||
| 		</tr> | ||||
| 		<tr> | ||||
| 			<td> | ||||
| 			<p><img src="https://oscimg.oschina.net/oscnet/up-e56e3828f48cd9886d88731766f06d5f3c1.png" width="1920" /></p> | ||||
| 			</td> | ||||
| 			<td><img src="https://oscimg.oschina.net/oscnet/up-0715990ea1a9f254ec2138fcd063c1f556a.png" width="1920" /></td> | ||||
| 		</tr> | ||||
| 		<tr> | ||||
| 			<td><img src="https://oscimg.oschina.net/oscnet/up-eaf5417ccf921bb64abb959e3d8e290467f.png" width="1920" /></td> | ||||
| 			<td><img src="https://oscimg.oschina.net/oscnet/up-fc285cf33095ebf8318de6999af0f473861.png" width="1920" /></td> | ||||
| 		</tr> | ||||
| 		<tr> | ||||
| 			<td><img src="https://oscimg.oschina.net/oscnet/up-60c83fd8bd61c29df6dbf47c88355e9c272.png" width="1920" /></td> | ||||
| 			<td><img src="https://oscimg.oschina.net/oscnet/up-7f731948c8b73c7d90f67f9e1c7a534d5c3.png" width="1920" /></td> | ||||
| 		</tr> | ||||
| 		<tr> | ||||
| 			<td><img src="https://oscimg.oschina.net/oscnet/up-e4de89b5e2d20c52d3c3a47f9eb88eb8526.png" width="1920" /></td> | ||||
| 			<td><img src="https://oscimg.oschina.net/oscnet/up-8791d823a508eb90e67c604f36f57491a67.png" width="1920" /></td> | ||||
| 		</tr> | ||||
| 		<tr> | ||||
| 			<td><img src="https://oscimg.oschina.net/oscnet/up-4589afd99982ead331785299b894174feb6.png" width="1920" /></td> | ||||
| 			<td><img src="https://oscimg.oschina.net/oscnet/up-8ea177cdacaea20995daf2f596b15232561.png" width="1920" /></td> | ||||
| 		</tr> | ||||
| 		<tr> | ||||
| 			<td><img src="https://oscimg.oschina.net/oscnet/up-32d1d04c55c11f74c9129fbbc58399728c4.png" width="1920" /></td> | ||||
| 			<td><img src="https://oscimg.oschina.net/oscnet/up-04fa118f7631b7ae6fd72299ca0a1430a63.png" width="1920" /></td> | ||||
| 		</tr> | ||||
| 		<tr> | ||||
| 			<td><img src="https://oscimg.oschina.net/oscnet/up-fe7e85b65827802bfaadf3acd42568b58c7.png" width="1920" /></td> | ||||
| 			<td><img src="https://oscimg.oschina.net/oscnet/up-eff2b02a54f8188022d8498cfe6af6fcc06.png" width="1920" /></td> | ||||
| 		</tr> | ||||
| 	</tbody> | ||||
| </table> | ||||
|  | ||||
| ## 在线体验 | ||||
|  | ||||
| - admin/admin123   | ||||
| - 陆陆续续收到一些打赏,为了更好的体验已用于演示服务器升级。谢谢各位小伙伴。 | ||||
|  | ||||
| 演示地址:http://vue.ruoyi.vip   | ||||
| 文档地址:http://doc.ruoyi.vip | ||||
							
								
								
									
										12
									
								
								bin/clean.bat
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								bin/clean.bat
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,12 @@ | ||||
| @echo off | ||||
| echo. | ||||
| echo [<5B><>Ϣ] <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>·<EFBFBD><C2B7><EFBFBD><EFBFBD> | ||||
| echo. | ||||
|  | ||||
| %~d0 | ||||
| cd %~dp0 | ||||
|  | ||||
| cd .. | ||||
| call mvn clean | ||||
|  | ||||
| pause | ||||
							
								
								
									
										12
									
								
								bin/package.bat
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								bin/package.bat
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,12 @@ | ||||
| @echo off | ||||
| echo. | ||||
| echo [<5B><>Ϣ] <20><><EFBFBD><EFBFBD>Web<65><62><EFBFBD>̣<EFBFBD><CCA3><EFBFBD><EFBFBD><EFBFBD>war/jar<61><72><EFBFBD>ļ<EFBFBD><C4BC><EFBFBD> | ||||
| echo. | ||||
|  | ||||
| %~d0 | ||||
| cd %~dp0 | ||||
|  | ||||
| cd .. | ||||
| call mvn clean package -Dmaven.test.skip=true | ||||
|  | ||||
| pause | ||||
							
								
								
									
										14
									
								
								bin/run.bat
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								bin/run.bat
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,14 @@ | ||||
| @echo off | ||||
| echo. | ||||
| echo [<5B><>Ϣ] <20><><EFBFBD><EFBFBD>Web<65><62><EFBFBD>̡<EFBFBD> | ||||
| echo. | ||||
|  | ||||
| cd %~dp0 | ||||
| cd ../ruoyi-admin/target | ||||
|  | ||||
| set JAVA_OPTS=-Xms256m -Xmx1024m -XX:MetaspaceSize=128m -XX:MaxMetaspaceSize=512m | ||||
|  | ||||
| java -jar %JAVA_OPTS% ruoyi-admin.jar | ||||
|  | ||||
| cd bin | ||||
| pause | ||||
							
								
								
									
										
											BIN
										
									
								
								doc/若依环境使用手册.docx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								doc/若依环境使用手册.docx
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										320
									
								
								pom.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										320
									
								
								pom.xml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,320 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project xmlns="http://maven.apache.org/POM/4.0.0" | ||||
|          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" | ||||
|          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||||
| 	<modelVersion>4.0.0</modelVersion> | ||||
|  | ||||
|     <groupId>com.ruoyi</groupId> | ||||
|     <artifactId>ruoyi-vue-plus</artifactId> | ||||
|     <version>${ruoyi-vue-plus.version}</version> | ||||
|  | ||||
|     <name>old-tress-sys</name> | ||||
|     <url>https://gitee.com/JavaLionLi/RuoYi-Vue-Plus</url> | ||||
|     <description>RuoYi-Vue-Plus后台管理系统</description> | ||||
|  | ||||
|     <properties> | ||||
|         <ruoyi-vue-plus.version>2.4.0</ruoyi-vue-plus.version> | ||||
|         <spring-boot.version>2.4.7</spring-boot.version> | ||||
|         <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> | ||||
|         <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding> | ||||
|         <java.version>1.8</java.version> | ||||
|         <maven-jar-plugin.version>3.1.1</maven-jar-plugin.version> | ||||
|         <druid.version>1.2.6</druid.version> | ||||
|         <knife4j.version>3.0.2</knife4j.version> | ||||
|         <poi.version>4.1.2</poi.version> | ||||
|         <velocity.version>1.7</velocity.version> | ||||
|         <jwt.version>0.9.1</jwt.version> | ||||
|         <mybatis-plus.version>3.4.3</mybatis-plus.version> | ||||
|         <hutool.version>5.7.2</hutool.version> | ||||
|         <feign.version>3.0.3</feign.version> | ||||
|         <feign-okhttp.version>11.0</feign-okhttp.version> | ||||
|         <spring-boot-admin.version>2.4.1</spring-boot-admin.version> | ||||
|         <redisson.version>3.15.2</redisson.version> | ||||
|         <lock4j.version>2.2.1</lock4j.version> | ||||
|         <datasource.version>3.4.0</datasource.version> | ||||
|     </properties> | ||||
|  | ||||
|     <!-- 依赖声明 --> | ||||
|     <dependencyManagement> | ||||
|         <dependencies> | ||||
|  | ||||
|             <!-- SpringBoot的依赖配置--> | ||||
|             <dependency> | ||||
|                 <groupId>org.springframework.boot</groupId> | ||||
|                 <artifactId>spring-boot-dependencies</artifactId> | ||||
|                 <version>${spring-boot.version}</version> | ||||
|                 <type>pom</type> | ||||
|                 <scope>import</scope> | ||||
|             </dependency> | ||||
|  | ||||
|             <!-- 阿里数据库连接池 --> | ||||
|             <dependency> | ||||
|                 <groupId>com.alibaba</groupId> | ||||
|                 <artifactId>druid-spring-boot-starter</artifactId> | ||||
|                 <version>${druid.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <dependency> | ||||
|                 <groupId>com.github.xiaoymin</groupId> | ||||
|                 <artifactId>knife4j-spring-boot-starter</artifactId> | ||||
|                 <version>${knife4j.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <!-- excel工具 --> | ||||
|             <dependency> | ||||
|                 <groupId>org.apache.poi</groupId> | ||||
|                 <artifactId>poi-ooxml</artifactId> | ||||
|                 <version>${poi.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <!-- velocity代码生成使用模板 --> | ||||
|             <dependency> | ||||
|                 <groupId>org.apache.velocity</groupId> | ||||
|                 <artifactId>velocity</artifactId> | ||||
|                 <version>${velocity.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <!-- Token生成与解析--> | ||||
|             <dependency> | ||||
|                 <groupId>io.jsonwebtoken</groupId> | ||||
|                 <artifactId>jjwt</artifactId> | ||||
|                 <version>${jwt.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <!-- dynamic-datasource 多数据源--> | ||||
|             <dependency> | ||||
|                 <groupId>com.baomidou</groupId> | ||||
|                 <artifactId>dynamic-datasource-spring-boot-starter</artifactId> | ||||
|                 <version>${datasource.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <dependency> | ||||
|                 <groupId>com.baomidou</groupId> | ||||
|                 <artifactId>mybatis-plus-boot-starter</artifactId> | ||||
|                 <version>${mybatis-plus.version}</version> | ||||
|             </dependency> | ||||
|             <dependency> | ||||
|                 <groupId>com.baomidou</groupId> | ||||
|                 <artifactId>mybatis-plus-extension</artifactId> | ||||
|                 <version>${mybatis-plus.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <dependency> | ||||
|                 <groupId>cn.hutool</groupId> | ||||
|                 <artifactId>hutool-all</artifactId> | ||||
|                 <version>${hutool.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <dependency> | ||||
|                 <groupId>org.springframework.cloud</groupId> | ||||
|                 <artifactId>spring-cloud-starter-openfeign</artifactId> | ||||
|                 <version>${feign.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <dependency> | ||||
|                 <groupId>io.github.openfeign</groupId> | ||||
|                 <artifactId>feign-okhttp</artifactId> | ||||
|                 <version>${feign-okhttp.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <dependency> | ||||
|                 <groupId>de.codecentric</groupId> | ||||
|                 <artifactId>spring-boot-admin-starter-server</artifactId> | ||||
|                 <version>${spring-boot-admin.version}</version> | ||||
|             </dependency> | ||||
|             <dependency> | ||||
|                 <groupId>de.codecentric</groupId> | ||||
|                 <artifactId>spring-boot-admin-starter-client</artifactId> | ||||
|                 <version>${spring-boot-admin.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <!--redisson--> | ||||
|             <dependency> | ||||
|                 <groupId>org.redisson</groupId> | ||||
|                 <artifactId>redisson-spring-boot-starter</artifactId> | ||||
|                 <version>${redisson.version}</version> | ||||
|             </dependency> | ||||
|             <dependency> | ||||
|                 <groupId>com.baomidou</groupId> | ||||
|                 <artifactId>lock4j-redisson-spring-boot-starter</artifactId> | ||||
|                 <version>${lock4j.version}</version> | ||||
|             </dependency> | ||||
|             <!-- 定时任务--> | ||||
|             <dependency> | ||||
|                 <groupId>com.ruoyi</groupId> | ||||
|                 <artifactId>ruoyi-quartz</artifactId> | ||||
|                 <version>${ruoyi-vue-plus.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <!-- 代码生成--> | ||||
|             <dependency> | ||||
|                 <groupId>com.ruoyi</groupId> | ||||
|                 <artifactId>ruoyi-generator</artifactId> | ||||
|                 <version>${ruoyi-vue-plus.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <!-- 核心模块--> | ||||
|             <dependency> | ||||
|                 <groupId>com.ruoyi</groupId> | ||||
|                 <artifactId>ruoyi-framework</artifactId> | ||||
|                 <version>${ruoyi-vue-plus.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <!-- 系统模块--> | ||||
|             <dependency> | ||||
|                 <groupId>com.ruoyi</groupId> | ||||
|                 <artifactId>ruoyi-system</artifactId> | ||||
|                 <version>${ruoyi-vue-plus.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <!-- 通用工具--> | ||||
|             <dependency> | ||||
|                 <groupId>com.ruoyi</groupId> | ||||
|                 <artifactId>ruoyi-common</artifactId> | ||||
|                 <version>${ruoyi-vue-plus.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <!-- demo模块 --> | ||||
| <!--            <dependency>--> | ||||
| <!--                <groupId>com.ruoyi</groupId>--> | ||||
| <!--                <artifactId>ruoyi-demo</artifactId>--> | ||||
| <!--                <version>${ruoyi-vue-plus.version}</version>--> | ||||
| <!--            </dependency>--> | ||||
|  | ||||
|         </dependencies> | ||||
|     </dependencyManagement> | ||||
|  | ||||
|     <modules> | ||||
|         <module>ruoyi-admin</module> | ||||
|         <module>ruoyi-framework</module> | ||||
|         <module>ruoyi-system</module> | ||||
|         <module>ruoyi-quartz</module> | ||||
|         <module>ruoyi-generator</module> | ||||
|         <module>ruoyi-common</module> | ||||
| <!--        <module>ruoyi-demo</module>--> | ||||
|     </modules> | ||||
|     <packaging>pom</packaging> | ||||
|  | ||||
|  | ||||
|     <dependencies> | ||||
|  | ||||
|     </dependencies> | ||||
|  | ||||
|     <build> | ||||
|         <plugins> | ||||
|             <plugin> | ||||
|                 <groupId>org.apache.maven.plugins</groupId> | ||||
|                 <artifactId>maven-compiler-plugin</artifactId> | ||||
|                 <version>3.1</version> | ||||
|                 <configuration> | ||||
|                     <source>${java.version}</source> | ||||
|                     <target>${java.version}</target> | ||||
|                     <encoding>${project.build.sourceEncoding}</encoding> | ||||
|                 </configuration> | ||||
|             </plugin> | ||||
|         </plugins> | ||||
|         <resources> | ||||
|             <resource> | ||||
|                 <!--打包该目录下的 application.yml --> | ||||
|                 <directory>src/main/resources</directory> | ||||
|                 <!-- 启用过滤 即该资源中的变量将会被过滤器中的值替换 --> | ||||
|                 <filtering>true</filtering> | ||||
|             </resource> | ||||
|         </resources> | ||||
|     </build> | ||||
|  | ||||
|     <repositories> | ||||
|         <repository> | ||||
|             <id>public</id> | ||||
|             <name>aliyun nexus</name> | ||||
|             <url>http://maven.aliyun.com/nexus/content/groups/public/</url> | ||||
|             <releases> | ||||
|                 <enabled>true</enabled> | ||||
|             </releases> | ||||
|         </repository> | ||||
|     </repositories> | ||||
|  | ||||
|     <pluginRepositories> | ||||
|         <pluginRepository> | ||||
|             <id>public</id> | ||||
|             <name>aliyun nexus</name> | ||||
|             <url>http://maven.aliyun.com/nexus/content/groups/public/</url> | ||||
|             <releases> | ||||
|                 <enabled>true</enabled> | ||||
|             </releases> | ||||
|             <snapshots> | ||||
|                 <enabled>false</enabled> | ||||
|             </snapshots> | ||||
|         </pluginRepository> | ||||
|     </pluginRepositories> | ||||
|  | ||||
| <!--    <profiles>--> | ||||
| <!--        <profile>--> | ||||
| <!--            <id>local</id>--> | ||||
| <!--            <properties>--> | ||||
| <!--                <!– 环境标识,需要与配置文件的名称相对应 –>--> | ||||
| <!--                <profiles.active>local</profiles.active>--> | ||||
| <!--                <logging.level>debug</logging.level>--> | ||||
| <!--            </properties>--> | ||||
| <!--        </profile>--> | ||||
| <!--        <profile>--> | ||||
| <!--            <id>dev</id>--> | ||||
| <!--            <properties>--> | ||||
| <!--                <!– 环境标识,需要与配置文件的名称相对应 –>--> | ||||
| <!--                <profiles.active>dev</profiles.active>--> | ||||
| <!--                <logging.level>debug</logging.level>--> | ||||
| <!--            </properties>--> | ||||
| <!--            <activation>--> | ||||
| <!--                <!– 默认环境 –>--> | ||||
| <!--                <activeByDefault>true</activeByDefault>--> | ||||
| <!--            </activation>--> | ||||
| <!--        </profile>--> | ||||
| <!--        <profile>--> | ||||
| <!--            <id>prod</id>--> | ||||
| <!--            <properties>--> | ||||
| <!--                <profiles.active>prod</profiles.active>--> | ||||
| <!--                <logging.level>warn</logging.level>--> | ||||
| <!--            </properties>--> | ||||
| <!--        </profile>--> | ||||
|  | ||||
| <!--        <!– jdk多版本配置 –>--> | ||||
| <!--        <profile>--> | ||||
| <!--            <id>jdk8</id>--> | ||||
| <!--            <activation>--> | ||||
| <!--                <activeByDefault>true</activeByDefault>--> | ||||
| <!--                <jdk>1.8</jdk>--> | ||||
| <!--            </activation>--> | ||||
| <!--            <properties>--> | ||||
| <!--                <java.version>1.8</java.version>--> | ||||
| <!--            </properties>--> | ||||
| <!--        </profile>--> | ||||
| <!--        <profile>--> | ||||
| <!--            <id>jdk11</id>--> | ||||
| <!--            <activation>--> | ||||
| <!--                <jdk>11</jdk>--> | ||||
| <!--            </activation>--> | ||||
| <!--            <properties>--> | ||||
| <!--                <java.version>11</java.version>--> | ||||
| <!--                <jaxb.version>3.0.1</jaxb.version>--> | ||||
| <!--            </properties>--> | ||||
| <!--            <dependencyManagement>--> | ||||
| <!--                <dependencies>--> | ||||
| <!--                    <!– jdk11 缺失依赖 jaxb–>--> | ||||
| <!--                    <dependency>--> | ||||
| <!--                        <groupId>com.sun.xml.bind</groupId>--> | ||||
| <!--                        <artifactId>jaxb-impl</artifactId>--> | ||||
| <!--                        <version>${jaxb.version}</version>--> | ||||
| <!--                    </dependency>--> | ||||
| <!--                </dependencies>--> | ||||
| <!--            </dependencyManagement>--> | ||||
| <!--            <dependencies>--> | ||||
| <!--                <!–jaxb–>--> | ||||
| <!--                <dependency>--> | ||||
| <!--                    <groupId>com.sun.xml.bind</groupId>--> | ||||
| <!--                    <artifactId>jaxb-impl</artifactId>--> | ||||
| <!--                </dependency>--> | ||||
| <!--            </dependencies>--> | ||||
| <!--        </profile>--> | ||||
| <!--    </profiles>--> | ||||
|  | ||||
| </project> | ||||
							
								
								
									
										89
									
								
								ruoyi-admin/pom.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								ruoyi-admin/pom.xml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,89 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project xmlns="http://maven.apache.org/POM/4.0.0" | ||||
|          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" | ||||
|          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||||
|     <parent> | ||||
|         <artifactId>ruoyi-vue-plus</artifactId> | ||||
|         <groupId>com.ruoyi</groupId> | ||||
|         <version>${ruoyi-vue-plus.version}</version> | ||||
|     </parent> | ||||
|     <modelVersion>4.0.0</modelVersion> | ||||
|     <packaging>jar</packaging> | ||||
|     <artifactId>ruoyi-admin</artifactId> | ||||
|  | ||||
|     <description> | ||||
|         web服务入口 | ||||
|     </description> | ||||
|  | ||||
|     <dependencies> | ||||
|  | ||||
|         <!-- spring-boot-devtools --> | ||||
|         <dependency> | ||||
|             <groupId>org.springframework.boot</groupId> | ||||
|             <artifactId>spring-boot-devtools</artifactId> | ||||
|             <optional>true</optional> <!-- 表示依赖不会传递 --> | ||||
|         </dependency> | ||||
|  | ||||
|          <!-- Mysql驱动包 --> | ||||
|         <dependency> | ||||
|             <groupId>mysql</groupId> | ||||
|             <artifactId>mysql-connector-java</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- 核心模块--> | ||||
|         <dependency> | ||||
|             <groupId>com.ruoyi</groupId> | ||||
|             <artifactId>ruoyi-framework</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- 定时任务--> | ||||
|         <dependency> | ||||
|             <groupId>com.ruoyi</groupId> | ||||
|             <artifactId>ruoyi-quartz</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- 代码生成--> | ||||
|         <dependency> | ||||
|             <groupId>com.ruoyi</groupId> | ||||
|             <artifactId>ruoyi-generator</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!--  demo模块  --> | ||||
| <!--        <dependency>--> | ||||
| <!--            <groupId>com.ruoyi</groupId>--> | ||||
| <!--            <artifactId>ruoyi-demo</artifactId>--> | ||||
| <!--        </dependency>--> | ||||
|  | ||||
|     </dependencies> | ||||
|  | ||||
|     <build> | ||||
|         <finalName>${project.artifactId}</finalName> | ||||
|         <plugins> | ||||
|             <plugin> | ||||
|                 <groupId>org.springframework.boot</groupId> | ||||
|                 <artifactId>spring-boot-maven-plugin</artifactId> | ||||
|                 <version>${spring-boot.version}</version> | ||||
|                 <configuration> | ||||
|                     <fork>true</fork> <!-- 如果没有该配置,devtools不会生效 --> | ||||
|                 </configuration> | ||||
|                 <executions> | ||||
|                     <execution> | ||||
|                         <goals> | ||||
|                             <goal>repackage</goal> | ||||
|                         </goals> | ||||
|                     </execution> | ||||
|                 </executions> | ||||
|             </plugin> | ||||
|             <plugin> | ||||
|                 <groupId>org.apache.maven.plugins</groupId> | ||||
|                 <artifactId>maven-war-plugin</artifactId> | ||||
|                 <version>3.1.0</version> | ||||
|                 <configuration> | ||||
|                     <failOnMissingWebXml>false</failOnMissingWebXml> | ||||
|                     <warName>${project.artifactId}</warName> | ||||
|                 </configuration> | ||||
|            </plugin> | ||||
|         </plugins> | ||||
|     </build> | ||||
|  | ||||
| </project> | ||||
							
								
								
									
										21
									
								
								ruoyi-admin/src/main/java/com/ruoyi/RuoYiApplication.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								ruoyi-admin/src/main/java/com/ruoyi/RuoYiApplication.java
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | ||||
| package com.ruoyi; | ||||
|  | ||||
| import org.springframework.boot.SpringApplication; | ||||
| import org.springframework.boot.autoconfigure.SpringBootApplication; | ||||
|  | ||||
| /** | ||||
|  * 启动程序 | ||||
|  * | ||||
|  * @author ruoyi | ||||
|  */ | ||||
|  | ||||
| @SpringBootApplication | ||||
| public class RuoYiApplication | ||||
| { | ||||
|     public static void main(String[] args) | ||||
|     { | ||||
|         System.setProperty("spring.devtools.restart.enabled", "false"); | ||||
|         SpringApplication.run(RuoYiApplication.class, args); | ||||
|         System.out.println("(♥◠‿◠)ノ゙  RuoYi-Vue-Plus启动成功   ლ(´ڡ`ლ)゙"); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,19 @@ | ||||
| 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); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,120 @@ | ||||
| package com.ruoyi.web.controller.common; | ||||
|  | ||||
| import cn.hutool.captcha.AbstractCaptcha; | ||||
| import cn.hutool.captcha.CircleCaptcha; | ||||
| import cn.hutool.captcha.LineCaptcha; | ||||
| import cn.hutool.captcha.ShearCaptcha; | ||||
| import cn.hutool.captcha.generator.CodeGenerator; | ||||
| import cn.hutool.captcha.generator.RandomGenerator; | ||||
| import cn.hutool.core.convert.Convert; | ||||
| import cn.hutool.core.util.IdUtil; | ||||
| import cn.hutool.core.util.StrUtil; | ||||
| import com.ruoyi.common.constant.Constants; | ||||
| import com.ruoyi.common.core.domain.AjaxResult; | ||||
| import com.ruoyi.common.core.redis.RedisCache; | ||||
| import com.ruoyi.framework.captcha.UnsignedMathGenerator; | ||||
| import com.ruoyi.framework.config.properties.CaptchaProperties; | ||||
| import org.springframework.beans.factory.annotation.Autowired; | ||||
| import org.springframework.web.bind.annotation.GetMapping; | ||||
| import org.springframework.web.bind.annotation.RestController; | ||||
|  | ||||
| import javax.annotation.Resource; | ||||
| import java.util.HashMap; | ||||
| import java.util.Map; | ||||
| import java.util.concurrent.TimeUnit; | ||||
|  | ||||
| /** | ||||
|  * 验证码操作处理 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @RestController | ||||
| public class CaptchaController { | ||||
|  | ||||
| 	// 圆圈干扰验证码 | ||||
| 	@Resource(name = "CircleCaptcha") | ||||
| 	private CircleCaptcha circleCaptcha; | ||||
| 	// 线段干扰的验证码 | ||||
| 	@Resource(name = "LineCaptcha") | ||||
| 	private LineCaptcha lineCaptcha; | ||||
| 	// 扭曲干扰验证码 | ||||
| 	@Resource(name = "ShearCaptcha") | ||||
| 	private ShearCaptcha shearCaptcha; | ||||
|  | ||||
| 	@Autowired | ||||
| 	private RedisCache redisCache; | ||||
|  | ||||
| 	@Autowired | ||||
| 	private CaptchaProperties captchaProperties; | ||||
|  | ||||
| 	/** | ||||
| 	 * 生成验证码 | ||||
| 	 */ | ||||
| 	@GetMapping("/captchaImage") | ||||
| 	public AjaxResult getCode() { | ||||
| 		Map<String, Object> ajax = new HashMap<>(); | ||||
| 		Boolean enabled = captchaProperties.getEnabled(); | ||||
| 		ajax.put("enabled", enabled); | ||||
| 		if (!enabled) { | ||||
| 			return AjaxResult.success(ajax); | ||||
| 		} | ||||
| 		// 保存验证码信息 | ||||
| 		String uuid = IdUtil.simpleUUID(); | ||||
| 		String verifyKey = Constants.CAPTCHA_CODE_KEY + uuid; | ||||
| 		String code = null; | ||||
| 		// 生成验证码 | ||||
| 		CodeGenerator codeGenerator; | ||||
| 		AbstractCaptcha captcha; | ||||
| 		switch (captchaProperties.getType()) { | ||||
| 			case "math": | ||||
| 				codeGenerator = new UnsignedMathGenerator(captchaProperties.getNumberLength()); | ||||
| 				break; | ||||
| 			case "char": | ||||
| 				codeGenerator = new RandomGenerator(captchaProperties.getCharLength()); | ||||
| 				break; | ||||
| 			default: | ||||
| 				throw new IllegalArgumentException("验证码类型异常"); | ||||
| 		} | ||||
| 		switch (captchaProperties.getCategory()) { | ||||
| 			case "line": | ||||
| 				captcha = lineCaptcha; | ||||
| 				break; | ||||
| 			case "circle": | ||||
| 				captcha = circleCaptcha; | ||||
| 				break; | ||||
| 			case "shear": | ||||
| 				captcha = shearCaptcha; | ||||
| 				break; | ||||
| 			default: | ||||
| 				throw new IllegalArgumentException("验证码类别异常"); | ||||
| 		} | ||||
| 		captcha.setGenerator(codeGenerator); | ||||
| 		captcha.createCode(); | ||||
| 		if ("math".equals(captchaProperties.getType())) { | ||||
| 			code = getCodeResult(captcha.getCode()); | ||||
| 		} else if ("char".equals(captchaProperties.getType())) { | ||||
| 			code = captcha.getCode(); | ||||
| 		} | ||||
| 		redisCache.setCacheObject(verifyKey, code, Constants.CAPTCHA_EXPIRATION, TimeUnit.MINUTES); | ||||
| 		ajax.put("uuid", uuid); | ||||
| 		ajax.put("img", captcha.getImageBase64()); | ||||
| 		return AjaxResult.success(ajax); | ||||
| 	} | ||||
|  | ||||
| 	private String getCodeResult(String capStr) { | ||||
| 		int numberLength = captchaProperties.getNumberLength(); | ||||
| 		int a = Convert.toInt(StrUtil.sub(capStr, 0, numberLength).trim()); | ||||
| 		char operator = capStr.charAt(numberLength); | ||||
| 		int b = Convert.toInt(StrUtil.sub(capStr, numberLength + 1, numberLength + 1 + numberLength).trim()); | ||||
| 		switch (operator) { | ||||
| 			case '*': | ||||
| 				return a * b + ""; | ||||
| 			case '+': | ||||
| 				return a + b + ""; | ||||
| 			case '-': | ||||
| 				return a - b + ""; | ||||
| 			default: | ||||
| 				return ""; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| @ -0,0 +1,123 @@ | ||||
| package com.ruoyi.web.controller.common; | ||||
|  | ||||
| import cn.hutool.core.util.StrUtil; | ||||
| import com.ruoyi.common.config.RuoYiConfig; | ||||
| import com.ruoyi.common.constant.Constants; | ||||
| import com.ruoyi.common.core.domain.AjaxResult; | ||||
| 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.GetMapping; | ||||
| import org.springframework.web.bind.annotation.PostMapping; | ||||
| import org.springframework.web.bind.annotation.RestController; | ||||
| import org.springframework.web.multipart.MultipartFile; | ||||
|  | ||||
| import javax.servlet.http.HttpServletRequest; | ||||
| import javax.servlet.http.HttpServletResponse; | ||||
| import java.io.File; | ||||
| import java.util.HashMap; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * 通用请求处理 | ||||
|  * | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| @RestController | ||||
| public class CommonController | ||||
| { | ||||
|     private static final Logger log = LoggerFactory.getLogger(CommonController.class); | ||||
|  | ||||
|     @Autowired | ||||
|     private ServerConfig serverConfig; | ||||
|  | ||||
|     /** | ||||
|      * 通用下载请求 | ||||
|      * | ||||
|      * @param fileName 文件名称 | ||||
|      * @param delete 是否删除 | ||||
|      */ | ||||
|     @GetMapping("common/download") | ||||
|     public void fileDownload(String fileName, Boolean delete, HttpServletResponse response, HttpServletRequest request) | ||||
|     { | ||||
|         try | ||||
|         { | ||||
|             if (!FileUtils.checkAllowDownload(fileName)) | ||||
|             { | ||||
|                 throw new Exception(StrUtil.format("文件名称({})非法,不允许下载。 ", fileName)); | ||||
|             } | ||||
|             String realFileName = System.currentTimeMillis() + fileName.substring(fileName.indexOf("_") + 1); | ||||
|             String filePath = RuoYiConfig.getDownloadPath() + fileName; | ||||
| 			File file = new File(filePath); | ||||
|             response.setContentType(MediaType.APPLICATION_OCTET_STREAM_VALUE); | ||||
|             FileUtils.setAttachmentResponseHeader(response, realFileName); | ||||
| 			FileUtils.writeToStream(file, response.getOutputStream()); | ||||
|             if (delete) | ||||
|             { | ||||
| 				FileUtils.del(file); | ||||
|             } | ||||
|         } | ||||
|         catch (Exception e) | ||||
|         { | ||||
|             log.error("下载文件失败", e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 通用上传请求 | ||||
|      */ | ||||
|     @PostMapping("/common/upload") | ||||
|     public AjaxResult uploadFile(MultipartFile file) throws Exception | ||||
|     { | ||||
|         try | ||||
|         { | ||||
|             // 上传文件路径 | ||||
|             String filePath = RuoYiConfig.getUploadPath(); | ||||
|             // 上传并返回新文件名称 | ||||
|             String fileName = FileUploadUtils.upload(filePath, file); | ||||
|             String url = serverConfig.getUrl() + fileName; | ||||
| 			Map<String,Object> ajax = new HashMap<>(); | ||||
|             ajax.put("fileName", fileName); | ||||
|             ajax.put("url", url); | ||||
|             return AjaxResult.success(ajax); | ||||
|         } | ||||
|         catch (Exception e) | ||||
|         { | ||||
|             return AjaxResult.error(e.getMessage()); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 本地资源通用下载 | ||||
|      */ | ||||
|     @GetMapping("/common/download/resource") | ||||
|     public void resourceDownload(String resource, HttpServletRequest request, HttpServletResponse response) | ||||
|             throws Exception | ||||
|     { | ||||
|         try | ||||
|         { | ||||
|             if (!FileUtils.checkAllowDownload(resource)) | ||||
|             { | ||||
|                 throw new Exception(StrUtil.format("资源文件({})非法,不允许下载。 ", resource)); | ||||
|             } | ||||
|             // 本地资源路径 | ||||
|             String localPath = RuoYiConfig.getProfile(); | ||||
|             // 数据库资源地址 | ||||
|             String downloadPath = localPath + StrUtil.subAfter(resource, Constants.RESOURCE_PREFIX,false); | ||||
|             // 下载名称 | ||||
|             String downloadName = StrUtil.subAfter(downloadPath, "/",true); | ||||
|             response.setContentType(MediaType.APPLICATION_OCTET_STREAM_VALUE); | ||||
| 			File file = new File(downloadPath); | ||||
|             FileUtils.setAttachmentResponseHeader(response, downloadName); | ||||
|             FileUtils.writeToStream(file, response.getOutputStream()); | ||||
|         } | ||||
|         catch (Exception e) | ||||
|         { | ||||
|             log.error("下载文件失败", e); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,50 @@ | ||||
| package com.ruoyi.web.controller.monitor; | ||||
|  | ||||
| import cn.hutool.core.util.StrUtil; | ||||
| import com.ruoyi.common.core.domain.AjaxResult; | ||||
| 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.GetMapping; | ||||
| import org.springframework.web.bind.annotation.RequestMapping; | ||||
| import org.springframework.web.bind.annotation.RestController; | ||||
|  | ||||
| import java.util.*; | ||||
|  | ||||
| /** | ||||
|  * 缓存监控 | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| @RestController | ||||
| @RequestMapping("/monitor/cache") | ||||
| public class CacheController | ||||
| { | ||||
|     @Autowired | ||||
|     private RedisTemplate<String, String> redisTemplate; | ||||
|  | ||||
|     @PreAuthorize("@ss.hasPermi('monitor:cache:list')") | ||||
|     @GetMapping() | ||||
|     public AjaxResult getInfo() throws Exception | ||||
|     { | ||||
|         Properties info = (Properties) redisTemplate.execute((RedisCallback<Object>) connection -> connection.info()); | ||||
|         Properties commandStats = (Properties) redisTemplate.execute((RedisCallback<Object>) connection -> connection.info("commandstats")); | ||||
|         Object dbSize = redisTemplate.execute((RedisCallback<Object>) connection -> connection.dbSize()); | ||||
|  | ||||
|         Map<String, Object> result = new HashMap<>(3); | ||||
|         result.put("info", info); | ||||
|         result.put("dbSize", dbSize); | ||||
|  | ||||
|         List<Map<String, String>> pieList = new ArrayList<>(); | ||||
|         commandStats.stringPropertyNames().forEach(key -> { | ||||
|             Map<String, String> data = new HashMap<>(2); | ||||
|             String property = commandStats.getProperty(key); | ||||
|             data.put("name", StrUtil.removePrefix(key, "cmdstat_")); | ||||
|             data.put("value", StrUtil.subBetween(property, "calls=", ",usec")); | ||||
|             pieList.add(data); | ||||
|         }); | ||||
|         result.put("commandStats", pieList); | ||||
|         return AjaxResult.success(result); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,62 @@ | ||||
| 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.SysLogininfor; | ||||
| import com.ruoyi.system.service.ISysLogininforService; | ||||
| import org.springframework.beans.factory.annotation.Autowired; | ||||
| import org.springframework.security.access.prepost.PreAuthorize; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| /** | ||||
|  * 系统访问记录 | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| @RestController | ||||
| @RequestMapping("/monitor/logininfor") | ||||
| public class SysLogininforController extends BaseController | ||||
| { | ||||
|     @Autowired | ||||
|     private ISysLogininforService logininforService; | ||||
|  | ||||
|     @PreAuthorize("@ss.hasPermi('monitor:logininfor:list')") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo list(SysLogininfor logininfor) | ||||
|     { | ||||
|         return logininforService.selectPageLogininforList(logininfor); | ||||
|     } | ||||
|  | ||||
|     @Log(title = "登录日志", businessType = BusinessType.EXPORT) | ||||
|     @PreAuthorize("@ss.hasPermi('monitor:logininfor:export')") | ||||
|     @GetMapping("/export") | ||||
|     public AjaxResult export(SysLogininfor logininfor) | ||||
|     { | ||||
|         List<SysLogininfor> list = logininforService.selectLogininforList(logininfor); | ||||
|         ExcelUtil<SysLogininfor> util = new ExcelUtil<SysLogininfor>(SysLogininfor.class); | ||||
|         return util.exportExcel(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 AjaxResult.success(); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,62 @@ | ||||
| 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 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) | ||||
|     { | ||||
|         return operLogService.selectPageOperLogList(operLog); | ||||
|     } | ||||
|  | ||||
|     @Log(title = "操作日志", businessType = BusinessType.EXPORT) | ||||
|     @PreAuthorize("@ss.hasPermi('monitor:operlog:export')") | ||||
|     @GetMapping("/export") | ||||
|     public AjaxResult export(SysOperLog operLog) | ||||
|     { | ||||
|         List<SysOperLog> list = operLogService.selectOperLogList(operLog); | ||||
|         ExcelUtil<SysOperLog> util = new ExcelUtil<SysOperLog>(SysOperLog.class); | ||||
|         return util.exportExcel(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 AjaxResult.success(); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,91 @@ | ||||
| package com.ruoyi.web.controller.monitor; | ||||
|  | ||||
| import cn.hutool.core.lang.Validator; | ||||
| import cn.hutool.core.util.StrUtil; | ||||
| 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.domain.model.LoginUser; | ||||
| import com.ruoyi.common.core.page.TableDataInfo; | ||||
| import com.ruoyi.common.core.redis.RedisCache; | ||||
| import com.ruoyi.common.enums.BusinessType; | ||||
| import com.ruoyi.common.utils.PageUtils; | ||||
| import com.ruoyi.system.domain.SysUserOnline; | ||||
| import com.ruoyi.system.service.ISysUserOnlineService; | ||||
| import org.springframework.beans.factory.annotation.Autowired; | ||||
| import org.springframework.security.access.prepost.PreAuthorize; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.Collection; | ||||
| import java.util.Collections; | ||||
| import java.util.List; | ||||
|  | ||||
| /** | ||||
|  * 在线用户监控 | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| @RestController | ||||
| @RequestMapping("/monitor/online") | ||||
| public class SysUserOnlineController extends BaseController | ||||
| { | ||||
|     @Autowired | ||||
|     private ISysUserOnlineService userOnlineService; | ||||
|  | ||||
|     @Autowired | ||||
|     private RedisCache redisCache; | ||||
|  | ||||
|     @PreAuthorize("@ss.hasPermi('monitor:online:list')") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo list(String ipaddr, String userName) | ||||
|     { | ||||
|         Collection<String> keys = redisCache.keys(Constants.LOGIN_TOKEN_KEY + "*"); | ||||
|         List<SysUserOnline> userOnlineList = new ArrayList<SysUserOnline>(); | ||||
|         for (String key : keys) | ||||
|         { | ||||
|             LoginUser user = redisCache.getCacheObject(key); | ||||
|             if (Validator.isNotEmpty(ipaddr) && Validator.isNotEmpty(userName)) | ||||
|             { | ||||
|                 if (StrUtil.equals(ipaddr, user.getIpaddr()) && StrUtil.equals(userName, user.getUsername())) | ||||
|                 { | ||||
|                     userOnlineList.add(userOnlineService.selectOnlineByInfo(ipaddr, userName, user)); | ||||
|                 } | ||||
|             } | ||||
|             else if (Validator.isNotEmpty(ipaddr)) | ||||
|             { | ||||
|                 if (StrUtil.equals(ipaddr, user.getIpaddr())) | ||||
|                 { | ||||
|                     userOnlineList.add(userOnlineService.selectOnlineByIpaddr(ipaddr, user)); | ||||
|                 } | ||||
|             } | ||||
|             else if (Validator.isNotEmpty(userName) && Validator.isNotNull(user.getUser())) | ||||
|             { | ||||
|                 if (StrUtil.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 PageUtils.buildDataInfo(userOnlineList); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 强退用户 | ||||
|      */ | ||||
|     @PreAuthorize("@ss.hasPermi('monitor:online:forceLogout')") | ||||
|     @Log(title = "在线用户", businessType = BusinessType.FORCE) | ||||
|     @DeleteMapping("/{tokenId}") | ||||
|     public AjaxResult forceLogout(@PathVariable String tokenId) | ||||
|     { | ||||
|         redisCache.deleteObject(Constants.LOGIN_TOKEN_KEY + tokenId); | ||||
|         return AjaxResult.success(); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,128 @@ | ||||
| package com.ruoyi.web.controller.system; | ||||
|  | ||||
| import com.ruoyi.common.annotation.Log; | ||||
| import com.ruoyi.common.annotation.RepeatSubmit; | ||||
| 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.SecurityUtils; | ||||
| 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 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) | ||||
|     { | ||||
|         return configService.selectPageConfigList(config); | ||||
|     } | ||||
|  | ||||
|     @Log(title = "参数管理", businessType = BusinessType.EXPORT) | ||||
|     @PreAuthorize("@ss.hasPermi('system:config:export')") | ||||
|     @GetMapping("/export") | ||||
|     public AjaxResult export(SysConfig config) | ||||
|     { | ||||
|         List<SysConfig> list = configService.selectConfigList(config); | ||||
|         ExcelUtil<SysConfig> util = new ExcelUtil<SysConfig>(SysConfig.class); | ||||
|         return util.exportExcel(list, "参数数据"); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据参数编号获取详细信息 | ||||
|      */ | ||||
|     @PreAuthorize("@ss.hasPermi('system:config:query')") | ||||
|     @GetMapping(value = "/{configId}") | ||||
|     public AjaxResult getInfo(@PathVariable Long configId) | ||||
|     { | ||||
|         return AjaxResult.success(configService.selectConfigById(configId)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据参数键名查询参数值 | ||||
|      */ | ||||
|     @GetMapping(value = "/configKey/{configKey}") | ||||
|     public AjaxResult getConfigKey(@PathVariable String configKey) | ||||
|     { | ||||
|         return AjaxResult.success(configService.selectConfigByKey(configKey)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 新增参数配置 | ||||
|      */ | ||||
|     @PreAuthorize("@ss.hasPermi('system:config:add')") | ||||
|     @Log(title = "参数管理", businessType = BusinessType.INSERT) | ||||
|     @PostMapping | ||||
|     @RepeatSubmit | ||||
|     public AjaxResult add(@Validated @RequestBody SysConfig config) | ||||
|     { | ||||
|         if (UserConstants.NOT_UNIQUE.equals(configService.checkConfigKeyUnique(config))) | ||||
|         { | ||||
|             return AjaxResult.error("新增参数'" + config.getConfigName() + "'失败,参数键名已存在"); | ||||
|         } | ||||
|         config.setCreateBy(SecurityUtils.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 AjaxResult.error("修改参数'" + config.getConfigName() + "'失败,参数键名已存在"); | ||||
|         } | ||||
|         config.setUpdateBy(SecurityUtils.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 AjaxResult.success(); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,159 @@ | ||||
| package com.ruoyi.web.controller.system; | ||||
|  | ||||
| import cn.hutool.core.util.StrUtil; | ||||
| 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.SecurityUtils; | ||||
| 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.HashMap; | ||||
| import java.util.Iterator; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * 部门信息 | ||||
|  * | ||||
|  * @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<SysDept> depts = deptService.selectDeptList(dept); | ||||
|         return AjaxResult.success(depts); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 查询部门列表(排除节点) | ||||
|      */ | ||||
|     @PreAuthorize("@ss.hasPermi('system:dept:list')") | ||||
|     @GetMapping("/list/exclude/{deptId}") | ||||
|     public AjaxResult excludeChild(@PathVariable(value = "deptId", required = false) Long deptId) | ||||
|     { | ||||
|         List<SysDept> depts = deptService.selectDeptList(new SysDept()); | ||||
|         Iterator<SysDept> it = depts.iterator(); | ||||
|         while (it.hasNext()) | ||||
|         { | ||||
|             SysDept d = (SysDept) it.next(); | ||||
|             if (d.getDeptId().intValue() == deptId | ||||
|                     || ArrayUtils.contains(StrUtil.splitToArray(d.getAncestors(), ","), deptId + "")) | ||||
|             { | ||||
|                 it.remove(); | ||||
|             } | ||||
|         } | ||||
|         return AjaxResult.success(depts); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据部门编号获取详细信息 | ||||
|      */ | ||||
|     @PreAuthorize("@ss.hasPermi('system:dept:query')") | ||||
|     @GetMapping(value = "/{deptId}") | ||||
|     public AjaxResult getInfo(@PathVariable Long deptId) | ||||
|     { | ||||
|         return AjaxResult.success(deptService.selectDeptById(deptId)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取部门下拉树列表 | ||||
|      */ | ||||
|     @GetMapping("/treeselect") | ||||
|     public AjaxResult treeselect(SysDept dept) | ||||
|     { | ||||
|         List<SysDept> depts = deptService.selectDeptList(dept); | ||||
|         return AjaxResult.success(deptService.buildDeptTreeSelect(depts)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 加载对应角色部门列表树 | ||||
|      */ | ||||
|     @GetMapping(value = "/roleDeptTreeselect/{roleId}") | ||||
|     public AjaxResult roleDeptTreeselect(@PathVariable("roleId") Long roleId) | ||||
|     { | ||||
|         List<SysDept> depts = deptService.selectDeptList(new SysDept()); | ||||
| 		Map<String,Object> ajax = new HashMap<>(); | ||||
|         ajax.put("checkedKeys", deptService.selectDeptListByRoleId(roleId)); | ||||
|         ajax.put("depts", deptService.buildDeptTreeSelect(depts)); | ||||
|         return AjaxResult.success(ajax); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 新增部门 | ||||
|      */ | ||||
|     @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 AjaxResult.error("新增部门'" + dept.getDeptName() + "'失败,部门名称已存在"); | ||||
|         } | ||||
|         dept.setCreateBy(SecurityUtils.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) | ||||
|     { | ||||
|         if (UserConstants.NOT_UNIQUE.equals(deptService.checkDeptNameUnique(dept))) | ||||
|         { | ||||
|             return AjaxResult.error("修改部门'" + dept.getDeptName() + "'失败,部门名称已存在"); | ||||
|         } | ||||
|         else if (dept.getParentId().equals(dept.getDeptId())) | ||||
|         { | ||||
|             return AjaxResult.error("修改部门'" + dept.getDeptName() + "'失败,上级部门不能是自己"); | ||||
|         } | ||||
|         else if (StrUtil.equals(UserConstants.DEPT_DISABLE, dept.getStatus()) | ||||
|                 && deptService.selectNormalChildrenDeptById(dept.getDeptId()) > 0) | ||||
|         { | ||||
|             return AjaxResult.error("该部门包含未停用的子部门!"); | ||||
|         } | ||||
|         dept.setUpdateBy(SecurityUtils.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 AjaxResult.error("存在下级部门,不允许删除"); | ||||
|         } | ||||
|         if (deptService.checkDeptExistUser(deptId)) | ||||
|         { | ||||
|             return AjaxResult.error("部门存在用户,不允许删除"); | ||||
|         } | ||||
|         return toAjax(deptService.deleteDeptById(deptId)); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,113 @@ | ||||
| package com.ruoyi.web.controller.system; | ||||
|  | ||||
| import cn.hutool.core.lang.Validator; | ||||
| 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.SecurityUtils; | ||||
| 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 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) | ||||
|     { | ||||
|         return dictDataService.selectPageDictDataList(dictData); | ||||
|     } | ||||
|  | ||||
|     @Log(title = "字典数据", businessType = BusinessType.EXPORT) | ||||
|     @PreAuthorize("@ss.hasPermi('system:dict:export')") | ||||
|     @GetMapping("/export") | ||||
|     public AjaxResult export(SysDictData dictData) | ||||
|     { | ||||
|         List<SysDictData> list = dictDataService.selectDictDataList(dictData); | ||||
|         ExcelUtil<SysDictData> util = new ExcelUtil<SysDictData>(SysDictData.class); | ||||
|         return util.exportExcel(list, "字典数据"); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 查询字典数据详细 | ||||
|      */ | ||||
|     @PreAuthorize("@ss.hasPermi('system:dict:query')") | ||||
|     @GetMapping(value = "/{dictCode}") | ||||
|     public AjaxResult getInfo(@PathVariable Long dictCode) | ||||
|     { | ||||
|         return AjaxResult.success(dictDataService.selectDictDataById(dictCode)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据字典类型查询字典数据信息 | ||||
|      */ | ||||
|     @GetMapping(value = "/type/{dictType}") | ||||
|     public AjaxResult dictType(@PathVariable String dictType) | ||||
|     { | ||||
|         List<SysDictData> data = dictTypeService.selectDictDataByType(dictType); | ||||
|         if (Validator.isNull(data)) | ||||
|         { | ||||
|             data = new ArrayList<SysDictData>(); | ||||
|         } | ||||
|         return AjaxResult.success(data); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 新增字典类型 | ||||
|      */ | ||||
|     @PreAuthorize("@ss.hasPermi('system:dict:add')") | ||||
|     @Log(title = "字典数据", businessType = BusinessType.INSERT) | ||||
|     @PostMapping | ||||
|     public AjaxResult add(@Validated @RequestBody SysDictData dict) | ||||
|     { | ||||
|         dict.setCreateBy(SecurityUtils.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(SecurityUtils.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(); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,124 @@ | ||||
| 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.SecurityUtils; | ||||
| 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 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) | ||||
|     { | ||||
|         return dictTypeService.selectPageDictTypeList(dictType); | ||||
|     } | ||||
|  | ||||
|     @Log(title = "字典类型", businessType = BusinessType.EXPORT) | ||||
|     @PreAuthorize("@ss.hasPermi('system:dict:export')") | ||||
|     @GetMapping("/export") | ||||
|     public AjaxResult export(SysDictType dictType) | ||||
|     { | ||||
|         List<SysDictType> list = dictTypeService.selectDictTypeList(dictType); | ||||
|         ExcelUtil<SysDictType> util = new ExcelUtil<SysDictType>(SysDictType.class); | ||||
|         return util.exportExcel(list, "字典类型"); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 查询字典类型详细 | ||||
|      */ | ||||
|     @PreAuthorize("@ss.hasPermi('system:dict:query')") | ||||
|     @GetMapping(value = "/{dictId}") | ||||
|     public AjaxResult getInfo(@PathVariable Long dictId) | ||||
|     { | ||||
|         return AjaxResult.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 AjaxResult.error("新增字典'" + dict.getDictName() + "'失败,字典类型已存在"); | ||||
|         } | ||||
|         dict.setCreateBy(SecurityUtils.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 AjaxResult.error("修改字典'" + dict.getDictName() + "'失败,字典类型已存在"); | ||||
|         } | ||||
|         dict.setUpdateBy(SecurityUtils.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 AjaxResult.success(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取字典选择框列表 | ||||
|      */ | ||||
|     @GetMapping("/optionselect") | ||||
|     public AjaxResult optionselect() | ||||
|     { | ||||
|         List<SysDictType> dictTypes = dictTypeService.selectDictTypeAll(); | ||||
|         return AjaxResult.success(dictTypes); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,97 @@ | ||||
| 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.core.domain.model.LoginUser; | ||||
| import com.ruoyi.common.utils.ServletUtils; | ||||
| import com.ruoyi.framework.web.service.SysLoginService; | ||||
| import com.ruoyi.framework.web.service.SysPermissionService; | ||||
| import com.ruoyi.framework.web.service.TokenService; | ||||
| 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.HashMap; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.Set; | ||||
|  | ||||
| /** | ||||
|  * 登录验证 | ||||
|  * | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| @RestController | ||||
| public class SysLoginController | ||||
| { | ||||
|     @Autowired | ||||
|     private SysLoginService loginService; | ||||
|  | ||||
|     @Autowired | ||||
|     private ISysMenuService menuService; | ||||
|  | ||||
|     @Autowired | ||||
|     private SysPermissionService permissionService; | ||||
|  | ||||
|     @Autowired | ||||
|     private TokenService tokenService; | ||||
|  | ||||
|     /** | ||||
|      * 登录方法 | ||||
|      * | ||||
|      * @param loginBody 登录信息 | ||||
|      * @return 结果 | ||||
|      */ | ||||
|     @PostMapping("/login") | ||||
|     public AjaxResult login(@RequestBody LoginBody loginBody) | ||||
|     { | ||||
| 		Map<String,Object> ajax = new HashMap<>(); | ||||
|         // 生成令牌 | ||||
|         String token = loginService.login(loginBody.getUsername(), loginBody.getPassword(), loginBody.getCode(), | ||||
|                 loginBody.getUuid()); | ||||
|         ajax.put(Constants.TOKEN, token); | ||||
|         return AjaxResult.success(ajax); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取用户信息 | ||||
|      * | ||||
|      * @return 用户信息 | ||||
|      */ | ||||
|     @GetMapping("getInfo") | ||||
|     public AjaxResult getInfo() | ||||
|     { | ||||
|         LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest()); | ||||
|         SysUser user = loginUser.getUser(); | ||||
|         // 角色集合 | ||||
|         Set<String> roles = permissionService.getRolePermission(user); | ||||
|         // 权限集合 | ||||
|         Set<String> permissions = permissionService.getMenuPermission(user); | ||||
| 		Map<String,Object> ajax = new HashMap<>(); | ||||
|         ajax.put("user", user); | ||||
|         ajax.put("roles", roles); | ||||
|         ajax.put("permissions", permissions); | ||||
|         return AjaxResult.success(ajax); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取路由信息 | ||||
|      * | ||||
|      * @return 路由信息 | ||||
|      */ | ||||
|     @GetMapping("getRouters") | ||||
|     public AjaxResult getRouters() | ||||
|     { | ||||
|         LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest()); | ||||
|         // 用户信息 | ||||
|         SysUser user = loginUser.getUser(); | ||||
|         List<SysMenu> menus = menuService.selectMenuTreeByUserId(user.getUserId()); | ||||
|         return AjaxResult.success(menuService.buildMenus(menus)); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,153 @@ | ||||
| package com.ruoyi.web.controller.system; | ||||
|  | ||||
| import cn.hutool.core.util.StrUtil; | ||||
| import com.ruoyi.common.annotation.Log; | ||||
| import com.ruoyi.common.constant.Constants; | ||||
| 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.SysMenu; | ||||
| 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.ServletUtils; | ||||
| import com.ruoyi.framework.web.service.TokenService; | ||||
| import com.ruoyi.system.service.ISysMenuService; | ||||
| 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.HashMap; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * 菜单信息 | ||||
|  * | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| @RestController | ||||
| @RequestMapping("/system/menu") | ||||
| public class SysMenuController extends BaseController | ||||
| { | ||||
|     @Autowired | ||||
|     private ISysMenuService menuService; | ||||
|  | ||||
|     @Autowired | ||||
|     private TokenService tokenService; | ||||
|  | ||||
|     /** | ||||
|      * 获取菜单列表 | ||||
|      */ | ||||
|     @PreAuthorize("@ss.hasPermi('system:menu:list')") | ||||
|     @GetMapping("/list") | ||||
|     public AjaxResult list(SysMenu menu) | ||||
|     { | ||||
|         LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest()); | ||||
|         Long userId = loginUser.getUser().getUserId(); | ||||
|         List<SysMenu> menus = menuService.selectMenuList(menu, userId); | ||||
|         return AjaxResult.success(menus); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据菜单编号获取详细信息 | ||||
|      */ | ||||
|     @PreAuthorize("@ss.hasPermi('system:menu:query')") | ||||
|     @GetMapping(value = "/{menuId}") | ||||
|     public AjaxResult getInfo(@PathVariable Long menuId) | ||||
|     { | ||||
|         return AjaxResult.success(menuService.selectMenuById(menuId)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取菜单下拉树列表 | ||||
|      */ | ||||
|     @GetMapping("/treeselect") | ||||
|     public AjaxResult treeselect(SysMenu menu) | ||||
|     { | ||||
|         LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest()); | ||||
|         Long userId = loginUser.getUser().getUserId(); | ||||
|         List<SysMenu> menus = menuService.selectMenuList(menu, userId); | ||||
|         return AjaxResult.success(menuService.buildMenuTreeSelect(menus)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 加载对应角色菜单列表树 | ||||
|      */ | ||||
|     @GetMapping(value = "/roleMenuTreeselect/{roleId}") | ||||
|     public AjaxResult roleMenuTreeselect(@PathVariable("roleId") Long roleId) | ||||
|     { | ||||
|         LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest()); | ||||
|         List<SysMenu> menus = menuService.selectMenuList(loginUser.getUser().getUserId()); | ||||
| 		Map<String,Object> ajax = new HashMap<>(); | ||||
|         ajax.put("checkedKeys", menuService.selectMenuListByRoleId(roleId)); | ||||
|         ajax.put("menus", menuService.buildMenuTreeSelect(menus)); | ||||
|         return AjaxResult.success(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 AjaxResult.error("新增菜单'" + menu.getMenuName() + "'失败,菜单名称已存在"); | ||||
|         } | ||||
|         else if (UserConstants.YES_FRAME.equals(menu.getIsFrame()) | ||||
|                 && !StrUtil.startWithAny(menu.getPath(), Constants.HTTP, Constants.HTTPS)) | ||||
|         { | ||||
|             return AjaxResult.error("新增菜单'" + menu.getMenuName() + "'失败,地址必须以http(s)://开头"); | ||||
|         } | ||||
|         menu.setCreateBy(SecurityUtils.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 AjaxResult.error("修改菜单'" + menu.getMenuName() + "'失败,菜单名称已存在"); | ||||
|         } | ||||
|         else if (UserConstants.YES_FRAME.equals(menu.getIsFrame()) | ||||
|                 && !StrUtil.startWithAny(menu.getPath(), Constants.HTTP, Constants.HTTPS)) | ||||
|         { | ||||
|             return AjaxResult.error("修改菜单'" + menu.getMenuName() + "'失败,地址必须以http(s)://开头"); | ||||
|         } | ||||
|         else if (menu.getMenuId().equals(menu.getParentId())) | ||||
|         { | ||||
|             return AjaxResult.error("修改菜单'" + menu.getMenuName() + "'失败,上级菜单不能选择自己"); | ||||
|         } | ||||
|         menu.setUpdateBy(SecurityUtils.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 AjaxResult.error("存在子菜单,不允许删除"); | ||||
|         } | ||||
|         if (menuService.checkMenuExistRole(menuId)) | ||||
|         { | ||||
|             return AjaxResult.error("菜单已分配,不允许删除"); | ||||
|         } | ||||
|         return toAjax(menuService.deleteMenuById(menuId)); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,90 @@ | ||||
| package com.ruoyi.web.controller.system; | ||||
|  | ||||
| import java.util.List; | ||||
| 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.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.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.SecurityUtils; | ||||
| import com.ruoyi.system.domain.SysNotice; | ||||
| import com.ruoyi.system.service.ISysNoticeService; | ||||
|  | ||||
| /** | ||||
|  * 公告 信息操作处理 | ||||
|  *  | ||||
|  * @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) | ||||
|     { | ||||
|         return noticeService.selectPageNoticeList(notice); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据通知公告编号获取详细信息 | ||||
|      */ | ||||
|     @PreAuthorize("@ss.hasPermi('system:notice:query')") | ||||
|     @GetMapping(value = "/{noticeId}") | ||||
|     public AjaxResult getInfo(@PathVariable Long noticeId) | ||||
|     { | ||||
|         return AjaxResult.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(SecurityUtils.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(SecurityUtils.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)); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,128 @@ | ||||
| package com.ruoyi.web.controller.system; | ||||
|  | ||||
| import java.util.List; | ||||
| 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.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.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.SecurityUtils; | ||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | ||||
| import com.ruoyi.system.domain.SysPost; | ||||
| import com.ruoyi.system.service.ISysPostService; | ||||
|  | ||||
| /** | ||||
|  * 岗位信息操作处理 | ||||
|  *  | ||||
|  * @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) | ||||
|     { | ||||
|         return postService.selectPagePostList(post); | ||||
|     } | ||||
|      | ||||
|     @Log(title = "岗位管理", businessType = BusinessType.EXPORT) | ||||
|     @PreAuthorize("@ss.hasPermi('system:post:export')") | ||||
|     @GetMapping("/export") | ||||
|     public AjaxResult export(SysPost post) | ||||
|     { | ||||
|         List<SysPost> list = postService.selectPostList(post); | ||||
|         ExcelUtil<SysPost> util = new ExcelUtil<SysPost>(SysPost.class); | ||||
|         return util.exportExcel(list, "岗位数据"); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据岗位编号获取详细信息 | ||||
|      */ | ||||
|     @PreAuthorize("@ss.hasPermi('system:post:query')") | ||||
|     @GetMapping(value = "/{postId}") | ||||
|     public AjaxResult getInfo(@PathVariable Long postId) | ||||
|     { | ||||
|         return AjaxResult.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 AjaxResult.error("新增岗位'" + post.getPostName() + "'失败,岗位名称已存在"); | ||||
|         } | ||||
|         else if (UserConstants.NOT_UNIQUE.equals(postService.checkPostCodeUnique(post))) | ||||
|         { | ||||
|             return AjaxResult.error("新增岗位'" + post.getPostName() + "'失败,岗位编码已存在"); | ||||
|         } | ||||
|         post.setCreateBy(SecurityUtils.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 AjaxResult.error("修改岗位'" + post.getPostName() + "'失败,岗位名称已存在"); | ||||
|         } | ||||
|         else if (UserConstants.NOT_UNIQUE.equals(postService.checkPostCodeUnique(post))) | ||||
|         { | ||||
|             return AjaxResult.error("修改岗位'" + post.getPostName() + "'失败,岗位编码已存在"); | ||||
|         } | ||||
|         post.setUpdateBy(SecurityUtils.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<SysPost> posts = postService.selectPostAll(); | ||||
|         return AjaxResult.success(posts); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,137 @@ | ||||
| package com.ruoyi.web.controller.system; | ||||
|  | ||||
| import cn.hutool.core.util.StrUtil; | ||||
| 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.ServletUtils; | ||||
| import com.ruoyi.common.utils.file.FileUploadUtils; | ||||
| 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; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.util.HashMap; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * 个人信息 业务处理 | ||||
|  * | ||||
|  * @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 = tokenService.getLoginUser(ServletUtils.getRequest()); | ||||
|         SysUser user = loginUser.getUser(); | ||||
| 		Map<String,Object> ajax = new HashMap<>(); | ||||
| 		ajax.put("user", user); | ||||
|         ajax.put("roleGroup", userService.selectUserRoleGroup(loginUser.getUsername())); | ||||
|         ajax.put("postGroup", userService.selectUserPostGroup(loginUser.getUsername())); | ||||
|         return AjaxResult.success(ajax); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 修改用户 | ||||
|      */ | ||||
|     @Log(title = "个人信息", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping | ||||
|     public AjaxResult updateProfile(@RequestBody SysUser user) | ||||
|     { | ||||
|         if (StrUtil.isNotEmpty(user.getPhonenumber()) | ||||
|                 && UserConstants.NOT_UNIQUE.equals(userService.checkPhoneUnique(user))) | ||||
|         { | ||||
|             return AjaxResult.error("修改用户'" + user.getUserName() + "'失败,手机号码已存在"); | ||||
|         } | ||||
|         if (StrUtil.isNotEmpty(user.getEmail()) | ||||
|                 && UserConstants.NOT_UNIQUE.equals(userService.checkEmailUnique(user))) | ||||
|         { | ||||
|             return AjaxResult.error("修改用户'" + user.getUserName() + "'失败,邮箱账号已存在"); | ||||
|         } | ||||
|         if (userService.updateUserProfile(user) > 0) | ||||
|         { | ||||
|             LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest()); | ||||
|             // 更新缓存用户信息 | ||||
|             loginUser.getUser().setNickName(user.getNickName()); | ||||
|             loginUser.getUser().setPhonenumber(user.getPhonenumber()); | ||||
|             loginUser.getUser().setEmail(user.getEmail()); | ||||
|             loginUser.getUser().setSex(user.getSex()); | ||||
|             tokenService.setLoginUser(loginUser); | ||||
|             return AjaxResult.success(); | ||||
|         } | ||||
|         return AjaxResult.error("修改个人信息异常,请联系管理员"); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 重置密码 | ||||
|      */ | ||||
|     @Log(title = "个人信息", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping("/updatePwd") | ||||
|     public AjaxResult updatePwd(String oldPassword, String newPassword) | ||||
|     { | ||||
|         LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest()); | ||||
|         String userName = loginUser.getUsername(); | ||||
|         String password = loginUser.getPassword(); | ||||
|         if (!SecurityUtils.matchesPassword(oldPassword, password)) | ||||
|         { | ||||
|             return AjaxResult.error("修改密码失败,旧密码错误"); | ||||
|         } | ||||
|         if (SecurityUtils.matchesPassword(newPassword, password)) | ||||
|         { | ||||
|             return AjaxResult.error("新密码不能与旧密码相同"); | ||||
|         } | ||||
|         if (userService.resetUserPwd(userName, SecurityUtils.encryptPassword(newPassword)) > 0) | ||||
|         { | ||||
|             // 更新缓存用户密码 | ||||
|             loginUser.getUser().setPassword(SecurityUtils.encryptPassword(newPassword)); | ||||
|             tokenService.setLoginUser(loginUser); | ||||
|             return AjaxResult.success(); | ||||
|         } | ||||
|         return AjaxResult.error("修改密码异常,请联系管理员"); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 头像上传 | ||||
|      */ | ||||
|     @Log(title = "用户头像", businessType = BusinessType.UPDATE) | ||||
|     @PostMapping("/avatar") | ||||
|     public AjaxResult avatar(@RequestParam("avatarfile") MultipartFile file) throws IOException | ||||
|     { | ||||
|         if (!file.isEmpty()) | ||||
|         { | ||||
|             LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest()); | ||||
|             String avatar = FileUploadUtils.upload(RuoYiConfig.getAvatarPath(), file); | ||||
|             if (userService.updateUserAvatar(loginUser.getUsername(), avatar)) | ||||
|             { | ||||
| 				Map<String,Object> ajax = new HashMap<>(); | ||||
|                 ajax.put("imgUrl", avatar); | ||||
|                 // 更新缓存用户头像 | ||||
|                 loginUser.getUser().setAvatar(avatar); | ||||
|                 tokenService.setLoginUser(loginUser); | ||||
|                 return AjaxResult.success(ajax); | ||||
|             } | ||||
|         } | ||||
|         return AjaxResult.error("上传图片异常,请联系管理员"); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,174 @@ | ||||
| package com.ruoyi.web.controller.system; | ||||
|  | ||||
| import cn.hutool.core.lang.Validator; | ||||
| 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.SysRole; | ||||
| 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.SecurityUtils; | ||||
| import com.ruoyi.common.utils.ServletUtils; | ||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | ||||
| import com.ruoyi.framework.web.service.SysPermissionService; | ||||
| import com.ruoyi.framework.web.service.TokenService; | ||||
| import com.ruoyi.system.service.ISysRoleService; | ||||
| import com.ruoyi.system.service.ISysUserService; | ||||
| 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/role") | ||||
| public class SysRoleController extends BaseController | ||||
| { | ||||
|     @Autowired | ||||
|     private ISysRoleService roleService; | ||||
|  | ||||
|     @Autowired | ||||
|     private TokenService tokenService; | ||||
|  | ||||
|     @Autowired | ||||
|     private SysPermissionService permissionService; | ||||
|  | ||||
|     @Autowired | ||||
|     private ISysUserService userService; | ||||
|  | ||||
|     @PreAuthorize("@ss.hasPermi('system:role:list')") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo list(SysRole role) | ||||
|     { | ||||
|         return roleService.selectPageRoleList(role); | ||||
|     } | ||||
|  | ||||
|     @Log(title = "角色管理", businessType = BusinessType.EXPORT) | ||||
|     @PreAuthorize("@ss.hasPermi('system:role:export')") | ||||
|     @GetMapping("/export") | ||||
|     public AjaxResult export(SysRole role) | ||||
|     { | ||||
|         List<SysRole> list = roleService.selectRoleList(role); | ||||
|         ExcelUtil<SysRole> util = new ExcelUtil<SysRole>(SysRole.class); | ||||
|         return util.exportExcel(list, "角色数据"); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据角色编号获取详细信息 | ||||
|      */ | ||||
|     @PreAuthorize("@ss.hasPermi('system:role:query')") | ||||
|     @GetMapping(value = "/{roleId}") | ||||
|     public AjaxResult getInfo(@PathVariable Long roleId) | ||||
|     { | ||||
|         return AjaxResult.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 AjaxResult.error("新增角色'" + role.getRoleName() + "'失败,角色名称已存在"); | ||||
|         } | ||||
|         else if (UserConstants.NOT_UNIQUE.equals(roleService.checkRoleKeyUnique(role))) | ||||
|         { | ||||
|             return AjaxResult.error("新增角色'" + role.getRoleName() + "'失败,角色权限已存在"); | ||||
|         } | ||||
|         role.setCreateBy(SecurityUtils.getUsername()); | ||||
|         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); | ||||
|         if (UserConstants.NOT_UNIQUE.equals(roleService.checkRoleNameUnique(role))) | ||||
|         { | ||||
|             return AjaxResult.error("修改角色'" + role.getRoleName() + "'失败,角色名称已存在"); | ||||
|         } | ||||
|         else if (UserConstants.NOT_UNIQUE.equals(roleService.checkRoleKeyUnique(role))) | ||||
|         { | ||||
|             return AjaxResult.error("修改角色'" + role.getRoleName() + "'失败,角色权限已存在"); | ||||
|         } | ||||
|         role.setUpdateBy(SecurityUtils.getUsername()); | ||||
|  | ||||
|         if (roleService.updateRole(role) > 0) | ||||
|         { | ||||
|             // 更新缓存用户权限 | ||||
|             LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest()); | ||||
|             if (Validator.isNotNull(loginUser.getUser()) && !loginUser.getUser().isAdmin()) | ||||
|             { | ||||
|                 loginUser.setPermissions(permissionService.getMenuPermission(loginUser.getUser())); | ||||
|                 loginUser.setUser(userService.selectUserByUserName(loginUser.getUser().getUserName())); | ||||
|                 tokenService.setLoginUser(loginUser); | ||||
|             } | ||||
|             return AjaxResult.success(); | ||||
|         } | ||||
|         return AjaxResult.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); | ||||
|         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); | ||||
|         role.setUpdateBy(SecurityUtils.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 AjaxResult.success(roleService.selectRoleAll()); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,199 @@ | ||||
| package com.ruoyi.web.controller.system; | ||||
|  | ||||
| import cn.hutool.core.lang.Validator; | ||||
| 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.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.SecurityUtils; | ||||
| import com.ruoyi.common.utils.ServletUtils; | ||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | ||||
| import com.ruoyi.framework.web.service.TokenService; | ||||
| import com.ruoyi.system.service.ISysPostService; | ||||
| import com.ruoyi.system.service.ISysRoleService; | ||||
| import com.ruoyi.system.service.ISysUserService; | ||||
| 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 java.util.HashMap; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| 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 ISysPostService postService; | ||||
|  | ||||
|     @Autowired | ||||
|     private TokenService tokenService; | ||||
|  | ||||
|     /** | ||||
|      * 获取用户列表 | ||||
|      */ | ||||
|     @PreAuthorize("@ss.hasPermi('system:user:list')") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo list(SysUser user) | ||||
|     { | ||||
|         return userService.selectPageUserList(user); | ||||
|     } | ||||
|  | ||||
|     @Log(title = "用户管理", businessType = BusinessType.EXPORT) | ||||
|     @PreAuthorize("@ss.hasPermi('system:user:export')") | ||||
|     @GetMapping("/export") | ||||
|     public AjaxResult export(SysUser user) | ||||
|     { | ||||
|         List<SysUser> list = userService.selectUserList(user); | ||||
|         ExcelUtil<SysUser> util = new ExcelUtil<SysUser>(SysUser.class); | ||||
|         return util.exportExcel(list, "用户数据"); | ||||
|     } | ||||
|  | ||||
|     @Log(title = "用户管理", businessType = BusinessType.IMPORT) | ||||
|     @PreAuthorize("@ss.hasPermi('system:user:import')") | ||||
|     @PostMapping("/importData") | ||||
|     public AjaxResult importData(MultipartFile file, boolean updateSupport) throws Exception | ||||
|     { | ||||
|         ExcelUtil<SysUser> util = new ExcelUtil<SysUser>(SysUser.class); | ||||
|         List<SysUser> userList = util.importExcel(file.getInputStream()); | ||||
|         LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest()); | ||||
|         String operName = loginUser.getUsername(); | ||||
|         String message = userService.importUser(userList, updateSupport, operName); | ||||
|         return AjaxResult.success(message); | ||||
|     } | ||||
|  | ||||
|     @GetMapping("/importTemplate") | ||||
|     public AjaxResult importTemplate() | ||||
|     { | ||||
|         ExcelUtil<SysUser> util = new ExcelUtil<SysUser>(SysUser.class); | ||||
|         return util.importTemplateExcel("用户数据"); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据用户编号获取详细信息 | ||||
|      */ | ||||
|     @PreAuthorize("@ss.hasPermi('system:user:query')") | ||||
|     @GetMapping(value = { "/", "/{userId}" }) | ||||
|     public AjaxResult getInfo(@PathVariable(value = "userId", required = false) Long userId) | ||||
|     { | ||||
|         Map<String, Object> ajax = new HashMap<>(); | ||||
|         List<SysRole> 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 (Validator.isNotNull(userId)) | ||||
|         { | ||||
|             ajax.put("user", userService.selectUserById(userId)); | ||||
|             ajax.put("postIds", postService.selectPostListByUserId(userId)); | ||||
|             ajax.put("roleIds", roleService.selectRoleListByUserId(userId)); | ||||
|         } | ||||
|         return AjaxResult.success(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.getUserName()))) | ||||
|         { | ||||
|             return AjaxResult.error("新增用户'" + user.getUserName() + "'失败,登录账号已存在"); | ||||
|         } | ||||
|         else if (Validator.isNotEmpty(user.getPhonenumber()) | ||||
|                 && UserConstants.NOT_UNIQUE.equals(userService.checkPhoneUnique(user))) | ||||
|         { | ||||
|             return AjaxResult.error("新增用户'" + user.getUserName() + "'失败,手机号码已存在"); | ||||
|         } | ||||
|         else if (Validator.isNotEmpty(user.getEmail()) | ||||
|                 && UserConstants.NOT_UNIQUE.equals(userService.checkEmailUnique(user))) | ||||
|         { | ||||
|             return AjaxResult.error("新增用户'" + user.getUserName() + "'失败,邮箱账号已存在"); | ||||
|         } | ||||
|         user.setCreateBy(SecurityUtils.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); | ||||
|         if (Validator.isNotEmpty(user.getPhonenumber()) | ||||
|                 && UserConstants.NOT_UNIQUE.equals(userService.checkPhoneUnique(user))) | ||||
|         { | ||||
|             return AjaxResult.error("修改用户'" + user.getUserName() + "'失败,手机号码已存在"); | ||||
|         } | ||||
|         else if (Validator.isNotEmpty(user.getEmail()) | ||||
|                 && UserConstants.NOT_UNIQUE.equals(userService.checkEmailUnique(user))) | ||||
|         { | ||||
|             return AjaxResult.error("修改用户'" + user.getUserName() + "'失败,邮箱账号已存在"); | ||||
|         } | ||||
|         user.setUpdateBy(SecurityUtils.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) | ||||
|     { | ||||
|         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); | ||||
|         user.setPassword(SecurityUtils.encryptPassword(user.getPassword())); | ||||
|         user.setUpdateBy(SecurityUtils.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); | ||||
|         user.setUpdateBy(SecurityUtils.getUsername()); | ||||
|         return toAjax(userService.updateUserStatus(user)); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1 @@ | ||||
| restart.include.json=/com.alibaba.fastjson.*.jar | ||||
							
								
								
									
										112
									
								
								ruoyi-admin/src/main/resources/application-dev.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										112
									
								
								ruoyi-admin/src/main/resources/application-dev.yml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,112 @@ | ||||
| # 数据源配置 | ||||
| spring: | ||||
|   autoconfigure: | ||||
|     exclude: com.alibaba.druid.spring.boot.autoconfigure.DruidDataSourceAutoConfigure | ||||
|   datasource: | ||||
|     type: com.alibaba.druid.pool.DruidDataSource | ||||
|     # 动态数据源文档 https://www.kancloud.cn/tracy5546/dynamic-datasource/content | ||||
|     dynamic: | ||||
|       #设置默认的数据源或者数据源组,默认值即为 master | ||||
|       primary: master | ||||
|       datasource: | ||||
|         # 主库数据源 | ||||
|         master: | ||||
|           driverClassName: com.mysql.cj.jdbc.Driver | ||||
|           url: jdbc:mysql://192.168.110.65:6975/work_platform?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8&autoReconnect=true | ||||
|           username: work_platform | ||||
|           password: work_platform | ||||
|         # 从库数据源 | ||||
| #        slave: | ||||
| #          driverClassName: com.mysql.cj.jdbc.Driver | ||||
| #          url: | ||||
| #          username: | ||||
| #          password: | ||||
|     druid: | ||||
|       # 初始连接数 | ||||
|       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 | ||||
|       # 注意这个值和druid原生不一致,默认启动了stat | ||||
|       filters: stat | ||||
|       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 | ||||
|   # redis 配置 | ||||
|   redis: | ||||
|     # 地址 | ||||
|     host: 192.168.110.65 | ||||
|     # 端口,默认为6379 | ||||
|     port: 6379 | ||||
|     # 数据库索引 | ||||
|     database: 15 | ||||
|     # 密码 | ||||
|     password: Yjmysql123 | ||||
|     # 连接超时时间 | ||||
|     timeout: 10s | ||||
|     # 是否开启ssl | ||||
|     ssl: false | ||||
|  | ||||
| --- # redisson 客户端配置 | ||||
| redisson: | ||||
|   # 线程池数量 | ||||
|   threads: 16 | ||||
|   # Netty线程池数量 | ||||
|   nettyThreads: 32 | ||||
|   # 传输模式 | ||||
|   transportMode: "NIO" | ||||
|   # 单节点配置 | ||||
|   singleServerConfig: | ||||
|     # 客户端名称 | ||||
|     clientName: ${ruoyi.name} | ||||
|     # 最小空闲连接数 | ||||
|     connectionMinimumIdleSize: 32 | ||||
|     # 连接池大小 | ||||
|     connectionPoolSize: 64 | ||||
|     # 连接空闲超时,单位:毫秒 | ||||
|     idleConnectionTimeout: 10000 | ||||
|     # 命令等待超时,单位:毫秒 | ||||
|     timeout: 3000 | ||||
|     # 如果尝试在此限制之内发送成功,则开始启用 timeout 计时。 | ||||
|     retryAttempts: 3 | ||||
|     # 命令重试发送时间间隔,单位:毫秒 | ||||
|     retryInterval: 1500 | ||||
|     # 发布和订阅连接的最小空闲连接数 | ||||
|     subscriptionConnectionMinimumIdleSize: 1 | ||||
|     # 发布和订阅连接池大小 | ||||
|     subscriptionConnectionPoolSize: 50 | ||||
|     # 单个连接最大订阅数量 | ||||
|     subscriptionsPerConnection: 5 | ||||
|     # DNS监测时间间隔,单位:毫秒 | ||||
|     dnsMonitoringInterval: 5000 | ||||
							
								
								
									
										112
									
								
								ruoyi-admin/src/main/resources/application-prod.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										112
									
								
								ruoyi-admin/src/main/resources/application-prod.yml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,112 @@ | ||||
| # 数据源配置 | ||||
| spring: | ||||
|   autoconfigure: | ||||
|     exclude: com.alibaba.druid.spring.boot.autoconfigure.DruidDataSourceAutoConfigure | ||||
|   datasource: | ||||
|     type: com.alibaba.druid.pool.DruidDataSource | ||||
|     # 动态数据源文档 https://www.kancloud.cn/tracy5546/dynamic-datasource/content | ||||
|     dynamic: | ||||
|       #设置默认的数据源或者数据源组,默认值即为 master | ||||
|       primary: master | ||||
|       datasource: | ||||
|         # 主库数据源 | ||||
|         master: | ||||
|           driverClassName: com.mysql.cj.jdbc.Driver | ||||
|           url: jdbc:mysql://localhost:3306/ry-vue?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8&autoReconnect=true | ||||
|           username: root | ||||
|           password: root | ||||
|         # 从库数据源 | ||||
| #        slave: | ||||
| #          driverClassName: com.mysql.cj.jdbc.Driver | ||||
| #          url: | ||||
| #          username: | ||||
| #          password: | ||||
|     druid: | ||||
|       # 初始连接数 | ||||
|       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 | ||||
|       # 注意这个值和druid原生不一致,默认启动了stat | ||||
|       filters: stat | ||||
|       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 | ||||
|   # redis 配置 | ||||
|   redis: | ||||
|     # 地址 | ||||
|     host: localhost | ||||
|     # 端口,默认为6379 | ||||
|     port: 6379 | ||||
|     # 数据库索引 | ||||
|     database: 0 | ||||
|     # 密码 | ||||
|     password: | ||||
|     # 连接超时时间 | ||||
|     timeout: 10s | ||||
|     # 是否开启ssl | ||||
|     ssl: false | ||||
|  | ||||
| --- # redisson 客户端配置 | ||||
| redisson: | ||||
|   # 线程池数量 | ||||
|   threads: 16 | ||||
|   # Netty线程池数量 | ||||
|   nettyThreads: 32 | ||||
|   # 传输模式 | ||||
|   transportMode: "NIO" | ||||
|   # 单节点配置 | ||||
|   singleServerConfig: | ||||
|     # 客户端名称 | ||||
|     clientName: ${ruoyi.name} | ||||
|     # 最小空闲连接数 | ||||
|     connectionMinimumIdleSize: 32 | ||||
|     # 连接池大小 | ||||
|     connectionPoolSize: 64 | ||||
|     # 连接空闲超时,单位:毫秒 | ||||
|     idleConnectionTimeout: 10000 | ||||
|     # 命令等待超时,单位:毫秒 | ||||
|     timeout: 3000 | ||||
|     # 如果尝试在此限制之内发送成功,则开始启用 timeout 计时。 | ||||
|     retryAttempts: 3 | ||||
|     # 命令重试发送时间间隔,单位:毫秒 | ||||
|     retryInterval: 1500 | ||||
|     # 发布和订阅连接的最小空闲连接数 | ||||
|     subscriptionConnectionMinimumIdleSize: 1 | ||||
|     # 发布和订阅连接池大小 | ||||
|     subscriptionConnectionPoolSize: 50 | ||||
|     # 单个连接最大订阅数量 | ||||
|     subscriptionsPerConnection: 5 | ||||
|     # DNS监测时间间隔,单位:毫秒 | ||||
|     dnsMonitoringInterval: 5000 | ||||
							
								
								
									
										324
									
								
								ruoyi-admin/src/main/resources/application.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										324
									
								
								ruoyi-admin/src/main/resources/application.yml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,324 @@ | ||||
| # 项目相关配置 | ||||
| ruoyi: | ||||
|   # 名称 | ||||
|   name: RuoYi-Vue-Plus | ||||
|   # 版本 | ||||
|   version: ${ruoyi-vue-plus.version} | ||||
|   # 版权年份 | ||||
|   copyrightYear: 2021 | ||||
|   # 实例演示开关 | ||||
|   demoEnabled: true | ||||
|   # 文件路径 | ||||
|   profile: ./ruoyi/uploadPath | ||||
|   # 获取ip地址开关 | ||||
|   addressEnabled: true | ||||
|  | ||||
| captcha: | ||||
|   # 验证码开关 | ||||
|   enabled: false | ||||
|   # 验证码类型 math 数组计算 char 字符验证 | ||||
|   type: math | ||||
|   # line 线段干扰 circle 圆圈干扰 shear 扭曲干扰 | ||||
|   category: circle | ||||
|   # 数字验证码位数 | ||||
|   numberLength: 1 | ||||
|   # 字符验证码长度 | ||||
|   charLength: 4 | ||||
|  | ||||
| # 开发环境配置 | ||||
| server: | ||||
|   # 服务器的HTTP端口,默认为8080 | ||||
|   port: 9099 | ||||
|   servlet: | ||||
|     # 应用的访问路径 | ||||
|     context-path: /ruoyi | ||||
|   # undertow 配置 | ||||
|   undertow: | ||||
|     # HTTP post内容的最大大小。当值为-1时,默认值为大小是无限的 | ||||
|     max-http-post-size: -1 | ||||
|     # 以下的配置会影响buffer,这些buffer会用于服务器连接的IO操作,有点类似netty的池化内存管理 | ||||
|     # 每块buffer的空间大小,越小的空间被利用越充分 | ||||
|     buffer-size: 512 | ||||
|     # 是否分配的直接内存 | ||||
|     direct-buffers: true | ||||
|     threads: | ||||
|       # 设置IO线程数, 它主要执行非阻塞的任务,它们会负责多个连接, 默认设置每个CPU核心一个线程 | ||||
|       io: 8 | ||||
|       # 阻塞任务线程池, 当执行类似servlet请求阻塞操作, undertow会从这个线程池中取得线程,它的值设置取决于系统的负载 | ||||
|       worker: 256 | ||||
| #  # tomcat 配置 | ||||
| #  tomcat: | ||||
| #    # tomcat的URI编码 | ||||
| #    uri-encoding: UTF-8 | ||||
| #    # tomcat最大线程数,默认为200 | ||||
| #    max-threads: 500 | ||||
| #    # Tomcat启动初始化的线程数,默认值25 | ||||
| #    min-spare-threads: 30 | ||||
|  | ||||
| # 日志配置 | ||||
| logging: | ||||
|   level: | ||||
|     com.ruoyi: debug | ||||
|     org.springframework: warn | ||||
|   config: classpath:logback.xml | ||||
|  | ||||
| # Spring配置 | ||||
| spring: | ||||
|   # 资源信息 | ||||
|   messages: | ||||
|     # 国际化资源文件路径 | ||||
|     basename: i18n/messages | ||||
|   # dev 开发环境 pord 正式环境 | ||||
|   profiles: | ||||
|     active: dev | ||||
|   # 文件上传 | ||||
|   servlet: | ||||
|     multipart: | ||||
|       # 单个文件大小 | ||||
|       max-file-size: 100GB | ||||
|       # 设置总上传的文件大小 | ||||
|       max-request-size: 200GB | ||||
|   # 服务模块 | ||||
|   devtools: | ||||
|     restart: | ||||
|       # 热部署开关 | ||||
|       enabled: false | ||||
|   # 与vue整合部署使用 | ||||
|   thymeleaf: | ||||
|     # 将系统模板放置到最前面 否则会与 springboot-admin 页面冲突 | ||||
|     template-resolver-order: 1 | ||||
|   jackson: | ||||
|     # 日期格式化 | ||||
|     date-format: yyyy-MM-dd HH:mm:ss | ||||
|     serialization: | ||||
|       # 格式化输出 | ||||
|       indent_output: false | ||||
|       # 忽略无法转换的对象 | ||||
|       fail_on_empty_beans: false | ||||
|     deserialization: | ||||
|       # 允许对象忽略json中不存在的属性 | ||||
|       fail_on_unknown_properties: false | ||||
|  | ||||
| # token配置 | ||||
| token: | ||||
|   # 令牌自定义标识 | ||||
|   header: Authorization | ||||
|   # 令牌密钥 | ||||
|   secret: abcdefghijklmnopqrstuvwxyz | ||||
|   # 令牌有效期(默认30分钟) | ||||
|   expireTime: 30000 | ||||
|  | ||||
| # MyBatisPlus配置 | ||||
| # https://baomidou.com/config/ | ||||
| mybatis-plus: | ||||
|   mapperPackage: com.ruoyi.**.mapper | ||||
|   # 对应的 XML 文件位置 | ||||
|   mapperLocations: classpath*:mapper/**/*Mapper.xml | ||||
|   # 实体扫描,多个package用逗号或者分号分隔 | ||||
|   typeAliasesPackage: com.ruoyi.**.domain | ||||
|   # 针对 typeAliasesPackage,如果配置了该属性,则仅仅会扫描路径下以该类作为父类的域对象 | ||||
|   #typeAliasesSuperType: Class<?> | ||||
|   # 如果配置了该属性,SqlSessionFactoryBean 会把该包下面的类注册为对应的 TypeHandler | ||||
|   #typeHandlersPackage: null | ||||
|   # 如果配置了该属性,会将路径下的枚举类进行注入,让实体类字段能够简单快捷的使用枚举属性 | ||||
|   #typeEnumsPackage: null | ||||
|   # 启动时是否检查 MyBatis XML 文件的存在,默认不检查 | ||||
|   checkConfigLocation: false | ||||
|   # 通过该属性可指定 MyBatis 的执行器,MyBatis 的执行器总共有三种: | ||||
|   # SIMPLE:该执行器类型不做特殊的事情,为每个语句的执行创建一个新的预处理语句(PreparedStatement) | ||||
|   # REUSE:该执行器类型会复用预处理语句(PreparedStatement) | ||||
|   # BATCH:该执行器类型会批量执行所有的更新语句 | ||||
|   executorType: SIMPLE | ||||
|   # 指定外部化 MyBatis Properties 配置,通过该配置可以抽离配置,实现不同环境的配置部署 | ||||
| #  configurationProperties: null | ||||
|   configuration: | ||||
|     # 自动驼峰命名规则(camel case)映射 | ||||
|     # 如果您的数据库命名符合规则无需使用 @TableField 注解指定数据库字段名 | ||||
|     mapUnderscoreToCamelCase: true | ||||
|     # 默认枚举处理类,如果配置了该属性,枚举将统一使用指定处理器进行处理 | ||||
|     # org.apache.ibatis.type.EnumTypeHandler : 存储枚举的名称 | ||||
|     # org.apache.ibatis.type.EnumOrdinalTypeHandler : 存储枚举的索引 | ||||
|     # com.baomidou.mybatisplus.extension.handlers.MybatisEnumTypeHandler : 枚举类需要实现IEnum接口或字段标记@EnumValue注解. | ||||
|     defaultEnumTypeHandler: org.apache.ibatis.type.EnumTypeHandler | ||||
|     # 当设置为 true 的时候,懒加载的对象可能被任何懒属性全部加载,否则,每个属性都按需加载。需要和 lazyLoadingEnabled 一起使用。 | ||||
|     aggressiveLazyLoading: true | ||||
|     # MyBatis 自动映射策略 | ||||
|     # NONE:不启用自动映射 | ||||
|     # PARTIAL:只对非嵌套的 resultMap 进行自动映射 | ||||
|     # FULL:对所有的 resultMap 都进行自动映射 | ||||
|     autoMappingBehavior: PARTIAL | ||||
|     # MyBatis 自动映射时未知列或未知属性处理策 | ||||
|     # NONE:不做任何处理 (默认值) | ||||
|     # WARNING:以日志的形式打印相关警告信息 | ||||
|     # FAILING:当作映射失败处理,并抛出异常和详细信息 | ||||
|     autoMappingUnknownColumnBehavior: NONE | ||||
|     # Mybatis一级缓存,默认为 SESSION | ||||
|     # SESSION session级别缓存,同一个session相同查询语句不会再次查询数据库 | ||||
|     # STATEMENT 关闭一级缓存 | ||||
|     localCacheScope: SESSION | ||||
|     # 开启Mybatis二级缓存,默认为 true | ||||
|     cacheEnabled: true | ||||
|   global-config: | ||||
|     # 是否打印 Logo banner | ||||
|     banner: true | ||||
|     # 是否初始化 SqlRunner | ||||
|     enableSqlRunner: false | ||||
|     dbConfig: | ||||
|       # 主键类型 | ||||
|       # AUTO 数据库ID自增 | ||||
|       # NONE 空 | ||||
|       # INPUT 用户输入ID | ||||
|       # ASSIGN_ID 全局唯一ID | ||||
|       # ASSIGN_UUID 全局唯一ID UUID | ||||
|       idType: AUTO | ||||
|       # 表名前缀 | ||||
|       tablePrefix: null | ||||
|       # 字段 format,例: %s,(对主键无效) | ||||
|       columnFormat: null | ||||
|       # 表名是否使用驼峰转下划线命名,只对表名生效 | ||||
|       tableUnderline: true | ||||
|       # 大写命名,对表名和字段名均生效 | ||||
|       capitalMode: false | ||||
|       # 全局的entity的逻辑删除字段属性名 | ||||
|       logicDeleteField: null | ||||
|       # 逻辑已删除值 | ||||
|       logicDeleteValue: 2 | ||||
|       # 逻辑未删除值 | ||||
|       logicNotDeleteValue: 0 | ||||
|       # 字段验证策略之 insert,在 insert 的时候的字段验证策略 | ||||
|       # IGNORED 忽略判断 | ||||
|       # NOT_NULL 非NULL判断 | ||||
|       # NOT_EMPTY 非空判断(只对字符串类型字段,其他类型字段依然为非NULL判断) | ||||
|       # DEFAULT 默认的,一般只用于注解里 | ||||
|       # NEVER 不加入 SQL | ||||
|       insertStrategy: NOT_EMPTY | ||||
|       # 字段验证策略之 update,在 update 的时候的字段验证策略 | ||||
|       updateStrategy: NOT_EMPTY | ||||
|       # 字段验证策略之 select,在 select 的时候的字段验证策略既 wrapper 根据内部 entity 生成的 where 条件 | ||||
|       selectStrategy: NOT_EMPTY | ||||
|  | ||||
| # Swagger配置 | ||||
| swagger: | ||||
|   # 是否开启swagger | ||||
|   enabled: true | ||||
|   # 请求前缀 | ||||
|   pathMapping: / | ||||
|   # 标题 | ||||
|   title: '标题:RuoYi-Vue-Plus后台管理系统_接口文档' | ||||
|   # 描述 | ||||
|   description: '描述:RuoYi-Vue-Plus后台管理系统接口文档,主要包含系统管理、系统监控、系统工具、代码生成、定时任务等模块的接口文档。' | ||||
|   # 版本 | ||||
|   version: '版本号: ${ruoyi-vue-plus.version}' | ||||
|   # 作者信息 | ||||
|   contact: | ||||
|     name: qjq | ||||
|     email: 1766193529@qq.com | ||||
|     url: https://gitee.com/JavaLionLi/RuoYi-Vue-Plus | ||||
|  | ||||
| # 防止XSS攻击 | ||||
| xss: | ||||
|   # 过滤开关 | ||||
|   enabled: true | ||||
|   # 排除链接(多个用逗号分隔) | ||||
|   excludes: /system/notice/* | ||||
|   # 匹配链接 | ||||
|   urlPatterns: /system/*,/monitor/*,/tool/* | ||||
|  | ||||
| # 全局线程池相关配置 | ||||
| thread-pool: | ||||
|   # 是否开启线程池 | ||||
|   enabled: false | ||||
|   # 核心线程池大小 | ||||
|   corePoolSize: 8 | ||||
|   # 最大可创建的线程数 | ||||
|   maxPoolSize: 16 | ||||
|   # 队列最大长度 | ||||
|   queueCapacity: 128 | ||||
|   # 线程池维护线程所允许的空闲时间 | ||||
|   keepAliveSeconds: 300 | ||||
|   # 线程池对拒绝任务(无线程可用)的处理策略 | ||||
|   # CallerRunsPolicy 等待 | ||||
|   # DiscardOldestPolicy 放弃最旧的 | ||||
|   # DiscardPolicy 丢弃 | ||||
|   # AbortPolicy 中止 | ||||
|   rejectedExecutionHandler: CallerRunsPolicy | ||||
|  | ||||
| # feign 相关配置 | ||||
| feign: | ||||
|   package: com.ruoyi.**.feign | ||||
|   # 开启压缩 | ||||
|   compression: | ||||
|     request: | ||||
|       enabled: true | ||||
|     response: | ||||
|       enabled: true | ||||
|   okhttp: | ||||
|     enabled: true | ||||
|   circuitbreaker: | ||||
|     enabled: true | ||||
|  | ||||
| --- # 分布式锁 lock4j 全局配置 | ||||
| lock4j: | ||||
|   # 获取分布式锁超时时间,默认为 3000 毫秒 | ||||
|   acquire-timeout: 3000 | ||||
|   # 分布式锁的超时时间,默认为 30 毫秒 | ||||
|   expire: 30000 | ||||
|  | ||||
| --- # 定时任务配置 | ||||
| spring: | ||||
|   quartz: | ||||
|     scheduler-name: RuoyiScheduler | ||||
|     startup-delay: 1s | ||||
|     overwrite-existing-jobs: true | ||||
|     auto-startup: true | ||||
|     job-store-type: jdbc | ||||
|     properties: | ||||
|       org: | ||||
|         quartz: | ||||
|           # Scheduler 相关配置 | ||||
|           scheduler: | ||||
|             instanceName: RuoyiScheduler | ||||
|             instanceId: AUTO | ||||
|           # 线程池相关配置 | ||||
|           threadPool: | ||||
|             class: org.quartz.simpl.SimpleThreadPool | ||||
|             threadCount: 20 | ||||
|             threadPriority: 5 | ||||
|           # JobStore 集群配置 | ||||
|           jobStore: | ||||
|             class: org.quartz.impl.jdbcjobstore.JobStoreTX | ||||
|             isClustered: true | ||||
|             clusterCheckinInterval: 15000 | ||||
|             txIsolationLevelSerializable: true | ||||
|             misfireThreshold: 60000 | ||||
|             tablePrefix: QRTZ_ | ||||
|             # sqlserver 启用 | ||||
|             # selectWithLockSQL: SELECT * FROM {0}LOCKS UPDLOCK WHERE LOCK_NAME = ? | ||||
|  | ||||
| --- # 监控配置 | ||||
| spring: | ||||
|   application: | ||||
|     name: ruoyi-vue-plus | ||||
|   boot: | ||||
|     admin: | ||||
|       # Spring Boot Admin Client 客户端的相关配置 | ||||
|       client: | ||||
|         # 设置 Spring Boot Admin Server 地址 | ||||
|         url: http://localhost:${server.port}${server.servlet.context-path}${spring.boot.admin.context-path} | ||||
|         instance: | ||||
|           prefer-ip: true # 注册实例时,优先使用 IP | ||||
|       # Spring Boot Admin Server 服务端的相关配置 | ||||
|       context-path: /admin # 配置 Spring | ||||
|  | ||||
| # Actuator 监控端点的配置项 | ||||
| management: | ||||
|   endpoints: | ||||
|     web: | ||||
|       # Actuator 提供的 API 接口的根目录。默认为 /actuator | ||||
|       base-path: /actuator | ||||
|       exposure: | ||||
|         # 需要开放的端点。默认值只打开 health 和 info 两个端点。通过设置 * ,可以开放所有端点。 | ||||
|         include: '*' | ||||
|   endpoint: | ||||
|     logfile: | ||||
|       external-file: ./logs/sys-console.log | ||||
							
								
								
									
										8
									
								
								ruoyi-admin/src/main/resources/banner.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								ruoyi-admin/src/main/resources/banner.txt
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,8 @@ | ||||
| Application Version: ${ruoyi-vue-plus.version} | ||||
| Spring Boot Version: ${spring-boot.version} | ||||
| __________            _____.___.__         ____   ____                     __________.__ | ||||
| \______   \__ __  ____\__  |   |__|        \   \ /   /_ __   ____          \______   \  |  __ __  ______ | ||||
|  |       _/  |  \/  _ \/   |   |  |  ______ \   Y   /  |  \_/ __ \   ______ |     ___/  | |  |  \/  ___/ | ||||
|  |    |   \  |  (  <_> )____   |  | /_____/  \     /|  |  /\  ___/  /_____/ |    |   |  |_|  |  /\___ \ | ||||
|  |____|_  /____/ \____// ______|__|           \___/ |____/  \___  >         |____|   |____/____//____  > | ||||
|         \/             \/                                       \/                                   \/ | ||||
							
								
								
									
										36
									
								
								ruoyi-admin/src/main/resources/i18n/messages.properties
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								ruoyi-admin/src/main/resources/i18n/messages.properties
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,36 @@ | ||||
| #错误消息 | ||||
| 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}次,帐户锁定10分钟 | ||||
| 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.notfound=请重新登录 | ||||
| user.forcelogout=管理员强制退出,请重新登录 | ||||
| user.unknown.error=未知错误,请重新登录 | ||||
|  | ||||
| ##文件上传消息 | ||||
| upload.exceed.maxSize=上传的文件大小超出限制的文件大小!<br/>允许的文件最大大小是:{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}] | ||||
							
								
								
									
										113
									
								
								ruoyi-admin/src/main/resources/logback.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										113
									
								
								ruoyi-admin/src/main/resources/logback.xml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,113 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <configuration> | ||||
|     <property name="log.path" value="./logs"/> | ||||
|     <property name="console.log.pattern" | ||||
|               value="%red(%d{yyyy-MM-dd HH:mm:ss}) %green([%thread]) %highlight(%-5level) %boldMagenta(%logger{36}%n) - %msg%n"/> | ||||
|     <property name="log.pattern" value="%d{yyyy-MM-dd HH:mm:ss} [%thread] %-5level %logger{36} - %msg%n"/> | ||||
|  | ||||
| 	<!-- 控制台输出 --> | ||||
| 	<appender name="console" class="ch.qos.logback.core.ConsoleAppender"> | ||||
| 		<encoder> | ||||
| 			<pattern>${console.log.pattern}</pattern> | ||||
|             <charset>utf-8</charset> | ||||
| 		</encoder> | ||||
| 	</appender> | ||||
|  | ||||
|     <!-- 控制台输出 --> | ||||
|     <appender name="file_console" class="ch.qos.logback.core.rolling.RollingFileAppender"> | ||||
|         <file>${log.path}/sys-console.log</file> | ||||
|         <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy"> | ||||
|             <!-- 日志文件名格式 --> | ||||
|             <fileNamePattern>${log.path}/sys-console.%d{yyyy-MM-dd}.log</fileNamePattern> | ||||
|             <!-- 日志最大 1天 --> | ||||
|             <maxHistory>1</maxHistory> | ||||
|         </rollingPolicy> | ||||
|         <encoder> | ||||
|             <pattern>${log.pattern}</pattern> | ||||
|             <charset>utf-8</charset> | ||||
|         </encoder> | ||||
|         <filter class="ch.qos.logback.classic.filter.ThresholdFilter"> | ||||
|             <!-- 过滤的级别 --> | ||||
|             <level>INFO</level> | ||||
|         </filter> | ||||
|     </appender> | ||||
|  | ||||
| 	<!-- 系统日志输出 --> | ||||
| 	<appender name="file_info" class="ch.qos.logback.core.rolling.RollingFileAppender"> | ||||
| 	    <file>${log.path}/sys-info.log</file> | ||||
|         <!-- 循环政策:基于时间创建日志文件 --> | ||||
| 		<rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy"> | ||||
|             <!-- 日志文件名格式 --> | ||||
| 			<fileNamePattern>${log.path}/sys-info.%d{yyyy-MM-dd}.log</fileNamePattern> | ||||
| 			<!-- 日志最大的历史 60天 --> | ||||
| 			<maxHistory>60</maxHistory> | ||||
| 		</rollingPolicy> | ||||
| 		<encoder> | ||||
| 			<pattern>${log.pattern}</pattern> | ||||
| 		</encoder> | ||||
| 		<filter class="ch.qos.logback.classic.filter.LevelFilter"> | ||||
|             <!-- 过滤的级别 --> | ||||
|             <level>INFO</level> | ||||
|             <!-- 匹配时的操作:接收(记录) --> | ||||
|             <onMatch>ACCEPT</onMatch> | ||||
|             <!-- 不匹配时的操作:拒绝(不记录) --> | ||||
|             <onMismatch>DENY</onMismatch> | ||||
|         </filter> | ||||
| 	</appender> | ||||
|  | ||||
| 	<appender name="file_error" class="ch.qos.logback.core.rolling.RollingFileAppender"> | ||||
| 	    <file>${log.path}/sys-error.log</file> | ||||
|         <!-- 循环政策:基于时间创建日志文件 --> | ||||
|         <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy"> | ||||
|             <!-- 日志文件名格式 --> | ||||
|             <fileNamePattern>${log.path}/sys-error.%d{yyyy-MM-dd}.log</fileNamePattern> | ||||
| 			<!-- 日志最大的历史 60天 --> | ||||
| 			<maxHistory>60</maxHistory> | ||||
|         </rollingPolicy> | ||||
|         <encoder> | ||||
|             <pattern>${log.pattern}</pattern> | ||||
|         </encoder> | ||||
|         <filter class="ch.qos.logback.classic.filter.LevelFilter"> | ||||
|             <!-- 过滤的级别 --> | ||||
|             <level>ERROR</level> | ||||
| 			<!-- 匹配时的操作:接收(记录) --> | ||||
|             <onMatch>ACCEPT</onMatch> | ||||
| 			<!-- 不匹配时的操作:拒绝(不记录) --> | ||||
|             <onMismatch>DENY</onMismatch> | ||||
|         </filter> | ||||
|     </appender> | ||||
|  | ||||
| 	<!-- 用户访问日志输出  --> | ||||
|     <appender name="sys-user" class="ch.qos.logback.core.rolling.RollingFileAppender"> | ||||
| 		<file>${log.path}/sys-user.log</file> | ||||
|         <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy"> | ||||
|             <!-- 按天回滚 daily --> | ||||
|             <fileNamePattern>${log.path}/sys-user.%d{yyyy-MM-dd}.log</fileNamePattern> | ||||
|             <!-- 日志最大的历史 60天 --> | ||||
|             <maxHistory>60</maxHistory> | ||||
|         </rollingPolicy> | ||||
|         <encoder> | ||||
|             <pattern>${log.pattern}</pattern> | ||||
|         </encoder> | ||||
|     </appender> | ||||
|  | ||||
| 	<!-- 系统模块日志级别控制  --> | ||||
| 	<logger name="com.ruoyi" level="info" /> | ||||
| 	<!-- Spring日志级别控制  --> | ||||
| 	<logger name="org.springframework" level="warn" /> | ||||
| 	<root level="info"> | ||||
| 		<appender-ref ref="console" /> | ||||
| 	</root> | ||||
|  | ||||
| 	<!--系统操作日志--> | ||||
|     <root level="info"> | ||||
|         <appender-ref ref="file_info" /> | ||||
|         <appender-ref ref="file_error" /> | ||||
|         <appender-ref ref="file_console" /> | ||||
|     </root> | ||||
|  | ||||
| 	<!--系统用户操作日志--> | ||||
|     <logger name="sys-user" level="info"> | ||||
|         <appender-ref ref="sys-user"/> | ||||
|     </logger> | ||||
| </configuration> | ||||
							
								
								
									
										16
									
								
								ruoyi-admin/src/main/resources/rebel.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								ruoyi-admin/src/main/resources/rebel.xml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,16 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
|  | ||||
| <!-- | ||||
|   This is the JRebel configuration file. It maps the running application to your IDE workspace, enabling JRebel reloading for this project. | ||||
|   Refer to https://manuals.jrebel.com/jrebel/standalone/config.html for more information. | ||||
| --> | ||||
| <application generated-by="intellij" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://www.zeroturnaround.com" xsi:schemaLocation="http://www.zeroturnaround.com http://update.zeroturnaround.com/jrebel/rebel-2_3.xsd"> | ||||
|  | ||||
| 	<id>ruoyi-admin</id> | ||||
|  | ||||
| 	<classpath> | ||||
| 		<dir name="E:/intellijWork/old-tress-sys/ruoyi-admin/target/classes"> | ||||
| 		</dir> | ||||
| 	</classpath> | ||||
|  | ||||
| </application> | ||||
							
								
								
									
										162
									
								
								ruoyi-common/pom.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										162
									
								
								ruoyi-common/pom.xml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,162 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project xmlns="http://maven.apache.org/POM/4.0.0" | ||||
|          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" | ||||
|          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||||
|     <parent> | ||||
|         <artifactId>ruoyi-vue-plus</artifactId> | ||||
|         <groupId>com.ruoyi</groupId> | ||||
|         <version>${ruoyi-vue-plus.version}</version> | ||||
|     </parent> | ||||
|     <modelVersion>4.0.0</modelVersion> | ||||
|  | ||||
|     <artifactId>ruoyi-common</artifactId> | ||||
|  | ||||
|     <description> | ||||
|         common通用工具 | ||||
|     </description> | ||||
|  | ||||
|     <dependencies> | ||||
|  | ||||
|         <!-- Spring框架基本的核心工具 --> | ||||
|         <dependency> | ||||
|             <groupId>org.springframework</groupId> | ||||
|             <artifactId>spring-context-support</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- SpringWeb模块 --> | ||||
|         <dependency> | ||||
|             <groupId>org.springframework</groupId> | ||||
|             <artifactId>spring-web</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- spring security 安全认证 --> | ||||
|         <dependency> | ||||
|             <groupId>org.springframework.boot</groupId> | ||||
|             <artifactId>spring-boot-starter-security</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- 自定义验证注解 --> | ||||
|         <dependency> | ||||
|             <groupId>javax.validation</groupId> | ||||
|             <artifactId>validation-api</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!--常用工具类 --> | ||||
|         <dependency> | ||||
|             <groupId>org.apache.commons</groupId> | ||||
|             <artifactId>commons-lang3</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- JSON工具类 --> | ||||
|         <dependency> | ||||
|             <groupId>com.fasterxml.jackson.core</groupId> | ||||
|             <artifactId>jackson-databind</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- excel工具 --> | ||||
|         <dependency> | ||||
|             <groupId>org.apache.poi</groupId> | ||||
|             <artifactId>poi-ooxml</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- yml解析器 --> | ||||
|         <dependency> | ||||
|             <groupId>org.yaml</groupId> | ||||
|             <artifactId>snakeyaml</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!--Token生成与解析--> | ||||
|         <dependency> | ||||
|             <groupId>io.jsonwebtoken</groupId> | ||||
|             <artifactId>jjwt</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- redis 缓存操作 --> | ||||
|         <dependency> | ||||
|             <groupId>org.springframework.boot</groupId> | ||||
|             <artifactId>spring-boot-starter-data-redis</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- pool 对象池 --> | ||||
|         <dependency> | ||||
|             <groupId>org.apache.commons</groupId> | ||||
|             <artifactId>commons-pool2</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- servlet包 --> | ||||
|         <dependency> | ||||
|             <groupId>javax.servlet</groupId> | ||||
|             <artifactId>javax.servlet-api</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <dependency> | ||||
|             <groupId>com.baomidou</groupId> | ||||
|             <artifactId>mybatis-plus-boot-starter</artifactId> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>com.baomidou</groupId> | ||||
|             <artifactId>mybatis-plus-extension</artifactId> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>cn.hutool</groupId> | ||||
|             <artifactId>hutool-all</artifactId> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>org.projectlombok</groupId> | ||||
|             <artifactId>lombok</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <dependency> | ||||
|             <groupId>org.springframework.cloud</groupId> | ||||
|             <artifactId>spring-cloud-starter-openfeign</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <dependency> | ||||
|             <groupId>io.github.openfeign</groupId> | ||||
|             <artifactId>feign-okhttp</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <dependency> | ||||
|             <groupId>de.codecentric</groupId> | ||||
|             <artifactId>spring-boot-admin-starter-server</artifactId> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>de.codecentric</groupId> | ||||
|             <artifactId>spring-boot-admin-starter-client</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <dependency> | ||||
|             <groupId>com.github.xiaoymin</groupId> | ||||
|             <artifactId>knife4j-spring-boot-starter</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <dependency> | ||||
|             <groupId>org.springframework.boot</groupId> | ||||
|             <artifactId>spring-boot-starter-actuator</artifactId> | ||||
|         </dependency> | ||||
|         <!--  自动生成YML配置关联JSON文件  --> | ||||
|         <dependency> | ||||
|             <groupId>org.springframework.boot</groupId> | ||||
|             <artifactId>spring-boot-configuration-processor</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!--redisson--> | ||||
|         <dependency> | ||||
|             <groupId>org.redisson</groupId> | ||||
|             <artifactId>redisson-spring-boot-starter</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- dynamic-datasource 多数据源--> | ||||
|         <dependency> | ||||
|             <groupId>com.baomidou</groupId> | ||||
|             <artifactId>dynamic-datasource-spring-boot-starter</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <dependency> | ||||
|             <groupId>com.baomidou</groupId> | ||||
|             <artifactId>lock4j-redisson-spring-boot-starter</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|     </dependencies> | ||||
|  | ||||
| </project> | ||||
| @ -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 ""; | ||||
|  | ||||
| 	/** | ||||
| 	 * 是否过滤用户权限 | ||||
| 	 */ | ||||
| 	public boolean isUser() default false; | ||||
| } | ||||
| @ -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; | ||||
| } | ||||
| @ -0,0 +1,165 @@ | ||||
| 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; | ||||
|  | ||||
| /** | ||||
|  * 自定义导出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; | ||||
|  | ||||
|     /** | ||||
|      * 导出类型(0数字 1字符串) | ||||
|      */ | ||||
|     public ColumnType cellType() default ColumnType.STRING; | ||||
|  | ||||
|     /** | ||||
|      * 导出时在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 {}; | ||||
|  | ||||
|     /** | ||||
|      * 是否导出数据,应对需求:有时我们需要导出一份模板,这是标题需要但内容需要用户手工填写. | ||||
|      */ | ||||
|     public boolean isExport() default true; | ||||
|  | ||||
|     /** | ||||
|      * 另一个类中的属性名称,支持多级获取,以小数点隔开 | ||||
|      */ | ||||
|     public String targetAttr() default ""; | ||||
|  | ||||
|     /** | ||||
|      * 是否自动统计数据,在最后追加一行统计数据总和 | ||||
|      */ | ||||
|     public boolean isStatistics() default false; | ||||
|  | ||||
|     /** | ||||
|      * 导出字段对齐方式(0:默认;1:靠左;2:居中;3:靠右) | ||||
|      */ | ||||
|     Align align() default Align.AUTO; | ||||
|  | ||||
|     public enum Align | ||||
|     { | ||||
|         AUTO(0), LEFT(1), CENTER(2), RIGHT(3); | ||||
|         private final int value; | ||||
|  | ||||
|         Align(int value) | ||||
|         { | ||||
|             this.value = value; | ||||
|         } | ||||
|  | ||||
|         public int value() | ||||
|         { | ||||
|             return this.value; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 字段类型(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; | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @ -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 | ||||
| { | ||||
|     Excel[] value(); | ||||
| } | ||||
| @ -0,0 +1,41 @@ | ||||
| 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; | ||||
| } | ||||
| @ -0,0 +1,23 @@ | ||||
| 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 | ||||
| { | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,76 @@ | ||||
| package com.ruoyi.common.config; | ||||
|  | ||||
| import lombok.Data; | ||||
| import lombok.Getter; | ||||
| import lombok.NoArgsConstructor; | ||||
| import lombok.experimental.Accessors; | ||||
| import org.springframework.boot.context.properties.ConfigurationProperties; | ||||
| import org.springframework.stereotype.Component; | ||||
|  | ||||
| /** | ||||
|  * 读取项目相关配置 | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
|  | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| @Accessors(chain = true) | ||||
| @Component | ||||
| @ConfigurationProperties(prefix = "ruoyi") | ||||
| public class RuoYiConfig | ||||
| { | ||||
|     /** 项目名称 */ | ||||
|     private String name; | ||||
|  | ||||
|     /** 版本 */ | ||||
|     private String version; | ||||
|  | ||||
|     /** 版权年份 */ | ||||
|     private String copyrightYear; | ||||
|  | ||||
|     /** 实例演示开关 */ | ||||
|     private boolean demoEnabled; | ||||
|  | ||||
|     /** 上传路径 */ | ||||
|     @Getter | ||||
|     private static String profile; | ||||
|  | ||||
|     /** 获取地址开关 */ | ||||
|     @Getter | ||||
|     private static boolean addressEnabled; | ||||
|  | ||||
|     public void setProfile(String profile) | ||||
|     { | ||||
|         RuoYiConfig.profile = profile; | ||||
|     } | ||||
|  | ||||
|     public void setAddressEnabled(boolean addressEnabled) | ||||
|     { | ||||
|         RuoYiConfig.addressEnabled = addressEnabled; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取头像上传路径 | ||||
|      */ | ||||
|     public static String getAvatarPath() | ||||
|     { | ||||
|         return getProfile() + "/avatar"; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取下载路径 | ||||
|      */ | ||||
|     public static String getDownloadPath() | ||||
|     { | ||||
|         return getProfile() + "/download/"; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取上传路径 | ||||
|      */ | ||||
|     public static String getUploadPath() | ||||
|     { | ||||
|         return getProfile() + "/upload"; | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,139 @@ | ||||
| package com.ruoyi.common.constant; | ||||
|  | ||||
| /** | ||||
|  * 通用常量信息 | ||||
|  * | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| public class Constants | ||||
| { | ||||
|     /** | ||||
|      * UTF-8 字符集 | ||||
|      */ | ||||
|     public static final String UTF8 = "UTF-8"; | ||||
|  | ||||
|     /** | ||||
|      * GBK 字符集 | ||||
|      */ | ||||
|     public static final String GBK = "GBK"; | ||||
|  | ||||
|     /** | ||||
|      * 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"; | ||||
|  | ||||
|     /** | ||||
|      * 登录成功 | ||||
|      */ | ||||
|     public static final String LOGIN_SUCCESS = "Success"; | ||||
|  | ||||
|     /** | ||||
|      * 注销 | ||||
|      */ | ||||
|     public static final String LOGOUT = "Logout"; | ||||
|  | ||||
|     /** | ||||
|      * 登录失败 | ||||
|      */ | ||||
|     public static final String LOGIN_FAIL = "Error"; | ||||
|  | ||||
|     /** | ||||
|      * 验证码 redis key | ||||
|      */ | ||||
|     public static final String CAPTCHA_CODE_KEY = "captcha_codes:"; | ||||
|  | ||||
|     /** | ||||
|      * 登录用户 redis key | ||||
|      */ | ||||
|     public static final String LOGIN_TOKEN_KEY = "login_tokens:"; | ||||
|  | ||||
|     /** | ||||
|      * 防重提交 redis key | ||||
|      */ | ||||
|     public static final String REPEAT_SUBMIT_KEY = "repeat_submit:"; | ||||
|  | ||||
|     /** | ||||
|      * 验证码有效期(分钟) | ||||
|      */ | ||||
|     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 = "sub"; | ||||
|  | ||||
|     /** | ||||
|      * 用户头像 | ||||
|      */ | ||||
|     public static final String JWT_AVATAR = "avatar"; | ||||
|  | ||||
|     /** | ||||
|      * 创建时间 | ||||
|      */ | ||||
|     public static final String JWT_CREATED = "created"; | ||||
|  | ||||
|     /** | ||||
|      * 用户权限 | ||||
|      */ | ||||
|     public static final String JWT_AUTHORITIES = "authorities"; | ||||
|  | ||||
|     /** | ||||
|      * 参数管理 cache key | ||||
|      */ | ||||
|     public static final String SYS_CONFIG_KEY = "sys_config:"; | ||||
|  | ||||
|     /** | ||||
|      * 字典管理 cache key | ||||
|      */ | ||||
|     public static final String SYS_DICT_KEY = "sys_dict:"; | ||||
|  | ||||
|     /** | ||||
|      * 资源映射路径 前缀 | ||||
|      */ | ||||
|     public static final String RESOURCE_PREFIX = "/profile"; | ||||
|  | ||||
| 	/** | ||||
| 	 * RMI 远程方法调用 | ||||
| 	 */ | ||||
| 	public static final String LOOKUP_RMI = "rmi://"; | ||||
|  | ||||
| 	/** | ||||
| 	 * 资源映射路径 前缀 | ||||
| 	 */ | ||||
| 	public static final String REDIS_LOCK_KEY = "redis_lock:"; | ||||
| } | ||||
| @ -0,0 +1,114 @@ | ||||
| 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 REQUIRE = "1"; | ||||
| } | ||||
| @ -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; | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,63 @@ | ||||
| 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"; | ||||
|  | ||||
|     /** 校验返回结果码 */ | ||||
|     public final static String UNIQUE = "0"; | ||||
|     public final static String NOT_UNIQUE = "1"; | ||||
| } | ||||
| @ -0,0 +1,78 @@ | ||||
| package com.ruoyi.common.core.controller; | ||||
|  | ||||
| import cn.hutool.core.util.StrUtil; | ||||
| import com.ruoyi.common.core.domain.AjaxResult; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| /** | ||||
|  * web层通用数据处理 | ||||
|  * | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| public class BaseController | ||||
| { | ||||
|     protected final Logger logger = LoggerFactory.getLogger(this.getClass()); | ||||
|  | ||||
|     /** | ||||
|      * 响应返回结果 | ||||
|      * | ||||
|      * @param rows 影响行数 | ||||
|      * @return 操作结果 | ||||
|      */ | ||||
|     protected AjaxResult<Void> toAjax(int rows) | ||||
|     { | ||||
|         return rows > 0 ? AjaxResult.success() : AjaxResult.error(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 响应返回结果 | ||||
|      * | ||||
|      * @param result 结果 | ||||
|      * @return 操作结果 | ||||
|      */ | ||||
|     protected AjaxResult<Void> toAjax(boolean result) | ||||
|     { | ||||
|         return result ? success() : error(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 返回成功 | ||||
|      */ | ||||
|     public AjaxResult<Void> success() | ||||
|     { | ||||
|         return AjaxResult.success(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 返回失败消息 | ||||
|      */ | ||||
|     public AjaxResult<Void> error() | ||||
|     { | ||||
|         return AjaxResult.error(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 返回成功消息 | ||||
|      */ | ||||
|     public AjaxResult<Void> success(String message) | ||||
|     { | ||||
|         return AjaxResult.success(message); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 返回失败消息 | ||||
|      */ | ||||
|     public AjaxResult<Void> error(String message) | ||||
|     { | ||||
|         return AjaxResult.error(message); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 页面跳转 | ||||
|      */ | ||||
|     public String redirect(String url) | ||||
|     { | ||||
|         return StrUtil.format("redirect:{}", url); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,134 @@ | ||||
| package com.ruoyi.common.core.domain; | ||||
|  | ||||
| import cn.hutool.http.HttpStatus; | ||||
| import io.swagger.annotations.ApiModel; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.AllArgsConstructor; | ||||
| import lombok.Data; | ||||
| import lombok.NoArgsConstructor; | ||||
| import lombok.experimental.Accessors; | ||||
|  | ||||
| /** | ||||
|  * 操作消息提醒 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| @AllArgsConstructor | ||||
| @Accessors(chain = true) | ||||
| @ApiModel("请求响应对象") | ||||
| public class AjaxResult<T> { | ||||
|  | ||||
| 	private static final long serialVersionUID = 1L; | ||||
|  | ||||
| 	/** | ||||
| 	 * 状态码 | ||||
| 	 */ | ||||
| 	@ApiModelProperty("消息状态码") | ||||
| 	private int code; | ||||
|  | ||||
| 	/** | ||||
| 	 * 返回内容 | ||||
| 	 */ | ||||
| 	@ApiModelProperty("消息内容") | ||||
| 	private String msg; | ||||
|  | ||||
| 	/** | ||||
| 	 * 数据对象 | ||||
| 	 */ | ||||
| 	@ApiModelProperty("数据对象") | ||||
| 	private T data; | ||||
|  | ||||
| 	/** | ||||
| 	 * 初始化一个新创建的 AjaxResult 对象 | ||||
| 	 * | ||||
| 	 * @param code 状态码 | ||||
| 	 * @param msg  返回内容 | ||||
| 	 */ | ||||
| 	public AjaxResult(int code, String msg) { | ||||
| 		this.code = code; | ||||
| 		this.msg = msg; | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 返回成功消息 | ||||
| 	 * | ||||
| 	 * @return 成功消息 | ||||
| 	 */ | ||||
| 	public static AjaxResult<Void> success() { | ||||
| 		return AjaxResult.success("操作成功"); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 返回成功数据 | ||||
| 	 * | ||||
| 	 * @return 成功消息 | ||||
| 	 */ | ||||
| 	public static <T> AjaxResult<T> success(T data) { | ||||
| 		return AjaxResult.success("操作成功", data); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 返回成功消息 | ||||
| 	 * | ||||
| 	 * @param msg 返回内容 | ||||
| 	 * @return 成功消息 | ||||
| 	 */ | ||||
| 	public static AjaxResult<Void> success(String msg) { | ||||
| 		return AjaxResult.success(msg, null); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 返回成功消息 | ||||
| 	 * | ||||
| 	 * @param msg  返回内容 | ||||
| 	 * @param data 数据对象 | ||||
| 	 * @return 成功消息 | ||||
| 	 */ | ||||
| 	public static <T> AjaxResult<T> success(String msg, T data) { | ||||
| 		return new AjaxResult<>(HttpStatus.HTTP_OK, msg, data); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 返回错误消息 | ||||
| 	 * | ||||
| 	 * @return | ||||
| 	 */ | ||||
| 	public static AjaxResult<Void> error() { | ||||
| 		return AjaxResult.error("操作失败"); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 返回错误消息 | ||||
| 	 * | ||||
| 	 * @param msg 返回内容 | ||||
| 	 * @return 警告消息 | ||||
| 	 */ | ||||
| 	public static AjaxResult<Void> error(String msg) { | ||||
| 		return AjaxResult.error(msg, null); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 返回错误消息 | ||||
| 	 * | ||||
| 	 * @param msg  返回内容 | ||||
| 	 * @param data 数据对象 | ||||
| 	 * @return 警告消息 | ||||
| 	 */ | ||||
| 	public static <T> AjaxResult<T> error(String msg, T data) { | ||||
| 		return new AjaxResult<>(HttpStatus.HTTP_INTERNAL_ERROR, msg, data); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 返回错误消息 | ||||
| 	 * | ||||
| 	 * @param code 状态码 | ||||
| 	 * @param msg  返回内容 | ||||
| 	 * @return 警告消息 | ||||
| 	 */ | ||||
| 	public static AjaxResult<Void> error(int code, String msg) { | ||||
| 		return new AjaxResult<>(code, msg, null); | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,46 @@ | ||||
| package com.ruoyi.common.core.domain; | ||||
|  | ||||
| import lombok.Data; | ||||
| import lombok.NoArgsConstructor; | ||||
| import lombok.experimental.Accessors; | ||||
|  | ||||
| import java.io.Serializable; | ||||
| import java.util.Date; | ||||
| import java.util.HashMap; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * Entity基类 | ||||
|  * | ||||
|  * @author ruoyi | ||||
|  */ | ||||
|  | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| @Accessors(chain = true) | ||||
| public class BaseEntity implements Serializable | ||||
| { | ||||
|     private static final long serialVersionUID = 1L; | ||||
|  | ||||
|     /** 搜索值 */ | ||||
|     private String searchValue; | ||||
|  | ||||
|     /** 创建者 */ | ||||
|     private String createBy; | ||||
|  | ||||
|     /** 创建时间 */ | ||||
|     private Date createTime; | ||||
|  | ||||
|     /** 更新者 */ | ||||
|     private String updateBy; | ||||
|  | ||||
|     /** 更新时间 */ | ||||
|     private Date updateTime; | ||||
|  | ||||
|     /** 备注 */ | ||||
|     private String remark; | ||||
|  | ||||
|     /** 请求参数 */ | ||||
|     private Map<String, Object> params = new HashMap<>(); | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,38 @@ | ||||
| package com.ruoyi.common.core.domain; | ||||
|  | ||||
| import lombok.*; | ||||
| import lombok.experimental.Accessors; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
|  | ||||
| /** | ||||
|  * Tree基类 | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
|  | ||||
| @EqualsAndHashCode(callSuper = true) | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| @Accessors(chain = true) | ||||
| 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<>(); | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,50 @@ | ||||
| package com.ruoyi.common.core.domain; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
| import com.ruoyi.common.core.domain.entity.SysDept; | ||||
| import com.ruoyi.common.core.domain.entity.SysMenu; | ||||
| import lombok.*; | ||||
| import lombok.experimental.Accessors; | ||||
|  | ||||
| import java.io.Serializable; | ||||
| import java.util.List; | ||||
| import java.util.stream.Collectors; | ||||
|  | ||||
| /** | ||||
|  * Treeselect树结构实体类 | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
|  | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| @Accessors(chain = true) | ||||
| 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<TreeSelect> children; | ||||
|  | ||||
|     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()); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,99 @@ | ||||
| package com.ruoyi.common.core.domain.entity; | ||||
|  | ||||
| import com.baomidou.mybatisplus.annotation.*; | ||||
| import com.fasterxml.jackson.annotation.JsonFormat; | ||||
| import lombok.Data; | ||||
| import lombok.NoArgsConstructor; | ||||
| import lombok.experimental.Accessors; | ||||
|  | ||||
| import javax.validation.constraints.Email; | ||||
| import javax.validation.constraints.NotBlank; | ||||
| import javax.validation.constraints.Size; | ||||
| import java.io.Serializable; | ||||
| import java.util.*; | ||||
|  | ||||
| /** | ||||
|  * 部门表 sys_dept | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
|  | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| @Accessors(chain = true) | ||||
| @TableName("sys_dept") | ||||
| public class SysDept implements Serializable | ||||
| { | ||||
|     private static final long serialVersionUID = 1L; | ||||
|  | ||||
|     /** 部门ID */ | ||||
|     @TableId(value = "dept_id",type = IdType.AUTO) | ||||
|     private Long deptId; | ||||
|  | ||||
|     /** 父部门ID */ | ||||
|     private Long parentId; | ||||
|  | ||||
|     /** 祖级列表 */ | ||||
|     private String ancestors; | ||||
|  | ||||
|     /** 部门名称 */ | ||||
|     @NotBlank(message = "部门名称不能为空") | ||||
|     @Size(min = 0, max = 30, message = "部门名称长度不能超过30个字符") | ||||
|     private String deptName; | ||||
|  | ||||
|     /** 显示顺序 */ | ||||
|     @NotBlank(message = "显示顺序不能为空") | ||||
|     private String orderNum; | ||||
|  | ||||
|     /** 负责人 */ | ||||
|     private String leader; | ||||
|  | ||||
|     /** 联系电话 */ | ||||
|     @Size(min = 0, max = 11, message = "联系电话长度不能超过11个字符") | ||||
|     private String phone; | ||||
|  | ||||
|     /** 邮箱 */ | ||||
|     @Email(message = "邮箱格式不正确") | ||||
|     @Size(min = 0, max = 50, message = "邮箱长度不能超过50个字符") | ||||
|     private String email; | ||||
|  | ||||
|     /** 部门状态:0正常,1停用 */ | ||||
|     private String status; | ||||
|  | ||||
|     /** 删除标志(0代表存在 2代表删除) */ | ||||
|     @TableLogic | ||||
|     private String delFlag; | ||||
|  | ||||
|     /** 父部门名称 */ | ||||
|     @TableField(exist = false) | ||||
|     private String parentName; | ||||
|  | ||||
|     /** 创建者 */ | ||||
|     @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; | ||||
|      | ||||
|     /** 子部门 */ | ||||
|     @TableField(exist = false) | ||||
|     private List<SysDept> children = new ArrayList<SysDept>(); | ||||
|  | ||||
|     /** | ||||
|      * 请求参数 | ||||
|      */ | ||||
|     @TableField(exist = false) | ||||
|     private Map<String, Object> params = new HashMap<>(); | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,107 @@ | ||||
| package com.ruoyi.common.core.domain.entity; | ||||
|  | ||||
| import com.baomidou.mybatisplus.annotation.*; | ||||
| import com.fasterxml.jackson.annotation.JsonFormat; | ||||
| import com.ruoyi.common.annotation.Excel; | ||||
| import com.ruoyi.common.annotation.Excel.ColumnType; | ||||
| import com.ruoyi.common.constant.UserConstants; | ||||
| import lombok.Data; | ||||
| import lombok.NoArgsConstructor; | ||||
| import lombok.experimental.Accessors; | ||||
|  | ||||
| import javax.validation.constraints.NotBlank; | ||||
| import javax.validation.constraints.Size; | ||||
| import java.io.Serializable; | ||||
| import java.util.Date; | ||||
| import java.util.HashMap; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * 字典数据表 sys_dict_data | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
|  | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| @Accessors(chain = true) | ||||
| @TableName("sys_dict_data") | ||||
| public class SysDictData implements Serializable | ||||
| { | ||||
|     private static final long serialVersionUID = 1L; | ||||
|  | ||||
|     /** 字典编码 */ | ||||
|     @Excel(name = "字典编码", cellType = ColumnType.NUMERIC) | ||||
|     @TableId(value = "dict_code",type = IdType.AUTO) | ||||
|     private Long dictCode; | ||||
|  | ||||
|     /** 字典排序 */ | ||||
|     @Excel(name = "字典排序", cellType = ColumnType.NUMERIC) | ||||
|     private Long dictSort; | ||||
|  | ||||
|     /** 字典标签 */ | ||||
|     @Excel(name = "字典标签") | ||||
|     @NotBlank(message = "字典标签不能为空") | ||||
|     @Size(min = 0, max = 100, message = "字典标签长度不能超过100个字符") | ||||
|     private String dictLabel; | ||||
|  | ||||
|     /** 字典键值 */ | ||||
|     @Excel(name = "字典键值") | ||||
|     @NotBlank(message = "字典键值不能为空") | ||||
|     @Size(min = 0, max = 100, message = "字典键值长度不能超过100个字符") | ||||
|     private String dictValue; | ||||
|  | ||||
|     /** 字典类型 */ | ||||
|     @Excel(name = "字典类型") | ||||
|     @NotBlank(message = "字典类型不能为空") | ||||
|     @Size(min = 0, max = 100, message = "字典类型长度不能超过100个字符") | ||||
|     private String dictType; | ||||
|  | ||||
|     /** 样式属性(其他样式扩展) */ | ||||
|     @Size(min = 0, max = 100, message = "样式属性长度不能超过100个字符") | ||||
|     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; | ||||
|  | ||||
|     /** 创建者 */ | ||||
|     @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; | ||||
|  | ||||
|     /** 备注 */ | ||||
|     private String remark; | ||||
|  | ||||
|     /** | ||||
|      * 请求参数 | ||||
|      */ | ||||
|     @TableField(exist = false) | ||||
|     private Map<String, Object> params = new HashMap<>(); | ||||
|  | ||||
|     public boolean getDefault() | ||||
|     { | ||||
|         return UserConstants.YES.equals(this.isDefault) ? true : false; | ||||
|     } | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,80 @@ | ||||
| package com.ruoyi.common.core.domain.entity; | ||||
|  | ||||
| import com.baomidou.mybatisplus.annotation.*; | ||||
| import com.fasterxml.jackson.annotation.JsonFormat; | ||||
| import com.ruoyi.common.annotation.Excel; | ||||
| import com.ruoyi.common.annotation.Excel.ColumnType; | ||||
| import lombok.Data; | ||||
| import lombok.NoArgsConstructor; | ||||
| import lombok.experimental.Accessors; | ||||
|  | ||||
| import javax.validation.constraints.NotBlank; | ||||
| import javax.validation.constraints.Size; | ||||
| import java.io.Serializable; | ||||
| import java.util.Date; | ||||
| import java.util.HashMap; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * 字典类型表 sys_dict_type | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
|  | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| @Accessors(chain = true) | ||||
| @TableName("sys_dict_type") | ||||
| public class SysDictType implements Serializable | ||||
| { | ||||
|     private static final long serialVersionUID = 1L; | ||||
|  | ||||
|     /** 字典主键 */ | ||||
|     @Excel(name = "字典主键", cellType = ColumnType.NUMERIC) | ||||
|     @TableId(value = "dict_id",type = IdType.AUTO) | ||||
|     private Long dictId; | ||||
|  | ||||
|     /** 字典名称 */ | ||||
|     @Excel(name = "字典名称") | ||||
|     @NotBlank(message = "字典名称不能为空") | ||||
|     @Size(min = 0, max = 100, message = "字典类型名称长度不能超过100个字符") | ||||
|     private String dictName; | ||||
|  | ||||
|     /** 字典类型 */ | ||||
|     @Excel(name = "字典类型") | ||||
|     @NotBlank(message = "字典类型不能为空") | ||||
|     @Size(min = 0, max = 100, message = "字典类型类型长度不能超过100个字符") | ||||
|     private String dictType; | ||||
|  | ||||
|     /** 状态(0正常 1停用) */ | ||||
|     @Excel(name = "状态", readConverterExp = "0=正常,1=停用") | ||||
|     private String status; | ||||
|  | ||||
|     /** 创建者 */ | ||||
|     @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; | ||||
|  | ||||
|     /** 备注 */ | ||||
|     private String remark; | ||||
|  | ||||
|     /** | ||||
|      * 请求参数 | ||||
|      */ | ||||
|     @TableField(exist = false) | ||||
|     private Map<String, Object> params = new HashMap<>(); | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,108 @@ | ||||
| package com.ruoyi.common.core.domain.entity; | ||||
|  | ||||
| import com.baomidou.mybatisplus.annotation.*; | ||||
| import com.fasterxml.jackson.annotation.JsonFormat; | ||||
| import lombok.Data; | ||||
| import lombok.NoArgsConstructor; | ||||
| import lombok.experimental.Accessors; | ||||
|  | ||||
| import javax.validation.constraints.NotBlank; | ||||
| import javax.validation.constraints.Size; | ||||
| import java.io.Serializable; | ||||
| import java.util.*; | ||||
|  | ||||
| /** | ||||
|  * 菜单权限表 sys_menu | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
|  | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| @Accessors(chain = true) | ||||
| @TableName("sys_menu") | ||||
| public class SysMenu implements Serializable | ||||
| { | ||||
|     private static final long serialVersionUID = 1L; | ||||
|  | ||||
|     /** 菜单ID */ | ||||
|     @TableId(value = "menu_id",type = IdType.AUTO) | ||||
|     private Long menuId; | ||||
|  | ||||
|     /** 菜单名称 */ | ||||
|     @NotBlank(message = "菜单名称不能为空") | ||||
|     @Size(min = 0, max = 50, message = "菜单名称长度不能超过50个字符") | ||||
|     private String menuName; | ||||
|  | ||||
|     /** 父菜单名称 */ | ||||
|     @TableField(exist = false) | ||||
|     private String parentName; | ||||
|  | ||||
|     /** 父菜单ID */ | ||||
|     private Long parentId; | ||||
|  | ||||
|     /** 显示顺序 */ | ||||
|     @NotBlank(message = "显示顺序不能为空") | ||||
|     private String orderNum; | ||||
|  | ||||
|     /** 路由地址 */ | ||||
|     @Size(min = 0, max = 200, message = "路由地址不能超过200个字符") | ||||
|     private String path; | ||||
|  | ||||
|     /** 组件路径 */ | ||||
|     @Size(min = 0, max = 200, message = "组件路径不能超过255个字符") | ||||
|     private String component; | ||||
|  | ||||
|     /** 是否为外链(0是 1否) */ | ||||
|     private String isFrame; | ||||
|  | ||||
|     /** 是否缓存(0缓存 1不缓存) */ | ||||
|     private String isCache; | ||||
|  | ||||
|     /** 类型(M目录 C菜单 F按钮) */ | ||||
|     @NotBlank(message = "菜单类型不能为空") | ||||
|     private String menuType; | ||||
|  | ||||
|     /** 显示状态(0显示 1隐藏) */ | ||||
|     private String visible; | ||||
|      | ||||
|     /** 菜单状态(0显示 1隐藏) */ | ||||
|     private String status; | ||||
|  | ||||
|     /** 权限字符串 */ | ||||
|     @Size(min = 0, max = 100, message = "权限标识长度不能超过100个字符") | ||||
|     private String perms; | ||||
|  | ||||
|     /** 菜单图标 */ | ||||
|     private String icon; | ||||
|  | ||||
|     /** 创建者 */ | ||||
|     @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; | ||||
|  | ||||
|     /** 备注 */ | ||||
|     private String remark; | ||||
|  | ||||
|     /** 请求参数 */ | ||||
|     @TableField(exist = false) | ||||
|     private Map<String, Object> params = new HashMap<>(); | ||||
|  | ||||
|     /** 子菜单 */ | ||||
|     @TableField(exist = false) | ||||
|     private List<SysMenu> children = new ArrayList<SysMenu>(); | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,126 @@ | ||||
| package com.ruoyi.common.core.domain.entity; | ||||
|  | ||||
| import com.baomidou.mybatisplus.annotation.*; | ||||
| import com.fasterxml.jackson.annotation.JsonFormat; | ||||
| import com.ruoyi.common.annotation.Excel; | ||||
| import com.ruoyi.common.annotation.Excel.ColumnType; | ||||
| import lombok.Data; | ||||
| import lombok.NoArgsConstructor; | ||||
| import lombok.experimental.Accessors; | ||||
|  | ||||
| import javax.validation.constraints.NotBlank; | ||||
| import javax.validation.constraints.Size; | ||||
| import java.io.Serializable; | ||||
| import java.util.Date; | ||||
| import java.util.HashMap; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * 角色表 sys_role | ||||
|  * | ||||
|  * @author ruoyi | ||||
|  */ | ||||
|  | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| @Accessors(chain = true) | ||||
| @TableName("sys_role") | ||||
| public class SysRole implements Serializable | ||||
| { | ||||
|     private static final long serialVersionUID = 1L; | ||||
|  | ||||
|     /** 角色ID */ | ||||
|     @Excel(name = "角色序号", cellType = ColumnType.NUMERIC) | ||||
|     @TableId(value = "role_id",type = IdType.AUTO) | ||||
|     private Long roleId; | ||||
|  | ||||
|     /** 角色名称 */ | ||||
|     @Excel(name = "角色名称") | ||||
|     @NotBlank(message = "角色名称不能为空") | ||||
|     @Size(min = 0, max = 30, message = "角色名称长度不能超过30个字符") | ||||
|     private String roleName; | ||||
|  | ||||
|     /** 角色权限 */ | ||||
|     @Excel(name = "角色权限") | ||||
|     @NotBlank(message = "权限字符不能为空") | ||||
|     @Size(min = 0, max = 100, message = "权限字符长度不能超过100个字符") | ||||
|     private String roleKey; | ||||
|  | ||||
|     /** 角色排序 */ | ||||
|     @Excel(name = "角色排序") | ||||
|     @NotBlank(message = "显示顺序不能为空") | ||||
|     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代表删除) */ | ||||
|     @TableLogic | ||||
|     private String delFlag; | ||||
|  | ||||
|     /** 创建者 */ | ||||
|     @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; | ||||
|  | ||||
|     /** 备注 */ | ||||
|     private String remark; | ||||
|  | ||||
|     /** | ||||
|      * 请求参数 | ||||
|      */ | ||||
|     @TableField(exist = false) | ||||
|     private Map<String, Object> params = new HashMap<>(); | ||||
|  | ||||
|     /** 用户是否存在此角色标识 默认不存在 */ | ||||
|     @TableField(exist = false) | ||||
|     private boolean flag = false; | ||||
|  | ||||
|     /** 菜单组 */ | ||||
|     @TableField(exist = false) | ||||
|     private Long[] menuIds; | ||||
|  | ||||
|     /** 部门组(数据权限) */ | ||||
|     @TableField(exist = false) | ||||
|     private Long[] deptIds; | ||||
|  | ||||
|     public SysRole(Long roleId) | ||||
|     { | ||||
|         this.roleId = roleId; | ||||
|     } | ||||
|  | ||||
|     public boolean isAdmin() | ||||
|     { | ||||
|         return isAdmin(this.roleId); | ||||
|     } | ||||
|  | ||||
|     public static boolean isAdmin(Long roleId) | ||||
|     { | ||||
|         return roleId != null && 1L == roleId; | ||||
|     } | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,166 @@ | ||||
| package com.ruoyi.common.core.domain.entity; | ||||
|  | ||||
| import com.baomidou.mybatisplus.annotation.*; | ||||
| import com.fasterxml.jackson.annotation.JsonFormat; | ||||
| import com.fasterxml.jackson.annotation.JsonIgnore; | ||||
| import com.fasterxml.jackson.annotation.JsonProperty; | ||||
| 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 lombok.Data; | ||||
| import lombok.NoArgsConstructor; | ||||
| import lombok.experimental.Accessors; | ||||
|  | ||||
| import javax.validation.constraints.Email; | ||||
| import javax.validation.constraints.NotBlank; | ||||
| import javax.validation.constraints.Size; | ||||
| import java.io.Serializable; | ||||
| import java.util.Date; | ||||
| import java.util.HashMap; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * 用户对象 sys_user | ||||
|  * | ||||
|  * @author ruoyi | ||||
|  */ | ||||
|  | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| @Accessors(chain = true) | ||||
| @TableName("sys_user") | ||||
| public class SysUser implements Serializable | ||||
| { | ||||
|     private static final long serialVersionUID = 1L; | ||||
|  | ||||
|     /** 用户ID */ | ||||
|     @Excel(name = "用户序号", cellType = ColumnType.NUMERIC, prompt = "用户编号") | ||||
|     @TableId(value = "user_id",type = IdType.AUTO) | ||||
|     private Long userId; | ||||
|  | ||||
|     /** 部门ID */ | ||||
|     @Excel(name = "部门编号", type = Type.IMPORT) | ||||
|     private Long deptId; | ||||
|  | ||||
|     /** 用户账号 */ | ||||
|     @NotBlank(message = "用户账号不能为空") | ||||
|     @Size(min = 0, max = 30, message = "用户账号长度不能超过30个字符") | ||||
|     @Excel(name = "登录名称") | ||||
|     private String userName; | ||||
|  | ||||
|     /** 用户昵称 */ | ||||
|     @Size(min = 0, max = 30, message = "用户昵称长度不能超过30个字符") | ||||
|     @Excel(name = "用户名称") | ||||
|     private String nickName; | ||||
|  | ||||
|     /** 用户邮箱 */ | ||||
|     @Email(message = "邮箱格式不正确") | ||||
|     @Size(min = 0, max = 50, message = "邮箱长度不能超过50个字符") | ||||
|     @Excel(name = "用户邮箱") | ||||
|     private String email; | ||||
|  | ||||
|     /** 手机号码 */ | ||||
|     @Size(min = 0, max = 11, message = "手机号码长度不能超过11个字符") | ||||
|     @Excel(name = "手机号码") | ||||
|     private String phonenumber; | ||||
|  | ||||
|     /** 用户性别 */ | ||||
|     @Excel(name = "用户性别", readConverterExp = "0=男,1=女,2=未知") | ||||
|     private String sex; | ||||
|  | ||||
|     /** 用户头像 */ | ||||
|     private String avatar; | ||||
|  | ||||
|     /** 密码 */ | ||||
|     private String password; | ||||
|  | ||||
|     @JsonIgnore | ||||
|     @JsonProperty | ||||
|     public String getPassword() { | ||||
|         return password; | ||||
|     } | ||||
|  | ||||
|     /** 盐加密 */ | ||||
|     private String salt; | ||||
|  | ||||
|     /** 帐号状态(0正常 1停用) */ | ||||
|     @Excel(name = "帐号状态", readConverterExp = "0=正常,1=停用") | ||||
|     private String status; | ||||
|  | ||||
|     /** 删除标志(0代表存在 2代表删除) */ | ||||
|     @TableLogic | ||||
|     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; | ||||
|  | ||||
|     /** 创建者 */ | ||||
|     @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; | ||||
|  | ||||
|     /** 备注 */ | ||||
|     private String remark; | ||||
|  | ||||
|     /** | ||||
|      * 请求参数 | ||||
|      */ | ||||
|     @TableField(exist = false) | ||||
|     private Map<String, Object> params = new HashMap<>(); | ||||
|  | ||||
|     /** 部门对象 */ | ||||
|     @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<SysRole> roles; | ||||
|  | ||||
|     /** 角色组 */ | ||||
|     @TableField(exist = false) | ||||
|     private Long[] roleIds; | ||||
|  | ||||
|     /** 岗位组 */ | ||||
|     @TableField(exist = false) | ||||
|     private Long[] postIds; | ||||
|  | ||||
|     public SysUser(Long userId) | ||||
|     { | ||||
|         this.userId = userId; | ||||
|     } | ||||
|  | ||||
|     public boolean isAdmin() | ||||
|     { | ||||
|         return isAdmin(this.userId); | ||||
|     } | ||||
|  | ||||
|     public static boolean isAdmin(Long userId) | ||||
|     { | ||||
|         return userId != null && 1L == userId; | ||||
|     } | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,37 @@ | ||||
| package com.ruoyi.common.core.domain.model; | ||||
|  | ||||
| import lombok.*; | ||||
| import lombok.experimental.Accessors; | ||||
|  | ||||
| /** | ||||
|  * 用户登录对象 | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
|  | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| @Accessors(chain = true) | ||||
| public class LoginBody | ||||
| { | ||||
|     /** | ||||
|      * 用户名 | ||||
|      */ | ||||
|     private String username; | ||||
|  | ||||
|     /** | ||||
|      * 用户密码 | ||||
|      */ | ||||
|     private String password; | ||||
|  | ||||
|     /** | ||||
|      * 验证码 | ||||
|      */ | ||||
|     private String code; | ||||
|  | ||||
|     /** | ||||
|      * 唯一标识 | ||||
|      */ | ||||
|     private String uuid = ""; | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,143 @@ | ||||
| package com.ruoyi.common.core.domain.model; | ||||
|  | ||||
| import com.fasterxml.jackson.annotation.JsonIgnore; | ||||
| import com.ruoyi.common.core.domain.entity.SysUser; | ||||
| import lombok.*; | ||||
| import lombok.experimental.Accessors; | ||||
| import org.springframework.security.core.GrantedAuthority; | ||||
| import org.springframework.security.core.userdetails.UserDetails; | ||||
|  | ||||
| import java.util.Collection; | ||||
| import java.util.Set; | ||||
|  | ||||
| /** | ||||
|  * 登录用户身份权限 | ||||
|  * | ||||
|  * @author ruoyi | ||||
|  */ | ||||
|  | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| @Accessors(chain = true) | ||||
| public class LoginUser implements UserDetails | ||||
| { | ||||
|     private static final long serialVersionUID = 1L; | ||||
|  | ||||
|     /** | ||||
|      * 用户唯一标识 | ||||
|      */ | ||||
|     private String token; | ||||
|  | ||||
|     /** | ||||
|      * 登录时间 | ||||
|      */ | ||||
|     private Long loginTime; | ||||
|  | ||||
|     /** | ||||
|      * 过期时间 | ||||
|      */ | ||||
|     private Long expireTime; | ||||
|  | ||||
|     /** | ||||
|      * 登录IP地址 | ||||
|      */ | ||||
|     private String ipaddr; | ||||
|  | ||||
|     /** | ||||
|      * 登录地点 | ||||
|      */ | ||||
|     private String loginLocation; | ||||
|  | ||||
|     /** | ||||
|      * 浏览器类型 | ||||
|      */ | ||||
|     private String browser; | ||||
|  | ||||
|     /** | ||||
|      * 操作系统 | ||||
|      */ | ||||
|     private String os; | ||||
|  | ||||
|     /** | ||||
|      * 权限列表 | ||||
|      */ | ||||
|     private Set<String> permissions; | ||||
|  | ||||
|     /** | ||||
|      * 用户信息 | ||||
|      */ | ||||
|     private SysUser user; | ||||
|  | ||||
|     public LoginUser(SysUser user, Set<String> permissions) | ||||
|     { | ||||
|         this.user = user; | ||||
|         this.permissions = permissions; | ||||
|     } | ||||
|  | ||||
|     @JsonIgnore | ||||
|     @Override | ||||
|     public String getPassword() | ||||
|     { | ||||
|         return user.getPassword(); | ||||
|     } | ||||
|  | ||||
| 	@JsonIgnore | ||||
|     @Override | ||||
|     public String getUsername() | ||||
|     { | ||||
|         return user.getUserName(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 账户是否未过期,过期无法验证 | ||||
|      */ | ||||
|     @JsonIgnore | ||||
|     @Override | ||||
|     public boolean isAccountNonExpired() | ||||
|     { | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 指定用户是否解锁,锁定的用户无法进行身份验证 | ||||
|      * | ||||
|      * @return | ||||
|      */ | ||||
|     @JsonIgnore | ||||
|     @Override | ||||
|     public boolean isAccountNonLocked() | ||||
|     { | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 指示是否已过期的用户的凭据(密码),过期的凭据防止认证 | ||||
|      * | ||||
|      * @return | ||||
|      */ | ||||
|     @JsonIgnore | ||||
|     @Override | ||||
|     public boolean isCredentialsNonExpired() | ||||
|     { | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 是否可用 ,禁用的用户不能身份验证 | ||||
|      * | ||||
|      * @return | ||||
|      */ | ||||
|     @JsonIgnore | ||||
|     @Override | ||||
|     public boolean isEnabled() | ||||
|     { | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
| 	@JsonIgnore | ||||
|     @Override | ||||
|     public Collection<? extends GrantedAuthority> getAuthorities() | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										102
									
								
								ruoyi-common/src/main/java/com/ruoyi/common/core/mybatisplus/cache/MybatisPlusRedisCache.java
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										102
									
								
								ruoyi-common/src/main/java/com/ruoyi/common/core/mybatisplus/cache/MybatisPlusRedisCache.java
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,102 @@ | ||||
| package com.ruoyi.common.core.mybatisplus.cache; | ||||
|  | ||||
| import cn.hutool.extra.spring.SpringUtil; | ||||
| import com.ruoyi.common.core.redis.RedisCache; | ||||
| import lombok.extern.slf4j.Slf4j; | ||||
| import org.apache.ibatis.cache.Cache; | ||||
| import org.springframework.data.redis.connection.RedisServerCommands; | ||||
| import org.springframework.data.redis.core.RedisTemplate; | ||||
| import org.springframework.util.CollectionUtils; | ||||
|  | ||||
| import java.util.Collection; | ||||
| import java.util.concurrent.locks.ReadWriteLock; | ||||
| import java.util.concurrent.locks.ReentrantReadWriteLock; | ||||
|  | ||||
| /** | ||||
|  * mybatis-redis 二级缓存 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Slf4j | ||||
| public class MybatisPlusRedisCache implements Cache { | ||||
|  | ||||
| 	private final ReadWriteLock readWriteLock = new ReentrantReadWriteLock(true); | ||||
|  | ||||
| 	private RedisCache redisCache; | ||||
|  | ||||
| 	private String id; | ||||
|  | ||||
| 	public MybatisPlusRedisCache(final String id) { | ||||
| 		if (id == null) { | ||||
| 			throw new IllegalArgumentException("Cache instances require an ID"); | ||||
| 		} | ||||
| 		this.id = id; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public String getId() { | ||||
| 		return this.id; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void putObject(Object key, Object value) { | ||||
| 		if (redisCache == null) { | ||||
| 			redisCache = SpringUtil.getBean(RedisCache.class); | ||||
| 		} | ||||
| 		if (value != null) { | ||||
| 			redisCache.setCacheObject(key.toString(), value); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object getObject(Object key) { | ||||
| 		if (redisCache == null) { | ||||
| 			//由于启动期间注入失败,只能运行期间注入,这段代码可以删除 | ||||
| 			redisCache = SpringUtil.getBean(RedisCache.class); | ||||
| 		} | ||||
| 		try { | ||||
| 			if (key != null) { | ||||
| 				return redisCache.getCacheObject(key.toString()); | ||||
| 			} | ||||
| 		} catch (Exception e) { | ||||
| 			e.printStackTrace(); | ||||
| 			log.error("缓存出错"); | ||||
| 		} | ||||
| 		return null; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public Object removeObject(Object key) { | ||||
| 		if (redisCache == null) { | ||||
| 			redisCache = SpringUtil.getBean(RedisCache.class); | ||||
| 		} | ||||
| 		if (key != null) { | ||||
| 			redisCache.deleteObject(key.toString()); | ||||
| 		} | ||||
| 		return null; | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public void clear() { | ||||
| 		log.debug("清空缓存"); | ||||
| 		if (redisCache == null) { | ||||
| 			redisCache = SpringUtil.getBean(RedisCache.class); | ||||
| 		} | ||||
| 		Collection<String> keys = redisCache.keys("*:" + this.id + "*"); | ||||
| 		if (!CollectionUtils.isEmpty(keys)) { | ||||
| 			redisCache.deleteObject(keys); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public int getSize() { | ||||
| 		RedisTemplate<String, Object> redisTemplate = SpringUtil.getBean("redisTemplate"); | ||||
| 		Long size = redisTemplate.execute(RedisServerCommands::dbSize); | ||||
| 		return size.intValue(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public ReadWriteLock getReadWriteLock() { | ||||
| 		return this.readWriteLock; | ||||
| 	} | ||||
| } | ||||
| @ -0,0 +1,18 @@ | ||||
| package com.ruoyi.common.core.mybatisplus.core; | ||||
|  | ||||
| import com.baomidou.mybatisplus.core.mapper.BaseMapper; | ||||
| import org.apache.ibatis.annotations.Param; | ||||
|  | ||||
| import java.util.Collection; | ||||
|  | ||||
| /** | ||||
|  * 自定义 Mapper 接口, 实现 自定义扩展 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  * @since 2021-05-13 | ||||
|  */ | ||||
| public interface BaseMapperPlus<T> extends BaseMapper<T> { | ||||
|  | ||||
| 	int insertAll(@Param("list") Collection<T> batchList); | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,245 @@ | ||||
| package com.ruoyi.common.core.mybatisplus.core; | ||||
|  | ||||
| import cn.hutool.core.bean.BeanUtil; | ||||
| import com.baomidou.mybatisplus.core.conditions.Wrapper; | ||||
| import com.baomidou.mybatisplus.core.toolkit.Wrappers; | ||||
| import com.baomidou.mybatisplus.extension.service.IService; | ||||
| import com.ruoyi.common.core.page.PagePlus; | ||||
|  | ||||
| import java.io.Serializable; | ||||
| import java.util.Collection; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.function.Function; | ||||
| import java.util.stream.Collectors; | ||||
|  | ||||
| /** | ||||
|  * 自定义 Service 接口, 实现 数据库实体与 vo 对象转换返回 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  * @since 2021-05-13 | ||||
|  */ | ||||
| public interface IServicePlus<T> extends IService<T> { | ||||
|  | ||||
|     /** | ||||
|      * 根据 ID 查询 | ||||
|      * | ||||
|      * @param kClass vo类型 | ||||
|      * @param id     主键ID | ||||
|      */ | ||||
|     default <K> K getVoById(Serializable id, Class<K> kClass) { | ||||
|         T t = getBaseMapper().selectById(id); | ||||
|         return BeanUtil.toBean(t, kClass); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据 ID 查询 | ||||
|      * | ||||
|      * @param id        主键ID | ||||
|      * @param convertor 转换函数 | ||||
|      * @param <K>       vo类型 | ||||
|      */ | ||||
|     default <K> K getVoById(Serializable id, Function<T, K> convertor) { | ||||
|         T t = getBaseMapper().selectById(id); | ||||
|         return convertor.apply(t); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 查询(根据ID 批量查询) | ||||
|      * | ||||
|      * @param kClass vo类型 | ||||
|      * @param idList 主键ID列表 | ||||
|      */ | ||||
|     default <K> List<K> listVoByIds(Collection<? extends Serializable> idList, Class<K> kClass) { | ||||
|         List<T> list = getBaseMapper().selectBatchIds(idList); | ||||
|         if (list == null) { | ||||
|             return null; | ||||
|         } | ||||
|         return list.stream() | ||||
|                 .map(any -> BeanUtil.toBean(any, kClass)) | ||||
|                 .collect(Collectors.toList()); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 查询(根据ID 批量查询) | ||||
|      * | ||||
|      * @param convertor 转换函数 | ||||
|      * @param idList    主键ID列表 | ||||
|      */ | ||||
|     default <K> List<K> listVoByIds(Collection<? extends Serializable> idList, | ||||
|                                     Function<Collection<T>, List<K>> convertor) { | ||||
|         List<T> list = getBaseMapper().selectBatchIds(idList); | ||||
|         if (list == null) { | ||||
|             return null; | ||||
|         } | ||||
|         return convertor.apply(list); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 查询(根据 columnMap 条件) | ||||
|      * | ||||
|      * @param kClass    vo类型 | ||||
|      * @param columnMap 表字段 map 对象 | ||||
|      */ | ||||
|     default <K> List<K> listVoByMap(Map<String, Object> columnMap, Class<K> kClass) { | ||||
|         List<T> list = getBaseMapper().selectByMap(columnMap); | ||||
|         if (list == null) { | ||||
|             return null; | ||||
|         } | ||||
|         return list.stream() | ||||
|                 .map(any -> BeanUtil.toBean(any, kClass)) | ||||
|                 .collect(Collectors.toList()); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 查询(根据 columnMap 条件) | ||||
|      * | ||||
|      * @param convertor 转换函数 | ||||
|      * @param columnMap 表字段 map 对象 | ||||
|      */ | ||||
|     default <K> List<K> listVoByMap(Map<String, Object> columnMap, | ||||
|                                     Function<Collection<T>, List<K>> convertor) { | ||||
|         List<T> list = getBaseMapper().selectByMap(columnMap); | ||||
|         if (list == null) { | ||||
|             return null; | ||||
|         } | ||||
|         return convertor.apply(list); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据 Wrapper,查询一条记录 <br/> | ||||
|      * <p>结果集,如果是多个会抛出异常,随机取一条加上限制条件 wrapper.last("LIMIT 1")</p> | ||||
|      * | ||||
|      * @param kClass       vo类型 | ||||
|      * @param queryWrapper 实体对象封装操作类 {@link com.baomidou.mybatisplus.core.conditions.query.QueryWrapper} | ||||
|      */ | ||||
|     default <K> K getVoOne(Wrapper<T> queryWrapper, Class<K> kClass) { | ||||
|         return BeanUtil.toBean(getOne(queryWrapper, true), kClass); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据 Wrapper,查询一条记录 <br/> | ||||
|      * <p>结果集,如果是多个会抛出异常,随机取一条加上限制条件 wrapper.last("LIMIT 1")</p> | ||||
|      * | ||||
|      * @param convertor    转换函数 | ||||
|      * @param queryWrapper 实体对象封装操作类 {@link com.baomidou.mybatisplus.core.conditions.query.QueryWrapper} | ||||
|      */ | ||||
|     default <K> K getVoOne(Wrapper<T> queryWrapper, Function<T, K> convertor) { | ||||
|         return convertor.apply(getOne(queryWrapper, true)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 查询列表 | ||||
|      * | ||||
|      * @param kClass       vo类型 | ||||
|      * @param queryWrapper 实体对象封装操作类 {@link com.baomidou.mybatisplus.core.conditions.query.QueryWrapper} | ||||
|      */ | ||||
|     default <K> List<K> listVo(Wrapper<T> queryWrapper, Class<K> kClass) { | ||||
|         List<T> list = getBaseMapper().selectList(queryWrapper); | ||||
|         if (list == null) { | ||||
|             return null; | ||||
|         } | ||||
|         return list.stream() | ||||
|                 .map(any -> BeanUtil.toBean(any, kClass)) | ||||
|                 .collect(Collectors.toList()); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 查询列表 | ||||
|      * | ||||
|      * @param convertor    转换函数 | ||||
|      * @param queryWrapper 实体对象封装操作类 {@link com.baomidou.mybatisplus.core.conditions.query.QueryWrapper} | ||||
|      */ | ||||
|     default <K> List<K> listVo(Wrapper<T> queryWrapper, Function<Collection<T>, List<K>> convertor) { | ||||
|         List<T> list = getBaseMapper().selectList(queryWrapper); | ||||
|         if (list == null) { | ||||
|             return null; | ||||
|         } | ||||
|         return convertor.apply(list); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 查询所有 | ||||
|      * | ||||
|      * @param kClass vo类型 | ||||
|      * @see Wrappers#emptyWrapper() | ||||
|      */ | ||||
|     default <K> List<K> listVo(Class<K> kClass) { | ||||
|         return listVo(Wrappers.emptyWrapper(), kClass); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 查询所有 | ||||
|      * | ||||
|      * @param convertor 转换函数 | ||||
|      * @see Wrappers#emptyWrapper() | ||||
|      */ | ||||
|     default <K> List<K> listVo(Function<Collection<T>, List<K>> convertor) { | ||||
|         return listVo(Wrappers.emptyWrapper(), convertor); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 翻页查询 | ||||
|      * | ||||
|      * @param page         翻页对象 | ||||
|      * @param queryWrapper 实体对象封装操作类 | ||||
|      */ | ||||
|     default <K> PagePlus<T, K> pageVo(PagePlus<T, K> page, Wrapper<T> queryWrapper, Class<K> kClass) { | ||||
|         PagePlus<T, K> result = getBaseMapper().selectPage(page, queryWrapper); | ||||
|         List<K> volist = result.getRecords().stream() | ||||
|                 .map(any -> BeanUtil.toBean(any, kClass)) | ||||
|                 .collect(Collectors.toList()); | ||||
|         result.setRecordsVo(volist); | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 翻页查询 | ||||
|      * | ||||
|      * @param page         翻页对象 | ||||
|      * @param queryWrapper 实体对象封装操作类 | ||||
|      * @param convertor    转换函数 | ||||
|      */ | ||||
|     default <K> PagePlus<T, K> pageVo(PagePlus<T, K> page, Wrapper<T> queryWrapper, | ||||
|                                       Function<Collection<T>, List<K>> convertor) { | ||||
|         PagePlus<T, K> result = getBaseMapper().selectPage(page, queryWrapper); | ||||
|         return result.setRecordsVo(convertor.apply(result.getRecords())); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 无条件翻页查询 | ||||
|      * | ||||
|      * @param page   翻页对象 | ||||
|      */ | ||||
|     default <K> PagePlus<T, K> pageVo(PagePlus<T, K> page, Class<K> kClass) { | ||||
|         return pageVo(page, Wrappers.emptyWrapper(), kClass); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 无条件翻页查询 | ||||
|      * | ||||
|      * @param page      翻页对象 | ||||
|      * @param convertor 转换函数 | ||||
|      */ | ||||
|     default <K> PagePlus<T, K> pageVo(PagePlus<T, K> page, Function<Collection<T>, List<K>> convertor) { | ||||
|         return pageVo(page, Wrappers.emptyWrapper(), convertor); | ||||
|     } | ||||
|  | ||||
| 	@Override | ||||
| 	default boolean saveBatch(Collection<T> entityList) { | ||||
| 		return saveBatch(entityList, DEFAULT_BATCH_SIZE); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	default boolean saveOrUpdateBatch(Collection<T> entityList) { | ||||
| 		return saveOrUpdateBatch(entityList, DEFAULT_BATCH_SIZE); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	default boolean updateBatchById(Collection<T> entityList) { | ||||
| 		return updateBatchById(entityList, DEFAULT_BATCH_SIZE); | ||||
| 	} | ||||
|  | ||||
| 	boolean saveAll(Collection<T> entityList); | ||||
| } | ||||
|  | ||||
| @ -0,0 +1,90 @@ | ||||
| package com.ruoyi.common.core.mybatisplus.core; | ||||
|  | ||||
| import com.baomidou.mybatisplus.core.toolkit.ClassUtils; | ||||
| import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; | ||||
| import lombok.extern.slf4j.Slf4j; | ||||
| import org.springframework.beans.factory.annotation.Autowired; | ||||
| import org.springframework.core.ResolvableType; | ||||
|  | ||||
| import java.util.Collection; | ||||
|  | ||||
| /** | ||||
|  * IServicePlus 实现类 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Slf4j | ||||
| @SuppressWarnings("unchecked") | ||||
| public class ServicePlusImpl<M extends BaseMapperPlus<T>, T> extends ServiceImpl<M, T> implements IServicePlus<T> { | ||||
|  | ||||
| 	@Autowired | ||||
| 	protected M baseMapper; | ||||
|  | ||||
| 	@Override | ||||
| 	public M getBaseMapper() { | ||||
| 		return baseMapper; | ||||
| 	} | ||||
|  | ||||
|  | ||||
| 	protected Class<T> entityClass = currentModelClass(); | ||||
|  | ||||
| 	@Override | ||||
| 	public Class<T> getEntityClass() { | ||||
| 		return entityClass; | ||||
| 	} | ||||
|  | ||||
| 	protected Class<T> mapperClass = currentMapperClass(); | ||||
|  | ||||
| 	@Override | ||||
| 	protected Class<T> currentMapperClass() { | ||||
| 		return (Class<T>) this.getResolvableType().as(ServicePlusImpl.class).getGeneric(0).getType(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	protected Class<T> currentModelClass() { | ||||
| 		return (Class<T>) this.getResolvableType().as(ServicePlusImpl.class).getGeneric(1).getType(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	protected ResolvableType getResolvableType() { | ||||
| 		return ResolvableType.forClass(ClassUtils.getUserClass(getClass())); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 单条执行性能差 | ||||
| 	 * | ||||
| 	 * {@link #saveAll(Collection)} | ||||
| 	 */ | ||||
| 	@Deprecated | ||||
| 	@Override | ||||
| 	public boolean saveBatch(Collection<T> entityList, int batchSize) { | ||||
| 		return super.saveBatch(entityList, batchSize); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean saveOrUpdate(T entity) { | ||||
| 		return super.saveOrUpdate(entity); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 单条执行性能差 | ||||
| 	 * | ||||
| 	 * {@link #saveAll(Collection)} | ||||
| 	 */ | ||||
| 	@Deprecated | ||||
| 	@Override | ||||
| 	public boolean saveOrUpdateBatch(Collection<T> entityList, int batchSize) { | ||||
| 		return super.saveOrUpdateBatch(entityList, batchSize); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean updateBatchById(Collection<T> entityList, int batchSize) { | ||||
| 		return super.updateBatchById(entityList, batchSize); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| 	public boolean saveAll(Collection<T> entityList) { | ||||
| 		return baseMapper.insertAll(entityList) == entityList.size(); | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,52 @@ | ||||
| package com.ruoyi.common.core.mybatisplus.methods; | ||||
|  | ||||
| import cn.hutool.core.util.StrUtil; | ||||
| import org.apache.ibatis.executor.keygen.NoKeyGenerator; | ||||
| import org.apache.ibatis.mapping.MappedStatement; | ||||
| import org.apache.ibatis.mapping.SqlSource; | ||||
|  | ||||
| import com.baomidou.mybatisplus.core.injector.AbstractMethod; | ||||
| import com.baomidou.mybatisplus.core.metadata.TableInfo; | ||||
|  | ||||
| /** | ||||
|  * 单sql批量插入 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| public class InsertAll extends AbstractMethod { | ||||
|  | ||||
| 	@Override | ||||
| 	public MappedStatement injectMappedStatement(Class<?> mapperClass, Class<?> modelClass, TableInfo tableInfo) { | ||||
| 		final String sql = "<script>insert into %s %s values %s</script>"; | ||||
| 		final String fieldSql = prepareFieldSql(tableInfo); | ||||
| 		final String valueSql = prepareValuesSqlForMysqlBatch(tableInfo); | ||||
| 		final String sqlResult = String.format(sql, tableInfo.getTableName(), fieldSql, valueSql); | ||||
| 		SqlSource sqlSource = languageDriver.createSqlSource(configuration, sqlResult, modelClass); | ||||
| 		return this.addInsertMappedStatement(mapperClass, modelClass, "insertAll", sqlSource, new NoKeyGenerator(), null, null); | ||||
| 	} | ||||
|  | ||||
| 	private String prepareFieldSql(TableInfo tableInfo) { | ||||
| 		StringBuilder fieldSql = new StringBuilder(); | ||||
| 		if (StrUtil.isNotBlank(tableInfo.getKeyColumn())) { | ||||
| 			fieldSql.append(tableInfo.getKeyColumn()).append(","); | ||||
| 		} | ||||
| 		tableInfo.getFieldList().forEach(x -> fieldSql.append(x.getColumn()).append(",")); | ||||
| 		fieldSql.delete(fieldSql.length() - 1, fieldSql.length()); | ||||
| 		fieldSql.insert(0, "("); | ||||
| 		fieldSql.append(")"); | ||||
| 		return fieldSql.toString(); | ||||
| 	} | ||||
|  | ||||
| 	private String prepareValuesSqlForMysqlBatch(TableInfo tableInfo) { | ||||
| 		final StringBuilder valueSql = new StringBuilder(); | ||||
| 		valueSql.append("<foreach collection=\"list\" item=\"item\" index=\"index\" open=\"(\" separator=\"),(\" close=\")\">"); | ||||
| 		if (StrUtil.isNotBlank(tableInfo.getKeyColumn())) { | ||||
| 			valueSql.append("#{item.").append(tableInfo.getKeyProperty()).append("},"); | ||||
| 		} | ||||
| 		tableInfo.getFieldList().forEach(x -> valueSql.append("#{item.").append(x.getProperty()).append("},")); | ||||
| 		valueSql.delete(valueSql.length() - 1, valueSql.length()); | ||||
| 		valueSql.append("</foreach>"); | ||||
| 		return valueSql.toString(); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @ -0,0 +1,156 @@ | ||||
| package com.ruoyi.common.core.page; | ||||
|  | ||||
| import com.baomidou.mybatisplus.core.metadata.IPage; | ||||
| import com.baomidou.mybatisplus.core.metadata.OrderItem; | ||||
| import lombok.Data; | ||||
| import lombok.experimental.Accessors; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.Arrays; | ||||
| import java.util.Collections; | ||||
| import java.util.List; | ||||
|  | ||||
| /** | ||||
|  * 分页 Page 增强对象 | ||||
|  * | ||||
|  * @param <T> 数据库实体 | ||||
|  * @param <K> vo实体 | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Data | ||||
| @Accessors(chain = true) | ||||
| public class PagePlus<T,K> implements IPage<T> { | ||||
|  | ||||
|     /** | ||||
|      * domain实体列表 | ||||
|      */ | ||||
|     private List<T> records = Collections.emptyList(); | ||||
|  | ||||
|     /** | ||||
|      * vo实体列表 | ||||
|      */ | ||||
|     private List<K> recordsVo = Collections.emptyList(); | ||||
|  | ||||
|     /** | ||||
|      * 总数 | ||||
|      */ | ||||
|     private long total = 0L; | ||||
|  | ||||
|     /** | ||||
|      * 页长度 | ||||
|      */ | ||||
|     private long size = 10L; | ||||
|  | ||||
|     /** | ||||
|      * 当前页 | ||||
|      */ | ||||
|     private long current = 1L; | ||||
|  | ||||
|     /** | ||||
|      * 排序字段信息 | ||||
|      */ | ||||
|     private List<OrderItem> orders = new ArrayList<>(); | ||||
|  | ||||
|     /** | ||||
|      * 自动优化 COUNT SQL | ||||
|      */ | ||||
|     private boolean optimizeCountSql = true; | ||||
|  | ||||
|     /** | ||||
|      * 是否进行 count 查询 | ||||
|      */ | ||||
|     private boolean isSearchCount = true; | ||||
|  | ||||
|     /** | ||||
|      * 是否命中count缓存 | ||||
|      */ | ||||
|     private boolean hitCount = false; | ||||
|  | ||||
|     /** | ||||
|      * countId | ||||
|      */ | ||||
|     private String countId; | ||||
|  | ||||
|     /** | ||||
|      * 最大limit | ||||
|      */ | ||||
|     private Long maxLimit; | ||||
|  | ||||
|     public PagePlus() { | ||||
|     } | ||||
|  | ||||
|     public PagePlus(long current, long size) { | ||||
|         this(current, size, 0L); | ||||
|     } | ||||
|  | ||||
|     public PagePlus(long current, long size, long total) { | ||||
|         this(current, size, total, true); | ||||
|     } | ||||
|  | ||||
|     public PagePlus(long current, long size, boolean isSearchCount) { | ||||
|         this(current, size, 0L, isSearchCount); | ||||
|     } | ||||
|  | ||||
|     public PagePlus(long current, long size, long total, boolean isSearchCount) { | ||||
|         if (current > 1L) { | ||||
|             this.current = current; | ||||
|         } | ||||
|         this.size = size; | ||||
|         this.total = total; | ||||
|         this.isSearchCount = isSearchCount; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public String countId() { | ||||
|         return this.getCountId(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public Long maxLimit() { | ||||
|         return this.getMaxLimit(); | ||||
|     } | ||||
|  | ||||
|     public PagePlus<T, K> addOrder(OrderItem... items) { | ||||
|         this.orders.addAll(Arrays.asList(items)); | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     public PagePlus<T, K> addOrder(List<OrderItem> items) { | ||||
|         this.orders.addAll(items); | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public List<OrderItem> orders() { | ||||
|         return this.getOrders(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean optimizeCountSql() { | ||||
|         return this.optimizeCountSql; | ||||
|     } | ||||
|  | ||||
| 	@Override | ||||
| 	public long getPages() { | ||||
| 		// 解决 github issues/3208 | ||||
| 		return IPage.super.getPages(); | ||||
| 	} | ||||
|  | ||||
| 	public static <T,K> PagePlus<T,K> of(long current, long size) { | ||||
| 		return of(current, size, 0); | ||||
| 	} | ||||
|  | ||||
| 	public static <T,K> PagePlus<T,K> of(long current, long size, long total) { | ||||
| 		return of(current, size, total, true); | ||||
| 	} | ||||
|  | ||||
| 	public static <T,K> PagePlus<T,K> of(long current, long size, boolean searchCount) { | ||||
| 		return of(current, size, 0, searchCount); | ||||
| 	} | ||||
|  | ||||
| 	public static <T,K> PagePlus<T,K> of(long current, long size, long total, boolean searchCount) { | ||||
| 		return new PagePlus<>(current, size, total, searchCount); | ||||
| 	} | ||||
|  | ||||
| } | ||||
|  | ||||
| @ -0,0 +1,60 @@ | ||||
| package com.ruoyi.common.core.page; | ||||
|  | ||||
| import io.swagger.annotations.ApiModel; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.Data; | ||||
| import lombok.NoArgsConstructor; | ||||
| import lombok.experimental.Accessors; | ||||
|  | ||||
| import java.io.Serializable; | ||||
| import java.util.List; | ||||
|  | ||||
| /** | ||||
|  * 表格分页数据对象 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
|  | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| @Accessors(chain = true) | ||||
| @ApiModel("分页响应对象") | ||||
| public class TableDataInfo<T> implements Serializable { | ||||
| 	private static final long serialVersionUID = 1L; | ||||
|  | ||||
| 	/** | ||||
| 	 * 总记录数 | ||||
| 	 */ | ||||
| 	@ApiModelProperty("总记录数") | ||||
| 	private long total; | ||||
|  | ||||
| 	/** | ||||
| 	 * 列表数据 | ||||
| 	 */ | ||||
| 	@ApiModelProperty("列表数据") | ||||
| 	private List<T> rows; | ||||
|  | ||||
| 	/** | ||||
| 	 * 消息状态码 | ||||
| 	 */ | ||||
| 	@ApiModelProperty("消息状态码") | ||||
| 	private int code; | ||||
|  | ||||
| 	/** | ||||
| 	 * 消息内容 | ||||
| 	 */ | ||||
| 	@ApiModelProperty("消息内容") | ||||
| 	private String msg; | ||||
|  | ||||
| 	/** | ||||
| 	 * 分页 | ||||
| 	 * | ||||
| 	 * @param list  列表数据 | ||||
| 	 * @param total 总记录数 | ||||
| 	 */ | ||||
| 	public TableDataInfo(List<T> list, long total) { | ||||
| 		this.rows = list; | ||||
| 		this.total = total; | ||||
| 	} | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,223 @@ | ||||
| package com.ruoyi.common.core.redis; | ||||
|  | ||||
| import com.google.common.collect.Lists; | ||||
| import org.redisson.api.*; | ||||
| import org.springframework.beans.factory.annotation.Autowired; | ||||
| import org.springframework.stereotype.Component; | ||||
|  | ||||
| import java.util.*; | ||||
| import java.util.concurrent.TimeUnit; | ||||
|  | ||||
| /** | ||||
|  * spring redis 工具类 | ||||
|  * | ||||
|  * @author shenxinquan | ||||
|  **/ | ||||
| @SuppressWarnings(value = {"unchecked", "rawtypes"}) | ||||
| @Component | ||||
| public class RedisCache { | ||||
|  | ||||
| 	@Autowired | ||||
| 	private RedissonClient redissonClient; | ||||
|  | ||||
| 	/** | ||||
| 	 * 缓存基本的对象,Integer、String、实体类等 | ||||
| 	 * | ||||
| 	 * @param key   缓存的键值 | ||||
| 	 * @param value 缓存的值 | ||||
| 	 */ | ||||
| 	public <T> void setCacheObject(final String key, final T value) { | ||||
| 		redissonClient.getBucket(key).set(value); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 缓存基本的对象,Integer、String、实体类等 | ||||
| 	 * | ||||
| 	 * @param key      缓存的键值 | ||||
| 	 * @param value    缓存的值 | ||||
| 	 * @param timeout  时间 | ||||
| 	 * @param timeUnit 时间颗粒度 | ||||
| 	 */ | ||||
| 	public <T> void setCacheObject(final String key, final T value, final Integer timeout, final TimeUnit timeUnit) { | ||||
| 		RBucket<T> result = redissonClient.getBucket(key); | ||||
| 		result.set(value); | ||||
| 		result.expire(timeout, timeUnit); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 设置有效时间 | ||||
| 	 * | ||||
| 	 * @param key     Redis键 | ||||
| 	 * @param timeout 超时时间 | ||||
| 	 * @return true=设置成功;false=设置失败 | ||||
| 	 */ | ||||
| 	public boolean expire(final String key, final long timeout) { | ||||
| 		return expire(key, timeout, TimeUnit.SECONDS); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 设置有效时间 | ||||
| 	 * | ||||
| 	 * @param key     Redis键 | ||||
| 	 * @param timeout 超时时间 | ||||
| 	 * @param unit    时间单位 | ||||
| 	 * @return true=设置成功;false=设置失败 | ||||
| 	 */ | ||||
| 	public boolean expire(final String key, final long timeout, final TimeUnit unit) { | ||||
| 		RBucket rBucket = redissonClient.getBucket(key); | ||||
| 		return rBucket.expire(timeout, unit); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 获得缓存的基本对象。 | ||||
| 	 * | ||||
| 	 * @param key 缓存键值 | ||||
| 	 * @return 缓存键值对应的数据 | ||||
| 	 */ | ||||
| 	public <T> T getCacheObject(final String key) { | ||||
| 		RBucket<T> rBucket = redissonClient.getBucket(key); | ||||
| 		return rBucket.get(); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 删除单个对象 | ||||
| 	 * | ||||
| 	 * @param key | ||||
| 	 */ | ||||
| 	public boolean deleteObject(final String key) { | ||||
| 		return redissonClient.getBucket(key).delete(); | ||||
| 	} | ||||
|  | ||||
| 	/* */ | ||||
|  | ||||
| 	/** | ||||
| 	 * 删除集合对象 | ||||
| 	 * | ||||
| 	 * @param collection 多个对象 | ||||
| 	 * @return | ||||
| 	 */ | ||||
| 	public void deleteObject(final Collection collection) { | ||||
| 		RBatch batch = redissonClient.createBatch(); | ||||
| 		collection.forEach(t->{ | ||||
| 			batch.getBucket(t.toString()).deleteAsync(); | ||||
| 		}); | ||||
| 		batch.execute(); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 缓存List数据 | ||||
| 	 * | ||||
| 	 * @param key      缓存的键值 | ||||
| 	 * @param dataList 待缓存的List数据 | ||||
| 	 * @return 缓存的对象 | ||||
| 	 */ | ||||
| 	public <T> boolean setCacheList(final String key, final List<T> dataList) { | ||||
| 		RList<T> rList = redissonClient.getList(key); | ||||
| 		return rList.addAll(dataList); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 获得缓存的list对象 | ||||
| 	 * | ||||
| 	 * @param key 缓存的键值 | ||||
| 	 * @return 缓存键值对应的数据 | ||||
| 	 */ | ||||
| 	public <T> List<T> getCacheList(final String key) { | ||||
| 		RList<T> rList = redissonClient.getList(key); | ||||
| 		return rList.readAll(); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 缓存Set | ||||
| 	 * | ||||
| 	 * @param key     缓存键值 | ||||
| 	 * @param dataSet 缓存的数据 | ||||
| 	 * @return 缓存数据的对象 | ||||
| 	 */ | ||||
| 	public <T> boolean setCacheSet(final String key, final Set<T> dataSet) { | ||||
| 		RSet<T> rSet = redissonClient.getSet(key); | ||||
| 		return rSet.addAll(dataSet); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 获得缓存的set | ||||
| 	 * | ||||
| 	 * @param key | ||||
| 	 * @return | ||||
| 	 */ | ||||
| 	public <T> Set<T> getCacheSet(final String key) { | ||||
| 		RSet<T> rSet = redissonClient.getSet(key); | ||||
| 		return rSet.readAll(); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 缓存Map | ||||
| 	 * | ||||
| 	 * @param key | ||||
| 	 * @param dataMap | ||||
| 	 */ | ||||
| 	public <T> void setCacheMap(final String key, final Map<String, T> dataMap) { | ||||
| 		if (dataMap != null) { | ||||
| 			RMap<String, T> rMap = redissonClient.getMap(key); | ||||
| 			rMap.putAll(dataMap); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 获得缓存的Map | ||||
| 	 * | ||||
| 	 * @param key | ||||
| 	 * @return | ||||
| 	 */ | ||||
| 	public <T> Map<String, T> getCacheMap(final String key) { | ||||
| 		RMap<String, T> rMap = redissonClient.getMap(key); | ||||
| 		return rMap.getAll(rMap.keySet()); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 往Hash中存入数据 | ||||
| 	 * | ||||
| 	 * @param key   Redis键 | ||||
| 	 * @param hKey  Hash键 | ||||
| 	 * @param value 值 | ||||
| 	 */ | ||||
| 	public <T> void setCacheMapValue(final String key, final String hKey, final T value) { | ||||
| 		RMap<String, T> rMap = redissonClient.getMap(key); | ||||
| 		rMap.put(hKey, value); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 获取Hash中的数据 | ||||
| 	 * | ||||
| 	 * @param key  Redis键 | ||||
| 	 * @param hKey Hash键 | ||||
| 	 * @return Hash中的对象 | ||||
| 	 */ | ||||
| 	public <T> T getCacheMapValue(final String key, final String hKey) { | ||||
| 		RMap<String, T> rMap = redissonClient.getMap(key); | ||||
| 		return rMap.get(hKey); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 获取多个Hash中的数据 | ||||
| 	 * | ||||
| 	 * @param key   Redis键 | ||||
| 	 * @param hKeys Hash键集合 | ||||
| 	 * @return Hash对象集合 | ||||
| 	 */ | ||||
| 	public <T> List<T> getMultiCacheMapValue(final String key, final Collection<Object> hKeys) { | ||||
| 		RListMultimap rListMultimap = redissonClient.getListMultimap(key); | ||||
| 		return rListMultimap.getAll(hKeys); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 获得缓存的基本对象列表 | ||||
| 	 * | ||||
| 	 * @param pattern 字符串前缀 | ||||
| 	 * @return 对象列表 | ||||
| 	 */ | ||||
| 	public Collection<String> keys(final String pattern) { | ||||
| 		Iterable<String> iterable = redissonClient.getKeys().getKeysByPattern(pattern); | ||||
| 		return Lists.newArrayList(iterable); | ||||
| 	} | ||||
| } | ||||
| @ -0,0 +1,20 @@ | ||||
| package com.ruoyi.common.enums; | ||||
|  | ||||
| /** | ||||
|  * 操作状态 | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  * | ||||
|  */ | ||||
| public enum BusinessStatus | ||||
| { | ||||
|     /** | ||||
|      * 成功 | ||||
|      */ | ||||
|     SUCCESS, | ||||
|  | ||||
|     /** | ||||
|      * 失败 | ||||
|      */ | ||||
|     FAIL, | ||||
| } | ||||
| @ -0,0 +1,59 @@ | ||||
| package com.ruoyi.common.enums; | ||||
|  | ||||
| /** | ||||
|  * 业务操作类型 | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| public enum BusinessType | ||||
| { | ||||
|     /** | ||||
|      * 其它 | ||||
|      */ | ||||
|     OTHER, | ||||
|  | ||||
|     /** | ||||
|      * 新增 | ||||
|      */ | ||||
|     INSERT, | ||||
|  | ||||
|     /** | ||||
|      * 修改 | ||||
|      */ | ||||
|     UPDATE, | ||||
|  | ||||
|     /** | ||||
|      * 删除 | ||||
|      */ | ||||
|     DELETE, | ||||
|  | ||||
|     /** | ||||
|      * 授权 | ||||
|      */ | ||||
|     GRANT, | ||||
|  | ||||
|     /** | ||||
|      * 导出 | ||||
|      */ | ||||
|     EXPORT, | ||||
|  | ||||
|     /** | ||||
|      * 导入 | ||||
|      */ | ||||
|     IMPORT, | ||||
|  | ||||
|     /** | ||||
|      * 强退 | ||||
|      */ | ||||
|     FORCE, | ||||
|  | ||||
|     /** | ||||
|      * 生成代码 | ||||
|      */ | ||||
|     GENCODE, | ||||
|      | ||||
|     /** | ||||
|      * 清空数据 | ||||
|      */ | ||||
|     CLEAN, | ||||
| } | ||||
| @ -0,0 +1,25 @@ | ||||
| package com.ruoyi.common.enums; | ||||
|  | ||||
| import lombok.AllArgsConstructor; | ||||
| import lombok.Getter; | ||||
|  | ||||
| /** | ||||
|  * 数据源 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @AllArgsConstructor | ||||
| public enum DataSourceType { | ||||
| 	/** | ||||
| 	 * 主库 | ||||
| 	 */ | ||||
| 	MASTER("master"), | ||||
|  | ||||
| 	/** | ||||
| 	 * 从库 | ||||
| 	 */ | ||||
| 	SLAVE("slave"); | ||||
|  | ||||
| 	@Getter | ||||
| 	private final String source; | ||||
| } | ||||
| @ -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<String, HttpMethod> 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)); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,24 @@ | ||||
| package com.ruoyi.common.enums; | ||||
|  | ||||
| /** | ||||
|  * 操作人类别 | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| public enum OperatorType | ||||
| { | ||||
|     /** | ||||
|      * 其它 | ||||
|      */ | ||||
|     OTHER, | ||||
|  | ||||
|     /** | ||||
|      * 后台用户 | ||||
|      */ | ||||
|     MANAGE, | ||||
|  | ||||
|     /** | ||||
|      * 手机端用户 | ||||
|      */ | ||||
|     MOBILE | ||||
| } | ||||
| @ -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; | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,97 @@ | ||||
| package com.ruoyi.common.exception; | ||||
|  | ||||
| import cn.hutool.core.lang.Validator; | ||||
| import com.ruoyi.common.utils.MessageUtils; | ||||
|  | ||||
| /** | ||||
|  * 基础异常 | ||||
|  *  | ||||
|  * @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 (!Validator.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; | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,43 @@ | ||||
| package com.ruoyi.common.exception; | ||||
|  | ||||
| /** | ||||
|  * 自定义异常 | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| public class CustomException extends RuntimeException | ||||
| { | ||||
|     private static final long serialVersionUID = 1L; | ||||
|  | ||||
|     private Integer code; | ||||
|  | ||||
|     private String message; | ||||
|  | ||||
|     public CustomException(String message) | ||||
|     { | ||||
|         this.message = message; | ||||
|     } | ||||
|  | ||||
|     public CustomException(String message, Integer code) | ||||
|     { | ||||
|         this.message = message; | ||||
|         this.code = code; | ||||
|     } | ||||
|  | ||||
|     public CustomException(String message, Throwable e) | ||||
|     { | ||||
|         super(message, e); | ||||
|         this.message = message; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public String getMessage() | ||||
|     { | ||||
|         return message; | ||||
|     } | ||||
|  | ||||
|     public Integer getCode() | ||||
|     { | ||||
|         return code; | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,15 @@ | ||||
| package com.ruoyi.common.exception; | ||||
|  | ||||
| /** | ||||
|  * 演示模式异常 | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| public class DemoModeException extends RuntimeException | ||||
| { | ||||
|     private static final long serialVersionUID = 1L; | ||||
|  | ||||
|     public DemoModeException() | ||||
|     { | ||||
|     } | ||||
| } | ||||
| @ -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); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,19 @@ | ||||
| package com.ruoyi.common.exception.file; | ||||
|  | ||||
| import com.ruoyi.common.exception.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); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @ -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 }); | ||||
|     } | ||||
| } | ||||
| @ -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 }); | ||||
|     } | ||||
| } | ||||
| @ -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 : [" + filename + "], extension : [" + extension + "], allowed 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); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @ -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 | ||||
|     } | ||||
| } | ||||
| @ -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); | ||||
|     } | ||||
| } | ||||
| @ -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); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,18 @@ | ||||
| package com.ruoyi.common.exception.user; | ||||
|  | ||||
| import com.ruoyi.common.exception.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); | ||||
|     } | ||||
| } | ||||
| @ -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); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,48 @@ | ||||
| package com.ruoyi.common.filter; | ||||
|  | ||||
| import cn.hutool.core.util.StrUtil; | ||||
| import org.springframework.http.MediaType; | ||||
|  | ||||
| import javax.servlet.*; | ||||
| import javax.servlet.http.HttpServletRequest; | ||||
| import java.io.IOException; | ||||
|  | ||||
| /** | ||||
|  * 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 | ||||
|                 && StrUtil.startWithIgnoreCase(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() | ||||
|     { | ||||
|  | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,77 @@ | ||||
| package com.ruoyi.common.filter; | ||||
|  | ||||
| import cn.hutool.core.io.IoUtil; | ||||
|  | ||||
| import javax.servlet.ReadListener; | ||||
| import javax.servlet.ServletInputStream; | ||||
| import javax.servlet.ServletResponse; | ||||
| import javax.servlet.http.HttpServletRequest; | ||||
| import javax.servlet.http.HttpServletRequestWrapper; | ||||
| import java.io.BufferedReader; | ||||
| import java.io.ByteArrayInputStream; | ||||
| import java.io.IOException; | ||||
| import java.io.InputStreamReader; | ||||
| import java.nio.charset.StandardCharsets; | ||||
|  | ||||
| /** | ||||
|  * 构建可重复读取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("UTF-8"); | ||||
|         response.setCharacterEncoding("UTF-8"); | ||||
|  | ||||
|         body = IoUtil.readUtf8(request.getInputStream()).getBytes(StandardCharsets.UTF_8); | ||||
|     } | ||||
|  | ||||
|     @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) | ||||
|             { | ||||
|  | ||||
|             } | ||||
|         }; | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,93 @@ | ||||
| package com.ruoyi.common.filter; | ||||
|  | ||||
| import cn.hutool.core.util.StrUtil; | ||||
|  | ||||
| import javax.servlet.*; | ||||
| import javax.servlet.http.HttpServletRequest; | ||||
| import javax.servlet.http.HttpServletResponse; | ||||
| import java.io.IOException; | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| import java.util.regex.Matcher; | ||||
| import java.util.regex.Pattern; | ||||
|  | ||||
| /** | ||||
|  * 防止XSS攻击的过滤器 | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| public class XssFilter implements Filter | ||||
| { | ||||
|     /** | ||||
|      * 排除链接 | ||||
|      */ | ||||
|     public List<String> excludes = new ArrayList<>(); | ||||
|  | ||||
|     /** | ||||
|      * xss过滤开关 | ||||
|      */ | ||||
|     public boolean enabled = false; | ||||
|  | ||||
|     @Override | ||||
|     public void init(FilterConfig filterConfig) throws ServletException | ||||
|     { | ||||
|         String tempExcludes = filterConfig.getInitParameter("excludes"); | ||||
|         String tempEnabled = filterConfig.getInitParameter("enabled"); | ||||
|         if (StrUtil.isNotEmpty(tempExcludes)) | ||||
|         { | ||||
|             String[] url = tempExcludes.split(","); | ||||
|             for (int i = 0; url != null && i < url.length; i++) | ||||
|             { | ||||
|                 excludes.add(url[i]); | ||||
|             } | ||||
|         } | ||||
|         if (StrUtil.isNotEmpty(tempEnabled)) | ||||
|         { | ||||
|             enabled = Boolean.valueOf(tempEnabled); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @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) | ||||
|     { | ||||
|         if (!enabled) | ||||
|         { | ||||
|             return true; | ||||
|         } | ||||
|         if (excludes == null || excludes.isEmpty()) | ||||
|         { | ||||
|             return false; | ||||
|         } | ||||
|         String url = request.getServletPath(); | ||||
|         for (String pattern : excludes) | ||||
|         { | ||||
|             Pattern p = Pattern.compile("^" + pattern); | ||||
|             Matcher m = p.matcher(url); | ||||
|             if (m.find()) | ||||
|             { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void destroy() | ||||
|     { | ||||
|  | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,108 @@ | ||||
| package com.ruoyi.common.filter; | ||||
|  | ||||
| import cn.hutool.core.io.IoUtil; | ||||
| import cn.hutool.core.lang.Validator; | ||||
| import cn.hutool.core.util.StrUtil; | ||||
| import cn.hutool.http.HtmlUtil; | ||||
| import org.springframework.http.HttpHeaders; | ||||
| import org.springframework.http.MediaType; | ||||
|  | ||||
| import javax.servlet.ReadListener; | ||||
| import javax.servlet.ServletInputStream; | ||||
| import javax.servlet.http.HttpServletRequest; | ||||
| import javax.servlet.http.HttpServletRequestWrapper; | ||||
| import java.io.ByteArrayInputStream; | ||||
| import java.io.IOException; | ||||
| import java.nio.charset.StandardCharsets; | ||||
|  | ||||
| /** | ||||
|  * 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] = HtmlUtil.cleanHtmlTag(values[i]).trim(); | ||||
|             } | ||||
|             return escapseValues; | ||||
|         } | ||||
|         return super.getParameterValues(name); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public ServletInputStream getInputStream() throws IOException | ||||
|     { | ||||
|         // 非json类型,直接返回 | ||||
|         if (!isJsonRequest()) | ||||
|         { | ||||
|             return super.getInputStream(); | ||||
|         } | ||||
|  | ||||
|         // 为空,直接返回 | ||||
|         String json = IoUtil.read(super.getInputStream(), StandardCharsets.UTF_8); | ||||
|         if (Validator.isEmpty(json)) | ||||
|         { | ||||
|             return super.getInputStream(); | ||||
|         } | ||||
|  | ||||
|         // xss过滤 | ||||
|         json = HtmlUtil.cleanHtmlTag(json).trim(); | ||||
|  | ||||
|         final ByteArrayInputStream bis = IoUtil.toStream(json, StandardCharsets.UTF_8); | ||||
|         return new ServletInputStream() | ||||
|         { | ||||
|             @Override | ||||
|             public boolean isFinished() | ||||
|             { | ||||
|                 return true; | ||||
|             } | ||||
|  | ||||
|             @Override | ||||
|             public boolean isReady() | ||||
|             { | ||||
|                 return true; | ||||
|             } | ||||
|  | ||||
|             @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 StrUtil.startWithIgnoreCase(header, MediaType.APPLICATION_JSON_VALUE); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										155
									
								
								ruoyi-common/src/main/java/com/ruoyi/common/utils/DateUtils.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										155
									
								
								ruoyi-common/src/main/java/com/ruoyi/common/utils/DateUtils.java
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,155 @@ | ||||
| package com.ruoyi.common.utils; | ||||
|  | ||||
| import java.lang.management.ManagementFactory; | ||||
| import java.text.ParseException; | ||||
| import java.text.SimpleDateFormat; | ||||
| 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(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取当前日期, 默认格式为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 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 + "分钟"; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										187
									
								
								ruoyi-common/src/main/java/com/ruoyi/common/utils/DictUtils.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										187
									
								
								ruoyi-common/src/main/java/com/ruoyi/common/utils/DictUtils.java
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,187 @@ | ||||
| package com.ruoyi.common.utils; | ||||
|  | ||||
| import cn.hutool.core.collection.CollUtil; | ||||
| import cn.hutool.core.lang.Validator; | ||||
| import cn.hutool.core.util.StrUtil; | ||||
| import com.ruoyi.common.constant.Constants; | ||||
| import com.ruoyi.common.core.domain.entity.SysDictData; | ||||
| import com.ruoyi.common.core.redis.RedisCache; | ||||
| import com.ruoyi.common.utils.spring.SpringUtils; | ||||
|  | ||||
| import java.util.Collection; | ||||
| import java.util.List; | ||||
|  | ||||
| /** | ||||
|  * 字典工具类 | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| public class DictUtils | ||||
| { | ||||
|     /** | ||||
|      * 分隔符 | ||||
|      */ | ||||
|     public static final String SEPARATOR = ","; | ||||
|  | ||||
|     /** | ||||
|      * 设置字典缓存 | ||||
|      *  | ||||
|      * @param key 参数键 | ||||
|      * @param dictDatas 字典数据列表 | ||||
|      */ | ||||
|     public static void setDictCache(String key, List<SysDictData> dictDatas) | ||||
|     { | ||||
|         SpringUtils.getBean(RedisCache.class).setCacheObject(getCacheKey(key), dictDatas); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取字典缓存 | ||||
|      *  | ||||
|      * @param key 参数键 | ||||
|      * @return dictDatas 字典数据列表 | ||||
|      */ | ||||
|     public static List<SysDictData> getDictCache(String key) | ||||
|     { | ||||
|         Object cacheObj = SpringUtils.getBean(RedisCache.class).getCacheObject(getCacheKey(key)); | ||||
|         if (Validator.isNotNull(cacheObj)) | ||||
|         { | ||||
|             List<SysDictData> dictDatas = (List<SysDictData>)cacheObj; | ||||
|             return dictDatas; | ||||
|         } | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据字典类型和字典值获取字典标签 | ||||
|      *  | ||||
|      * @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<SysDictData> datas = getDictCache(dictType); | ||||
|  | ||||
|         if (StrUtil.containsAny(separator, dictValue) && CollUtil.isNotEmpty(datas)) | ||||
|         { | ||||
|             for (SysDictData dict : datas) | ||||
|             { | ||||
|                 for (String value : dictValue.split(separator)) | ||||
|                 { | ||||
|                     if (value.equals(dict.getDictValue())) | ||||
|                     { | ||||
|                         propertyString.append(dict.getDictLabel() + separator); | ||||
|                         break; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             for (SysDictData dict : datas) | ||||
|             { | ||||
|                 if (dictValue.equals(dict.getDictValue())) | ||||
|                 { | ||||
|                     return dict.getDictLabel(); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         return StrUtil.strip(propertyString.toString(), null, separator); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据字典类型和字典标签获取字典值 | ||||
|      *  | ||||
|      * @param dictType 字典类型 | ||||
|      * @param dictLabel 字典标签 | ||||
|      * @param separator 分隔符 | ||||
|      * @return 字典值 | ||||
|      */ | ||||
|     public static String getDictValue(String dictType, String dictLabel, String separator) | ||||
|     { | ||||
|         StringBuilder propertyString = new StringBuilder(); | ||||
|         List<SysDictData> datas = getDictCache(dictType); | ||||
|  | ||||
|         if (StrUtil.containsAny(separator, dictLabel) && CollUtil.isNotEmpty(datas)) | ||||
|         { | ||||
|             for (SysDictData dict : datas) | ||||
|             { | ||||
|                 for (String label : dictLabel.split(separator)) | ||||
|                 { | ||||
|                     if (label.equals(dict.getDictLabel())) | ||||
|                     { | ||||
|                         propertyString.append(dict.getDictValue() + separator); | ||||
|                         break; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             for (SysDictData dict : datas) | ||||
|             { | ||||
|                 if (dictLabel.equals(dict.getDictLabel())) | ||||
|                 { | ||||
|                     return dict.getDictValue(); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         return StrUtil.strip(propertyString.toString(), null, separator); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 删除指定字典缓存 | ||||
|      *  | ||||
|      * @param key 字典键 | ||||
|      */ | ||||
|     public static void removeDictCache(String key) | ||||
|     { | ||||
|         SpringUtils.getBean(RedisCache.class).deleteObject(getCacheKey(key)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 清空字典缓存 | ||||
|      */ | ||||
|     public static void clearDictCache() | ||||
|     { | ||||
|         Collection<String> keys = SpringUtils.getBean(RedisCache.class).keys(Constants.SYS_DICT_KEY + "*"); | ||||
|         SpringUtils.getBean(RedisCache.class).deleteObject(keys); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 设置cache key | ||||
|      *  | ||||
|      * @param configKey 参数键 | ||||
|      * @return 缓存键key | ||||
|      */ | ||||
|     public static String getCacheKey(String configKey) | ||||
|     { | ||||
|         return Constants.SYS_DICT_KEY + configKey; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										101
									
								
								ruoyi-common/src/main/java/com/ruoyi/common/utils/JsonUtils.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										101
									
								
								ruoyi-common/src/main/java/com/ruoyi/common/utils/JsonUtils.java
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,101 @@ | ||||
| package com.ruoyi.common.utils; | ||||
|  | ||||
| import cn.hutool.core.lang.Validator; | ||||
| import cn.hutool.core.util.ArrayUtil; | ||||
| import cn.hutool.core.util.StrUtil; | ||||
| import com.fasterxml.jackson.core.JsonProcessingException; | ||||
| import com.fasterxml.jackson.core.type.TypeReference; | ||||
| import com.fasterxml.jackson.databind.ObjectMapper; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * JSON 工具类 | ||||
|  * | ||||
|  * @author 芋道源码 | ||||
|  */ | ||||
| public class JsonUtils { | ||||
|  | ||||
|     private static ObjectMapper objectMapper = new ObjectMapper(); | ||||
|  | ||||
|     /** | ||||
|      * 初始化 objectMapper 属性 | ||||
|      * <p> | ||||
|      * 通过这样的方式,使用 Spring 创建的 ObjectMapper Bean | ||||
|      * | ||||
|      * @param objectMapper ObjectMapper 对象 | ||||
|      */ | ||||
|     public static void init(ObjectMapper objectMapper) { | ||||
|         JsonUtils.objectMapper = objectMapper; | ||||
|     } | ||||
|  | ||||
|     public static String toJsonString(Object object) { | ||||
| 		if (Validator.isEmpty(object)) { | ||||
| 			return null; | ||||
| 		} | ||||
|         try { | ||||
|             return objectMapper.writeValueAsString(object); | ||||
|         } catch (JsonProcessingException e) { | ||||
|             throw new RuntimeException(e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public static <T> T parseObject(String text, Class<T> clazz) { | ||||
|         if (StrUtil.isEmpty(text)) { | ||||
|             return null; | ||||
|         } | ||||
|         try { | ||||
|             return objectMapper.readValue(text, clazz); | ||||
|         } catch (IOException e) { | ||||
|             throw new RuntimeException(e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public static <T> T parseObject(byte[] bytes, Class<T> clazz) { | ||||
|         if (ArrayUtil.isEmpty(bytes)) { | ||||
|             return null; | ||||
|         } | ||||
|         try { | ||||
|             return objectMapper.readValue(bytes, clazz); | ||||
|         } catch (IOException e) { | ||||
|             throw new RuntimeException(e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public static <T> T parseObject(String text, TypeReference<T> typeReference) { | ||||
| 		if (StrUtil.isBlank(text)) { | ||||
| 			return null; | ||||
| 		} | ||||
|         try { | ||||
|             return objectMapper.readValue(text, typeReference); | ||||
|         } catch (IOException e) { | ||||
|             throw new RuntimeException(e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| 	public static <T> Map<String, T> parseMap(String text) { | ||||
| 		if (StrUtil.isBlank(text)) { | ||||
| 			return null; | ||||
| 		} | ||||
| 		try { | ||||
| 			return objectMapper.readValue(text, new TypeReference<Map<String, T>>() {}); | ||||
| 		} catch (IOException e) { | ||||
| 			throw new RuntimeException(e); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
|     public static <T> List<T> parseArray(String text, Class<T> clazz) { | ||||
|         if (StrUtil.isEmpty(text)) { | ||||
|             return new ArrayList<>(); | ||||
|         } | ||||
|         try { | ||||
|             return objectMapper.readValue(text, objectMapper.getTypeFactory().constructCollectionType(List.class, clazz)); | ||||
|         } catch (IOException e) { | ||||
|             throw new RuntimeException(e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
| @ -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()); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										133
									
								
								ruoyi-common/src/main/java/com/ruoyi/common/utils/PageUtils.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										133
									
								
								ruoyi-common/src/main/java/com/ruoyi/common/utils/PageUtils.java
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,133 @@ | ||||
| package com.ruoyi.common.utils; | ||||
|  | ||||
| import cn.hutool.core.util.StrUtil; | ||||
| import cn.hutool.http.HttpStatus; | ||||
| import com.baomidou.mybatisplus.core.metadata.OrderItem; | ||||
| import com.baomidou.mybatisplus.extension.plugins.pagination.Page; | ||||
| import com.ruoyi.common.core.page.PagePlus; | ||||
| import com.ruoyi.common.core.page.TableDataInfo; | ||||
| import com.ruoyi.common.utils.sql.SqlUtil; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| /** | ||||
|  * 分页工具 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| public class PageUtils { | ||||
|  | ||||
|     /** | ||||
|      * 当前记录起始索引 | ||||
|      */ | ||||
|     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 int DEFAULT_PAGE_NUM = 1; | ||||
|  | ||||
|     /** | ||||
|      * 每页显示记录数 默认值 默认查全部 | ||||
|      */ | ||||
|     public static final int DEFAULT_PAGE_SIZE = Integer.MAX_VALUE; | ||||
|  | ||||
|     /** | ||||
|      * 构建 plus 分页对象 | ||||
|      * @param <T> domain 实体 | ||||
|      * @param <K> vo 实体 | ||||
|      * @return 分页对象 | ||||
|      */ | ||||
|     public static <T, K> PagePlus<T, K> buildPagePlus() { | ||||
|         Integer pageNum = ServletUtils.getParameterToInt(PAGE_NUM, DEFAULT_PAGE_NUM); | ||||
|         Integer pageSize = ServletUtils.getParameterToInt(PAGE_SIZE, DEFAULT_PAGE_SIZE); | ||||
|         String orderByColumn = ServletUtils.getParameter(ORDER_BY_COLUMN); | ||||
|         String isAsc = ServletUtils.getParameter(IS_ASC); | ||||
|         PagePlus<T, K> page = new PagePlus<>(pageNum, pageSize); | ||||
|         if (StrUtil.isNotBlank(orderByColumn)) { | ||||
|             String orderBy = SqlUtil.escapeOrderBySql(orderByColumn); | ||||
|             if ("asc".equals(isAsc)) { | ||||
|                 page.addOrder(OrderItem.asc(orderBy)); | ||||
|             } else if ("desc".equals(isAsc)) { | ||||
|                 page.addOrder(OrderItem.desc(orderBy)); | ||||
|             } | ||||
|         } | ||||
|         return page; | ||||
|     } | ||||
|  | ||||
| 	public static <T> Page<T> buildPage() { | ||||
| 		return buildPage(null, null); | ||||
| 	} | ||||
|  | ||||
|     /** | ||||
|      * 构建 MP 普通分页对象 | ||||
|      * @param <T> domain 实体 | ||||
|      * @return 分页对象 | ||||
|      */ | ||||
|     public static <T> Page<T> buildPage(String defaultOrderByColumn, String defaultIsAsc) { | ||||
|         Integer pageNum = ServletUtils.getParameterToInt(PAGE_NUM, DEFAULT_PAGE_NUM); | ||||
|         Integer pageSize = ServletUtils.getParameterToInt(PAGE_SIZE, DEFAULT_PAGE_SIZE); | ||||
|         String orderByColumn = ServletUtils.getParameter(ORDER_BY_COLUMN, defaultOrderByColumn); | ||||
|         String isAsc = ServletUtils.getParameter(IS_ASC, defaultIsAsc); | ||||
| 		// 兼容前端排序类型 | ||||
| 		if ("ascending".equals(isAsc)) { | ||||
| 			isAsc = "asc"; | ||||
| 		} else if ("descending".equals(isAsc)) { | ||||
| 			isAsc = "desc"; | ||||
| 		} | ||||
|         Page<T> page = new Page<>(pageNum, pageSize); | ||||
|         if (StrUtil.isNotBlank(orderByColumn)) { | ||||
|             String orderBy = SqlUtil.escapeOrderBySql(orderByColumn); | ||||
| 			orderBy = StrUtil.toUnderlineCase(orderBy); | ||||
| 			if ("asc".equals(isAsc)) { | ||||
|                 page.addOrder(OrderItem.asc(orderBy)); | ||||
|             } else if ("desc".equals(isAsc)) { | ||||
|                 page.addOrder(OrderItem.desc(orderBy)); | ||||
|             } | ||||
|         } | ||||
|         return page; | ||||
|     } | ||||
|  | ||||
|     public static <T, K> TableDataInfo<K> buildDataInfo(PagePlus<T, K> page) { | ||||
|         TableDataInfo<K> rspData = new TableDataInfo<>(); | ||||
|         rspData.setCode(HttpStatus.HTTP_OK); | ||||
|         rspData.setMsg("查询成功"); | ||||
|         rspData.setRows(page.getRecordsVo()); | ||||
|         rspData.setTotal(page.getTotal()); | ||||
|         return rspData; | ||||
|     } | ||||
|  | ||||
|     public static <T> TableDataInfo<T> buildDataInfo(Page<T> page) { | ||||
|         TableDataInfo<T> rspData = new TableDataInfo<>(); | ||||
|         rspData.setCode(HttpStatus.HTTP_OK); | ||||
|         rspData.setMsg("查询成功"); | ||||
|         rspData.setRows(page.getRecords()); | ||||
|         rspData.setTotal(page.getTotal()); | ||||
|         return rspData; | ||||
|     } | ||||
|  | ||||
|     public static <T> TableDataInfo<T> buildDataInfo(List<T> list) { | ||||
|         TableDataInfo<T> rspData = new TableDataInfo<>(); | ||||
|         rspData.setCode(HttpStatus.HTTP_OK); | ||||
|         rspData.setMsg("查询成功"); | ||||
|         rspData.setRows(list); | ||||
|         rspData.setTotal(list.size()); | ||||
|         return rspData; | ||||
|     } | ||||
|  | ||||
| } | ||||
| @ -0,0 +1,90 @@ | ||||
| package com.ruoyi.common.utils; | ||||
|  | ||||
| import cn.hutool.http.HttpStatus; | ||||
| import org.springframework.security.core.Authentication; | ||||
| import org.springframework.security.core.context.SecurityContextHolder; | ||||
| import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder; | ||||
| import com.ruoyi.common.core.domain.model.LoginUser; | ||||
| import com.ruoyi.common.exception.CustomException; | ||||
|  | ||||
| /** | ||||
|  * 安全服务工具类 | ||||
|  *  | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| public class SecurityUtils | ||||
| { | ||||
|     /** | ||||
|      * 获取用户账户 | ||||
|      **/ | ||||
|     public static String getUsername() | ||||
|     { | ||||
|         try | ||||
|         { | ||||
|             return getLoginUser().getUsername(); | ||||
|         } | ||||
|         catch (Exception e) | ||||
|         { | ||||
|             throw new CustomException("获取用户账户异常", HttpStatus.HTTP_UNAUTHORIZED); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取用户 | ||||
|      **/ | ||||
|     public static LoginUser getLoginUser() | ||||
|     { | ||||
|         try | ||||
|         { | ||||
|             return (LoginUser) getAuthentication().getPrincipal(); | ||||
|         } | ||||
|         catch (Exception e) | ||||
|         { | ||||
|             throw new CustomException("获取用户信息异常", HttpStatus.HTTP_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; | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,130 @@ | ||||
| package com.ruoyi.common.utils; | ||||
|  | ||||
| import cn.hutool.core.convert.Convert; | ||||
| import cn.hutool.core.util.StrUtil; | ||||
| import cn.hutool.extra.servlet.ServletUtil; | ||||
| import cn.hutool.http.HttpStatus; | ||||
| 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 javax.servlet.http.HttpServletRequest; | ||||
| import javax.servlet.http.HttpServletResponse; | ||||
| import javax.servlet.http.HttpSession; | ||||
| import java.io.IOException; | ||||
| import java.nio.charset.StandardCharsets; | ||||
|  | ||||
| /** | ||||
|  * 客户端工具类 | ||||
|  * | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| public class ServletUtils extends ServletUtil { | ||||
| 	/** | ||||
| 	 * 获取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); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 获取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   待渲染的字符串 | ||||
| 	 * @return null | ||||
| 	 */ | ||||
| 	public static String renderString(HttpServletResponse response, String string) { | ||||
| 		try { | ||||
| 			response.setStatus(HttpStatus.HTTP_OK); | ||||
| 			response.setContentType(MediaType.APPLICATION_JSON_VALUE); | ||||
| 			response.setCharacterEncoding(StandardCharsets.UTF_8.toString()); | ||||
| 			response.getWriter().print(string); | ||||
| 		} catch (IOException e) { | ||||
| 			e.printStackTrace(); | ||||
| 		} | ||||
| 		return null; | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 是否是Ajax异步请求 | ||||
| 	 * | ||||
| 	 * @param request | ||||
| 	 */ | ||||
| 	public static boolean isAjaxRequest(HttpServletRequest request) { | ||||
|  | ||||
| 		String accept = request.getHeader("accept"); | ||||
| 		if (accept != null && accept.indexOf("application/json") != -1) { | ||||
| 			return true; | ||||
| 		} | ||||
|  | ||||
| 		String xRequestedWith = request.getHeader("X-Requested-With"); | ||||
| 		if (xRequestedWith != null && xRequestedWith.indexOf("XMLHttpRequest") != -1) { | ||||
| 			return true; | ||||
| 		} | ||||
|  | ||||
| 		String uri = request.getRequestURI(); | ||||
| 		if (StrUtil.equalsAnyIgnoreCase(uri, ".json", ".xml")) { | ||||
| 			return true; | ||||
| 		} | ||||
|  | ||||
| 		String ajax = request.getParameter("__ajax"); | ||||
| 		if (StrUtil.equalsAnyIgnoreCase(ajax, "json", "xml")) { | ||||
| 			return true; | ||||
| 		} | ||||
| 		return false; | ||||
| 	} | ||||
|  | ||||
| 	public static String getClientIP() { | ||||
| 		return getClientIP(getRequest()); | ||||
| 	} | ||||
|  | ||||
| } | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	 zt
					zt