mirror of
				https://github.com/dromara/RuoYi-Vue-Plus.git
				synced 2025-10-26 11:53:46 +08:00 
			
		
		
		
	Compare commits
	
		
			358 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 8f34644692 | ||
|  | 7135a917c5 | ||
|  | e2bcd943f5 | ||
|  | d7c855b6f5 | ||
|  | d2774d3706 | ||
|  | adfae03953 | ||
|  | 8298a08592 | ||
|  | 9a3111487a | ||
|  | 0788bcfcb0 | ||
|  | e4fff795b8 | ||
|  | fd7dc204de | ||
|  | 4e807455e0 | ||
|  | 8c9c52417e | ||
|  | be51b4bf25 | ||
|  | 5761ba4271 | ||
|  | 9e075f5c85 | ||
|  | 37d6cc8146 | ||
|  | 2293822dc2 | ||
|  | f5bec70911 | ||
|  | 87c53a049b | ||
|  | 5bcd2825b6 | ||
|  | fc442f482d | ||
|  | a26c02006f | ||
|  | be37190c09 | ||
|  | 036144b9ea | ||
|  | 2f2021caaf | ||
|  | 4a00e4c9b1 | ||
|  | 541c2df8db | ||
|  | 0e39f6fa2d | ||
|  | 17f0c20242 | ||
|  | 24be022d4c | ||
|  | 1f41b1db02 | ||
|  | bd338dd934 | ||
|  | 1920ba94b7 | ||
|  | ccb1449fb2 | ||
|  | 3fb15d8ec3 | ||
|  | 3f4d51b485 | ||
|  | 64ce9daf88 | ||
|  | 0a07236347 | ||
|  | 42d602b7a8 | ||
|  | d4b800036c | ||
|  | 0b0a52c699 | ||
|  | b289f4c1b1 | ||
|  | edbde96487 | ||
|  | 2f380f0c41 | ||
|  | bc76319f28 | ||
|  | 5c2313d622 | ||
|  | aa638475e9 | ||
|  | 4af334f2ad | ||
|  | 21ab56ff78 | ||
|  | 09c4c40d4d | ||
|  | 2cb1b84204 | ||
|  | d1ef8f5eb3 | ||
|  | 39efed1710 | ||
|  | 9a895763d5 | ||
|  | 3950f3c869 | ||
|  | 7347cbaedf | ||
|  | ba2994210c | ||
|  | f1a4b363c6 | ||
|  | a78b5b7b2a | ||
|  | e104f0cce5 | ||
|  | 5d6f76d11a | ||
|  | 99d1760b98 | ||
|  | 4d71cfa90a | ||
|  | e02f692359 | ||
|  | 9a7e66826c | ||
|  | 77ec8d1c9a | ||
|  | 66ce21ec29 | ||
|  | 27e0937235 | ||
|  | aadb7a41cb | ||
|  | 9b3767a954 | ||
|  | 67e20711d4 | ||
|  | d1681dc18c | ||
|  | 0c58ba5057 | ||
|  | 34997ef3e1 | ||
|  | 7d57725490 | ||
|  | 0fa0070062 | ||
|  | 851dc54b49 | ||
|  | 9f2fe90e50 | ||
|  | cf2c4e02c6 | ||
|  | d351c59b38 | ||
|  | 8837119aad | ||
|  | f9a9431958 | ||
|  | 56d209cd20 | ||
|  | 0936aaccea | ||
|  | 89c1e4f91d | ||
|  | 88b5715eae | ||
|  | 5b05d4a123 | ||
|  | 893ef39401 | ||
|  | 890ad682d7 | ||
|  | 5d367b7bf8 | ||
|  | b9f45057b8 | ||
|  | 89008c28df | ||
|  | af7d0a3409 | ||
|  | 1cd2eef899 | ||
|  | 8c7b93ec4f | ||
|  | 3f7bf545b5 | ||
|  | 138842e9ba | ||
|  | 07f3517b6f | ||
|  | 1a8e2b3e63 | ||
|  | e57682aa53 | ||
|  | 46287da60e | ||
|  | 0cecbec3b3 | ||
|  | 515657616a | ||
|  | be7766b5f0 | ||
|  | 168d49fe0b | ||
|  | cbedec7ca6 | ||
|  | fb1bac2114 | ||
|  | d0f399a66a | ||
|  | e73dbd470a | ||
|  | e0cd5381e2 | ||
|  | 8fe07a7e6d | ||
|  | 4a796d0e81 | ||
|  | aee5d417ed | ||
|  | 250c5ba226 | ||
|  | b91c848962 | ||
|  | 027cd821a2 | ||
|  | e3e9fe5106 | ||
|  | 26b0dc336a | ||
|  | d4475d0e8d | ||
|  | 1c935819db | ||
|  | 0c9398776a | ||
|  | aff54ab5fe | ||
|  | 06177addf5 | ||
|  | 9f3b91fe57 | ||
|  | 80d25863db | ||
|  | cf8a7f8678 | ||
|  | a8d798c38a | ||
|  | 975b84a394 | ||
|  | ef121fa664 | ||
|  | f4c1816084 | ||
|  | 86fc709b03 | ||
|  | afc4f9552f | ||
|  | b8b58cb202 | ||
|  | 96970ff951 | ||
|  | c2c0a03932 | ||
|  | 9bc2d2981b | ||
|  | 7cc9d17424 | ||
|  | bc8b5f1079 | ||
|  | 5ec5e1a65d | ||
|  | 1f0e742710 | ||
|  | 801615f780 | ||
|  | 28b9fbb4d2 | ||
|  | d51b77f42b | ||
|  | 6f83ed3285 | ||
|  | 369e4be31c | ||
|  | 5c9fe22780 | ||
|  | a2843b599d | ||
|  | 55ba098e50 | ||
|  | 1fdd81b4fa | ||
|  | d4af02f600 | ||
|  | d7cf341eb3 | ||
|  | 079e6f7c20 | ||
|  | 8a930bd7d5 | ||
|  | 2c79dc906e | ||
|  | 1852017ecc | ||
|  | ca301891db | ||
|  | ce7536df9f | ||
|  | 424d11896a | ||
|  | ae707d340b | ||
|  | edefee46b2 | ||
|  | 77ac99a1d0 | ||
|  | 918708a227 | ||
|  | 3ba8cf4102 | ||
|  | 73db68b08b | ||
|  | ecc4aa5571 | ||
|  | a1a13708be | ||
|  | c7c3da2038 | ||
|  | eb9f3d3772 | ||
|  | 1a61790407 | ||
|  | fe98fba5b5 | ||
|  | ff3ad74fb4 | ||
|  | 7cf93278d3 | ||
|  | 1d9f2d2ce7 | ||
|  | f109bd4a02 | ||
|  | aeebf7fa95 | ||
|  | c57ab693e4 | ||
|  | af54608f62 | ||
|  | f2d502faf8 | ||
|  | 6c1e146bc1 | ||
|  | 15d5eb858c | ||
|  | ac1d7aa69f | ||
|  | da4077f3b7 | ||
|  | 25f9f72366 | ||
|  | 7f04327625 | ||
|  | dd25573ebc | ||
|  | d9e54388e7 | ||
|  | 0b07780619 | ||
|  | 48cb0a1bb1 | ||
|  | 84f00e7cad | ||
|  | a46a7458e1 | ||
|  | c33aa5c969 | ||
|  | 0ee5fd1ac0 | ||
|  | 4eb8809a8a | ||
|  | 9315417935 | ||
|  | 1bb1b3886b | ||
|  | 9cd4f0c332 | ||
|  | 0d8510b8b3 | ||
|  | 6b57a8161c | ||
|  | 0a893d196e | ||
|  | a0a09c23da | ||
|  | 1270b6717f | ||
|  | d1f8b2ed17 | ||
|  | 765deae84d | ||
|  | 686068c8ed | ||
|  | ec45cf04af | ||
|  | b9931cda30 | ||
|  | 8a97d2717b | ||
|  | b80f733cdb | ||
|  | c6b0e61f44 | ||
|  | b2a927cc5e | ||
|  | b73324e800 | ||
|  | 23cbed3aac | ||
|  | 2bd6ad9963 | ||
|  | ea3d66fda3 | ||
|  | 8acce4a7fc | ||
|  | 181eaefea8 | ||
|  | 5e73b88a8a | ||
|  | 5e6d0b79e3 | ||
|  | bf362b9aba | ||
|  | 6b0d9214a9 | ||
|  | 9b57c67d3e | ||
|  | 7545323eba | ||
|  | 5eff9a50b6 | ||
|  | 7dc33c9247 | ||
|  | 6f48fc3c58 | ||
|  | 8558954da7 | ||
|  | 2ae41df23b | ||
|  | 47b8daf69c | ||
|  | ef31a0de42 | ||
|  | 8c956d681b | ||
|  | 89eb44afbc | ||
|  | 9d7e32fb07 | ||
|  | abb88d622a | ||
|  | 1ccdd75019 | ||
|  | aaedafff52 | ||
|  | 3ccf5c0e50 | ||
|  | f25a9832b8 | ||
|  | 11d5e2cdbb | ||
|  | 15f7a7db65 | ||
|  | aec194ba61 | ||
|  | 63eb145772 | ||
|  | db5fea922f | ||
|  | 0214d93299 | ||
|  | 1124a203cc | ||
|  | e8fbf5fdbd | ||
|  | 8fd4ae1282 | ||
|  | eab1b450d5 | ||
|  | 98fc30786e | ||
|  | 4cb8e239ae | ||
|  | ea781293df | ||
|  | 0a2979360e | ||
|  | 0c75048e70 | ||
|  | a07c004983 | ||
|  | 1018a0eda6 | ||
|  | d1e4898924 | ||
|  | a46c4bf04e | ||
|  | d40667a798 | ||
|  | 0eca9e7401 | ||
|  | b76af31155 | ||
|  | 426aeb0a7f | ||
|  | 8a222df60e | ||
|  | 4c79fcb6ed | ||
|  | b27e55f4de | ||
|  | a246cd044e | ||
|  | 880101f338 | ||
|  | a2d0ff1328 | ||
|  | 5479b4db2b | ||
|  | da0c86dd54 | ||
|  | e9f29cd147 | ||
|  | 263448faa2 | ||
|  | 34396de03a | ||
|  | f6fa17a3a8 | ||
|  | 1dceadba4c | ||
|  | 666c80877c | ||
|  | 4cb9aab9ce | ||
|  | 51593aafb2 | ||
|  | 5ced7e05f5 | ||
|  | b97a662f2e | ||
|  | 9639c096ba | ||
|  | e4af41e89a | ||
|  | 52e1717014 | ||
|  | 0ced5d2eaa | ||
|  | ab8111bdeb | ||
|  | 8dc23ea1ce | ||
|  | 8a2e3d13f2 | ||
|  | a29a0648ad | ||
|  | 63d471ec94 | ||
|  | 9fa3eac3aa | ||
|  | a3ba78d7bd | ||
|  | 4b07be6bfd | ||
|  | e320c50e49 | ||
|  | 524ad4e6dd | ||
|  | e515d6f776 | ||
|  | c137965dec | ||
|  | a9e68e13a8 | ||
|  | 77fffda10e | ||
|  | 88fe5c641f | ||
|  | da0309eb09 | ||
|  | 4e6f47d50a | ||
|  | 7e626ea219 | ||
|  | 8a6ccabffb | ||
|  | f546ba85aa | ||
|  | 73e1d3b046 | ||
|  | b2c8062131 | ||
|  | 6805a96e53 | ||
|  | a92667c000 | ||
|  | 4a0619cadb | ||
|  | a7e16d582d | ||
|  | b3a4da5e4d | ||
|  | 5ea298a238 | ||
|  | 859c0915fb | ||
|  | 45fd96cd0d | ||
|  | ffe2b3c24d | ||
|  | b5b8d941f3 | ||
|  | 970561b956 | ||
|  | 9c3b53b701 | ||
|  | 6a69be88c5 | ||
|  | 3a918ae8f6 | ||
|  | f668089e64 | ||
|  | af6a08398e | ||
|  | 4d7def4311 | ||
|  | ae960f5220 | ||
|  | 5be9ce1823 | ||
|  | 5693bbe1f5 | ||
|  | e4267c5c56 | ||
|  | 99dfd42372 | ||
|  | 8d36924cb6 | ||
|  | 80d1e28d0f | ||
|  | 60132e9947 | ||
|  | 940b996e2d | ||
|  | 0055f479cb | ||
|  | 1c41b701dd | ||
|  | e1e26d53e6 | ||
|  | f9131832c7 | ||
|  | ce14b5e9fb | ||
|  | c40aa1f950 | ||
|  | 7d1d3d3f05 | ||
|  | 5e6a895fef | ||
|  | d2fe2a7fc5 | ||
|  | 44a8025e0f | ||
|  | 71e392c1f9 | ||
|  | 416088a2de | ||
|  | e57d11d55a | ||
|  | 781ae8d5c8 | ||
|  | b01d45cf5c | ||
|  | 4941aaa5c1 | ||
|  | 07559a5aaa | ||
|  | 92f030887b | ||
|  | 9476f7f616 | ||
|  | cbe405f6ea | ||
|  | e18cf51c01 | ||
|  | f32813951f | ||
|  | 7ba8fc256b | ||
|  | c7972aa5e6 | ||
|  | eaef38f79c | ||
|  | 7414bc492e | ||
|  | 6f14087a16 | 
| @@ -10,7 +10,7 @@ end_of_line = lf | ||||
| trim_trailing_whitespace = true | ||||
| insert_final_newline = true | ||||
|  | ||||
| [*.{json,yml}] | ||||
| [*.{json,yml,yaml}] | ||||
| indent_size = 2 | ||||
|  | ||||
| [*.md] | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| ### 更改目的 解决了什么问题 | ||||
| ### 更改目的 解决了什么问题(请提交到dev分支) | ||||
|  | ||||
|  | ||||
| ### 描述 做了哪些改动 | ||||
|   | ||||
							
								
								
									
										12
									
								
								.run/ruoyi-admin.run.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								.run/ruoyi-admin.run.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| <component name="ProjectRunConfigurationManager"> | ||||
|   <configuration default="false" name="ruoyi-server" type="docker-deploy" factoryName="dockerfile" server-name="Docker"> | ||||
|     <deployment type="dockerfile"> | ||||
|       <settings> | ||||
|         <option name="imageTag" value="ruoyi/ruoyi-server:4.3.0" /> | ||||
|         <option name="buildOnly" value="true" /> | ||||
|         <option name="sourceFilePath" value="ruoyi-admin/Dockerfile" /> | ||||
|       </settings> | ||||
|     </deployment> | ||||
|     <method v="2" /> | ||||
|   </configuration> | ||||
| </component> | ||||
							
								
								
									
										12
									
								
								.run/ruoyi-monitor-admin.run.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								.run/ruoyi-monitor-admin.run.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| <component name="ProjectRunConfigurationManager"> | ||||
|   <configuration default="false" name="ruoyi-monitor-admin" type="docker-deploy" factoryName="dockerfile" server-name="Docker"> | ||||
|     <deployment type="dockerfile"> | ||||
|       <settings> | ||||
|         <option name="imageTag" value="ruoyi/ruoyi-monitor-admin:4.3.0" /> | ||||
|         <option name="buildOnly" value="true" /> | ||||
|         <option name="sourceFilePath" value="ruoyi-extend/ruoyi-monitor-admin/Dockerfile" /> | ||||
|       </settings> | ||||
|     </deployment> | ||||
|     <method v="2" /> | ||||
|   </configuration> | ||||
| </component> | ||||
							
								
								
									
										12
									
								
								.run/ruoyi-xxl-job-admin.run.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								.run/ruoyi-xxl-job-admin.run.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| <component name="ProjectRunConfigurationManager"> | ||||
|   <configuration default="false" name="ruoyi-xxl-job-admin" type="docker-deploy" factoryName="dockerfile" server-name="Docker"> | ||||
|     <deployment type="dockerfile"> | ||||
|       <settings> | ||||
|         <option name="imageTag" value="ruoyi/ruoyi-xxl-job-admin:4.3.0" /> | ||||
|         <option name="buildOnly" value="true" /> | ||||
|         <option name="sourceFilePath" value="ruoyi-extend/ruoyi-xxl-job-admin/Dockerfile" /> | ||||
|       </settings> | ||||
|     </deployment> | ||||
|     <method v="2" /> | ||||
|   </configuration> | ||||
| </component> | ||||
							
								
								
									
										21
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								README.md
									
									
									
									
									
								
							| @@ -4,13 +4,16 @@ | ||||
| [](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) | ||||
| []() | ||||
| [](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus) | ||||
| []() | ||||
| []() | ||||
| []() | ||||
|  | ||||
| > RuoYi-Vue-Plus 是重写 RuoYi-Vue 针对 `分布式集群` 场景全方位升级(不兼容原框架) | ||||
|  | ||||
| > 项目代码、文档 均开源免费可商用 遵循开源协议在项目中保留开源协议文件即可<br> | ||||
| 活到老写到老 为兴趣而开源 为学习而开源 为让大家真正可以学到技术而开源 | ||||
|  | ||||
| > 系统演示: [传送门](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/系统演示?sort_id=4836388) | ||||
|  | ||||
| | 功能介绍     | 使用技术                | 文档地址                                                                                              | 特性注意事项                     | | ||||
| @@ -42,10 +45,11 @@ | ||||
| | 分布式任务调度  | Xxl-Job             | [Xxl-Job官网](https://www.xuxueli.com/xxl-job/)                                                     | 高性能 高可靠 易扩展                | | ||||
| | 文件存储     | Minio               | [Minio文档](https://docs.min.io/)                                                                   | 本地存储                       | | ||||
| | 文件存储     | 七牛、阿里、腾讯            | [OSS使用文档](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=4359146&doc_id=1469725) | 云存储                        | | ||||
| | 短信模块     | 阿里、腾讯               | [短信使用文档](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=5578491&doc_id=1469725)  | 短信发送                       | | ||||
| | 监控框架     | SpringBoot-Admin    | [SpringBoot-Admin文档](https://codecentric.github.io/spring-boot-admin/current/)                    | 全方位服务监控                    | | ||||
| | 校验框架     | Validation          | [Validation文档](https://docs.jboss.org/hibernate/stable/validator/reference/en-US/html_single/)    | 增强接口安全性、严谨性 支持国际化          | | ||||
| | Excel框架  | Alibaba EasyExcel   | [EasyExcel文档](https://www.yuque.com/easyexcel/doc/easyexcel)                                      | 性能优异 扩展性强                  | | ||||
| | 文档框架     | Knife4j             | [Knife4j文档](https://doc.xiaominfo.com/knife4j/documentation/)                                     | 美化接口文档                     | | ||||
| | 文档框架     | SpringDoc、javadoc   | [接口文档](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=5805266&doc_id=1469725)    | 无注解零入侵基于java注释             | | ||||
| | 工具类框架    | Hutool、Lombok       | [Hutool文档](https://www.hutool.cn/docs/)                                                           | 减少代码冗余 增加安全性               | | ||||
| | 代码生成器    | 适配MP、Knife4j规范化代码   | [Hutool文档](https://www.hutool.cn/docs/)                                                           | 一键生成前后端代码                  | | ||||
| | 部署方式     | Docker              | [Docker文档](https://docs.docker.com/)                                                              | 容器编排 一键部署业务集群              | | ||||
| @@ -55,11 +59,14 @@ | ||||
|  | ||||
| 使用框架前请仔细阅读文档重点注意事项 | ||||
| <br> | ||||
| >[初始化项目 必看](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/关于初始化项目?sort_id=4164117) | ||||
| >>[https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/关于初始化项目?sort_id=4164117](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/关于初始化项目?sort_id=4164117) | ||||
| >[初始化项目 必看](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=4164117&doc_id=1469725) | ||||
| >>[https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=4164117&doc_id=1469725](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=4164117&doc_id=1469725) | ||||
| > | ||||
| >[部署项目 必看](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/关于应用部署?sort_id=4219382) | ||||
| >>[https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/关于应用部署?sort_id=4219382](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/关于应用部署?sort_id=4219382) | ||||
| >[专栏与视频 入门必看](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=5473272&doc_id=1469725) | ||||
| >>[https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=5473272&doc_id=1469725](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=5473272&doc_id=1469725) | ||||
| > | ||||
| >[部署项目 必看](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=4219382&doc_id=1469725) | ||||
| >>[https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=4219382&doc_id=1469725](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=4219382&doc_id=1469725) | ||||
| >  | ||||
| >[参考文档 Wiki](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages) | ||||
| >>[https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages) | ||||
|   | ||||
							
								
								
									
										163
									
								
								pom.xml
									
									
									
									
									
								
							
							
						
						
									
										163
									
								
								pom.xml
									
									
									
									
									
								
							| @@ -6,56 +6,46 @@ | ||||
|  | ||||
|     <groupId>com.ruoyi</groupId> | ||||
|     <artifactId>ruoyi-vue-plus</artifactId> | ||||
|     <version>4.1.0</version> | ||||
|     <version>4.3.0</version> | ||||
|  | ||||
|     <name>RuoYi-Vue-Plus</name> | ||||
|     <url>https://gitee.com/JavaLionLi/RuoYi-Vue-Plus</url> | ||||
|     <description>RuoYi-Vue-Plus后台管理系统</description> | ||||
|  | ||||
|     <properties> | ||||
|         <ruoyi-vue-plus.version>4.1.0</ruoyi-vue-plus.version> | ||||
|         <spring-boot.version>2.6.7</spring-boot.version> | ||||
|         <ruoyi-vue-plus.version>4.3.0</ruoyi-vue-plus.version> | ||||
|         <spring-boot.version>2.7.3</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.2.2</maven-jar-plugin.version> | ||||
|         <spring-boot.mybatis>2.2.0</spring-boot.mybatis> | ||||
|         <druid.version>1.2.8</druid.version> | ||||
|         <knife4j.version>3.0.3</knife4j.version> | ||||
|         <swagger-annotations.version>1.5.22</swagger-annotations.version> | ||||
|         <poi.version>4.1.2</poi.version> | ||||
|         <commons-compress.version>1.21</commons-compress.version> | ||||
|         <easyexcel.version>3.0.5</easyexcel.version> | ||||
|         <spring-boot.mybatis>2.2.2</spring-boot.mybatis> | ||||
|         <druid.version>1.2.12</druid.version> | ||||
|         <springdoc.version>1.6.11</springdoc.version> | ||||
|         <poi.version>5.2.2</poi.version> | ||||
|         <easyexcel.version>3.1.1</easyexcel.version> | ||||
|         <velocity.version>2.3</velocity.version> | ||||
|         <satoken.version>1.29.0</satoken.version> | ||||
|         <mybatis-plus.version>3.5.1</mybatis-plus.version> | ||||
|         <satoken.version>1.30.0</satoken.version> | ||||
|         <mybatis-plus.version>3.5.2</mybatis-plus.version> | ||||
|         <p6spy.version>3.9.1</p6spy.version> | ||||
|         <hutool.version>5.7.22</hutool.version> | ||||
|         <okhttp.version>4.9.3</okhttp.version> | ||||
|         <spring-boot-admin.version>2.6.6</spring-boot-admin.version> | ||||
|         <redisson.version>3.17.0</redisson.version> | ||||
|         <lock4j.version>2.2.1</lock4j.version> | ||||
|         <dynamic-ds.version>3.5.1</dynamic-ds.version> | ||||
|         <tlog.version>1.3.6</tlog.version> | ||||
|         <xxl-job.version>2.3.0</xxl-job.version> | ||||
|         <hutool.version>5.8.6</hutool.version> | ||||
|         <okhttp.version>4.10.0</okhttp.version> | ||||
|         <spring-boot-admin.version>2.7.4</spring-boot-admin.version> | ||||
|         <redisson.version>3.17.6</redisson.version> | ||||
|         <lock4j.version>2.2.2</lock4j.version> | ||||
|         <dynamic-ds.version>3.5.2</dynamic-ds.version> | ||||
|         <tlog.version>1.4.3</tlog.version> | ||||
|         <xxl-job.version>2.3.1</xxl-job.version> | ||||
|         <lombok.version>1.18.24</lombok.version> | ||||
|  | ||||
|         <!-- jdk11 缺失依赖 jaxb--> | ||||
|         <jaxb.version>3.0.1</jaxb.version> | ||||
|         <!-- 统一 guava 版本 解决隐式漏洞问题 --> | ||||
|         <guava.version>30.0-jre</guava.version> | ||||
|         <guava.version>31.1-jre</guava.version> | ||||
|  | ||||
|         <!-- OSS 配置 --> | ||||
|         <qiniu.version>7.9.5</qiniu.version> | ||||
|         <aliyun.oss.version>3.14.0</aliyun.oss.version> | ||||
|         <qcloud.cos.version>5.6.72</qcloud.cos.version> | ||||
|         <minio.version>8.3.8</minio.version> | ||||
|  | ||||
|         <!-- docker 配置 --> | ||||
|         <docker.registry.url>localhost</docker.registry.url> | ||||
|         <docker.registry.host>http://${docker.registry.url}:2375</docker.registry.host> | ||||
|         <docker.namespace>ruoyi</docker.namespace> | ||||
|         <docker.plugin.version>1.2.2</docker.plugin.version> | ||||
|  | ||||
|         <aws-java-sdk-s3.version>1.12.300</aws-java-sdk-s3.version> | ||||
|         <!-- SMS 配置 --> | ||||
|         <aliyun.sms.version>2.0.18</aliyun.sms.version> | ||||
|         <tencent.sms.version>3.1.591</tencent.sms.version> | ||||
|     </properties> | ||||
|  | ||||
|     <!-- 依赖声明 --> | ||||
| @@ -88,46 +78,38 @@ | ||||
|             </dependency> | ||||
|  | ||||
|             <dependency> | ||||
|                 <groupId>com.github.xiaoymin</groupId> | ||||
|                 <artifactId>knife4j-spring-boot-starter</artifactId> | ||||
|                 <version>${knife4j.version}</version> | ||||
|                 <exclusions> | ||||
|                     <exclusion> | ||||
|                         <artifactId>swagger-annotations</artifactId> | ||||
|                         <groupId>io.swagger</groupId> | ||||
|                     </exclusion> | ||||
|                 </exclusions> | ||||
|                 <groupId>org.springdoc</groupId> | ||||
|                 <artifactId>springdoc-openapi-webmvc-core</artifactId> | ||||
|                 <version>${springdoc.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <dependency> | ||||
|                 <groupId>io.swagger</groupId> | ||||
|                 <artifactId>swagger-annotations</artifactId> | ||||
|                 <version>${swagger-annotations.version}</version> | ||||
|                 <groupId>org.springdoc</groupId> | ||||
|                 <artifactId>springdoc-openapi-javadoc</artifactId> | ||||
|                 <version>${springdoc.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <!-- excel工具 --> | ||||
|             <dependency> | ||||
|                 <groupId>org.projectlombok</groupId> | ||||
|                 <artifactId>lombok</artifactId> | ||||
|                 <version>${lombok.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <dependency> | ||||
|                 <groupId>org.apache.poi</groupId> | ||||
|                 <artifactId>poi</artifactId> | ||||
|                 <version>${poi.version}</version> | ||||
|             </dependency> | ||||
|             <dependency> | ||||
|                 <groupId>org.apache.poi</groupId> | ||||
|                 <artifactId>poi-ooxml</artifactId> | ||||
|                 <version>${poi.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <!-- 修复poi漏洞 --> | ||||
|             <dependency> | ||||
|                 <groupId>org.apache.commons</groupId> | ||||
|                 <artifactId>commons-compress</artifactId> | ||||
|                 <version>${commons-compress.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <dependency> | ||||
|                 <groupId>com.alibaba</groupId> | ||||
|                 <artifactId>easyexcel</artifactId> | ||||
|                 <version>${easyexcel.version}</version> | ||||
|                 <exclusions> | ||||
|                     <exclusion> | ||||
|                         <groupId>org.apache.poi</groupId> | ||||
|                         <artifactId>poi</artifactId> | ||||
|                     </exclusion> | ||||
|                     <exclusion> | ||||
|                         <groupId>org.apache.poi</groupId> | ||||
|                         <artifactId>poi-ooxml-schemas</artifactId> | ||||
| @@ -153,13 +135,12 @@ | ||||
|                 <groupId>cn.dev33</groupId> | ||||
|                 <artifactId>sa-token-jwt</artifactId> | ||||
|                 <version>${satoken.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <!-- jdk11 缺失依赖 jaxb--> | ||||
|             <dependency> | ||||
|                 <groupId>com.sun.xml.bind</groupId> | ||||
|                 <artifactId>jaxb-impl</artifactId> | ||||
|                 <version>${jaxb.version}</version> | ||||
|                 <exclusions> | ||||
|                     <exclusion> | ||||
|                         <groupId>cn.hutool</groupId> | ||||
|                         <artifactId>hutool-all</artifactId> | ||||
|                     </exclusion> | ||||
|                 </exclusions> | ||||
|             </dependency> | ||||
|  | ||||
|             <!-- dynamic-datasource 多数据源--> | ||||
| @@ -192,6 +173,24 @@ | ||||
|                 <version>${okhttp.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <dependency> | ||||
|                 <groupId>com.amazonaws</groupId> | ||||
|                 <artifactId>aws-java-sdk-s3</artifactId> | ||||
|                 <version>${aws-java-sdk-s3.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <dependency> | ||||
|                 <groupId>com.aliyun</groupId> | ||||
|                 <artifactId>dysmsapi20170525</artifactId> | ||||
|                 <version>${aliyun.sms.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <dependency> | ||||
|                 <groupId>com.tencentcloudapi</groupId> | ||||
|                 <artifactId>tencentcloud-sdk-java</artifactId> | ||||
|                 <version>${tencent.sms.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <dependency> | ||||
|                 <groupId>de.codecentric</groupId> | ||||
|                 <artifactId>spring-boot-admin-starter-server</artifactId> | ||||
| @@ -297,6 +296,13 @@ | ||||
|                 <version>${ruoyi-vue-plus.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <!-- SMS短信模块 --> | ||||
|             <dependency> | ||||
|                 <groupId>com.ruoyi</groupId> | ||||
|                 <artifactId>ruoyi-sms</artifactId> | ||||
|                 <version>${ruoyi-vue-plus.version}</version> | ||||
|             </dependency> | ||||
|  | ||||
|             <!-- demo模块 --> | ||||
|             <dependency> | ||||
|                 <groupId>com.ruoyi</groupId> | ||||
| @@ -317,6 +323,7 @@ | ||||
|         <module>ruoyi-demo</module> | ||||
|         <module>ruoyi-extend</module> | ||||
|         <module>ruoyi-oss</module> | ||||
|         <module>ruoyi-sms</module> | ||||
|     </modules> | ||||
|     <packaging>pom</packaging> | ||||
|  | ||||
| @@ -330,6 +337,23 @@ | ||||
|                     <source>${java.version}</source> | ||||
|                     <target>${java.version}</target> | ||||
|                     <encoding>${project.build.sourceEncoding}</encoding> | ||||
|                     <annotationProcessorPaths> | ||||
|                         <path> | ||||
|                             <groupId>com.github.therapi</groupId> | ||||
|                             <artifactId>therapi-runtime-javadoc-scribe</artifactId> | ||||
|                             <version>0.15.0</version> | ||||
|                         </path> | ||||
|                         <path> | ||||
|                             <groupId>org.projectlombok</groupId> | ||||
|                             <artifactId>lombok</artifactId> | ||||
|                             <version>${lombok.version}</version> | ||||
|                         </path> | ||||
|                         <path> | ||||
|                             <groupId>org.springframework.boot</groupId> | ||||
|                             <artifactId>spring-boot-configuration-processor</artifactId> | ||||
|                             <version>${spring-boot.version}</version> | ||||
|                         </path> | ||||
|                     </annotationProcessorPaths> | ||||
|                 </configuration> | ||||
|             </plugin> | ||||
|         </plugins> | ||||
| @@ -385,8 +409,6 @@ | ||||
|                 <!-- 环境标识,需要与配置文件的名称相对应 --> | ||||
|                 <profiles.active>local</profiles.active> | ||||
|                 <logging.level>debug</logging.level> | ||||
|                 <knife4j.production>false</knife4j.production> | ||||
|                 <endpoints.include>'*'</endpoints.include> | ||||
|             </properties> | ||||
|         </profile> | ||||
|         <profile> | ||||
| @@ -395,8 +417,6 @@ | ||||
|                 <!-- 环境标识,需要与配置文件的名称相对应 --> | ||||
|                 <profiles.active>dev</profiles.active> | ||||
|                 <logging.level>debug</logging.level> | ||||
|                 <knife4j.production>false</knife4j.production> | ||||
|                 <endpoints.include>'*'</endpoints.include> | ||||
|             </properties> | ||||
|             <activation> | ||||
|                 <!-- 默认环境 --> | ||||
| @@ -408,10 +428,9 @@ | ||||
|             <properties> | ||||
|                 <profiles.active>prod</profiles.active> | ||||
|                 <logging.level>warn</logging.level> | ||||
|                 <knife4j.production>true</knife4j.production> | ||||
|                 <endpoints.include>health, info, logfile</endpoints.include> | ||||
|             </properties> | ||||
|         </profile> | ||||
|     </profiles> | ||||
|  | ||||
| </project> | ||||
|  | ||||
|   | ||||
| @@ -8,8 +8,10 @@ RUN mkdir -p /ruoyi/server/temp | ||||
|  | ||||
| WORKDIR /ruoyi/server | ||||
|  | ||||
| EXPOSE 8080 | ||||
| ENV SERVER_PORT=8080 | ||||
|  | ||||
| EXPOSE ${SERVER_PORT} | ||||
|  | ||||
| ADD ./target/ruoyi-admin.jar ./app.jar | ||||
|  | ||||
| ENTRYPOINT ["java", "-Djava.security.egd=file:/dev/./urandom", "-jar", "app.jar"] | ||||
| ENTRYPOINT ["java", "-Djava.security.egd=file:/dev/./urandom", "-Dserver.port=${SERVER_PORT}","-jar", "app.jar"] | ||||
|   | ||||
| @@ -5,7 +5,7 @@ | ||||
|     <parent> | ||||
|         <artifactId>ruoyi-vue-plus</artifactId> | ||||
|         <groupId>com.ruoyi</groupId> | ||||
|         <version>4.1.0</version> | ||||
|         <version>4.3.0</version> | ||||
|     </parent> | ||||
|     <modelVersion>4.0.0</modelVersion> | ||||
|     <packaging>jar</packaging> | ||||
| @@ -107,25 +107,6 @@ | ||||
|                     <warName>${project.artifactId}</warName> | ||||
|                 </configuration> | ||||
|             </plugin> | ||||
|             <plugin> | ||||
|                 <groupId>com.spotify</groupId> | ||||
|                 <artifactId>docker-maven-plugin</artifactId> | ||||
|                 <version>${docker.plugin.version}</version> | ||||
|                 <configuration> | ||||
|                     <imageName>${docker.namespace}/ruoyi-server:${project.version}</imageName> | ||||
|                     <dockerDirectory>${project.basedir}</dockerDirectory> | ||||
|                     <dockerHost>${docker.registry.host}</dockerHost> | ||||
|                     <registryUrl>${docker.registry.url}</registryUrl> | ||||
|                     <serverId>${docker.registry.url}</serverId> | ||||
|                     <resources> | ||||
|                         <resource> | ||||
|                             <targetPath>/</targetPath> | ||||
|                             <directory>${project.build.directory}</directory> | ||||
|                             <include>${project.build.finalName}.jar</include> | ||||
|                         </resource> | ||||
|                     </resources> | ||||
|                 </configuration> | ||||
|             </plugin> | ||||
|         </plugins> | ||||
|     </build> | ||||
|  | ||||
|   | ||||
| @@ -4,6 +4,9 @@ import cn.hutool.captcha.AbstractCaptcha; | ||||
| import cn.hutool.captcha.generator.CodeGenerator; | ||||
| import cn.hutool.core.convert.Convert; | ||||
| import cn.hutool.core.util.IdUtil; | ||||
| import cn.hutool.core.util.RandomUtil; | ||||
| import com.ruoyi.common.annotation.Anonymous; | ||||
| import com.ruoyi.common.constant.CacheConstants; | ||||
| import com.ruoyi.common.constant.Constants; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.enums.CaptchaType; | ||||
| @@ -12,45 +15,78 @@ import com.ruoyi.common.utils.redis.RedisUtils; | ||||
| import com.ruoyi.common.utils.reflect.ReflectUtils; | ||||
| import com.ruoyi.common.utils.spring.SpringUtils; | ||||
| import com.ruoyi.framework.config.properties.CaptchaProperties; | ||||
| import com.ruoyi.sms.config.properties.SmsProperties; | ||||
| import com.ruoyi.sms.core.SmsTemplate; | ||||
| import com.ruoyi.sms.entity.SmsResult; | ||||
| import com.ruoyi.system.service.ISysConfigService; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import lombok.extern.slf4j.Slf4j; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.GetMapping; | ||||
| import org.springframework.web.bind.annotation.RestController; | ||||
|  | ||||
| import javax.validation.constraints.NotBlank; | ||||
| import java.time.Duration; | ||||
| import java.util.HashMap; | ||||
| import java.util.Map; | ||||
| import java.util.concurrent.TimeUnit; | ||||
|  | ||||
| /** | ||||
|  * 验证码操作处理 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Api(value = "验证码操作处理", tags = {"验证码管理"}) | ||||
| @Anonymous | ||||
| @Slf4j | ||||
| @Validated | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| public class CaptchaController { | ||||
|  | ||||
|     private final CaptchaProperties captchaProperties; | ||||
|     private final SmsProperties smsProperties; | ||||
|     private final ISysConfigService configService; | ||||
|  | ||||
|     /** | ||||
|      * 短信验证码 | ||||
|      * | ||||
|      * @param phonenumber 用户手机号 | ||||
|      */ | ||||
|     @GetMapping("/captchaSms") | ||||
|     public R<Void> smsCaptcha(@NotBlank(message = "{user.phonenumber.not.blank}") | ||||
|                               String phonenumber) { | ||||
|         if (!smsProperties.getEnabled()) { | ||||
|             return R.fail("当前系统没有开启短信功能!"); | ||||
|         } | ||||
|         String key = CacheConstants.CAPTCHA_CODE_KEY + phonenumber; | ||||
|         String code = RandomUtil.randomNumbers(4); | ||||
|         RedisUtils.setCacheObject(key, code, Duration.ofMinutes(Constants.CAPTCHA_EXPIRATION)); | ||||
|         // 验证码模板id 自行处理 (查数据库或写死均可) | ||||
|         String templateId = ""; | ||||
|         Map<String, String> map = new HashMap<>(1); | ||||
|         map.put("code", code); | ||||
|         SmsTemplate smsTemplate = SpringUtils.getBean(SmsTemplate.class); | ||||
|         SmsResult result = smsTemplate.send(phonenumber, templateId, map); | ||||
|         if (!result.isSuccess()) { | ||||
|             log.error("验证码短信发送异常 => {}", result); | ||||
|             return R.fail(result.getMessage()); | ||||
|         } | ||||
|         return R.ok(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 生成验证码 | ||||
|      */ | ||||
|     @ApiOperation("生成验证码") | ||||
|     @GetMapping("/captchaImage") | ||||
|     public R<Map<String, Object>> getCode() { | ||||
|         Map<String, Object> ajax = new HashMap<>(); | ||||
|         boolean captchaOnOff = configService.selectCaptchaOnOff(); | ||||
|         ajax.put("captchaOnOff", captchaOnOff); | ||||
|         if (!captchaOnOff) { | ||||
|         boolean captchaEnabled = configService.selectCaptchaEnabled(); | ||||
|         ajax.put("captchaEnabled", captchaEnabled); | ||||
|         if (!captchaEnabled) { | ||||
|             return R.ok(ajax); | ||||
|         } | ||||
|         // 保存验证码信息 | ||||
|         String uuid = IdUtil.simpleUUID(); | ||||
|         String verifyKey = Constants.CAPTCHA_CODE_KEY + uuid; | ||||
|         String verifyKey = CacheConstants.CAPTCHA_CODE_KEY + uuid; | ||||
|         // 生成验证码 | ||||
|         CaptchaType captchaType = captchaProperties.getType(); | ||||
|         boolean isMath = CaptchaType.MATH == captchaType; | ||||
| @@ -60,7 +96,7 @@ public class CaptchaController { | ||||
|         captcha.setGenerator(codeGenerator); | ||||
|         captcha.createCode(); | ||||
|         String code = isMath ? getCodeResult(captcha.getCode()) : captcha.getCode(); | ||||
|         RedisUtils.setCacheObject(verifyKey, code, Constants.CAPTCHA_EXPIRATION, TimeUnit.MINUTES); | ||||
|         RedisUtils.setCacheObject(verifyKey, code, Duration.ofMinutes(Constants.CAPTCHA_EXPIRATION)); | ||||
|         ajax.put("uuid", uuid); | ||||
|         ajax.put("img", captcha.getImageBase64()); | ||||
|         return R.ok(ajax); | ||||
|   | ||||
| @@ -1,40 +1,59 @@ | ||||
| package com.ruoyi.web.controller.monitor; | ||||
|  | ||||
| import cn.dev33.satoken.annotation.SaCheckPermission; | ||||
| import cn.hutool.core.collection.CollUtil; | ||||
| import com.ruoyi.common.constant.CacheConstants; | ||||
| import com.ruoyi.common.constant.CacheNames; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.utils.JsonUtils; | ||||
| import com.ruoyi.common.utils.StringUtils; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import com.ruoyi.common.utils.redis.CacheUtils; | ||||
| import com.ruoyi.common.utils.redis.RedisUtils; | ||||
| import com.ruoyi.system.domain.SysCache; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.data.redis.connection.RedisServerCommands; | ||||
| import org.springframework.data.redis.core.RedisCallback; | ||||
| import org.springframework.data.redis.core.RedisTemplate; | ||||
| import org.springframework.web.bind.annotation.GetMapping; | ||||
| import org.springframework.web.bind.annotation.RequestMapping; | ||||
| import org.springframework.web.bind.annotation.RestController; | ||||
| import org.redisson.spring.data.connection.RedissonConnectionFactory; | ||||
| import org.springframework.data.redis.connection.RedisConnection; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
|  | ||||
| import java.util.*; | ||||
| import java.util.stream.Collectors; | ||||
|  | ||||
| /** | ||||
|  * 缓存监控 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Api(value = "缓存监控", tags = {"缓存监控管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/monitor/cache") | ||||
| public class CacheController { | ||||
|  | ||||
|     private final RedisTemplate<String, String> redisTemplate; | ||||
|     private final RedissonConnectionFactory connectionFactory; | ||||
|  | ||||
|     @ApiOperation("获取缓存监控详细信息") | ||||
|     private final static List<SysCache> CACHES = new ArrayList<>(); | ||||
|  | ||||
|     static { | ||||
|         CACHES.add(new SysCache(CacheConstants.LOGIN_TOKEN_KEY, "用户信息")); | ||||
|         CACHES.add(new SysCache(CacheConstants.ONLINE_TOKEN_KEY, "在线用户")); | ||||
|         CACHES.add(new SysCache(CacheNames.SYS_CONFIG, "配置信息")); | ||||
|         CACHES.add(new SysCache(CacheNames.SYS_DICT, "数据字典")); | ||||
|         CACHES.add(new SysCache(CacheConstants.CAPTCHA_CODE_KEY, "验证码")); | ||||
|         CACHES.add(new SysCache(CacheConstants.REPEAT_SUBMIT_KEY, "防重提交")); | ||||
|         CACHES.add(new SysCache(CacheConstants.RATE_LIMIT_KEY, "限流处理")); | ||||
|         CACHES.add(new SysCache(CacheNames.SYS_OSS_CONFIG, "OSS配置")); | ||||
|         CACHES.add(new SysCache(CacheConstants.PWD_ERR_CNT_KEY, "密码错误次数")); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取缓存监控列表 | ||||
|      */ | ||||
|     @SaCheckPermission("monitor:cache:list") | ||||
|     @GetMapping() | ||||
|     public R<Map<String, Object>> getInfo() throws Exception { | ||||
|         Properties info = (Properties) redisTemplate.execute((RedisCallback<Object>) RedisServerCommands::info); | ||||
|         Properties commandStats = (Properties) redisTemplate.execute((RedisCallback<Object>) connection -> connection.info("commandstats")); | ||||
|         Object dbSize = redisTemplate.execute((RedisCallback<Object>) RedisServerCommands::dbSize); | ||||
|         RedisConnection connection = connectionFactory.getConnection(); | ||||
|         Properties info = connection.info(); | ||||
|         Properties commandStats = connection.info("commandstats"); | ||||
|         Long dbSize = connection.dbSize(); | ||||
|  | ||||
|         Map<String, Object> result = new HashMap<>(3); | ||||
|         result.put("info", info); | ||||
| @@ -53,4 +72,98 @@ public class CacheController { | ||||
|         result.put("commandStats", pieList); | ||||
|         return R.ok(result); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取缓存监控缓存名列表 | ||||
|      */ | ||||
|     @SaCheckPermission("monitor:cache:list") | ||||
|     @GetMapping("/getNames") | ||||
|     public R<List<SysCache>> cache() { | ||||
|         return R.ok(CACHES); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取缓存监控Key列表 | ||||
|      * | ||||
|      * @param cacheName 缓存名 | ||||
|      */ | ||||
|     @SaCheckPermission("monitor:cache:list") | ||||
|     @GetMapping("/getKeys/{cacheName}") | ||||
|     public R<Collection<String>> getCacheKeys(@PathVariable String cacheName) { | ||||
|         Collection<String> cacheKeys = new HashSet<>(0); | ||||
|         if (isCacheNames(cacheName)) { | ||||
|             Set<Object> keys = CacheUtils.keys(cacheName); | ||||
|             if (CollUtil.isNotEmpty(keys)) { | ||||
|                 cacheKeys = keys.stream().map(Object::toString).collect(Collectors.toList()); | ||||
|             } | ||||
|         } else { | ||||
|             cacheKeys = RedisUtils.keys(cacheName + "*"); | ||||
|         } | ||||
|         return R.ok(cacheKeys); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取缓存监控缓存值详情 | ||||
|      * | ||||
|      * @param cacheName 缓存名 | ||||
|      * @param cacheKey  缓存key | ||||
|      */ | ||||
|     @SaCheckPermission("monitor:cache:list") | ||||
|     @GetMapping("/getValue/{cacheName}/{cacheKey}") | ||||
|     public R<SysCache> getCacheValue(@PathVariable String cacheName, @PathVariable String cacheKey) { | ||||
|         Object cacheValue; | ||||
|         if (isCacheNames(cacheName)) { | ||||
|             cacheValue = CacheUtils.get(cacheName, cacheKey); | ||||
|         } else { | ||||
|             cacheValue = RedisUtils.getCacheObject(cacheKey); | ||||
|         } | ||||
|         SysCache sysCache = new SysCache(cacheName, cacheKey, JsonUtils.toJsonString(cacheValue)); | ||||
|         return R.ok(sysCache); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 清理缓存监控缓存名 | ||||
|      * | ||||
|      * @param cacheName 缓存名 | ||||
|      */ | ||||
|     @SaCheckPermission("monitor:cache:list") | ||||
|     @DeleteMapping("/clearCacheName/{cacheName}") | ||||
|     public R<Void> clearCacheName(@PathVariable String cacheName) { | ||||
|         if (isCacheNames(cacheName)) { | ||||
|             CacheUtils.clear(cacheName); | ||||
|         } else { | ||||
|             RedisUtils.deleteKeys(cacheName + "*"); | ||||
|         } | ||||
|         return R.ok(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 清理缓存监控Key | ||||
|      * | ||||
|      * @param cacheKey key名 | ||||
|      */ | ||||
|     @SaCheckPermission("monitor:cache:list") | ||||
|     @DeleteMapping("/clearCacheKey/{cacheName}/{cacheKey}") | ||||
|     public R<Void> clearCacheKey(@PathVariable String cacheName, @PathVariable String cacheKey) { | ||||
|         if (isCacheNames(cacheName)) { | ||||
|             CacheUtils.evict(cacheName, cacheKey); | ||||
|         } else { | ||||
|             RedisUtils.deleteObject(cacheKey); | ||||
|         } | ||||
|         return R.ok(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 清理全部缓存监控 | ||||
|      */ | ||||
|     @SaCheckPermission("monitor:cache:list") | ||||
|     @DeleteMapping("/clearCacheAll") | ||||
|     public R<Void> clearCacheAll() { | ||||
|         RedisUtils.deleteKeys("*"); | ||||
|         return R.ok(); | ||||
|     } | ||||
|  | ||||
|     private boolean isCacheNames(String cacheName) { | ||||
|         return !StringUtils.contains(cacheName, ":"); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -2,16 +2,16 @@ package com.ruoyi.web.controller.monitor; | ||||
|  | ||||
| import cn.dev33.satoken.annotation.SaCheckPermission; | ||||
| import com.ruoyi.common.annotation.Log; | ||||
| import com.ruoyi.common.constant.CacheConstants; | ||||
| import com.ruoyi.common.core.controller.BaseController; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.domain.PageQuery; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.page.TableDataInfo; | ||||
| import com.ruoyi.common.enums.BusinessType; | ||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | ||||
| import com.ruoyi.common.utils.redis.RedisUtils; | ||||
| import com.ruoyi.system.domain.SysLogininfor; | ||||
| import com.ruoyi.system.service.ISysLogininforService; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
| @@ -25,7 +25,6 @@ import java.util.List; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "系统访问记录", tags = {"系统访问记录管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/monitor/logininfor") | ||||
| @@ -33,14 +32,18 @@ public class SysLogininforController extends BaseController { | ||||
|  | ||||
|     private final ISysLogininforService logininforService; | ||||
|  | ||||
|     @ApiOperation("查询系统访问记录列表") | ||||
|     /** | ||||
|      * 获取系统访问记录列表 | ||||
|      */ | ||||
|     @SaCheckPermission("monitor:logininfor:list") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo<SysLogininfor> list(SysLogininfor logininfor, PageQuery pageQuery) { | ||||
|         return logininforService.selectPageLogininforList(logininfor, pageQuery); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("导出系统访问记录列表") | ||||
|     /** | ||||
|      * 导出系统访问记录列表 | ||||
|      */ | ||||
|     @Log(title = "登录日志", businessType = BusinessType.EXPORT) | ||||
|     @SaCheckPermission("monitor:logininfor:export") | ||||
|     @PostMapping("/export") | ||||
| @@ -49,7 +52,10 @@ public class SysLogininforController extends BaseController { | ||||
|         ExcelUtil.exportExcel(list, "登录日志", SysLogininfor.class, response); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("删除系统访问记录") | ||||
|     /** | ||||
|      * 批量删除登录日志 | ||||
|      * @param infoIds 日志ids | ||||
|      */ | ||||
|     @SaCheckPermission("monitor:logininfor:remove") | ||||
|     @Log(title = "登录日志", businessType = BusinessType.DELETE) | ||||
|     @DeleteMapping("/{infoIds}") | ||||
| @@ -57,7 +63,9 @@ public class SysLogininforController extends BaseController { | ||||
|         return toAjax(logininforService.deleteLogininforByIds(infoIds)); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("清空系统访问记录") | ||||
|     /** | ||||
|      * 清理系统访问记录 | ||||
|      */ | ||||
|     @SaCheckPermission("monitor:logininfor:remove") | ||||
|     @Log(title = "登录日志", businessType = BusinessType.CLEAN) | ||||
|     @DeleteMapping("/clean") | ||||
| @@ -65,4 +73,16 @@ public class SysLogininforController extends BaseController { | ||||
|         logininforService.cleanLogininfor(); | ||||
|         return R.ok(); | ||||
|     } | ||||
|  | ||||
|     @SaCheckPermission("monitor:logininfor:unlock") | ||||
|     @Log(title = "账户解锁", businessType = BusinessType.OTHER) | ||||
|     @GetMapping("/unlock/{userName}") | ||||
|     public R<Void> unlock(@PathVariable("userName") String userName) { | ||||
|         String loginName = CacheConstants.PWD_ERR_CNT_KEY + userName; | ||||
|         if (RedisUtils.hasKey(loginName)) { | ||||
|             RedisUtils.deleteObject(loginName); | ||||
|         } | ||||
|         return R.ok(); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -3,15 +3,13 @@ package com.ruoyi.web.controller.monitor; | ||||
| import cn.dev33.satoken.annotation.SaCheckPermission; | ||||
| import com.ruoyi.common.annotation.Log; | ||||
| import com.ruoyi.common.core.controller.BaseController; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.domain.PageQuery; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| 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 io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
| @@ -25,7 +23,6 @@ import java.util.List; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "操作日志记录", tags = {"操作日志记录管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/monitor/operlog") | ||||
| @@ -33,14 +30,18 @@ public class SysOperlogController extends BaseController { | ||||
|  | ||||
|     private final ISysOperLogService operLogService; | ||||
|  | ||||
|     @ApiOperation("查询操作日志记录列表") | ||||
|     /** | ||||
|      * 获取操作日志记录列表 | ||||
|      */ | ||||
|     @SaCheckPermission("monitor:operlog:list") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo<SysOperLog> list(SysOperLog operLog, PageQuery pageQuery) { | ||||
|         return operLogService.selectPageOperLogList(operLog, pageQuery); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("导出操作日志记录列表") | ||||
|     /** | ||||
|      * 导出操作日志记录列表 | ||||
|      */ | ||||
|     @Log(title = "操作日志", businessType = BusinessType.EXPORT) | ||||
|     @SaCheckPermission("monitor:operlog:export") | ||||
|     @PostMapping("/export") | ||||
| @@ -49,7 +50,10 @@ public class SysOperlogController extends BaseController { | ||||
|         ExcelUtil.exportExcel(list, "操作日志", SysOperLog.class, response); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("删除操作日志记录") | ||||
|     /** | ||||
|      * 批量删除操作日志记录 | ||||
|      * @param operIds 日志ids | ||||
|      */ | ||||
|     @Log(title = "操作日志", businessType = BusinessType.DELETE) | ||||
|     @SaCheckPermission("monitor:operlog:remove") | ||||
|     @DeleteMapping("/{operIds}") | ||||
| @@ -57,7 +61,9 @@ public class SysOperlogController extends BaseController { | ||||
|         return toAjax(operLogService.deleteOperLogByIds(operIds)); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("清空操作日志记录") | ||||
|     /** | ||||
|      * 清理操作日志记录 | ||||
|      */ | ||||
|     @Log(title = "操作日志", businessType = BusinessType.CLEAN) | ||||
|     @SaCheckPermission("monitor:operlog:remove") | ||||
|     @DeleteMapping("/clean") | ||||
|   | ||||
| @@ -5,37 +5,39 @@ import cn.dev33.satoken.exception.NotLoginException; | ||||
| import cn.dev33.satoken.stp.StpUtil; | ||||
| import cn.hutool.core.bean.BeanUtil; | ||||
| import com.ruoyi.common.annotation.Log; | ||||
| import com.ruoyi.common.constant.Constants; | ||||
| import com.ruoyi.common.constant.CacheConstants; | ||||
| import com.ruoyi.common.core.controller.BaseController; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.domain.dto.UserOnlineDTO; | ||||
| import com.ruoyi.common.core.page.TableDataInfo; | ||||
| import com.ruoyi.common.enums.BusinessType; | ||||
| import com.ruoyi.common.utils.StreamUtils; | ||||
| import com.ruoyi.common.utils.StringUtils; | ||||
| import com.ruoyi.common.utils.redis.RedisUtils; | ||||
| import com.ruoyi.system.domain.SysUserOnline; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.Collections; | ||||
| import java.util.List; | ||||
| import java.util.stream.Collectors; | ||||
|  | ||||
| /** | ||||
|  * 在线用户监控 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Api(value = "在线用户监控", tags = {"在线用户监控管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/monitor/online") | ||||
| public class SysUserOnlineController extends BaseController { | ||||
|  | ||||
|     @ApiOperation("在线用户列表") | ||||
|     /** | ||||
|      * 获取在线用户监控列表 | ||||
|      * | ||||
|      * @param ipaddr   IP地址 | ||||
|      * @param userName 用户名 | ||||
|      */ | ||||
|     @SaCheckPermission("monitor:online:list") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo<SysUserOnline> list(String ipaddr, String userName) { | ||||
| @@ -43,26 +45,26 @@ public class SysUserOnlineController extends BaseController { | ||||
|         List<String> keys = StpUtil.searchTokenValue("", -1, 0); | ||||
|         List<UserOnlineDTO> userOnlineDTOList = new ArrayList<>(); | ||||
|         for (String key : keys) { | ||||
|             String token = key.replace(Constants.LOGIN_TOKEN_KEY, ""); | ||||
|             // 如果已经过期则踢下线 | ||||
|             String token = key.replace(CacheConstants.LOGIN_TOKEN_KEY, ""); | ||||
|             // 如果已经过期则跳过 | ||||
|             if (StpUtil.stpLogic.getTokenActivityTimeoutByToken(token) < 0) { | ||||
|                 continue; | ||||
|             } | ||||
|             userOnlineDTOList.add(RedisUtils.getCacheObject(Constants.ONLINE_TOKEN_KEY + token)); | ||||
|             userOnlineDTOList.add(RedisUtils.getCacheObject(CacheConstants.ONLINE_TOKEN_KEY + token)); | ||||
|         } | ||||
|         if (StringUtils.isNotEmpty(ipaddr) && StringUtils.isNotEmpty(userName)) { | ||||
|             userOnlineDTOList = userOnlineDTOList.stream().filter(userOnline -> | ||||
|             userOnlineDTOList = StreamUtils.filter(userOnlineDTOList, userOnline -> | ||||
|                 StringUtils.equals(ipaddr, userOnline.getIpaddr()) && | ||||
|                     StringUtils.equals(userName, userOnline.getUserName()) | ||||
|             ).collect(Collectors.toList()); | ||||
|             ); | ||||
|         } else if (StringUtils.isNotEmpty(ipaddr)) { | ||||
|             userOnlineDTOList = userOnlineDTOList.stream().filter(userOnline -> | ||||
|                     StringUtils.equals(ipaddr, userOnline.getIpaddr())) | ||||
|                 .collect(Collectors.toList()); | ||||
|             userOnlineDTOList = StreamUtils.filter(userOnlineDTOList, userOnline -> | ||||
|                 StringUtils.equals(ipaddr, userOnline.getIpaddr()) | ||||
|             ); | ||||
|         } else if (StringUtils.isNotEmpty(userName)) { | ||||
|             userOnlineDTOList = userOnlineDTOList.stream().filter(userOnline -> | ||||
|             userOnlineDTOList = StreamUtils.filter(userOnlineDTOList, userOnline -> | ||||
|                 StringUtils.equals(userName, userOnline.getUserName()) | ||||
|             ).collect(Collectors.toList()); | ||||
|             ); | ||||
|         } | ||||
|         Collections.reverse(userOnlineDTOList); | ||||
|         userOnlineDTOList.removeAll(Collections.singleton(null)); | ||||
| @@ -72,8 +74,9 @@ public class SysUserOnlineController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 强退用户 | ||||
|      * | ||||
|      * @param tokenId token值 | ||||
|      */ | ||||
|     @ApiOperation("强退用户") | ||||
|     @SaCheckPermission("monitor:online:forceLogout") | ||||
|     @Log(title = "在线用户", businessType = BusinessType.FORCE) | ||||
|     @DeleteMapping("/{tokenId}") | ||||
|   | ||||
| @@ -4,16 +4,13 @@ import cn.dev33.satoken.annotation.SaCheckPermission; | ||||
| 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.R; | ||||
| import com.ruoyi.common.core.domain.PageQuery; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.page.TableDataInfo; | ||||
| import com.ruoyi.common.enums.BusinessType; | ||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | ||||
| import com.ruoyi.system.domain.SysConfig; | ||||
| import com.ruoyi.system.service.ISysConfigService; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import io.swagger.annotations.ApiParam; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
| @@ -27,7 +24,6 @@ import java.util.List; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "参数配置控制器", tags = {"参数配置管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/system/config") | ||||
| @@ -38,14 +34,15 @@ public class SysConfigController extends BaseController { | ||||
|     /** | ||||
|      * 获取参数配置列表 | ||||
|      */ | ||||
|     @ApiOperation("获取参数配置列表") | ||||
|     @SaCheckPermission("system:config:list") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo<SysConfig> list(SysConfig config, PageQuery pageQuery) { | ||||
|         return configService.selectPageConfigList(config, pageQuery); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("导出参数配置列表") | ||||
|     /** | ||||
|      * 导出参数配置列表 | ||||
|      */ | ||||
|     @Log(title = "参数管理", businessType = BusinessType.EXPORT) | ||||
|     @SaCheckPermission("system:config:export") | ||||
|     @PostMapping("/export") | ||||
| @@ -56,27 +53,28 @@ public class SysConfigController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 根据参数编号获取详细信息 | ||||
|      * | ||||
|      * @param configId 参数ID | ||||
|      */ | ||||
|     @ApiOperation("根据参数编号获取详细信息") | ||||
|     @SaCheckPermission("system:config:query") | ||||
|     @GetMapping(value = "/{configId}") | ||||
|     public R<SysConfig> getInfo(@ApiParam("参数ID") @PathVariable Long configId) { | ||||
|     public R<SysConfig> getInfo(@PathVariable Long configId) { | ||||
|         return R.ok(configService.selectConfigById(configId)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据参数键名查询参数值 | ||||
|      * | ||||
|      * @param configKey 参数Key | ||||
|      */ | ||||
|     @ApiOperation("根据参数键名查询参数值") | ||||
|     @GetMapping(value = "/configKey/{configKey}") | ||||
|     public R<Void> getConfigKey(@ApiParam("参数Key") @PathVariable String configKey) { | ||||
|     public R<Void> getConfigKey(@PathVariable String configKey) { | ||||
|         return R.ok(configService.selectConfigByKey(configKey)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 新增参数配置 | ||||
|      */ | ||||
|     @ApiOperation("新增参数配置") | ||||
|     @SaCheckPermission("system:config:add") | ||||
|     @Log(title = "参数管理", businessType = BusinessType.INSERT) | ||||
|     @PostMapping | ||||
| @@ -84,13 +82,13 @@ public class SysConfigController extends BaseController { | ||||
|         if (UserConstants.NOT_UNIQUE.equals(configService.checkConfigKeyUnique(config))) { | ||||
|             return R.fail("新增参数'" + config.getConfigName() + "'失败,参数键名已存在"); | ||||
|         } | ||||
|         return toAjax(configService.insertConfig(config)); | ||||
|         configService.insertConfig(config); | ||||
|         return R.ok(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 修改参数配置 | ||||
|      */ | ||||
|     @ApiOperation("修改参数配置") | ||||
|     @SaCheckPermission("system:config:edit") | ||||
|     @Log(title = "参数管理", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping | ||||
| @@ -98,28 +96,30 @@ public class SysConfigController extends BaseController { | ||||
|         if (UserConstants.NOT_UNIQUE.equals(configService.checkConfigKeyUnique(config))) { | ||||
|             return R.fail("修改参数'" + config.getConfigName() + "'失败,参数键名已存在"); | ||||
|         } | ||||
|         return toAjax(configService.updateConfig(config)); | ||||
|         configService.updateConfig(config); | ||||
|         return R.ok(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据参数键名修改参数配置 | ||||
|      */ | ||||
|     @ApiOperation("根据参数键名修改参数配置") | ||||
|     @SaCheckPermission("system:config:edit") | ||||
|     @Log(title = "参数管理", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping("/updateByKey") | ||||
|     public R<Void> updateByKey(@RequestBody SysConfig config) { | ||||
|         return toAjax(configService.updateConfig(config)); | ||||
|         configService.updateConfig(config); | ||||
|         return R.ok(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 删除参数配置 | ||||
|      * | ||||
|      * @param configIds 参数ID串 | ||||
|      */ | ||||
|     @ApiOperation("删除参数配置") | ||||
|     @SaCheckPermission("system:config:remove") | ||||
|     @Log(title = "参数管理", businessType = BusinessType.DELETE) | ||||
|     @DeleteMapping("/{configIds}") | ||||
|     public R<Void> remove(@ApiParam("参数ID串") @PathVariable Long[] configIds) { | ||||
|     public R<Void> remove(@PathVariable Long[] configIds) { | ||||
|         configService.deleteConfigByIds(configIds); | ||||
|         return R.ok(); | ||||
|     } | ||||
| @@ -127,7 +127,6 @@ public class SysConfigController extends BaseController { | ||||
|     /** | ||||
|      * 刷新参数缓存 | ||||
|      */ | ||||
|     @ApiOperation("刷新参数缓存") | ||||
|     @SaCheckPermission("system:config:remove") | ||||
|     @Log(title = "参数管理", businessType = BusinessType.CLEAN) | ||||
|     @DeleteMapping("/refreshCache") | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| package com.ruoyi.web.controller.system; | ||||
|  | ||||
| import cn.dev33.satoken.annotation.SaCheckPermission; | ||||
| import cn.hutool.core.lang.tree.Tree; | ||||
| import cn.hutool.core.util.ArrayUtil; | ||||
| import com.ruoyi.common.annotation.Log; | ||||
| import com.ruoyi.common.constant.UserConstants; | ||||
| @@ -11,16 +10,11 @@ import com.ruoyi.common.core.domain.entity.SysDept; | ||||
| import com.ruoyi.common.enums.BusinessType; | ||||
| import com.ruoyi.common.utils.StringUtils; | ||||
| import com.ruoyi.system.service.ISysDeptService; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import io.swagger.annotations.ApiParam; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
|  | ||||
| import java.util.HashMap; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * 部门信息 | ||||
| @@ -28,7 +22,6 @@ import java.util.Map; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "部门控制器", tags = {"部门管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/system/dept") | ||||
| @@ -39,7 +32,6 @@ public class SysDeptController extends BaseController { | ||||
|     /** | ||||
|      * 获取部门列表 | ||||
|      */ | ||||
|     @ApiOperation("获取部门列表") | ||||
|     @SaCheckPermission("system:dept:list") | ||||
|     @GetMapping("/list") | ||||
|     public R<List<SysDept>> list(SysDept dept) { | ||||
| @@ -49,11 +41,12 @@ public class SysDeptController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 查询部门列表(排除节点) | ||||
|      * | ||||
|      * @param deptId 部门ID | ||||
|      */ | ||||
|     @ApiOperation("查询部门列表(排除节点)") | ||||
|     @SaCheckPermission("system:dept:list") | ||||
|     @GetMapping("/list/exclude/{deptId}") | ||||
|     public R<List<SysDept>> excludeChild(@ApiParam("部门ID") @PathVariable(value = "deptId", required = false) Long deptId) { | ||||
|     public R<List<SysDept>> excludeChild(@PathVariable(value = "deptId", required = false) Long deptId) { | ||||
|         List<SysDept> depts = deptService.selectDeptList(new SysDept()); | ||||
|         depts.removeIf(d -> d.getDeptId().equals(deptId) | ||||
|             || ArrayUtil.contains(StringUtils.split(d.getAncestors(), ","), deptId + "")); | ||||
| @@ -62,42 +55,19 @@ public class SysDeptController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 根据部门编号获取详细信息 | ||||
|      * | ||||
|      * @param deptId 部门ID | ||||
|      */ | ||||
|     @ApiOperation("根据部门编号获取详细信息") | ||||
|     @SaCheckPermission("system:dept:query") | ||||
|     @GetMapping(value = "/{deptId}") | ||||
|     public R<SysDept> getInfo(@ApiParam("部门ID") @PathVariable Long deptId) { | ||||
|     public R<SysDept> getInfo(@PathVariable Long deptId) { | ||||
|         deptService.checkDeptDataScope(deptId); | ||||
|         return R.ok(deptService.selectDeptById(deptId)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取部门下拉树列表 | ||||
|      */ | ||||
|     @ApiOperation("获取部门下拉树列表") | ||||
|     @GetMapping("/treeselect") | ||||
|     public R<List<Tree<Long>>> treeselect(SysDept dept) { | ||||
|         List<SysDept> depts = deptService.selectDeptList(dept); | ||||
|         return R.ok(deptService.buildDeptTreeSelect(depts)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 加载对应角色部门列表树 | ||||
|      */ | ||||
|     @ApiOperation("加载对应角色部门列表树") | ||||
|     @GetMapping(value = "/roleDeptTreeselect/{roleId}") | ||||
|     public R<Map<String, Object>> roleDeptTreeselect(@ApiParam("角色ID") @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 R.ok(ajax); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 新增部门 | ||||
|      */ | ||||
|     @ApiOperation("新增部门") | ||||
|     @SaCheckPermission("system:dept:add") | ||||
|     @Log(title = "部门管理", businessType = BusinessType.INSERT) | ||||
|     @PostMapping | ||||
| @@ -111,7 +81,6 @@ public class SysDeptController extends BaseController { | ||||
|     /** | ||||
|      * 修改部门 | ||||
|      */ | ||||
|     @ApiOperation("修改部门") | ||||
|     @SaCheckPermission("system:dept:edit") | ||||
|     @Log(title = "部门管理", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping | ||||
| @@ -131,12 +100,13 @@ public class SysDeptController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 删除部门 | ||||
|      * | ||||
|      * @param deptId 部门ID | ||||
|      */ | ||||
|     @ApiOperation("删除部门") | ||||
|     @SaCheckPermission("system:dept:remove") | ||||
|     @Log(title = "部门管理", businessType = BusinessType.DELETE) | ||||
|     @DeleteMapping("/{deptId}") | ||||
|     public R<Void> remove(@ApiParam("部门ID串") @PathVariable Long deptId) { | ||||
|     public R<Void> remove(@PathVariable Long deptId) { | ||||
|         if (deptService.hasChildByDeptId(deptId)) { | ||||
|             return R.fail("存在下级部门,不允许删除"); | ||||
|         } | ||||
|   | ||||
| @@ -4,17 +4,14 @@ import cn.dev33.satoken.annotation.SaCheckPermission; | ||||
| import cn.hutool.core.util.ObjectUtil; | ||||
| import com.ruoyi.common.annotation.Log; | ||||
| import com.ruoyi.common.core.controller.BaseController; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.domain.PageQuery; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| 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.poi.ExcelUtil; | ||||
| import com.ruoyi.system.service.ISysDictDataService; | ||||
| import com.ruoyi.system.service.ISysDictTypeService; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import io.swagger.annotations.ApiParam; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
| @@ -29,7 +26,6 @@ import java.util.List; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "数据字典信息控制器", tags = {"数据字典信息管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/system/dict/data") | ||||
| @@ -38,14 +34,18 @@ public class SysDictDataController extends BaseController { | ||||
|     private final ISysDictDataService dictDataService; | ||||
|     private final ISysDictTypeService dictTypeService; | ||||
|  | ||||
|     @ApiOperation("查询字典数据列表") | ||||
|     /** | ||||
|      * 查询字典数据列表 | ||||
|      */ | ||||
|     @SaCheckPermission("system:dict:list") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo<SysDictData> list(SysDictData dictData, PageQuery pageQuery) { | ||||
|         return dictDataService.selectPageDictDataList(dictData, pageQuery); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("导出字典数据列表") | ||||
|     /** | ||||
|      * 导出字典数据列表 | ||||
|      */ | ||||
|     @Log(title = "字典数据", businessType = BusinessType.EXPORT) | ||||
|     @SaCheckPermission("system:dict:export") | ||||
|     @PostMapping("/export") | ||||
| @@ -56,20 +56,22 @@ public class SysDictDataController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 查询字典数据详细 | ||||
|      * | ||||
|      * @param dictCode 字典code | ||||
|      */ | ||||
|     @ApiOperation("查询字典数据详细") | ||||
|     @SaCheckPermission("system:dict:query") | ||||
|     @GetMapping(value = "/{dictCode}") | ||||
|     public R<SysDictData> getInfo(@ApiParam("字典code") @PathVariable Long dictCode) { | ||||
|     public R<SysDictData> getInfo(@PathVariable Long dictCode) { | ||||
|         return R.ok(dictDataService.selectDictDataById(dictCode)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据字典类型查询字典数据信息 | ||||
|      * | ||||
|      * @param dictType 字典类型 | ||||
|      */ | ||||
|     @ApiOperation("根据字典类型查询字典数据信息") | ||||
|     @GetMapping(value = "/type/{dictType}") | ||||
|     public R<List<SysDictData>> dictType(@ApiParam("字典类型") @PathVariable String dictType) { | ||||
|     public R<List<SysDictData>> dictType(@PathVariable String dictType) { | ||||
|         List<SysDictData> data = dictTypeService.selectDictDataByType(dictType); | ||||
|         if (ObjectUtil.isNull(data)) { | ||||
|             data = new ArrayList<>(); | ||||
| @@ -80,33 +82,34 @@ public class SysDictDataController extends BaseController { | ||||
|     /** | ||||
|      * 新增字典类型 | ||||
|      */ | ||||
|     @ApiOperation("新增字典类型") | ||||
|     @SaCheckPermission("system:dict:add") | ||||
|     @Log(title = "字典数据", businessType = BusinessType.INSERT) | ||||
|     @PostMapping | ||||
|     public R<Void> add(@Validated @RequestBody SysDictData dict) { | ||||
|         return toAjax(dictDataService.insertDictData(dict)); | ||||
|         dictDataService.insertDictData(dict); | ||||
|         return R.ok(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 修改保存字典类型 | ||||
|      */ | ||||
|     @ApiOperation("修改保存字典类型") | ||||
|     @SaCheckPermission("system:dict:edit") | ||||
|     @Log(title = "字典数据", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping | ||||
|     public R<Void> edit(@Validated @RequestBody SysDictData dict) { | ||||
|         return toAjax(dictDataService.updateDictData(dict)); | ||||
|         dictDataService.updateDictData(dict); | ||||
|         return R.ok(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 删除字典类型 | ||||
|      * | ||||
|      * @param dictCodes 字典code串 | ||||
|      */ | ||||
|     @ApiOperation("删除字典类型") | ||||
|     @SaCheckPermission("system:dict:remove") | ||||
|     @Log(title = "字典类型", businessType = BusinessType.DELETE) | ||||
|     @DeleteMapping("/{dictCodes}") | ||||
|     public R<Void> remove(@ApiParam("字典code串") @PathVariable Long[] dictCodes) { | ||||
|     public R<Void> remove(@PathVariable Long[] dictCodes) { | ||||
|         dictDataService.deleteDictDataByIds(dictCodes); | ||||
|         return R.ok(); | ||||
|     } | ||||
|   | ||||
| @@ -4,16 +4,13 @@ import cn.dev33.satoken.annotation.SaCheckPermission; | ||||
| 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.R; | ||||
| import com.ruoyi.common.core.domain.PageQuery; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.domain.entity.SysDictType; | ||||
| import com.ruoyi.common.core.page.TableDataInfo; | ||||
| import com.ruoyi.common.enums.BusinessType; | ||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | ||||
| import com.ruoyi.system.service.ISysDictTypeService; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import io.swagger.annotations.ApiParam; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
| @@ -27,7 +24,6 @@ import java.util.List; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "数据字典信息控制器", tags = {"数据字典信息管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/system/dict/type") | ||||
| @@ -35,14 +31,18 @@ public class SysDictTypeController extends BaseController { | ||||
|  | ||||
|     private final ISysDictTypeService dictTypeService; | ||||
|  | ||||
|     @ApiOperation("查询字典类型列表") | ||||
|     /** | ||||
|      * 查询字典类型列表 | ||||
|      */ | ||||
|     @SaCheckPermission("system:dict:list") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo<SysDictType> list(SysDictType dictType, PageQuery pageQuery) { | ||||
|         return dictTypeService.selectPageDictTypeList(dictType, pageQuery); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("导出字典类型列表") | ||||
|     /** | ||||
|      * 导出字典类型列表 | ||||
|      */ | ||||
|     @Log(title = "字典类型", businessType = BusinessType.EXPORT) | ||||
|     @SaCheckPermission("system:dict:export") | ||||
|     @PostMapping("/export") | ||||
| @@ -53,18 +53,18 @@ public class SysDictTypeController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 查询字典类型详细 | ||||
|      * | ||||
|      * @param dictId 字典ID | ||||
|      */ | ||||
|     @ApiOperation("查询字典类型详细") | ||||
|     @SaCheckPermission("system:dict:query") | ||||
|     @GetMapping(value = "/{dictId}") | ||||
|     public R<SysDictType> getInfo(@ApiParam("字典ID") @PathVariable Long dictId) { | ||||
|     public R<SysDictType> getInfo(@PathVariable Long dictId) { | ||||
|         return R.ok(dictTypeService.selectDictTypeById(dictId)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 新增字典类型 | ||||
|      */ | ||||
|     @ApiOperation("新增字典类型") | ||||
|     @SaCheckPermission("system:dict:add") | ||||
|     @Log(title = "字典类型", businessType = BusinessType.INSERT) | ||||
|     @PostMapping | ||||
| @@ -72,13 +72,13 @@ public class SysDictTypeController extends BaseController { | ||||
|         if (UserConstants.NOT_UNIQUE.equals(dictTypeService.checkDictTypeUnique(dict))) { | ||||
|             return R.fail("新增字典'" + dict.getDictName() + "'失败,字典类型已存在"); | ||||
|         } | ||||
|         return toAjax(dictTypeService.insertDictType(dict)); | ||||
|         dictTypeService.insertDictType(dict); | ||||
|         return R.ok(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 修改字典类型 | ||||
|      */ | ||||
|     @ApiOperation("修改字典类型") | ||||
|     @SaCheckPermission("system:dict:edit") | ||||
|     @Log(title = "字典类型", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping | ||||
| @@ -86,17 +86,19 @@ public class SysDictTypeController extends BaseController { | ||||
|         if (UserConstants.NOT_UNIQUE.equals(dictTypeService.checkDictTypeUnique(dict))) { | ||||
|             return R.fail("修改字典'" + dict.getDictName() + "'失败,字典类型已存在"); | ||||
|         } | ||||
|         return toAjax(dictTypeService.updateDictType(dict)); | ||||
|         dictTypeService.updateDictType(dict); | ||||
|         return R.ok(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 删除字典类型 | ||||
|      * | ||||
|      * @param dictIds 字典ID串 | ||||
|      */ | ||||
|     @ApiOperation("删除字典类型") | ||||
|     @SaCheckPermission("system:dict:remove") | ||||
|     @Log(title = "字典类型", businessType = BusinessType.DELETE) | ||||
|     @DeleteMapping("/{dictIds}") | ||||
|     public R<Void> remove(@ApiParam("字典ID串") @PathVariable Long[] dictIds) { | ||||
|     public R<Void> remove(@PathVariable Long[] dictIds) { | ||||
|         dictTypeService.deleteDictTypeByIds(dictIds); | ||||
|         return R.ok(); | ||||
|     } | ||||
| @@ -104,7 +106,6 @@ public class SysDictTypeController extends BaseController { | ||||
|     /** | ||||
|      * 刷新字典缓存 | ||||
|      */ | ||||
|     @ApiOperation("刷新字典缓存") | ||||
|     @SaCheckPermission("system:dict:remove") | ||||
|     @Log(title = "字典类型", businessType = BusinessType.CLEAN) | ||||
|     @DeleteMapping("/refreshCache") | ||||
| @@ -116,7 +117,6 @@ public class SysDictTypeController extends BaseController { | ||||
|     /** | ||||
|      * 获取字典选择框列表 | ||||
|      */ | ||||
|     @ApiOperation("获取字典选择框列表") | ||||
|     @GetMapping("/optionselect") | ||||
|     public R<List<SysDictType>> optionselect() { | ||||
|         List<SysDictType> dictTypes = dictTypeService.selectDictTypeAll(); | ||||
|   | ||||
| @@ -1,9 +1,8 @@ | ||||
| package com.ruoyi.web.controller.system; | ||||
|  | ||||
| import com.ruoyi.common.annotation.Anonymous; | ||||
| import com.ruoyi.common.config.RuoYiConfig; | ||||
| import com.ruoyi.common.utils.StringUtils; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.web.bind.annotation.GetMapping; | ||||
| import org.springframework.web.bind.annotation.RestController; | ||||
| @@ -13,7 +12,6 @@ import org.springframework.web.bind.annotation.RestController; | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Api(value = "首页控制器", tags = {"首页管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| public class SysIndexController { | ||||
| @@ -26,7 +24,7 @@ public class SysIndexController { | ||||
|     /** | ||||
|      * 访问首页,提示语 | ||||
|      */ | ||||
|     @ApiOperation("访问首页,提示语") | ||||
|     @Anonymous | ||||
|     @GetMapping("/") | ||||
|     public String index() { | ||||
|         return StringUtils.format("欢迎使用{}后台管理框架,当前版本:v{},请通过前端地址访问。", ruoyiConfig.getName(), ruoyiConfig.getVersion()); | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| package com.ruoyi.web.controller.system; | ||||
|  | ||||
| import cn.dev33.satoken.exception.NotLoginException; | ||||
| import cn.dev33.satoken.stp.StpUtil; | ||||
| import com.ruoyi.common.annotation.Anonymous; | ||||
| import com.ruoyi.common.constant.Constants; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.domain.entity.SysMenu; | ||||
| @@ -14,8 +13,6 @@ import com.ruoyi.system.service.ISysMenuService; | ||||
| import com.ruoyi.system.service.ISysUserService; | ||||
| import com.ruoyi.system.service.SysLoginService; | ||||
| import com.ruoyi.system.service.SysPermissionService; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.GetMapping; | ||||
| @@ -35,7 +32,6 @@ import java.util.Set; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "登录验证控制器", tags = {"登录验证管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| public class SysLoginController { | ||||
| @@ -51,7 +47,7 @@ public class SysLoginController { | ||||
|      * @param loginBody 登录信息 | ||||
|      * @return 结果 | ||||
|      */ | ||||
|     @ApiOperation("登录方法") | ||||
|     @Anonymous | ||||
|     @PostMapping("/login") | ||||
|     public R<Map<String, Object>> login(@Validated @RequestBody LoginBody loginBody) { | ||||
|         Map<String, Object> ajax = new HashMap<>(); | ||||
| @@ -68,7 +64,7 @@ public class SysLoginController { | ||||
|      * @param smsLoginBody 登录信息 | ||||
|      * @return 结果 | ||||
|      */ | ||||
|     @ApiOperation("短信登录(示例)") | ||||
|     @Anonymous | ||||
|     @PostMapping("/smsLogin") | ||||
|     public R<Map<String, Object>> smsLogin(@Validated @RequestBody SmsLoginBody smsLoginBody) { | ||||
|         Map<String, Object> ajax = new HashMap<>(); | ||||
| @@ -84,7 +80,7 @@ public class SysLoginController { | ||||
|      * @param xcxCode 小程序code | ||||
|      * @return 结果 | ||||
|      */ | ||||
|     @ApiOperation("小程序登录(示例)") | ||||
|     @Anonymous | ||||
|     @PostMapping("/xcxLogin") | ||||
|     public R<Map<String, Object>> xcxLogin(@NotBlank(message = "{xcx.code.not.blank}") String xcxCode) { | ||||
|         Map<String, Object> ajax = new HashMap<>(); | ||||
| @@ -94,14 +90,13 @@ public class SysLoginController { | ||||
|         return R.ok(ajax); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("登出方法") | ||||
|     /** | ||||
|      * 退出登录 | ||||
|      */ | ||||
|     @Anonymous | ||||
|     @PostMapping("/logout") | ||||
|     public R<Void> logout() { | ||||
|         try { | ||||
|             StpUtil.logout(); | ||||
|             loginService.logout(LoginHelper.getUsername()); | ||||
|         } catch (NotLoginException e) { | ||||
|         } | ||||
|         loginService.logout(); | ||||
|         return R.ok("退出成功"); | ||||
|     } | ||||
|  | ||||
| @@ -110,7 +105,6 @@ public class SysLoginController { | ||||
|      * | ||||
|      * @return 用户信息 | ||||
|      */ | ||||
|     @ApiOperation("获取用户信息") | ||||
|     @GetMapping("getInfo") | ||||
|     public R<Map<String, Object>> getInfo() { | ||||
|         SysUser user = userService.selectUserById(LoginHelper.getUserId()); | ||||
| @@ -130,7 +124,6 @@ public class SysLoginController { | ||||
|      * | ||||
|      * @return 路由信息 | ||||
|      */ | ||||
|     @ApiOperation("获取路由信息") | ||||
|     @GetMapping("getRouters") | ||||
|     public R<List<RouterVo>> getRouters() { | ||||
|         Long userId = LoginHelper.getUserId(); | ||||
|   | ||||
| @@ -10,9 +10,6 @@ import com.ruoyi.common.core.domain.entity.SysMenu; | ||||
| import com.ruoyi.common.enums.BusinessType; | ||||
| import com.ruoyi.common.utils.StringUtils; | ||||
| import com.ruoyi.system.service.ISysMenuService; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import io.swagger.annotations.ApiParam; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
| @@ -27,7 +24,6 @@ import java.util.Map; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "菜单信息控制器", tags = {"菜单信息管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/system/menu") | ||||
| @@ -38,7 +34,6 @@ public class SysMenuController extends BaseController { | ||||
|     /** | ||||
|      * 获取菜单列表 | ||||
|      */ | ||||
|     @ApiOperation("获取菜单列表") | ||||
|     @SaCheckPermission("system:menu:list") | ||||
|     @GetMapping("/list") | ||||
|     public R<List<SysMenu>> list(SysMenu menu) { | ||||
| @@ -48,18 +43,18 @@ public class SysMenuController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 根据菜单编号获取详细信息 | ||||
|      * | ||||
|      * @param menuId 菜单ID | ||||
|      */ | ||||
|     @ApiOperation("根据菜单编号获取详细信息") | ||||
|     @SaCheckPermission("system:menu:query") | ||||
|     @GetMapping(value = "/{menuId}") | ||||
|     public R<SysMenu> getInfo(@ApiParam("菜单ID") @PathVariable Long menuId) { | ||||
|     public R<SysMenu> getInfo(@PathVariable Long menuId) { | ||||
|         return R.ok(menuService.selectMenuById(menuId)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取菜单下拉树列表 | ||||
|      */ | ||||
|     @ApiOperation("获取菜单下拉树列表") | ||||
|     @GetMapping("/treeselect") | ||||
|     public R<List<Tree<Long>>> treeselect(SysMenu menu) { | ||||
|         List<SysMenu> menus = menuService.selectMenuList(menu, getUserId()); | ||||
| @@ -68,10 +63,11 @@ public class SysMenuController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 加载对应角色菜单列表树 | ||||
|      * | ||||
|      * @param roleId 角色ID | ||||
|      */ | ||||
|     @ApiOperation("加载对应角色菜单列表树") | ||||
|     @GetMapping(value = "/roleMenuTreeselect/{roleId}") | ||||
|     public R<Map<String, Object>> roleMenuTreeselect(@ApiParam("角色ID") @PathVariable("roleId") Long roleId) { | ||||
|     public R<Map<String, Object>> roleMenuTreeselect(@PathVariable("roleId") Long roleId) { | ||||
|         List<SysMenu> menus = menuService.selectMenuList(getUserId()); | ||||
|         Map<String, Object> ajax = new HashMap<>(); | ||||
|         ajax.put("checkedKeys", menuService.selectMenuListByRoleId(roleId)); | ||||
| @@ -82,7 +78,6 @@ public class SysMenuController extends BaseController { | ||||
|     /** | ||||
|      * 新增菜单 | ||||
|      */ | ||||
|     @ApiOperation("新增菜单") | ||||
|     @SaCheckPermission("system:menu:add") | ||||
|     @Log(title = "菜单管理", businessType = BusinessType.INSERT) | ||||
|     @PostMapping | ||||
| @@ -98,7 +93,6 @@ public class SysMenuController extends BaseController { | ||||
|     /** | ||||
|      * 修改菜单 | ||||
|      */ | ||||
|     @ApiOperation("修改菜单") | ||||
|     @SaCheckPermission("system:menu:edit") | ||||
|     @Log(title = "菜单管理", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping | ||||
| @@ -115,12 +109,13 @@ public class SysMenuController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 删除菜单 | ||||
|      * | ||||
|      * @param menuId 菜单ID | ||||
|      */ | ||||
|     @ApiOperation("删除菜单") | ||||
|     @SaCheckPermission("system:menu:remove") | ||||
|     @Log(title = "菜单管理", businessType = BusinessType.DELETE) | ||||
|     @DeleteMapping("/{menuId}") | ||||
|     public R<Void> remove(@ApiParam("菜单ID") @PathVariable("menuId") Long menuId) { | ||||
|     public R<Void> remove(@PathVariable("menuId") Long menuId) { | ||||
|         if (menuService.hasChildByMenuId(menuId)) { | ||||
|             return R.fail("存在子菜单,不允许删除"); | ||||
|         } | ||||
|   | ||||
| @@ -3,15 +3,12 @@ package com.ruoyi.web.controller.system; | ||||
| import cn.dev33.satoken.annotation.SaCheckPermission; | ||||
| import com.ruoyi.common.annotation.Log; | ||||
| import com.ruoyi.common.core.controller.BaseController; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.domain.PageQuery; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.page.TableDataInfo; | ||||
| import com.ruoyi.common.enums.BusinessType; | ||||
| import com.ruoyi.system.domain.SysNotice; | ||||
| import com.ruoyi.system.service.ISysNoticeService; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import io.swagger.annotations.ApiParam; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
| @@ -22,7 +19,6 @@ import org.springframework.web.bind.annotation.*; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "公告信息控制器", tags = {"公告信息管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/system/notice") | ||||
| @@ -33,7 +29,6 @@ public class SysNoticeController extends BaseController { | ||||
|     /** | ||||
|      * 获取通知公告列表 | ||||
|      */ | ||||
|     @ApiOperation("获取通知公告列表") | ||||
|     @SaCheckPermission("system:notice:list") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo<SysNotice> list(SysNotice notice, PageQuery pageQuery) { | ||||
| @@ -42,18 +37,18 @@ public class SysNoticeController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 根据通知公告编号获取详细信息 | ||||
|      * | ||||
|      * @param noticeId 公告ID | ||||
|      */ | ||||
|     @ApiOperation("根据通知公告编号获取详细信息") | ||||
|     @SaCheckPermission("system:notice:query") | ||||
|     @GetMapping(value = "/{noticeId}") | ||||
|     public R<SysNotice> getInfo(@ApiParam("公告ID") @PathVariable Long noticeId) { | ||||
|     public R<SysNotice> getInfo(@PathVariable Long noticeId) { | ||||
|         return R.ok(noticeService.selectNoticeById(noticeId)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 新增通知公告 | ||||
|      */ | ||||
|     @ApiOperation("新增通知公告") | ||||
|     @SaCheckPermission("system:notice:add") | ||||
|     @Log(title = "通知公告", businessType = BusinessType.INSERT) | ||||
|     @PostMapping | ||||
| @@ -64,7 +59,6 @@ public class SysNoticeController extends BaseController { | ||||
|     /** | ||||
|      * 修改通知公告 | ||||
|      */ | ||||
|     @ApiOperation("修改通知公告") | ||||
|     @SaCheckPermission("system:notice:edit") | ||||
|     @Log(title = "通知公告", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping | ||||
| @@ -74,12 +68,13 @@ public class SysNoticeController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 删除通知公告 | ||||
|      * | ||||
|      * @param noticeIds 公告ID串 | ||||
|      */ | ||||
|     @ApiOperation("删除通知公告") | ||||
|     @SaCheckPermission("system:notice:remove") | ||||
|     @Log(title = "通知公告", businessType = BusinessType.DELETE) | ||||
|     @DeleteMapping("/{noticeIds}") | ||||
|     public R<Void> remove(@ApiParam("公告ID串") @PathVariable Long[] noticeIds) { | ||||
|     public R<Void> remove(@PathVariable Long[] noticeIds) { | ||||
|         return toAjax(noticeService.deleteNoticeByIds(noticeIds)); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -4,8 +4,8 @@ import cn.dev33.satoken.annotation.SaCheckPermission; | ||||
| import com.ruoyi.common.annotation.Log; | ||||
| import com.ruoyi.common.annotation.RepeatSubmit; | ||||
| import com.ruoyi.common.core.controller.BaseController; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.domain.PageQuery; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.page.TableDataInfo; | ||||
| import com.ruoyi.common.core.validate.AddGroup; | ||||
| import com.ruoyi.common.core.validate.EditGroup; | ||||
| @@ -14,9 +14,6 @@ import com.ruoyi.common.enums.BusinessType; | ||||
| import com.ruoyi.system.domain.bo.SysOssConfigBo; | ||||
| import com.ruoyi.system.domain.vo.SysOssConfigVo; | ||||
| import com.ruoyi.system.service.ISysOssConfigService; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import io.swagger.annotations.ApiParam; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
| @@ -26,14 +23,13 @@ import javax.validation.constraints.NotNull; | ||||
| import java.util.Arrays; | ||||
|  | ||||
| /** | ||||
|  * 对象存储配置Controller | ||||
|  * 对象存储配置 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  * @author 孤舟烟雨 | ||||
|  * @date 2021-08-13 | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "对象存储配置控制器", tags = {"对象存储配置管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/system/oss/config") | ||||
| @@ -44,7 +40,6 @@ public class SysOssConfigController extends BaseController { | ||||
|     /** | ||||
|      * 查询对象存储配置列表 | ||||
|      */ | ||||
|     @ApiOperation("查询对象存储配置列表") | ||||
|     @SaCheckPermission("system:oss:list") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo<SysOssConfigVo> list(@Validated(QueryGroup.class) SysOssConfigBo bo, PageQuery pageQuery) { | ||||
| @@ -53,20 +48,19 @@ public class SysOssConfigController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 获取对象存储配置详细信息 | ||||
|      * | ||||
|      * @param ossConfigId OSS配置ID | ||||
|      */ | ||||
|     @ApiOperation("获取对象存储配置详细信息") | ||||
|     @SaCheckPermission("system:oss:query") | ||||
|     @GetMapping("/{ossConfigId}") | ||||
|     public R<SysOssConfigVo> getInfo(@ApiParam("OSS配置ID") | ||||
|                                               @NotNull(message = "主键不能为空") | ||||
|                                               @PathVariable("ossConfigId") Long ossConfigId) { | ||||
|     public R<SysOssConfigVo> getInfo(@NotNull(message = "主键不能为空") | ||||
|                                      @PathVariable Long ossConfigId) { | ||||
|         return R.ok(iSysOssConfigService.queryById(ossConfigId)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 新增对象存储配置 | ||||
|      */ | ||||
|     @ApiOperation("新增对象存储配置") | ||||
|     @SaCheckPermission("system:oss:add") | ||||
|     @Log(title = "对象存储配置", businessType = BusinessType.INSERT) | ||||
|     @RepeatSubmit() | ||||
| @@ -78,7 +72,6 @@ public class SysOssConfigController extends BaseController { | ||||
|     /** | ||||
|      * 修改对象存储配置 | ||||
|      */ | ||||
|     @ApiOperation("修改对象存储配置") | ||||
|     @SaCheckPermission("system:oss:edit") | ||||
|     @Log(title = "对象存储配置", businessType = BusinessType.UPDATE) | ||||
|     @RepeatSubmit() | ||||
| @@ -89,13 +82,13 @@ public class SysOssConfigController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 删除对象存储配置 | ||||
|      * | ||||
|      * @param ossConfigIds OSS配置ID串 | ||||
|      */ | ||||
|     @ApiOperation("删除对象存储配置") | ||||
|     @SaCheckPermission("system:oss:remove") | ||||
|     @Log(title = "对象存储配置", businessType = BusinessType.DELETE) | ||||
|     @DeleteMapping("/{ossConfigIds}") | ||||
|     public R<Void> remove(@ApiParam("OSS配置ID串") | ||||
|                                    @NotEmpty(message = "主键不能为空") | ||||
|     public R<Void> remove(@NotEmpty(message = "主键不能为空") | ||||
|                           @PathVariable Long[] ossConfigIds) { | ||||
|         return toAjax(iSysOssConfigService.deleteWithValidByIds(Arrays.asList(ossConfigIds), true) ? 1 : 0); | ||||
|     } | ||||
| @@ -103,7 +96,6 @@ public class SysOssConfigController extends BaseController { | ||||
|     /** | ||||
|      * 状态修改 | ||||
|      */ | ||||
|     @ApiOperation("状态修改") | ||||
|     @SaCheckPermission("system:oss:edit") | ||||
|     @Log(title = "对象存储状态修改", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping("/changeStatus") | ||||
|   | ||||
| @@ -19,7 +19,6 @@ import com.ruoyi.system.domain.SysOss; | ||||
| import com.ruoyi.system.domain.bo.SysOssBo; | ||||
| import com.ruoyi.system.domain.vo.SysOssVo; | ||||
| import com.ruoyi.system.service.ISysOssService; | ||||
| import io.swagger.annotations.*; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.http.MediaType; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| @@ -28,10 +27,10 @@ import org.springframework.web.multipart.MultipartFile; | ||||
|  | ||||
| import javax.servlet.http.HttpServletResponse; | ||||
| import javax.validation.constraints.NotEmpty; | ||||
| import java.io.File; | ||||
| import java.io.IOException; | ||||
| import java.util.Arrays; | ||||
| import java.util.HashMap; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
| @@ -40,7 +39,6 @@ import java.util.Map; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "对象存储控制器", tags = {"对象存储管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/system/oss") | ||||
| @@ -51,7 +49,6 @@ public class SysOssController extends BaseController { | ||||
|     /** | ||||
|      * 查询OSS对象存储列表 | ||||
|      */ | ||||
|     @ApiOperation("查询OSS对象存储列表") | ||||
|     @SaCheckPermission("system:oss:list") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo<SysOssVo> list(@Validated(QueryGroup.class) SysOssBo bo, PageQuery pageQuery) { | ||||
| @@ -59,15 +56,26 @@ public class SysOssController extends BaseController { | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 上传OSS对象存储 | ||||
|      * 查询OSS对象基于id串 | ||||
|      * | ||||
|      * @param ossIds OSS对象ID串 | ||||
|      */ | ||||
|     @SaCheckPermission("system:oss:list") | ||||
|     @GetMapping("/listByIds/{ossIds}") | ||||
|     public R<List<SysOssVo>> listByIds(@NotEmpty(message = "主键不能为空") | ||||
|                                        @PathVariable Long[] ossIds) { | ||||
|         List<SysOssVo> list = iSysOssService.listByIds(Arrays.asList(ossIds)); | ||||
|         return R.ok(list); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 上传OSS对象存储 | ||||
|      * | ||||
|      * @param file 文件 | ||||
|      */ | ||||
|     @ApiOperation("上传OSS对象存储") | ||||
|     @ApiImplicitParams({ | ||||
|         @ApiImplicitParam(name = "file", value = "文件", paramType = "query", dataTypeClass = File.class, required = true) | ||||
|     }) | ||||
|     @SaCheckPermission("system:oss:upload") | ||||
|     @Log(title = "OSS对象存储", businessType = BusinessType.INSERT) | ||||
|     @PostMapping("/upload") | ||||
|     @PostMapping(value = "/upload", consumes = MediaType.MULTIPART_FORM_DATA_VALUE) | ||||
|     public R<Map<String, String>> upload(@RequestPart("file") MultipartFile file) { | ||||
|         if (ObjectUtil.isNull(file)) { | ||||
|             throw new ServiceException("上传文件不能为空"); | ||||
| @@ -80,11 +88,15 @@ public class SysOssController extends BaseController { | ||||
|         return R.ok(map); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("下载OSS对象存储") | ||||
|     /** | ||||
|      * 下载OSS对象 | ||||
|      * | ||||
|      * @param ossId OSS对象ID | ||||
|      */ | ||||
|     @SaCheckPermission("system:oss:download") | ||||
|     @GetMapping("/download/{ossId}") | ||||
|     public void download(@ApiParam("OSS对象ID") @PathVariable Long ossId, HttpServletResponse response) throws IOException { | ||||
|         SysOss sysOss = iSysOssService.getById(ossId); | ||||
|     public void download(@PathVariable Long ossId, HttpServletResponse response) throws IOException { | ||||
|         SysOssVo sysOss = iSysOssService.getById(ossId); | ||||
|         if (ObjectUtil.isNull(sysOss)) { | ||||
|             throw new ServiceException("文件数据不存在!"); | ||||
|         } | ||||
| @@ -106,13 +118,13 @@ public class SysOssController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 删除OSS对象存储 | ||||
|      * | ||||
|      * @param ossIds OSS对象ID串 | ||||
|      */ | ||||
|     @ApiOperation("删除OSS对象存储") | ||||
|     @SaCheckPermission("system:oss:remove") | ||||
|     @Log(title = "OSS对象存储", businessType = BusinessType.DELETE) | ||||
|     @DeleteMapping("/{ossIds}") | ||||
|     public R<Void> remove(@ApiParam("OSS对象ID串") | ||||
|                                    @NotEmpty(message = "主键不能为空") | ||||
|     public R<Void> remove(@NotEmpty(message = "主键不能为空") | ||||
|                           @PathVariable Long[] ossIds) { | ||||
|         return toAjax(iSysOssService.deleteWithValidByIds(Arrays.asList(ossIds), true) ? 1 : 0); | ||||
|     } | ||||
|   | ||||
| @@ -4,16 +4,13 @@ import cn.dev33.satoken.annotation.SaCheckPermission; | ||||
| 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.R; | ||||
| import com.ruoyi.common.core.domain.PageQuery; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.page.TableDataInfo; | ||||
| import com.ruoyi.common.enums.BusinessType; | ||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | ||||
| import com.ruoyi.system.domain.SysPost; | ||||
| import com.ruoyi.system.service.ISysPostService; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import io.swagger.annotations.ApiParam; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
| @@ -27,7 +24,6 @@ import java.util.List; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "岗位信息控制器", tags = {"岗位信息管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/system/post") | ||||
| @@ -38,14 +34,15 @@ public class SysPostController extends BaseController { | ||||
|     /** | ||||
|      * 获取岗位列表 | ||||
|      */ | ||||
|     @ApiOperation("获取岗位列表") | ||||
|     @SaCheckPermission("system:post:list") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo<SysPost> list(SysPost post, PageQuery pageQuery) { | ||||
|         return postService.selectPagePostList(post, pageQuery); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("导出岗位列表") | ||||
|     /** | ||||
|      * 导出岗位列表 | ||||
|      */ | ||||
|     @Log(title = "岗位管理", businessType = BusinessType.EXPORT) | ||||
|     @SaCheckPermission("system:post:export") | ||||
|     @PostMapping("/export") | ||||
| @@ -56,18 +53,18 @@ public class SysPostController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 根据岗位编号获取详细信息 | ||||
|      * | ||||
|      * @param postId 岗位ID | ||||
|      */ | ||||
|     @ApiOperation("根据岗位编号获取详细信息") | ||||
|     @SaCheckPermission("system:post:query") | ||||
|     @GetMapping(value = "/{postId}") | ||||
|     public R<SysPost> getInfo(@ApiParam("岗位ID") @PathVariable Long postId) { | ||||
|     public R<SysPost> getInfo(@PathVariable Long postId) { | ||||
|         return R.ok(postService.selectPostById(postId)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 新增岗位 | ||||
|      */ | ||||
|     @ApiOperation("新增岗位") | ||||
|     @SaCheckPermission("system:post:add") | ||||
|     @Log(title = "岗位管理", businessType = BusinessType.INSERT) | ||||
|     @PostMapping | ||||
| @@ -83,7 +80,6 @@ public class SysPostController extends BaseController { | ||||
|     /** | ||||
|      * 修改岗位 | ||||
|      */ | ||||
|     @ApiOperation("修改岗位") | ||||
|     @SaCheckPermission("system:post:edit") | ||||
|     @Log(title = "岗位管理", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping | ||||
| @@ -98,19 +94,19 @@ public class SysPostController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 删除岗位 | ||||
|      * | ||||
|      * @param postIds 岗位ID串 | ||||
|      */ | ||||
|     @ApiOperation("删除岗位") | ||||
|     @SaCheckPermission("system:post:remove") | ||||
|     @Log(title = "岗位管理", businessType = BusinessType.DELETE) | ||||
|     @DeleteMapping("/{postIds}") | ||||
|     public R<Void> remove(@ApiParam("岗位ID串") @PathVariable Long[] postIds) { | ||||
|     public R<Void> remove(@PathVariable Long[] postIds) { | ||||
|         return toAjax(postService.deletePostByIds(postIds)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取岗位选择框列表 | ||||
|      */ | ||||
|     @ApiOperation("获取岗位选择框列表") | ||||
|     @GetMapping("/optionselect") | ||||
|     public R<List<SysPost>> optionselect() { | ||||
|         List<SysPost> posts = postService.selectPostAll(); | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| package com.ruoyi.web.controller.system; | ||||
|  | ||||
| import cn.dev33.satoken.secure.BCrypt; | ||||
| import cn.hutool.core.io.FileUtil; | ||||
| import com.ruoyi.common.annotation.Log; | ||||
| import com.ruoyi.common.constant.UserConstants; | ||||
| import com.ruoyi.common.core.controller.BaseController; | ||||
| @@ -9,19 +10,17 @@ import com.ruoyi.common.core.domain.entity.SysUser; | ||||
| import com.ruoyi.common.enums.BusinessType; | ||||
| import com.ruoyi.common.helper.LoginHelper; | ||||
| import com.ruoyi.common.utils.StringUtils; | ||||
| import com.ruoyi.common.utils.file.MimeTypeUtils; | ||||
| import com.ruoyi.system.domain.SysOss; | ||||
| import com.ruoyi.system.service.ISysOssService; | ||||
| import com.ruoyi.system.service.ISysUserService; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiImplicitParam; | ||||
| import io.swagger.annotations.ApiImplicitParams; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.http.MediaType; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
| import org.springframework.web.multipart.MultipartFile; | ||||
|  | ||||
| import java.io.File; | ||||
| import java.util.Arrays; | ||||
| import java.util.HashMap; | ||||
| import java.util.Map; | ||||
|  | ||||
| @@ -31,7 +30,6 @@ import java.util.Map; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "个人信息控制器", tags = {"个人信息管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/system/user/profile") | ||||
| @@ -43,7 +41,6 @@ public class SysProfileController extends BaseController { | ||||
|     /** | ||||
|      * 个人信息 | ||||
|      */ | ||||
|     @ApiOperation("个人信息") | ||||
|     @GetMapping | ||||
|     public R<Map<String, Object>> profile() { | ||||
|         SysUser user = userService.selectUserById(getUserId()); | ||||
| @@ -57,7 +54,6 @@ public class SysProfileController extends BaseController { | ||||
|     /** | ||||
|      * 修改用户 | ||||
|      */ | ||||
|     @ApiOperation("修改用户") | ||||
|     @Log(title = "个人信息", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping | ||||
|     public R<Void> updateProfile(@RequestBody SysUser user) { | ||||
| @@ -72,6 +68,8 @@ public class SysProfileController extends BaseController { | ||||
|         user.setUserId(getUserId()); | ||||
|         user.setUserName(null); | ||||
|         user.setPassword(null); | ||||
|         user.setAvatar(null); | ||||
|         user.setDeptId(null); | ||||
|         if (userService.updateUserProfile(user) > 0) { | ||||
|             return R.ok(); | ||||
|         } | ||||
| @@ -80,12 +78,10 @@ public class SysProfileController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 重置密码 | ||||
|      * | ||||
|      * @param newPassword 旧密码 | ||||
|      * @param oldPassword 新密码 | ||||
|      */ | ||||
|     @ApiOperation("重置密码") | ||||
|     @ApiImplicitParams({ | ||||
|         @ApiImplicitParam(name = "oldPassword", value = "旧密码", paramType = "query", dataTypeClass = String.class), | ||||
|         @ApiImplicitParam(name = "newPassword", value = "新密码", paramType = "query", dataTypeClass = String.class) | ||||
|     }) | ||||
|     @Log(title = "个人信息", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping("/updatePwd") | ||||
|     public R<Void> updatePwd(String oldPassword, String newPassword) { | ||||
| @@ -107,17 +103,19 @@ public class SysProfileController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 头像上传 | ||||
|      * | ||||
|      * @param avatarfile 用户头像 | ||||
|      */ | ||||
|     @ApiOperation("头像上传") | ||||
|     @ApiImplicitParams({ | ||||
|         @ApiImplicitParam(name = "avatarfile", value = "用户头像", paramType = "query", dataTypeClass = File.class, required = true) | ||||
|     }) | ||||
|     @Log(title = "用户头像", businessType = BusinessType.UPDATE) | ||||
|     @PostMapping("/avatar") | ||||
|     public R<Map<String, Object>> avatar(@RequestPart("avatarfile") MultipartFile file) { | ||||
|     @PostMapping(value = "/avatar", consumes = MediaType.MULTIPART_FORM_DATA_VALUE) | ||||
|     public R<Map<String, Object>> avatar(@RequestPart("avatarfile") MultipartFile avatarfile) { | ||||
|         Map<String, Object> ajax = new HashMap<>(); | ||||
|         if (!file.isEmpty()) { | ||||
|             SysOss oss = iSysOssService.upload(file); | ||||
|         if (!avatarfile.isEmpty()) { | ||||
|             String extension = FileUtil.extName(avatarfile.getOriginalFilename()); | ||||
|             if (!StringUtils.equalsAnyIgnoreCase(extension, MimeTypeUtils.IMAGE_EXTENSION)) { | ||||
|                 return R.fail("文件格式不正确,请上传" + Arrays.toString(MimeTypeUtils.IMAGE_EXTENSION) + "格式"); | ||||
|             } | ||||
|             SysOss oss = iSysOssService.upload(avatarfile); | ||||
|             String avatar = oss.getUrl(); | ||||
|             if (userService.updateUserAvatar(getUsername(), avatar)) { | ||||
|                 ajax.put("imgUrl", avatar); | ||||
|   | ||||
| @@ -1,12 +1,11 @@ | ||||
| package com.ruoyi.web.controller.system; | ||||
|  | ||||
| import com.ruoyi.common.annotation.Anonymous; | ||||
| import com.ruoyi.common.core.controller.BaseController; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.domain.model.RegisterBody; | ||||
| import com.ruoyi.system.service.ISysConfigService; | ||||
| import com.ruoyi.system.service.SysRegisterService; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.PostMapping; | ||||
| @@ -19,7 +18,6 @@ import org.springframework.web.bind.annotation.RestController; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "注册验证控制器", tags = {"注册验证管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| public class SysRegisterController extends BaseController { | ||||
| @@ -27,7 +25,10 @@ public class SysRegisterController extends BaseController { | ||||
|     private final SysRegisterService registerService; | ||||
|     private final ISysConfigService configService; | ||||
|  | ||||
|     @ApiOperation("用户注册") | ||||
|     /** | ||||
|      * 用户注册 | ||||
|      */ | ||||
|     @Anonymous | ||||
|     @PostMapping("/register") | ||||
|     public R<Void> register(@Validated @RequestBody RegisterBody user) { | ||||
|         if (!("true".equals(configService.selectConfigByKey("sys.account.registerUser")))) { | ||||
|   | ||||
| @@ -5,8 +5,9 @@ import cn.hutool.core.util.ObjectUtil; | ||||
| 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.R; | ||||
| import com.ruoyi.common.core.domain.PageQuery; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.domain.entity.SysDept; | ||||
| import com.ruoyi.common.core.domain.entity.SysRole; | ||||
| import com.ruoyi.common.core.domain.entity.SysUser; | ||||
| import com.ruoyi.common.core.domain.model.LoginUser; | ||||
| @@ -15,16 +16,18 @@ import com.ruoyi.common.enums.BusinessType; | ||||
| import com.ruoyi.common.helper.LoginHelper; | ||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | ||||
| import com.ruoyi.system.domain.SysUserRole; | ||||
| import com.ruoyi.system.service.ISysDeptService; | ||||
| import com.ruoyi.system.service.ISysRoleService; | ||||
| import com.ruoyi.system.service.ISysUserService; | ||||
| import com.ruoyi.system.service.SysPermissionService; | ||||
| import io.swagger.annotations.*; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
|  | ||||
| import javax.servlet.http.HttpServletResponse; | ||||
| import java.util.HashMap; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * 角色信息 | ||||
| @@ -32,7 +35,6 @@ import java.util.List; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "角色信息控制器", tags = {"角色信息管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/system/role") | ||||
| @@ -40,16 +42,21 @@ public class SysRoleController extends BaseController { | ||||
|  | ||||
|     private final ISysRoleService roleService; | ||||
|     private final ISysUserService userService; | ||||
|     private final ISysDeptService deptService; | ||||
|     private final SysPermissionService permissionService; | ||||
|  | ||||
|     @ApiOperation("查询角色信息列表") | ||||
|     /** | ||||
|      * 获取角色信息列表 | ||||
|      */ | ||||
|     @SaCheckPermission("system:role:list") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo<SysRole> list(SysRole role, PageQuery pageQuery) { | ||||
|         return roleService.selectPageRoleList(role, pageQuery); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("导出角色信息列表") | ||||
|     /** | ||||
|      * 导出角色信息列表 | ||||
|      */ | ||||
|     @Log(title = "角色管理", businessType = BusinessType.EXPORT) | ||||
|     @SaCheckPermission("system:role:export") | ||||
|     @PostMapping("/export") | ||||
| @@ -60,11 +67,12 @@ public class SysRoleController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 根据角色编号获取详细信息 | ||||
|      * | ||||
|      * @param roleId 角色ID | ||||
|      */ | ||||
|     @ApiOperation("根据角色编号获取详细信息") | ||||
|     @SaCheckPermission("system:role:query") | ||||
|     @GetMapping(value = "/{roleId}") | ||||
|     public R<SysRole> getInfo(@ApiParam("角色ID") @PathVariable Long roleId) { | ||||
|     public R<SysRole> getInfo(@PathVariable Long roleId) { | ||||
|         roleService.checkRoleDataScope(roleId); | ||||
|         return R.ok(roleService.selectRoleById(roleId)); | ||||
|     } | ||||
| @@ -72,7 +80,6 @@ public class SysRoleController extends BaseController { | ||||
|     /** | ||||
|      * 新增角色 | ||||
|      */ | ||||
|     @ApiOperation("新增角色") | ||||
|     @SaCheckPermission("system:role:add") | ||||
|     @Log(title = "角色管理", businessType = BusinessType.INSERT) | ||||
|     @PostMapping | ||||
| @@ -89,7 +96,6 @@ public class SysRoleController extends BaseController { | ||||
|     /** | ||||
|      * 修改保存角色 | ||||
|      */ | ||||
|     @ApiOperation("修改保存角色") | ||||
|     @SaCheckPermission("system:role:edit") | ||||
|     @Log(title = "角色管理", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping | ||||
| @@ -118,7 +124,6 @@ public class SysRoleController extends BaseController { | ||||
|     /** | ||||
|      * 修改保存数据权限 | ||||
|      */ | ||||
|     @ApiOperation("修改保存数据权限") | ||||
|     @SaCheckPermission("system:role:edit") | ||||
|     @Log(title = "角色管理", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping("/dataScope") | ||||
| @@ -131,7 +136,6 @@ public class SysRoleController extends BaseController { | ||||
|     /** | ||||
|      * 状态修改 | ||||
|      */ | ||||
|     @ApiOperation("状态修改") | ||||
|     @SaCheckPermission("system:role:edit") | ||||
|     @Log(title = "角色管理", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping("/changeStatus") | ||||
| @@ -143,19 +147,19 @@ public class SysRoleController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 删除角色 | ||||
|      * | ||||
|      * @param roleIds 角色ID串 | ||||
|      */ | ||||
|     @ApiOperation("删除角色") | ||||
|     @SaCheckPermission("system:role:remove") | ||||
|     @Log(title = "角色管理", businessType = BusinessType.DELETE) | ||||
|     @DeleteMapping("/{roleIds}") | ||||
|     public R<Void> remove(@ApiParam("岗位ID串") @PathVariable Long[] roleIds) { | ||||
|     public R<Void> remove(@PathVariable Long[] roleIds) { | ||||
|         return toAjax(roleService.deleteRoleByIds(roleIds)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取角色选择框列表 | ||||
|      */ | ||||
|     @ApiOperation("获取角色选择框列表") | ||||
|     @SaCheckPermission("system:role:query") | ||||
|     @GetMapping("/optionselect") | ||||
|     public R<List<SysRole>> optionselect() { | ||||
| @@ -165,7 +169,6 @@ public class SysRoleController extends BaseController { | ||||
|     /** | ||||
|      * 查询已分配用户角色列表 | ||||
|      */ | ||||
|     @ApiOperation("查询已分配用户角色列表") | ||||
|     @SaCheckPermission("system:role:list") | ||||
|     @GetMapping("/authUser/allocatedList") | ||||
|     public TableDataInfo<SysUser> allocatedList(SysUser user, PageQuery pageQuery) { | ||||
| @@ -175,7 +178,6 @@ public class SysRoleController extends BaseController { | ||||
|     /** | ||||
|      * 查询未分配用户角色列表 | ||||
|      */ | ||||
|     @ApiOperation("查询未分配用户角色列表") | ||||
|     @SaCheckPermission("system:role:list") | ||||
|     @GetMapping("/authUser/unallocatedList") | ||||
|     public TableDataInfo<SysUser> unallocatedList(SysUser user, PageQuery pageQuery) { | ||||
| @@ -185,7 +187,6 @@ public class SysRoleController extends BaseController { | ||||
|     /** | ||||
|      * 取消授权用户 | ||||
|      */ | ||||
|     @ApiOperation("取消授权用户") | ||||
|     @SaCheckPermission("system:role:edit") | ||||
|     @Log(title = "角色管理", businessType = BusinessType.GRANT) | ||||
|     @PutMapping("/authUser/cancel") | ||||
| @@ -195,12 +196,10 @@ public class SysRoleController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 批量取消授权用户 | ||||
|      * | ||||
|      * @param roleId  角色ID | ||||
|      * @param userIds 用户ID串 | ||||
|      */ | ||||
|     @ApiOperation("批量取消授权用户") | ||||
|     @ApiImplicitParams({ | ||||
|         @ApiImplicitParam(name = "roleId", value = "角色ID", paramType = "query", dataTypeClass = String.class), | ||||
|         @ApiImplicitParam(name = "userIds", value = "用户ID串", paramType = "query", dataTypeClass = String.class) | ||||
|     }) | ||||
|     @SaCheckPermission("system:role:edit") | ||||
|     @Log(title = "角色管理", businessType = BusinessType.GRANT) | ||||
|     @PutMapping("/authUser/cancelAll") | ||||
| @@ -210,12 +209,10 @@ public class SysRoleController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 批量选择用户授权 | ||||
|      * | ||||
|      * @param roleId  角色ID | ||||
|      * @param userIds 用户ID串 | ||||
|      */ | ||||
|     @ApiOperation("批量选择用户授权") | ||||
|     @ApiImplicitParams({ | ||||
|         @ApiImplicitParam(name = "roleId", value = "角色ID", paramType = "query", dataTypeClass = String.class), | ||||
|         @ApiImplicitParam(name = "userIds", value = "用户ID串", paramType = "query", dataTypeClass = String.class) | ||||
|     }) | ||||
|     @SaCheckPermission("system:role:edit") | ||||
|     @Log(title = "角色管理", businessType = BusinessType.GRANT) | ||||
|     @PutMapping("/authUser/selectAll") | ||||
| @@ -223,4 +220,18 @@ public class SysRoleController extends BaseController { | ||||
|         roleService.checkRoleDataScope(roleId); | ||||
|         return toAjax(roleService.insertAuthUsers(roleId, userIds)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取对应角色部门树列表 | ||||
|      * | ||||
|      * @param roleId 角色ID | ||||
|      */ | ||||
|     @SaCheckPermission("system:role:list") | ||||
|     @GetMapping(value = "/deptTree/{roleId}") | ||||
|     public R<Map<String, Object>> roleDeptTreeselect(@PathVariable("roleId") Long roleId) { | ||||
|         Map<String, Object> ajax = new HashMap<>(); | ||||
|         ajax.put("checkedKeys", deptService.selectDeptListByRoleId(roleId)); | ||||
|         ajax.put("depts", deptService.selectDeptTreeList(new SysDept())); | ||||
|         return R.ok(ajax); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -3,6 +3,7 @@ package com.ruoyi.web.controller.system; | ||||
| import cn.dev33.satoken.annotation.SaCheckPermission; | ||||
| import cn.dev33.satoken.secure.BCrypt; | ||||
| import cn.hutool.core.bean.BeanUtil; | ||||
| import cn.hutool.core.lang.tree.Tree; | ||||
| import cn.hutool.core.util.ArrayUtil; | ||||
| import cn.hutool.core.util.ObjectUtil; | ||||
| import com.ruoyi.common.annotation.Log; | ||||
| @@ -17,16 +18,18 @@ import com.ruoyi.common.core.page.TableDataInfo; | ||||
| import com.ruoyi.common.enums.BusinessType; | ||||
| import com.ruoyi.common.excel.ExcelResult; | ||||
| import com.ruoyi.common.helper.LoginHelper; | ||||
| import com.ruoyi.common.utils.StreamUtils; | ||||
| import com.ruoyi.common.utils.StringUtils; | ||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | ||||
| import com.ruoyi.system.domain.vo.SysUserExportVo; | ||||
| import com.ruoyi.system.domain.vo.SysUserImportVo; | ||||
| import com.ruoyi.system.listener.SysUserImportListener; | ||||
| import com.ruoyi.system.service.ISysDeptService; | ||||
| import com.ruoyi.system.service.ISysPostService; | ||||
| import com.ruoyi.system.service.ISysRoleService; | ||||
| import com.ruoyi.system.service.ISysUserService; | ||||
| import io.swagger.annotations.*; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.http.MediaType; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
| import org.springframework.web.multipart.MultipartFile; | ||||
| @@ -36,7 +39,6 @@ import java.util.ArrayList; | ||||
| import java.util.HashMap; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.stream.Collectors; | ||||
|  | ||||
| /** | ||||
|  * 用户信息 | ||||
| @@ -44,7 +46,6 @@ import java.util.stream.Collectors; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "用户信息控制器", tags = {"用户信息管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/system/user") | ||||
| @@ -53,18 +54,20 @@ public class SysUserController extends BaseController { | ||||
|     private final ISysUserService userService; | ||||
|     private final ISysRoleService roleService; | ||||
|     private final ISysPostService postService; | ||||
|     private final ISysDeptService deptService; | ||||
|  | ||||
|     /** | ||||
|      * 获取用户列表 | ||||
|      */ | ||||
|     @ApiOperation("获取用户列表") | ||||
|     @SaCheckPermission("system:user:list") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo<SysUser> list(SysUser user, PageQuery pageQuery) { | ||||
|         return userService.selectPageUserList(user, pageQuery); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("导出用户列表") | ||||
|     /** | ||||
|      * 导出用户列表 | ||||
|      */ | ||||
|     @Log(title = "用户管理", businessType = BusinessType.EXPORT) | ||||
|     @SaCheckPermission("system:user:export") | ||||
|     @PostMapping("/export") | ||||
| @@ -82,19 +85,23 @@ public class SysUserController extends BaseController { | ||||
|         ExcelUtil.exportExcel(listVo, "用户数据", SysUserExportVo.class, response); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("导入用户列表") | ||||
|     @ApiImplicitParams({ | ||||
|         @ApiImplicitParam(name = "file", value = "导入文件", dataType = "java.io.File", required = true), | ||||
|     }) | ||||
|     /** | ||||
|      * 导入数据 | ||||
|      * | ||||
|      * @param file          导入文件 | ||||
|      * @param updateSupport 是否更新已存在数据 | ||||
|      */ | ||||
|     @Log(title = "用户管理", businessType = BusinessType.IMPORT) | ||||
|     @SaCheckPermission("system:user:import") | ||||
|     @PostMapping("/importData") | ||||
|     @PostMapping(value = "/importData", consumes = MediaType.MULTIPART_FORM_DATA_VALUE) | ||||
|     public R<Void> importData(@RequestPart("file") MultipartFile file, boolean updateSupport) throws Exception { | ||||
|         ExcelResult<SysUserImportVo> result = ExcelUtil.importExcel(file.getInputStream(), SysUserImportVo.class, new SysUserImportListener(updateSupport)); | ||||
|         return R.ok(result.getAnalysis()); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("下载导入模板") | ||||
|     /** | ||||
|      * 获取导入模板 | ||||
|      */ | ||||
|     @PostMapping("/importTemplate") | ||||
|     public void importTemplate(HttpServletResponse response) { | ||||
|         ExcelUtil.exportExcel(new ArrayList<>(), "用户数据", SysUserImportVo.class, response); | ||||
| @@ -102,21 +109,22 @@ public class SysUserController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 根据用户编号获取详细信息 | ||||
|      * | ||||
|      * @param userId 用户ID | ||||
|      */ | ||||
|     @ApiOperation("根据用户编号获取详细信息") | ||||
|     @SaCheckPermission("system:user:query") | ||||
|     @GetMapping(value = {"/", "/{userId}"}) | ||||
|     public R<Map<String, Object>> getInfo(@ApiParam("用户ID") @PathVariable(value = "userId", required = false) Long userId) { | ||||
|     public R<Map<String, Object>> getInfo(@PathVariable(value = "userId", required = false) Long userId) { | ||||
|         userService.checkUserDataScope(userId); | ||||
|         Map<String, Object> ajax = new HashMap<>(); | ||||
|         List<SysRole> roles = roleService.selectRoleAll(); | ||||
|         ajax.put("roles", LoginHelper.isAdmin(userId) ? roles : roles.stream().filter(r -> !r.isAdmin()).collect(Collectors.toList())); | ||||
|         ajax.put("roles", LoginHelper.isAdmin(userId) ? roles : StreamUtils.filter(roles, r -> !r.isAdmin())); | ||||
|         ajax.put("posts", postService.selectPostAll()); | ||||
|         if (ObjectUtil.isNotNull(userId)) { | ||||
|             SysUser sysUser = userService.selectUserById(userId); | ||||
|             ajax.put("user", sysUser); | ||||
|             ajax.put("postIds", postService.selectPostListByUserId(userId)); | ||||
|             ajax.put("roleIds", sysUser.getRoles().stream().map(SysRole::getRoleId).collect(Collectors.toList())); | ||||
|             ajax.put("roleIds", StreamUtils.toList(sysUser.getRoles(), SysRole::getRoleId)); | ||||
|         } | ||||
|         return R.ok(ajax); | ||||
|     } | ||||
| @@ -124,7 +132,6 @@ public class SysUserController extends BaseController { | ||||
|     /** | ||||
|      * 新增用户 | ||||
|      */ | ||||
|     @ApiOperation("新增用户") | ||||
|     @SaCheckPermission("system:user:add") | ||||
|     @Log(title = "用户管理", businessType = BusinessType.INSERT) | ||||
|     @PostMapping | ||||
| @@ -145,7 +152,6 @@ public class SysUserController extends BaseController { | ||||
|     /** | ||||
|      * 修改用户 | ||||
|      */ | ||||
|     @ApiOperation("修改用户") | ||||
|     @SaCheckPermission("system:user:edit") | ||||
|     @Log(title = "用户管理", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping | ||||
| @@ -164,12 +170,13 @@ public class SysUserController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 删除用户 | ||||
|      * | ||||
|      * @param userIds 角色ID串 | ||||
|      */ | ||||
|     @ApiOperation("删除用户") | ||||
|     @SaCheckPermission("system:user:remove") | ||||
|     @Log(title = "用户管理", businessType = BusinessType.DELETE) | ||||
|     @DeleteMapping("/{userIds}") | ||||
|     public R<Void> remove(@ApiParam("角色ID串") @PathVariable Long[] userIds) { | ||||
|     public R<Void> remove(@PathVariable Long[] userIds) { | ||||
|         if (ArrayUtil.contains(userIds, getUserId())) { | ||||
|             return R.fail("当前用户不能删除"); | ||||
|         } | ||||
| @@ -179,7 +186,6 @@ public class SysUserController extends BaseController { | ||||
|     /** | ||||
|      * 重置密码 | ||||
|      */ | ||||
|     @ApiOperation("重置密码") | ||||
|     @SaCheckPermission("system:user:resetPwd") | ||||
|     @Log(title = "用户管理", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping("/resetPwd") | ||||
| @@ -193,7 +199,6 @@ public class SysUserController extends BaseController { | ||||
|     /** | ||||
|      * 状态修改 | ||||
|      */ | ||||
|     @ApiOperation("状态修改") | ||||
|     @SaCheckPermission("system:user:edit") | ||||
|     @Log(title = "用户管理", businessType = BusinessType.UPDATE) | ||||
|     @PutMapping("/changeStatus") | ||||
| @@ -205,27 +210,26 @@ public class SysUserController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 根据用户编号获取授权角色 | ||||
|      * | ||||
|      * @param userId 用户ID | ||||
|      */ | ||||
|     @ApiOperation("根据用户编号获取授权角色") | ||||
|     @SaCheckPermission("system:user:query") | ||||
|     @GetMapping("/authRole/{userId}") | ||||
|     public R<Map<String, Object>> authRole(@ApiParam("用户ID") @PathVariable("userId") Long userId) { | ||||
|     public R<Map<String, Object>> authRole(@PathVariable Long userId) { | ||||
|         SysUser user = userService.selectUserById(userId); | ||||
|         List<SysRole> roles = roleService.selectRolesByUserId(userId); | ||||
|         Map<String, Object> ajax = new HashMap<>(); | ||||
|         ajax.put("user", user); | ||||
|         ajax.put("roles", LoginHelper.isAdmin(userId) ? roles : roles.stream().filter(r -> !r.isAdmin()).collect(Collectors.toList())); | ||||
|         ajax.put("roles", LoginHelper.isAdmin(userId) ? roles : StreamUtils.filter(roles, r -> !r.isAdmin())); | ||||
|         return R.ok(ajax); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 用户授权角色 | ||||
|      * | ||||
|      * @param userId  用户Id | ||||
|      * @param roleIds 角色ID串 | ||||
|      */ | ||||
|     @ApiOperation("用户授权角色") | ||||
|     @ApiImplicitParams({ | ||||
|         @ApiImplicitParam(name = "userId", value = "用户Id", paramType = "query", dataTypeClass = String.class), | ||||
|         @ApiImplicitParam(name = "roleIds", value = "角色ID串", paramType = "query", dataTypeClass = String.class) | ||||
|     }) | ||||
|     @SaCheckPermission("system:user:edit") | ||||
|     @Log(title = "用户管理", businessType = BusinessType.GRANT) | ||||
|     @PutMapping("/authRole") | ||||
| @@ -234,4 +238,14 @@ public class SysUserController extends BaseController { | ||||
|         userService.insertUserAuth(userId, roleIds); | ||||
|         return R.ok(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取部门树列表 | ||||
|      */ | ||||
|     @SaCheckPermission("system:user:list") | ||||
|     @GetMapping("/deptTree") | ||||
|     public R<List<Tree<Long>>> deptTree(SysDept dept) { | ||||
|         return R.ok(deptService.selectDeptTreeList(dept)); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -130,14 +130,16 @@ spring: | ||||
|     port: 6379 | ||||
|     # 数据库索引 | ||||
|     database: 0 | ||||
|     # 密码 | ||||
|     password: | ||||
|     # 密码(如没有密码请注释掉) | ||||
|     # password: | ||||
|     # 连接超时时间 | ||||
|     timeout: 10s | ||||
|     # 是否开启ssl | ||||
|     ssl: false | ||||
|  | ||||
| redisson: | ||||
|   # redis key前缀 | ||||
|   keyPrefix: | ||||
|   # 线程池数量 | ||||
|   threads: 4 | ||||
|   # Netty线程池数量 | ||||
| @@ -156,3 +158,37 @@ redisson: | ||||
|     timeout: 3000 | ||||
|     # 发布和订阅连接池大小 | ||||
|     subscriptionConnectionPoolSize: 50 | ||||
|  | ||||
| --- # mail 邮件发送 | ||||
| mail: | ||||
|   enabled: false | ||||
|   host: smtp.163.com | ||||
|   port: 465 | ||||
|   # 是否需要用户名密码验证 | ||||
|   auth: true | ||||
|   # 发送方,遵循RFC-822标准 | ||||
|   from: xxx@163.com | ||||
|   # 用户名(注意:如果使用foxmail邮箱,此处user为qq号) | ||||
|   user: xxx@163.com | ||||
|   # 密码(注意,某些邮箱需要为SMTP服务单独设置密码,详情查看相关帮助) | ||||
|   pass: xxxxxxxxxx | ||||
|   # 使用 STARTTLS安全连接,STARTTLS是对纯文本通信协议的扩展。 | ||||
|   starttlsEnable: true | ||||
|   # 使用SSL安全连接 | ||||
|   sslEnable: true | ||||
|   # SMTP超时时长,单位毫秒,缺省值不超时 | ||||
|   timeout: 0 | ||||
|   # Socket连接超时值,单位毫秒,缺省值不超时 | ||||
|   connectionTimeout: 0 | ||||
|  | ||||
| --- # sms 短信 | ||||
| sms: | ||||
|   enabled: false | ||||
|   # 阿里云 dysmsapi.aliyuncs.com | ||||
|   # 腾讯云 sms.tencentcloudapi.com | ||||
|   endpoint: "dysmsapi.aliyuncs.com" | ||||
|   accessKeyId: xxxxxxx | ||||
|   accessKeySecret: xxxxxx | ||||
|   signName: 测试 | ||||
|   # 腾讯专用 | ||||
|   sdkAppId: | ||||
|   | ||||
| @@ -5,7 +5,7 @@ spring.servlet.multipart.location: /ruoyi/server/temp | ||||
| spring.boot.admin.client: | ||||
|   # 增加客户端开关 | ||||
|   enabled: true | ||||
|   url: http://172.30.0.90:9090/admin | ||||
|   url: http://localhost:9090/admin | ||||
|   instance: | ||||
|     service-host-type: IP | ||||
|   username: ruoyi | ||||
| @@ -16,7 +16,7 @@ xxl.job: | ||||
|   # 执行器开关 | ||||
|   enabled: true | ||||
|   # 调度中心地址:如调度中心集群部署存在多个地址则用逗号分隔。 | ||||
|   admin-addresses: http://172.30.0.92:9100/xxl-job-admin | ||||
|   admin-addresses: http://localhost:9100/xxl-job-admin | ||||
|   # 执行器通讯TOKEN:非空时启用 | ||||
|   access-token: xxl-job | ||||
|   executor: | ||||
| @@ -51,7 +51,7 @@ spring: | ||||
|           driverClassName: com.mysql.cj.jdbc.Driver | ||||
|           # jdbc 所有参数配置参考 https://lionli.blog.csdn.net/article/details/122018562 | ||||
|           # rewriteBatchedStatements=true 批处理优化 大幅提升批量插入更新删除性能(对数据库有性能损耗 使用批量操作应考虑性能问题) | ||||
|           url: jdbc:mysql://172.30.0.36:3306/ry-vue?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8&autoReconnect=true&rewriteBatchedStatements=true | ||||
|           url: jdbc:mysql://localhost:3306/ry-vue?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8&autoReconnect=true&rewriteBatchedStatements=true | ||||
|           username: root | ||||
|           password: root | ||||
|         # 从库数据源 | ||||
| @@ -63,19 +63,19 @@ spring: | ||||
|           password: | ||||
| #        oracle: | ||||
| #          driverClassName: oracle.jdbc.OracleDriver | ||||
| #          url: jdbc:oracle:thin:@//172.30.0.36:1521/XE | ||||
| #          url: jdbc:oracle:thin:@//localhost:1521/XE | ||||
| #          username: ROOT | ||||
| #          password: root | ||||
| #          druid: | ||||
| #            validationQuery: SELECT 1 FROM DUAL | ||||
| #        postgres: | ||||
| #          driverClassName: org.postgresql.Driver | ||||
| #          url: jdbc:postgresql://172.30.0.36:5432/postgres?useUnicode=true&characterEncoding=utf8&useSSL=true&autoReconnect=true&reWriteBatchedInserts=true | ||||
| #          url: jdbc:postgresql://localhost:5432/postgres?useUnicode=true&characterEncoding=utf8&useSSL=true&autoReconnect=true&reWriteBatchedInserts=true | ||||
| #          username: root | ||||
| #          password: root | ||||
| #        sqlserver: | ||||
| #          driverClassName: com.microsoft.sqlserver.jdbc.SQLServerDriver | ||||
| #          url: jdbc:sqlserver://172.30.0.36:1433;DatabaseName=tempdb;SelectMethod=cursor;rewriteBatchedStatements=true | ||||
| #          url: jdbc:sqlserver://localhost:1433;DatabaseName=tempdb;SelectMethod=cursor;rewriteBatchedStatements=true | ||||
| #          username: SA | ||||
| #          password: root | ||||
|       druid: | ||||
| @@ -128,19 +128,21 @@ spring.datasource.druid: | ||||
| spring: | ||||
|   redis: | ||||
|     # 地址 | ||||
|     host: 172.30.0.48 | ||||
|     host: localhost | ||||
|     # 端口,默认为6379 | ||||
|     port: 6379 | ||||
|     # 数据库索引 | ||||
|     database: 0 | ||||
|     # 密码 | ||||
|     password: | ||||
|     # 密码(如没有密码请注释掉) | ||||
|     # password: | ||||
|     # 连接超时时间 | ||||
|     timeout: 10s | ||||
|     # 是否开启ssl | ||||
|     ssl: false | ||||
|  | ||||
| redisson: | ||||
|   # redis key前缀 | ||||
|   keyPrefix: | ||||
|   # 线程池数量 | ||||
|   threads: 16 | ||||
|   # Netty线程池数量 | ||||
| @@ -159,3 +161,37 @@ redisson: | ||||
|     timeout: 3000 | ||||
|     # 发布和订阅连接池大小 | ||||
|     subscriptionConnectionPoolSize: 50 | ||||
|  | ||||
| --- # mail 邮件发送 | ||||
| mail: | ||||
|   enabled: false | ||||
|   host: smtp.163.com | ||||
|   port: 465 | ||||
|   # 是否需要用户名密码验证 | ||||
|   auth: true | ||||
|   # 发送方,遵循RFC-822标准 | ||||
|   from: xxx@163.com | ||||
|   # 用户名(注意:如果使用foxmail邮箱,此处user为qq号) | ||||
|   user: xxx@163.com | ||||
|   # 密码(注意,某些邮箱需要为SMTP服务单独设置密码,详情查看相关帮助) | ||||
|   pass: xxxxxxxxxx | ||||
|   # 使用 STARTTLS安全连接,STARTTLS是对纯文本通信协议的扩展。 | ||||
|   starttlsEnable: true | ||||
|   # 使用SSL安全连接 | ||||
|   sslEnable: true | ||||
|   # SMTP超时时长,单位毫秒,缺省值不超时 | ||||
|   timeout: 0 | ||||
|   # Socket连接超时值,单位毫秒,缺省值不超时 | ||||
|   connectionTimeout: 0 | ||||
|  | ||||
| --- # sms 短信 | ||||
| sms: | ||||
|   enabled: false | ||||
|   # 阿里云 dysmsapi.aliyuncs.com | ||||
|   # 腾讯云 sms.tencentcloudapi.com | ||||
|   endpoint: "dysmsapi.aliyuncs.com" | ||||
|   accessKeyId: xxxxxxx | ||||
|   accessKeySecret: xxxxxx | ||||
|   signName: 测试 | ||||
|   # 腾讯专用 | ||||
|   sdkAppId: | ||||
|   | ||||
| @@ -53,6 +53,14 @@ logging: | ||||
|     org.springframework: warn | ||||
|   config: classpath:logback.xml | ||||
|  | ||||
| # 用户配置 | ||||
| user: | ||||
|   password: | ||||
|     # 密码最大错误次数 | ||||
|     maxRetryCount: 5 | ||||
|     # 密码锁定时间(默认10分钟) | ||||
|     lockTime: 10 | ||||
|  | ||||
| # Spring配置 | ||||
| spring: | ||||
|   application: | ||||
| @@ -75,10 +83,6 @@ spring: | ||||
|     restart: | ||||
|       # 热部署开关 | ||||
|       enabled: true | ||||
|   mvc: | ||||
|     pathmatch: | ||||
|       # 适配 boot 2.6 路由与 springfox 兼容 | ||||
|       matching-strategy: ANT_PATH_MATCHER | ||||
|   jackson: | ||||
|     # 日期格式化 | ||||
|     date-format: yyyy-MM-dd HH:mm:ss | ||||
| @@ -118,12 +122,6 @@ sa-token: | ||||
| security: | ||||
|   # 排除路径 | ||||
|   excludes: | ||||
|     - /login | ||||
|     - /smsLogin | ||||
|     - /xcxLogin | ||||
|     - /logout | ||||
|     - /register | ||||
|     - /captchaImage | ||||
|     # 静态资源 | ||||
|     - /*.html | ||||
|     - /**/*.html | ||||
| @@ -131,10 +129,8 @@ security: | ||||
|     - /**/*.js | ||||
|     # swagger 文档配置 | ||||
|     - /favicon.ico | ||||
|     - /doc.html | ||||
|     - /swagger-resources/** | ||||
|     - /webjars/** | ||||
|     - /*/api-docs | ||||
|     - /*/api-docs/** | ||||
|     # druid 监控配置 | ||||
|     - /druid/** | ||||
|     # actuator 监控配置 | ||||
| @@ -189,8 +185,7 @@ mybatis-plus: | ||||
| swagger: | ||||
|   # 是否开启swagger | ||||
|   enabled: true | ||||
|   # 请求前缀 | ||||
|   pathMapping: /dev-api | ||||
|   info: | ||||
|     # 标题 | ||||
|     title: '标题:${ruoyi.name}后台管理系统_接口文档' | ||||
|     # 描述 | ||||
| @@ -202,33 +197,23 @@ swagger: | ||||
|       name: Lion Li | ||||
|       email: crazylionli@163.com | ||||
|       url: https://gitee.com/JavaLionLi/RuoYi-Vue-Plus | ||||
|   groups: | ||||
|     - name: 1.演示案例 | ||||
|       basePackage: com.ruoyi.demo | ||||
|     - name: 2.系统模块 | ||||
|       basePackage: com.ruoyi.web | ||||
|     - name: 3.代码生成模块 | ||||
|       basePackage: com.ruoyi.generator | ||||
|   components: | ||||
|     # 鉴权方式配置 | ||||
|     security-schemes: | ||||
|       apiKey: | ||||
|         type: APIKEY | ||||
|         in: HEADER | ||||
|         name: ${sa-token.token-name} | ||||
|  | ||||
| knife4j: | ||||
|   # 是否开启Knife4j增强模式 | ||||
|   enable: true | ||||
|   # 是否开启生产环境保护策略 | ||||
|   production: @knife4j.production@ | ||||
|   basic: | ||||
|     enable: true | ||||
|     username: ruoyi | ||||
|     password: 123456 | ||||
|   # 前端Ui的个性化配置属性 | ||||
|   setting: | ||||
|     # 默认语言 | ||||
|     language: zh-CN | ||||
|     # 是否显示Footer | ||||
|     enableFooter: false | ||||
|     # 是否开启动态参数调试功能 | ||||
|     enableDynamicParameter: true | ||||
|     # 是否在每个Debug调试栏后显示刷新变量按钮 | ||||
|     enableReloadCacheParameter: true | ||||
| springdoc: | ||||
|   #这里定义了两个分组,可定义多个,也可以不定义 | ||||
|   group-configs: | ||||
|     - group: 1.演示模块 | ||||
|       packages-to-scan: com.ruoyi.demo | ||||
|     - group: 2.系统模块 | ||||
|       packages-to-scan: com.ruoyi.web | ||||
|     - group: 3.代码生成模块 | ||||
|       packages-to-scan: com.ruoyi.generator | ||||
|  | ||||
| # 防止XSS攻击 | ||||
| xss: | ||||
| @@ -248,26 +233,11 @@ thread-pool: | ||||
|   # 线程池维护线程所允许的空闲时间 | ||||
|   keepAliveSeconds: 300 | ||||
|  | ||||
| --- # redisson 缓存配置 | ||||
| redisson: | ||||
|   cacheGroup: | ||||
|     # 用例: @Cacheable(cacheNames="groupId", key="#XXX") 方可使用缓存组配置 | ||||
|     - groupId: redissonCacheMap | ||||
|       # 组过期时间(脚本监控) | ||||
|       ttl: 60000 | ||||
|       # 组最大空闲时间(脚本监控) | ||||
|       maxIdleTime: 60000 | ||||
|       # 组最大长度 | ||||
|       maxSize: 0 | ||||
|     - groupId: testCache | ||||
|       ttl: 1000 | ||||
|       maxIdleTime: 500 | ||||
|  | ||||
| --- # 分布式锁 lock4j 全局配置 | ||||
| lock4j: | ||||
|   # 获取分布式锁超时时间,默认为 3000 毫秒 | ||||
|   acquire-timeout: 3000 | ||||
|   # 分布式锁的超时时间,默认为 30 毫秒 | ||||
|   # 分布式锁的超时时间,默认为 30 秒 | ||||
|   expire: 30000 | ||||
|  | ||||
| --- # Actuator 监控端点的配置项 | ||||
| @@ -275,9 +245,9 @@ management: | ||||
|   endpoints: | ||||
|     web: | ||||
|       exposure: | ||||
|         # 需要开放的端点。默认值只打开 health 和 info 两个端点。通过设置 * ,可以开放所有端点。 | ||||
|         # 生产环境不建议放开所有 根据项目需求放开即可 | ||||
|         include: @endpoints.include@ | ||||
|         include: '*' | ||||
|   endpoint: | ||||
|     health: | ||||
|       show-details: ALWAYS | ||||
|     logfile: | ||||
|       external-file: ./logs/sys-console.log | ||||
|   | ||||
| @@ -5,7 +5,7 @@ user.jcaptcha.expire=验证码已失效 | ||||
| user.not.exists=对不起, 您的账号:{0} 不存在. | ||||
| user.password.not.match=用户不存在/密码错误 | ||||
| user.password.retry.limit.count=密码输入错误{0}次 | ||||
| user.password.retry.limit.exceed=密码错误次数过多,帐户锁定{0}分钟 | ||||
| user.password.retry.limit.exceed=密码输入错误{0}次,帐户锁定{1}分钟 | ||||
| user.password.delete=对不起,您的账号:{0} 已被删除 | ||||
| user.blocked=对不起,您的账号:{0} 已禁用,请联系管理员 | ||||
| role.blocked=角色已封禁,请联系管理员 | ||||
| @@ -41,5 +41,5 @@ repeat.submit.message=不允许重复提交,请稍候再试 | ||||
| rate.limiter.message=访问过于频繁,请稍候再试 | ||||
| sms.code.not.blank=短信验证码不能为空 | ||||
| sms.code.retry.limit.count=短信验证码输入错误{0}次 | ||||
| sms.code.retry.limit.exceed=短信验证码错误次数过多,帐户锁定{0}分钟 | ||||
| sms.code.retry.limit.exceed=短信验证码输入错误{0}次,帐户锁定{0}分钟 | ||||
| xcx.code.not.blank=小程序code不能为空 | ||||
|   | ||||
| @@ -5,7 +5,7 @@ user.jcaptcha.expire=Captcha invalid | ||||
| user.not.exists=Sorry, your account: {0} does not exist | ||||
| user.password.not.match=User does not exist/Password error | ||||
| user.password.retry.limit.count=Password input error {0} times | ||||
| user.password.retry.limit.exceed=Too many password errors, account locked for {0} minutes | ||||
| user.password.retry.limit.exceed=Password input error {0} times, account locked for {1} minutes | ||||
| user.password.delete=Sorry, your account:{0} has been deleted | ||||
| user.blocked=Sorry, your account: {0} has been disabled. Please contact the administrator | ||||
| role.blocked=Role disabled,please contact administrators | ||||
| @@ -41,5 +41,5 @@ repeat.submit.message=Repeat submit is not allowed, please try again later | ||||
| rate.limiter.message=Visit too frequently, please try again later | ||||
| sms.code.not.blank=Sms code cannot be blank | ||||
| sms.code.retry.limit.count=Sms code input error {0} times | ||||
| sms.code.retry.limit.exceed=Too many sms code errors, account locked for {0} minutes | ||||
| sms.code.retry.limit.exceed=Sms code input error {0} times, account locked for {0} minutes | ||||
| xcx.code.not.blank=Mini program code cannot be blank | ||||
|   | ||||
| @@ -5,7 +5,7 @@ user.jcaptcha.expire=验证码已失效 | ||||
| user.not.exists=对不起, 您的账号:{0} 不存在. | ||||
| user.password.not.match=用户不存在/密码错误 | ||||
| user.password.retry.limit.count=密码输入错误{0}次 | ||||
| user.password.retry.limit.exceed=密码错误次数过多,帐户锁定{0}分钟 | ||||
| user.password.retry.limit.exceed=密码输入错误{0}次,帐户锁定{1}分钟 | ||||
| user.password.delete=对不起,您的账号:{0} 已被删除 | ||||
| user.blocked=对不起,您的账号:{0} 已禁用,请联系管理员 | ||||
| role.blocked=角色已封禁,请联系管理员 | ||||
| @@ -41,5 +41,5 @@ repeat.submit.message=不允许重复提交,请稍候再试 | ||||
| rate.limiter.message=访问过于频繁,请稍候再试 | ||||
| sms.code.not.blank=短信验证码不能为空 | ||||
| sms.code.retry.limit.count=短信验证码输入错误{0}次 | ||||
| sms.code.retry.limit.exceed=短信验证码错误次数过多,帐户锁定{0}分钟 | ||||
| sms.code.retry.limit.exceed=短信验证码输入错误{0}次,帐户锁定{0}分钟 | ||||
| xcx.code.not.blank=小程序code不能为空 | ||||
|   | ||||
| @@ -97,17 +97,9 @@ | ||||
|         <appender-ref ref="file_error"/> | ||||
|     </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="console" /> | ||||
|         <appender-ref ref="async_info" /> | ||||
|         <appender-ref ref="async_error" /> | ||||
|         <appender-ref ref="file_console" /> | ||||
|   | ||||
| @@ -5,7 +5,7 @@ | ||||
|     <parent> | ||||
|         <artifactId>ruoyi-vue-plus</artifactId> | ||||
|         <groupId>com.ruoyi</groupId> | ||||
|         <version>4.1.0</version> | ||||
|         <version>4.3.0</version> | ||||
|     </parent> | ||||
|     <modelVersion>4.0.0</modelVersion> | ||||
|  | ||||
| @@ -58,12 +58,6 @@ | ||||
|             <artifactId>jackson-databind</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- excel工具 --> | ||||
|         <dependency> | ||||
|             <groupId>org.apache.poi</groupId> | ||||
|             <artifactId>poi-ooxml</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <dependency> | ||||
|             <groupId>com.alibaba</groupId> | ||||
|             <artifactId>easyexcel</artifactId> | ||||
| @@ -75,12 +69,6 @@ | ||||
|             <artifactId>snakeyaml</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- jdk11 缺失依赖 jaxb--> | ||||
|         <dependency> | ||||
|             <groupId>com.sun.xml.bind</groupId> | ||||
|             <artifactId>jaxb-impl</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- servlet包 --> | ||||
|         <dependency> | ||||
|             <groupId>javax.servlet</groupId> | ||||
| @@ -127,19 +115,24 @@ | ||||
|             <artifactId>hutool-extra</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <dependency> | ||||
|             <groupId>com.sun.mail</groupId> | ||||
|             <artifactId>jakarta.mail</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <dependency> | ||||
|             <groupId>org.projectlombok</groupId> | ||||
|             <artifactId>lombok</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <dependency> | ||||
|             <groupId>com.github.xiaoymin</groupId> | ||||
|             <artifactId>knife4j-spring-boot-starter</artifactId> | ||||
|             <groupId>org.springdoc</groupId> | ||||
|             <artifactId>springdoc-openapi-webmvc-core</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <dependency> | ||||
|             <groupId>io.swagger</groupId> | ||||
|             <artifactId>swagger-annotations</artifactId> | ||||
|             <groupId>org.springdoc</groupId> | ||||
|             <artifactId>springdoc-openapi-javadoc</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!--  自动生成YML配置关联JSON文件  --> | ||||
|   | ||||
| @@ -0,0 +1,18 @@ | ||||
| package com.ruoyi.common.annotation; | ||||
|  | ||||
| import java.lang.annotation.Documented; | ||||
| import java.lang.annotation.ElementType; | ||||
| import java.lang.annotation.Retention; | ||||
| import java.lang.annotation.RetentionPolicy; | ||||
| import java.lang.annotation.Target; | ||||
|  | ||||
| /** | ||||
|  * 匿名访问不鉴权注解 | ||||
|  * | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| @Target({ElementType.METHOD, ElementType.TYPE}) | ||||
| @Retention(RetentionPolicy.RUNTIME) | ||||
| @Documented | ||||
| public @interface Anonymous { | ||||
| } | ||||
| @@ -0,0 +1,24 @@ | ||||
| package com.ruoyi.common.annotation; | ||||
|  | ||||
| import com.ruoyi.common.excel.CellMergeStrategy; | ||||
|  | ||||
| import java.lang.annotation.*; | ||||
|  | ||||
| /** | ||||
|  * excel 列单元格合并(合并列相同项) | ||||
|  * | ||||
|  * 需搭配 {@link CellMergeStrategy} 策略使用 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Target(ElementType.FIELD) | ||||
| @Retention(RetentionPolicy.RUNTIME) | ||||
| @Inherited | ||||
| public @interface CellMerge { | ||||
|  | ||||
| 	/** | ||||
| 	 * col index | ||||
| 	 */ | ||||
| 	int index() default -1; | ||||
|  | ||||
| } | ||||
| @@ -5,6 +5,8 @@ import java.lang.annotation.*; | ||||
| /** | ||||
|  * 数据权限 | ||||
|  * | ||||
|  * 一个注解只能对应一个模板 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  * @version 3.5.0 | ||||
|  */ | ||||
| @@ -16,11 +18,11 @@ public @interface DataColumn { | ||||
|     /** | ||||
|      * 占位符关键字 | ||||
|      */ | ||||
|     String key() default "deptName"; | ||||
|     String[] key() default "deptName"; | ||||
|  | ||||
|     /** | ||||
|      * 占位符替换值 | ||||
|      */ | ||||
|     String value() default "dept_id"; | ||||
|     String[] value() default "dept_id"; | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| package com.ruoyi.common.annotation; | ||||
|  | ||||
| import com.ruoyi.common.constant.Constants; | ||||
| import com.ruoyi.common.constant.CacheConstants; | ||||
| import com.ruoyi.common.enums.LimitType; | ||||
|  | ||||
| import java.lang.annotation.*; | ||||
| @@ -17,7 +17,7 @@ public @interface RateLimiter { | ||||
|     /** | ||||
|      * 限流key | ||||
|      */ | ||||
|     String key() default Constants.RATE_LIMIT_KEY; | ||||
|     String key() default CacheConstants.RATE_LIMIT_KEY; | ||||
|  | ||||
|     /** | ||||
|      * 限流时间,单位秒 | ||||
|   | ||||
| @@ -0,0 +1,49 @@ | ||||
| package com.ruoyi.common.constant; | ||||
|  | ||||
| /** | ||||
|  * 缓存的key 常量 | ||||
|  * | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| public interface CacheConstants { | ||||
|  | ||||
|     /** | ||||
|      * 登录用户 redis key | ||||
|      */ | ||||
|     String LOGIN_TOKEN_KEY = "Authorization:login:token:"; | ||||
|  | ||||
|     /** | ||||
|      * 在线用户 redis key | ||||
|      */ | ||||
|     String ONLINE_TOKEN_KEY = "online_tokens:"; | ||||
|  | ||||
|     /** | ||||
|      * 验证码 redis key | ||||
|      */ | ||||
|     String CAPTCHA_CODE_KEY = "captcha_codes:"; | ||||
|  | ||||
|     /** | ||||
|      * 参数管理 cache key | ||||
|      */ | ||||
|     String SYS_CONFIG_KEY = "sys_config:"; | ||||
|  | ||||
|     /** | ||||
|      * 字典管理 cache key | ||||
|      */ | ||||
|     String SYS_DICT_KEY = "sys_dict:"; | ||||
|  | ||||
|     /** | ||||
|      * 防重提交 redis key | ||||
|      */ | ||||
|     String REPEAT_SUBMIT_KEY = "repeat_submit:"; | ||||
|  | ||||
|     /** | ||||
|      * 限流 redis key | ||||
|      */ | ||||
|     String RATE_LIMIT_KEY = "rate_limit:"; | ||||
|  | ||||
|     /** | ||||
|      * 登录账户密码错误次数 redis key | ||||
|      */ | ||||
|     String PWD_ERR_CNT_KEY = "pwd_err_cnt:"; | ||||
| } | ||||
| @@ -0,0 +1,48 @@ | ||||
| package com.ruoyi.common.constant; | ||||
|  | ||||
| /** | ||||
|  * 缓存组名称常量 | ||||
|  * <p> | ||||
|  * key 格式为 cacheNames#ttl#maxIdleTime#maxSize | ||||
|  * <p> | ||||
|  * ttl 过期时间 如果设置为0则不过期 默认为0 | ||||
|  * maxIdleTime 最大空闲时间 根据LRU算法清理空闲数据 如果设置为0则不检测 默认为0 | ||||
|  * maxSize 组最大长度 根据LRU算法清理溢出数据 如果设置为0则无限长 默认为0 | ||||
|  * <p> | ||||
|  * 例子: test#60s、test#0#60s、test#0#1m#1000、test#1h#0#500 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| public interface CacheNames { | ||||
|  | ||||
|     /** | ||||
|      * 演示案例 | ||||
|      */ | ||||
|     String DEMO_CACHE = "demo:cache#60s#10m#20"; | ||||
|  | ||||
|     /** | ||||
|      * 系统配置 | ||||
|      */ | ||||
|     String SYS_CONFIG = "sys_config"; | ||||
|  | ||||
|     /** | ||||
|      * 数据字典 | ||||
|      */ | ||||
|     String SYS_DICT = "sys_dict"; | ||||
|  | ||||
|     /** | ||||
|      * OSS内容 | ||||
|      */ | ||||
|     String SYS_OSS = "sys_oss#30d"; | ||||
|  | ||||
|     /** | ||||
|      * OSS配置 | ||||
|      */ | ||||
|     String SYS_OSS_CONFIG = "sys_oss_config"; | ||||
|  | ||||
|     /** | ||||
|      * 在线用户 | ||||
|      */ | ||||
|     String ONLINE_TOKEN = "online_tokens"; | ||||
|  | ||||
| } | ||||
| @@ -17,6 +17,11 @@ public interface Constants { | ||||
|      */ | ||||
|     String GBK = "GBK"; | ||||
|  | ||||
|     /** | ||||
|      * www主域 | ||||
|      */ | ||||
|     String WWW = "www."; | ||||
|  | ||||
|     /** | ||||
|      * http请求 | ||||
|      */ | ||||
| @@ -57,70 +62,15 @@ public interface Constants { | ||||
|      */ | ||||
|     String LOGIN_FAIL = "Error"; | ||||
|  | ||||
|     /** | ||||
|      * 验证码 redis key | ||||
|      */ | ||||
|     String CAPTCHA_CODE_KEY = "captcha_codes:"; | ||||
|  | ||||
|     /** | ||||
|      * 登录用户 redis key | ||||
|      */ | ||||
|     String LOGIN_TOKEN_KEY = "Authorization:login:token:"; | ||||
|  | ||||
|     /** | ||||
|      * 在线用户 redis key | ||||
|      */ | ||||
|     String ONLINE_TOKEN_KEY = "online_tokens:"; | ||||
|  | ||||
|     /** | ||||
|      * 防重提交 redis key | ||||
|      */ | ||||
|     String REPEAT_SUBMIT_KEY = "repeat_submit:"; | ||||
|  | ||||
|     /** | ||||
|      * 限流 redis key | ||||
|      */ | ||||
|     String RATE_LIMIT_KEY = "rate_limit:"; | ||||
|  | ||||
|     /** | ||||
|      * 验证码有效期(分钟) | ||||
|      */ | ||||
|     Integer CAPTCHA_EXPIRATION = 2; | ||||
|  | ||||
|     /** | ||||
|      * 登陆错误 redis key | ||||
|      */ | ||||
|     String LOGIN_ERROR = "login_error:"; | ||||
|  | ||||
|     /** | ||||
|      * 登录错误次数 | ||||
|      */ | ||||
|     Integer LOGIN_ERROR_NUMBER = 5; | ||||
|  | ||||
|     /** | ||||
|      * 登录错误限制时间(分钟) | ||||
|      */ | ||||
|     Integer LOGIN_ERROR_LIMIT_TIME = 10; | ||||
|  | ||||
|     /** | ||||
|      * 令牌 | ||||
|      */ | ||||
|     String TOKEN = "token"; | ||||
|  | ||||
|     /** | ||||
|      * 令牌前缀 | ||||
|      */ | ||||
|     String LOGIN_USER_KEY = "login_user_key"; | ||||
|  | ||||
|     /** | ||||
|      * 参数管理 cache key | ||||
|      */ | ||||
|     String SYS_CONFIG_KEY = "sys_config:"; | ||||
|  | ||||
|     /** | ||||
|      * 字典管理 cache key | ||||
|      */ | ||||
|     String SYS_DICT_KEY = "sys_dict:"; | ||||
|  | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -2,7 +2,6 @@ package com.ruoyi.common.core.domain; | ||||
|  | ||||
| import com.baomidou.mybatisplus.annotation.FieldFill; | ||||
| import com.baomidou.mybatisplus.annotation.TableField; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.Data; | ||||
|  | ||||
| import java.io.Serializable; | ||||
| @@ -24,42 +23,36 @@ public class BaseEntity implements Serializable { | ||||
|     /** | ||||
|      * 搜索值 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "搜索值") | ||||
|     @TableField(exist = false) | ||||
|     private String searchValue; | ||||
|  | ||||
|     /** | ||||
|      * 创建者 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "创建者") | ||||
|     @TableField(fill = FieldFill.INSERT) | ||||
|     private String createBy; | ||||
|  | ||||
|     /** | ||||
|      * 创建时间 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "创建时间") | ||||
|     @TableField(fill = FieldFill.INSERT) | ||||
|     private Date createTime; | ||||
|  | ||||
|     /** | ||||
|      * 更新者 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "更新者") | ||||
|     @TableField(fill = FieldFill.INSERT_UPDATE) | ||||
|     private String updateBy; | ||||
|  | ||||
|     /** | ||||
|      * 更新时间 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "更新时间") | ||||
|     @TableField(fill = FieldFill.INSERT_UPDATE) | ||||
|     private Date updateTime; | ||||
|  | ||||
|     /** | ||||
|      * 请求参数 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "请求参数") | ||||
|     @TableField(exist = false) | ||||
|     private Map<String, Object> params = new HashMap<>(); | ||||
|  | ||||
|   | ||||
| @@ -5,7 +5,6 @@ import com.baomidou.mybatisplus.core.metadata.OrderItem; | ||||
| import com.baomidou.mybatisplus.extension.plugins.pagination.Page; | ||||
| import com.ruoyi.common.utils.StringUtils; | ||||
| import com.ruoyi.common.utils.sql.SqlUtil; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.Data; | ||||
|  | ||||
| import java.io.Serializable; | ||||
| @@ -24,25 +23,21 @@ public class PageQuery implements Serializable { | ||||
|     /** | ||||
|      * 分页大小 | ||||
|      */ | ||||
|     @ApiModelProperty("分页大小") | ||||
|     private Integer pageSize; | ||||
|  | ||||
|     /** | ||||
|      * 当前页数 | ||||
|      */ | ||||
|     @ApiModelProperty("当前页数") | ||||
|     private Integer pageNum; | ||||
|  | ||||
|     /** | ||||
|      * 排序列 | ||||
|      */ | ||||
|     @ApiModelProperty("排序列") | ||||
|     private String orderByColumn; | ||||
|  | ||||
|     /** | ||||
|      * 排序的方向desc或者asc | ||||
|      */ | ||||
|     @ApiModelProperty(value = "排序的方向", example = "asc,desc") | ||||
|     private String isAsc; | ||||
|  | ||||
|     /** | ||||
|   | ||||
| @@ -1,7 +1,5 @@ | ||||
| package com.ruoyi.common.core.domain; | ||||
|  | ||||
| import io.swagger.annotations.ApiModel; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.Data; | ||||
| import lombok.NoArgsConstructor; | ||||
|  | ||||
| @@ -14,7 +12,6 @@ import java.io.Serializable; | ||||
|  */ | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| @ApiModel("请求响应对象") | ||||
| public class R<T> implements Serializable { | ||||
|     private static final long serialVersionUID = 1L; | ||||
|  | ||||
| @@ -28,13 +25,10 @@ public class R<T> implements Serializable { | ||||
|      */ | ||||
|     public static final int FAIL = 500; | ||||
|  | ||||
|     @ApiModelProperty("消息状态码") | ||||
|     private int code; | ||||
|  | ||||
|     @ApiModelProperty("消息内容") | ||||
|     private String msg; | ||||
|  | ||||
|     @ApiModelProperty("数据对象") | ||||
|     private T data; | ||||
|  | ||||
|     public static <T> R<T> ok() { | ||||
| @@ -81,4 +75,11 @@ public class R<T> implements Serializable { | ||||
|         return r; | ||||
|     } | ||||
|  | ||||
|     public Boolean isError() { | ||||
|         return !isSuccess(); | ||||
|     } | ||||
|  | ||||
|     public Boolean isSuccess() { | ||||
|         return R.SUCCESS == getCode(); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| package com.ruoyi.common.core.domain; | ||||
|  | ||||
| import com.baomidou.mybatisplus.annotation.TableField; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
|  | ||||
| @@ -24,20 +23,17 @@ public class TreeEntity<T> extends BaseEntity { | ||||
|      * 父菜单名称 | ||||
|      */ | ||||
|     @TableField(exist = false) | ||||
|     @ApiModelProperty(value = "父菜单名称") | ||||
|     private String parentName; | ||||
|  | ||||
|     /** | ||||
|      * 父菜单ID | ||||
|      */ | ||||
|     @ApiModelProperty(value = "父菜单ID") | ||||
|     private Long parentId; | ||||
|  | ||||
|     /** | ||||
|      * 子部门 | ||||
|      */ | ||||
|     @TableField(exist = false) | ||||
|     @ApiModelProperty(value = "子部门") | ||||
|     private List<T> children = new ArrayList<>(); | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -4,8 +4,6 @@ import com.baomidou.mybatisplus.annotation.TableId; | ||||
| import com.baomidou.mybatisplus.annotation.TableLogic; | ||||
| import com.baomidou.mybatisplus.annotation.TableName; | ||||
| import com.ruoyi.common.core.domain.TreeEntity; | ||||
| import io.swagger.annotations.ApiModel; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
|  | ||||
| @@ -23,21 +21,18 @@ import javax.validation.constraints.Size; | ||||
| @Data | ||||
| @EqualsAndHashCode(callSuper = true) | ||||
| @TableName("sys_dept") | ||||
| @ApiModel("部门业务对象") | ||||
| public class SysDept extends TreeEntity<SysDept> { | ||||
|     private static final long serialVersionUID = 1L; | ||||
|  | ||||
|     /** | ||||
|      * 部门ID | ||||
|      */ | ||||
|     @ApiModelProperty(value = "部门id") | ||||
|     @TableId(value = "dept_id") | ||||
|     private Long deptId; | ||||
|  | ||||
|     /** | ||||
|      * 部门名称 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "部门名称") | ||||
|     @NotBlank(message = "部门名称不能为空") | ||||
|     @Size(min = 0, max = 30, message = "部门名称长度不能超过30个字符") | ||||
|     private String deptName; | ||||
| @@ -45,27 +40,23 @@ public class SysDept extends TreeEntity<SysDept> { | ||||
|     /** | ||||
|      * 显示顺序 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "显示顺序") | ||||
|     @NotNull(message = "显示顺序不能为空") | ||||
|     private Integer orderNum; | ||||
|  | ||||
|     /** | ||||
|      * 负责人 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "负责人") | ||||
|     private String leader; | ||||
|  | ||||
|     /** | ||||
|      * 联系电话 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "联系电话") | ||||
|     @Size(min = 0, max = 11, message = "联系电话长度不能超过11个字符") | ||||
|     private String phone; | ||||
|  | ||||
|     /** | ||||
|      * 邮箱 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "邮箱") | ||||
|     @Email(message = "邮箱格式不正确") | ||||
|     @Size(min = 0, max = 50, message = "邮箱长度不能超过50个字符") | ||||
|     private String email; | ||||
| @@ -73,20 +64,17 @@ public class SysDept extends TreeEntity<SysDept> { | ||||
|     /** | ||||
|      * 部门状态:0正常,1停用 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "部门状态:0正常,1停用") | ||||
|     private String status; | ||||
|  | ||||
|     /** | ||||
|      * 删除标志(0代表存在 2代表删除) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "删除标志(0代表存在 2代表删除)") | ||||
|     @TableLogic | ||||
|     private String delFlag; | ||||
|  | ||||
|     /** | ||||
|      * 祖级列表 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "祖级列表") | ||||
|     private String ancestors; | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,8 +8,6 @@ import com.ruoyi.common.annotation.ExcelDictFormat; | ||||
| import com.ruoyi.common.constant.UserConstants; | ||||
| import com.ruoyi.common.convert.ExcelDictConvert; | ||||
| import com.ruoyi.common.core.domain.BaseEntity; | ||||
| import io.swagger.annotations.ApiModel; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
|  | ||||
| @@ -26,13 +24,11 @@ import javax.validation.constraints.Size; | ||||
| @EqualsAndHashCode(callSuper = true) | ||||
| @TableName("sys_dict_data") | ||||
| @ExcelIgnoreUnannotated | ||||
| @ApiModel("字典数据业务对象") | ||||
| public class SysDictData extends BaseEntity { | ||||
|  | ||||
|     /** | ||||
|      * 字典编码 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "字典编码") | ||||
|     @ExcelProperty(value = "字典编码") | ||||
|     @TableId(value = "dict_code") | ||||
|     private Long dictCode; | ||||
| @@ -40,14 +36,12 @@ public class SysDictData extends BaseEntity { | ||||
|     /** | ||||
|      * 字典排序 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "字典排序") | ||||
|     @ExcelProperty(value = "字典排序") | ||||
|     private Integer dictSort; | ||||
|  | ||||
|     /** | ||||
|      * 字典标签 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "字典标签") | ||||
|     @ExcelProperty(value = "字典标签") | ||||
|     @NotBlank(message = "字典标签不能为空") | ||||
|     @Size(min = 0, max = 100, message = "字典标签长度不能超过100个字符") | ||||
| @@ -56,7 +50,6 @@ public class SysDictData extends BaseEntity { | ||||
|     /** | ||||
|      * 字典键值 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "字典键值") | ||||
|     @ExcelProperty(value = "字典键值") | ||||
|     @NotBlank(message = "字典键值不能为空") | ||||
|     @Size(min = 0, max = 100, message = "字典键值长度不能超过100个字符") | ||||
| @@ -65,7 +58,6 @@ public class SysDictData extends BaseEntity { | ||||
|     /** | ||||
|      * 字典类型 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "字典类型") | ||||
|     @ExcelProperty(value = "字典类型") | ||||
|     @NotBlank(message = "字典类型不能为空") | ||||
|     @Size(min = 0, max = 100, message = "字典类型长度不能超过100个字符") | ||||
| @@ -74,20 +66,17 @@ public class SysDictData extends BaseEntity { | ||||
|     /** | ||||
|      * 样式属性(其他样式扩展) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "样式属性(其他样式扩展)") | ||||
|     @Size(min = 0, max = 100, message = "样式属性长度不能超过100个字符") | ||||
|     private String cssClass; | ||||
|  | ||||
|     /** | ||||
|      * 表格字典样式 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "表格字典样式") | ||||
|     private String listClass; | ||||
|  | ||||
|     /** | ||||
|      * 是否默认(Y是 N否) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "是否默认(Y是 N否)") | ||||
|     @ExcelProperty(value = "是否默认", converter = ExcelDictConvert.class) | ||||
|     @ExcelDictFormat(dictType = "sys_yes_no") | ||||
|     private String isDefault; | ||||
| @@ -95,7 +84,6 @@ public class SysDictData extends BaseEntity { | ||||
|     /** | ||||
|      * 状态(0正常 1停用) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "状态(0正常 1停用)") | ||||
|     @ExcelProperty(value = "状态", converter = ExcelDictConvert.class) | ||||
|     @ExcelDictFormat(dictType = "sys_normal_disable") | ||||
|     private String status; | ||||
| @@ -103,7 +91,6 @@ public class SysDictData extends BaseEntity { | ||||
|     /** | ||||
|      * 备注 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "备注") | ||||
|     private String remark; | ||||
|  | ||||
|     public boolean getDefault() { | ||||
|   | ||||
| @@ -7,11 +7,8 @@ import com.baomidou.mybatisplus.annotation.TableName; | ||||
| import com.ruoyi.common.annotation.ExcelDictFormat; | ||||
| import com.ruoyi.common.convert.ExcelDictConvert; | ||||
| import com.ruoyi.common.core.domain.BaseEntity; | ||||
| import io.swagger.annotations.ApiModel; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.experimental.Accessors; | ||||
|  | ||||
| import javax.validation.constraints.NotBlank; | ||||
| import javax.validation.constraints.Pattern; | ||||
| @@ -27,13 +24,11 @@ import javax.validation.constraints.Size; | ||||
| @EqualsAndHashCode(callSuper = true) | ||||
| @TableName("sys_dict_type") | ||||
| @ExcelIgnoreUnannotated | ||||
| @ApiModel("字典类型业务对象") | ||||
| public class SysDictType extends BaseEntity { | ||||
|  | ||||
|     /** | ||||
|      * 字典主键 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "字典主键") | ||||
|     @ExcelProperty(value = "字典主键") | ||||
|     @TableId(value = "dict_id") | ||||
|     private Long dictId; | ||||
| @@ -41,7 +36,6 @@ public class SysDictType extends BaseEntity { | ||||
|     /** | ||||
|      * 字典名称 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "字典名称") | ||||
|     @ExcelProperty(value = "字典名称") | ||||
|     @NotBlank(message = "字典名称不能为空") | ||||
|     @Size(min = 0, max = 100, message = "字典类型名称长度不能超过100个字符") | ||||
| @@ -50,7 +44,6 @@ public class SysDictType extends BaseEntity { | ||||
|     /** | ||||
|      * 字典类型 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "字典类型") | ||||
|     @ExcelProperty(value = "字典类型") | ||||
|     @NotBlank(message = "字典类型不能为空") | ||||
|     @Size(min = 0, max = 100, message = "字典类型类型长度不能超过100个字符") | ||||
| @@ -60,7 +53,6 @@ public class SysDictType extends BaseEntity { | ||||
|     /** | ||||
|      * 状态(0正常 1停用) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "状态(0正常 1停用)") | ||||
|     @ExcelProperty(value = "状态", converter = ExcelDictConvert.class) | ||||
|     @ExcelDictFormat(dictType = "sys_normal_disable") | ||||
|     private String status; | ||||
| @@ -68,7 +60,6 @@ public class SysDictType extends BaseEntity { | ||||
|     /** | ||||
|      * 备注 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "备注") | ||||
|     private String remark; | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -4,8 +4,6 @@ import com.baomidou.mybatisplus.annotation.TableId; | ||||
| import com.baomidou.mybatisplus.annotation.TableName; | ||||
| import com.fasterxml.jackson.annotation.JsonInclude; | ||||
| import com.ruoyi.common.core.domain.TreeEntity; | ||||
| import io.swagger.annotations.ApiModel; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
|  | ||||
| @@ -22,20 +20,17 @@ import javax.validation.constraints.Size; | ||||
| @Data | ||||
| @EqualsAndHashCode(callSuper = true) | ||||
| @TableName("sys_menu") | ||||
| @ApiModel("菜单权限业务对象") | ||||
| public class SysMenu extends TreeEntity<SysMenu> { | ||||
|  | ||||
|     /** | ||||
|      * 菜单ID | ||||
|      */ | ||||
|     @ApiModelProperty(value = "菜单ID") | ||||
|     @TableId(value = "menu_id") | ||||
|     private Long menuId; | ||||
|  | ||||
|     /** | ||||
|      * 菜单名称 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "菜单名称") | ||||
|     @NotBlank(message = "菜单名称不能为空") | ||||
|     @Size(min = 0, max = 50, message = "菜单名称长度不能超过50个字符") | ||||
|     private String menuName; | ||||
| @@ -43,65 +38,55 @@ public class SysMenu extends TreeEntity<SysMenu> { | ||||
|     /** | ||||
|      * 显示顺序 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "显示顺序") | ||||
|     @NotNull(message = "显示顺序不能为空") | ||||
|     private Integer orderNum; | ||||
|  | ||||
|     /** | ||||
|      * 路由地址 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "路由地址") | ||||
|     @Size(min = 0, max = 200, message = "路由地址不能超过200个字符") | ||||
|     private String path; | ||||
|  | ||||
|     /** | ||||
|      * 组件路径 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "组件路径") | ||||
|     @Size(min = 0, max = 200, message = "组件路径不能超过255个字符") | ||||
|     private String component; | ||||
|  | ||||
|     /** | ||||
|      * 路由参数 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "路由参数") | ||||
|     private String queryParam; | ||||
|  | ||||
|     /** | ||||
|      * 是否为外链(0是 1否) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "是否为外链(0是 1否)") | ||||
|     private String isFrame; | ||||
|  | ||||
|     /** | ||||
|      * 是否缓存(0缓存 1不缓存) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "是否缓存(0缓存 1不缓存)") | ||||
|     private String isCache; | ||||
|  | ||||
|     /** | ||||
|      * 类型(M目录 C菜单 F按钮) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "类型(M目录 C菜单 F按钮)") | ||||
|     @NotBlank(message = "菜单类型不能为空") | ||||
|     private String menuType; | ||||
|  | ||||
|     /** | ||||
|      * 显示状态(0显示 1隐藏) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "显示状态(0显示 1隐藏)") | ||||
|     private String visible; | ||||
|  | ||||
|     /** | ||||
|      * 菜单状态(0显示 1隐藏) | ||||
|      * 菜单状态(0正常 1停用) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "菜单状态(0显示 1隐藏)") | ||||
|     private String status; | ||||
|  | ||||
|     /** | ||||
|      * 权限字符串 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "权限字符串") | ||||
|     @JsonInclude(JsonInclude.Include.NON_NULL) | ||||
|     @Size(min = 0, max = 100, message = "权限标识长度不能超过100个字符") | ||||
|     private String perms; | ||||
| @@ -109,13 +94,11 @@ public class SysMenu extends TreeEntity<SysMenu> { | ||||
|     /** | ||||
|      * 菜单图标 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "菜单图标") | ||||
|     private String icon; | ||||
|  | ||||
|     /** | ||||
|      * 备注 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "备注") | ||||
|     private String remark; | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -10,7 +10,6 @@ import com.ruoyi.common.annotation.ExcelDictFormat; | ||||
| import com.ruoyi.common.constant.UserConstants; | ||||
| import com.ruoyi.common.convert.ExcelDictConvert; | ||||
| import com.ruoyi.common.core.domain.BaseEntity; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.NoArgsConstructor; | ||||
| @@ -18,6 +17,7 @@ import lombok.NoArgsConstructor; | ||||
| import javax.validation.constraints.NotBlank; | ||||
| import javax.validation.constraints.NotNull; | ||||
| import javax.validation.constraints.Size; | ||||
| import java.util.Set; | ||||
|  | ||||
| /** | ||||
|  * 角色表 sys_role | ||||
| @@ -35,7 +35,6 @@ public class SysRole extends BaseEntity { | ||||
|     /** | ||||
|      * 角色ID | ||||
|      */ | ||||
|     @ApiModelProperty(value = "角色ID") | ||||
|     @ExcelProperty(value = "角色序号") | ||||
|     @TableId(value = "role_id") | ||||
|     private Long roleId; | ||||
| @@ -43,7 +42,6 @@ public class SysRole extends BaseEntity { | ||||
|     /** | ||||
|      * 角色名称 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "角色名称") | ||||
|     @ExcelProperty(value = "角色名称") | ||||
|     @NotBlank(message = "角色名称不能为空") | ||||
|     @Size(min = 0, max = 30, message = "角色名称长度不能超过30个字符") | ||||
| @@ -52,7 +50,6 @@ public class SysRole extends BaseEntity { | ||||
|     /** | ||||
|      * 角色权限 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "角色权限") | ||||
|     @ExcelProperty(value = "角色权限") | ||||
|     @NotBlank(message = "权限字符不能为空") | ||||
|     @Size(min = 0, max = 100, message = "权限字符长度不能超过100个字符") | ||||
| @@ -61,7 +58,6 @@ public class SysRole extends BaseEntity { | ||||
|     /** | ||||
|      * 角色排序 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "角色排序") | ||||
|     @ExcelProperty(value = "角色排序") | ||||
|     @NotNull(message = "显示顺序不能为空") | ||||
|     private Integer roleSort; | ||||
| @@ -69,7 +65,6 @@ public class SysRole extends BaseEntity { | ||||
|     /** | ||||
|      * 数据范围(1:所有数据权限;2:自定义数据权限;3:本部门数据权限;4:本部门及以下数据权限;5:仅本人数据权限) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "数据范围(1:所有数据权限;2:自定义数据权限;3:本部门数据权限;4:本部门及以下数据权限;5:仅本人数据权限)") | ||||
|     @ExcelProperty(value = "数据范围", converter = ExcelDictConvert.class) | ||||
|     @ExcelDictFormat(readConverterExp = "1=所有数据权限,2=自定义数据权限,3=本部门数据权限,4=本部门及以下数据权限,5=仅本人数据权限") | ||||
|     private String dataScope; | ||||
| @@ -77,62 +72,59 @@ public class SysRole extends BaseEntity { | ||||
|     /** | ||||
|      * 菜单树选择项是否关联显示( 0:父子不互相关联显示 1:父子互相关联显示) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "菜单树选择项是否关联显示( 0:父子不互相关联显示 1:父子互相关联显示)") | ||||
|     private Boolean menuCheckStrictly; | ||||
|  | ||||
|     /** | ||||
|      * 部门树选择项是否关联显示(0:父子不互相关联显示 1:父子互相关联显示 ) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "部门树选择项是否关联显示(0:父子不互相关联显示 1:父子互相关联显示 )") | ||||
|     private Boolean deptCheckStrictly; | ||||
|  | ||||
|     /** | ||||
|      * 角色状态(0正常 1停用) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "角色状态(0正常 1停用)") | ||||
|     @ExcelProperty(value = "角色状态", converter = ExcelDictConvert.class) | ||||
|     @ExcelDictFormat(dictType = "sys_common_status") | ||||
|     @ExcelDictFormat(dictType = "sys_normal_disable") | ||||
|     private String status; | ||||
|  | ||||
|     /** | ||||
|      * 删除标志(0代表存在 2代表删除) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "删除标志(0代表存在 2代表删除)") | ||||
|     @TableLogic | ||||
|     private String delFlag; | ||||
|  | ||||
|     /** | ||||
|      * 备注 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "备注") | ||||
|     private String remark; | ||||
|  | ||||
|     /** | ||||
|      * 用户是否存在此角色标识 默认不存在 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "用户是否存在此角色标识 默认不存在") | ||||
|     @TableField(exist = false) | ||||
|     private boolean flag = false; | ||||
|  | ||||
|     /** | ||||
|      * 菜单组 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "菜单组") | ||||
|     @TableField(exist = false) | ||||
|     private Long[] menuIds; | ||||
|  | ||||
|     /** | ||||
|      * 部门组(数据权限) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "部门组(数据权限)") | ||||
|     @TableField(exist = false) | ||||
|     private Long[] deptIds; | ||||
|  | ||||
|     /** | ||||
|      * 角色菜单权限 | ||||
|      */ | ||||
|     @TableField(exist = false) | ||||
|     private Set<String> permissions; | ||||
|  | ||||
|     public SysRole(Long roleId) { | ||||
|         this.roleId = roleId; | ||||
|     } | ||||
|  | ||||
|     @ApiModelProperty(value = "是否管理员") | ||||
|     public boolean isAdmin() { | ||||
|         return UserConstants.ADMIN_ID.equals(this.roleId); | ||||
|     } | ||||
|   | ||||
| @@ -1,15 +1,11 @@ | ||||
| package com.ruoyi.common.core.domain.entity; | ||||
|  | ||||
| import com.baomidou.mybatisplus.annotation.*; | ||||
| import com.fasterxml.jackson.annotation.JsonIgnore; | ||||
| import com.fasterxml.jackson.annotation.JsonProperty; | ||||
| import com.ruoyi.common.annotation.Sensitive; | ||||
| import com.ruoyi.common.constant.UserConstants; | ||||
| import com.ruoyi.common.core.domain.BaseEntity; | ||||
| import com.ruoyi.common.enums.SensitiveStrategy; | ||||
| import com.ruoyi.common.xss.Xss; | ||||
| import io.swagger.annotations.ApiModel; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
| import lombok.NoArgsConstructor; | ||||
| @@ -30,26 +26,22 @@ import java.util.List; | ||||
| @NoArgsConstructor | ||||
| @EqualsAndHashCode(callSuper = true) | ||||
| @TableName("sys_user") | ||||
| @ApiModel("用户信息业务对象") | ||||
| public class SysUser extends BaseEntity { | ||||
|  | ||||
|     /** | ||||
|      * 用户ID | ||||
|      */ | ||||
|     @ApiModelProperty(value = "用户ID") | ||||
|     @TableId(value = "user_id") | ||||
|     private Long userId; | ||||
|  | ||||
|     /** | ||||
|      * 部门ID | ||||
|      */ | ||||
|     @ApiModelProperty(value = "部门ID") | ||||
|     private Long deptId; | ||||
|  | ||||
|     /** | ||||
|      * 用户账号 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "用户账号") | ||||
|     @Xss(message = "用户账号不能包含脚本字符") | ||||
|     @NotBlank(message = "用户账号不能为空") | ||||
|     @Size(min = 0, max = 30, message = "用户账号长度不能超过30个字符") | ||||
| @@ -58,7 +50,6 @@ public class SysUser extends BaseEntity { | ||||
|     /** | ||||
|      * 用户昵称 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "用户昵称") | ||||
|     @Xss(message = "用户昵称不能包含脚本字符") | ||||
|     @Size(min = 0, max = 30, message = "用户昵称长度不能超过30个字符") | ||||
|     private String nickName; | ||||
| @@ -66,14 +57,12 @@ public class SysUser extends BaseEntity { | ||||
|     /** | ||||
|      * 用户类型(sys_user系统用户) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "用户类型") | ||||
|     private String userType; | ||||
|  | ||||
|     /** | ||||
|      * 用户邮箱 | ||||
|      */ | ||||
|     @Sensitive(strategy = SensitiveStrategy.EMAIL) | ||||
|     @ApiModelProperty(value = "用户邮箱") | ||||
|     @Email(message = "邮箱格式不正确") | ||||
|     @Size(min = 0, max = 50, message = "邮箱长度不能超过50个字符") | ||||
|     private String email; | ||||
| @@ -82,25 +71,21 @@ public class SysUser extends BaseEntity { | ||||
|      * 手机号码 | ||||
|      */ | ||||
|     @Sensitive(strategy = SensitiveStrategy.PHONE) | ||||
|     @ApiModelProperty(value = "手机号码") | ||||
|     private String phonenumber; | ||||
|  | ||||
|     /** | ||||
|      * 用户性别 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "用户性别") | ||||
|     private String sex; | ||||
|  | ||||
|     /** | ||||
|      * 用户头像 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "用户头像") | ||||
|     private String avatar; | ||||
|  | ||||
|     /** | ||||
|      * 密码 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "密码") | ||||
|     @TableField( | ||||
|         insertStrategy = FieldStrategy.NOT_EMPTY, | ||||
|         updateStrategy = FieldStrategy.NOT_EMPTY, | ||||
| @@ -108,75 +93,59 @@ public class SysUser extends BaseEntity { | ||||
|     ) | ||||
|     private String password; | ||||
|  | ||||
|     @JsonIgnore | ||||
|     @JsonProperty | ||||
|     public String getPassword() { | ||||
|         return password; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 帐号状态(0正常 1停用) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "帐号状态(0正常 1停用)") | ||||
|     private String status; | ||||
|  | ||||
|     /** | ||||
|      * 删除标志(0代表存在 2代表删除) | ||||
|      */ | ||||
|     @ApiModelProperty(value = "删除标志(0代表存在 2代表删除)") | ||||
|     @TableLogic | ||||
|     private String delFlag; | ||||
|  | ||||
|     /** | ||||
|      * 最后登录IP | ||||
|      */ | ||||
|     @ApiModelProperty(value = "最后登录IP") | ||||
|     private String loginIp; | ||||
|  | ||||
|     /** | ||||
|      * 最后登录时间 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "最后登录时间") | ||||
|     private Date loginDate; | ||||
|  | ||||
|     /** | ||||
|      * 备注 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "备注") | ||||
|     private String remark; | ||||
|  | ||||
|     /** | ||||
|      * 部门对象 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "部门对象") | ||||
|     @TableField(exist = false) | ||||
|     private SysDept dept; | ||||
|  | ||||
|     /** | ||||
|      * 角色对象 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "角色对象") | ||||
|     @TableField(exist = false) | ||||
|     private List<SysRole> roles; | ||||
|  | ||||
|     /** | ||||
|      * 角色组 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "角色组") | ||||
|     @TableField(exist = false) | ||||
|     private Long[] roleIds; | ||||
|  | ||||
|     /** | ||||
|      * 岗位组 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "岗位组") | ||||
|     @TableField(exist = false) | ||||
|     private Long[] postIds; | ||||
|  | ||||
|     /** | ||||
|      * 数据权限 当前角色ID | ||||
|      */ | ||||
|     @ApiModelProperty(value = "角色ID") | ||||
|     @TableField(exist = false) | ||||
|     private Long roleId; | ||||
|  | ||||
| @@ -184,7 +153,6 @@ public class SysUser extends BaseEntity { | ||||
|         this.userId = userId; | ||||
|     } | ||||
|  | ||||
|     @ApiModelProperty(value = "是否管理员") | ||||
|     public boolean isAdmin() { | ||||
|         return UserConstants.ADMIN_ID.equals(this.userId); | ||||
|     } | ||||
|   | ||||
| @@ -1,8 +1,6 @@ | ||||
| package com.ruoyi.common.core.domain.model; | ||||
|  | ||||
| import com.ruoyi.common.constant.UserConstants; | ||||
| import io.swagger.annotations.ApiModel; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.Data; | ||||
| import org.hibernate.validator.constraints.Length; | ||||
|  | ||||
| @@ -15,7 +13,6 @@ import javax.validation.constraints.NotBlank; | ||||
|  */ | ||||
|  | ||||
| @Data | ||||
| @ApiModel("用户登录对象") | ||||
| public class LoginBody { | ||||
|  | ||||
|     /** | ||||
| @@ -23,7 +20,6 @@ public class LoginBody { | ||||
|      */ | ||||
|     @NotBlank(message = "{user.username.not.blank}") | ||||
|     @Length(min = UserConstants.USERNAME_MIN_LENGTH, max = UserConstants.USERNAME_MAX_LENGTH, message = "{user.username.length.valid}") | ||||
|     @ApiModelProperty(value = "用户名") | ||||
|     private String username; | ||||
|  | ||||
|     /** | ||||
| @@ -31,19 +27,16 @@ public class LoginBody { | ||||
|      */ | ||||
|     @NotBlank(message = "{user.password.not.blank}") | ||||
|     @Length(min = UserConstants.PASSWORD_MIN_LENGTH, max = UserConstants.PASSWORD_MAX_LENGTH, message = "{user.password.length.valid}") | ||||
|     @ApiModelProperty(value = "用户密码") | ||||
|     private String password; | ||||
|  | ||||
|     /** | ||||
|      * 验证码 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "验证码") | ||||
|     private String code; | ||||
|  | ||||
|     /** | ||||
|      * 唯一标识 | ||||
|      */ | ||||
|     @ApiModelProperty(value = "唯一标识") | ||||
|     private String uuid; | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -105,6 +105,12 @@ public class LoginUser implements Serializable { | ||||
|      * 获取登录id | ||||
|      */ | ||||
|     public String getLoginId() { | ||||
|         if (userType == null) { | ||||
|             throw new IllegalArgumentException("用户类型不能为空"); | ||||
|         } | ||||
|         if (userId == null) { | ||||
|             throw new IllegalArgumentException("用户ID不能为空"); | ||||
|         } | ||||
|         return userType + LoginHelper.JOIN_CODE + userId; | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -1,7 +1,5 @@ | ||||
| package com.ruoyi.common.core.domain.model; | ||||
|  | ||||
| import io.swagger.annotations.ApiModel; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
|  | ||||
| @@ -12,10 +10,8 @@ import lombok.EqualsAndHashCode; | ||||
|  */ | ||||
| @Data | ||||
| @EqualsAndHashCode(callSuper = true) | ||||
| @ApiModel("用户注册对象") | ||||
| public class RegisterBody extends LoginBody { | ||||
|  | ||||
|     @ApiModelProperty(value = "用户类型") | ||||
|     private String userType; | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,7 +1,5 @@ | ||||
| package com.ruoyi.common.core.domain.model; | ||||
|  | ||||
| import io.swagger.annotations.ApiModel; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.Data; | ||||
|  | ||||
| import javax.validation.constraints.NotBlank; | ||||
| @@ -13,21 +11,18 @@ import javax.validation.constraints.NotBlank; | ||||
|  */ | ||||
|  | ||||
| @Data | ||||
| @ApiModel("短信登录对象") | ||||
| public class SmsLoginBody { | ||||
|  | ||||
|     /** | ||||
|      * 用户名 | ||||
|      */ | ||||
|     @NotBlank(message = "{user.phonenumber.not.blank}") | ||||
|     @ApiModelProperty(value = "用户手机号") | ||||
|     private String phonenumber; | ||||
|  | ||||
|     /** | ||||
|      * 用户密码 | ||||
|      */ | ||||
|     @NotBlank(message = "{sms.code.not.blank}") | ||||
|     @ApiModelProperty(value = "短信验证码") | ||||
|     private String smsCode; | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -149,7 +149,7 @@ public interface BaseMapperPlus<M, T, V> extends BaseMapper<T> { | ||||
|         return BeanCopyUtils.copy(obj, voClass); | ||||
|     } | ||||
|  | ||||
|     default List<V> selectVoById(Collection<? extends Serializable> idList) { | ||||
|     default List<V> selectVoBatchIds(Collection<? extends Serializable> idList) { | ||||
|         return selectVoBatchIds(idList, this.currentVoClass()); | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -2,11 +2,8 @@ package com.ruoyi.common.core.page; | ||||
|  | ||||
| import cn.hutool.http.HttpStatus; | ||||
| import com.baomidou.mybatisplus.core.metadata.IPage; | ||||
| 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; | ||||
| @@ -19,32 +16,27 @@ import java.util.List; | ||||
|  | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| @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; | ||||
|  | ||||
|     /** | ||||
|   | ||||
| @@ -0,0 +1,114 @@ | ||||
| package com.ruoyi.common.excel; | ||||
|  | ||||
| import com.alibaba.excel.metadata.Head; | ||||
| import com.alibaba.excel.write.merge.AbstractMergeStrategy; | ||||
| import com.ruoyi.common.annotation.CellMerge; | ||||
| import lombok.AllArgsConstructor; | ||||
| import lombok.Data; | ||||
| import lombok.SneakyThrows; | ||||
| import lombok.extern.slf4j.Slf4j; | ||||
| import org.apache.commons.collections4.CollectionUtils; | ||||
| import org.apache.poi.ss.usermodel.Cell; | ||||
| import org.apache.poi.ss.usermodel.Sheet; | ||||
| import org.apache.poi.ss.util.CellRangeAddress; | ||||
|  | ||||
| import java.lang.reflect.Field; | ||||
| import java.lang.reflect.Method; | ||||
| import java.util.ArrayList; | ||||
| import java.util.HashMap; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * 列值重复合并策略 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @AllArgsConstructor | ||||
| @Slf4j | ||||
| public class CellMergeStrategy extends AbstractMergeStrategy { | ||||
|  | ||||
| 	private List<?> list; | ||||
| 	private boolean hasTitle; | ||||
|  | ||||
| 	@Override | ||||
| 	protected void merge(Sheet sheet, Cell cell, Head head, Integer relativeRowIndex) { | ||||
| 		List<CellRangeAddress> cellList = handle(list, hasTitle); | ||||
| 		// judge the list is not null | ||||
| 		if (CollectionUtils.isNotEmpty(cellList)) { | ||||
| 			// the judge is necessary | ||||
| 			if (cell.getRowIndex() == 1 && cell.getColumnIndex() == 0) { | ||||
| 				for (CellRangeAddress item : cellList) { | ||||
| 					sheet.addMergedRegion(item); | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@SneakyThrows | ||||
| 	private static List<CellRangeAddress> handle(List<?> list, boolean hasTitle) { | ||||
| 		List<CellRangeAddress> cellList = new ArrayList<>(); | ||||
| 		if (CollectionUtils.isEmpty(list)) { | ||||
| 			return cellList; | ||||
| 		} | ||||
| 		Class<?> clazz = list.get(0).getClass(); | ||||
| 		Field[] fields = clazz.getDeclaredFields(); | ||||
| 		// 有注解的字段 | ||||
| 		List<Field> mergeFields = new ArrayList<>(); | ||||
| 		List<Integer> mergeFieldsIndex = new ArrayList<>(); | ||||
| 		for (int i = 0; i < fields.length; i++) { | ||||
| 			Field field = fields[i]; | ||||
| 			if (field.isAnnotationPresent(CellMerge.class)) { | ||||
| 				CellMerge cm = field.getAnnotation(CellMerge.class); | ||||
| 				mergeFields.add(field); | ||||
| 				mergeFieldsIndex.add(cm.index() == -1 ? i : cm.index()); | ||||
| 			} | ||||
| 		} | ||||
| 		// 行合并开始下标 | ||||
| 		int rowIndex = hasTitle ? 1 : 0; | ||||
| 		Map<Field, RepeatCell> map = new HashMap<>(); | ||||
| 		// 生成两两合并单元格 | ||||
| 		for (int i = 0; i < list.size(); i++) { | ||||
| 			for (int j = 0; j < mergeFields.size(); j++) { | ||||
| 				Field field = mergeFields.get(j); | ||||
| 				String name = field.getName(); | ||||
| 				String methodName = "get" + name.substring(0, 1).toUpperCase() + name.substring(1); | ||||
| 				Method readMethod = clazz.getMethod(methodName); | ||||
| 				Object val = readMethod.invoke(list.get(i)); | ||||
|  | ||||
| 				int colNum = mergeFieldsIndex.get(j); | ||||
| 				if (!map.containsKey(field)) { | ||||
| 					map.put(field, new RepeatCell(val, i)); | ||||
| 				} else { | ||||
| 					RepeatCell repeatCell = map.get(field); | ||||
| 					Object cellValue = repeatCell.getValue(); | ||||
| 					if (cellValue == null || "".equals(cellValue)) { | ||||
| 						// 空值跳过不合并 | ||||
| 						continue; | ||||
| 					} | ||||
| 					if (cellValue != val) { | ||||
| 						if (i - repeatCell.getCurrent() > 1) { | ||||
| 							cellList.add(new CellRangeAddress(repeatCell.getCurrent() + rowIndex, i + rowIndex - 1, colNum, colNum)); | ||||
| 						} | ||||
| 						map.put(field, new RepeatCell(val, i)); | ||||
| 					} else if (i == list.size() - 1) { | ||||
| 						if (i > repeatCell.getCurrent()) { | ||||
| 							cellList.add(new CellRangeAddress(repeatCell.getCurrent() + rowIndex, i + rowIndex, colNum, colNum)); | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		return cellList; | ||||
| 	} | ||||
|  | ||||
| 	@Data | ||||
| 	@AllArgsConstructor | ||||
| 	static class RepeatCell { | ||||
|  | ||||
| 		private Object value; | ||||
|  | ||||
| 		private int current; | ||||
|  | ||||
| 	} | ||||
| } | ||||
| @@ -6,6 +6,7 @@ import com.alibaba.excel.event.AnalysisEventListener; | ||||
| import com.alibaba.excel.exception.ExcelAnalysisException; | ||||
| import com.alibaba.excel.exception.ExcelDataConvertException; | ||||
| import com.ruoyi.common.utils.JsonUtils; | ||||
| import com.ruoyi.common.utils.StreamUtils; | ||||
| import com.ruoyi.common.utils.ValidatorUtils; | ||||
| import lombok.NoArgsConstructor; | ||||
| import lombok.extern.slf4j.Slf4j; | ||||
| @@ -14,7 +15,6 @@ import javax.validation.ConstraintViolation; | ||||
| import javax.validation.ConstraintViolationException; | ||||
| import java.util.Map; | ||||
| import java.util.Set; | ||||
| import java.util.stream.Collectors; | ||||
|  | ||||
| /** | ||||
|  * Excel 导入监听 | ||||
| @@ -69,9 +69,7 @@ public class DefaultExcelListener<T> extends AnalysisEventListener<T> implements | ||||
|         if (exception instanceof ConstraintViolationException) { | ||||
|             ConstraintViolationException constraintViolationException = (ConstraintViolationException) exception; | ||||
|             Set<ConstraintViolation<?>> constraintViolations = constraintViolationException.getConstraintViolations(); | ||||
|             String constraintViolationsMsg = constraintViolations.stream() | ||||
|                 .map(ConstraintViolation::getMessage) | ||||
|                 .collect(Collectors.joining(", ")); | ||||
|             String constraintViolationsMsg = StreamUtils.join(constraintViolations, ConstraintViolation::getMessage, ", "); | ||||
|             errMsg = StrUtil.format("第{}行数据校验异常: {}", context.readRowHolder().getRowIndex() + 1, constraintViolationsMsg); | ||||
|             if (log.isDebugEnabled()) { | ||||
|                 log.error(errMsg); | ||||
|   | ||||
| @@ -0,0 +1,16 @@ | ||||
| package com.ruoyi.common.exception.user; | ||||
|  | ||||
| /** | ||||
|  * 用户错误最大次数异常类 | ||||
|  * | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| public class UserPasswordRetryLimitExceedException extends UserException { | ||||
|  | ||||
|     private static final long serialVersionUID = 1L; | ||||
|  | ||||
|     public UserPasswordRetryLimitExceedException(int retryLimitCount, int lockTime) { | ||||
|         super("user.password.retry.limit.exceed", retryLimitCount, lockTime); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,6 +1,7 @@ | ||||
| package com.ruoyi.common.filter; | ||||
|  | ||||
| import cn.hutool.core.io.IoUtil; | ||||
| import com.ruoyi.common.constant.Constants; | ||||
|  | ||||
| import javax.servlet.ReadListener; | ||||
| import javax.servlet.ServletInputStream; | ||||
| @@ -23,8 +24,8 @@ public class RepeatedlyRequestWrapper extends HttpServletRequestWrapper { | ||||
|  | ||||
|     public RepeatedlyRequestWrapper(HttpServletRequest request, ServletResponse response) throws IOException { | ||||
|         super(request); | ||||
|         request.setCharacterEncoding("UTF-8"); | ||||
|         response.setCharacterEncoding("UTF-8"); | ||||
|         request.setCharacterEncoding(Constants.UTF8); | ||||
|         response.setCharacterEncoding(Constants.UTF8); | ||||
|  | ||||
|         body = IoUtil.readUtf8(request.getInputStream()).getBytes(StandardCharsets.UTF_8); | ||||
|     } | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| package com.ruoyi.common.filter; | ||||
|  | ||||
| import com.ruoyi.common.enums.HttpMethod; | ||||
| import com.ruoyi.common.utils.StringUtils; | ||||
|  | ||||
| import javax.servlet.*; | ||||
| @@ -48,7 +49,7 @@ public class XssFilter implements Filter { | ||||
|         String url = request.getServletPath(); | ||||
|         String method = request.getMethod(); | ||||
|         // GET DELETE 不过滤 | ||||
|         if (method == null || method.matches("GET") || method.matches("DELETE")) { | ||||
|         if (method == null || HttpMethod.GET.matches(method) || HttpMethod.DELETE.matches(method)) { | ||||
|             return true; | ||||
|         } | ||||
|         return StringUtils.matches(url, excludes); | ||||
|   | ||||
| @@ -89,7 +89,6 @@ public class XssHttpServletRequestWrapper extends HttpServletRequestWrapper { | ||||
|     /** | ||||
|      * 是否是Json请求 | ||||
|      * | ||||
|      * @param request | ||||
|      */ | ||||
|     public boolean isJsonRequest() { | ||||
|         String header = super.getHeader(HttpHeaders.CONTENT_TYPE); | ||||
|   | ||||
| @@ -1,11 +1,11 @@ | ||||
| package com.ruoyi.common.helper; | ||||
|  | ||||
| import cn.dev33.satoken.context.SaHolder; | ||||
| import cn.dev33.satoken.context.model.SaStorage; | ||||
| import cn.hutool.core.util.ObjectUtil; | ||||
| import com.ruoyi.common.utils.ServletUtils; | ||||
| import lombok.AccessLevel; | ||||
| import lombok.NoArgsConstructor; | ||||
|  | ||||
| import javax.servlet.http.HttpServletRequest; | ||||
| import java.util.HashMap; | ||||
| import java.util.Map; | ||||
|  | ||||
| @@ -33,11 +33,11 @@ public class DataPermissionHelper { | ||||
|     } | ||||
|  | ||||
|     public static Map<String, Object> getContext() { | ||||
|         HttpServletRequest request = ServletUtils.getRequest(); | ||||
|         Object attribute = request.getAttribute(DATA_PERMISSION_KEY); | ||||
|         SaStorage saStorage = SaHolder.getStorage(); | ||||
|         Object attribute = saStorage.get(DATA_PERMISSION_KEY); | ||||
|         if (ObjectUtil.isNull(attribute)) { | ||||
|             request.setAttribute(DATA_PERMISSION_KEY, new HashMap<>()); | ||||
|             attribute = request.getAttribute(DATA_PERMISSION_KEY); | ||||
|             saStorage.set(DATA_PERMISSION_KEY, new HashMap<>()); | ||||
|             attribute = saStorage.get(DATA_PERMISSION_KEY); | ||||
|         } | ||||
|         if (attribute instanceof Map) { | ||||
|             return (Map<String, Object>) attribute; | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| package com.ruoyi.common.helper; | ||||
|  | ||||
| import cn.dev33.satoken.context.SaHolder; | ||||
| import cn.dev33.satoken.stp.StpUtil; | ||||
| import cn.hutool.core.util.ObjectUtil; | ||||
| import com.ruoyi.common.constant.UserConstants; | ||||
| @@ -29,15 +30,13 @@ public class LoginHelper { | ||||
|     public static final String JOIN_CODE = ":"; | ||||
|     public static final String LOGIN_USER_KEY = "loginUser"; | ||||
|  | ||||
|     private static final ThreadLocal<LoginUser> LOGIN_CACHE = new ThreadLocal<>(); | ||||
|  | ||||
|     /** | ||||
|      * 登录系统 | ||||
|      * | ||||
|      * @param loginUser 登录用户信息 | ||||
|      */ | ||||
|     public static void login(LoginUser loginUser) { | ||||
|         LOGIN_CACHE.set(loginUser); | ||||
|         SaHolder.getStorage().set(LOGIN_USER_KEY, loginUser); | ||||
|         StpUtil.login(loginUser.getLoginId()); | ||||
|         setLoginUser(loginUser); | ||||
|     } | ||||
| @@ -49,7 +48,7 @@ public class LoginHelper { | ||||
|      * @param loginUser 登录用户信息 | ||||
|      */ | ||||
|     public static void loginByDevice(LoginUser loginUser, DeviceType deviceType) { | ||||
|         LOGIN_CACHE.set(loginUser); | ||||
|         SaHolder.getStorage().set(LOGIN_USER_KEY, loginUser); | ||||
|         StpUtil.login(loginUser.getLoginId(), deviceType.getDevice()); | ||||
|         setLoginUser(loginUser); | ||||
|     } | ||||
| @@ -65,18 +64,13 @@ public class LoginHelper { | ||||
|      * 获取用户(多级缓存) | ||||
|      */ | ||||
|     public static LoginUser getLoginUser() { | ||||
|         LoginUser loginUser = LOGIN_CACHE.get(); | ||||
|         LoginUser loginUser = (LoginUser) SaHolder.getStorage().get(LOGIN_USER_KEY); | ||||
|         if (loginUser != null) { | ||||
|             return loginUser; | ||||
|         } | ||||
|         return (LoginUser) StpUtil.getTokenSession().get(LOGIN_USER_KEY); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 清除一级缓存 防止内存问题 | ||||
|      */ | ||||
|     public static void clearCache() { | ||||
|         LOGIN_CACHE.remove(); | ||||
|         loginUser = (LoginUser) StpUtil.getTokenSession().get(LOGIN_USER_KEY); | ||||
|         SaHolder.getStorage().set(LOGIN_USER_KEY, loginUser); | ||||
|         return loginUser; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| package com.ruoyi.common.jackson; | ||||
|  | ||||
| import cn.hutool.core.util.ObjectUtil; | ||||
| import com.fasterxml.jackson.core.JsonGenerator; | ||||
| import com.fasterxml.jackson.databind.BeanProperty; | ||||
| import com.fasterxml.jackson.databind.JsonMappingException; | ||||
| @@ -10,6 +11,8 @@ import com.ruoyi.common.annotation.Sensitive; | ||||
| import com.ruoyi.common.core.service.SensitiveService; | ||||
| import com.ruoyi.common.enums.SensitiveStrategy; | ||||
| import com.ruoyi.common.utils.spring.SpringUtils; | ||||
| import lombok.extern.slf4j.Slf4j; | ||||
| import org.springframework.beans.BeansException; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.util.Objects; | ||||
| @@ -19,19 +22,24 @@ import java.util.Objects; | ||||
|  * | ||||
|  * @author Yjoioooo | ||||
|  */ | ||||
| @Slf4j | ||||
| public class SensitiveJsonSerializer extends JsonSerializer<String> implements ContextualSerializer { | ||||
|  | ||||
|     private SensitiveStrategy strategy; | ||||
|  | ||||
|     @Override | ||||
|     public void serialize(String value, JsonGenerator gen, SerializerProvider serializers) throws IOException { | ||||
|         try { | ||||
|             SensitiveService sensitiveService = SpringUtils.getBean(SensitiveService.class); | ||||
|         if (sensitiveService.isSensitive()) { | ||||
|             gen.writeString(value); | ||||
|         } else { | ||||
|             if (ObjectUtil.isNotNull(sensitiveService) && sensitiveService.isSensitive()) { | ||||
|                 gen.writeString(strategy.desensitizer().apply(value)); | ||||
|             } else { | ||||
|                 gen.writeString(value); | ||||
|             } | ||||
|         } catch (BeansException e) { | ||||
|             log.error("脱敏实现不存在, 采用默认处理 => {}", e.getMessage()); | ||||
|             gen.writeString(value); | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|   | ||||
| @@ -14,7 +14,6 @@ import org.springframework.cglib.core.Converter; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.stream.Collectors; | ||||
|  | ||||
| /** | ||||
|  * bean深拷贝工具(基于 cglib 性能优异) | ||||
| @@ -79,11 +78,11 @@ public class BeanCopyUtils { | ||||
|         if (CollUtil.isEmpty(sourceList)) { | ||||
|             return CollUtil.newArrayList(); | ||||
|         } | ||||
|         return sourceList.stream().map(source -> { | ||||
|         return StreamUtils.toList(sourceList, source -> { | ||||
|             V target = ReflectUtil.newInstanceIfPossible(desc); | ||||
|             copy(source, target); | ||||
|             return target; | ||||
|         }).collect(Collectors.toList()); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|   | ||||
| @@ -6,6 +6,7 @@ import cn.hutool.core.util.ObjectUtil; | ||||
| import com.fasterxml.jackson.core.JsonProcessingException; | ||||
| import com.fasterxml.jackson.core.type.TypeReference; | ||||
| import com.fasterxml.jackson.databind.ObjectMapper; | ||||
| import com.fasterxml.jackson.databind.exc.MismatchedInputException; | ||||
| import com.ruoyi.common.utils.spring.SpringUtils; | ||||
| import lombok.AccessLevel; | ||||
| import lombok.NoArgsConstructor; | ||||
| @@ -78,6 +79,9 @@ public class JsonUtils { | ||||
|         } | ||||
|         try { | ||||
|             return OBJECT_MAPPER.readValue(text, OBJECT_MAPPER.getTypeFactory().constructType(Dict.class)); | ||||
|         } catch (MismatchedInputException e) { | ||||
|             // 类型不匹配说明不是json | ||||
|             return null; | ||||
|         } catch (IOException e) { | ||||
|             throw new RuntimeException(e); | ||||
|         } | ||||
|   | ||||
| @@ -3,6 +3,7 @@ package com.ruoyi.common.utils; | ||||
| import cn.hutool.core.convert.Convert; | ||||
| import cn.hutool.extra.servlet.ServletUtil; | ||||
| import cn.hutool.http.HttpStatus; | ||||
| import com.ruoyi.common.constant.Constants; | ||||
| import lombok.AccessLevel; | ||||
| import lombok.NoArgsConstructor; | ||||
| import org.springframework.http.MediaType; | ||||
| @@ -14,6 +15,9 @@ import javax.servlet.http.HttpServletRequest; | ||||
| import javax.servlet.http.HttpServletResponse; | ||||
| import javax.servlet.http.HttpSession; | ||||
| import java.io.IOException; | ||||
| import java.io.UnsupportedEncodingException; | ||||
| import java.net.URLDecoder; | ||||
| import java.net.URLEncoder; | ||||
| import java.nio.charset.StandardCharsets; | ||||
|  | ||||
| /** | ||||
| @@ -117,7 +121,7 @@ public class ServletUtils extends ServletUtil { | ||||
|     public static boolean isAjaxRequest(HttpServletRequest request) { | ||||
|  | ||||
|         String accept = request.getHeader("accept"); | ||||
|         if (accept != null && accept.contains("application/json")) { | ||||
|         if (accept != null && accept.contains(MediaType.APPLICATION_JSON_VALUE)) { | ||||
|             return true; | ||||
|         } | ||||
|  | ||||
| @@ -139,4 +143,32 @@ public class ServletUtils extends ServletUtil { | ||||
|         return getClientIP(getRequest()); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 内容编码 | ||||
|      * | ||||
|      * @param str 内容 | ||||
|      * @return 编码后的内容 | ||||
|      */ | ||||
|     public static String urlEncode(String str) { | ||||
|         try { | ||||
|             return URLEncoder.encode(str, Constants.UTF8); | ||||
|         } catch (UnsupportedEncodingException e) { | ||||
|             return StringUtils.EMPTY; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 内容解码 | ||||
|      * | ||||
|      * @param str 内容 | ||||
|      * @return 解码后的内容 | ||||
|      */ | ||||
|     public static String urlDecode(String str) { | ||||
|         try { | ||||
|             return URLDecoder.decode(str, Constants.UTF8); | ||||
|         } catch (UnsupportedEncodingException e) { | ||||
|             return StringUtils.EMPTY; | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,251 @@ | ||||
| package com.ruoyi.common.utils; | ||||
|  | ||||
| import cn.hutool.core.collection.CollUtil; | ||||
| import cn.hutool.core.map.MapUtil; | ||||
| import lombok.AccessLevel; | ||||
| import lombok.NoArgsConstructor; | ||||
|  | ||||
| import java.util.*; | ||||
| import java.util.function.BiFunction; | ||||
| import java.util.function.Function; | ||||
| import java.util.function.Predicate; | ||||
| import java.util.stream.Collectors; | ||||
|  | ||||
| /** | ||||
|  * stream 流工具类 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @NoArgsConstructor(access = AccessLevel.PRIVATE) | ||||
| public class StreamUtils { | ||||
|  | ||||
|     /** | ||||
|      * 将collection过滤 | ||||
|      * | ||||
|      * @param collection 需要转化的集合 | ||||
|      * @param function   过滤方法 | ||||
|      * @return 过滤后的list | ||||
|      */ | ||||
|     public static <E> List<E> filter(Collection<E> collection, Predicate<E> function) { | ||||
|         if (CollUtil.isEmpty(collection)) { | ||||
|             return CollUtil.newArrayList(); | ||||
|         } | ||||
|         return collection.stream().filter(function).collect(Collectors.toList()); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 将collection拼接 | ||||
|      * | ||||
|      * @param collection 需要转化的集合 | ||||
|      * @param function   拼接方法 | ||||
|      * @return 拼接后的list | ||||
|      */ | ||||
|     public static <E> String join(Collection<E> collection, Function<E, String> function) { | ||||
|         return join(collection, function, ","); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 将collection拼接 | ||||
|      * | ||||
|      * @param collection 需要转化的集合 | ||||
|      * @param function   拼接方法 | ||||
|      * @param delimiter  拼接符 | ||||
|      * @return 拼接后的list | ||||
|      */ | ||||
|     public static <E> String join(Collection<E> collection, Function<E, String> function, CharSequence delimiter) { | ||||
|         if (CollUtil.isEmpty(collection)) { | ||||
|             return StringUtils.EMPTY; | ||||
|         } | ||||
|         return collection.stream().map(function).filter(Objects::nonNull).collect(Collectors.joining(delimiter)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 将collection排序 | ||||
|      * | ||||
|      * @param collection 需要转化的集合 | ||||
|      * @param comparing  排序方法 | ||||
|      * @return 排序后的list | ||||
|      */ | ||||
|     public static <E> List<E> sorted(Collection<E> collection, Comparator<E> comparing) { | ||||
|         if (CollUtil.isEmpty(collection)) { | ||||
|             return CollUtil.newArrayList(); | ||||
|         } | ||||
|         return collection.stream().sorted(comparing).collect(Collectors.toList()); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 将collection转化为类型不变的map<br> | ||||
|      * <B>{@code Collection<V>  ---->  Map<K,V>}</B> | ||||
|      * | ||||
|      * @param collection 需要转化的集合 | ||||
|      * @param key        V类型转化为K类型的lambda方法 | ||||
|      * @param <V>        collection中的泛型 | ||||
|      * @param <K>        map中的key类型 | ||||
|      * @return 转化后的map | ||||
|      */ | ||||
|     public static <V, K> Map<K, V> toIdentityMap(Collection<V> collection, Function<V, K> key) { | ||||
|         if (CollUtil.isEmpty(collection)) { | ||||
|             return MapUtil.newHashMap(); | ||||
|         } | ||||
|         return collection.stream().collect(Collectors.toMap(key, Function.identity(), (l, r) -> l)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 将Collection转化为map(value类型与collection的泛型不同)<br> | ||||
|      * <B>{@code Collection<E> -----> Map<K,V>  }</B> | ||||
|      * | ||||
|      * @param collection 需要转化的集合 | ||||
|      * @param key        E类型转化为K类型的lambda方法 | ||||
|      * @param value      E类型转化为V类型的lambda方法 | ||||
|      * @param <E>        collection中的泛型 | ||||
|      * @param <K>        map中的key类型 | ||||
|      * @param <V>        map中的value类型 | ||||
|      * @return 转化后的map | ||||
|      */ | ||||
|     public static <E, K, V> Map<K, V> toMap(Collection<E> collection, Function<E, K> key, Function<E, V> value) { | ||||
|         if (CollUtil.isEmpty(collection)) { | ||||
|             return MapUtil.newHashMap(); | ||||
|         } | ||||
|         return collection.stream().collect(Collectors.toMap(key, value, (l, r) -> l)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 将collection按照规则(比如有相同的班级id)分类成map<br> | ||||
|      * <B>{@code Collection<E> -------> Map<K,List<E>> } </B> | ||||
|      * | ||||
|      * @param collection 需要分类的集合 | ||||
|      * @param key        分类的规则 | ||||
|      * @param <E>        collection中的泛型 | ||||
|      * @param <K>        map中的key类型 | ||||
|      * @return 分类后的map | ||||
|      */ | ||||
|     public static <E, K> Map<K, List<E>> groupByKey(Collection<E> collection, Function<E, K> key) { | ||||
|         if (CollUtil.isEmpty(collection)) { | ||||
|             return MapUtil.newHashMap(); | ||||
|         } | ||||
|         return collection | ||||
|             .stream() | ||||
|             .collect(Collectors.groupingBy(key, LinkedHashMap::new, Collectors.toList())); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 将collection按照两个规则(比如有相同的年级id,班级id)分类成双层map<br> | ||||
|      * <B>{@code Collection<E>  --->  Map<T,Map<U,List<E>>> } </B> | ||||
|      * | ||||
|      * @param collection 需要分类的集合 | ||||
|      * @param key1       第一个分类的规则 | ||||
|      * @param key2       第二个分类的规则 | ||||
|      * @param <E>        集合元素类型 | ||||
|      * @param <K>        第一个map中的key类型 | ||||
|      * @param <U>        第二个map中的key类型 | ||||
|      * @return 分类后的map | ||||
|      */ | ||||
|     public static <E, K, U> Map<K, Map<U, List<E>>> groupBy2Key(Collection<E> collection, Function<E, K> key1, Function<E, U> key2) { | ||||
|         if (CollUtil.isEmpty(collection)) { | ||||
|             return MapUtil.newHashMap(); | ||||
|         } | ||||
|         return collection | ||||
|             .stream() | ||||
|             .collect(Collectors.groupingBy(key1, LinkedHashMap::new, Collectors.groupingBy(key2, LinkedHashMap::new, Collectors.toList()))); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 将collection按照两个规则(比如有相同的年级id,班级id)分类成双层map<br> | ||||
|      * <B>{@code Collection<E>  --->  Map<T,Map<U,E>> } </B> | ||||
|      * | ||||
|      * @param collection 需要分类的集合 | ||||
|      * @param key1       第一个分类的规则 | ||||
|      * @param key2       第二个分类的规则 | ||||
|      * @param <T>        第一个map中的key类型 | ||||
|      * @param <U>        第二个map中的key类型 | ||||
|      * @param <E>        collection中的泛型 | ||||
|      * @return 分类后的map | ||||
|      */ | ||||
|     public static <E, T, U> Map<T, Map<U, E>> group2Map(Collection<E> collection, Function<E, T> key1, Function<E, U> key2) { | ||||
|         if (CollUtil.isEmpty(collection) || key1 == null || key2 == null) { | ||||
|             return MapUtil.newHashMap(); | ||||
|         } | ||||
|         return collection | ||||
|             .stream() | ||||
|             .collect(Collectors.groupingBy(key1, LinkedHashMap::new, Collectors.toMap(key2, Function.identity(), (l, r) -> l))); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 将collection转化为List集合,但是两者的泛型不同<br> | ||||
|      * <B>{@code Collection<E>  ------>  List<T> } </B> | ||||
|      * | ||||
|      * @param collection 需要转化的集合 | ||||
|      * @param function   collection中的泛型转化为list泛型的lambda表达式 | ||||
|      * @param <E>        collection中的泛型 | ||||
|      * @param <T>        List中的泛型 | ||||
|      * @return 转化后的list | ||||
|      */ | ||||
|     public static <E, T> List<T> toList(Collection<E> collection, Function<E, T> function) { | ||||
|         if (CollUtil.isEmpty(collection)) { | ||||
|             return CollUtil.newArrayList(); | ||||
|         } | ||||
|         return collection | ||||
|             .stream() | ||||
|             .map(function) | ||||
|             .filter(Objects::nonNull) | ||||
|             .collect(Collectors.toList()); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 将collection转化为Set集合,但是两者的泛型不同<br> | ||||
|      * <B>{@code Collection<E>  ------>  Set<T> } </B> | ||||
|      * | ||||
|      * @param collection 需要转化的集合 | ||||
|      * @param function   collection中的泛型转化为set泛型的lambda表达式 | ||||
|      * @param <E>        collection中的泛型 | ||||
|      * @param <T>        Set中的泛型 | ||||
|      * @return 转化后的Set | ||||
|      */ | ||||
|     public static <E, T> Set<T> toSet(Collection<E> collection, Function<E, T> function) { | ||||
|         if (CollUtil.isEmpty(collection) || function == null) { | ||||
|             return CollUtil.newHashSet(); | ||||
|         } | ||||
|         return collection | ||||
|             .stream() | ||||
|             .map(function) | ||||
|             .filter(Objects::nonNull) | ||||
|             .collect(Collectors.toSet()); | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /** | ||||
|      * 合并两个相同key类型的map | ||||
|      * | ||||
|      * @param map1  第一个需要合并的 map | ||||
|      * @param map2  第二个需要合并的 map | ||||
|      * @param merge 合并的lambda,将key  value1 value2合并成最终的类型,注意value可能为空的情况 | ||||
|      * @param <K>   map中的key类型 | ||||
|      * @param <X>   第一个 map的value类型 | ||||
|      * @param <Y>   第二个 map的value类型 | ||||
|      * @param <V>   最终map的value类型 | ||||
|      * @return 合并后的map | ||||
|      */ | ||||
|     public static <K, X, Y, V> Map<K, V> merge(Map<K, X> map1, Map<K, Y> map2, BiFunction<X, Y, V> merge) { | ||||
|         if (MapUtil.isEmpty(map1) && MapUtil.isEmpty(map2)) { | ||||
|             return MapUtil.newHashMap(); | ||||
|         } else if (MapUtil.isEmpty(map1)) { | ||||
|             map1 = MapUtil.newHashMap(); | ||||
|         } else if (MapUtil.isEmpty(map2)) { | ||||
|             map2 = MapUtil.newHashMap(); | ||||
|         } | ||||
|         Set<K> key = new HashSet<>(); | ||||
|         key.addAll(map1.keySet()); | ||||
|         key.addAll(map2.keySet()); | ||||
|         Map<K, V> map = new HashMap<>(); | ||||
|         for (K t : key) { | ||||
|             X x = map1.get(t); | ||||
|             Y y = map2.get(t); | ||||
|             V z = merge.apply(x, y); | ||||
|             if (z != null) { | ||||
|                 map.put(t, z); | ||||
|             } | ||||
|         } | ||||
|         return map; | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,468 @@ | ||||
| package com.ruoyi.common.utils.email; | ||||
|  | ||||
| import cn.hutool.core.collection.CollUtil; | ||||
| import cn.hutool.core.io.IoUtil; | ||||
| import cn.hutool.core.map.MapUtil; | ||||
| import cn.hutool.core.util.CharUtil; | ||||
| import cn.hutool.core.util.StrUtil; | ||||
| import cn.hutool.extra.mail.*; | ||||
| import com.ruoyi.common.utils.StringUtils; | ||||
| import com.ruoyi.common.utils.spring.SpringUtils; | ||||
| import lombok.AccessLevel; | ||||
| import lombok.NoArgsConstructor; | ||||
|  | ||||
| import javax.mail.Authenticator; | ||||
| import javax.mail.Session; | ||||
| import java.io.File; | ||||
| import java.io.InputStream; | ||||
| import java.util.Collection; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
|  | ||||
|  | ||||
| /** | ||||
|  * 邮件工具类 | ||||
|  */ | ||||
| @NoArgsConstructor(access = AccessLevel.PRIVATE) | ||||
| public class MailUtils { | ||||
|  | ||||
|     private static final MailAccount ACCOUNT = SpringUtils.getBean(MailAccount.class); | ||||
|  | ||||
|     /** | ||||
|      * 获取邮件发送实例 | ||||
|      */ | ||||
|     public static MailAccount getMailAccount() { | ||||
|         return ACCOUNT; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取邮件发送实例 (自定义发送人以及授权码) | ||||
|      * | ||||
|      * @param user 发送人 | ||||
|      * @param pass 授权码 | ||||
|      */ | ||||
|     public static MailAccount getMailAccount(String from, String user, String pass) { | ||||
|         ACCOUNT.setFrom(StringUtils.blankToDefault(from, ACCOUNT.getFrom())); | ||||
|         ACCOUNT.setUser(StringUtils.blankToDefault(user, ACCOUNT.getUser())); | ||||
|         ACCOUNT.setPass(StringUtils.blankToDefault(pass, ACCOUNT.getPass())); | ||||
|         return ACCOUNT; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 使用配置文件中设置的账户发送文本邮件,发送给单个或多个收件人<br> | ||||
|      * 多个收件人可以使用逗号“,”分隔,也可以通过分号“;”分隔 | ||||
|      * | ||||
|      * @param to      收件人 | ||||
|      * @param subject 标题 | ||||
|      * @param content 正文 | ||||
|      * @param files   附件列表 | ||||
|      * @return message-id | ||||
|      * @since 3.2.0 | ||||
|      */ | ||||
|     public static String sendText(String to, String subject, String content, File... files) { | ||||
|         return send(to, subject, content, false, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 使用配置文件中设置的账户发送HTML邮件,发送给单个或多个收件人<br> | ||||
|      * 多个收件人可以使用逗号“,”分隔,也可以通过分号“;”分隔 | ||||
|      * | ||||
|      * @param to      收件人 | ||||
|      * @param subject 标题 | ||||
|      * @param content 正文 | ||||
|      * @param files   附件列表 | ||||
|      * @return message-id | ||||
|      * @since 3.2.0 | ||||
|      */ | ||||
|     public static String sendHtml(String to, String subject, String content, File... files) { | ||||
|         return send(to, subject, content, true, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 使用配置文件中设置的账户发送邮件,发送单个或多个收件人<br> | ||||
|      * 多个收件人可以使用逗号“,”分隔,也可以通过分号“;”分隔 | ||||
|      * | ||||
|      * @param to      收件人 | ||||
|      * @param subject 标题 | ||||
|      * @param content 正文 | ||||
|      * @param isHtml  是否为HTML | ||||
|      * @param files   附件列表 | ||||
|      * @return message-id | ||||
|      */ | ||||
|     public static String send(String to, String subject, String content, boolean isHtml, File... files) { | ||||
|         return send(splitAddress(to), subject, content, isHtml, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 使用配置文件中设置的账户发送邮件,发送单个或多个收件人<br> | ||||
|      * 多个收件人、抄送人、密送人可以使用逗号“,”分隔,也可以通过分号“;”分隔 | ||||
|      * | ||||
|      * @param to      收件人,可以使用逗号“,”分隔,也可以通过分号“;”分隔 | ||||
|      * @param cc      抄送人,可以使用逗号“,”分隔,也可以通过分号“;”分隔 | ||||
|      * @param bcc     密送人,可以使用逗号“,”分隔,也可以通过分号“;”分隔 | ||||
|      * @param subject 标题 | ||||
|      * @param content 正文 | ||||
|      * @param isHtml  是否为HTML | ||||
|      * @param files   附件列表 | ||||
|      * @return message-id | ||||
|      * @since 4.0.3 | ||||
|      */ | ||||
|     public static String send(String to, String cc, String bcc, String subject, String content, boolean isHtml, File... files) { | ||||
|         return send(splitAddress(to), splitAddress(cc), splitAddress(bcc), subject, content, isHtml, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 使用配置文件中设置的账户发送文本邮件,发送给多人 | ||||
|      * | ||||
|      * @param tos     收件人列表 | ||||
|      * @param subject 标题 | ||||
|      * @param content 正文 | ||||
|      * @param files   附件列表 | ||||
|      * @return message-id | ||||
|      */ | ||||
|     public static String sendText(Collection<String> tos, String subject, String content, File... files) { | ||||
|         return send(tos, subject, content, false, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 使用配置文件中设置的账户发送HTML邮件,发送给多人 | ||||
|      * | ||||
|      * @param tos     收件人列表 | ||||
|      * @param subject 标题 | ||||
|      * @param content 正文 | ||||
|      * @param files   附件列表 | ||||
|      * @return message-id | ||||
|      * @since 3.2.0 | ||||
|      */ | ||||
|     public static String sendHtml(Collection<String> tos, String subject, String content, File... files) { | ||||
|         return send(tos, subject, content, true, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 使用配置文件中设置的账户发送邮件,发送给多人 | ||||
|      * | ||||
|      * @param tos     收件人列表 | ||||
|      * @param subject 标题 | ||||
|      * @param content 正文 | ||||
|      * @param isHtml  是否为HTML | ||||
|      * @param files   附件列表 | ||||
|      * @return message-id | ||||
|      */ | ||||
|     public static String send(Collection<String> tos, String subject, String content, boolean isHtml, File... files) { | ||||
|         return send(tos, null, null, subject, content, isHtml, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 使用配置文件中设置的账户发送邮件,发送给多人 | ||||
|      * | ||||
|      * @param tos     收件人列表 | ||||
|      * @param ccs     抄送人列表,可以为null或空 | ||||
|      * @param bccs    密送人列表,可以为null或空 | ||||
|      * @param subject 标题 | ||||
|      * @param content 正文 | ||||
|      * @param isHtml  是否为HTML | ||||
|      * @param files   附件列表 | ||||
|      * @return message-id | ||||
|      * @since 4.0.3 | ||||
|      */ | ||||
|     public static String send(Collection<String> tos, Collection<String> ccs, Collection<String> bccs, String subject, String content, boolean isHtml, File... files) { | ||||
|         return send(getMailAccount(), true, tos, ccs, bccs, subject, content, null, isHtml, files); | ||||
|     } | ||||
|  | ||||
|     // ------------------------------------------------------------------------------------------------------------------------------- Custom MailAccount | ||||
|  | ||||
|     /** | ||||
|      * 发送邮件给多人 | ||||
|      * | ||||
|      * @param mailAccount 邮件认证对象 | ||||
|      * @param to          收件人,多个收件人逗号或者分号隔开 | ||||
|      * @param subject     标题 | ||||
|      * @param content     正文 | ||||
|      * @param isHtml      是否为HTML格式 | ||||
|      * @param files       附件列表 | ||||
|      * @return message-id | ||||
|      * @since 3.2.0 | ||||
|      */ | ||||
|     public static String send(MailAccount mailAccount, String to, String subject, String content, boolean isHtml, File... files) { | ||||
|         return send(mailAccount, splitAddress(to), subject, content, isHtml, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 发送邮件给多人 | ||||
|      * | ||||
|      * @param mailAccount 邮件帐户信息 | ||||
|      * @param tos         收件人列表 | ||||
|      * @param subject     标题 | ||||
|      * @param content     正文 | ||||
|      * @param isHtml      是否为HTML格式 | ||||
|      * @param files       附件列表 | ||||
|      * @return message-id | ||||
|      */ | ||||
|     public static String send(MailAccount mailAccount, Collection<String> tos, String subject, String content, boolean isHtml, File... files) { | ||||
|         return send(mailAccount, tos, null, null, subject, content, isHtml, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 发送邮件给多人 | ||||
|      * | ||||
|      * @param mailAccount 邮件帐户信息 | ||||
|      * @param tos         收件人列表 | ||||
|      * @param ccs         抄送人列表,可以为null或空 | ||||
|      * @param bccs        密送人列表,可以为null或空 | ||||
|      * @param subject     标题 | ||||
|      * @param content     正文 | ||||
|      * @param isHtml      是否为HTML格式 | ||||
|      * @param files       附件列表 | ||||
|      * @return message-id | ||||
|      * @since 4.0.3 | ||||
|      */ | ||||
|     public static String send(MailAccount mailAccount, Collection<String> tos, Collection<String> ccs, Collection<String> bccs, String subject, String content, boolean isHtml, File... files) { | ||||
|         return send(mailAccount, false, tos, ccs, bccs, subject, content, null, isHtml, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 使用配置文件中设置的账户发送HTML邮件,发送给单个或多个收件人<br> | ||||
|      * 多个收件人可以使用逗号“,”分隔,也可以通过分号“;”分隔 | ||||
|      * | ||||
|      * @param to       收件人 | ||||
|      * @param subject  标题 | ||||
|      * @param content  正文 | ||||
|      * @param imageMap 图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER | ||||
|      * @param files    附件列表 | ||||
|      * @return message-id | ||||
|      * @since 3.2.0 | ||||
|      */ | ||||
|     public static String sendHtml(String to, String subject, String content, Map<String, InputStream> imageMap, File... files) { | ||||
|         return send(to, subject, content, imageMap, true, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 使用配置文件中设置的账户发送邮件,发送单个或多个收件人<br> | ||||
|      * 多个收件人可以使用逗号“,”分隔,也可以通过分号“;”分隔 | ||||
|      * | ||||
|      * @param to       收件人 | ||||
|      * @param subject  标题 | ||||
|      * @param content  正文 | ||||
|      * @param imageMap 图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER | ||||
|      * @param isHtml   是否为HTML | ||||
|      * @param files    附件列表 | ||||
|      * @return message-id | ||||
|      */ | ||||
|     public static String send(String to, String subject, String content, Map<String, InputStream> imageMap, boolean isHtml, File... files) { | ||||
|         return send(splitAddress(to), subject, content, imageMap, isHtml, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 使用配置文件中设置的账户发送邮件,发送单个或多个收件人<br> | ||||
|      * 多个收件人、抄送人、密送人可以使用逗号“,”分隔,也可以通过分号“;”分隔 | ||||
|      * | ||||
|      * @param to       收件人,可以使用逗号“,”分隔,也可以通过分号“;”分隔 | ||||
|      * @param cc       抄送人,可以使用逗号“,”分隔,也可以通过分号“;”分隔 | ||||
|      * @param bcc      密送人,可以使用逗号“,”分隔,也可以通过分号“;”分隔 | ||||
|      * @param subject  标题 | ||||
|      * @param content  正文 | ||||
|      * @param imageMap 图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER | ||||
|      * @param isHtml   是否为HTML | ||||
|      * @param files    附件列表 | ||||
|      * @return message-id | ||||
|      * @since 4.0.3 | ||||
|      */ | ||||
|     public static String send(String to, String cc, String bcc, String subject, String content, Map<String, InputStream> imageMap, boolean isHtml, File... files) { | ||||
|         return send(splitAddress(to), splitAddress(cc), splitAddress(bcc), subject, content, imageMap, isHtml, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 使用配置文件中设置的账户发送HTML邮件,发送给多人 | ||||
|      * | ||||
|      * @param tos      收件人列表 | ||||
|      * @param subject  标题 | ||||
|      * @param content  正文 | ||||
|      * @param imageMap 图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER | ||||
|      * @param files    附件列表 | ||||
|      * @return message-id | ||||
|      * @since 3.2.0 | ||||
|      */ | ||||
|     public static String sendHtml(Collection<String> tos, String subject, String content, Map<String, InputStream> imageMap, File... files) { | ||||
|         return send(tos, subject, content, imageMap, true, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 使用配置文件中设置的账户发送邮件,发送给多人 | ||||
|      * | ||||
|      * @param tos      收件人列表 | ||||
|      * @param subject  标题 | ||||
|      * @param content  正文 | ||||
|      * @param imageMap 图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER | ||||
|      * @param isHtml   是否为HTML | ||||
|      * @param files    附件列表 | ||||
|      * @return message-id | ||||
|      */ | ||||
|     public static String send(Collection<String> tos, String subject, String content, Map<String, InputStream> imageMap, boolean isHtml, File... files) { | ||||
|         return send(tos, null, null, subject, content, imageMap, isHtml, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 使用配置文件中设置的账户发送邮件,发送给多人 | ||||
|      * | ||||
|      * @param tos      收件人列表 | ||||
|      * @param ccs      抄送人列表,可以为null或空 | ||||
|      * @param bccs     密送人列表,可以为null或空 | ||||
|      * @param subject  标题 | ||||
|      * @param content  正文 | ||||
|      * @param imageMap 图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER | ||||
|      * @param isHtml   是否为HTML | ||||
|      * @param files    附件列表 | ||||
|      * @return message-id | ||||
|      * @since 4.0.3 | ||||
|      */ | ||||
|     public static String send(Collection<String> tos, Collection<String> ccs, Collection<String> bccs, String subject, String content, Map<String, InputStream> imageMap, boolean isHtml, File... files) { | ||||
|         return send(getMailAccount(), true, tos, ccs, bccs, subject, content, imageMap, isHtml, files); | ||||
|     } | ||||
|  | ||||
|     // ------------------------------------------------------------------------------------------------------------------------------- Custom MailAccount | ||||
|  | ||||
|     /** | ||||
|      * 发送邮件给多人 | ||||
|      * | ||||
|      * @param mailAccount 邮件认证对象 | ||||
|      * @param to          收件人,多个收件人逗号或者分号隔开 | ||||
|      * @param subject     标题 | ||||
|      * @param content     正文 | ||||
|      * @param imageMap    图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER | ||||
|      * @param isHtml      是否为HTML格式 | ||||
|      * @param files       附件列表 | ||||
|      * @return message-id | ||||
|      * @since 3.2.0 | ||||
|      */ | ||||
|     public static String send(MailAccount mailAccount, String to, String subject, String content, Map<String, InputStream> imageMap, boolean isHtml, File... files) { | ||||
|         return send(mailAccount, splitAddress(to), subject, content, imageMap, isHtml, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 发送邮件给多人 | ||||
|      * | ||||
|      * @param mailAccount 邮件帐户信息 | ||||
|      * @param tos         收件人列表 | ||||
|      * @param subject     标题 | ||||
|      * @param content     正文 | ||||
|      * @param imageMap    图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER | ||||
|      * @param isHtml      是否为HTML格式 | ||||
|      * @param files       附件列表 | ||||
|      * @return message-id | ||||
|      * @since 4.6.3 | ||||
|      */ | ||||
|     public static String send(MailAccount mailAccount, Collection<String> tos, String subject, String content, Map<String, InputStream> imageMap, boolean isHtml, File... files) { | ||||
|         return send(mailAccount, tos, null, null, subject, content, imageMap, isHtml, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 发送邮件给多人 | ||||
|      * | ||||
|      * @param mailAccount 邮件帐户信息 | ||||
|      * @param tos         收件人列表 | ||||
|      * @param ccs         抄送人列表,可以为null或空 | ||||
|      * @param bccs        密送人列表,可以为null或空 | ||||
|      * @param subject     标题 | ||||
|      * @param content     正文 | ||||
|      * @param imageMap    图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER | ||||
|      * @param isHtml      是否为HTML格式 | ||||
|      * @param files       附件列表 | ||||
|      * @return message-id | ||||
|      * @since 4.6.3 | ||||
|      */ | ||||
|     public static String send(MailAccount mailAccount, Collection<String> tos, Collection<String> ccs, Collection<String> bccs, String subject, String content, Map<String, InputStream> imageMap, | ||||
|                               boolean isHtml, File... files) { | ||||
|         return send(mailAccount, false, tos, ccs, bccs, subject, content, imageMap, isHtml, files); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 根据配置文件,获取邮件客户端会话 | ||||
|      * | ||||
|      * @param mailAccount 邮件账户配置 | ||||
|      * @param isSingleton 是否单例(全局共享会话) | ||||
|      * @return {@link Session} | ||||
|      * @since 5.5.7 | ||||
|      */ | ||||
|     public static Session getSession(MailAccount mailAccount, boolean isSingleton) { | ||||
|         Authenticator authenticator = null; | ||||
|         if (mailAccount.isAuth()) { | ||||
|             authenticator = new UserPassAuthenticator(mailAccount.getUser(), mailAccount.getPass()); | ||||
|         } | ||||
|  | ||||
|         return isSingleton ? Session.getDefaultInstance(mailAccount.getSmtpProps(), authenticator) // | ||||
|             : Session.getInstance(mailAccount.getSmtpProps(), authenticator); | ||||
|     } | ||||
|  | ||||
|     // ------------------------------------------------------------------------------------------------------------------------ Private method start | ||||
|  | ||||
|     /** | ||||
|      * 发送邮件给多人 | ||||
|      * | ||||
|      * @param mailAccount      邮件帐户信息 | ||||
|      * @param useGlobalSession 是否全局共享Session | ||||
|      * @param tos              收件人列表 | ||||
|      * @param ccs              抄送人列表,可以为null或空 | ||||
|      * @param bccs             密送人列表,可以为null或空 | ||||
|      * @param subject          标题 | ||||
|      * @param content          正文 | ||||
|      * @param imageMap         图片与占位符,占位符格式为cid:${cid} | ||||
|      * @param isHtml           是否为HTML格式 | ||||
|      * @param files            附件列表 | ||||
|      * @return message-id | ||||
|      * @since 4.6.3 | ||||
|      */ | ||||
|     private static String send(MailAccount mailAccount, boolean useGlobalSession, Collection<String> tos, Collection<String> ccs, Collection<String> bccs, String subject, String content, | ||||
|                                Map<String, InputStream> imageMap, boolean isHtml, File... files) { | ||||
|         final Mail mail = Mail.create(mailAccount).setUseGlobalSession(useGlobalSession); | ||||
|  | ||||
|         // 可选抄送人 | ||||
|         if (CollUtil.isNotEmpty(ccs)) { | ||||
|             mail.setCcs(ccs.toArray(new String[0])); | ||||
|         } | ||||
|         // 可选密送人 | ||||
|         if (CollUtil.isNotEmpty(bccs)) { | ||||
|             mail.setBccs(bccs.toArray(new String[0])); | ||||
|         } | ||||
|  | ||||
|         mail.setTos(tos.toArray(new String[0])); | ||||
|         mail.setTitle(subject); | ||||
|         mail.setContent(content); | ||||
|         mail.setHtml(isHtml); | ||||
|         mail.setFiles(files); | ||||
|  | ||||
|         // 图片 | ||||
|         if (MapUtil.isNotEmpty(imageMap)) { | ||||
|             for (Map.Entry<String, InputStream> entry : imageMap.entrySet()) { | ||||
|                 mail.addImage(entry.getKey(), entry.getValue()); | ||||
|                 // 关闭流 | ||||
|                 IoUtil.close(entry.getValue()); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return mail.send(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 将多个联系人转为列表,分隔符为逗号或者分号 | ||||
|      * | ||||
|      * @param addresses 多个联系人,如果为空返回null | ||||
|      * @return 联系人列表 | ||||
|      */ | ||||
|     private static List<String> splitAddress(String addresses) { | ||||
|         if (StrUtil.isBlank(addresses)) { | ||||
|             return null; | ||||
|         } | ||||
|  | ||||
|         List<String> result; | ||||
|         if (StrUtil.contains(addresses, CharUtil.COMMA)) { | ||||
|             result = StrUtil.splitTrim(addresses, CharUtil.COMMA); | ||||
|         } else if (StrUtil.contains(addresses, ';')) { | ||||
|             result = StrUtil.splitTrim(addresses, ';'); | ||||
|         } else { | ||||
|             result = CollUtil.newArrayList(addresses); | ||||
|         } | ||||
|         return result; | ||||
|     } | ||||
|     // ------------------------------------------------------------------------------------------------------------------------ Private method end | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,40 @@ | ||||
| package com.ruoyi.common.utils.file; | ||||
|  | ||||
| /** | ||||
|  * 媒体类型工具类 | ||||
|  * | ||||
|  * @author ruoyi | ||||
|  */ | ||||
| public class MimeTypeUtils { | ||||
|     public static final String IMAGE_PNG = "image/png"; | ||||
|  | ||||
|     public static final String IMAGE_JPG = "image/jpg"; | ||||
|  | ||||
|     public static final String IMAGE_JPEG = "image/jpeg"; | ||||
|  | ||||
|     public static final String IMAGE_BMP = "image/bmp"; | ||||
|  | ||||
|     public static final String IMAGE_GIF = "image/gif"; | ||||
|  | ||||
|     public static final String[] IMAGE_EXTENSION = {"bmp", "gif", "jpg", "jpeg", "png"}; | ||||
|  | ||||
|     public static final String[] FLASH_EXTENSION = {"swf", "flv"}; | ||||
|  | ||||
|     public static final String[] MEDIA_EXTENSION = {"swf", "flv", "mp3", "wav", "wma", "wmv", "mid", "avi", "mpg", | ||||
|         "asf", "rm", "rmvb"}; | ||||
|  | ||||
|     public static final String[] VIDEO_EXTENSION = {"mp4", "avi", "rmvb"}; | ||||
|  | ||||
|     public static final String[] DEFAULT_ALLOWED_EXTENSION = { | ||||
|         // 图片 | ||||
|         "bmp", "gif", "jpg", "jpeg", "png", | ||||
|         // word excel powerpoint | ||||
|         "doc", "docx", "xls", "xlsx", "ppt", "pptx", "html", "htm", "txt", | ||||
|         // 压缩文件 | ||||
|         "rar", "zip", "gz", "bz2", | ||||
|         // 视频格式 | ||||
|         "mp4", "avi", "rmvb", | ||||
|         // pdf | ||||
|         "pdf"}; | ||||
|  | ||||
| } | ||||
| @@ -1,9 +1,17 @@ | ||||
| package com.ruoyi.common.utils.poi; | ||||
|  | ||||
| import cn.hutool.core.collection.CollUtil; | ||||
| import cn.hutool.core.io.resource.ClassPathResource; | ||||
| import cn.hutool.core.util.IdUtil; | ||||
| import com.alibaba.excel.EasyExcel; | ||||
| import com.alibaba.excel.ExcelWriter; | ||||
| import com.alibaba.excel.write.builder.ExcelWriterSheetBuilder; | ||||
| import com.alibaba.excel.write.metadata.WriteSheet; | ||||
| import com.alibaba.excel.write.metadata.fill.FillConfig; | ||||
| import com.alibaba.excel.write.metadata.fill.FillWrapper; | ||||
| import com.alibaba.excel.write.style.column.LongestMatchColumnWidthStyleStrategy; | ||||
| import com.ruoyi.common.convert.ExcelBigNumberConvert; | ||||
| import com.ruoyi.common.excel.CellMergeStrategy; | ||||
| import com.ruoyi.common.excel.DefaultExcelListener; | ||||
| import com.ruoyi.common.excel.ExcelListener; | ||||
| import com.ruoyi.common.excel.ExcelResult; | ||||
| @@ -16,7 +24,10 @@ import javax.servlet.ServletOutputStream; | ||||
| import javax.servlet.http.HttpServletResponse; | ||||
| import java.io.IOException; | ||||
| import java.io.InputStream; | ||||
| import java.io.UnsupportedEncodingException; | ||||
| import java.util.Collection; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * Excel相关处理 | ||||
| @@ -69,27 +80,125 @@ public class ExcelUtil { | ||||
|      * | ||||
|      * @param list      导出数据集合 | ||||
|      * @param sheetName 工作表的名称 | ||||
|      * @return 结果 | ||||
|      * @param clazz     实体类 | ||||
|      * @param response  响应体 | ||||
|      */ | ||||
|     public static <T> void exportExcel(List<T> list, String sheetName, Class<T> clazz, HttpServletResponse response) { | ||||
|         exportExcel(list, sheetName, clazz, false, response); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 导出excel | ||||
|      * | ||||
|      * @param list      导出数据集合 | ||||
|      * @param sheetName 工作表的名称 | ||||
|      * @param clazz     实体类 | ||||
|      * @param merge     是否合并单元格 | ||||
|      * @param response  响应体 | ||||
|      */ | ||||
|     public static <T> void exportExcel(List<T> list, String sheetName, Class<T> clazz, boolean merge, HttpServletResponse response) { | ||||
|         try { | ||||
|             String filename = encodingFilename(sheetName); | ||||
|             response.reset(); | ||||
|             FileUtils.setAttachmentResponseHeader(response, filename); | ||||
|             response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=UTF-8"); | ||||
|             resetResponse(sheetName, response); | ||||
|             ServletOutputStream os = response.getOutputStream(); | ||||
|             EasyExcel.write(os, clazz) | ||||
|             ExcelWriterSheetBuilder builder = EasyExcel.write(os, clazz) | ||||
|                 .autoCloseStream(false) | ||||
|                 // 自动适配 | ||||
|                 .registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()) | ||||
|                 // 大数值自动转换 防止失真 | ||||
|                 .registerConverter(new ExcelBigNumberConvert()) | ||||
|                 .sheet(sheetName).doWrite(list); | ||||
|                 .sheet(sheetName); | ||||
|             if (merge) { | ||||
|                 // 合并处理器 | ||||
|                 builder.registerWriteHandler(new CellMergeStrategy(list, true)); | ||||
|             } | ||||
|             builder.doWrite(list); | ||||
|         } catch (IOException e) { | ||||
|             throw new RuntimeException("导出Excel异常"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 单表多数据模板导出 模板格式为 {.属性} | ||||
|      * | ||||
|      * @param filename     文件名 | ||||
|      * @param templatePath 模板路径 resource 目录下的路径包括模板文件名 | ||||
|      *                     例如: excel/temp.xlsx | ||||
|      *                     重点: 模板文件必须放置到启动类对应的 resource 目录下 | ||||
|      * @param data         模板需要的数据 | ||||
|      */ | ||||
|     public static void exportTemplate(List<Object> data, String filename, String templatePath, HttpServletResponse response) { | ||||
|         try { | ||||
|             resetResponse(filename, response); | ||||
|             ClassPathResource templateResource = new ClassPathResource(templatePath); | ||||
|             ExcelWriter excelWriter = EasyExcel.write(response.getOutputStream()) | ||||
|                 .withTemplate(templateResource.getStream()) | ||||
|                 .autoCloseStream(false) | ||||
|                 // 大数值自动转换 防止失真 | ||||
|                 .registerConverter(new ExcelBigNumberConvert()) | ||||
|                 .build(); | ||||
|             WriteSheet writeSheet = EasyExcel.writerSheet().build(); | ||||
|             if (CollUtil.isEmpty(data)) { | ||||
|                 throw new IllegalArgumentException("数据为空"); | ||||
|             } | ||||
|             // 单表多数据导出 模板格式为 {.属性} | ||||
|             for (Object d : data) { | ||||
|                 excelWriter.fill(d, writeSheet); | ||||
|             } | ||||
|             excelWriter.finish(); | ||||
|         } catch (IOException e) { | ||||
|             throw new RuntimeException("导出Excel异常"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 多表多数据模板导出 模板格式为 {key.属性} | ||||
|      * | ||||
|      * @param filename     文件名 | ||||
|      * @param templatePath 模板路径 resource 目录下的路径包括模板文件名 | ||||
|      *                     例如: excel/temp.xlsx | ||||
|      *                     重点: 模板文件必须放置到启动类对应的 resource 目录下 | ||||
|      * @param data         模板需要的数据 | ||||
|      */ | ||||
|     public static void exportTemplateMultiList(Map<String, Object> data, String filename, String templatePath, HttpServletResponse response) { | ||||
|         try { | ||||
|             resetResponse(filename, response); | ||||
|             ClassPathResource templateResource = new ClassPathResource(templatePath); | ||||
|             ExcelWriter excelWriter = EasyExcel.write(response.getOutputStream()) | ||||
|                 .withTemplate(templateResource.getStream()) | ||||
|                 .autoCloseStream(false) | ||||
|                 // 大数值自动转换 防止失真 | ||||
|                 .registerConverter(new ExcelBigNumberConvert()) | ||||
|                 .build(); | ||||
|             WriteSheet writeSheet = EasyExcel.writerSheet().build(); | ||||
|             if (CollUtil.isEmpty(data)) { | ||||
|                 throw new IllegalArgumentException("数据为空"); | ||||
|             } | ||||
|             for (Map.Entry<String, Object> map : data.entrySet()) { | ||||
|                 // 设置列表后续还有数据 | ||||
|                 FillConfig fillConfig = FillConfig.builder().forceNewRow(Boolean.TRUE).build(); | ||||
|                 if (map.getValue() instanceof Collection) { | ||||
|                     // 多表导出必须使用 FillWrapper | ||||
|                     excelWriter.fill(new FillWrapper(map.getKey(), (Collection<?>) map.getValue()), fillConfig, writeSheet); | ||||
|                 } else { | ||||
|                     excelWriter.fill(map.getValue(), writeSheet); | ||||
|                 } | ||||
|             } | ||||
|             excelWriter.finish(); | ||||
|         } catch (IOException e) { | ||||
|             throw new RuntimeException("导出Excel异常"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 重置响应体 | ||||
|      */ | ||||
|     private static void resetResponse(String sheetName, HttpServletResponse response) throws UnsupportedEncodingException { | ||||
|         String filename = encodingFilename(sheetName); | ||||
|         response.reset(); | ||||
|         FileUtils.setAttachmentResponseHeader(response, filename); | ||||
|         response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=UTF-8"); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 解析导出值 0=男,1=女,2=未知 | ||||
|      * | ||||
| @@ -103,7 +212,7 @@ public class ExcelUtil { | ||||
|         String[] convertSource = converterExp.split(","); | ||||
|         for (String item : convertSource) { | ||||
|             String[] itemArray = item.split("="); | ||||
|             if (StringUtils.containsAny(separator, propertyValue)) { | ||||
|             if (StringUtils.containsAny(propertyValue, separator)) { | ||||
|                 for (String value : propertyValue.split(separator)) { | ||||
|                     if (itemArray[0].equals(value)) { | ||||
|                         propertyString.append(itemArray[1] + separator); | ||||
| @@ -132,7 +241,7 @@ public class ExcelUtil { | ||||
|         String[] convertSource = converterExp.split(","); | ||||
|         for (String item : convertSource) { | ||||
|             String[] itemArray = item.split("="); | ||||
|             if (StringUtils.containsAny(separator, propertyValue)) { | ||||
|             if (StringUtils.containsAny(propertyValue, separator)) { | ||||
|                 for (String value : propertyValue.split(separator)) { | ||||
|                     if (itemArray[1].equals(value)) { | ||||
|                         propertyString.append(itemArray[0] + separator); | ||||
|   | ||||
| @@ -0,0 +1,75 @@ | ||||
| package com.ruoyi.common.utils.redis; | ||||
|  | ||||
| import com.ruoyi.common.utils.spring.SpringUtils; | ||||
| import lombok.AccessLevel; | ||||
| import lombok.NoArgsConstructor; | ||||
| import org.redisson.api.RMap; | ||||
| import org.springframework.cache.Cache; | ||||
| import org.springframework.cache.CacheManager; | ||||
|  | ||||
| import java.util.Set; | ||||
|  | ||||
| /** | ||||
|  * 缓存操作工具类 {@link } | ||||
|  * | ||||
|  * @author Michelle.Chung | ||||
|  * @date 2022/8/13 | ||||
|  */ | ||||
| @NoArgsConstructor(access = AccessLevel.PRIVATE) | ||||
| @SuppressWarnings(value = {"unchecked"}) | ||||
| public class CacheUtils { | ||||
|  | ||||
|     private static final CacheManager CACHE_MANAGER = SpringUtils.getBean(CacheManager.class); | ||||
|  | ||||
|     /** | ||||
|      * 获取缓存组内所有的KEY | ||||
|      * | ||||
|      * @param cacheNames 缓存组名称 | ||||
|      */ | ||||
|     public static Set<Object> keys(String cacheNames) { | ||||
|         RMap<Object, Object> rmap = (RMap<Object, Object>) CACHE_MANAGER.getCache(cacheNames).getNativeCache(); | ||||
|         return rmap.keySet(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取缓存值 | ||||
|      * | ||||
|      * @param cacheNames 缓存组名称 | ||||
|      * @param key        缓存key | ||||
|      */ | ||||
|     public static <T> T get(String cacheNames, Object key) { | ||||
|         Cache.ValueWrapper wrapper = CACHE_MANAGER.getCache(cacheNames).get(key); | ||||
|         return wrapper != null ? (T) wrapper.get() : null; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 保存缓存值 | ||||
|      * | ||||
|      * @param cacheNames 缓存组名称 | ||||
|      * @param key        缓存key | ||||
|      * @param value      缓存值 | ||||
|      */ | ||||
|     public static void put(String cacheNames, Object key, Object value) { | ||||
|         CACHE_MANAGER.getCache(cacheNames).put(key, value); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 删除缓存值 | ||||
|      * | ||||
|      * @param cacheNames 缓存组名称 | ||||
|      * @param key        缓存key | ||||
|      */ | ||||
|     public static void evict(String cacheNames, Object key) { | ||||
|         CACHE_MANAGER.getCache(cacheNames).evict(key); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 清空缓存值 | ||||
|      * | ||||
|      * @param cacheNames 缓存组名称 | ||||
|      */ | ||||
|     public static void clear(String cacheNames) { | ||||
|         CACHE_MANAGER.getCache(cacheNames).clear(); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -5,7 +5,6 @@ import lombok.AccessLevel; | ||||
| import lombok.NoArgsConstructor; | ||||
| import org.redisson.api.*; | ||||
|  | ||||
| import java.util.Comparator; | ||||
| import java.util.concurrent.TimeUnit; | ||||
| import java.util.function.Consumer; | ||||
|  | ||||
| @@ -30,6 +29,43 @@ public class QueueUtils { | ||||
|         return CLIENT; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 添加普通队列数据 | ||||
|      * | ||||
|      * @param queueName 队列名 | ||||
|      * @param data      数据 | ||||
|      */ | ||||
|     public static <T> boolean addQueueObject(String queueName, T data) { | ||||
|         RBlockingQueue<T> queue = CLIENT.getBlockingQueue(queueName); | ||||
|         return queue.offer(data); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 通用获取一个队列数据 没有数据返回 null(不支持延迟队列) | ||||
|      * | ||||
|      * @param queueName 队列名 | ||||
|      */ | ||||
|     public static <T> T getQueueObject(String queueName) { | ||||
|         RBlockingQueue<T> queue = CLIENT.getBlockingQueue(queueName); | ||||
|         return queue.poll(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 通用删除队列数据(不支持延迟队列) | ||||
|      */ | ||||
|     public static <T> boolean removeQueueObject(String queueName, T data) { | ||||
|         RBlockingQueue<T> queue = CLIENT.getBlockingQueue(queueName); | ||||
|         return queue.remove(data); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 通用销毁队列 所有阻塞监听 报错(不支持延迟队列) | ||||
|      */ | ||||
|     public static <T> boolean destroyQueue(String queueName) { | ||||
|         RBlockingQueue<T> queue = CLIENT.getBlockingQueue(queueName); | ||||
|         return queue.delete(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 添加延迟队列数据 默认毫秒 | ||||
|      * | ||||
| @@ -84,32 +120,6 @@ public class QueueUtils { | ||||
|         delayedQueue.destroy(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 尝试设置 优先队列比较器 用于排序优先级 | ||||
|      * | ||||
|      * @param queueName  队列名 | ||||
|      * @param comparator 比较器 | ||||
|      */ | ||||
|     public static <T> boolean trySetPriorityQueueComparator(String queueName, Comparator<T> comparator) { | ||||
|         RPriorityBlockingQueue<T> priorityBlockingQueue = CLIENT.getPriorityBlockingQueue(queueName); | ||||
|         return priorityBlockingQueue.trySetComparator(comparator); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 尝试设置 优先队列比较器 用于排序优先级 | ||||
|      * | ||||
|      * @param queueName  队列名 | ||||
|      * @param comparator 比较器 | ||||
|      * @param destroy    已存在是否销毁 | ||||
|      */ | ||||
|     public static <T> boolean trySetPriorityQueueComparator(String queueName, Comparator<T> comparator, boolean destroy) { | ||||
|         RPriorityBlockingQueue<T> priorityBlockingQueue = CLIENT.getPriorityBlockingQueue(queueName); | ||||
|         if (priorityBlockingQueue.isExists() && destroy) { | ||||
|             destroyPriorityQueueObject(queueName); | ||||
|         } | ||||
|         return priorityBlockingQueue.trySetComparator(comparator); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 添加优先队列数据 | ||||
|      * | ||||
| @@ -121,32 +131,6 @@ public class QueueUtils { | ||||
|         return priorityBlockingQueue.offer(data); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取一个优先队列数据 没有数据返回 null | ||||
|      * | ||||
|      * @param queueName 队列名 | ||||
|      */ | ||||
|     public static <T> T getPriorityQueueObject(String queueName) { | ||||
|         RPriorityBlockingQueue<T> priorityBlockingQueue = CLIENT.getPriorityBlockingQueue(queueName); | ||||
|         return priorityBlockingQueue.poll(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 删除优先队列数据 | ||||
|      */ | ||||
|     public static <T> boolean removePriorityQueueObject(String queueName, T data) { | ||||
|         RPriorityBlockingQueue<T> priorityBlockingQueue = CLIENT.getPriorityBlockingQueue(queueName); | ||||
|         return priorityBlockingQueue.remove(data); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 销毁优先队列 | ||||
|      */ | ||||
|     public static boolean destroyPriorityQueueObject(String queueName) { | ||||
|         RPriorityBlockingQueue<?> priorityBlockingQueue = CLIENT.getPriorityBlockingQueue(queueName); | ||||
|         return priorityBlockingQueue.delete(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 尝试设置 有界队列 容量 用于限制数量 | ||||
|      * | ||||
| @@ -168,7 +152,7 @@ public class QueueUtils { | ||||
|     public static <T> boolean trySetBoundedQueueCapacity(String queueName, int capacity, boolean destroy) { | ||||
|         RBoundedBlockingQueue<T> boundedBlockingQueue = CLIENT.getBoundedBlockingQueue(queueName); | ||||
|         if (boundedBlockingQueue.isExists() && destroy) { | ||||
|             destroyBoundedQueueObject(queueName); | ||||
|             destroyQueue(queueName); | ||||
|         } | ||||
|         return boundedBlockingQueue.trySetCapacity(capacity); | ||||
|     } | ||||
| @@ -185,32 +169,6 @@ public class QueueUtils { | ||||
|         return boundedBlockingQueue.offer(data); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取一个有界队列数据 没有数据返回 null | ||||
|      * | ||||
|      * @param queueName 队列名 | ||||
|      */ | ||||
|     public static <T> T getBoundedQueueObject(String queueName) { | ||||
|         RBoundedBlockingQueue<T> boundedBlockingQueue = CLIENT.getBoundedBlockingQueue(queueName); | ||||
|         return boundedBlockingQueue.poll(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 删除有界队列数据 | ||||
|      */ | ||||
|     public static <T> boolean removeBoundedQueueObject(String queueName, T data) { | ||||
|         RBoundedBlockingQueue<T> boundedBlockingQueue = CLIENT.getBoundedBlockingQueue(queueName); | ||||
|         return boundedBlockingQueue.remove(data); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 销毁有界队列 | ||||
|      */ | ||||
|     public static boolean destroyBoundedQueueObject(String queueName) { | ||||
|         RBoundedBlockingQueue<?> boundedBlockingQueue = CLIENT.getBoundedBlockingQueue(queueName); | ||||
|         return boundedBlockingQueue.delete(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 订阅阻塞队列(可订阅所有实现类 例如: 延迟 优先 有界 等) | ||||
|      */ | ||||
|   | ||||
| @@ -1,17 +1,19 @@ | ||||
| package com.ruoyi.common.utils.redis; | ||||
|  | ||||
| import cn.hutool.core.collection.IterUtil; | ||||
| import com.ruoyi.common.utils.spring.SpringUtils; | ||||
| import lombok.AccessLevel; | ||||
| import lombok.NoArgsConstructor; | ||||
| import org.redisson.api.*; | ||||
| import org.redisson.config.Config; | ||||
|  | ||||
| import java.time.Duration; | ||||
| import java.util.Collection; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.Set; | ||||
| import java.util.concurrent.TimeUnit; | ||||
| import java.util.function.Consumer; | ||||
| import java.util.stream.Collectors; | ||||
| import java.util.stream.Stream; | ||||
|  | ||||
| /** | ||||
|  * redis 工具类 | ||||
| @@ -25,6 +27,14 @@ public class RedisUtils { | ||||
|  | ||||
|     private static final RedissonClient CLIENT = SpringUtils.getBean(RedissonClient.class); | ||||
|  | ||||
|     public static NameMapper getNameMapper() { | ||||
|         Config config = CLIENT.getConfig(); | ||||
|         if (config.isClusterConfig()) { | ||||
|             return config.useClusterServers().getNameMapper(); | ||||
|         } | ||||
|         return config.useSingleServer().getNameMapper(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 限流 | ||||
|      * | ||||
| @@ -100,14 +110,13 @@ public class RedisUtils { | ||||
|      * @since Redis 6.X 以上使用 setAndKeepTTL 兼容 5.X 方案 | ||||
|      */ | ||||
|     public static <T> void setCacheObject(final String key, final T value, final boolean isSaveTtl) { | ||||
|         RBucket<Object> bucket = CLIENT.getBucket(key); | ||||
|         RBucket<T> bucket = CLIENT.getBucket(key); | ||||
|         if (isSaveTtl) { | ||||
|             try { | ||||
|                 bucket.setAndKeepTTL(value); | ||||
|             } catch (Exception e) { | ||||
|                 long timeToLive = bucket.remainTimeToLive(); | ||||
|                 bucket.set(value); | ||||
|                 bucket.expire(timeToLive, TimeUnit.MILLISECONDS); | ||||
|                 setCacheObject(key, value, Duration.ofMillis(timeToLive)); | ||||
|             } | ||||
|         } else { | ||||
|             bucket.set(value); | ||||
| @@ -119,13 +128,14 @@ public class RedisUtils { | ||||
|      * | ||||
|      * @param key      缓存的键值 | ||||
|      * @param value    缓存的值 | ||||
|      * @param timeout  时间 | ||||
|      * @param timeUnit 时间颗粒度 | ||||
|      * @param duration 时间 | ||||
|      */ | ||||
|     public static <T> void setCacheObject(final String key, final T value, final long timeout, final TimeUnit timeUnit) { | ||||
|         RBucket<T> result = CLIENT.getBucket(key); | ||||
|         result.set(value); | ||||
|         result.expire(timeout, timeUnit); | ||||
|     public static <T> void setCacheObject(final String key, final T value, final Duration duration) { | ||||
|         RBatch batch = CLIENT.createBatch(); | ||||
|         RBucketAsync<T> bucket = batch.getBucket(key); | ||||
|         bucket.setAsync(value); | ||||
|         bucket.expireAsync(duration); | ||||
|         batch.execute(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -149,20 +159,19 @@ public class RedisUtils { | ||||
|      * @return true=设置成功;false=设置失败 | ||||
|      */ | ||||
|     public static boolean expire(final String key, final long timeout) { | ||||
|         return expire(key, timeout, TimeUnit.SECONDS); | ||||
|         return expire(key, Duration.ofSeconds(timeout)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 设置有效时间 | ||||
|      * | ||||
|      * @param key      Redis键 | ||||
|      * @param timeout 超时时间 | ||||
|      * @param unit    时间单位 | ||||
|      * @param duration 超时时间 | ||||
|      * @return true=设置成功;false=设置失败 | ||||
|      */ | ||||
|     public static boolean expire(final String key, final long timeout, final TimeUnit unit) { | ||||
|     public static boolean expire(final String key, final Duration duration) { | ||||
|         RBucket rBucket = CLIENT.getBucket(key); | ||||
|         return rBucket.expire(timeout, unit); | ||||
|         return rBucket.expire(duration); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -366,6 +375,50 @@ public class RedisUtils { | ||||
|         return rMap.getAll(hKeys); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 设置原子值 | ||||
|      * | ||||
|      * @param key   Redis键 | ||||
|      * @param value 值 | ||||
|      */ | ||||
|     public static void setAtomicValue(String key, long value) { | ||||
|         RAtomicLong atomic = CLIENT.getAtomicLong(key); | ||||
|         atomic.set(value); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获取原子值 | ||||
|      * | ||||
|      * @param key Redis键 | ||||
|      * @return 当前值 | ||||
|      */ | ||||
|     public static long getAtomicValue(String key) { | ||||
|         RAtomicLong atomic = CLIENT.getAtomicLong(key); | ||||
|         return atomic.get(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 递增原子值 | ||||
|      * | ||||
|      * @param key Redis键 | ||||
|      * @return 当前值 | ||||
|      */ | ||||
|     public static long incrAtomicValue(String key) { | ||||
|         RAtomicLong atomic = CLIENT.getAtomicLong(key); | ||||
|         return atomic.incrementAndGet(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 递减原子值 | ||||
|      * | ||||
|      * @param key Redis键 | ||||
|      * @return 当前值 | ||||
|      */ | ||||
|     public static long decrAtomicValue(String key) { | ||||
|         RAtomicLong atomic = CLIENT.getAtomicLong(key); | ||||
|         return atomic.decrementAndGet(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 获得缓存的基本对象列表 | ||||
|      * | ||||
| @@ -373,8 +426,17 @@ public class RedisUtils { | ||||
|      * @return 对象列表 | ||||
|      */ | ||||
|     public static Collection<String> keys(final String pattern) { | ||||
|         Iterable<String> iterable = CLIENT.getKeys().getKeysByPattern(pattern); | ||||
|         return IterUtil.toList(iterable); | ||||
|         Stream<String> stream = CLIENT.getKeys().getKeysStreamByPattern(getNameMapper().map(pattern)); | ||||
|         return stream.map(key -> getNameMapper().unmap(key)).collect(Collectors.toList()); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 删除缓存的基本对象列表 | ||||
|      * | ||||
|      * @param pattern 字符串前缀 | ||||
|      */ | ||||
|     public static void deleteKeys(final String pattern) { | ||||
|         CLIENT.getKeys().deleteByPattern(getNameMapper().map(pattern)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -384,6 +446,6 @@ public class RedisUtils { | ||||
|      */ | ||||
|     public static Boolean hasKey(String key) { | ||||
|         RKeys rKeys = CLIENT.getKeys(); | ||||
|         return rKeys.countExists(key) > 0; | ||||
|         return rKeys.countExists(getNameMapper().map(key)) > 0; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -5,7 +5,7 @@ | ||||
|     <parent> | ||||
|         <artifactId>ruoyi-vue-plus</artifactId> | ||||
|         <groupId>com.ruoyi</groupId> | ||||
|         <version>4.1.0</version> | ||||
|         <version>4.3.0</version> | ||||
|     </parent> | ||||
|     <modelVersion>4.0.0</modelVersion> | ||||
|  | ||||
| @@ -23,6 +23,22 @@ | ||||
|             <artifactId>ruoyi-common</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <dependency> | ||||
|             <groupId>com.ruoyi</groupId> | ||||
|             <artifactId>ruoyi-sms</artifactId> | ||||
|         </dependency> | ||||
|  | ||||
|         <!-- 短信 用哪个导入哪个依赖 --> | ||||
| <!--        <dependency>--> | ||||
| <!--            <groupId>com.aliyun</groupId>--> | ||||
| <!--            <artifactId>dysmsapi20170525</artifactId>--> | ||||
| <!--        </dependency>--> | ||||
|  | ||||
| <!--        <dependency>--> | ||||
| <!--            <groupId>com.tencentcloudapi</groupId>--> | ||||
| <!--            <artifactId>tencentcloud-sdk-java</artifactId>--> | ||||
| <!--        </dependency>--> | ||||
|  | ||||
|     </dependencies> | ||||
|  | ||||
| </project> | ||||
|   | ||||
| @@ -0,0 +1,52 @@ | ||||
| package com.ruoyi.demo.controller; | ||||
|  | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.utils.email.MailUtils; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.GetMapping; | ||||
| import org.springframework.web.bind.annotation.RequestMapping; | ||||
| import org.springframework.web.bind.annotation.RestController; | ||||
|  | ||||
| import java.io.File; | ||||
|  | ||||
|  | ||||
| /** | ||||
|  * 邮件发送案例 | ||||
|  * | ||||
|  * @author Michelle.Chung | ||||
|  */ | ||||
| @Validated | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/demo/mail") | ||||
| public class MailController { | ||||
|  | ||||
|     /** | ||||
|      * 发送邮件 | ||||
|      * | ||||
|      * @param to      接收人 | ||||
|      * @param subject 标题 | ||||
|      * @param text    内容 | ||||
|      */ | ||||
|     @GetMapping("/sendSimpleMessage") | ||||
|     public R<Void> sendSimpleMessage(String to, String subject, String text) { | ||||
|         MailUtils.sendText(to, subject, text); | ||||
|         return R.ok(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 发送邮件(带附件) | ||||
|      * | ||||
|      * @param to       接收人 | ||||
|      * @param subject  标题 | ||||
|      * @param text     内容 | ||||
|      * @param filePath 附件路径 | ||||
|      */ | ||||
|     @GetMapping("/sendMessageWithAttachment") | ||||
|     public R<Void> sendMessageWithAttachment(String to, String subject, String text, String filePath) { | ||||
|         MailUtils.sendText(to, subject, text, new File(filePath)); | ||||
|         return R.ok(); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,9 +1,8 @@ | ||||
| package com.ruoyi.demo.controller; | ||||
|  | ||||
| import com.ruoyi.common.constant.CacheNames; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.utils.redis.RedisUtils; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.cache.annotation.CacheEvict; | ||||
| import org.springframework.cache.annotation.CachePut; | ||||
| @@ -12,7 +11,7 @@ import org.springframework.web.bind.annotation.GetMapping; | ||||
| import org.springframework.web.bind.annotation.RequestMapping; | ||||
| import org.springframework.web.bind.annotation.RestController; | ||||
|  | ||||
| import java.util.concurrent.TimeUnit; | ||||
| import java.time.Duration; | ||||
|  | ||||
| /** | ||||
|  * spring-cache 演示案例 | ||||
| @@ -20,8 +19,7 @@ import java.util.concurrent.TimeUnit; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| // 类级别 缓存统一配置 | ||||
| //@CacheConfig(cacheNames = "redissonCacheMap") | ||||
| @Api(value = "spring-cache 演示案例", tags = {"spring-cache 演示案例"}) | ||||
| //@CacheConfig(cacheNames = CacheNames.DEMO_CACHE) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/demo/cache") | ||||
| @@ -39,10 +37,9 @@ public class RedisCacheController { | ||||
|      * 重点说明: 缓存注解严谨与其他筛选数据功能一起使用 | ||||
|      * 例如: 数据权限注解 会造成 缓存击穿 与 数据不一致问题 | ||||
|      * <p> | ||||
|      * cacheNames 为配置文件内 groupId | ||||
|      * cacheNames 命名规则 查看 {@link CacheNames} 注释 支持多参数 | ||||
|      */ | ||||
|     @ApiOperation("测试 @Cacheable") | ||||
|     @Cacheable(cacheNames = "redissonCacheMap", key = "#key", condition = "#key != null") | ||||
|     @Cacheable(cacheNames = "demo:cache#60s#10m#20", key = "#key", condition = "#key != null") | ||||
|     @GetMapping("/test1") | ||||
|     public R<String> test1(String key, String value) { | ||||
|         return R.ok("操作成功", value); | ||||
| @@ -52,12 +49,11 @@ public class RedisCacheController { | ||||
|      * 测试 @CachePut | ||||
|      * <p> | ||||
|      * 加了@CachePut注解的方法,会把方法的返回值put到缓存里面缓存起来,供其它地方使用 | ||||
|      * 它「通常用在新增方法上」 | ||||
|      * 它「通常用在新增或者实时更新方法上」 | ||||
|      * <p> | ||||
|      * cacheNames 为 配置文件内 groupId | ||||
|      * cacheNames 命名规则 查看 {@link CacheNames} 注释 支持多参数 | ||||
|      */ | ||||
|     @ApiOperation("测试 @CachePut") | ||||
|     @CachePut(cacheNames = "redissonCacheMap", key = "#key", condition = "#key != null") | ||||
|     @CachePut(cacheNames = CacheNames.DEMO_CACHE, key = "#key", condition = "#key != null") | ||||
|     @GetMapping("/test2") | ||||
|     public R<String> test2(String key, String value) { | ||||
|         return R.ok("操作成功", value); | ||||
| @@ -67,12 +63,11 @@ public class RedisCacheController { | ||||
|      * 测试 @CacheEvict | ||||
|      * <p> | ||||
|      * 使用了CacheEvict注解的方法,会清空指定缓存 | ||||
|      * 「一般用在更新或者删除的方法上」 | ||||
|      * 「一般用在删除的方法上」 | ||||
|      * <p> | ||||
|      * cacheNames 为 配置文件内 groupId | ||||
|      * cacheNames 命名规则 查看 {@link CacheNames} 注释 支持多参数 | ||||
|      */ | ||||
|     @ApiOperation("测试 @CacheEvict") | ||||
|     @CacheEvict(cacheNames = "redissonCacheMap", key = "#key", condition = "#key != null") | ||||
|     @CacheEvict(cacheNames = CacheNames.DEMO_CACHE, key = "#key", condition = "#key != null") | ||||
|     @GetMapping("/test3") | ||||
|     public R<String> test3(String key, String value) { | ||||
|         return R.ok("操作成功", value); | ||||
| @@ -83,11 +78,10 @@ public class RedisCacheController { | ||||
|      * 手动设置过期时间10秒 | ||||
|      * 11秒后获取 判断是否相等 | ||||
|      */ | ||||
|     @ApiOperation("测试设置过期时间") | ||||
|     @GetMapping("/test6") | ||||
|     public R<Boolean> test6(String key, String value) { | ||||
|         RedisUtils.setCacheObject(key, value); | ||||
|         boolean flag = RedisUtils.expire(key, 10, TimeUnit.SECONDS); | ||||
|         boolean flag = RedisUtils.expire(key, Duration.ofSeconds(10)); | ||||
|         System.out.println("***********" + flag); | ||||
|         try { | ||||
|             Thread.sleep(11 * 1000); | ||||
|   | ||||
| @@ -5,8 +5,6 @@ import com.baomidou.lock.LockTemplate; | ||||
| import com.baomidou.lock.annotation.Lock4j; | ||||
| import com.baomidou.lock.executor.RedissonLockExecutor; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import lombok.extern.slf4j.Slf4j; | ||||
| import org.springframework.beans.factory.annotation.Autowired; | ||||
| import org.springframework.web.bind.annotation.GetMapping; | ||||
| @@ -21,7 +19,6 @@ import java.time.LocalTime; | ||||
|  * | ||||
|  * @author shenxinquan | ||||
|  */ | ||||
| @Api(value = "测试分布式锁的样例", tags = {"测试分布式锁的样例"}) | ||||
| @Slf4j | ||||
| @RestController | ||||
| @RequestMapping("/demo/redisLock") | ||||
| @@ -33,7 +30,6 @@ public class RedisLockController { | ||||
|     /** | ||||
|      * 测试lock4j 注解 | ||||
|      */ | ||||
|     @ApiOperation("测试lock4j 注解") | ||||
|     @Lock4j(keys = {"#key"}) | ||||
|     @GetMapping("/testLock4j") | ||||
|     public R<String> testLock4j(String key, String value) { | ||||
| @@ -50,7 +46,6 @@ public class RedisLockController { | ||||
|     /** | ||||
|      * 测试lock4j 工具 | ||||
|      */ | ||||
|     @ApiOperation("测试lock4j 工具") | ||||
|     @GetMapping("/testLock4jLockTemplate") | ||||
|     public R<String> testLock4jLockTemplate(String key, String value) { | ||||
|         final LockInfo lockInfo = lockTemplate.lock(key, 30000L, 5000L, RedissonLockExecutor.class); | ||||
|   | ||||
| @@ -2,9 +2,6 @@ package com.ruoyi.demo.controller; | ||||
|  | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.utils.redis.RedisUtils; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import io.swagger.annotations.ApiParam; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.web.bind.annotation.GetMapping; | ||||
| import org.springframework.web.bind.annotation.RequestMapping; | ||||
| @@ -15,24 +12,32 @@ import org.springframework.web.bind.annotation.RestController; | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Api(value = "Redis发布订阅 演示案例", tags = {"Redis发布订阅"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/demo/redis/pubsub") | ||||
| public class RedisPubSubController { | ||||
|  | ||||
|     @ApiOperation("发布消息") | ||||
|     /** | ||||
|      * 发布消息 | ||||
|      * | ||||
|      * @param key   通道Key | ||||
|      * @param value 发送内容 | ||||
|      */ | ||||
|     @GetMapping("/pub") | ||||
|     public R<Void> pub(@ApiParam("通道Key") String key, @ApiParam("发送内容") String value) { | ||||
|     public R<Void> pub(String key, String value) { | ||||
|         RedisUtils.publish(key, value, consumer -> { | ||||
|             System.out.println("发布通道 => " + key + ", 发送值 => " + value); | ||||
|         }); | ||||
|         return R.ok("操作成功"); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("订阅消息") | ||||
|     /** | ||||
|      * 订阅消息 | ||||
|      * | ||||
|      * @param key 通道Key | ||||
|      */ | ||||
|     @GetMapping("/sub") | ||||
|     public R<Void> sub(@ApiParam("通道Key") String key) { | ||||
|     public R<Void> sub(String key) { | ||||
|         RedisUtils.subscribe(key, String.class, msg -> { | ||||
|             System.out.println("订阅通道 => " + key + ", 接收值 => " + msg); | ||||
|         }); | ||||
|   | ||||
| @@ -3,8 +3,6 @@ package com.ruoyi.demo.controller; | ||||
| import com.ruoyi.common.annotation.RateLimiter; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.enums.LimitType; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import lombok.extern.slf4j.Slf4j; | ||||
| import org.springframework.web.bind.annotation.GetMapping; | ||||
| import org.springframework.web.bind.annotation.RequestMapping; | ||||
| @@ -16,7 +14,6 @@ import org.springframework.web.bind.annotation.RestController; | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Api(value = "测试分布式限流样例", tags = {"测试分布式限流样例"}) | ||||
| @Slf4j | ||||
| @RestController | ||||
| @RequestMapping("/demo/rateLimiter") | ||||
| @@ -26,7 +23,6 @@ public class RedisRateLimiterController { | ||||
|      * 测试全局限流 | ||||
|      * 全局影响 | ||||
|      */ | ||||
|     @ApiOperation("测试全局限流") | ||||
|     @RateLimiter(count = 2, time = 10) | ||||
|     @GetMapping("/test") | ||||
|     public R<String> test(String value) { | ||||
| @@ -37,7 +33,6 @@ public class RedisRateLimiterController { | ||||
|      * 测试请求IP限流 | ||||
|      * 同一IP请求受影响 | ||||
|      */ | ||||
|     @ApiOperation("测试请求IP限流") | ||||
|     @RateLimiter(count = 2, time = 10, limitType = LimitType.IP) | ||||
|     @GetMapping("/testip") | ||||
|     public R<String> testip(String value) { | ||||
| @@ -48,7 +43,6 @@ public class RedisRateLimiterController { | ||||
|      * 测试集群实例限流 | ||||
|      * 启动两个后端服务互不影响 | ||||
|      */ | ||||
|     @ApiOperation("测试集群实例限流") | ||||
|     @RateLimiter(count = 2, time = 10, limitType = LimitType.CLUSTER) | ||||
|     @GetMapping("/testcluster") | ||||
|     public R<String> testcluster(String value) { | ||||
|   | ||||
| @@ -0,0 +1,76 @@ | ||||
| package com.ruoyi.demo.controller; | ||||
|  | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.utils.spring.SpringUtils; | ||||
| import com.ruoyi.sms.config.properties.SmsProperties; | ||||
| import com.ruoyi.sms.core.SmsTemplate; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.GetMapping; | ||||
| import org.springframework.web.bind.annotation.RequestMapping; | ||||
| import org.springframework.web.bind.annotation.RestController; | ||||
|  | ||||
| import java.util.HashMap; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * 短信演示案例 | ||||
|  * 请先阅读文档 否则无法使用 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  * @version 4.2.0 | ||||
|  */ | ||||
| @Validated | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/demo/sms") | ||||
| public class SmsController { | ||||
|  | ||||
|     private final SmsProperties smsProperties; | ||||
| //    private final SmsTemplate smsTemplate; // 可以使用spring注入 | ||||
| //    private final AliyunSmsTemplate smsTemplate; // 也可以注入某个厂家的模板工具 | ||||
|  | ||||
|     /** | ||||
|      * 发送短信Aliyun | ||||
|      * | ||||
|      * @param phones     电话号 | ||||
|      * @param templateId 模板ID | ||||
|      */ | ||||
|     @GetMapping("/sendAliyun") | ||||
|     public R<Object> sendAliyun(String phones, String templateId) { | ||||
|         if (!smsProperties.getEnabled()) { | ||||
|             return R.fail("当前系统没有开启短信功能!"); | ||||
|         } | ||||
|         if (!SpringUtils.containsBean("aliyunSmsTemplate")) { | ||||
|             return R.fail("阿里云依赖未引入!"); | ||||
|         } | ||||
|         SmsTemplate smsTemplate = SpringUtils.getBean(SmsTemplate.class); | ||||
|         Map<String, String> map = new HashMap<>(1); | ||||
|         map.put("code", "1234"); | ||||
|         Object send = smsTemplate.send(phones, templateId, map); | ||||
|         return R.ok(send); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 发送短信Tencent | ||||
|      * | ||||
|      * @param phones     电话号 | ||||
|      * @param templateId 模板ID | ||||
|      */ | ||||
|     @GetMapping("/sendTencent") | ||||
|     public R<Object> sendTencent(String phones, String templateId) { | ||||
|         if (!smsProperties.getEnabled()) { | ||||
|             return R.fail("当前系统没有开启短信功能!"); | ||||
|         } | ||||
|         if (!SpringUtils.containsBean("tencentSmsTemplate")) { | ||||
|             return R.fail("腾讯云依赖未引入!"); | ||||
|         } | ||||
|         SmsTemplate smsTemplate = SpringUtils.getBean(SmsTemplate.class); | ||||
|         Map<String, String> map = new HashMap<>(1); | ||||
| //        map.put("2", "测试测试"); | ||||
|         map.put("1", "1234"); | ||||
|         Object send = smsTemplate.send(phones, templateId, map); | ||||
|         return R.ok(send); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,24 +1,18 @@ | ||||
| package com.ruoyi.demo.controller; | ||||
|  | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiImplicitParam; | ||||
| import io.swagger.annotations.ApiImplicitParams; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import org.springframework.http.MediaType; | ||||
| import org.springframework.web.bind.annotation.PostMapping; | ||||
| import org.springframework.web.bind.annotation.RequestMapping; | ||||
| import org.springframework.web.bind.annotation.RequestPart; | ||||
| import org.springframework.web.bind.annotation.RestController; | ||||
| import org.springframework.web.multipart.MultipartFile; | ||||
|  | ||||
| import java.io.File; | ||||
|  | ||||
| /** | ||||
|  * swagger3 用法示例 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Api(value = "演示swagger3控制器", tags = {"演示swagger3接口"}) | ||||
| @RestController | ||||
| @RequestMapping("/swagger/demo") | ||||
| public class Swagger3DemoController { | ||||
| @@ -26,12 +20,10 @@ public class Swagger3DemoController { | ||||
|     /** | ||||
|      * 上传请求 | ||||
|      * 必须使用 @RequestPart 注解标注为文件 | ||||
|      * | ||||
|      * @param file 文件 | ||||
|      */ | ||||
|     @ApiOperation(value = "通用上传请求") | ||||
|     @ApiImplicitParams({ | ||||
|         @ApiImplicitParam(name = "file", value = "文件", paramType = "query", dataTypeClass = File.class, required = true) | ||||
|     }) | ||||
|     @PostMapping(value = "/upload") | ||||
|     @PostMapping(value = "/upload", consumes = MediaType.MULTIPART_FORM_DATA_VALUE) | ||||
|     public R<String> upload(@RequestPart("file") MultipartFile file) { | ||||
|         return R.ok("操作成功", file.getOriginalFilename()); | ||||
|     } | ||||
|   | ||||
| @@ -5,8 +5,6 @@ import com.ruoyi.common.core.controller.BaseController; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.demo.domain.TestDemo; | ||||
| import com.ruoyi.demo.mapper.TestDemoMapper; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.web.bind.annotation.DeleteMapping; | ||||
| import org.springframework.web.bind.annotation.PostMapping; | ||||
| @@ -22,7 +20,6 @@ import java.util.List; | ||||
|  * @author Lion Li | ||||
|  * @date 2021-05-30 | ||||
|  */ | ||||
| @Api(value = "测试批量方法", tags = {"测试批量方法"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/demo/batch") | ||||
| @@ -38,7 +35,6 @@ public class TestBatchController extends BaseController { | ||||
|      * <p> | ||||
|      * 3.5.0 版本 增加 rewriteBatchedStatements=true 批处理参数 使 MP 原生批处理可以达到同样的速度 | ||||
|      */ | ||||
|     @ApiOperation(value = "新增批量方法") | ||||
|     @PostMapping("/add") | ||||
| //    @DS("slave") | ||||
|     public R<Void> add() { | ||||
| @@ -58,7 +54,6 @@ public class TestBatchController extends BaseController { | ||||
|      * <p> | ||||
|      * 3.5.0 版本 增加 rewriteBatchedStatements=true 批处理参数 使 MP 原生批处理可以达到同样的速度 | ||||
|      */ | ||||
|     @ApiOperation(value = "新增或更新批量方法") | ||||
|     @PostMapping("/addOrUpdate") | ||||
| //    @DS("slave") | ||||
|     public R<Void> addOrUpdate() { | ||||
| @@ -68,7 +63,8 @@ public class TestBatchController extends BaseController { | ||||
|             testDemo.setOrderNum(-1); | ||||
|             testDemo.setTestKey("批量新增"); | ||||
|             testDemo.setValue("测试新增"); | ||||
|             list.add(testDemo);        } | ||||
|             list.add(testDemo); | ||||
|         } | ||||
|         testDemoMapper.insertBatch(list); | ||||
|         for (int i = 0; i < list.size(); i++) { | ||||
|             TestDemo testDemo = list.get(i); | ||||
| @@ -84,7 +80,6 @@ public class TestBatchController extends BaseController { | ||||
|     /** | ||||
|      * 删除批量方法 | ||||
|      */ | ||||
|     @ApiOperation(value = "删除批量方法") | ||||
|     @DeleteMapping() | ||||
| //    @DS("slave") | ||||
|     public R<Void> remove() { | ||||
|   | ||||
| @@ -5,8 +5,8 @@ import cn.hutool.core.bean.BeanUtil; | ||||
| import com.ruoyi.common.annotation.Log; | ||||
| import com.ruoyi.common.annotation.RepeatSubmit; | ||||
| import com.ruoyi.common.core.controller.BaseController; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.domain.PageQuery; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.core.page.TableDataInfo; | ||||
| import com.ruoyi.common.core.validate.AddGroup; | ||||
| import com.ruoyi.common.core.validate.EditGroup; | ||||
| @@ -20,8 +20,8 @@ import com.ruoyi.demo.domain.bo.TestDemoBo; | ||||
| import com.ruoyi.demo.domain.bo.TestDemoImportVo; | ||||
| import com.ruoyi.demo.domain.vo.TestDemoVo; | ||||
| import com.ruoyi.demo.service.ITestDemoService; | ||||
| import io.swagger.annotations.*; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.http.MediaType; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
| import org.springframework.web.multipart.MultipartFile; | ||||
| @@ -40,7 +40,6 @@ import java.util.concurrent.TimeUnit; | ||||
|  * @date 2021-07-26 | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "测试单表控制器", tags = {"测试单表管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/demo/demo") | ||||
| @@ -51,7 +50,6 @@ public class TestDemoController extends BaseController { | ||||
|     /** | ||||
|      * 查询测试单表列表 | ||||
|      */ | ||||
|     @ApiOperation("查询测试单表列表") | ||||
|     @SaCheckPermission("demo:demo:list") | ||||
|     @GetMapping("/list") | ||||
|     public TableDataInfo<TestDemoVo> list(@Validated(QueryGroup.class) TestDemoBo bo, PageQuery pageQuery) { | ||||
| @@ -61,20 +59,20 @@ public class TestDemoController extends BaseController { | ||||
|     /** | ||||
|      * 自定义分页查询 | ||||
|      */ | ||||
|     @ApiOperation("自定义分页查询") | ||||
|     @SaCheckPermission("demo:demo:list") | ||||
|     @GetMapping("/page") | ||||
|     public TableDataInfo<TestDemoVo> page(@Validated(QueryGroup.class) TestDemoBo bo, PageQuery pageQuery) { | ||||
|         return iTestDemoService.customPageList(bo, pageQuery); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("导入测试-校验") | ||||
|     @ApiImplicitParams({ | ||||
|         @ApiImplicitParam(name = "file", value = "导入文件", dataType = "java.io.File", required = true), | ||||
|     }) | ||||
|     /** | ||||
|      * 导入数据 | ||||
|      * | ||||
|      * @param file 导入文件 | ||||
|      */ | ||||
|     @Log(title = "测试单表", businessType = BusinessType.IMPORT) | ||||
|     @SaCheckPermission("demo:demo:import") | ||||
|     @PostMapping("/importData") | ||||
|     @PostMapping(value = "/importData", consumes = MediaType.MULTIPART_FORM_DATA_VALUE) | ||||
|     public R<Void> importData(@RequestPart("file") MultipartFile file) throws Exception { | ||||
|         ExcelResult<TestDemoImportVo> excelResult = ExcelUtil.importExcel(file.getInputStream(), TestDemoImportVo.class, true); | ||||
|         List<TestDemoImportVo> volist = excelResult.getList(); | ||||
| @@ -86,7 +84,6 @@ public class TestDemoController extends BaseController { | ||||
|     /** | ||||
|      * 导出测试单表列表 | ||||
|      */ | ||||
|     @ApiOperation("导出测试单表列表") | ||||
|     @SaCheckPermission("demo:demo:export") | ||||
|     @Log(title = "测试单表", businessType = BusinessType.EXPORT) | ||||
|     @PostMapping("/export") | ||||
| @@ -101,12 +98,12 @@ public class TestDemoController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 获取测试单表详细信息 | ||||
|      * | ||||
|      * @param id 测试ID | ||||
|      */ | ||||
|     @ApiOperation("获取测试单表详细信息") | ||||
|     @SaCheckPermission("demo:demo:query") | ||||
|     @GetMapping("/{id}") | ||||
|     public R<TestDemoVo> getInfo(@ApiParam("测试ID") | ||||
|                                           @NotNull(message = "主键不能为空") | ||||
|     public R<TestDemoVo> getInfo(@NotNull(message = "主键不能为空") | ||||
|                                  @PathVariable("id") Long id) { | ||||
|         return R.ok(iTestDemoService.queryById(id)); | ||||
|     } | ||||
| @@ -114,7 +111,6 @@ public class TestDemoController extends BaseController { | ||||
|     /** | ||||
|      * 新增测试单表 | ||||
|      */ | ||||
|     @ApiOperation("新增测试单表") | ||||
|     @SaCheckPermission("demo:demo:add") | ||||
|     @Log(title = "测试单表", businessType = BusinessType.INSERT) | ||||
|     @RepeatSubmit(interval = 2, timeUnit = TimeUnit.SECONDS, message = "{repeat.submit.message}") | ||||
| @@ -129,7 +125,6 @@ public class TestDemoController extends BaseController { | ||||
|     /** | ||||
|      * 修改测试单表 | ||||
|      */ | ||||
|     @ApiOperation("修改测试单表") | ||||
|     @SaCheckPermission("demo:demo:edit") | ||||
|     @Log(title = "测试单表", businessType = BusinessType.UPDATE) | ||||
|     @RepeatSubmit | ||||
| @@ -140,13 +135,13 @@ public class TestDemoController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 删除测试单表 | ||||
|      * | ||||
|      * @param ids 测试ID串 | ||||
|      */ | ||||
|     @ApiOperation("删除测试单表") | ||||
|     @SaCheckPermission("demo:demo:remove") | ||||
|     @Log(title = "测试单表", businessType = BusinessType.DELETE) | ||||
|     @DeleteMapping("/{ids}") | ||||
|     public R<Void> remove(@ApiParam("测试ID串") | ||||
|                                    @NotEmpty(message = "主键不能为空") | ||||
|     public R<Void> remove(@NotEmpty(message = "主键不能为空") | ||||
|                           @PathVariable Long[] ids) { | ||||
|         return toAjax(iTestDemoService.deleteWithValidByIds(Arrays.asList(ids), true) ? 1 : 0); | ||||
|     } | ||||
|   | ||||
| @@ -0,0 +1,97 @@ | ||||
| package com.ruoyi.demo.controller; | ||||
|  | ||||
| import cn.hutool.core.collection.CollUtil; | ||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | ||||
| import lombok.AllArgsConstructor; | ||||
| import lombok.Data; | ||||
| import org.springframework.web.bind.annotation.GetMapping; | ||||
| import org.springframework.web.bind.annotation.RequestMapping; | ||||
| import org.springframework.web.bind.annotation.RestController; | ||||
|  | ||||
| import javax.servlet.http.HttpServletResponse; | ||||
| import java.util.ArrayList; | ||||
| import java.util.HashMap; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * 测试Excel功能 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @RestController | ||||
| @RequestMapping("/demo/excel") | ||||
| public class TestExcelController { | ||||
|  | ||||
|     /** | ||||
|      * 单列表多数据 | ||||
|      */ | ||||
|     @GetMapping("/exportTemplateOne") | ||||
|     public void exportTemplateOne(HttpServletResponse response) { | ||||
|         Map<String, String> map = new HashMap<>(); | ||||
|         map.put("title", "单列表多数据"); | ||||
|         map.put("test1", "数据测试1"); | ||||
|         map.put("test2", "数据测试2"); | ||||
|         map.put("test3", "数据测试3"); | ||||
|         map.put("test4", "数据测试4"); | ||||
|         map.put("testTest", "666"); | ||||
|         List<TestObj> list = new ArrayList<>(); | ||||
|         list.add(new TestObj("单列表测试1", "列表测试1", "列表测试2", "列表测试3", "列表测试4")); | ||||
|         list.add(new TestObj("单列表测试2", "列表测试5", "列表测试6", "列表测试7", "列表测试8")); | ||||
|         list.add(new TestObj("单列表测试3", "列表测试9", "列表测试10", "列表测试11", "列表测试12")); | ||||
|         ExcelUtil.exportTemplate(CollUtil.newArrayList(map, list), "单列表.xlsx", "excel/单列表.xlsx", response); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 多列表多数据 | ||||
|      */ | ||||
|     @GetMapping("/exportTemplateMuliti") | ||||
|     public void exportTemplateMuliti(HttpServletResponse response) { | ||||
|         Map<String, String> map = new HashMap<>(); | ||||
|         map.put("title1", "标题1"); | ||||
|         map.put("title2", "标题2"); | ||||
|         map.put("title3", "标题3"); | ||||
|         map.put("title4", "标题4"); | ||||
|         map.put("author", "Lion Li"); | ||||
|         List<TestObj1> list1 = new ArrayList<>(); | ||||
|         list1.add(new TestObj1("list1测试1", "list1测试2", "list1测试3")); | ||||
|         list1.add(new TestObj1("list1测试4", "list1测试5", "list1测试6")); | ||||
|         list1.add(new TestObj1("list1测试7", "list1测试8", "list1测试9")); | ||||
|         List<TestObj1> list2 = new ArrayList<>(); | ||||
|         list2.add(new TestObj1("list2测试1", "list2测试2", "list2测试3")); | ||||
|         list2.add(new TestObj1("list2测试4", "list2测试5", "list2测试6")); | ||||
|         List<TestObj1> list3 = new ArrayList<>(); | ||||
|         list3.add(new TestObj1("list3测试1", "list3测试2", "list3测试3")); | ||||
|         List<TestObj1> list4 = new ArrayList<>(); | ||||
|         list4.add(new TestObj1("list4测试1", "list4测试2", "list4测试3")); | ||||
|         list4.add(new TestObj1("list4测试4", "list4测试5", "list4测试6")); | ||||
|         list4.add(new TestObj1("list4测试7", "list4测试8", "list4测试9")); | ||||
|         list4.add(new TestObj1("list4测试10", "list4测试11", "list4测试12")); | ||||
|         Map<String, Object> multiListMap = new HashMap<>(); | ||||
|         multiListMap.put("map", map); | ||||
|         multiListMap.put("data1", list1); | ||||
|         multiListMap.put("data2", list2); | ||||
|         multiListMap.put("data3", list3); | ||||
|         multiListMap.put("data4", list4); | ||||
|         ExcelUtil.exportTemplateMultiList(multiListMap, "多列表.xlsx", "excel/多列表.xlsx", response); | ||||
|     } | ||||
|  | ||||
|     @Data | ||||
|     @AllArgsConstructor | ||||
|     static class TestObj1 { | ||||
|         private String test1; | ||||
|         private String test2; | ||||
|         private String test3; | ||||
|     } | ||||
|  | ||||
|     @Data | ||||
|     @AllArgsConstructor | ||||
|     static class TestObj { | ||||
|         private String name; | ||||
|         private String list1; | ||||
|         private String list2; | ||||
|         private String list3; | ||||
|         private String list4; | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -2,9 +2,6 @@ package com.ruoyi.demo.controller; | ||||
|  | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.utils.MessageUtils; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import io.swagger.annotations.ApiParam; | ||||
| import lombok.Data; | ||||
| import org.hibernate.validator.constraints.Range; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| @@ -22,7 +19,6 @@ import javax.validation.constraints.NotNull; | ||||
|  * @author Lion Li | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "测试国际化控制器", tags = {"测试国际化管理"}) | ||||
| @RestController | ||||
| @RequestMapping("/demo/i18n") | ||||
| public class TestI18nController { | ||||
| @@ -32,10 +28,11 @@ public class TestI18nController { | ||||
|      * code为 messages.properties 中的 key | ||||
|      * <p> | ||||
|      * 测试使用 user.register.success | ||||
|      * | ||||
|      * @param code 国际化code | ||||
|      */ | ||||
|     @ApiOperation("通过code获取国际化内容") | ||||
|     @GetMapping() | ||||
|     public R<Void> get(@ApiParam("国际化code") String code) { | ||||
|     public R<Void> get(String code) { | ||||
|         return R.ok(MessageUtils.message(code)); | ||||
|     } | ||||
|  | ||||
| @@ -45,7 +42,6 @@ public class TestI18nController { | ||||
|      * <p> | ||||
|      * 测试使用 not.null | ||||
|      */ | ||||
|     @ApiOperation("Validator 校验国际化") | ||||
|     @GetMapping("/test1") | ||||
|     public R<Void> test1(@NotBlank(message = "{not.null}") String str) { | ||||
|         return R.ok(str); | ||||
| @@ -57,7 +53,6 @@ public class TestI18nController { | ||||
|      * <p> | ||||
|      * 测试使用 not.null | ||||
|      */ | ||||
|     @ApiOperation("Bean 校验国际化") | ||||
|     @GetMapping("/test2") | ||||
|     public R<TestI18nBo> test2(@Validated TestI18nBo bo) { | ||||
|         return R.ok(bo); | ||||
|   | ||||
| @@ -4,8 +4,6 @@ import com.ruoyi.common.annotation.Sensitive; | ||||
| import com.ruoyi.common.core.controller.BaseController; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.enums.SensitiveStrategy; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import lombok.Data; | ||||
| import org.springframework.web.bind.annotation.GetMapping; | ||||
| import org.springframework.web.bind.annotation.RequestMapping; | ||||
| @@ -21,7 +19,6 @@ import org.springframework.web.bind.annotation.RestController; | ||||
|  * @version 3.6.0 | ||||
|  * @see com.ruoyi.common.core.service.SensitiveService | ||||
|  */ | ||||
| @Api(value = "测试数据脱敏控制器", tags = {"测试数据脱敏管理"}) | ||||
| @RestController | ||||
| @RequestMapping("/demo/sensitive") | ||||
| public class TestSensitiveController extends BaseController { | ||||
| @@ -29,7 +26,6 @@ public class TestSensitiveController extends BaseController { | ||||
|     /** | ||||
|      * 测试数据脱敏 | ||||
|      */ | ||||
|     @ApiOperation("查询测试单表列表") | ||||
|     @GetMapping("/test") | ||||
|     public R<TestSensitive> test() { | ||||
|         TestSensitive testSensitive = new TestSensitive(); | ||||
|   | ||||
| @@ -13,9 +13,6 @@ import com.ruoyi.common.utils.poi.ExcelUtil; | ||||
| import com.ruoyi.demo.domain.bo.TestTreeBo; | ||||
| import com.ruoyi.demo.domain.vo.TestTreeVo; | ||||
| import com.ruoyi.demo.service.ITestTreeService; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import io.swagger.annotations.ApiParam; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import org.springframework.validation.annotation.Validated; | ||||
| import org.springframework.web.bind.annotation.*; | ||||
| @@ -33,7 +30,6 @@ import java.util.List; | ||||
|  * @date 2021-07-26 | ||||
|  */ | ||||
| @Validated | ||||
| @Api(value = "测试树表控制器", tags = {"测试树表管理"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/demo/tree") | ||||
| @@ -44,7 +40,6 @@ public class TestTreeController extends BaseController { | ||||
|     /** | ||||
|      * 查询测试树表列表 | ||||
|      */ | ||||
|     @ApiOperation("查询测试树表列表") | ||||
|     @SaCheckPermission("demo:tree:list") | ||||
|     @GetMapping("/list") | ||||
|     public R<List<TestTreeVo>> list(@Validated(QueryGroup.class) TestTreeBo bo) { | ||||
| @@ -55,7 +50,6 @@ public class TestTreeController extends BaseController { | ||||
|     /** | ||||
|      * 导出测试树表列表 | ||||
|      */ | ||||
|     @ApiOperation("导出测试树表列表") | ||||
|     @SaCheckPermission("demo:tree:export") | ||||
|     @Log(title = "测试树表", businessType = BusinessType.EXPORT) | ||||
|     @GetMapping("/export") | ||||
| @@ -66,12 +60,12 @@ public class TestTreeController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 获取测试树表详细信息 | ||||
|      * | ||||
|      * @param id 测试树ID | ||||
|      */ | ||||
|     @ApiOperation("获取测试树表详细信息") | ||||
|     @SaCheckPermission("demo:tree:query") | ||||
|     @GetMapping("/{id}") | ||||
|     public R<TestTreeVo> getInfo(@ApiParam("测试树ID") | ||||
|                                           @NotNull(message = "主键不能为空") | ||||
|     public R<TestTreeVo> getInfo(@NotNull(message = "主键不能为空") | ||||
|                                  @PathVariable("id") Long id) { | ||||
|         return R.ok(iTestTreeService.queryById(id)); | ||||
|     } | ||||
| @@ -79,7 +73,6 @@ public class TestTreeController extends BaseController { | ||||
|     /** | ||||
|      * 新增测试树表 | ||||
|      */ | ||||
|     @ApiOperation("新增测试树表") | ||||
|     @SaCheckPermission("demo:tree:add") | ||||
|     @Log(title = "测试树表", businessType = BusinessType.INSERT) | ||||
|     @RepeatSubmit | ||||
| @@ -91,7 +84,6 @@ public class TestTreeController extends BaseController { | ||||
|     /** | ||||
|      * 修改测试树表 | ||||
|      */ | ||||
|     @ApiOperation("修改测试树表") | ||||
|     @SaCheckPermission("demo:tree:edit") | ||||
|     @Log(title = "测试树表", businessType = BusinessType.UPDATE) | ||||
|     @RepeatSubmit | ||||
| @@ -102,13 +94,13 @@ public class TestTreeController extends BaseController { | ||||
|  | ||||
|     /** | ||||
|      * 删除测试树表 | ||||
|      * | ||||
|      * @param ids 测试树ID串 | ||||
|      */ | ||||
|     @ApiOperation("删除测试树表") | ||||
|     @SaCheckPermission("demo:tree:remove") | ||||
|     @Log(title = "测试树表", businessType = BusinessType.DELETE) | ||||
|     @DeleteMapping("/{ids}") | ||||
|     public R<Void> remove(@ApiParam("测试树ID串") | ||||
|                                    @NotEmpty(message = "主键不能为空") | ||||
|     public R<Void> remove(@NotEmpty(message = "主键不能为空") | ||||
|                           @PathVariable Long[] ids) { | ||||
|         return toAjax(iTestTreeService.deleteWithValidByIds(Arrays.asList(ids), true) ? 1 : 0); | ||||
|     } | ||||
|   | ||||
| @@ -2,9 +2,6 @@ package com.ruoyi.demo.controller.queue; | ||||
|  | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.utils.redis.QueueUtils; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import io.swagger.annotations.ApiParam; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import lombok.extern.slf4j.Slf4j; | ||||
| import org.springframework.web.bind.annotation.GetMapping; | ||||
| @@ -23,19 +20,22 @@ import org.springframework.web.bind.annotation.RestController; | ||||
|  * @version 3.6.0 | ||||
|  */ | ||||
| @Slf4j | ||||
| @Api(value = "有界队列 演示案例", tags = {"有界队列"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/demo/queue/bounded") | ||||
| public class BoundedQueueController { | ||||
|  | ||||
|  | ||||
|     @ApiOperation("添加队列数据") | ||||
|     /** | ||||
|      * 添加队列数据 | ||||
|      * | ||||
|      * @param queueName 队列名 | ||||
|      * @param capacity  容量 | ||||
|      */ | ||||
|     @GetMapping("/add") | ||||
|     public R<Void> add(@ApiParam("队列名") String queueName, | ||||
|                                 @ApiParam("容量") int capacity) { | ||||
|     public R<Void> add(String queueName, int capacity) { | ||||
|         // 用完了一定要销毁 否则会一直存在 | ||||
|         boolean b = QueueUtils.destroyBoundedQueueObject(queueName); | ||||
|         boolean b = QueueUtils.destroyQueue(queueName); | ||||
|         log.info("通道: {} , 删除: {}", queueName, b); | ||||
|         // 初始化设置一次即可 | ||||
|         if (QueueUtils.trySetBoundedQueueCapacity(queueName, capacity)) { | ||||
| @@ -56,11 +56,15 @@ public class BoundedQueueController { | ||||
|         return R.ok("操作成功"); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("删除队列数据") | ||||
|     /** | ||||
|      * 删除队列数据 | ||||
|      * | ||||
|      * @param queueName 队列名 | ||||
|      */ | ||||
|     @GetMapping("/remove") | ||||
|     public R<Void> remove(@ApiParam("队列名") String queueName) { | ||||
|     public R<Void> remove(String queueName) { | ||||
|         String data = "data-" + 5; | ||||
|         if (QueueUtils.removeBoundedQueueObject(queueName, data)) { | ||||
|         if (QueueUtils.removeQueueObject(queueName, data)) { | ||||
|             log.info("通道: {} , 删除数据: {}", queueName, data); | ||||
|         } else { | ||||
|             return R.fail("操作失败"); | ||||
| @@ -68,12 +72,16 @@ public class BoundedQueueController { | ||||
|         return R.ok("操作成功"); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("获取队列数据") | ||||
|     /** | ||||
|      * 获取队列数据 | ||||
|      * | ||||
|      * @param queueName 队列名 | ||||
|      */ | ||||
|     @GetMapping("/get") | ||||
|     public R<Void> get(@ApiParam("队列名") String queueName) { | ||||
|     public R<Void> get(String queueName) { | ||||
|         String data; | ||||
|         do { | ||||
|             data = QueueUtils.getBoundedQueueObject(queueName); | ||||
|             data = QueueUtils.getQueueObject(queueName); | ||||
|             log.info("通道: {} , 获取数据: {}", queueName, data); | ||||
|         } while (data != null); | ||||
|         return R.ok("操作成功"); | ||||
|   | ||||
| @@ -2,9 +2,6 @@ package com.ruoyi.demo.controller.queue; | ||||
|  | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.utils.redis.QueueUtils; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import io.swagger.annotations.ApiParam; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import lombok.extern.slf4j.Slf4j; | ||||
| import org.springframework.web.bind.annotation.GetMapping; | ||||
| @@ -26,15 +23,18 @@ import java.util.concurrent.TimeUnit; | ||||
|  * @version 3.6.0 | ||||
|  */ | ||||
| @Slf4j | ||||
| @Api(value = "延迟队列 演示案例", tags = {"延迟队列"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/demo/queue/delayed") | ||||
| public class DelayedQueueController { | ||||
|  | ||||
|     @ApiOperation("订阅队列") | ||||
|     /** | ||||
|      * 订阅队列 | ||||
|      * | ||||
|      * @param queueName 队列名 | ||||
|      */ | ||||
|     @GetMapping("/subscribe") | ||||
|     public R<Void> subscribe(@ApiParam("队列名") String queueName) { | ||||
|     public R<Void> subscribe(String queueName) { | ||||
|         log.info("通道: {} 监听中......", queueName); | ||||
|         // 项目初始化设置一次即可 | ||||
|         QueueUtils.subscribeBlockingQueue(queueName, (String orderNum) -> { | ||||
| @@ -44,21 +44,29 @@ public class DelayedQueueController { | ||||
|         return R.ok("操作成功"); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("添加队列数据") | ||||
|     /** | ||||
|      * 添加队列数据 | ||||
|      * | ||||
|      * @param queueName 队列名 | ||||
|      * @param orderNum  订单号 | ||||
|      * @param time      延迟时间(秒) | ||||
|      */ | ||||
|     @GetMapping("/add") | ||||
|     public R<Void> add(@ApiParam("队列名") String queueName, | ||||
|                                 @ApiParam("订单号") String orderNum, | ||||
|                                 @ApiParam("延迟时间(秒)") Long time) { | ||||
|     public R<Void> add(String queueName, String orderNum, Long time) { | ||||
|         QueueUtils.addDelayedQueueObject(queueName, orderNum, time, TimeUnit.SECONDS); | ||||
|         // 观察发送时间 | ||||
|         log.info("通道: {} , 发送数据: {}", queueName, orderNum); | ||||
|         return R.ok("操作成功"); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("删除队列数据") | ||||
|     /** | ||||
|      * 删除队列数据 | ||||
|      * | ||||
|      * @param queueName 队列名 | ||||
|      * @param orderNum  订单号 | ||||
|      */ | ||||
|     @GetMapping("/remove") | ||||
|     public R<Void> remove(@ApiParam("队列名") String queueName, | ||||
|                                    @ApiParam("订单号") String orderNum) { | ||||
|     public R<Void> remove(String queueName, String orderNum) { | ||||
|         if (QueueUtils.removeDelayedQueueObject(queueName, orderNum)) { | ||||
|             log.info("通道: {} , 删除数据: {}", queueName, orderNum); | ||||
|         } else { | ||||
| @@ -67,9 +75,13 @@ public class DelayedQueueController { | ||||
|         return R.ok("操作成功"); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("销毁队列") | ||||
|     /** | ||||
|      * 销毁队列 | ||||
|      * | ||||
|      * @param queueName 队列名 | ||||
|      */ | ||||
|     @GetMapping("/destroy") | ||||
|     public R<Void> destroy(@ApiParam("队列名") String queueName) { | ||||
|     public R<Void> destroy(String queueName) { | ||||
|         // 用完了一定要销毁 否则会一直存在 | ||||
|         QueueUtils.destroyDelayedQueue(queueName); | ||||
|         return R.ok("操作成功"); | ||||
|   | ||||
| @@ -2,6 +2,7 @@ package com.ruoyi.demo.controller.queue; | ||||
|  | ||||
| import lombok.Data; | ||||
| import lombok.NoArgsConstructor; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
|  | ||||
| /** | ||||
|  * 实体类 注意不允许使用内部类 否则会找不到类 | ||||
| @@ -11,7 +12,12 @@ import lombok.NoArgsConstructor; | ||||
|  */ | ||||
| @Data | ||||
| @NoArgsConstructor | ||||
| public class PriorityDemo { | ||||
| public class PriorityDemo implements Comparable<PriorityDemo> { | ||||
|     private String name; | ||||
|     private Integer orderNum; | ||||
|  | ||||
|     @Override | ||||
|     public int compareTo(@NotNull PriorityDemo other) { | ||||
|         return Integer.compare(getOrderNum(), other.getOrderNum()); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,16 +0,0 @@ | ||||
| package com.ruoyi.demo.controller.queue; | ||||
|  | ||||
| import java.util.Comparator; | ||||
|  | ||||
| /** | ||||
|  * 比较器 注意不允许使用 内部类或匿名类或lambda表达式 会找不到类 | ||||
|  * | ||||
|  * @author Lion Li | ||||
|  * @version 3.6.0 | ||||
|  */ | ||||
| public class PriorityDemoComparator implements Comparator<PriorityDemo> { | ||||
|     @Override | ||||
|     public int compare(PriorityDemo pd1, PriorityDemo pd2) { | ||||
|         return Integer.compare(pd1.getOrderNum(), pd2.getOrderNum()); | ||||
|     } | ||||
| } | ||||
| @@ -3,9 +3,6 @@ package com.ruoyi.demo.controller.queue; | ||||
| import cn.hutool.core.util.RandomUtil; | ||||
| import com.ruoyi.common.core.domain.R; | ||||
| import com.ruoyi.common.utils.redis.QueueUtils; | ||||
| import io.swagger.annotations.Api; | ||||
| import io.swagger.annotations.ApiOperation; | ||||
| import io.swagger.annotations.ApiParam; | ||||
| import lombok.RequiredArgsConstructor; | ||||
| import lombok.extern.slf4j.Slf4j; | ||||
| import org.springframework.web.bind.annotation.GetMapping; | ||||
| @@ -24,26 +21,22 @@ import org.springframework.web.bind.annotation.RestController; | ||||
|  * @version 3.6.0 | ||||
|  */ | ||||
| @Slf4j | ||||
| @Api(value = "优先队列 演示案例", tags = {"优先队列"}) | ||||
| @RequiredArgsConstructor | ||||
| @RestController | ||||
| @RequestMapping("/demo/queue/priority") | ||||
| public class PriorityQueueController { | ||||
|  | ||||
|     @ApiOperation("添加队列数据") | ||||
|     /** | ||||
|      * 添加队列数据 | ||||
|      * | ||||
|      * @param queueName 队列名 | ||||
|      */ | ||||
|     @GetMapping("/add") | ||||
|     public R<Void> add(@ApiParam("队列名") String queueName) { | ||||
|     public R<Void> add(String queueName) { | ||||
|         // 用完了一定要销毁 否则会一直存在 | ||||
|         boolean b = QueueUtils.destroyPriorityQueueObject(queueName); | ||||
|         boolean b = QueueUtils.destroyQueue(queueName); | ||||
|         log.info("通道: {} , 删除: {}", queueName, b); | ||||
|         // 初始化设置一次即可 此处注意 不允许用内部类或匿名类 | ||||
|         boolean flag = QueueUtils.trySetPriorityQueueComparator(queueName, new PriorityDemoComparator()); | ||||
|         if (flag) { | ||||
|             log.info("通道: {} , 设置比较器成功", queueName); | ||||
|         } else { | ||||
|             log.info("通道: {} , 设置比较器失败", queueName); | ||||
|             return R.fail("操作失败"); | ||||
|         } | ||||
|  | ||||
|         for (int i = 0; i < 10; i++) { | ||||
|             int randomNum = RandomUtil.randomInt(10); | ||||
|             PriorityDemo data = new PriorityDemo(); | ||||
| @@ -58,15 +51,19 @@ public class PriorityQueueController { | ||||
|         return R.ok("操作成功"); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("删除队列数据") | ||||
|     /** | ||||
|      * 删除队列数据 | ||||
|      * | ||||
|      * @param queueName 队列名 | ||||
|      * @param name      对象名 | ||||
|      * @param orderNum  排序号 | ||||
|      */ | ||||
|     @GetMapping("/remove") | ||||
|     public R<Void> remove(@ApiParam("队列名") String queueName, | ||||
|                                    @ApiParam("对象名") String name, | ||||
|                                    @ApiParam("排序号") Integer orderNum) { | ||||
|     public R<Void> remove(String queueName, String name, Integer orderNum) { | ||||
|         PriorityDemo data = new PriorityDemo(); | ||||
|         data.setName(name); | ||||
|         data.setOrderNum(orderNum); | ||||
|         if (QueueUtils.removePriorityQueueObject(queueName, data)) { | ||||
|         if (QueueUtils.removeQueueObject(queueName, data)) { | ||||
|             log.info("通道: {} , 删除数据: {}", queueName, data); | ||||
|         } else { | ||||
|             return R.fail("操作失败"); | ||||
| @@ -74,12 +71,16 @@ public class PriorityQueueController { | ||||
|         return R.ok("操作成功"); | ||||
|     } | ||||
|  | ||||
|     @ApiOperation("获取队列数据") | ||||
|     /** | ||||
|      * 获取队列数据 | ||||
|      * | ||||
|      * @param queueName 队列名 | ||||
|      */ | ||||
|     @GetMapping("/get") | ||||
|     public R<Void> get(@ApiParam("队列名") String queueName) { | ||||
|     public R<Void> get(String queueName) { | ||||
|         PriorityDemo data; | ||||
|         do { | ||||
|             data = QueueUtils.getPriorityQueueObject(queueName); | ||||
|             data = QueueUtils.getQueueObject(queueName); | ||||
|             log.info("通道: {} , 获取数据: {}", queueName, data); | ||||
|         } while (data != null); | ||||
|         return R.ok("操作成功"); | ||||
|   | ||||
| @@ -3,8 +3,6 @@ package com.ruoyi.demo.domain.bo; | ||||
| import com.ruoyi.common.core.domain.BaseEntity; | ||||
| import com.ruoyi.common.core.validate.AddGroup; | ||||
| import com.ruoyi.common.core.validate.EditGroup; | ||||
| import io.swagger.annotations.ApiModel; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.Data; | ||||
| import lombok.EqualsAndHashCode; | ||||
|  | ||||
| @@ -20,48 +18,41 @@ import javax.validation.constraints.NotNull; | ||||
|  | ||||
| @Data | ||||
| @EqualsAndHashCode(callSuper = true) | ||||
| @ApiModel("测试单表业务对象") | ||||
| public class TestDemoBo extends BaseEntity { | ||||
|  | ||||
|     /** | ||||
|      * 主键 | ||||
|      */ | ||||
|     @ApiModelProperty("主键") | ||||
|     @NotNull(message = "主键不能为空", groups = {EditGroup.class}) | ||||
|     private Long id; | ||||
|  | ||||
|     /** | ||||
|      * 部门id | ||||
|      */ | ||||
|     @ApiModelProperty("部门id") | ||||
|     @NotNull(message = "部门id不能为空", groups = {AddGroup.class, EditGroup.class}) | ||||
|     private Long deptId; | ||||
|  | ||||
|     /** | ||||
|      * 用户id | ||||
|      */ | ||||
|     @ApiModelProperty("用户id") | ||||
|     @NotNull(message = "用户id不能为空", groups = {AddGroup.class, EditGroup.class}) | ||||
|     private Long userId; | ||||
|  | ||||
|     /** | ||||
|      * 排序号 | ||||
|      */ | ||||
|     @ApiModelProperty("排序号") | ||||
|     @NotNull(message = "排序号不能为空", groups = {AddGroup.class, EditGroup.class}) | ||||
|     private Long orderNum; | ||||
|     private Integer orderNum; | ||||
|  | ||||
|     /** | ||||
|      * key键 | ||||
|      */ | ||||
|     @ApiModelProperty("key键") | ||||
|     @NotBlank(message = "key键不能为空", groups = {AddGroup.class, EditGroup.class}) | ||||
|     private String testKey; | ||||
|  | ||||
|     /** | ||||
|      * 值 | ||||
|      */ | ||||
|     @ApiModelProperty("值") | ||||
|     @NotBlank(message = "值不能为空", groups = {AddGroup.class, EditGroup.class}) | ||||
|     private String value; | ||||
|  | ||||
|   | ||||
| @@ -1,8 +1,6 @@ | ||||
| package com.ruoyi.demo.domain.bo; | ||||
|  | ||||
| import com.alibaba.excel.annotation.ExcelProperty; | ||||
| import io.swagger.annotations.ApiModel; | ||||
| import io.swagger.annotations.ApiModelProperty; | ||||
| import lombok.Data; | ||||
|  | ||||
| import javax.validation.constraints.NotBlank; | ||||
| @@ -15,13 +13,11 @@ import javax.validation.constraints.NotNull; | ||||
|  * @date 2021-07-26 | ||||
|  */ | ||||
| @Data | ||||
| @ApiModel("测试单表业务对象") | ||||
| public class TestDemoImportVo { | ||||
|  | ||||
|     /** | ||||
|      * 部门id | ||||
|      */ | ||||
|     @ApiModelProperty("部门id") | ||||
|     @NotNull(message = "部门id不能为空") | ||||
|     @ExcelProperty(value = "部门id") | ||||
|     private Long deptId; | ||||
| @@ -29,7 +25,6 @@ public class TestDemoImportVo { | ||||
|     /** | ||||
|      * 用户id | ||||
|      */ | ||||
|     @ApiModelProperty("用户id") | ||||
|     @NotNull(message = "用户id不能为空") | ||||
|     @ExcelProperty(value = "用户id") | ||||
|     private Long userId; | ||||
| @@ -37,7 +32,6 @@ public class TestDemoImportVo { | ||||
|     /** | ||||
|      * 排序号 | ||||
|      */ | ||||
|     @ApiModelProperty("排序号") | ||||
|     @NotNull(message = "排序号不能为空") | ||||
|     @ExcelProperty(value = "排序号") | ||||
|     private Long orderNum; | ||||
| @@ -45,7 +39,6 @@ public class TestDemoImportVo { | ||||
|     /** | ||||
|      * key键 | ||||
|      */ | ||||
|     @ApiModelProperty("key键") | ||||
|     @NotBlank(message = "key键不能为空") | ||||
|     @ExcelProperty(value = "key键") | ||||
|     private String testKey; | ||||
| @@ -53,7 +46,6 @@ public class TestDemoImportVo { | ||||
|     /** | ||||
|      * 值 | ||||
|      */ | ||||
|     @ApiModelProperty("值") | ||||
|     @NotBlank(message = "值不能为空") | ||||
|     @ExcelProperty(value = "值") | ||||
|     private String value; | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user