mirror of
				https://github.com/dromara/RuoYi-Vue-Plus.git
				synced 2025-10-26 20:03:46 +08:00 
			
		
		
		
	Compare commits
	
		
			735 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 1a70cf658c | ||
|  | 242e26abee | ||
|  | 52d33195ec | ||
|  | d8b486c793 | ||
|  | 00b6ca5dfe | ||
|  | 9ebaf936c1 | ||
|  | dac463efa6 | ||
|  | da1b18792b | ||
|  | 85b4adf400 | ||
|  | 70f39cb1ee | ||
|  | d87dfd6397 | ||
|  | 1496220a74 | ||
|  | 996835d124 | ||
|  | f6f3db1701 | ||
|  | 6ee3085260 | ||
|  | a6f6562693 | ||
|  | d8585d0ee7 | ||
|  | 4761237849 | ||
|  | f1a09711c1 | ||
|  | 346a3cf7f1 | ||
|  | c42229bc54 | ||
|  | 317ab351ce | ||
|  | e1579622de | ||
|  | f2ff839106 | ||
|  | c494ab9963 | ||
|  | 15f7d4dc47 | ||
|  | dd9ef8431b | ||
|  | e729a924d4 | ||
|  | aeacfea536 | ||
|  | 43b31edb75 | ||
|  | 8448bcdc6d | ||
|  | e1efdb8602 | ||
|  | 409849db1a | ||
|  | 9d6b0b35a3 | ||
|  | 7f00861e68 | ||
|  | d2b1177580 | ||
|  | ef91551449 | ||
|  | 4063b0c2dd | ||
|  | 5117fe2958 | ||
|  | 9ed60f22de | ||
|  | 8498ead56c | ||
|  | 53ae4ea6bc | ||
|  | 4c8137daf2 | ||
|  | 6e017c35f7 | ||
|  | 06cda3fb18 | ||
|  | b38151a0bb | ||
|  | 2790b5ddc4 | ||
|  | f70951d374 | ||
|  | 9ebe1ae918 | ||
|  | 06e6f7af0c | ||
|  | d6894c89ed | ||
|  | ebf780617a | ||
|  | e130475259 | ||
|  | 115af9c402 | ||
|  | 8b914c1091 | ||
|  | 9c368e8e68 | ||
|  | 700db1fc15 | ||
|  | 35f9c0fb72 | ||
|  | 3091241279 | ||
|  | 56050aa650 | ||
|  | b5cbb58c2d | ||
|  | adaa93acfb | ||
|  | 5d043078b2 | ||
|  | fe8db721d7 | ||
|  | 4d02466fed | ||
|  | ca69c00cf4 | ||
|  | 6733e48ea1 | ||
|  | dca5d69dd2 | ||
|  | 842af57d74 | ||
|  | 8dc92383d9 | ||
|  | 1ae35032a5 | ||
|  | 223ae7ae72 | ||
|  | 87a212f093 | ||
|  | 5952b1c90e | ||
|  | 83ba9be342 | ||
|  | a1431c5cb1 | ||
|  | e41a58b10d | ||
|  | da94e89825 | ||
|  | 0381df6b17 | ||
|  | 31acd6b301 | ||
|  | 3c5dfdcf57 | ||
|  | e0b65b857d | ||
|  | 83395d4e2f | ||
|  | a6bed384d7 | ||
|  | afac00de33 | ||
|  | ee80e97578 | ||
|  | 35fac6cc0c | ||
|  | e5515751fd | ||
|  | c9c1e5ca7f | ||
|  | b47e67ff30 | ||
|  | f5de8b9ddf | ||
|  | a63abbf268 | ||
|  | 1f42bd3d22 | ||
|  | 93c4cc2cd1 | ||
|  | ab4b21ff58 | ||
|  | 83a09c475d | ||
|  | 9d99afd159 | ||
|  | 1858ac6b25 | ||
|  | 78290128e7 | ||
|  | 68193fd37f | ||
|  | cd9c3c3f4f | ||
|  | 8bd023b49f | ||
|  | 57daef9ced | ||
|  | fb134da091 | ||
|  | c1a3eaaffb | ||
|  | b6dbe7b5a5 | ||
|  | 8138063bfe | ||
|  | 4c8dc500c7 | ||
|  | 85247991b6 | ||
|  | a127eaa0e9 | ||
|  | ebb9f15a75 | ||
|  | cd137bd9fc | ||
|  | 07bde5f88a | ||
|  | 6fffa02acf | ||
|  | 5830c3c96d | ||
|  | 9f2dc5c233 | ||
|  | ec076c1e0d | ||
|  | 6e8ef308ed | ||
|  | dc48f9858b | ||
|  | a6b2ac5dcd | ||
|  | c0685b7f7f | ||
|  | fb2d616c57 | ||
|  | 6ff6853082 | ||
|  | f8014ae969 | ||
|  | 88ed6e25ca | ||
|  | 1de2b7a57e | ||
|  | 95f7ba0080 | ||
|  | bb73b31e6b | ||
|  | 9ecc4475dd | ||
|  | f01aa37394 | ||
|  | 0208fa14af | ||
|  | efa2f23510 | ||
|  | 408155ff5c | ||
|  | 785c7495c1 | ||
|  | 4517dea98d | ||
|  | 0358dc233a | ||
|  | e21396870f | ||
|  | b67f6a0fec | ||
|  | 42d8104505 | ||
|  | f40a0eab23 | ||
|  | b6153d1aef | ||
|  | a2c585daa4 | ||
|  | 5a60bf0b0a | ||
|  | d2e58ba46b | ||
|  | 9dcddc7876 | ||
|  | b970185536 | ||
|  | a64a029323 | ||
|  | 98bb1f00ee | ||
|  | 78f4d1c85b | ||
|  | bf83fe568b | ||
|  | 02f4fdb2e5 | ||
|  | cb61a84acb | ||
|  | dc56e211e6 | ||
|  | e4d33b7d13 | ||
|  | f7de561ac5 | ||
|  | fde624998f | ||
|  | 6505432bf4 | ||
|  | 7d953c1a8b | ||
|  | 95010d4a4c | ||
|  | a32a931d24 | ||
|  | 4d72fb4289 | ||
|  | 186c04d90a | ||
|  | 84de5a5d42 | ||
|  | 6fe80cff6b | ||
|  | c542b7ac32 | ||
|  | 3607e008a3 | ||
|  | 5b8edbf381 | ||
|  | 03f7bc0f3f | ||
|  | 0c54416040 | ||
|  | 0d188d6a39 | ||
|  | e95a1245c5 | ||
|  | a3b761f4e0 | ||
|  | 6e0c5817d5 | ||
|  | e410fc8c41 | ||
|  | 9e7141f97c | ||
|  | d7441edae7 | ||
|  | 5053361a7f | ||
|  | 91c99d305c | ||
|  | 2c39f26de3 | ||
|  | a680a84d91 | ||
|  | 390edc5e3e | ||
|  | 663024ac7f | ||
|  | 0e75aa6250 | ||
|  | 693a33f660 | ||
|  | 8af85e8066 | ||
|  | 513bda53cd | ||
|  | cf7df6a14c | ||
|  | acbdd6365a | ||
|  | e781f88eca | ||
|  | 86d16d070d | ||
|  | 3c4c91dc7d | ||
|  | 574837a92a | ||
|  | 854b405d89 | ||
|  | 7d66a5bd84 | ||
|  | 2c7663d971 | ||
|  | 6281ec0456 | ||
|  | c3faa1a925 | ||
|  | 111db70908 | ||
|  | d6400e382a | ||
|  | 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 | ||
|  | 0700c60636 | ||
|  | aac6e0521a | ||
|  | 7e25a80c5b | ||
|  | 328d4b916d | ||
|  | 596ff28207 | ||
|  | 70c1a37bbd | ||
|  | 4902276921 | ||
|  | 83cdeacaa8 | ||
|  | e6e2f48f7e | ||
|  | e316f613d0 | ||
|  | 9a8c87a43e | ||
|  | 67140f996b | ||
|  | 2f5d9ced6d | ||
|  | 3dfb116fdd | ||
|  | eb1047a149 | ||
|  | b42a8176ca | ||
|  | 2a9beec31d | ||
|  | 910fe6261b | ||
|  | 0c820b96d8 | ||
|  | 9689efedc4 | ||
|  | 97561f428e | ||
|  | df0497fb58 | ||
|  | 30f0bdc754 | ||
|  | c3c49a05c2 | ||
|  | e72b878380 | ||
|  | d51d8ad877 | ||
|  | 68a616d7c7 | ||
|  | a0447de061 | ||
|  | 46db06a22b | ||
|  | 7984fdfd20 | ||
|  | d637f03067 | ||
|  | 4c57c2a044 | ||
|  | 94b09c3fc0 | ||
|  | de5ae4a05e | ||
|  | 517c5705b6 | ||
|  | 5d3e977761 | ||
|  | c0d430f4e5 | ||
|  | d910888181 | ||
|  | db2dfee6ae | ||
|  | d311d56bbf | ||
|  | a33898c2a0 | ||
|  | e6940d48e3 | ||
|  | 3bc5ef38d3 | ||
|  | bac3b3a0dc | ||
|  | 07fded4da9 | ||
|  | 4f0a584efc | ||
|  | 1139fb39c5 | ||
|  | 4cd7c07344 | ||
|  | a854e0ca8e | ||
|  | de1766abde | ||
|  | ebd005e1b1 | ||
|  | 9ba61037a4 | ||
|  | c74b878372 | ||
|  | 01a566c794 | ||
|  | 5362a633e6 | ||
|  | b2c3f45141 | ||
|  | d5f9b5b74a | ||
|  | 95a2c74462 | ||
|  | d217ef37bf | ||
|  | 95bea7f64a | ||
|  | 58b63a24c2 | ||
|  | 53c90cfc99 | ||
|  | 2043d1f439 | ||
|  | 61034d4dde | ||
|  | 6605bf35a8 | ||
|  | b2b93e5060 | ||
|  | 098286fcaf | ||
|  | a6f30412ad | ||
|  | b624648115 | ||
|  | 9d1d74c022 | ||
|  | 0c79d1a619 | ||
|  | 1ea8eebfec | ||
|  | 1dcc03aae0 | ||
|  | 3be9535225 | ||
|  | 8cad83a9dc | ||
|  | 0488495a34 | ||
|  | a5eb8ce21e | ||
|  | 1dbcd4ee6e | ||
|  | 7e1f024d28 | ||
|  | 4a353896e3 | ||
|  | eaa12de740 | ||
|  | 0e4372b4a3 | ||
|  | b0c31ae486 | ||
|  | a43e4b8aa7 | ||
|  | ce3f80144a | ||
|  | cd33cd66ae | ||
|  | f2a2c2d407 | ||
|  | 5f0481c067 | ||
|  | f30f3ac37b | ||
|  | d31ed1f2e1 | ||
|  | cf67d606ae | ||
|  | 76b859f6ba | ||
|  | a326e301f0 | ||
|  | a16ae7dadc | ||
|  | ac47375aba | ||
|  | 857054179c | ||
|  | fd5a87eedf | ||
|  | 4770b0d3e3 | ||
|  | 181ec4633b | ||
|  | dc5e5c6c08 | ||
|  | dcdf423d66 | ||
|  | 1a19c3b3a5 | ||
|  | 0a3fcfd9f4 | ||
|  | 49ac180329 | ||
|  | 0271aa5414 | ||
|  | 880b3290e8 | ||
|  | a0364f0758 | ||
|  | ed693e89c9 | ||
|  | 526957e0ac | ||
|  | 3341eb5f2c | ||
|  | 51fba42b83 | ||
|  | 12f5327e0e | ||
|  | 40f9fac1a5 | ||
|  | 0b5e924a21 | ||
|  | cd312f1348 | ||
|  | e480f616e0 | ||
|  | 2285a94958 | ||
|  | 7f1356a447 | ||
|  | 420a43cdd5 | ||
|  | 1aab88859d | ||
|  | 3210fce113 | ||
|  | 0a1b026d92 | ||
|  | 29e1bf3f2e | ||
|  | 333a38978e | ||
|  | 91c7b46d52 | ||
|  | 870d51bf10 | ||
|  | 7c0f41c785 | ||
|  | 647d3391bb | ||
|  | 0d24f34d83 | ||
|  | a68a32d9b6 | ||
|  | 8a89054c2f | ||
|  | cd46d08afd | ||
|  | 5e8ccda522 | ||
|  | 78d0e67987 | ||
|  | 62fc38078c | ||
|  | bc4f844cd8 | ||
|  | d73cd5ed27 | ||
|  | 2bf3d3a9c5 | ||
|  | e3b64387a5 | ||
|  | bd0454309f | ||
|  | a72fa8038f | ||
|  | f15f8e3295 | ||
|  | 604c671d41 | ||
|  | 6130bebbb3 | ||
|  | 9f944c043f | ||
|  | 092b3214c5 | ||
|  | 07dfef48be | ||
|  | e9571fc2e2 | ||
|  | db3a83d163 | ||
|  | 22f7ccc11d | ||
|  | ea53b3faba | ||
|  | 645ccb2f87 | ||
|  | b40cc02e2a | ||
|  | 98418272ab | ||
|  | a32cc879cc | ||
|  | adf329cf1a | ||
|  | 4bd5009ec1 | ||
|  | 74a9b4c058 | ||
|  | 8ee749f137 | ||
|  | 46fa404d83 | ||
|  | cb8f5de5af | ||
|  | 3c7c5380f4 | ||
|  | 04d611e1c2 | ||
|  | 2d813b2191 | ||
|  | f1bf549e0f | ||
|  | d3a3ad5cbd | ||
|  | 8091a46005 | ||
|  | 18cffc4678 | ||
|  | c1d01fb1e1 | ||
|  | 4d0c84bcf5 | ||
|  | 8db5c27d81 | ||
|  | 2ff36d81a1 | ||
|  | 56feff0455 | ||
|  | cf17bcb15c | ||
|  | b2d3abd9d6 | ||
|  | a54cb7d3e4 | ||
|  | 00549c267f | ||
|  | f41bfdb8eb | 
| @@ -10,7 +10,7 @@ end_of_line = lf | |||||||
| trim_trailing_whitespace = true | trim_trailing_whitespace = true | ||||||
| insert_final_newline = true | insert_final_newline = true | ||||||
|  |  | ||||||
| [*.{json,yml}] | [*.{json,yml,yaml}] | ||||||
| indent_size = 2 | indent_size = 2 | ||||||
|  |  | ||||||
| [*.md] | [*.md] | ||||||
|   | |||||||
| @@ -1,13 +1,49 @@ | |||||||
| ### 使用版本 | ### 使用版本(未按照模板填写直接删除) | ||||||
|  |  | ||||||
|  | - jdk版本(带上尾号): 例如 1.8.0_202 | ||||||
|  | - 框架版本(项目启动时输出的版本号): 例如 4.4.0 | ||||||
|  | - 其他依赖版本(你觉得有必要的): | ||||||
|  |  | ||||||
|  | ### 问题前提 | ||||||
|  |  | ||||||
|  | > 功能不好用 不会用 是否已经看过项目文档 | ||||||
|  | > 项目运行报错 是否已经拿着报错信息去百度 常见报错百度百度足以 | ||||||
|  | > 是否搜索过其他issue 一些已经解决的问题 会在issue内留下解决方法 | ||||||
|  | > 无法线上解决或者与框架无关的问题的欢迎加VIP群跟作者一对一谈 | ||||||
|  |  | ||||||
|  | ### 异常模块 | ||||||
|  |  | ||||||
|  | > 此报错都涉及到那些系统模块 | ||||||
|  |  | ||||||
|  | 例如 ruoyi-system ruoyi-auth 等等 | ||||||
|  |  | ||||||
| ### 问题描述 | ### 问题描述 | ||||||
|  |  | ||||||
|  | > 越详细越容易直击问题所在 | ||||||
|  |  | ||||||
|  | 已知: XXX功能不好用 或 XXX数据不正常 等等 | ||||||
|  |  | ||||||
| ### 希望结果 | ### 希望结果 | ||||||
|  |  | ||||||
|  | > 想知道你觉得怎么样是正常或者合理的 | ||||||
|  |  | ||||||
|  | 希望功能可以有XXX结果 或者 XXX现象 | ||||||
|  |  | ||||||
| ### 重现步骤 | ### 重现步骤 | ||||||
|  |  | ||||||
|  | > 作者并不知道这个问题是如何出现的 | ||||||
|  |  | ||||||
| ### 报错信息(截图为主 请勿发混乱格式) | - 1 | ||||||
|  | - 2 | ||||||
|  | - 3 | ||||||
|  |  | ||||||
|  | ### 相关代码与报错信息(请勿发混乱格式) | ||||||
|  |  | ||||||
|  | > 代码可按照如下形式提供或者截图均可 越详细越好 | ||||||
|  | > 大多数问题都是 代码编写错误问题 逻辑问题 或者用法错误等问题 | ||||||
|  |  | ||||||
|  | ```java | ||||||
|  | public class XXX { | ||||||
|  |  | ||||||
|  | } | ||||||
|  | ``` | ||||||
| @@ -1,7 +1,7 @@ | |||||||
| ### 更改目的 解决了什么问题 | ### 更改目的 解决了什么问题(请提交到dev分支) | ||||||
|  |  | ||||||
|  |  | ||||||
| ### 描述 做了哪些改动 | ### 改动逻辑 这么写的思路(让作者更好的理解你的意图) | ||||||
|  |  | ||||||
|  |  | ||||||
| ### 测试 都做了哪些测试(未经过测试不采纳) | ### 测试 都做了哪些测试(未经过测试不采纳) | ||||||
							
								
								
									
										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.5.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-server.run.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								.run/ruoyi-server.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.5.0" /> | ||||||
|  |         <option name="buildOnly" value="true" /> | ||||||
|  |         <option name="sourceFilePath" value="ruoyi-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.5.0" /> | ||||||
|  |         <option name="buildOnly" value="true" /> | ||||||
|  |         <option name="sourceFilePath" value="ruoyi-extend/ruoyi-xxl-job-admin/Dockerfile" /> | ||||||
|  |       </settings> | ||||||
|  |     </deployment> | ||||||
|  |     <method v="2" /> | ||||||
|  |   </configuration> | ||||||
|  | </component> | ||||||
							
								
								
									
										32
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										32
									
								
								README.md
									
									
									
									
									
								
							| @@ -4,14 +4,17 @@ | |||||||
| [](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/blob/master/LICENSE) | [](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/blob/master/LICENSE) | ||||||
| [](https://www.jetbrains.com/?from=RuoYi-Vue-Plus) | [](https://www.jetbrains.com/?from=RuoYi-Vue-Plus) | ||||||
| <br> | <br> | ||||||
| [](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus) | [](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus) | ||||||
| []() | []() | ||||||
| []() | []() | ||||||
| []() | []() | ||||||
|  |  | ||||||
| > RuoYi-Vue-Plus 是重写 RuoYi-Vue 针对 `分布式集群` 场景全方位升级(不兼容原框架) | > RuoYi-Vue-Plus 是重写 RuoYi-Vue 针对 `分布式集群` 场景全方位升级(不兼容原框架) | ||||||
|  |  | ||||||
| > 系统演示: [传送门](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/系统演示?sort_id=4836388) | > 项目代码、文档 均开源免费可商用 遵循开源协议在项目中保留开源协议文件即可<br> | ||||||
|  | 活到老写到老 为兴趣而开源 为学习而开源 为让大家真正可以学到技术而开源 | ||||||
|  |  | ||||||
|  | > 系统演示: [传送门](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=4836388&doc_id=1469725) | ||||||
|  |  | ||||||
| | 功能介绍     | 使用技术                | 文档地址                                                                                              | 特性注意事项                     | | | 功能介绍     | 使用技术                | 文档地址                                                                                              | 特性注意事项                     | | ||||||
| |----------|---------------------|---------------------------------------------------------------------------------------------------|----------------------------| | |----------|---------------------|---------------------------------------------------------------------------------------------------|----------------------------| | ||||||
| @@ -25,6 +28,9 @@ | |||||||
| | 容器框架     | Undertow            | [Undertow官网](https://undertow.io/)                                                                | 基于 XNIO 的高性能容器             | | | 容器框架     | Undertow            | [Undertow官网](https://undertow.io/)                                                                | 基于 XNIO 的高性能容器             | | ||||||
| | 权限认证框架   | Sa-Token、Jwt        | [Sa-Token官网](https://sa-token.dev33.cn/)                                                          | 强解耦、强扩展                    | | | 权限认证框架   | Sa-Token、Jwt        | [Sa-Token官网](https://sa-token.dev33.cn/)                                                          | 强解耦、强扩展                    | | ||||||
| | 关系数据库    | MySQL               | [MySQL官网](https://dev.mysql.com/)                                                                 | 适配 8.X 最低 5.7              | | | 关系数据库    | MySQL               | [MySQL官网](https://dev.mysql.com/)                                                                 | 适配 8.X 最低 5.7              | | ||||||
|  | | 关系数据库    | Oracle              | [Oracle官网](https://www.oracle.com/cn/database/)                                                   | 适配 11g 12c                 | | ||||||
|  | | 关系数据库    | PostgreSQL          | [PostgreSQL官网](https://www.postgresql.org/)                                                       | 适配 13 14                   | | ||||||
|  | | 关系数据库    | SQLServer           | [SQLServer官网](https://docs.microsoft.com/zh-cn/sql/sql-server)                                    | 适配 2017 2019               | | ||||||
| | 缓存数据库    | Redis               | [Redis官网](https://redis.io/)                                                                      | 适配 6.X 最低 4.X              | | | 缓存数据库    | Redis               | [Redis官网](https://redis.io/)                                                                      | 适配 6.X 最低 4.X              | | ||||||
| | 数据库框架    | Mybatis-Plus        | [Mybatis-Plus文档](https://baomidou.com/guide/)                                                     | 快速 CRUD 增加开发效率             | | | 数据库框架    | Mybatis-Plus        | [Mybatis-Plus文档](https://baomidou.com/guide/)                                                     | 快速 CRUD 增加开发效率             | | ||||||
| | 数据库框架    | p6spy               | [p6spy官网](https://p6spy.readthedocs.io/)                                                          | 更强劲的 SQL 分析                | | | 数据库框架    | p6spy               | [p6spy官网](https://p6spy.readthedocs.io/)                                                          | 更强劲的 SQL 分析                | | ||||||
| @@ -35,16 +41,17 @@ | |||||||
| | 分布式队列    | Redisson            | [Redisson文档](https://github.com/redisson/redisson/wiki/%E7%9B%AE%E5%BD%95)                        | 普通队列、延迟队列、优先队列 等           | | | 分布式队列    | Redisson            | [Redisson文档](https://github.com/redisson/redisson/wiki/%E7%9B%AE%E5%BD%95)                        | 普通队列、延迟队列、优先队列 等           | | ||||||
| | 分布式锁     | Lock4j              | [Lock4j官网](https://gitee.com/baomidou/lock4j)                                                     | 注解锁、工具锁 多种多样               | | | 分布式锁     | Lock4j              | [Lock4j官网](https://gitee.com/baomidou/lock4j)                                                     | 注解锁、工具锁 多种多样               | | ||||||
| | 分布式幂等    | Redisson            | [Lock4j文档](https://gitee.com/baomidou/lock4j)                                                     | 拦截重复提交                     | | | 分布式幂等    | Redisson            | [Lock4j文档](https://gitee.com/baomidou/lock4j)                                                     | 拦截重复提交                     | | ||||||
| | 分布式日志    | TLog                | [TLog文档](https://yomahub.com/tlog/docs)                                                           | 支持跟踪链路日志记录、性能分析、链路排查       | | | 分布式链路追踪  | Apache SkyWalking   | [Apache SkyWalking文档](https://skywalking.apache.org/docs/)                                        | 链路追踪、网格分析、度量聚合、可视化         | | ||||||
| | 分布式任务调度  | Xxl-Job             | [Xxl-Job官网](https://www.xuxueli.com/xxl-job/)                                                     | 高性能 高可靠 易扩展                | | | 分布式任务调度  | Xxl-Job             | [Xxl-Job官网](https://www.xuxueli.com/xxl-job/)                                                     | 高性能 高可靠 易扩展                | | ||||||
| | 文件存储     | Minio               | [Minio文档](https://docs.min.io/)                                                                   | 本地存储                       | | | 文件存储     | Minio               | [Minio文档](https://docs.min.io/)                                                                   | 本地存储                       | | ||||||
| | 文件存储     | 七牛、阿里、腾讯            | [OSS使用文档](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=4359146&doc_id=1469725) | 云存储                        | | | 文件存储     | 七牛、阿里、腾讯            | [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/)                    | 全方位服务监控                    | | | 监控框架     | 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/)    | 增强接口安全性、严谨性 支持国际化          | | | 校验框架     | 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)                                      | 性能优异 扩展性强                  | | | 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/)                                                           | 减少代码冗余 增加安全性               | | | 工具类框架    | Hutool、Lombok       | [Hutool文档](https://www.hutool.cn/docs/)                                                           | 减少代码冗余 增加安全性               | | ||||||
| | 代码生成器    | 适配MP、Knife4j规范化代码   | [Hutool文档](https://www.hutool.cn/docs/)                                                           | 一键生成前后端代码                  | | | 代码生成器    | 适配MP、SpringDoc规范化代码 | [代码生成文档](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=5522329&doc_id=1469725)  | 一键生成前后端代码                  | | ||||||
| | 部署方式     | Docker              | [Docker文档](https://docs.docker.com/)                                                              | 容器编排 一键部署业务集群              | | | 部署方式     | Docker              | [Docker文档](https://docs.docker.com/)                                                              | 容器编排 一键部署业务集群              | | ||||||
| | 国际化      | SpringMessage       | [SpringMVC文档](https://docs.spring.io/spring-framework/docs/current/reference/html/web.html#mvc)   | Spring标准国际化方案              | | | 国际化      | SpringMessage       | [SpringMVC文档](https://docs.spring.io/spring-framework/docs/current/reference/html/web.html#mvc)   | Spring标准国际化方案              | | ||||||
|  |  | ||||||
| @@ -52,11 +59,14 @@ | |||||||
|  |  | ||||||
| 使用框架前请仔细阅读文档重点注意事项 | 使用框架前请仔细阅读文档重点注意事项 | ||||||
| <br> | <br> | ||||||
| >[初始化项目 必看](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/关于初始化项目?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/关于应用部署?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/关于应用部署?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=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) | >[参考文档 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) | >>[https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages) | ||||||
|   | |||||||
							
								
								
									
										254
									
								
								pom.xml
									
									
									
									
									
								
							
							
						
						
									
										254
									
								
								pom.xml
									
									
									
									
									
								
							| @@ -6,55 +6,77 @@ | |||||||
|  |  | ||||||
|     <groupId>com.ruoyi</groupId> |     <groupId>com.ruoyi</groupId> | ||||||
|     <artifactId>ruoyi-vue-plus</artifactId> |     <artifactId>ruoyi-vue-plus</artifactId> | ||||||
|     <version>4.0.1</version> |     <version>4.5.0</version> | ||||||
|  |  | ||||||
|     <name>RuoYi-Vue-Plus</name> |     <name>RuoYi-Vue-Plus</name> | ||||||
|     <url>https://gitee.com/JavaLionLi/RuoYi-Vue-Plus</url> |     <url>https://gitee.com/JavaLionLi/RuoYi-Vue-Plus</url> | ||||||
|     <description>RuoYi-Vue-Plus后台管理系统</description> |     <description>RuoYi-Vue-Plus后台管理系统</description> | ||||||
|  |  | ||||||
|     <properties> |     <properties> | ||||||
|         <ruoyi-vue-plus.version>4.0.1</ruoyi-vue-plus.version> |         <ruoyi-vue-plus.version>4.5.0</ruoyi-vue-plus.version> | ||||||
|         <spring-boot.version>2.6.4</spring-boot.version> |         <spring-boot.version>2.7.7</spring-boot.version> | ||||||
|         <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> |         <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> | ||||||
|         <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding> |         <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding> | ||||||
|         <java.version>1.8</java.version> |         <java.version>1.8</java.version> | ||||||
|         <maven-jar-plugin.version>3.2.2</maven-jar-plugin.version> |         <maven-jar-plugin.version>3.2.2</maven-jar-plugin.version> | ||||||
|         <druid.version>1.2.8</druid.version> |         <spring-boot.mybatis>2.2.2</spring-boot.mybatis> | ||||||
|         <knife4j.version>3.0.3</knife4j.version> |         <springdoc.version>1.6.14</springdoc.version> | ||||||
|         <swagger-annotations.version>1.5.22</swagger-annotations.version> |         <poi.version>5.2.3</poi.version> | ||||||
|         <poi.version>4.1.2</poi.version> |         <easyexcel.version>3.1.5</easyexcel.version> | ||||||
|         <easyexcel.version>3.0.5</easyexcel.version> |  | ||||||
|         <cglib.version>3.3.0</cglib.version> |  | ||||||
|         <velocity.version>2.3</velocity.version> |         <velocity.version>2.3</velocity.version> | ||||||
|         <satoken.version>1.29.0</satoken.version> |         <satoken.version>1.34.0</satoken.version> | ||||||
|         <mybatis-plus.version>3.5.1</mybatis-plus.version> |         <mybatis-plus.version>3.5.3.1</mybatis-plus.version> | ||||||
|         <p6spy.version>3.9.1</p6spy.version> |         <p6spy.version>3.9.1</p6spy.version> | ||||||
|         <hutool.version>5.7.21</hutool.version> |         <hutool.version>5.8.11</hutool.version> | ||||||
|         <okhttp.version>4.9.2</okhttp.version> |         <okhttp.version>4.10.0</okhttp.version> | ||||||
|         <spring-boot-admin.version>2.6.2</spring-boot-admin.version> |         <spring-boot-admin.version>2.7.10</spring-boot-admin.version> | ||||||
|         <redisson.version>3.16.8</redisson.version> |         <redisson.version>3.19.1</redisson.version> | ||||||
|         <lock4j.version>2.2.1</lock4j.version> |         <lock4j.version>2.2.3</lock4j.version> | ||||||
|         <dynamic-ds.version>3.5.0</dynamic-ds.version> |         <dynamic-ds.version>3.5.2</dynamic-ds.version> | ||||||
|         <tlog.version>1.3.6</tlog.version> |         <alibaba-ttl.version>2.14.2</alibaba-ttl.version> | ||||||
|         <xxl-job.version>2.3.0</xxl-job.version> |         <xxl-job.version>2.3.1</xxl-job.version> | ||||||
|  |         <lombok.version>1.18.24</lombok.version> | ||||||
|  |  | ||||||
|         <!-- jdk11 缺失依赖 jaxb--> |         <!-- 临时修复 snakeyaml 漏洞 --> | ||||||
|         <jaxb.version>3.0.1</jaxb.version> |         <snakeyaml.version>1.33</snakeyaml.version> | ||||||
|  |  | ||||||
|         <!-- OSS 配置 --> |         <!-- OSS 配置 --> | ||||||
|         <qiniu.version>7.9.3</qiniu.version> |         <aws-java-sdk-s3.version>1.12.373</aws-java-sdk-s3.version> | ||||||
|         <aliyun.oss.version>3.14.0</aliyun.oss.version> |         <!-- SMS 配置 --> | ||||||
|         <qcloud.cos.version>5.6.68</qcloud.cos.version> |         <aliyun.sms.version>2.0.23</aliyun.sms.version> | ||||||
|         <minio.version>8.3.7</minio.version> |         <tencent.sms.version>3.1.660</tencent.sms.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> |  | ||||||
|  |  | ||||||
|     </properties> |     </properties> | ||||||
|  |  | ||||||
|  |     <profiles> | ||||||
|  |         <profile> | ||||||
|  |             <id>local</id> | ||||||
|  |             <properties> | ||||||
|  |                 <!-- 环境标识,需要与配置文件的名称相对应 --> | ||||||
|  |                 <profiles.active>local</profiles.active> | ||||||
|  |                 <logging.level>debug</logging.level> | ||||||
|  |             </properties> | ||||||
|  |         </profile> | ||||||
|  |         <profile> | ||||||
|  |             <id>dev</id> | ||||||
|  |             <properties> | ||||||
|  |                 <!-- 环境标识,需要与配置文件的名称相对应 --> | ||||||
|  |                 <profiles.active>dev</profiles.active> | ||||||
|  |                 <logging.level>debug</logging.level> | ||||||
|  |             </properties> | ||||||
|  |             <activation> | ||||||
|  |                 <!-- 默认环境 --> | ||||||
|  |                 <activeByDefault>true</activeByDefault> | ||||||
|  |             </activation> | ||||||
|  |         </profile> | ||||||
|  |         <profile> | ||||||
|  |             <id>prod</id> | ||||||
|  |             <properties> | ||||||
|  |                 <profiles.active>prod</profiles.active> | ||||||
|  |                 <logging.level>warn</logging.level> | ||||||
|  |             </properties> | ||||||
|  |         </profile> | ||||||
|  |     </profiles> | ||||||
|  |  | ||||||
|     <!-- 依赖声明 --> |     <!-- 依赖声明 --> | ||||||
|     <dependencyManagement> |     <dependencyManagement> | ||||||
|         <dependencies> |         <dependencies> | ||||||
| @@ -77,47 +99,39 @@ | |||||||
|                 <scope>import</scope> |                 <scope>import</scope> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <!-- 阿里数据库连接池 --> |  | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.alibaba</groupId> |                 <groupId>org.springdoc</groupId> | ||||||
|                 <artifactId>druid-spring-boot-starter</artifactId> |                 <artifactId>springdoc-openapi-webmvc-core</artifactId> | ||||||
|                 <version>${druid.version}</version> |                 <version>${springdoc.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.github.xiaoymin</groupId> |                 <groupId>org.springdoc</groupId> | ||||||
|                 <artifactId>knife4j-spring-boot-starter</artifactId> |                 <artifactId>springdoc-openapi-javadoc</artifactId> | ||||||
|                 <version>${knife4j.version}</version> |                 <version>${springdoc.version}</version> | ||||||
|                 <exclusions> |  | ||||||
|                     <exclusion> |  | ||||||
|                         <artifactId>swagger-annotations</artifactId> |  | ||||||
|                         <groupId>io.swagger</groupId> |  | ||||||
|                     </exclusion> |  | ||||||
|                 </exclusions> |  | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>io.swagger</groupId> |                 <groupId>org.projectlombok</groupId> | ||||||
|                 <artifactId>swagger-annotations</artifactId> |                 <artifactId>lombok</artifactId> | ||||||
|                 <version>${swagger-annotations.version}</version> |                 <version>${lombok.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <!-- excel工具 --> |             <dependency> | ||||||
|  |                 <groupId>org.apache.poi</groupId> | ||||||
|  |                 <artifactId>poi</artifactId> | ||||||
|  |                 <version>${poi.version}</version> | ||||||
|  |             </dependency> | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>org.apache.poi</groupId> |                 <groupId>org.apache.poi</groupId> | ||||||
|                 <artifactId>poi-ooxml</artifactId> |                 <artifactId>poi-ooxml</artifactId> | ||||||
|                 <version>${poi.version}</version> |                 <version>${poi.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.alibaba</groupId> |                 <groupId>com.alibaba</groupId> | ||||||
|                 <artifactId>easyexcel</artifactId> |                 <artifactId>easyexcel</artifactId> | ||||||
|                 <version>${easyexcel.version}</version> |                 <version>${easyexcel.version}</version> | ||||||
|                 <exclusions> |                 <exclusions> | ||||||
|                     <exclusion> |  | ||||||
|                         <groupId>org.apache.poi</groupId> |  | ||||||
|                         <artifactId>poi</artifactId> |  | ||||||
|                     </exclusion> |  | ||||||
|                     <exclusion> |                     <exclusion> | ||||||
|                         <groupId>org.apache.poi</groupId> |                         <groupId>org.apache.poi</groupId> | ||||||
|                         <artifactId>poi-ooxml-schemas</artifactId> |                         <artifactId>poi-ooxml-schemas</artifactId> | ||||||
| @@ -125,12 +139,6 @@ | |||||||
|                 </exclusions> |                 </exclusions> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <dependency> |  | ||||||
|                 <groupId>cglib</groupId> |  | ||||||
|                 <artifactId>cglib</artifactId> |  | ||||||
|                 <version>${cglib.version}</version> |  | ||||||
|             </dependency> |  | ||||||
|  |  | ||||||
|             <!-- velocity代码生成使用模板 --> |             <!-- velocity代码生成使用模板 --> | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>org.apache.velocity</groupId> |                 <groupId>org.apache.velocity</groupId> | ||||||
| @@ -149,13 +157,12 @@ | |||||||
|                 <groupId>cn.dev33</groupId> |                 <groupId>cn.dev33</groupId> | ||||||
|                 <artifactId>sa-token-jwt</artifactId> |                 <artifactId>sa-token-jwt</artifactId> | ||||||
|                 <version>${satoken.version}</version> |                 <version>${satoken.version}</version> | ||||||
|             </dependency> |                 <exclusions> | ||||||
|  |                     <exclusion> | ||||||
|             <!-- jdk11 缺失依赖 jaxb--> |                         <groupId>cn.hutool</groupId> | ||||||
|             <dependency> |                         <artifactId>hutool-all</artifactId> | ||||||
|                 <groupId>com.sun.xml.bind</groupId> |                     </exclusion> | ||||||
|                 <artifactId>jaxb-impl</artifactId> |                 </exclusions> | ||||||
|                 <version>${jaxb.version}</version> |  | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <!-- dynamic-datasource 多数据源--> |             <!-- dynamic-datasource 多数据源--> | ||||||
| @@ -170,11 +177,7 @@ | |||||||
|                 <artifactId>mybatis-plus-boot-starter</artifactId> |                 <artifactId>mybatis-plus-boot-starter</artifactId> | ||||||
|                 <version>${mybatis-plus.version}</version> |                 <version>${mybatis-plus.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|             <dependency> |  | ||||||
|                 <groupId>com.baomidou</groupId> |  | ||||||
|                 <artifactId>mybatis-plus-extension</artifactId> |  | ||||||
|                 <version>${mybatis-plus.version}</version> |  | ||||||
|             </dependency> |  | ||||||
|             <!-- sql性能分析插件 --> |             <!-- sql性能分析插件 --> | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>p6spy</groupId> |                 <groupId>p6spy</groupId> | ||||||
| @@ -188,6 +191,24 @@ | |||||||
|                 <version>${okhttp.version}</version> |                 <version>${okhttp.version}</version> | ||||||
|             </dependency> |             </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-sms</artifactId> | ||||||
|  |                 <version>${tencent.sms.version}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>de.codecentric</groupId> |                 <groupId>de.codecentric</groupId> | ||||||
|                 <artifactId>spring-boot-admin-starter-server</artifactId> |                 <artifactId>spring-boot-admin-starter-server</artifactId> | ||||||
| @@ -219,15 +240,16 @@ | |||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.yomahub</groupId> |                 <groupId>com.alibaba</groupId> | ||||||
|                 <artifactId>tlog-web-spring-boot-starter</artifactId> |                 <artifactId>transmittable-thread-local</artifactId> | ||||||
|                 <version>${tlog.version}</version> |                 <version>${alibaba-ttl.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- 临时修复 snakeyaml 漏洞 --> | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.yomahub</groupId> |                 <groupId>org.yaml</groupId> | ||||||
|                 <artifactId>tlog-xxljob-spring-boot-starter</artifactId> |                 <artifactId>snakeyaml</artifactId> | ||||||
|                 <version>${tlog.version}</version> |                 <version>${snakeyaml.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <!-- 定时任务 --> |             <!-- 定时任务 --> | ||||||
| @@ -272,6 +294,13 @@ | |||||||
|                 <version>${ruoyi-vue-plus.version}</version> |                 <version>${ruoyi-vue-plus.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- SMS短信模块 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>com.ruoyi</groupId> | ||||||
|  |                 <artifactId>ruoyi-sms</artifactId> | ||||||
|  |                 <version>${ruoyi-vue-plus.version}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|             <!-- demo模块 --> |             <!-- demo模块 --> | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.ruoyi</groupId> |                 <groupId>com.ruoyi</groupId> | ||||||
| @@ -292,6 +321,7 @@ | |||||||
|         <module>ruoyi-demo</module> |         <module>ruoyi-demo</module> | ||||||
|         <module>ruoyi-extend</module> |         <module>ruoyi-extend</module> | ||||||
|         <module>ruoyi-oss</module> |         <module>ruoyi-oss</module> | ||||||
|  |         <module>ruoyi-sms</module> | ||||||
|     </modules> |     </modules> | ||||||
|     <packaging>pom</packaging> |     <packaging>pom</packaging> | ||||||
|  |  | ||||||
| @@ -305,6 +335,36 @@ | |||||||
|                     <source>${java.version}</source> |                     <source>${java.version}</source> | ||||||
|                     <target>${java.version}</target> |                     <target>${java.version}</target> | ||||||
|                     <encoding>${project.build.sourceEncoding}</encoding> |                     <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> | ||||||
|  |             <!-- 单元测试使用 --> | ||||||
|  |             <plugin> | ||||||
|  |                 <groupId>org.apache.maven.plugins</groupId> | ||||||
|  |                 <artifactId>maven-surefire-plugin</artifactId> | ||||||
|  |                 <version>2.22.2</version> | ||||||
|  |                 <configuration> | ||||||
|  |                     <argLine>-Dfile.encoding=UTF-8</argLine> | ||||||
|  |                     <!-- 根据打包环境执行对应的@Tag测试方法 --> | ||||||
|  |                     <groups>${profiles.active}</groups> | ||||||
|  |                     <!-- 排除标签 --> | ||||||
|  |                     <excludedGroups>exclude</excludedGroups> | ||||||
|                 </configuration> |                 </configuration> | ||||||
|             </plugin> |             </plugin> | ||||||
|         </plugins> |         </plugins> | ||||||
| @@ -353,40 +413,6 @@ | |||||||
|         </pluginRepository> |         </pluginRepository> | ||||||
|     </pluginRepositories> |     </pluginRepositories> | ||||||
|  |  | ||||||
|     <profiles> |  | ||||||
|         <profile> |  | ||||||
|             <id>local</id> |  | ||||||
|             <properties> |  | ||||||
|                 <!-- 环境标识,需要与配置文件的名称相对应 --> |  | ||||||
|                 <profiles.active>local</profiles.active> |  | ||||||
|                 <logging.level>debug</logging.level> |  | ||||||
|                 <knife4j.production>false</knife4j.production> |  | ||||||
|                 <endpoints.include>'*'</endpoints.include> |  | ||||||
|             </properties> |  | ||||||
|         </profile> |  | ||||||
|         <profile> |  | ||||||
|             <id>dev</id> |  | ||||||
|             <properties> |  | ||||||
|                 <!-- 环境标识,需要与配置文件的名称相对应 --> |  | ||||||
|                 <profiles.active>dev</profiles.active> |  | ||||||
|                 <logging.level>debug</logging.level> |  | ||||||
|                 <knife4j.production>false</knife4j.production> |  | ||||||
|                 <endpoints.include>'*'</endpoints.include> |  | ||||||
|             </properties> |  | ||||||
|             <activation> |  | ||||||
|                 <!-- 默认环境 --> |  | ||||||
|                 <activeByDefault>true</activeByDefault> |  | ||||||
|             </activation> |  | ||||||
|         </profile> |  | ||||||
|         <profile> |  | ||||||
|             <id>prod</id> |  | ||||||
|             <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> | </project> | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,14 +2,22 @@ FROM anapsix/alpine-java:8_server-jre_unlimited | |||||||
|  |  | ||||||
| MAINTAINER Lion Li | MAINTAINER Lion Li | ||||||
|  |  | ||||||
| RUN mkdir -p /ruoyi/server | RUN mkdir -p /ruoyi/server/logs \ | ||||||
| RUN mkdir -p /ruoyi/server/logs |     /ruoyi/server/temp \ | ||||||
| RUN mkdir -p /ruoyi/server/temp |     /ruoyi/skywalking/agent | ||||||
|  |  | ||||||
| WORKDIR /ruoyi/server | WORKDIR /ruoyi/server | ||||||
|  |  | ||||||
| EXPOSE 8080 | ENV SERVER_PORT=8080 | ||||||
|  |  | ||||||
|  | EXPOSE ${SERVER_PORT} | ||||||
|  |  | ||||||
| ADD ./target/ruoyi-admin.jar ./app.jar | 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}", \ | ||||||
|  |             # 应用名称 如果想区分集群节点监控 改成不同的名称即可 | ||||||
|  | #            "-Dskywalking.agent.service_name=ruoyi-server", \ | ||||||
|  | #            "-javaagent:/ruoyi/skywalking/agent/skywalking-agent.jar", \ | ||||||
|  |             "-jar", "app.jar"] | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ | |||||||
|     <parent> |     <parent> | ||||||
|         <artifactId>ruoyi-vue-plus</artifactId> |         <artifactId>ruoyi-vue-plus</artifactId> | ||||||
|         <groupId>com.ruoyi</groupId> |         <groupId>com.ruoyi</groupId> | ||||||
|         <version>4.0.1</version> |         <version>4.5.0</version> | ||||||
|     </parent> |     </parent> | ||||||
|     <modelVersion>4.0.0</modelVersion> |     <modelVersion>4.0.0</modelVersion> | ||||||
|     <packaging>jar</packaging> |     <packaging>jar</packaging> | ||||||
| @@ -26,8 +26,23 @@ | |||||||
|  |  | ||||||
|         <!-- Mysql驱动包 --> |         <!-- Mysql驱动包 --> | ||||||
|         <dependency> |         <dependency> | ||||||
|             <groupId>mysql</groupId> |             <groupId>com.mysql</groupId> | ||||||
|             <artifactId>mysql-connector-java</artifactId> |             <artifactId>mysql-connector-j</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |         <!-- Oracle --> | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>com.oracle.database.jdbc</groupId> | ||||||
|  |             <artifactId>ojdbc8</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |         <!-- PostgreSql --> | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>org.postgresql</groupId> | ||||||
|  |             <artifactId>postgresql</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |         <!-- SqlServer --> | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>com.microsoft.sqlserver</groupId> | ||||||
|  |             <artifactId>mssql-jdbc</artifactId> | ||||||
|         </dependency> |         </dependency> | ||||||
|  |  | ||||||
|         <!-- 核心模块--> |         <!-- 核心模块--> | ||||||
| @@ -63,6 +78,24 @@ | |||||||
|             <artifactId>ruoyi-demo</artifactId> |             <artifactId>ruoyi-demo</artifactId> | ||||||
|         </dependency> |         </dependency> | ||||||
|  |  | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>org.springframework.boot</groupId> | ||||||
|  |             <artifactId>spring-boot-starter-test</artifactId> | ||||||
|  |             <scope>test</scope> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |         <!-- skywalking 整合 logback --> | ||||||
|  | <!--        <dependency>--> | ||||||
|  | <!--            <groupId>org.apache.skywalking</groupId>--> | ||||||
|  | <!--            <artifactId>apm-toolkit-logback-1.x</artifactId>--> | ||||||
|  | <!--            <version>${与你的agent探针版本保持一致}</version>--> | ||||||
|  | <!--        </dependency>--> | ||||||
|  | <!--        <dependency>--> | ||||||
|  | <!--            <groupId>org.apache.skywalking</groupId>--> | ||||||
|  | <!--            <artifactId>apm-toolkit-trace</artifactId>--> | ||||||
|  | <!--            <version>${与你的agent探针版本保持一致}</version>--> | ||||||
|  | <!--        </dependency>--> | ||||||
|  |  | ||||||
|     </dependencies> |     </dependencies> | ||||||
|  |  | ||||||
|     <build> |     <build> | ||||||
| @@ -92,25 +125,6 @@ | |||||||
|                     <warName>${project.artifactId}</warName> |                     <warName>${project.artifactId}</warName> | ||||||
|                 </configuration> |                 </configuration> | ||||||
|             </plugin> |             </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> |         </plugins> | ||||||
|     </build> |     </build> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,9 +1,11 @@ | |||||||
| package com.ruoyi.web.controller.common; | package com.ruoyi.web.controller.common; | ||||||
|  |  | ||||||
|  | import cn.dev33.satoken.annotation.SaIgnore; | ||||||
| import cn.hutool.captcha.AbstractCaptcha; | import cn.hutool.captcha.AbstractCaptcha; | ||||||
| import cn.hutool.captcha.generator.CodeGenerator; | import cn.hutool.captcha.generator.CodeGenerator; | ||||||
| import cn.hutool.core.convert.Convert; |  | ||||||
| import cn.hutool.core.util.IdUtil; | import cn.hutool.core.util.IdUtil; | ||||||
|  | import cn.hutool.core.util.RandomUtil; | ||||||
|  | import com.ruoyi.common.constant.CacheConstants; | ||||||
| import com.ruoyi.common.constant.Constants; | import com.ruoyi.common.constant.Constants; | ||||||
| import com.ruoyi.common.core.domain.R; | import com.ruoyi.common.core.domain.R; | ||||||
| import com.ruoyi.common.enums.CaptchaType; | import com.ruoyi.common.enums.CaptchaType; | ||||||
| @@ -12,45 +14,81 @@ import com.ruoyi.common.utils.redis.RedisUtils; | |||||||
| import com.ruoyi.common.utils.reflect.ReflectUtils; | import com.ruoyi.common.utils.reflect.ReflectUtils; | ||||||
| import com.ruoyi.common.utils.spring.SpringUtils; | import com.ruoyi.common.utils.spring.SpringUtils; | ||||||
| import com.ruoyi.framework.config.properties.CaptchaProperties; | 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 com.ruoyi.system.service.ISysConfigService; | ||||||
| import io.swagger.annotations.Api; |  | ||||||
| import io.swagger.annotations.ApiOperation; |  | ||||||
| import lombok.RequiredArgsConstructor; | import lombok.RequiredArgsConstructor; | ||||||
|  | import lombok.extern.slf4j.Slf4j; | ||||||
|  | import org.springframework.expression.Expression; | ||||||
|  | import org.springframework.expression.ExpressionParser; | ||||||
|  | import org.springframework.expression.spel.standard.SpelExpressionParser; | ||||||
|  | import org.springframework.validation.annotation.Validated; | ||||||
| import org.springframework.web.bind.annotation.GetMapping; | import org.springframework.web.bind.annotation.GetMapping; | ||||||
| import org.springframework.web.bind.annotation.RestController; | import org.springframework.web.bind.annotation.RestController; | ||||||
|  |  | ||||||
|  | import javax.validation.constraints.NotBlank; | ||||||
|  | import java.time.Duration; | ||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
| import java.util.concurrent.TimeUnit; |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 验证码操作处理 |  * 验证码操作处理 | ||||||
|  * |  * | ||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Api(value = "验证码操作处理", tags = {"验证码管理"}) | @SaIgnore | ||||||
|  | @Slf4j | ||||||
|  | @Validated | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| public class CaptchaController { | public class CaptchaController { | ||||||
|  |  | ||||||
|     private final CaptchaProperties captchaProperties; |     private final CaptchaProperties captchaProperties; | ||||||
|  |     private final SmsProperties smsProperties; | ||||||
|     private final ISysConfigService configService; |     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") |     @GetMapping("/captchaImage") | ||||||
|     public R<Map<String, Object>> getCode() { |     public R<Map<String, Object>> getCode() { | ||||||
|         Map<String, Object> ajax = new HashMap<>(); |         Map<String, Object> ajax = new HashMap<>(); | ||||||
|         boolean captchaOnOff = configService.selectCaptchaOnOff(); |         boolean captchaEnabled = configService.selectCaptchaEnabled(); | ||||||
|         ajax.put("captchaOnOff", captchaOnOff); |         ajax.put("captchaEnabled", captchaEnabled); | ||||||
|         if (!captchaOnOff) { |         if (!captchaEnabled) { | ||||||
|             return R.ok(ajax); |             return R.ok(ajax); | ||||||
|         } |         } | ||||||
|         // 保存验证码信息 |         // 保存验证码信息 | ||||||
|         String uuid = IdUtil.simpleUUID(); |         String uuid = IdUtil.simpleUUID(); | ||||||
|         String verifyKey = Constants.CAPTCHA_CODE_KEY + uuid; |         String verifyKey = CacheConstants.CAPTCHA_CODE_KEY + uuid; | ||||||
|         // 生成验证码 |         // 生成验证码 | ||||||
|         CaptchaType captchaType = captchaProperties.getType(); |         CaptchaType captchaType = captchaProperties.getType(); | ||||||
|         boolean isMath = CaptchaType.MATH == captchaType; |         boolean isMath = CaptchaType.MATH == captchaType; | ||||||
| @@ -59,28 +97,16 @@ public class CaptchaController { | |||||||
|         AbstractCaptcha captcha = SpringUtils.getBean(captchaProperties.getCategory().getClazz()); |         AbstractCaptcha captcha = SpringUtils.getBean(captchaProperties.getCategory().getClazz()); | ||||||
|         captcha.setGenerator(codeGenerator); |         captcha.setGenerator(codeGenerator); | ||||||
|         captcha.createCode(); |         captcha.createCode(); | ||||||
|         String code = isMath ? getCodeResult(captcha.getCode()) : captcha.getCode(); |         String code = captcha.getCode(); | ||||||
|         RedisUtils.setCacheObject(verifyKey, code, Constants.CAPTCHA_EXPIRATION, TimeUnit.MINUTES); |         if (isMath) { | ||||||
|  |             ExpressionParser parser = new SpelExpressionParser(); | ||||||
|  |             Expression exp = parser.parseExpression(StringUtils.remove(code, "=")); | ||||||
|  |             code = exp.getValue(String.class); | ||||||
|  |         } | ||||||
|  |         RedisUtils.setCacheObject(verifyKey, code, Duration.ofMinutes(Constants.CAPTCHA_EXPIRATION)); | ||||||
|         ajax.put("uuid", uuid); |         ajax.put("uuid", uuid); | ||||||
|         ajax.put("img", captcha.getImageBase64()); |         ajax.put("img", captcha.getImageBase64()); | ||||||
|         return R.ok(ajax); |         return R.ok(ajax); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private String getCodeResult(String capStr) { |  | ||||||
|         int numberLength = captchaProperties.getNumberLength(); |  | ||||||
|         int a = Convert.toInt(StringUtils.substring(capStr, 0, numberLength).trim()); |  | ||||||
|         char operator = capStr.charAt(numberLength); |  | ||||||
|         int b = Convert.toInt(StringUtils.substring(capStr, numberLength + 1, numberLength + 1 + numberLength).trim()); |  | ||||||
|         switch (operator) { |  | ||||||
|             case '*': |  | ||||||
|                 return Convert.toStr(a * b); |  | ||||||
|             case '+': |  | ||||||
|                 return Convert.toStr(a + b); |  | ||||||
|             case '-': |  | ||||||
|                 return Convert.toStr(a - b); |  | ||||||
|             default: |  | ||||||
|                 return StringUtils.EMPTY; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,40 +1,59 @@ | |||||||
| package com.ruoyi.web.controller.monitor; | package com.ruoyi.web.controller.monitor; | ||||||
|  |  | ||||||
| import cn.dev33.satoken.annotation.SaCheckPermission; | 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.core.domain.R; | ||||||
|  | import com.ruoyi.common.utils.JsonUtils; | ||||||
| import com.ruoyi.common.utils.StringUtils; | import com.ruoyi.common.utils.StringUtils; | ||||||
| import io.swagger.annotations.Api; | import com.ruoyi.common.utils.redis.CacheUtils; | ||||||
| import io.swagger.annotations.ApiOperation; | import com.ruoyi.common.utils.redis.RedisUtils; | ||||||
|  | import com.ruoyi.system.domain.SysCache; | ||||||
| import lombok.RequiredArgsConstructor; | import lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.data.redis.connection.RedisServerCommands; | import org.redisson.spring.data.connection.RedissonConnectionFactory; | ||||||
| import org.springframework.data.redis.core.RedisCallback; | import org.springframework.data.redis.connection.RedisConnection; | ||||||
| import org.springframework.data.redis.core.RedisTemplate; | import org.springframework.web.bind.annotation.*; | ||||||
| import org.springframework.web.bind.annotation.GetMapping; |  | ||||||
| import org.springframework.web.bind.annotation.RequestMapping; |  | ||||||
| import org.springframework.web.bind.annotation.RestController; |  | ||||||
|  |  | ||||||
| import java.util.*; | import java.util.*; | ||||||
|  | import java.util.stream.Collectors; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 缓存监控 |  * 缓存监控 | ||||||
|  * |  * | ||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Api(value = "缓存监控", tags = {"缓存监控管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| @RequestMapping("/monitor/cache") | @RequestMapping("/monitor/cache") | ||||||
| public class CacheController { | 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") |     @SaCheckPermission("monitor:cache:list") | ||||||
|     @GetMapping() |     @GetMapping() | ||||||
|     public R<Map<String, Object>> getInfo() throws Exception { |     public R<Map<String, Object>> getInfo() throws Exception { | ||||||
|         Properties info = (Properties) redisTemplate.execute((RedisCallback<Object>) RedisServerCommands::info); |         RedisConnection connection = connectionFactory.getConnection(); | ||||||
|         Properties commandStats = (Properties) redisTemplate.execute((RedisCallback<Object>) connection -> connection.info("commandstats")); |         Properties info = connection.info(); | ||||||
|         Object dbSize = redisTemplate.execute((RedisCallback<Object>) RedisServerCommands::dbSize); |         Properties commandStats = connection.info("commandstats"); | ||||||
|  |         Long dbSize = connection.dbSize(); | ||||||
|  |  | ||||||
|         Map<String, Object> result = new HashMap<>(3); |         Map<String, Object> result = new HashMap<>(3); | ||||||
|         result.put("info", info); |         result.put("info", info); | ||||||
| @@ -53,4 +72,98 @@ public class CacheController { | |||||||
|         result.put("commandStats", pieList); |         result.put("commandStats", pieList); | ||||||
|         return R.ok(result); |         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 cn.dev33.satoken.annotation.SaCheckPermission; | ||||||
| import com.ruoyi.common.annotation.Log; | import com.ruoyi.common.annotation.Log; | ||||||
|  | import com.ruoyi.common.constant.CacheConstants; | ||||||
| import com.ruoyi.common.core.controller.BaseController; | 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.PageQuery; | ||||||
|  | import com.ruoyi.common.core.domain.R; | ||||||
| import com.ruoyi.common.core.page.TableDataInfo; | import com.ruoyi.common.core.page.TableDataInfo; | ||||||
| import com.ruoyi.common.enums.BusinessType; | import com.ruoyi.common.enums.BusinessType; | ||||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | import com.ruoyi.common.utils.poi.ExcelUtil; | ||||||
|  | import com.ruoyi.common.utils.redis.RedisUtils; | ||||||
| import com.ruoyi.system.domain.SysLogininfor; | import com.ruoyi.system.domain.SysLogininfor; | ||||||
| import com.ruoyi.system.service.ISysLogininforService; | import com.ruoyi.system.service.ISysLogininforService; | ||||||
| import io.swagger.annotations.Api; |  | ||||||
| import io.swagger.annotations.ApiOperation; |  | ||||||
| import lombok.RequiredArgsConstructor; | import lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.validation.annotation.Validated; | import org.springframework.validation.annotation.Validated; | ||||||
| import org.springframework.web.bind.annotation.*; | import org.springframework.web.bind.annotation.*; | ||||||
| @@ -25,7 +25,6 @@ import java.util.List; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Validated | @Validated | ||||||
| @Api(value = "系统访问记录", tags = {"系统访问记录管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| @RequestMapping("/monitor/logininfor") | @RequestMapping("/monitor/logininfor") | ||||||
| @@ -33,14 +32,18 @@ public class SysLogininforController extends BaseController { | |||||||
|  |  | ||||||
|     private final ISysLogininforService logininforService; |     private final ISysLogininforService logininforService; | ||||||
|  |  | ||||||
|     @ApiOperation("查询系统访问记录列表") |     /** | ||||||
|  |      * 获取系统访问记录列表 | ||||||
|  |      */ | ||||||
|     @SaCheckPermission("monitor:logininfor:list") |     @SaCheckPermission("monitor:logininfor:list") | ||||||
|     @GetMapping("/list") |     @GetMapping("/list") | ||||||
|     public TableDataInfo<SysLogininfor> list(SysLogininfor logininfor, PageQuery pageQuery) { |     public TableDataInfo<SysLogininfor> list(SysLogininfor logininfor, PageQuery pageQuery) { | ||||||
|         return logininforService.selectPageLogininforList(logininfor, pageQuery); |         return logininforService.selectPageLogininforList(logininfor, pageQuery); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiOperation("导出系统访问记录列表") |     /** | ||||||
|  |      * 导出系统访问记录列表 | ||||||
|  |      */ | ||||||
|     @Log(title = "登录日志", businessType = BusinessType.EXPORT) |     @Log(title = "登录日志", businessType = BusinessType.EXPORT) | ||||||
|     @SaCheckPermission("monitor:logininfor:export") |     @SaCheckPermission("monitor:logininfor:export") | ||||||
|     @PostMapping("/export") |     @PostMapping("/export") | ||||||
| @@ -49,7 +52,10 @@ public class SysLogininforController extends BaseController { | |||||||
|         ExcelUtil.exportExcel(list, "登录日志", SysLogininfor.class, response); |         ExcelUtil.exportExcel(list, "登录日志", SysLogininfor.class, response); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiOperation("删除系统访问记录") |     /** | ||||||
|  |      * 批量删除登录日志 | ||||||
|  |      * @param infoIds 日志ids | ||||||
|  |      */ | ||||||
|     @SaCheckPermission("monitor:logininfor:remove") |     @SaCheckPermission("monitor:logininfor:remove") | ||||||
|     @Log(title = "登录日志", businessType = BusinessType.DELETE) |     @Log(title = "登录日志", businessType = BusinessType.DELETE) | ||||||
|     @DeleteMapping("/{infoIds}") |     @DeleteMapping("/{infoIds}") | ||||||
| @@ -57,7 +63,9 @@ public class SysLogininforController extends BaseController { | |||||||
|         return toAjax(logininforService.deleteLogininforByIds(infoIds)); |         return toAjax(logininforService.deleteLogininforByIds(infoIds)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiOperation("清空系统访问记录") |     /** | ||||||
|  |      * 清理系统访问记录 | ||||||
|  |      */ | ||||||
|     @SaCheckPermission("monitor:logininfor:remove") |     @SaCheckPermission("monitor:logininfor:remove") | ||||||
|     @Log(title = "登录日志", businessType = BusinessType.CLEAN) |     @Log(title = "登录日志", businessType = BusinessType.CLEAN) | ||||||
|     @DeleteMapping("/clean") |     @DeleteMapping("/clean") | ||||||
| @@ -65,4 +73,16 @@ public class SysLogininforController extends BaseController { | |||||||
|         logininforService.cleanLogininfor(); |         logininforService.cleanLogininfor(); | ||||||
|         return R.ok(); |         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 cn.dev33.satoken.annotation.SaCheckPermission; | ||||||
| import com.ruoyi.common.annotation.Log; | import com.ruoyi.common.annotation.Log; | ||||||
| import com.ruoyi.common.core.controller.BaseController; | 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.PageQuery; | ||||||
|  | import com.ruoyi.common.core.domain.R; | ||||||
| import com.ruoyi.common.core.page.TableDataInfo; | import com.ruoyi.common.core.page.TableDataInfo; | ||||||
| import com.ruoyi.common.enums.BusinessType; | import com.ruoyi.common.enums.BusinessType; | ||||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | import com.ruoyi.common.utils.poi.ExcelUtil; | ||||||
| import com.ruoyi.system.domain.SysOperLog; | import com.ruoyi.system.domain.SysOperLog; | ||||||
| import com.ruoyi.system.service.ISysOperLogService; | import com.ruoyi.system.service.ISysOperLogService; | ||||||
| import io.swagger.annotations.Api; |  | ||||||
| import io.swagger.annotations.ApiOperation; |  | ||||||
| import lombok.RequiredArgsConstructor; | import lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.validation.annotation.Validated; | import org.springframework.validation.annotation.Validated; | ||||||
| import org.springframework.web.bind.annotation.*; | import org.springframework.web.bind.annotation.*; | ||||||
| @@ -25,7 +23,6 @@ import java.util.List; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Validated | @Validated | ||||||
| @Api(value = "操作日志记录", tags = {"操作日志记录管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| @RequestMapping("/monitor/operlog") | @RequestMapping("/monitor/operlog") | ||||||
| @@ -33,14 +30,18 @@ public class SysOperlogController extends BaseController { | |||||||
|  |  | ||||||
|     private final ISysOperLogService operLogService; |     private final ISysOperLogService operLogService; | ||||||
|  |  | ||||||
|     @ApiOperation("查询操作日志记录列表") |     /** | ||||||
|  |      * 获取操作日志记录列表 | ||||||
|  |      */ | ||||||
|     @SaCheckPermission("monitor:operlog:list") |     @SaCheckPermission("monitor:operlog:list") | ||||||
|     @GetMapping("/list") |     @GetMapping("/list") | ||||||
|     public TableDataInfo<SysOperLog> list(SysOperLog operLog, PageQuery pageQuery) { |     public TableDataInfo<SysOperLog> list(SysOperLog operLog, PageQuery pageQuery) { | ||||||
|         return operLogService.selectPageOperLogList(operLog, pageQuery); |         return operLogService.selectPageOperLogList(operLog, pageQuery); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiOperation("导出操作日志记录列表") |     /** | ||||||
|  |      * 导出操作日志记录列表 | ||||||
|  |      */ | ||||||
|     @Log(title = "操作日志", businessType = BusinessType.EXPORT) |     @Log(title = "操作日志", businessType = BusinessType.EXPORT) | ||||||
|     @SaCheckPermission("monitor:operlog:export") |     @SaCheckPermission("monitor:operlog:export") | ||||||
|     @PostMapping("/export") |     @PostMapping("/export") | ||||||
| @@ -49,7 +50,10 @@ public class SysOperlogController extends BaseController { | |||||||
|         ExcelUtil.exportExcel(list, "操作日志", SysOperLog.class, response); |         ExcelUtil.exportExcel(list, "操作日志", SysOperLog.class, response); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiOperation("删除操作日志记录") |     /** | ||||||
|  |      * 批量删除操作日志记录 | ||||||
|  |      * @param operIds 日志ids | ||||||
|  |      */ | ||||||
|     @Log(title = "操作日志", businessType = BusinessType.DELETE) |     @Log(title = "操作日志", businessType = BusinessType.DELETE) | ||||||
|     @SaCheckPermission("monitor:operlog:remove") |     @SaCheckPermission("monitor:operlog:remove") | ||||||
|     @DeleteMapping("/{operIds}") |     @DeleteMapping("/{operIds}") | ||||||
| @@ -57,7 +61,9 @@ public class SysOperlogController extends BaseController { | |||||||
|         return toAjax(operLogService.deleteOperLogByIds(operIds)); |         return toAjax(operLogService.deleteOperLogByIds(operIds)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiOperation("清空操作日志记录") |     /** | ||||||
|  |      * 清理操作日志记录 | ||||||
|  |      */ | ||||||
|     @Log(title = "操作日志", businessType = BusinessType.CLEAN) |     @Log(title = "操作日志", businessType = BusinessType.CLEAN) | ||||||
|     @SaCheckPermission("monitor:operlog:remove") |     @SaCheckPermission("monitor:operlog:remove") | ||||||
|     @DeleteMapping("/clean") |     @DeleteMapping("/clean") | ||||||
|   | |||||||
| @@ -5,64 +5,66 @@ import cn.dev33.satoken.exception.NotLoginException; | |||||||
| import cn.dev33.satoken.stp.StpUtil; | import cn.dev33.satoken.stp.StpUtil; | ||||||
| import cn.hutool.core.bean.BeanUtil; | import cn.hutool.core.bean.BeanUtil; | ||||||
| import com.ruoyi.common.annotation.Log; | 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.controller.BaseController; | ||||||
| import com.ruoyi.common.core.domain.R; | import com.ruoyi.common.core.domain.R; | ||||||
| import com.ruoyi.common.core.domain.dto.UserOnlineDTO; | import com.ruoyi.common.core.domain.dto.UserOnlineDTO; | ||||||
| import com.ruoyi.common.core.page.TableDataInfo; | import com.ruoyi.common.core.page.TableDataInfo; | ||||||
| import com.ruoyi.common.enums.BusinessType; | import com.ruoyi.common.enums.BusinessType; | ||||||
|  | import com.ruoyi.common.utils.StreamUtils; | ||||||
| import com.ruoyi.common.utils.StringUtils; | import com.ruoyi.common.utils.StringUtils; | ||||||
| import com.ruoyi.common.utils.redis.RedisUtils; | import com.ruoyi.common.utils.redis.RedisUtils; | ||||||
| import com.ruoyi.system.domain.SysUserOnline; | import com.ruoyi.system.domain.SysUserOnline; | ||||||
| import io.swagger.annotations.Api; |  | ||||||
| import io.swagger.annotations.ApiOperation; |  | ||||||
| import lombok.RequiredArgsConstructor; | import lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.web.bind.annotation.*; | import org.springframework.web.bind.annotation.*; | ||||||
|  |  | ||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
| import java.util.Collections; | import java.util.Collections; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.stream.Collectors; |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 在线用户监控 |  * 在线用户监控 | ||||||
|  * |  * | ||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Api(value = "在线用户监控", tags = {"在线用户监控管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| @RequestMapping("/monitor/online") | @RequestMapping("/monitor/online") | ||||||
| public class SysUserOnlineController extends BaseController { | public class SysUserOnlineController extends BaseController { | ||||||
|  |  | ||||||
|     @ApiOperation("在线用户列表") |     /** | ||||||
|  |      * 获取在线用户监控列表 | ||||||
|  |      * | ||||||
|  |      * @param ipaddr   IP地址 | ||||||
|  |      * @param userName 用户名 | ||||||
|  |      */ | ||||||
|     @SaCheckPermission("monitor:online:list") |     @SaCheckPermission("monitor:online:list") | ||||||
|     @GetMapping("/list") |     @GetMapping("/list") | ||||||
|     public TableDataInfo<SysUserOnline> list(String ipaddr, String userName) { |     public TableDataInfo<SysUserOnline> list(String ipaddr, String userName) { | ||||||
|         // 获取所有未过期的 token |         // 获取所有未过期的 token | ||||||
|         List<String> keys = StpUtil.searchTokenValue("", -1, 0); |         List<String> keys = StpUtil.searchTokenValue("", 0, -1, false); | ||||||
|         List<UserOnlineDTO> userOnlineDTOList = new ArrayList<>(); |         List<UserOnlineDTO> userOnlineDTOList = new ArrayList<>(); | ||||||
|         for (String key : keys) { |         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) { |             if (StpUtil.stpLogic.getTokenActivityTimeoutByToken(token) < -1) { | ||||||
|                 continue; |                 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)) { |         if (StringUtils.isNotEmpty(ipaddr) && StringUtils.isNotEmpty(userName)) { | ||||||
|             userOnlineDTOList = userOnlineDTOList.stream().filter(userOnline -> |             userOnlineDTOList = StreamUtils.filter(userOnlineDTOList, userOnline -> | ||||||
|                 StringUtils.equals(ipaddr, userOnline.getIpaddr()) && |                 StringUtils.equals(ipaddr, userOnline.getIpaddr()) && | ||||||
|                     StringUtils.equals(userName, userOnline.getUserName()) |                     StringUtils.equals(userName, userOnline.getUserName()) | ||||||
|             ).collect(Collectors.toList()); |             ); | ||||||
|         } else if (StringUtils.isNotEmpty(ipaddr)) { |         } else if (StringUtils.isNotEmpty(ipaddr)) { | ||||||
|             userOnlineDTOList = userOnlineDTOList.stream().filter(userOnline -> |             userOnlineDTOList = StreamUtils.filter(userOnlineDTOList, userOnline -> | ||||||
|                     StringUtils.equals(ipaddr, userOnline.getIpaddr())) |                 StringUtils.equals(ipaddr, userOnline.getIpaddr()) | ||||||
|                 .collect(Collectors.toList()); |             ); | ||||||
|         } else if (StringUtils.isNotEmpty(userName)) { |         } else if (StringUtils.isNotEmpty(userName)) { | ||||||
|             userOnlineDTOList = userOnlineDTOList.stream().filter(userOnline -> |             userOnlineDTOList = StreamUtils.filter(userOnlineDTOList, userOnline -> | ||||||
|                 StringUtils.equals(userName, userOnline.getUserName()) |                 StringUtils.equals(userName, userOnline.getUserName()) | ||||||
|             ).collect(Collectors.toList()); |             ); | ||||||
|         } |         } | ||||||
|         Collections.reverse(userOnlineDTOList); |         Collections.reverse(userOnlineDTOList); | ||||||
|         userOnlineDTOList.removeAll(Collections.singleton(null)); |         userOnlineDTOList.removeAll(Collections.singleton(null)); | ||||||
| @@ -72,15 +74,16 @@ public class SysUserOnlineController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 强退用户 |      * 强退用户 | ||||||
|  |      * | ||||||
|  |      * @param tokenId token值 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("强退用户") |  | ||||||
|     @SaCheckPermission("monitor:online:forceLogout") |     @SaCheckPermission("monitor:online:forceLogout") | ||||||
|     @Log(title = "在线用户", businessType = BusinessType.FORCE) |     @Log(title = "在线用户", businessType = BusinessType.FORCE) | ||||||
|     @DeleteMapping("/{tokenId}") |     @DeleteMapping("/{tokenId}") | ||||||
|     public R<Void> forceLogout(@PathVariable String tokenId) { |     public R<Void> forceLogout(@PathVariable String tokenId) { | ||||||
|         try { |         try { | ||||||
|             StpUtil.kickoutByTokenValue(tokenId); |             StpUtil.kickoutByTokenValue(tokenId); | ||||||
|         } catch (NotLoginException e) { |         } catch (NotLoginException ignored) { | ||||||
|         } |         } | ||||||
|         return R.ok(); |         return R.ok(); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -4,16 +4,13 @@ import cn.dev33.satoken.annotation.SaCheckPermission; | |||||||
| import com.ruoyi.common.annotation.Log; | import com.ruoyi.common.annotation.Log; | ||||||
| import com.ruoyi.common.constant.UserConstants; | import com.ruoyi.common.constant.UserConstants; | ||||||
| import com.ruoyi.common.core.controller.BaseController; | 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.PageQuery; | ||||||
|  | import com.ruoyi.common.core.domain.R; | ||||||
| import com.ruoyi.common.core.page.TableDataInfo; | import com.ruoyi.common.core.page.TableDataInfo; | ||||||
| import com.ruoyi.common.enums.BusinessType; | import com.ruoyi.common.enums.BusinessType; | ||||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | import com.ruoyi.common.utils.poi.ExcelUtil; | ||||||
| import com.ruoyi.system.domain.SysConfig; | import com.ruoyi.system.domain.SysConfig; | ||||||
| import com.ruoyi.system.service.ISysConfigService; | 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 lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.validation.annotation.Validated; | import org.springframework.validation.annotation.Validated; | ||||||
| import org.springframework.web.bind.annotation.*; | import org.springframework.web.bind.annotation.*; | ||||||
| @@ -27,7 +24,6 @@ import java.util.List; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Validated | @Validated | ||||||
| @Api(value = "参数配置控制器", tags = {"参数配置管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| @RequestMapping("/system/config") | @RequestMapping("/system/config") | ||||||
| @@ -38,14 +34,15 @@ public class SysConfigController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 获取参数配置列表 |      * 获取参数配置列表 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("获取参数配置列表") |  | ||||||
|     @SaCheckPermission("system:config:list") |     @SaCheckPermission("system:config:list") | ||||||
|     @GetMapping("/list") |     @GetMapping("/list") | ||||||
|     public TableDataInfo<SysConfig> list(SysConfig config, PageQuery pageQuery) { |     public TableDataInfo<SysConfig> list(SysConfig config, PageQuery pageQuery) { | ||||||
|         return configService.selectPageConfigList(config, pageQuery); |         return configService.selectPageConfigList(config, pageQuery); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiOperation("导出参数配置列表") |     /** | ||||||
|  |      * 导出参数配置列表 | ||||||
|  |      */ | ||||||
|     @Log(title = "参数管理", businessType = BusinessType.EXPORT) |     @Log(title = "参数管理", businessType = BusinessType.EXPORT) | ||||||
|     @SaCheckPermission("system:config:export") |     @SaCheckPermission("system:config:export") | ||||||
|     @PostMapping("/export") |     @PostMapping("/export") | ||||||
| @@ -56,27 +53,28 @@ public class SysConfigController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 根据参数编号获取详细信息 |      * 根据参数编号获取详细信息 | ||||||
|  |      * | ||||||
|  |      * @param configId 参数ID | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("根据参数编号获取详细信息") |  | ||||||
|     @SaCheckPermission("system:config:query") |     @SaCheckPermission("system:config:query") | ||||||
|     @GetMapping(value = "/{configId}") |     @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)); |         return R.ok(configService.selectConfigById(configId)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 根据参数键名查询参数值 |      * 根据参数键名查询参数值 | ||||||
|  |      * | ||||||
|  |      * @param configKey 参数Key | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("根据参数键名查询参数值") |  | ||||||
|     @GetMapping(value = "/configKey/{configKey}") |     @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)); |         return R.ok(configService.selectConfigByKey(configKey)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 新增参数配置 |      * 新增参数配置 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("新增参数配置") |  | ||||||
|     @SaCheckPermission("system:config:add") |     @SaCheckPermission("system:config:add") | ||||||
|     @Log(title = "参数管理", businessType = BusinessType.INSERT) |     @Log(title = "参数管理", businessType = BusinessType.INSERT) | ||||||
|     @PostMapping |     @PostMapping | ||||||
| @@ -84,13 +82,13 @@ public class SysConfigController extends BaseController { | |||||||
|         if (UserConstants.NOT_UNIQUE.equals(configService.checkConfigKeyUnique(config))) { |         if (UserConstants.NOT_UNIQUE.equals(configService.checkConfigKeyUnique(config))) { | ||||||
|             return R.fail("新增参数'" + config.getConfigName() + "'失败,参数键名已存在"); |             return R.fail("新增参数'" + config.getConfigName() + "'失败,参数键名已存在"); | ||||||
|         } |         } | ||||||
|         return toAjax(configService.insertConfig(config)); |         configService.insertConfig(config); | ||||||
|  |         return R.ok(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 修改参数配置 |      * 修改参数配置 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("修改参数配置") |  | ||||||
|     @SaCheckPermission("system:config:edit") |     @SaCheckPermission("system:config:edit") | ||||||
|     @Log(title = "参数管理", businessType = BusinessType.UPDATE) |     @Log(title = "参数管理", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping |     @PutMapping | ||||||
| @@ -98,28 +96,30 @@ public class SysConfigController extends BaseController { | |||||||
|         if (UserConstants.NOT_UNIQUE.equals(configService.checkConfigKeyUnique(config))) { |         if (UserConstants.NOT_UNIQUE.equals(configService.checkConfigKeyUnique(config))) { | ||||||
|             return R.fail("修改参数'" + config.getConfigName() + "'失败,参数键名已存在"); |             return R.fail("修改参数'" + config.getConfigName() + "'失败,参数键名已存在"); | ||||||
|         } |         } | ||||||
|         return toAjax(configService.updateConfig(config)); |         configService.updateConfig(config); | ||||||
|  |         return R.ok(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 根据参数键名修改参数配置 |      * 根据参数键名修改参数配置 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("根据参数键名修改参数配置") |  | ||||||
|     @SaCheckPermission("system:config:edit") |     @SaCheckPermission("system:config:edit") | ||||||
|     @Log(title = "参数管理", businessType = BusinessType.UPDATE) |     @Log(title = "参数管理", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping("/updateByKey") |     @PutMapping("/updateByKey") | ||||||
|     public R<Void> updateByKey(@RequestBody SysConfig config) { |     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") |     @SaCheckPermission("system:config:remove") | ||||||
|     @Log(title = "参数管理", businessType = BusinessType.DELETE) |     @Log(title = "参数管理", businessType = BusinessType.DELETE) | ||||||
|     @DeleteMapping("/{configIds}") |     @DeleteMapping("/{configIds}") | ||||||
|     public R<Void> remove(@ApiParam("参数ID串") @PathVariable Long[] configIds) { |     public R<Void> remove(@PathVariable Long[] configIds) { | ||||||
|         configService.deleteConfigByIds(configIds); |         configService.deleteConfigByIds(configIds); | ||||||
|         return R.ok(); |         return R.ok(); | ||||||
|     } |     } | ||||||
| @@ -127,7 +127,6 @@ public class SysConfigController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 刷新参数缓存 |      * 刷新参数缓存 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("刷新参数缓存") |  | ||||||
|     @SaCheckPermission("system:config:remove") |     @SaCheckPermission("system:config:remove") | ||||||
|     @Log(title = "参数管理", businessType = BusinessType.CLEAN) |     @Log(title = "参数管理", businessType = BusinessType.CLEAN) | ||||||
|     @DeleteMapping("/refreshCache") |     @DeleteMapping("/refreshCache") | ||||||
|   | |||||||
| @@ -1,7 +1,6 @@ | |||||||
| package com.ruoyi.web.controller.system; | package com.ruoyi.web.controller.system; | ||||||
|  |  | ||||||
| import cn.dev33.satoken.annotation.SaCheckPermission; | import cn.dev33.satoken.annotation.SaCheckPermission; | ||||||
| import cn.hutool.core.lang.tree.Tree; |  | ||||||
| import cn.hutool.core.util.ArrayUtil; | import cn.hutool.core.util.ArrayUtil; | ||||||
| import com.ruoyi.common.annotation.Log; | import com.ruoyi.common.annotation.Log; | ||||||
| import com.ruoyi.common.constant.UserConstants; | 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.enums.BusinessType; | ||||||
| import com.ruoyi.common.utils.StringUtils; | import com.ruoyi.common.utils.StringUtils; | ||||||
| import com.ruoyi.system.service.ISysDeptService; | 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 lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.validation.annotation.Validated; | import org.springframework.validation.annotation.Validated; | ||||||
| import org.springframework.web.bind.annotation.*; | import org.springframework.web.bind.annotation.*; | ||||||
|  |  | ||||||
| import java.util.HashMap; |  | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Map; |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 部门信息 |  * 部门信息 | ||||||
| @@ -28,7 +22,6 @@ import java.util.Map; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Validated | @Validated | ||||||
| @Api(value = "部门控制器", tags = {"部门管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| @RequestMapping("/system/dept") | @RequestMapping("/system/dept") | ||||||
| @@ -39,7 +32,6 @@ public class SysDeptController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 获取部门列表 |      * 获取部门列表 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("获取部门列表") |  | ||||||
|     @SaCheckPermission("system:dept:list") |     @SaCheckPermission("system:dept:list") | ||||||
|     @GetMapping("/list") |     @GetMapping("/list") | ||||||
|     public R<List<SysDept>> list(SysDept dept) { |     public R<List<SysDept>> list(SysDept dept) { | ||||||
| @@ -49,11 +41,12 @@ public class SysDeptController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 查询部门列表(排除节点) |      * 查询部门列表(排除节点) | ||||||
|  |      * | ||||||
|  |      * @param deptId 部门ID | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("查询部门列表(排除节点)") |  | ||||||
|     @SaCheckPermission("system:dept:list") |     @SaCheckPermission("system:dept:list") | ||||||
|     @GetMapping("/list/exclude/{deptId}") |     @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()); |         List<SysDept> depts = deptService.selectDeptList(new SysDept()); | ||||||
|         depts.removeIf(d -> d.getDeptId().equals(deptId) |         depts.removeIf(d -> d.getDeptId().equals(deptId) | ||||||
|             || ArrayUtil.contains(StringUtils.split(d.getAncestors(), ","), 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") |     @SaCheckPermission("system:dept:query") | ||||||
|     @GetMapping(value = "/{deptId}") |     @GetMapping(value = "/{deptId}") | ||||||
|     public R<SysDept> getInfo(@ApiParam("部门ID") @PathVariable Long deptId) { |     public R<SysDept> getInfo(@PathVariable Long deptId) { | ||||||
|         deptService.checkDeptDataScope(deptId); |         deptService.checkDeptDataScope(deptId); | ||||||
|         return R.ok(deptService.selectDeptById(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") |     @SaCheckPermission("system:dept:add") | ||||||
|     @Log(title = "部门管理", businessType = BusinessType.INSERT) |     @Log(title = "部门管理", businessType = BusinessType.INSERT) | ||||||
|     @PostMapping |     @PostMapping | ||||||
| @@ -111,7 +81,6 @@ public class SysDeptController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 修改部门 |      * 修改部门 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("修改部门") |  | ||||||
|     @SaCheckPermission("system:dept:edit") |     @SaCheckPermission("system:dept:edit") | ||||||
|     @Log(title = "部门管理", businessType = BusinessType.UPDATE) |     @Log(title = "部门管理", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping |     @PutMapping | ||||||
| @@ -131,17 +100,18 @@ public class SysDeptController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 删除部门 |      * 删除部门 | ||||||
|  |      * | ||||||
|  |      * @param deptId 部门ID | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("删除部门") |  | ||||||
|     @SaCheckPermission("system:dept:remove") |     @SaCheckPermission("system:dept:remove") | ||||||
|     @Log(title = "部门管理", businessType = BusinessType.DELETE) |     @Log(title = "部门管理", businessType = BusinessType.DELETE) | ||||||
|     @DeleteMapping("/{deptId}") |     @DeleteMapping("/{deptId}") | ||||||
|     public R<Void> remove(@ApiParam("部门ID串") @PathVariable Long deptId) { |     public R<Void> remove(@PathVariable Long deptId) { | ||||||
|         if (deptService.hasChildByDeptId(deptId)) { |         if (deptService.hasChildByDeptId(deptId)) { | ||||||
|             return R.fail("存在下级部门,不允许删除"); |             return R.warn("存在下级部门,不允许删除"); | ||||||
|         } |         } | ||||||
|         if (deptService.checkDeptExistUser(deptId)) { |         if (deptService.checkDeptExistUser(deptId)) { | ||||||
|             return R.fail("部门存在用户,不允许删除"); |             return R.warn("部门存在用户,不允许删除"); | ||||||
|         } |         } | ||||||
|         deptService.checkDeptDataScope(deptId); |         deptService.checkDeptDataScope(deptId); | ||||||
|         return toAjax(deptService.deleteDeptById(deptId)); |         return toAjax(deptService.deleteDeptById(deptId)); | ||||||
|   | |||||||
| @@ -4,17 +4,14 @@ import cn.dev33.satoken.annotation.SaCheckPermission; | |||||||
| import cn.hutool.core.util.ObjectUtil; | import cn.hutool.core.util.ObjectUtil; | ||||||
| import com.ruoyi.common.annotation.Log; | import com.ruoyi.common.annotation.Log; | ||||||
| import com.ruoyi.common.core.controller.BaseController; | 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.PageQuery; | ||||||
|  | import com.ruoyi.common.core.domain.R; | ||||||
| import com.ruoyi.common.core.domain.entity.SysDictData; | import com.ruoyi.common.core.domain.entity.SysDictData; | ||||||
| import com.ruoyi.common.core.page.TableDataInfo; | import com.ruoyi.common.core.page.TableDataInfo; | ||||||
| import com.ruoyi.common.enums.BusinessType; | import com.ruoyi.common.enums.BusinessType; | ||||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | import com.ruoyi.common.utils.poi.ExcelUtil; | ||||||
| import com.ruoyi.system.service.ISysDictDataService; | import com.ruoyi.system.service.ISysDictDataService; | ||||||
| import com.ruoyi.system.service.ISysDictTypeService; | 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 lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.validation.annotation.Validated; | import org.springframework.validation.annotation.Validated; | ||||||
| import org.springframework.web.bind.annotation.*; | import org.springframework.web.bind.annotation.*; | ||||||
| @@ -29,7 +26,6 @@ import java.util.List; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Validated | @Validated | ||||||
| @Api(value = "数据字典信息控制器", tags = {"数据字典信息管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| @RequestMapping("/system/dict/data") | @RequestMapping("/system/dict/data") | ||||||
| @@ -38,14 +34,18 @@ public class SysDictDataController extends BaseController { | |||||||
|     private final ISysDictDataService dictDataService; |     private final ISysDictDataService dictDataService; | ||||||
|     private final ISysDictTypeService dictTypeService; |     private final ISysDictTypeService dictTypeService; | ||||||
|  |  | ||||||
|     @ApiOperation("查询字典数据列表") |     /** | ||||||
|  |      * 查询字典数据列表 | ||||||
|  |      */ | ||||||
|     @SaCheckPermission("system:dict:list") |     @SaCheckPermission("system:dict:list") | ||||||
|     @GetMapping("/list") |     @GetMapping("/list") | ||||||
|     public TableDataInfo<SysDictData> list(SysDictData dictData, PageQuery pageQuery) { |     public TableDataInfo<SysDictData> list(SysDictData dictData, PageQuery pageQuery) { | ||||||
|         return dictDataService.selectPageDictDataList(dictData, pageQuery); |         return dictDataService.selectPageDictDataList(dictData, pageQuery); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiOperation("导出字典数据列表") |     /** | ||||||
|  |      * 导出字典数据列表 | ||||||
|  |      */ | ||||||
|     @Log(title = "字典数据", businessType = BusinessType.EXPORT) |     @Log(title = "字典数据", businessType = BusinessType.EXPORT) | ||||||
|     @SaCheckPermission("system:dict:export") |     @SaCheckPermission("system:dict:export") | ||||||
|     @PostMapping("/export") |     @PostMapping("/export") | ||||||
| @@ -56,20 +56,22 @@ public class SysDictDataController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 查询字典数据详细 |      * 查询字典数据详细 | ||||||
|  |      * | ||||||
|  |      * @param dictCode 字典code | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("查询字典数据详细") |  | ||||||
|     @SaCheckPermission("system:dict:query") |     @SaCheckPermission("system:dict:query") | ||||||
|     @GetMapping(value = "/{dictCode}") |     @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)); |         return R.ok(dictDataService.selectDictDataById(dictCode)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 根据字典类型查询字典数据信息 |      * 根据字典类型查询字典数据信息 | ||||||
|  |      * | ||||||
|  |      * @param dictType 字典类型 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("根据字典类型查询字典数据信息") |  | ||||||
|     @GetMapping(value = "/type/{dictType}") |     @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); |         List<SysDictData> data = dictTypeService.selectDictDataByType(dictType); | ||||||
|         if (ObjectUtil.isNull(data)) { |         if (ObjectUtil.isNull(data)) { | ||||||
|             data = new ArrayList<>(); |             data = new ArrayList<>(); | ||||||
| @@ -80,33 +82,34 @@ public class SysDictDataController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 新增字典类型 |      * 新增字典类型 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("新增字典类型") |  | ||||||
|     @SaCheckPermission("system:dict:add") |     @SaCheckPermission("system:dict:add") | ||||||
|     @Log(title = "字典数据", businessType = BusinessType.INSERT) |     @Log(title = "字典数据", businessType = BusinessType.INSERT) | ||||||
|     @PostMapping |     @PostMapping | ||||||
|     public R<Void> add(@Validated @RequestBody SysDictData dict) { |     public R<Void> add(@Validated @RequestBody SysDictData dict) { | ||||||
|         return toAjax(dictDataService.insertDictData(dict)); |         dictDataService.insertDictData(dict); | ||||||
|  |         return R.ok(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 修改保存字典类型 |      * 修改保存字典类型 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("修改保存字典类型") |  | ||||||
|     @SaCheckPermission("system:dict:edit") |     @SaCheckPermission("system:dict:edit") | ||||||
|     @Log(title = "字典数据", businessType = BusinessType.UPDATE) |     @Log(title = "字典数据", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping |     @PutMapping | ||||||
|     public R<Void> edit(@Validated @RequestBody SysDictData dict) { |     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") |     @SaCheckPermission("system:dict:remove") | ||||||
|     @Log(title = "字典类型", businessType = BusinessType.DELETE) |     @Log(title = "字典类型", businessType = BusinessType.DELETE) | ||||||
|     @DeleteMapping("/{dictCodes}") |     @DeleteMapping("/{dictCodes}") | ||||||
|     public R<Void> remove(@ApiParam("字典code串") @PathVariable Long[] dictCodes) { |     public R<Void> remove(@PathVariable Long[] dictCodes) { | ||||||
|         dictDataService.deleteDictDataByIds(dictCodes); |         dictDataService.deleteDictDataByIds(dictCodes); | ||||||
|         return R.ok(); |         return R.ok(); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -4,16 +4,13 @@ import cn.dev33.satoken.annotation.SaCheckPermission; | |||||||
| import com.ruoyi.common.annotation.Log; | import com.ruoyi.common.annotation.Log; | ||||||
| import com.ruoyi.common.constant.UserConstants; | import com.ruoyi.common.constant.UserConstants; | ||||||
| import com.ruoyi.common.core.controller.BaseController; | 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.PageQuery; | ||||||
|  | import com.ruoyi.common.core.domain.R; | ||||||
| import com.ruoyi.common.core.domain.entity.SysDictType; | import com.ruoyi.common.core.domain.entity.SysDictType; | ||||||
| import com.ruoyi.common.core.page.TableDataInfo; | import com.ruoyi.common.core.page.TableDataInfo; | ||||||
| import com.ruoyi.common.enums.BusinessType; | import com.ruoyi.common.enums.BusinessType; | ||||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | import com.ruoyi.common.utils.poi.ExcelUtil; | ||||||
| import com.ruoyi.system.service.ISysDictTypeService; | 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 lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.validation.annotation.Validated; | import org.springframework.validation.annotation.Validated; | ||||||
| import org.springframework.web.bind.annotation.*; | import org.springframework.web.bind.annotation.*; | ||||||
| @@ -27,7 +24,6 @@ import java.util.List; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Validated | @Validated | ||||||
| @Api(value = "数据字典信息控制器", tags = {"数据字典信息管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| @RequestMapping("/system/dict/type") | @RequestMapping("/system/dict/type") | ||||||
| @@ -35,14 +31,18 @@ public class SysDictTypeController extends BaseController { | |||||||
|  |  | ||||||
|     private final ISysDictTypeService dictTypeService; |     private final ISysDictTypeService dictTypeService; | ||||||
|  |  | ||||||
|     @ApiOperation("查询字典类型列表") |     /** | ||||||
|  |      * 查询字典类型列表 | ||||||
|  |      */ | ||||||
|     @SaCheckPermission("system:dict:list") |     @SaCheckPermission("system:dict:list") | ||||||
|     @GetMapping("/list") |     @GetMapping("/list") | ||||||
|     public TableDataInfo<SysDictType> list(SysDictType dictType, PageQuery pageQuery) { |     public TableDataInfo<SysDictType> list(SysDictType dictType, PageQuery pageQuery) { | ||||||
|         return dictTypeService.selectPageDictTypeList(dictType, pageQuery); |         return dictTypeService.selectPageDictTypeList(dictType, pageQuery); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiOperation("导出字典类型列表") |     /** | ||||||
|  |      * 导出字典类型列表 | ||||||
|  |      */ | ||||||
|     @Log(title = "字典类型", businessType = BusinessType.EXPORT) |     @Log(title = "字典类型", businessType = BusinessType.EXPORT) | ||||||
|     @SaCheckPermission("system:dict:export") |     @SaCheckPermission("system:dict:export") | ||||||
|     @PostMapping("/export") |     @PostMapping("/export") | ||||||
| @@ -53,18 +53,18 @@ public class SysDictTypeController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 查询字典类型详细 |      * 查询字典类型详细 | ||||||
|  |      * | ||||||
|  |      * @param dictId 字典ID | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("查询字典类型详细") |  | ||||||
|     @SaCheckPermission("system:dict:query") |     @SaCheckPermission("system:dict:query") | ||||||
|     @GetMapping(value = "/{dictId}") |     @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)); |         return R.ok(dictTypeService.selectDictTypeById(dictId)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 新增字典类型 |      * 新增字典类型 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("新增字典类型") |  | ||||||
|     @SaCheckPermission("system:dict:add") |     @SaCheckPermission("system:dict:add") | ||||||
|     @Log(title = "字典类型", businessType = BusinessType.INSERT) |     @Log(title = "字典类型", businessType = BusinessType.INSERT) | ||||||
|     @PostMapping |     @PostMapping | ||||||
| @@ -72,13 +72,13 @@ public class SysDictTypeController extends BaseController { | |||||||
|         if (UserConstants.NOT_UNIQUE.equals(dictTypeService.checkDictTypeUnique(dict))) { |         if (UserConstants.NOT_UNIQUE.equals(dictTypeService.checkDictTypeUnique(dict))) { | ||||||
|             return R.fail("新增字典'" + dict.getDictName() + "'失败,字典类型已存在"); |             return R.fail("新增字典'" + dict.getDictName() + "'失败,字典类型已存在"); | ||||||
|         } |         } | ||||||
|         return toAjax(dictTypeService.insertDictType(dict)); |         dictTypeService.insertDictType(dict); | ||||||
|  |         return R.ok(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 修改字典类型 |      * 修改字典类型 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("修改字典类型") |  | ||||||
|     @SaCheckPermission("system:dict:edit") |     @SaCheckPermission("system:dict:edit") | ||||||
|     @Log(title = "字典类型", businessType = BusinessType.UPDATE) |     @Log(title = "字典类型", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping |     @PutMapping | ||||||
| @@ -86,17 +86,19 @@ public class SysDictTypeController extends BaseController { | |||||||
|         if (UserConstants.NOT_UNIQUE.equals(dictTypeService.checkDictTypeUnique(dict))) { |         if (UserConstants.NOT_UNIQUE.equals(dictTypeService.checkDictTypeUnique(dict))) { | ||||||
|             return R.fail("修改字典'" + dict.getDictName() + "'失败,字典类型已存在"); |             return R.fail("修改字典'" + dict.getDictName() + "'失败,字典类型已存在"); | ||||||
|         } |         } | ||||||
|         return toAjax(dictTypeService.updateDictType(dict)); |         dictTypeService.updateDictType(dict); | ||||||
|  |         return R.ok(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 删除字典类型 |      * 删除字典类型 | ||||||
|  |      * | ||||||
|  |      * @param dictIds 字典ID串 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("删除字典类型") |  | ||||||
|     @SaCheckPermission("system:dict:remove") |     @SaCheckPermission("system:dict:remove") | ||||||
|     @Log(title = "字典类型", businessType = BusinessType.DELETE) |     @Log(title = "字典类型", businessType = BusinessType.DELETE) | ||||||
|     @DeleteMapping("/{dictIds}") |     @DeleteMapping("/{dictIds}") | ||||||
|     public R<Void> remove(@ApiParam("字典ID串") @PathVariable Long[] dictIds) { |     public R<Void> remove(@PathVariable Long[] dictIds) { | ||||||
|         dictTypeService.deleteDictTypeByIds(dictIds); |         dictTypeService.deleteDictTypeByIds(dictIds); | ||||||
|         return R.ok(); |         return R.ok(); | ||||||
|     } |     } | ||||||
| @@ -104,7 +106,6 @@ public class SysDictTypeController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 刷新字典缓存 |      * 刷新字典缓存 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("刷新字典缓存") |  | ||||||
|     @SaCheckPermission("system:dict:remove") |     @SaCheckPermission("system:dict:remove") | ||||||
|     @Log(title = "字典类型", businessType = BusinessType.CLEAN) |     @Log(title = "字典类型", businessType = BusinessType.CLEAN) | ||||||
|     @DeleteMapping("/refreshCache") |     @DeleteMapping("/refreshCache") | ||||||
| @@ -116,7 +117,6 @@ public class SysDictTypeController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 获取字典选择框列表 |      * 获取字典选择框列表 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("获取字典选择框列表") |  | ||||||
|     @GetMapping("/optionselect") |     @GetMapping("/optionselect") | ||||||
|     public R<List<SysDictType>> optionselect() { |     public R<List<SysDictType>> optionselect() { | ||||||
|         List<SysDictType> dictTypes = dictTypeService.selectDictTypeAll(); |         List<SysDictType> dictTypes = dictTypeService.selectDictTypeAll(); | ||||||
|   | |||||||
| @@ -1,9 +1,8 @@ | |||||||
| package com.ruoyi.web.controller.system; | package com.ruoyi.web.controller.system; | ||||||
|  |  | ||||||
|  | import cn.dev33.satoken.annotation.SaIgnore; | ||||||
| import com.ruoyi.common.config.RuoYiConfig; | import com.ruoyi.common.config.RuoYiConfig; | ||||||
| import com.ruoyi.common.utils.StringUtils; | import com.ruoyi.common.utils.StringUtils; | ||||||
| import io.swagger.annotations.Api; |  | ||||||
| import io.swagger.annotations.ApiOperation; |  | ||||||
| import lombok.RequiredArgsConstructor; | import lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.web.bind.annotation.GetMapping; | import org.springframework.web.bind.annotation.GetMapping; | ||||||
| import org.springframework.web.bind.annotation.RestController; | import org.springframework.web.bind.annotation.RestController; | ||||||
| @@ -13,7 +12,6 @@ import org.springframework.web.bind.annotation.RestController; | |||||||
|  * |  * | ||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Api(value = "首页控制器", tags = {"首页管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| public class SysIndexController { | public class SysIndexController { | ||||||
| @@ -26,7 +24,7 @@ public class SysIndexController { | |||||||
|     /** |     /** | ||||||
|      * 访问首页,提示语 |      * 访问首页,提示语 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("访问首页,提示语") |     @SaIgnore | ||||||
|     @GetMapping("/") |     @GetMapping("/") | ||||||
|     public String index() { |     public String index() { | ||||||
|         return StringUtils.format("欢迎使用{}后台管理框架,当前版本:v{},请通过前端地址访问。", ruoyiConfig.getName(), ruoyiConfig.getVersion()); |         return StringUtils.format("欢迎使用{}后台管理框架,当前版本:v{},请通过前端地址访问。", ruoyiConfig.getName(), ruoyiConfig.getVersion()); | ||||||
|   | |||||||
| @@ -1,20 +1,18 @@ | |||||||
| package com.ruoyi.web.controller.system; | package com.ruoyi.web.controller.system; | ||||||
|  |  | ||||||
| import cn.dev33.satoken.exception.NotLoginException; | import cn.dev33.satoken.annotation.SaIgnore; | ||||||
| import cn.dev33.satoken.stp.StpUtil; |  | ||||||
| import com.ruoyi.common.constant.Constants; | import com.ruoyi.common.constant.Constants; | ||||||
| import com.ruoyi.common.core.domain.R; | import com.ruoyi.common.core.domain.R; | ||||||
| import com.ruoyi.common.core.domain.entity.SysMenu; | import com.ruoyi.common.core.domain.entity.SysMenu; | ||||||
| import com.ruoyi.common.core.domain.entity.SysUser; | import com.ruoyi.common.core.domain.entity.SysUser; | ||||||
| import com.ruoyi.common.core.domain.model.LoginBody; | import com.ruoyi.common.core.domain.model.LoginBody; | ||||||
|  | import com.ruoyi.common.core.domain.model.LoginUser; | ||||||
|  | import com.ruoyi.common.core.domain.model.SmsLoginBody; | ||||||
| import com.ruoyi.common.helper.LoginHelper; | import com.ruoyi.common.helper.LoginHelper; | ||||||
| import com.ruoyi.system.domain.vo.RouterVo; | import com.ruoyi.system.domain.vo.RouterVo; | ||||||
| import com.ruoyi.system.service.ISysMenuService; | import com.ruoyi.system.service.ISysMenuService; | ||||||
| import com.ruoyi.system.service.ISysUserService; | import com.ruoyi.system.service.ISysUserService; | ||||||
| import com.ruoyi.system.service.SysLoginService; | 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 lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.validation.annotation.Validated; | import org.springframework.validation.annotation.Validated; | ||||||
| import org.springframework.web.bind.annotation.GetMapping; | import org.springframework.web.bind.annotation.GetMapping; | ||||||
| @@ -22,10 +20,10 @@ import org.springframework.web.bind.annotation.PostMapping; | |||||||
| import org.springframework.web.bind.annotation.RequestBody; | import org.springframework.web.bind.annotation.RequestBody; | ||||||
| import org.springframework.web.bind.annotation.RestController; | import org.springframework.web.bind.annotation.RestController; | ||||||
|  |  | ||||||
|  | import javax.validation.constraints.NotBlank; | ||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
| import java.util.Set; |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 登录验证 |  * 登录验证 | ||||||
| @@ -33,7 +31,6 @@ import java.util.Set; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Validated | @Validated | ||||||
| @Api(value = "登录验证控制器", tags = {"登录验证管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| public class SysLoginController { | public class SysLoginController { | ||||||
| @@ -41,7 +38,6 @@ public class SysLoginController { | |||||||
|     private final SysLoginService loginService; |     private final SysLoginService loginService; | ||||||
|     private final ISysMenuService menuService; |     private final ISysMenuService menuService; | ||||||
|     private final ISysUserService userService; |     private final ISysUserService userService; | ||||||
|     private final SysPermissionService permissionService; |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 登录方法 |      * 登录方法 | ||||||
| @@ -49,7 +45,7 @@ public class SysLoginController { | |||||||
|      * @param loginBody 登录信息 |      * @param loginBody 登录信息 | ||||||
|      * @return 结果 |      * @return 结果 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("登录方法") |     @SaIgnore | ||||||
|     @PostMapping("/login") |     @PostMapping("/login") | ||||||
|     public R<Map<String, Object>> login(@Validated @RequestBody LoginBody loginBody) { |     public R<Map<String, Object>> login(@Validated @RequestBody LoginBody loginBody) { | ||||||
|         Map<String, Object> ajax = new HashMap<>(); |         Map<String, Object> ajax = new HashMap<>(); | ||||||
| @@ -60,14 +56,45 @@ public class SysLoginController { | |||||||
|         return R.ok(ajax); |         return R.ok(ajax); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiOperation("登出方法") |     /** | ||||||
|  |      * 短信登录(示例) | ||||||
|  |      * | ||||||
|  |      * @param smsLoginBody 登录信息 | ||||||
|  |      * @return 结果 | ||||||
|  |      */ | ||||||
|  |     @SaIgnore | ||||||
|  |     @PostMapping("/smsLogin") | ||||||
|  |     public R<Map<String, Object>> smsLogin(@Validated @RequestBody SmsLoginBody smsLoginBody) { | ||||||
|  |         Map<String, Object> ajax = new HashMap<>(); | ||||||
|  |         // 生成令牌 | ||||||
|  |         String token = loginService.smsLogin(smsLoginBody.getPhonenumber(), smsLoginBody.getSmsCode()); | ||||||
|  |         ajax.put(Constants.TOKEN, token); | ||||||
|  |         return R.ok(ajax); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 小程序登录(示例) | ||||||
|  |      * | ||||||
|  |      * @param xcxCode 小程序code | ||||||
|  |      * @return 结果 | ||||||
|  |      */ | ||||||
|  |     @SaIgnore | ||||||
|  |     @PostMapping("/xcxLogin") | ||||||
|  |     public R<Map<String, Object>> xcxLogin(@NotBlank(message = "{xcx.code.not.blank}") String xcxCode) { | ||||||
|  |         Map<String, Object> ajax = new HashMap<>(); | ||||||
|  |         // 生成令牌 | ||||||
|  |         String token = loginService.xcxLogin(xcxCode); | ||||||
|  |         ajax.put(Constants.TOKEN, token); | ||||||
|  |         return R.ok(ajax); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 退出登录 | ||||||
|  |      */ | ||||||
|  |     @SaIgnore | ||||||
|     @PostMapping("/logout") |     @PostMapping("/logout") | ||||||
|     public R<Void> logout() { |     public R<Void> logout() { | ||||||
|         try { |         loginService.logout(); | ||||||
|             StpUtil.logout(); |  | ||||||
|             loginService.logout(LoginHelper.getUsername()); |  | ||||||
|         } catch (NotLoginException e) { |  | ||||||
|         } |  | ||||||
|         return R.ok("退出成功"); |         return R.ok("退出成功"); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -76,18 +103,14 @@ public class SysLoginController { | |||||||
|      * |      * | ||||||
|      * @return 用户信息 |      * @return 用户信息 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("获取用户信息") |  | ||||||
|     @GetMapping("getInfo") |     @GetMapping("getInfo") | ||||||
|     public R<Map<String, Object>> getInfo() { |     public R<Map<String, Object>> getInfo() { | ||||||
|         SysUser user = userService.selectUserById(LoginHelper.getUserId()); |         LoginUser loginUser = LoginHelper.getLoginUser(); | ||||||
|         // 角色集合 |         SysUser user = userService.selectUserById(loginUser.getUserId()); | ||||||
|         Set<String> roles = permissionService.getRolePermission(user); |  | ||||||
|         // 权限集合 |  | ||||||
|         Set<String> permissions = permissionService.getMenuPermission(user); |  | ||||||
|         Map<String, Object> ajax = new HashMap<>(); |         Map<String, Object> ajax = new HashMap<>(); | ||||||
|         ajax.put("user", user); |         ajax.put("user", user); | ||||||
|         ajax.put("roles", roles); |         ajax.put("roles", loginUser.getRolePermission()); | ||||||
|         ajax.put("permissions", permissions); |         ajax.put("permissions", loginUser.getMenuPermission()); | ||||||
|         return R.ok(ajax); |         return R.ok(ajax); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -96,7 +119,6 @@ public class SysLoginController { | |||||||
|      * |      * | ||||||
|      * @return 路由信息 |      * @return 路由信息 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("获取路由信息") |  | ||||||
|     @GetMapping("getRouters") |     @GetMapping("getRouters") | ||||||
|     public R<List<RouterVo>> getRouters() { |     public R<List<RouterVo>> getRouters() { | ||||||
|         Long userId = LoginHelper.getUserId(); |         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.enums.BusinessType; | ||||||
| import com.ruoyi.common.utils.StringUtils; | import com.ruoyi.common.utils.StringUtils; | ||||||
| import com.ruoyi.system.service.ISysMenuService; | 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 lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.validation.annotation.Validated; | import org.springframework.validation.annotation.Validated; | ||||||
| import org.springframework.web.bind.annotation.*; | import org.springframework.web.bind.annotation.*; | ||||||
| @@ -27,7 +24,6 @@ import java.util.Map; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Validated | @Validated | ||||||
| @Api(value = "菜单信息控制器", tags = {"菜单信息管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| @RequestMapping("/system/menu") | @RequestMapping("/system/menu") | ||||||
| @@ -38,7 +34,6 @@ public class SysMenuController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 获取菜单列表 |      * 获取菜单列表 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("获取菜单列表") |  | ||||||
|     @SaCheckPermission("system:menu:list") |     @SaCheckPermission("system:menu:list") | ||||||
|     @GetMapping("/list") |     @GetMapping("/list") | ||||||
|     public R<List<SysMenu>> list(SysMenu menu) { |     public R<List<SysMenu>> list(SysMenu menu) { | ||||||
| @@ -48,18 +43,18 @@ public class SysMenuController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 根据菜单编号获取详细信息 |      * 根据菜单编号获取详细信息 | ||||||
|  |      * | ||||||
|  |      * @param menuId 菜单ID | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("根据菜单编号获取详细信息") |  | ||||||
|     @SaCheckPermission("system:menu:query") |     @SaCheckPermission("system:menu:query") | ||||||
|     @GetMapping(value = "/{menuId}") |     @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)); |         return R.ok(menuService.selectMenuById(menuId)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 获取菜单下拉树列表 |      * 获取菜单下拉树列表 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("获取菜单下拉树列表") |  | ||||||
|     @GetMapping("/treeselect") |     @GetMapping("/treeselect") | ||||||
|     public R<List<Tree<Long>>> treeselect(SysMenu menu) { |     public R<List<Tree<Long>>> treeselect(SysMenu menu) { | ||||||
|         List<SysMenu> menus = menuService.selectMenuList(menu, getUserId()); |         List<SysMenu> menus = menuService.selectMenuList(menu, getUserId()); | ||||||
| @@ -68,10 +63,11 @@ public class SysMenuController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 加载对应角色菜单列表树 |      * 加载对应角色菜单列表树 | ||||||
|  |      * | ||||||
|  |      * @param roleId 角色ID | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("加载对应角色菜单列表树") |  | ||||||
|     @GetMapping(value = "/roleMenuTreeselect/{roleId}") |     @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()); |         List<SysMenu> menus = menuService.selectMenuList(getUserId()); | ||||||
|         Map<String, Object> ajax = new HashMap<>(); |         Map<String, Object> ajax = new HashMap<>(); | ||||||
|         ajax.put("checkedKeys", menuService.selectMenuListByRoleId(roleId)); |         ajax.put("checkedKeys", menuService.selectMenuListByRoleId(roleId)); | ||||||
| @@ -82,7 +78,6 @@ public class SysMenuController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 新增菜单 |      * 新增菜单 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("新增菜单") |  | ||||||
|     @SaCheckPermission("system:menu:add") |     @SaCheckPermission("system:menu:add") | ||||||
|     @Log(title = "菜单管理", businessType = BusinessType.INSERT) |     @Log(title = "菜单管理", businessType = BusinessType.INSERT) | ||||||
|     @PostMapping |     @PostMapping | ||||||
| @@ -98,7 +93,6 @@ public class SysMenuController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 修改菜单 |      * 修改菜单 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("修改菜单") |  | ||||||
|     @SaCheckPermission("system:menu:edit") |     @SaCheckPermission("system:menu:edit") | ||||||
|     @Log(title = "菜单管理", businessType = BusinessType.UPDATE) |     @Log(title = "菜单管理", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping |     @PutMapping | ||||||
| @@ -115,17 +109,18 @@ public class SysMenuController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 删除菜单 |      * 删除菜单 | ||||||
|  |      * | ||||||
|  |      * @param menuId 菜单ID | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("删除菜单") |  | ||||||
|     @SaCheckPermission("system:menu:remove") |     @SaCheckPermission("system:menu:remove") | ||||||
|     @Log(title = "菜单管理", businessType = BusinessType.DELETE) |     @Log(title = "菜单管理", businessType = BusinessType.DELETE) | ||||||
|     @DeleteMapping("/{menuId}") |     @DeleteMapping("/{menuId}") | ||||||
|     public R<Void> remove(@ApiParam("菜单ID") @PathVariable("menuId") Long menuId) { |     public R<Void> remove(@PathVariable("menuId") Long menuId) { | ||||||
|         if (menuService.hasChildByMenuId(menuId)) { |         if (menuService.hasChildByMenuId(menuId)) { | ||||||
|             return R.fail("存在子菜单,不允许删除"); |             return R.warn("存在子菜单,不允许删除"); | ||||||
|         } |         } | ||||||
|         if (menuService.checkMenuExistRole(menuId)) { |         if (menuService.checkMenuExistRole(menuId)) { | ||||||
|             return R.fail("菜单已分配,不允许删除"); |             return R.warn("菜单已分配,不允许删除"); | ||||||
|         } |         } | ||||||
|         return toAjax(menuService.deleteMenuById(menuId)); |         return toAjax(menuService.deleteMenuById(menuId)); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -3,15 +3,12 @@ package com.ruoyi.web.controller.system; | |||||||
| import cn.dev33.satoken.annotation.SaCheckPermission; | import cn.dev33.satoken.annotation.SaCheckPermission; | ||||||
| import com.ruoyi.common.annotation.Log; | import com.ruoyi.common.annotation.Log; | ||||||
| import com.ruoyi.common.core.controller.BaseController; | 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.PageQuery; | ||||||
|  | import com.ruoyi.common.core.domain.R; | ||||||
| import com.ruoyi.common.core.page.TableDataInfo; | import com.ruoyi.common.core.page.TableDataInfo; | ||||||
| import com.ruoyi.common.enums.BusinessType; | import com.ruoyi.common.enums.BusinessType; | ||||||
| import com.ruoyi.system.domain.SysNotice; | import com.ruoyi.system.domain.SysNotice; | ||||||
| import com.ruoyi.system.service.ISysNoticeService; | 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 lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.validation.annotation.Validated; | import org.springframework.validation.annotation.Validated; | ||||||
| import org.springframework.web.bind.annotation.*; | import org.springframework.web.bind.annotation.*; | ||||||
| @@ -22,7 +19,6 @@ import org.springframework.web.bind.annotation.*; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Validated | @Validated | ||||||
| @Api(value = "公告信息控制器", tags = {"公告信息管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| @RequestMapping("/system/notice") | @RequestMapping("/system/notice") | ||||||
| @@ -33,7 +29,6 @@ public class SysNoticeController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 获取通知公告列表 |      * 获取通知公告列表 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("获取通知公告列表") |  | ||||||
|     @SaCheckPermission("system:notice:list") |     @SaCheckPermission("system:notice:list") | ||||||
|     @GetMapping("/list") |     @GetMapping("/list") | ||||||
|     public TableDataInfo<SysNotice> list(SysNotice notice, PageQuery pageQuery) { |     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") |     @SaCheckPermission("system:notice:query") | ||||||
|     @GetMapping(value = "/{noticeId}") |     @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)); |         return R.ok(noticeService.selectNoticeById(noticeId)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 新增通知公告 |      * 新增通知公告 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("新增通知公告") |  | ||||||
|     @SaCheckPermission("system:notice:add") |     @SaCheckPermission("system:notice:add") | ||||||
|     @Log(title = "通知公告", businessType = BusinessType.INSERT) |     @Log(title = "通知公告", businessType = BusinessType.INSERT) | ||||||
|     @PostMapping |     @PostMapping | ||||||
| @@ -64,7 +59,6 @@ public class SysNoticeController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 修改通知公告 |      * 修改通知公告 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("修改通知公告") |  | ||||||
|     @SaCheckPermission("system:notice:edit") |     @SaCheckPermission("system:notice:edit") | ||||||
|     @Log(title = "通知公告", businessType = BusinessType.UPDATE) |     @Log(title = "通知公告", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping |     @PutMapping | ||||||
| @@ -74,12 +68,13 @@ public class SysNoticeController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 删除通知公告 |      * 删除通知公告 | ||||||
|  |      * | ||||||
|  |      * @param noticeIds 公告ID串 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("删除通知公告") |  | ||||||
|     @SaCheckPermission("system:notice:remove") |     @SaCheckPermission("system:notice:remove") | ||||||
|     @Log(title = "通知公告", businessType = BusinessType.DELETE) |     @Log(title = "通知公告", businessType = BusinessType.DELETE) | ||||||
|     @DeleteMapping("/{noticeIds}") |     @DeleteMapping("/{noticeIds}") | ||||||
|     public R<Void> remove(@ApiParam("公告ID串") @PathVariable Long[] noticeIds) { |     public R<Void> remove(@PathVariable Long[] noticeIds) { | ||||||
|         return toAjax(noticeService.deleteNoticeByIds(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.Log; | ||||||
| import com.ruoyi.common.annotation.RepeatSubmit; | import com.ruoyi.common.annotation.RepeatSubmit; | ||||||
| import com.ruoyi.common.core.controller.BaseController; | 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.PageQuery; | ||||||
|  | import com.ruoyi.common.core.domain.R; | ||||||
| import com.ruoyi.common.core.page.TableDataInfo; | import com.ruoyi.common.core.page.TableDataInfo; | ||||||
| import com.ruoyi.common.core.validate.AddGroup; | import com.ruoyi.common.core.validate.AddGroup; | ||||||
| import com.ruoyi.common.core.validate.EditGroup; | 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.bo.SysOssConfigBo; | ||||||
| import com.ruoyi.system.domain.vo.SysOssConfigVo; | import com.ruoyi.system.domain.vo.SysOssConfigVo; | ||||||
| import com.ruoyi.system.service.ISysOssConfigService; | 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 lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.validation.annotation.Validated; | import org.springframework.validation.annotation.Validated; | ||||||
| import org.springframework.web.bind.annotation.*; | import org.springframework.web.bind.annotation.*; | ||||||
| @@ -26,14 +23,13 @@ import javax.validation.constraints.NotNull; | |||||||
| import java.util.Arrays; | import java.util.Arrays; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 对象存储配置Controller |  * 对象存储配置 | ||||||
|  * |  * | ||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  * @author 孤舟烟雨 |  * @author 孤舟烟雨 | ||||||
|  * @date 2021-08-13 |  * @date 2021-08-13 | ||||||
|  */ |  */ | ||||||
| @Validated | @Validated | ||||||
| @Api(value = "对象存储配置控制器", tags = {"对象存储配置管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| @RequestMapping("/system/oss/config") | @RequestMapping("/system/oss/config") | ||||||
| @@ -44,7 +40,6 @@ public class SysOssConfigController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 查询对象存储配置列表 |      * 查询对象存储配置列表 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("查询对象存储配置列表") |  | ||||||
|     @SaCheckPermission("system:oss:list") |     @SaCheckPermission("system:oss:list") | ||||||
|     @GetMapping("/list") |     @GetMapping("/list") | ||||||
|     public TableDataInfo<SysOssConfigVo> list(@Validated(QueryGroup.class) SysOssConfigBo bo, PageQuery pageQuery) { |     public TableDataInfo<SysOssConfigVo> list(@Validated(QueryGroup.class) SysOssConfigBo bo, PageQuery pageQuery) { | ||||||
| @@ -53,57 +48,54 @@ public class SysOssConfigController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 获取对象存储配置详细信息 |      * 获取对象存储配置详细信息 | ||||||
|  |      * | ||||||
|  |      * @param ossConfigId OSS配置ID | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("获取对象存储配置详细信息") |  | ||||||
|     @SaCheckPermission("system:oss:query") |     @SaCheckPermission("system:oss:query") | ||||||
|     @GetMapping("/{ossConfigId}") |     @GetMapping("/{ossConfigId}") | ||||||
|     public R<SysOssConfigVo> getInfo(@ApiParam("OSS配置ID") |     public R<SysOssConfigVo> getInfo(@NotNull(message = "主键不能为空") | ||||||
|                                               @NotNull(message = "主键不能为空") |                                      @PathVariable Long ossConfigId) { | ||||||
|                                               @PathVariable("ossConfigId") Integer ossConfigId) { |  | ||||||
|         return R.ok(iSysOssConfigService.queryById(ossConfigId)); |         return R.ok(iSysOssConfigService.queryById(ossConfigId)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 新增对象存储配置 |      * 新增对象存储配置 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("新增对象存储配置") |  | ||||||
|     @SaCheckPermission("system:oss:add") |     @SaCheckPermission("system:oss:add") | ||||||
|     @Log(title = "对象存储配置", businessType = BusinessType.INSERT) |     @Log(title = "对象存储配置", businessType = BusinessType.INSERT) | ||||||
|     @RepeatSubmit() |     @RepeatSubmit() | ||||||
|     @PostMapping() |     @PostMapping() | ||||||
|     public R<Void> add(@Validated(AddGroup.class) @RequestBody SysOssConfigBo bo) { |     public R<Void> add(@Validated(AddGroup.class) @RequestBody SysOssConfigBo bo) { | ||||||
|         return toAjax(iSysOssConfigService.insertByBo(bo) ? 1 : 0); |         return toAjax(iSysOssConfigService.insertByBo(bo)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 修改对象存储配置 |      * 修改对象存储配置 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("修改对象存储配置") |  | ||||||
|     @SaCheckPermission("system:oss:edit") |     @SaCheckPermission("system:oss:edit") | ||||||
|     @Log(title = "对象存储配置", businessType = BusinessType.UPDATE) |     @Log(title = "对象存储配置", businessType = BusinessType.UPDATE) | ||||||
|     @RepeatSubmit() |     @RepeatSubmit() | ||||||
|     @PutMapping() |     @PutMapping() | ||||||
|     public R<Void> edit(@Validated(EditGroup.class) @RequestBody SysOssConfigBo bo) { |     public R<Void> edit(@Validated(EditGroup.class) @RequestBody SysOssConfigBo bo) { | ||||||
|         return toAjax(iSysOssConfigService.updateByBo(bo) ? 1 : 0); |         return toAjax(iSysOssConfigService.updateByBo(bo)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 删除对象存储配置 |      * 删除对象存储配置 | ||||||
|  |      * | ||||||
|  |      * @param ossConfigIds OSS配置ID串 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("删除对象存储配置") |  | ||||||
|     @SaCheckPermission("system:oss:remove") |     @SaCheckPermission("system:oss:remove") | ||||||
|     @Log(title = "对象存储配置", businessType = BusinessType.DELETE) |     @Log(title = "对象存储配置", businessType = BusinessType.DELETE) | ||||||
|     @DeleteMapping("/{ossConfigIds}") |     @DeleteMapping("/{ossConfigIds}") | ||||||
|     public R<Void> remove(@ApiParam("OSS配置ID串") |     public R<Void> remove(@NotEmpty(message = "主键不能为空") | ||||||
|                                    @NotEmpty(message = "主键不能为空") |                           @PathVariable Long[] ossConfigIds) { | ||||||
|                                    @PathVariable Long[] ossConfigIds) { |         return toAjax(iSysOssConfigService.deleteWithValidByIds(Arrays.asList(ossConfigIds), true)); | ||||||
|         return toAjax(iSysOssConfigService.deleteWithValidByIds(Arrays.asList(ossConfigIds), true) ? 1 : 0); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 状态修改 |      * 状态修改 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("状态修改") |  | ||||||
|     @SaCheckPermission("system:oss:edit") |     @SaCheckPermission("system:oss:edit") | ||||||
|     @Log(title = "对象存储状态修改", businessType = BusinessType.UPDATE) |     @Log(title = "对象存储状态修改", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping("/changeStatus") |     @PutMapping("/changeStatus") | ||||||
|   | |||||||
| @@ -15,12 +15,12 @@ import com.ruoyi.common.core.validate.QueryGroup; | |||||||
| import com.ruoyi.common.enums.BusinessType; | import com.ruoyi.common.enums.BusinessType; | ||||||
| import com.ruoyi.common.exception.ServiceException; | import com.ruoyi.common.exception.ServiceException; | ||||||
| import com.ruoyi.common.utils.file.FileUtils; | import com.ruoyi.common.utils.file.FileUtils; | ||||||
|  | import com.ruoyi.oss.core.OssClient; | ||||||
|  | import com.ruoyi.oss.factory.OssFactory; | ||||||
| import com.ruoyi.system.domain.SysOss; | import com.ruoyi.system.domain.SysOss; | ||||||
| import com.ruoyi.system.domain.bo.SysOssBo; | import com.ruoyi.system.domain.bo.SysOssBo; | ||||||
| import com.ruoyi.system.domain.vo.SysOssVo; | import com.ruoyi.system.domain.vo.SysOssVo; | ||||||
| import com.ruoyi.system.service.ISysConfigService; |  | ||||||
| import com.ruoyi.system.service.ISysOssService; | import com.ruoyi.system.service.ISysOssService; | ||||||
| import io.swagger.annotations.*; |  | ||||||
| import lombok.RequiredArgsConstructor; | import lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.http.MediaType; | import org.springframework.http.MediaType; | ||||||
| import org.springframework.validation.annotation.Validated; | import org.springframework.validation.annotation.Validated; | ||||||
| @@ -29,10 +29,10 @@ import org.springframework.web.multipart.MultipartFile; | |||||||
|  |  | ||||||
| import javax.servlet.http.HttpServletResponse; | import javax.servlet.http.HttpServletResponse; | ||||||
| import javax.validation.constraints.NotEmpty; | import javax.validation.constraints.NotEmpty; | ||||||
| import java.io.File; |  | ||||||
| import java.io.IOException; | import java.io.IOException; | ||||||
| import java.util.Arrays; | import java.util.Arrays; | ||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
|  | import java.util.List; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -41,19 +41,16 @@ import java.util.Map; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Validated | @Validated | ||||||
| @Api(value = "对象存储控制器", tags = {"对象存储管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| @RequestMapping("/system/oss") | @RequestMapping("/system/oss") | ||||||
| public class SysOssController extends BaseController { | public class SysOssController extends BaseController { | ||||||
|  |  | ||||||
|     private final ISysOssService iSysOssService; |     private final ISysOssService iSysOssService; | ||||||
|     private final ISysConfigService iSysConfigService; |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 查询OSS对象存储列表 |      * 查询OSS对象存储列表 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("查询OSS对象存储列表") |  | ||||||
|     @SaCheckPermission("system:oss:list") |     @SaCheckPermission("system:oss:list") | ||||||
|     @GetMapping("/list") |     @GetMapping("/list") | ||||||
|     public TableDataInfo<SysOssVo> list(@Validated(QueryGroup.class) SysOssBo bo, PageQuery pageQuery) { |     public TableDataInfo<SysOssVo> list(@Validated(QueryGroup.class) SysOssBo bo, PageQuery pageQuery) { | ||||||
| @@ -61,61 +58,60 @@ 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") |     @SaCheckPermission("system:oss:upload") | ||||||
|     @Log(title = "OSS对象存储", businessType = BusinessType.INSERT) |     @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) { |     public R<Map<String, String>> upload(@RequestPart("file") MultipartFile file) { | ||||||
|         if (ObjectUtil.isNull(file)) { |         if (ObjectUtil.isNull(file)) { | ||||||
|             throw new ServiceException("上传文件不能为空"); |             throw new ServiceException("上传文件不能为空"); | ||||||
|         } |         } | ||||||
|         SysOss oss = iSysOssService.upload(file); |         SysOssVo oss = iSysOssService.upload(file); | ||||||
|         Map<String, String> map = new HashMap<>(2); |         Map<String, String> map = new HashMap<>(2); | ||||||
|         map.put("url", oss.getUrl()); |         map.put("url", oss.getUrl()); | ||||||
|         map.put("fileName", oss.getFileName()); |         map.put("fileName", oss.getOriginalName()); | ||||||
|  |         map.put("ossId", oss.getOssId().toString()); | ||||||
|         return R.ok(map); |         return R.ok(map); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiOperation("下载OSS对象存储") |     /** | ||||||
|  |      * 下载OSS对象 | ||||||
|  |      * | ||||||
|  |      * @param ossId OSS对象ID | ||||||
|  |      */ | ||||||
|     @SaCheckPermission("system:oss:download") |     @SaCheckPermission("system:oss:download") | ||||||
|     @GetMapping("/download/{ossId}") |     @GetMapping("/download/{ossId}") | ||||||
|     public void download(@ApiParam("OSS对象ID") @PathVariable Long ossId, HttpServletResponse response) throws IOException { |     public void download(@PathVariable Long ossId, HttpServletResponse response) throws IOException { | ||||||
|         SysOss sysOss = iSysOssService.getById(ossId); |         iSysOssService.download(ossId,response); | ||||||
|         if (ObjectUtil.isNull(sysOss)) { |  | ||||||
|             throw new ServiceException("文件数据不存在!"); |  | ||||||
|         } |  | ||||||
|         response.reset(); |  | ||||||
|         FileUtils.setAttachmentResponseHeader(response, sysOss.getOriginalName()); |  | ||||||
|         response.setContentType(MediaType.APPLICATION_OCTET_STREAM_VALUE + "; charset=UTF-8"); |  | ||||||
|         long data; |  | ||||||
|         try { |  | ||||||
|             data = HttpUtil.download(sysOss.getUrl(), response.getOutputStream(), false); |  | ||||||
|         } catch (HttpException e) { |  | ||||||
|             if (e.getMessage().contains("403")) { |  | ||||||
|                 throw new ServiceException("无读取权限, 请在对应的OSS开启'公有读'权限!"); |  | ||||||
|             } else { |  | ||||||
|                 throw new ServiceException(e.getMessage()); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         response.setContentLength(Convert.toInt(data)); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 删除OSS对象存储 |      * 删除OSS对象存储 | ||||||
|  |      * | ||||||
|  |      * @param ossIds OSS对象ID串 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("删除OSS对象存储") |  | ||||||
|     @SaCheckPermission("system:oss:remove") |     @SaCheckPermission("system:oss:remove") | ||||||
|     @Log(title = "OSS对象存储", businessType = BusinessType.DELETE) |     @Log(title = "OSS对象存储", businessType = BusinessType.DELETE) | ||||||
|     @DeleteMapping("/{ossIds}") |     @DeleteMapping("/{ossIds}") | ||||||
|     public R<Void> remove(@ApiParam("OSS对象ID串") |     public R<Void> remove(@NotEmpty(message = "主键不能为空") | ||||||
|                                    @NotEmpty(message = "主键不能为空") |                           @PathVariable Long[] ossIds) { | ||||||
|                                    @PathVariable Long[] ossIds) { |         return toAjax(iSysOssService.deleteWithValidByIds(Arrays.asList(ossIds), true)); | ||||||
|         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.annotation.Log; | ||||||
| import com.ruoyi.common.constant.UserConstants; | import com.ruoyi.common.constant.UserConstants; | ||||||
| import com.ruoyi.common.core.controller.BaseController; | 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.PageQuery; | ||||||
|  | import com.ruoyi.common.core.domain.R; | ||||||
| import com.ruoyi.common.core.page.TableDataInfo; | import com.ruoyi.common.core.page.TableDataInfo; | ||||||
| import com.ruoyi.common.enums.BusinessType; | import com.ruoyi.common.enums.BusinessType; | ||||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | import com.ruoyi.common.utils.poi.ExcelUtil; | ||||||
| import com.ruoyi.system.domain.SysPost; | import com.ruoyi.system.domain.SysPost; | ||||||
| import com.ruoyi.system.service.ISysPostService; | 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 lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.validation.annotation.Validated; | import org.springframework.validation.annotation.Validated; | ||||||
| import org.springframework.web.bind.annotation.*; | import org.springframework.web.bind.annotation.*; | ||||||
| @@ -27,7 +24,6 @@ import java.util.List; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Validated | @Validated | ||||||
| @Api(value = "岗位信息控制器", tags = {"岗位信息管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| @RequestMapping("/system/post") | @RequestMapping("/system/post") | ||||||
| @@ -38,14 +34,15 @@ public class SysPostController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 获取岗位列表 |      * 获取岗位列表 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("获取岗位列表") |  | ||||||
|     @SaCheckPermission("system:post:list") |     @SaCheckPermission("system:post:list") | ||||||
|     @GetMapping("/list") |     @GetMapping("/list") | ||||||
|     public TableDataInfo<SysPost> list(SysPost post, PageQuery pageQuery) { |     public TableDataInfo<SysPost> list(SysPost post, PageQuery pageQuery) { | ||||||
|         return postService.selectPagePostList(post, pageQuery); |         return postService.selectPagePostList(post, pageQuery); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiOperation("导出岗位列表") |     /** | ||||||
|  |      * 导出岗位列表 | ||||||
|  |      */ | ||||||
|     @Log(title = "岗位管理", businessType = BusinessType.EXPORT) |     @Log(title = "岗位管理", businessType = BusinessType.EXPORT) | ||||||
|     @SaCheckPermission("system:post:export") |     @SaCheckPermission("system:post:export") | ||||||
|     @PostMapping("/export") |     @PostMapping("/export") | ||||||
| @@ -56,18 +53,18 @@ public class SysPostController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 根据岗位编号获取详细信息 |      * 根据岗位编号获取详细信息 | ||||||
|  |      * | ||||||
|  |      * @param postId 岗位ID | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("根据岗位编号获取详细信息") |  | ||||||
|     @SaCheckPermission("system:post:query") |     @SaCheckPermission("system:post:query") | ||||||
|     @GetMapping(value = "/{postId}") |     @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)); |         return R.ok(postService.selectPostById(postId)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 新增岗位 |      * 新增岗位 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("新增岗位") |  | ||||||
|     @SaCheckPermission("system:post:add") |     @SaCheckPermission("system:post:add") | ||||||
|     @Log(title = "岗位管理", businessType = BusinessType.INSERT) |     @Log(title = "岗位管理", businessType = BusinessType.INSERT) | ||||||
|     @PostMapping |     @PostMapping | ||||||
| @@ -83,7 +80,6 @@ public class SysPostController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 修改岗位 |      * 修改岗位 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("修改岗位") |  | ||||||
|     @SaCheckPermission("system:post:edit") |     @SaCheckPermission("system:post:edit") | ||||||
|     @Log(title = "岗位管理", businessType = BusinessType.UPDATE) |     @Log(title = "岗位管理", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping |     @PutMapping | ||||||
| @@ -98,19 +94,19 @@ public class SysPostController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 删除岗位 |      * 删除岗位 | ||||||
|  |      * | ||||||
|  |      * @param postIds 岗位ID串 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("删除岗位") |  | ||||||
|     @SaCheckPermission("system:post:remove") |     @SaCheckPermission("system:post:remove") | ||||||
|     @Log(title = "岗位管理", businessType = BusinessType.DELETE) |     @Log(title = "岗位管理", businessType = BusinessType.DELETE) | ||||||
|     @DeleteMapping("/{postIds}") |     @DeleteMapping("/{postIds}") | ||||||
|     public R<Void> remove(@ApiParam("岗位ID串") @PathVariable Long[] postIds) { |     public R<Void> remove(@PathVariable Long[] postIds) { | ||||||
|         return toAjax(postService.deletePostByIds(postIds)); |         return toAjax(postService.deletePostByIds(postIds)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 获取岗位选择框列表 |      * 获取岗位选择框列表 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("获取岗位选择框列表") |  | ||||||
|     @GetMapping("/optionselect") |     @GetMapping("/optionselect") | ||||||
|     public R<List<SysPost>> optionselect() { |     public R<List<SysPost>> optionselect() { | ||||||
|         List<SysPost> posts = postService.selectPostAll(); |         List<SysPost> posts = postService.selectPostAll(); | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| package com.ruoyi.web.controller.system; | package com.ruoyi.web.controller.system; | ||||||
|  |  | ||||||
| import cn.dev33.satoken.secure.BCrypt; | import cn.dev33.satoken.secure.BCrypt; | ||||||
|  | import cn.hutool.core.io.FileUtil; | ||||||
| import com.ruoyi.common.annotation.Log; | import com.ruoyi.common.annotation.Log; | ||||||
| import com.ruoyi.common.constant.UserConstants; | import com.ruoyi.common.constant.UserConstants; | ||||||
| import com.ruoyi.common.core.controller.BaseController; | import com.ruoyi.common.core.controller.BaseController; | ||||||
| @@ -9,19 +10,18 @@ import com.ruoyi.common.core.domain.entity.SysUser; | |||||||
| import com.ruoyi.common.enums.BusinessType; | import com.ruoyi.common.enums.BusinessType; | ||||||
| import com.ruoyi.common.helper.LoginHelper; | import com.ruoyi.common.helper.LoginHelper; | ||||||
| import com.ruoyi.common.utils.StringUtils; | import com.ruoyi.common.utils.StringUtils; | ||||||
|  | import com.ruoyi.common.utils.file.MimeTypeUtils; | ||||||
| import com.ruoyi.system.domain.SysOss; | import com.ruoyi.system.domain.SysOss; | ||||||
|  | import com.ruoyi.system.domain.vo.SysOssVo; | ||||||
| import com.ruoyi.system.service.ISysOssService; | import com.ruoyi.system.service.ISysOssService; | ||||||
| import com.ruoyi.system.service.ISysUserService; | 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 lombok.RequiredArgsConstructor; | ||||||
|  | import org.springframework.http.MediaType; | ||||||
| import org.springframework.validation.annotation.Validated; | import org.springframework.validation.annotation.Validated; | ||||||
| import org.springframework.web.bind.annotation.*; | import org.springframework.web.bind.annotation.*; | ||||||
| import org.springframework.web.multipart.MultipartFile; | import org.springframework.web.multipart.MultipartFile; | ||||||
|  |  | ||||||
| import java.io.File; | import java.util.Arrays; | ||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
|  |  | ||||||
| @@ -31,7 +31,6 @@ import java.util.Map; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Validated | @Validated | ||||||
| @Api(value = "个人信息控制器", tags = {"个人信息管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| @RequestMapping("/system/user/profile") | @RequestMapping("/system/user/profile") | ||||||
| @@ -43,7 +42,6 @@ public class SysProfileController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 个人信息 |      * 个人信息 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("个人信息") |  | ||||||
|     @GetMapping |     @GetMapping | ||||||
|     public R<Map<String, Object>> profile() { |     public R<Map<String, Object>> profile() { | ||||||
|         SysUser user = userService.selectUserById(getUserId()); |         SysUser user = userService.selectUserById(getUserId()); | ||||||
| @@ -57,7 +55,6 @@ public class SysProfileController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 修改用户 |      * 修改用户 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("修改用户") |  | ||||||
|     @Log(title = "个人信息", businessType = BusinessType.UPDATE) |     @Log(title = "个人信息", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping |     @PutMapping | ||||||
|     public R<Void> updateProfile(@RequestBody SysUser user) { |     public R<Void> updateProfile(@RequestBody SysUser user) { | ||||||
| @@ -72,6 +69,8 @@ public class SysProfileController extends BaseController { | |||||||
|         user.setUserId(getUserId()); |         user.setUserId(getUserId()); | ||||||
|         user.setUserName(null); |         user.setUserName(null); | ||||||
|         user.setPassword(null); |         user.setPassword(null); | ||||||
|  |         user.setAvatar(null); | ||||||
|  |         user.setDeptId(null); | ||||||
|         if (userService.updateUserProfile(user) > 0) { |         if (userService.updateUserProfile(user) > 0) { | ||||||
|             return R.ok(); |             return R.ok(); | ||||||
|         } |         } | ||||||
| @@ -80,12 +79,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) |     @Log(title = "个人信息", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping("/updatePwd") |     @PutMapping("/updatePwd") | ||||||
|     public R<Void> updatePwd(String oldPassword, String newPassword) { |     public R<Void> updatePwd(String oldPassword, String newPassword) { | ||||||
| @@ -107,17 +104,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) |     @Log(title = "用户头像", businessType = BusinessType.UPDATE) | ||||||
|     @PostMapping("/avatar") |     @PostMapping(value = "/avatar", consumes = MediaType.MULTIPART_FORM_DATA_VALUE) | ||||||
|     public R<Map<String, Object>> avatar(@RequestPart("avatarfile") MultipartFile file) { |     public R<Map<String, Object>> avatar(@RequestPart("avatarfile") MultipartFile avatarfile) { | ||||||
|         Map<String, Object> ajax = new HashMap<>(); |         Map<String, Object> ajax = new HashMap<>(); | ||||||
|         if (!file.isEmpty()) { |         if (!avatarfile.isEmpty()) { | ||||||
|             SysOss oss = iSysOssService.upload(file); |             String extension = FileUtil.extName(avatarfile.getOriginalFilename()); | ||||||
|  |             if (!StringUtils.equalsAnyIgnoreCase(extension, MimeTypeUtils.IMAGE_EXTENSION)) { | ||||||
|  |                 return R.fail("文件格式不正确,请上传" + Arrays.toString(MimeTypeUtils.IMAGE_EXTENSION) + "格式"); | ||||||
|  |             } | ||||||
|  |             SysOssVo oss = iSysOssService.upload(avatarfile); | ||||||
|             String avatar = oss.getUrl(); |             String avatar = oss.getUrl(); | ||||||
|             if (userService.updateUserAvatar(getUsername(), avatar)) { |             if (userService.updateUserAvatar(getUsername(), avatar)) { | ||||||
|                 ajax.put("imgUrl", avatar); |                 ajax.put("imgUrl", avatar); | ||||||
|   | |||||||
| @@ -1,12 +1,11 @@ | |||||||
| package com.ruoyi.web.controller.system; | package com.ruoyi.web.controller.system; | ||||||
|  |  | ||||||
|  | import cn.dev33.satoken.annotation.SaIgnore; | ||||||
| import com.ruoyi.common.core.controller.BaseController; | import com.ruoyi.common.core.controller.BaseController; | ||||||
| import com.ruoyi.common.core.domain.R; | import com.ruoyi.common.core.domain.R; | ||||||
| import com.ruoyi.common.core.domain.model.RegisterBody; | import com.ruoyi.common.core.domain.model.RegisterBody; | ||||||
| import com.ruoyi.system.service.ISysConfigService; | import com.ruoyi.system.service.ISysConfigService; | ||||||
| import com.ruoyi.system.service.SysRegisterService; | import com.ruoyi.system.service.SysRegisterService; | ||||||
| import io.swagger.annotations.Api; |  | ||||||
| import io.swagger.annotations.ApiOperation; |  | ||||||
| import lombok.RequiredArgsConstructor; | import lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.validation.annotation.Validated; | import org.springframework.validation.annotation.Validated; | ||||||
| import org.springframework.web.bind.annotation.PostMapping; | import org.springframework.web.bind.annotation.PostMapping; | ||||||
| @@ -19,7 +18,6 @@ import org.springframework.web.bind.annotation.RestController; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Validated | @Validated | ||||||
| @Api(value = "注册验证控制器", tags = {"注册验证管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| public class SysRegisterController extends BaseController { | public class SysRegisterController extends BaseController { | ||||||
| @@ -27,7 +25,10 @@ public class SysRegisterController extends BaseController { | |||||||
|     private final SysRegisterService registerService; |     private final SysRegisterService registerService; | ||||||
|     private final ISysConfigService configService; |     private final ISysConfigService configService; | ||||||
|  |  | ||||||
|     @ApiOperation("用户注册") |     /** | ||||||
|  |      * 用户注册 | ||||||
|  |      */ | ||||||
|  |     @SaIgnore | ||||||
|     @PostMapping("/register") |     @PostMapping("/register") | ||||||
|     public R<Void> register(@Validated @RequestBody RegisterBody user) { |     public R<Void> register(@Validated @RequestBody RegisterBody user) { | ||||||
|         if (!("true".equals(configService.selectConfigByKey("sys.account.registerUser")))) { |         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.annotation.Log; | ||||||
| import com.ruoyi.common.constant.UserConstants; | import com.ruoyi.common.constant.UserConstants; | ||||||
| import com.ruoyi.common.core.controller.BaseController; | 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.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.SysRole; | ||||||
| import com.ruoyi.common.core.domain.entity.SysUser; | import com.ruoyi.common.core.domain.entity.SysUser; | ||||||
| import com.ruoyi.common.core.domain.model.LoginUser; | 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.helper.LoginHelper; | ||||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | import com.ruoyi.common.utils.poi.ExcelUtil; | ||||||
| import com.ruoyi.system.domain.SysUserRole; | import com.ruoyi.system.domain.SysUserRole; | ||||||
|  | import com.ruoyi.system.service.ISysDeptService; | ||||||
| import com.ruoyi.system.service.ISysRoleService; | import com.ruoyi.system.service.ISysRoleService; | ||||||
| import com.ruoyi.system.service.ISysUserService; | import com.ruoyi.system.service.ISysUserService; | ||||||
| import com.ruoyi.system.service.SysPermissionService; | import com.ruoyi.system.service.SysPermissionService; | ||||||
| import io.swagger.annotations.*; |  | ||||||
| import lombok.RequiredArgsConstructor; | import lombok.RequiredArgsConstructor; | ||||||
| import org.springframework.validation.annotation.Validated; | import org.springframework.validation.annotation.Validated; | ||||||
| import org.springframework.web.bind.annotation.*; | import org.springframework.web.bind.annotation.*; | ||||||
|  |  | ||||||
| import javax.servlet.http.HttpServletResponse; | import javax.servlet.http.HttpServletResponse; | ||||||
|  | import java.util.HashMap; | ||||||
| import java.util.List; | import java.util.List; | ||||||
|  | import java.util.Map; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 角色信息 |  * 角色信息 | ||||||
| @@ -32,7 +35,6 @@ import java.util.List; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Validated | @Validated | ||||||
| @Api(value = "角色信息控制器", tags = {"角色信息管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| @RequestMapping("/system/role") | @RequestMapping("/system/role") | ||||||
| @@ -40,16 +42,21 @@ public class SysRoleController extends BaseController { | |||||||
|  |  | ||||||
|     private final ISysRoleService roleService; |     private final ISysRoleService roleService; | ||||||
|     private final ISysUserService userService; |     private final ISysUserService userService; | ||||||
|  |     private final ISysDeptService deptService; | ||||||
|     private final SysPermissionService permissionService; |     private final SysPermissionService permissionService; | ||||||
|  |  | ||||||
|     @ApiOperation("查询角色信息列表") |     /** | ||||||
|  |      * 获取角色信息列表 | ||||||
|  |      */ | ||||||
|     @SaCheckPermission("system:role:list") |     @SaCheckPermission("system:role:list") | ||||||
|     @GetMapping("/list") |     @GetMapping("/list") | ||||||
|     public TableDataInfo<SysRole> list(SysRole role, PageQuery pageQuery) { |     public TableDataInfo<SysRole> list(SysRole role, PageQuery pageQuery) { | ||||||
|         return roleService.selectPageRoleList(role, pageQuery); |         return roleService.selectPageRoleList(role, pageQuery); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiOperation("导出角色信息列表") |     /** | ||||||
|  |      * 导出角色信息列表 | ||||||
|  |      */ | ||||||
|     @Log(title = "角色管理", businessType = BusinessType.EXPORT) |     @Log(title = "角色管理", businessType = BusinessType.EXPORT) | ||||||
|     @SaCheckPermission("system:role:export") |     @SaCheckPermission("system:role:export") | ||||||
|     @PostMapping("/export") |     @PostMapping("/export") | ||||||
| @@ -60,11 +67,12 @@ public class SysRoleController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 根据角色编号获取详细信息 |      * 根据角色编号获取详细信息 | ||||||
|  |      * | ||||||
|  |      * @param roleId 角色ID | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("根据角色编号获取详细信息") |  | ||||||
|     @SaCheckPermission("system:role:query") |     @SaCheckPermission("system:role:query") | ||||||
|     @GetMapping(value = "/{roleId}") |     @GetMapping(value = "/{roleId}") | ||||||
|     public R<SysRole> getInfo(@ApiParam("角色ID") @PathVariable Long roleId) { |     public R<SysRole> getInfo(@PathVariable Long roleId) { | ||||||
|         roleService.checkRoleDataScope(roleId); |         roleService.checkRoleDataScope(roleId); | ||||||
|         return R.ok(roleService.selectRoleById(roleId)); |         return R.ok(roleService.selectRoleById(roleId)); | ||||||
|     } |     } | ||||||
| @@ -72,7 +80,6 @@ public class SysRoleController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 新增角色 |      * 新增角色 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("新增角色") |  | ||||||
|     @SaCheckPermission("system:role:add") |     @SaCheckPermission("system:role:add") | ||||||
|     @Log(title = "角色管理", businessType = BusinessType.INSERT) |     @Log(title = "角色管理", businessType = BusinessType.INSERT) | ||||||
|     @PostMapping |     @PostMapping | ||||||
| @@ -89,7 +96,6 @@ public class SysRoleController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 修改保存角色 |      * 修改保存角色 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("修改保存角色") |  | ||||||
|     @SaCheckPermission("system:role:edit") |     @SaCheckPermission("system:role:edit") | ||||||
|     @Log(title = "角色管理", businessType = BusinessType.UPDATE) |     @Log(title = "角色管理", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping |     @PutMapping | ||||||
| @@ -118,7 +124,6 @@ public class SysRoleController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 修改保存数据权限 |      * 修改保存数据权限 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("修改保存数据权限") |  | ||||||
|     @SaCheckPermission("system:role:edit") |     @SaCheckPermission("system:role:edit") | ||||||
|     @Log(title = "角色管理", businessType = BusinessType.UPDATE) |     @Log(title = "角色管理", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping("/dataScope") |     @PutMapping("/dataScope") | ||||||
| @@ -131,7 +136,6 @@ public class SysRoleController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 状态修改 |      * 状态修改 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("状态修改") |  | ||||||
|     @SaCheckPermission("system:role:edit") |     @SaCheckPermission("system:role:edit") | ||||||
|     @Log(title = "角色管理", businessType = BusinessType.UPDATE) |     @Log(title = "角色管理", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping("/changeStatus") |     @PutMapping("/changeStatus") | ||||||
| @@ -143,19 +147,19 @@ public class SysRoleController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 删除角色 |      * 删除角色 | ||||||
|  |      * | ||||||
|  |      * @param roleIds 角色ID串 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("删除角色") |  | ||||||
|     @SaCheckPermission("system:role:remove") |     @SaCheckPermission("system:role:remove") | ||||||
|     @Log(title = "角色管理", businessType = BusinessType.DELETE) |     @Log(title = "角色管理", businessType = BusinessType.DELETE) | ||||||
|     @DeleteMapping("/{roleIds}") |     @DeleteMapping("/{roleIds}") | ||||||
|     public R<Void> remove(@ApiParam("岗位ID串") @PathVariable Long[] roleIds) { |     public R<Void> remove(@PathVariable Long[] roleIds) { | ||||||
|         return toAjax(roleService.deleteRoleByIds(roleIds)); |         return toAjax(roleService.deleteRoleByIds(roleIds)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 获取角色选择框列表 |      * 获取角色选择框列表 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("获取角色选择框列表") |  | ||||||
|     @SaCheckPermission("system:role:query") |     @SaCheckPermission("system:role:query") | ||||||
|     @GetMapping("/optionselect") |     @GetMapping("/optionselect") | ||||||
|     public R<List<SysRole>> optionselect() { |     public R<List<SysRole>> optionselect() { | ||||||
| @@ -165,7 +169,6 @@ public class SysRoleController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 查询已分配用户角色列表 |      * 查询已分配用户角色列表 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("查询已分配用户角色列表") |  | ||||||
|     @SaCheckPermission("system:role:list") |     @SaCheckPermission("system:role:list") | ||||||
|     @GetMapping("/authUser/allocatedList") |     @GetMapping("/authUser/allocatedList") | ||||||
|     public TableDataInfo<SysUser> allocatedList(SysUser user, PageQuery pageQuery) { |     public TableDataInfo<SysUser> allocatedList(SysUser user, PageQuery pageQuery) { | ||||||
| @@ -175,7 +178,6 @@ public class SysRoleController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 查询未分配用户角色列表 |      * 查询未分配用户角色列表 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("查询未分配用户角色列表") |  | ||||||
|     @SaCheckPermission("system:role:list") |     @SaCheckPermission("system:role:list") | ||||||
|     @GetMapping("/authUser/unallocatedList") |     @GetMapping("/authUser/unallocatedList") | ||||||
|     public TableDataInfo<SysUser> unallocatedList(SysUser user, PageQuery pageQuery) { |     public TableDataInfo<SysUser> unallocatedList(SysUser user, PageQuery pageQuery) { | ||||||
| @@ -185,7 +187,6 @@ public class SysRoleController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 取消授权用户 |      * 取消授权用户 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("取消授权用户") |  | ||||||
|     @SaCheckPermission("system:role:edit") |     @SaCheckPermission("system:role:edit") | ||||||
|     @Log(title = "角色管理", businessType = BusinessType.GRANT) |     @Log(title = "角色管理", businessType = BusinessType.GRANT) | ||||||
|     @PutMapping("/authUser/cancel") |     @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") |     @SaCheckPermission("system:role:edit") | ||||||
|     @Log(title = "角色管理", businessType = BusinessType.GRANT) |     @Log(title = "角色管理", businessType = BusinessType.GRANT) | ||||||
|     @PutMapping("/authUser/cancelAll") |     @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") |     @SaCheckPermission("system:role:edit") | ||||||
|     @Log(title = "角色管理", businessType = BusinessType.GRANT) |     @Log(title = "角色管理", businessType = BusinessType.GRANT) | ||||||
|     @PutMapping("/authUser/selectAll") |     @PutMapping("/authUser/selectAll") | ||||||
| @@ -223,4 +220,18 @@ public class SysRoleController extends BaseController { | |||||||
|         roleService.checkRoleDataScope(roleId); |         roleService.checkRoleDataScope(roleId); | ||||||
|         return toAjax(roleService.insertAuthUsers(roleId, userIds)); |         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.annotation.SaCheckPermission; | ||||||
| import cn.dev33.satoken.secure.BCrypt; | import cn.dev33.satoken.secure.BCrypt; | ||||||
| import cn.hutool.core.bean.BeanUtil; | import cn.hutool.core.bean.BeanUtil; | ||||||
|  | import cn.hutool.core.lang.tree.Tree; | ||||||
| import cn.hutool.core.util.ArrayUtil; | import cn.hutool.core.util.ArrayUtil; | ||||||
| import cn.hutool.core.util.ObjectUtil; | import cn.hutool.core.util.ObjectUtil; | ||||||
| import com.ruoyi.common.annotation.Log; | 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.enums.BusinessType; | ||||||
| import com.ruoyi.common.excel.ExcelResult; | import com.ruoyi.common.excel.ExcelResult; | ||||||
| import com.ruoyi.common.helper.LoginHelper; | import com.ruoyi.common.helper.LoginHelper; | ||||||
|  | import com.ruoyi.common.utils.StreamUtils; | ||||||
| import com.ruoyi.common.utils.StringUtils; | import com.ruoyi.common.utils.StringUtils; | ||||||
| import com.ruoyi.common.utils.poi.ExcelUtil; | import com.ruoyi.common.utils.poi.ExcelUtil; | ||||||
| import com.ruoyi.system.domain.vo.SysUserExportVo; | import com.ruoyi.system.domain.vo.SysUserExportVo; | ||||||
| import com.ruoyi.system.domain.vo.SysUserImportVo; | import com.ruoyi.system.domain.vo.SysUserImportVo; | ||||||
| import com.ruoyi.system.listener.SysUserImportListener; | import com.ruoyi.system.listener.SysUserImportListener; | ||||||
|  | import com.ruoyi.system.service.ISysDeptService; | ||||||
| import com.ruoyi.system.service.ISysPostService; | import com.ruoyi.system.service.ISysPostService; | ||||||
| import com.ruoyi.system.service.ISysRoleService; | import com.ruoyi.system.service.ISysRoleService; | ||||||
| import com.ruoyi.system.service.ISysUserService; | import com.ruoyi.system.service.ISysUserService; | ||||||
| import io.swagger.annotations.*; |  | ||||||
| import lombok.RequiredArgsConstructor; | import lombok.RequiredArgsConstructor; | ||||||
|  | import org.springframework.http.MediaType; | ||||||
| import org.springframework.validation.annotation.Validated; | import org.springframework.validation.annotation.Validated; | ||||||
| import org.springframework.web.bind.annotation.*; | import org.springframework.web.bind.annotation.*; | ||||||
| import org.springframework.web.multipart.MultipartFile; | import org.springframework.web.multipart.MultipartFile; | ||||||
| @@ -36,7 +39,6 @@ import java.util.ArrayList; | |||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
| import java.util.stream.Collectors; |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 用户信息 |  * 用户信息 | ||||||
| @@ -44,7 +46,6 @@ import java.util.stream.Collectors; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Validated | @Validated | ||||||
| @Api(value = "用户信息控制器", tags = {"用户信息管理"}) |  | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| @RequestMapping("/system/user") | @RequestMapping("/system/user") | ||||||
| @@ -53,18 +54,20 @@ public class SysUserController extends BaseController { | |||||||
|     private final ISysUserService userService; |     private final ISysUserService userService; | ||||||
|     private final ISysRoleService roleService; |     private final ISysRoleService roleService; | ||||||
|     private final ISysPostService postService; |     private final ISysPostService postService; | ||||||
|  |     private final ISysDeptService deptService; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 获取用户列表 |      * 获取用户列表 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("获取用户列表") |  | ||||||
|     @SaCheckPermission("system:user:list") |     @SaCheckPermission("system:user:list") | ||||||
|     @GetMapping("/list") |     @GetMapping("/list") | ||||||
|     public TableDataInfo<SysUser> list(SysUser user, PageQuery pageQuery) { |     public TableDataInfo<SysUser> list(SysUser user, PageQuery pageQuery) { | ||||||
|         return userService.selectPageUserList(user, pageQuery); |         return userService.selectPageUserList(user, pageQuery); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiOperation("导出用户列表") |     /** | ||||||
|  |      * 导出用户列表 | ||||||
|  |      */ | ||||||
|     @Log(title = "用户管理", businessType = BusinessType.EXPORT) |     @Log(title = "用户管理", businessType = BusinessType.EXPORT) | ||||||
|     @SaCheckPermission("system:user:export") |     @SaCheckPermission("system:user:export") | ||||||
|     @PostMapping("/export") |     @PostMapping("/export") | ||||||
| @@ -82,19 +85,23 @@ public class SysUserController extends BaseController { | |||||||
|         ExcelUtil.exportExcel(listVo, "用户数据", SysUserExportVo.class, response); |         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) |     @Log(title = "用户管理", businessType = BusinessType.IMPORT) | ||||||
|     @SaCheckPermission("system:user: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 { |     public R<Void> importData(@RequestPart("file") MultipartFile file, boolean updateSupport) throws Exception { | ||||||
|         ExcelResult<SysUserImportVo> result = ExcelUtil.importExcel(file.getInputStream(), SysUserImportVo.class, new SysUserImportListener(updateSupport)); |         ExcelResult<SysUserImportVo> result = ExcelUtil.importExcel(file.getInputStream(), SysUserImportVo.class, new SysUserImportListener(updateSupport)); | ||||||
|         return R.ok(result.getAnalysis()); |         return R.ok(result.getAnalysis()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiOperation("下载导入模板") |     /** | ||||||
|  |      * 获取导入模板 | ||||||
|  |      */ | ||||||
|     @PostMapping("/importTemplate") |     @PostMapping("/importTemplate") | ||||||
|     public void importTemplate(HttpServletResponse response) { |     public void importTemplate(HttpServletResponse response) { | ||||||
|         ExcelUtil.exportExcel(new ArrayList<>(), "用户数据", SysUserImportVo.class, 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") |     @SaCheckPermission("system:user:query") | ||||||
|     @GetMapping(value = {"/", "/{userId}"}) |     @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); |         userService.checkUserDataScope(userId); | ||||||
|         Map<String, Object> ajax = new HashMap<>(); |         Map<String, Object> ajax = new HashMap<>(); | ||||||
|         List<SysRole> roles = roleService.selectRoleAll(); |         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()); |         ajax.put("posts", postService.selectPostAll()); | ||||||
|         if (ObjectUtil.isNotNull(userId)) { |         if (ObjectUtil.isNotNull(userId)) { | ||||||
|             SysUser sysUser = userService.selectUserById(userId); |             SysUser sysUser = userService.selectUserById(userId); | ||||||
|             ajax.put("user", sysUser); |             ajax.put("user", sysUser); | ||||||
|             ajax.put("postIds", postService.selectPostListByUserId(userId)); |             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); |         return R.ok(ajax); | ||||||
|     } |     } | ||||||
| @@ -124,12 +132,11 @@ public class SysUserController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 新增用户 |      * 新增用户 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("新增用户") |  | ||||||
|     @SaCheckPermission("system:user:add") |     @SaCheckPermission("system:user:add") | ||||||
|     @Log(title = "用户管理", businessType = BusinessType.INSERT) |     @Log(title = "用户管理", businessType = BusinessType.INSERT) | ||||||
|     @PostMapping |     @PostMapping | ||||||
|     public R<Void> add(@Validated @RequestBody SysUser user) { |     public R<Void> add(@Validated @RequestBody SysUser user) { | ||||||
|         if (UserConstants.NOT_UNIQUE.equals(userService.checkUserNameUnique(user.getUserName()))) { |         if (UserConstants.NOT_UNIQUE.equals(userService.checkUserNameUnique(user))) { | ||||||
|             return R.fail("新增用户'" + user.getUserName() + "'失败,登录账号已存在"); |             return R.fail("新增用户'" + user.getUserName() + "'失败,登录账号已存在"); | ||||||
|         } else if (StringUtils.isNotEmpty(user.getPhonenumber()) |         } else if (StringUtils.isNotEmpty(user.getPhonenumber()) | ||||||
|             && UserConstants.NOT_UNIQUE.equals(userService.checkPhoneUnique(user))) { |             && UserConstants.NOT_UNIQUE.equals(userService.checkPhoneUnique(user))) { | ||||||
| @@ -145,14 +152,15 @@ public class SysUserController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 修改用户 |      * 修改用户 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("修改用户") |  | ||||||
|     @SaCheckPermission("system:user:edit") |     @SaCheckPermission("system:user:edit") | ||||||
|     @Log(title = "用户管理", businessType = BusinessType.UPDATE) |     @Log(title = "用户管理", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping |     @PutMapping | ||||||
|     public R<Void> edit(@Validated @RequestBody SysUser user) { |     public R<Void> edit(@Validated @RequestBody SysUser user) { | ||||||
|         userService.checkUserAllowed(user); |         userService.checkUserAllowed(user); | ||||||
|         userService.checkUserDataScope(user.getUserId()); |         userService.checkUserDataScope(user.getUserId()); | ||||||
|         if (StringUtils.isNotEmpty(user.getPhonenumber()) |         if (UserConstants.NOT_UNIQUE.equals(userService.checkUserNameUnique(user))) { | ||||||
|  |             return R.fail("修改用户'" + user.getUserName() + "'失败,登录账号已存在"); | ||||||
|  |         } else if (StringUtils.isNotEmpty(user.getPhonenumber()) | ||||||
|             && UserConstants.NOT_UNIQUE.equals(userService.checkPhoneUnique(user))) { |             && UserConstants.NOT_UNIQUE.equals(userService.checkPhoneUnique(user))) { | ||||||
|             return R.fail("修改用户'" + user.getUserName() + "'失败,手机号码已存在"); |             return R.fail("修改用户'" + user.getUserName() + "'失败,手机号码已存在"); | ||||||
|         } else if (StringUtils.isNotEmpty(user.getEmail()) |         } else if (StringUtils.isNotEmpty(user.getEmail()) | ||||||
| @@ -164,12 +172,13 @@ public class SysUserController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 删除用户 |      * 删除用户 | ||||||
|  |      * | ||||||
|  |      * @param userIds 角色ID串 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("删除用户") |  | ||||||
|     @SaCheckPermission("system:user:remove") |     @SaCheckPermission("system:user:remove") | ||||||
|     @Log(title = "用户管理", businessType = BusinessType.DELETE) |     @Log(title = "用户管理", businessType = BusinessType.DELETE) | ||||||
|     @DeleteMapping("/{userIds}") |     @DeleteMapping("/{userIds}") | ||||||
|     public R<Void> remove(@ApiParam("角色ID串") @PathVariable Long[] userIds) { |     public R<Void> remove(@PathVariable Long[] userIds) { | ||||||
|         if (ArrayUtil.contains(userIds, getUserId())) { |         if (ArrayUtil.contains(userIds, getUserId())) { | ||||||
|             return R.fail("当前用户不能删除"); |             return R.fail("当前用户不能删除"); | ||||||
|         } |         } | ||||||
| @@ -179,7 +188,6 @@ public class SysUserController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 重置密码 |      * 重置密码 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("重置密码") |  | ||||||
|     @SaCheckPermission("system:user:resetPwd") |     @SaCheckPermission("system:user:resetPwd") | ||||||
|     @Log(title = "用户管理", businessType = BusinessType.UPDATE) |     @Log(title = "用户管理", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping("/resetPwd") |     @PutMapping("/resetPwd") | ||||||
| @@ -193,7 +201,6 @@ public class SysUserController extends BaseController { | |||||||
|     /** |     /** | ||||||
|      * 状态修改 |      * 状态修改 | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("状态修改") |  | ||||||
|     @SaCheckPermission("system:user:edit") |     @SaCheckPermission("system:user:edit") | ||||||
|     @Log(title = "用户管理", businessType = BusinessType.UPDATE) |     @Log(title = "用户管理", businessType = BusinessType.UPDATE) | ||||||
|     @PutMapping("/changeStatus") |     @PutMapping("/changeStatus") | ||||||
| @@ -205,27 +212,26 @@ public class SysUserController extends BaseController { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 根据用户编号获取授权角色 |      * 根据用户编号获取授权角色 | ||||||
|  |      * | ||||||
|  |      * @param userId 用户ID | ||||||
|      */ |      */ | ||||||
|     @ApiOperation("根据用户编号获取授权角色") |  | ||||||
|     @SaCheckPermission("system:user:query") |     @SaCheckPermission("system:user:query") | ||||||
|     @GetMapping("/authRole/{userId}") |     @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); |         SysUser user = userService.selectUserById(userId); | ||||||
|         List<SysRole> roles = roleService.selectRolesByUserId(userId); |         List<SysRole> roles = roleService.selectRolesByUserId(userId); | ||||||
|         Map<String, Object> ajax = new HashMap<>(); |         Map<String, Object> ajax = new HashMap<>(); | ||||||
|         ajax.put("user", user); |         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); |         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") |     @SaCheckPermission("system:user:edit") | ||||||
|     @Log(title = "用户管理", businessType = BusinessType.GRANT) |     @Log(title = "用户管理", businessType = BusinessType.GRANT) | ||||||
|     @PutMapping("/authRole") |     @PutMapping("/authRole") | ||||||
| @@ -234,4 +240,14 @@ public class SysUserController extends BaseController { | |||||||
|         userService.insertUserAuth(userId, roleIds); |         userService.insertUserAuth(userId, roleIds); | ||||||
|         return R.ok(); |         return R.ok(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 获取部门树列表 | ||||||
|  |      */ | ||||||
|  |     @SaCheckPermission("system:user:list") | ||||||
|  |     @GetMapping("/deptTree") | ||||||
|  |     public R<List<Tree<Long>>> deptTree(SysDept dept) { | ||||||
|  |         return R.ok(deptService.selectDeptTreeList(dept)); | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,55 +1,51 @@ | |||||||
| --- # 监控配置 | --- # 监控中心配置 | ||||||
| spring: | spring.boot.admin.client: | ||||||
|   boot: |   # 增加客户端开关 | ||||||
|     admin: |   enabled: true | ||||||
|       # Spring Boot Admin Client 客户端的相关配置 |   url: http://localhost:9090/admin | ||||||
|       client: |   instance: | ||||||
|         # 增加客户端开关 |     service-host-type: IP | ||||||
|         enabled: true |   username: ruoyi | ||||||
|         # 设置 Spring Boot Admin Server 地址 |   password: 123456 | ||||||
|         url: http://localhost:9090/admin |  | ||||||
|         instance: |  | ||||||
|           service-host-type: IP |  | ||||||
|         username: ruoyi |  | ||||||
|         password: 123456 |  | ||||||
|  |  | ||||||
| --- # xxl-job 配置 | --- # xxl-job 配置 | ||||||
| xxl: | xxl.job: | ||||||
|   job: |   # 执行器开关 | ||||||
|     # 执行器开关 |   enabled: true | ||||||
|     enabled: true |   # 调度中心地址:如调度中心集群部署存在多个地址则用逗号分隔。 | ||||||
|     # 调度中心地址:如调度中心集群部署存在多个地址则用逗号分隔。 |   admin-addresses: http://localhost:9100/xxl-job-admin | ||||||
|     admin-addresses: http://localhost:9100/xxl-job-admin |   # 执行器通讯TOKEN:非空时启用 | ||||||
|     # 执行器通讯TOKEN:非空时启用 |   access-token: xxl-job | ||||||
|     access-token: xxl-job |   executor: | ||||||
|     # 执行器配置 |     # 执行器AppName:执行器心跳注册分组依据;为空则关闭自动注册 | ||||||
|     executor: |     appname: xxl-job-executor | ||||||
|       # 执行器AppName:执行器心跳注册分组依据;为空则关闭自动注册 |     # 执行器端口号 执行器从9101开始往后写 | ||||||
|       appname: xxl-job-executor |     port: 9101 | ||||||
|       # 执行器端口号 执行器从9101开始往后写 |     # 执行器注册:默认IP:PORT | ||||||
|       port: 9101 |     address: | ||||||
|       # 执行器注册:默认IP:PORT |     # 执行器IP:默认自动获取IP | ||||||
|       address: |     ip: | ||||||
|       # 执行器IP:默认自动获取IP |     # 执行器运行日志文件存储磁盘路径 | ||||||
|       ip: |     logpath: ./logs/xxl-job | ||||||
|       # 执行器运行日志文件存储磁盘路径 |     # 执行器日志文件保存天数:大于3生效 | ||||||
|       logpath: ./logs/xxl-job |     logretentiondays: 30 | ||||||
|       # 执行器日志文件保存天数:大于3生效 |  | ||||||
|       logretentiondays: 30 |  | ||||||
|  |  | ||||||
| --- # 数据源配置 | --- # 数据源配置 | ||||||
| spring: | spring: | ||||||
|   datasource: |   datasource: | ||||||
|     type: com.alibaba.druid.pool.DruidDataSource |     type: com.zaxxer.hikari.HikariDataSource | ||||||
|     # 动态数据源文档 https://www.kancloud.cn/tracy5546/dynamic-datasource/content |     # 动态数据源文档 https://www.kancloud.cn/tracy5546/dynamic-datasource/content | ||||||
|     dynamic: |     dynamic: | ||||||
|       # 性能分析插件(有性能损耗 不建议生产环境使用) |       # 性能分析插件(有性能损耗 不建议生产环境使用) | ||||||
|       p6spy: true |       p6spy: true | ||||||
|       # 设置默认的数据源或者数据源组,默认值即为 master |       # 设置默认的数据源或者数据源组,默认值即为 master | ||||||
|       primary: master |       primary: master | ||||||
|  |       # 严格模式 匹配不到数据源则报错 | ||||||
|  |       strict: true | ||||||
|       datasource: |       datasource: | ||||||
|         # 主库数据源 |         # 主库数据源 | ||||||
|         master: |         master: | ||||||
|  |           type: ${spring.datasource.type} | ||||||
|           driverClassName: com.mysql.cj.jdbc.Driver |           driverClassName: com.mysql.cj.jdbc.Driver | ||||||
|           # jdbc 所有参数配置参考 https://lionli.blog.csdn.net/article/details/122018562 |           # jdbc 所有参数配置参考 https://lionli.blog.csdn.net/article/details/122018562 | ||||||
|           # rewriteBatchedStatements=true 批处理优化 大幅提升批量插入更新删除性能(对数据库有性能损耗 使用批量操作应考虑性能问题) |           # rewriteBatchedStatements=true 批处理优化 大幅提升批量插入更新删除性能(对数据库有性能损耗 使用批量操作应考虑性能问题) | ||||||
| @@ -59,57 +55,46 @@ spring: | |||||||
|         # 从库数据源 |         # 从库数据源 | ||||||
|         slave: |         slave: | ||||||
|           lazy: true |           lazy: true | ||||||
|  |           type: ${spring.datasource.type} | ||||||
|           driverClassName: com.mysql.cj.jdbc.Driver |           driverClassName: com.mysql.cj.jdbc.Driver | ||||||
|           url: |           url: jdbc:mysql://localhost:3306/ry-vue?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8&autoReconnect=true&rewriteBatchedStatements=true | ||||||
|           username: |           username: | ||||||
|           password: |           password: | ||||||
|       druid: | #        oracle: | ||||||
|         # 初始连接数 | #          type: ${spring.datasource.type} | ||||||
|         initialSize: 5 | #          driverClassName: oracle.jdbc.OracleDriver | ||||||
|         # 最小连接池数量 | #          url: jdbc:oracle:thin:@//localhost:1521/XE | ||||||
|         minIdle: 10 | #          username: ROOT | ||||||
|  | #          password: root | ||||||
|  | #          hikari: | ||||||
|  | #            connectionTestQuery: SELECT 1 FROM DUAL | ||||||
|  | #        postgres: | ||||||
|  | #          type: ${spring.datasource.type} | ||||||
|  | #          driverClassName: org.postgresql.Driver | ||||||
|  | #          url: jdbc:postgresql://localhost:5432/postgres?useUnicode=true&characterEncoding=utf8&useSSL=true&autoReconnect=true&reWriteBatchedInserts=true | ||||||
|  | #          username: root | ||||||
|  | #          password: root | ||||||
|  | #        sqlserver: | ||||||
|  | #          type: ${spring.datasource.type} | ||||||
|  | #          driverClassName: com.microsoft.sqlserver.jdbc.SQLServerDriver | ||||||
|  | #          url: jdbc:sqlserver://localhost:1433;DatabaseName=tempdb;SelectMethod=cursor;encrypt=false;rewriteBatchedStatements=true | ||||||
|  | #          username: SA | ||||||
|  | #          password: root | ||||||
|  |       hikari: | ||||||
|         # 最大连接池数量 |         # 最大连接池数量 | ||||||
|         maxActive: 20 |         maxPoolSize: 20 | ||||||
|  |         # 最小空闲线程数量 | ||||||
|  |         minIdle: 10 | ||||||
|         # 配置获取连接等待超时的时间 |         # 配置获取连接等待超时的时间 | ||||||
|         maxWait: 60000 |         connectionTimeout: 10000 | ||||||
|         # 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 |         # 校验超时时间 | ||||||
|         timeBetweenEvictionRunsMillis: 60000 |         validationTimeout: 5000 | ||||||
|         # 配置一个连接在池中最小生存的时间,单位是毫秒 |         # 空闲连接存活最大时间,默认10分钟 | ||||||
|         minEvictableIdleTimeMillis: 300000 |         idleTimeout: 60000 | ||||||
|         # 配置一个连接在池中最大生存的时间,单位是毫秒 |         # 此属性控制池中连接的最长生命周期,值0表示无限生命周期,默认30分钟 | ||||||
|         maxEvictableIdleTimeMillis: 900000 |         maxLifetime: 900000 | ||||||
|         # 配置检测连接是否有效 |         # 连接测试query(配置检测连接是否有效) | ||||||
|         validationQuery: SELECT 1 FROM DUAL |         connectionTestQuery: SELECT 1 | ||||||
|         testWhileIdle: true |  | ||||||
|         testOnBorrow: false |  | ||||||
|         testOnReturn: false |  | ||||||
|         # 注意这个值和druid原生不一致,默认启动了stat |  | ||||||
|         filters: stat |  | ||||||
|  |  | ||||||
| --- # druid 配置 |  | ||||||
| spring: |  | ||||||
|   datasource: |  | ||||||
|     druid: |  | ||||||
|       webStatFilter: |  | ||||||
|         enabled: true |  | ||||||
|       statViewServlet: |  | ||||||
|         enabled: true |  | ||||||
|         # 设置白名单,不填则允许所有访问 |  | ||||||
|         allow: |  | ||||||
|         url-pattern: /druid/* |  | ||||||
|         # 控制台管理用户名和密码 |  | ||||||
|         login-username: ruoyi |  | ||||||
|         login-password: 123456 |  | ||||||
|       filter: |  | ||||||
|         stat: |  | ||||||
|           enabled: true |  | ||||||
|           # 慢SQL记录 |  | ||||||
|           log-slow-sql: true |  | ||||||
|           slow-sql-millis: 1000 |  | ||||||
|           merge-sql: true |  | ||||||
|         wall: |  | ||||||
|           config: |  | ||||||
|             multi-statement-allow: true |  | ||||||
|  |  | ||||||
| --- # redis 单机配置(单机与集群只能开启一个另一个需要注释掉) | --- # redis 单机配置(单机与集群只能开启一个另一个需要注释掉) | ||||||
| spring: | spring: | ||||||
| @@ -120,20 +105,20 @@ spring: | |||||||
|     port: 6379 |     port: 6379 | ||||||
|     # 数据库索引 |     # 数据库索引 | ||||||
|     database: 0 |     database: 0 | ||||||
|     # 密码 |     # 密码(如没有密码请注释掉) | ||||||
|     password: |     # password: | ||||||
|     # 连接超时时间 |     # 连接超时时间 | ||||||
|     timeout: 10s |     timeout: 10s | ||||||
|     # 是否开启ssl |     # 是否开启ssl | ||||||
|     ssl: false |     ssl: false | ||||||
|  |  | ||||||
| redisson: | redisson: | ||||||
|  |   # redis key前缀 | ||||||
|  |   keyPrefix: | ||||||
|   # 线程池数量 |   # 线程池数量 | ||||||
|   threads: 4 |   threads: 4 | ||||||
|   # Netty线程池数量 |   # Netty线程池数量 | ||||||
|   nettyThreads: 8 |   nettyThreads: 8 | ||||||
|   # 传输模式 |  | ||||||
|   transportMode: "NIO" |  | ||||||
|   # 单节点配置 |   # 单节点配置 | ||||||
|   singleServerConfig: |   singleServerConfig: | ||||||
|     # 客户端名称 |     # 客户端名称 | ||||||
| @@ -146,9 +131,39 @@ redisson: | |||||||
|     idleConnectionTimeout: 10000 |     idleConnectionTimeout: 10000 | ||||||
|     # 命令等待超时,单位:毫秒 |     # 命令等待超时,单位:毫秒 | ||||||
|     timeout: 3000 |     timeout: 3000 | ||||||
|     # 如果尝试在此限制之内发送成功,则开始启用 timeout 计时。 |  | ||||||
|     retryAttempts: 3 |  | ||||||
|     # 命令重试发送时间间隔,单位:毫秒 |  | ||||||
|     retryInterval: 1500 |  | ||||||
|     # 发布和订阅连接池大小 |     # 发布和订阅连接池大小 | ||||||
|     subscriptionConnectionPoolSize: 50 |     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: | ||||||
|   | |||||||
| @@ -1,146 +1,127 @@ | |||||||
| --- # 配置临时路径存储 | --- # 临时文件存储位置 避免临时文件被系统清理报错 | ||||||
| spring: | spring.servlet.multipart.location: /ruoyi/server/temp | ||||||
|   servlet: |  | ||||||
|     multipart: |  | ||||||
|       # 临时文件存储位置 避免临时文件被系统清理报错 |  | ||||||
|       location: /ruoyi/server/temp |  | ||||||
|  |  | ||||||
| --- # 监控配置 | --- # 监控中心配置 | ||||||
| spring: | spring.boot.admin.client: | ||||||
|   boot: |   # 增加客户端开关 | ||||||
|     admin: |   enabled: true | ||||||
|       # Spring Boot Admin Client 客户端的相关配置 |   url: http://localhost:9090/admin | ||||||
|       client: |   instance: | ||||||
|         # 增加客户端开关 |     service-host-type: IP | ||||||
|         enabled: true |   username: ruoyi | ||||||
|         # 设置 Spring Boot Admin Server 地址 |   password: 123456 | ||||||
|         url: http://172.30.0.90:9090/admin |  | ||||||
|         instance: |  | ||||||
|           service-host-type: IP |  | ||||||
|         username: ruoyi |  | ||||||
|         password: 123456 |  | ||||||
|  |  | ||||||
| --- # xxl-job 配置 | --- # xxl-job 配置 | ||||||
| xxl: | xxl.job: | ||||||
|   job: |   # 执行器开关 | ||||||
|     # 执行器开关 |   enabled: true | ||||||
|     enabled: true |   # 调度中心地址:如调度中心集群部署存在多个地址则用逗号分隔。 | ||||||
|     # 调度中心地址:如调度中心集群部署存在多个地址则用逗号分隔。 |   admin-addresses: http://localhost:9100/xxl-job-admin | ||||||
|     admin-addresses: http://172.30.0.92:9100/xxl-job-admin |   # 执行器通讯TOKEN:非空时启用 | ||||||
|     # 执行器通讯TOKEN:非空时启用 |   access-token: xxl-job | ||||||
|     access-token: xxl-job |   executor: | ||||||
|     # 执行器配置 |     # 执行器AppName:执行器心跳注册分组依据;为空则关闭自动注册 | ||||||
|     executor: |     appname: xxl-job-executor | ||||||
|       # 执行器AppName:执行器心跳注册分组依据;为空则关闭自动注册 |     # 执行器端口号 执行器从9101开始往后写 | ||||||
|       appname: xxl-job-executor |     port: 9101 | ||||||
|       # 执行器端口号 执行器从9101开始往后写 |     # 执行器注册:默认IP:PORT | ||||||
|       port: 9101 |     address: | ||||||
|       # 执行器注册:默认IP:PORT |     # 执行器IP:默认自动获取IP | ||||||
|       address: |     ip: | ||||||
|       # 执行器IP:默认自动获取IP |     # 执行器运行日志文件存储磁盘路径 | ||||||
|       ip: |     logpath: ./logs/xxl-job | ||||||
|       # 执行器运行日志文件存储磁盘路径 |     # 执行器日志文件保存天数:大于3生效 | ||||||
|       logpath: ./logs/xxl-job |     logretentiondays: 30 | ||||||
|       # 执行器日志文件保存天数:大于3生效 |  | ||||||
|       logretentiondays: 30 |  | ||||||
|  |  | ||||||
| --- # 数据源配置 | --- # 数据源配置 | ||||||
| spring: | spring: | ||||||
|   datasource: |   datasource: | ||||||
|     type: com.alibaba.druid.pool.DruidDataSource |     type: com.zaxxer.hikari.HikariDataSource | ||||||
|     # 动态数据源文档 https://www.kancloud.cn/tracy5546/dynamic-datasource/content |     # 动态数据源文档 https://www.kancloud.cn/tracy5546/dynamic-datasource/content | ||||||
|     dynamic: |     dynamic: | ||||||
|       # 性能分析插件(有性能损耗 不建议生产环境使用) |       # 性能分析插件(有性能损耗 不建议生产环境使用) | ||||||
|       p6spy: false |       p6spy: false | ||||||
|       # 设置默认的数据源或者数据源组,默认值即为 master |       # 设置默认的数据源或者数据源组,默认值即为 master | ||||||
|       primary: master |       primary: master | ||||||
|  |       # 严格模式 匹配不到数据源则报错 | ||||||
|  |       strict: true | ||||||
|       datasource: |       datasource: | ||||||
|         # 主库数据源 |         # 主库数据源 | ||||||
|         master: |         master: | ||||||
|  |           type: ${spring.datasource.type} | ||||||
|           driverClassName: com.mysql.cj.jdbc.Driver |           driverClassName: com.mysql.cj.jdbc.Driver | ||||||
|           # jdbc 所有参数配置参考 https://lionli.blog.csdn.net/article/details/122018562 |           # jdbc 所有参数配置参考 https://lionli.blog.csdn.net/article/details/122018562 | ||||||
|           # rewriteBatchedStatements=true 批处理优化 大幅提升批量插入更新删除性能(对数据库有性能损耗 使用批量操作应考虑性能问题) |           # 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 |           username: root | ||||||
|           password: root |           password: root | ||||||
|         # 从库数据源 |         # 从库数据源 | ||||||
|         slave: |         slave: | ||||||
|           lazy: true |           lazy: true | ||||||
|  |           type: ${spring.datasource.type} | ||||||
|           driverClassName: com.mysql.cj.jdbc.Driver |           driverClassName: com.mysql.cj.jdbc.Driver | ||||||
|           url: |           url: jdbc:mysql://localhost:3306/ry-vue?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8&autoReconnect=true&rewriteBatchedStatements=true | ||||||
|           username: |           username: | ||||||
|           password: |           password: | ||||||
|       druid: | #        oracle: | ||||||
|         # 初始连接数 | #          type: ${spring.datasource.type} | ||||||
|         initialSize: 5 | #          driverClassName: oracle.jdbc.OracleDriver | ||||||
|         # 最小连接池数量 | #          url: jdbc:oracle:thin:@//localhost:1521/XE | ||||||
|         minIdle: 10 | #          username: ROOT | ||||||
|  | #          password: root | ||||||
|  | #          hikari: | ||||||
|  | #            connectionTestQuery: SELECT 1 FROM DUAL | ||||||
|  | #        postgres: | ||||||
|  | #          type: ${spring.datasource.type} | ||||||
|  | #          driverClassName: org.postgresql.Driver | ||||||
|  | #          url: jdbc:postgresql://localhost:5432/postgres?useUnicode=true&characterEncoding=utf8&useSSL=true&autoReconnect=true&reWriteBatchedInserts=true | ||||||
|  | #          username: root | ||||||
|  | #          password: root | ||||||
|  | #        sqlserver: | ||||||
|  | #          type: ${spring.datasource.type} | ||||||
|  | #          driverClassName: com.microsoft.sqlserver.jdbc.SQLServerDriver | ||||||
|  | #          url: jdbc:sqlserver://localhost:1433;DatabaseName=tempdb;SelectMethod=cursor;encrypt=false;rewriteBatchedStatements=true | ||||||
|  | #          username: SA | ||||||
|  | #          password: root | ||||||
|  |       hikari: | ||||||
|         # 最大连接池数量 |         # 最大连接池数量 | ||||||
|         maxActive: 20 |         maxPoolSize: 20 | ||||||
|  |         # 最小空闲线程数量 | ||||||
|  |         minIdle: 10 | ||||||
|         # 配置获取连接等待超时的时间 |         # 配置获取连接等待超时的时间 | ||||||
|         maxWait: 60000 |         connectionTimeout: 10000 | ||||||
|         # 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 |         # 校验超时时间 | ||||||
|         timeBetweenEvictionRunsMillis: 60000 |         validationTimeout: 5000 | ||||||
|         # 配置一个连接在池中最小生存的时间,单位是毫秒 |         # 空闲连接存活最大时间,默认10分钟 | ||||||
|         minEvictableIdleTimeMillis: 300000 |         idleTimeout: 60000 | ||||||
|         # 配置一个连接在池中最大生存的时间,单位是毫秒 |         # 此属性控制池中连接的最长生命周期,值0表示无限生命周期,默认30分钟 | ||||||
|         maxEvictableIdleTimeMillis: 900000 |         maxLifetime: 900000 | ||||||
|         # 配置检测连接是否有效 |         # 连接测试query(配置检测连接是否有效) | ||||||
|         validationQuery: SELECT 1 FROM DUAL |         connectionTestQuery: SELECT 1 | ||||||
|         testWhileIdle: true |  | ||||||
|         testOnBorrow: false |  | ||||||
|         testOnReturn: false |  | ||||||
|         # 注意这个值和druid原生不一致,默认启动了stat |  | ||||||
|         filters: stat |  | ||||||
|  |  | ||||||
| --- # druid 配置 |  | ||||||
| spring: |  | ||||||
|   datasource: |  | ||||||
|     druid: |  | ||||||
|       webStatFilter: |  | ||||||
|         enabled: true |  | ||||||
|       statViewServlet: |  | ||||||
|         enabled: true |  | ||||||
|         # 设置白名单,不填则允许所有访问 |  | ||||||
|         allow: |  | ||||||
|         url-pattern: /druid/* |  | ||||||
|         # 控制台管理用户名和密码 |  | ||||||
|         login-username: ruoyi |  | ||||||
|         login-password: 123456 |  | ||||||
|       filter: |  | ||||||
|         stat: |  | ||||||
|           enabled: true |  | ||||||
|           # 慢SQL记录 |  | ||||||
|           log-slow-sql: true |  | ||||||
|           slow-sql-millis: 1000 |  | ||||||
|           merge-sql: true |  | ||||||
|         wall: |  | ||||||
|           config: |  | ||||||
|             multi-statement-allow: true |  | ||||||
|  |  | ||||||
| --- # redis 单机配置(单机与集群只能开启一个另一个需要注释掉) | --- # redis 单机配置(单机与集群只能开启一个另一个需要注释掉) | ||||||
| spring: | spring: | ||||||
|   redis: |   redis: | ||||||
|     # 地址 |     # 地址 | ||||||
|     host: 172.30.0.48 |     host: localhost | ||||||
|     # 端口,默认为6379 |     # 端口,默认为6379 | ||||||
|     port: 6379 |     port: 6379 | ||||||
|     # 数据库索引 |     # 数据库索引 | ||||||
|     database: 0 |     database: 0 | ||||||
|     # 密码 |     # 密码(如没有密码请注释掉) | ||||||
|     password: |     # password: | ||||||
|     # 连接超时时间 |     # 连接超时时间 | ||||||
|     timeout: 10s |     timeout: 10s | ||||||
|     # 是否开启ssl |     # 是否开启ssl | ||||||
|     ssl: false |     ssl: false | ||||||
|  |  | ||||||
| redisson: | redisson: | ||||||
|  |   # redis key前缀 | ||||||
|  |   keyPrefix: | ||||||
|   # 线程池数量 |   # 线程池数量 | ||||||
|   threads: 16 |   threads: 16 | ||||||
|   # Netty线程池数量 |   # Netty线程池数量 | ||||||
|   nettyThreads: 32 |   nettyThreads: 32 | ||||||
|   # 传输模式 |  | ||||||
|   transportMode: "NIO" |  | ||||||
|   # 单节点配置 |   # 单节点配置 | ||||||
|   singleServerConfig: |   singleServerConfig: | ||||||
|     # 客户端名称 |     # 客户端名称 | ||||||
| @@ -153,9 +134,39 @@ redisson: | |||||||
|     idleConnectionTimeout: 10000 |     idleConnectionTimeout: 10000 | ||||||
|     # 命令等待超时,单位:毫秒 |     # 命令等待超时,单位:毫秒 | ||||||
|     timeout: 3000 |     timeout: 3000 | ||||||
|     # 如果尝试在此限制之内发送成功,则开始启用 timeout 计时。 |  | ||||||
|     retryAttempts: 3 |  | ||||||
|     # 命令重试发送时间间隔,单位:毫秒 |  | ||||||
|     retryInterval: 1500 |  | ||||||
|     # 发布和订阅连接池大小 |     # 发布和订阅连接池大小 | ||||||
|     subscriptionConnectionPoolSize: 50 |     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 |     org.springframework: warn | ||||||
|   config: classpath:logback.xml |   config: classpath:logback.xml | ||||||
|  |  | ||||||
|  | # 用户配置 | ||||||
|  | user: | ||||||
|  |   password: | ||||||
|  |     # 密码最大错误次数 | ||||||
|  |     maxRetryCount: 5 | ||||||
|  |     # 密码锁定时间(默认10分钟) | ||||||
|  |     lockTime: 10 | ||||||
|  |  | ||||||
| # Spring配置 | # Spring配置 | ||||||
| spring: | spring: | ||||||
|   application: |   application: | ||||||
| @@ -75,14 +83,9 @@ spring: | |||||||
|     restart: |     restart: | ||||||
|       # 热部署开关 |       # 热部署开关 | ||||||
|       enabled: true |       enabled: true | ||||||
|   # 与vue整合部署使用 |  | ||||||
|   thymeleaf: |  | ||||||
|     # 将系统模板放置到最前面 否则会与 springboot-admin 页面冲突 |  | ||||||
|     template-resolver-order: 1 |  | ||||||
|   mvc: |   mvc: | ||||||
|     pathmatch: |     format: | ||||||
|       # 适配 boot 2.6 路由与 springfox 兼容 |       date-time: yyyy-MM-dd HH:mm:ss | ||||||
|       matching-strategy: ANT_PATH_MATCHER |  | ||||||
|   jackson: |   jackson: | ||||||
|     # 日期格式化 |     # 日期格式化 | ||||||
|     date-format: yyyy-MM-dd HH:mm:ss |     date-format: yyyy-MM-dd HH:mm:ss | ||||||
| @@ -107,52 +110,32 @@ sa-token: | |||||||
|   is-concurrent: true |   is-concurrent: true | ||||||
|   # 在多人登录同一账号时,是否共用一个token (为true时所有登录共用一个token, 为false时每次登录新建一个token) |   # 在多人登录同一账号时,是否共用一个token (为true时所有登录共用一个token, 为false时每次登录新建一个token) | ||||||
|   is-share: false |   is-share: false | ||||||
|   # 是否尝试从请求体里读取token |  | ||||||
|   is-read-body: false |  | ||||||
|   # 是否尝试从header里读取token |   # 是否尝试从header里读取token | ||||||
|   is-read-head: true |   is-read-header: true | ||||||
|   # 是否尝试从cookie里读取token |   # 是否尝试从cookie里读取token | ||||||
|   is-read-cookie: false |   is-read-cookie: false | ||||||
|   # token前缀 |   # token前缀 | ||||||
|   token-prefix: "Bearer" |   token-prefix: "Bearer" | ||||||
|   # token风格 |  | ||||||
|   token-style: uuid |  | ||||||
|   # jwt秘钥 |   # jwt秘钥 | ||||||
|   jwt-secret-key: abcdefghijklmnopqrstuvwxyz |   jwt-secret-key: abcdefghijklmnopqrstuvwxyz | ||||||
|   # 是否输出操作日志 |  | ||||||
|   is-log: true |  | ||||||
|  |  | ||||||
| # security配置 | # security配置 | ||||||
| security: | security: | ||||||
|   # 排除路径 |   # 排除路径 | ||||||
|   excludes: |   excludes: | ||||||
|     - /login |  | ||||||
|     - /logout |  | ||||||
|     - /register |  | ||||||
|     - /captchaImage |  | ||||||
|     # 静态资源 |     # 静态资源 | ||||||
|     - /*.html |     - /*.html | ||||||
|     - /**/*.html |     - /**/*.html | ||||||
|     - /**/*.css |     - /**/*.css | ||||||
|     - /**/*.js |     - /**/*.js | ||||||
|     # swagger 文档配置 |     # swagger 文档配置 | ||||||
|     - /doc.html |     - /favicon.ico | ||||||
|     - /swagger-resources/** |  | ||||||
|     - /webjars/** |  | ||||||
|     - /*/api-docs |     - /*/api-docs | ||||||
|     # druid 监控配置 |     - /*/api-docs/** | ||||||
|     - /druid/** |  | ||||||
|   # 用户放行 |  | ||||||
|   permit-all: |  | ||||||
|     # actuator 监控配置 |     # actuator 监控配置 | ||||||
|     - /actuator |     - /actuator | ||||||
|     - /actuator/** |     - /actuator/** | ||||||
|  |  | ||||||
| # 重复提交 |  | ||||||
| repeat-submit: |  | ||||||
|   # 全局间隔时间(毫秒) |  | ||||||
|   interval: 5000 |  | ||||||
|  |  | ||||||
| # MyBatisPlus配置 | # MyBatisPlus配置 | ||||||
| # https://baomidou.com/config/ | # https://baomidou.com/config/ | ||||||
| mybatis-plus: | mybatis-plus: | ||||||
| @@ -165,25 +148,15 @@ mybatis-plus: | |||||||
|   typeAliasesPackage: com.ruoyi.**.domain |   typeAliasesPackage: com.ruoyi.**.domain | ||||||
|   # 启动时是否检查 MyBatis XML 文件的存在,默认不检查 |   # 启动时是否检查 MyBatis XML 文件的存在,默认不检查 | ||||||
|   checkConfigLocation: false |   checkConfigLocation: false | ||||||
|   # 通过该属性可指定 MyBatis 的执行器,MyBatis 的执行器总共有三种: |  | ||||||
|   # SIMPLE:每个语句创建新的预处理器 REUSE:会复用预处理器 BATCH:批量执行所有的更新 |  | ||||||
|   executorType: SIMPLE |  | ||||||
|   configuration: |   configuration: | ||||||
|     # 自动驼峰命名规则(camel case)映射 |     # 自动驼峰命名规则(camel case)映射 | ||||||
|     mapUnderscoreToCamelCase: true |     mapUnderscoreToCamelCase: true | ||||||
|     # 当设置为 true 的时候,懒加载的对象可能被任何懒属性全部加载,否则,每个属性都按需加载。需要和 lazyLoadingEnabled 一起使用。 |  | ||||||
|     aggressiveLazyLoading: true |  | ||||||
|     # MyBatis 自动映射策略 |     # MyBatis 自动映射策略 | ||||||
|     # NONE:不启用 PARTIAL:只对非嵌套 resultMap 自动映射 FULL:对所有 resultMap 自动映射 |     # NONE:不启用 PARTIAL:只对非嵌套 resultMap 自动映射 FULL:对所有 resultMap 自动映射 | ||||||
|     autoMappingBehavior: PARTIAL |     autoMappingBehavior: PARTIAL | ||||||
|     # MyBatis 自动映射时未知列或未知属性处理策 |     # MyBatis 自动映射时未知列或未知属性处理策 | ||||||
|     # NONE:不做处理 WARNING:打印相关警告 FAILING:抛出异常和详细信息 |     # NONE:不做处理 WARNING:打印相关警告 FAILING:抛出异常和详细信息 | ||||||
|     autoMappingUnknownColumnBehavior: NONE |     autoMappingUnknownColumnBehavior: NONE | ||||||
|     # Mybatis一级缓存,默认为 SESSION |  | ||||||
|     # SESSION session级别缓存 STATEMENT 关闭一级缓存 |  | ||||||
|     localCacheScope: SESSION |  | ||||||
|     # 开启Mybatis二级缓存,默认为 true |  | ||||||
|     cacheEnabled: false |  | ||||||
|     # 更详细的日志输出 会有性能损耗 org.apache.ibatis.logging.stdout.StdOutImpl |     # 更详细的日志输出 会有性能损耗 org.apache.ibatis.logging.stdout.StdOutImpl | ||||||
|     # 关闭日志记录 (可单纯使用 p6spy 分析) org.apache.ibatis.logging.nologging.NoLoggingImpl |     # 关闭日志记录 (可单纯使用 p6spy 分析) org.apache.ibatis.logging.nologging.NoLoggingImpl | ||||||
|     # 默认日志输出 org.apache.ibatis.logging.slf4j.Slf4jImpl |     # 默认日志输出 org.apache.ibatis.logging.slf4j.Slf4jImpl | ||||||
| @@ -191,16 +164,10 @@ mybatis-plus: | |||||||
|   global-config: |   global-config: | ||||||
|     # 是否打印 Logo banner |     # 是否打印 Logo banner | ||||||
|     banner: true |     banner: true | ||||||
|     # 是否初始化 SqlRunner |  | ||||||
|     enableSqlRunner: false |  | ||||||
|     dbConfig: |     dbConfig: | ||||||
|       # 主键类型 |       # 主键类型 | ||||||
|       # AUTO 自增 NONE 空 INPUT 用户输入 ASSIGN_ID 雪花 ASSIGN_UUID 唯一 UUID |       # AUTO 自增 NONE 空 INPUT 用户输入 ASSIGN_ID 雪花 ASSIGN_UUID 唯一 UUID | ||||||
|       idType: AUTO |       idType: ASSIGN_ID | ||||||
|       # 表名是否使用驼峰转下划线命名,只对表名生效 |  | ||||||
|       tableUnderline: true |  | ||||||
|       # 大写命名,对表名和字段名均生效 |  | ||||||
|       capitalMode: false |  | ||||||
|       # 逻辑已删除值 |       # 逻辑已删除值 | ||||||
|       logicDeleteValue: 2 |       logicDeleteValue: 2 | ||||||
|       # 逻辑未删除值 |       # 逻辑未删除值 | ||||||
| @@ -217,46 +184,38 @@ mybatis-plus: | |||||||
| swagger: | swagger: | ||||||
|   # 是否开启swagger |   # 是否开启swagger | ||||||
|   enabled: true |   enabled: true | ||||||
|   # 请求前缀 |   info: | ||||||
|   pathMapping: /dev-api |     # 标题 | ||||||
|   # 标题 |     title: '标题:${ruoyi.name}后台管理系统_接口文档' | ||||||
|   title: '标题:${ruoyi.name}后台管理系统_接口文档' |     # 描述 | ||||||
|   # 描述 |     description: '描述:用于管理集团旗下公司的人员信息,具体包括XXX,XXX模块...' | ||||||
|   description: '描述:用于管理集团旗下公司的人员信息,具体包括XXX,XXX模块...' |     # 版本 | ||||||
|   # 版本 |     version: '版本号: ${ruoyi-vue-plus.version}' | ||||||
|   version: '版本号: ${ruoyi-vue-plus.version}' |     # 作者信息 | ||||||
|   # 作者信息 |     contact: | ||||||
|   contact: |       name: Lion Li | ||||||
|     name: Lion Li |       email: crazylionli@163.com | ||||||
|     email: crazylionli@163.com |       url: https://gitee.com/JavaLionLi/RuoYi-Vue-Plus | ||||||
|     url: https://gitee.com/JavaLionLi/RuoYi-Vue-Plus |   components: | ||||||
|   groups: |     # 鉴权方式配置 | ||||||
|     - name: 1.演示案例 |     security-schemes: | ||||||
|       basePackage: com.ruoyi.demo |       apiKey: | ||||||
|     - name: 2.系统模块 |         type: APIKEY | ||||||
|       basePackage: com.ruoyi.web |         in: HEADER | ||||||
|     - name: 3.代码生成模块 |         name: ${sa-token.token-name} | ||||||
|       basePackage: com.ruoyi.generator |  | ||||||
|  |  | ||||||
| knife4j: | springdoc: | ||||||
|   # 是否开启Knife4j增强模式 |   swagger-ui: | ||||||
|   enable: true |     # 持久化认证数据 | ||||||
|   # 是否开启生产环境保护策略 |     persistAuthorization: true | ||||||
|   production: @knife4j.production@ |   #这里定义了两个分组,可定义多个,也可以不定义 | ||||||
|   basic: |   group-configs: | ||||||
|     enable: true |     - group: 1.演示模块 | ||||||
|     username: ruoyi |       packages-to-scan: com.ruoyi.demo | ||||||
|     password: 123456 |     - group: 2.系统模块 | ||||||
|   # 前端Ui的个性化配置属性 |       packages-to-scan: com.ruoyi.web | ||||||
|   setting: |     - group: 3.代码生成模块 | ||||||
|     # 默认语言 |       packages-to-scan: com.ruoyi.generator | ||||||
|     language: zh-CN |  | ||||||
|     # 是否显示Footer |  | ||||||
|     enableFooter: false |  | ||||||
|     # 是否开启动态参数调试功能 |  | ||||||
|     enableDynamicParameter: true |  | ||||||
|     # 是否在每个Debug调试栏后显示刷新变量按钮 |  | ||||||
|     enableReloadCacheParameter: true |  | ||||||
|  |  | ||||||
| # 防止XSS攻击 | # 防止XSS攻击 | ||||||
| xss: | xss: | ||||||
| @@ -271,53 +230,26 @@ xss: | |||||||
| thread-pool: | thread-pool: | ||||||
|   # 是否开启线程池 |   # 是否开启线程池 | ||||||
|   enabled: false |   enabled: false | ||||||
|   # 核心线程池大小 |  | ||||||
|   corePoolSize: 8 |  | ||||||
|   # 最大可创建的线程数 |  | ||||||
|   maxPoolSize: 16 |  | ||||||
|   # 队列最大长度 |   # 队列最大长度 | ||||||
|   queueCapacity: 128 |   queueCapacity: 128 | ||||||
|   # 线程池维护线程所允许的空闲时间 |   # 线程池维护线程所允许的空闲时间 | ||||||
|   keepAliveSeconds: 300 |   keepAliveSeconds: 300 | ||||||
|   # 线程池对拒绝任务(无线程可用)的处理策略 |  | ||||||
|   # CALLER_RUNS_POLICY 调用方执行 |  | ||||||
|   # DISCARD_OLDEST_POLICY 放弃最旧的 |  | ||||||
|   # DISCARD_POLICY 丢弃 |  | ||||||
|   # ABORT_POLICY 中止 |  | ||||||
|   rejectedExecutionHandler: CALLER_RUNS_POLICY |  | ||||||
|  |  | ||||||
| --- # redisson 缓存配置 |  | ||||||
| redisson: |  | ||||||
|   cacheGroup: |  | ||||||
|     # 用例: @Cacheable(cacheNames="groupId", key="#XXX") 方可使用缓存组配置 |  | ||||||
|     - groupId: redissonCacheMap |  | ||||||
|       # 组过期时间(脚本监控) |  | ||||||
|       ttl: 60000 |  | ||||||
|       # 组最大空闲时间(脚本监控) |  | ||||||
|       maxIdleTime: 60000 |  | ||||||
|       # 组最大长度 |  | ||||||
|       maxSize: 0 |  | ||||||
|     - groupId: testCache |  | ||||||
|       ttl: 1000 |  | ||||||
|       maxIdleTime: 500 |  | ||||||
|  |  | ||||||
| --- # 分布式锁 lock4j 全局配置 | --- # 分布式锁 lock4j 全局配置 | ||||||
| lock4j: | lock4j: | ||||||
|   # 获取分布式锁超时时间,默认为 3000 毫秒 |   # 获取分布式锁超时时间,默认为 3000 毫秒 | ||||||
|   acquire-timeout: 3000 |   acquire-timeout: 3000 | ||||||
|   # 分布式锁的超时时间,默认为 30 毫秒 |   # 分布式锁的超时时间,默认为 30 秒 | ||||||
|   expire: 30000 |   expire: 30000 | ||||||
|  |  | ||||||
| --- # Actuator 监控端点的配置项 | --- # Actuator 监控端点的配置项 | ||||||
| management: | management: | ||||||
|   endpoints: |   endpoints: | ||||||
|     web: |     web: | ||||||
|       # Actuator 提供的 API 接口的根目录。默认为 /actuator |  | ||||||
|       base-path: /actuator |  | ||||||
|       exposure: |       exposure: | ||||||
|         # 需要开放的端点。默认值只打开 health 和 info 两个端点。通过设置 * ,可以开放所有端点。 |         include: '*' | ||||||
|         # 生产环境不建议放开所有 根据项目需求放开即可 |  | ||||||
|         include: @endpoints.include@ |  | ||||||
|   endpoint: |   endpoint: | ||||||
|  |     health: | ||||||
|  |       show-details: ALWAYS | ||||||
|     logfile: |     logfile: | ||||||
|       external-file: ./logs/sys-console.log |       external-file: ./logs/sys-console.log | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ user.jcaptcha.expire=验证码已失效 | |||||||
| user.not.exists=对不起, 您的账号:{0} 不存在. | user.not.exists=对不起, 您的账号:{0} 不存在. | ||||||
| user.password.not.match=用户不存在/密码错误 | user.password.not.match=用户不存在/密码错误 | ||||||
| user.password.retry.limit.count=密码输入错误{0}次 | user.password.retry.limit.count=密码输入错误{0}次 | ||||||
| user.password.retry.limit.exceed=密码错误次数过多,帐户锁定{0}分钟 | user.password.retry.limit.exceed=密码输入错误{0}次,帐户锁定{1}分钟 | ||||||
| user.password.delete=对不起,您的账号:{0} 已被删除 | user.password.delete=对不起,您的账号:{0} 已被删除 | ||||||
| user.blocked=对不起,您的账号:{0} 已禁用,请联系管理员 | user.blocked=对不起,您的账号:{0} 已禁用,请联系管理员 | ||||||
| role.blocked=角色已封禁,请联系管理员 | role.blocked=角色已封禁,请联系管理员 | ||||||
| @@ -18,6 +18,7 @@ user.password.not.blank=用户密码不能为空 | |||||||
| user.password.length.valid=用户密码长度必须在{min}到{max}个字符之间 | user.password.length.valid=用户密码长度必须在{min}到{max}个字符之间 | ||||||
| user.password.not.valid=* 5-50个字符 | user.password.not.valid=* 5-50个字符 | ||||||
| user.email.not.valid=邮箱格式错误 | user.email.not.valid=邮箱格式错误 | ||||||
|  | user.phonenumber.not.blank=用户手机号不能为空 | ||||||
| user.mobile.phone.number.not.valid=手机号格式错误 | user.mobile.phone.number.not.valid=手机号格式错误 | ||||||
| user.login.success=登录成功 | user.login.success=登录成功 | ||||||
| user.register.success=注册成功 | user.register.success=注册成功 | ||||||
| @@ -36,3 +37,9 @@ no.update.permission=您没有修改数据的权限,请联系管理员添加 | |||||||
| no.delete.permission=您没有删除数据的权限,请联系管理员添加权限 [{0}] | no.delete.permission=您没有删除数据的权限,请联系管理员添加权限 [{0}] | ||||||
| no.export.permission=您没有导出数据的权限,请联系管理员添加权限 [{0}] | no.export.permission=您没有导出数据的权限,请联系管理员添加权限 [{0}] | ||||||
| no.view.permission=您没有查看数据的权限,请联系管理员添加权限 [{0}] | no.view.permission=您没有查看数据的权限,请联系管理员添加权限 [{0}] | ||||||
|  | repeat.submit.message=不允许重复提交,请稍候再试 | ||||||
|  | rate.limiter.message=访问过于频繁,请稍候再试 | ||||||
|  | sms.code.not.blank=短信验证码不能为空 | ||||||
|  | sms.code.retry.limit.count=短信验证码输入错误{0}次 | ||||||
|  | sms.code.retry.limit.exceed=短信验证码输入错误{0}次,帐户锁定{1}分钟 | ||||||
|  | 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.not.exists=Sorry, your account: {0} does not exist | ||||||
| user.password.not.match=User does not exist/Password error | user.password.not.match=User does not exist/Password error | ||||||
| user.password.retry.limit.count=Password input error {0} times | 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.password.delete=Sorry, your account:{0} has been deleted | ||||||
| user.blocked=Sorry, your account: {0} has been disabled. Please contact the administrator | user.blocked=Sorry, your account: {0} has been disabled. Please contact the administrator | ||||||
| role.blocked=Role disabled,please contact administrators | role.blocked=Role disabled,please contact administrators | ||||||
| @@ -18,6 +18,7 @@ user.password.not.blank=Password cannot be empty | |||||||
| user.password.length.valid=Password length must be between {min} and {max} characters | user.password.length.valid=Password length must be between {min} and {max} characters | ||||||
| user.password.not.valid=* 5-50 characters | user.password.not.valid=* 5-50 characters | ||||||
| user.email.not.valid=Mailbox format error | user.email.not.valid=Mailbox format error | ||||||
|  | user.phonenumber.not.blank=Phone number cannot be blank | ||||||
| user.mobile.phone.number.not.valid=Phone number format error | user.mobile.phone.number.not.valid=Phone number format error | ||||||
| user.login.success=Login successful | user.login.success=Login successful | ||||||
| user.register.success=Register successful | user.register.success=Register successful | ||||||
| @@ -36,3 +37,9 @@ no.update.permission=You do not have permission to modify data,please contact | |||||||
| no.delete.permission=You do not have permission to delete data,please contact your administrator to add permissions [{0}] | no.delete.permission=You do not have permission to delete data,please contact your administrator to add permissions [{0}] | ||||||
| no.export.permission=You do not have permission to export data,please contact your administrator to add permissions [{0}] | no.export.permission=You do not have permission to export data,please contact your administrator to add permissions [{0}] | ||||||
| no.view.permission=You do not have permission to view data,please contact your administrator to add permissions [{0}] | no.view.permission=You do not have permission to view data,please contact your administrator to add permissions [{0}] | ||||||
|  | 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=Sms code input error {0} times, account locked for {1} minutes | ||||||
|  | xcx.code.not.blank=Mini program code cannot be blank | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ user.jcaptcha.expire=验证码已失效 | |||||||
| user.not.exists=对不起, 您的账号:{0} 不存在. | user.not.exists=对不起, 您的账号:{0} 不存在. | ||||||
| user.password.not.match=用户不存在/密码错误 | user.password.not.match=用户不存在/密码错误 | ||||||
| user.password.retry.limit.count=密码输入错误{0}次 | user.password.retry.limit.count=密码输入错误{0}次 | ||||||
| user.password.retry.limit.exceed=密码错误次数过多,帐户锁定{0}分钟 | user.password.retry.limit.exceed=密码输入错误{0}次,帐户锁定{1}分钟 | ||||||
| user.password.delete=对不起,您的账号:{0} 已被删除 | user.password.delete=对不起,您的账号:{0} 已被删除 | ||||||
| user.blocked=对不起,您的账号:{0} 已禁用,请联系管理员 | user.blocked=对不起,您的账号:{0} 已禁用,请联系管理员 | ||||||
| role.blocked=角色已封禁,请联系管理员 | role.blocked=角色已封禁,请联系管理员 | ||||||
| @@ -18,6 +18,7 @@ user.password.not.blank=用户密码不能为空 | |||||||
| user.password.length.valid=用户密码长度必须在{min}到{max}个字符之间 | user.password.length.valid=用户密码长度必须在{min}到{max}个字符之间 | ||||||
| user.password.not.valid=* 5-50个字符 | user.password.not.valid=* 5-50个字符 | ||||||
| user.email.not.valid=邮箱格式错误 | user.email.not.valid=邮箱格式错误 | ||||||
|  | user.phonenumber.not.blank=用户手机号不能为空 | ||||||
| user.mobile.phone.number.not.valid=手机号格式错误 | user.mobile.phone.number.not.valid=手机号格式错误 | ||||||
| user.login.success=登录成功 | user.login.success=登录成功 | ||||||
| user.register.success=注册成功 | user.register.success=注册成功 | ||||||
| @@ -36,3 +37,9 @@ no.update.permission=您没有修改数据的权限,请联系管理员添加 | |||||||
| no.delete.permission=您没有删除数据的权限,请联系管理员添加权限 [{0}] | no.delete.permission=您没有删除数据的权限,请联系管理员添加权限 [{0}] | ||||||
| no.export.permission=您没有导出数据的权限,请联系管理员添加权限 [{0}] | no.export.permission=您没有导出数据的权限,请联系管理员添加权限 [{0}] | ||||||
| no.view.permission=您没有查看数据的权限,请联系管理员添加权限 [{0}] | no.view.permission=您没有查看数据的权限,请联系管理员添加权限 [{0}] | ||||||
|  | repeat.submit.message=不允许重复提交,请稍候再试 | ||||||
|  | rate.limiter.message=访问过于频繁,请稍候再试 | ||||||
|  | sms.code.not.blank=短信验证码不能为空 | ||||||
|  | sms.code.retry.limit.count=短信验证码输入错误{0}次 | ||||||
|  | sms.code.retry.limit.exceed=短信验证码输入错误{0}次,帐户锁定{1}分钟 | ||||||
|  | xcx.code.not.blank=小程序code不能为空 | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ | |||||||
|  |  | ||||||
|     <!-- 控制台输出 --> |     <!-- 控制台输出 --> | ||||||
|     <appender name="console" class="ch.qos.logback.core.ConsoleAppender"> |     <appender name="console" class="ch.qos.logback.core.ConsoleAppender"> | ||||||
|         <encoder class="com.yomahub.tlog.core.enhance.logback.AspectLogbackEncoder"> |         <encoder> | ||||||
|             <pattern>${console.log.pattern}</pattern> |             <pattern>${console.log.pattern}</pattern> | ||||||
|             <charset>utf-8</charset> |             <charset>utf-8</charset> | ||||||
|         </encoder> |         </encoder> | ||||||
| @@ -22,7 +22,7 @@ | |||||||
|             <!-- 日志最大 1天 --> |             <!-- 日志最大 1天 --> | ||||||
|             <maxHistory>1</maxHistory> |             <maxHistory>1</maxHistory> | ||||||
|         </rollingPolicy> |         </rollingPolicy> | ||||||
|         <encoder class="com.yomahub.tlog.core.enhance.logback.AspectLogbackEncoder"> |         <encoder> | ||||||
|             <pattern>${log.pattern}</pattern> |             <pattern>${log.pattern}</pattern> | ||||||
|             <charset>utf-8</charset> |             <charset>utf-8</charset> | ||||||
|         </encoder> |         </encoder> | ||||||
| @@ -42,7 +42,7 @@ | |||||||
|             <!-- 日志最大的历史 60天 --> |             <!-- 日志最大的历史 60天 --> | ||||||
|             <maxHistory>60</maxHistory> |             <maxHistory>60</maxHistory> | ||||||
|         </rollingPolicy> |         </rollingPolicy> | ||||||
|         <encoder class="com.yomahub.tlog.core.enhance.logback.AspectLogbackEncoder"> |         <encoder> | ||||||
|             <pattern>${log.pattern}</pattern> |             <pattern>${log.pattern}</pattern> | ||||||
|         </encoder> |         </encoder> | ||||||
|         <filter class="ch.qos.logback.classic.filter.LevelFilter"> |         <filter class="ch.qos.logback.classic.filter.LevelFilter"> | ||||||
| @@ -64,7 +64,7 @@ | |||||||
|             <!-- 日志最大的历史 60天 --> |             <!-- 日志最大的历史 60天 --> | ||||||
|             <maxHistory>60</maxHistory> |             <maxHistory>60</maxHistory> | ||||||
|         </rollingPolicy> |         </rollingPolicy> | ||||||
|         <encoder class="com.yomahub.tlog.core.enhance.logback.AspectLogbackEncoder"> |         <encoder> | ||||||
|             <pattern>${log.pattern}</pattern> |             <pattern>${log.pattern}</pattern> | ||||||
|         </encoder> |         </encoder> | ||||||
|         <filter class="ch.qos.logback.classic.filter.LevelFilter"> |         <filter class="ch.qos.logback.classic.filter.LevelFilter"> | ||||||
| @@ -77,20 +77,53 @@ | |||||||
|         </filter> |         </filter> | ||||||
|     </appender> |     </appender> | ||||||
|  |  | ||||||
|     <!-- 系统模块日志级别控制  --> |     <!-- info异步输出 --> | ||||||
|     <logger name="com.ruoyi" level="info" /> |     <appender name="async_info" class="ch.qos.logback.classic.AsyncAppender"> | ||||||
|     <!-- Spring日志级别控制  --> |         <!-- 不丢失日志.默认的,如果队列的80%已满,则会丢弃TRACT、DEBUG、INFO级别的日志 --> | ||||||
|     <logger name="org.springframework" level="warn" /> |         <discardingThreshold>0</discardingThreshold> | ||||||
|  |         <!-- 更改默认的队列的深度,该值会影响性能.默认值为256 --> | ||||||
|  |         <queueSize>512</queueSize> | ||||||
|  |         <!-- 添加附加的appender,最多只能添加一个 --> | ||||||
|  |         <appender-ref ref="file_info"/> | ||||||
|  |     </appender> | ||||||
|  |  | ||||||
|     <root level="info"> |     <!-- error异步输出 --> | ||||||
|         <appender-ref ref="console" /> |     <appender name="async_error" class="ch.qos.logback.classic.AsyncAppender"> | ||||||
|     </root> |         <!-- 不丢失日志.默认的,如果队列的80%已满,则会丢弃TRACT、DEBUG、INFO级别的日志 --> | ||||||
|  |         <discardingThreshold>0</discardingThreshold> | ||||||
|  |         <!-- 更改默认的队列的深度,该值会影响性能.默认值为256 --> | ||||||
|  |         <queueSize>512</queueSize> | ||||||
|  |         <!-- 添加附加的appender,最多只能添加一个 --> | ||||||
|  |         <appender-ref ref="file_error"/> | ||||||
|  |     </appender> | ||||||
|  |  | ||||||
|  |     <!-- 整合 skywalking 控制台输出 tid --> | ||||||
|  | <!--    <appender name="console" class="ch.qos.logback.core.ConsoleAppender">--> | ||||||
|  | <!--        <encoder class="ch.qos.logback.core.encoder.LayoutWrappingEncoder">--> | ||||||
|  | <!--            <layout class="org.apache.skywalking.apm.toolkit.log.logback.v1.x.TraceIdPatternLogbackLayout">--> | ||||||
|  | <!--                <pattern>[%tid] ${console.log.pattern}</pattern>--> | ||||||
|  | <!--            </layout>--> | ||||||
|  | <!--            <charset>utf-8</charset>--> | ||||||
|  | <!--        </encoder>--> | ||||||
|  | <!--    </appender>--> | ||||||
|  |  | ||||||
|  |     <!-- 整合 skywalking 推送采集日志 --> | ||||||
|  | <!--    <appender name="sky_log" class="org.apache.skywalking.apm.toolkit.log.logback.v1.x.log.GRPCLogClientAppender">--> | ||||||
|  | <!--        <encoder class="ch.qos.logback.core.encoder.LayoutWrappingEncoder">--> | ||||||
|  | <!--            <layout class="org.apache.skywalking.apm.toolkit.log.logback.v1.x.TraceIdPatternLogbackLayout">--> | ||||||
|  | <!--                <pattern>[%tid] ${console.log.pattern}</pattern>--> | ||||||
|  | <!--            </layout>--> | ||||||
|  | <!--            <charset>utf-8</charset>--> | ||||||
|  | <!--        </encoder>--> | ||||||
|  | <!--    </appender>--> | ||||||
|  |  | ||||||
|     <!--系统操作日志--> |     <!--系统操作日志--> | ||||||
|     <root level="info"> |     <root level="info"> | ||||||
|         <appender-ref ref="file_info" /> |         <appender-ref ref="console" /> | ||||||
|         <appender-ref ref="file_error" /> |         <appender-ref ref="async_info" /> | ||||||
|  |         <appender-ref ref="async_error" /> | ||||||
|         <appender-ref ref="file_console" /> |         <appender-ref ref="file_console" /> | ||||||
|  | <!--        <appender-ref ref="sky_log"/>--> | ||||||
|     </root> |     </root> | ||||||
|  |  | ||||||
| </configuration> | </configuration> | ||||||
|   | |||||||
| @@ -14,6 +14,8 @@ useprefix=true | |||||||
| excludecategories=info,debug,result,commit,resultset | excludecategories=info,debug,result,commit,resultset | ||||||
| # 日期格式 | # 日期格式 | ||||||
| dateformat=yyyy-MM-dd HH:mm:ss | dateformat=yyyy-MM-dd HH:mm:ss | ||||||
|  | # SQL语句打印时间格式 | ||||||
|  | databaseDialectTimestampFormat=yyyy-MM-dd HH:mm:ss | ||||||
| # 实际驱动可多个 | # 实际驱动可多个 | ||||||
| #driverlist=org.h2.Driver | #driverlist=org.h2.Driver | ||||||
| # 是否开启慢SQL记录 | # 是否开启慢SQL记录 | ||||||
| @@ -22,3 +24,5 @@ outagedetection=true | |||||||
| outagedetectioninterval=2 | outagedetectioninterval=2 | ||||||
| # 是否过滤 Log | # 是否过滤 Log | ||||||
| filter=true | filter=true | ||||||
|  | # 过滤 Log 时所排除的 sql 关键字,以逗号分隔 | ||||||
|  | exclude=SELECT 1 | ||||||
|   | |||||||
							
								
								
									
										45
									
								
								ruoyi-admin/src/test/java/com/ruoyi/test/AssertUnitTest.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								ruoyi-admin/src/test/java/com/ruoyi/test/AssertUnitTest.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | |||||||
|  | package com.ruoyi.test; | ||||||
|  |  | ||||||
|  | import org.junit.jupiter.api.Assertions; | ||||||
|  | import org.junit.jupiter.api.DisplayName; | ||||||
|  | import org.junit.jupiter.api.Test; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 断言单元测试案例 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | @DisplayName("断言单元测试案例") | ||||||
|  | public class AssertUnitTest { | ||||||
|  |  | ||||||
|  |     @DisplayName("测试 assertEquals 方法") | ||||||
|  |     @Test | ||||||
|  |     public void testAssertEquals() { | ||||||
|  |         Assertions.assertEquals("666", new String("666")); | ||||||
|  |         Assertions.assertNotEquals("666", new String("666")); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @DisplayName("测试 assertSame 方法") | ||||||
|  |     @Test | ||||||
|  |     public void testAssertSame() { | ||||||
|  |         Object obj = new Object(); | ||||||
|  |         Object obj1 = obj; | ||||||
|  |         Assertions.assertSame(obj, obj1); | ||||||
|  |         Assertions.assertNotSame(obj, obj1); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @DisplayName("测试 assertTrue 方法") | ||||||
|  |     @Test | ||||||
|  |     public void testAssertTrue() { | ||||||
|  |         Assertions.assertTrue(true); | ||||||
|  |         Assertions.assertFalse(true); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @DisplayName("测试 assertNull 方法") | ||||||
|  |     @Test | ||||||
|  |     public void testAssertNull() { | ||||||
|  |         Assertions.assertNull(null); | ||||||
|  |         Assertions.assertNotNull(null); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										70
									
								
								ruoyi-admin/src/test/java/com/ruoyi/test/DemoUnitTest.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								ruoyi-admin/src/test/java/com/ruoyi/test/DemoUnitTest.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,70 @@ | |||||||
|  | package com.ruoyi.test; | ||||||
|  |  | ||||||
|  | import com.ruoyi.common.config.RuoYiConfig; | ||||||
|  | import org.junit.jupiter.api.*; | ||||||
|  | import org.springframework.beans.factory.annotation.Autowired; | ||||||
|  | import org.springframework.boot.test.context.SpringBootTest; | ||||||
|  |  | ||||||
|  | import java.util.concurrent.TimeUnit; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 单元测试案例 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | @SpringBootTest // 此注解只能在 springboot 主包下使用 需包含 main 方法与 yml 配置文件 | ||||||
|  | @DisplayName("单元测试案例") | ||||||
|  | public class DemoUnitTest { | ||||||
|  |  | ||||||
|  |     @Autowired | ||||||
|  |     private RuoYiConfig ruoYiConfig; | ||||||
|  |  | ||||||
|  |     @DisplayName("测试 @SpringBootTest @Test @DisplayName 注解") | ||||||
|  |     @Test | ||||||
|  |     public void testTest() { | ||||||
|  |         System.out.println(ruoYiConfig); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Disabled | ||||||
|  |     @DisplayName("测试 @Disabled 注解") | ||||||
|  |     @Test | ||||||
|  |     public void testDisabled() { | ||||||
|  |         System.out.println(ruoYiConfig); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Timeout(value = 2L, unit = TimeUnit.SECONDS) | ||||||
|  |     @DisplayName("测试 @Timeout 注解") | ||||||
|  |     @Test | ||||||
|  |     public void testTimeout() throws InterruptedException { | ||||||
|  |         Thread.sleep(3000); | ||||||
|  |         System.out.println(ruoYiConfig); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     @DisplayName("测试 @RepeatedTest 注解") | ||||||
|  |     @RepeatedTest(3) | ||||||
|  |     public void testRepeatedTest() { | ||||||
|  |         System.out.println(666); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @BeforeAll | ||||||
|  |     public static void testBeforeAll() { | ||||||
|  |         System.out.println("@BeforeAll =================="); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @BeforeEach | ||||||
|  |     public void testBeforeEach() { | ||||||
|  |         System.out.println("@BeforeEach =================="); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @AfterEach | ||||||
|  |     public void testAfterEach() { | ||||||
|  |         System.out.println("@AfterEach =================="); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @AfterAll | ||||||
|  |     public static void testAfterAll() { | ||||||
|  |         System.out.println("@AfterAll =================="); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										72
									
								
								ruoyi-admin/src/test/java/com/ruoyi/test/ParamUnitTest.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								ruoyi-admin/src/test/java/com/ruoyi/test/ParamUnitTest.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,72 @@ | |||||||
|  | package com.ruoyi.test; | ||||||
|  |  | ||||||
|  | import com.ruoyi.common.enums.UserType; | ||||||
|  | import org.junit.jupiter.api.AfterEach; | ||||||
|  | import org.junit.jupiter.api.BeforeEach; | ||||||
|  | import org.junit.jupiter.api.DisplayName; | ||||||
|  | import org.junit.jupiter.params.ParameterizedTest; | ||||||
|  | import org.junit.jupiter.params.provider.EnumSource; | ||||||
|  | import org.junit.jupiter.params.provider.MethodSource; | ||||||
|  | import org.junit.jupiter.params.provider.NullSource; | ||||||
|  | import org.junit.jupiter.params.provider.ValueSource; | ||||||
|  |  | ||||||
|  | import java.util.ArrayList; | ||||||
|  | import java.util.List; | ||||||
|  | import java.util.stream.Stream; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 带参数单元测试案例 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | @DisplayName("带参数单元测试案例") | ||||||
|  | public class ParamUnitTest { | ||||||
|  |  | ||||||
|  |     @DisplayName("测试 @ValueSource 注解") | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @ValueSource(strings = {"t1", "t2", "t3"}) | ||||||
|  |     public void testValueSource(String str) { | ||||||
|  |         System.out.println(str); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @DisplayName("测试 @NullSource 注解") | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @NullSource | ||||||
|  |     public void testNullSource(String str) { | ||||||
|  |         System.out.println(str); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @DisplayName("测试 @EnumSource 注解") | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @EnumSource(UserType.class) | ||||||
|  |     public void testEnumSource(UserType type) { | ||||||
|  |         System.out.println(type.getUserType()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @DisplayName("测试 @MethodSource 注解") | ||||||
|  |     @ParameterizedTest | ||||||
|  |     @MethodSource("getParam") | ||||||
|  |     public void testMethodSource(String str) { | ||||||
|  |         System.out.println(str); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static Stream<String> getParam() { | ||||||
|  |         List<String> list = new ArrayList<>(); | ||||||
|  |         list.add("t1"); | ||||||
|  |         list.add("t2"); | ||||||
|  |         list.add("t3"); | ||||||
|  |         return list.stream(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @BeforeEach | ||||||
|  |     public void testBeforeEach() { | ||||||
|  |         System.out.println("@BeforeEach =================="); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @AfterEach | ||||||
|  |     public void testAfterEach() { | ||||||
|  |         System.out.println("@AfterEach =================="); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										54
									
								
								ruoyi-admin/src/test/java/com/ruoyi/test/TagUnitTest.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								ruoyi-admin/src/test/java/com/ruoyi/test/TagUnitTest.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,54 @@ | |||||||
|  | package com.ruoyi.test; | ||||||
|  |  | ||||||
|  | import org.junit.jupiter.api.*; | ||||||
|  | import org.springframework.boot.test.context.SpringBootTest; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 标签单元测试案例 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | @SpringBootTest | ||||||
|  | @DisplayName("标签单元测试案例") | ||||||
|  | public class TagUnitTest { | ||||||
|  |  | ||||||
|  |     @Tag("dev") | ||||||
|  |     @DisplayName("测试 @Tag dev") | ||||||
|  |     @Test | ||||||
|  |     public void testTagDev() { | ||||||
|  |         System.out.println("dev"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Tag("prod") | ||||||
|  |     @DisplayName("测试 @Tag prod") | ||||||
|  |     @Test | ||||||
|  |     public void testTagProd() { | ||||||
|  |         System.out.println("prod"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Tag("local") | ||||||
|  |     @DisplayName("测试 @Tag local") | ||||||
|  |     @Test | ||||||
|  |     public void testTagLocal() { | ||||||
|  |         System.out.println("local"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Tag("exclude") | ||||||
|  |     @DisplayName("测试 @Tag exclude") | ||||||
|  |     @Test | ||||||
|  |     public void testTagExclude() { | ||||||
|  |         System.out.println("exclude"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @BeforeEach | ||||||
|  |     public void testBeforeEach() { | ||||||
|  |         System.out.println("@BeforeEach =================="); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @AfterEach | ||||||
|  |     public void testAfterEach() { | ||||||
|  |         System.out.println("@AfterEach =================="); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -5,7 +5,7 @@ | |||||||
|     <parent> |     <parent> | ||||||
|         <artifactId>ruoyi-vue-plus</artifactId> |         <artifactId>ruoyi-vue-plus</artifactId> | ||||||
|         <groupId>com.ruoyi</groupId> |         <groupId>com.ruoyi</groupId> | ||||||
|         <version>4.0.1</version> |         <version>4.5.0</version> | ||||||
|     </parent> |     </parent> | ||||||
|     <modelVersion>4.0.0</modelVersion> |     <modelVersion>4.0.0</modelVersion> | ||||||
|  |  | ||||||
| @@ -58,34 +58,17 @@ | |||||||
|             <artifactId>jackson-databind</artifactId> |             <artifactId>jackson-databind</artifactId> | ||||||
|         </dependency> |         </dependency> | ||||||
|  |  | ||||||
|         <!-- excel工具 --> |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>org.apache.poi</groupId> |  | ||||||
|             <artifactId>poi-ooxml</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <dependency> |         <dependency> | ||||||
|             <groupId>com.alibaba</groupId> |             <groupId>com.alibaba</groupId> | ||||||
|             <artifactId>easyexcel</artifactId> |             <artifactId>easyexcel</artifactId> | ||||||
|         </dependency> |         </dependency> | ||||||
|  |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>cglib</groupId> |  | ||||||
|             <artifactId>cglib</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <!-- yml解析器 --> |         <!-- yml解析器 --> | ||||||
|         <dependency> |         <dependency> | ||||||
|             <groupId>org.yaml</groupId> |             <groupId>org.yaml</groupId> | ||||||
|             <artifactId>snakeyaml</artifactId> |             <artifactId>snakeyaml</artifactId> | ||||||
|         </dependency> |         </dependency> | ||||||
|  |  | ||||||
|         <!-- jdk11 缺失依赖 jaxb--> |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>com.sun.xml.bind</groupId> |  | ||||||
|             <artifactId>jaxb-impl</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <!-- servlet包 --> |         <!-- servlet包 --> | ||||||
|         <dependency> |         <dependency> | ||||||
|             <groupId>javax.servlet</groupId> |             <groupId>javax.servlet</groupId> | ||||||
| @@ -96,10 +79,6 @@ | |||||||
|             <groupId>com.baomidou</groupId> |             <groupId>com.baomidou</groupId> | ||||||
|             <artifactId>mybatis-plus-boot-starter</artifactId> |             <artifactId>mybatis-plus-boot-starter</artifactId> | ||||||
|         </dependency> |         </dependency> | ||||||
|         <dependency> |  | ||||||
|             <groupId>com.baomidou</groupId> |  | ||||||
|             <artifactId>mybatis-plus-extension</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <!-- dynamic-datasource 多数据源--> |         <!-- dynamic-datasource 多数据源--> | ||||||
|         <dependency> |         <dependency> | ||||||
| @@ -132,19 +111,24 @@ | |||||||
|             <artifactId>hutool-extra</artifactId> |             <artifactId>hutool-extra</artifactId> | ||||||
|         </dependency> |         </dependency> | ||||||
|  |  | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>com.sun.mail</groupId> | ||||||
|  |             <artifactId>jakarta.mail</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|         <dependency> |         <dependency> | ||||||
|             <groupId>org.projectlombok</groupId> |             <groupId>org.projectlombok</groupId> | ||||||
|             <artifactId>lombok</artifactId> |             <artifactId>lombok</artifactId> | ||||||
|         </dependency> |         </dependency> | ||||||
|  |  | ||||||
|         <dependency> |         <dependency> | ||||||
|             <groupId>com.github.xiaoymin</groupId> |             <groupId>org.springdoc</groupId> | ||||||
|             <artifactId>knife4j-spring-boot-starter</artifactId> |             <artifactId>springdoc-openapi-webmvc-core</artifactId> | ||||||
|         </dependency> |         </dependency> | ||||||
|  |  | ||||||
|         <dependency> |         <dependency> | ||||||
|             <groupId>io.swagger</groupId> |             <groupId>org.springdoc</groupId> | ||||||
|             <artifactId>swagger-annotations</artifactId> |             <artifactId>springdoc-openapi-javadoc</artifactId> | ||||||
|         </dependency> |         </dependency> | ||||||
|  |  | ||||||
|         <!--  自动生成YML配置关联JSON文件  --> |         <!--  自动生成YML配置关联JSON文件  --> | ||||||
|   | |||||||
| @@ -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 |  * @author Lion Li | ||||||
|  * @version 3.5.0 |  * @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"; | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,27 @@ | |||||||
|  | package com.ruoyi.common.annotation; | ||||||
|  |  | ||||||
|  | import com.fasterxml.jackson.annotation.JacksonAnnotationsInside; | ||||||
|  | import com.fasterxml.jackson.databind.annotation.JsonSerialize; | ||||||
|  | import com.ruoyi.common.jackson.DictDataJsonSerializer; | ||||||
|  |  | ||||||
|  | import java.lang.annotation.ElementType; | ||||||
|  | import java.lang.annotation.Retention; | ||||||
|  | import java.lang.annotation.RetentionPolicy; | ||||||
|  | import java.lang.annotation.Target; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 字典数据映射注解 | ||||||
|  |  * | ||||||
|  |  * @author itino | ||||||
|  |  */ | ||||||
|  | @Retention(RetentionPolicy.RUNTIME) | ||||||
|  | @Target({ElementType.FIELD, ElementType.METHOD}) | ||||||
|  | @JacksonAnnotationsInside | ||||||
|  | @JsonSerialize(using = DictDataJsonSerializer.class) | ||||||
|  | public @interface DictDataMapper { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 设置字典的type值 (如: sys_user_sex) | ||||||
|  |      */ | ||||||
|  |     String dictType() default ""; | ||||||
|  | } | ||||||
| @@ -1,36 +1,36 @@ | |||||||
| package com.ruoyi.common.annotation; | package com.ruoyi.common.annotation; | ||||||
|  |  | ||||||
| import com.ruoyi.common.constant.Constants; | import com.ruoyi.common.constant.CacheConstants; | ||||||
| import com.ruoyi.common.enums.LimitType; | import com.ruoyi.common.enums.LimitType; | ||||||
|  |  | ||||||
| import java.lang.annotation.*; | import java.lang.annotation.*; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 限流注解 |  * 限流注解 | ||||||
|  * |  * | ||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Target(ElementType.METHOD) | @Target(ElementType.METHOD) | ||||||
| @Retention(RetentionPolicy.RUNTIME) | @Retention(RetentionPolicy.RUNTIME) | ||||||
| @Documented | @Documented | ||||||
| public @interface RateLimiter { | public @interface RateLimiter { | ||||||
|     /** |     /** | ||||||
|      * 限流key |      * 限流key | ||||||
|      */ |      */ | ||||||
|     String key() default Constants.RATE_LIMIT_KEY; |     String key() default CacheConstants.RATE_LIMIT_KEY; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 限流时间,单位秒 |      * 限流时间,单位秒 | ||||||
|      */ |      */ | ||||||
|     int time() default 60; |     int time() default 60; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 限流次数 |      * 限流次数 | ||||||
|      */ |      */ | ||||||
|     int count() default 100; |     int count() default 100; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 限流类型 |      * 限流类型 | ||||||
|      */ |      */ | ||||||
|     LimitType limitType() default LimitType.DEFAULT; |     LimitType limitType() default LimitType.DEFAULT; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -22,8 +22,8 @@ public @interface RepeatSubmit { | |||||||
|     TimeUnit timeUnit() default TimeUnit.MILLISECONDS; |     TimeUnit timeUnit() default TimeUnit.MILLISECONDS; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 提示消息 |      * 提示消息 支持国际化 格式为 {code} | ||||||
|      */ |      */ | ||||||
|     String message() default "不允许重复提交,请稍候再试"; |     String message() default "{repeat.submit.message}"; | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -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"; |     String GBK = "GBK"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * www主域 | ||||||
|  |      */ | ||||||
|  |     String WWW = "www."; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * http请求 |      * http请求 | ||||||
|      */ |      */ | ||||||
| @@ -57,70 +62,15 @@ public interface Constants { | |||||||
|      */ |      */ | ||||||
|     String LOGIN_FAIL = "Error"; |     String LOGIN_FAIL = "Error"; | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 验证码 redis key |  | ||||||
|      */ |  | ||||||
|     String CAPTCHA_CODE_KEY = "captcha_codes:"; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 登录用户 redis key |  | ||||||
|      */ |  | ||||||
|     public static final String LOGIN_TOKEN_KEY = "Authorization:login:token:"; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 在线用户 redis key |  | ||||||
|      */ |  | ||||||
|     public static final 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; |     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 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:"; |  | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -0,0 +1,93 @@ | |||||||
|  | package com.ruoyi.common.constant; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 返回状态码 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | public interface HttpStatus { | ||||||
|  |     /** | ||||||
|  |      * 操作成功 | ||||||
|  |      */ | ||||||
|  |     int SUCCESS = 200; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 对象创建成功 | ||||||
|  |      */ | ||||||
|  |     int CREATED = 201; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 请求已经被接受 | ||||||
|  |      */ | ||||||
|  |     int ACCEPTED = 202; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 操作已经执行成功,但是没有返回数据 | ||||||
|  |      */ | ||||||
|  |     int NO_CONTENT = 204; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 资源已被移除 | ||||||
|  |      */ | ||||||
|  |     int MOVED_PERM = 301; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 重定向 | ||||||
|  |      */ | ||||||
|  |     int SEE_OTHER = 303; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 资源没有被修改 | ||||||
|  |      */ | ||||||
|  |     int NOT_MODIFIED = 304; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 参数列表错误(缺少,格式不匹配) | ||||||
|  |      */ | ||||||
|  |     int BAD_REQUEST = 400; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 未授权 | ||||||
|  |      */ | ||||||
|  |     int UNAUTHORIZED = 401; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 访问受限,授权过期 | ||||||
|  |      */ | ||||||
|  |     int FORBIDDEN = 403; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 资源,服务未找到 | ||||||
|  |      */ | ||||||
|  |     int NOT_FOUND = 404; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 不允许的http方法 | ||||||
|  |      */ | ||||||
|  |     int BAD_METHOD = 405; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 资源冲突,或者资源被锁 | ||||||
|  |      */ | ||||||
|  |     int CONFLICT = 409; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 不支持的数据,媒体类型 | ||||||
|  |      */ | ||||||
|  |     int UNSUPPORTED_TYPE = 415; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 系统内部错误 | ||||||
|  |      */ | ||||||
|  |     int ERROR = 500; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 接口未实现 | ||||||
|  |      */ | ||||||
|  |     int NOT_IMPLEMENTED = 501; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 系统警告消息 | ||||||
|  |      */ | ||||||
|  |     int WARN = 601; | ||||||
|  | } | ||||||
| @@ -22,11 +22,21 @@ public interface UserConstants { | |||||||
|      */ |      */ | ||||||
|     String EXCEPTION = "1"; |     String EXCEPTION = "1"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 用户正常状态 | ||||||
|  |      */ | ||||||
|  |     String USER_NORMAL = "0"; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 用户封禁状态 |      * 用户封禁状态 | ||||||
|      */ |      */ | ||||||
|     String USER_DISABLE = "1"; |     String USER_DISABLE = "1"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 角色正常状态 | ||||||
|  |      */ | ||||||
|  |     String ROLE_NORMAL = "0"; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 角色封禁状态 |      * 角色封禁状态 | ||||||
|      */ |      */ | ||||||
| @@ -62,6 +72,16 @@ public interface UserConstants { | |||||||
|      */ |      */ | ||||||
|     String NO_FRAME = "1"; |     String NO_FRAME = "1"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 菜单正常状态 | ||||||
|  |      */ | ||||||
|  |     String MENU_NORMAL = "0"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 菜单停用状态 | ||||||
|  |      */ | ||||||
|  |     String MENU_DISABLE = "1"; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 菜单类型(目录) |      * 菜单类型(目录) | ||||||
|      */ |      */ | ||||||
|   | |||||||
| @@ -2,7 +2,8 @@ package com.ruoyi.common.core.domain; | |||||||
|  |  | ||||||
| import com.baomidou.mybatisplus.annotation.FieldFill; | import com.baomidou.mybatisplus.annotation.FieldFill; | ||||||
| import com.baomidou.mybatisplus.annotation.TableField; | import com.baomidou.mybatisplus.annotation.TableField; | ||||||
| import io.swagger.annotations.ApiModelProperty; | import com.fasterxml.jackson.annotation.JsonIgnore; | ||||||
|  | import com.fasterxml.jackson.annotation.JsonInclude; | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
|  |  | ||||||
| import java.io.Serializable; | import java.io.Serializable; | ||||||
| @@ -24,42 +25,38 @@ public class BaseEntity implements Serializable { | |||||||
|     /** |     /** | ||||||
|      * 搜索值 |      * 搜索值 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "搜索值") |     @JsonIgnore | ||||||
|     @TableField(exist = false) |     @TableField(exist = false) | ||||||
|     private String searchValue; |     private String searchValue; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 创建者 |      * 创建者 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "创建者") |  | ||||||
|     @TableField(fill = FieldFill.INSERT) |     @TableField(fill = FieldFill.INSERT) | ||||||
|     private String createBy; |     private String createBy; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 创建时间 |      * 创建时间 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "创建时间") |  | ||||||
|     @TableField(fill = FieldFill.INSERT) |     @TableField(fill = FieldFill.INSERT) | ||||||
|     private Date createTime; |     private Date createTime; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 更新者 |      * 更新者 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "更新者") |  | ||||||
|     @TableField(fill = FieldFill.INSERT_UPDATE) |     @TableField(fill = FieldFill.INSERT_UPDATE) | ||||||
|     private String updateBy; |     private String updateBy; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 更新时间 |      * 更新时间 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "更新时间") |  | ||||||
|     @TableField(fill = FieldFill.INSERT_UPDATE) |     @TableField(fill = FieldFill.INSERT_UPDATE) | ||||||
|     private Date updateTime; |     private Date updateTime; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 请求参数 |      * 请求参数 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "请求参数") |     @JsonInclude(JsonInclude.Include.NON_EMPTY) | ||||||
|     @TableField(exist = false) |     @TableField(exist = false) | ||||||
|     private Map<String, Object> params = new HashMap<>(); |     private Map<String, Object> params = new HashMap<>(); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,14 +1,17 @@ | |||||||
| package com.ruoyi.common.core.domain; | package com.ruoyi.common.core.domain; | ||||||
|  |  | ||||||
|  | import cn.hutool.core.collection.CollUtil; | ||||||
| import cn.hutool.core.util.ObjectUtil; | import cn.hutool.core.util.ObjectUtil; | ||||||
| import com.baomidou.mybatisplus.core.metadata.OrderItem; | import com.baomidou.mybatisplus.core.metadata.OrderItem; | ||||||
| import com.baomidou.mybatisplus.extension.plugins.pagination.Page; | import com.baomidou.mybatisplus.extension.plugins.pagination.Page; | ||||||
|  | import com.ruoyi.common.exception.ServiceException; | ||||||
| import com.ruoyi.common.utils.StringUtils; | import com.ruoyi.common.utils.StringUtils; | ||||||
| import com.ruoyi.common.utils.sql.SqlUtil; | import com.ruoyi.common.utils.sql.SqlUtil; | ||||||
| import io.swagger.annotations.ApiModelProperty; |  | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
|  |  | ||||||
| import java.io.Serializable; | import java.io.Serializable; | ||||||
|  | import java.util.ArrayList; | ||||||
|  | import java.util.List; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 分页查询实体类 |  * 分页查询实体类 | ||||||
| @@ -24,25 +27,21 @@ public class PageQuery implements Serializable { | |||||||
|     /** |     /** | ||||||
|      * 分页大小 |      * 分页大小 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty("分页大小") |  | ||||||
|     private Integer pageSize; |     private Integer pageSize; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 当前页数 |      * 当前页数 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty("当前页数") |  | ||||||
|     private Integer pageNum; |     private Integer pageNum; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 排序列 |      * 排序列 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty("排序列") |  | ||||||
|     private String orderByColumn; |     private String orderByColumn; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 排序的方向desc或者asc |      * 排序的方向desc或者asc | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "排序的方向", example = "asc,desc") |  | ||||||
|     private String isAsc; |     private String isAsc; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -62,30 +61,52 @@ public class PageQuery implements Serializable { | |||||||
|             pageNum = DEFAULT_PAGE_NUM; |             pageNum = DEFAULT_PAGE_NUM; | ||||||
|         } |         } | ||||||
|         Page<T> page = new Page<>(pageNum, pageSize); |         Page<T> page = new Page<>(pageNum, pageSize); | ||||||
|         OrderItem orderItem = buildOrderItem(); |         List<OrderItem> orderItems = buildOrderItem(); | ||||||
|         if (ObjectUtil.isNotNull(orderItem)) { |         if (CollUtil.isNotEmpty(orderItems)) { | ||||||
|             page.addOrder(orderItem); |             page.addOrder(orderItems); | ||||||
|         } |         } | ||||||
|         return page; |         return page; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private OrderItem buildOrderItem() { |     /** | ||||||
|         // 兼容前端排序类型 |      * 构建排序 | ||||||
|         if ("ascending".equals(isAsc)) { |      * | ||||||
|             isAsc = "asc"; |      * 支持的用法如下: | ||||||
|         } else if ("descending".equals(isAsc)) { |      * {isAsc:"asc",orderByColumn:"id"} order by id asc | ||||||
|             isAsc = "desc"; |      * {isAsc:"asc",orderByColumn:"id,createTime"} order by id asc,create_time asc | ||||||
|  |      * {isAsc:"desc",orderByColumn:"id,createTime"} order by id desc,create_time desc | ||||||
|  |      * {isAsc:"asc,desc",orderByColumn:"id,createTime"} order by id asc,create_time desc | ||||||
|  |      */ | ||||||
|  |     private List<OrderItem> buildOrderItem() { | ||||||
|  |         if (StringUtils.isBlank(orderByColumn) || StringUtils.isBlank(isAsc)) { | ||||||
|  |             return null; | ||||||
|         } |         } | ||||||
|         if (StringUtils.isNotBlank(orderByColumn)) { |         String orderBy = SqlUtil.escapeOrderBySql(orderByColumn); | ||||||
|             String orderBy = SqlUtil.escapeOrderBySql(orderByColumn); |         orderBy = StringUtils.toUnderScoreCase(orderBy); | ||||||
|             orderBy = StringUtils.toUnderScoreCase(orderBy); |  | ||||||
|             if ("asc".equals(isAsc)) { |         // 兼容前端排序类型 | ||||||
|                 return OrderItem.asc(orderBy); |         isAsc = StringUtils.replaceEach(isAsc, new String[]{"ascending", "descending"}, new String[]{"asc", "desc"}); | ||||||
|             } else if ("desc".equals(isAsc)) { |  | ||||||
|                 return OrderItem.desc(orderBy); |         String[] orderByArr = orderBy.split(","); | ||||||
|  |         String[] isAscArr = isAsc.split(","); | ||||||
|  |         if (isAscArr.length != 1 && isAscArr.length != orderByArr.length) { | ||||||
|  |             throw new ServiceException("排序参数有误"); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         List<OrderItem> list = new ArrayList<>(); | ||||||
|  |         // 每个字段各自排序 | ||||||
|  |         for (int i = 0; i < orderByArr.length; i++) { | ||||||
|  |             String orderByStr = orderByArr[i]; | ||||||
|  |             String isAscStr = isAscArr.length == 1 ? isAscArr[0] : isAscArr[i]; | ||||||
|  |             if ("asc".equals(isAscStr)) { | ||||||
|  |                 list.add(OrderItem.asc(orderByStr)); | ||||||
|  |             } else if ("desc".equals(isAscStr)) { | ||||||
|  |                 list.add(OrderItem.desc(orderByStr)); | ||||||
|  |             } else { | ||||||
|  |                 throw new ServiceException("排序参数有误"); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         return null; |         return list; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,7 +1,6 @@ | |||||||
| package com.ruoyi.common.core.domain; | package com.ruoyi.common.core.domain; | ||||||
|  |  | ||||||
| import io.swagger.annotations.ApiModel; | import com.ruoyi.common.constant.HttpStatus; | ||||||
| import io.swagger.annotations.ApiModelProperty; |  | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
|  |  | ||||||
| @@ -14,7 +13,6 @@ import java.io.Serializable; | |||||||
|  */ |  */ | ||||||
| @Data | @Data | ||||||
| @NoArgsConstructor | @NoArgsConstructor | ||||||
| @ApiModel("请求响应对象") |  | ||||||
| public class R<T> implements Serializable { | public class R<T> implements Serializable { | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
|  |  | ||||||
| @@ -28,13 +26,10 @@ public class R<T> implements Serializable { | |||||||
|      */ |      */ | ||||||
|     public static final int FAIL = 500; |     public static final int FAIL = 500; | ||||||
|  |  | ||||||
|     @ApiModelProperty("消息状态码") |  | ||||||
|     private int code; |     private int code; | ||||||
|  |  | ||||||
|     @ApiModelProperty("消息内容") |  | ||||||
|     private String msg; |     private String msg; | ||||||
|  |  | ||||||
|     @ApiModelProperty("数据对象") |  | ||||||
|     private T data; |     private T data; | ||||||
|  |  | ||||||
|     public static <T> R<T> ok() { |     public static <T> R<T> ok() { | ||||||
| @@ -73,6 +68,27 @@ public class R<T> implements Serializable { | |||||||
|         return restResult(null, code, msg); |         return restResult(null, code, msg); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 返回警告消息 | ||||||
|  |      * | ||||||
|  |      * @param msg 返回内容 | ||||||
|  |      * @return 警告消息 | ||||||
|  |      */ | ||||||
|  |     public static <T> R<T> warn(String msg) { | ||||||
|  |         return restResult(null, HttpStatus.WARN, msg); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 返回警告消息 | ||||||
|  |      * | ||||||
|  |      * @param msg 返回内容 | ||||||
|  |      * @param data 数据对象 | ||||||
|  |      * @return 警告消息 | ||||||
|  |      */ | ||||||
|  |     public static <T> R<T> warn(String msg, T data) { | ||||||
|  |         return restResult(data, HttpStatus.WARN, msg); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     private static <T> R<T> restResult(T data, int code, String msg) { |     private static <T> R<T> restResult(T data, int code, String msg) { | ||||||
|         R<T> r = new R<>(); |         R<T> r = new R<>(); | ||||||
|         r.setCode(code); |         r.setCode(code); | ||||||
| @@ -81,4 +97,11 @@ public class R<T> implements Serializable { | |||||||
|         return r; |         return r; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     public static <T> Boolean isError(R<T> ret) { | ||||||
|  |         return !isSuccess(ret); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static <T> Boolean isSuccess(R<T> ret) { | ||||||
|  |         return R.SUCCESS == ret.getCode(); | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,7 +1,6 @@ | |||||||
| package com.ruoyi.common.core.domain; | package com.ruoyi.common.core.domain; | ||||||
|  |  | ||||||
| import com.baomidou.mybatisplus.annotation.TableField; | import com.baomidou.mybatisplus.annotation.TableField; | ||||||
| import io.swagger.annotations.ApiModelProperty; |  | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.EqualsAndHashCode; | import lombok.EqualsAndHashCode; | ||||||
|  |  | ||||||
| @@ -16,7 +15,7 @@ import java.util.List; | |||||||
|  |  | ||||||
| @Data | @Data | ||||||
| @EqualsAndHashCode(callSuper = true) | @EqualsAndHashCode(callSuper = true) | ||||||
| public class TreeEntity extends BaseEntity { | public class TreeEntity<T> extends BaseEntity { | ||||||
|  |  | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
|  |  | ||||||
| @@ -24,20 +23,17 @@ public class TreeEntity extends BaseEntity { | |||||||
|      * 父菜单名称 |      * 父菜单名称 | ||||||
|      */ |      */ | ||||||
|     @TableField(exist = false) |     @TableField(exist = false) | ||||||
|     @ApiModelProperty(value = "父菜单名称") |  | ||||||
|     private String parentName; |     private String parentName; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 父菜单ID |      * 父菜单ID | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "父菜单ID") |  | ||||||
|     private Long parentId; |     private Long parentId; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 子部门 |      * 子部门 | ||||||
|      */ |      */ | ||||||
|     @TableField(exist = false) |     @TableField(exist = false) | ||||||
|     @ApiModelProperty(value = "子部门") |     private List<T> children = new ArrayList<>(); | ||||||
|     private List<?> children = new ArrayList<>(); |  | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,6 +1,5 @@ | |||||||
| package com.ruoyi.common.core.domain.dto; | package com.ruoyi.common.core.domain.dto; | ||||||
|  |  | ||||||
| import lombok.Builder; |  | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -4,8 +4,6 @@ import com.baomidou.mybatisplus.annotation.TableId; | |||||||
| import com.baomidou.mybatisplus.annotation.TableLogic; | import com.baomidou.mybatisplus.annotation.TableLogic; | ||||||
| import com.baomidou.mybatisplus.annotation.TableName; | import com.baomidou.mybatisplus.annotation.TableName; | ||||||
| import com.ruoyi.common.core.domain.TreeEntity; | import com.ruoyi.common.core.domain.TreeEntity; | ||||||
| import io.swagger.annotations.ApiModel; |  | ||||||
| import io.swagger.annotations.ApiModelProperty; |  | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.EqualsAndHashCode; | import lombok.EqualsAndHashCode; | ||||||
|  |  | ||||||
| @@ -23,21 +21,18 @@ import javax.validation.constraints.Size; | |||||||
| @Data | @Data | ||||||
| @EqualsAndHashCode(callSuper = true) | @EqualsAndHashCode(callSuper = true) | ||||||
| @TableName("sys_dept") | @TableName("sys_dept") | ||||||
| @ApiModel("部门业务对象") | public class SysDept extends TreeEntity<SysDept> { | ||||||
| public class SysDept extends TreeEntity { |  | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 部门ID |      * 部门ID | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "部门id") |  | ||||||
|     @TableId(value = "dept_id") |     @TableId(value = "dept_id") | ||||||
|     private Long deptId; |     private Long deptId; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 部门名称 |      * 部门名称 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "部门名称") |  | ||||||
|     @NotBlank(message = "部门名称不能为空") |     @NotBlank(message = "部门名称不能为空") | ||||||
|     @Size(min = 0, max = 30, message = "部门名称长度不能超过30个字符") |     @Size(min = 0, max = 30, message = "部门名称长度不能超过30个字符") | ||||||
|     private String deptName; |     private String deptName; | ||||||
| @@ -45,27 +40,23 @@ public class SysDept extends TreeEntity { | |||||||
|     /** |     /** | ||||||
|      * 显示顺序 |      * 显示顺序 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "显示顺序") |  | ||||||
|     @NotNull(message = "显示顺序不能为空") |     @NotNull(message = "显示顺序不能为空") | ||||||
|     private Long orderNum; |     private Integer orderNum; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 负责人 |      * 负责人 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "负责人") |  | ||||||
|     private String leader; |     private String leader; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 联系电话 |      * 联系电话 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "联系电话") |  | ||||||
|     @Size(min = 0, max = 11, message = "联系电话长度不能超过11个字符") |     @Size(min = 0, max = 11, message = "联系电话长度不能超过11个字符") | ||||||
|     private String phone; |     private String phone; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 邮箱 |      * 邮箱 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "邮箱") |  | ||||||
|     @Email(message = "邮箱格式不正确") |     @Email(message = "邮箱格式不正确") | ||||||
|     @Size(min = 0, max = 50, message = "邮箱长度不能超过50个字符") |     @Size(min = 0, max = 50, message = "邮箱长度不能超过50个字符") | ||||||
|     private String email; |     private String email; | ||||||
| @@ -73,20 +64,17 @@ public class SysDept extends TreeEntity { | |||||||
|     /** |     /** | ||||||
|      * 部门状态:0正常,1停用 |      * 部门状态:0正常,1停用 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "部门状态:0正常,1停用") |  | ||||||
|     private String status; |     private String status; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 删除标志(0代表存在 2代表删除) |      * 删除标志(0代表存在 2代表删除) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "删除标志(0代表存在 2代表删除)") |  | ||||||
|     @TableLogic |     @TableLogic | ||||||
|     private String delFlag; |     private String delFlag; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 祖级列表 |      * 祖级列表 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "祖级列表") |  | ||||||
|     private String ancestors; |     private String ancestors; | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,8 +8,6 @@ import com.ruoyi.common.annotation.ExcelDictFormat; | |||||||
| import com.ruoyi.common.constant.UserConstants; | import com.ruoyi.common.constant.UserConstants; | ||||||
| import com.ruoyi.common.convert.ExcelDictConvert; | import com.ruoyi.common.convert.ExcelDictConvert; | ||||||
| import com.ruoyi.common.core.domain.BaseEntity; | import com.ruoyi.common.core.domain.BaseEntity; | ||||||
| import io.swagger.annotations.ApiModel; |  | ||||||
| import io.swagger.annotations.ApiModelProperty; |  | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.EqualsAndHashCode; | import lombok.EqualsAndHashCode; | ||||||
|  |  | ||||||
| @@ -26,13 +24,11 @@ import javax.validation.constraints.Size; | |||||||
| @EqualsAndHashCode(callSuper = true) | @EqualsAndHashCode(callSuper = true) | ||||||
| @TableName("sys_dict_data") | @TableName("sys_dict_data") | ||||||
| @ExcelIgnoreUnannotated | @ExcelIgnoreUnannotated | ||||||
| @ApiModel("字典数据业务对象") |  | ||||||
| public class SysDictData extends BaseEntity { | public class SysDictData extends BaseEntity { | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 字典编码 |      * 字典编码 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "字典编码") |  | ||||||
|     @ExcelProperty(value = "字典编码") |     @ExcelProperty(value = "字典编码") | ||||||
|     @TableId(value = "dict_code") |     @TableId(value = "dict_code") | ||||||
|     private Long dictCode; |     private Long dictCode; | ||||||
| @@ -40,14 +36,12 @@ public class SysDictData extends BaseEntity { | |||||||
|     /** |     /** | ||||||
|      * 字典排序 |      * 字典排序 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "字典排序") |  | ||||||
|     @ExcelProperty(value = "字典排序") |     @ExcelProperty(value = "字典排序") | ||||||
|     private Long dictSort; |     private Integer dictSort; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 字典标签 |      * 字典标签 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "字典标签") |  | ||||||
|     @ExcelProperty(value = "字典标签") |     @ExcelProperty(value = "字典标签") | ||||||
|     @NotBlank(message = "字典标签不能为空") |     @NotBlank(message = "字典标签不能为空") | ||||||
|     @Size(min = 0, max = 100, message = "字典标签长度不能超过100个字符") |     @Size(min = 0, max = 100, message = "字典标签长度不能超过100个字符") | ||||||
| @@ -56,7 +50,6 @@ public class SysDictData extends BaseEntity { | |||||||
|     /** |     /** | ||||||
|      * 字典键值 |      * 字典键值 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "字典键值") |  | ||||||
|     @ExcelProperty(value = "字典键值") |     @ExcelProperty(value = "字典键值") | ||||||
|     @NotBlank(message = "字典键值不能为空") |     @NotBlank(message = "字典键值不能为空") | ||||||
|     @Size(min = 0, max = 100, message = "字典键值长度不能超过100个字符") |     @Size(min = 0, max = 100, message = "字典键值长度不能超过100个字符") | ||||||
| @@ -65,7 +58,6 @@ public class SysDictData extends BaseEntity { | |||||||
|     /** |     /** | ||||||
|      * 字典类型 |      * 字典类型 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "字典类型") |  | ||||||
|     @ExcelProperty(value = "字典类型") |     @ExcelProperty(value = "字典类型") | ||||||
|     @NotBlank(message = "字典类型不能为空") |     @NotBlank(message = "字典类型不能为空") | ||||||
|     @Size(min = 0, max = 100, message = "字典类型长度不能超过100个字符") |     @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个字符") |     @Size(min = 0, max = 100, message = "样式属性长度不能超过100个字符") | ||||||
|     private String cssClass; |     private String cssClass; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 表格字典样式 |      * 表格字典样式 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "表格字典样式") |  | ||||||
|     private String listClass; |     private String listClass; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 是否默认(Y是 N否) |      * 是否默认(Y是 N否) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "是否默认(Y是 N否)") |  | ||||||
|     @ExcelProperty(value = "是否默认", converter = ExcelDictConvert.class) |     @ExcelProperty(value = "是否默认", converter = ExcelDictConvert.class) | ||||||
|     @ExcelDictFormat(dictType = "sys_yes_no") |     @ExcelDictFormat(dictType = "sys_yes_no") | ||||||
|     private String isDefault; |     private String isDefault; | ||||||
| @@ -95,7 +84,6 @@ public class SysDictData extends BaseEntity { | |||||||
|     /** |     /** | ||||||
|      * 状态(0正常 1停用) |      * 状态(0正常 1停用) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "状态(0正常 1停用)") |  | ||||||
|     @ExcelProperty(value = "状态", converter = ExcelDictConvert.class) |     @ExcelProperty(value = "状态", converter = ExcelDictConvert.class) | ||||||
|     @ExcelDictFormat(dictType = "sys_normal_disable") |     @ExcelDictFormat(dictType = "sys_normal_disable") | ||||||
|     private String status; |     private String status; | ||||||
| @@ -103,7 +91,6 @@ public class SysDictData extends BaseEntity { | |||||||
|     /** |     /** | ||||||
|      * 备注 |      * 备注 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "备注") |  | ||||||
|     private String remark; |     private String remark; | ||||||
|  |  | ||||||
|     public boolean getDefault() { |     public boolean getDefault() { | ||||||
|   | |||||||
| @@ -7,13 +7,11 @@ import com.baomidou.mybatisplus.annotation.TableName; | |||||||
| import com.ruoyi.common.annotation.ExcelDictFormat; | import com.ruoyi.common.annotation.ExcelDictFormat; | ||||||
| import com.ruoyi.common.convert.ExcelDictConvert; | import com.ruoyi.common.convert.ExcelDictConvert; | ||||||
| import com.ruoyi.common.core.domain.BaseEntity; | import com.ruoyi.common.core.domain.BaseEntity; | ||||||
| import io.swagger.annotations.ApiModel; |  | ||||||
| import io.swagger.annotations.ApiModelProperty; |  | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.EqualsAndHashCode; | import lombok.EqualsAndHashCode; | ||||||
| import lombok.experimental.Accessors; |  | ||||||
|  |  | ||||||
| import javax.validation.constraints.NotBlank; | import javax.validation.constraints.NotBlank; | ||||||
|  | import javax.validation.constraints.Pattern; | ||||||
| import javax.validation.constraints.Size; | import javax.validation.constraints.Size; | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -26,13 +24,11 @@ import javax.validation.constraints.Size; | |||||||
| @EqualsAndHashCode(callSuper = true) | @EqualsAndHashCode(callSuper = true) | ||||||
| @TableName("sys_dict_type") | @TableName("sys_dict_type") | ||||||
| @ExcelIgnoreUnannotated | @ExcelIgnoreUnannotated | ||||||
| @ApiModel("字典类型业务对象") |  | ||||||
| public class SysDictType extends BaseEntity { | public class SysDictType extends BaseEntity { | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 字典主键 |      * 字典主键 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "字典主键") |  | ||||||
|     @ExcelProperty(value = "字典主键") |     @ExcelProperty(value = "字典主键") | ||||||
|     @TableId(value = "dict_id") |     @TableId(value = "dict_id") | ||||||
|     private Long dictId; |     private Long dictId; | ||||||
| @@ -40,7 +36,6 @@ public class SysDictType extends BaseEntity { | |||||||
|     /** |     /** | ||||||
|      * 字典名称 |      * 字典名称 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "字典名称") |  | ||||||
|     @ExcelProperty(value = "字典名称") |     @ExcelProperty(value = "字典名称") | ||||||
|     @NotBlank(message = "字典名称不能为空") |     @NotBlank(message = "字典名称不能为空") | ||||||
|     @Size(min = 0, max = 100, message = "字典类型名称长度不能超过100个字符") |     @Size(min = 0, max = 100, message = "字典类型名称长度不能超过100个字符") | ||||||
| @@ -49,16 +44,15 @@ public class SysDictType extends BaseEntity { | |||||||
|     /** |     /** | ||||||
|      * 字典类型 |      * 字典类型 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "字典类型") |  | ||||||
|     @ExcelProperty(value = "字典类型") |     @ExcelProperty(value = "字典类型") | ||||||
|     @NotBlank(message = "字典类型不能为空") |     @NotBlank(message = "字典类型不能为空") | ||||||
|     @Size(min = 0, max = 100, message = "字典类型类型长度不能超过100个字符") |     @Size(min = 0, max = 100, message = "字典类型类型长度不能超过100个字符") | ||||||
|  |     @Pattern(regexp = "^[a-z][a-z0-9_]*$", message = "字典类型必须以字母开头,且只能为(小写字母,数字,下滑线)") | ||||||
|     private String dictType; |     private String dictType; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 状态(0正常 1停用) |      * 状态(0正常 1停用) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "状态(0正常 1停用)") |  | ||||||
|     @ExcelProperty(value = "状态", converter = ExcelDictConvert.class) |     @ExcelProperty(value = "状态", converter = ExcelDictConvert.class) | ||||||
|     @ExcelDictFormat(dictType = "sys_normal_disable") |     @ExcelDictFormat(dictType = "sys_normal_disable") | ||||||
|     private String status; |     private String status; | ||||||
| @@ -66,7 +60,6 @@ public class SysDictType extends BaseEntity { | |||||||
|     /** |     /** | ||||||
|      * 备注 |      * 备注 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "备注") |  | ||||||
|     private String remark; |     private String remark; | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,11 +1,9 @@ | |||||||
| package com.ruoyi.common.core.domain.entity; | package com.ruoyi.common.core.domain.entity; | ||||||
|  |  | ||||||
| import com.baomidou.mybatisplus.annotation.TableField; |  | ||||||
| import com.baomidou.mybatisplus.annotation.TableId; | import com.baomidou.mybatisplus.annotation.TableId; | ||||||
| import com.baomidou.mybatisplus.annotation.TableName; | import com.baomidou.mybatisplus.annotation.TableName; | ||||||
|  | import com.fasterxml.jackson.annotation.JsonInclude; | ||||||
| import com.ruoyi.common.core.domain.TreeEntity; | import com.ruoyi.common.core.domain.TreeEntity; | ||||||
| import io.swagger.annotations.ApiModel; |  | ||||||
| import io.swagger.annotations.ApiModelProperty; |  | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.EqualsAndHashCode; | import lombok.EqualsAndHashCode; | ||||||
|  |  | ||||||
| @@ -22,20 +20,17 @@ import javax.validation.constraints.Size; | |||||||
| @Data | @Data | ||||||
| @EqualsAndHashCode(callSuper = true) | @EqualsAndHashCode(callSuper = true) | ||||||
| @TableName("sys_menu") | @TableName("sys_menu") | ||||||
| @ApiModel("菜单权限业务对象") | public class SysMenu extends TreeEntity<SysMenu> { | ||||||
| public class SysMenu extends TreeEntity { |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 菜单ID |      * 菜单ID | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "菜单ID") |  | ||||||
|     @TableId(value = "menu_id") |     @TableId(value = "menu_id") | ||||||
|     private Long menuId; |     private Long menuId; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 菜单名称 |      * 菜单名称 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "菜单名称") |  | ||||||
|     @NotBlank(message = "菜单名称不能为空") |     @NotBlank(message = "菜单名称不能为空") | ||||||
|     @Size(min = 0, max = 50, message = "菜单名称长度不能超过50个字符") |     @Size(min = 0, max = 50, message = "菜单名称长度不能超过50个字符") | ||||||
|     private String menuName; |     private String menuName; | ||||||
| @@ -43,79 +38,67 @@ public class SysMenu extends TreeEntity { | |||||||
|     /** |     /** | ||||||
|      * 显示顺序 |      * 显示顺序 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "显示顺序") |  | ||||||
|     @NotNull(message = "显示顺序不能为空") |     @NotNull(message = "显示顺序不能为空") | ||||||
|     private Long orderNum; |     private Integer orderNum; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 路由地址 |      * 路由地址 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "路由地址") |  | ||||||
|     @Size(min = 0, max = 200, message = "路由地址不能超过200个字符") |     @Size(min = 0, max = 200, message = "路由地址不能超过200个字符") | ||||||
|     private String path; |     private String path; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 组件路径 |      * 组件路径 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "组件路径") |  | ||||||
|     @Size(min = 0, max = 200, message = "组件路径不能超过255个字符") |     @Size(min = 0, max = 200, message = "组件路径不能超过255个字符") | ||||||
|     private String component; |     private String component; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 路由参数 |      * 路由参数 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "路由参数") |     private String queryParam; | ||||||
|     @TableField("`query`") |  | ||||||
|     private String query; |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 是否为外链(0是 1否) |      * 是否为外链(0是 1否) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "是否为外链(0是 1否)") |  | ||||||
|     private String isFrame; |     private String isFrame; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 是否缓存(0缓存 1不缓存) |      * 是否缓存(0缓存 1不缓存) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "是否缓存(0缓存 1不缓存)") |  | ||||||
|     private String isCache; |     private String isCache; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 类型(M目录 C菜单 F按钮) |      * 类型(M目录 C菜单 F按钮) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "类型(M目录 C菜单 F按钮)") |  | ||||||
|     @NotBlank(message = "菜单类型不能为空") |     @NotBlank(message = "菜单类型不能为空") | ||||||
|     private String menuType; |     private String menuType; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 显示状态(0显示 1隐藏) |      * 显示状态(0显示 1隐藏) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "显示状态(0显示 1隐藏)") |  | ||||||
|     private String visible; |     private String visible; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 菜单状态(0显示 1隐藏) |      * 菜单状态(0正常 1停用) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "菜单状态(0显示 1隐藏)") |  | ||||||
|     private String status; |     private String status; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 权限字符串 |      * 权限字符串 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "权限字符串") |     @JsonInclude(JsonInclude.Include.NON_NULL) | ||||||
|     @Size(min = 0, max = 100, message = "权限标识长度不能超过100个字符") |     @Size(min = 0, max = 100, message = "权限标识长度不能超过100个字符") | ||||||
|     private String perms; |     private String perms; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 菜单图标 |      * 菜单图标 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "菜单图标") |  | ||||||
|     private String icon; |     private String icon; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 备注 |      * 备注 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "备注") |  | ||||||
|     private String remark; |     private String remark; | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -10,13 +10,14 @@ import com.ruoyi.common.annotation.ExcelDictFormat; | |||||||
| import com.ruoyi.common.constant.UserConstants; | import com.ruoyi.common.constant.UserConstants; | ||||||
| import com.ruoyi.common.convert.ExcelDictConvert; | import com.ruoyi.common.convert.ExcelDictConvert; | ||||||
| import com.ruoyi.common.core.domain.BaseEntity; | import com.ruoyi.common.core.domain.BaseEntity; | ||||||
| import io.swagger.annotations.ApiModelProperty; |  | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.EqualsAndHashCode; | import lombok.EqualsAndHashCode; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
|  |  | ||||||
| import javax.validation.constraints.NotBlank; | import javax.validation.constraints.NotBlank; | ||||||
|  | import javax.validation.constraints.NotNull; | ||||||
| import javax.validation.constraints.Size; | import javax.validation.constraints.Size; | ||||||
|  | import java.util.Set; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 角色表 sys_role |  * 角色表 sys_role | ||||||
| @@ -34,7 +35,6 @@ public class SysRole extends BaseEntity { | |||||||
|     /** |     /** | ||||||
|      * 角色ID |      * 角色ID | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "角色ID") |  | ||||||
|     @ExcelProperty(value = "角色序号") |     @ExcelProperty(value = "角色序号") | ||||||
|     @TableId(value = "role_id") |     @TableId(value = "role_id") | ||||||
|     private Long roleId; |     private Long roleId; | ||||||
| @@ -42,7 +42,6 @@ public class SysRole extends BaseEntity { | |||||||
|     /** |     /** | ||||||
|      * 角色名称 |      * 角色名称 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "角色名称") |  | ||||||
|     @ExcelProperty(value = "角色名称") |     @ExcelProperty(value = "角色名称") | ||||||
|     @NotBlank(message = "角色名称不能为空") |     @NotBlank(message = "角色名称不能为空") | ||||||
|     @Size(min = 0, max = 30, message = "角色名称长度不能超过30个字符") |     @Size(min = 0, max = 30, message = "角色名称长度不能超过30个字符") | ||||||
| @@ -51,7 +50,6 @@ public class SysRole extends BaseEntity { | |||||||
|     /** |     /** | ||||||
|      * 角色权限 |      * 角色权限 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "角色权限") |  | ||||||
|     @ExcelProperty(value = "角色权限") |     @ExcelProperty(value = "角色权限") | ||||||
|     @NotBlank(message = "权限字符不能为空") |     @NotBlank(message = "权限字符不能为空") | ||||||
|     @Size(min = 0, max = 100, message = "权限字符长度不能超过100个字符") |     @Size(min = 0, max = 100, message = "权限字符长度不能超过100个字符") | ||||||
| @@ -60,15 +58,13 @@ public class SysRole extends BaseEntity { | |||||||
|     /** |     /** | ||||||
|      * 角色排序 |      * 角色排序 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "角色排序") |  | ||||||
|     @ExcelProperty(value = "角色排序") |     @ExcelProperty(value = "角色排序") | ||||||
|     @NotBlank(message = "显示顺序不能为空") |     @NotNull(message = "显示顺序不能为空") | ||||||
|     private String roleSort; |     private Integer roleSort; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 数据范围(1:所有数据权限;2:自定义数据权限;3:本部门数据权限;4:本部门及以下数据权限;5:仅本人数据权限) |      * 数据范围(1:所有数据权限;2:自定义数据权限;3:本部门数据权限;4:本部门及以下数据权限;5:仅本人数据权限) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "数据范围(1:所有数据权限;2:自定义数据权限;3:本部门数据权限;4:本部门及以下数据权限;5:仅本人数据权限)") |  | ||||||
|     @ExcelProperty(value = "数据范围", converter = ExcelDictConvert.class) |     @ExcelProperty(value = "数据范围", converter = ExcelDictConvert.class) | ||||||
|     @ExcelDictFormat(readConverterExp = "1=所有数据权限,2=自定义数据权限,3=本部门数据权限,4=本部门及以下数据权限,5=仅本人数据权限") |     @ExcelDictFormat(readConverterExp = "1=所有数据权限,2=自定义数据权限,3=本部门数据权限,4=本部门及以下数据权限,5=仅本人数据权限") | ||||||
|     private String dataScope; |     private String dataScope; | ||||||
| @@ -76,64 +72,60 @@ public class SysRole extends BaseEntity { | |||||||
|     /** |     /** | ||||||
|      * 菜单树选择项是否关联显示( 0:父子不互相关联显示 1:父子互相关联显示) |      * 菜单树选择项是否关联显示( 0:父子不互相关联显示 1:父子互相关联显示) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "菜单树选择项是否关联显示( 0:父子不互相关联显示 1:父子互相关联显示)") |     private Boolean menuCheckStrictly; | ||||||
|     private boolean menuCheckStrictly; |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 部门树选择项是否关联显示(0:父子不互相关联显示 1:父子互相关联显示 ) |      * 部门树选择项是否关联显示(0:父子不互相关联显示 1:父子互相关联显示 ) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "部门树选择项是否关联显示(0:父子不互相关联显示 1:父子互相关联显示 )") |     private Boolean deptCheckStrictly; | ||||||
|     private boolean deptCheckStrictly; |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 角色状态(0正常 1停用) |      * 角色状态(0正常 1停用) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "角色状态(0正常 1停用)") |  | ||||||
|     @ExcelProperty(value = "角色状态", converter = ExcelDictConvert.class) |     @ExcelProperty(value = "角色状态", converter = ExcelDictConvert.class) | ||||||
|     @ExcelDictFormat(dictType = "sys_common_status") |     @ExcelDictFormat(dictType = "sys_normal_disable") | ||||||
|     private String status; |     private String status; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 删除标志(0代表存在 2代表删除) |      * 删除标志(0代表存在 2代表删除) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "删除标志(0代表存在 2代表删除)") |  | ||||||
|     @TableLogic |     @TableLogic | ||||||
|     private String delFlag; |     private String delFlag; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 备注 |      * 备注 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "备注") |  | ||||||
|     private String remark; |     private String remark; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 用户是否存在此角色标识 默认不存在 |      * 用户是否存在此角色标识 默认不存在 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "用户是否存在此角色标识 默认不存在") |  | ||||||
|     @TableField(exist = false) |     @TableField(exist = false) | ||||||
|     private boolean flag = false; |     private boolean flag = false; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 菜单组 |      * 菜单组 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "菜单组") |  | ||||||
|     @TableField(exist = false) |     @TableField(exist = false) | ||||||
|     private Long[] menuIds; |     private Long[] menuIds; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 部门组(数据权限) |      * 部门组(数据权限) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "部门组(数据权限)") |  | ||||||
|     @TableField(exist = false) |     @TableField(exist = false) | ||||||
|     private Long[] deptIds; |     private Long[] deptIds; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 角色菜单权限 | ||||||
|  |      */ | ||||||
|  |     @TableField(exist = false) | ||||||
|  |     private Set<String> permissions; | ||||||
|  |  | ||||||
|     public SysRole(Long roleId) { |     public SysRole(Long roleId) { | ||||||
|         this.roleId = roleId; |         this.roleId = roleId; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiModelProperty(value = "是否管理员") |  | ||||||
|     public boolean isAdmin() { |     public boolean isAdmin() { | ||||||
|         return UserConstants.ADMIN_ID.equals(this.roleId); |         return UserConstants.ADMIN_ID.equals(this.roleId); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,15 +1,11 @@ | |||||||
| package com.ruoyi.common.core.domain.entity; | package com.ruoyi.common.core.domain.entity; | ||||||
|  |  | ||||||
| import com.baomidou.mybatisplus.annotation.*; | 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.annotation.Sensitive; | ||||||
| import com.ruoyi.common.constant.UserConstants; | import com.ruoyi.common.constant.UserConstants; | ||||||
| import com.ruoyi.common.core.domain.BaseEntity; | import com.ruoyi.common.core.domain.BaseEntity; | ||||||
| import com.ruoyi.common.enums.SensitiveStrategy; | import com.ruoyi.common.enums.SensitiveStrategy; | ||||||
| import com.ruoyi.common.xss.Xss; | import com.ruoyi.common.xss.Xss; | ||||||
| import io.swagger.annotations.ApiModel; |  | ||||||
| import io.swagger.annotations.ApiModelProperty; |  | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.EqualsAndHashCode; | import lombok.EqualsAndHashCode; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
| @@ -30,26 +26,22 @@ import java.util.List; | |||||||
| @NoArgsConstructor | @NoArgsConstructor | ||||||
| @EqualsAndHashCode(callSuper = true) | @EqualsAndHashCode(callSuper = true) | ||||||
| @TableName("sys_user") | @TableName("sys_user") | ||||||
| @ApiModel("用户信息业务对象") |  | ||||||
| public class SysUser extends BaseEntity { | public class SysUser extends BaseEntity { | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 用户ID |      * 用户ID | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "用户ID") |  | ||||||
|     @TableId(value = "user_id") |     @TableId(value = "user_id") | ||||||
|     private Long userId; |     private Long userId; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 部门ID |      * 部门ID | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "部门ID") |  | ||||||
|     private Long deptId; |     private Long deptId; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 用户账号 |      * 用户账号 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "用户账号") |  | ||||||
|     @Xss(message = "用户账号不能包含脚本字符") |     @Xss(message = "用户账号不能包含脚本字符") | ||||||
|     @NotBlank(message = "用户账号不能为空") |     @NotBlank(message = "用户账号不能为空") | ||||||
|     @Size(min = 0, max = 30, message = "用户账号长度不能超过30个字符") |     @Size(min = 0, max = 30, message = "用户账号长度不能超过30个字符") | ||||||
| @@ -58,7 +50,6 @@ public class SysUser extends BaseEntity { | |||||||
|     /** |     /** | ||||||
|      * 用户昵称 |      * 用户昵称 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "用户昵称") |  | ||||||
|     @Xss(message = "用户昵称不能包含脚本字符") |     @Xss(message = "用户昵称不能包含脚本字符") | ||||||
|     @Size(min = 0, max = 30, message = "用户昵称长度不能超过30个字符") |     @Size(min = 0, max = 30, message = "用户昵称长度不能超过30个字符") | ||||||
|     private String nickName; |     private String nickName; | ||||||
| @@ -66,14 +57,12 @@ public class SysUser extends BaseEntity { | |||||||
|     /** |     /** | ||||||
|      * 用户类型(sys_user系统用户) |      * 用户类型(sys_user系统用户) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "用户类型") |  | ||||||
|     private String userType; |     private String userType; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 用户邮箱 |      * 用户邮箱 | ||||||
|      */ |      */ | ||||||
|     @Sensitive(strategy = SensitiveStrategy.EMAIL) |     @Sensitive(strategy = SensitiveStrategy.EMAIL) | ||||||
|     @ApiModelProperty(value = "用户邮箱") |  | ||||||
|     @Email(message = "邮箱格式不正确") |     @Email(message = "邮箱格式不正确") | ||||||
|     @Size(min = 0, max = 50, message = "邮箱长度不能超过50个字符") |     @Size(min = 0, max = 50, message = "邮箱长度不能超过50个字符") | ||||||
|     private String email; |     private String email; | ||||||
| @@ -82,25 +71,21 @@ public class SysUser extends BaseEntity { | |||||||
|      * 手机号码 |      * 手机号码 | ||||||
|      */ |      */ | ||||||
|     @Sensitive(strategy = SensitiveStrategy.PHONE) |     @Sensitive(strategy = SensitiveStrategy.PHONE) | ||||||
|     @ApiModelProperty(value = "手机号码") |  | ||||||
|     private String phonenumber; |     private String phonenumber; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 用户性别 |      * 用户性别 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "用户性别") |  | ||||||
|     private String sex; |     private String sex; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 用户头像 |      * 用户头像 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "用户头像") |  | ||||||
|     private String avatar; |     private String avatar; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 密码 |      * 密码 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "密码") |  | ||||||
|     @TableField( |     @TableField( | ||||||
|         insertStrategy = FieldStrategy.NOT_EMPTY, |         insertStrategy = FieldStrategy.NOT_EMPTY, | ||||||
|         updateStrategy = FieldStrategy.NOT_EMPTY, |         updateStrategy = FieldStrategy.NOT_EMPTY, | ||||||
| @@ -108,75 +93,59 @@ public class SysUser extends BaseEntity { | |||||||
|     ) |     ) | ||||||
|     private String password; |     private String password; | ||||||
|  |  | ||||||
|     @JsonIgnore |  | ||||||
|     @JsonProperty |  | ||||||
|     public String getPassword() { |  | ||||||
|         return password; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 帐号状态(0正常 1停用) |      * 帐号状态(0正常 1停用) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "帐号状态(0正常 1停用)") |  | ||||||
|     private String status; |     private String status; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 删除标志(0代表存在 2代表删除) |      * 删除标志(0代表存在 2代表删除) | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "删除标志(0代表存在 2代表删除)") |  | ||||||
|     @TableLogic |     @TableLogic | ||||||
|     private String delFlag; |     private String delFlag; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 最后登录IP |      * 最后登录IP | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "最后登录IP") |  | ||||||
|     private String loginIp; |     private String loginIp; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 最后登录时间 |      * 最后登录时间 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "最后登录时间") |  | ||||||
|     private Date loginDate; |     private Date loginDate; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 备注 |      * 备注 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "备注") |  | ||||||
|     private String remark; |     private String remark; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 部门对象 |      * 部门对象 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "部门对象") |  | ||||||
|     @TableField(exist = false) |     @TableField(exist = false) | ||||||
|     private SysDept dept; |     private SysDept dept; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 角色对象 |      * 角色对象 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "角色对象") |  | ||||||
|     @TableField(exist = false) |     @TableField(exist = false) | ||||||
|     private List<SysRole> roles; |     private List<SysRole> roles; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 角色组 |      * 角色组 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "角色组") |  | ||||||
|     @TableField(exist = false) |     @TableField(exist = false) | ||||||
|     private Long[] roleIds; |     private Long[] roleIds; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 岗位组 |      * 岗位组 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "岗位组") |  | ||||||
|     @TableField(exist = false) |     @TableField(exist = false) | ||||||
|     private Long[] postIds; |     private Long[] postIds; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 数据权限 当前角色ID |      * 数据权限 当前角色ID | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "角色ID") |  | ||||||
|     @TableField(exist = false) |     @TableField(exist = false) | ||||||
|     private Long roleId; |     private Long roleId; | ||||||
|  |  | ||||||
| @@ -184,7 +153,6 @@ public class SysUser extends BaseEntity { | |||||||
|         this.userId = userId; |         this.userId = userId; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiModelProperty(value = "是否管理员") |  | ||||||
|     public boolean isAdmin() { |     public boolean isAdmin() { | ||||||
|         return UserConstants.ADMIN_ID.equals(this.userId); |         return UserConstants.ADMIN_ID.equals(this.userId); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -0,0 +1,44 @@ | |||||||
|  | package com.ruoyi.common.core.domain.event; | ||||||
|  |  | ||||||
|  | import lombok.Data; | ||||||
|  |  | ||||||
|  | import javax.servlet.http.HttpServletRequest; | ||||||
|  | import java.io.Serializable; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 登录事件 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | @Data | ||||||
|  | public class LogininforEvent implements Serializable { | ||||||
|  |  | ||||||
|  |     private static final long serialVersionUID = 1L; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 用户账号 | ||||||
|  |      */ | ||||||
|  |     private String username; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 登录状态 0成功 1失败 | ||||||
|  |      */ | ||||||
|  |     private String status; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 提示消息 | ||||||
|  |      */ | ||||||
|  |     private String message; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 请求体 | ||||||
|  |      */ | ||||||
|  |     private HttpServletRequest request; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 其他参数 | ||||||
|  |      */ | ||||||
|  |     private Object[] args; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| package com.ruoyi.common.core.domain.dto; | package com.ruoyi.common.core.domain.event; | ||||||
| 
 | 
 | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| 
 | 
 | ||||||
| @@ -6,13 +6,13 @@ import java.io.Serializable; | |||||||
| import java.util.Date; | import java.util.Date; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 通用操作日志实体 |  * 操作日志事件 | ||||||
|  * |  * | ||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| @Data | @Data | ||||||
| public class OperLogDTO implements Serializable { | public class OperLogEvent implements Serializable { | ||||||
| 
 | 
 | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
| 
 | 
 | ||||||
| @@ -1,8 +1,6 @@ | |||||||
| package com.ruoyi.common.core.domain.model; | package com.ruoyi.common.core.domain.model; | ||||||
|  |  | ||||||
| import com.ruoyi.common.constant.UserConstants; | import com.ruoyi.common.constant.UserConstants; | ||||||
| import io.swagger.annotations.ApiModel; |  | ||||||
| import io.swagger.annotations.ApiModelProperty; |  | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import org.hibernate.validator.constraints.Length; | import org.hibernate.validator.constraints.Length; | ||||||
|  |  | ||||||
| @@ -15,7 +13,6 @@ import javax.validation.constraints.NotBlank; | |||||||
|  */ |  */ | ||||||
|  |  | ||||||
| @Data | @Data | ||||||
| @ApiModel("用户登录对象") |  | ||||||
| public class LoginBody { | public class LoginBody { | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -23,7 +20,6 @@ public class LoginBody { | |||||||
|      */ |      */ | ||||||
|     @NotBlank(message = "{user.username.not.blank}") |     @NotBlank(message = "{user.username.not.blank}") | ||||||
|     @Length(min = UserConstants.USERNAME_MIN_LENGTH, max = UserConstants.USERNAME_MAX_LENGTH, message = "{user.username.length.valid}") |     @Length(min = UserConstants.USERNAME_MIN_LENGTH, max = UserConstants.USERNAME_MAX_LENGTH, message = "{user.username.length.valid}") | ||||||
|     @ApiModelProperty(value = "用户名") |  | ||||||
|     private String username; |     private String username; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -31,19 +27,16 @@ public class LoginBody { | |||||||
|      */ |      */ | ||||||
|     @NotBlank(message = "{user.password.not.blank}") |     @NotBlank(message = "{user.password.not.blank}") | ||||||
|     @Length(min = UserConstants.PASSWORD_MIN_LENGTH, max = UserConstants.PASSWORD_MAX_LENGTH, message = "{user.password.length.valid}") |     @Length(min = UserConstants.PASSWORD_MIN_LENGTH, max = UserConstants.PASSWORD_MAX_LENGTH, message = "{user.password.length.valid}") | ||||||
|     @ApiModelProperty(value = "用户密码") |  | ||||||
|     private String password; |     private String password; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 验证码 |      * 验证码 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "验证码") |  | ||||||
|     private String code; |     private String code; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 唯一标识 |      * 唯一标识 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty(value = "唯一标识") |  | ||||||
|     private String uuid; |     private String uuid; | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -105,6 +105,12 @@ public class LoginUser implements Serializable { | |||||||
|      * 获取登录id |      * 获取登录id | ||||||
|      */ |      */ | ||||||
|     public String getLoginId() { |     public String getLoginId() { | ||||||
|  |         if (userType == null) { | ||||||
|  |             throw new IllegalArgumentException("用户类型不能为空"); | ||||||
|  |         } | ||||||
|  |         if (userId == null) { | ||||||
|  |             throw new IllegalArgumentException("用户ID不能为空"); | ||||||
|  |         } | ||||||
|         return userType + LoginHelper.JOIN_CODE + userId; |         return userType + LoginHelper.JOIN_CODE + userId; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,7 +1,5 @@ | |||||||
| package com.ruoyi.common.core.domain.model; | package com.ruoyi.common.core.domain.model; | ||||||
|  |  | ||||||
| import io.swagger.annotations.ApiModel; |  | ||||||
| import io.swagger.annotations.ApiModelProperty; |  | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.EqualsAndHashCode; | import lombok.EqualsAndHashCode; | ||||||
|  |  | ||||||
| @@ -12,10 +10,8 @@ import lombok.EqualsAndHashCode; | |||||||
|  */ |  */ | ||||||
| @Data | @Data | ||||||
| @EqualsAndHashCode(callSuper = true) | @EqualsAndHashCode(callSuper = true) | ||||||
| @ApiModel("用户注册对象") |  | ||||||
| public class RegisterBody extends LoginBody { | public class RegisterBody extends LoginBody { | ||||||
|  |  | ||||||
|     @ApiModelProperty(value = "用户类型") |  | ||||||
|     private String userType; |     private String userType; | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,28 @@ | |||||||
|  | package com.ruoyi.common.core.domain.model; | ||||||
|  |  | ||||||
|  | import lombok.Data; | ||||||
|  |  | ||||||
|  | import javax.validation.constraints.NotBlank; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 短信登录对象 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | @Data | ||||||
|  | public class SmsLoginBody { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 用户名 | ||||||
|  |      */ | ||||||
|  |     @NotBlank(message = "{user.phonenumber.not.blank}") | ||||||
|  |     private String phonenumber; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 用户密码 | ||||||
|  |      */ | ||||||
|  |     @NotBlank(message = "{sms.code.not.blank}") | ||||||
|  |     private String smsCode; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,24 @@ | |||||||
|  | package com.ruoyi.common.core.domain.model; | ||||||
|  |  | ||||||
|  | import lombok.Data; | ||||||
|  | import lombok.EqualsAndHashCode; | ||||||
|  | import lombok.NoArgsConstructor; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 小程序登录用户身份权限 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | @Data | ||||||
|  | @EqualsAndHashCode(callSuper = true) | ||||||
|  | @NoArgsConstructor | ||||||
|  | public class XcxLoginUser extends LoginUser { | ||||||
|  |  | ||||||
|  |     private static final long serialVersionUID = 1L; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * openid | ||||||
|  |      */ | ||||||
|  |     private String openid; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -4,16 +4,12 @@ import cn.hutool.core.collection.CollUtil; | |||||||
| import cn.hutool.core.util.ObjectUtil; | import cn.hutool.core.util.ObjectUtil; | ||||||
| import com.baomidou.mybatisplus.core.conditions.Wrapper; | import com.baomidou.mybatisplus.core.conditions.Wrapper; | ||||||
| import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; | import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; | ||||||
| import com.baomidou.mybatisplus.core.enums.SqlMethod; |  | ||||||
| import com.baomidou.mybatisplus.core.mapper.BaseMapper; | import com.baomidou.mybatisplus.core.mapper.BaseMapper; | ||||||
| import com.baomidou.mybatisplus.core.metadata.IPage; | import com.baomidou.mybatisplus.core.metadata.IPage; | ||||||
| import com.baomidou.mybatisplus.core.metadata.TableInfo; |  | ||||||
| import com.baomidou.mybatisplus.core.metadata.TableInfoHelper; |  | ||||||
| import com.baomidou.mybatisplus.core.toolkit.*; | import com.baomidou.mybatisplus.core.toolkit.*; | ||||||
| import com.baomidou.mybatisplus.extension.plugins.pagination.Page; | import com.baomidou.mybatisplus.extension.plugins.pagination.Page; | ||||||
| import com.baomidou.mybatisplus.extension.toolkit.SqlHelper; | import com.baomidou.mybatisplus.extension.toolkit.Db; | ||||||
| import com.ruoyi.common.utils.BeanCopyUtils; | import com.ruoyi.common.utils.BeanCopyUtils; | ||||||
| import org.apache.ibatis.binding.MapperMethod; |  | ||||||
| import org.apache.ibatis.logging.Log; | import org.apache.ibatis.logging.Log; | ||||||
| import org.apache.ibatis.logging.LogFactory; | import org.apache.ibatis.logging.LogFactory; | ||||||
|  |  | ||||||
| @@ -21,7 +17,6 @@ import java.io.Serializable; | |||||||
| import java.util.Collection; | import java.util.Collection; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
| import java.util.Objects; |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 自定义 Mapper 接口, 实现 自定义扩展 |  * 自定义 Mapper 接口, 实现 自定义扩展 | ||||||
| @@ -37,8 +32,6 @@ public interface BaseMapperPlus<M, T, V> extends BaseMapper<T> { | |||||||
|  |  | ||||||
|     Log log = LogFactory.getLog(BaseMapperPlus.class); |     Log log = LogFactory.getLog(BaseMapperPlus.class); | ||||||
|  |  | ||||||
|     int DEFAULT_BATCH_SIZE = 1000; |  | ||||||
|  |  | ||||||
|     default Class<V> currentVoClass() { |     default Class<V> currentVoClass() { | ||||||
|         return (Class<V>) ReflectionKit.getSuperClassGenericType(this.getClass(), BaseMapperPlus.class, 2); |         return (Class<V>) ReflectionKit.getSuperClassGenericType(this.getClass(), BaseMapperPlus.class, 2); | ||||||
|     } |     } | ||||||
| @@ -59,79 +52,49 @@ public interface BaseMapperPlus<M, T, V> extends BaseMapper<T> { | |||||||
|      * 批量插入 |      * 批量插入 | ||||||
|      */ |      */ | ||||||
|     default boolean insertBatch(Collection<T> entityList) { |     default boolean insertBatch(Collection<T> entityList) { | ||||||
|         return insertBatch(entityList, DEFAULT_BATCH_SIZE); |         return Db.saveBatch(entityList); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 批量更新 |      * 批量更新 | ||||||
|      */ |      */ | ||||||
|     default boolean updateBatchById(Collection<T> entityList) { |     default boolean updateBatchById(Collection<T> entityList) { | ||||||
|         return updateBatchById(entityList, DEFAULT_BATCH_SIZE); |         return Db.updateBatchById(entityList); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 批量插入或更新 |      * 批量插入或更新 | ||||||
|      */ |      */ | ||||||
|     default boolean insertOrUpdateBatch(Collection<T> entityList) { |     default boolean insertOrUpdateBatch(Collection<T> entityList) { | ||||||
|         return insertOrUpdateBatch(entityList, DEFAULT_BATCH_SIZE); |         return Db.saveOrUpdateBatch(entityList); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 批量插入(包含限制条数) |      * 批量插入(包含限制条数) | ||||||
|      */ |      */ | ||||||
|     default boolean insertBatch(Collection<T> entityList, int batchSize) { |     default boolean insertBatch(Collection<T> entityList, int batchSize) { | ||||||
|         String sqlStatement = SqlHelper.getSqlStatement(this.currentMapperClass(), SqlMethod.INSERT_ONE); |         return Db.saveBatch(entityList, batchSize); | ||||||
|         return SqlHelper.executeBatch(this.currentModelClass(), log, entityList, batchSize, |  | ||||||
|             (sqlSession, entity) -> sqlSession.insert(sqlStatement, entity)); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 批量更新(包含限制条数) |      * 批量更新(包含限制条数) | ||||||
|      */ |      */ | ||||||
|     default boolean updateBatchById(Collection<T> entityList, int batchSize) { |     default boolean updateBatchById(Collection<T> entityList, int batchSize) { | ||||||
|         String sqlStatement = SqlHelper.getSqlStatement(this.currentMapperClass(), SqlMethod.UPDATE_BY_ID); |         return Db.updateBatchById(entityList, batchSize); | ||||||
|         return SqlHelper.executeBatch(this.currentModelClass(), log, entityList, batchSize, |  | ||||||
|             (sqlSession, entity) -> { |  | ||||||
|                 MapperMethod.ParamMap<T> param = new MapperMethod.ParamMap<>(); |  | ||||||
|                 param.put(Constants.ENTITY, entity); |  | ||||||
|                 sqlSession.update(sqlStatement, param); |  | ||||||
|             }); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 批量插入或更新(包含限制条数) |      * 批量插入或更新(包含限制条数) | ||||||
|      */ |      */ | ||||||
|     default boolean insertOrUpdateBatch(Collection<T> entityList, int batchSize) { |     default boolean insertOrUpdateBatch(Collection<T> entityList, int batchSize) { | ||||||
|         TableInfo tableInfo = TableInfoHelper.getTableInfo(this.currentModelClass()); |         return Db.saveOrUpdateBatch(entityList, batchSize); | ||||||
|         Assert.notNull(tableInfo, "error: can not execute. because can not find cache of TableInfo for entity!"); |  | ||||||
|         String keyProperty = tableInfo.getKeyProperty(); |  | ||||||
|         Assert.notEmpty(keyProperty, "error: can not execute. because can not find column for id from entity!"); |  | ||||||
|         return SqlHelper.saveOrUpdateBatch(this.currentModelClass(), this.currentMapperClass(), log, entityList, batchSize, (sqlSession, entity) -> { |  | ||||||
|             Object idVal = tableInfo.getPropertyValue(entity, keyProperty); |  | ||||||
|             String sqlStatement = SqlHelper.getSqlStatement(this.currentMapperClass(), SqlMethod.SELECT_BY_ID); |  | ||||||
|             return StringUtils.checkValNull(idVal) |  | ||||||
|                 || CollectionUtils.isEmpty(sqlSession.selectList(sqlStatement, entity)); |  | ||||||
|         }, (sqlSession, entity) -> { |  | ||||||
|             MapperMethod.ParamMap<T> param = new MapperMethod.ParamMap<>(); |  | ||||||
|             param.put(Constants.ENTITY, entity); |  | ||||||
|             String sqlStatement = SqlHelper.getSqlStatement(this.currentMapperClass(), SqlMethod.UPDATE_BY_ID); |  | ||||||
|             sqlSession.update(sqlStatement, param); |  | ||||||
|         }); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 插入或更新(包含限制条数) |      * 插入或更新(包含限制条数) | ||||||
|      */ |      */ | ||||||
|     default boolean insertOrUpdate(T entity) { |     default boolean insertOrUpdate(T entity) { | ||||||
|         if (null != entity) { |         return Db.saveOrUpdate(entity); | ||||||
|             TableInfo tableInfo = TableInfoHelper.getTableInfo(this.currentModelClass()); |  | ||||||
|             Assert.notNull(tableInfo, "error: can not execute. because can not find cache of TableInfo for entity!"); |  | ||||||
|             String keyProperty = tableInfo.getKeyProperty(); |  | ||||||
|             Assert.notEmpty(keyProperty, "error: can not execute. because can not find column for id from entity!"); |  | ||||||
|             Object idVal = tableInfo.getPropertyValue(entity, tableInfo.getKeyProperty()); |  | ||||||
|             return StringUtils.checkValNull(idVal) || Objects.isNull(selectById((Serializable) idVal)) ? insert(entity) > 0 : updateById(entity) > 0; |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     default V selectVoById(Serializable id) { |     default V selectVoById(Serializable id) { | ||||||
| @@ -149,7 +112,7 @@ public interface BaseMapperPlus<M, T, V> extends BaseMapper<T> { | |||||||
|         return BeanCopyUtils.copy(obj, voClass); |         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()); |         return selectVoBatchIds(idList, this.currentVoClass()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,11 +2,8 @@ package com.ruoyi.common.core.page; | |||||||
|  |  | ||||||
| import cn.hutool.http.HttpStatus; | import cn.hutool.http.HttpStatus; | ||||||
| import com.baomidou.mybatisplus.core.metadata.IPage; | import com.baomidou.mybatisplus.core.metadata.IPage; | ||||||
| import io.swagger.annotations.ApiModel; |  | ||||||
| import io.swagger.annotations.ApiModelProperty; |  | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
| import lombok.experimental.Accessors; |  | ||||||
|  |  | ||||||
| import java.io.Serializable; | import java.io.Serializable; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| @@ -19,32 +16,27 @@ import java.util.List; | |||||||
|  |  | ||||||
| @Data | @Data | ||||||
| @NoArgsConstructor | @NoArgsConstructor | ||||||
| @ApiModel("分页响应对象") |  | ||||||
| public class TableDataInfo<T> implements Serializable { | public class TableDataInfo<T> implements Serializable { | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 总记录数 |      * 总记录数 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty("总记录数") |  | ||||||
|     private long total; |     private long total; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 列表数据 |      * 列表数据 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty("列表数据") |  | ||||||
|     private List<T> rows; |     private List<T> rows; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 消息状态码 |      * 消息状态码 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty("消息状态码") |  | ||||||
|     private int code; |     private int code; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 消息内容 |      * 消息内容 | ||||||
|      */ |      */ | ||||||
|     @ApiModelProperty("消息内容") |  | ||||||
|     private String msg; |     private String msg; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|   | |||||||
| @@ -1,14 +0,0 @@ | |||||||
| package com.ruoyi.common.core.service; |  | ||||||
|  |  | ||||||
| import javax.servlet.http.HttpServletRequest; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * 通用 系统访问日志 |  | ||||||
|  * |  | ||||||
|  * @author Lion Li |  | ||||||
|  */ |  | ||||||
| public interface LogininforService { |  | ||||||
|  |  | ||||||
|     void recordLogininfor(String username, String status, String message, |  | ||||||
|                           HttpServletRequest request, Object... args); |  | ||||||
| } |  | ||||||
| @@ -1,15 +0,0 @@ | |||||||
| package com.ruoyi.common.core.service; |  | ||||||
|  |  | ||||||
| import com.ruoyi.common.core.domain.dto.OperLogDTO; |  | ||||||
| import org.springframework.scheduling.annotation.Async; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * 通用 操作日志 |  | ||||||
|  * |  | ||||||
|  * @author Lion Li |  | ||||||
|  */ |  | ||||||
| public interface OperLogService { |  | ||||||
|  |  | ||||||
|     @Async |  | ||||||
|     void recordOper(OperLogDTO operLogDTO); |  | ||||||
| } |  | ||||||
| @@ -0,0 +1,49 @@ | |||||||
|  | package com.ruoyi.common.enums; | ||||||
|  |  | ||||||
|  | import com.ruoyi.common.utils.StringUtils; | ||||||
|  | import lombok.AllArgsConstructor; | ||||||
|  | import lombok.Getter; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 数据库类型 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | @Getter | ||||||
|  | @AllArgsConstructor | ||||||
|  | public enum DataBaseType { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * MySQL | ||||||
|  |      */ | ||||||
|  |     MY_SQL("MySQL"), | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Oracle | ||||||
|  |      */ | ||||||
|  |     ORACLE("Oracle"), | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * PostgreSQL | ||||||
|  |      */ | ||||||
|  |     POSTGRE_SQL("PostgreSQL"), | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * SQL Server | ||||||
|  |      */ | ||||||
|  |     SQL_SERVER("Microsoft SQL Server"); | ||||||
|  |  | ||||||
|  |     private final String type; | ||||||
|  |  | ||||||
|  |     public static DataBaseType find(String databaseProductName) { | ||||||
|  |         if (StringUtils.isBlank(databaseProductName)) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         for (DataBaseType type : values()) { | ||||||
|  |             if (type.getType().equals(databaseProductName)) { | ||||||
|  |                 return type; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -21,7 +21,12 @@ public enum DeviceType { | |||||||
|     /** |     /** | ||||||
|      * app端 |      * app端 | ||||||
|      */ |      */ | ||||||
|     APP("app"); |     APP("app"), | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 小程序端 | ||||||
|  |      */ | ||||||
|  |     XCX("xcx"); | ||||||
|  |  | ||||||
|     private final String device; |     private final String device; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,39 @@ | |||||||
|  | package com.ruoyi.common.enums; | ||||||
|  |  | ||||||
|  | import lombok.AllArgsConstructor; | ||||||
|  | import lombok.Getter; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 登录类型 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | @Getter | ||||||
|  | @AllArgsConstructor | ||||||
|  | public enum LoginType { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 密码登录 | ||||||
|  |      */ | ||||||
|  |     PASSWORD("user.password.retry.limit.exceed", "user.password.retry.limit.count"), | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 短信登录 | ||||||
|  |      */ | ||||||
|  |     SMS("sms.code.retry.limit.exceed", "sms.code.retry.limit.count"), | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 小程序登录 | ||||||
|  |      */ | ||||||
|  |     XCX("", ""); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 登录重试超出限制提示 | ||||||
|  |      */ | ||||||
|  |     final String retryLimitExceed; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 登录重试限制计数提示 | ||||||
|  |      */ | ||||||
|  |     final String retryLimitCount; | ||||||
|  | } | ||||||
| @@ -1,26 +0,0 @@ | |||||||
| package com.ruoyi.common.enums; |  | ||||||
|  |  | ||||||
| import lombok.AllArgsConstructor; |  | ||||||
| import lombok.Getter; |  | ||||||
|  |  | ||||||
| import java.util.concurrent.RejectedExecutionHandler; |  | ||||||
| import java.util.concurrent.ThreadPoolExecutor; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * 线程池 拒绝策略 泛型 |  | ||||||
|  * |  | ||||||
|  * @author Lion Li |  | ||||||
|  */ |  | ||||||
| @Getter |  | ||||||
| @AllArgsConstructor |  | ||||||
| public enum ThreadPoolRejectedPolicy { |  | ||||||
|  |  | ||||||
|     CALLER_RUNS_POLICY("调用方执行", ThreadPoolExecutor.CallerRunsPolicy.class), |  | ||||||
|     DISCARD_OLDEST_POLICY("放弃最旧的", ThreadPoolExecutor.DiscardOldestPolicy.class), |  | ||||||
|     DISCARD_POLICY("丢弃", ThreadPoolExecutor.DiscardPolicy.class), |  | ||||||
|     ABORT_POLICY("中止", ThreadPoolExecutor.AbortPolicy.class); |  | ||||||
|  |  | ||||||
|     private final String name; |  | ||||||
|     private final Class<? extends RejectedExecutionHandler> clazz; |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -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.equals(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.ExcelAnalysisException; | ||||||
| import com.alibaba.excel.exception.ExcelDataConvertException; | import com.alibaba.excel.exception.ExcelDataConvertException; | ||||||
| import com.ruoyi.common.utils.JsonUtils; | import com.ruoyi.common.utils.JsonUtils; | ||||||
|  | import com.ruoyi.common.utils.StreamUtils; | ||||||
| import com.ruoyi.common.utils.ValidatorUtils; | import com.ruoyi.common.utils.ValidatorUtils; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
| import lombok.extern.slf4j.Slf4j; | import lombok.extern.slf4j.Slf4j; | ||||||
| @@ -14,7 +15,6 @@ import javax.validation.ConstraintViolation; | |||||||
| import javax.validation.ConstraintViolationException; | import javax.validation.ConstraintViolationException; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
| import java.util.Set; | import java.util.Set; | ||||||
| import java.util.stream.Collectors; |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Excel 导入监听 |  * Excel 导入监听 | ||||||
| @@ -69,9 +69,7 @@ public class DefaultExcelListener<T> extends AnalysisEventListener<T> implements | |||||||
|         if (exception instanceof ConstraintViolationException) { |         if (exception instanceof ConstraintViolationException) { | ||||||
|             ConstraintViolationException constraintViolationException = (ConstraintViolationException) exception; |             ConstraintViolationException constraintViolationException = (ConstraintViolationException) exception; | ||||||
|             Set<ConstraintViolation<?>> constraintViolations = constraintViolationException.getConstraintViolations(); |             Set<ConstraintViolation<?>> constraintViolations = constraintViolationException.getConstraintViolations(); | ||||||
|             String constraintViolationsMsg = constraintViolations.stream() |             String constraintViolationsMsg = StreamUtils.join(constraintViolations, ConstraintViolation::getMessage, ", "); | ||||||
|                 .map(ConstraintViolation::getMessage) |  | ||||||
|                 .collect(Collectors.joining(", ")); |  | ||||||
|             errMsg = StrUtil.format("第{}行数据校验异常: {}", context.readRowHolder().getRowIndex() + 1, constraintViolationsMsg); |             errMsg = StrUtil.format("第{}行数据校验异常: {}", context.readRowHolder().getRowIndex() + 1, constraintViolationsMsg); | ||||||
|             if (log.isDebugEnabled()) { |             if (log.isDebugEnabled()) { | ||||||
|                 log.error(errMsg); |                 log.error(errMsg); | ||||||
|   | |||||||
| @@ -1,52 +1,52 @@ | |||||||
| package com.ruoyi.common.exception; | package com.ruoyi.common.exception; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 全局异常 |  * 全局异常 | ||||||
|  * |  * | ||||||
|  * @author ruoyi |  * @author ruoyi | ||||||
|  */ |  */ | ||||||
| public class GlobalException extends RuntimeException { | public class GlobalException extends RuntimeException { | ||||||
|  |  | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 错误提示 |      * 错误提示 | ||||||
|      */ |      */ | ||||||
|     private String message; |     private String message; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 错误明细,内部调试错误 |      * 错误明细,内部调试错误 | ||||||
|      * <p> |      * <p> | ||||||
|      * 和 {@link CommonResult#getDetailMessage()} 一致的设计 |      * 和 {@link CommonResult#getDetailMessage()} 一致的设计 | ||||||
|      */ |      */ | ||||||
|     private String detailMessage; |     private String detailMessage; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 空构造方法,避免反序列化问题 |      * 空构造方法,避免反序列化问题 | ||||||
|      */ |      */ | ||||||
|     public GlobalException() { |     public GlobalException() { | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public GlobalException(String message) { |     public GlobalException(String message) { | ||||||
|         this.message = message; |         this.message = message; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public String getDetailMessage() { |     public String getDetailMessage() { | ||||||
|         return detailMessage; |         return detailMessage; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public GlobalException setDetailMessage(String detailMessage) { |     public GlobalException setDetailMessage(String detailMessage) { | ||||||
|         this.detailMessage = detailMessage; |         this.detailMessage = detailMessage; | ||||||
|         return this; |         return this; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public String getMessage() { |     public String getMessage() { | ||||||
|         return message; |         return message; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public GlobalException setMessage(String message) { |     public GlobalException setMessage(String message) { | ||||||
|         this.message = message; |         this.message = message; | ||||||
|         return this; |         return this; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,65 +1,65 @@ | |||||||
| package com.ruoyi.common.exception; | package com.ruoyi.common.exception; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 业务异常 |  * 业务异常 | ||||||
|  * |  * | ||||||
|  * @author ruoyi |  * @author ruoyi | ||||||
|  */ |  */ | ||||||
| public final class ServiceException extends RuntimeException { | public final class ServiceException extends RuntimeException { | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 错误码 |      * 错误码 | ||||||
|      */ |      */ | ||||||
|     private Integer code; |     private Integer code; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 错误提示 |      * 错误提示 | ||||||
|      */ |      */ | ||||||
|     private String message; |     private String message; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 错误明细,内部调试错误 |      * 错误明细,内部调试错误 | ||||||
|      * <p> |      * <p> | ||||||
|      * 和 {@link CommonResult#getDetailMessage()} 一致的设计 |      * 和 {@link CommonResult#getDetailMessage()} 一致的设计 | ||||||
|      */ |      */ | ||||||
|     private String detailMessage; |     private String detailMessage; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 空构造方法,避免反序列化问题 |      * 空构造方法,避免反序列化问题 | ||||||
|      */ |      */ | ||||||
|     public ServiceException() { |     public ServiceException() { | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public ServiceException(String message) { |     public ServiceException(String message) { | ||||||
|         this.message = message; |         this.message = message; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public ServiceException(String message, Integer code) { |     public ServiceException(String message, Integer code) { | ||||||
|         this.message = message; |         this.message = message; | ||||||
|         this.code = code; |         this.code = code; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public String getDetailMessage() { |     public String getDetailMessage() { | ||||||
|         return detailMessage; |         return detailMessage; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public String getMessage() { |     public String getMessage() { | ||||||
|         return message; |         return message; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public Integer getCode() { |     public Integer getCode() { | ||||||
|         return code; |         return code; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public ServiceException setMessage(String message) { |     public ServiceException setMessage(String message) { | ||||||
|         this.message = message; |         this.message = message; | ||||||
|         return this; |         return this; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public ServiceException setDetailMessage(String detailMessage) { |     public ServiceException setDetailMessage(String detailMessage) { | ||||||
|         this.detailMessage = detailMessage; |         this.detailMessage = detailMessage; | ||||||
|         return this; |         return this; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -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; | package com.ruoyi.common.filter; | ||||||
|  |  | ||||||
| import cn.hutool.core.io.IoUtil; | import cn.hutool.core.io.IoUtil; | ||||||
|  | import com.ruoyi.common.constant.Constants; | ||||||
|  |  | ||||||
| import javax.servlet.ReadListener; | import javax.servlet.ReadListener; | ||||||
| import javax.servlet.ServletInputStream; | import javax.servlet.ServletInputStream; | ||||||
| @@ -11,7 +12,6 @@ import java.io.BufferedReader; | |||||||
| import java.io.ByteArrayInputStream; | import java.io.ByteArrayInputStream; | ||||||
| import java.io.IOException; | import java.io.IOException; | ||||||
| import java.io.InputStreamReader; | import java.io.InputStreamReader; | ||||||
| import java.nio.charset.StandardCharsets; |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 构建可重复读取inputStream的request |  * 构建可重复读取inputStream的request | ||||||
| @@ -23,10 +23,10 @@ public class RepeatedlyRequestWrapper extends HttpServletRequestWrapper { | |||||||
|  |  | ||||||
|     public RepeatedlyRequestWrapper(HttpServletRequest request, ServletResponse response) throws IOException { |     public RepeatedlyRequestWrapper(HttpServletRequest request, ServletResponse response) throws IOException { | ||||||
|         super(request); |         super(request); | ||||||
|         request.setCharacterEncoding("UTF-8"); |         request.setCharacterEncoding(Constants.UTF8); | ||||||
|         response.setCharacterEncoding("UTF-8"); |         response.setCharacterEncoding(Constants.UTF8); | ||||||
|  |  | ||||||
|         body = IoUtil.readUtf8(request.getInputStream()).getBytes(StandardCharsets.UTF_8); |         body = IoUtil.readBytes(request.getInputStream(), false); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| package com.ruoyi.common.filter; | package com.ruoyi.common.filter; | ||||||
|  |  | ||||||
|  | import com.ruoyi.common.enums.HttpMethod; | ||||||
| import com.ruoyi.common.utils.StringUtils; | import com.ruoyi.common.utils.StringUtils; | ||||||
|  |  | ||||||
| import javax.servlet.*; | import javax.servlet.*; | ||||||
| @@ -48,7 +49,7 @@ public class XssFilter implements Filter { | |||||||
|         String url = request.getServletPath(); |         String url = request.getServletPath(); | ||||||
|         String method = request.getMethod(); |         String method = request.getMethod(); | ||||||
|         // GET DELETE 不过滤 |         // 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 true; | ||||||
|         } |         } | ||||||
|         return StringUtils.matches(url, excludes); |         return StringUtils.matches(url, excludes); | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| package com.ruoyi.common.filter; | package com.ruoyi.common.filter; | ||||||
|  |  | ||||||
| import cn.hutool.core.io.IoUtil; | import cn.hutool.core.io.IoUtil; | ||||||
|  | import cn.hutool.core.util.StrUtil; | ||||||
| import cn.hutool.http.HtmlUtil; | import cn.hutool.http.HtmlUtil; | ||||||
| import com.ruoyi.common.utils.StringUtils; | import com.ruoyi.common.utils.StringUtils; | ||||||
| import org.springframework.http.HttpHeaders; | import org.springframework.http.HttpHeaders; | ||||||
| @@ -32,12 +33,12 @@ public class XssHttpServletRequestWrapper extends HttpServletRequestWrapper { | |||||||
|         String[] values = super.getParameterValues(name); |         String[] values = super.getParameterValues(name); | ||||||
|         if (values != null) { |         if (values != null) { | ||||||
|             int length = values.length; |             int length = values.length; | ||||||
|             String[] escapseValues = new String[length]; |             String[] escapesValues = new String[length]; | ||||||
|             for (int i = 0; i < length; i++) { |             for (int i = 0; i < length; i++) { | ||||||
|                 // 防xss攻击和过滤前后空格 |                 // 防xss攻击和过滤前后空格 | ||||||
|                 escapseValues[i] = HtmlUtil.cleanHtmlTag(values[i]).trim(); |                 escapesValues[i] = HtmlUtil.cleanHtmlTag(values[i]).trim(); | ||||||
|             } |             } | ||||||
|             return escapseValues; |             return escapesValues; | ||||||
|         } |         } | ||||||
|         return super.getParameterValues(name); |         return super.getParameterValues(name); | ||||||
|     } |     } | ||||||
| @@ -50,7 +51,7 @@ public class XssHttpServletRequestWrapper extends HttpServletRequestWrapper { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         // 为空,直接返回 |         // 为空,直接返回 | ||||||
|         String json = IoUtil.read(super.getInputStream(), StandardCharsets.UTF_8); |         String json = StrUtil.str(IoUtil.readBytes(super.getInputStream(), false), StandardCharsets.UTF_8); | ||||||
|         if (StringUtils.isEmpty(json)) { |         if (StringUtils.isEmpty(json)) { | ||||||
|             return super.getInputStream(); |             return super.getInputStream(); | ||||||
|         } |         } | ||||||
| @@ -88,11 +89,9 @@ public class XssHttpServletRequestWrapper extends HttpServletRequestWrapper { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 是否是Json请求 |      * 是否是Json请求 | ||||||
|      * |  | ||||||
|      * @param request |  | ||||||
|      */ |      */ | ||||||
|     public boolean isJsonRequest() { |     public boolean isJsonRequest() { | ||||||
|         String header = super.getHeader(HttpHeaders.CONTENT_TYPE); |         String header = super.getHeader(HttpHeaders.CONTENT_TYPE); | ||||||
|         return StringUtils.startsWithIgnoreCase(header, MediaType.APPLICATION_JSON_VALUE); |         return StringUtils.startsWithIgnoreCase(header, MediaType.APPLICATION_JSON_VALUE); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,72 @@ | |||||||
|  | package com.ruoyi.common.helper; | ||||||
|  |  | ||||||
|  | import cn.hutool.core.convert.Convert; | ||||||
|  | import com.baomidou.dynamic.datasource.DynamicRoutingDataSource; | ||||||
|  | import com.ruoyi.common.enums.DataBaseType; | ||||||
|  | import com.ruoyi.common.exception.ServiceException; | ||||||
|  | import com.ruoyi.common.utils.spring.SpringUtils; | ||||||
|  | import lombok.AccessLevel; | ||||||
|  | import lombok.NoArgsConstructor; | ||||||
|  |  | ||||||
|  | import javax.sql.DataSource; | ||||||
|  | import java.sql.Connection; | ||||||
|  | import java.sql.DatabaseMetaData; | ||||||
|  | import java.sql.SQLException; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 数据库助手 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | @NoArgsConstructor(access = AccessLevel.PRIVATE) | ||||||
|  | public class DataBaseHelper { | ||||||
|  |  | ||||||
|  |     private static final DynamicRoutingDataSource DS = SpringUtils.getBean(DynamicRoutingDataSource.class); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 获取当前数据库类型 | ||||||
|  |      */ | ||||||
|  |     public static DataBaseType getDataBaseType() { | ||||||
|  |         DataSource dataSource = DS.determineDataSource(); | ||||||
|  |         try (Connection conn = dataSource.getConnection()) { | ||||||
|  |             DatabaseMetaData metaData = conn.getMetaData(); | ||||||
|  |             String databaseProductName = metaData.getDatabaseProductName(); | ||||||
|  |             return DataBaseType.find(databaseProductName); | ||||||
|  |         } catch (SQLException e) { | ||||||
|  |             throw new ServiceException(e.getMessage()); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static boolean isMySql() { | ||||||
|  |         return DataBaseType.MY_SQL == getDataBaseType(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static boolean isOracle() { | ||||||
|  |         return DataBaseType.ORACLE == getDataBaseType(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static boolean isPostgerSql() { | ||||||
|  |         return DataBaseType.POSTGRE_SQL == getDataBaseType(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static boolean isSqlServer() { | ||||||
|  |         return DataBaseType.SQL_SERVER == getDataBaseType(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static String findInSet(Object var1, String var2) { | ||||||
|  |         DataBaseType dataBasyType = getDataBaseType(); | ||||||
|  |         String var = Convert.toStr(var1); | ||||||
|  |         if (dataBasyType == DataBaseType.SQL_SERVER) { | ||||||
|  |             // charindex(',100,' , ',0,100,101,') <> 0 | ||||||
|  |             return "charindex('," + var + ",' , ','+" + var2 + "+',') <> 0"; | ||||||
|  |         } else if (dataBasyType == DataBaseType.POSTGRE_SQL) { | ||||||
|  |             // (select position(',100,' in ',0,100,101,')) <> 0 | ||||||
|  |             return "(select position('," + var + ",' in ','||" + var2 + "||',')) <> 0"; | ||||||
|  |         } else if (dataBasyType == DataBaseType.ORACLE) { | ||||||
|  |             // instr(',0,100,101,' , ',100,') <> 0 | ||||||
|  |             return "instr(','||" + var2 + "||',' , '," + var + ",') <> 0"; | ||||||
|  |         } | ||||||
|  |         // find_in_set(100 , '0,100,101') | ||||||
|  |         return "find_in_set(" + var + " , " + var2 + ") <> 0"; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,11 +1,11 @@ | |||||||
| package com.ruoyi.common.helper; | 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 cn.hutool.core.util.ObjectUtil; | ||||||
| import com.ruoyi.common.utils.ServletUtils; |  | ||||||
| import lombok.AccessLevel; | import lombok.AccessLevel; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
|  |  | ||||||
| import javax.servlet.http.HttpServletRequest; |  | ||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
|  |  | ||||||
| @@ -33,11 +33,11 @@ public class DataPermissionHelper { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static Map<String, Object> getContext() { |     public static Map<String, Object> getContext() { | ||||||
|         HttpServletRequest request = ServletUtils.getRequest(); |         SaStorage saStorage = SaHolder.getStorage(); | ||||||
|         Object attribute = request.getAttribute(DATA_PERMISSION_KEY); |         Object attribute = saStorage.get(DATA_PERMISSION_KEY); | ||||||
|         if (ObjectUtil.isNull(attribute)) { |         if (ObjectUtil.isNull(attribute)) { | ||||||
|             request.setAttribute(DATA_PERMISSION_KEY, new HashMap<>()); |             saStorage.set(DATA_PERMISSION_KEY, new HashMap<>()); | ||||||
|             attribute = request.getAttribute(DATA_PERMISSION_KEY); |             attribute = saStorage.get(DATA_PERMISSION_KEY); | ||||||
|         } |         } | ||||||
|         if (attribute instanceof Map) { |         if (attribute instanceof Map) { | ||||||
|             return (Map<String, Object>) attribute; |             return (Map<String, Object>) attribute; | ||||||
|   | |||||||
| @@ -1,6 +1,8 @@ | |||||||
| package com.ruoyi.common.helper; | package com.ruoyi.common.helper; | ||||||
|  |  | ||||||
|  | import cn.dev33.satoken.context.SaHolder; | ||||||
| import cn.dev33.satoken.stp.StpUtil; | import cn.dev33.satoken.stp.StpUtil; | ||||||
|  | import cn.hutool.core.util.ArrayUtil; | ||||||
| import cn.hutool.core.util.ObjectUtil; | import cn.hutool.core.util.ObjectUtil; | ||||||
| import com.ruoyi.common.constant.UserConstants; | import com.ruoyi.common.constant.UserConstants; | ||||||
| import com.ruoyi.common.core.domain.model.LoginUser; | import com.ruoyi.common.core.domain.model.LoginUser; | ||||||
| @@ -13,7 +15,7 @@ import lombok.NoArgsConstructor; | |||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 登录鉴权助手 |  * 登录鉴权助手 | ||||||
|  *  |  * | ||||||
|  * user_type 为 用户类型 同一个用户表 可以有多种用户类型 例如 pc,app |  * user_type 为 用户类型 同一个用户表 可以有多种用户类型 例如 pc,app | ||||||
|  * deivce 为 设备类型 同一个用户类型 可以有 多种设备类型 例如 web,ios |  * deivce 为 设备类型 同一个用户类型 可以有 多种设备类型 例如 web,ios | ||||||
|  * 可以组成 用户类型与设备类型多对多的 权限灵活控制 |  * 可以组成 用户类型与设备类型多对多的 权限灵活控制 | ||||||
| @@ -29,15 +31,13 @@ public class LoginHelper { | |||||||
|     public static final String JOIN_CODE = ":"; |     public static final String JOIN_CODE = ":"; | ||||||
|     public static final String LOGIN_USER_KEY = "loginUser"; |     public static final String LOGIN_USER_KEY = "loginUser"; | ||||||
|  |  | ||||||
|     private static final ThreadLocal<LoginUser> LOGIN_CACHE = new ThreadLocal<>(); |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 登录系统 |      * 登录系统 | ||||||
|      * |      * | ||||||
|      * @param loginUser 登录用户信息 |      * @param loginUser 登录用户信息 | ||||||
|      */ |      */ | ||||||
|     public static void login(LoginUser loginUser) { |     public static void login(LoginUser loginUser) { | ||||||
|         LOGIN_CACHE.set(loginUser); |         SaHolder.getStorage().set(LOGIN_USER_KEY, loginUser); | ||||||
|         StpUtil.login(loginUser.getLoginId()); |         StpUtil.login(loginUser.getLoginId()); | ||||||
|         setLoginUser(loginUser); |         setLoginUser(loginUser); | ||||||
|     } |     } | ||||||
| @@ -49,7 +49,7 @@ public class LoginHelper { | |||||||
|      * @param loginUser 登录用户信息 |      * @param loginUser 登录用户信息 | ||||||
|      */ |      */ | ||||||
|     public static void loginByDevice(LoginUser loginUser, DeviceType deviceType) { |     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()); |         StpUtil.login(loginUser.getLoginId(), deviceType.getDevice()); | ||||||
|         setLoginUser(loginUser); |         setLoginUser(loginUser); | ||||||
|     } |     } | ||||||
| @@ -65,18 +65,13 @@ public class LoginHelper { | |||||||
|      * 获取用户(多级缓存) |      * 获取用户(多级缓存) | ||||||
|      */ |      */ | ||||||
|     public static LoginUser getLoginUser() { |     public static LoginUser getLoginUser() { | ||||||
|         LoginUser loginUser = LOGIN_CACHE.get(); |         LoginUser loginUser = (LoginUser) SaHolder.getStorage().get(LOGIN_USER_KEY); | ||||||
|         if (loginUser != null) { |         if (loginUser != null) { | ||||||
|             return loginUser; |             return loginUser; | ||||||
|         } |         } | ||||||
|         return (LoginUser) StpUtil.getTokenSession().get(LOGIN_USER_KEY); |         loginUser = (LoginUser) StpUtil.getTokenSession().get(LOGIN_USER_KEY); | ||||||
|     } |         SaHolder.getStorage().set(LOGIN_USER_KEY, loginUser); | ||||||
|  |         return loginUser; | ||||||
|     /** |  | ||||||
|      * 清除一级缓存 防止内存问题 |  | ||||||
|      */ |  | ||||||
|     public static void clearCache() { |  | ||||||
|         LOGIN_CACHE.remove(); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|   | |||||||
| @@ -0,0 +1,56 @@ | |||||||
|  | package com.ruoyi.common.jackson; | ||||||
|  |  | ||||||
|  | import cn.hutool.core.util.ObjectUtil; | ||||||
|  | import cn.hutool.core.util.StrUtil; | ||||||
|  | import com.fasterxml.jackson.core.JsonGenerator; | ||||||
|  | import com.fasterxml.jackson.databind.BeanProperty; | ||||||
|  | import com.fasterxml.jackson.databind.JsonMappingException; | ||||||
|  | import com.fasterxml.jackson.databind.JsonSerializer; | ||||||
|  | import com.fasterxml.jackson.databind.SerializerProvider; | ||||||
|  | import com.fasterxml.jackson.databind.ser.ContextualSerializer; | ||||||
|  | import com.ruoyi.common.annotation.DictDataMapper; | ||||||
|  | import com.ruoyi.common.core.service.DictService; | ||||||
|  | import com.ruoyi.common.utils.StringUtils; | ||||||
|  | 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; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 字典数据json序列化工具 | ||||||
|  |  * | ||||||
|  |  * @author itino | ||||||
|  |  */ | ||||||
|  | @Slf4j | ||||||
|  | public class DictDataJsonSerializer extends JsonSerializer<String> implements ContextualSerializer { | ||||||
|  |  | ||||||
|  |     private String dictType; | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public void serialize(String value, JsonGenerator gen, SerializerProvider serializers) throws IOException { | ||||||
|  |         try { | ||||||
|  |             DictService dictService = SpringUtils.getBean(DictService.class); | ||||||
|  |             if (ObjectUtil.isNotNull(dictService)) { | ||||||
|  |                 String label = dictService.getDictLabel(dictType, value); | ||||||
|  |                 gen.writeString(StringUtils.isNotBlank(label) ? label : value); | ||||||
|  |             } else { | ||||||
|  |                 gen.writeString(value); | ||||||
|  |             } | ||||||
|  |         } catch (BeansException e) { | ||||||
|  |             log.error("字典数据未查到, 采用默认处理 => {}", e.getMessage()); | ||||||
|  |             gen.writeString(value); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public JsonSerializer<?> createContextual(SerializerProvider prov, BeanProperty property) throws JsonMappingException { | ||||||
|  |         DictDataMapper anno = property.getAnnotation(DictDataMapper.class); | ||||||
|  |         if (Objects.nonNull(anno) && StrUtil.isNotBlank(anno.dictType())) { | ||||||
|  |             this.dictType = anno.dictType(); | ||||||
|  |             return this; | ||||||
|  |         } | ||||||
|  |         return prov.findValueSerializer(property.getType(), property); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,5 +1,6 @@ | |||||||
| package com.ruoyi.common.jackson; | package com.ruoyi.common.jackson; | ||||||
|  |  | ||||||
|  | import cn.hutool.core.util.ObjectUtil; | ||||||
| import com.fasterxml.jackson.core.JsonGenerator; | import com.fasterxml.jackson.core.JsonGenerator; | ||||||
| import com.fasterxml.jackson.databind.BeanProperty; | import com.fasterxml.jackson.databind.BeanProperty; | ||||||
| import com.fasterxml.jackson.databind.JsonMappingException; | 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.core.service.SensitiveService; | ||||||
| import com.ruoyi.common.enums.SensitiveStrategy; | import com.ruoyi.common.enums.SensitiveStrategy; | ||||||
| import com.ruoyi.common.utils.spring.SpringUtils; | import com.ruoyi.common.utils.spring.SpringUtils; | ||||||
|  | import lombok.extern.slf4j.Slf4j; | ||||||
|  | import org.springframework.beans.BeansException; | ||||||
|  |  | ||||||
| import java.io.IOException; | import java.io.IOException; | ||||||
| import java.util.Objects; | import java.util.Objects; | ||||||
| @@ -19,19 +22,24 @@ import java.util.Objects; | |||||||
|  * |  * | ||||||
|  * @author Yjoioooo |  * @author Yjoioooo | ||||||
|  */ |  */ | ||||||
|  | @Slf4j | ||||||
| public class SensitiveJsonSerializer extends JsonSerializer<String> implements ContextualSerializer { | public class SensitiveJsonSerializer extends JsonSerializer<String> implements ContextualSerializer { | ||||||
|  |  | ||||||
|     private SensitiveStrategy strategy; |     private SensitiveStrategy strategy; | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void serialize(String value, JsonGenerator gen, SerializerProvider serializers) throws IOException { |     public void serialize(String value, JsonGenerator gen, SerializerProvider serializers) throws IOException { | ||||||
|         SensitiveService sensitiveService = SpringUtils.getBean(SensitiveService.class); |         try { | ||||||
|         if (sensitiveService.isSensitive()) { |             SensitiveService sensitiveService = SpringUtils.getBean(SensitiveService.class); | ||||||
|  |             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); |             gen.writeString(value); | ||||||
|         } else { |  | ||||||
|             gen.writeString(strategy.desensitizer().apply(value)); |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|   | |||||||
| @@ -1,12 +1,16 @@ | |||||||
| package com.ruoyi.common.utils; | package com.ruoyi.common.utils; | ||||||
|  |  | ||||||
| import cn.hutool.core.collection.CollUtil; | import cn.hutool.core.collection.CollUtil; | ||||||
|  | import cn.hutool.core.lang.SimpleCache; | ||||||
| import cn.hutool.core.map.MapUtil; | import cn.hutool.core.map.MapUtil; | ||||||
| import cn.hutool.core.util.ObjectUtil; | import cn.hutool.core.util.ObjectUtil; | ||||||
| import cn.hutool.core.util.ReflectUtil; | import cn.hutool.core.util.ReflectUtil; | ||||||
| import cn.hutool.extra.cglib.CglibUtil; | import cn.hutool.core.util.StrUtil; | ||||||
| import lombok.AccessLevel; | import lombok.AccessLevel; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
|  | import org.springframework.cglib.beans.BeanCopier; | ||||||
|  | import org.springframework.cglib.beans.BeanMap; | ||||||
|  | import org.springframework.cglib.core.Converter; | ||||||
|  |  | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
| @@ -37,7 +41,8 @@ public class BeanCopyUtils { | |||||||
|         if (ObjectUtil.isNull(desc)) { |         if (ObjectUtil.isNull(desc)) { | ||||||
|             return null; |             return null; | ||||||
|         } |         } | ||||||
|         return CglibUtil.copy(source, desc); |         final V target = ReflectUtil.newInstanceIfPossible(desc); | ||||||
|  |         return copy(source, target); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -54,7 +59,8 @@ public class BeanCopyUtils { | |||||||
|         if (ObjectUtil.isNull(desc)) { |         if (ObjectUtil.isNull(desc)) { | ||||||
|             return null; |             return null; | ||||||
|         } |         } | ||||||
|         CglibUtil.copy(source, desc); |         BeanCopier beanCopier = BeanCopierCache.INSTANCE.get(source.getClass(), desc.getClass(), null); | ||||||
|  |         beanCopier.copy(source, desc, null); | ||||||
|         return desc; |         return desc; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -72,7 +78,11 @@ public class BeanCopyUtils { | |||||||
|         if (CollUtil.isEmpty(sourceList)) { |         if (CollUtil.isEmpty(sourceList)) { | ||||||
|             return CollUtil.newArrayList(); |             return CollUtil.newArrayList(); | ||||||
|         } |         } | ||||||
|         return CglibUtil.copyList(sourceList, () -> ReflectUtil.newInstanceIfPossible(desc)); |         return StreamUtils.toList(sourceList, source -> { | ||||||
|  |             V target = ReflectUtil.newInstanceIfPossible(desc); | ||||||
|  |             copy(source, target); | ||||||
|  |             return target; | ||||||
|  |         }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -81,11 +91,12 @@ public class BeanCopyUtils { | |||||||
|      * @param bean 数据来源实体 |      * @param bean 数据来源实体 | ||||||
|      * @return map对象 |      * @return map对象 | ||||||
|      */ |      */ | ||||||
|  |     @SuppressWarnings("unchecked") | ||||||
|     public static <T> Map<String, Object> copyToMap(T bean) { |     public static <T> Map<String, Object> copyToMap(T bean) { | ||||||
|         if (ObjectUtil.isNull(bean)) { |         if (ObjectUtil.isNull(bean)) { | ||||||
|             return null; |             return null; | ||||||
|         } |         } | ||||||
|         return CglibUtil.toMap(bean); |         return BeanMap.create(bean); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -102,7 +113,8 @@ public class BeanCopyUtils { | |||||||
|         if (ObjectUtil.isNull(beanClass)) { |         if (ObjectUtil.isNull(beanClass)) { | ||||||
|             return null; |             return null; | ||||||
|         } |         } | ||||||
|         return CglibUtil.toBean(map, beanClass); |         T bean = ReflectUtil.newInstanceIfPossible(beanClass); | ||||||
|  |         return mapToBean(map, bean); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -119,6 +131,54 @@ public class BeanCopyUtils { | |||||||
|         if (ObjectUtil.isNull(bean)) { |         if (ObjectUtil.isNull(bean)) { | ||||||
|             return null; |             return null; | ||||||
|         } |         } | ||||||
|         return CglibUtil.fillBean(map, bean); |         BeanMap.create(bean).putAll(map); | ||||||
|  |         return bean; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * BeanCopier属性缓存<br> | ||||||
|  |      * 缓存用于防止多次反射造成的性能问题 | ||||||
|  |      * | ||||||
|  |      * @author Looly | ||||||
|  |      * @since 5.4.1 | ||||||
|  |      */ | ||||||
|  |     public enum BeanCopierCache { | ||||||
|  |         /** | ||||||
|  |          * BeanCopier属性缓存单例 | ||||||
|  |          */ | ||||||
|  |         INSTANCE; | ||||||
|  |  | ||||||
|  |         private final SimpleCache<String, BeanCopier> cache = new SimpleCache<>(); | ||||||
|  |  | ||||||
|  |         /** | ||||||
|  |          * 获得类与转换器生成的key在{@link BeanCopier}的Map中对应的元素 | ||||||
|  |          * | ||||||
|  |          * @param srcClass    源Bean的类 | ||||||
|  |          * @param targetClass 目标Bean的类 | ||||||
|  |          * @param converter   转换器 | ||||||
|  |          * @return Map中对应的BeanCopier | ||||||
|  |          */ | ||||||
|  |         public BeanCopier get(Class<?> srcClass, Class<?> targetClass, Converter converter) { | ||||||
|  |             final String key = genKey(srcClass, targetClass, converter); | ||||||
|  |             return cache.get(key, () -> BeanCopier.create(srcClass, targetClass, converter != null)); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /** | ||||||
|  |          * 获得类与转换器生成的key | ||||||
|  |          * | ||||||
|  |          * @param srcClass    源Bean的类 | ||||||
|  |          * @param targetClass 目标Bean的类 | ||||||
|  |          * @param converter   转换器 | ||||||
|  |          * @return 属性名和Map映射的key | ||||||
|  |          */ | ||||||
|  |         private String genKey(Class<?> srcClass, Class<?> targetClass, Converter converter) { | ||||||
|  |             final StringBuilder key = StrUtil.builder() | ||||||
|  |                 .append(srcClass.getName()).append('#').append(targetClass.getName()); | ||||||
|  |             if(null != converter){ | ||||||
|  |                 key.append('#').append(converter.getClass().getName()); | ||||||
|  |             } | ||||||
|  |             return key.toString(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -22,17 +22,17 @@ import java.util.Date; | |||||||
| @NoArgsConstructor(access = AccessLevel.PRIVATE) | @NoArgsConstructor(access = AccessLevel.PRIVATE) | ||||||
| public class DateUtils extends org.apache.commons.lang3.time.DateUtils { | public class DateUtils extends org.apache.commons.lang3.time.DateUtils { | ||||||
|  |  | ||||||
|     public static String YYYY = "yyyy"; |     public static final String YYYY = "yyyy"; | ||||||
|  |  | ||||||
|     public static String YYYY_MM = "yyyy-MM"; |     public static final String YYYY_MM = "yyyy-MM"; | ||||||
|  |  | ||||||
|     public static String YYYY_MM_DD = "yyyy-MM-dd"; |     public static final String YYYY_MM_DD = "yyyy-MM-dd"; | ||||||
|  |  | ||||||
|     public static String YYYYMMDDHHMMSS = "yyyyMMddHHmmss"; |     public static final String YYYYMMDDHHMMSS = "yyyyMMddHHmmss"; | ||||||
|  |  | ||||||
|     public static String YYYY_MM_DD_HH_MM_SS = "yyyy-MM-dd HH:mm:ss"; |     public static final String YYYY_MM_DD_HH_MM_SS = "yyyy-MM-dd HH:mm:ss"; | ||||||
|  |  | ||||||
|     private static String[] parsePatterns = { |     private static final String[] PARSE_PATTERNS = { | ||||||
|         "yyyy-MM-dd", "yyyy-MM-dd HH:mm:ss", "yyyy-MM-dd HH:mm", "yyyy-MM", |         "yyyy-MM-dd", "yyyy-MM-dd HH:mm:ss", "yyyy-MM-dd HH:mm", "yyyy-MM", | ||||||
|         "yyyy/MM/dd", "yyyy/MM/dd HH:mm:ss", "yyyy/MM/dd HH:mm", "yyyy/MM", |         "yyyy/MM/dd", "yyyy/MM/dd HH:mm:ss", "yyyy/MM/dd HH:mm", "yyyy/MM", | ||||||
|         "yyyy.MM.dd", "yyyy.MM.dd HH:mm:ss", "yyyy.MM.dd HH:mm", "yyyy.MM"}; |         "yyyy.MM.dd", "yyyy.MM.dd HH:mm:ss", "yyyy.MM.dd HH:mm", "yyyy.MM"}; | ||||||
| @@ -55,27 +55,27 @@ public class DateUtils extends org.apache.commons.lang3.time.DateUtils { | |||||||
|         return dateTimeNow(YYYY_MM_DD); |         return dateTimeNow(YYYY_MM_DD); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static final String getTime() { |     public static String getTime() { | ||||||
|         return dateTimeNow(YYYY_MM_DD_HH_MM_SS); |         return dateTimeNow(YYYY_MM_DD_HH_MM_SS); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static final String dateTimeNow() { |     public static String dateTimeNow() { | ||||||
|         return dateTimeNow(YYYYMMDDHHMMSS); |         return dateTimeNow(YYYYMMDDHHMMSS); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static final String dateTimeNow(final String format) { |     public static String dateTimeNow(final String format) { | ||||||
|         return parseDateToStr(format, new Date()); |         return parseDateToStr(format, new Date()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static final String dateTime(final Date date) { |     public static String dateTime(final Date date) { | ||||||
|         return parseDateToStr(YYYY_MM_DD, date); |         return parseDateToStr(YYYY_MM_DD, date); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static final String parseDateToStr(final String format, final Date date) { |     public static String parseDateToStr(final String format, final Date date) { | ||||||
|         return new SimpleDateFormat(format).format(date); |         return new SimpleDateFormat(format).format(date); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static final Date dateTime(final String format, final String ts) { |     public static Date dateTime(final String format, final String ts) { | ||||||
|         try { |         try { | ||||||
|             return new SimpleDateFormat(format).parse(ts); |             return new SimpleDateFormat(format).parse(ts); | ||||||
|         } catch (ParseException e) { |         } catch (ParseException e) { | ||||||
| @@ -86,7 +86,7 @@ public class DateUtils extends org.apache.commons.lang3.time.DateUtils { | |||||||
|     /** |     /** | ||||||
|      * 日期路径 即年/月/日 如2018/08/08 |      * 日期路径 即年/月/日 如2018/08/08 | ||||||
|      */ |      */ | ||||||
|     public static final String datePath() { |     public static String datePath() { | ||||||
|         Date now = new Date(); |         Date now = new Date(); | ||||||
|         return DateFormatUtils.format(now, "yyyy/MM/dd"); |         return DateFormatUtils.format(now, "yyyy/MM/dd"); | ||||||
|     } |     } | ||||||
| @@ -94,7 +94,7 @@ public class DateUtils extends org.apache.commons.lang3.time.DateUtils { | |||||||
|     /** |     /** | ||||||
|      * 日期路径 即年/月/日 如20180808 |      * 日期路径 即年/月/日 如20180808 | ||||||
|      */ |      */ | ||||||
|     public static final String dateTime() { |     public static String dateTime() { | ||||||
|         Date now = new Date(); |         Date now = new Date(); | ||||||
|         return DateFormatUtils.format(now, "yyyyMMdd"); |         return DateFormatUtils.format(now, "yyyyMMdd"); | ||||||
|     } |     } | ||||||
| @@ -107,7 +107,7 @@ public class DateUtils extends org.apache.commons.lang3.time.DateUtils { | |||||||
|             return null; |             return null; | ||||||
|         } |         } | ||||||
|         try { |         try { | ||||||
|             return parseDate(str.toString(), parsePatterns); |             return parseDate(str.toString(), PARSE_PATTERNS); | ||||||
|         } catch (ParseException e) { |         } catch (ParseException e) { | ||||||
|             return null; |             return null; | ||||||
|         } |         } | ||||||
| @@ -124,8 +124,7 @@ public class DateUtils extends org.apache.commons.lang3.time.DateUtils { | |||||||
|     /** |     /** | ||||||
|      * 计算相差天数 |      * 计算相差天数 | ||||||
|      */ |      */ | ||||||
|     public static int differentDaysByMillisecond(Date date1, Date date2) |     public static int differentDaysByMillisecond(Date date1, Date date2) { | ||||||
|     { |  | ||||||
|         return Math.abs((int) ((date2.getTime() - date1.getTime()) / (1000 * 3600 * 24))); |         return Math.abs((int) ((date2.getTime() - date1.getTime()) / (1000 * 3600 * 24))); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ import cn.hutool.core.util.ObjectUtil; | |||||||
| import com.fasterxml.jackson.core.JsonProcessingException; | import com.fasterxml.jackson.core.JsonProcessingException; | ||||||
| import com.fasterxml.jackson.core.type.TypeReference; | import com.fasterxml.jackson.core.type.TypeReference; | ||||||
| import com.fasterxml.jackson.databind.ObjectMapper; | import com.fasterxml.jackson.databind.ObjectMapper; | ||||||
|  | import com.fasterxml.jackson.databind.exc.MismatchedInputException; | ||||||
| import com.ruoyi.common.utils.spring.SpringUtils; | import com.ruoyi.common.utils.spring.SpringUtils; | ||||||
| import lombok.AccessLevel; | import lombok.AccessLevel; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
| @@ -78,6 +79,9 @@ public class JsonUtils { | |||||||
|         } |         } | ||||||
|         try { |         try { | ||||||
|             return OBJECT_MAPPER.readValue(text, OBJECT_MAPPER.getTypeFactory().constructType(Dict.class)); |             return OBJECT_MAPPER.readValue(text, OBJECT_MAPPER.getTypeFactory().constructType(Dict.class)); | ||||||
|  |         } catch (MismatchedInputException e) { | ||||||
|  |             // 类型不匹配说明不是json | ||||||
|  |             return null; | ||||||
|         } catch (IOException e) { |         } catch (IOException e) { | ||||||
|             throw new RuntimeException(e); |             throw new RuntimeException(e); | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ package com.ruoyi.common.utils; | |||||||
| import cn.hutool.core.convert.Convert; | import cn.hutool.core.convert.Convert; | ||||||
| import cn.hutool.extra.servlet.ServletUtil; | import cn.hutool.extra.servlet.ServletUtil; | ||||||
| import cn.hutool.http.HttpStatus; | import cn.hutool.http.HttpStatus; | ||||||
|  | import com.ruoyi.common.constant.Constants; | ||||||
| import lombok.AccessLevel; | import lombok.AccessLevel; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
| import org.springframework.http.MediaType; | import org.springframework.http.MediaType; | ||||||
| @@ -10,11 +11,18 @@ import org.springframework.web.context.request.RequestAttributes; | |||||||
| import org.springframework.web.context.request.RequestContextHolder; | import org.springframework.web.context.request.RequestContextHolder; | ||||||
| import org.springframework.web.context.request.ServletRequestAttributes; | import org.springframework.web.context.request.ServletRequestAttributes; | ||||||
|  |  | ||||||
|  | import javax.servlet.ServletRequest; | ||||||
| import javax.servlet.http.HttpServletRequest; | import javax.servlet.http.HttpServletRequest; | ||||||
| import javax.servlet.http.HttpServletResponse; | import javax.servlet.http.HttpServletResponse; | ||||||
| import javax.servlet.http.HttpSession; | import javax.servlet.http.HttpSession; | ||||||
| import java.io.IOException; | import java.io.IOException; | ||||||
|  | import java.io.UnsupportedEncodingException; | ||||||
|  | import java.net.URLDecoder; | ||||||
|  | import java.net.URLEncoder; | ||||||
| import java.nio.charset.StandardCharsets; | import java.nio.charset.StandardCharsets; | ||||||
|  | import java.util.Collections; | ||||||
|  | import java.util.HashMap; | ||||||
|  | import java.util.Map; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * 客户端工具类 |  * 客户端工具类 | ||||||
| @@ -66,6 +74,31 @@ public class ServletUtils extends ServletUtil { | |||||||
|         return Convert.toBool(getRequest().getParameter(name), defaultValue); |         return Convert.toBool(getRequest().getParameter(name), defaultValue); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 获得所有请求参数 | ||||||
|  |      * | ||||||
|  |      * @param request 请求对象{@link ServletRequest} | ||||||
|  |      * @return Map | ||||||
|  |      */ | ||||||
|  |     public static Map<String, String[]> getParams(ServletRequest request) { | ||||||
|  |         final Map<String, String[]> map = request.getParameterMap(); | ||||||
|  |         return Collections.unmodifiableMap(map); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 获得所有请求参数 | ||||||
|  |      * | ||||||
|  |      * @param request 请求对象{@link ServletRequest} | ||||||
|  |      * @return Map | ||||||
|  |      */ | ||||||
|  |     public static Map<String, String> getParamMap(ServletRequest request) { | ||||||
|  |         Map<String, String> params = new HashMap<>(); | ||||||
|  |         for (Map.Entry<String, String[]> entry : getParams(request).entrySet()) { | ||||||
|  |             params.put(entry.getKey(), StringUtils.join(entry.getValue(), ",")); | ||||||
|  |         } | ||||||
|  |         return params; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 获取request |      * 获取request | ||||||
|      */ |      */ | ||||||
| @@ -117,7 +150,7 @@ public class ServletUtils extends ServletUtil { | |||||||
|     public static boolean isAjaxRequest(HttpServletRequest request) { |     public static boolean isAjaxRequest(HttpServletRequest request) { | ||||||
|  |  | ||||||
|         String accept = request.getHeader("accept"); |         String accept = request.getHeader("accept"); | ||||||
|         if (accept != null && accept.contains("application/json")) { |         if (accept != null && accept.contains(MediaType.APPLICATION_JSON_VALUE)) { | ||||||
|             return true; |             return true; | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -139,4 +172,32 @@ public class ServletUtils extends ServletUtil { | |||||||
|         return getClientIP(getRequest()); |         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; | 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 cn.hutool.core.util.IdUtil; | ||||||
| import com.alibaba.excel.EasyExcel; | 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.alibaba.excel.write.style.column.LongestMatchColumnWidthStyleStrategy; | ||||||
| import com.ruoyi.common.convert.ExcelBigNumberConvert; | import com.ruoyi.common.convert.ExcelBigNumberConvert; | ||||||
|  | import com.ruoyi.common.excel.CellMergeStrategy; | ||||||
| import com.ruoyi.common.excel.DefaultExcelListener; | import com.ruoyi.common.excel.DefaultExcelListener; | ||||||
| import com.ruoyi.common.excel.ExcelListener; | import com.ruoyi.common.excel.ExcelListener; | ||||||
| import com.ruoyi.common.excel.ExcelResult; | import com.ruoyi.common.excel.ExcelResult; | ||||||
| @@ -16,7 +24,11 @@ import javax.servlet.ServletOutputStream; | |||||||
| import javax.servlet.http.HttpServletResponse; | import javax.servlet.http.HttpServletResponse; | ||||||
| import java.io.IOException; | import java.io.IOException; | ||||||
| import java.io.InputStream; | import java.io.InputStream; | ||||||
|  | import java.io.OutputStream; | ||||||
|  | import java.io.UnsupportedEncodingException; | ||||||
|  | import java.util.Collection; | ||||||
| import java.util.List; | import java.util.List; | ||||||
|  | import java.util.Map; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Excel相关处理 |  * Excel相关处理 | ||||||
| @@ -69,27 +81,185 @@ public class ExcelUtil { | |||||||
|      * |      * | ||||||
|      * @param list      导出数据集合 |      * @param list      导出数据集合 | ||||||
|      * @param sheetName 工作表的名称 |      * @param sheetName 工作表的名称 | ||||||
|      * @return 结果 |      * @param clazz     实体类 | ||||||
|  |      * @param response  响应体 | ||||||
|      */ |      */ | ||||||
|     public static <T> void exportExcel(List<T> list, String sheetName, Class<T> clazz, HttpServletResponse response) { |     public static <T> void exportExcel(List<T> list, String sheetName, Class<T> clazz, HttpServletResponse response) { | ||||||
|         try { |         try { | ||||||
|             String filename = encodingFilename(sheetName); |             resetResponse(sheetName, response); | ||||||
|             response.reset(); |  | ||||||
|             FileUtils.setAttachmentResponseHeader(response, filename); |  | ||||||
|             response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=UTF-8"); |  | ||||||
|             ServletOutputStream os = response.getOutputStream(); |             ServletOutputStream os = response.getOutputStream(); | ||||||
|             EasyExcel.write(os, clazz) |             exportExcel(list, sheetName, clazz, false, os); | ||||||
|                 .autoCloseStream(false) |  | ||||||
|                 // 自动适配 |  | ||||||
|                 .registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()) |  | ||||||
|                 // 大数值自动转换 防止失真 |  | ||||||
|                 .registerConverter(new ExcelBigNumberConvert()) |  | ||||||
|                 .sheet(sheetName).doWrite(list); |  | ||||||
|         } catch (IOException e) { |         } catch (IOException e) { | ||||||
|             throw new RuntimeException("导出Excel异常"); |             throw new RuntimeException("导出Excel异常"); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 导出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 { | ||||||
|  |             resetResponse(sheetName, response); | ||||||
|  |             ServletOutputStream os = response.getOutputStream(); | ||||||
|  |             exportExcel(list, sheetName, clazz, merge, os); | ||||||
|  |         } catch (IOException e) { | ||||||
|  |             throw new RuntimeException("导出Excel异常"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 导出excel | ||||||
|  |      * | ||||||
|  |      * @param list      导出数据集合 | ||||||
|  |      * @param sheetName 工作表的名称 | ||||||
|  |      * @param clazz     实体类 | ||||||
|  |      * @param os        输出流 | ||||||
|  |      */ | ||||||
|  |     public static <T> void exportExcel(List<T> list, String sheetName, Class<T> clazz, OutputStream os) { | ||||||
|  |         exportExcel(list, sheetName, clazz, false, os); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 导出excel | ||||||
|  |      * | ||||||
|  |      * @param list      导出数据集合 | ||||||
|  |      * @param sheetName 工作表的名称 | ||||||
|  |      * @param clazz     实体类 | ||||||
|  |      * @param merge     是否合并单元格 | ||||||
|  |      * @param os        输出流 | ||||||
|  |      */ | ||||||
|  |     public static <T> void exportExcel(List<T> list, String sheetName, Class<T> clazz, boolean merge, OutputStream os) { | ||||||
|  |         ExcelWriterSheetBuilder builder = EasyExcel.write(os, clazz) | ||||||
|  |             .autoCloseStream(false) | ||||||
|  |             // 自动适配 | ||||||
|  |             .registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()) | ||||||
|  |             // 大数值自动转换 防止失真 | ||||||
|  |             .registerConverter(new ExcelBigNumberConvert()) | ||||||
|  |             .sheet(sheetName); | ||||||
|  |         if (merge) { | ||||||
|  |             // 合并处理器 | ||||||
|  |             builder.registerWriteHandler(new CellMergeStrategy(list, true)); | ||||||
|  |         } | ||||||
|  |         builder.doWrite(list); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 单表多数据模板导出 模板格式为 {.属性} | ||||||
|  |      * | ||||||
|  |      * @param filename     文件名 | ||||||
|  |      * @param templatePath 模板路径 resource 目录下的路径包括模板文件名 | ||||||
|  |      *                     例如: excel/temp.xlsx | ||||||
|  |      *                     重点: 模板文件必须放置到启动类对应的 resource 目录下 | ||||||
|  |      * @param data         模板需要的数据 | ||||||
|  |      * @param response     响应体 | ||||||
|  |      */ | ||||||
|  |     public static void exportTemplate(List<Object> data, String filename, String templatePath, HttpServletResponse response) { | ||||||
|  |         try { | ||||||
|  |             resetResponse(filename, response); | ||||||
|  |             ServletOutputStream os = response.getOutputStream(); | ||||||
|  |             exportTemplate(data, templatePath, os); | ||||||
|  |         } catch (IOException e) { | ||||||
|  |             throw new RuntimeException("导出Excel异常"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 单表多数据模板导出 模板格式为 {.属性} | ||||||
|  |      * | ||||||
|  |      * @param templatePath 模板路径 resource 目录下的路径包括模板文件名 | ||||||
|  |      *                     例如: excel/temp.xlsx | ||||||
|  |      *                     重点: 模板文件必须放置到启动类对应的 resource 目录下 | ||||||
|  |      * @param data         模板需要的数据 | ||||||
|  |      * @param os           输出流 | ||||||
|  |      */ | ||||||
|  |     public static void exportTemplate(List<Object> data, String templatePath, OutputStream os) { | ||||||
|  |         ClassPathResource templateResource = new ClassPathResource(templatePath); | ||||||
|  |         ExcelWriter excelWriter = EasyExcel.write(os) | ||||||
|  |             .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(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 多表多数据模板导出 模板格式为 {key.属性} | ||||||
|  |      * | ||||||
|  |      * @param filename     文件名 | ||||||
|  |      * @param templatePath 模板路径 resource 目录下的路径包括模板文件名 | ||||||
|  |      *                     例如: excel/temp.xlsx | ||||||
|  |      *                     重点: 模板文件必须放置到启动类对应的 resource 目录下 | ||||||
|  |      * @param data         模板需要的数据 | ||||||
|  |      * @param response     响应体 | ||||||
|  |      */ | ||||||
|  |     public static void exportTemplateMultiList(Map<String, Object> data, String filename, String templatePath, HttpServletResponse response) { | ||||||
|  |         try { | ||||||
|  |             resetResponse(filename, response); | ||||||
|  |             ServletOutputStream os = response.getOutputStream(); | ||||||
|  |             exportTemplateMultiList(data, templatePath, os); | ||||||
|  |         } catch (IOException e) { | ||||||
|  |             throw new RuntimeException("导出Excel异常"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 多表多数据模板导出 模板格式为 {key.属性} | ||||||
|  |      * | ||||||
|  |      * @param templatePath 模板路径 resource 目录下的路径包括模板文件名 | ||||||
|  |      *                     例如: excel/temp.xlsx | ||||||
|  |      *                     重点: 模板文件必须放置到启动类对应的 resource 目录下 | ||||||
|  |      * @param data         模板需要的数据 | ||||||
|  |      * @param os           输出流 | ||||||
|  |      */ | ||||||
|  |     public static void exportTemplateMultiList(Map<String, Object> data, String templatePath, OutputStream os) { | ||||||
|  |         ClassPathResource templateResource = new ClassPathResource(templatePath); | ||||||
|  |         ExcelWriter excelWriter = EasyExcel.write(os) | ||||||
|  |             .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(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 重置响应体 | ||||||
|  |      */ | ||||||
|  |     private static void resetResponse(String sheetName, HttpServletResponse response) throws UnsupportedEncodingException { | ||||||
|  |         String filename = encodingFilename(sheetName); | ||||||
|  |         FileUtils.setAttachmentResponseHeader(response, filename); | ||||||
|  |         response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=UTF-8"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 解析导出值 0=男,1=女,2=未知 |      * 解析导出值 0=男,1=女,2=未知 | ||||||
|      * |      * | ||||||
| @@ -103,7 +273,7 @@ public class ExcelUtil { | |||||||
|         String[] convertSource = converterExp.split(","); |         String[] convertSource = converterExp.split(","); | ||||||
|         for (String item : convertSource) { |         for (String item : convertSource) { | ||||||
|             String[] itemArray = item.split("="); |             String[] itemArray = item.split("="); | ||||||
|             if (StringUtils.containsAny(separator, propertyValue)) { |             if (StringUtils.containsAny(propertyValue, separator)) { | ||||||
|                 for (String value : propertyValue.split(separator)) { |                 for (String value : propertyValue.split(separator)) { | ||||||
|                     if (itemArray[0].equals(value)) { |                     if (itemArray[0].equals(value)) { | ||||||
|                         propertyString.append(itemArray[1] + separator); |                         propertyString.append(itemArray[1] + separator); | ||||||
| @@ -132,7 +302,7 @@ public class ExcelUtil { | |||||||
|         String[] convertSource = converterExp.split(","); |         String[] convertSource = converterExp.split(","); | ||||||
|         for (String item : convertSource) { |         for (String item : convertSource) { | ||||||
|             String[] itemArray = item.split("="); |             String[] itemArray = item.split("="); | ||||||
|             if (StringUtils.containsAny(separator, propertyValue)) { |             if (StringUtils.containsAny(propertyValue, separator)) { | ||||||
|                 for (String value : propertyValue.split(separator)) { |                 for (String value : propertyValue.split(separator)) { | ||||||
|                     if (itemArray[1].equals(value)) { |                     if (itemArray[1].equals(value)) { | ||||||
|                         propertyString.append(itemArray[0] + separator); |                         propertyString.append(itemArray[0] + separator); | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user