mirror of
				https://github.com/dromara/RuoYi-Vue-Plus.git
				synced 2025-10-25 11:23:43 +08:00 
			
		
		
		
	Compare commits
	
		
			595 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 203233fbaf | ||
|  | 5bc6b1de3a | ||
|  | 7445934371 | ||
|  | 521596bc12 | ||
|  | 00d85947b0 | ||
|  | a62eba5c37 | ||
|  | 102d854743 | ||
|  | 953de2fb1c | ||
|  | c241a715a4 | ||
|  | 3a51c58b82 | ||
|  | 0ac4a63e91 | ||
|  | e981e2f69f | ||
|  | 296392b094 | ||
|  | e1e409d89e | ||
|  | 7fb4d17913 | ||
|  | 469c334b95 | ||
|  | ddcb13c7c3 | ||
|  | c4582b085f | ||
|  | 45097ccb8f | ||
|  | 663e22ac29 | ||
|  | 0403c2c274 | ||
|  | 48fdfffc42 | ||
|  | 6037edb621 | ||
|  | ce8a82383d | ||
|  | e1390615b7 | ||
|  | 860afb8738 | ||
|  | e509d95af9 | ||
|  | 4cefcf6ada | ||
|  | affbce1957 | ||
|  | 5d97541b8f | ||
|  | 8ed5c75c6a | ||
|  | a1adfa2e93 | ||
|  | 1999000322 | ||
|  | 677fe7e2a5 | ||
|  | 8704cae182 | ||
|  | 7dde869eba | ||
|  | cb9a3c36e6 | ||
|  | d131a833ab | ||
|  | be1398137a | ||
|  | 183e39f1cd | ||
|  | a6fc191df2 | ||
|  | 87981fb80c | ||
|  | 96ab48396c | ||
|  | 40ec6584c5 | ||
|  | d811cb8e04 | ||
|  | d3ad832ded | ||
|  | b0cc1a913e | ||
|  | b0faebc5e6 | ||
|  | c45ffaec0f | ||
|  | 78c91d0733 | ||
|  | 3f1e5702a2 | ||
|  | 6bed331971 | ||
|  | 793737d69d | ||
|  | 4e8f2b130e | ||
|  | ac6fe634dc | ||
|  | 1e1616ceb0 | ||
|  | 3046362ff4 | ||
|  | f2956c322e | ||
|  | 754c22f8a6 | ||
|  | 16685a8e0c | ||
|  | e2692aa9e9 | ||
|  | 9b0938e0d6 | ||
|  | c4f69b466a | ||
|  | 3acbf6efee | ||
|  | 17cf957052 | ||
|  | 624a14d2de | ||
|  | 0682efb472 | ||
|  | abc3acf489 | ||
|  | f9ba1df798 | ||
|  | 9c7274b776 | ||
|  | bf9f54c021 | ||
|  | 26369657e3 | ||
|  | 3d9b05a61a | ||
|  | c93b666140 | ||
|  | 0f16051024 | ||
|  | 81f7a59caa | ||
|  | 2c4306b436 | ||
|  | 3a9379555e | ||
|   | e7ed4afe79 | ||
|  | 213cc8a3aa | ||
|  | 323c290ee9 | ||
|  | 5602976ee1 | ||
|  | c709b44786 | ||
|  | f34740ad03 | ||
|  | 51feaf7b99 | ||
|  | c5bc6a97c9 | ||
|  | 864e5d695f | ||
|  | 635e36a882 | ||
|  | 002a880e8b | ||
|  | 5d58c27720 | ||
|  | 2709690f81 | ||
|  | 5f84ab968b | ||
|  | cf1c18184e | ||
|  | 8d681eda21 | ||
|  | e2b169e07a | ||
|  | e58b010f4a | ||
|  | 23e85c5b9c | ||
|  | 4198dce4f0 | ||
|  | b537b1ecd2 | ||
|  | 22a8057ea4 | ||
|  | f1232a60f3 | ||
|  | e2e7cee58b | ||
|  | 7abd419d8c | ||
|  | 3774d71bac | ||
|  | 29118ae78c | ||
|  | f708492681 | ||
|  | bf34781f04 | ||
|  | 0e6464d344 | ||
|  | de3adb2230 | ||
|  | 653bf84929 | ||
|  | 9e796943b8 | ||
|  | c4daabafbe | ||
|  | 8b45cb1963 | ||
|  | ee5fc9507a | ||
|  | 905637d70f | ||
|  | cb3ead140b | ||
|  | aabd3990a0 | ||
|   | 937de85647 | ||
|  | be700f7ddc | ||
|  | b2eceed943 | ||
|  | 40d652ab82 | ||
|  | b24e6348c3 | ||
|  | 2c64c66ed1 | ||
|  | 977e9a119c | ||
|  | bfe1b2ae50 | ||
|  | 23b7e5f8c7 | ||
|  | de0fe48a72 | ||
|  | 6bc2bac957 | ||
|  | 0cd6712b2a | ||
|  | 7e62401bac | ||
|  | 3a67a6599f | ||
|  | 6381034b1f | ||
|  | eb9e4b5eef | ||
|  | 052a8f71a3 | ||
|  | 58b4df7bb5 | ||
|  | 57bbf64fbf | ||
|  | d0aa35e47e | ||
|  | 680ec7bbb7 | ||
|  | 116927e903 | ||
|  | 146c268dff | ||
|  | 992595f999 | ||
|  | ef2bd5e6a4 | ||
|  | c1b9b95581 | ||
|  | 5ebf079b4f | ||
|  | 301111fffd | ||
|  | 2a74329206 | ||
|  | c1ed482c59 | ||
|  | 5429bb091c | ||
|  | 818e681cdf | ||
|  | 1962ebc370 | ||
|  | c97cdbe5d4 | ||
|  | eb22895abd | ||
|  | 72e0d9aeb6 | ||
|  | b4cbefd2b8 | ||
|  | 84429206dc | ||
|  | 15a7a83966 | ||
|  | f12afaab5d | ||
|  | 60c48fd27c | ||
|  | 1e423facaa | ||
|  | af4b0cb107 | ||
|  | 703815dae7 | ||
|  | 29e7c5dc41 | ||
|  | eb16b9e6a0 | ||
|  | 743ebc77d6 | ||
|  | 58e7849fc4 | ||
|  | 49f20f7d9f | ||
|  | ae0d4dd6e5 | ||
|  | afdc4b103d | ||
|  | ddc43b3057 | ||
|  | 86a8f5a700 | ||
|  | 4a00998f13 | ||
|  | 9465803730 | ||
|  | e042f7f6b3 | ||
|  | b96f2c4f27 | ||
|  | f1990e6ce8 | ||
|  | 6c9f1f1994 | ||
|  | 776b235cec | ||
|  | bbe672208f | ||
|  | c92c99c8da | ||
|  | 359ec45b05 | ||
|  | 352caa3e93 | ||
|  | 348938eb96 | ||
|  | cb59082bad | ||
|  | 1cabccc9a8 | ||
|  | 0c09adfe0a | ||
|  | e86765c6bc | ||
|  | 16e47a8466 | ||
|  | 17acf2ba7a | ||
|  | 0217706960 | ||
|  | 9e9a1920ac | ||
|   | fe03c2fb92 | ||
|  | 6d7f1adea8 | ||
|  | a37b7cafad | ||
|  | ea0b82997f | ||
|  | 3480e36dcd | ||
|  | 8df34e50fa | ||
|  | d82a753ed5 | ||
|  | 9004f8fd9f | ||
|  | 77ea3ce514 | ||
|  | a2e9a4e899 | ||
|  | f51f96d01c | ||
|  | 27090fe116 | ||
|  | 66540b5e56 | ||
|  | 9a53ed796b | ||
|  | 1c081a5892 | ||
|  | 508882d21e | ||
|  | beaf3efdf5 | ||
|  | 8eb43cbea1 | ||
|  | 17ecce9978 | ||
|  | 32e96434d2 | ||
|  | e2525b1666 | ||
|  | 2dd7462053 | ||
|  | 525921c129 | ||
|  | 6df7b34b4a | ||
|  | dbf4d6c713 | ||
|  | 1dc718291b | ||
|  | 82aeb75e31 | ||
|  | 5796c0aa9f | ||
|  | 1b85f0f0d3 | ||
|  | af08632c37 | ||
|  | 10b5b0e82a | ||
|  | ee8922729e | ||
|  | 1f303ff1de | ||
|  | 64b0748bf1 | ||
|  | 34639a8943 | ||
|  | da69cfad3e | ||
|  | 2f5a6e996e | ||
|  | 53bad85b00 | ||
|  | 73f39deb01 | ||
|  | cd4520836e | ||
|  | 69c5e781ac | ||
|  | 7a6660dd8d | ||
|  | ab711047b4 | ||
|  | bd50596038 | ||
|  | 01d94ecd89 | ||
|  | c8ea31541d | ||
|  | b6c1c0917d | ||
|  | f89bc278f2 | ||
|  | c4032aae88 | ||
|  | c763a9ab83 | ||
|  | 53ad4969b0 | ||
|  | 611b147cc3 | ||
|  | 3166b99f74 | ||
|  | bb2836b6f1 | ||
|  | 06588f3ad4 | ||
|  | 6b14778691 | ||
|  | 6bbe11d494 | ||
|  | 8bff33d672 | ||
|  | c616a55647 | ||
|  | a7586639ab | ||
|  | 55c8ea7082 | ||
|  | 7bde8054d9 | ||
|  | b0909dbe3d | ||
|  | 19da4f7522 | ||
|  | cd6bfb2a3a | ||
|  | 9b9f424e6c | ||
|  | 08ba802e12 | ||
|  | 33c2d8e833 | ||
|  | 7193ee08c9 | ||
|  | 8573ffdd7f | ||
|  | d7947b8be5 | ||
|  | 4a9aafe853 | ||
|  | 47e64c9290 | ||
|  | 5cbd1d26c2 | ||
|  | 02423ec320 | ||
|  | 7accca933f | ||
|  | 6f4e3b478e | ||
|  | b7c0e76da2 | ||
|  | 9448782f45 | ||
|  | 50fbfe2cb4 | ||
|  | b7ce933971 | ||
|  | 35ad306730 | ||
|  | c8d94da4fb | ||
|  | 77a849992e | ||
|  | 98aa0b9f18 | ||
|  | bda0e0ec64 | ||
|  | effd504d48 | ||
|  | aec0e22747 | ||
|  | bbe9dc7dc8 | ||
|  | da41c65213 | ||
|  | ec2cbac35d | ||
|  | 005e7df1b8 | ||
|  | cf4bc969be | ||
|  | 9bc1e4ac1e | ||
|  | 9ec982a23e | ||
|  | 5110961eb9 | ||
|  | 76dc239875 | ||
|  | be774f5ca6 | ||
|  | 1275e416cd | ||
|  | 2d894c1309 | ||
|  | e4b405491f | ||
|  | 745c40cd00 | ||
|  | 014550c816 | ||
|  | 9968912322 | ||
|  | 0b79ada24c | ||
|  | 60edd6148b | ||
|  | bb587607e1 | ||
|  | d34be960cb | ||
|  | 204d882a8e | ||
|  | ffedfd89a1 | ||
|  | ef77672466 | ||
|  | 789858199b | ||
|  | eb37e25ad3 | ||
|  | 2e5702dc86 | ||
|  | d058fd01ca | ||
|  | 28d69701c9 | ||
|  | 3303b9aa59 | ||
|  | 7b6dc06a06 | ||
|  | 512108cf5a | ||
|  | d5c33b5604 | ||
|  | f78df97bcc | ||
|  | 8de7fef44a | ||
|  | f51c654c58 | ||
|  | ee7c5fbbd9 | ||
|  | f2892a672a | ||
|  | 8d69be093e | ||
|  | 13c776ec34 | ||
|  | 2f0bc7489b | ||
|  | 21b0b4baae | ||
|  | feb36d6f39 | ||
|  | 3a052c3ef8 | ||
|  | 1b1f4d097e | ||
|  | af3a8dd245 | ||
|  | 9e061669b1 | ||
|  | b9578e28b7 | ||
|  | b8e389bb7e | ||
|  | f687bde93b | ||
|  | bdddef0aa0 | ||
|  | d3a1b496fb | ||
|  | cff90251ef | ||
|  | 491e6f6ebc | ||
|  | 9121a5ac9b | ||
|  | cbd05fbc89 | ||
|  | f2b2fe3103 | ||
|  | 48ee686ade | ||
|  | a558939d0f | ||
|  | f4e639ac32 | ||
|  | 8f307bf05b | ||
|  | b25137f129 | ||
|  | ddc8acd978 | ||
|  | 4450c06163 | ||
|  | 210d198fa3 | ||
|  | 3f50eb158d | ||
|  | 0d9cc23d3a | ||
|  | 21968ea8e6 | ||
|  | 4e1f970195 | ||
|  | 093e7212cf | ||
|  | 1bfcbfa456 | ||
|  | 8d0d85444a | ||
|  | a345bcc0a0 | ||
|  | bec7f8c802 | ||
|  | 5b7c787b31 | ||
|  | 0ebcac34e3 | ||
|  | 8ecbdfd472 | ||
|  | 4fc7401b67 | ||
|  | 0f161d7d6c | ||
|  | 5cfffb6367 | ||
|  | fd8d1a243d | ||
|  | 44a796ed57 | ||
|  | 4e911a1a98 | ||
|  | 8a555eb191 | ||
|  | 9c22f7a0d0 | ||
|  | 2ac0468288 | ||
|  | 9d84e4065e | ||
|  | d868d6c8f1 | ||
|  | efc4f86d88 | ||
|  | 1983c51747 | ||
|  | 30819dab84 | ||
|  | 083011c018 | ||
|  | fd6db2de2c | ||
|  | 750cbe9312 | ||
|  | 851f92cf80 | ||
|  | 798bc43f6a | ||
|  | 77fa4e0aa8 | ||
|  | 7862d58f01 | ||
|  | 27be95a15b | ||
|  | 4ba109a2b7 | ||
|  | bfd5a07b13 | ||
|  | dffd5bf02a | ||
|  | df38aad2b5 | ||
|  | c36bc040d0 | ||
|  | 1dd9a952d5 | ||
|  | 2e92773b8c | ||
|  | 92e15c2f16 | ||
|  | ae21e0c3d0 | ||
|  | 1ed1752f67 | ||
|  | 181166f4c0 | ||
|  | 5b39017708 | ||
|  | 210570c005 | ||
|  | b022c12419 | ||
|  | 42f5735c55 | ||
|  | 4db6a638d4 | ||
|  | a29550445d | ||
|  | 63a240b1d4 | ||
|  | 6ed424f89e | ||
|  | c4b62788b5 | ||
|  | 0c1d6e111a | ||
|  | 65ae5ab362 | ||
|  | abca91c18f | ||
|  | 5e0dcea4f1 | ||
|  | c3f23156ee | ||
|  | e0d921b1ac | ||
|  | 930eebdd1e | ||
|  | 2670877343 | ||
|  | 92dfee0275 | ||
|  | 3b6ced8185 | ||
|  | c00c9ba386 | ||
|  | 7dfa67c88c | ||
|  | 64cfb487d6 | ||
|  | 94846d281f | ||
|  | aa29f1cec5 | ||
|  | 91ce812225 | ||
|  | 22c681a8d4 | ||
|  | 50cae03b6f | ||
|  | f1dd87abd5 | ||
|  | 04c2d845dc | ||
|  | 6ed2779d15 | ||
|  | 1bab808025 | ||
|  | fbfb53b169 | ||
|  | 520cbec1b3 | ||
|  | 53cb1c801f | ||
|  | 169e2e9ae9 | ||
|  | 2406d1063d | ||
|  | 3bf946392d | ||
|  | f8c8770737 | ||
|  | daa2b734ef | ||
|  | 0673aeec7d | ||
|  | 99ce2887f4 | ||
|  | a45fa3e2ec | ||
|  | 7ac4ca554e | ||
|  | dbb7ce4c29 | ||
|  | c55157dce0 | ||
|  | 51a558d690 | ||
|  | 7b8747c831 | ||
|  | 9df7f92d9c | ||
|  | aa81264d04 | ||
|  | 9dd57179da | ||
|  | 700b441bbb | ||
|  | 82345cbf47 | ||
|  | bde5353547 | ||
|  | d303e6524a | ||
|  | 0ad45df788 | ||
|  | b14fa6e12f | ||
|  | 5a150544c6 | ||
|  | c82bde9792 | ||
|  | 0ec6fbd34c | ||
|  | 4f301205fe | ||
|  | 876145dcec | ||
|  | 662df23d4d | ||
|  | 4ffb0acc35 | ||
|  | d1b1f06fc4 | ||
|  | 408da809a0 | ||
|  | f0a9768d8e | ||
|  | 69edf436da | ||
|  | cdb509a4fa | ||
|  | f8c98a1f48 | ||
|  | 6a813c0e41 | ||
|  | 891f71b1c2 | ||
|  | a3ecc71558 | ||
|  | 8095db9e28 | ||
|  | 183af84117 | ||
|  | f55fa60f4d | ||
|  | 0b761fe52b | ||
|  | d4b4eb4004 | ||
|  | 3cd8b6da30 | ||
|  | 9ed5b521d6 | ||
|  | 84871d4cfb | ||
|  | a28dd0fe40 | ||
|  | 5cf45e8663 | ||
|  | ca59c2a512 | ||
|  | 856f0767b4 | ||
|  | 0fdd97203f | ||
|  | b4645daf2a | ||
|  | 11e4b56313 | ||
|  | 5e3f504c80 | ||
|  | f6c96bd246 | ||
|  | 2f558094d1 | ||
|  | 6e21c64728 | ||
|  | 37794b5738 | ||
|  | 4b814886b2 | ||
|  | 1fd09f8ca3 | ||
|  | 37290d023e | ||
|  | 7c5d645639 | ||
|  | 4bfdf357a2 | ||
|  | 06f42fa0ea | ||
|  | 6fde4fb4ef | ||
|  | 402bad1e88 | ||
|  | 3ab03bf0e7 | ||
|  | d957914c30 | ||
|  | a49ed727b6 | ||
|  | c6746ed788 | ||
|  | e7dc439ed3 | ||
|  | 380cb2db29 | ||
|  | c1036a3df2 | ||
|  | 70b8340a5f | ||
|  | 80aa636348 | ||
|  | d56c77d6ba | ||
|  | 1ac98f522f | ||
|  | 0af7c94ae6 | ||
|  | e7242d4011 | ||
|  | fcea03a975 | ||
|  | 9b2a7c4b45 | ||
|  | 0ea206dfe1 | ||
|  | cb80568828 | ||
|  | 3baad24912 | ||
|  | f8bb302fef | ||
|  | d696ed9841 | ||
|  | 2aabdb83ad | ||
|  | 895e19768b | ||
|  | 173261e644 | ||
|  | 45ac0f23e1 | ||
|  | a8d5644f2e | ||
|  | c59122a194 | ||
|  | 287253166a | ||
|  | f0bbc53a0c | ||
|  | 07590f30f6 | ||
|  | 08565363a9 | ||
|  | 69b8a3cfec | ||
|  | d09ea16f46 | ||
|  | 234f74f8f0 | ||
|  | fdc6c89bc2 | ||
|  | 4d22f5928b | ||
|  | 296437569c | ||
|  | 922a8345e2 | ||
|  | 2787212362 | ||
|  | 5cce09b5c2 | ||
|  | 0d755d2266 | ||
|  | c681862724 | ||
|  | e8483729b1 | ||
|  | e8225fbdee | ||
|  | 075ef635c6 | ||
|  | fbdae78675 | ||
|  | 949eeb213d | ||
|  | 771a1045ee | ||
|  | bf344753fb | ||
|  | c67cc725d4 | ||
|  | bff9720b6f | ||
|  | a5835aa0c2 | ||
|  | 5057628b52 | ||
|  | c2208270bd | ||
|  | c19b3c6a7a | ||
|  | a0eaa5a0f6 | ||
|  | 2d2e06dfc0 | ||
|  | ee11117fd7 | ||
|  | 13d2ce975f | ||
|  | 1f963caed6 | ||
|  | f0eae7cdb5 | ||
|  | e94ed91f0b | ||
|  | 97f453efc7 | ||
|  | 27a2943362 | ||
|  | 226f1ae89a | ||
|  | 3ba8323cb7 | ||
|  | 1e1c79f163 | ||
|  | f93ccd68e9 | ||
|  | 912b14c288 | ||
|  | 2d264ad558 | ||
|  | 1d90aee7ce | ||
|  | be27328d81 | ||
|  | fedb04611b | ||
|  | 0faf7a5c59 | ||
|  | 5493557820 | ||
|  | 4c75b5b514 | ||
|  | eca2bb4824 | ||
|  | 558d76c6bb | ||
|  | 27df5aba2f | ||
|  | 8557765c9b | ||
|  | 57da2e33a6 | ||
|  | afbc78b672 | ||
|  | 4ce596aa9c | ||
|  | 2ae3681286 | ||
|  | 71a2a8245d | ||
|  | 40f28e9869 | ||
|  | 04eefe2ba9 | ||
|  | 7d1c53f927 | ||
|  | a986b0e75d | ||
|  | a12f6472da | ||
|  | ec004e9ba3 | ||
|  | f3aff34602 | ||
|  | df6867ee05 | ||
|  | 5911e94b65 | ||
|  | 8ae8c5498d | ||
|  | 61ef4c71bc | ||
|  | 2023de973e | ||
|  | 0fc796b90d | ||
|  | ac9e3f2ca4 | ||
|  | 68e57803f3 | ||
|  | 25f89bd236 | ||
|  | d270f7ac8b | ||
|  | c7dcbd5c3b | ||
|  | a3e23b259c | ||
|  | 02bd7ec552 | ||
|  | 41242cd4aa | ||
|  | b0996853d1 | ||
|  | c1df1dc7a0 | ||
|  | f3c6513d5a | 
							
								
								
									
										50
									
								
								.gitee/ISSUE_TEMPLATE/bug.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								.gitee/ISSUE_TEMPLATE/bug.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,50 @@ | |||||||
|  | name: Bug 反馈 | ||||||
|  | description: 当你使用过程中发现了一个 Bug,导致应用崩溃或抛出异常,或者有一个组件存在问题,或者某些地方看起来不对劲,请在这里反馈。 | ||||||
|  | title: "[Bug]: " | ||||||
|  | labels: ["bug"] | ||||||
|  | body: | ||||||
|  |   - type: textarea | ||||||
|  |     id: version | ||||||
|  |     attributes: | ||||||
|  |       label: 版本 | ||||||
|  |       description: 你当前正在使用我们软件的哪个版本(pom文件内的版本号)? | ||||||
|  |       value: | | ||||||
|  |         jdk版本(带上尾号): 例如 17.0.8 | ||||||
|  |         框架版本(项目启动时输出的版本号): 例如 5.1.1 | ||||||
|  |         其他依赖版本(你觉得有必要的): | ||||||
|  |     validations: | ||||||
|  |       required: true | ||||||
|  |   - type: checkboxes | ||||||
|  |     attributes: | ||||||
|  |       label: 功能不好用不会用是否已经看过项目文档? | ||||||
|  |       options: | ||||||
|  |         - label: https://plus-doc.dromara.org | ||||||
|  |           required: true | ||||||
|  |   - type: checkboxes | ||||||
|  |     attributes: | ||||||
|  |       label: 这个问题是否已经存在? | ||||||
|  |       options: | ||||||
|  |         - label: 我已经搜索过现有的问题 (https://gitee.com/dromara/RuoYi-Vue-Plus/issues) | ||||||
|  |           required: true | ||||||
|  |   - type: textarea | ||||||
|  |     attributes: | ||||||
|  |       label: 希望结果 | ||||||
|  |       description: 想知道你觉得怎么样是正常或者合理的。 | ||||||
|  |     validations: | ||||||
|  |       required: true | ||||||
|  |   - type: markdown | ||||||
|  |     attributes: | ||||||
|  |       label: 如何复现 | ||||||
|  |       description: 请详细告诉我们如何复现你遇到的问题。 | ||||||
|  |       value: | | ||||||
|  |         如涉及代码,可提供一个最小代码示例,并使用```附上它,或者截图均可,越详细越好。<br> | ||||||
|  |         大多数问题都是:代码编写错误问题,逻辑问题,或者用法错误等问题。 | ||||||
|  |     validations: | ||||||
|  |       required: true | ||||||
|  |   - type: textarea | ||||||
|  |     attributes: | ||||||
|  |       label: 相关代码与报错信息(请勿发混乱格式) | ||||||
|  |       description: 如果可以的话,上传任何关于 bug 的截图。 | ||||||
|  |       value: | | ||||||
|  |         [在这里上传图片] | ||||||
|  |  | ||||||
							
								
								
									
										5
									
								
								.gitee/ISSUE_TEMPLATE/config.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.gitee/ISSUE_TEMPLATE/config.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | blank_issues_enabled: false | ||||||
|  | contact_links: | ||||||
|  |   - name: RuoYi-Vue-Plus 文档中心 | ||||||
|  |     url: https://plus-doc.dromara.org | ||||||
|  |     about: 提供 RuoYi-Vue-Plus 搭建使用指南、平台基本开发使用方式、介绍、基础知识和常见问题解答 | ||||||
							
								
								
									
										43
									
								
								.gitee/ISSUE_TEMPLATE/feature.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								.gitee/ISSUE_TEMPLATE/feature.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | |||||||
|  | name: 功能建议 | ||||||
|  | description: 对本项目提出一个功能建议。 | ||||||
|  | title: "[功能建议]: " | ||||||
|  | labels: ["enhancement"] | ||||||
|  | body: | ||||||
|  |   - type: markdown | ||||||
|  |     attributes: | ||||||
|  |       value: | | ||||||
|  |         感谢提出功能建议,我们将仔细考虑!请持续关注该issues,在加入计划后我们会有贡献者设置为负责人,同时状态成为进行中。 | ||||||
|  |   - type: textarea | ||||||
|  |     id: related-problem | ||||||
|  |     attributes: | ||||||
|  |       label: 你的功能建议是否和某个问题相关? | ||||||
|  |       description: 清晰并简洁地描述问题是什么,例如,当我...时,我总是感到困扰。 | ||||||
|  |     validations: | ||||||
|  |       required: false | ||||||
|  |   - type: textarea | ||||||
|  |     id: desired-solution | ||||||
|  |     attributes: | ||||||
|  |       label: 你希望看到什么解决方案? | ||||||
|  |       description: 清晰并简洁地描述你希望发生的事情。 | ||||||
|  |     validations: | ||||||
|  |       required: true | ||||||
|  |   - type: textarea | ||||||
|  |     id: alternatives | ||||||
|  |     attributes: | ||||||
|  |       label: 你考虑过哪些替代方案? | ||||||
|  |       description: 清晰并简洁地描述你考虑过的任何替代解决方案或功能。 | ||||||
|  |     validations: | ||||||
|  |       required: false | ||||||
|  |   - type: textarea | ||||||
|  |     id: additional-context | ||||||
|  |     attributes: | ||||||
|  |       label: 你有其他上下文或截图吗? | ||||||
|  |       description: 在此处添加有关功能请求的任何其他上下文或截图。 | ||||||
|  |     validations: | ||||||
|  |       required: false | ||||||
|  |   - type: checkboxes | ||||||
|  |     attributes: | ||||||
|  |       label: 意向参与贡献 | ||||||
|  |       options: | ||||||
|  |         - label: 我有意向参与具体功能的开发实现并将代码贡献回到上游社区。 | ||||||
|  |           required: false | ||||||
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -44,3 +44,5 @@ nbdist/ | |||||||
| !*/build/*.java | !*/build/*.java | ||||||
| !*/build/*.html | !*/build/*.html | ||||||
| !*/build/*.xml | !*/build/*.xml | ||||||
|  |  | ||||||
|  | .flattened-pom.xml | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
|   <configuration default="false" name="ruoyi-monitor-admin" type="docker-deploy" factoryName="dockerfile" server-name="Docker"> |   <configuration default="false" name="ruoyi-monitor-admin" type="docker-deploy" factoryName="dockerfile" server-name="Docker"> | ||||||
|     <deployment type="dockerfile"> |     <deployment type="dockerfile"> | ||||||
|       <settings> |       <settings> | ||||||
|         <option name="imageTag" value="ruoyi/ruoyi-monitor-admin:4.5.0" /> |         <option name="imageTag" value="ruoyi/ruoyi-monitor-admin:5.1.2" /> | ||||||
|         <option name="buildOnly" value="true" /> |         <option name="buildOnly" value="true" /> | ||||||
|         <option name="sourceFilePath" value="ruoyi-extend/ruoyi-monitor-admin/Dockerfile" /> |         <option name="sourceFilePath" value="ruoyi-extend/ruoyi-monitor-admin/Dockerfile" /> | ||||||
|       </settings> |       </settings> | ||||||
|   | |||||||
| @@ -1,10 +1,10 @@ | |||||||
| <component name="ProjectRunConfigurationManager"> | <component name="ProjectRunConfigurationManager"> | ||||||
|   <configuration default="false" name="ruoyi-xxl-job-admin" type="docker-deploy" factoryName="dockerfile" server-name="Docker"> |   <configuration default="false" name="ruoyi-powerjob-server" type="docker-deploy" factoryName="dockerfile" server-name="Docker"> | ||||||
|     <deployment type="dockerfile"> |     <deployment type="dockerfile"> | ||||||
|       <settings> |       <settings> | ||||||
|         <option name="imageTag" value="ruoyi/ruoyi-xxl-job-admin:4.5.0" /> |         <option name="imageTag" value="ruoyi/ruoyi-powerjob-server:5.1.2" /> | ||||||
|         <option name="buildOnly" value="true" /> |         <option name="buildOnly" value="true" /> | ||||||
|         <option name="sourceFilePath" value="ruoyi-extend/ruoyi-xxl-job-admin/Dockerfile" /> |         <option name="sourceFilePath" value="ruoyi-extend/ruoyi-powerjob-server/Dockerfile" /> | ||||||
|       </settings> |       </settings> | ||||||
|     </deployment> |     </deployment> | ||||||
|     <method v="2" /> |     <method v="2" /> | ||||||
| @@ -2,7 +2,7 @@ | |||||||
|   <configuration default="false" name="ruoyi-server" type="docker-deploy" factoryName="dockerfile" server-name="Docker"> |   <configuration default="false" name="ruoyi-server" type="docker-deploy" factoryName="dockerfile" server-name="Docker"> | ||||||
|     <deployment type="dockerfile"> |     <deployment type="dockerfile"> | ||||||
|       <settings> |       <settings> | ||||||
|         <option name="imageTag" value="ruoyi/ruoyi-server:4.5.0" /> |         <option name="imageTag" value="ruoyi/ruoyi-server:5.1.2" /> | ||||||
|         <option name="buildOnly" value="true" /> |         <option name="buildOnly" value="true" /> | ||||||
|         <option name="sourceFilePath" value="ruoyi-admin/Dockerfile" /> |         <option name="sourceFilePath" value="ruoyi-admin/Dockerfile" /> | ||||||
|       </settings> |       </settings> | ||||||
|   | |||||||
							
								
								
									
										280
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										280
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,163 +1,185 @@ | |||||||
|  | <img src="https://foruda.gitee.com/images/1679673773341074847/178e8451_1766278.png" width="50%" height="50%"> | ||||||
|  | <div style="height: 10px; clear: both;"></div> | ||||||
|  |  | ||||||
|  | - - - | ||||||
| ## 平台简介 | ## 平台简介 | ||||||
| [](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus) |  | ||||||
| [](https://github.com/JavaLionLi/RuoYi-Vue-Plus) | [](https://gitee.com/dromara/RuoYi-Vue-Plus) | ||||||
| [](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/blob/master/LICENSE) | [](https://github.com/dromara/RuoYi-Vue-Plus) | ||||||
|  | [](https://gitee.com/dromara/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/dromara/RuoYi-Vue-Plus) | ||||||
| []() | []() | ||||||
| []() | []() | ||||||
| []() | []() | ||||||
|  |  | ||||||
| > RuoYi-Vue-Plus 是重写 RuoYi-Vue 针对 `分布式集群` 场景全方位升级(不兼容原框架) | > RuoYi-Vue-Plus 是重写 RuoYi-Vue 针对 `分布式集群与多租户` 场景全方位升级(不兼容原框架) | ||||||
|  |  | ||||||
| > 项目代码、文档 均开源免费可商用 遵循开源协议在项目中保留开源协议文件即可<br> | > 项目代码、文档 均开源免费可商用 遵循开源协议在项目中保留开源协议文件即可<br> | ||||||
| 活到老写到老 为兴趣而开源 为学习而开源 为让大家真正可以学到技术而开源 | 活到老写到老 为兴趣而开源 为学习而开源 为让大家真正可以学到技术而开源 | ||||||
|  |  | ||||||
| > 系统演示: [传送门](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=4836388&doc_id=1469725) | > 系统演示: [传送门](https://plus-doc.dromara.org/#/common/demo_system) | ||||||
|  |  | ||||||
| | 功能介绍     | 使用技术                | 文档地址                                                                                              | 特性注意事项                     | | > 前端项目地址: [plus-ui](https://gitee.com/JavaLionLi/plus-ui) | ||||||
| |----------|---------------------|---------------------------------------------------------------------------------------------------|----------------------------| |  | ||||||
| | 当前框架     | RuoYi-Vue-Plus      | [RuoYi-Vue-Plus文档](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages)                       | 重写RuoYi-Vue全方位升级(不兼容原框架)   | | > 文档地址: [plus-doc](https://plus-doc.dromara.org) - [plus-doc(国内备用)](https://dromara.gitee.io/plus-doc) | ||||||
| | 微服务分支    | RuoYi-Cloud-Plus    | [微服务分支地址](https://gitee.com/JavaLionLi/RuoYi-Cloud-Plus)                                          | 重写RuoYi-Cloud全方位升级(不兼容原框架) | |  | ||||||
| | 单体分支     | RuoYi-Vue-Plus-fast | [fast分支地址](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/tree/fast/)                                | 单体应用结构                     | | # 本框架与RuoYi的功能差异 | ||||||
| | Vue3分支   | RuoYi-Vue-Plus-UI   | [UI地址](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus-UI)                                            | 由于组件还未完善 仅供学习              | |  | ||||||
| | 原框架      | RuoYi-Vue           | [RuoYi-Vue官网](http://ruoyi.vip/)                                                                  | 定期同步需要的功能                  | | | 功能          | 本框架                                                                                                               | RuoYi                                                                              | | ||||||
| | 前端开发框架   | Vue、Element UI      | [Element UI官网](https://element.eleme.cn/#/zh-CN)                                                  |                            | | |-------------|-------------------------------------------------------------------------------------------------------------------|------------------------------------------------------------------------------------| | ||||||
| | 后端开发框架   | SpringBoot          | [SpringBoot官网](https://spring.io/projects/spring-boot/#learn)                                     |                            | | | 前端项目        | 采用 Vue3 + TS + ElementPlus 重写                                                                                     | 基于Vue2/Vue3 + JS                                                                   |  | ||||||
| | 容器框架     | Undertow            | [Undertow官网](https://undertow.io/)                                                                | 基于 XNIO 的高性能容器             | | | 后端项目结构      | 采用插件化 + 扩展包形式 结构解耦 易于扩展                                                                                           | 模块相互注入耦合严重难以扩展                                                                     |  | ||||||
| | 权限认证框架   | Sa-Token、Jwt        | [Sa-Token官网](https://sa-token.dev33.cn/)                                                          | 强解耦、强扩展                    | | | 后端代码风格      | 严格遵守Alibaba规范与项目统一配置的代码格式化                                                                                        | 代码书写与常规结构不同阅读障碍大                                                                   | | ||||||
| | 关系数据库    | MySQL               | [MySQL官网](https://dev.mysql.com/)                                                                 | 适配 8.X 最低 5.7              | | | Web容器       | 采用 Undertow 基于 XNIO 的高性能容器                                                                                        | 采用 Tomcat                                                                          | | ||||||
| | 关系数据库    | Oracle              | [Oracle官网](https://www.oracle.com/cn/database/)                                                   | 适配 11g 12c                 | | | 权限认证        | 采用 Sa-Token、Jwt 静态使用功能齐全 低耦合 高扩展                                                                                  | Spring Security 配置繁琐扩展性极差                                                          | | ||||||
| | 关系数据库    | PostgreSQL          | [PostgreSQL官网](https://www.postgresql.org/)                                                       | 适配 13 14                   | | | 权限注解        | 采用 Sa-Token 支持注解 登录校验、角色校验、权限校验、二级认证校验、HttpBasic校验、忽略校验<br/>角色与权限校验支持多种条件 如 `AND` `OR` 或 `权限 OR 角色` 等复杂表达式        | 只支持是否存在匹配                                                                          | | ||||||
| | 关系数据库    | SQLServer           | [SQLServer官网](https://docs.microsoft.com/zh-cn/sql/sql-server)                                    | 适配 2017 2019               | | | 三方鉴权        | 采用 JustAuth 第三方登录组件 支持微信、钉钉等数十种三方认证                                                                               | 无                                                                                  | | ||||||
| | 缓存数据库    | Redis               | [Redis官网](https://redis.io/)                                                                      | 适配 6.X 最低 4.X              | | | 关系数据库支持     | 原生支持 MySQL、Oracle、PostgreSQL、SQLServer<br/>可同时使用异构切换(支持其他 mybatis-plus 支持的所有数据库 只需要增加jdbc依赖即可使用 达梦金仓等均有成功案例)      | 支持 Mysql、Oracle 不支持同时使用、不支持异构切换                                                    | | ||||||
| | 数据库框架    | Mybatis-Plus        | [Mybatis-Plus文档](https://baomidou.com/guide/)                                                     | 快速 CRUD 增加开发效率             | | | 缓存数据库       | 支持 Redis 5-7 支持大部分新功能特性 如 分布式限流、分布式队列                                                                             | Redis 简单 get set 支持                                                                | | ||||||
| | 数据库框架    | p6spy               | [p6spy官网](https://p6spy.readthedocs.io/)                                                          | 更强劲的 SQL 分析                | | | Redis客户端    | 采用 Redisson Redis官方推荐 基于Netty的客户端工具<br/>支持Redis 90%以上的命令 底层优化规避很多不正确的用法 例如: keys被转换为scan<br/>支持单机、哨兵、单主集群、多主集群等模式 | Lettuce + RedisTemplate 支持模式少 工具使用繁琐<br/>连接池采用 common-pool Bug多经常性出问题              | | ||||||
| | 多数据源框架   | dynamic-datasource  | [dynamic-ds文档](https://www.kancloud.cn/tracy5546/dynamic-datasource/content)                      | 支持主从与多种类数据库异构              | | | 缓存注解        | 采用 Spring-Cache 注解 对其扩展了实现支持了更多功能<br/>例如 过期时间 最大空闲时间 组最大长度等 只需一个注解即可完成数据自动缓存                                      | 需手动编写Redis代码逻辑                                                                     | | ||||||
| | 序列化框架    | Jackson             | [Jackson官网](https://github.com/FasterXML/jackson)                                                 | 统一使用 jackson 高效可靠          | | | ORM框架       | 采用 Mybatis-Plus 基于对象几乎不用写SQL全java操作 功能强大插件众多<br/>例如多租户插件 分页插件 乐观锁插件等等                                             | 采用 Mybatis 基于XML需要手写SQL                                                            | | ||||||
| | Redis客户端 | Redisson            | [Redisson文档](https://github.com/redisson/redisson/wiki/%E7%9B%AE%E5%BD%95)                        | 支持单机、集群配置                  | | | SQL监控       | 采用 p6spy 可输出完整SQL与执行时间监控                                                                                          | log输出 需手动拼接sql与参数无法快速查看调试问题                                                        | | ||||||
| | 分布式限流    | Redisson            | [Redisson文档](https://github.com/redisson/redisson/wiki/%E7%9B%AE%E5%BD%95)                        | 全局、请求IP、集群ID 多种限流          | | | 数据分页        | 采用 Mybatis-Plus 分页插件<br/>框架对其进行了扩展 对象化分页对象 支持多种方式传参 支持前端多排序 复杂排序                                                  | 采用 PageHelper 仅支持单查询分页 参数只能从param传 只能单排序 功能扩展性差 体验不好                               | | ||||||
| | 分布式队列    | Redisson            | [Redisson文档](https://github.com/redisson/redisson/wiki/%E7%9B%AE%E5%BD%95)                        | 普通队列、延迟队列、优先队列 等           | | | 数据权限        | 采用 Mybatis-Plus 插件 自行分析拼接SQL 无感式过滤<br/>只需为Mapper设置好注解条件 支持多种自定义 不限于部门角色                                           | 采用 注解+aop 实现 基于部门角色 生成的sql兼容性差 不支持其他业务扩展<br/>生成sql后需手动拼接到具体业务sql上 对于多个Mapper查询不起作用 | | ||||||
| | 分布式锁     | Lock4j              | [Lock4j官网](https://gitee.com/baomidou/lock4j)                                                     | 注解锁、工具锁 多种多样               | | | 数据脱敏        | 采用 注解 + jackson 序列化期间脱敏 支持不同模块不同的脱敏条件<br/>支持多种策略 如身份证、手机号、地址、邮箱、银行卡等 可自行扩展                                        | 无                                                                                  | | ||||||
| | 分布式幂等    | Redisson            | [Lock4j文档](https://gitee.com/baomidou/lock4j)                                                     | 拦截重复提交                     | | | 数据加解密       | 采用 注解 + mybatis 拦截器 对存取数据期间自动加解密<br/>支持多种策略 如BASE64、AES、RSA、SM2、SM4等                                              | 无                                                                                  | | ||||||
| | 分布式链路追踪  | Apache SkyWalking   | [Apache SkyWalking文档](https://skywalking.apache.org/docs/)                                        | 链路追踪、网格分析、度量聚合、可视化         | | | 接口传输加密      | 采用 动态 AES + RSA 加密请求 body 每一次请求秘钥都不同大幅度降低可破解性                                                                     | 无                                                                                  | | ||||||
| | 分布式任务调度  | Xxl-Job             | [Xxl-Job官网](https://www.xuxueli.com/xxl-job/)                                                     | 高性能 高可靠 易扩展                | | | 数据翻译        | 采用 注解 + jackson 序列化期间动态修改数据 数据进行翻译<br/>支持多种模式: `映射翻译` `直接翻译` `其他扩展条件翻译` 接口化两步即可完成自定义扩展 内置多种翻译实现                   | 无                                                                                  | | ||||||
| | 文件存储     | Minio               | [Minio文档](https://docs.min.io/)                                                                   | 本地存储                       | | | 多数据源框架      | 采用 dynamic-datasource 支持世面大部分数据库<br/>通过yml配置即可动态管理异构不同种类的数据库 也可通过前端页面添加数据源<br/>支持spel表达式从请求头参数等条件切换数据源            | 基于 druid 手动编写代码配置数据源 配置繁琐 支持性差                                                     | | ||||||
| | 文件存储     | 七牛、阿里、腾讯            | [OSS使用文档](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=4359146&doc_id=1469725) | 云存储                        | | | 多数据源事务      | 采用 dynamic-datasource 支持多数据源不同种类的数据库事务回滚                                                                          | 不支持                                                                                | | ||||||
| | 短信模块     | 阿里、腾讯               | [短信使用文档](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=5578491&doc_id=1469725)  | 短信发送                       | | | 数据库连接池      | 采用 HikariCP Spring官方内置连接池 配置简单 以性能与稳定性闻名天下                                                                        | 采用 druid bug众多 社区维护差 活跃度低 配置众多繁琐性能一般                                               | | ||||||
| | 监控框架     | SpringBoot-Admin    | [SpringBoot-Admin文档](https://codecentric.github.io/spring-boot-admin/current/)                    | 全方位服务监控                    | | | 数据库主键       | 采用 雪花ID 基于时间戳的 有序增长 唯一ID 再也不用为分库分表 数据合并主键冲突重复而发愁                                                                  | 采用 数据库自增ID 支持数据量有限 不支持多数据源主键唯一                                                     | | ||||||
| | 校验框架     | Validation          | [Validation文档](https://docs.jboss.org/hibernate/stable/validator/reference/en-US/html_single/)    | 增强接口安全性、严谨性 支持国际化          | | | WebSocket协议 | 基于 Spring 封装的 WebSocket 协议 扩展了Token鉴权与分布式会话同步 不再只是基于单机的废物                                                         | 无                                                                                  | | ||||||
| | Excel框架  | Alibaba EasyExcel   | [EasyExcel文档](https://www.yuque.com/easyexcel/doc/easyexcel)                                      | 性能优异 扩展性强                  | | | 序列化         | 采用 Jackson Spring官方内置序列化 靠谱!!!                                                                                    | 采用 fastjson bugjson 远近闻名                                                           |  | ||||||
| | 文档框架     | SpringDoc、javadoc   | [接口文档](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=5805266&doc_id=1469725)    | 无注解零入侵基于java注释             | | | 分布式幂等       | 参考美团GTIS防重系统简化实现(细节可看文档)                                                                                          | 手动编写注解基于aop实现                                                                      | | ||||||
| | 工具类框架    | Hutool、Lombok       | [Hutool文档](https://www.hutool.cn/docs/)                                                           | 减少代码冗余 增加安全性               | | | 分布式锁        | 采用 Lock4j 底层基于 Redisson                                                                                           | 无                                                                                  | | ||||||
| | 代码生成器    | 适配MP、SpringDoc规范化代码 | [代码生成文档](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=5522329&doc_id=1469725)  | 一键生成前后端代码                  | | | 分布式任务调度     | 采用 PowerJob 天生支持分布式 统一的管理中心                                                                                       | 采用 Quartz 基于数据库锁性能差 集群需要做很多配置与改造                                                   |  | ||||||
| | 部署方式     | Docker              | [Docker文档](https://docs.docker.com/)                                                              | 容器编排 一键部署业务集群              | | | 文件存储        | 采用 Minio 分布式文件存储 天生支持多机、多硬盘、多分片、多副本存储<br/>支持权限管理 安全可靠 文件可加密存储                                                     | 采用 本机文件存储 文件裸漏 易丢失泄漏 不支持集群有单点效应                                                    | | ||||||
| | 国际化      | SpringMessage       | [SpringMVC文档](https://docs.spring.io/spring-framework/docs/current/reference/html/web.html#mvc)   | Spring标准国际化方案              | | | 云存储         | 采用 AWS S3 协议客户端 支持 七牛、阿里、腾讯 等一切支持S3协议的厂家                                                                          | 不支持                                                                                | | ||||||
|  | | 短信          | 采用 sms4j 短信融合包 支持数十种短信厂家 只需在yml配置好厂家密钥即可使用 可多厂家共用                                                                 | 不支持                                                                                | | ||||||
|  | | 邮件          | 采用 mail-api 通用协议支持大部分邮件厂商                                                                                         | 不支持                                                                                | | ||||||
|  | | 接口文档        | 采用 SpringDoc、javadoc 无注解零入侵基于java注释<br/>只需把注释写好 无需再写一大堆的文档注解了                                                     | 采用 Springfox 已停止维护 需要编写大量的注解来支持文档生成                                                |  | ||||||
|  | | 校验框架        | 采用 Validation 支持注解与工具类校验 注解支持国际化                                                                                  | 仅支持注解 且注解不支持国际化                                                                    | | ||||||
|  | | Excel框架     | 采用 Alibaba EasyExcel 基于插件化<br/>框架对其增加了很多功能 例如 自动合并相同内容 自动排列布局 字典翻译等                                               | 基于 POI 手写实现 功能有限 复杂 扩展性差                                                           | | ||||||
|  | | 工具类框架       | 采用 Hutool、Lombok 上百种工具覆盖90%的使用需求 基于注解自动生成 get set 等简化框架大量代码                                                       | 手写工具稳定性差易出问题 工具数量有限 代码臃肿需自己手写 get set 等                                            |  | ||||||
|  | | 监控框架        | 采用 SpringBoot-Admin 基于SpringBoot官方 actuator 探针机制<br/>实时监控服务状态 框架还为其扩展了在线日志查看监控                                    | 无                                                                                  |  | ||||||
|  | | 链路追踪        | 采用 Apache SkyWalking 还在为请求不知道去哪了 到哪出了问题而烦恼吗<br/>用了它即可实时查看请求经过的每一处每一个节点                                            | 无                                                                                  | | ||||||
|  | | 代码生成器       | 只需设计好表结构 一键生成所有crud代码与页面<br/>降低80%的开发量 把精力都投入到业务设计上<br/>框架为其适配MP、SpringDoc规范化代码 同时支持动态多数据源代码生成                    | 代码生成原生结构 只支持单数据源生成                                                                 | | ||||||
|  | | 部署方式        | 支持 Docker 编排 一键搭建所有环境 让开发人员从此不再为搭建环境而烦恼                                                                           | 原生jar部署 其他环境需手动下载安装 自行搭建                                                           |  | ||||||
|  | | 项目路径修改      | 提供详细的修改方案文档 并为其做了一些改动 非常简单即可修改成自己想要的                                                                              | 需要做很多改造 文档说明有限                                                                     | | ||||||
|  | | 国际化         | 基于请求头动态返回不同语种的文本内容 开发难度低 有对应的工具类 支持大部分注解内容国际化                                                                     | 只提供基础功能 其他需自行编写扩展                                                                  | | ||||||
|  | | 代码单例测试      | 提供单例测试 使用方式编写方法与maven多环境单测插件                                                                                      | 只提供基础功能 其他需自行编写扩展                                                                  | | ||||||
|  | | Demo案例      | 提供框架功能的实际使用案例 单独一个模块提供了很多很全                                                                                       | 无                                                                                  | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## 本框架与RuoYi的业务差异 | ||||||
|  |  | ||||||
|  | | 业务     | 功能说明                                                                 | 本框架 | RuoYi            | | ||||||
|  | |--------|----------------------------------------------------------------------|-----|------------------| | ||||||
|  | | 租户管理   | 系统内租户的管理 如:租户套餐、过期时间、用户数量、企业信息等                                      | 支持  | 无                | | ||||||
|  | | 租户套餐管理 | 系统内租户所能使用的套餐管理 如:套餐内所包含的菜单等                                          | 支持  | 无                | | ||||||
|  | | 客户端管理  | 系统内对接的所有客户端管理 如: pc端、小程序端等<br>支持动态授权登录方式 如: 短信登录、密码登录等 支持动态控制token时效 | 支持  | 无                | | ||||||
|  | | 用户管理   | 用户的管理配置 如:新增用户、分配用户所属部门、角色、岗位等                                       | 支持  | 支持               | | ||||||
|  | | 部门管理   | 配置系统组织机构(公司、部门、小组) 树结构展现支持数据权限                                       | 支持  | 支持               | | ||||||
|  | | 岗位管理   | 配置系统用户所属担任职务                                                         | 支持  | 支持               | | ||||||
|  | | 菜单管理   | 配置系统菜单、操作权限、按钮权限标识等                                                  | 支持  | 支持               | | ||||||
|  | | 角色管理   | 角色菜单权限分配、设置角色按机构进行数据范围权限划分                                           | 支持  | 支持               | | ||||||
|  | | 字典管理   | 对系统中经常使用的一些较为固定的数据进行维护                                               | 支持  | 支持               | | ||||||
|  | | 参数管理   | 对系统动态配置常用参数                                                          | 支持  | 支持               | | ||||||
|  | | 通知公告   | 系统通知公告信息发布维护                                                         | 支持  | 支持               | | ||||||
|  | | 操作日志   | 系统正常操作日志记录和查询 系统异常信息日志记录和查询                                          | 支持  | 支持               | | ||||||
|  | | 登录日志   | 系统登录日志记录查询包含登录异常                                                     | 支持  | 支持               | | ||||||
|  | | 文件管理   | 系统文件展示、上传、下载、删除等管理                                                   | 支持  | 无                | | ||||||
|  | | 文件配置管理 | 系统文件上传、下载所需要的配置信息动态添加、修改、删除等管理                                       | 支持  | 无                | | ||||||
|  | | 在线用户管理 | 已登录系统的在线用户信息监控与强制踢出操作                                                | 支持  | 支持               | | ||||||
|  | | 定时任务   | 运行报表、任务管理(添加、修改、删除)、日志管理、执行器管理等                                      | 支持  | 仅支持任务与日志管理       | | ||||||
|  | | 代码生成   | 多数据源前后端代码的生成(java、html、xml、sql)支持CRUD下载                              | 支持  | 仅支持单数据源          | | ||||||
|  | | 系统接口   | 根据业务代码自动生成相关的api接口文档                                                 | 支持  | 支持               | | ||||||
|  | | 服务监控   | 监视集群系统CPU、内存、磁盘、堆栈、在线日志、Spring相关配置等                                  | 支持  | 仅支持单机CPU、内存、磁盘监控 | | ||||||
|  | | 缓存监控   | 对系统的缓存信息查询,命令统计等。                                                    | 支持  | 支持               | | ||||||
|  | | 在线构建器  | 拖动表单元素生成相应的HTML代码。                                                   | 支持  | 支持               | | ||||||
|  | | 使用案例   | 系统的一些功能案例                                                            | 支持  | 不支持              | | ||||||
|  |  | ||||||
| ## 参考文档 | ## 参考文档 | ||||||
|  |  | ||||||
| 使用框架前请仔细阅读文档重点注意事项 | 使用框架前请仔细阅读文档重点注意事项 | ||||||
| <br> | <br> | ||||||
| >[初始化项目 必看](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=4164117&doc_id=1469725) | >[初始化项目 必看](https://plus-doc.dromara.org/#/ruoyi-vue-plus/quickstart/init) | ||||||
| >>[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://plus-doc.dromara.org/#/ruoyi-vue-plus/quickstart/init](https://plus-doc.dromara.org/#/ruoyi-vue-plus/quickstart/init) | ||||||
| > | > | ||||||
| >[专栏与视频 入门必看](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=5473272&doc_id=1469725) | >[专栏与视频 入门必看](https://plus-doc.dromara.org/#/common/column) | ||||||
| >>[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://plus-doc.dromara.org/#/common/column](https://plus-doc.dromara.org/#/common/column) | ||||||
| > | > | ||||||
| >[部署项目 必看](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=4219382&doc_id=1469725) | >[部署项目 必看](https://plus-doc.dromara.org/#/ruoyi-vue-plus/quickstart/deploy) | ||||||
| >>[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://plus-doc.dromara.org/#/ruoyi-vue-plus/quickstart/deploy](https://plus-doc.dromara.org/#/ruoyi-vue-plus/quickstart/deploy) | ||||||
| >  | >  | ||||||
| >[参考文档 Wiki](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages) | >[参考文档 Wiki](https://plus-doc.dromara.org) | ||||||
| >>[https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages) | >>[https://plus-doc.dromara.org](https://plus-doc.dromara.org) | ||||||
|  | >  | ||||||
|  | >[参考文档(国内备份)](https://dromara.gitee.io/plus-doc) | ||||||
|  | >>[https://dromara.gitee.io/plus-doc](https://dromara.gitee.io/plus-doc) | ||||||
|  |  | ||||||
|  |  | ||||||
| ## 软件架构图 | ## 软件架构图 | ||||||
|  |  | ||||||
|  |  | ||||||
| ## 贡献代码 |  | ||||||
|  |  | ||||||
| 欢迎各路英雄豪杰 `PR` 代码 请提交到 `dev` 开发分支 统一测试发版 | ## 如何参与贡献 | ||||||
|  |  | ||||||
| 框架定位为 `通用后台管理系统(分布式集群强化)` 原则上不接受业务 `PR` | [参与贡献的方式 https://plus-doc.dromara.org/#/common/contribution](https://plus-doc.dromara.org/#/common/contribution) | ||||||
|  |  | ||||||
| ### 其他 | ### 其他 | ||||||
|  |  | ||||||
| * 同步升级 RuoYi-Vue | * 定期同步升级 RuoYi-Vue 有用的更新 | ||||||
| * GitHub 地址 [RuoYi-Vue-Plus-github](https://github.com/JavaLionLi/RuoYi-Vue-Plus) | * GitHub 地址 [RuoYi-Vue-Plus](https://github.com/dromara/RuoYi-Vue-Plus) | ||||||
| * 单模块 分支 [RuoYi-Vue-Plus-fast](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/tree/fast/) |  | ||||||
| * 微服务 分支 [RuoYi-Cloud-Plus](https://gitee.com/JavaLionLi/RuoYi-Cloud-Plus) | * 微服务 分支 [RuoYi-Cloud-Plus](https://gitee.com/JavaLionLi/RuoYi-Cloud-Plus) | ||||||
| * 用户扩展项目 [扩展项目列表](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/pages?sort_id=4478302&doc_id=1469725) | * 前端项目 地址 [plus-ui](https://gitee.com/JavaLionLi/plus-ui) | ||||||
|  | * 用户扩展项目 [扩展项目列表](https://plus-doc.dromara.org/#/ruoyi-vue-plus/extend-project/list) | ||||||
|  |  | ||||||
| ## 加群与捐献 | ## 加群与捐献 | ||||||
| >[加群与捐献](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/加群与捐献?sort_id=4104598) | >[加群与捐献](https://plus-doc.dromara.org/#/ruoyi-vue-plus/other/group_chat) | ||||||
| >>[https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/加群与捐献?sort_id=4104598](https://gitee.com/JavaLionLi/RuoYi-Vue-Plus/wikis/加群与捐献?sort_id=4104598) | >>[https://plus-doc.dromara.org/#/ruoyi-vue-plus/other/group_chat](https://plus-doc.dromara.org/#/ruoyi-vue-plus/other/group_chat) | ||||||
|  |  | ||||||
| ## 捐献作者 | ## 捐献作者 | ||||||
| 作者为兼职做开源,平时还需要工作,如果帮到了您可以请作者吃个盒饭   | 作者为兼职做开源,平时还需要工作,如果帮到了您可以请作者吃个盒饭   | ||||||
| <img src="https://images.gitee.com/uploads/images/2022/0218/213734_b1b8197f_1766278.jpeg" width="300px" height="450px" /> | <img src="https://foruda.gitee.com/images/1678975784848381069/d8661ed9_1766278.png" width="300px" height="450px" /> | ||||||
| <img src="https://images.gitee.com/uploads/images/2021/0525/101713_3d18b119_1766278.jpeg" width="300px" height="450px" /> | <img src="https://foruda.gitee.com/images/1678975801230205215/6f96229d_1766278.png" width="300px" height="450px" /> | ||||||
|  |  | ||||||
| ## 业务功能 |  | ||||||
|  |  | ||||||
| | 功能 | 介绍 | |  | ||||||
| |---|---| |  | ||||||
| | 用户管理 | 用户是系统操作者,该功能主要完成系统用户配置。 | |  | ||||||
| | 部门管理 | 配置系统组织机构(公司、部门、小组),树结构展现支持数据权限。 | |  | ||||||
| | 岗位管理 | 配置系统用户所属担任职务。 | |  | ||||||
| | 菜单管理 | 配置系统菜单,操作权限,按钮权限标识等。 | |  | ||||||
| | 角色管理 | 角色菜单权限分配、设置角色按机构进行数据范围权限划分。 | |  | ||||||
| | 字典管理 | 对系统中经常使用的一些较为固定的数据进行维护。 | |  | ||||||
| | 参数管理 | 对系统动态配置常用参数。 | |  | ||||||
| | 通知公告 | 系统通知公告信息发布维护。 | |  | ||||||
| | 操作日志 | 系统正常操作日志记录和查询;系统异常信息日志记录和查询。 | |  | ||||||
| | 登录日志 | 系统登录日志记录查询包含登录异常。 | |  | ||||||
| | 文件管理 | 系统文件上传、下载等管理。 | |  | ||||||
| | 定时任务 | 在线(添加、修改、删除)任务调度包含执行结果日志。 | |  | ||||||
| | 代码生成 | 前后端代码的生成(java、html、xml、sql)支持CRUD下载 。 | |  | ||||||
| | 系统接口 | 根据业务代码自动生成相关的api接口文档。 | |  | ||||||
| | 服务监控 | 监视集群系统CPU、内存、磁盘、堆栈、在线日志、Spring相关配置等。 | |  | ||||||
| | 缓存监控 | 对系统的缓存信息查询,命令统计等。 | |  | ||||||
| | 在线构建器 | 拖动表单元素生成相应的HTML代码。 | |  | ||||||
| | 连接池监视 | 监视当前系统数据库连接池状态,可进行分析SQL找出系统性能瓶颈。 | |  | ||||||
| | 使用案例 | 系统的一些功能案例 | |  | ||||||
|  |  | ||||||
| ## 演示图例 | ## 演示图例 | ||||||
|  |  | ||||||
| <table border="1" cellpadding="1" cellspacing="1" style="width:500px"> | |                                                                                            |                                                                                            | | ||||||
|     <tbody> | |--------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------| | ||||||
|         <tr> | |  |  | | ||||||
|             <td><img src="https://oscimg.oschina.net/oscnet/up-972235bcbe3518dedd351ff0e2ee7d1031c.png" width="1920" /></td> | |  |  | | ||||||
|             <td><img src="https://oscimg.oschina.net/oscnet/up-5e0097702fa91e2e36391de8127676a7fa1.png" width="1920" /></td> | |  |  | | ||||||
|         </tr> | |  |  | | ||||||
|         <tr> | |  |  | | ||||||
|             <td> | |  |  | | ||||||
|             <p><img src="https://oscimg.oschina.net/oscnet/up-e56e3828f48cd9886d88731766f06d5f3c1.png" width="1920" /></p> | |  |  | | ||||||
|             </td> | |  |  | | ||||||
|             <td><img src="https://oscimg.oschina.net/oscnet/up-0715990ea1a9f254ec2138fcd063c1f556a.png" width="1920" /></td> | |  |  | | ||||||
|         </tr> | |  |  | | ||||||
|         <tr> | |  |  | | ||||||
|             <td><img src="https://oscimg.oschina.net/oscnet/up-eaf5417ccf921bb64abb959e3d8e290467f.png" width="1920" /></td> | |  |  | | ||||||
|             <td><img src="https://oscimg.oschina.net/oscnet/up-fc285cf33095ebf8318de6999af0f473861.png" width="1920" /></td> | |  |  | | ||||||
|         </tr> | |  |  | | ||||||
|         <tr> | |  |  | | ||||||
|             <td><img src="https://oscimg.oschina.net/oscnet/up-60c83fd8bd61c29df6dbf47c88355e9c272.png" width="1920" /></td> | |  |  | | ||||||
|             <td><img src="https://oscimg.oschina.net/oscnet/up-7f731948c8b73c7d90f67f9e1c7a534d5c3.png" width="1920" /></td> | |  |  | | ||||||
|         </tr> | |  |  | | ||||||
|         <tr> | |  |  | | ||||||
|             <td><img src="https://oscimg.oschina.net/oscnet/up-e4de89b5e2d20c52d3c3a47f9eb88eb8526.png" width="1920" /></td> |  | ||||||
|             <td><img src="https://oscimg.oschina.net/oscnet/up-8791d823a508eb90e67c604f36f57491a67.png" width="1920" /></td> |  | ||||||
|         </tr> |  | ||||||
|         <tr> |  | ||||||
|             <td><img src="https://oscimg.oschina.net/oscnet/up-4589afd99982ead331785299b894174feb6.png" width="1920" /></td> |  | ||||||
|             <td><img src="https://oscimg.oschina.net/oscnet/up-8ea177cdacaea20995daf2f596b15232561.png" width="1920" /></td> |  | ||||||
|         </tr> |  | ||||||
|         <tr> |  | ||||||
|             <td><img src="https://oscimg.oschina.net/oscnet/up-32d1d04c55c11f74c9129fbbc58399728c4.png" width="1920" /></td> |  | ||||||
|             <td><img src="https://oscimg.oschina.net/oscnet/up-04fa118f7631b7ae6fd72299ca0a1430a63.png" width="1920" /></td> |  | ||||||
|         </tr> |  | ||||||
|         <tr> |  | ||||||
|             <td><img src="https://oscimg.oschina.net/oscnet/up-fe7e85b65827802bfaadf3acd42568b58c7.png" width="1920" /></td> |  | ||||||
|             <td><img src="https://oscimg.oschina.net/oscnet/up-eff2b02a54f8188022d8498cfe6af6fcc06.png" width="1920" /></td> |  | ||||||
|         </tr> |  | ||||||
|     </tbody> |  | ||||||
| </table> |  | ||||||
|   | |||||||
							
								
								
									
										275
									
								
								pom.xml
									
									
									
									
									
								
							
							
						
						
									
										275
									
								
								pom.xml
									
									
									
									
									
								
							| @@ -4,47 +4,58 @@ | |||||||
|          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> |          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||||||
|     <modelVersion>4.0.0</modelVersion> |     <modelVersion>4.0.0</modelVersion> | ||||||
|  |  | ||||||
|     <groupId>com.ruoyi</groupId> |     <groupId>org.dromara</groupId> | ||||||
|     <artifactId>ruoyi-vue-plus</artifactId> |     <artifactId>ruoyi-vue-plus</artifactId> | ||||||
|     <version>4.5.0</version> |     <version>${revision}</version> | ||||||
|  |  | ||||||
|     <name>RuoYi-Vue-Plus</name> |     <name>RuoYi-Vue-Plus</name> | ||||||
|     <url>https://gitee.com/JavaLionLi/RuoYi-Vue-Plus</url> |     <url>https://gitee.com/dromara/RuoYi-Vue-Plus</url> | ||||||
|     <description>RuoYi-Vue-Plus后台管理系统</description> |     <description>RuoYi-Vue-Plus多租户管理系统</description> | ||||||
|  |  | ||||||
|     <properties> |     <properties> | ||||||
|         <ruoyi-vue-plus.version>4.5.0</ruoyi-vue-plus.version> |         <revision>5.1.2</revision> | ||||||
|         <spring-boot.version>2.7.7</spring-boot.version> |         <spring-boot.version>3.1.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>17</java.version> | ||||||
|         <maven-jar-plugin.version>3.2.2</maven-jar-plugin.version> |         <spring-boot.mybatis>3.0.3</spring-boot.mybatis> | ||||||
|         <spring-boot.mybatis>2.2.2</spring-boot.mybatis> |         <springdoc.version>2.2.0</springdoc.version> | ||||||
|         <springdoc.version>1.6.14</springdoc.version> |         <therapi-javadoc.version>0.15.0</therapi-javadoc.version> | ||||||
|         <poi.version>5.2.3</poi.version> |         <poi.version>5.2.3</poi.version> | ||||||
|         <easyexcel.version>3.1.5</easyexcel.version> |         <easyexcel.version>3.3.3</easyexcel.version> | ||||||
|         <velocity.version>2.3</velocity.version> |         <velocity.version>2.3</velocity.version> | ||||||
|         <satoken.version>1.34.0</satoken.version> |         <satoken.version>1.37.0</satoken.version> | ||||||
|         <mybatis-plus.version>3.5.3.1</mybatis-plus.version> |         <mybatis-plus.version>3.5.4</mybatis-plus.version> | ||||||
|         <p6spy.version>3.9.1</p6spy.version> |         <p6spy.version>3.9.1</p6spy.version> | ||||||
|         <hutool.version>5.8.11</hutool.version> |         <hutool.version>5.8.22</hutool.version> | ||||||
|         <okhttp.version>4.10.0</okhttp.version> |         <okhttp.version>4.10.0</okhttp.version> | ||||||
|         <spring-boot-admin.version>2.7.10</spring-boot-admin.version> |         <spring-boot-admin.version>3.1.8</spring-boot-admin.version> | ||||||
|         <redisson.version>3.19.1</redisson.version> |         <redisson.version>3.24.3</redisson.version> | ||||||
|         <lock4j.version>2.2.3</lock4j.version> |         <lock4j.version>2.2.5</lock4j.version> | ||||||
|         <dynamic-ds.version>3.5.2</dynamic-ds.version> |         <dynamic-ds.version>4.2.0</dynamic-ds.version> | ||||||
|         <alibaba-ttl.version>2.14.2</alibaba-ttl.version> |         <alibaba-ttl.version>2.14.4</alibaba-ttl.version> | ||||||
|         <xxl-job.version>2.3.1</xxl-job.version> |         <powerjob.version>4.3.6</powerjob.version> | ||||||
|         <lombok.version>1.18.24</lombok.version> |         <mapstruct-plus.version>1.3.5</mapstruct-plus.version> | ||||||
|  |         <mapstruct-plus.lombok.version>0.2.0</mapstruct-plus.lombok.version> | ||||||
|         <!-- 临时修复 snakeyaml 漏洞 --> |         <lombok.version>1.18.30</lombok.version> | ||||||
|         <snakeyaml.version>1.33</snakeyaml.version> |         <bouncycastle.version>1.76</bouncycastle.version> | ||||||
|  |         <justauth.version>1.16.6</justauth.version> | ||||||
|  |         <!-- 离线IP地址定位库 --> | ||||||
|  |         <ip2region.version>2.7.0</ip2region.version> | ||||||
|  |  | ||||||
|         <!-- OSS 配置 --> |         <!-- OSS 配置 --> | ||||||
|         <aws-java-sdk-s3.version>1.12.373</aws-java-sdk-s3.version> |         <aws-java-sdk-s3.version>1.12.600</aws-java-sdk-s3.version> | ||||||
|         <!-- SMS 配置 --> |         <!-- SMS 配置 --> | ||||||
|         <aliyun.sms.version>2.0.23</aliyun.sms.version> |         <sms4j.version>2.2.0</sms4j.version> | ||||||
|         <tencent.sms.version>3.1.660</tencent.sms.version> |         <!-- 限制框架中的fastjson版本 --> | ||||||
|  |         <fastjson.version>1.2.83</fastjson.version> | ||||||
|  |  | ||||||
|  |         <!-- 插件版本 --> | ||||||
|  |         <maven-jar-plugin.version>3.2.2</maven-jar-plugin.version> | ||||||
|  |         <maven-war-plugin.version>3.2.2</maven-war-plugin.version> | ||||||
|  |         <maven-compiler-plugin.verison>3.11.0</maven-compiler-plugin.verison> | ||||||
|  |         <maven-surefire-plugin.version>3.1.2</maven-surefire-plugin.version> | ||||||
|  |         <flatten-maven-plugin.version>1.3.0</flatten-maven-plugin.version> | ||||||
|     </properties> |     </properties> | ||||||
|  |  | ||||||
|     <profiles> |     <profiles> | ||||||
| @@ -53,7 +64,7 @@ | |||||||
|             <properties> |             <properties> | ||||||
|                 <!-- 环境标识,需要与配置文件的名称相对应 --> |                 <!-- 环境标识,需要与配置文件的名称相对应 --> | ||||||
|                 <profiles.active>local</profiles.active> |                 <profiles.active>local</profiles.active> | ||||||
|                 <logging.level>debug</logging.level> |                 <logging.level>info</logging.level> | ||||||
|             </properties> |             </properties> | ||||||
|         </profile> |         </profile> | ||||||
|         <profile> |         <profile> | ||||||
| @@ -61,7 +72,7 @@ | |||||||
|             <properties> |             <properties> | ||||||
|                 <!-- 环境标识,需要与配置文件的名称相对应 --> |                 <!-- 环境标识,需要与配置文件的名称相对应 --> | ||||||
|                 <profiles.active>dev</profiles.active> |                 <profiles.active>dev</profiles.active> | ||||||
|                 <logging.level>debug</logging.level> |                 <logging.level>info</logging.level> | ||||||
|             </properties> |             </properties> | ||||||
|             <activation> |             <activation> | ||||||
|                 <!-- 默认环境 --> |                 <!-- 默认环境 --> | ||||||
| @@ -99,18 +110,34 @@ | |||||||
|                 <scope>import</scope> |                 <scope>import</scope> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- JustAuth 的依赖配置--> | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>org.springdoc</groupId> |                 <groupId>me.zhyd.oauth</groupId> | ||||||
|                 <artifactId>springdoc-openapi-webmvc-core</artifactId> |                 <artifactId>JustAuth</artifactId> | ||||||
|                 <version>${springdoc.version}</version> |                 <version>${justauth.version}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- common 的依赖配置--> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-bom</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |                 <type>pom</type> | ||||||
|  |                 <scope>import</scope> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>org.springdoc</groupId> |                 <groupId>org.springdoc</groupId> | ||||||
|                 <artifactId>springdoc-openapi-javadoc</artifactId> |                 <artifactId>springdoc-openapi-starter-webmvc-api</artifactId> | ||||||
|                 <version>${springdoc.version}</version> |                 <version>${springdoc.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>com.github.therapi</groupId> | ||||||
|  |                 <artifactId>therapi-runtime-javadoc</artifactId> | ||||||
|  |                 <version>${therapi-javadoc.version}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>org.projectlombok</groupId> |                 <groupId>org.projectlombok</groupId> | ||||||
|                 <artifactId>lombok</artifactId> |                 <artifactId>lombok</artifactId> | ||||||
| @@ -149,7 +176,7 @@ | |||||||
|             <!-- Sa-Token 权限认证, 在线文档:http://sa-token.dev33.cn/ --> |             <!-- Sa-Token 权限认证, 在线文档:http://sa-token.dev33.cn/ --> | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>cn.dev33</groupId> |                 <groupId>cn.dev33</groupId> | ||||||
|                 <artifactId>sa-token-spring-boot-starter</artifactId> |                 <artifactId>sa-token-spring-boot3-starter</artifactId> | ||||||
|                 <version>${satoken.version}</version> |                 <version>${satoken.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|             <!-- Sa-Token 整合 jwt --> |             <!-- Sa-Token 整合 jwt --> | ||||||
| @@ -164,20 +191,37 @@ | |||||||
|                     </exclusion> |                     </exclusion> | ||||||
|                 </exclusions> |                 </exclusions> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>cn.dev33</groupId> | ||||||
|  |                 <artifactId>sa-token-core</artifactId> | ||||||
|  |                 <version>${satoken.version}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|             <!-- dynamic-datasource 多数据源--> |             <!-- dynamic-datasource 多数据源--> | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.baomidou</groupId> |                 <groupId>com.baomidou</groupId> | ||||||
|                 <artifactId>dynamic-datasource-spring-boot-starter</artifactId> |                 <artifactId>dynamic-datasource-spring-boot3-starter</artifactId> | ||||||
|                 <version>${dynamic-ds.version}</version> |                 <version>${dynamic-ds.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.mybatis.spring.boot</groupId> | ||||||
|  |                 <artifactId>mybatis-spring-boot-starter</artifactId> | ||||||
|  |                 <version>${spring-boot.mybatis}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.baomidou</groupId> |                 <groupId>com.baomidou</groupId> | ||||||
|                 <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-annotation</artifactId> | ||||||
|  |                 <version>${mybatis-plus.version}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|             <!-- sql性能分析插件 --> |             <!-- sql性能分析插件 --> | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>p6spy</groupId> |                 <groupId>p6spy</groupId> | ||||||
| @@ -196,17 +240,11 @@ | |||||||
|                 <artifactId>aws-java-sdk-s3</artifactId> |                 <artifactId>aws-java-sdk-s3</artifactId> | ||||||
|                 <version>${aws-java-sdk-s3.version}</version> |                 <version>${aws-java-sdk-s3.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |             <!--短信sms4j--> | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.aliyun</groupId> |                 <groupId>org.dromara.sms4j</groupId> | ||||||
|                 <artifactId>dysmsapi20170525</artifactId> |                 <artifactId>sms4j-spring-boot-starter</artifactId> | ||||||
|                 <version>${aliyun.sms.version}</version> |                 <version>${sms4j.version}</version> | ||||||
|             </dependency> |  | ||||||
|  |  | ||||||
|             <dependency> |  | ||||||
|                 <groupId>com.tencentcloudapi</groupId> |  | ||||||
|                 <artifactId>tencentcloud-sdk-java-sms</artifactId> |  | ||||||
|                 <version>${tencent.sms.version}</version> |  | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <dependency> |             <dependency> | ||||||
| @@ -225,17 +263,6 @@ | |||||||
|                 <groupId>org.redisson</groupId> |                 <groupId>org.redisson</groupId> | ||||||
|                 <artifactId>redisson-spring-boot-starter</artifactId> |                 <artifactId>redisson-spring-boot-starter</artifactId> | ||||||
|                 <version>${redisson.version}</version> |                 <version>${redisson.version}</version> | ||||||
|                 <exclusions> |  | ||||||
|                     <exclusion> |  | ||||||
|                         <groupId>org.redisson</groupId> |  | ||||||
|                         <artifactId>redisson-spring-data-30</artifactId> |  | ||||||
|                     </exclusion> |  | ||||||
|                 </exclusions> |  | ||||||
|             </dependency> |  | ||||||
|             <dependency> |  | ||||||
|                 <groupId>org.redisson</groupId> |  | ||||||
|                 <artifactId>redisson-spring-data-27</artifactId> |  | ||||||
|                 <version>${redisson.version}</version> |  | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <dependency> |             <dependency> | ||||||
| @@ -244,11 +271,16 @@ | |||||||
|                 <version>${lock4j.version}</version> |                 <version>${lock4j.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <!-- xxl-job-core --> |             <!-- PowerJob --> | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.xuxueli</groupId> |                 <groupId>tech.powerjob</groupId> | ||||||
|                 <artifactId>xxl-job-core</artifactId> |                 <artifactId>powerjob-worker-spring-boot-starter</artifactId> | ||||||
|                 <version>${xxl-job.version}</version> |                 <version>${powerjob.version}</version> | ||||||
|  |             </dependency> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>tech.powerjob</groupId> | ||||||
|  |                 <artifactId>powerjob-official-processors</artifactId> | ||||||
|  |                 <version>${powerjob.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <dependency> |             <dependency> | ||||||
| @@ -257,67 +289,54 @@ | |||||||
|                 <version>${alibaba-ttl.version}</version> |                 <version>${alibaba-ttl.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <!-- 临时修复 snakeyaml 漏洞 --> |             <!-- 加密包引入 --> | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>org.yaml</groupId> |                 <groupId>org.bouncycastle</groupId> | ||||||
|                 <artifactId>snakeyaml</artifactId> |                 <artifactId>bcprov-jdk15to18</artifactId> | ||||||
|                 <version>${snakeyaml.version}</version> |                 <version>${bouncycastle.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <!-- 定时任务 --> |  | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.ruoyi</groupId> |                 <groupId>io.github.linpeilie</groupId> | ||||||
|                 <artifactId>ruoyi-job</artifactId> |                 <artifactId>mapstruct-plus-spring-boot-starter</artifactId> | ||||||
|                 <version>${ruoyi-vue-plus.version}</version> |                 <version>${mapstruct-plus.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <!-- 代码生成--> |             <!-- 离线IP地址定位库 ip2region --> | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.ruoyi</groupId> |                 <groupId>org.lionsoul</groupId> | ||||||
|                 <artifactId>ruoyi-generator</artifactId> |                 <artifactId>ip2region</artifactId> | ||||||
|                 <version>${ruoyi-vue-plus.version}</version> |                 <version>${ip2region.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <!-- 核心模块--> |  | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.ruoyi</groupId> |                 <groupId>com.alibaba</groupId> | ||||||
|                 <artifactId>ruoyi-framework</artifactId> |                 <artifactId>fastjson</artifactId> | ||||||
|                 <version>${ruoyi-vue-plus.version}</version> |                 <version>${fastjson.version}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <!-- 系统模块--> |  | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.ruoyi</groupId> |                 <groupId>org.dromara</groupId> | ||||||
|                 <artifactId>ruoyi-system</artifactId> |                 <artifactId>ruoyi-system</artifactId> | ||||||
|                 <version>${ruoyi-vue-plus.version}</version> |                 <version>${revision}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <!-- 通用工具--> |  | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.ruoyi</groupId> |                 <groupId>org.dromara</groupId> | ||||||
|                 <artifactId>ruoyi-common</artifactId> |                 <artifactId>ruoyi-job</artifactId> | ||||||
|                 <version>${ruoyi-vue-plus.version}</version> |                 <version>${revision}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <!-- OSS对象存储模块 --> |  | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.ruoyi</groupId> |                 <groupId>org.dromara</groupId> | ||||||
|                 <artifactId>ruoyi-oss</artifactId> |                 <artifactId>ruoyi-generator</artifactId> | ||||||
|                 <version>${ruoyi-vue-plus.version}</version> |                 <version>${revision}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|             <!-- SMS短信模块 --> |  | ||||||
|             <dependency> |             <dependency> | ||||||
|                 <groupId>com.ruoyi</groupId> |                 <groupId>org.dromara</groupId> | ||||||
|                 <artifactId>ruoyi-sms</artifactId> |  | ||||||
|                 <version>${ruoyi-vue-plus.version}</version> |  | ||||||
|             </dependency> |  | ||||||
|  |  | ||||||
|             <!-- demo模块 --> |  | ||||||
|             <dependency> |  | ||||||
|                 <groupId>com.ruoyi</groupId> |  | ||||||
|                 <artifactId>ruoyi-demo</artifactId> |                 <artifactId>ruoyi-demo</artifactId> | ||||||
|                 <version>${ruoyi-vue-plus.version}</version> |                 <version>${revision}</version> | ||||||
|             </dependency> |             </dependency> | ||||||
|  |  | ||||||
|         </dependencies> |         </dependencies> | ||||||
| @@ -325,15 +344,9 @@ | |||||||
|  |  | ||||||
|     <modules> |     <modules> | ||||||
|         <module>ruoyi-admin</module> |         <module>ruoyi-admin</module> | ||||||
|         <module>ruoyi-framework</module> |  | ||||||
|         <module>ruoyi-system</module> |  | ||||||
|         <module>ruoyi-job</module> |  | ||||||
|         <module>ruoyi-generator</module> |  | ||||||
|         <module>ruoyi-common</module> |         <module>ruoyi-common</module> | ||||||
|         <module>ruoyi-demo</module> |  | ||||||
|         <module>ruoyi-extend</module> |         <module>ruoyi-extend</module> | ||||||
|         <module>ruoyi-oss</module> |         <module>ruoyi-modules</module> | ||||||
|         <module>ruoyi-sms</module> |  | ||||||
|     </modules> |     </modules> | ||||||
|     <packaging>pom</packaging> |     <packaging>pom</packaging> | ||||||
|  |  | ||||||
| @@ -342,7 +355,7 @@ | |||||||
|             <plugin> |             <plugin> | ||||||
|                 <groupId>org.apache.maven.plugins</groupId> |                 <groupId>org.apache.maven.plugins</groupId> | ||||||
|                 <artifactId>maven-compiler-plugin</artifactId> |                 <artifactId>maven-compiler-plugin</artifactId> | ||||||
|                 <version>3.9.0</version> |                 <version>${maven-compiler-plugin.verison}</version> | ||||||
|                 <configuration> |                 <configuration> | ||||||
|                     <source>${java.version}</source> |                     <source>${java.version}</source> | ||||||
|                     <target>${java.version}</target> |                     <target>${java.version}</target> | ||||||
| @@ -351,7 +364,7 @@ | |||||||
|                         <path> |                         <path> | ||||||
|                             <groupId>com.github.therapi</groupId> |                             <groupId>com.github.therapi</groupId> | ||||||
|                             <artifactId>therapi-runtime-javadoc-scribe</artifactId> |                             <artifactId>therapi-runtime-javadoc-scribe</artifactId> | ||||||
|                             <version>0.15.0</version> |                             <version>${therapi-javadoc.version}</version> | ||||||
|                         </path> |                         </path> | ||||||
|                         <path> |                         <path> | ||||||
|                             <groupId>org.projectlombok</groupId> |                             <groupId>org.projectlombok</groupId> | ||||||
| @@ -363,22 +376,60 @@ | |||||||
|                             <artifactId>spring-boot-configuration-processor</artifactId> |                             <artifactId>spring-boot-configuration-processor</artifactId> | ||||||
|                             <version>${spring-boot.version}</version> |                             <version>${spring-boot.version}</version> | ||||||
|                         </path> |                         </path> | ||||||
|  |                         <path> | ||||||
|  |                             <groupId>io.github.linpeilie</groupId> | ||||||
|  |                             <artifactId>mapstruct-plus-processor</artifactId> | ||||||
|  |                             <version>${mapstruct-plus.version}</version> | ||||||
|  |                         </path> | ||||||
|  |                         <path> | ||||||
|  |                             <groupId>org.projectlombok</groupId> | ||||||
|  |                             <artifactId>lombok-mapstruct-binding</artifactId> | ||||||
|  |                             <version>${mapstruct-plus.lombok.version}</version> | ||||||
|  |                         </path> | ||||||
|                     </annotationProcessorPaths> |                     </annotationProcessorPaths> | ||||||
|  |                     <compilerArgs> | ||||||
|  |                         <arg>-parameters</arg> | ||||||
|  |                     </compilerArgs> | ||||||
|                 </configuration> |                 </configuration> | ||||||
|             </plugin> |             </plugin> | ||||||
|             <!-- 单元测试使用 --> |             <!-- 单元测试使用 --> | ||||||
|             <plugin> |             <plugin> | ||||||
|                 <groupId>org.apache.maven.plugins</groupId> |                 <groupId>org.apache.maven.plugins</groupId> | ||||||
|                 <artifactId>maven-surefire-plugin</artifactId> |                 <artifactId>maven-surefire-plugin</artifactId> | ||||||
|                 <version>2.22.2</version> |                 <version>${maven-surefire-plugin.version}</version> | ||||||
|                 <configuration> |                 <configuration> | ||||||
|                     <argLine>-Dfile.encoding=UTF-8</argLine> |  | ||||||
|                     <!-- 根据打包环境执行对应的@Tag测试方法 --> |                     <!-- 根据打包环境执行对应的@Tag测试方法 --> | ||||||
|                     <groups>${profiles.active}</groups> |                     <groups>${profiles.active}</groups> | ||||||
|                     <!-- 排除标签 --> |                     <!-- 排除标签 --> | ||||||
|                     <excludedGroups>exclude</excludedGroups> |                     <excludedGroups>exclude</excludedGroups> | ||||||
|                 </configuration> |                 </configuration> | ||||||
|             </plugin> |             </plugin> | ||||||
|  |             <!-- 统一版本号管理 --> | ||||||
|  |             <plugin> | ||||||
|  |                 <groupId>org.codehaus.mojo</groupId> | ||||||
|  |                 <artifactId>flatten-maven-plugin</artifactId> | ||||||
|  |                 <version>${flatten-maven-plugin.version}</version> | ||||||
|  |                 <configuration> | ||||||
|  |                     <updatePomFile>true</updatePomFile> | ||||||
|  |                     <flattenMode>resolveCiFriendliesOnly</flattenMode> | ||||||
|  |                 </configuration> | ||||||
|  |                 <executions> | ||||||
|  |                     <execution> | ||||||
|  |                         <id>flatten</id> | ||||||
|  |                         <phase>process-resources</phase> | ||||||
|  |                         <goals> | ||||||
|  |                             <goal>flatten</goal> | ||||||
|  |                         </goals> | ||||||
|  |                     </execution> | ||||||
|  |                     <execution> | ||||||
|  |                         <id>flatten.clean</id> | ||||||
|  |                         <phase>clean</phase> | ||||||
|  |                         <goals> | ||||||
|  |                             <goal>clean</goal> | ||||||
|  |                         </goals> | ||||||
|  |                     </execution> | ||||||
|  |                 </executions> | ||||||
|  |             </plugin> | ||||||
|         </plugins> |         </plugins> | ||||||
|         <resources> |         <resources> | ||||||
|             <resource> |             <resource> | ||||||
| @@ -403,8 +454,8 @@ | |||||||
|     <repositories> |     <repositories> | ||||||
|         <repository> |         <repository> | ||||||
|             <id>public</id> |             <id>public</id> | ||||||
|             <name>aliyun nexus</name> |             <name>huawei nexus</name> | ||||||
|             <url>https://maven.aliyun.com/repository/public/</url> |             <url>https://mirrors.huaweicloud.com/repository/maven/</url> | ||||||
|             <releases> |             <releases> | ||||||
|                 <enabled>true</enabled> |                 <enabled>true</enabled> | ||||||
|             </releases> |             </releases> | ||||||
| @@ -414,8 +465,8 @@ | |||||||
|     <pluginRepositories> |     <pluginRepositories> | ||||||
|         <pluginRepository> |         <pluginRepository> | ||||||
|             <id>public</id> |             <id>public</id> | ||||||
|             <name>aliyun nexus</name> |             <name>huawei nexus</name> | ||||||
|             <url>https://maven.aliyun.com/repository/public/</url> |             <url>https://mirrors.huaweicloud.com/repository/maven/</url> | ||||||
|             <releases> |             <releases> | ||||||
|                 <enabled>true</enabled> |                 <enabled>true</enabled> | ||||||
|             </releases> |             </releases> | ||||||
|   | |||||||
| @@ -1,4 +1,5 @@ | |||||||
| FROM anapsix/alpine-java:8_server-jre_unlimited | #FROM findepi/graalvm:java17-native | ||||||
|  | FROM openjdk:17.0.2-oraclelinux8 | ||||||
|  |  | ||||||
| MAINTAINER Lion Li | MAINTAINER Lion Li | ||||||
|  |  | ||||||
| @@ -8,16 +9,16 @@ RUN mkdir -p /ruoyi/server/logs \ | |||||||
|  |  | ||||||
| WORKDIR /ruoyi/server | WORKDIR /ruoyi/server | ||||||
|  |  | ||||||
| ENV SERVER_PORT=8080 | ENV SERVER_PORT=8080 LANG=C.UTF-8 LC_ALL=C.UTF-8 JAVA_OPTS="" | ||||||
|  |  | ||||||
| EXPOSE ${SERVER_PORT} | EXPOSE ${SERVER_PORT} | ||||||
|  |  | ||||||
| ADD ./target/ruoyi-admin.jar ./app.jar | ADD ./target/ruoyi-admin.jar ./app.jar | ||||||
|  |  | ||||||
| ENTRYPOINT ["java", \ | ENTRYPOINT java -Djava.security.egd=file:/dev/./urandom -Dserver.port=${SERVER_PORT} \ | ||||||
|             "-Djava.security.egd=file:/dev/./urandom", \ |            # 应用名称 如果想区分集群节点监控 改成不同的名称即可 | ||||||
|             "-Dserver.port=${SERVER_PORT}", \ |            #-Dskywalking.agent.service_name=ruoyi-server \ | ||||||
|             # 应用名称 如果想区分集群节点监控 改成不同的名称即可 |            #-javaagent:/ruoyi/skywalking/agent/skywalking-agent.jar \ | ||||||
| #            "-Dskywalking.agent.service_name=ruoyi-server", \ |            -jar app.jar \ | ||||||
| #            "-javaagent:/ruoyi/skywalking/agent/skywalking-agent.jar", \ |            -XX:+HeapDumpOnOutOfMemoryError -Xlog:gc*,:time,tags,level -XX:+UseZGC ${JAVA_OPTS} | ||||||
|             "-jar", "app.jar"] |  | ||||||
|   | |||||||
| @@ -4,8 +4,8 @@ | |||||||
|          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> |          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||||||
|     <parent> |     <parent> | ||||||
|         <artifactId>ruoyi-vue-plus</artifactId> |         <artifactId>ruoyi-vue-plus</artifactId> | ||||||
|         <groupId>com.ruoyi</groupId> |         <groupId>org.dromara</groupId> | ||||||
|         <version>4.5.0</version> |         <version>${revision}</version> | ||||||
|     </parent> |     </parent> | ||||||
|     <modelVersion>4.0.0</modelVersion> |     <modelVersion>4.0.0</modelVersion> | ||||||
|     <packaging>jar</packaging> |     <packaging>jar</packaging> | ||||||
| @@ -17,13 +17,6 @@ | |||||||
|  |  | ||||||
|     <dependencies> |     <dependencies> | ||||||
|  |  | ||||||
|         <!-- spring-boot-devtools --> |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>org.springframework.boot</groupId> |  | ||||||
|             <artifactId>spring-boot-devtools</artifactId> |  | ||||||
|             <optional>true</optional> <!-- 表示依赖不会传递 --> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <!-- Mysql驱动包 --> |         <!-- Mysql驱动包 --> | ||||||
|         <dependency> |         <dependency> | ||||||
|             <groupId>com.mysql</groupId> |             <groupId>com.mysql</groupId> | ||||||
| @@ -45,45 +38,59 @@ | |||||||
|             <artifactId>mssql-jdbc</artifactId> |             <artifactId>mssql-jdbc</artifactId> | ||||||
|         </dependency> |         </dependency> | ||||||
|  |  | ||||||
|         <!-- 核心模块--> |  | ||||||
|         <dependency> |         <dependency> | ||||||
|             <groupId>com.ruoyi</groupId> |             <groupId>org.dromara</groupId> | ||||||
|             <artifactId>ruoyi-framework</artifactId> |             <artifactId>ruoyi-common-doc</artifactId> | ||||||
|         </dependency> |         </dependency> | ||||||
|  |  | ||||||
|         <dependency> |         <dependency> | ||||||
|             <groupId>com.ruoyi</groupId> |             <groupId>org.dromara</groupId> | ||||||
|  |             <artifactId>ruoyi-common-social</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>org.dromara</groupId> | ||||||
|  |             <artifactId>ruoyi-common-ratelimiter</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>org.dromara</groupId> | ||||||
|             <artifactId>ruoyi-system</artifactId> |             <artifactId>ruoyi-system</artifactId> | ||||||
|         </dependency> |         </dependency> | ||||||
|  |  | ||||||
|         <dependency> |         <dependency> | ||||||
|             <groupId>com.ruoyi</groupId> |             <groupId>org.dromara</groupId> | ||||||
|             <artifactId>ruoyi-job</artifactId> |             <artifactId>ruoyi-job</artifactId> | ||||||
|         </dependency> |         </dependency> | ||||||
|  |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>com.ruoyi</groupId> |  | ||||||
|             <artifactId>ruoyi-oss</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <!-- 代码生成--> |         <!-- 代码生成--> | ||||||
|         <dependency> |         <dependency> | ||||||
|             <groupId>com.ruoyi</groupId> |             <groupId>org.dromara</groupId> | ||||||
|             <artifactId>ruoyi-generator</artifactId> |             <artifactId>ruoyi-generator</artifactId> | ||||||
|         </dependency> |         </dependency> | ||||||
|  |  | ||||||
|         <!--  demo模块  --> |         <!--  demo模块  --> | ||||||
|         <dependency> |         <dependency> | ||||||
|             <groupId>com.ruoyi</groupId> |             <groupId>org.dromara</groupId> | ||||||
|             <artifactId>ruoyi-demo</artifactId> |             <artifactId>ruoyi-demo</artifactId> | ||||||
|         </dependency> |         </dependency> | ||||||
|  |  | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>de.codecentric</groupId> | ||||||
|  |             <artifactId>spring-boot-admin-starter-client</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|         <dependency> |         <dependency> | ||||||
|             <groupId>org.springframework.boot</groupId> |             <groupId>org.springframework.boot</groupId> | ||||||
|             <artifactId>spring-boot-starter-test</artifactId> |             <artifactId>spring-boot-starter-test</artifactId> | ||||||
|             <scope>test</scope> |             <scope>test</scope> | ||||||
|         </dependency> |         </dependency> | ||||||
|  |  | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>me.zhyd.oauth</groupId> | ||||||
|  |             <artifactId>JustAuth</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|         <!-- skywalking 整合 logback --> |         <!-- skywalking 整合 logback --> | ||||||
| <!--        <dependency>--> | <!--        <dependency>--> | ||||||
| <!--            <groupId>org.apache.skywalking</groupId>--> | <!--            <groupId>org.apache.skywalking</groupId>--> | ||||||
| @@ -105,9 +112,6 @@ | |||||||
|                 <groupId>org.springframework.boot</groupId> |                 <groupId>org.springframework.boot</groupId> | ||||||
|                 <artifactId>spring-boot-maven-plugin</artifactId> |                 <artifactId>spring-boot-maven-plugin</artifactId> | ||||||
|                 <version>${spring-boot.version}</version> |                 <version>${spring-boot.version}</version> | ||||||
|                 <configuration> |  | ||||||
|                     <fork>true</fork> <!-- 如果没有该配置,devtools不会生效 --> |  | ||||||
|                 </configuration> |  | ||||||
|                 <executions> |                 <executions> | ||||||
|                     <execution> |                     <execution> | ||||||
|                         <goals> |                         <goals> | ||||||
| @@ -116,10 +120,15 @@ | |||||||
|                     </execution> |                     </execution> | ||||||
|                 </executions> |                 </executions> | ||||||
|             </plugin> |             </plugin> | ||||||
|  |             <plugin> | ||||||
|  |                 <groupId>org.apache.maven.plugins</groupId> | ||||||
|  |                 <artifactId>maven-jar-plugin</artifactId> | ||||||
|  |                 <version>${maven-jar-plugin.version}</version> | ||||||
|  |             </plugin> | ||||||
|             <plugin> |             <plugin> | ||||||
|                 <groupId>org.apache.maven.plugins</groupId> |                 <groupId>org.apache.maven.plugins</groupId> | ||||||
|                 <artifactId>maven-war-plugin</artifactId> |                 <artifactId>maven-war-plugin</artifactId> | ||||||
|                 <version>3.2.2</version> |                 <version>${maven-war-plugin.version}</version> | ||||||
|                 <configuration> |                 <configuration> | ||||||
|                     <failOnMissingWebXml>false</failOnMissingWebXml> |                     <failOnMissingWebXml>false</failOnMissingWebXml> | ||||||
|                     <warName>${project.artifactId}</warName> |                     <warName>${project.artifactId}</warName> | ||||||
|   | |||||||
| @@ -1,112 +0,0 @@ | |||||||
| package com.ruoyi.web.controller.common; |  | ||||||
|  |  | ||||||
| import cn.dev33.satoken.annotation.SaIgnore; |  | ||||||
| import cn.hutool.captcha.AbstractCaptcha; |  | ||||||
| import cn.hutool.captcha.generator.CodeGenerator; |  | ||||||
| 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.core.domain.R; |  | ||||||
| import com.ruoyi.common.enums.CaptchaType; |  | ||||||
| import com.ruoyi.common.utils.StringUtils; |  | ||||||
| import com.ruoyi.common.utils.redis.RedisUtils; |  | ||||||
| import com.ruoyi.common.utils.reflect.ReflectUtils; |  | ||||||
| import com.ruoyi.common.utils.spring.SpringUtils; |  | ||||||
| import com.ruoyi.framework.config.properties.CaptchaProperties; |  | ||||||
| import com.ruoyi.sms.config.properties.SmsProperties; |  | ||||||
| import com.ruoyi.sms.core.SmsTemplate; |  | ||||||
| import com.ruoyi.sms.entity.SmsResult; |  | ||||||
| import com.ruoyi.system.service.ISysConfigService; |  | ||||||
| import 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.RestController; |  | ||||||
|  |  | ||||||
| import javax.validation.constraints.NotBlank; |  | ||||||
| import java.time.Duration; |  | ||||||
| import java.util.HashMap; |  | ||||||
| import java.util.Map; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * 验证码操作处理 |  | ||||||
|  * |  | ||||||
|  * @author Lion Li |  | ||||||
|  */ |  | ||||||
| @SaIgnore |  | ||||||
| @Slf4j |  | ||||||
| @Validated |  | ||||||
| @RequiredArgsConstructor |  | ||||||
| @RestController |  | ||||||
| public class CaptchaController { |  | ||||||
|  |  | ||||||
|     private final CaptchaProperties captchaProperties; |  | ||||||
|     private final SmsProperties smsProperties; |  | ||||||
|     private final ISysConfigService configService; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 短信验证码 |  | ||||||
|      * |  | ||||||
|      * @param phonenumber 用户手机号 |  | ||||||
|      */ |  | ||||||
|     @GetMapping("/captchaSms") |  | ||||||
|     public R<Void> smsCaptcha(@NotBlank(message = "{user.phonenumber.not.blank}") |  | ||||||
|                               String phonenumber) { |  | ||||||
|         if (!smsProperties.getEnabled()) { |  | ||||||
|             return R.fail("当前系统没有开启短信功能!"); |  | ||||||
|         } |  | ||||||
|         String key = CacheConstants.CAPTCHA_CODE_KEY + phonenumber; |  | ||||||
|         String code = RandomUtil.randomNumbers(4); |  | ||||||
|         RedisUtils.setCacheObject(key, code, Duration.ofMinutes(Constants.CAPTCHA_EXPIRATION)); |  | ||||||
|         // 验证码模板id 自行处理 (查数据库或写死均可) |  | ||||||
|         String templateId = ""; |  | ||||||
|         Map<String, String> map = new HashMap<>(1); |  | ||||||
|         map.put("code", code); |  | ||||||
|         SmsTemplate smsTemplate = SpringUtils.getBean(SmsTemplate.class); |  | ||||||
|         SmsResult result = smsTemplate.send(phonenumber, templateId, map); |  | ||||||
|         if (!result.isSuccess()) { |  | ||||||
|             log.error("验证码短信发送异常 => {}", result); |  | ||||||
|             return R.fail(result.getMessage()); |  | ||||||
|         } |  | ||||||
|         return R.ok(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 生成验证码 |  | ||||||
|      */ |  | ||||||
|     @GetMapping("/captchaImage") |  | ||||||
|     public R<Map<String, Object>> getCode() { |  | ||||||
|         Map<String, Object> ajax = new HashMap<>(); |  | ||||||
|         boolean captchaEnabled = configService.selectCaptchaEnabled(); |  | ||||||
|         ajax.put("captchaEnabled", captchaEnabled); |  | ||||||
|         if (!captchaEnabled) { |  | ||||||
|             return R.ok(ajax); |  | ||||||
|         } |  | ||||||
|         // 保存验证码信息 |  | ||||||
|         String uuid = IdUtil.simpleUUID(); |  | ||||||
|         String verifyKey = CacheConstants.CAPTCHA_CODE_KEY + uuid; |  | ||||||
|         // 生成验证码 |  | ||||||
|         CaptchaType captchaType = captchaProperties.getType(); |  | ||||||
|         boolean isMath = CaptchaType.MATH == captchaType; |  | ||||||
|         Integer length = isMath ? captchaProperties.getNumberLength() : captchaProperties.getCharLength(); |  | ||||||
|         CodeGenerator codeGenerator = ReflectUtils.newInstance(captchaType.getClazz(), length); |  | ||||||
|         AbstractCaptcha captcha = SpringUtils.getBean(captchaProperties.getCategory().getClazz()); |  | ||||||
|         captcha.setGenerator(codeGenerator); |  | ||||||
|         captcha.createCode(); |  | ||||||
|         String code = captcha.getCode(); |  | ||||||
|         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("img", captcha.getImageBase64()); |  | ||||||
|         return R.ok(ajax); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -1,169 +0,0 @@ | |||||||
| package com.ruoyi.web.controller.monitor; |  | ||||||
|  |  | ||||||
| import cn.dev33.satoken.annotation.SaCheckPermission; |  | ||||||
| import cn.hutool.core.collection.CollUtil; |  | ||||||
| import com.ruoyi.common.constant.CacheConstants; |  | ||||||
| import com.ruoyi.common.constant.CacheNames; |  | ||||||
| import com.ruoyi.common.core.domain.R; |  | ||||||
| import com.ruoyi.common.utils.JsonUtils; |  | ||||||
| import com.ruoyi.common.utils.StringUtils; |  | ||||||
| import com.ruoyi.common.utils.redis.CacheUtils; |  | ||||||
| import com.ruoyi.common.utils.redis.RedisUtils; |  | ||||||
| import com.ruoyi.system.domain.SysCache; |  | ||||||
| import lombok.RequiredArgsConstructor; |  | ||||||
| import org.redisson.spring.data.connection.RedissonConnectionFactory; |  | ||||||
| import org.springframework.data.redis.connection.RedisConnection; |  | ||||||
| import org.springframework.web.bind.annotation.*; |  | ||||||
|  |  | ||||||
| import java.util.*; |  | ||||||
| import java.util.stream.Collectors; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * 缓存监控 |  | ||||||
|  * |  | ||||||
|  * @author Lion Li |  | ||||||
|  */ |  | ||||||
| @RequiredArgsConstructor |  | ||||||
| @RestController |  | ||||||
| @RequestMapping("/monitor/cache") |  | ||||||
| public class CacheController { |  | ||||||
|  |  | ||||||
|     private final RedissonConnectionFactory connectionFactory; |  | ||||||
|  |  | ||||||
|     private final static List<SysCache> CACHES = new ArrayList<>(); |  | ||||||
|  |  | ||||||
|     static { |  | ||||||
|         CACHES.add(new SysCache(CacheConstants.LOGIN_TOKEN_KEY, "用户信息")); |  | ||||||
|         CACHES.add(new SysCache(CacheConstants.ONLINE_TOKEN_KEY, "在线用户")); |  | ||||||
|         CACHES.add(new SysCache(CacheNames.SYS_CONFIG, "配置信息")); |  | ||||||
|         CACHES.add(new SysCache(CacheNames.SYS_DICT, "数据字典")); |  | ||||||
|         CACHES.add(new SysCache(CacheConstants.CAPTCHA_CODE_KEY, "验证码")); |  | ||||||
|         CACHES.add(new SysCache(CacheConstants.REPEAT_SUBMIT_KEY, "防重提交")); |  | ||||||
|         CACHES.add(new SysCache(CacheConstants.RATE_LIMIT_KEY, "限流处理")); |  | ||||||
|         CACHES.add(new SysCache(CacheNames.SYS_OSS_CONFIG, "OSS配置")); |  | ||||||
|         CACHES.add(new SysCache(CacheConstants.PWD_ERR_CNT_KEY, "密码错误次数")); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 获取缓存监控列表 |  | ||||||
|      */ |  | ||||||
|     @SaCheckPermission("monitor:cache:list") |  | ||||||
|     @GetMapping() |  | ||||||
|     public R<Map<String, Object>> getInfo() throws Exception { |  | ||||||
|         RedisConnection connection = connectionFactory.getConnection(); |  | ||||||
|         Properties info = connection.info(); |  | ||||||
|         Properties commandStats = connection.info("commandstats"); |  | ||||||
|         Long dbSize = connection.dbSize(); |  | ||||||
|  |  | ||||||
|         Map<String, Object> result = new HashMap<>(3); |  | ||||||
|         result.put("info", info); |  | ||||||
|         result.put("dbSize", dbSize); |  | ||||||
|  |  | ||||||
|         List<Map<String, String>> pieList = new ArrayList<>(); |  | ||||||
|         if (commandStats != null) { |  | ||||||
|             commandStats.stringPropertyNames().forEach(key -> { |  | ||||||
|                 Map<String, String> data = new HashMap<>(2); |  | ||||||
|                 String property = commandStats.getProperty(key); |  | ||||||
|                 data.put("name", StringUtils.removeStart(key, "cmdstat_")); |  | ||||||
|                 data.put("value", StringUtils.substringBetween(property, "calls=", ",usec")); |  | ||||||
|                 pieList.add(data); |  | ||||||
|             }); |  | ||||||
|         } |  | ||||||
|         result.put("commandStats", pieList); |  | ||||||
|         return 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, ":"); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,128 +0,0 @@ | |||||||
| package com.ruoyi.web.controller.system; |  | ||||||
|  |  | ||||||
| import cn.dev33.satoken.annotation.SaIgnore; |  | ||||||
| import com.ruoyi.common.constant.Constants; |  | ||||||
| import com.ruoyi.common.core.domain.R; |  | ||||||
| import com.ruoyi.common.core.domain.entity.SysMenu; |  | ||||||
| import com.ruoyi.common.core.domain.entity.SysUser; |  | ||||||
| import com.ruoyi.common.core.domain.model.LoginBody; |  | ||||||
| import com.ruoyi.common.core.domain.model.LoginUser; |  | ||||||
| import com.ruoyi.common.core.domain.model.SmsLoginBody; |  | ||||||
| import com.ruoyi.common.helper.LoginHelper; |  | ||||||
| import com.ruoyi.system.domain.vo.RouterVo; |  | ||||||
| import com.ruoyi.system.service.ISysMenuService; |  | ||||||
| import com.ruoyi.system.service.ISysUserService; |  | ||||||
| import com.ruoyi.system.service.SysLoginService; |  | ||||||
| import lombok.RequiredArgsConstructor; |  | ||||||
| import org.springframework.validation.annotation.Validated; |  | ||||||
| import org.springframework.web.bind.annotation.GetMapping; |  | ||||||
| import org.springframework.web.bind.annotation.PostMapping; |  | ||||||
| import org.springframework.web.bind.annotation.RequestBody; |  | ||||||
| import org.springframework.web.bind.annotation.RestController; |  | ||||||
|  |  | ||||||
| import javax.validation.constraints.NotBlank; |  | ||||||
| import java.util.HashMap; |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.Map; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * 登录验证 |  | ||||||
|  * |  | ||||||
|  * @author Lion Li |  | ||||||
|  */ |  | ||||||
| @Validated |  | ||||||
| @RequiredArgsConstructor |  | ||||||
| @RestController |  | ||||||
| public class SysLoginController { |  | ||||||
|  |  | ||||||
|     private final SysLoginService loginService; |  | ||||||
|     private final ISysMenuService menuService; |  | ||||||
|     private final ISysUserService userService; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 登录方法 |  | ||||||
|      * |  | ||||||
|      * @param loginBody 登录信息 |  | ||||||
|      * @return 结果 |  | ||||||
|      */ |  | ||||||
|     @SaIgnore |  | ||||||
|     @PostMapping("/login") |  | ||||||
|     public R<Map<String, Object>> login(@Validated @RequestBody LoginBody loginBody) { |  | ||||||
|         Map<String, Object> ajax = new HashMap<>(); |  | ||||||
|         // 生成令牌 |  | ||||||
|         String token = loginService.login(loginBody.getUsername(), loginBody.getPassword(), loginBody.getCode(), |  | ||||||
|             loginBody.getUuid()); |  | ||||||
|         ajax.put(Constants.TOKEN, token); |  | ||||||
|         return R.ok(ajax); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 短信登录(示例) |  | ||||||
|      * |  | ||||||
|      * @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") |  | ||||||
|     public R<Void> logout() { |  | ||||||
|         loginService.logout(); |  | ||||||
|         return R.ok("退出成功"); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 获取用户信息 |  | ||||||
|      * |  | ||||||
|      * @return 用户信息 |  | ||||||
|      */ |  | ||||||
|     @GetMapping("getInfo") |  | ||||||
|     public R<Map<String, Object>> getInfo() { |  | ||||||
|         LoginUser loginUser = LoginHelper.getLoginUser(); |  | ||||||
|         SysUser user = userService.selectUserById(loginUser.getUserId()); |  | ||||||
|         Map<String, Object> ajax = new HashMap<>(); |  | ||||||
|         ajax.put("user", user); |  | ||||||
|         ajax.put("roles", loginUser.getRolePermission()); |  | ||||||
|         ajax.put("permissions", loginUser.getMenuPermission()); |  | ||||||
|         return R.ok(ajax); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 获取路由信息 |  | ||||||
|      * |  | ||||||
|      * @return 路由信息 |  | ||||||
|      */ |  | ||||||
|     @GetMapping("getRouters") |  | ||||||
|     public R<List<RouterVo>> getRouters() { |  | ||||||
|         Long userId = LoginHelper.getUserId(); |  | ||||||
|         List<SysMenu> menus = menuService.selectMenuTreeByUserId(userId); |  | ||||||
|         return R.ok(menuService.buildMenus(menus)); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,127 +0,0 @@ | |||||||
| package com.ruoyi.web.controller.system; |  | ||||||
|  |  | ||||||
| import cn.dev33.satoken.annotation.SaCheckPermission; |  | ||||||
| import cn.hutool.core.lang.tree.Tree; |  | ||||||
| import com.ruoyi.common.annotation.Log; |  | ||||||
| import com.ruoyi.common.constant.UserConstants; |  | ||||||
| import com.ruoyi.common.core.controller.BaseController; |  | ||||||
| import com.ruoyi.common.core.domain.R; |  | ||||||
| import com.ruoyi.common.core.domain.entity.SysMenu; |  | ||||||
| import com.ruoyi.common.enums.BusinessType; |  | ||||||
| import com.ruoyi.common.utils.StringUtils; |  | ||||||
| import com.ruoyi.system.service.ISysMenuService; |  | ||||||
| import lombok.RequiredArgsConstructor; |  | ||||||
| import org.springframework.validation.annotation.Validated; |  | ||||||
| import org.springframework.web.bind.annotation.*; |  | ||||||
|  |  | ||||||
| import java.util.HashMap; |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.Map; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * 菜单信息 |  | ||||||
|  * |  | ||||||
|  * @author Lion Li |  | ||||||
|  */ |  | ||||||
| @Validated |  | ||||||
| @RequiredArgsConstructor |  | ||||||
| @RestController |  | ||||||
| @RequestMapping("/system/menu") |  | ||||||
| public class SysMenuController extends BaseController { |  | ||||||
|  |  | ||||||
|     private final ISysMenuService menuService; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 获取菜单列表 |  | ||||||
|      */ |  | ||||||
|     @SaCheckPermission("system:menu:list") |  | ||||||
|     @GetMapping("/list") |  | ||||||
|     public R<List<SysMenu>> list(SysMenu menu) { |  | ||||||
|         List<SysMenu> menus = menuService.selectMenuList(menu, getUserId()); |  | ||||||
|         return R.ok(menus); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 根据菜单编号获取详细信息 |  | ||||||
|      * |  | ||||||
|      * @param menuId 菜单ID |  | ||||||
|      */ |  | ||||||
|     @SaCheckPermission("system:menu:query") |  | ||||||
|     @GetMapping(value = "/{menuId}") |  | ||||||
|     public R<SysMenu> getInfo(@PathVariable Long menuId) { |  | ||||||
|         return R.ok(menuService.selectMenuById(menuId)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 获取菜单下拉树列表 |  | ||||||
|      */ |  | ||||||
|     @GetMapping("/treeselect") |  | ||||||
|     public R<List<Tree<Long>>> treeselect(SysMenu menu) { |  | ||||||
|         List<SysMenu> menus = menuService.selectMenuList(menu, getUserId()); |  | ||||||
|         return R.ok(menuService.buildMenuTreeSelect(menus)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 加载对应角色菜单列表树 |  | ||||||
|      * |  | ||||||
|      * @param roleId 角色ID |  | ||||||
|      */ |  | ||||||
|     @GetMapping(value = "/roleMenuTreeselect/{roleId}") |  | ||||||
|     public R<Map<String, Object>> roleMenuTreeselect(@PathVariable("roleId") Long roleId) { |  | ||||||
|         List<SysMenu> menus = menuService.selectMenuList(getUserId()); |  | ||||||
|         Map<String, Object> ajax = new HashMap<>(); |  | ||||||
|         ajax.put("checkedKeys", menuService.selectMenuListByRoleId(roleId)); |  | ||||||
|         ajax.put("menus", menuService.buildMenuTreeSelect(menus)); |  | ||||||
|         return R.ok(ajax); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 新增菜单 |  | ||||||
|      */ |  | ||||||
|     @SaCheckPermission("system:menu:add") |  | ||||||
|     @Log(title = "菜单管理", businessType = BusinessType.INSERT) |  | ||||||
|     @PostMapping |  | ||||||
|     public R<Void> add(@Validated @RequestBody SysMenu menu) { |  | ||||||
|         if (UserConstants.NOT_UNIQUE.equals(menuService.checkMenuNameUnique(menu))) { |  | ||||||
|             return R.fail("新增菜单'" + menu.getMenuName() + "'失败,菜单名称已存在"); |  | ||||||
|         } else if (UserConstants.YES_FRAME.equals(menu.getIsFrame()) && !StringUtils.ishttp(menu.getPath())) { |  | ||||||
|             return R.fail("新增菜单'" + menu.getMenuName() + "'失败,地址必须以http(s)://开头"); |  | ||||||
|         } |  | ||||||
|         return toAjax(menuService.insertMenu(menu)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 修改菜单 |  | ||||||
|      */ |  | ||||||
|     @SaCheckPermission("system:menu:edit") |  | ||||||
|     @Log(title = "菜单管理", businessType = BusinessType.UPDATE) |  | ||||||
|     @PutMapping |  | ||||||
|     public R<Void> edit(@Validated @RequestBody SysMenu menu) { |  | ||||||
|         if (UserConstants.NOT_UNIQUE.equals(menuService.checkMenuNameUnique(menu))) { |  | ||||||
|             return R.fail("修改菜单'" + menu.getMenuName() + "'失败,菜单名称已存在"); |  | ||||||
|         } else if (UserConstants.YES_FRAME.equals(menu.getIsFrame()) && !StringUtils.ishttp(menu.getPath())) { |  | ||||||
|             return R.fail("修改菜单'" + menu.getMenuName() + "'失败,地址必须以http(s)://开头"); |  | ||||||
|         } else if (menu.getMenuId().equals(menu.getParentId())) { |  | ||||||
|             return R.fail("修改菜单'" + menu.getMenuName() + "'失败,上级菜单不能选择自己"); |  | ||||||
|         } |  | ||||||
|         return toAjax(menuService.updateMenu(menu)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 删除菜单 |  | ||||||
|      * |  | ||||||
|      * @param menuId 菜单ID |  | ||||||
|      */ |  | ||||||
|     @SaCheckPermission("system:menu:remove") |  | ||||||
|     @Log(title = "菜单管理", businessType = BusinessType.DELETE) |  | ||||||
|     @DeleteMapping("/{menuId}") |  | ||||||
|     public R<Void> remove(@PathVariable("menuId") Long menuId) { |  | ||||||
|         if (menuService.hasChildByMenuId(menuId)) { |  | ||||||
|             return R.warn("存在子菜单,不允许删除"); |  | ||||||
|         } |  | ||||||
|         if (menuService.checkMenuExistRole(menuId)) { |  | ||||||
|             return R.warn("菜单已分配,不允许删除"); |  | ||||||
|         } |  | ||||||
|         return toAjax(menuService.deleteMenuById(menuId)); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,128 +0,0 @@ | |||||||
| package com.ruoyi.web.controller.system; |  | ||||||
|  |  | ||||||
| import cn.dev33.satoken.secure.BCrypt; |  | ||||||
| import cn.hutool.core.io.FileUtil; |  | ||||||
| import com.ruoyi.common.annotation.Log; |  | ||||||
| import com.ruoyi.common.constant.UserConstants; |  | ||||||
| import com.ruoyi.common.core.controller.BaseController; |  | ||||||
| import com.ruoyi.common.core.domain.R; |  | ||||||
| import com.ruoyi.common.core.domain.entity.SysUser; |  | ||||||
| import com.ruoyi.common.enums.BusinessType; |  | ||||||
| import com.ruoyi.common.helper.LoginHelper; |  | ||||||
| import com.ruoyi.common.utils.StringUtils; |  | ||||||
| import com.ruoyi.common.utils.file.MimeTypeUtils; |  | ||||||
| import com.ruoyi.system.domain.SysOss; |  | ||||||
| import com.ruoyi.system.domain.vo.SysOssVo; |  | ||||||
| import com.ruoyi.system.service.ISysOssService; |  | ||||||
| import com.ruoyi.system.service.ISysUserService; |  | ||||||
| import lombok.RequiredArgsConstructor; |  | ||||||
| import org.springframework.http.MediaType; |  | ||||||
| import org.springframework.validation.annotation.Validated; |  | ||||||
| import org.springframework.web.bind.annotation.*; |  | ||||||
| import org.springframework.web.multipart.MultipartFile; |  | ||||||
|  |  | ||||||
| import java.util.Arrays; |  | ||||||
| import java.util.HashMap; |  | ||||||
| import java.util.Map; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * 个人信息 业务处理 |  | ||||||
|  * |  | ||||||
|  * @author Lion Li |  | ||||||
|  */ |  | ||||||
| @Validated |  | ||||||
| @RequiredArgsConstructor |  | ||||||
| @RestController |  | ||||||
| @RequestMapping("/system/user/profile") |  | ||||||
| public class SysProfileController extends BaseController { |  | ||||||
|  |  | ||||||
|     private final ISysUserService userService; |  | ||||||
|     private final ISysOssService iSysOssService; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 个人信息 |  | ||||||
|      */ |  | ||||||
|     @GetMapping |  | ||||||
|     public R<Map<String, Object>> profile() { |  | ||||||
|         SysUser user = userService.selectUserById(getUserId()); |  | ||||||
|         Map<String, Object> ajax = new HashMap<>(); |  | ||||||
|         ajax.put("user", user); |  | ||||||
|         ajax.put("roleGroup", userService.selectUserRoleGroup(user.getUserName())); |  | ||||||
|         ajax.put("postGroup", userService.selectUserPostGroup(user.getUserName())); |  | ||||||
|         return R.ok(ajax); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 修改用户 |  | ||||||
|      */ |  | ||||||
|     @Log(title = "个人信息", businessType = BusinessType.UPDATE) |  | ||||||
|     @PutMapping |  | ||||||
|     public R<Void> updateProfile(@RequestBody SysUser user) { |  | ||||||
|         if (StringUtils.isNotEmpty(user.getPhonenumber()) |  | ||||||
|             && UserConstants.NOT_UNIQUE.equals(userService.checkPhoneUnique(user))) { |  | ||||||
|             return R.fail("修改用户'" + user.getUserName() + "'失败,手机号码已存在"); |  | ||||||
|         } |  | ||||||
|         if (StringUtils.isNotEmpty(user.getEmail()) |  | ||||||
|             && UserConstants.NOT_UNIQUE.equals(userService.checkEmailUnique(user))) { |  | ||||||
|             return R.fail("修改用户'" + user.getUserName() + "'失败,邮箱账号已存在"); |  | ||||||
|         } |  | ||||||
|         user.setUserId(getUserId()); |  | ||||||
|         user.setUserName(null); |  | ||||||
|         user.setPassword(null); |  | ||||||
|         user.setAvatar(null); |  | ||||||
|         user.setDeptId(null); |  | ||||||
|         if (userService.updateUserProfile(user) > 0) { |  | ||||||
|             return R.ok(); |  | ||||||
|         } |  | ||||||
|         return R.fail("修改个人信息异常,请联系管理员"); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 重置密码 |  | ||||||
|      * |  | ||||||
|      * @param newPassword 旧密码 |  | ||||||
|      * @param oldPassword 新密码 |  | ||||||
|      */ |  | ||||||
|     @Log(title = "个人信息", businessType = BusinessType.UPDATE) |  | ||||||
|     @PutMapping("/updatePwd") |  | ||||||
|     public R<Void> updatePwd(String oldPassword, String newPassword) { |  | ||||||
|         SysUser user = userService.selectUserById(LoginHelper.getUserId()); |  | ||||||
|         String userName = user.getUserName(); |  | ||||||
|         String password = user.getPassword(); |  | ||||||
|         if (!BCrypt.checkpw(oldPassword, password)) { |  | ||||||
|             return R.fail("修改密码失败,旧密码错误"); |  | ||||||
|         } |  | ||||||
|         if (BCrypt.checkpw(newPassword, password)) { |  | ||||||
|             return R.fail("新密码不能与旧密码相同"); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if (userService.resetUserPwd(userName, BCrypt.hashpw(newPassword)) > 0) { |  | ||||||
|             return R.ok(); |  | ||||||
|         } |  | ||||||
|         return R.fail("修改密码异常,请联系管理员"); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 头像上传 |  | ||||||
|      * |  | ||||||
|      * @param avatarfile 用户头像 |  | ||||||
|      */ |  | ||||||
|     @Log(title = "用户头像", businessType = BusinessType.UPDATE) |  | ||||||
|     @PostMapping(value = "/avatar", consumes = MediaType.MULTIPART_FORM_DATA_VALUE) |  | ||||||
|     public R<Map<String, Object>> avatar(@RequestPart("avatarfile") MultipartFile avatarfile) { |  | ||||||
|         Map<String, Object> ajax = new HashMap<>(); |  | ||||||
|         if (!avatarfile.isEmpty()) { |  | ||||||
|             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(); |  | ||||||
|             if (userService.updateUserAvatar(getUsername(), avatar)) { |  | ||||||
|                 ajax.put("imgUrl", avatar); |  | ||||||
|                 return R.ok(ajax); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return R.fail("上传图片异常,请联系管理员"); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,40 +0,0 @@ | |||||||
| package com.ruoyi.web.controller.system; |  | ||||||
|  |  | ||||||
| import cn.dev33.satoken.annotation.SaIgnore; |  | ||||||
| import com.ruoyi.common.core.controller.BaseController; |  | ||||||
| import com.ruoyi.common.core.domain.R; |  | ||||||
| import com.ruoyi.common.core.domain.model.RegisterBody; |  | ||||||
| import com.ruoyi.system.service.ISysConfigService; |  | ||||||
| import com.ruoyi.system.service.SysRegisterService; |  | ||||||
| import lombok.RequiredArgsConstructor; |  | ||||||
| import org.springframework.validation.annotation.Validated; |  | ||||||
| import org.springframework.web.bind.annotation.PostMapping; |  | ||||||
| import org.springframework.web.bind.annotation.RequestBody; |  | ||||||
| import org.springframework.web.bind.annotation.RestController; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * 注册验证 |  | ||||||
|  * |  | ||||||
|  * @author Lion Li |  | ||||||
|  */ |  | ||||||
| @Validated |  | ||||||
| @RequiredArgsConstructor |  | ||||||
| @RestController |  | ||||||
| public class SysRegisterController extends BaseController { |  | ||||||
|  |  | ||||||
|     private final SysRegisterService registerService; |  | ||||||
|     private final ISysConfigService configService; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 用户注册 |  | ||||||
|      */ |  | ||||||
|     @SaIgnore |  | ||||||
|     @PostMapping("/register") |  | ||||||
|     public R<Void> register(@Validated @RequestBody RegisterBody user) { |  | ||||||
|         if (!("true".equals(configService.selectConfigByKey("sys.account.registerUser")))) { |  | ||||||
|             return R.fail("当前系统没有开启注册功能!"); |  | ||||||
|         } |  | ||||||
|         registerService.register(user); |  | ||||||
|         return R.ok(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| package com.ruoyi; | package org.dromara; | ||||||
| 
 | 
 | ||||||
| import org.springframework.boot.SpringApplication; | import org.springframework.boot.SpringApplication; | ||||||
| import org.springframework.boot.autoconfigure.SpringBootApplication; | import org.springframework.boot.autoconfigure.SpringBootApplication; | ||||||
| @@ -7,15 +7,14 @@ import org.springframework.boot.context.metrics.buffering.BufferingApplicationSt | |||||||
| /** | /** | ||||||
|  * 启动程序 |  * 启动程序 | ||||||
|  * |  * | ||||||
|  * @author ruoyi |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| @SpringBootApplication | @SpringBootApplication | ||||||
| public class RuoYiApplication { | public class DromaraApplication { | ||||||
| 
 | 
 | ||||||
|     public static void main(String[] args) { |     public static void main(String[] args) { | ||||||
|         System.setProperty("spring.devtools.restart.enabled", "false"); |         SpringApplication application = new SpringApplication(DromaraApplication.class); | ||||||
|         SpringApplication application = new SpringApplication(RuoYiApplication.class); |  | ||||||
|         application.setApplicationStartup(new BufferingApplicationStartup(2048)); |         application.setApplicationStartup(new BufferingApplicationStartup(2048)); | ||||||
|         application.run(args); |         application.run(args); | ||||||
|         System.out.println("(♥◠‿◠)ノ゙  RuoYi-Vue-Plus启动成功   ლ(´ڡ`ლ)゙"); |         System.out.println("(♥◠‿◠)ノ゙  RuoYi-Vue-Plus启动成功   ლ(´ڡ`ლ)゙"); | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| package com.ruoyi; | package org.dromara; | ||||||
| 
 | 
 | ||||||
| import org.springframework.boot.builder.SpringApplicationBuilder; | import org.springframework.boot.builder.SpringApplicationBuilder; | ||||||
| import org.springframework.boot.web.servlet.support.SpringBootServletInitializer; | import org.springframework.boot.web.servlet.support.SpringBootServletInitializer; | ||||||
| @@ -6,13 +6,13 @@ import org.springframework.boot.web.servlet.support.SpringBootServletInitializer | |||||||
| /** | /** | ||||||
|  * web容器中进行部署 |  * web容器中进行部署 | ||||||
|  * |  * | ||||||
|  * @author ruoyi |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| public class RuoYiServletInitializer extends SpringBootServletInitializer { | public class DromaraServletInitializer extends SpringBootServletInitializer { | ||||||
| 
 | 
 | ||||||
|     @Override |     @Override | ||||||
|     protected SpringApplicationBuilder configure(SpringApplicationBuilder application) { |     protected SpringApplicationBuilder configure(SpringApplicationBuilder application) { | ||||||
|         return application.sources(RuoYiApplication.class); |         return application.sources(DromaraApplication.class); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
| @@ -0,0 +1,205 @@ | |||||||
|  | package org.dromara.web.controller; | ||||||
|  |  | ||||||
|  | import cn.dev33.satoken.annotation.SaIgnore; | ||||||
|  | import cn.hutool.core.collection.CollUtil; | ||||||
|  | import cn.hutool.core.util.ObjectUtil; | ||||||
|  | import jakarta.servlet.http.HttpServletRequest; | ||||||
|  | import lombok.RequiredArgsConstructor; | ||||||
|  | import lombok.extern.slf4j.Slf4j; | ||||||
|  | import me.zhyd.oauth.model.AuthResponse; | ||||||
|  | import me.zhyd.oauth.model.AuthUser; | ||||||
|  | import me.zhyd.oauth.request.AuthRequest; | ||||||
|  | import me.zhyd.oauth.utils.AuthStateUtils; | ||||||
|  | import org.dromara.common.core.constant.UserConstants; | ||||||
|  | import org.dromara.common.core.domain.R; | ||||||
|  | import org.dromara.common.core.domain.model.LoginBody; | ||||||
|  | import org.dromara.common.core.domain.model.RegisterBody; | ||||||
|  | import org.dromara.common.core.domain.model.SocialLoginBody; | ||||||
|  | import org.dromara.common.core.utils.*; | ||||||
|  | import org.dromara.common.encrypt.annotation.ApiEncrypt; | ||||||
|  | import org.dromara.common.json.utils.JsonUtils; | ||||||
|  | import org.dromara.common.satoken.utils.LoginHelper; | ||||||
|  | import org.dromara.common.social.config.properties.SocialLoginConfigProperties; | ||||||
|  | import org.dromara.common.social.config.properties.SocialProperties; | ||||||
|  | import org.dromara.common.social.utils.SocialUtils; | ||||||
|  | import org.dromara.common.tenant.helper.TenantHelper; | ||||||
|  | import org.dromara.common.websocket.utils.WebSocketUtils; | ||||||
|  | import org.dromara.system.domain.SysClient; | ||||||
|  | import org.dromara.system.domain.bo.SysTenantBo; | ||||||
|  | import org.dromara.system.domain.vo.SysTenantVo; | ||||||
|  | import org.dromara.system.service.ISysClientService; | ||||||
|  | import org.dromara.system.service.ISysConfigService; | ||||||
|  | import org.dromara.system.service.ISysSocialService; | ||||||
|  | import org.dromara.system.service.ISysTenantService; | ||||||
|  | import org.dromara.web.domain.vo.LoginTenantVo; | ||||||
|  | import org.dromara.web.domain.vo.LoginVo; | ||||||
|  | import org.dromara.web.domain.vo.TenantListVo; | ||||||
|  | import org.dromara.web.service.IAuthStrategy; | ||||||
|  | import org.dromara.web.service.SysLoginService; | ||||||
|  | import org.dromara.web.service.SysRegisterService; | ||||||
|  | import org.springframework.validation.annotation.Validated; | ||||||
|  | import org.springframework.web.bind.annotation.*; | ||||||
|  |  | ||||||
|  | import java.net.URL; | ||||||
|  | import java.util.List; | ||||||
|  | import java.util.concurrent.ScheduledExecutorService; | ||||||
|  | import java.util.concurrent.TimeUnit; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 认证 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | @Slf4j | ||||||
|  | @SaIgnore | ||||||
|  | @RequiredArgsConstructor | ||||||
|  | @RestController | ||||||
|  | @RequestMapping("/auth") | ||||||
|  | public class AuthController { | ||||||
|  |  | ||||||
|  |     private final SocialProperties socialProperties; | ||||||
|  |     private final SysLoginService loginService; | ||||||
|  |     private final SysRegisterService registerService; | ||||||
|  |     private final ISysConfigService configService; | ||||||
|  |     private final ISysTenantService tenantService; | ||||||
|  |     private final ISysSocialService socialUserService; | ||||||
|  |     private final ISysClientService clientService; | ||||||
|  |     private final ScheduledExecutorService scheduledExecutorService; | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 登录方法 | ||||||
|  |      * | ||||||
|  |      * @param body 登录信息 | ||||||
|  |      * @return 结果 | ||||||
|  |      */ | ||||||
|  |     @ApiEncrypt | ||||||
|  |     @PostMapping("/login") | ||||||
|  |     public R<LoginVo> login(@RequestBody String body) { | ||||||
|  |         LoginBody loginBody = JsonUtils.parseObject(body, LoginBody.class); | ||||||
|  |         ValidatorUtils.validate(loginBody); | ||||||
|  |         // 授权类型和客户端id | ||||||
|  |         String clientId = loginBody.getClientId(); | ||||||
|  |         String grantType = loginBody.getGrantType(); | ||||||
|  |         SysClient client = clientService.queryByClientId(clientId); | ||||||
|  |         // 查询不到 client 或 client 内不包含 grantType | ||||||
|  |         if (ObjectUtil.isNull(client) || !StringUtils.contains(client.getGrantType(), grantType)) { | ||||||
|  |             log.info("客户端id: {} 认证类型:{} 异常!.", clientId, grantType); | ||||||
|  |             return R.fail(MessageUtils.message("auth.grant.type.error")); | ||||||
|  |         } else if (!UserConstants.NORMAL.equals(client.getStatus())) { | ||||||
|  |             return R.fail(MessageUtils.message("auth.grant.type.blocked")); | ||||||
|  |         } | ||||||
|  |         // 校验租户 | ||||||
|  |         loginService.checkTenant(loginBody.getTenantId()); | ||||||
|  |         // 登录 | ||||||
|  |         LoginVo loginVo = IAuthStrategy.login(body, client, grantType); | ||||||
|  |  | ||||||
|  |         Long userId = LoginHelper.getUserId(); | ||||||
|  |         scheduledExecutorService.schedule(() -> { | ||||||
|  |             WebSocketUtils.sendMessage(userId, "欢迎登录RuoYi-Vue-Plus后台管理系统"); | ||||||
|  |         }, 3, TimeUnit.SECONDS); | ||||||
|  |         return R.ok(loginVo); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 第三方登录请求 | ||||||
|  |      * | ||||||
|  |      * @param source 登录来源 | ||||||
|  |      * @return 结果 | ||||||
|  |      */ | ||||||
|  |     @GetMapping("/binding/{source}") | ||||||
|  |     public R<String> authBinding(@PathVariable("source") String source) { | ||||||
|  |         SocialLoginConfigProperties obj = socialProperties.getType().get(source); | ||||||
|  |         if (ObjectUtil.isNull(obj)) { | ||||||
|  |             return R.fail(source + "平台账号暂不支持"); | ||||||
|  |         } | ||||||
|  |         AuthRequest authRequest = SocialUtils.getAuthRequest(source, socialProperties); | ||||||
|  |         String authorizeUrl = authRequest.authorize(AuthStateUtils.createState()); | ||||||
|  |         return R.ok("操作成功", authorizeUrl); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 第三方登录回调业务处理 绑定授权 | ||||||
|  |      * | ||||||
|  |      * @param loginBody 请求体 | ||||||
|  |      * @return 结果 | ||||||
|  |      */ | ||||||
|  |     @PostMapping("/social/callback") | ||||||
|  |     public R<Void> socialCallback(@RequestBody SocialLoginBody loginBody) { | ||||||
|  |         // 获取第三方登录信息 | ||||||
|  |         AuthResponse<AuthUser> response = SocialUtils.loginAuth( | ||||||
|  |                 loginBody.getSource(), loginBody.getSocialCode(), | ||||||
|  |                 loginBody.getSocialState(), socialProperties); | ||||||
|  |         AuthUser authUserData = response.getData(); | ||||||
|  |         // 判断授权响应是否成功 | ||||||
|  |         if (!response.ok()) { | ||||||
|  |             return R.fail(response.getMsg()); | ||||||
|  |         } | ||||||
|  |         loginService.socialRegister(authUserData); | ||||||
|  |         return R.ok(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 取消授权 | ||||||
|  |      * | ||||||
|  |      * @param socialId socialId | ||||||
|  |      */ | ||||||
|  |     @DeleteMapping(value = "/unlock/{socialId}") | ||||||
|  |     public R<Void> unlockSocial(@PathVariable Long socialId) { | ||||||
|  |         Boolean rows = socialUserService.deleteWithValidById(socialId); | ||||||
|  |         return rows ? R.ok() : R.fail("取消授权失败"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 退出登录 | ||||||
|  |      */ | ||||||
|  |     @PostMapping("/logout") | ||||||
|  |     public R<Void> logout() { | ||||||
|  |         loginService.logout(); | ||||||
|  |         return R.ok("退出成功"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 用户注册 | ||||||
|  |      */ | ||||||
|  |     @ApiEncrypt | ||||||
|  |     @PostMapping("/register") | ||||||
|  |     public R<Void> register(@Validated @RequestBody RegisterBody user) { | ||||||
|  |         if (!configService.selectRegisterEnabled(user.getTenantId())) { | ||||||
|  |             return R.fail("当前系统没有开启注册功能!"); | ||||||
|  |         } | ||||||
|  |         registerService.register(user); | ||||||
|  |         return R.ok(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 登录页面租户下拉框 | ||||||
|  |      * | ||||||
|  |      * @return 租户列表 | ||||||
|  |      */ | ||||||
|  |     @GetMapping("/tenant/list") | ||||||
|  |     public R<LoginTenantVo> tenantList(HttpServletRequest request) throws Exception { | ||||||
|  |         List<SysTenantVo> tenantList = tenantService.queryList(new SysTenantBo()); | ||||||
|  |         List<TenantListVo> voList = MapstructUtils.convert(tenantList, TenantListVo.class); | ||||||
|  |         // 获取域名 | ||||||
|  |         String host; | ||||||
|  |         String referer = request.getHeader("referer"); | ||||||
|  |         if (StringUtils.isNotBlank(referer)) { | ||||||
|  |             // 这里从referer中取值是为了本地使用hosts添加虚拟域名,方便本地环境调试 | ||||||
|  |             host = referer.split("//")[1].split("/")[0]; | ||||||
|  |         } else { | ||||||
|  |             host = new URL(request.getRequestURL().toString()).getHost(); | ||||||
|  |         } | ||||||
|  |         // 根据域名进行筛选 | ||||||
|  |         List<TenantListVo> list = StreamUtils.filter(voList, vo -> | ||||||
|  |                 StringUtils.equals(vo.getDomain(), host)); | ||||||
|  |         // 返回对象 | ||||||
|  |         LoginTenantVo vo = new LoginTenantVo(); | ||||||
|  |         vo.setVoList(CollUtil.isNotEmpty(list) ? list : voList); | ||||||
|  |         vo.setTenantEnabled(TenantHelper.isEnable()); | ||||||
|  |         return R.ok(vo); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,136 @@ | |||||||
|  | package org.dromara.web.controller; | ||||||
|  |  | ||||||
|  | import cn.dev33.satoken.annotation.SaIgnore; | ||||||
|  | import cn.hutool.captcha.AbstractCaptcha; | ||||||
|  | import cn.hutool.captcha.generator.CodeGenerator; | ||||||
|  | import cn.hutool.core.util.IdUtil; | ||||||
|  | import cn.hutool.core.util.RandomUtil; | ||||||
|  | import org.dromara.common.core.constant.Constants; | ||||||
|  | import org.dromara.common.core.constant.GlobalConstants; | ||||||
|  | import org.dromara.common.core.domain.R; | ||||||
|  | import org.dromara.common.core.utils.SpringUtils; | ||||||
|  | import org.dromara.common.core.utils.StringUtils; | ||||||
|  | import org.dromara.common.core.utils.reflect.ReflectUtils; | ||||||
|  | import org.dromara.common.mail.config.properties.MailProperties; | ||||||
|  | import org.dromara.common.mail.utils.MailUtils; | ||||||
|  | import org.dromara.common.ratelimiter.annotation.RateLimiter; | ||||||
|  | import org.dromara.common.ratelimiter.enums.LimitType; | ||||||
|  | import org.dromara.common.redis.utils.RedisUtils; | ||||||
|  | import org.dromara.common.web.config.properties.CaptchaProperties; | ||||||
|  | import org.dromara.common.web.enums.CaptchaType; | ||||||
|  | import org.dromara.sms4j.api.SmsBlend; | ||||||
|  | import org.dromara.sms4j.api.entity.SmsResponse; | ||||||
|  | import org.dromara.sms4j.core.factory.SmsFactory; | ||||||
|  | import org.dromara.sms4j.provider.enumerate.SupplierType; | ||||||
|  | import org.dromara.web.domain.vo.CaptchaVo; | ||||||
|  | import jakarta.validation.constraints.NotBlank; | ||||||
|  | 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.RestController; | ||||||
|  |  | ||||||
|  | import java.time.Duration; | ||||||
|  | import java.util.LinkedHashMap; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 验证码操作处理 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | @SaIgnore | ||||||
|  | @Slf4j | ||||||
|  | @Validated | ||||||
|  | @RequiredArgsConstructor | ||||||
|  | @RestController | ||||||
|  | public class CaptchaController { | ||||||
|  |  | ||||||
|  |     private final CaptchaProperties captchaProperties; | ||||||
|  |     private final MailProperties mailProperties; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 短信验证码 | ||||||
|  |      * | ||||||
|  |      * @param phonenumber 用户手机号 | ||||||
|  |      */ | ||||||
|  |     @RateLimiter(key = "#phonenumber", time = 60, count = 1) | ||||||
|  |     @GetMapping("/resource/sms/code") | ||||||
|  |     public R<Void> smsCode(@NotBlank(message = "{user.phonenumber.not.blank}") String phonenumber) { | ||||||
|  |         String key = GlobalConstants.CAPTCHA_CODE_KEY + phonenumber; | ||||||
|  |         String code = RandomUtil.randomNumbers(4); | ||||||
|  |         RedisUtils.setCacheObject(key, code, Duration.ofMinutes(Constants.CAPTCHA_EXPIRATION)); | ||||||
|  |         // 验证码模板id 自行处理 (查数据库或写死均可) | ||||||
|  |         String templateId = ""; | ||||||
|  |         LinkedHashMap<String, String> map = new LinkedHashMap<>(1); | ||||||
|  |         map.put("code", code); | ||||||
|  |         SmsBlend smsBlend = SmsFactory.createSmsBlend(SupplierType.ALIBABA); | ||||||
|  |         SmsResponse smsResponse = smsBlend.sendMessage(phonenumber, templateId, map); | ||||||
|  |         if (!"OK".equals(smsResponse.getCode())) { | ||||||
|  |             log.error("验证码短信发送异常 => {}", smsResponse); | ||||||
|  |             return R.fail(smsResponse.getMessage()); | ||||||
|  |         } | ||||||
|  |         return R.ok(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 邮箱验证码 | ||||||
|  |      * | ||||||
|  |      * @param email 邮箱 | ||||||
|  |      */ | ||||||
|  |     @RateLimiter(key = "#email", time = 60, count = 1) | ||||||
|  |     @GetMapping("/resource/email/code") | ||||||
|  |     public R<Void> emailCode(@NotBlank(message = "{user.email.not.blank}") String email) { | ||||||
|  |         if (!mailProperties.getEnabled()) { | ||||||
|  |             return R.fail("当前系统没有开启邮箱功能!"); | ||||||
|  |         } | ||||||
|  |         String key = GlobalConstants.CAPTCHA_CODE_KEY + email; | ||||||
|  |         String code = RandomUtil.randomNumbers(4); | ||||||
|  |         RedisUtils.setCacheObject(key, code, Duration.ofMinutes(Constants.CAPTCHA_EXPIRATION)); | ||||||
|  |         try { | ||||||
|  |             MailUtils.sendText(email, "登录验证码", "您本次验证码为:" + code + ",有效性为" + Constants.CAPTCHA_EXPIRATION + "分钟,请尽快填写。"); | ||||||
|  |         } catch (Exception e) { | ||||||
|  |             log.error("验证码短信发送异常 => {}", e.getMessage()); | ||||||
|  |             return R.fail(e.getMessage()); | ||||||
|  |         } | ||||||
|  |         return R.ok(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 生成验证码 | ||||||
|  |      */ | ||||||
|  |     @RateLimiter(time = 60, count = 10, limitType = LimitType.IP) | ||||||
|  |     @GetMapping("/auth/code") | ||||||
|  |     public R<CaptchaVo> getCode() { | ||||||
|  |         CaptchaVo captchaVo = new CaptchaVo(); | ||||||
|  |         boolean captchaEnabled = captchaProperties.getEnable(); | ||||||
|  |         if (!captchaEnabled) { | ||||||
|  |             captchaVo.setCaptchaEnabled(false); | ||||||
|  |             return R.ok(captchaVo); | ||||||
|  |         } | ||||||
|  |         // 保存验证码信息 | ||||||
|  |         String uuid = IdUtil.simpleUUID(); | ||||||
|  |         String verifyKey = GlobalConstants.CAPTCHA_CODE_KEY + uuid; | ||||||
|  |         // 生成验证码 | ||||||
|  |         CaptchaType captchaType = captchaProperties.getType(); | ||||||
|  |         boolean isMath = CaptchaType.MATH == captchaType; | ||||||
|  |         Integer length = isMath ? captchaProperties.getNumberLength() : captchaProperties.getCharLength(); | ||||||
|  |         CodeGenerator codeGenerator = ReflectUtils.newInstance(captchaType.getClazz(), length); | ||||||
|  |         AbstractCaptcha captcha = SpringUtils.getBean(captchaProperties.getCategory().getClazz()); | ||||||
|  |         captcha.setGenerator(codeGenerator); | ||||||
|  |         captcha.createCode(); | ||||||
|  |         String code = captcha.getCode(); | ||||||
|  |         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)); | ||||||
|  |         captchaVo.setUuid(uuid); | ||||||
|  |         captchaVo.setImg(captcha.getImageBase64()); | ||||||
|  |         return R.ok(captchaVo); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -1,8 +1,8 @@ | |||||||
| package com.ruoyi.web.controller.system; | package org.dromara.web.controller; | ||||||
| 
 | 
 | ||||||
| import cn.dev33.satoken.annotation.SaIgnore; | import cn.dev33.satoken.annotation.SaIgnore; | ||||||
| import com.ruoyi.common.config.RuoYiConfig; | import org.dromara.common.core.config.RuoYiConfig; | ||||||
| import com.ruoyi.common.utils.StringUtils; | import org.dromara.common.core.utils.StringUtils; | ||||||
| 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; | ||||||
| @@ -12,9 +12,10 @@ import org.springframework.web.bind.annotation.RestController; | |||||||
|  * |  * | ||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
|  | @SaIgnore | ||||||
| @RequiredArgsConstructor | @RequiredArgsConstructor | ||||||
| @RestController | @RestController | ||||||
| public class SysIndexController { | public class IndexController { | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * 系统基础配置 |      * 系统基础配置 | ||||||
| @@ -24,7 +25,6 @@ public class SysIndexController { | |||||||
|     /** |     /** | ||||||
|      * 访问首页,提示语 |      * 访问首页,提示语 | ||||||
|      */ |      */ | ||||||
|     @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()); | ||||||
| @@ -0,0 +1,25 @@ | |||||||
|  | package org.dromara.web.domain.vo; | ||||||
|  |  | ||||||
|  | import lombok.Data; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 验证码信息 | ||||||
|  |  * | ||||||
|  |  * @author Michelle.Chung | ||||||
|  |  */ | ||||||
|  | @Data | ||||||
|  | public class CaptchaVo { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 是否开启验证码 | ||||||
|  |      */ | ||||||
|  |     private Boolean captchaEnabled = true; | ||||||
|  |  | ||||||
|  |     private String uuid; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 验证码图片 | ||||||
|  |      */ | ||||||
|  |     private String img; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,25 @@ | |||||||
|  | package org.dromara.web.domain.vo; | ||||||
|  |  | ||||||
|  | import lombok.Data; | ||||||
|  |  | ||||||
|  | import java.util.List; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 登录租户对象 | ||||||
|  |  * | ||||||
|  |  * @author Michelle.Chung | ||||||
|  |  */ | ||||||
|  | @Data | ||||||
|  | public class LoginTenantVo { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 租户开关 | ||||||
|  |      */ | ||||||
|  |     private Boolean tenantEnabled; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 租户对象列表 | ||||||
|  |      */ | ||||||
|  |     private List<TenantListVo> voList; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,54 @@ | |||||||
|  | package org.dromara.web.domain.vo; | ||||||
|  |  | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty; | ||||||
|  | import lombok.Data; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 登录验证信息 | ||||||
|  |  * | ||||||
|  |  * @author Michelle.Chung | ||||||
|  |  */ | ||||||
|  | @Data | ||||||
|  | public class LoginVo { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 授权令牌 | ||||||
|  |      */ | ||||||
|  |     @JsonProperty("access_token") | ||||||
|  |     private String accessToken; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 刷新令牌 | ||||||
|  |      */ | ||||||
|  |     @JsonProperty("refresh_token") | ||||||
|  |     private String refreshToken; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 授权令牌 access_token 的有效期 | ||||||
|  |      */ | ||||||
|  |     @JsonProperty("expire_in") | ||||||
|  |     private Long expireIn; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 刷新令牌 refresh_token 的有效期 | ||||||
|  |      */ | ||||||
|  |     @JsonProperty("refresh_expire_in") | ||||||
|  |     private Long refreshExpireIn; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 应用id | ||||||
|  |      */ | ||||||
|  |     @JsonProperty("client_id") | ||||||
|  |     private String clientId; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 令牌权限 | ||||||
|  |      */ | ||||||
|  |     private String scope; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 用户 openid | ||||||
|  |      */ | ||||||
|  |     private String openid; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,22 @@ | |||||||
|  | package org.dromara.web.domain.vo; | ||||||
|  |  | ||||||
|  | import org.dromara.system.domain.vo.SysTenantVo; | ||||||
|  | import io.github.linpeilie.annotations.AutoMapper; | ||||||
|  | import lombok.Data; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 租户列表 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | @Data | ||||||
|  | @AutoMapper(target = SysTenantVo.class) | ||||||
|  | public class TenantListVo { | ||||||
|  |  | ||||||
|  |     private String tenantId; | ||||||
|  |  | ||||||
|  |     private String companyName; | ||||||
|  |  | ||||||
|  |     private String domain; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -1,20 +1,24 @@ | |||||||
| package com.ruoyi.framework.listener; | package org.dromara.web.listener; | ||||||
| 
 | 
 | ||||||
| import cn.dev33.satoken.config.SaTokenConfig; | import cn.dev33.satoken.config.SaTokenConfig; | ||||||
| import cn.dev33.satoken.listener.SaTokenListener; | import cn.dev33.satoken.listener.SaTokenListener; | ||||||
| import cn.dev33.satoken.stp.SaLoginModel; | import cn.dev33.satoken.stp.SaLoginModel; | ||||||
| import cn.hutool.http.useragent.UserAgent; | import cn.hutool.http.useragent.UserAgent; | ||||||
| import cn.hutool.http.useragent.UserAgentUtil; | import cn.hutool.http.useragent.UserAgentUtil; | ||||||
| import com.ruoyi.common.constant.CacheConstants; |  | ||||||
| import com.ruoyi.common.core.domain.dto.UserOnlineDTO; |  | ||||||
| import com.ruoyi.common.core.domain.model.LoginUser; |  | ||||||
| import com.ruoyi.common.enums.UserType; |  | ||||||
| import com.ruoyi.common.helper.LoginHelper; |  | ||||||
| import com.ruoyi.common.utils.ServletUtils; |  | ||||||
| import com.ruoyi.common.utils.ip.AddressUtils; |  | ||||||
| import com.ruoyi.common.utils.redis.RedisUtils; |  | ||||||
| import lombok.RequiredArgsConstructor; | import lombok.RequiredArgsConstructor; | ||||||
| import lombok.extern.slf4j.Slf4j; | import lombok.extern.slf4j.Slf4j; | ||||||
|  | import org.dromara.common.core.constant.CacheConstants; | ||||||
|  | import org.dromara.common.core.constant.Constants; | ||||||
|  | import org.dromara.common.core.domain.dto.UserOnlineDTO; | ||||||
|  | import org.dromara.common.core.domain.model.LoginUser; | ||||||
|  | import org.dromara.common.core.utils.MessageUtils; | ||||||
|  | import org.dromara.common.core.utils.ServletUtils; | ||||||
|  | import org.dromara.common.core.utils.SpringUtils; | ||||||
|  | import org.dromara.common.core.utils.ip.AddressUtils; | ||||||
|  | import org.dromara.common.log.event.LogininforEvent; | ||||||
|  | import org.dromara.common.redis.utils.RedisUtils; | ||||||
|  | import org.dromara.common.satoken.utils.LoginHelper; | ||||||
|  | import org.dromara.web.service.SysLoginService; | ||||||
| import org.springframework.stereotype.Component; | import org.springframework.stereotype.Component; | ||||||
| 
 | 
 | ||||||
| import java.time.Duration; | import java.time.Duration; | ||||||
| @@ -30,35 +34,43 @@ import java.time.Duration; | |||||||
| public class UserActionListener implements SaTokenListener { | public class UserActionListener implements SaTokenListener { | ||||||
| 
 | 
 | ||||||
|     private final SaTokenConfig tokenConfig; |     private final SaTokenConfig tokenConfig; | ||||||
|  |     private final SysLoginService loginService; | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * 每次登录时触发 |      * 每次登录时触发 | ||||||
|      */ |      */ | ||||||
|     @Override |     @Override | ||||||
|     public void doLogin(String loginType, Object loginId, String tokenValue, SaLoginModel loginModel) { |     public void doLogin(String loginType, Object loginId, String tokenValue, SaLoginModel loginModel) { | ||||||
|         UserType userType = UserType.getUserType(loginId.toString()); |         UserAgent userAgent = UserAgentUtil.parse(ServletUtils.getRequest().getHeader("User-Agent")); | ||||||
|         if (userType == UserType.SYS_USER) { |         String ip = ServletUtils.getClientIP(); | ||||||
|             UserAgent userAgent = UserAgentUtil.parse(ServletUtils.getRequest().getHeader("User-Agent")); |         LoginUser user = LoginHelper.getLoginUser(); | ||||||
|             String ip = ServletUtils.getClientIP(); |         UserOnlineDTO dto = new UserOnlineDTO(); | ||||||
|             LoginUser user = LoginHelper.getLoginUser(); |         dto.setIpaddr(ip); | ||||||
|             UserOnlineDTO dto = new UserOnlineDTO(); |         dto.setLoginLocation(AddressUtils.getRealAddressByIP(ip)); | ||||||
|             dto.setIpaddr(ip); |         dto.setBrowser(userAgent.getBrowser().getName()); | ||||||
|             dto.setLoginLocation(AddressUtils.getRealAddressByIP(ip)); |         dto.setOs(userAgent.getOs().getName()); | ||||||
|             dto.setBrowser(userAgent.getBrowser().getName()); |         dto.setLoginTime(System.currentTimeMillis()); | ||||||
|             dto.setOs(userAgent.getOs().getName()); |         dto.setTokenId(tokenValue); | ||||||
|             dto.setLoginTime(System.currentTimeMillis()); |         dto.setUserName(user.getUsername()); | ||||||
|             dto.setTokenId(tokenValue); |         dto.setClientKey(user.getClientKey()); | ||||||
|             dto.setUserName(user.getUsername()); |         dto.setDeviceType(user.getDeviceType()); | ||||||
|             dto.setDeptName(user.getDeptName()); |         dto.setDeptName(user.getDeptName()); | ||||||
|             if(tokenConfig.getTimeout() == -1) { |         if(tokenConfig.getTimeout() == -1) { | ||||||
|                 RedisUtils.setCacheObject(CacheConstants.ONLINE_TOKEN_KEY + tokenValue, dto); |             RedisUtils.setCacheObject(CacheConstants.ONLINE_TOKEN_KEY + tokenValue, dto); | ||||||
|             } else { |         } else { | ||||||
|                 RedisUtils.setCacheObject(CacheConstants.ONLINE_TOKEN_KEY + tokenValue, dto, Duration.ofSeconds(tokenConfig.getTimeout())); |             RedisUtils.setCacheObject(CacheConstants.ONLINE_TOKEN_KEY + tokenValue, dto, Duration.ofSeconds(tokenConfig.getTimeout())); | ||||||
|             } |  | ||||||
|             log.info("user doLogin, userId:{}, token:{}", loginId, tokenValue); |  | ||||||
|         } else if (userType == UserType.APP_USER) { |  | ||||||
|             // app端 自行根据业务编写 |  | ||||||
|         } |         } | ||||||
|  |         // 记录登录日志 | ||||||
|  |         LogininforEvent logininforEvent = new LogininforEvent(); | ||||||
|  |         logininforEvent.setTenantId(user.getTenantId()); | ||||||
|  |         logininforEvent.setUsername(user.getUsername()); | ||||||
|  |         logininforEvent.setStatus(Constants.LOGIN_SUCCESS); | ||||||
|  |         logininforEvent.setMessage(MessageUtils.message("user.login.success")); | ||||||
|  |         logininforEvent.setRequest(ServletUtils.getRequest()); | ||||||
|  |         SpringUtils.context().publishEvent(logininforEvent); | ||||||
|  |         // 更新登录信息 | ||||||
|  |         loginService.recordLoginInfo(user.getUserId(), ip); | ||||||
|  |         log.info("user doLogin, userId:{}, token:{}", loginId, tokenValue); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
| @@ -76,7 +88,7 @@ public class UserActionListener implements SaTokenListener { | |||||||
|     @Override |     @Override | ||||||
|     public void doKickout(String loginType, Object loginId, String tokenValue) { |     public void doKickout(String loginType, Object loginId, String tokenValue) { | ||||||
|         RedisUtils.deleteObject(CacheConstants.ONLINE_TOKEN_KEY + tokenValue); |         RedisUtils.deleteObject(CacheConstants.ONLINE_TOKEN_KEY + tokenValue); | ||||||
|         log.info("user doLogoutByLoginId, userId:{}, token:{}", loginId, tokenValue); |         log.info("user doKickout, userId:{}, token:{}", loginId, tokenValue); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
| @@ -0,0 +1,36 @@ | |||||||
|  | package org.dromara.web.service; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | import org.dromara.common.core.exception.ServiceException; | ||||||
|  | import org.dromara.common.core.utils.SpringUtils; | ||||||
|  | import org.dromara.system.domain.SysClient; | ||||||
|  | import org.dromara.web.domain.vo.LoginVo; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 授权策略 | ||||||
|  |  * | ||||||
|  |  * @author Michelle.Chung | ||||||
|  |  */ | ||||||
|  | public interface IAuthStrategy { | ||||||
|  |  | ||||||
|  |     String BASE_NAME = "AuthStrategy"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 登录 | ||||||
|  |      */ | ||||||
|  |     static LoginVo login(String body, SysClient client, String grantType) { | ||||||
|  |         // 授权类型和客户端id | ||||||
|  |         String beanName = grantType + BASE_NAME; | ||||||
|  |         if (!SpringUtils.containsBean(beanName)) { | ||||||
|  |             throw new ServiceException("授权类型不正确!"); | ||||||
|  |         } | ||||||
|  |         IAuthStrategy instance = SpringUtils.getBean(beanName); | ||||||
|  |         return instance.login(body, client); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 登录 | ||||||
|  |      */ | ||||||
|  |     LoginVo login(String body, SysClient client); | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,232 @@ | |||||||
|  | package org.dromara.web.service; | ||||||
|  |  | ||||||
|  | import cn.dev33.satoken.exception.NotLoginException; | ||||||
|  | import cn.dev33.satoken.stp.StpUtil; | ||||||
|  | import cn.hutool.core.bean.BeanUtil; | ||||||
|  | import cn.hutool.core.collection.CollUtil; | ||||||
|  | import cn.hutool.core.util.ObjectUtil; | ||||||
|  | import lombok.RequiredArgsConstructor; | ||||||
|  | import lombok.extern.slf4j.Slf4j; | ||||||
|  | import me.zhyd.oauth.model.AuthUser; | ||||||
|  | import org.dromara.common.core.constant.Constants; | ||||||
|  | import org.dromara.common.core.constant.GlobalConstants; | ||||||
|  | import org.dromara.common.core.constant.TenantConstants; | ||||||
|  | import org.dromara.common.core.domain.dto.RoleDTO; | ||||||
|  | import org.dromara.common.core.domain.model.LoginUser; | ||||||
|  | import org.dromara.common.core.enums.LoginType; | ||||||
|  | import org.dromara.common.core.enums.TenantStatus; | ||||||
|  | import org.dromara.common.core.exception.user.UserException; | ||||||
|  | import org.dromara.common.core.utils.*; | ||||||
|  | import org.dromara.common.log.event.LogininforEvent; | ||||||
|  | import org.dromara.common.mybatis.helper.DataPermissionHelper; | ||||||
|  | import org.dromara.common.redis.utils.RedisUtils; | ||||||
|  | import org.dromara.common.satoken.utils.LoginHelper; | ||||||
|  | import org.dromara.common.tenant.exception.TenantException; | ||||||
|  | import org.dromara.common.tenant.helper.TenantHelper; | ||||||
|  | import org.dromara.system.domain.SysUser; | ||||||
|  | import org.dromara.system.domain.bo.SysSocialBo; | ||||||
|  | import org.dromara.system.domain.vo.SysSocialVo; | ||||||
|  | import org.dromara.system.domain.vo.SysTenantVo; | ||||||
|  | import org.dromara.system.domain.vo.SysUserVo; | ||||||
|  | import org.dromara.system.mapper.SysUserMapper; | ||||||
|  | import org.dromara.system.service.ISysPermissionService; | ||||||
|  | import org.dromara.system.service.ISysSocialService; | ||||||
|  | import org.dromara.system.service.ISysTenantService; | ||||||
|  | import org.springframework.beans.factory.annotation.Value; | ||||||
|  | import org.springframework.stereotype.Service; | ||||||
|  |  | ||||||
|  | import java.time.Duration; | ||||||
|  | import java.util.Date; | ||||||
|  | import java.util.List; | ||||||
|  | import java.util.function.Supplier; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 登录校验方法 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | @RequiredArgsConstructor | ||||||
|  | @Slf4j | ||||||
|  | @Service | ||||||
|  | public class SysLoginService { | ||||||
|  |  | ||||||
|  |     @Value("${user.password.maxRetryCount}") | ||||||
|  |     private Integer maxRetryCount; | ||||||
|  |  | ||||||
|  |     @Value("${user.password.lockTime}") | ||||||
|  |     private Integer lockTime; | ||||||
|  |  | ||||||
|  |     private final ISysTenantService tenantService; | ||||||
|  |     private final ISysPermissionService permissionService; | ||||||
|  |     private final ISysSocialService sysSocialService; | ||||||
|  |     private final SysUserMapper userMapper; | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 绑定第三方用户 | ||||||
|  |      * | ||||||
|  |      * @param authUserData 授权响应实体 | ||||||
|  |      * @return 统一响应实体 | ||||||
|  |      */ | ||||||
|  |     public void socialRegister(AuthUser authUserData) { | ||||||
|  |         String authId = authUserData.getSource() + authUserData.getUuid(); | ||||||
|  |         // 第三方用户信息 | ||||||
|  |         SysSocialBo bo = BeanUtil.toBean(authUserData, SysSocialBo.class); | ||||||
|  |         BeanUtil.copyProperties(authUserData.getToken(), bo); | ||||||
|  |         bo.setUserId(LoginHelper.getUserId()); | ||||||
|  |         bo.setAuthId(authId); | ||||||
|  |         bo.setOpenId(authUserData.getUuid()); | ||||||
|  |         bo.setUserName(authUserData.getUsername()); | ||||||
|  |         bo.setNickName(authUserData.getNickname()); | ||||||
|  |         // 查询是否已经绑定用户 | ||||||
|  |         List<SysSocialVo> list = sysSocialService.selectByAuthId(authId); | ||||||
|  |         if (CollUtil.isEmpty(list)) { | ||||||
|  |             // 没有绑定用户, 新增用户信息 | ||||||
|  |             sysSocialService.insertByBo(bo); | ||||||
|  |         } else { | ||||||
|  |             // 更新用户信息 | ||||||
|  |             bo.setId(list.get(0).getId()); | ||||||
|  |             sysSocialService.updateByBo(bo); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 退出登录 | ||||||
|  |      */ | ||||||
|  |     public void logout() { | ||||||
|  |         try { | ||||||
|  |             LoginUser loginUser = LoginHelper.getLoginUser(); | ||||||
|  |             if (ObjectUtil.isNull(loginUser)) { | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |             if (TenantHelper.isEnable() && LoginHelper.isSuperAdmin()) { | ||||||
|  |                 // 超级管理员 登出清除动态租户 | ||||||
|  |                 TenantHelper.clearDynamic(); | ||||||
|  |             } | ||||||
|  |             recordLogininfor(loginUser.getTenantId(), loginUser.getUsername(), Constants.LOGOUT, MessageUtils.message("user.logout.success")); | ||||||
|  |         } catch (NotLoginException ignored) { | ||||||
|  |         } finally { | ||||||
|  |             try { | ||||||
|  |                 StpUtil.logout(); | ||||||
|  |             } catch (NotLoginException ignored) { | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 记录登录信息 | ||||||
|  |      * | ||||||
|  |      * @param tenantId 租户ID | ||||||
|  |      * @param username 用户名 | ||||||
|  |      * @param status   状态 | ||||||
|  |      * @param message  消息内容 | ||||||
|  |      */ | ||||||
|  |     public void recordLogininfor(String tenantId, String username, String status, String message) { | ||||||
|  |         LogininforEvent logininforEvent = new LogininforEvent(); | ||||||
|  |         logininforEvent.setTenantId(tenantId); | ||||||
|  |         logininforEvent.setUsername(username); | ||||||
|  |         logininforEvent.setStatus(status); | ||||||
|  |         logininforEvent.setMessage(message); | ||||||
|  |         logininforEvent.setRequest(ServletUtils.getRequest()); | ||||||
|  |         SpringUtils.context().publishEvent(logininforEvent); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 构建登录用户 | ||||||
|  |      */ | ||||||
|  |     public LoginUser buildLoginUser(SysUserVo user) { | ||||||
|  |         LoginUser loginUser = new LoginUser(); | ||||||
|  |         loginUser.setTenantId(user.getTenantId()); | ||||||
|  |         loginUser.setUserId(user.getUserId()); | ||||||
|  |         loginUser.setDeptId(user.getDeptId()); | ||||||
|  |         loginUser.setUsername(user.getUserName()); | ||||||
|  |         loginUser.setNickname(user.getNickName()); | ||||||
|  |         loginUser.setUserType(user.getUserType()); | ||||||
|  |         loginUser.setMenuPermission(permissionService.getMenuPermission(user.getUserId())); | ||||||
|  |         loginUser.setRolePermission(permissionService.getRolePermission(user.getUserId())); | ||||||
|  |         loginUser.setDeptName(ObjectUtil.isNull(user.getDept()) ? "" : user.getDept().getDeptName()); | ||||||
|  |         List<RoleDTO> roles = BeanUtil.copyToList(user.getRoles(), RoleDTO.class); | ||||||
|  |         loginUser.setRoles(roles); | ||||||
|  |         return loginUser; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 记录登录信息 | ||||||
|  |      * | ||||||
|  |      * @param userId 用户ID | ||||||
|  |      */ | ||||||
|  |     public void recordLoginInfo(Long userId, String ip) { | ||||||
|  |         SysUser sysUser = new SysUser(); | ||||||
|  |         sysUser.setUserId(userId); | ||||||
|  |         sysUser.setLoginIp(ip); | ||||||
|  |         sysUser.setLoginDate(DateUtils.getNowDate()); | ||||||
|  |         sysUser.setUpdateBy(userId); | ||||||
|  |         DataPermissionHelper.ignore(() -> userMapper.updateById(sysUser)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 登录校验 | ||||||
|  |      */ | ||||||
|  |     public void checkLogin(LoginType loginType, String tenantId, String username, Supplier<Boolean> supplier) { | ||||||
|  |         String errorKey = GlobalConstants.PWD_ERR_CNT_KEY + username; | ||||||
|  |         String loginFail = Constants.LOGIN_FAIL; | ||||||
|  |  | ||||||
|  |         // 获取用户登录错误次数,默认为0 (可自定义限制策略 例如: key + username + ip) | ||||||
|  |         int errorNumber = ObjectUtil.defaultIfNull(RedisUtils.getCacheObject(errorKey), 0); | ||||||
|  |         // 锁定时间内登录 则踢出 | ||||||
|  |         if (errorNumber >= maxRetryCount) { | ||||||
|  |             recordLogininfor(tenantId, username, loginFail, MessageUtils.message(loginType.getRetryLimitExceed(), maxRetryCount, lockTime)); | ||||||
|  |             throw new UserException(loginType.getRetryLimitExceed(), maxRetryCount, lockTime); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (supplier.get()) { | ||||||
|  |             // 错误次数递增 | ||||||
|  |             errorNumber++; | ||||||
|  |             RedisUtils.setCacheObject(errorKey, errorNumber, Duration.ofMinutes(lockTime)); | ||||||
|  |             // 达到规定错误次数 则锁定登录 | ||||||
|  |             if (errorNumber >= maxRetryCount) { | ||||||
|  |                 recordLogininfor(tenantId, username, loginFail, MessageUtils.message(loginType.getRetryLimitExceed(), maxRetryCount, lockTime)); | ||||||
|  |                 throw new UserException(loginType.getRetryLimitExceed(), maxRetryCount, lockTime); | ||||||
|  |             } else { | ||||||
|  |                 // 未达到规定错误次数 | ||||||
|  |                 recordLogininfor(tenantId, username, loginFail, MessageUtils.message(loginType.getRetryLimitCount(), errorNumber)); | ||||||
|  |                 throw new UserException(loginType.getRetryLimitCount(), errorNumber); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // 登录成功 清空错误次数 | ||||||
|  |         RedisUtils.deleteObject(errorKey); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 校验租户 | ||||||
|  |      * | ||||||
|  |      * @param tenantId 租户ID | ||||||
|  |      */ | ||||||
|  |     public void checkTenant(String tenantId) { | ||||||
|  |         if (!TenantHelper.isEnable()) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         if (TenantConstants.DEFAULT_TENANT_ID.equals(tenantId)) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         if (StringUtils.isBlank(tenantId)) { | ||||||
|  |             throw new TenantException("tenant.number.not.blank"); | ||||||
|  |         } | ||||||
|  |         SysTenantVo tenant = tenantService.queryByTenantId(tenantId); | ||||||
|  |         if (ObjectUtil.isNull(tenant)) { | ||||||
|  |             log.info("登录租户:{} 不存在.", tenantId); | ||||||
|  |             throw new TenantException("tenant.not.exists"); | ||||||
|  |         } else if (TenantStatus.DISABLE.getCode().equals(tenant.getStatus())) { | ||||||
|  |             log.info("登录租户:{} 已被停用.", tenantId); | ||||||
|  |             throw new TenantException("tenant.blocked"); | ||||||
|  |         } else if (ObjectUtil.isNotNull(tenant.getExpireTime()) | ||||||
|  |             && new Date().after(tenant.getExpireTime())) { | ||||||
|  |             log.info("登录租户:{} 已超过有效期.", tenantId); | ||||||
|  |             throw new TenantException("tenant.expired"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,117 @@ | |||||||
|  | package org.dromara.web.service; | ||||||
|  |  | ||||||
|  | import cn.dev33.satoken.secure.BCrypt; | ||||||
|  | import cn.hutool.core.util.ObjectUtil; | ||||||
|  | import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; | ||||||
|  | import lombok.RequiredArgsConstructor; | ||||||
|  | import org.dromara.common.core.constant.Constants; | ||||||
|  | import org.dromara.common.core.constant.GlobalConstants; | ||||||
|  | import org.dromara.common.core.domain.model.RegisterBody; | ||||||
|  | import org.dromara.common.core.enums.UserType; | ||||||
|  | import org.dromara.common.core.exception.user.CaptchaException; | ||||||
|  | import org.dromara.common.core.exception.user.CaptchaExpireException; | ||||||
|  | import org.dromara.common.core.exception.user.UserException; | ||||||
|  | import org.dromara.common.core.utils.MessageUtils; | ||||||
|  | import org.dromara.common.core.utils.ServletUtils; | ||||||
|  | import org.dromara.common.core.utils.SpringUtils; | ||||||
|  | import org.dromara.common.core.utils.StringUtils; | ||||||
|  | import org.dromara.common.log.event.LogininforEvent; | ||||||
|  | import org.dromara.common.redis.utils.RedisUtils; | ||||||
|  | import org.dromara.common.tenant.helper.TenantHelper; | ||||||
|  | import org.dromara.common.web.config.properties.CaptchaProperties; | ||||||
|  | import org.dromara.system.domain.SysUser; | ||||||
|  | import org.dromara.system.domain.bo.SysUserBo; | ||||||
|  | import org.dromara.system.mapper.SysUserMapper; | ||||||
|  | import org.dromara.system.service.ISysUserService; | ||||||
|  | import org.springframework.stereotype.Service; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 注册校验方法 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | @RequiredArgsConstructor | ||||||
|  | @Service | ||||||
|  | public class SysRegisterService { | ||||||
|  |  | ||||||
|  |     private final ISysUserService userService; | ||||||
|  |     private final SysUserMapper userMapper; | ||||||
|  |     private final CaptchaProperties captchaProperties; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 注册 | ||||||
|  |      */ | ||||||
|  |     public void register(RegisterBody registerBody) { | ||||||
|  |         String tenantId = registerBody.getTenantId(); | ||||||
|  |         String username = registerBody.getUsername(); | ||||||
|  |         String password = registerBody.getPassword(); | ||||||
|  |         // 校验用户类型是否存在 | ||||||
|  |         String userType = UserType.getUserType(registerBody.getUserType()).getUserType(); | ||||||
|  |  | ||||||
|  |         boolean captchaEnabled = captchaProperties.getEnable(); | ||||||
|  |         // 验证码开关 | ||||||
|  |         if (captchaEnabled) { | ||||||
|  |             validateCaptcha(tenantId, username, registerBody.getCode(), registerBody.getUuid()); | ||||||
|  |         } | ||||||
|  |         SysUserBo sysUser = new SysUserBo(); | ||||||
|  |         sysUser.setUserName(username); | ||||||
|  |         sysUser.setNickName(username); | ||||||
|  |         sysUser.setPassword(BCrypt.hashpw(password)); | ||||||
|  |         sysUser.setUserType(userType); | ||||||
|  |  | ||||||
|  |         boolean exist = TenantHelper.dynamic(tenantId, () -> { | ||||||
|  |             return userMapper.exists(new LambdaQueryWrapper<SysUser>() | ||||||
|  |                 .eq(SysUser::getUserName, sysUser.getUserName()) | ||||||
|  |                 .ne(ObjectUtil.isNotNull(sysUser.getUserId()), SysUser::getUserId, sysUser.getUserId())); | ||||||
|  |         }); | ||||||
|  |         if (exist) { | ||||||
|  |             throw new UserException("user.register.save.error", username); | ||||||
|  |         } | ||||||
|  |         boolean regFlag = userService.registerUser(sysUser, tenantId); | ||||||
|  |         if (!regFlag) { | ||||||
|  |             throw new UserException("user.register.error"); | ||||||
|  |         } | ||||||
|  |         recordLogininfor(tenantId, username, Constants.REGISTER, MessageUtils.message("user.register.success")); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 校验验证码 | ||||||
|  |      * | ||||||
|  |      * @param username 用户名 | ||||||
|  |      * @param code     验证码 | ||||||
|  |      * @param uuid     唯一标识 | ||||||
|  |      */ | ||||||
|  |     public void validateCaptcha(String tenantId, String username, String code, String uuid) { | ||||||
|  |         String verifyKey = GlobalConstants.CAPTCHA_CODE_KEY + StringUtils.defaultString(uuid, ""); | ||||||
|  |         String captcha = RedisUtils.getCacheObject(verifyKey); | ||||||
|  |         RedisUtils.deleteObject(verifyKey); | ||||||
|  |         if (captcha == null) { | ||||||
|  |             recordLogininfor(tenantId, username, Constants.REGISTER, MessageUtils.message("user.jcaptcha.expire")); | ||||||
|  |             throw new CaptchaExpireException(); | ||||||
|  |         } | ||||||
|  |         if (!code.equalsIgnoreCase(captcha)) { | ||||||
|  |             recordLogininfor(tenantId, username, Constants.REGISTER, MessageUtils.message("user.jcaptcha.error")); | ||||||
|  |             throw new CaptchaException(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 记录登录信息 | ||||||
|  |      * | ||||||
|  |      * @param tenantId 租户ID | ||||||
|  |      * @param username 用户名 | ||||||
|  |      * @param status   状态 | ||||||
|  |      * @param message  消息内容 | ||||||
|  |      * @return | ||||||
|  |      */ | ||||||
|  |     private void recordLogininfor(String tenantId, String username, String status, String message) { | ||||||
|  |         LogininforEvent logininforEvent = new LogininforEvent(); | ||||||
|  |         logininforEvent.setTenantId(tenantId); | ||||||
|  |         logininforEvent.setUsername(username); | ||||||
|  |         logininforEvent.setStatus(status); | ||||||
|  |         logininforEvent.setMessage(message); | ||||||
|  |         logininforEvent.setRequest(ServletUtils.getRequest()); | ||||||
|  |         SpringUtils.context().publishEvent(logininforEvent); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,107 @@ | |||||||
|  | package org.dromara.web.service.impl; | ||||||
|  |  | ||||||
|  | import cn.dev33.satoken.stp.SaLoginModel; | ||||||
|  | import cn.dev33.satoken.stp.StpUtil; | ||||||
|  | import cn.hutool.core.util.ObjectUtil; | ||||||
|  | import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; | ||||||
|  | import lombok.RequiredArgsConstructor; | ||||||
|  | import lombok.extern.slf4j.Slf4j; | ||||||
|  | import org.dromara.common.core.constant.Constants; | ||||||
|  | import org.dromara.common.core.constant.GlobalConstants; | ||||||
|  | import org.dromara.common.core.domain.model.EmailLoginBody; | ||||||
|  | import org.dromara.common.core.domain.model.LoginUser; | ||||||
|  | import org.dromara.common.core.enums.LoginType; | ||||||
|  | import org.dromara.common.core.enums.UserStatus; | ||||||
|  | import org.dromara.common.core.exception.user.CaptchaExpireException; | ||||||
|  | import org.dromara.common.core.exception.user.UserException; | ||||||
|  | import org.dromara.common.core.utils.MessageUtils; | ||||||
|  | import org.dromara.common.core.utils.StringUtils; | ||||||
|  | import org.dromara.common.core.utils.ValidatorUtils; | ||||||
|  | import org.dromara.common.json.utils.JsonUtils; | ||||||
|  | import org.dromara.common.redis.utils.RedisUtils; | ||||||
|  | import org.dromara.common.satoken.utils.LoginHelper; | ||||||
|  | import org.dromara.common.tenant.helper.TenantHelper; | ||||||
|  | import org.dromara.system.domain.SysClient; | ||||||
|  | import org.dromara.system.domain.SysUser; | ||||||
|  | import org.dromara.system.domain.vo.SysUserVo; | ||||||
|  | import org.dromara.system.mapper.SysUserMapper; | ||||||
|  | import org.dromara.web.domain.vo.LoginVo; | ||||||
|  | import org.dromara.web.service.IAuthStrategy; | ||||||
|  | import org.dromara.web.service.SysLoginService; | ||||||
|  | import org.springframework.stereotype.Service; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 邮件认证策略 | ||||||
|  |  * | ||||||
|  |  * @author Michelle.Chung | ||||||
|  |  */ | ||||||
|  | @Slf4j | ||||||
|  | @Service("email" + IAuthStrategy.BASE_NAME) | ||||||
|  | @RequiredArgsConstructor | ||||||
|  | public class EmailAuthStrategy implements IAuthStrategy { | ||||||
|  |  | ||||||
|  |     private final SysLoginService loginService; | ||||||
|  |     private final SysUserMapper userMapper; | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public LoginVo login(String body, SysClient client) { | ||||||
|  |         EmailLoginBody loginBody = JsonUtils.parseObject(body, EmailLoginBody.class); | ||||||
|  |         ValidatorUtils.validate(loginBody); | ||||||
|  |         String tenantId = loginBody.getTenantId(); | ||||||
|  |         String email = loginBody.getEmail(); | ||||||
|  |         String emailCode = loginBody.getEmailCode(); | ||||||
|  |  | ||||||
|  |         // 通过邮箱查找用户 | ||||||
|  |         SysUserVo user = loadUserByEmail(tenantId, email); | ||||||
|  |  | ||||||
|  |         loginService.checkLogin(LoginType.EMAIL, tenantId, user.getUserName(), () -> !validateEmailCode(tenantId, email, emailCode)); | ||||||
|  |         // 此处可根据登录用户的数据不同 自行创建 loginUser 属性不够用继承扩展就行了 | ||||||
|  |         LoginUser loginUser = loginService.buildLoginUser(user); | ||||||
|  |         loginUser.setClientKey(client.getClientKey()); | ||||||
|  |         loginUser.setDeviceType(client.getDeviceType()); | ||||||
|  |         SaLoginModel model = new SaLoginModel(); | ||||||
|  |         model.setDevice(client.getDeviceType()); | ||||||
|  |         // 自定义分配 不同用户体系 不同 token 授权时间 不设置默认走全局 yml 配置 | ||||||
|  |         // 例如: 后台用户30分钟过期 app用户1天过期 | ||||||
|  |         model.setTimeout(client.getTimeout()); | ||||||
|  |         model.setActiveTimeout(client.getActiveTimeout()); | ||||||
|  |         model.setExtra(LoginHelper.CLIENT_KEY, client.getClientId()); | ||||||
|  |         // 生成token | ||||||
|  |         LoginHelper.login(loginUser, model); | ||||||
|  |  | ||||||
|  |         LoginVo loginVo = new LoginVo(); | ||||||
|  |         loginVo.setAccessToken(StpUtil.getTokenValue()); | ||||||
|  |         loginVo.setExpireIn(StpUtil.getTokenTimeout()); | ||||||
|  |         loginVo.setClientId(client.getClientId()); | ||||||
|  |         return loginVo; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 校验邮箱验证码 | ||||||
|  |      */ | ||||||
|  |     private boolean validateEmailCode(String tenantId, String email, String emailCode) { | ||||||
|  |         String code = RedisUtils.getCacheObject(GlobalConstants.CAPTCHA_CODE_KEY + email); | ||||||
|  |         if (StringUtils.isBlank(code)) { | ||||||
|  |             loginService.recordLogininfor(tenantId, email, Constants.LOGIN_FAIL, MessageUtils.message("user.jcaptcha.expire")); | ||||||
|  |             throw new CaptchaExpireException(); | ||||||
|  |         } | ||||||
|  |         return code.equals(emailCode); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private SysUserVo loadUserByEmail(String tenantId, String email) { | ||||||
|  |         return TenantHelper.dynamic(tenantId, () -> { | ||||||
|  |             SysUser user = userMapper.selectOne(new LambdaQueryWrapper<SysUser>() | ||||||
|  |                 .select(SysUser::getEmail, SysUser::getStatus) | ||||||
|  |                 .eq(SysUser::getEmail, email)); | ||||||
|  |             if (ObjectUtil.isNull(user)) { | ||||||
|  |                 log.info("登录用户:{} 不存在.", email); | ||||||
|  |                 throw new UserException("user.not.exists", email); | ||||||
|  |             } else if (UserStatus.DISABLE.getCode().equals(user.getStatus())) { | ||||||
|  |                 log.info("登录用户:{} 已被停用.", email); | ||||||
|  |                 throw new UserException("user.blocked", email); | ||||||
|  |             } | ||||||
|  |             return userMapper.selectUserByEmail(email); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,126 @@ | |||||||
|  | package org.dromara.web.service.impl; | ||||||
|  |  | ||||||
|  | import cn.dev33.satoken.secure.BCrypt; | ||||||
|  | import cn.dev33.satoken.stp.SaLoginModel; | ||||||
|  | import cn.dev33.satoken.stp.StpUtil; | ||||||
|  | import cn.hutool.core.util.ObjectUtil; | ||||||
|  | import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; | ||||||
|  | import lombok.RequiredArgsConstructor; | ||||||
|  | import lombok.extern.slf4j.Slf4j; | ||||||
|  | import org.dromara.common.core.constant.Constants; | ||||||
|  | import org.dromara.common.core.constant.GlobalConstants; | ||||||
|  | import org.dromara.common.core.domain.model.LoginUser; | ||||||
|  | import org.dromara.common.core.domain.model.PasswordLoginBody; | ||||||
|  | import org.dromara.common.core.enums.LoginType; | ||||||
|  | import org.dromara.common.core.enums.UserStatus; | ||||||
|  | import org.dromara.common.core.exception.user.CaptchaException; | ||||||
|  | import org.dromara.common.core.exception.user.CaptchaExpireException; | ||||||
|  | import org.dromara.common.core.exception.user.UserException; | ||||||
|  | import org.dromara.common.core.utils.MessageUtils; | ||||||
|  | import org.dromara.common.core.utils.StringUtils; | ||||||
|  | import org.dromara.common.core.utils.ValidatorUtils; | ||||||
|  | import org.dromara.common.json.utils.JsonUtils; | ||||||
|  | import org.dromara.common.redis.utils.RedisUtils; | ||||||
|  | import org.dromara.common.satoken.utils.LoginHelper; | ||||||
|  | import org.dromara.common.tenant.helper.TenantHelper; | ||||||
|  | import org.dromara.common.web.config.properties.CaptchaProperties; | ||||||
|  | import org.dromara.system.domain.SysClient; | ||||||
|  | import org.dromara.system.domain.SysUser; | ||||||
|  | import org.dromara.system.domain.vo.SysUserVo; | ||||||
|  | import org.dromara.system.mapper.SysUserMapper; | ||||||
|  | import org.dromara.web.domain.vo.LoginVo; | ||||||
|  | import org.dromara.web.service.IAuthStrategy; | ||||||
|  | import org.dromara.web.service.SysLoginService; | ||||||
|  | import org.springframework.stereotype.Service; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 密码认证策略 | ||||||
|  |  * | ||||||
|  |  * @author Michelle.Chung | ||||||
|  |  */ | ||||||
|  | @Slf4j | ||||||
|  | @Service("password" + IAuthStrategy.BASE_NAME) | ||||||
|  | @RequiredArgsConstructor | ||||||
|  | public class PasswordAuthStrategy implements IAuthStrategy { | ||||||
|  |  | ||||||
|  |     private final CaptchaProperties captchaProperties; | ||||||
|  |     private final SysLoginService loginService; | ||||||
|  |     private final SysUserMapper userMapper; | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public LoginVo login(String body, SysClient client) { | ||||||
|  |         PasswordLoginBody loginBody = JsonUtils.parseObject(body, PasswordLoginBody.class); | ||||||
|  |         ValidatorUtils.validate(loginBody); | ||||||
|  |         String tenantId = loginBody.getTenantId(); | ||||||
|  |         String username = loginBody.getUsername(); | ||||||
|  |         String password = loginBody.getPassword(); | ||||||
|  |         String code = loginBody.getCode(); | ||||||
|  |         String uuid = loginBody.getUuid(); | ||||||
|  |  | ||||||
|  |         boolean captchaEnabled = captchaProperties.getEnable(); | ||||||
|  |         // 验证码开关 | ||||||
|  |         if (captchaEnabled) { | ||||||
|  |             validateCaptcha(tenantId, username, code, uuid); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         SysUserVo user = loadUserByUsername(tenantId, username); | ||||||
|  |         loginService.checkLogin(LoginType.PASSWORD, tenantId, username, () -> !BCrypt.checkpw(password, user.getPassword())); | ||||||
|  |         // 此处可根据登录用户的数据不同 自行创建 loginUser | ||||||
|  |         LoginUser loginUser = loginService.buildLoginUser(user); | ||||||
|  |         loginUser.setClientKey(client.getClientKey()); | ||||||
|  |         loginUser.setDeviceType(client.getDeviceType()); | ||||||
|  |         SaLoginModel model = new SaLoginModel(); | ||||||
|  |         model.setDevice(client.getDeviceType()); | ||||||
|  |         // 自定义分配 不同用户体系 不同 token 授权时间 不设置默认走全局 yml 配置 | ||||||
|  |         // 例如: 后台用户30分钟过期 app用户1天过期 | ||||||
|  |         model.setTimeout(client.getTimeout()); | ||||||
|  |         model.setActiveTimeout(client.getActiveTimeout()); | ||||||
|  |         model.setExtra(LoginHelper.CLIENT_KEY, client.getClientId()); | ||||||
|  |         // 生成token | ||||||
|  |         LoginHelper.login(loginUser, model); | ||||||
|  |  | ||||||
|  |         LoginVo loginVo = new LoginVo(); | ||||||
|  |         loginVo.setAccessToken(StpUtil.getTokenValue()); | ||||||
|  |         loginVo.setExpireIn(StpUtil.getTokenTimeout()); | ||||||
|  |         loginVo.setClientId(client.getClientId()); | ||||||
|  |         return loginVo; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 校验验证码 | ||||||
|  |      * | ||||||
|  |      * @param username 用户名 | ||||||
|  |      * @param code     验证码 | ||||||
|  |      * @param uuid     唯一标识 | ||||||
|  |      */ | ||||||
|  |     private void validateCaptcha(String tenantId, String username, String code, String uuid) { | ||||||
|  |         String verifyKey = GlobalConstants.CAPTCHA_CODE_KEY + StringUtils.defaultString(uuid, ""); | ||||||
|  |         String captcha = RedisUtils.getCacheObject(verifyKey); | ||||||
|  |         RedisUtils.deleteObject(verifyKey); | ||||||
|  |         if (captcha == null) { | ||||||
|  |             loginService.recordLogininfor(tenantId, username, Constants.LOGIN_FAIL, MessageUtils.message("user.jcaptcha.expire")); | ||||||
|  |             throw new CaptchaExpireException(); | ||||||
|  |         } | ||||||
|  |         if (!code.equalsIgnoreCase(captcha)) { | ||||||
|  |             loginService.recordLogininfor(tenantId, username, Constants.LOGIN_FAIL, MessageUtils.message("user.jcaptcha.error")); | ||||||
|  |             throw new CaptchaException(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private SysUserVo loadUserByUsername(String tenantId, String username) { | ||||||
|  |         return TenantHelper.dynamic(tenantId, () -> { | ||||||
|  |             SysUser user = userMapper.selectOne(new LambdaQueryWrapper<SysUser>() | ||||||
|  |                 .select(SysUser::getUserName, SysUser::getStatus) | ||||||
|  |                 .eq(SysUser::getUserName, username)); | ||||||
|  |             if (ObjectUtil.isNull(user)) { | ||||||
|  |                 log.info("登录用户:{} 不存在.", username); | ||||||
|  |                 throw new UserException("user.not.exists", username); | ||||||
|  |             } else if (UserStatus.DISABLE.getCode().equals(user.getStatus())) { | ||||||
|  |                 log.info("登录用户:{} 已被停用.", username); | ||||||
|  |                 throw new UserException("user.blocked", username); | ||||||
|  |             } | ||||||
|  |             return userMapper.selectUserByUserName(username); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,107 @@ | |||||||
|  | package org.dromara.web.service.impl; | ||||||
|  |  | ||||||
|  | import cn.dev33.satoken.stp.SaLoginModel; | ||||||
|  | import cn.dev33.satoken.stp.StpUtil; | ||||||
|  | import cn.hutool.core.util.ObjectUtil; | ||||||
|  | import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; | ||||||
|  | import lombok.RequiredArgsConstructor; | ||||||
|  | import lombok.extern.slf4j.Slf4j; | ||||||
|  | import org.dromara.common.core.constant.Constants; | ||||||
|  | import org.dromara.common.core.constant.GlobalConstants; | ||||||
|  | import org.dromara.common.core.domain.model.LoginUser; | ||||||
|  | import org.dromara.common.core.domain.model.SmsLoginBody; | ||||||
|  | import org.dromara.common.core.enums.LoginType; | ||||||
|  | import org.dromara.common.core.enums.UserStatus; | ||||||
|  | import org.dromara.common.core.exception.user.CaptchaExpireException; | ||||||
|  | import org.dromara.common.core.exception.user.UserException; | ||||||
|  | import org.dromara.common.core.utils.MessageUtils; | ||||||
|  | import org.dromara.common.core.utils.StringUtils; | ||||||
|  | import org.dromara.common.core.utils.ValidatorUtils; | ||||||
|  | import org.dromara.common.json.utils.JsonUtils; | ||||||
|  | import org.dromara.common.redis.utils.RedisUtils; | ||||||
|  | import org.dromara.common.satoken.utils.LoginHelper; | ||||||
|  | import org.dromara.common.tenant.helper.TenantHelper; | ||||||
|  | import org.dromara.system.domain.SysClient; | ||||||
|  | import org.dromara.system.domain.SysUser; | ||||||
|  | import org.dromara.system.domain.vo.SysUserVo; | ||||||
|  | import org.dromara.system.mapper.SysUserMapper; | ||||||
|  | import org.dromara.web.domain.vo.LoginVo; | ||||||
|  | import org.dromara.web.service.IAuthStrategy; | ||||||
|  | import org.dromara.web.service.SysLoginService; | ||||||
|  | import org.springframework.stereotype.Service; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 短信认证策略 | ||||||
|  |  * | ||||||
|  |  * @author Michelle.Chung | ||||||
|  |  */ | ||||||
|  | @Slf4j | ||||||
|  | @Service("sms" + IAuthStrategy.BASE_NAME) | ||||||
|  | @RequiredArgsConstructor | ||||||
|  | public class SmsAuthStrategy implements IAuthStrategy { | ||||||
|  |  | ||||||
|  |     private final SysLoginService loginService; | ||||||
|  |     private final SysUserMapper userMapper; | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public LoginVo login(String body, SysClient client) { | ||||||
|  |         SmsLoginBody loginBody = JsonUtils.parseObject(body, SmsLoginBody.class); | ||||||
|  |         ValidatorUtils.validate(loginBody); | ||||||
|  |         String tenantId = loginBody.getTenantId(); | ||||||
|  |         String phonenumber = loginBody.getPhonenumber(); | ||||||
|  |         String smsCode = loginBody.getSmsCode(); | ||||||
|  |  | ||||||
|  |         // 通过手机号查找用户 | ||||||
|  |         SysUserVo user = loadUserByPhonenumber(tenantId, phonenumber); | ||||||
|  |  | ||||||
|  |         loginService.checkLogin(LoginType.SMS, tenantId, user.getUserName(), () -> !validateSmsCode(tenantId, phonenumber, smsCode)); | ||||||
|  |         // 此处可根据登录用户的数据不同 自行创建 loginUser 属性不够用继承扩展就行了 | ||||||
|  |         LoginUser loginUser = loginService.buildLoginUser(user); | ||||||
|  |         loginUser.setClientKey(client.getClientKey()); | ||||||
|  |         loginUser.setDeviceType(client.getDeviceType()); | ||||||
|  |         SaLoginModel model = new SaLoginModel(); | ||||||
|  |         model.setDevice(client.getDeviceType()); | ||||||
|  |         // 自定义分配 不同用户体系 不同 token 授权时间 不设置默认走全局 yml 配置 | ||||||
|  |         // 例如: 后台用户30分钟过期 app用户1天过期 | ||||||
|  |         model.setTimeout(client.getTimeout()); | ||||||
|  |         model.setActiveTimeout(client.getActiveTimeout()); | ||||||
|  |         model.setExtra(LoginHelper.CLIENT_KEY, client.getClientId()); | ||||||
|  |         // 生成token | ||||||
|  |         LoginHelper.login(loginUser, model); | ||||||
|  |  | ||||||
|  |         LoginVo loginVo = new LoginVo(); | ||||||
|  |         loginVo.setAccessToken(StpUtil.getTokenValue()); | ||||||
|  |         loginVo.setExpireIn(StpUtil.getTokenTimeout()); | ||||||
|  |         loginVo.setClientId(client.getClientId()); | ||||||
|  |         return loginVo; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 校验短信验证码 | ||||||
|  |      */ | ||||||
|  |     private boolean validateSmsCode(String tenantId, String phonenumber, String smsCode) { | ||||||
|  |         String code = RedisUtils.getCacheObject(GlobalConstants.CAPTCHA_CODE_KEY + phonenumber); | ||||||
|  |         if (StringUtils.isBlank(code)) { | ||||||
|  |             loginService.recordLogininfor(tenantId, phonenumber, Constants.LOGIN_FAIL, MessageUtils.message("user.jcaptcha.expire")); | ||||||
|  |             throw new CaptchaExpireException(); | ||||||
|  |         } | ||||||
|  |         return code.equals(smsCode); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private SysUserVo loadUserByPhonenumber(String tenantId, String phonenumber) { | ||||||
|  |         return TenantHelper.dynamic(tenantId, () -> { | ||||||
|  |             SysUser user = userMapper.selectOne(new LambdaQueryWrapper<SysUser>() | ||||||
|  |                 .select(SysUser::getPhonenumber, SysUser::getStatus) | ||||||
|  |                 .eq(SysUser::getPhonenumber, phonenumber)); | ||||||
|  |             if (ObjectUtil.isNull(user)) { | ||||||
|  |                 log.info("登录用户:{} 不存在.", phonenumber); | ||||||
|  |                 throw new UserException("user.not.exists", phonenumber); | ||||||
|  |             } else if (UserStatus.DISABLE.getCode().equals(user.getStatus())) { | ||||||
|  |                 log.info("登录用户:{} 已被停用.", phonenumber); | ||||||
|  |                 throw new UserException("user.blocked", phonenumber); | ||||||
|  |             } | ||||||
|  |             return userMapper.selectUserByPhonenumber(phonenumber); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,131 @@ | |||||||
|  | package org.dromara.web.service.impl; | ||||||
|  |  | ||||||
|  | import cn.dev33.satoken.stp.SaLoginModel; | ||||||
|  | import cn.dev33.satoken.stp.StpUtil; | ||||||
|  | import cn.hutool.core.collection.CollUtil; | ||||||
|  | import cn.hutool.core.map.MapUtil; | ||||||
|  | import cn.hutool.core.util.ObjectUtil; | ||||||
|  | import cn.hutool.http.HttpUtil; | ||||||
|  | import cn.hutool.http.Method; | ||||||
|  | import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; | ||||||
|  | import lombok.RequiredArgsConstructor; | ||||||
|  | import lombok.extern.slf4j.Slf4j; | ||||||
|  | import me.zhyd.oauth.model.AuthResponse; | ||||||
|  | import me.zhyd.oauth.model.AuthUser; | ||||||
|  | import org.dromara.common.core.domain.model.LoginUser; | ||||||
|  | import org.dromara.common.core.domain.model.SocialLoginBody; | ||||||
|  | import org.dromara.common.core.enums.UserStatus; | ||||||
|  | import org.dromara.common.core.exception.ServiceException; | ||||||
|  | import org.dromara.common.core.exception.user.UserException; | ||||||
|  | import org.dromara.common.core.utils.ValidatorUtils; | ||||||
|  | import org.dromara.common.json.utils.JsonUtils; | ||||||
|  | import org.dromara.common.satoken.utils.LoginHelper; | ||||||
|  | import org.dromara.common.social.config.properties.SocialProperties; | ||||||
|  | import org.dromara.common.social.utils.SocialUtils; | ||||||
|  | import org.dromara.common.tenant.helper.TenantHelper; | ||||||
|  | import org.dromara.system.domain.SysClient; | ||||||
|  | import org.dromara.system.domain.SysUser; | ||||||
|  | import org.dromara.system.domain.vo.SysSocialVo; | ||||||
|  | import org.dromara.system.domain.vo.SysUserVo; | ||||||
|  | import org.dromara.system.mapper.SysUserMapper; | ||||||
|  | import org.dromara.system.service.ISysSocialService; | ||||||
|  | import org.dromara.web.domain.vo.LoginVo; | ||||||
|  | import org.dromara.web.service.IAuthStrategy; | ||||||
|  | import org.dromara.web.service.SysLoginService; | ||||||
|  | import org.springframework.stereotype.Service; | ||||||
|  |  | ||||||
|  | import java.util.List; | ||||||
|  | import java.util.Optional; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 第三方授权策略 | ||||||
|  |  * | ||||||
|  |  * @author thiszhc is 三三 | ||||||
|  |  */ | ||||||
|  | @Slf4j | ||||||
|  | @Service("social" + IAuthStrategy.BASE_NAME) | ||||||
|  | @RequiredArgsConstructor | ||||||
|  | public class SocialAuthStrategy implements IAuthStrategy { | ||||||
|  |  | ||||||
|  |     private final SocialProperties socialProperties; | ||||||
|  |     private final ISysSocialService sysSocialService; | ||||||
|  |     private final SysUserMapper userMapper; | ||||||
|  |     private final SysLoginService loginService; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 登录-第三方授权登录 | ||||||
|  |      * | ||||||
|  |      * @param body     登录信息 | ||||||
|  |      * @param client   客户端信息 | ||||||
|  |      */ | ||||||
|  |     @Override | ||||||
|  |     public LoginVo login(String body, SysClient client) { | ||||||
|  |         SocialLoginBody loginBody = JsonUtils.parseObject(body, SocialLoginBody.class); | ||||||
|  |         ValidatorUtils.validate(loginBody); | ||||||
|  |         AuthResponse<AuthUser> response = SocialUtils.loginAuth( | ||||||
|  |                 loginBody.getSource(), loginBody.getSocialCode(), | ||||||
|  |                 loginBody.getSocialState(), socialProperties); | ||||||
|  |         if (!response.ok()) { | ||||||
|  |             throw new ServiceException(response.getMsg()); | ||||||
|  |         } | ||||||
|  |         AuthUser authUserData = response.getData(); | ||||||
|  |         if ("GITEE".equals(authUserData.getSource())) { | ||||||
|  |             // 如用户使用 gitee 登录顺手 star 给作者一点支持 拒绝白嫖 | ||||||
|  |             HttpUtil.createRequest(Method.PUT, "https://gitee.com/api/v5/user/starred/dromara/RuoYi-Vue-Plus") | ||||||
|  |                     .formStr(MapUtil.of("access_token", authUserData.getToken().getAccessToken())) | ||||||
|  |                     .executeAsync(); | ||||||
|  |             HttpUtil.createRequest(Method.PUT, "https://gitee.com/api/v5/user/starred/dromara/RuoYi-Cloud-Plus") | ||||||
|  |                     .formStr(MapUtil.of("access_token", authUserData.getToken().getAccessToken())) | ||||||
|  |                     .executeAsync(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         List<SysSocialVo> list = sysSocialService.selectByAuthId(authUserData.getSource() + authUserData.getUuid()); | ||||||
|  |         if (CollUtil.isEmpty(list)) { | ||||||
|  |             throw new ServiceException("你还没有绑定第三方账号,绑定后才可以登录!"); | ||||||
|  |         } | ||||||
|  |         Optional<SysSocialVo> opt = list.stream().filter(x -> x.getTenantId().equals(loginBody.getTenantId())).findAny(); | ||||||
|  |         if (opt.isEmpty()) { | ||||||
|  |             throw new ServiceException("对不起,你没有权限登录当前租户!"); | ||||||
|  |         } | ||||||
|  |         SysSocialVo social = opt.get(); | ||||||
|  |         // 查找用户 | ||||||
|  |         SysUserVo user = loadUser(social.getTenantId(), social.getUserId()); | ||||||
|  |  | ||||||
|  |         // 此处可根据登录用户的数据不同 自行创建 loginUser 属性不够用继承扩展就行了 | ||||||
|  |         LoginUser loginUser = loginService.buildLoginUser(user); | ||||||
|  |         loginUser.setClientKey(client.getClientKey()); | ||||||
|  |         loginUser.setDeviceType(client.getDeviceType()); | ||||||
|  |         SaLoginModel model = new SaLoginModel(); | ||||||
|  |         model.setDevice(client.getDeviceType()); | ||||||
|  |         // 自定义分配 不同用户体系 不同 token 授权时间 不设置默认走全局 yml 配置 | ||||||
|  |         // 例如: 后台用户30分钟过期 app用户1天过期 | ||||||
|  |         model.setTimeout(client.getTimeout()); | ||||||
|  |         model.setActiveTimeout(client.getActiveTimeout()); | ||||||
|  |         model.setExtra(LoginHelper.CLIENT_KEY, client.getClientId()); | ||||||
|  |         // 生成token | ||||||
|  |         LoginHelper.login(loginUser, model); | ||||||
|  |  | ||||||
|  |         LoginVo loginVo = new LoginVo(); | ||||||
|  |         loginVo.setAccessToken(StpUtil.getTokenValue()); | ||||||
|  |         loginVo.setExpireIn(StpUtil.getTokenTimeout()); | ||||||
|  |         loginVo.setClientId(client.getClientId()); | ||||||
|  |         return loginVo; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private SysUserVo loadUser(String tenantId, Long userId) { | ||||||
|  |         return TenantHelper.dynamic(tenantId, () -> { | ||||||
|  |             SysUser user = userMapper.selectOne(new LambdaQueryWrapper<SysUser>() | ||||||
|  |                 .select(SysUser::getUserName, SysUser::getStatus) | ||||||
|  |                 .eq(SysUser::getUserId, userId)); | ||||||
|  |             if (ObjectUtil.isNull(user)) { | ||||||
|  |                 log.info("登录用户:{} 不存在.", ""); | ||||||
|  |                 throw new UserException("user.not.exists", ""); | ||||||
|  |             } else if (UserStatus.DISABLE.getCode().equals(user.getStatus())) { | ||||||
|  |                 log.info("登录用户:{} 已被停用.", ""); | ||||||
|  |                 throw new UserException("user.blocked", ""); | ||||||
|  |             } | ||||||
|  |             return userMapper.selectUserByUserName(user.getUserName()); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,91 @@ | |||||||
|  | package org.dromara.web.service.impl; | ||||||
|  |  | ||||||
|  | import cn.dev33.satoken.stp.SaLoginModel; | ||||||
|  | import cn.dev33.satoken.stp.StpUtil; | ||||||
|  | import cn.hutool.core.util.ObjectUtil; | ||||||
|  | import lombok.RequiredArgsConstructor; | ||||||
|  | import lombok.extern.slf4j.Slf4j; | ||||||
|  | import org.dromara.common.core.domain.model.XcxLoginBody; | ||||||
|  | import org.dromara.common.core.domain.model.XcxLoginUser; | ||||||
|  | import org.dromara.common.core.enums.UserStatus; | ||||||
|  | import org.dromara.common.core.utils.ValidatorUtils; | ||||||
|  | import org.dromara.common.json.utils.JsonUtils; | ||||||
|  | import org.dromara.common.satoken.utils.LoginHelper; | ||||||
|  | import org.dromara.system.domain.SysClient; | ||||||
|  | import org.dromara.system.domain.vo.SysUserVo; | ||||||
|  | import org.dromara.web.domain.vo.LoginVo; | ||||||
|  | import org.dromara.web.service.IAuthStrategy; | ||||||
|  | import org.dromara.web.service.SysLoginService; | ||||||
|  | import org.springframework.stereotype.Service; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 邮件认证策略 | ||||||
|  |  * | ||||||
|  |  * @author Michelle.Chung | ||||||
|  |  */ | ||||||
|  | @Slf4j | ||||||
|  | @Service("xcx" + IAuthStrategy.BASE_NAME) | ||||||
|  | @RequiredArgsConstructor | ||||||
|  | public class XcxAuthStrategy implements IAuthStrategy { | ||||||
|  |  | ||||||
|  |     private final SysLoginService loginService; | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public LoginVo login(String body, SysClient client) { | ||||||
|  |         XcxLoginBody loginBody = JsonUtils.parseObject(body, XcxLoginBody.class); | ||||||
|  |         ValidatorUtils.validate(loginBody); | ||||||
|  |         // xcxCode 为 小程序调用 wx.login 授权后获取 | ||||||
|  |         String xcxCode = loginBody.getXcxCode(); | ||||||
|  |         // 多个小程序识别使用 | ||||||
|  |         String appid = loginBody.getAppid(); | ||||||
|  |  | ||||||
|  |         // todo 以下自行实现 | ||||||
|  |         // 校验 appid + appsrcret + xcxCode 调用登录凭证校验接口 获取 session_key 与 openid | ||||||
|  |         String openid = ""; | ||||||
|  |         // 框架登录不限制从什么表查询 只要最终构建出 LoginUser 即可 | ||||||
|  |         SysUserVo user = loadUserByOpenid(openid); | ||||||
|  |  | ||||||
|  |         // 此处可根据登录用户的数据不同 自行创建 loginUser 属性不够用继承扩展就行了 | ||||||
|  |         XcxLoginUser loginUser = new XcxLoginUser(); | ||||||
|  |         loginUser.setTenantId(user.getTenantId()); | ||||||
|  |         loginUser.setUserId(user.getUserId()); | ||||||
|  |         loginUser.setUsername(user.getUserName()); | ||||||
|  |         loginUser.setNickname(user.getNickName()); | ||||||
|  |         loginUser.setUserType(user.getUserType()); | ||||||
|  |         loginUser.setClientKey(client.getClientKey()); | ||||||
|  |         loginUser.setDeviceType(client.getDeviceType()); | ||||||
|  |         loginUser.setOpenid(openid); | ||||||
|  |  | ||||||
|  |         SaLoginModel model = new SaLoginModel(); | ||||||
|  |         model.setDevice(client.getDeviceType()); | ||||||
|  |         // 自定义分配 不同用户体系 不同 token 授权时间 不设置默认走全局 yml 配置 | ||||||
|  |         // 例如: 后台用户30分钟过期 app用户1天过期 | ||||||
|  |         model.setTimeout(client.getTimeout()); | ||||||
|  |         model.setActiveTimeout(client.getActiveTimeout()); | ||||||
|  |         model.setExtra(LoginHelper.CLIENT_KEY, client.getClientId()); | ||||||
|  |         // 生成token | ||||||
|  |         LoginHelper.login(loginUser, model); | ||||||
|  |  | ||||||
|  |         LoginVo loginVo = new LoginVo(); | ||||||
|  |         loginVo.setAccessToken(StpUtil.getTokenValue()); | ||||||
|  |         loginVo.setExpireIn(StpUtil.getTokenTimeout()); | ||||||
|  |         loginVo.setClientId(client.getClientId()); | ||||||
|  |         loginVo.setOpenid(openid); | ||||||
|  |         return loginVo; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private SysUserVo loadUserByOpenid(String openid) { | ||||||
|  |         // 使用 openid 查询绑定用户 如未绑定用户 则根据业务自行处理 例如 创建默认用户 | ||||||
|  |         // todo 自行实现 userService.selectUserByOpenid(openid); | ||||||
|  |         SysUserVo user = new SysUserVo(); | ||||||
|  |         if (ObjectUtil.isNull(user)) { | ||||||
|  |             log.info("登录用户:{} 不存在.", openid); | ||||||
|  |             // todo 用户不存在 业务逻辑自行实现 | ||||||
|  |         } else if (UserStatus.DISABLE.getCode().equals(user.getStatus())) { | ||||||
|  |             log.info("登录用户:{} 已被停用.", openid); | ||||||
|  |             // todo 用户已被停用 业务逻辑自行实现 | ||||||
|  |         } | ||||||
|  |         return user; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -8,27 +8,21 @@ spring.boot.admin.client: | |||||||
|   username: ruoyi |   username: ruoyi | ||||||
|   password: 123456 |   password: 123456 | ||||||
|  |  | ||||||
| --- # xxl-job 配置 | --- # powerjob 配置 | ||||||
| xxl.job: | powerjob: | ||||||
|   # 执行器开关 |   worker: | ||||||
|   enabled: true |     # 如何开启调度中心请查看文档教程 | ||||||
|   # 调度中心地址:如调度中心集群部署存在多个地址则用逗号分隔。 |     enabled: false | ||||||
|   admin-addresses: http://localhost:9100/xxl-job-admin |     # 需要先在 powerjob 登录页执行应用注册后才能使用 | ||||||
|   # 执行器通讯TOKEN:非空时启用 |     app-name: ruoyi-worker | ||||||
|   access-token: xxl-job |     allow-lazy-connect-server: false | ||||||
|   executor: |     max-appended-wf-context-length: 4096 | ||||||
|     # 执行器AppName:执行器心跳注册分组依据;为空则关闭自动注册 |     max-result-length: 4096 | ||||||
|     appname: xxl-job-executor |     # 28080 端口 随着主应用端口飘逸 避免集群冲突 | ||||||
|     # 执行器端口号 执行器从9101开始往后写 |     port: 2${server.port} | ||||||
|     port: 9101 |     protocol: http | ||||||
|     # 执行器注册:默认IP:PORT |     server-address: 127.0.0.1:7700 | ||||||
|     address: |     store-strategy: disk | ||||||
|     # 执行器IP:默认自动获取IP |  | ||||||
|     ip: |  | ||||||
|     # 执行器运行日志文件存储磁盘路径 |  | ||||||
|     logpath: ./logs/xxl-job |  | ||||||
|     # 执行器日志文件保存天数:大于3生效 |  | ||||||
|     logretentiondays: 30 |  | ||||||
|  |  | ||||||
| --- # 数据源配置 | --- # 数据源配置 | ||||||
| spring: | spring: | ||||||
| @@ -49,7 +43,7 @@ spring: | |||||||
|           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://localhost: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&allowPublicKeyRetrieval=true | ||||||
|           username: root |           username: root | ||||||
|           password: root |           password: root | ||||||
|         # 从库数据源 |         # 从库数据源 | ||||||
| @@ -57,7 +51,7 @@ spring: | |||||||
|           lazy: true |           lazy: true | ||||||
|           type: ${spring.datasource.type} |           type: ${spring.datasource.type} | ||||||
|           driverClassName: com.mysql.cj.jdbc.Driver |           driverClassName: com.mysql.cj.jdbc.Driver | ||||||
|           url: jdbc:mysql://localhost:3306/ry-vue?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8&autoReconnect=true&rewriteBatchedStatements=true |           url: jdbc:mysql://localhost:3306/ry-vue?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8&autoReconnect=true&rewriteBatchedStatements=true&allowPublicKeyRetrieval=true | ||||||
|           username: |           username: | ||||||
|           password: |           password: | ||||||
| #        oracle: | #        oracle: | ||||||
| @@ -66,8 +60,6 @@ spring: | |||||||
| #          url: jdbc:oracle:thin:@//localhost:1521/XE | #          url: jdbc:oracle:thin:@//localhost:1521/XE | ||||||
| #          username: ROOT | #          username: ROOT | ||||||
| #          password: root | #          password: root | ||||||
| #          hikari: |  | ||||||
| #            connectionTestQuery: SELECT 1 FROM DUAL |  | ||||||
| #        postgres: | #        postgres: | ||||||
| #          type: ${spring.datasource.type} | #          type: ${spring.datasource.type} | ||||||
| #          driverClassName: org.postgresql.Driver | #          driverClassName: org.postgresql.Driver | ||||||
| @@ -86,18 +78,18 @@ spring: | |||||||
|         # 最小空闲线程数量 |         # 最小空闲线程数量 | ||||||
|         minIdle: 10 |         minIdle: 10 | ||||||
|         # 配置获取连接等待超时的时间 |         # 配置获取连接等待超时的时间 | ||||||
|         connectionTimeout: 10000 |         connectionTimeout: 30000 | ||||||
|         # 校验超时时间 |         # 校验超时时间 | ||||||
|         validationTimeout: 5000 |         validationTimeout: 5000 | ||||||
|         # 空闲连接存活最大时间,默认10分钟 |         # 空闲连接存活最大时间,默认10分钟 | ||||||
|         idleTimeout: 60000 |         idleTimeout: 600000 | ||||||
|         # 此属性控制池中连接的最长生命周期,值0表示无限生命周期,默认30分钟 |         # 此属性控制池中连接的最长生命周期,值0表示无限生命周期,默认30分钟 | ||||||
|         maxLifetime: 900000 |         maxLifetime: 1800000 | ||||||
|         # 连接测试query(配置检测连接是否有效) |         # 多久检查一次连接的活性 | ||||||
|         connectionTestQuery: SELECT 1 |         keepaliveTime: 30000 | ||||||
|  |  | ||||||
| --- # redis 单机配置(单机与集群只能开启一个另一个需要注释掉) | --- # redis 单机配置(单机与集群只能开启一个另一个需要注释掉) | ||||||
| spring: | spring.data: | ||||||
|   redis: |   redis: | ||||||
|     # 地址 |     # 地址 | ||||||
|     host: localhost |     host: localhost | ||||||
| @@ -110,7 +102,7 @@ spring: | |||||||
|     # 连接超时时间 |     # 连接超时时间 | ||||||
|     timeout: 10s |     timeout: 10s | ||||||
|     # 是否开启ssl |     # 是否开启ssl | ||||||
|     ssl: false |     ssl.enabled: false | ||||||
|  |  | ||||||
| redisson: | redisson: | ||||||
|   # redis key前缀 |   # redis key前缀 | ||||||
| @@ -156,14 +148,100 @@ mail: | |||||||
|   # Socket连接超时值,单位毫秒,缺省值不超时 |   # Socket连接超时值,单位毫秒,缺省值不超时 | ||||||
|   connectionTimeout: 0 |   connectionTimeout: 0 | ||||||
|  |  | ||||||
| --- # sms 短信 | --- # sms 短信 支持 阿里云 腾讯云 云片 等等各式各样的短信服务商 | ||||||
|  | # https://wind.kim/doc/start 文档地址 各个厂商可同时使用 | ||||||
| sms: | sms: | ||||||
|   enabled: false |  | ||||||
|   # 阿里云 dysmsapi.aliyuncs.com |   # 阿里云 dysmsapi.aliyuncs.com | ||||||
|   # 腾讯云 sms.tencentcloudapi.com |   alibaba: | ||||||
|   endpoint: "dysmsapi.aliyuncs.com" |     #请求地址 默认为 dysmsapi.aliyuncs.com 如无特殊改变可以不用设置 | ||||||
|   accessKeyId: xxxxxxx |     requestUrl: dysmsapi.aliyuncs.com | ||||||
|   accessKeySecret: xxxxxx |     #阿里云的accessKey | ||||||
|   signName: 测试 |     accessKeyId: xxxxxxx | ||||||
|   # 腾讯专用 |     #阿里云的accessKeySecret | ||||||
|   sdkAppId: |     accessKeySecret: xxxxxxx | ||||||
|  |     #短信签名 | ||||||
|  |     signature: 测试 | ||||||
|  |   tencent: | ||||||
|  |     #请求地址默认为 sms.tencentcloudapi.com 如无特殊改变可不用设置 | ||||||
|  |     requestUrl: sms.tencentcloudapi.com | ||||||
|  |     #腾讯云的accessKey | ||||||
|  |     accessKeyId: xxxxxxx | ||||||
|  |     #腾讯云的accessKeySecret | ||||||
|  |     accessKeySecret: xxxxxxx | ||||||
|  |     #短信签名 | ||||||
|  |     signature: 测试 | ||||||
|  |     #短信sdkAppId | ||||||
|  |     sdkAppId: appid | ||||||
|  |     #地域信息默认为 ap-guangzhou 如无特殊改变可不用设置 | ||||||
|  |     territory: ap-guangzhou | ||||||
|  |  | ||||||
|  |  | ||||||
|  | --- # 三方授权 | ||||||
|  | justauth: | ||||||
|  |   enabled: true | ||||||
|  |   # 前端外网访问地址 | ||||||
|  |   address: http://localhost:80 | ||||||
|  |   type: | ||||||
|  |     maxkey: | ||||||
|  |       # maxkey 服务器地址 | ||||||
|  |       # 注意 如下均配置均不需要修改 maxkey 已经内置好了数据 | ||||||
|  |       server-url: http://sso.maxkey.top | ||||||
|  |       client-id: 876892492581044224 | ||||||
|  |       client-secret: x1Y5MTMwNzIwMjMxNTM4NDc3Mzche8 | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=maxkey | ||||||
|  |     qq: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=qq | ||||||
|  |       union-id: false | ||||||
|  |     weibo: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=weibo | ||||||
|  |     gitee: | ||||||
|  |       client-id: 91436b7940090d09c72c7daf85b959cfd5f215d67eea73acbf61b6b590751a98 | ||||||
|  |       client-secret: 02c6fcfd70342980cd8dd2f2c06c1a350645d76c754d7a264c4e125f9ba915ac | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=gitee | ||||||
|  |     dingtalk: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=dingtalk | ||||||
|  |     baidu: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=baidu | ||||||
|  |     csdn: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=csdn | ||||||
|  |     coding: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=coding | ||||||
|  |       coding-group-name: xx | ||||||
|  |     oschina: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=oschina | ||||||
|  |     alipay_wallet: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=alipay_wallet | ||||||
|  |       alipay-public-key: MIIB**************DAQAB | ||||||
|  |     wechat_open: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=wechat_open | ||||||
|  |     wechat_mp: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=wechat_mp | ||||||
|  |     wechat_enterprise: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=wechat_enterprise | ||||||
|  |       agent-id: 1000002 | ||||||
|  |     gitlab: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=gitlab | ||||||
|   | |||||||
| @@ -11,27 +11,21 @@ spring.boot.admin.client: | |||||||
|   username: ruoyi |   username: ruoyi | ||||||
|   password: 123456 |   password: 123456 | ||||||
|  |  | ||||||
| --- # xxl-job 配置 | --- # powerjob 配置 | ||||||
| xxl.job: | powerjob: | ||||||
|   # 执行器开关 |   worker: | ||||||
|   enabled: true |     # 如何开启调度中心请查看文档教程 | ||||||
|   # 调度中心地址:如调度中心集群部署存在多个地址则用逗号分隔。 |     enabled: false | ||||||
|   admin-addresses: http://localhost:9100/xxl-job-admin |     # 需要先在 powerjob 登录页执行应用注册后才能使用 | ||||||
|   # 执行器通讯TOKEN:非空时启用 |     app-name: ruoyi-worker | ||||||
|   access-token: xxl-job |     allow-lazy-connect-server: false | ||||||
|   executor: |     max-appended-wf-context-length: 4096 | ||||||
|     # 执行器AppName:执行器心跳注册分组依据;为空则关闭自动注册 |     max-result-length: 4096 | ||||||
|     appname: xxl-job-executor |     # 28080 端口 随着主应用端口飘逸 避免集群冲突 | ||||||
|     # 执行器端口号 执行器从9101开始往后写 |     port: 2${server.port} | ||||||
|     port: 9101 |     protocol: http | ||||||
|     # 执行器注册:默认IP:PORT |     server-address: 127.0.0.1:7700 | ||||||
|     address: |     store-strategy: disk | ||||||
|     # 执行器IP:默认自动获取IP |  | ||||||
|     ip: |  | ||||||
|     # 执行器运行日志文件存储磁盘路径 |  | ||||||
|     logpath: ./logs/xxl-job |  | ||||||
|     # 执行器日志文件保存天数:大于3生效 |  | ||||||
|     logretentiondays: 30 |  | ||||||
|  |  | ||||||
| --- # 数据源配置 | --- # 数据源配置 | ||||||
| spring: | spring: | ||||||
| @@ -52,7 +46,7 @@ spring: | |||||||
|           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://localhost: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&allowPublicKeyRetrieval=true | ||||||
|           username: root |           username: root | ||||||
|           password: root |           password: root | ||||||
|         # 从库数据源 |         # 从库数据源 | ||||||
| @@ -60,7 +54,7 @@ spring: | |||||||
|           lazy: true |           lazy: true | ||||||
|           type: ${spring.datasource.type} |           type: ${spring.datasource.type} | ||||||
|           driverClassName: com.mysql.cj.jdbc.Driver |           driverClassName: com.mysql.cj.jdbc.Driver | ||||||
|           url: jdbc:mysql://localhost:3306/ry-vue?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8&autoReconnect=true&rewriteBatchedStatements=true |           url: jdbc:mysql://localhost:3306/ry-vue?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8&autoReconnect=true&rewriteBatchedStatements=true&allowPublicKeyRetrieval=true | ||||||
|           username: |           username: | ||||||
|           password: |           password: | ||||||
| #        oracle: | #        oracle: | ||||||
| @@ -69,8 +63,6 @@ spring: | |||||||
| #          url: jdbc:oracle:thin:@//localhost:1521/XE | #          url: jdbc:oracle:thin:@//localhost:1521/XE | ||||||
| #          username: ROOT | #          username: ROOT | ||||||
| #          password: root | #          password: root | ||||||
| #          hikari: |  | ||||||
| #            connectionTestQuery: SELECT 1 FROM DUAL |  | ||||||
| #        postgres: | #        postgres: | ||||||
| #          type: ${spring.datasource.type} | #          type: ${spring.datasource.type} | ||||||
| #          driverClassName: org.postgresql.Driver | #          driverClassName: org.postgresql.Driver | ||||||
| @@ -89,18 +81,18 @@ spring: | |||||||
|         # 最小空闲线程数量 |         # 最小空闲线程数量 | ||||||
|         minIdle: 10 |         minIdle: 10 | ||||||
|         # 配置获取连接等待超时的时间 |         # 配置获取连接等待超时的时间 | ||||||
|         connectionTimeout: 10000 |         connectionTimeout: 30000 | ||||||
|         # 校验超时时间 |         # 校验超时时间 | ||||||
|         validationTimeout: 5000 |         validationTimeout: 5000 | ||||||
|         # 空闲连接存活最大时间,默认10分钟 |         # 空闲连接存活最大时间,默认10分钟 | ||||||
|         idleTimeout: 60000 |         idleTimeout: 600000 | ||||||
|         # 此属性控制池中连接的最长生命周期,值0表示无限生命周期,默认30分钟 |         # 此属性控制池中连接的最长生命周期,值0表示无限生命周期,默认30分钟 | ||||||
|         maxLifetime: 900000 |         maxLifetime: 1800000 | ||||||
|         # 连接测试query(配置检测连接是否有效) |         # 多久检查一次连接的活性 | ||||||
|         connectionTestQuery: SELECT 1 |         keepaliveTime: 30000 | ||||||
|  |  | ||||||
| --- # redis 单机配置(单机与集群只能开启一个另一个需要注释掉) | --- # redis 单机配置(单机与集群只能开启一个另一个需要注释掉) | ||||||
| spring: | spring.data: | ||||||
|   redis: |   redis: | ||||||
|     # 地址 |     # 地址 | ||||||
|     host: localhost |     host: localhost | ||||||
| @@ -113,7 +105,7 @@ spring: | |||||||
|     # 连接超时时间 |     # 连接超时时间 | ||||||
|     timeout: 10s |     timeout: 10s | ||||||
|     # 是否开启ssl |     # 是否开启ssl | ||||||
|     ssl: false |     ssl.enabled: false | ||||||
|  |  | ||||||
| redisson: | redisson: | ||||||
|   # redis key前缀 |   # redis key前缀 | ||||||
| @@ -159,14 +151,99 @@ mail: | |||||||
|   # Socket连接超时值,单位毫秒,缺省值不超时 |   # Socket连接超时值,单位毫秒,缺省值不超时 | ||||||
|   connectionTimeout: 0 |   connectionTimeout: 0 | ||||||
|  |  | ||||||
| --- # sms 短信 | --- # sms 短信 支持 阿里云 腾讯云 云片 等等各式各样的短信服务商 | ||||||
|  | # https://wind.kim/doc/start 文档地址 各个厂商可同时使用 | ||||||
| sms: | sms: | ||||||
|   enabled: false |  | ||||||
|   # 阿里云 dysmsapi.aliyuncs.com |   # 阿里云 dysmsapi.aliyuncs.com | ||||||
|   # 腾讯云 sms.tencentcloudapi.com |   alibaba: | ||||||
|   endpoint: "dysmsapi.aliyuncs.com" |     #请求地址 默认为 dysmsapi.aliyuncs.com 如无特殊改变可以不用设置 | ||||||
|   accessKeyId: xxxxxxx |     requestUrl: dysmsapi.aliyuncs.com | ||||||
|   accessKeySecret: xxxxxx |     #阿里云的accessKey | ||||||
|   signName: 测试 |     accessKeyId: xxxxxxx | ||||||
|   # 腾讯专用 |     #阿里云的accessKeySecret | ||||||
|   sdkAppId: |     accessKeySecret: xxxxxxx | ||||||
|  |     #短信签名 | ||||||
|  |     signature: 测试 | ||||||
|  |   tencent: | ||||||
|  |     #请求地址默认为 sms.tencentcloudapi.com 如无特殊改变可不用设置 | ||||||
|  |     requestUrl: sms.tencentcloudapi.com | ||||||
|  |     #腾讯云的accessKey | ||||||
|  |     accessKeyId: xxxxxxx | ||||||
|  |     #腾讯云的accessKeySecret | ||||||
|  |     accessKeySecret: xxxxxxx | ||||||
|  |     #短信签名 | ||||||
|  |     signature: 测试 | ||||||
|  |     #短信sdkAppId | ||||||
|  |     sdkAppId: appid | ||||||
|  |     #地域信息默认为 ap-guangzhou 如无特殊改变可不用设置 | ||||||
|  |     territory: ap-guangzhou | ||||||
|  |  | ||||||
|  | --- # 三方授权 | ||||||
|  | justauth: | ||||||
|  |   enabled: true | ||||||
|  |   # 前端外网访问地址 | ||||||
|  |   address: http://localhost:80 | ||||||
|  |   type: | ||||||
|  |     maxkey: | ||||||
|  |       # maxkey 服务器地址 | ||||||
|  |       # 注意 如下均配置均不需要修改 maxkey 已经内置好了数据 | ||||||
|  |       server-url: http://sso.maxkey.top | ||||||
|  |       client-id: 876892492581044224 | ||||||
|  |       client-secret: x1Y5MTMwNzIwMjMxNTM4NDc3Mzche8 | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=maxkey | ||||||
|  |     qq: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=qq | ||||||
|  |       union-id: false | ||||||
|  |     weibo: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=weibo | ||||||
|  |     gitee: | ||||||
|  |       client-id: 91436b7940090d09c72c7daf85b959cfd5f215d67eea73acbf61b6b590751a98 | ||||||
|  |       client-secret: 02c6fcfd70342980cd8dd2f2c06c1a350645d76c754d7a264c4e125f9ba915ac | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=gitee | ||||||
|  |     dingtalk: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=dingtalk | ||||||
|  |     baidu: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=baidu | ||||||
|  |     csdn: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=csdn | ||||||
|  |     coding: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=coding | ||||||
|  |       coding-group-name: xx | ||||||
|  |     oschina: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=oschina | ||||||
|  |     alipay_wallet: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=alipay_wallet | ||||||
|  |       alipay-public-key: MIIB**************DAQAB | ||||||
|  |     wechat_open: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=wechat_open | ||||||
|  |     wechat_mp: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=wechat_mp | ||||||
|  |     wechat_enterprise: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=wechat_enterprise | ||||||
|  |       agent-id: 1000002 | ||||||
|  |     gitlab: | ||||||
|  |       client-id: 10**********6 | ||||||
|  |       client-secret: 1f7d08**********5b7**********29e | ||||||
|  |       redirect-uri: ${justauth.address}/social-callback?source=gitlab | ||||||
|   | |||||||
| @@ -3,17 +3,12 @@ ruoyi: | |||||||
|   # 名称 |   # 名称 | ||||||
|   name: RuoYi-Vue-Plus |   name: RuoYi-Vue-Plus | ||||||
|   # 版本 |   # 版本 | ||||||
|   version: ${ruoyi-vue-plus.version} |   version: ${revision} | ||||||
|   # 版权年份 |   # 版权年份 | ||||||
|   copyrightYear: 2022 |   copyrightYear: 2023 | ||||||
|   # 实例演示开关 |  | ||||||
|   demoEnabled: true |  | ||||||
|   # 获取ip地址开关 |  | ||||||
|   addressEnabled: true |  | ||||||
|   # 缓存懒加载 |  | ||||||
|   cacheLazy: false |  | ||||||
|  |  | ||||||
| captcha: | captcha: | ||||||
|  |   enable: true | ||||||
|   # 页面 <参数设置> 可开启关闭 验证码校验 |   # 页面 <参数设置> 可开启关闭 验证码校验 | ||||||
|   # 验证码类型 math 数组计算 char 字符验证 |   # 验证码类型 math 数组计算 char 字符验证 | ||||||
|   type: MATH |   type: MATH | ||||||
| @@ -49,9 +44,10 @@ server: | |||||||
| # 日志配置 | # 日志配置 | ||||||
| logging: | logging: | ||||||
|   level: |   level: | ||||||
|     com.ruoyi: @logging.level@ |     org.dromara: @logging.level@ | ||||||
|     org.springframework: warn |     org.springframework: warn | ||||||
|   config: classpath:logback.xml |     tech.powerjob.worker.background: warn | ||||||
|  |   config: classpath:logback-plus.xml | ||||||
|  |  | ||||||
| # 用户配置 | # 用户配置 | ||||||
| user: | user: | ||||||
| @@ -78,11 +74,6 @@ spring: | |||||||
|       max-file-size: 10MB |       max-file-size: 10MB | ||||||
|       # 设置总上传的文件大小 |       # 设置总上传的文件大小 | ||||||
|       max-request-size: 20MB |       max-request-size: 20MB | ||||||
|   # 服务模块 |  | ||||||
|   devtools: |  | ||||||
|     restart: |  | ||||||
|       # 热部署开关 |  | ||||||
|       enabled: true |  | ||||||
|   mvc: |   mvc: | ||||||
|     format: |     format: | ||||||
|       date-time: yyyy-MM-dd HH:mm:ss |       date-time: yyyy-MM-dd HH:mm:ss | ||||||
| @@ -102,20 +93,10 @@ spring: | |||||||
| sa-token: | sa-token: | ||||||
|   # token名称 (同时也是cookie名称) |   # token名称 (同时也是cookie名称) | ||||||
|   token-name: Authorization |   token-name: Authorization | ||||||
|   # token有效期 设为一天 (必定过期) 单位: 秒 |  | ||||||
|   timeout: 86400 |  | ||||||
|   # token临时有效期 (指定时间无操作就过期) 单位: 秒 |  | ||||||
|   activity-timeout: 1800 |  | ||||||
|   # 是否允许同一账号并发登录 (为true时允许一起登录, 为false时新登录挤掉旧登录) |   # 是否允许同一账号并发登录 (为true时允许一起登录, 为false时新登录挤掉旧登录) | ||||||
|   is-concurrent: true |   is-concurrent: true | ||||||
|   # 在多人登录同一账号时,是否共用一个token (为true时所有登录共用一个token, 为false时每次登录新建一个token) |   # 在多人登录同一账号时,是否共用一个token (为true时所有登录共用一个token, 为false时每次登录新建一个token) | ||||||
|   is-share: false |   is-share: false | ||||||
|   # 是否尝试从header里读取token |  | ||||||
|   is-read-header: true |  | ||||||
|   # 是否尝试从cookie里读取token |  | ||||||
|   is-read-cookie: false |  | ||||||
|   # token前缀 |  | ||||||
|   token-prefix: "Bearer" |  | ||||||
|   # jwt秘钥 |   # jwt秘钥 | ||||||
|   jwt-secret-key: abcdefghijklmnopqrstuvwxyz |   jwt-secret-key: abcdefghijklmnopqrstuvwxyz | ||||||
|  |  | ||||||
| @@ -128,74 +109,95 @@ security: | |||||||
|     - /**/*.html |     - /**/*.html | ||||||
|     - /**/*.css |     - /**/*.css | ||||||
|     - /**/*.js |     - /**/*.js | ||||||
|     # swagger 文档配置 |     # 公共路径 | ||||||
|     - /favicon.ico |     - /favicon.ico | ||||||
|  |     - /error | ||||||
|  |     # swagger 文档配置 | ||||||
|     - /*/api-docs |     - /*/api-docs | ||||||
|     - /*/api-docs/** |     - /*/api-docs/** | ||||||
|     # actuator 监控配置 |     # actuator 监控配置 | ||||||
|     - /actuator |     - /actuator | ||||||
|     - /actuator/** |     - /actuator/** | ||||||
|  |  | ||||||
|  | # 多租户配置 | ||||||
|  | tenant: | ||||||
|  |   # 是否开启 | ||||||
|  |   enable: true | ||||||
|  |   # 排除表 | ||||||
|  |   excludes: | ||||||
|  |     - sys_menu | ||||||
|  |     - sys_tenant | ||||||
|  |     - sys_tenant_package | ||||||
|  |     - sys_role_dept | ||||||
|  |     - sys_role_menu | ||||||
|  |     - sys_user_post | ||||||
|  |     - sys_user_role | ||||||
|  |     - sys_client | ||||||
|  |     - sys_oss_config | ||||||
|  |  | ||||||
| # MyBatisPlus配置 | # MyBatisPlus配置 | ||||||
| # https://baomidou.com/config/ | # https://baomidou.com/config/ | ||||||
| mybatis-plus: | mybatis-plus: | ||||||
|   # 不支持多包, 如有需要可在注解配置 或 提升扫包等级 |   # 不支持多包, 如有需要可在注解配置 或 提升扫包等级 | ||||||
|   # 例如 com.**.**.mapper |   # 例如 com.**.**.mapper | ||||||
|   mapperPackage: com.ruoyi.**.mapper |   mapperPackage: org.dromara.**.mapper | ||||||
|   # 对应的 XML 文件位置 |   # 对应的 XML 文件位置 | ||||||
|   mapperLocations: classpath*:mapper/**/*Mapper.xml |   mapperLocations: classpath*:mapper/**/*Mapper.xml | ||||||
|   # 实体扫描,多个package用逗号或者分号分隔 |   # 实体扫描,多个package用逗号或者分号分隔 | ||||||
|   typeAliasesPackage: com.ruoyi.**.domain |   typeAliasesPackage: org.dromara.**.domain | ||||||
|   # 启动时是否检查 MyBatis XML 文件的存在,默认不检查 |  | ||||||
|   checkConfigLocation: false |  | ||||||
|   configuration: |  | ||||||
|     # 自动驼峰命名规则(camel case)映射 |  | ||||||
|     mapUnderscoreToCamelCase: true |  | ||||||
|     # MyBatis 自动映射策略 |  | ||||||
|     # NONE:不启用 PARTIAL:只对非嵌套 resultMap 自动映射 FULL:对所有 resultMap 自动映射 |  | ||||||
|     autoMappingBehavior: PARTIAL |  | ||||||
|     # MyBatis 自动映射时未知列或未知属性处理策 |  | ||||||
|     # NONE:不做处理 WARNING:打印相关警告 FAILING:抛出异常和详细信息 |  | ||||||
|     autoMappingUnknownColumnBehavior: NONE |  | ||||||
|     # 更详细的日志输出 会有性能损耗 org.apache.ibatis.logging.stdout.StdOutImpl |  | ||||||
|     # 关闭日志记录 (可单纯使用 p6spy 分析) org.apache.ibatis.logging.nologging.NoLoggingImpl |  | ||||||
|     # 默认日志输出 org.apache.ibatis.logging.slf4j.Slf4jImpl |  | ||||||
|     logImpl: org.apache.ibatis.logging.nologging.NoLoggingImpl |  | ||||||
|   global-config: |   global-config: | ||||||
|     # 是否打印 Logo banner |  | ||||||
|     banner: true |  | ||||||
|     dbConfig: |     dbConfig: | ||||||
|       # 主键类型 |       # 主键类型 | ||||||
|       # AUTO 自增 NONE 空 INPUT 用户输入 ASSIGN_ID 雪花 ASSIGN_UUID 唯一 UUID |       # AUTO 自增 NONE 空 INPUT 用户输入 ASSIGN_ID 雪花 ASSIGN_UUID 唯一 UUID | ||||||
|  |       # 如需改为自增 需要将数据库表全部设置为自增 | ||||||
|       idType: ASSIGN_ID |       idType: ASSIGN_ID | ||||||
|       # 逻辑已删除值 |  | ||||||
|       logicDeleteValue: 2 |  | ||||||
|       # 逻辑未删除值 |  | ||||||
|       logicNotDeleteValue: 0 |  | ||||||
|       # 字段验证策略之 insert,在 insert 的时候的字段验证策略 |  | ||||||
|       # IGNORED 忽略 NOT_NULL 非NULL NOT_EMPTY 非空 DEFAULT 默认 NEVER 不加入 SQL |  | ||||||
|       insertStrategy: NOT_NULL |  | ||||||
|       # 字段验证策略之 update,在 update 的时候的字段验证策略 |  | ||||||
|       updateStrategy: NOT_NULL |  | ||||||
|       # 字段验证策略之 select,在 select 的时候的字段验证策略既 wrapper 根据内部 entity 生成的 where 条件 |  | ||||||
|       where-strategy: NOT_NULL |  | ||||||
|  |  | ||||||
| # Swagger配置 | # 数据加密 | ||||||
| swagger: | mybatis-encryptor: | ||||||
|   # 是否开启swagger |   # 是否开启加密 | ||||||
|  |   enable: false | ||||||
|  |   # 默认加密算法 | ||||||
|  |   algorithm: BASE64 | ||||||
|  |   # 编码方式 BASE64/HEX。默认BASE64 | ||||||
|  |   encode: BASE64 | ||||||
|  |   # 安全秘钥 对称算法的秘钥 如:AES,SM4 | ||||||
|  |   password: | ||||||
|  |   # 公私钥 非对称算法的公私钥 如:SM2,RSA | ||||||
|  |   publicKey: | ||||||
|  |   privateKey: | ||||||
|  |  | ||||||
|  | # api接口加密 | ||||||
|  | api-decrypt: | ||||||
|  |   # 是否开启全局接口加密 | ||||||
|   enabled: true |   enabled: true | ||||||
|  |   # AES 加密头标识 | ||||||
|  |   headerFlag: encrypt-key | ||||||
|  |   # 响应加密公钥 非对称算法的公私钥 如:SM2,RSA 使用者请自行更换 | ||||||
|  |   # 对应前端解密私钥 MIIBVAIBADANBgkqhkiG9w0BAQEFAASCAT4wggE6AgEAAkEAmc3CuPiGL/LcIIm7zryCEIbl1SPzBkr75E2VMtxegyZ1lYRD+7TZGAPkvIsBcaMs6Nsy0L78n2qh+lIZMpLH8wIDAQABAkEAk82Mhz0tlv6IVCyIcw/s3f0E+WLmtPFyR9/WtV3Y5aaejUkU60JpX4m5xNR2VaqOLTZAYjW8Wy0aXr3zYIhhQQIhAMfqR9oFdYw1J9SsNc+CrhugAvKTi0+BF6VoL6psWhvbAiEAxPPNTmrkmrXwdm/pQQu3UOQmc2vCZ5tiKpW10CgJi8kCIFGkL6utxw93Ncj4exE/gPLvKcT+1Emnoox+O9kRXss5AiAMtYLJDaLEzPrAWcZeeSgSIzbL+ecokmFKSDDcRske6QIgSMkHedwND1olF8vlKsJUGK3BcdtM8w4Xq7BpSBwsloE= | ||||||
|  |   publicKey: MFwwDQYJKoZIhvcNAQEBBQADSwAwSAJBAJnNwrj4hi/y3CCJu868ghCG5dUj8wZK++RNlTLcXoMmdZWEQ/u02RgD5LyLAXGjLOjbMtC+/J9qofpSGTKSx/MCAwEAAQ== | ||||||
|  |   # 请求解密私钥 非对称算法的公私钥 如:SM2,RSA 使用者请自行更换 | ||||||
|  |   # 对应前端加密公钥 MFwwDQYJKoZIhvcNAQEBBQADSwAwSAJBAKoR8mX0rGKLqzcWmOzbfj64K8ZIgOdHnzkXSOVOZbFu/TJhZ7rFAN+eaGkl3C4buccQd/EjEsj9ir7ijT7h96MCAwEAAQ== | ||||||
|  |   privateKey: MIIBVAIBADANBgkqhkiG9w0BAQEFAASCAT4wggE6AgEAAkEAqhHyZfSsYourNxaY7Nt+PrgrxkiA50efORdI5U5lsW79MmFnusUA355oaSXcLhu5xxB38SMSyP2KvuKNPuH3owIDAQABAkAfoiLyL+Z4lf4Myxk6xUDgLaWGximj20CUf+5BKKnlrK+Ed8gAkM0HqoTt2UZwA5E2MzS4EI2gjfQhz5X28uqxAiEA3wNFxfrCZlSZHb0gn2zDpWowcSxQAgiCstxGUoOqlW8CIQDDOerGKH5OmCJ4Z21v+F25WaHYPxCFMvwxpcw99EcvDQIgIdhDTIqD2jfYjPTY8Jj3EDGPbH2HHuffvflECt3Ek60CIQCFRlCkHpi7hthhYhovyloRYsM+IS9h/0BzlEAuO0ktMQIgSPT3aFAgJYwKpqRYKlLDVcflZFCKY7u3UP8iWi1Qw0Y= | ||||||
|  |  | ||||||
|  | springdoc: | ||||||
|  |   api-docs: | ||||||
|  |     # 是否开启接口文档 | ||||||
|  |     enabled: true | ||||||
|  | #  swagger-ui: | ||||||
|  | #    # 持久化认证数据 | ||||||
|  | #    persistAuthorization: true | ||||||
|   info: |   info: | ||||||
|     # 标题 |     # 标题 | ||||||
|     title: '标题:${ruoyi.name}后台管理系统_接口文档' |     title: '标题:${ruoyi.name}多租户管理系统_接口文档' | ||||||
|     # 描述 |     # 描述 | ||||||
|     description: '描述:用于管理集团旗下公司的人员信息,具体包括XXX,XXX模块...' |     description: '描述:用于管理集团旗下公司的人员信息,具体包括XXX,XXX模块...' | ||||||
|     # 版本 |     # 版本 | ||||||
|     version: '版本号: ${ruoyi-vue-plus.version}' |     version: '版本号: ${ruoyi.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/dromara/RuoYi-Vue-Plus | ||||||
|   components: |   components: | ||||||
|     # 鉴权方式配置 |     # 鉴权方式配置 | ||||||
|     security-schemes: |     security-schemes: | ||||||
| @@ -203,19 +205,16 @@ swagger: | |||||||
|         type: APIKEY |         type: APIKEY | ||||||
|         in: HEADER |         in: HEADER | ||||||
|         name: ${sa-token.token-name} |         name: ${sa-token.token-name} | ||||||
|  |  | ||||||
| springdoc: |  | ||||||
|   swagger-ui: |  | ||||||
|     # 持久化认证数据 |  | ||||||
|     persistAuthorization: true |  | ||||||
|   #这里定义了两个分组,可定义多个,也可以不定义 |   #这里定义了两个分组,可定义多个,也可以不定义 | ||||||
|   group-configs: |   group-configs: | ||||||
|     - group: 1.演示模块 |     - group: 1.演示模块 | ||||||
|       packages-to-scan: com.ruoyi.demo |       packages-to-scan: org.dromara.demo | ||||||
|     - group: 2.系统模块 |     - group: 2.通用模块 | ||||||
|       packages-to-scan: com.ruoyi.web |       packages-to-scan: org.dromara.web | ||||||
|     - group: 3.代码生成模块 |     - group: 3.系统模块 | ||||||
|       packages-to-scan: com.ruoyi.generator |       packages-to-scan: org.dromara.system | ||||||
|  |     - group: 4.代码生成模块 | ||||||
|  |       packages-to-scan: org.dromara.generator | ||||||
|  |  | ||||||
| # 防止XSS攻击 | # 防止XSS攻击 | ||||||
| xss: | xss: | ||||||
| @@ -253,3 +252,12 @@ management: | |||||||
|       show-details: ALWAYS |       show-details: ALWAYS | ||||||
|     logfile: |     logfile: | ||||||
|       external-file: ./logs/sys-console.log |       external-file: ./logs/sys-console.log | ||||||
|  |  | ||||||
|  | --- # websocket | ||||||
|  | websocket: | ||||||
|  |   # 如果关闭 需要和前端开关一起关闭 | ||||||
|  |   enabled: true | ||||||
|  |   # 路径 | ||||||
|  |   path: /resource/websocket | ||||||
|  |   # 设置访问源地址 | ||||||
|  |   allowedOrigins: '*' | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| Application Version: ${ruoyi-vue-plus.version} | Application Version: ${revision} | ||||||
| Spring Boot Version: ${spring-boot.version} | Spring Boot Version: ${spring-boot.version} | ||||||
| __________            _____.___.__         ____   ____                     __________.__ | __________            _____.___.__         ____   ____                     __________.__ | ||||||
| \______   \__ __  ____\__  |   |__|        \   \ /   /_ __   ____          \______   \  |  __ __  ______ | \______   \__ __  ____\__  |   |__|        \   \ /   /_ __   ____          \______   \  |  __ __  ______ | ||||||
|   | |||||||
| @@ -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.email.not.blank=邮箱不能为空 | ||||||
| user.phonenumber.not.blank=用户手机号不能为空 | user.phonenumber.not.blank=用户手机号不能为空 | ||||||
| user.mobile.phone.number.not.valid=手机号格式错误 | user.mobile.phone.number.not.valid=手机号格式错误 | ||||||
| user.login.success=登录成功 | user.login.success=登录成功 | ||||||
| @@ -27,6 +28,10 @@ user.register.error=注册失败,请联系系统管理人员 | |||||||
| user.notfound=请重新登录 | user.notfound=请重新登录 | ||||||
| user.forcelogout=管理员强制退出,请重新登录 | user.forcelogout=管理员强制退出,请重新登录 | ||||||
| user.unknown.error=未知错误,请重新登录 | user.unknown.error=未知错误,请重新登录 | ||||||
|  | auth.grant.type.error=认证权限类型错误 | ||||||
|  | auth.grant.type.blocked=认证权限类型已禁用 | ||||||
|  | auth.grant.type.not.blank=认证权限类型不能为空 | ||||||
|  | auth.clientid.not.blank=认证客户端id不能为空 | ||||||
| ##文件上传消息 | ##文件上传消息 | ||||||
| upload.exceed.maxSize=上传的文件大小超出限制的文件大小!<br/>允许的文件最大大小是:{0}MB! | upload.exceed.maxSize=上传的文件大小超出限制的文件大小!<br/>允许的文件最大大小是:{0}MB! | ||||||
| upload.filename.exceed.length=上传的文件名最长{0}个字符 | upload.filename.exceed.length=上传的文件名最长{0}个字符 | ||||||
| @@ -42,4 +47,15 @@ rate.limiter.message=访问过于频繁,请稍候再试 | |||||||
| sms.code.not.blank=短信验证码不能为空 | sms.code.not.blank=短信验证码不能为空 | ||||||
| sms.code.retry.limit.count=短信验证码输入错误{0}次 | sms.code.retry.limit.count=短信验证码输入错误{0}次 | ||||||
| sms.code.retry.limit.exceed=短信验证码输入错误{0}次,帐户锁定{1}分钟 | sms.code.retry.limit.exceed=短信验证码输入错误{0}次,帐户锁定{1}分钟 | ||||||
| xcx.code.not.blank=小程序code不能为空 | email.code.not.blank=邮箱验证码不能为空 | ||||||
|  | email.code.retry.limit.count=邮箱验证码输入错误{0}次 | ||||||
|  | email.code.retry.limit.exceed=邮箱验证码输入错误{0}次,帐户锁定{1}分钟 | ||||||
|  | xcx.code.not.blank=小程序[code]不能为空 | ||||||
|  | social.source.not.blank=第三方登录平台[source]不能为空 | ||||||
|  | social.code.not.blank=第三方登录平台[code]不能为空 | ||||||
|  | social.state.not.blank=第三方登录平台[state]不能为空 | ||||||
|  | ##租户 | ||||||
|  | tenant.number.not.blank=租户编号不能为空 | ||||||
|  | tenant.not.exists=对不起, 您的租户不存在,请联系管理员 | ||||||
|  | tenant.blocked=对不起,您的租户已禁用,请联系管理员 | ||||||
|  | tenant.expired=对不起,您的租户已过期,请联系管理员 | ||||||
|   | |||||||
| @@ -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.email.not.blank=Mailbox cannot be blank | ||||||
| user.phonenumber.not.blank=Phone number cannot be blank | 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 | ||||||
| @@ -27,6 +28,10 @@ user.register.error=Register failed, please contact system administrator | |||||||
| user.notfound=Please login again | user.notfound=Please login again | ||||||
| user.forcelogout=The administrator is forced to exit,please login again | user.forcelogout=The administrator is forced to exit,please login again | ||||||
| user.unknown.error=Unknown error, please login again | user.unknown.error=Unknown error, please login again | ||||||
|  | auth.grant.type.error=Auth grant type error | ||||||
|  | auth.grant.type.blocked=Auth grant type disabled | ||||||
|  | auth.grant.type.not.blank=Auth grant type cannot be blank | ||||||
|  | auth.clientid.not.blank=Auth clientid cannot be blank | ||||||
| ##文件上传消息 | ##文件上传消息 | ||||||
| upload.exceed.maxSize=The uploaded file size exceeds the limit file size!<br/>the maximum allowed file size is:{0}MB! | upload.exceed.maxSize=The uploaded file size exceeds the limit file size!<br/>the maximum allowed file size is:{0}MB! | ||||||
| upload.filename.exceed.length=The maximum length of uploaded file name is {0} characters | upload.filename.exceed.length=The maximum length of uploaded file name is {0} characters | ||||||
| @@ -42,4 +47,15 @@ rate.limiter.message=Visit too frequently, please try again later | |||||||
| sms.code.not.blank=Sms code cannot be blank | sms.code.not.blank=Sms code cannot be blank | ||||||
| sms.code.retry.limit.count=Sms code input error {0} times | 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 | 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 | email.code.not.blank=Email code cannot be blank | ||||||
|  | email.code.retry.limit.count=Email code input error {0} times | ||||||
|  | email.code.retry.limit.exceed=Email code input error {0} times, account locked for {1} minutes | ||||||
|  | xcx.code.not.blank=Mini program [code] cannot be blank | ||||||
|  | social.source.not.blank=Social login platform [source] cannot be blank | ||||||
|  | social.code.not.blank=Social login platform [code] cannot be blank | ||||||
|  | social.state.not.blank=Social login platform [state] cannot be blank | ||||||
|  | ##租户 | ||||||
|  | tenant.number.not.blank=Tenant number cannot be blank | ||||||
|  | tenant.not.exists=Sorry, your tenant does not exist. Please contact the administrator | ||||||
|  | tenant.blocked=Sorry, your tenant is disabled. Please contact the administrator | ||||||
|  | tenant.expired=Sorry, your tenant has expired. Please contact the administrator. | ||||||
|   | |||||||
| @@ -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.email.not.blank=邮箱不能为空 | ||||||
| user.phonenumber.not.blank=用户手机号不能为空 | user.phonenumber.not.blank=用户手机号不能为空 | ||||||
| user.mobile.phone.number.not.valid=手机号格式错误 | user.mobile.phone.number.not.valid=手机号格式错误 | ||||||
| user.login.success=登录成功 | user.login.success=登录成功 | ||||||
| @@ -27,6 +28,10 @@ user.register.error=注册失败,请联系系统管理人员 | |||||||
| user.notfound=请重新登录 | user.notfound=请重新登录 | ||||||
| user.forcelogout=管理员强制退出,请重新登录 | user.forcelogout=管理员强制退出,请重新登录 | ||||||
| user.unknown.error=未知错误,请重新登录 | user.unknown.error=未知错误,请重新登录 | ||||||
|  | auth.grant.type.error=认证权限类型错误 | ||||||
|  | auth.grant.type.blocked=认证权限类型已禁用 | ||||||
|  | auth.grant.type.not.blank=认证权限类型不能为空 | ||||||
|  | auth.clientid.not.blank=认证客户端id不能为空 | ||||||
| ##文件上传消息 | ##文件上传消息 | ||||||
| upload.exceed.maxSize=上传的文件大小超出限制的文件大小!<br/>允许的文件最大大小是:{0}MB! | upload.exceed.maxSize=上传的文件大小超出限制的文件大小!<br/>允许的文件最大大小是:{0}MB! | ||||||
| upload.filename.exceed.length=上传的文件名最长{0}个字符 | upload.filename.exceed.length=上传的文件名最长{0}个字符 | ||||||
| @@ -42,4 +47,15 @@ rate.limiter.message=访问过于频繁,请稍候再试 | |||||||
| sms.code.not.blank=短信验证码不能为空 | sms.code.not.blank=短信验证码不能为空 | ||||||
| sms.code.retry.limit.count=短信验证码输入错误{0}次 | sms.code.retry.limit.count=短信验证码输入错误{0}次 | ||||||
| sms.code.retry.limit.exceed=短信验证码输入错误{0}次,帐户锁定{1}分钟 | sms.code.retry.limit.exceed=短信验证码输入错误{0}次,帐户锁定{1}分钟 | ||||||
| xcx.code.not.blank=小程序code不能为空 | email.code.not.blank=邮箱验证码不能为空 | ||||||
|  | email.code.retry.limit.count=邮箱验证码输入错误{0}次 | ||||||
|  | email.code.retry.limit.exceed=邮箱验证码输入错误{0}次,帐户锁定{1}分钟 | ||||||
|  | xcx.code.not.blank=小程序[code]不能为空 | ||||||
|  | social.source.not.blank=第三方登录平台[source]不能为空 | ||||||
|  | social.code.not.blank=第三方登录平台[code]不能为空 | ||||||
|  | social.state.not.blank=第三方登录平台[state]不能为空 | ||||||
|  | ##租户 | ||||||
|  | tenant.number.not.blank=租户编号不能为空 | ||||||
|  | tenant.not.exists=对不起, 您的租户不存在,请联系管理员 | ||||||
|  | tenant.blocked=对不起,您的租户已禁用,请联系管理员 | ||||||
|  | tenant.expired=对不起,您的租户已过期,请联系管理员 | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								ruoyi-admin/src/main/resources/ip2region.xdb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								ruoyi-admin/src/main/resources/ip2region.xdb
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -1,4 +1,4 @@ | |||||||
| package com.ruoyi.test; | package org.dromara.test; | ||||||
| 
 | 
 | ||||||
| import org.junit.jupiter.api.Assertions; | import org.junit.jupiter.api.Assertions; | ||||||
| import org.junit.jupiter.api.DisplayName; | import org.junit.jupiter.api.DisplayName; | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| package com.ruoyi.test; | package org.dromara.test; | ||||||
| 
 | 
 | ||||||
| import com.ruoyi.common.config.RuoYiConfig; | import org.dromara.common.core.config.RuoYiConfig; | ||||||
| import org.junit.jupiter.api.*; | import org.junit.jupiter.api.*; | ||||||
| import org.springframework.beans.factory.annotation.Autowired; | import org.springframework.beans.factory.annotation.Autowired; | ||||||
| import org.springframework.boot.test.context.SpringBootTest; | import org.springframework.boot.test.context.SpringBootTest; | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| package com.ruoyi.test; | package org.dromara.test; | ||||||
| 
 | 
 | ||||||
| import com.ruoyi.common.enums.UserType; | import org.dromara.common.core.enums.UserType; | ||||||
| import org.junit.jupiter.api.AfterEach; | import org.junit.jupiter.api.AfterEach; | ||||||
| import org.junit.jupiter.api.BeforeEach; | import org.junit.jupiter.api.BeforeEach; | ||||||
| import org.junit.jupiter.api.DisplayName; | import org.junit.jupiter.api.DisplayName; | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| package com.ruoyi.test; | package org.dromara.test; | ||||||
| 
 | 
 | ||||||
| import org.junit.jupiter.api.*; | import org.junit.jupiter.api.*; | ||||||
| import org.springframework.boot.test.context.SpringBootTest; | import org.springframework.boot.test.context.SpringBootTest; | ||||||
| @@ -4,155 +4,42 @@ | |||||||
|          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> |          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||||||
|     <parent> |     <parent> | ||||||
|         <artifactId>ruoyi-vue-plus</artifactId> |         <artifactId>ruoyi-vue-plus</artifactId> | ||||||
|         <groupId>com.ruoyi</groupId> |         <groupId>org.dromara</groupId> | ||||||
|         <version>4.5.0</version> |         <version>${revision}</version> | ||||||
|     </parent> |     </parent> | ||||||
|     <modelVersion>4.0.0</modelVersion> |     <modelVersion>4.0.0</modelVersion> | ||||||
|  |  | ||||||
|  |     <modules> | ||||||
|  |         <module>ruoyi-common-bom</module> | ||||||
|  |         <module>ruoyi-common-social</module> | ||||||
|  |         <module>ruoyi-common-core</module> | ||||||
|  |         <module>ruoyi-common-doc</module> | ||||||
|  |         <module>ruoyi-common-excel</module> | ||||||
|  |         <module>ruoyi-common-idempotent</module> | ||||||
|  |         <module>ruoyi-common-job</module> | ||||||
|  |         <module>ruoyi-common-log</module> | ||||||
|  |         <module>ruoyi-common-mail</module> | ||||||
|  |         <module>ruoyi-common-mybatis</module> | ||||||
|  |         <module>ruoyi-common-oss</module> | ||||||
|  |         <module>ruoyi-common-ratelimiter</module> | ||||||
|  |         <module>ruoyi-common-redis</module> | ||||||
|  |         <module>ruoyi-common-satoken</module> | ||||||
|  |         <module>ruoyi-common-security</module> | ||||||
|  |         <module>ruoyi-common-sms</module> | ||||||
|  |         <module>ruoyi-common-web</module> | ||||||
|  |         <module>ruoyi-common-translation</module> | ||||||
|  |         <module>ruoyi-common-sensitive</module> | ||||||
|  |         <module>ruoyi-common-json</module> | ||||||
|  |         <module>ruoyi-common-encrypt</module> | ||||||
|  |         <module>ruoyi-common-tenant</module> | ||||||
|  |         <module>ruoyi-common-websocket</module> | ||||||
|  |     </modules> | ||||||
|  |  | ||||||
|     <artifactId>ruoyi-common</artifactId> |     <artifactId>ruoyi-common</artifactId> | ||||||
|  |     <packaging>pom</packaging> | ||||||
|  |  | ||||||
|     <description> |     <description> | ||||||
|         common通用工具 |         common 通用模块 | ||||||
|     </description> |     </description> | ||||||
|  |  | ||||||
|     <dependencies> |  | ||||||
|  |  | ||||||
|         <!-- Spring框架基本的核心工具 --> |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>org.springframework</groupId> |  | ||||||
|             <artifactId>spring-context-support</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <!-- SpringWeb模块 --> |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>org.springframework</groupId> |  | ||||||
|             <artifactId>spring-web</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <!-- Sa-Token 权限认证, 在线文档:http://sa-token.dev33.cn/ --> |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>cn.dev33</groupId> |  | ||||||
|             <artifactId>sa-token-spring-boot-starter</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|         <!-- Sa-Token 整合 jwt --> |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>cn.dev33</groupId> |  | ||||||
|             <artifactId>sa-token-jwt</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <!-- 自定义验证注解 --> |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>org.springframework.boot</groupId> |  | ||||||
|             <artifactId>spring-boot-starter-validation</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <!--常用工具类 --> |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>org.apache.commons</groupId> |  | ||||||
|             <artifactId>commons-lang3</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <!-- JSON工具类 --> |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>com.fasterxml.jackson.core</groupId> |  | ||||||
|             <artifactId>jackson-databind</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>com.alibaba</groupId> |  | ||||||
|             <artifactId>easyexcel</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <!-- yml解析器 --> |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>org.yaml</groupId> |  | ||||||
|             <artifactId>snakeyaml</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <!-- servlet包 --> |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>javax.servlet</groupId> |  | ||||||
|             <artifactId>javax.servlet-api</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>com.baomidou</groupId> |  | ||||||
|             <artifactId>mybatis-plus-boot-starter</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <!-- dynamic-datasource 多数据源--> |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>com.baomidou</groupId> |  | ||||||
|             <artifactId>dynamic-datasource-spring-boot-starter</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>cn.hutool</groupId> |  | ||||||
|             <artifactId>hutool-core</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>cn.hutool</groupId> |  | ||||||
|             <artifactId>hutool-http</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>cn.hutool</groupId> |  | ||||||
|             <artifactId>hutool-captcha</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>cn.hutool</groupId> |  | ||||||
|             <artifactId>hutool-jwt</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>cn.hutool</groupId> |  | ||||||
|             <artifactId>hutool-extra</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>com.sun.mail</groupId> |  | ||||||
|             <artifactId>jakarta.mail</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>org.projectlombok</groupId> |  | ||||||
|             <artifactId>lombok</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>org.springdoc</groupId> |  | ||||||
|             <artifactId>springdoc-openapi-webmvc-core</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>org.springdoc</groupId> |  | ||||||
|             <artifactId>springdoc-openapi-javadoc</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <!--  自动生成YML配置关联JSON文件  --> |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>org.springframework.boot</groupId> |  | ||||||
|             <artifactId>spring-boot-configuration-processor</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <!--redisson--> |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>org.redisson</groupId> |  | ||||||
|             <artifactId>redisson-spring-boot-starter</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>org.redisson</groupId> |  | ||||||
|             <artifactId>redisson-spring-data-27</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|         <dependency> |  | ||||||
|             <groupId>com.baomidou</groupId> |  | ||||||
|             <artifactId>lock4j-redisson-spring-boot-starter</artifactId> |  | ||||||
|         </dependency> |  | ||||||
|  |  | ||||||
|     </dependencies> |  | ||||||
|  |  | ||||||
| </project> | </project> | ||||||
|   | |||||||
							
								
								
									
										178
									
								
								ruoyi-common/ruoyi-common-bom/pom.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										178
									
								
								ruoyi-common/ruoyi-common-bom/pom.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,178 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <project xmlns="http://maven.apache.org/POM/4.0.0" | ||||||
|  |          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" | ||||||
|  |          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||||||
|  |     <modelVersion>4.0.0</modelVersion> | ||||||
|  |  | ||||||
|  |     <groupId>org.dromara</groupId> | ||||||
|  |     <artifactId>ruoyi-common-bom</artifactId> | ||||||
|  |     <version>${revision}</version> | ||||||
|  |     <packaging>pom</packaging> | ||||||
|  |  | ||||||
|  |     <description> | ||||||
|  |         ruoyi-common-bom common依赖项 | ||||||
|  |     </description> | ||||||
|  |  | ||||||
|  |     <properties> | ||||||
|  |         <revision>5.1.2</revision> | ||||||
|  |     </properties> | ||||||
|  |  | ||||||
|  |     <dependencyManagement> | ||||||
|  |         <dependencies> | ||||||
|  |             <!-- 核心模块 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-core</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- 接口模块 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-doc</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- excel --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-excel</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- 幂等 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-idempotent</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- 调度模块 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-job</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- 日志记录 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-log</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- 邮件服务 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-mail</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- 数据库服务 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-mybatis</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- OSS --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-oss</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- 限流 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-ratelimiter</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- 缓存服务 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-redis</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- satoken --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-satoken</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- 安全模块 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-security</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- 短信模块 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-sms</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-social</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- web服务 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-web</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- 翻译模块 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-translation</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- 脱敏模块 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-sensitive</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- 序列化模块 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-json</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- 数据库加解密模块 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-encrypt</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- 租户模块 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-tenant</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |             <!-- WebSocket模块 --> | ||||||
|  |             <dependency> | ||||||
|  |                 <groupId>org.dromara</groupId> | ||||||
|  |                 <artifactId>ruoyi-common-websocket</artifactId> | ||||||
|  |                 <version>${revision}</version> | ||||||
|  |             </dependency> | ||||||
|  |  | ||||||
|  |         </dependencies> | ||||||
|  |     </dependencyManagement> | ||||||
|  |  | ||||||
|  | </project> | ||||||
							
								
								
									
										99
									
								
								ruoyi-common/ruoyi-common-core/pom.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								ruoyi-common/ruoyi-common-core/pom.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,99 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <project xmlns="http://maven.apache.org/POM/4.0.0" | ||||||
|  |          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" | ||||||
|  |          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||||||
|  |     <parent> | ||||||
|  |         <groupId>org.dromara</groupId> | ||||||
|  |         <artifactId>ruoyi-common</artifactId> | ||||||
|  |         <version>${revision}</version> | ||||||
|  |     </parent> | ||||||
|  |     <modelVersion>4.0.0</modelVersion> | ||||||
|  |  | ||||||
|  |     <artifactId>ruoyi-common-core</artifactId> | ||||||
|  |  | ||||||
|  |     <description> | ||||||
|  |         ruoyi-common-core 核心模块 | ||||||
|  |     </description> | ||||||
|  |  | ||||||
|  |     <dependencies> | ||||||
|  |         <!-- Spring框架基本的核心工具 --> | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>org.springframework</groupId> | ||||||
|  |             <artifactId>spring-context-support</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |         <!-- SpringWeb模块 --> | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>org.springframework</groupId> | ||||||
|  |             <artifactId>spring-web</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |         <!-- 自定义验证注解 --> | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>org.springframework.boot</groupId> | ||||||
|  |             <artifactId>spring-boot-starter-validation</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>org.springframework.boot</groupId> | ||||||
|  |             <artifactId>spring-boot-starter-aop</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |         <!--常用工具类 --> | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>org.apache.commons</groupId> | ||||||
|  |             <artifactId>commons-lang3</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |         <!-- servlet包 --> | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>jakarta.servlet</groupId> | ||||||
|  |             <artifactId>jakarta.servlet-api</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>cn.hutool</groupId> | ||||||
|  |             <artifactId>hutool-core</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>cn.hutool</groupId> | ||||||
|  |             <artifactId>hutool-http</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>cn.hutool</groupId> | ||||||
|  |             <artifactId>hutool-extra</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>org.projectlombok</groupId> | ||||||
|  |             <artifactId>lombok</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |         <!--  自动生成YML配置关联JSON文件  --> | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>org.springframework.boot</groupId> | ||||||
|  |             <artifactId>spring-boot-configuration-processor</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>org.springframework.boot</groupId> | ||||||
|  |             <artifactId>spring-boot-properties-migrator</artifactId> | ||||||
|  |             <scope>runtime</scope> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>io.github.linpeilie</groupId> | ||||||
|  |             <artifactId>mapstruct-plus-spring-boot-starter</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |         <!-- 离线IP地址定位库 --> | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>org.lionsoul</groupId> | ||||||
|  |             <artifactId>ip2region</artifactId> | ||||||
|  |         </dependency> | ||||||
|  |  | ||||||
|  |     </dependencies> | ||||||
|  |  | ||||||
|  | </project> | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| package com.ruoyi.framework.config; | package org.dromara.common.core.config; | ||||||
| 
 | 
 | ||||||
| import org.springframework.context.annotation.Configuration; | import org.springframework.boot.autoconfigure.AutoConfiguration; | ||||||
| import org.springframework.context.annotation.EnableAspectJAutoProxy; | import org.springframework.context.annotation.EnableAspectJAutoProxy; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
| @@ -8,7 +8,7 @@ import org.springframework.context.annotation.EnableAspectJAutoProxy; | |||||||
|  * |  * | ||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Configuration | @AutoConfiguration | ||||||
| // 表示通过aop框架暴露该代理对象,AopContext能够访问 | // 表示通过aop框架暴露该代理对象,AopContext能够访问 | ||||||
| @EnableAspectJAutoProxy(exposeProxy = true) | @EnableAspectJAutoProxy(exposeProxy = true) | ||||||
| public class ApplicationConfig { | public class ApplicationConfig { | ||||||
| @@ -1,17 +1,15 @@ | |||||||
| package com.ruoyi.framework.config; | package org.dromara.common.core.config; | ||||||
| 
 | 
 | ||||||
| import cn.hutool.core.util.ArrayUtil; | import cn.hutool.core.util.ArrayUtil; | ||||||
| import com.ruoyi.common.exception.ServiceException; | import org.dromara.common.core.exception.ServiceException; | ||||||
|  | import org.dromara.common.core.utils.SpringUtils; | ||||||
| import org.springframework.aop.interceptor.AsyncUncaughtExceptionHandler; | import org.springframework.aop.interceptor.AsyncUncaughtExceptionHandler; | ||||||
| import org.springframework.beans.factory.annotation.Autowired; | import org.springframework.boot.autoconfigure.AutoConfiguration; | ||||||
| import org.springframework.beans.factory.annotation.Qualifier; | import org.springframework.scheduling.annotation.AsyncConfigurer; | ||||||
| import org.springframework.context.annotation.Configuration; |  | ||||||
| import org.springframework.scheduling.annotation.AsyncConfigurerSupport; |  | ||||||
| import org.springframework.scheduling.annotation.EnableAsync; | import org.springframework.scheduling.annotation.EnableAsync; | ||||||
| 
 | 
 | ||||||
| import java.util.Arrays; | import java.util.Arrays; | ||||||
| import java.util.concurrent.Executor; | import java.util.concurrent.Executor; | ||||||
| import java.util.concurrent.ScheduledExecutorService; |  | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 异步配置 |  * 异步配置 | ||||||
| @@ -19,19 +17,15 @@ import java.util.concurrent.ScheduledExecutorService; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @EnableAsync(proxyTargetClass = true) | @EnableAsync(proxyTargetClass = true) | ||||||
| @Configuration | @AutoConfiguration | ||||||
| public class AsyncConfig extends AsyncConfigurerSupport { | public class AsyncConfig implements AsyncConfigurer { | ||||||
| 
 |  | ||||||
|     @Autowired |  | ||||||
|     @Qualifier("scheduledExecutorService") |  | ||||||
|     private ScheduledExecutorService scheduledExecutorService; |  | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * 自定义 @Async 注解使用系统线程池 |      * 自定义 @Async 注解使用系统线程池 | ||||||
|      */ |      */ | ||||||
|     @Override |     @Override | ||||||
|     public Executor getAsyncExecutor() { |     public Executor getAsyncExecutor() { | ||||||
|         return scheduledExecutorService; |         return SpringUtils.getBean("scheduledExecutorService"); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
| @@ -1,7 +1,6 @@ | |||||||
| package com.ruoyi.common.config; | package org.dromara.common.core.config; | ||||||
| 
 | 
 | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.Getter; |  | ||||||
| import org.springframework.boot.context.properties.ConfigurationProperties; | import org.springframework.boot.context.properties.ConfigurationProperties; | ||||||
| import org.springframework.stereotype.Component; | import org.springframework.stereotype.Component; | ||||||
| 
 | 
 | ||||||
| @@ -31,24 +30,4 @@ public class RuoYiConfig { | |||||||
|      */ |      */ | ||||||
|     private String copyrightYear; |     private String copyrightYear; | ||||||
| 
 | 
 | ||||||
|     /** |  | ||||||
|      * 实例演示开关 |  | ||||||
|      */ |  | ||||||
|     private boolean demoEnabled; |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * 缓存懒加载 |  | ||||||
|      */ |  | ||||||
|     private boolean cacheLazy; |  | ||||||
| 
 |  | ||||||
|     /** |  | ||||||
|      * 获取地址开关 |  | ||||||
|      */ |  | ||||||
|     @Getter |  | ||||||
|     private static boolean addressEnabled; |  | ||||||
| 
 |  | ||||||
|     public void setAddressEnabled(boolean addressEnabled) { |  | ||||||
|         RuoYiConfig.addressEnabled = addressEnabled; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| } | } | ||||||
| @@ -1,12 +1,14 @@ | |||||||
| package com.ruoyi.framework.config; | package org.dromara.common.core.config; | ||||||
| 
 | 
 | ||||||
| import com.ruoyi.common.utils.Threads; | import jakarta.annotation.PreDestroy; | ||||||
| import com.ruoyi.framework.config.properties.ThreadPoolProperties; | import lombok.extern.slf4j.Slf4j; | ||||||
| import org.apache.commons.lang3.concurrent.BasicThreadFactory; | import org.apache.commons.lang3.concurrent.BasicThreadFactory; | ||||||
| import org.springframework.beans.factory.annotation.Autowired; | import org.dromara.common.core.config.properties.ThreadPoolProperties; | ||||||
|  | import org.dromara.common.core.utils.Threads; | ||||||
|  | import org.springframework.boot.autoconfigure.AutoConfiguration; | ||||||
| import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; | import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; | ||||||
|  | import org.springframework.boot.context.properties.EnableConfigurationProperties; | ||||||
| import org.springframework.context.annotation.Bean; | import org.springframework.context.annotation.Bean; | ||||||
| import org.springframework.context.annotation.Configuration; |  | ||||||
| import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor; | import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor; | ||||||
| 
 | 
 | ||||||
| import java.util.concurrent.ScheduledExecutorService; | import java.util.concurrent.ScheduledExecutorService; | ||||||
| @@ -18,7 +20,9 @@ import java.util.concurrent.ThreadPoolExecutor; | |||||||
|  * |  * | ||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  **/ |  **/ | ||||||
| @Configuration | @Slf4j | ||||||
|  | @AutoConfiguration | ||||||
|  | @EnableConfigurationProperties(ThreadPoolProperties.class) | ||||||
| public class ThreadPoolConfig { | public class ThreadPoolConfig { | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
| @@ -26,12 +30,11 @@ public class ThreadPoolConfig { | |||||||
|      */ |      */ | ||||||
|     private final int core = Runtime.getRuntime().availableProcessors() + 1; |     private final int core = Runtime.getRuntime().availableProcessors() + 1; | ||||||
| 
 | 
 | ||||||
|     @Autowired |     private ScheduledExecutorService scheduledExecutorService; | ||||||
|     private ThreadPoolProperties threadPoolProperties; |  | ||||||
| 
 | 
 | ||||||
|     @Bean(name = "threadPoolTaskExecutor") |     @Bean(name = "threadPoolTaskExecutor") | ||||||
|     @ConditionalOnProperty(prefix = "thread-pool", name = "enabled", havingValue = "true") |     @ConditionalOnProperty(prefix = "thread-pool", name = "enabled", havingValue = "true") | ||||||
|     public ThreadPoolTaskExecutor threadPoolTaskExecutor() { |     public ThreadPoolTaskExecutor threadPoolTaskExecutor(ThreadPoolProperties threadPoolProperties) { | ||||||
|         ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor(); |         ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor(); | ||||||
|         executor.setCorePoolSize(core); |         executor.setCorePoolSize(core); | ||||||
|         executor.setMaxPoolSize(core * 2); |         executor.setMaxPoolSize(core * 2); | ||||||
| @@ -46,7 +49,7 @@ public class ThreadPoolConfig { | |||||||
|      */ |      */ | ||||||
|     @Bean(name = "scheduledExecutorService") |     @Bean(name = "scheduledExecutorService") | ||||||
|     protected ScheduledExecutorService scheduledExecutorService() { |     protected ScheduledExecutorService scheduledExecutorService() { | ||||||
|         return new ScheduledThreadPoolExecutor(core, |         ScheduledThreadPoolExecutor scheduledThreadPoolExecutor = new ScheduledThreadPoolExecutor(core, | ||||||
|             new BasicThreadFactory.Builder().namingPattern("schedule-pool-%d").daemon(true).build(), |             new BasicThreadFactory.Builder().namingPattern("schedule-pool-%d").daemon(true).build(), | ||||||
|             new ThreadPoolExecutor.CallerRunsPolicy()) { |             new ThreadPoolExecutor.CallerRunsPolicy()) { | ||||||
|             @Override |             @Override | ||||||
| @@ -55,5 +58,21 @@ public class ThreadPoolConfig { | |||||||
|                 Threads.printException(r, t); |                 Threads.printException(r, t); | ||||||
|             } |             } | ||||||
|         }; |         }; | ||||||
|  |         this.scheduledExecutorService = scheduledThreadPoolExecutor; | ||||||
|  |         return scheduledThreadPoolExecutor; | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 销毁事件 | ||||||
|  |      */ | ||||||
|  |     @PreDestroy | ||||||
|  |     public void destroy() { | ||||||
|  |         try { | ||||||
|  |             log.info("====关闭后台任务任务线程池===="); | ||||||
|  |             Threads.shutdownAndAwaitTermination(scheduledExecutorService); | ||||||
|  |         } catch (Exception e) { | ||||||
|  |             log.error(e.getMessage(), e); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
| } | } | ||||||
| @@ -0,0 +1,40 @@ | |||||||
|  | package org.dromara.common.core.config; | ||||||
|  |  | ||||||
|  | import jakarta.validation.Validator; | ||||||
|  | import org.hibernate.validator.HibernateValidator; | ||||||
|  | import org.springframework.boot.autoconfigure.AutoConfiguration; | ||||||
|  | import org.springframework.context.MessageSource; | ||||||
|  | import org.springframework.context.annotation.Bean; | ||||||
|  | import org.springframework.validation.beanvalidation.LocalValidatorFactoryBean; | ||||||
|  |  | ||||||
|  | import java.util.Properties; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 校验框架配置类 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | @AutoConfiguration | ||||||
|  | public class ValidatorConfig { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 配置校验框架 快速返回模式 | ||||||
|  |      */ | ||||||
|  |     @Bean | ||||||
|  |     public Validator validator(MessageSource messageSource) { | ||||||
|  |         try (LocalValidatorFactoryBean factoryBean = new LocalValidatorFactoryBean()) { | ||||||
|  |             // 国际化 | ||||||
|  |             factoryBean.setValidationMessageSource(messageSource); | ||||||
|  |             // 设置使用 HibernateValidator 校验器 | ||||||
|  |             factoryBean.setProviderClass(HibernateValidator.class); | ||||||
|  |             Properties properties = new Properties(); | ||||||
|  |             // 设置 快速异常返回 | ||||||
|  |             properties.setProperty("hibernate.validator.fail_fast", "true"); | ||||||
|  |             factoryBean.setValidationProperties(properties); | ||||||
|  |             // 加载配置 | ||||||
|  |             factoryBean.afterPropertiesSet(); | ||||||
|  |             return factoryBean.getValidator(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -1,8 +1,7 @@ | |||||||
| package com.ruoyi.framework.config.properties; | package org.dromara.common.core.config.properties; | ||||||
| 
 | 
 | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import org.springframework.boot.context.properties.ConfigurationProperties; | import org.springframework.boot.context.properties.ConfigurationProperties; | ||||||
| import org.springframework.stereotype.Component; |  | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 线程池 配置属性 |  * 线程池 配置属性 | ||||||
| @@ -10,7 +9,6 @@ import org.springframework.stereotype.Component; | |||||||
|  * @author Lion Li |  * @author Lion Li | ||||||
|  */ |  */ | ||||||
| @Data | @Data | ||||||
| @Component |  | ||||||
| @ConfigurationProperties(prefix = "thread-pool") | @ConfigurationProperties(prefix = "thread-pool") | ||||||
| public class ThreadPoolProperties { | public class ThreadPoolProperties { | ||||||
| 
 | 
 | ||||||
| @@ -0,0 +1,25 @@ | |||||||
|  | package org.dromara.common.core.constant; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 缓存的key 常量 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | public interface CacheConstants { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 在线用户 redis key | ||||||
|  |      */ | ||||||
|  |     String ONLINE_TOKEN_KEY = "online_tokens:"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 参数管理 cache key | ||||||
|  |      */ | ||||||
|  |     String SYS_CONFIG_KEY = "sys_config:"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 字典管理 cache key | ||||||
|  |      */ | ||||||
|  |     String SYS_DICT_KEY = "sys_dict:"; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| package com.ruoyi.common.constant; | package org.dromara.common.core.constant; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 缓存组名称常量 |  * 缓存组名称常量 | ||||||
| @@ -30,6 +30,26 @@ public interface CacheNames { | |||||||
|      */ |      */ | ||||||
|     String SYS_DICT = "sys_dict"; |     String SYS_DICT = "sys_dict"; | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * 租户 | ||||||
|  |      */ | ||||||
|  |     String SYS_TENANT = GlobalConstants.GLOBAL_REDIS_KEY + "sys_tenant#30d"; | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 用户账户 | ||||||
|  |      */ | ||||||
|  |     String SYS_USER_NAME = "sys_user_name#30d"; | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 用户名称 | ||||||
|  |      */ | ||||||
|  |     String SYS_NICKNAME = "sys_nickname#30d"; | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 部门 | ||||||
|  |      */ | ||||||
|  |     String SYS_DEPT = "sys_dept#30d"; | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * OSS内容 |      * OSS内容 | ||||||
|      */ |      */ | ||||||
| @@ -38,7 +58,7 @@ public interface CacheNames { | |||||||
|     /** |     /** | ||||||
|      * OSS配置 |      * OSS配置 | ||||||
|      */ |      */ | ||||||
|     String SYS_OSS_CONFIG = "sys_oss_config"; |     String SYS_OSS_CONFIG = GlobalConstants.GLOBAL_REDIS_KEY + "sys_oss_config"; | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * 在线用户 |      * 在线用户 | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| package com.ruoyi.common.constant; | package org.dromara.common.core.constant; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 通用常量信息 |  * 通用常量信息 | ||||||
| @@ -72,5 +72,10 @@ public interface Constants { | |||||||
|      */ |      */ | ||||||
|     String TOKEN = "token"; |     String TOKEN = "token"; | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * 顶级部门id | ||||||
|  |      */ | ||||||
|  |     Long TOP_PARENT_ID = 0L; | ||||||
|  | 
 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @@ -0,0 +1,39 @@ | |||||||
|  | package org.dromara.common.core.constant; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 全局的key常量 (业务无关的key) | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | public interface GlobalConstants { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 全局 redis key (业务无关的key) | ||||||
|  |      */ | ||||||
|  |     String GLOBAL_REDIS_KEY = "global:"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 验证码 redis key | ||||||
|  |      */ | ||||||
|  |     String CAPTCHA_CODE_KEY = GLOBAL_REDIS_KEY + "captcha_codes:"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 防重提交 redis key | ||||||
|  |      */ | ||||||
|  |     String REPEAT_SUBMIT_KEY = GLOBAL_REDIS_KEY + "repeat_submit:"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 限流 redis key | ||||||
|  |      */ | ||||||
|  |     String RATE_LIMIT_KEY = GLOBAL_REDIS_KEY + "rate_limit:"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 登录账户密码错误次数 redis key | ||||||
|  |      */ | ||||||
|  |     String PWD_ERR_CNT_KEY = GLOBAL_REDIS_KEY + "pwd_err_cnt:"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 三方认证 redis key | ||||||
|  |      */ | ||||||
|  |     String SOCIAL_AUTH_CODE_KEY = GLOBAL_REDIS_KEY + "social_auth_codes:"; | ||||||
|  | } | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| package com.ruoyi.common.constant; | package org.dromara.common.core.constant; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 返回状态码 |  * 返回状态码 | ||||||
| @@ -0,0 +1,45 @@ | |||||||
|  | package org.dromara.common.core.constant; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 租户常量信息 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | public interface TenantConstants { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 租户正常状态 | ||||||
|  |      */ | ||||||
|  |     String NORMAL = "0"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 租户封禁状态 | ||||||
|  |      */ | ||||||
|  |     String DISABLE = "1"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 超级管理员ID | ||||||
|  |      */ | ||||||
|  |     Long SUPER_ADMIN_ID = 1L; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 超级管理员角色 roleKey | ||||||
|  |      */ | ||||||
|  |     String SUPER_ADMIN_ROLE_KEY = "superadmin"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 租户管理员角色 roleKey | ||||||
|  |      */ | ||||||
|  |     String TENANT_ADMIN_ROLE_KEY = "admin"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 租户管理员角色名称 | ||||||
|  |      */ | ||||||
|  |     String TENANT_ADMIN_ROLE_NAME = "管理员"; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 默认租户ID | ||||||
|  |      */ | ||||||
|  |     String DEFAULT_TENANT_ID = "000000"; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| package com.ruoyi.common.constant; | package org.dromara.common.core.constant; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 用户常量信息 |  * 用户常量信息 | ||||||
| @@ -52,6 +52,16 @@ public interface UserConstants { | |||||||
|      */ |      */ | ||||||
|     String DEPT_DISABLE = "1"; |     String DEPT_DISABLE = "1"; | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * 岗位正常状态 | ||||||
|  |      */ | ||||||
|  |     String POST_NORMAL = "0"; | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 岗位停用状态 | ||||||
|  |      */ | ||||||
|  |     String POST_DISABLE = "1"; | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * 字典正常状态 |      * 字典正常状态 | ||||||
|      */ |      */ | ||||||
| @@ -112,12 +122,6 @@ public interface UserConstants { | |||||||
|      */ |      */ | ||||||
|     String INNER_LINK = "InnerLink"; |     String INNER_LINK = "InnerLink"; | ||||||
| 
 | 
 | ||||||
|     /** |  | ||||||
|      * 校验返回结果码 |  | ||||||
|      */ |  | ||||||
|     String UNIQUE = "0"; |  | ||||||
|     String NOT_UNIQUE = "1"; |  | ||||||
| 
 |  | ||||||
|     /** |     /** | ||||||
|      * 用户名长度限制 |      * 用户名长度限制 | ||||||
|      */ |      */ | ||||||
| @@ -131,8 +135,8 @@ public interface UserConstants { | |||||||
|     int PASSWORD_MAX_LENGTH = 20; |     int PASSWORD_MAX_LENGTH = 20; | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * 管理员ID |      * 超级管理员ID | ||||||
|      */ |      */ | ||||||
|     Long ADMIN_ID = 1L; |     Long SUPER_ADMIN_ID = 1L; | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
| @@ -1,9 +1,10 @@ | |||||||
| package com.ruoyi.common.core.domain; | package org.dromara.common.core.domain; | ||||||
| 
 | 
 | ||||||
| import com.ruoyi.common.constant.HttpStatus; | import org.dromara.common.core.constant.HttpStatus; | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
| 
 | 
 | ||||||
|  | import java.io.Serial; | ||||||
| import java.io.Serializable; | import java.io.Serializable; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
| @@ -14,6 +15,8 @@ import java.io.Serializable; | |||||||
| @Data | @Data | ||||||
| @NoArgsConstructor | @NoArgsConstructor | ||||||
| public class R<T> implements Serializable { | public class R<T> implements Serializable { | ||||||
|  | 
 | ||||||
|  |     @Serial | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| package com.ruoyi.common.core.domain.dto; | package org.dromara.common.core.domain.dto; | ||||||
| 
 | 
 | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
| @@ -1,8 +1,9 @@ | |||||||
| package com.ruoyi.common.core.domain.dto; | package org.dromara.common.core.domain.dto; | ||||||
| 
 | 
 | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
| 
 | 
 | ||||||
|  | import java.io.Serial; | ||||||
| import java.io.Serializable; | import java.io.Serializable; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
| @@ -15,6 +16,7 @@ import java.io.Serializable; | |||||||
| @NoArgsConstructor | @NoArgsConstructor | ||||||
| public class UserOnlineDTO implements Serializable { | public class UserOnlineDTO implements Serializable { | ||||||
| 
 | 
 | ||||||
|  |     @Serial | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
| @@ -32,6 +34,16 @@ public class UserOnlineDTO implements Serializable { | |||||||
|      */ |      */ | ||||||
|     private String userName; |     private String userName; | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * 客户端 | ||||||
|  |      */ | ||||||
|  |     private String clientKey; | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 设备类型 | ||||||
|  |      */ | ||||||
|  |     private String deviceType; | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * 登录IP地址 |      * 登录IP地址 | ||||||
|      */ |      */ | ||||||
| @@ -0,0 +1,31 @@ | |||||||
|  | package org.dromara.common.core.domain.model; | ||||||
|  |  | ||||||
|  | import jakarta.validation.constraints.Email; | ||||||
|  | import jakarta.validation.constraints.NotBlank; | ||||||
|  | import lombok.Data; | ||||||
|  | import lombok.EqualsAndHashCode; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 邮件登录对象 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | @Data | ||||||
|  | @EqualsAndHashCode(callSuper = true) | ||||||
|  | public class EmailLoginBody extends LoginBody { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 邮箱 | ||||||
|  |      */ | ||||||
|  |     @NotBlank(message = "{user.email.not.blank}") | ||||||
|  |     @Email(message = "{user.email.not.valid}") | ||||||
|  |     private String email; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 邮箱code | ||||||
|  |      */ | ||||||
|  |     @NotBlank(message = "{email.code.not.blank}") | ||||||
|  |     private String emailCode; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,48 @@ | |||||||
|  | package org.dromara.common.core.domain.model; | ||||||
|  |  | ||||||
|  | import jakarta.validation.constraints.NotBlank; | ||||||
|  | import lombok.Data; | ||||||
|  |  | ||||||
|  | import java.io.Serial; | ||||||
|  | import java.io.Serializable; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 用户登录对象 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | @Data | ||||||
|  | public class LoginBody implements Serializable { | ||||||
|  |  | ||||||
|  |     @Serial | ||||||
|  |     private static final long serialVersionUID = 1L; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 客户端id | ||||||
|  |      */ | ||||||
|  |     @NotBlank(message = "{auth.clientid.not.blank}") | ||||||
|  |     private String clientId; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 授权类型 | ||||||
|  |      */ | ||||||
|  |     @NotBlank(message = "{auth.grant.type.not.blank}") | ||||||
|  |     private String grantType; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 租户ID | ||||||
|  |      */ | ||||||
|  |     private String tenantId; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 验证码 | ||||||
|  |      */ | ||||||
|  |     private String code; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 唯一标识 | ||||||
|  |      */ | ||||||
|  |     private String uuid; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -1,10 +1,10 @@ | |||||||
| package com.ruoyi.common.core.domain.model; | package org.dromara.common.core.domain.model; | ||||||
| 
 | 
 | ||||||
| import com.ruoyi.common.core.domain.dto.RoleDTO; | import org.dromara.common.core.domain.dto.RoleDTO; | ||||||
| import com.ruoyi.common.helper.LoginHelper; |  | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
| 
 | 
 | ||||||
|  | import java.io.Serial; | ||||||
| import java.io.Serializable; | import java.io.Serializable; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Set; | import java.util.Set; | ||||||
| @@ -19,8 +19,14 @@ import java.util.Set; | |||||||
| @NoArgsConstructor | @NoArgsConstructor | ||||||
| public class LoginUser implements Serializable { | public class LoginUser implements Serializable { | ||||||
| 
 | 
 | ||||||
|  |     @Serial | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * 租户ID | ||||||
|  |      */ | ||||||
|  |     private String tenantId; | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * 用户ID |      * 用户ID | ||||||
|      */ |      */ | ||||||
| @@ -91,6 +97,11 @@ public class LoginUser implements Serializable { | |||||||
|      */ |      */ | ||||||
|     private String username; |     private String username; | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * 用户昵称 | ||||||
|  |      */ | ||||||
|  |     private String nickname; | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * 角色对象 |      * 角色对象 | ||||||
|      */ |      */ | ||||||
| @@ -101,6 +112,16 @@ public class LoginUser implements Serializable { | |||||||
|      */ |      */ | ||||||
|     private Long roleId; |     private Long roleId; | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * 客户端 | ||||||
|  |      */ | ||||||
|  |     private String clientKey; | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 设备类型 | ||||||
|  |      */ | ||||||
|  |     private String deviceType; | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * 获取登录id |      * 获取登录id | ||||||
|      */ |      */ | ||||||
| @@ -111,7 +132,7 @@ public class LoginUser implements Serializable { | |||||||
|         if (userId == null) { |         if (userId == null) { | ||||||
|             throw new IllegalArgumentException("用户ID不能为空"); |             throw new IllegalArgumentException("用户ID不能为空"); | ||||||
|         } |         } | ||||||
|         return userType + LoginHelper.JOIN_CODE + userId; |         return userType + ":" + userId; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
| @@ -0,0 +1,33 @@ | |||||||
|  | package org.dromara.common.core.domain.model; | ||||||
|  |  | ||||||
|  | import jakarta.validation.constraints.NotBlank; | ||||||
|  | import lombok.Data; | ||||||
|  | import lombok.EqualsAndHashCode; | ||||||
|  | import org.hibernate.validator.constraints.Length; | ||||||
|  |  | ||||||
|  | import static org.dromara.common.core.constant.UserConstants.*; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 密码登录对象 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | @Data | ||||||
|  | @EqualsAndHashCode(callSuper = true) | ||||||
|  | public class PasswordLoginBody extends LoginBody { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 用户名 | ||||||
|  |      */ | ||||||
|  |     @NotBlank(message = "{user.username.not.blank}") | ||||||
|  |     @Length(min = USERNAME_MIN_LENGTH, max = USERNAME_MAX_LENGTH, message = "{user.username.length.valid}") | ||||||
|  |     private String username; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 用户密码 | ||||||
|  |      */ | ||||||
|  |     @NotBlank(message = "{user.password.not.blank}") | ||||||
|  |     @Length(min = PASSWORD_MIN_LENGTH, max = PASSWORD_MAX_LENGTH, message = "{user.password.length.valid}") | ||||||
|  |     private String password; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,35 @@ | |||||||
|  | package org.dromara.common.core.domain.model; | ||||||
|  |  | ||||||
|  | import jakarta.validation.constraints.NotBlank; | ||||||
|  | import lombok.Data; | ||||||
|  | import lombok.EqualsAndHashCode; | ||||||
|  | import org.hibernate.validator.constraints.Length; | ||||||
|  |  | ||||||
|  | import static org.dromara.common.core.constant.UserConstants.*; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 用户注册对象 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | @Data | ||||||
|  | @EqualsAndHashCode(callSuper = true) | ||||||
|  | public class RegisterBody extends LoginBody { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 用户名 | ||||||
|  |      */ | ||||||
|  |     @NotBlank(message = "{user.username.not.blank}") | ||||||
|  |     @Length(min = USERNAME_MIN_LENGTH, max = USERNAME_MAX_LENGTH, message = "{user.username.length.valid}") | ||||||
|  |     private String username; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 用户密码 | ||||||
|  |      */ | ||||||
|  |     @NotBlank(message = "{user.password.not.blank}") | ||||||
|  |     @Length(min = PASSWORD_MIN_LENGTH, max = PASSWORD_MAX_LENGTH, message = "{user.password.length.valid}") | ||||||
|  |     private String password; | ||||||
|  |  | ||||||
|  |     private String userType; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -1,8 +1,8 @@ | |||||||
| package com.ruoyi.common.core.domain.model; | package org.dromara.common.core.domain.model; | ||||||
| 
 | 
 | ||||||
|  | import jakarta.validation.constraints.NotBlank; | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| 
 | import lombok.EqualsAndHashCode; | ||||||
| import javax.validation.constraints.NotBlank; |  | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 短信登录对象 |  * 短信登录对象 | ||||||
| @@ -11,16 +11,17 @@ import javax.validation.constraints.NotBlank; | |||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| @Data | @Data | ||||||
| public class SmsLoginBody { | @EqualsAndHashCode(callSuper = true) | ||||||
|  | public class SmsLoginBody extends LoginBody { | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * 用户名 |      * 手机号 | ||||||
|      */ |      */ | ||||||
|     @NotBlank(message = "{user.phonenumber.not.blank}") |     @NotBlank(message = "{user.phonenumber.not.blank}") | ||||||
|     private String phonenumber; |     private String phonenumber; | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * 用户密码 |      * 短信code | ||||||
|      */ |      */ | ||||||
|     @NotBlank(message = "{sms.code.not.blank}") |     @NotBlank(message = "{sms.code.not.blank}") | ||||||
|     private String smsCode; |     private String smsCode; | ||||||
| @@ -0,0 +1,35 @@ | |||||||
|  | package org.dromara.common.core.domain.model; | ||||||
|  |  | ||||||
|  | import jakarta.validation.constraints.NotBlank; | ||||||
|  | import lombok.Data; | ||||||
|  | import lombok.EqualsAndHashCode; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 三方登录对象 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | @Data | ||||||
|  | @EqualsAndHashCode(callSuper = true) | ||||||
|  | public class SocialLoginBody extends LoginBody { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 第三方登录平台 | ||||||
|  |      */ | ||||||
|  |     @NotBlank(message = "{social.source.not.blank}") | ||||||
|  |     private String source; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 第三方登录code | ||||||
|  |      */ | ||||||
|  |     @NotBlank(message = "{social.code.not.blank}") | ||||||
|  |     private String socialCode; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 第三方登录socialState | ||||||
|  |      */ | ||||||
|  |     @NotBlank(message = "{social.state.not.blank}") | ||||||
|  |     private String socialState; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,28 @@ | |||||||
|  | package org.dromara.common.core.domain.model; | ||||||
|  |  | ||||||
|  | import jakarta.validation.constraints.NotBlank; | ||||||
|  | import lombok.Data; | ||||||
|  | import lombok.EqualsAndHashCode; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 三方登录对象 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | @Data | ||||||
|  | @EqualsAndHashCode(callSuper = true) | ||||||
|  | public class XcxLoginBody extends LoginBody { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 小程序id(多个小程序时使用) | ||||||
|  |      */ | ||||||
|  |     private String appid; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 小程序code | ||||||
|  |      */ | ||||||
|  |     @NotBlank(message = "{xcx.code.not.blank}") | ||||||
|  |     private String xcxCode; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -1,9 +1,11 @@ | |||||||
| package com.ruoyi.common.core.domain.model; | package org.dromara.common.core.domain.model; | ||||||
| 
 | 
 | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.EqualsAndHashCode; | import lombok.EqualsAndHashCode; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
| 
 | 
 | ||||||
|  | import java.io.Serial; | ||||||
|  | 
 | ||||||
| /** | /** | ||||||
|  * 小程序登录用户身份权限 |  * 小程序登录用户身份权限 | ||||||
|  * |  * | ||||||
| @@ -14,6 +16,7 @@ import lombok.NoArgsConstructor; | |||||||
| @NoArgsConstructor | @NoArgsConstructor | ||||||
| public class XcxLoginUser extends LoginUser { | public class XcxLoginUser extends LoginUser { | ||||||
| 
 | 
 | ||||||
|  |     @Serial | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| package com.ruoyi.common.enums; | package org.dromara.common.core.enums; | ||||||
| 
 | 
 | ||||||
| import lombok.AllArgsConstructor; | import lombok.AllArgsConstructor; | ||||||
| import lombok.Getter; | import lombok.Getter; | ||||||
| @@ -26,7 +26,12 @@ public enum DeviceType { | |||||||
|     /** |     /** | ||||||
|      * 小程序端 |      * 小程序端 | ||||||
|      */ |      */ | ||||||
|     XCX("xcx"); |     XCX("xcx"), | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * social第三方端 | ||||||
|  |      */ | ||||||
|  |     SOCIAL("social"); | ||||||
| 
 | 
 | ||||||
|     private final String device; |     private final String device; | ||||||
| } | } | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| package com.ruoyi.common.enums; | package org.dromara.common.core.enums; | ||||||
| 
 | 
 | ||||||
| import lombok.AllArgsConstructor; | import lombok.AllArgsConstructor; | ||||||
| import lombok.Getter; | import lombok.Getter; | ||||||
| @@ -22,6 +22,11 @@ public enum LoginType { | |||||||
|      */ |      */ | ||||||
|     SMS("sms.code.retry.limit.exceed", "sms.code.retry.limit.count"), |     SMS("sms.code.retry.limit.exceed", "sms.code.retry.limit.count"), | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * 邮箱登录 | ||||||
|  |      */ | ||||||
|  |     EMAIL("email.code.retry.limit.exceed", "email.code.retry.limit.count"), | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * 小程序登录 |      * 小程序登录 | ||||||
|      */ |      */ | ||||||
| @@ -0,0 +1,30 @@ | |||||||
|  | package org.dromara.common.core.enums; | ||||||
|  |  | ||||||
|  | import lombok.AllArgsConstructor; | ||||||
|  | import lombok.Getter; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 用户状态 | ||||||
|  |  * | ||||||
|  |  * @author LionLi | ||||||
|  |  */ | ||||||
|  | @Getter | ||||||
|  | @AllArgsConstructor | ||||||
|  | public enum TenantStatus { | ||||||
|  |     /** | ||||||
|  |      * 正常 | ||||||
|  |      */ | ||||||
|  |     OK("0", "正常"), | ||||||
|  |     /** | ||||||
|  |      * 停用 | ||||||
|  |      */ | ||||||
|  |     DISABLE("1", "停用"), | ||||||
|  |     /** | ||||||
|  |      * 删除 | ||||||
|  |      */ | ||||||
|  |     DELETED("2", "删除"); | ||||||
|  |  | ||||||
|  |     private final String code; | ||||||
|  |     private final String info; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,30 @@ | |||||||
|  | package org.dromara.common.core.enums; | ||||||
|  |  | ||||||
|  | import lombok.AllArgsConstructor; | ||||||
|  | import lombok.Getter; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 用户状态 | ||||||
|  |  * | ||||||
|  |  * @author ruoyi | ||||||
|  |  */ | ||||||
|  | @Getter | ||||||
|  | @AllArgsConstructor | ||||||
|  | public enum UserStatus { | ||||||
|  |     /** | ||||||
|  |      * 正常 | ||||||
|  |      */ | ||||||
|  |     OK("0", "正常"), | ||||||
|  |     /** | ||||||
|  |      * 停用 | ||||||
|  |      */ | ||||||
|  |     DISABLE("1", "停用"), | ||||||
|  |     /** | ||||||
|  |      * 删除 | ||||||
|  |      */ | ||||||
|  |     DELETED("2", "删除"); | ||||||
|  |  | ||||||
|  |     private final String code; | ||||||
|  |     private final String info; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| package com.ruoyi.common.enums; | package org.dromara.common.core.enums; | ||||||
| 
 | 
 | ||||||
| import com.ruoyi.common.utils.StringUtils; | import org.dromara.common.core.utils.StringUtils; | ||||||
| import lombok.AllArgsConstructor; | import lombok.AllArgsConstructor; | ||||||
| import lombok.Getter; | import lombok.Getter; | ||||||
| 
 | 
 | ||||||
| @@ -1,11 +1,24 @@ | |||||||
| package com.ruoyi.common.exception; | package org.dromara.common.core.exception; | ||||||
|  | 
 | ||||||
|  | import lombok.AllArgsConstructor; | ||||||
|  | import lombok.Data; | ||||||
|  | import lombok.EqualsAndHashCode; | ||||||
|  | import lombok.NoArgsConstructor; | ||||||
|  | 
 | ||||||
|  | import java.io.Serial; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 业务异常 |  * 业务异常 | ||||||
|  * |  * | ||||||
|  * @author ruoyi |  * @author ruoyi | ||||||
|  */ |  */ | ||||||
|  | @Data | ||||||
|  | @EqualsAndHashCode(callSuper = true) | ||||||
|  | @NoArgsConstructor | ||||||
|  | @AllArgsConstructor | ||||||
| public final class ServiceException extends RuntimeException { | public final class ServiceException extends RuntimeException { | ||||||
|  | 
 | ||||||
|  |     @Serial | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
| @@ -20,17 +33,9 @@ public final class ServiceException extends RuntimeException { | |||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * 错误明细,内部调试错误 |      * 错误明细,内部调试错误 | ||||||
|      * <p> |  | ||||||
|      * 和 {@link CommonResult#getDetailMessage()} 一致的设计 |  | ||||||
|      */ |      */ | ||||||
|     private String detailMessage; |     private String detailMessage; | ||||||
| 
 | 
 | ||||||
|     /** |  | ||||||
|      * 空构造方法,避免反序列化问题 |  | ||||||
|      */ |  | ||||||
|     public ServiceException() { |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public ServiceException(String message) { |     public ServiceException(String message) { | ||||||
|         this.message = message; |         this.message = message; | ||||||
|     } |     } | ||||||
| @@ -1,11 +1,14 @@ | |||||||
| package com.ruoyi.common.exception.base; | package org.dromara.common.core.exception.base; | ||||||
| 
 | 
 | ||||||
| import com.ruoyi.common.utils.MessageUtils; | import lombok.AllArgsConstructor; | ||||||
| import com.ruoyi.common.utils.StringUtils; | import org.dromara.common.core.utils.MessageUtils; | ||||||
|  | import org.dromara.common.core.utils.StringUtils; | ||||||
| import lombok.Data; | import lombok.Data; | ||||||
| import lombok.EqualsAndHashCode; | import lombok.EqualsAndHashCode; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
| 
 | 
 | ||||||
|  | import java.io.Serial; | ||||||
|  | 
 | ||||||
| /** | /** | ||||||
|  * 基础异常 |  * 基础异常 | ||||||
|  * |  * | ||||||
| @@ -14,7 +17,10 @@ import lombok.NoArgsConstructor; | |||||||
| @Data | @Data | ||||||
| @EqualsAndHashCode(callSuper = true) | @EqualsAndHashCode(callSuper = true) | ||||||
| @NoArgsConstructor | @NoArgsConstructor | ||||||
|  | @AllArgsConstructor | ||||||
| public class BaseException extends RuntimeException { | public class BaseException extends RuntimeException { | ||||||
|  | 
 | ||||||
|  |     @Serial | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
| @@ -37,13 +43,6 @@ public class BaseException extends RuntimeException { | |||||||
|      */ |      */ | ||||||
|     private String defaultMessage; |     private String defaultMessage; | ||||||
| 
 | 
 | ||||||
|     public BaseException(String module, String code, Object[] args, String defaultMessage) { |  | ||||||
|         this.module = module; |  | ||||||
|         this.code = code; |  | ||||||
|         this.args = args; |  | ||||||
|         this.defaultMessage = defaultMessage; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public BaseException(String module, String code, Object[] args) { |     public BaseException(String module, String code, Object[] args) { | ||||||
|         this(module, code, args, null); |         this(module, code, args, null); | ||||||
|     } |     } | ||||||
| @@ -1,6 +1,8 @@ | |||||||
| package com.ruoyi.common.exception.file; | package org.dromara.common.core.exception.file; | ||||||
| 
 | 
 | ||||||
| import com.ruoyi.common.exception.base.BaseException; | import org.dromara.common.core.exception.base.BaseException; | ||||||
|  | 
 | ||||||
|  | import java.io.Serial; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 文件信息异常类 |  * 文件信息异常类 | ||||||
| @@ -8,6 +10,8 @@ import com.ruoyi.common.exception.base.BaseException; | |||||||
|  * @author ruoyi |  * @author ruoyi | ||||||
|  */ |  */ | ||||||
| public class FileException extends BaseException { | public class FileException extends BaseException { | ||||||
|  | 
 | ||||||
|  |     @Serial | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
| 
 | 
 | ||||||
|     public FileException(String code, Object[] args) { |     public FileException(String code, Object[] args) { | ||||||
| @@ -1,4 +1,6 @@ | |||||||
| package com.ruoyi.common.exception.file; | package org.dromara.common.core.exception.file; | ||||||
|  | 
 | ||||||
|  | import java.io.Serial; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 文件名称超长限制异常类 |  * 文件名称超长限制异常类 | ||||||
| @@ -6,6 +8,8 @@ package com.ruoyi.common.exception.file; | |||||||
|  * @author ruoyi |  * @author ruoyi | ||||||
|  */ |  */ | ||||||
| public class FileNameLengthLimitExceededException extends FileException { | public class FileNameLengthLimitExceededException extends FileException { | ||||||
|  | 
 | ||||||
|  |     @Serial | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
| 
 | 
 | ||||||
|     public FileNameLengthLimitExceededException(int defaultFileNameLength) { |     public FileNameLengthLimitExceededException(int defaultFileNameLength) { | ||||||
| @@ -1,4 +1,6 @@ | |||||||
| package com.ruoyi.common.exception.file; | package org.dromara.common.core.exception.file; | ||||||
|  | 
 | ||||||
|  | import java.io.Serial; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 文件名大小限制异常类 |  * 文件名大小限制异常类 | ||||||
| @@ -6,6 +8,8 @@ package com.ruoyi.common.exception.file; | |||||||
|  * @author ruoyi |  * @author ruoyi | ||||||
|  */ |  */ | ||||||
| public class FileSizeLimitExceededException extends FileException { | public class FileSizeLimitExceededException extends FileException { | ||||||
|  | 
 | ||||||
|  |     @Serial | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
| 
 | 
 | ||||||
|     public FileSizeLimitExceededException(long defaultMaxSize) { |     public FileSizeLimitExceededException(long defaultMaxSize) { | ||||||
| @@ -1,4 +1,6 @@ | |||||||
| package com.ruoyi.common.exception.user; | package org.dromara.common.core.exception.user; | ||||||
|  | 
 | ||||||
|  | import java.io.Serial; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 验证码错误异常类 |  * 验证码错误异常类 | ||||||
| @@ -6,6 +8,8 @@ package com.ruoyi.common.exception.user; | |||||||
|  * @author ruoyi |  * @author ruoyi | ||||||
|  */ |  */ | ||||||
| public class CaptchaException extends UserException { | public class CaptchaException extends UserException { | ||||||
|  | 
 | ||||||
|  |     @Serial | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
| 
 | 
 | ||||||
|     public CaptchaException() { |     public CaptchaException() { | ||||||
| @@ -1,4 +1,6 @@ | |||||||
| package com.ruoyi.common.exception.user; | package org.dromara.common.core.exception.user; | ||||||
|  | 
 | ||||||
|  | import java.io.Serial; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 验证码失效异常类 |  * 验证码失效异常类 | ||||||
| @@ -6,6 +8,8 @@ package com.ruoyi.common.exception.user; | |||||||
|  * @author ruoyi |  * @author ruoyi | ||||||
|  */ |  */ | ||||||
| public class CaptchaExpireException extends UserException { | public class CaptchaExpireException extends UserException { | ||||||
|  | 
 | ||||||
|  |     @Serial | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
| 
 | 
 | ||||||
|     public CaptchaExpireException() { |     public CaptchaExpireException() { | ||||||
| @@ -1,6 +1,8 @@ | |||||||
| package com.ruoyi.common.exception.user; | package org.dromara.common.core.exception.user; | ||||||
| 
 | 
 | ||||||
| import com.ruoyi.common.exception.base.BaseException; | import org.dromara.common.core.exception.base.BaseException; | ||||||
|  | 
 | ||||||
|  | import java.io.Serial; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 用户信息异常类 |  * 用户信息异常类 | ||||||
| @@ -8,6 +10,8 @@ import com.ruoyi.common.exception.base.BaseException; | |||||||
|  * @author ruoyi |  * @author ruoyi | ||||||
|  */ |  */ | ||||||
| public class UserException extends BaseException { | public class UserException extends BaseException { | ||||||
|  | 
 | ||||||
|  |     @Serial | ||||||
|     private static final long serialVersionUID = 1L; |     private static final long serialVersionUID = 1L; | ||||||
| 
 | 
 | ||||||
|     public UserException(String code, Object... args) { |     public UserException(String code, Object... args) { | ||||||
| @@ -0,0 +1,31 @@ | |||||||
|  | package org.dromara.common.core.factory; | ||||||
|  |  | ||||||
|  | import org.dromara.common.core.utils.StringUtils; | ||||||
|  | import org.springframework.beans.factory.config.YamlPropertiesFactoryBean; | ||||||
|  | import org.springframework.core.env.PropertiesPropertySource; | ||||||
|  | import org.springframework.core.env.PropertySource; | ||||||
|  | import org.springframework.core.io.support.DefaultPropertySourceFactory; | ||||||
|  | import org.springframework.core.io.support.EncodedResource; | ||||||
|  |  | ||||||
|  | import java.io.IOException; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * yml 配置源工厂 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | public class YmlPropertySourceFactory extends DefaultPropertySourceFactory { | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public PropertySource<?> createPropertySource(String name, EncodedResource resource) throws IOException { | ||||||
|  |         String sourceName = resource.getResource().getFilename(); | ||||||
|  |         if (StringUtils.isNotBlank(sourceName) && StringUtils.endsWithAny(sourceName, ".yml", ".yaml")) { | ||||||
|  |             YamlPropertiesFactoryBean factory = new YamlPropertiesFactoryBean(); | ||||||
|  |             factory.setResources(resource.getResource()); | ||||||
|  |             factory.afterPropertiesSet(); | ||||||
|  |             return new PropertiesPropertySource(sourceName, factory.getObject()); | ||||||
|  |         } | ||||||
|  |         return super.createPropertySource(name, resource); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| package com.ruoyi.common.core.service; | package org.dromara.common.core.service; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 通用 参数配置服务 |  * 通用 参数配置服务 | ||||||
| @@ -0,0 +1,18 @@ | |||||||
|  | package org.dromara.common.core.service; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 通用 部门服务 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | public interface DeptService { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 通过部门ID查询部门名称 | ||||||
|  |      * | ||||||
|  |      * @param deptIds 部门ID串逗号分隔 | ||||||
|  |      * @return 部门名称串逗号分隔 | ||||||
|  |      */ | ||||||
|  |     String selectDeptNameByIds(String deptIds); | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -1,4 +1,6 @@ | |||||||
| package com.ruoyi.common.core.service; | package org.dromara.common.core.service; | ||||||
|  | 
 | ||||||
|  | import java.util.Map; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * 通用 字典服务 |  * 通用 字典服务 | ||||||
| @@ -54,4 +56,12 @@ public interface DictService { | |||||||
|      */ |      */ | ||||||
|     String getDictValue(String dictType, String dictLabel, String separator); |     String getDictValue(String dictType, String dictLabel, String separator); | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * 获取字典下所有的字典值与标签 | ||||||
|  |      * | ||||||
|  |      * @param dictType 字典类型 | ||||||
|  |      * @return dictValue为key,dictLabel为值组成的Map | ||||||
|  |      */ | ||||||
|  |     Map<String, String> getAllDictByDictType(String dictType); | ||||||
|  | 
 | ||||||
| } | } | ||||||
| @@ -0,0 +1,18 @@ | |||||||
|  | package org.dromara.common.core.service; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 通用 OSS服务 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | public interface OssService { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 通过ossId查询对应的url | ||||||
|  |      * | ||||||
|  |      * @param ossIds ossId串逗号分隔 | ||||||
|  |      * @return url串逗号分隔 | ||||||
|  |      */ | ||||||
|  |     String selectUrlByIds(String ossIds); | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,26 @@ | |||||||
|  | package org.dromara.common.core.service; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 通用 用户服务 | ||||||
|  |  * | ||||||
|  |  * @author Lion Li | ||||||
|  |  */ | ||||||
|  | public interface UserService { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 通过用户ID查询用户账户 | ||||||
|  |      * | ||||||
|  |      * @param userId 用户ID | ||||||
|  |      * @return 用户账户 | ||||||
|  |      */ | ||||||
|  |     String selectUserNameById(Long userId); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 通过用户ID查询用户账户 | ||||||
|  |      * | ||||||
|  |      * @param userId 用户ID | ||||||
|  |      * @return 用户账户 | ||||||
|  |      */ | ||||||
|  |     String selectNicknameById(Long userId); | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| package com.ruoyi.common.utils; | package org.dromara.common.core.utils; | ||||||
| 
 | 
 | ||||||
| import lombok.AccessLevel; | import lombok.AccessLevel; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
| @@ -0,0 +1,93 @@ | |||||||
|  | package org.dromara.common.core.utils; | ||||||
|  |  | ||||||
|  | import cn.hutool.core.collection.CollUtil; | ||||||
|  | import cn.hutool.core.map.MapUtil; | ||||||
|  | import cn.hutool.core.util.ObjectUtil; | ||||||
|  | import io.github.linpeilie.Converter; | ||||||
|  | import lombok.AccessLevel; | ||||||
|  | import lombok.NoArgsConstructor; | ||||||
|  |  | ||||||
|  | import java.util.List; | ||||||
|  | import java.util.Map; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Mapstruct 工具类 | ||||||
|  |  * <p>参考文档:<a href="https://mapstruct.plus/introduction/quick-start.html">mapstruct-plus</a></p> | ||||||
|  |  * | ||||||
|  |  * | ||||||
|  |  * @author Michelle.Chung | ||||||
|  |  */ | ||||||
|  | @NoArgsConstructor(access = AccessLevel.PRIVATE) | ||||||
|  | public class MapstructUtils { | ||||||
|  |  | ||||||
|  |     private final static Converter CONVERTER = SpringUtils.getBean(Converter.class); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 将 T 类型对象,转换为 desc 类型的对象并返回 | ||||||
|  |      * | ||||||
|  |      * @param source 数据来源实体 | ||||||
|  |      * @param desc   描述对象 转换后的对象 | ||||||
|  |      * @return desc | ||||||
|  |      */ | ||||||
|  |     public static <T, V> V convert(T source, Class<V> desc) { | ||||||
|  |         if (ObjectUtil.isNull(source)) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         if (ObjectUtil.isNull(desc)) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         return CONVERTER.convert(source, desc); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 将 T 类型对象,按照配置的映射字段规则,给 desc 类型的对象赋值并返回 desc 对象 | ||||||
|  |      * | ||||||
|  |      * @param source 数据来源实体 | ||||||
|  |      * @param desc   转换后的对象 | ||||||
|  |      * @return desc | ||||||
|  |      */ | ||||||
|  |     public static <T, V> V convert(T source, V desc) { | ||||||
|  |         if (ObjectUtil.isNull(source)) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         if (ObjectUtil.isNull(desc)) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         return CONVERTER.convert(source, desc); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 将 T 类型的集合,转换为 desc 类型的集合并返回 | ||||||
|  |      * | ||||||
|  |      * @param sourceList 数据来源实体列表 | ||||||
|  |      * @param desc       描述对象 转换后的对象 | ||||||
|  |      * @return desc | ||||||
|  |      */ | ||||||
|  |     public static <T, V> List<V> convert(List<T> sourceList, Class<V> desc) { | ||||||
|  |         if (ObjectUtil.isNull(sourceList)) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         if (CollUtil.isEmpty(sourceList)) { | ||||||
|  |             return CollUtil.newArrayList(); | ||||||
|  |         } | ||||||
|  |         return CONVERTER.convert(sourceList, desc); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 将 Map 转换为 beanClass 类型的集合并返回 | ||||||
|  |      * | ||||||
|  |      * @param map       数据来源 | ||||||
|  |      * @param beanClass bean类 | ||||||
|  |      * @return bean对象 | ||||||
|  |      */ | ||||||
|  |     public static <T> T convert(Map<String, Object> map, Class<T> beanClass) { | ||||||
|  |         if (MapUtil.isEmpty(map)) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         if (ObjectUtil.isNull(beanClass)) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         return CONVERTER.convert(map, beanClass); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -1,9 +1,9 @@ | |||||||
| package com.ruoyi.common.utils; | package org.dromara.common.core.utils; | ||||||
| 
 | 
 | ||||||
| import com.ruoyi.common.utils.spring.SpringUtils; |  | ||||||
| import lombok.AccessLevel; | import lombok.AccessLevel; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
| import org.springframework.context.MessageSource; | import org.springframework.context.MessageSource; | ||||||
|  | import org.springframework.context.NoSuchMessageException; | ||||||
| import org.springframework.context.i18n.LocaleContextHolder; | import org.springframework.context.i18n.LocaleContextHolder; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
| @@ -24,6 +24,10 @@ public class MessageUtils { | |||||||
|      * @return 获取国际化翻译值 |      * @return 获取国际化翻译值 | ||||||
|      */ |      */ | ||||||
|     public static String message(String code, Object... args) { |     public static String message(String code, Object... args) { | ||||||
|         return MESSAGE_SOURCE.getMessage(code, args, LocaleContextHolder.getLocale()); |         try { | ||||||
|  |             return MESSAGE_SOURCE.getMessage(code, args, LocaleContextHolder.getLocale()); | ||||||
|  |         } catch (NoSuchMessageException e) { | ||||||
|  |             return code; | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -1,26 +1,26 @@ | |||||||
| package com.ruoyi.common.utils; | package org.dromara.common.core.utils; | ||||||
| 
 | 
 | ||||||
| import cn.hutool.core.convert.Convert; | import cn.hutool.core.convert.Convert; | ||||||
| import cn.hutool.extra.servlet.ServletUtil; | import cn.hutool.extra.servlet.JakartaServletUtil; | ||||||
| import cn.hutool.http.HttpStatus; | import cn.hutool.http.HttpStatus; | ||||||
| import com.ruoyi.common.constant.Constants; | import jakarta.servlet.ServletRequest; | ||||||
|  | import jakarta.servlet.http.HttpServletRequest; | ||||||
|  | import jakarta.servlet.http.HttpServletResponse; | ||||||
|  | import jakarta.servlet.http.HttpSession; | ||||||
| import lombok.AccessLevel; | import lombok.AccessLevel; | ||||||
| import lombok.NoArgsConstructor; | import lombok.NoArgsConstructor; | ||||||
| import org.springframework.http.MediaType; | import org.springframework.http.MediaType; | ||||||
|  | import org.springframework.util.LinkedCaseInsensitiveMap; | ||||||
| import org.springframework.web.context.request.RequestAttributes; | 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.HttpServletResponse; |  | ||||||
| import javax.servlet.http.HttpSession; |  | ||||||
| import java.io.IOException; | import java.io.IOException; | ||||||
| import java.io.UnsupportedEncodingException; |  | ||||||
| import java.net.URLDecoder; | import java.net.URLDecoder; | ||||||
| import java.net.URLEncoder; | import java.net.URLEncoder; | ||||||
| import java.nio.charset.StandardCharsets; | import java.nio.charset.StandardCharsets; | ||||||
| import java.util.Collections; | import java.util.Collections; | ||||||
|  | import java.util.Enumeration; | ||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
| 
 | 
 | ||||||
| @@ -30,7 +30,7 @@ import java.util.Map; | |||||||
|  * @author ruoyi |  * @author ruoyi | ||||||
|  */ |  */ | ||||||
| @NoArgsConstructor(access = AccessLevel.PRIVATE) | @NoArgsConstructor(access = AccessLevel.PRIVATE) | ||||||
| public class ServletUtils extends ServletUtil { | public class ServletUtils extends JakartaServletUtil { | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * 获取String参数 |      * 获取String参数 | ||||||
| @@ -94,7 +94,7 @@ public class ServletUtils extends ServletUtil { | |||||||
|     public static Map<String, String> getParamMap(ServletRequest request) { |     public static Map<String, String> getParamMap(ServletRequest request) { | ||||||
|         Map<String, String> params = new HashMap<>(); |         Map<String, String> params = new HashMap<>(); | ||||||
|         for (Map.Entry<String, String[]> entry : getParams(request).entrySet()) { |         for (Map.Entry<String, String[]> entry : getParams(request).entrySet()) { | ||||||
|             params.put(entry.getKey(), StringUtils.join(entry.getValue(), ",")); |             params.put(entry.getKey(), StringUtils.join(entry.getValue(), StringUtils.SEPARATOR)); | ||||||
|         } |         } | ||||||
|         return params; |         return params; | ||||||
|     } |     } | ||||||
| @@ -103,14 +103,22 @@ public class ServletUtils extends ServletUtil { | |||||||
|      * 获取request |      * 获取request | ||||||
|      */ |      */ | ||||||
|     public static HttpServletRequest getRequest() { |     public static HttpServletRequest getRequest() { | ||||||
|         return getRequestAttributes().getRequest(); |         try { | ||||||
|  |             return getRequestAttributes().getRequest(); | ||||||
|  |         } catch (Exception e) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * 获取response |      * 获取response | ||||||
|      */ |      */ | ||||||
|     public static HttpServletResponse getResponse() { |     public static HttpServletResponse getResponse() { | ||||||
|         return getRequestAttributes().getResponse(); |         try { | ||||||
|  |             return getRequestAttributes().getResponse(); | ||||||
|  |         } catch (Exception e) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
| @@ -121,8 +129,33 @@ public class ServletUtils extends ServletUtil { | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static ServletRequestAttributes getRequestAttributes() { |     public static ServletRequestAttributes getRequestAttributes() { | ||||||
|         RequestAttributes attributes = RequestContextHolder.getRequestAttributes(); |         try { | ||||||
|         return (ServletRequestAttributes) attributes; |             RequestAttributes attributes = RequestContextHolder.getRequestAttributes(); | ||||||
|  |             return (ServletRequestAttributes) attributes; | ||||||
|  |         } catch (Exception e) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static String getHeader(HttpServletRequest request, String name) { | ||||||
|  |         String value = request.getHeader(name); | ||||||
|  |         if (StringUtils.isEmpty(value)) { | ||||||
|  |             return StringUtils.EMPTY; | ||||||
|  |         } | ||||||
|  |         return urlDecode(value); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static Map<String, String> getHeaders(HttpServletRequest request) { | ||||||
|  |         Map<String, String> map = new LinkedCaseInsensitiveMap<>(); | ||||||
|  |         Enumeration<String> enumeration = request.getHeaderNames(); | ||||||
|  |         if (enumeration != null) { | ||||||
|  |             while (enumeration.hasMoreElements()) { | ||||||
|  |                 String key = enumeration.nextElement(); | ||||||
|  |                 String value = request.getHeader(key); | ||||||
|  |                 map.put(key, value); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return map; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
| @@ -179,11 +212,7 @@ public class ServletUtils extends ServletUtil { | |||||||
|      * @return 编码后的内容 |      * @return 编码后的内容 | ||||||
|      */ |      */ | ||||||
|     public static String urlEncode(String str) { |     public static String urlEncode(String str) { | ||||||
|         try { |         return URLEncoder.encode(str, StandardCharsets.UTF_8); | ||||||
|             return URLEncoder.encode(str, Constants.UTF8); |  | ||||||
|         } catch (UnsupportedEncodingException e) { |  | ||||||
|             return StringUtils.EMPTY; |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
| @@ -193,11 +222,7 @@ public class ServletUtils extends ServletUtil { | |||||||
|      * @return 解码后的内容 |      * @return 解码后的内容 | ||||||
|      */ |      */ | ||||||
|     public static String urlDecode(String str) { |     public static String urlDecode(String str) { | ||||||
|         try { |         return URLDecoder.decode(str, StandardCharsets.UTF_8); | ||||||
|             return URLDecoder.decode(str, Constants.UTF8); |  | ||||||
|         } catch (UnsupportedEncodingException e) { |  | ||||||
|             return StringUtils.EMPTY; |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| package com.ruoyi.common.utils.spring; | package org.dromara.common.core.utils; | ||||||
| 
 | 
 | ||||||
| import cn.hutool.extra.spring.SpringUtil; | import cn.hutool.extra.spring.SpringUtil; | ||||||
| import org.springframework.aop.framework.AopContext; | import org.springframework.aop.framework.AopContext; | ||||||
| @@ -16,9 +16,6 @@ public final class SpringUtils extends SpringUtil { | |||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * 如果BeanFactory包含一个与所给名称匹配的bean定义,则返回true |      * 如果BeanFactory包含一个与所给名称匹配的bean定义,则返回true | ||||||
|      * |  | ||||||
|      * @param name |  | ||||||
|      * @return boolean |  | ||||||
|      */ |      */ | ||||||
|     public static boolean containsBean(String name) { |     public static boolean containsBean(String name) { | ||||||
|         return getBeanFactory().containsBean(name); |         return getBeanFactory().containsBean(name); | ||||||
| @@ -27,16 +24,12 @@ public final class SpringUtils extends SpringUtil { | |||||||
|     /** |     /** | ||||||
|      * 判断以给定名字注册的bean定义是一个singleton还是一个prototype。 |      * 判断以给定名字注册的bean定义是一个singleton还是一个prototype。 | ||||||
|      * 如果与给定名字相应的bean定义没有被找到,将会抛出一个异常(NoSuchBeanDefinitionException) |      * 如果与给定名字相应的bean定义没有被找到,将会抛出一个异常(NoSuchBeanDefinitionException) | ||||||
|      * |  | ||||||
|      * @param name |  | ||||||
|      * @return boolean |  | ||||||
|      */ |      */ | ||||||
|     public static boolean isSingleton(String name) throws NoSuchBeanDefinitionException { |     public static boolean isSingleton(String name) throws NoSuchBeanDefinitionException { | ||||||
|         return getBeanFactory().isSingleton(name); |         return getBeanFactory().isSingleton(name); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * @param name |  | ||||||
|      * @return Class 注册对象的类型 |      * @return Class 注册对象的类型 | ||||||
|      */ |      */ | ||||||
|     public static Class<?> getType(String name) throws NoSuchBeanDefinitionException { |     public static Class<?> getType(String name) throws NoSuchBeanDefinitionException { | ||||||
| @@ -45,8 +38,6 @@ public final class SpringUtils extends SpringUtil { | |||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * 如果给定的bean名字在bean定义中有别名,则返回这些别名 |      * 如果给定的bean名字在bean定义中有别名,则返回这些别名 | ||||||
|      * |  | ||||||
|      * @param name |  | ||||||
|      */ |      */ | ||||||
|     public static String[] getAliases(String name) throws NoSuchBeanDefinitionException { |     public static String[] getAliases(String name) throws NoSuchBeanDefinitionException { | ||||||
|         return getBeanFactory().getAliases(name); |         return getBeanFactory().getAliases(name); | ||||||
| @@ -54,9 +45,6 @@ public final class SpringUtils extends SpringUtil { | |||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * 获取aop代理对象 |      * 获取aop代理对象 | ||||||
|      * |  | ||||||
|      * @param invoker |  | ||||||
|      * @return |  | ||||||
|      */ |      */ | ||||||
|     @SuppressWarnings("unchecked") |     @SuppressWarnings("unchecked") | ||||||
|     public static <T> T getAopProxy(T invoker) { |     public static <T> T getAopProxy(T invoker) { | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user