Compare commits
	
		
			1733 Commits
		
	
	
		
			91d25081c0
			...
			main
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 90555fd34f | |||
| a05ada5df0 | |||
| 0dc430b098 | |||
| 34480385d3 | |||
| 1f2103c7fa | |||
| fd68ed87a3 | |||
| 062c17c51e | |||
| 779fc5c5a5 | |||
| de169b79a1 | |||
| 08ebb311fb | |||
| aa24de0a5a | |||
| 12cdd25be7 | |||
| 59700493eb | |||
| e5937d573a | |||
| 88c3a84972 | |||
| db0d3a6ef3 | |||
| 3d29d27441 | |||
| b5f66603bd | |||
| 6da86e12bd | |||
| 976eeaa443 | |||
| 25d1d813f1 | |||
| 778f0c3ba2 | |||
| 38c50a4f8a | |||
| 9049022a74 | |||
| c497f321bb | |||
| 7b6f3a7a5f | |||
| 84c0768c8b | |||
| 53e9678efa | |||
| efb8d8115f | |||
| e4f547fa92 | |||
| 41183b4648 | |||
| 36e20bf0d1 | |||
| 0308e9ad83 | |||
| 06c0374f16 | |||
| c5bc610e2f | |||
| a86a24ca34 | |||
| cb2e3ea581 | |||
| 42eaf1d5e3 | |||
| 02ef706fc2 | |||
| 085b217abb | |||
| 0866e0972a | |||
| 4b13265737 | |||
| 79cd2b8123 | |||
| 8cc9641bbf | |||
| 32935aed88 | |||
| c72adbfc4b | |||
| bc378cc619 | |||
| 6327a5d2bf | |||
| 2ab2a04748 | |||
| fb0a9e98a1 | |||
| e45fe1bf10 | |||
| 3d852a8356 | |||
| b244944f41 | |||
| 3c7ba669e2 | |||
| 81e7e7129c | |||
| d7ad110b9e | |||
| 0c17ea2dcd | |||
| 78ff13a654 | |||
| 863c285049 | |||
| a3d74c0b57 | |||
| 9016a72046 | |||
| 3c32614d1c | |||
| 51988471cf | |||
| 8990bd0722 | |||
| aab2417976 | |||
| 1bd6f8da4e | |||
| 22bd1bf042 | |||
| d536a65fb4 | |||
| 03149a637d | |||
| bc6c05b3ea | |||
| 59ca353b25 | |||
| 6bc65ec412 | |||
| 97e95b51ab | |||
| b69756ef81 | |||
| a6dfa81ba6 | |||
| dad517a953 | |||
| eb2d093b02 | |||
| 67186bba55 | |||
| 1a3a9149a2 | |||
| edeecad2ce | |||
| 387f5388d9 | |||
| adcaa0a5fd | |||
| 47b2c1cb93 | |||
| ce120a6d5d | |||
| 7f3589dcfb | |||
| b134c28c10 | |||
| 41c8d0367d | |||
| 3b148d549e | |||
| b6c96af8a2 | |||
| 4904625488 | |||
| 08b5fd23ab | |||
| 0574f4f629 | |||
| 4adc3e127c | |||
| dd0a1c2293 | |||
| a07407417c | |||
| e33e3b43b7 | |||
| 634bf759ca | |||
| 0ed29c6097 | |||
| b752434fbb | |||
| eec63cc7b2 | |||
| 3dc9dd1f35 | |||
| 88e287067b | |||
| eb18e2d009 | |||
| 27a3f450ef | |||
| 58a46a09c3 | |||
| 83a1316a64 | |||
| f05f146c89 | |||
| a27852ed44 | |||
| 3782062f4a | |||
| fd83abb46c | |||
| a9d1b9f4a6 | |||
| ad69dad725 | |||
| 2f55303d16 | |||
| 3a9128a894 | |||
| def6296d4d | |||
| 034472defa | |||
| 550e4ac9ce | |||
| d26e0a89f6 | |||
| 6767afdd35 | |||
| a58de0cf92 | |||
| df93f0e0ce | |||
| 0b54b126db | |||
| a94cf8dad9 | |||
| 2c3e12a42c | |||
| c4dbdc1b8e | |||
| 42ed4692af | |||
| 258943535c | |||
| 0347d767f0 | |||
| 48b0190242 | |||
| 15d0952de8 | |||
| 84ebc1762b | |||
| a096b16945 | |||
| 37ac52116a | |||
| fcb68be006 | |||
| 048c48d754 | |||
| 6ecac8d331 | |||
| 8b1dd7cb95 | |||
| 5a58fe9077 | |||
| 12574dbe46 | |||
| b3e7c00232 | |||
| 692e060f6d | |||
| 2ac0a5f896 | |||
| f8be99547a | |||
| 7dd585c3dd | |||
| 7355949c1e | |||
| 539b9fb2b2 | |||
| 99386c6d53 | |||
| abbd73ac00 | |||
| 4bee95c8a6 | |||
| 090fc81829 | |||
| 75100cacec | |||
| 13fd262c94 | |||
| 8451cdfb80 | |||
| c8841856c0 | |||
| 2a30b28e43 | |||
| dd6849b840 | |||
| ca27903e45 | |||
| aeab6eddc2 | |||
| 1c0d40aed9 | |||
| 1444afaae2 | |||
| a05bc369b7 | |||
| 6c7f411869 | |||
| f61c45e89a | |||
| 27ed9f61d0 | |||
| df77e31043 | |||
| 2d65bdb8ee | |||
| 4966aaeda9 | |||
| 28bd700b03 | |||
| f2ca013b96 | |||
| 6cf7dabaef | |||
| e6d63592ec | |||
| 3ac4ebded3 | |||
| 6f9fc659f3 | |||
| 005bb0ea2e | |||
| 80a0543e10 | |||
| 5d42805514 | |||
| 1b7ae8a2c5 | |||
| 168b0b13fb | |||
| d99fcba468 | |||
| 147b8b0a42 | |||
| eed755fd11 | |||
| 74a612704e | |||
| 8defc56d1e | |||
| 1db20d118d | |||
| 7a70a770bb | |||
| cc9e4f974f | |||
| 2965b8fea0 | |||
| 00c617ec2e | |||
| 01ef738d31 | |||
| 423cbe7315 | |||
| afb003c397 | |||
| 2dc5a29220 | |||
| c525ec0330 | |||
| 735f1e26df | |||
| 5129400a29 | |||
| a6a01aaa37 | |||
| b819df9656 | |||
| 5d1c5fcc44 | |||
| ebad3b31b7 | |||
| 3e9f7f9e29 | |||
| 4b3463e97c | |||
| 002f2c2834 | |||
| 1509ee0729 | |||
| 830e41dfa3 | |||
| 4d1f84cc5c | |||
| 1bafbed17c | |||
| 694d9cd05a | |||
| 60172ae84d | |||
| 7e7a1122fa | |||
| a1533c8e98 | |||
| b0a6fc6498 | |||
| 74ed7b20ba | |||
| 206c25985a | |||
| 0001697274 | |||
| add21c45c5 | |||
| ef8458c7a3 | |||
| 81f972edc1 | |||
| c729a402aa | |||
| 2335050834 | |||
| 6340ed27cf | |||
| 618f80fddc | |||
| 45b6c8db96 | |||
| 5132a6b9fa | |||
| de6642b675 | |||
| 3b42399726 | |||
| 689f9fe48f | |||
| 73038222cc | |||
| c7925c1706 | |||
| 2659adb7a9 | |||
| be59bd7e89 | |||
| fcb2ca1917 | |||
| 51ce143fc2 | |||
| 804e139385 | |||
| f0fc996426 | |||
| 89eb11f808 | |||
| efdb485a3b | |||
| 30d89987a4 | |||
| 3d695069a2 | |||
| e068b57062 | |||
| 811810cd36 | |||
| c90df4b02b | |||
| 7c1082f833 | |||
| 800b8d3216 | |||
| ab877beae1 | |||
| 046c163e6f | |||
| 7959d3e5ed | |||
| 8e877a6366 | |||
| d18c19dd35 | |||
| a99260209b | |||
| 2192ddc8fa | |||
| 741a1282a3 | |||
| 1a6a331ad8 | |||
| 1ba63e2cab | |||
| 5696240e03 | |||
| 885243a5b0 | |||
| a849d00c7f | |||
| d04b44c931 | |||
| a3aad9d2c9 | |||
| d98268f809 | |||
| 34440e9ba3 | |||
| d1c889e5f2 | |||
| 1e29573ef7 | |||
| 55da259510 | |||
| cc2f533dc6 | |||
| 4436e6f20a | |||
| 32b0c19f9d | |||
| 3cedd36e15 | |||
| ecbe9b2e93 | |||
| 9ad6b6ea48 | |||
| 0d2daf4d2c | |||
| edf16a6021 | |||
| 9af2d768e8 | |||
| 7551a19b34 | |||
| f59f45d9a4 | |||
| 81e82ad731 | |||
| ca870392e2 | |||
| a7e167a95f | |||
| a49b82a7c2 | |||
| 704ad12ccf | |||
| ab9fd2bc16 | |||
| 69a63a77d3 | |||
| da7e4c2156 | |||
| a4b5185f6b | |||
| 22fc8b22b8 | |||
| a8da17162a | |||
| 5677824cde | |||
| f13c221fd6 | |||
| 4ffa9363a8 | |||
| 6d2f48f86d | |||
| 8e01ced1f5 | |||
| e8f1bc09f9 | |||
| 640f5ce6f5 | |||
| c0be30027c | |||
| 832586bd41 | |||
| 1a774937b3 | |||
| d1a936d55b | |||
| e508dafb34 | |||
| 8335717741 | |||
| 16a2b82ffd | |||
| 8db5c6443d | |||
| 9ed717fb95 | |||
| dcd4497315 | |||
| 54c0322398 | |||
| e3c33c71a0 | |||
| dc97eaa835 | |||
| 7055bb9872 | |||
| dcbe57806c | |||
| fd1b17e356 | |||
| 28427a873a | |||
| 5bdb101b52 | |||
| 97b2b38f8e | |||
| 2268f4a3fc | |||
| 9eff828249 | |||
| b14438cc15 | |||
| 3275ac5036 | |||
| b27d3bd5c6 | |||
| e049e0fa3c | |||
| 03ebc9cfe9 | |||
| caee89cf53 | |||
| 24841b9850 | |||
| e67b798714 | |||
| dc13053825 | |||
| af352256e9 | |||
| d35a3d1a8c | |||
| b92810efd2 | |||
| fcbd809691 | |||
| 60c4e0b528 | |||
| d3ec13e6c0 | |||
| a36d9f02d8 | |||
| d6db862c9d | |||
| 56542a7bf1 | |||
| 36b8e8169e | |||
| b102241efd | |||
| f36010fefa | |||
| aa23d6d50f | |||
| 6df043dfac | |||
| fe84292483 | |||
| 0f48c71837 | |||
| 107e8fce55 | |||
| 3079998a5d | |||
| e2d0ae558a | |||
| 1bca1b27ed | |||
| 6fc372c898 | |||
| ddcd54d3b9 | |||
| eb8c8c14e8 | |||
| affc0cc235 | |||
| f23251f5bb | |||
| 84f33d1bc2 | |||
| 73c9a90ae3 | |||
| c4e1709b99 | |||
| ced35af66d | |||
| b915ace6ff | |||
| e7a5fd5819 | |||
| 2fd7419bdd | |||
| 4bde03643c | |||
| fd510710d9 | |||
| 8a924bd5be | |||
| 1bc52b56af | |||
| 73edc0515f | |||
| 9c33fd93f7 | |||
| 7870f8ea78 | |||
| 3c087bc275 | |||
| 27c5b991cf | |||
| 8a937f01a4 | |||
| 3940282ed8 | |||
| ca704f38b9 | |||
| 6ff044e4ab | |||
| fa98138541 | |||
| cb7917dc26 | |||
| 58d066af0a | |||
| e2daff6463 | |||
| 7c3b7cffc2 | |||
| 775391f590 | |||
| 57adfec490 | |||
| 24e62c1885 | |||
| a70b5d89ec | |||
| 761d56f4bd | |||
| e759f62b5f | |||
| 9e2d031b5d | |||
| b9cb8ad4a8 | |||
| c1d4c1ff1d | |||
| 971683a81e | |||
| 51dae0f02c | |||
| e2c70de2e0 | |||
| d94418067f | |||
| 1cb2ee77b5 | |||
| 336d3c9434 | |||
| 8ad13c289e | |||
| 7649ce6e52 | |||
| 7577f48a09 | |||
| 5759a51017 | |||
| 0251906964 | |||
| dd5c121f1f | |||
| 2723a5f134 | |||
| cae3a92a66 | |||
| c3c60605fd | |||
| 562550880c | |||
| 238f704b22 | |||
| a9c68f9971 | |||
| 5639d8ac8e | |||
| d822a4a8ac | |||
| e52c914000 | |||
| a301f854ba | |||
| 602d9625e2 | |||
| 5598bca8d3 | |||
| 1bbaf8f7b7 | |||
| 3bb2753607 | |||
| 08848c783d | |||
| 9aac591591 | |||
| 6e229af790 | |||
| ce8cc3eb29 | |||
| 198ecddc89 | |||
| ffa8e5aebb | |||
| ae439b7e64 | |||
| cbbfe014cc | |||
| 3f1101ff73 | |||
| 83028f7817 | |||
| 5777d9700f | |||
| e1e9f4588a | |||
| be2f013b9a | |||
| 70d1795557 | |||
| 0b03ebeb70 | |||
| c466ecb77c | |||
| 8c6c681424 | |||
| ba9c71a4ec | |||
| 50bc9f4ff3 | |||
| e33050a6d6 | |||
| f00ea03fad | |||
| 3595c02e74 | |||
| 3ff84074bd | |||
| 6dd6be183b | |||
| 0764247447 | |||
| f9f9b9aab9 | |||
| ec0252bae0 | |||
| dc74d203bd | |||
| 387d364861 | |||
| 82afdecf6c | |||
| 519c63a023 | |||
| f22e7b9ad1 | |||
| d45a25258e | |||
| c7ec95f4bb | |||
| bc822355df | |||
| 9535ff18de | |||
| da0a83bb6d | |||
| 4977ee99df | |||
| 229e7a8ccc | |||
| 9ed031e574 | |||
| 3c616474ff | |||
| b1fb62dd65 | |||
| b7b166c362 | |||
| 46321dd3c1 | |||
| 1998a95c35 | |||
| 13a1fa674b | |||
| e488f3419e | |||
| 56eb6b3ce3 | |||
| dc1c29b69d | |||
| c7eae53b22 | |||
| b3b3d46696 | |||
| 545836d43c | |||
| 537ec88d05 | |||
| 219f83dec0 | |||
| d54f05fa00 | |||
| 5708f4f059 | |||
| 353807404a | |||
| a76a841238 | |||
| 81fa445964 | |||
| f65ddbc5b8 | |||
| b817a230fd | |||
| 3a180d478c | |||
| 74fecddf95 | |||
| 1dec8913c5 | |||
| c26680de84 | |||
| b9063fb22f | |||
| 287d133080 | |||
| 3ef1a732e5 | |||
| 7cd95da83c | |||
| dd138bff86 | |||
| 8fffad9d3a | |||
| 327b0149d9 | |||
| b822cf47bb | |||
| f4f0f203a2 | |||
| 30e1e461e3 | |||
| b7196f5a0c | |||
| 3e25accaa3 | |||
| 5b3c5731ee | |||
| 84de4e0c5a | |||
| 5d33a18890 | |||
| 48677a5a24 | |||
| b0349ac133 | |||
| 96186a1a50 | |||
| 925c5203be | |||
| bc8bc479d1 | |||
| 89a8a145df | |||
| 83a938dc53 | |||
| 47595b1291 | |||
| 75940bbb23 | |||
| 37516a0072 | |||
| 01a88964df | |||
| 0f68b297a0 | |||
| d454acdcbe | |||
| 2ce13afc0a | |||
| 2dd75ae7e8 | |||
| a17a6a41da | |||
| 5db181aa74 | |||
| d74f1ddb81 | |||
| be12148d04 | |||
| 72d10f9443 | |||
| 81b11976a7 | |||
| f918e89307 | |||
| 83ed4b6961 | |||
| 3216c73ee8 | |||
| 3a2b77379f | |||
| 801b9934d6 | |||
| dc4e5f75cd | |||
| 7a745c2f4b | |||
| d0178d551c | |||
| 4b1c2e36ed | |||
| 827333108d | |||
| 2e05b25c41 | |||
| d4318cc48c | |||
| 587b90bd27 | |||
| 780088eb0c | |||
| 4dc20c5e90 | |||
| 2a3d7c9291 | |||
| ac25782f2b | |||
| 8ac9695535 | |||
| 20437d56e7 | |||
| 5933a74885 | |||
| f0b412828a | |||
| afb64eb8f2 | |||
| 367faac5c3 | |||
| 3c90f065fb | |||
| 8b731999a7 | |||
| 5182d03b16 | |||
| 84deaaa970 | |||
| 433a9a29c5 | |||
| a2b39466c2 | |||
| 3388bb4283 | |||
| 03586c4005 | |||
| 8ae225f434 | |||
| 6ea69e1510 | |||
| c8c1087b73 | |||
| 553c6dc539 | |||
| feae2f5f98 | |||
| 2a96467d9c | |||
| 6cc22f5b6d | |||
| 03bd915fa5 | |||
| 9103d67cc1 | |||
| 872ec7f13f | |||
| 7041aff350 | |||
| 000fb7c941 | |||
| e0d978621b | |||
| c29627bb64 | |||
| 839cbdeaec | |||
| 25083fb0e4 | |||
| 44e3eda145 | |||
| 00c705085e | |||
| 7b957c6732 | |||
| 03e1ef3271 | |||
| 4370fef5d5 | |||
| 93b0565368 | |||
| e0565f7eed | |||
| 43ea4191c3 | |||
| 308127d044 | |||
| ab2f581c9f | |||
| 51c4044e2f | |||
| fbfb951825 | |||
| b529d49e78 | |||
| 3344757af8 | |||
| 5521f39cc5 | |||
| f9c34d14c3 | |||
| dc0902c555 | |||
| 239516b98b | |||
| d2dc045255 | |||
| 3d1716d847 | |||
| 34452525d4 | |||
| 713d42a674 | |||
| e1bfd944e9 | |||
| a6f8f6a4d4 | |||
| cf538a2c36 | |||
| 3caaa151f4 | |||
| 60ce64d3e1 | |||
| 9c9aa33687 | |||
| a8589ef4e7 | |||
| 10dd50b332 | |||
| bebfda0343 | |||
| babfb27b1f | |||
| 258bd0796d | |||
| 2f0182e06c | |||
| ecddf9975f | |||
| 664677a005 | |||
| ca9e7da17e | |||
| 39eb3d48a8 | |||
| a6e949bdd6 | |||
| 63f952d390 | |||
| 7e9cb556d0 | |||
| dce1abaeff | |||
| c4602369ae | |||
| b7610641e5 | |||
| 8fb1247279 | |||
| b8621dfbb0 | |||
| 04f2ac6815 | |||
| 9be78062be | |||
| 3410045f83 | |||
| 14b0eeec7e | |||
| d1579126f3 | |||
| c5539bc7e3 | |||
| 0f8fcbcaed | |||
| b1f82f9abe | |||
| 27deff3ff3 | |||
| 04eb416a73 | |||
| 05e714fff1 | |||
| 55badb6206 | |||
| 1b782f3df8 | |||
| bbf3fc04b6 | |||
| 7657f779b5 | |||
| 93633940dd | |||
| 3e4bfef14e | |||
| 09df1eb896 | |||
| 23b3e6cdce | |||
| b6f5325351 | |||
| 32a71664a4 | |||
| ce881506f9 | |||
| 96b832983a | |||
| 8f2ec7f4dd | |||
| 705459ee90 | |||
| 155ea5c5e4 | |||
| 8904ef2247 | |||
| de07b3d7de | |||
| faf827de71 | |||
| d95f95899c | |||
| e9e538168c | |||
| 49a5e47f9d | |||
| 8285589b10 | |||
| 00ce6d6a7a | |||
| 7c32c08f1f | |||
| d36aada227 | |||
| 5be86bf7d6 | |||
| ddb49f6215 | |||
| 40c0b72450 | |||
| e0c9a2cea7 | |||
| df3f045209 | |||
| 6ccdfab551 | |||
| 24dc521f83 | |||
| cdf96f4f6a | |||
| 807de3db57 | |||
| e3e4151187 | |||
| 1d268da08d | |||
| b34459e6c6 | |||
| 797666ae0d | |||
| dd5e6c399b | |||
| 456372c7fb | |||
| b4cd489ee9 | |||
| b04f35c2da | |||
| dcf470997e | |||
| a26bb19b0f | |||
| 6182a7a77e | |||
| 0974d1dbf8 | |||
| d090631d1c | |||
| 12a35db6cd | |||
| c4d9d503ac | |||
| 824cd2f3ea | |||
| 47dfaec226 | |||
| eb36313c9b | |||
| 354fbf7e29 | |||
| 1ddd40948e | |||
| 64d9f3e362 | |||
| 460196dc4d | |||
| 80841fe543 | |||
| c8f96a10f0 | |||
| b10c102f94 | |||
| 82b109e3bd | |||
| cd0c066978 | |||
| 7a395a9906 | |||
| 96f571e0c4 | |||
| 8385800e48 | |||
| 9315447618 | |||
| 00c306475c | |||
| affbb3eba3 | |||
| ddd552deb4 | |||
| b56b2e15af | |||
| f77b5f67d0 | |||
| bb41a81eb1 | |||
| 44dfa45ca8 | |||
| 8cfe9ade9a | |||
| 6e6b27bb65 | |||
| df3a00f8c0 | |||
| 2e66b5fa45 | |||
| 1dba0a3d95 | |||
| c9e90974bd | |||
| 4f0a882b9e | |||
| a35b602f1a | |||
| a3e717f2f7 | |||
| 8b10e0e770 | |||
| 22c302efa0 | |||
| 86450533cf | |||
| d940b3092f | |||
| 99fdf473ae | |||
| bb3263dd68 | |||
| e29e71b8bd | |||
| 9abbb05ad8 | |||
| 0c4dc7e5df | |||
| 36052f034a | |||
| 00e4fefc8f | |||
| 1ecaf69b0b | |||
| a1f9b676b5 | |||
| 330e4945e1 | |||
| 0583a8a56f | |||
| bf62482137 | |||
| ba17095536 | |||
| 4ff5e9e163 | |||
| 8a03249759 | |||
| 72cb90357e | |||
| 72563e9bfa | |||
| e334d1e5d9 | |||
| f503492bf9 | |||
| b735e861d0 | |||
| c7513e9045 | |||
| 4eb433d372 | |||
| 368c647151 | |||
| 1ca676ce0b | |||
| b33945d21c | |||
| 1649c08356 | |||
| 2416ae61f3 | |||
| c54105e65b | |||
| 9039a7a2d0 | |||
| a1ef9a4970 | |||
| c1748001d5 | |||
| e470e70612 | |||
| e0d48712ac | |||
| 05592f94b9 | |||
| 4097e5a133 | |||
| 3093d2159d | |||
| d6e5a45be9 | |||
| 10a65294ce | |||
| 22c5c5be25 | |||
| 7f4de67d67 | |||
| 01fb336985 | |||
| 559df6c7b8 | |||
| b6af88a732 | |||
| b55e08a719 | |||
| cc72e44fca | |||
| 58a2a17d6d | |||
| 84804d32ad | |||
| fcae1b6770 | |||
| b7d7afb8a5 | |||
| e38ed331b6 | |||
| 2ba798b606 | |||
| ee0c99bec9 | |||
| e7232db2f3 | |||
| 4dc0a13203 | |||
| 2f2437e14d | |||
| 695ccf975b | |||
| 2d0492cafa | |||
| 68472b234e | |||
| 157e3a39b6 | |||
| 79f5a0f520 | |||
| 831bd731ca | |||
| 7f6c0f7f04 | |||
| 354ae68dd1 | |||
| 234a46d2ac | |||
| f658df4dca | |||
| 5c7bf8086c | |||
| 9d43b8e23a | |||
| dd614c07e2 | |||
| 4270aef79b | |||
| f134bc4599 | |||
| 19dc676c36 | |||
| adf181f790 | |||
| 82b07897f8 | |||
| f9dd3bc7e2 | |||
| 7cec01897f | |||
| 297f6555f3 | |||
| 6111409d66 | |||
| 1c0dc82d44 | |||
| 5820117c1a | |||
| cc695c115b | |||
| 86dac7e2b4 | |||
| 52ddefa631 | |||
| c46d6621ec | |||
| d94ef1eb25 | |||
| eee59855cc | |||
| c1e325aadf | |||
| efdf1d3eed | |||
| 65b28f92d5 | |||
| a5845c90c2 | |||
| cec87da69d | |||
| 8ea51774e6 | |||
| f68f24cb2c | |||
| baeea79e66 | |||
| 85f14edc0a | |||
| 116aea3431 | |||
| b8299bc139 | |||
| ed094347fc | |||
| 7f1fadf068 | |||
| b8afdffbe1 | |||
| 9fcd172581 | |||
| 9425889e93 | |||
| 4c9277f61a | |||
| f6ba79f31c | |||
| 91a8aa0afe | |||
| fa99c296a5 | |||
| dcf0637420 | |||
| 98b337c5ee | |||
| ba0151bca0 | |||
| c8c081b3fd | |||
| 5f3b1663d2 | |||
| 078e601100 | |||
| 4eb4c19386 | |||
| 656fd9a1fa | |||
| 7f70e508e4 | |||
| 66e786b4bb | |||
| ebb9e5f4ae | |||
| adfad4ad40 | |||
| 6409078c03 | |||
| f671114574 | |||
| 10b2bd1480 | |||
| ce37060d94 | |||
| 79e3d59a9a | |||
| 7d19a4d184 | |||
| cda42c26e8 | |||
| 170cf9f217 | |||
| 22f28a2f8a | |||
| 43662738fc | |||
| ceef9ca979 | |||
| 057ff17240 | |||
| b9ad3bdb72 | |||
| fc8b944031 | |||
| 5bfcbe9126 | |||
| 0d6e4804f5 | |||
| d024e0fa23 | |||
| f813d8eae4 | |||
| efe8f4f939 | |||
| e0424b1678 | |||
| ba692a1195 | |||
| afc7bd68c1 | |||
| d732bad042 | |||
| d70a70b19c | |||
| 4c935c3bee | |||
| f1611efe7c | |||
| 9f2b43ea0c | |||
| c160dd791f | |||
| 35e5e518f5 | |||
| afebc190f6 | |||
| 23cd1b4601 | |||
| e1b142a044 | |||
| 031fc8ba1b | |||
| 7204acb2bb | |||
| 738f1ea9fe | |||
| 2a70a7824f | |||
| 9f848e1bdc | |||
| 0c153aeb6a | |||
| 2c4c19990a | |||
| dd0b751a43 | |||
| a25b7d5cc2 | |||
| fd3d596d57 | |||
| 20938e7d43 | |||
| 9516ce1c0f | |||
| 10f484357c | |||
| 1051846c5b | |||
| c6853289ad | |||
| 4c50143834 | |||
| d5e6ea8677 | |||
| 2497bb69bc | |||
| edb77d7ad7 | |||
| 2f58bdb381 | |||
| a58a67e0a2 | |||
| 472cfdb45b | |||
| d0a0b5ecbe | |||
| 5e01ece02b | |||
| 47748875c0 | |||
| d804ad268a | |||
| 4315fe12a5 | |||
| 8e5b43a14e | |||
| 7f855bfc56 | |||
| 307eea3ea2 | |||
| e8cb4c6ea2 | |||
| da3175292b | |||
| b70c9518eb | |||
| 1cca469577 | |||
| 42f10a8899 | |||
| dd229f15ac | |||
| 2bd1c99409 | |||
| b45c762bbe | |||
| a7b58089dd | |||
| e8d16217b0 | |||
| ebd82ee2c7 | |||
| b0c7819b5a | |||
| 1e4b47f989 | |||
| 93410af224 | |||
| 45e8b0d155 | |||
| ff255dbfae | |||
| c8d7bdb8b7 | |||
| bb897fe965 | |||
| 1551e3231c | |||
| dbe9b72feb | |||
| 3a1db0f595 | |||
| ac7b91b7e8 | |||
| aa682aa10a | |||
| e373e6ab0f | |||
| 95a714b391 | |||
| 6dd2a3136f | |||
| 28f58c7f56 | |||
| 8b69458d75 | |||
| 8bd46d8f21 | |||
| 50974d55c2 | |||
| e1bb8e54ed | |||
| 9e679bf787 | |||
| 1de705b063 | |||
| 80b91aa445 | |||
| c2d7d12767 | |||
| 429dc2b848 | |||
| ca4ea0e5ea | |||
| f6926ad356 | |||
| 2860deb90a | |||
| 2cdbbb1b37 | |||
| 4a264d90d4 | |||
| 766d9668c2 | |||
| abb60f5743 | |||
| 4dce8c8f03 | |||
| 77a9f1a13c | |||
| bf01bb66e5 | |||
| 97a5bace6f | |||
| dbc5d6c31e | |||
| 586db3f008 | |||
| 331a1549da | |||
| a9d74605af | |||
| 9bb1195fb6 | |||
| 27bf60b94f | |||
| e3bacfb8cb | |||
| c4e58890ea | |||
| 80a78a036e | |||
| 8700030707 | |||
| 0e0114a7d1 | |||
| 263ca3ac9a | |||
| d4d51ec48f | |||
| 27fbfd49e2 | |||
| d3483c8062 | |||
| 2c77143fc2 | |||
| fded23b97d | |||
| 9d7bd8e9ab | |||
| fb91398462 | |||
| e87f19a8df | |||
| 105dadd798 | |||
| 6daaf22dc0 | |||
| 2abf2837d3 | |||
| 0ea985178e | |||
| 422aa67af6 | |||
| fc3b62ee37 | |||
| 7fffab6985 | |||
| e2fa126a67 | |||
| be4a58d1c6 | |||
| 9396f70f85 | |||
| 7f5e138cf7 | |||
| 5a4be3d2c1 | |||
| ffc146df06 | |||
| f39a7681db | |||
| b5f37cb54b | |||
| 6aaa4f8cf6 | |||
| c60a7580ba | |||
| 3dca59685e | |||
| a529f05825 | |||
| 4cee0e3585 | |||
| 46da172806 | |||
| b72a2884f6 | |||
| 26b8dcee1b | |||
| 0c0c4019aa | |||
| 97edb56edc | |||
| 78d13aee1a | |||
| e41ec1c91c | |||
| e3f65c8941 | |||
| f0aa0bc021 | |||
| 6ebca8d22b | |||
| 8595af7173 | |||
| 95371ad934 | |||
| 04f757a08c | |||
| 2c176825fd | |||
| cf08d0d490 | |||
| 808030100f | |||
| aa7c088504 | |||
| 0040a52169 | |||
| fae7de48d3 | |||
| a7db9bed44 | |||
| 21bf0910c5 | |||
| b8230646a2 | |||
| fcfcb9845f | |||
| 059d5260a9 | |||
| 37853bdedd | |||
| 87f22f45aa | |||
| d241b4fa7a | |||
| 685a3998a2 | |||
| dad72b904f | |||
| 43279541dd | |||
| c406d21674 | |||
| b4791977c1 | |||
| b0988cca70 | |||
| 81e1f7f6b1 | |||
| ef917ecc25 | |||
| fc6916fc2d | |||
| c63949992f | |||
| 39b27b2a17 | |||
| ae4a790236 | |||
| 33130140fd | |||
| a93faad951 | |||
| f9a8b431e0 | |||
| 9012dd14e2 | |||
| 4a6330d016 | |||
| 985e5e2bea | |||
| 4f77818406 | |||
| e6dec42a00 | |||
| fd001d24d3 | |||
| 849fd9d984 | |||
| 9b5c0696b1 | |||
| 7aa5884797 | |||
| a3442b8f2f | |||
| 5b237a1547 | |||
| 30793b75d5 | |||
| 0e3b4200d8 | |||
| 2e37990d87 | |||
| 6f0f53a9de | |||
| dd07d724a8 | |||
| 6b138246a9 | |||
| 82f49667a9 | |||
| 54072412f3 | |||
| 047446ba3a | |||
| 6d6e1dd9ea | |||
| 03ce8618e7 | |||
| 6cba58c301 | |||
| a168d90dd3 | |||
| db1a7a7fd6 | |||
| f4d9fa69e4 | |||
| 1a89177ecc | |||
| 382de101dd | |||
| cf03eae4ec | |||
| 36a82d7f53 | |||
| b775781fd7 | |||
| 3a34401113 | |||
| 2acf723c86 | |||
| 9927268330 | |||
| bbb193a787 | |||
| c45c97e29d | |||
| 976aeb0f75 | |||
| 3b807543b7 | |||
| 0350f86322 | |||
| 007be6b1ff | |||
| 3ec0cf4fac | |||
| bb8dda6da0 | |||
| c599e2ee00 | |||
| 1b49e3837c | |||
| 86e15d4155 | |||
| 6db8013e34 | |||
| b10af9d9f1 | |||
| 7b04803aa0 | |||
| d3b9fd7d78 | |||
| c077f7322d | |||
| c95b3db6eb | |||
| 0d709742ed | |||
| 246136b1ad | |||
| 44ffe20e88 | |||
| 290c4600fa | |||
| 84007e1b72 | |||
| 9bc1c610ac | |||
| 613298bdea | |||
| 0c325185fb | |||
| 0acf98aef3 | |||
| 2847cffa76 | |||
| 05c293ed6d | |||
| 3f86862ae9 | |||
| 9a80979a42 | |||
| 091ed270b0 | |||
| 92c19092fe | |||
| 42e55c0617 | |||
| 2792caa387 | |||
| a438aae9bc | |||
| 5f607e2b75 | |||
| 7333b5d755 | |||
| c64a315226 | |||
| 1d6c74162e | |||
| 8f84483826 | |||
| a4cafca6ab | |||
| 69b46d791f | |||
| 4512aed44a | |||
| 0bf2b1b4ae | |||
| 0185c09d55 | |||
| 02e6e77453 | |||
| b849de00dc | |||
| 50f02892e3 | |||
| 2a9eaead83 | |||
| 3b008155e1 | |||
| 501fa36fad | |||
| 4089590bdf | |||
| 46284a0660 | |||
| c40220f766 | |||
| dd3c1f45c8 | |||
| 05df86e15a | |||
| 4e84678a3c | |||
| 2ca4204775 | |||
| 8b433027e2 | |||
| 476e4e8eb1 | |||
| 205deea88a | |||
| 5bd4ff7610 | |||
| bfc78b0ef9 | |||
| d693c397ea | |||
| 3a403e5192 | |||
| 1d8d1ec9a5 | |||
| aad4f8dbb5 | |||
| 8d185c274a | |||
| 3b59d6c546 | |||
| 5e491f11ee | |||
| d33ed42853 | |||
| 7cedea06ac | |||
| 8e5ed5bbc1 | |||
| 2e5f750e50 | |||
| f8b02e5964 | |||
| 67457967c0 | |||
| b3b7ef90e8 | |||
| aa4af439be | |||
| 20289cad10 | |||
| b03d424e2f | |||
| e0d64c31c7 | |||
| 2dfaf4ebae | |||
| 8c1b95dc97 | |||
| ccf8b0220e | |||
| fb5641343e | |||
| 6b7b3efcb1 | |||
| b929b51525 | |||
| 327f7ad341 | |||
| 2d1f333095 | |||
| 17c5bade83 | |||
| 87765941eb | |||
| fbc90b69d9 | |||
| 1809862c16 | |||
| d72055c7d3 | |||
| 300f784f7d | |||
| 2e9a187935 | |||
| 67a045eae6 | |||
| e3f0145264 | |||
| 2a79903a28 | |||
| 34b5dcccfc | |||
| 4023476685 | |||
| df690dae0e | |||
| d17f76261e | |||
| d3222ce083 | |||
| 71dffc4c0b | |||
| 406a421742 | |||
| acb6eb237c | |||
| 10bf728faf | |||
| 663b50654d | |||
| 607617747c | |||
| 7c7f2e0f2c | |||
| f0a69eb1a2 | |||
| 1c8f5ef7ac | |||
| 69331edabb | |||
| 7c1c4b907b | |||
| 788a121994 | |||
| eb5710837b | |||
| 6b307a6e17 | |||
| c3e18d658c | |||
| b8d422f45c | |||
| 5732ecfbfa | |||
| 08d08a934a | |||
| 536923c00b | |||
| 28124fc059 | |||
| 48706f0bd5 | |||
| e06b4c3179 | |||
| e4d251a0b3 | |||
| c500c12668 | |||
| 3d581b1677 | |||
| 400d281cf6 | |||
| 64c6cc05de | |||
| 6d40ef6f4d | |||
| 71937ce89c | |||
| 0d402b608c | |||
| c66421b45d | |||
| 5609bdb6f4 | |||
| bbfcf5fd4f | |||
| 97d44104e1 | |||
| ae04b22b2a | |||
| 31e33d49df | |||
| 59a035e5c2 | |||
| 945fd5d5a3 | |||
| 62060adeba | |||
| 157936acef | |||
| b2fc75edb8 | |||
| 72b48e136f | |||
| a999dd2085 | |||
| 2f08149e48 | |||
| 48d5f1674f | |||
| 49f95ab100 | |||
| ede4c465c6 | |||
| 1a84d5b30c | |||
| 0826db0a5b | |||
| 9f3a25bd7d | |||
| 348936a67e | |||
| bcd094c5dd | |||
| 0e9863050f | |||
| c097cb54f1 | |||
| 01aeb8e759 | |||
| b163427514 | |||
| 7328283b6b | |||
| 4fb97a7c95 | |||
| 3e89025fd9 | |||
| 183f8098be | |||
| 6c9d57b18a | |||
| eb552f01f0 | |||
| 95717824c6 | |||
| f369650711 | |||
| 81a94082aa | |||
| dd8634c47b | |||
| f577e137c2 | |||
| 3b65050632 | |||
| 728b00abdb | |||
| d0df31674c | |||
| b6cdeee548 | |||
| 70255273ed | |||
| 1fe88402e2 | |||
| 5890f9932b | |||
| 88d326023b | |||
| 144bb4945b | |||
| d621e271a0 | |||
| 1d904c5cde | |||
| 67097696e6 | |||
| d32f503633 | |||
| 8e7e77067a | |||
| 537cdbb307 | |||
| 18f53df9f8 | |||
| 9899390b61 | |||
| dcd26308d3 | |||
| 0f1ed03caf | |||
| 80c476a908 | |||
| c935cdd8ee | |||
| 59da1d6e49 | |||
| c9bb3aa489 | |||
| 5aef7dac33 | |||
| aea1f1889e | |||
| 3df0b1fcec | |||
| 113d29fab0 | |||
| ebb969c039 | |||
| 9a30dd6de5 | |||
| 525f837e21 | |||
| db0e526896 | |||
| e76eed7274 | |||
| 286629836c | |||
| 354c8c3d4a | |||
| d3ffa1d40a | |||
| a6a279837d | |||
| e3cf7fbfa0 | |||
| 7bafb6ba0d | |||
| faf7aa06b6 | |||
| 738d7fba2a | |||
| 0aaac97915 | |||
| 38ef6e5583 | |||
| 720ee63505 | |||
| e1ea1f14a5 | |||
| 6efab40a85 | |||
| 22f274fd32 | |||
| c50f24b755 | |||
| b8b387c33d | |||
| a2e6d09ee8 | |||
| 8be2ec9319 | |||
| 34c08d4345 | |||
| c0b15b5d94 | |||
| dac5858541 | |||
| 2cfc067ea1 | |||
| 11dcb399f6 | |||
| a91db4f956 | |||
| 653f1f8d58 | |||
| bf634b09db | |||
| 89beba25b6 | |||
| 67522804d9 | |||
| da1c096b45 | |||
| 4a27a825a1 | |||
| 77ed131f89 | |||
| f70e5bae9a | |||
| cec4cd0d28 | |||
| 95e31bb629 | |||
| 443818efb5 | |||
| 711842f00d | |||
| 8a09780a02 | |||
| 3ea6b5824b | |||
| be8f0d66b9 | |||
| 45e8ec6505 | |||
| 1ec7a6f096 | |||
| 4554b85914 | |||
| 8f44d0b2cd | |||
| 8aa79c4a9c | |||
| 71a3c357e7 | |||
| c8d3210b57 | |||
| 5ac6750ffe | |||
| 92bcbbe065 | |||
| 844f9fd79b | |||
| d2ecca55b3 | |||
| 1b7ecc4afe | |||
| 2282a49563 | |||
| abd1626997 | |||
| afe529a116 | |||
| 3e8476431d | |||
| b82fdfb2c8 | |||
| 2d015d0a33 | |||
| 765aec3620 | |||
| 8a866df5a3 | |||
| 0dd1a706fd | |||
| f89a61e23e | |||
| c479e5ad81 | |||
| 1a02f2383e | |||
| 319893d60f | |||
| e007a95982 | |||
| ca2cc7a6b6 | |||
| 76ade3daa1 | |||
| 41be05093d | |||
| 40f4a12f9b | |||
| 5e093a5555 | |||
| 06d670df50 | |||
| 9192209ca7 | |||
| 26d9b6cf35 | |||
| 4923a04c9d | |||
| 2d17eac199 | |||
| 52a174d1b3 | |||
| e482bc3aad | |||
| 6101b964af | |||
| ec022b74d1 | |||
| a241b96d59 | |||
| dc42c09ce3 | |||
| 315c957d71 | |||
| 046a34d2a4 | |||
| 5a0bf61a36 | |||
| 3d76220660 | |||
| 02254c29e4 | |||
| c65bad63ca | |||
| 9ff6ec1888 | |||
| 538d4288bb | |||
| 75f3d1dab3 | |||
| fbaa1aa14c | |||
| fb66ea3347 | |||
| 123b21cab2 | |||
| f45e07c879 | |||
| d20f51ceac | |||
| f80b8248e8 | |||
| ba2530ba55 | |||
| 3b97364f24 | |||
| 209f1f4bd1 | |||
| 38cf9e453d | |||
| d2950106ec | |||
| f304242eb4 | |||
| 9f59578275 | |||
| 962f800d2e | |||
| 4a83ebd472 | |||
| 962107e507 | |||
| 81415e0854 | |||
| 039bd11963 | |||
| e6777962a9 | |||
| 5c250ea4ae | |||
| 8c5b7b811d | |||
| 7326babc49 | |||
| 555c3bfc73 | |||
| e4ca84252d | |||
| d1d9ef5d24 | |||
| e48fcc9d25 | |||
| 024b8ce872 | |||
| 421846e08a | |||
| 2735ac32bb | |||
| ee35a0c13e | |||
| 29c7d5c677 | |||
| 25022e4909 | |||
| 0df0d71660 | |||
| 7204233663 | |||
| e090a2fe7a | |||
| 290be744a3 | |||
| 13dc77666a | |||
| 8a3d11ae59 | |||
| e3405bcec6 | |||
| 7e02acd22c | |||
| da4ecf7d23 | |||
| 0fd1c2235f | |||
| 185d7e6666 | |||
| b20c29b022 | |||
| 80930ebd3d | |||
| 13088f53a6 | |||
| 12d5dcd298 | |||
| 0c3f190044 | |||
| 35289ea93e | |||
| 2c305dc6c6 | |||
| 63f690ee2d | |||
| 1a68746a7c | |||
| 62f76f7433 | |||
| 6ab9871fb9 | |||
| 7badf857cb | |||
| 612ca02461 | |||
| b5e6176885 | |||
| 6b90f42a0e | |||
| cae9f22e49 | |||
| aeed1dbd06 | |||
| 382364b5df | |||
| 0e8d3656b2 | |||
| 0d2e0a1af8 | |||
| 516853b05f | |||
| e96b3d9bd4 | |||
| 858ce524f9 | |||
| 1a5b4b364a | |||
| f196b20024 | |||
| 3795fb4a40 | |||
| 11bb799bd5 | |||
| 84102bb95a | |||
| 3ada2dea87 | |||
| 82ac381936 | |||
| 0c01aeec50 | |||
| 3c72ae048e | |||
| 892206744d | |||
| 61cf1577dc | |||
| 9e2c1474db | |||
| 7fefc9b0a6 | |||
| 5ce4078ba4 | |||
| 8718089483 | |||
| 16328f73d9 | |||
| 5a75972f26 | |||
| e0d4f53cf4 | |||
| dcd6933824 | |||
| 35f66e7e41 | |||
| e09a59c5b4 | |||
| 8d32f1b3bd | |||
| 04314c6256 | |||
| 68c2b505bb | |||
| 049e654535 | |||
| 9448c6a488 | |||
| c927dc4ecd | |||
| 481f2e1126 | |||
| 9751900193 | |||
| 07455cd5b1 | |||
| fe4ecd0ad8 | |||
| 216f983d36 | |||
| cb9ba824cf | |||
| a215d027ec | |||
| 78d476fe80 | |||
| 41a4802dc6 | |||
| a11c8465d5 | |||
| 06a890bc15 | |||
| 366304a9b7 | |||
| d382d85d26 | |||
| 4356663688 | |||
| b464f7ae4c | |||
| 3164232f9d | |||
| 8da7ee0bb3 | |||
| 9458a3976b | |||
| 26b55e6fcf | |||
| dbf795804c | |||
| 0d743f7204 | |||
| e3a31f16bc | |||
| 234c9adcdd | |||
| b7e4e6f43f | |||
| 725b959117 | |||
| 6cbe113b3e | |||
| 19875a841d | |||
| b3b7f739ff | |||
| 6409b69d6c | |||
| 9c367b400a | |||
| c5164c76fc | |||
| 0b0be28e1a | |||
| baade8e138 | |||
| fc17ba9aaa | |||
| 31208b5e99 | |||
| c590eff460 | |||
| 472b51dd85 | |||
| b848d6b4e0 | |||
| ad295564de | |||
| bfedc448af | |||
| 1a6a833b93 | |||
| 615164e488 | |||
| 8be8d15e4c | |||
| 60012a7aad | |||
| 75140a4055 | |||
| 6bc5143471 | |||
| 1eeea16792 | |||
| e0e24be688 | |||
| d8139d2ab0 | |||
| db3ac923e1 | |||
| e96d8f7469 | |||
| 24fc451574 | |||
| ee1c8d1f83 | |||
| fc39b6c7a0 | |||
| 62797eb3f5 | |||
| 2acffd8afc | |||
| 9afc44b7b1 | |||
| 3c8e72073c | |||
| 1b7fc14f00 | |||
| 724d7a9d9b | |||
| 9a394b7dae | |||
| ce15025c8d | |||
| 2da3b0db78 | |||
| 0e786f46cb | |||
| 5b218169c7 | |||
| 5c228af14b | |||
| 685ad7afaf | |||
| 976a504233 | |||
| 264cf75964 | |||
| 58a57c9a0e | |||
| c773dbc7b5 | |||
| 1d877255ac | |||
| 37cbc64f52 | |||
| 97e80a85e0 | |||
| cf8d94776e | |||
| cb1dde17bb | |||
| bc1db79430 | |||
| 42fbcdd1e8 | |||
| dad31fffcb | |||
| ed3e55514a | |||
| 275f6049d1 | |||
| d3b12eeef1 | |||
| b3d66151bc | |||
| 1cf70c25a9 | |||
| 695c8cbad8 | |||
| 8fb61e7689 | |||
| d45a34525b | |||
| c6c9073fa0 | |||
| c29988acf4 | |||
| 64a63fa8fa | |||
| eadbf56dae | |||
| 22b9dceea5 | |||
| 4b3b455135 | |||
| 899bc076a3 | |||
| e6ac177396 | |||
| 8017e837ef | |||
| 3d0e29003f | |||
| 4c8c6226a6 | |||
| 78b9b00f77 | |||
| fc99e6324a | |||
| 0ee7faa551 | |||
| 146205e4f7 | |||
| e5fdced681 | |||
| 1c7fdfac69 | |||
| 3ec16b5045 | |||
| afb99fef64 | |||
| 5b85c2e8a4 | |||
| 7dfaa36024 | |||
| 063ee78a8c | |||
| 5f3c7e7e90 | |||
| 1f055c2283 | |||
| 4eec062871 | |||
| e7a318d6b9 | |||
| 9a7e76ea7a | |||
| f45c6c7938 | |||
| c43faef14e | |||
| 487d3c9d6e | |||
| 04c682fc9b | |||
| 0496f665aa | |||
| ae6171e844 | |||
| 0d19e1be74 | |||
| fb236a374d | |||
| 4aff0111aa | |||
| 1be91d0de4 | |||
| cda2f1ec36 | |||
| e5c85287bb | |||
| 1fcb0ec5fd | |||
| 3537f22197 | |||
| 99d7510c32 | |||
| bcdd161205 | |||
| 0f6c3075bc | |||
| cd4b165f90 | |||
| 63b3ba2bb2 | |||
| 567c51f6a2 | |||
| 7444b41f60 | |||
| 0aecd36956 | |||
| 8e90dbc8b6 | |||
| 5068be1a4c | |||
| 9f70722521 | |||
| 7a22e7d887 | |||
| 52fae596fa | |||
| 6775d3d72d | |||
| ccb67957bc | |||
| 149a3ad2f1 | |||
| 9146e2e231 | |||
| 333458a184 | |||
| f889ae5232 | |||
| e507af8d5b | |||
| fb82538d0d | |||
| cccb76afc2 | |||
| 72ee39612e | |||
| d561ad6d41 | |||
| a7fc89cf40 | |||
| b59d7b5dca | |||
| 51fd5408dc | |||
| 4f0148d80e | |||
| ed4f0a62a1 | |||
| 3fae40fbef | |||
| 51260311a0 | |||
| 0745890af0 | |||
| 75efb564fc | |||
| f6e9c6d010 | |||
| 4abe1730a7 | |||
| 236aeb0258 | |||
| 626f0e6989 | |||
| 90732e137f | |||
| 9f42d9d173 | |||
| 5b0be30c5b | |||
| 13aa9838cd | |||
| 24b8618306 | |||
| f90a93c4bc | |||
| 75dbfad3a7 | |||
| 8000ad6c6a | |||
| ffdcd61894 | |||
| ba491420f5 | |||
| 1f1f1bea1a | |||
| f161d9a436 | |||
| fd460f2d3e | |||
| 468add0819 | |||
| 151cc10caf | |||
| 22a79c0be4 | |||
| 5970a9a5b6 | |||
| d95460c7cd | |||
| 8896233d78 | |||
| a3d93d4b08 | |||
| d0d6ef64df | |||
| ff10a9935b | |||
| d5cc28e50b | |||
| 07a92af982 | |||
| 615d4baef8 | |||
| f4618877d4 | |||
| 9578e54ea7 | |||
| 5ff288f739 | |||
| f081b9691f | |||
| 06bb2ea228 | |||
| 81e2b43231 | |||
| 27c40da7b4 | |||
| 8b30c1c319 | |||
| 300e20dcd0 | |||
| 0978549675 | |||
| ab4a2d0e6b | |||
| 74f7c75012 | |||
| 37f2f5e40b | |||
| 056e575e7c | |||
| febfa442fa | |||
| 27f4d78f0d | |||
| 858f1a9a32 | |||
| 2b914fd222 | |||
| d68a089235 | |||
| f3590dac7a | |||
| aa457cc2fb | |||
| 7ca71f90db | |||
| d2ae958847 | |||
| a285e30108 | |||
| 896246d9ed | |||
| d6dfa63bea | |||
| 8f50d05906 | |||
| 88520dbc7f | |||
| 109e42a5a3 | |||
| 6aec31711d | |||
| 1ba3cb6277 | |||
| eece72bc40 | |||
| 41fe37bdb2 | |||
| fa515ad39c | |||
| 57426b5b5b | |||
| f09673a795 | |||
| 42f5c49cbc | |||
| 378e8f13af | |||
| f71536c614 | |||
| 6cf401539f | |||
| 13761a4130 | |||
| 7bdddc7ae8 | |||
| fc2737921b | |||
| aa8926a624 | |||
| 4603ea4f96 | |||
| be71e59be2 | |||
| bd09bfacd0 | |||
| 4d7753378f | |||
| ed78959817 | |||
| ad66be3449 | |||
| 00a6fa0c91 | |||
| f25888c88e | |||
| 1629bc8bf7 | |||
| 29c9460362 | |||
| 60257c4ef4 | |||
| 7bb7e92137 | |||
| cd833dc21d | |||
| 1e0b79bf62 | |||
| fb8309c7b4 | |||
| 779c3f824e | |||
| 2abfb8ce58 | |||
| 9046c10d10 | |||
| a9d3427b6f | |||
| 2e249db5f5 | |||
| f2ecf3a676 | |||
| 53bc8ed05a | |||
| c5fd55a6f8 | |||
| 586f4d21d4 | |||
| 0d56b61b2e | |||
| 3b9d8a4fcb | |||
| 72adae3f54 | |||
| d1ea7b8704 | |||
| 41a6e05034 | |||
| 6883434d0d | |||
| 0679fdfb6d | |||
| eda2193e64 | |||
| 118c5c2eed | |||
| d865ec3ef8 | |||
| 7af970ace4 | |||
| 99bf829c88 | |||
| 65e0b87d79 | |||
| eab9ac5f05 | |||
| 5feafe1b48 | |||
| d414e3b300 | |||
| a47b40d922 | |||
| 9126671581 | |||
| c9292b7d04 | |||
| eaeca443cf | |||
| f49203cafc | |||
| 307c09a4c1 | |||
| c3cb8fe93b | |||
| ae7e1a91c1 | |||
| d0aebe906b | |||
| 3e1887e0d1 | |||
| fa8d8c4ab1 | |||
| ff81c6c5ca | |||
| 474646db47 | |||
| 4900b52d2b | |||
| 56f7b6c449 | |||
| 82a6161abf | |||
| 077af3a46d | |||
| 498d9c4893 | |||
| 2410d77cb7 | |||
| 40bdd5ba1c | |||
| 629528b6cf | |||
| 76b2b5f7e3 | |||
| a8c3f05ffa | |||
| e918d809eb | |||
| 2a24c376b3 | |||
| 7af059e543 | |||
| 73988d7b2c | |||
| 1d217b9f75 | |||
| 3ad2256a66 | |||
| 897726e1ec | |||
| 8b98a2dd07 | |||
| cca75420f0 | |||
| 86c627ed1d | |||
| d55514e3a7 | 
| @@ -7,5 +7,5 @@ indent_size = 4 | |||||||
| indent_style = space | indent_style = space | ||||||
| trim_trailing_whitespace = true | trim_trailing_whitespace = true | ||||||
| insert_final_newline = true | insert_final_newline = true | ||||||
| max_line_length = 120 | max_line_length = 130 | ||||||
| tab_width = 4 | tab_width = 4 | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -323,4 +323,7 @@ gradle-app.setting | |||||||
| ### Gradle Patch ### | ### Gradle Patch ### | ||||||
| **/build/ | **/build/ | ||||||
|  |  | ||||||
|  | .kiro/ | ||||||
|  | .junie | ||||||
|  |  | ||||||
| # End of https://www.toptal.com/developers/gitignore/api/visualstudiocode,intellij,java,kotlin,macos,windows,eclipse,gradle | # End of https://www.toptal.com/developers/gitignore/api/visualstudiocode,intellij,java,kotlin,macos,windows,eclipse,gradle | ||||||
|   | |||||||
| @@ -26,11 +26,14 @@ repositories { | |||||||
| } | } | ||||||
|  |  | ||||||
| dependencies { | dependencies { | ||||||
|  |     implementation("org.redisson:redisson-spring-data-27:3.19.2") | ||||||
|  |     implementation("org.springframework.boot:spring-boot-starter-aop") | ||||||
|     implementation("org.springframework.boot:spring-boot-starter-data-jpa") |     implementation("org.springframework.boot:spring-boot-starter-data-jpa") | ||||||
|     implementation("org.springframework.boot:spring-boot-starter-data-redis") |     implementation("org.springframework.boot:spring-boot-starter-data-redis") | ||||||
|     implementation("org.springframework.boot:spring-boot-starter-security") |     implementation("org.springframework.boot:spring-boot-starter-security") | ||||||
|     implementation("org.springframework.boot:spring-boot-starter-web") |     implementation("org.springframework.boot:spring-boot-starter-web") | ||||||
|     implementation("com.fasterxml.jackson.module:jackson-module-kotlin") |     implementation("com.fasterxml.jackson.module:jackson-module-kotlin") | ||||||
|  |     implementation("org.springframework.retry:spring-retry") | ||||||
|     implementation("org.jetbrains.kotlin:kotlin-reflect") |     implementation("org.jetbrains.kotlin:kotlin-reflect") | ||||||
|  |  | ||||||
|     // jwt |     // jwt | ||||||
| @@ -44,6 +47,7 @@ dependencies { | |||||||
|     kapt("org.springframework.boot:spring-boot-configuration-processor") |     kapt("org.springframework.boot:spring-boot-configuration-processor") | ||||||
|  |  | ||||||
|     // aws |     // aws | ||||||
|  |     implementation("com.amazonaws:aws-java-sdk-sqs:1.12.380") | ||||||
|     implementation("com.amazonaws:aws-java-sdk-ses:1.12.380") |     implementation("com.amazonaws:aws-java-sdk-ses:1.12.380") | ||||||
|     implementation("com.amazonaws:aws-java-sdk-s3:1.12.380") |     implementation("com.amazonaws:aws-java-sdk-s3:1.12.380") | ||||||
|     implementation("com.amazonaws:aws-java-sdk-cloudfront:1.12.380") |     implementation("com.amazonaws:aws-java-sdk-cloudfront:1.12.380") | ||||||
| @@ -58,6 +62,17 @@ dependencies { | |||||||
|     // firebase admin sdk |     // firebase admin sdk | ||||||
|     implementation("com.google.firebase:firebase-admin:9.2.0") |     implementation("com.google.firebase:firebase-admin:9.2.0") | ||||||
|  |  | ||||||
|  |     // android publisher | ||||||
|  |     implementation("com.google.apis:google-api-services-androidpublisher:v3-rev20240319-2.0.0") | ||||||
|  |  | ||||||
|  |     implementation("com.google.api-client:google-api-client:1.32.1") | ||||||
|  |  | ||||||
|  |     implementation("org.apache.poi:poi-ooxml:5.2.3") | ||||||
|  |     implementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.6.4") | ||||||
|  |  | ||||||
|  |     // file mimetype check | ||||||
|  |     implementation("org.apache.tika:tika-core:3.2.0") | ||||||
|  |  | ||||||
|     developmentOnly("org.springframework.boot:spring-boot-devtools") |     developmentOnly("org.springframework.boot:spring-boot-devtools") | ||||||
|     runtimeOnly("com.h2database:h2") |     runtimeOnly("com.h2database:h2") | ||||||
|     runtimeOnly("com.mysql:mysql-connector-j") |     runtimeOnly("com.mysql:mysql-connector-j") | ||||||
|   | |||||||
| @@ -2,10 +2,12 @@ package kr.co.vividnext.sodalive | |||||||
|  |  | ||||||
| import org.springframework.boot.autoconfigure.SpringBootApplication | import org.springframework.boot.autoconfigure.SpringBootApplication | ||||||
| import org.springframework.boot.runApplication | import org.springframework.boot.runApplication | ||||||
|  | import org.springframework.retry.annotation.EnableRetry | ||||||
| import org.springframework.scheduling.annotation.EnableAsync | import org.springframework.scheduling.annotation.EnableAsync | ||||||
|  |  | ||||||
| @SpringBootApplication | @SpringBootApplication | ||||||
| @EnableAsync | @EnableAsync | ||||||
|  | @EnableRetry | ||||||
| class SodaLiveApplication | class SodaLiveApplication | ||||||
|  |  | ||||||
| fun main(args: Array<String>) { | fun main(args: Array<String>) { | ||||||
|   | |||||||
| @@ -0,0 +1,43 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.common.ApiResponse | ||||||
|  | import org.springframework.data.domain.Pageable | ||||||
|  | import org.springframework.security.access.prepost.PreAuthorize | ||||||
|  | import org.springframework.web.bind.annotation.GetMapping | ||||||
|  | import org.springframework.web.bind.annotation.PathVariable | ||||||
|  | import org.springframework.web.bind.annotation.PostMapping | ||||||
|  | import org.springframework.web.bind.annotation.PutMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestPart | ||||||
|  | import org.springframework.web.bind.annotation.RestController | ||||||
|  | import org.springframework.web.multipart.MultipartFile | ||||||
|  |  | ||||||
|  | @RestController | ||||||
|  | @PreAuthorize("hasRole('ADMIN')") | ||||||
|  | @RequestMapping("/admin/audition") | ||||||
|  | class AdminAuditionController(private val service: AdminAuditionService) { | ||||||
|  |     @PostMapping | ||||||
|  |     fun createAudition( | ||||||
|  |         @RequestPart("image") image: MultipartFile, | ||||||
|  |         @RequestPart("request") requestString: String | ||||||
|  |     ) = ApiResponse.ok(service.createAudition(image, requestString), "등록되었습니다.") | ||||||
|  |  | ||||||
|  |     @PutMapping | ||||||
|  |     fun updateAudition( | ||||||
|  |         @RequestPart("image", required = false) image: MultipartFile? = null, | ||||||
|  |         @RequestPart("request") requestString: String | ||||||
|  |     ) = ApiResponse.ok(service.updateAudition(image, requestString), "수정되었습니다.") | ||||||
|  |  | ||||||
|  |     @GetMapping | ||||||
|  |     fun getAuditionList(pageable: Pageable) = ApiResponse.ok( | ||||||
|  |         service.getAuditionList( | ||||||
|  |             offset = pageable.offset, | ||||||
|  |             limit = pageable.pageSize.toLong() | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     @GetMapping("/{id}") | ||||||
|  |     fun getAuditionDetail(@PathVariable id: Long) = ApiResponse.ok( | ||||||
|  |         service.getAuditionDetail(auditionId = id) | ||||||
|  |     ) | ||||||
|  | } | ||||||
| @@ -0,0 +1,69 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition | ||||||
|  |  | ||||||
|  | import com.querydsl.jpa.impl.JPAQueryFactory | ||||||
|  | import kr.co.vividnext.sodalive.audition.Audition | ||||||
|  | import kr.co.vividnext.sodalive.audition.QAudition.audition | ||||||
|  | import org.springframework.beans.factory.annotation.Value | ||||||
|  | import org.springframework.data.jpa.repository.JpaRepository | ||||||
|  | import org.springframework.stereotype.Repository | ||||||
|  |  | ||||||
|  | @Repository | ||||||
|  | interface AdminAuditionRepository : JpaRepository<Audition, Long>, AdminAuditionQueryRepository | ||||||
|  |  | ||||||
|  | interface AdminAuditionQueryRepository { | ||||||
|  |     fun getAuditionList(offset: Long, limit: Long): List<GetAuditionListItem> | ||||||
|  |     fun getAuditionListCount(): Int | ||||||
|  |     fun getAuditionDetail(auditionId: Long): GetAuditionDetailRawData | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class AdminAuditionQueryRepositoryImpl( | ||||||
|  |     private val queryFactory: JPAQueryFactory, | ||||||
|  |  | ||||||
|  |     @Value("\${cloud.aws.cloud-front.host}") | ||||||
|  |     private val coverImageHost: String | ||||||
|  | ) : AdminAuditionQueryRepository { | ||||||
|  |     override fun getAuditionList(offset: Long, limit: Long): List<GetAuditionListItem> { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetAuditionListItem( | ||||||
|  |                     audition.id, | ||||||
|  |                     audition.title, | ||||||
|  |                     audition.imagePath.prepend("/").prepend(coverImageHost), | ||||||
|  |                     audition.isAdult, | ||||||
|  |                     audition.information, | ||||||
|  |                     audition.status, | ||||||
|  |                     audition.originalWorkUrl.coalesce("") | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(audition) | ||||||
|  |             .where(audition.isActive.isTrue) | ||||||
|  |             .offset(offset) | ||||||
|  |             .limit(limit) | ||||||
|  |             .orderBy(audition.isActive.desc(), audition.id.desc()) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun getAuditionListCount(): Int { | ||||||
|  |         return queryFactory | ||||||
|  |             .select(audition.id) | ||||||
|  |             .from(audition) | ||||||
|  |             .fetch() | ||||||
|  |             .size | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun getAuditionDetail(auditionId: Long): GetAuditionDetailRawData { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetAuditionDetailRawData( | ||||||
|  |                     audition.id, | ||||||
|  |                     audition.title, | ||||||
|  |                     audition.imagePath.prepend("/").prepend(coverImageHost), | ||||||
|  |                     audition.information, | ||||||
|  |                     audition.originalWorkUrl.coalesce("") | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(audition) | ||||||
|  |             .where(audition.id.eq(auditionId)) | ||||||
|  |             .fetchFirst() | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,122 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition | ||||||
|  |  | ||||||
|  | import com.fasterxml.jackson.databind.ObjectMapper | ||||||
|  | import kr.co.vividnext.sodalive.admin.audition.role.AdminAuditionRoleRepository | ||||||
|  | import kr.co.vividnext.sodalive.audition.AuditionStatus | ||||||
|  | import kr.co.vividnext.sodalive.aws.s3.S3Uploader | ||||||
|  | import kr.co.vividnext.sodalive.common.SodaException | ||||||
|  | import kr.co.vividnext.sodalive.fcm.FcmEvent | ||||||
|  | import kr.co.vividnext.sodalive.fcm.FcmEventType | ||||||
|  | import kr.co.vividnext.sodalive.utils.generateFileName | ||||||
|  | import org.springframework.beans.factory.annotation.Value | ||||||
|  | import org.springframework.context.ApplicationEventPublisher | ||||||
|  | import org.springframework.data.repository.findByIdOrNull | ||||||
|  | import org.springframework.stereotype.Service | ||||||
|  | import org.springframework.transaction.annotation.Transactional | ||||||
|  | import org.springframework.web.multipart.MultipartFile | ||||||
|  |  | ||||||
|  | @Service | ||||||
|  | class AdminAuditionService( | ||||||
|  |     private val s3Uploader: S3Uploader, | ||||||
|  |     private val objectMapper: ObjectMapper, | ||||||
|  |     private val repository: AdminAuditionRepository, | ||||||
|  |     private val roleRepository: AdminAuditionRoleRepository, | ||||||
|  |     private val applicationEventPublisher: ApplicationEventPublisher, | ||||||
|  |  | ||||||
|  |     @Value("\${cloud.aws.s3.bucket}") | ||||||
|  |     private val bucket: String | ||||||
|  | ) { | ||||||
|  |     @Transactional | ||||||
|  |     fun createAudition(image: MultipartFile, requestString: String) { | ||||||
|  |         val request = objectMapper.readValue(requestString, CreateAuditionRequest::class.java) | ||||||
|  |         val audition = repository.save(request.toAudition()) | ||||||
|  |  | ||||||
|  |         val fileName = generateFileName("audition") | ||||||
|  |         val imagePath = s3Uploader.upload( | ||||||
|  |             inputStream = image.inputStream, | ||||||
|  |             bucket = bucket, | ||||||
|  |             filePath = "audition/production/${audition.id}/$fileName" | ||||||
|  |         ) | ||||||
|  |         audition.imagePath = imagePath | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional | ||||||
|  |     fun updateAudition(image: MultipartFile?, requestString: String) { | ||||||
|  |         val request = objectMapper.readValue(requestString, UpdateAuditionRequest::class.java) | ||||||
|  |         val audition = repository.findByIdOrNull(id = request.id) | ||||||
|  |             ?: throw SodaException("잘못된 요청입니다.\n다시 시도해 주세요.") | ||||||
|  |  | ||||||
|  |         if (request.title != null) { | ||||||
|  |             audition.title = request.title | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (request.information != null) { | ||||||
|  |             audition.information = request.information | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (request.isAdult != null) { | ||||||
|  |             audition.isAdult = request.isAdult | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (request.status != null) { | ||||||
|  |             if ( | ||||||
|  |                 (audition.status == AuditionStatus.COMPLETED || audition.status == AuditionStatus.IN_PROGRESS) && | ||||||
|  |                 request.status == AuditionStatus.NOT_STARTED | ||||||
|  |             ) { | ||||||
|  |                 throw SodaException("모집전 상태로 변경할 수 없습니다.") | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             audition.status = request.status | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (request.originalWorkUrl != null) { | ||||||
|  |             audition.originalWorkUrl = request.originalWorkUrl | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (image != null) { | ||||||
|  |             val fileName = generateFileName("audition") | ||||||
|  |             val imagePath = s3Uploader.upload( | ||||||
|  |                 inputStream = image.inputStream, | ||||||
|  |                 bucket = bucket, | ||||||
|  |                 filePath = "audition/production/${audition.id}/$fileName" | ||||||
|  |             ) | ||||||
|  |             audition.imagePath = imagePath | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (request.isActive != null) { | ||||||
|  |             audition.isActive = request.isActive | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (request.status != null && request.status == AuditionStatus.IN_PROGRESS && audition.isActive) { | ||||||
|  |             applicationEventPublisher.publishEvent( | ||||||
|  |                 FcmEvent( | ||||||
|  |                     type = FcmEventType.IN_PROGRESS_AUDITION, | ||||||
|  |                     title = "새로운 오디션 등록!", | ||||||
|  |                     message = "'${audition.title}'이 등록되었습니다. 지금 바로 오리지널 오디오 드라마 오디션에 지원해보세요!", | ||||||
|  |                     isAuth = audition.isAdult, | ||||||
|  |                     auditionId = audition.id ?: -1 | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getAuditionList(offset: Long, limit: Long): GetAuditionListResponse { | ||||||
|  |         val totalCount = repository.getAuditionListCount() | ||||||
|  |         val items = repository.getAuditionList(offset = offset, limit = limit) | ||||||
|  |         return GetAuditionListResponse(totalCount, items) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getAuditionDetail(auditionId: Long): GetAuditionDetailResponse { | ||||||
|  |         val auditionDetail = repository.getAuditionDetail(auditionId = auditionId) | ||||||
|  |         val roleList = roleRepository.getAuditionRoleListByAuditionId(auditionId = auditionId) | ||||||
|  |  | ||||||
|  |         return GetAuditionDetailResponse( | ||||||
|  |             id = auditionDetail.id, | ||||||
|  |             title = auditionDetail.title, | ||||||
|  |             imageUrl = auditionDetail.imageUrl, | ||||||
|  |             information = auditionDetail.information, | ||||||
|  |             originalWorkUrl = auditionDetail.originalWorkUrl, | ||||||
|  |             roleList = roleList | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,30 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.audition.Audition | ||||||
|  | import kr.co.vividnext.sodalive.common.SodaException | ||||||
|  |  | ||||||
|  | data class CreateAuditionRequest( | ||||||
|  |     val title: String, | ||||||
|  |     val information: String, | ||||||
|  |     val isAdult: Boolean = false, | ||||||
|  |     val originalWorkUrl: String? = null | ||||||
|  | ) { | ||||||
|  |     init { | ||||||
|  |         if (title.isBlank()) { | ||||||
|  |             throw SodaException("오디션 제목을 입력하세요") | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (information.isBlank() || information.length < 10) { | ||||||
|  |             throw SodaException("오디션 정보는 최소 10글자 입니다") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun toAudition(): Audition { | ||||||
|  |         return Audition( | ||||||
|  |             title = title, | ||||||
|  |             information = information, | ||||||
|  |             isAdult = isAdult, | ||||||
|  |             originalWorkUrl = originalWorkUrl | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,30 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition | ||||||
|  |  | ||||||
|  | import com.querydsl.core.annotations.QueryProjection | ||||||
|  | import kr.co.vividnext.sodalive.audition.AuditionStatus | ||||||
|  |  | ||||||
|  | data class GetAuditionDetailRawData @QueryProjection constructor( | ||||||
|  |     val id: Long, | ||||||
|  |     val title: String, | ||||||
|  |     val imageUrl: String, | ||||||
|  |     val information: String, | ||||||
|  |     val originalWorkUrl: String | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class GetAuditionDetailResponse( | ||||||
|  |     val id: Long, | ||||||
|  |     val title: String, | ||||||
|  |     val imageUrl: String, | ||||||
|  |     val information: String, | ||||||
|  |     val originalWorkUrl: String, | ||||||
|  |     val roleList: List<GetAuditionRoleListData> | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class GetAuditionRoleListData @QueryProjection constructor( | ||||||
|  |     val id: Long, | ||||||
|  |     val name: String, | ||||||
|  |     val imageUrl: String, | ||||||
|  |     val information: String, | ||||||
|  |     val auditionScriptUrl: String, | ||||||
|  |     val status: AuditionStatus | ||||||
|  | ) | ||||||
| @@ -0,0 +1,19 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition | ||||||
|  |  | ||||||
|  | import com.querydsl.core.annotations.QueryProjection | ||||||
|  | import kr.co.vividnext.sodalive.audition.AuditionStatus | ||||||
|  |  | ||||||
|  | data class GetAuditionListResponse( | ||||||
|  |     val totalCount: Int, | ||||||
|  |     val items: List<GetAuditionListItem> | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class GetAuditionListItem @QueryProjection constructor( | ||||||
|  |     val id: Long, | ||||||
|  |     val title: String, | ||||||
|  |     val imageUrl: String, | ||||||
|  |     val isAdult: Boolean, | ||||||
|  |     val information: String, | ||||||
|  |     val status: AuditionStatus, | ||||||
|  |     val originalWorkUrl: String | ||||||
|  | ) | ||||||
| @@ -0,0 +1,13 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.audition.AuditionStatus | ||||||
|  |  | ||||||
|  | data class UpdateAuditionRequest( | ||||||
|  |     val id: Long, | ||||||
|  |     val title: String? = null, | ||||||
|  |     val information: String? = null, | ||||||
|  |     val isAdult: Boolean? = null, | ||||||
|  |     val status: AuditionStatus? = null, | ||||||
|  |     val originalWorkUrl: String? = null, | ||||||
|  |     val isActive: Boolean? = null | ||||||
|  | ) | ||||||
| @@ -0,0 +1,19 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition.applicant | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.common.ApiResponse | ||||||
|  | import org.springframework.security.access.prepost.PreAuthorize | ||||||
|  | import org.springframework.web.bind.annotation.DeleteMapping | ||||||
|  | import org.springframework.web.bind.annotation.PathVariable | ||||||
|  | import org.springframework.web.bind.annotation.RequestMapping | ||||||
|  | import org.springframework.web.bind.annotation.RestController | ||||||
|  |  | ||||||
|  | @RestController | ||||||
|  | @PreAuthorize("hasRole('ADMIN')") | ||||||
|  | @RequestMapping("/admin/audition/applicant") | ||||||
|  | class AdminAuditionApplicantController(private val service: AdminAuditionApplicantService) { | ||||||
|  |     @DeleteMapping("/{id}") | ||||||
|  |     fun deleteAuditionApplicant(@PathVariable id: Long) = ApiResponse.ok( | ||||||
|  |         service.deleteAuditionApplicant(id), | ||||||
|  |         "오디션 지원이 취소 되었습니다." | ||||||
|  |     ) | ||||||
|  | } | ||||||
| @@ -0,0 +1,6 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition.applicant | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.audition.AuditionApplicant | ||||||
|  | import org.springframework.data.jpa.repository.JpaRepository | ||||||
|  |  | ||||||
|  | interface AdminAuditionApplicantRepository : JpaRepository<AuditionApplicant, Long> | ||||||
| @@ -0,0 +1,17 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition.applicant | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.common.SodaException | ||||||
|  | import org.springframework.data.repository.findByIdOrNull | ||||||
|  | import org.springframework.stereotype.Service | ||||||
|  | import org.springframework.transaction.annotation.Transactional | ||||||
|  |  | ||||||
|  | @Service | ||||||
|  | class AdminAuditionApplicantService(private val repository: AdminAuditionApplicantRepository) { | ||||||
|  |     @Transactional | ||||||
|  |     fun deleteAuditionApplicant(id: Long) { | ||||||
|  |         val applicant = repository.findByIdOrNull(id) | ||||||
|  |             ?: throw SodaException("잘못된 요청입니다.") | ||||||
|  |  | ||||||
|  |         applicant.isActive = false | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,47 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition.role | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.common.ApiResponse | ||||||
|  | import org.springframework.data.domain.Pageable | ||||||
|  | import org.springframework.security.access.prepost.PreAuthorize | ||||||
|  | import org.springframework.web.bind.annotation.GetMapping | ||||||
|  | import org.springframework.web.bind.annotation.PathVariable | ||||||
|  | import org.springframework.web.bind.annotation.PostMapping | ||||||
|  | import org.springframework.web.bind.annotation.PutMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestPart | ||||||
|  | import org.springframework.web.bind.annotation.RestController | ||||||
|  | import org.springframework.web.multipart.MultipartFile | ||||||
|  |  | ||||||
|  | @RestController | ||||||
|  | @PreAuthorize("hasRole('ADMIN')") | ||||||
|  | @RequestMapping("/admin/audition/role") | ||||||
|  | class AdminAuditionRoleController(private val service: AdminAuditionRoleService) { | ||||||
|  |     @PostMapping | ||||||
|  |     fun createAuditionRole( | ||||||
|  |         @RequestPart("image") image: MultipartFile, | ||||||
|  |         @RequestPart("request") requestString: String | ||||||
|  |     ) = ApiResponse.ok(service.createAuditionRole(image, requestString), "등록되었습니다.") | ||||||
|  |  | ||||||
|  |     @PutMapping | ||||||
|  |     fun updateAuditionRole( | ||||||
|  |         @RequestPart("image", required = false) image: MultipartFile? = null, | ||||||
|  |         @RequestPart("request") requestString: String | ||||||
|  |     ) = ApiResponse.ok(service.updateAuditionRole(image, requestString), "수정되었습니다.") | ||||||
|  |  | ||||||
|  |     @GetMapping("/{id}") | ||||||
|  |     fun getAuditionRoleDetail(@PathVariable id: Long) = ApiResponse.ok( | ||||||
|  |         service.getAuditionRoleDetail(auditionRoleId = id) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     @GetMapping("/{id}/applicant") | ||||||
|  |     fun getAuditionApplicantList( | ||||||
|  |         @PathVariable id: Long, | ||||||
|  |         pageable: Pageable | ||||||
|  |     ) = ApiResponse.ok( | ||||||
|  |         service.getAuditionApplicantList( | ||||||
|  |             auditionRoleId = id, | ||||||
|  |             offset = pageable.offset, | ||||||
|  |             limit = pageable.pageSize.toLong() | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  | } | ||||||
| @@ -0,0 +1,106 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition.role | ||||||
|  |  | ||||||
|  | import com.querydsl.jpa.impl.JPAQueryFactory | ||||||
|  | import kr.co.vividnext.sodalive.admin.audition.GetAuditionRoleListData | ||||||
|  | import kr.co.vividnext.sodalive.admin.audition.QGetAuditionRoleListData | ||||||
|  | import kr.co.vividnext.sodalive.audition.AuditionRole | ||||||
|  | import kr.co.vividnext.sodalive.audition.QAudition.audition | ||||||
|  | import kr.co.vividnext.sodalive.audition.QAuditionApplicant.auditionApplicant | ||||||
|  | import kr.co.vividnext.sodalive.audition.QAuditionRole.auditionRole | ||||||
|  | import kr.co.vividnext.sodalive.audition.QAuditionVote.auditionVote | ||||||
|  | import kr.co.vividnext.sodalive.member.QMember.member | ||||||
|  | import org.springframework.beans.factory.annotation.Value | ||||||
|  | import org.springframework.data.jpa.repository.JpaRepository | ||||||
|  |  | ||||||
|  | interface AdminAuditionRoleRepository : JpaRepository<AuditionRole, Long>, AdminAuditionRoleQueryRepository | ||||||
|  |  | ||||||
|  | interface AdminAuditionRoleQueryRepository { | ||||||
|  |     fun getAuditionRoleListByAuditionId(auditionId: Long): List<GetAuditionRoleListData> | ||||||
|  |     fun getAuditionRoleDetail(auditionRoleId: Long): GetAuditionRoleDetailResponse | ||||||
|  |     fun getAuditionApplicantList(auditionRoleId: Long, offset: Long, limit: Long): List<GetAuditionRoleApplicantItem> | ||||||
|  |     fun getAuditionApplicantTotalCount(auditionRoleId: Long): Int | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class AdminAuditionRoleQueryRepositoryImpl( | ||||||
|  |     private val queryFactory: JPAQueryFactory, | ||||||
|  |  | ||||||
|  |     @Value("\${cloud.aws.cloud-front.host}") | ||||||
|  |     private val cloudfrontHost: String | ||||||
|  | ) : AdminAuditionRoleQueryRepository { | ||||||
|  |     override fun getAuditionRoleListByAuditionId(auditionId: Long): List<GetAuditionRoleListData> { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetAuditionRoleListData( | ||||||
|  |                     auditionRole.id, | ||||||
|  |                     auditionRole.name, | ||||||
|  |                     auditionRole.imagePath.prepend("/").prepend(cloudfrontHost), | ||||||
|  |                     auditionRole.information, | ||||||
|  |                     auditionRole.auditionScriptUrl, | ||||||
|  |                     auditionRole.status | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(auditionRole) | ||||||
|  |             .innerJoin(auditionRole.audition, audition) | ||||||
|  |             .where( | ||||||
|  |                 auditionRole.audition.id.eq(auditionId), | ||||||
|  |                 auditionRole.isActive.isTrue | ||||||
|  |             ) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun getAuditionRoleDetail(auditionRoleId: Long): GetAuditionRoleDetailResponse { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetAuditionRoleDetailResponse( | ||||||
|  |                     auditionRole.name, | ||||||
|  |                     auditionRole.imagePath.prepend("/").prepend(cloudfrontHost), | ||||||
|  |                     auditionRole.information, | ||||||
|  |                     auditionRole.auditionScriptUrl | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(auditionRole) | ||||||
|  |             .where(auditionRole.id.eq(auditionRoleId)) | ||||||
|  |             .fetchFirst() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun getAuditionApplicantList( | ||||||
|  |         auditionRoleId: Long, | ||||||
|  |         offset: Long, | ||||||
|  |         limit: Long | ||||||
|  |     ): List<GetAuditionRoleApplicantItem> { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetAuditionRoleApplicantItem( | ||||||
|  |                     auditionApplicant.id, | ||||||
|  |                     member.nickname, | ||||||
|  |                     member.profileImage.prepend("/").prepend(cloudfrontHost), | ||||||
|  |                     auditionApplicant.phoneNumber, | ||||||
|  |                     auditionApplicant.voicePath.prepend("/").prepend(cloudfrontHost), | ||||||
|  |                     auditionVote.id.count() | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(auditionApplicant) | ||||||
|  |             .innerJoin(auditionApplicant.member, member) | ||||||
|  |             .innerJoin(auditionApplicant.role, auditionRole) | ||||||
|  |             .leftJoin(auditionVote).on(auditionApplicant.id.eq(auditionVote.applicant.id)) | ||||||
|  |             .where( | ||||||
|  |                 auditionRole.id.eq(auditionRoleId), | ||||||
|  |                 auditionApplicant.isActive.isTrue | ||||||
|  |             ) | ||||||
|  |             .groupBy(auditionApplicant.id) | ||||||
|  |             .orderBy(auditionVote.id.count().desc()) | ||||||
|  |             .offset(offset) | ||||||
|  |             .limit(limit) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun getAuditionApplicantTotalCount(auditionRoleId: Long): Int { | ||||||
|  |         return queryFactory | ||||||
|  |             .select(auditionApplicant.id) | ||||||
|  |             .from(auditionApplicant) | ||||||
|  |             .innerJoin(auditionApplicant.role, auditionRole) | ||||||
|  |             .where(auditionRole.id.eq(auditionRoleId)) | ||||||
|  |             .fetch() | ||||||
|  |             .size | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,95 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition.role | ||||||
|  |  | ||||||
|  | import com.fasterxml.jackson.databind.ObjectMapper | ||||||
|  | import kr.co.vividnext.sodalive.admin.audition.AdminAuditionRepository | ||||||
|  | import kr.co.vividnext.sodalive.audition.AuditionRole | ||||||
|  | import kr.co.vividnext.sodalive.aws.s3.S3Uploader | ||||||
|  | import kr.co.vividnext.sodalive.common.SodaException | ||||||
|  | import kr.co.vividnext.sodalive.utils.generateFileName | ||||||
|  | import org.springframework.beans.factory.annotation.Value | ||||||
|  | import org.springframework.data.repository.findByIdOrNull | ||||||
|  | import org.springframework.stereotype.Service | ||||||
|  | import org.springframework.transaction.annotation.Transactional | ||||||
|  | import org.springframework.web.multipart.MultipartFile | ||||||
|  |  | ||||||
|  | @Service | ||||||
|  | class AdminAuditionRoleService( | ||||||
|  |     private val s3Uploader: S3Uploader, | ||||||
|  |     private val objectMapper: ObjectMapper, | ||||||
|  |     private val repository: AdminAuditionRoleRepository, | ||||||
|  |     private val auditionRepository: AdminAuditionRepository, | ||||||
|  |  | ||||||
|  |     @Value("\${cloud.aws.s3.bucket}") | ||||||
|  |     private val bucket: String | ||||||
|  | ) { | ||||||
|  |     @Transactional | ||||||
|  |     fun createAuditionRole(image: MultipartFile, requestString: String) { | ||||||
|  |         val request = objectMapper.readValue(requestString, CreateAuditionRoleRequest::class.java) | ||||||
|  |         val auditionRole = AuditionRole( | ||||||
|  |             name = request.name, | ||||||
|  |             information = request.information, | ||||||
|  |             auditionScriptUrl = request.auditionScriptUrl | ||||||
|  |         ) | ||||||
|  |         val audition = auditionRepository.findByIdOrNull(id = request.auditionId) | ||||||
|  |             ?: throw SodaException("잘못된 요청입니다.\n다시 시도해 주세요.") | ||||||
|  |         auditionRole.audition = audition | ||||||
|  |         repository.save(auditionRole) | ||||||
|  |  | ||||||
|  |         val fileName = generateFileName("audition_role") | ||||||
|  |         val imagePath = s3Uploader.upload( | ||||||
|  |             inputStream = image.inputStream, | ||||||
|  |             bucket = bucket, | ||||||
|  |             filePath = "audition/role/${auditionRole.id}/$fileName" | ||||||
|  |         ) | ||||||
|  |         auditionRole.imagePath = imagePath | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional | ||||||
|  |     fun updateAuditionRole(image: MultipartFile?, requestString: String) { | ||||||
|  |         val request = objectMapper.readValue(requestString, UpdateAuditionRoleRequest::class.java) | ||||||
|  |         val auditionRole = repository.findByIdOrNull(id = request.id) | ||||||
|  |             ?: throw SodaException("잘못된 요청입니다.\n다시 시도해 주세요.") | ||||||
|  |  | ||||||
|  |         if (!request.name.isNullOrBlank()) { | ||||||
|  |             if (request.name.length < 2) throw SodaException("배역 이름은 최소 2글자 입니다") | ||||||
|  |             auditionRole.name = request.name | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (!request.information.isNullOrBlank()) { | ||||||
|  |             if (request.information.length < 10) throw SodaException("오디션 배역 정보는 최소 10글자 입니다") | ||||||
|  |             auditionRole.information = request.information | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (!request.auditionScriptUrl.isNullOrBlank()) { | ||||||
|  |             auditionRole.auditionScriptUrl = request.auditionScriptUrl | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (request.status != null) { | ||||||
|  |             auditionRole.status = request.status | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (request.isActive != null) { | ||||||
|  |             auditionRole.isActive = request.isActive | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (image != null) { | ||||||
|  |             val fileName = generateFileName("audition_role") | ||||||
|  |             val imagePath = s3Uploader.upload( | ||||||
|  |                 inputStream = image.inputStream, | ||||||
|  |                 bucket = bucket, | ||||||
|  |                 filePath = "audition/role/${auditionRole.id}/$fileName" | ||||||
|  |             ) | ||||||
|  |             auditionRole.imagePath = imagePath | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getAuditionRoleDetail(auditionRoleId: Long): GetAuditionRoleDetailResponse { | ||||||
|  |         return repository.getAuditionRoleDetail(auditionRoleId = auditionRoleId) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getAuditionApplicantList(auditionRoleId: Long, offset: Long, limit: Long): GetAuditionRoleApplicantResponse { | ||||||
|  |         val totalCount = repository.getAuditionApplicantTotalCount(auditionRoleId = auditionRoleId) | ||||||
|  |         val items = repository.getAuditionApplicantList(auditionRoleId = auditionRoleId, offset = offset, limit = limit) | ||||||
|  |         return GetAuditionRoleApplicantResponse(totalCount, items) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,28 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition.role | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.common.SodaException | ||||||
|  |  | ||||||
|  | data class CreateAuditionRoleRequest( | ||||||
|  |     val auditionId: Long, | ||||||
|  |     val name: String, | ||||||
|  |     val information: String, | ||||||
|  |     val auditionScriptUrl: String | ||||||
|  | ) { | ||||||
|  |     init { | ||||||
|  |         if (auditionId < 0) { | ||||||
|  |             throw SodaException("캐릭터가 등록될 오디션을 선택하세요") | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (name.isBlank() || name.length < 2) { | ||||||
|  |             throw SodaException("캐릭터명을 입력하세요") | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (auditionScriptUrl.isBlank() || auditionScriptUrl.length < 10) { | ||||||
|  |             throw SodaException("오디션 대본 URL을 입력하세요") | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (information.isBlank() || information.length < 10) { | ||||||
|  |             throw SodaException("오디션 캐릭터 정보는 최소 10글자 입니다") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,17 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition.role | ||||||
|  |  | ||||||
|  | import com.querydsl.core.annotations.QueryProjection | ||||||
|  |  | ||||||
|  | data class GetAuditionRoleApplicantResponse( | ||||||
|  |     val totalCount: Int, | ||||||
|  |     val items: List<GetAuditionRoleApplicantItem> | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class GetAuditionRoleApplicantItem @QueryProjection constructor( | ||||||
|  |     val applicantId: Long, | ||||||
|  |     val nickname: String, | ||||||
|  |     val profileImageUrl: String, | ||||||
|  |     val phoneNumber: String, | ||||||
|  |     val voiceUrl: String, | ||||||
|  |     val voteCount: Long | ||||||
|  | ) | ||||||
| @@ -0,0 +1,10 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition.role | ||||||
|  |  | ||||||
|  | import com.querydsl.core.annotations.QueryProjection | ||||||
|  |  | ||||||
|  | data class GetAuditionRoleDetailResponse @QueryProjection constructor( | ||||||
|  |     val name: String, | ||||||
|  |     val imageUrl: String, | ||||||
|  |     val information: String, | ||||||
|  |     val auditionScriptUrl: String | ||||||
|  | ) | ||||||
| @@ -0,0 +1,19 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.audition.role | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.audition.AuditionStatus | ||||||
|  | import kr.co.vividnext.sodalive.common.SodaException | ||||||
|  |  | ||||||
|  | data class UpdateAuditionRoleRequest( | ||||||
|  |     val id: Long, | ||||||
|  |     val name: String? = null, | ||||||
|  |     val information: String? = null, | ||||||
|  |     val auditionScriptUrl: String? = null, | ||||||
|  |     val status: AuditionStatus? = null, | ||||||
|  |     val isActive: Boolean? = null | ||||||
|  | ) { | ||||||
|  |     init { | ||||||
|  |         if (id < 0) { | ||||||
|  |             throw SodaException("잘못된 요청입니다.") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,93 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.common.ApiResponse | ||||||
|  | import org.springframework.data.domain.Pageable | ||||||
|  | import org.springframework.security.access.prepost.PreAuthorize | ||||||
|  | import org.springframework.web.bind.annotation.GetMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestParam | ||||||
|  | import org.springframework.web.bind.annotation.RestController | ||||||
|  |  | ||||||
|  | @RestController | ||||||
|  | @PreAuthorize("hasRole('ADMIN')") | ||||||
|  | @RequestMapping("/admin/calculate") | ||||||
|  | class AdminCalculateController(private val service: AdminCalculateService) { | ||||||
|  |     @GetMapping("/live") | ||||||
|  |     fun getCalculateLive( | ||||||
|  |         @RequestParam startDateStr: String, | ||||||
|  |         @RequestParam endDateStr: String | ||||||
|  |     ) = ApiResponse.ok(service.getCalculateLive(startDateStr, endDateStr)) | ||||||
|  |  | ||||||
|  |     @GetMapping("/content-list") | ||||||
|  |     fun getCalculateContentList( | ||||||
|  |         @RequestParam startDateStr: String, | ||||||
|  |         @RequestParam endDateStr: String | ||||||
|  |     ) = ApiResponse.ok(service.getCalculateContentList(startDateStr, endDateStr)) | ||||||
|  |  | ||||||
|  |     @GetMapping("/cumulative-sales-by-content") | ||||||
|  |     fun getCumulativeSalesByContent(pageable: Pageable) = ApiResponse.ok( | ||||||
|  |         service.getCumulativeSalesByContent(pageable.offset, pageable.pageSize.toLong()) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     @GetMapping("/content-donation-list") | ||||||
|  |     fun getCalculateContentDonationList( | ||||||
|  |         @RequestParam startDateStr: String, | ||||||
|  |         @RequestParam endDateStr: String | ||||||
|  |     ) = ApiResponse.ok(service.getCalculateContentDonationList(startDateStr, endDateStr)) | ||||||
|  |  | ||||||
|  |     @GetMapping("/community-post") | ||||||
|  |     fun getCalculateCommunityPost( | ||||||
|  |         @RequestParam startDateStr: String, | ||||||
|  |         @RequestParam endDateStr: String, | ||||||
|  |         pageable: Pageable | ||||||
|  |     ) = ApiResponse.ok( | ||||||
|  |         service.getCalculateCommunityPost( | ||||||
|  |             startDateStr, | ||||||
|  |             endDateStr, | ||||||
|  |             pageable.offset, | ||||||
|  |             pageable.pageSize.toLong() | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     @GetMapping("/live-by-creator") | ||||||
|  |     fun getCalculateLiveByCreator( | ||||||
|  |         @RequestParam startDateStr: String, | ||||||
|  |         @RequestParam endDateStr: String, | ||||||
|  |         pageable: Pageable | ||||||
|  |     ) = ApiResponse.ok( | ||||||
|  |         service.getCalculateLiveByCreator( | ||||||
|  |             startDateStr, | ||||||
|  |             endDateStr, | ||||||
|  |             pageable.offset, | ||||||
|  |             pageable.pageSize.toLong() | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     @GetMapping("/content-by-creator") | ||||||
|  |     fun getCalculateContentByCreator( | ||||||
|  |         @RequestParam startDateStr: String, | ||||||
|  |         @RequestParam endDateStr: String, | ||||||
|  |         pageable: Pageable | ||||||
|  |     ) = ApiResponse.ok( | ||||||
|  |         service.getCalculateContentByCreator( | ||||||
|  |             startDateStr, | ||||||
|  |             endDateStr, | ||||||
|  |             pageable.offset, | ||||||
|  |             pageable.pageSize.toLong() | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     @GetMapping("/community-by-creator") | ||||||
|  |     fun getCalculateCommunityByCreator( | ||||||
|  |         @RequestParam startDateStr: String, | ||||||
|  |         @RequestParam endDateStr: String, | ||||||
|  |         pageable: Pageable | ||||||
|  |     ) = ApiResponse.ok( | ||||||
|  |         service.getCalculateCommunityByCreator( | ||||||
|  |             startDateStr, | ||||||
|  |             endDateStr, | ||||||
|  |             pageable.offset, | ||||||
|  |             pageable.pageSize.toLong() | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  | } | ||||||
| @@ -0,0 +1,428 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate | ||||||
|  |  | ||||||
|  | import com.querydsl.core.types.dsl.CaseBuilder | ||||||
|  | import com.querydsl.core.types.dsl.DateTimePath | ||||||
|  | import com.querydsl.core.types.dsl.Expressions | ||||||
|  | import com.querydsl.core.types.dsl.StringTemplate | ||||||
|  | import com.querydsl.jpa.impl.JPAQueryFactory | ||||||
|  | import kr.co.vividnext.sodalive.admin.calculate.ratio.QCreatorSettlementRatio.creatorSettlementRatio | ||||||
|  | import kr.co.vividnext.sodalive.can.use.CanUsage | ||||||
|  | import kr.co.vividnext.sodalive.can.use.QUseCan.useCan | ||||||
|  | import kr.co.vividnext.sodalive.content.QAudioContent.audioContent | ||||||
|  | import kr.co.vividnext.sodalive.content.order.QOrder.order | ||||||
|  | import kr.co.vividnext.sodalive.explorer.profile.creatorCommunity.QCreatorCommunity.creatorCommunity | ||||||
|  | import kr.co.vividnext.sodalive.live.room.QLiveRoom.liveRoom | ||||||
|  | import kr.co.vividnext.sodalive.member.QMember.member | ||||||
|  | import org.springframework.stereotype.Repository | ||||||
|  | import java.time.LocalDateTime | ||||||
|  |  | ||||||
|  | @Repository | ||||||
|  | class AdminCalculateQueryRepository(private val queryFactory: JPAQueryFactory) { | ||||||
|  |     fun getCalculateLive(startDate: LocalDateTime, endDate: LocalDateTime): List<GetCalculateLiveQueryData> { | ||||||
|  |         val formattedDate = getFormattedDate(liveRoom.beginDateTime) | ||||||
|  |  | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetCalculateLiveQueryData( | ||||||
|  |                     member.email, | ||||||
|  |                     member.nickname, | ||||||
|  |                     formattedDate, | ||||||
|  |                     liveRoom.title, | ||||||
|  |                     liveRoom.price, | ||||||
|  |                     useCan.canUsage, | ||||||
|  |                     useCan.id.count(), | ||||||
|  |                     useCan.can.add(useCan.rewardCan).sum(), | ||||||
|  |                     creatorSettlementRatio.liveSettlementRatio | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(useCan) | ||||||
|  |             .innerJoin(useCan.room, liveRoom) | ||||||
|  |             .innerJoin(liveRoom.member, member) | ||||||
|  |             .leftJoin(creatorSettlementRatio) | ||||||
|  |             .on( | ||||||
|  |                 member.id.eq(creatorSettlementRatio.member.id) | ||||||
|  |                     .and(creatorSettlementRatio.deletedAt.isNull) | ||||||
|  |             ) | ||||||
|  |             .where( | ||||||
|  |                 useCan.isRefund.isFalse | ||||||
|  |                     .and(useCan.createdAt.goe(startDate)) | ||||||
|  |                     .and(useCan.createdAt.loe(endDate)) | ||||||
|  |             ) | ||||||
|  |             .groupBy(liveRoom.id, useCan.canUsage, creatorSettlementRatio.liveSettlementRatio) | ||||||
|  |             .orderBy(member.nickname.desc(), liveRoom.id.desc(), useCan.canUsage.desc(), formattedDate.desc()) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCalculateContentList(startDate: LocalDateTime, endDate: LocalDateTime): List<GetCalculateContentQueryData> { | ||||||
|  |         val orderFormattedDate = getFormattedDate(order.createdAt) | ||||||
|  |         val pointGroup = CaseBuilder() | ||||||
|  |             .`when`(order.point.loe(0)).then(0) | ||||||
|  |             .otherwise(1) | ||||||
|  |  | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetCalculateContentQueryData( | ||||||
|  |                     member.nickname, | ||||||
|  |                     audioContent.title, | ||||||
|  |                     getFormattedDate(audioContent.createdAt), | ||||||
|  |                     orderFormattedDate, | ||||||
|  |                     order.type, | ||||||
|  |                     order.can, | ||||||
|  |                     order.id.count(), | ||||||
|  |                     order.can.sum(), | ||||||
|  |                     order.point.sum(), | ||||||
|  |                     creatorSettlementRatio.contentSettlementRatio | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(order) | ||||||
|  |             .innerJoin(order.audioContent, audioContent) | ||||||
|  |             .innerJoin(audioContent.member, member) | ||||||
|  |             .leftJoin(creatorSettlementRatio) | ||||||
|  |             .on( | ||||||
|  |                 member.id.eq(creatorSettlementRatio.member.id) | ||||||
|  |                     .and(creatorSettlementRatio.deletedAt.isNull) | ||||||
|  |             ) | ||||||
|  |             .where( | ||||||
|  |                 order.createdAt.goe(startDate) | ||||||
|  |                     .and(order.createdAt.loe(endDate)) | ||||||
|  |                     .and(order.isActive.isTrue) | ||||||
|  |             ) | ||||||
|  |             .groupBy( | ||||||
|  |                 audioContent.id, | ||||||
|  |                 order.type, | ||||||
|  |                 orderFormattedDate, | ||||||
|  |                 order.can, | ||||||
|  |                 pointGroup, | ||||||
|  |                 creatorSettlementRatio.contentSettlementRatio | ||||||
|  |             ) | ||||||
|  |             .orderBy(member.id.desc(), orderFormattedDate.desc(), audioContent.id.asc()) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun getFormattedDate(dateTimePath: DateTimePath<LocalDateTime>): StringTemplate { | ||||||
|  |         return Expressions.stringTemplate( | ||||||
|  |             "DATE_FORMAT({0}, {1})", | ||||||
|  |             Expressions.dateTimeTemplate( | ||||||
|  |                 LocalDateTime::class.java, | ||||||
|  |                 "CONVERT_TZ({0},{1},{2})", | ||||||
|  |                 dateTimePath, | ||||||
|  |                 "UTC", | ||||||
|  |                 "Asia/Seoul" | ||||||
|  |             ), | ||||||
|  |             "%Y-%m-%d" | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCumulativeSalesByContentTotalCount(): Int { | ||||||
|  |         return queryFactory | ||||||
|  |             .select(audioContent.id) | ||||||
|  |             .from(order) | ||||||
|  |             .innerJoin(order.audioContent, audioContent) | ||||||
|  |             .innerJoin(audioContent.member, member) | ||||||
|  |             .where(order.isActive.isTrue) | ||||||
|  |             .groupBy(member.id, audioContent.id, order.can) | ||||||
|  |             .fetch() | ||||||
|  |             .size | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCumulativeSalesByContent(offset: Long, limit: Long): List<GetCumulativeSalesByContentQueryData> { | ||||||
|  |         val pointGroup = CaseBuilder() | ||||||
|  |             .`when`(order.point.loe(0)).then(0) | ||||||
|  |             .otherwise(1) | ||||||
|  |  | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetCumulativeSalesByContentQueryData( | ||||||
|  |                     member.nickname, | ||||||
|  |                     audioContent.title, | ||||||
|  |                     getFormattedDate(audioContent.createdAt), | ||||||
|  |                     order.type, | ||||||
|  |                     order.can, | ||||||
|  |                     order.id.count(), | ||||||
|  |                     order.can.sum(), | ||||||
|  |                     order.point.sum(), | ||||||
|  |                     creatorSettlementRatio.contentSettlementRatio | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(order) | ||||||
|  |             .innerJoin(order.audioContent, audioContent) | ||||||
|  |             .innerJoin(audioContent.member, member) | ||||||
|  |             .leftJoin(creatorSettlementRatio) | ||||||
|  |             .on( | ||||||
|  |                 member.id.eq(creatorSettlementRatio.member.id) | ||||||
|  |                     .and(creatorSettlementRatio.deletedAt.isNull) | ||||||
|  |             ) | ||||||
|  |             .where(order.isActive.isTrue) | ||||||
|  |             .groupBy( | ||||||
|  |                 member.id, | ||||||
|  |                 audioContent.id, | ||||||
|  |                 order.type, | ||||||
|  |                 order.can, | ||||||
|  |                 pointGroup, | ||||||
|  |                 creatorSettlementRatio.contentSettlementRatio | ||||||
|  |             ) | ||||||
|  |             .offset(offset) | ||||||
|  |             .limit(limit) | ||||||
|  |             .orderBy(member.id.desc(), audioContent.id.desc()) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCalculateContentDonationList( | ||||||
|  |         startDate: LocalDateTime, | ||||||
|  |         endDate: LocalDateTime | ||||||
|  |     ): List<GetCalculateContentDonationQueryData> { | ||||||
|  |         val donationFormattedDate = getFormattedDate(useCan.createdAt) | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetCalculateContentDonationQueryData( | ||||||
|  |                     member.nickname, | ||||||
|  |                     audioContent.title, | ||||||
|  |                     audioContent.price, | ||||||
|  |                     getFormattedDate(audioContent.createdAt), | ||||||
|  |                     donationFormattedDate, | ||||||
|  |                     useCan.id.count(), | ||||||
|  |                     useCan.can.add(useCan.rewardCan).sum() | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(useCan) | ||||||
|  |             .innerJoin(useCan.audioContent, audioContent) | ||||||
|  |             .innerJoin(audioContent.member, member) | ||||||
|  |             .where( | ||||||
|  |                 useCan.isRefund.isFalse | ||||||
|  |                     .and(useCan.canUsage.eq(CanUsage.DONATION)) | ||||||
|  |                     .and(useCan.createdAt.goe(startDate)) | ||||||
|  |                     .and(useCan.createdAt.loe(endDate)) | ||||||
|  |             ) | ||||||
|  |             .groupBy(donationFormattedDate, audioContent.id) | ||||||
|  |             .orderBy(member.id.asc(), donationFormattedDate.desc(), audioContent.id.desc()) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCalculateCommunityPostTotalCount(startDate: LocalDateTime?, endDate: LocalDateTime?): Int { | ||||||
|  |         val formattedDate = getFormattedDate(useCan.createdAt) | ||||||
|  |         return queryFactory | ||||||
|  |             .select(creatorCommunity.id) | ||||||
|  |             .from(useCan) | ||||||
|  |             .innerJoin(useCan.communityPost, creatorCommunity) | ||||||
|  |             .innerJoin(creatorCommunity.member, member) | ||||||
|  |             .where( | ||||||
|  |                 useCan.isRefund.isFalse | ||||||
|  |                     .and(useCan.canUsage.eq(CanUsage.PAID_COMMUNITY_POST)) | ||||||
|  |                     .and(useCan.createdAt.goe(startDate)) | ||||||
|  |                     .and(useCan.createdAt.loe(endDate)) | ||||||
|  |             ) | ||||||
|  |             .groupBy(formattedDate, creatorCommunity.id) | ||||||
|  |             .fetch() | ||||||
|  |             .size | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCalculateCommunityPostList( | ||||||
|  |         startDate: LocalDateTime?, | ||||||
|  |         endDate: LocalDateTime?, | ||||||
|  |         offset: Long, | ||||||
|  |         limit: Long | ||||||
|  |     ): List<GetCalculateCommunityPostQueryData> { | ||||||
|  |         val formattedDate = getFormattedDate(useCan.createdAt) | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetCalculateCommunityPostQueryData( | ||||||
|  |                     member.nickname, | ||||||
|  |                     Expressions.stringTemplate("substring({0}, 1, 10)", creatorCommunity.content), | ||||||
|  |                     formattedDate, | ||||||
|  |                     creatorCommunity.price, | ||||||
|  |                     useCan.id.count(), | ||||||
|  |                     useCan.can.add(useCan.rewardCan).sum(), | ||||||
|  |                     creatorSettlementRatio.communitySettlementRatio | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(useCan) | ||||||
|  |             .innerJoin(useCan.communityPost, creatorCommunity) | ||||||
|  |             .innerJoin(creatorCommunity.member, member) | ||||||
|  |             .leftJoin(creatorSettlementRatio) | ||||||
|  |             .on( | ||||||
|  |                 member.id.eq(creatorSettlementRatio.member.id) | ||||||
|  |                     .and(creatorSettlementRatio.deletedAt.isNull) | ||||||
|  |             ) | ||||||
|  |             .where( | ||||||
|  |                 useCan.isRefund.isFalse | ||||||
|  |                     .and(useCan.canUsage.eq(CanUsage.PAID_COMMUNITY_POST)) | ||||||
|  |                     .and(useCan.createdAt.goe(startDate)) | ||||||
|  |                     .and(useCan.createdAt.loe(endDate)) | ||||||
|  |             ) | ||||||
|  |             .groupBy(formattedDate, creatorCommunity.id, creatorSettlementRatio.communitySettlementRatio) | ||||||
|  |             .orderBy(member.id.asc(), formattedDate.desc(), creatorCommunity.id.desc()) | ||||||
|  |             .offset(offset) | ||||||
|  |             .limit(limit) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCalculateLiveByCreatorTotalCount(startDate: LocalDateTime, endDate: LocalDateTime): Int { | ||||||
|  |         return queryFactory | ||||||
|  |             .select(member.id) | ||||||
|  |             .from(useCan) | ||||||
|  |             .innerJoin(useCan.room, liveRoom) | ||||||
|  |             .innerJoin(liveRoom.member, member) | ||||||
|  |             .leftJoin(creatorSettlementRatio) | ||||||
|  |             .on( | ||||||
|  |                 member.id.eq(creatorSettlementRatio.member.id) | ||||||
|  |                     .and(creatorSettlementRatio.deletedAt.isNull) | ||||||
|  |             ) | ||||||
|  |             .where( | ||||||
|  |                 useCan.isRefund.isFalse | ||||||
|  |                     .and(useCan.createdAt.goe(startDate)) | ||||||
|  |                     .and(useCan.createdAt.loe(endDate)) | ||||||
|  |             ) | ||||||
|  |             .groupBy(member.id) | ||||||
|  |             .fetch() | ||||||
|  |             .size | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCalculateLiveByCreator( | ||||||
|  |         startDate: LocalDateTime, | ||||||
|  |         endDate: LocalDateTime, | ||||||
|  |         offset: Long, | ||||||
|  |         limit: Long | ||||||
|  |     ): List<GetCalculateByCreatorQueryData> { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetCalculateByCreatorQueryData( | ||||||
|  |                     member.email, | ||||||
|  |                     member.nickname, | ||||||
|  |                     useCan.can.add(useCan.rewardCan).sum(), | ||||||
|  |                     creatorSettlementRatio.liveSettlementRatio | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(useCan) | ||||||
|  |             .innerJoin(useCan.room, liveRoom) | ||||||
|  |             .innerJoin(liveRoom.member, member) | ||||||
|  |             .leftJoin(creatorSettlementRatio) | ||||||
|  |             .on( | ||||||
|  |                 member.id.eq(creatorSettlementRatio.member.id) | ||||||
|  |                     .and(creatorSettlementRatio.deletedAt.isNull) | ||||||
|  |             ) | ||||||
|  |             .where( | ||||||
|  |                 useCan.isRefund.isFalse | ||||||
|  |                     .and(useCan.createdAt.goe(startDate)) | ||||||
|  |                     .and(useCan.createdAt.loe(endDate)) | ||||||
|  |             ) | ||||||
|  |             .groupBy(member.id, creatorSettlementRatio.liveSettlementRatio) | ||||||
|  |             .orderBy(member.id.desc()) | ||||||
|  |             .offset(offset) | ||||||
|  |             .limit(limit) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCalculateContentByCreatorTotalCount(startDate: LocalDateTime, endDate: LocalDateTime): Int { | ||||||
|  |         return queryFactory | ||||||
|  |             .select(member.id) | ||||||
|  |             .from(order) | ||||||
|  |             .innerJoin(order.audioContent, audioContent) | ||||||
|  |             .innerJoin(audioContent.member, member) | ||||||
|  |             .leftJoin(creatorSettlementRatio) | ||||||
|  |             .on( | ||||||
|  |                 member.id.eq(creatorSettlementRatio.member.id) | ||||||
|  |                     .and(creatorSettlementRatio.deletedAt.isNull) | ||||||
|  |             ) | ||||||
|  |             .where( | ||||||
|  |                 order.createdAt.goe(startDate) | ||||||
|  |                     .and(order.createdAt.loe(endDate)) | ||||||
|  |                     .and(order.isActive.isTrue) | ||||||
|  |             ) | ||||||
|  |             .groupBy(member.id) | ||||||
|  |             .fetch() | ||||||
|  |             .size | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCalculateContentByCreator( | ||||||
|  |         startDate: LocalDateTime, | ||||||
|  |         endDate: LocalDateTime, | ||||||
|  |         offset: Long, | ||||||
|  |         limit: Long | ||||||
|  |     ): List<GetCalculateByCreatorQueryData> { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetCalculateByCreatorQueryData( | ||||||
|  |                     member.email, | ||||||
|  |                     member.nickname, | ||||||
|  |                     order.can.sum(), | ||||||
|  |                     creatorSettlementRatio.contentSettlementRatio | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(order) | ||||||
|  |             .innerJoin(order.audioContent, audioContent) | ||||||
|  |             .innerJoin(audioContent.member, member) | ||||||
|  |             .leftJoin(creatorSettlementRatio) | ||||||
|  |             .on( | ||||||
|  |                 member.id.eq(creatorSettlementRatio.member.id) | ||||||
|  |                     .and(creatorSettlementRatio.deletedAt.isNull) | ||||||
|  |             ) | ||||||
|  |             .where( | ||||||
|  |                 order.createdAt.goe(startDate) | ||||||
|  |                     .and(order.createdAt.loe(endDate)) | ||||||
|  |                     .and(order.isActive.isTrue) | ||||||
|  |             ) | ||||||
|  |             .groupBy(member.id) | ||||||
|  |             .orderBy(member.id.desc()) | ||||||
|  |             .offset(offset) | ||||||
|  |             .limit(limit) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCalculateCommunityByCreatorTotalCount(startDate: LocalDateTime, endDate: LocalDateTime): Int { | ||||||
|  |         return queryFactory | ||||||
|  |             .select(member.id) | ||||||
|  |             .from(useCan) | ||||||
|  |             .innerJoin(useCan.communityPost, creatorCommunity) | ||||||
|  |             .innerJoin(creatorCommunity.member, member) | ||||||
|  |             .leftJoin(creatorSettlementRatio) | ||||||
|  |             .on( | ||||||
|  |                 member.id.eq(creatorSettlementRatio.member.id) | ||||||
|  |                     .and(creatorSettlementRatio.deletedAt.isNull) | ||||||
|  |             ) | ||||||
|  |             .where( | ||||||
|  |                 useCan.isRefund.isFalse | ||||||
|  |                     .and(useCan.canUsage.eq(CanUsage.PAID_COMMUNITY_POST)) | ||||||
|  |                     .and(useCan.createdAt.goe(startDate)) | ||||||
|  |                     .and(useCan.createdAt.loe(endDate)) | ||||||
|  |             ) | ||||||
|  |             .groupBy(member.id) | ||||||
|  |             .fetch() | ||||||
|  |             .size | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCalculateCommunityByCreator( | ||||||
|  |         startDate: LocalDateTime, | ||||||
|  |         endDate: LocalDateTime, | ||||||
|  |         offset: Long, | ||||||
|  |         limit: Long | ||||||
|  |     ): List<GetCalculateByCreatorQueryData> { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetCalculateByCreatorQueryData( | ||||||
|  |                     member.email, | ||||||
|  |                     member.nickname, | ||||||
|  |                     useCan.can.add(useCan.rewardCan).sum(), | ||||||
|  |                     creatorSettlementRatio.communitySettlementRatio | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(useCan) | ||||||
|  |             .innerJoin(useCan.communityPost, creatorCommunity) | ||||||
|  |             .innerJoin(creatorCommunity.member, member) | ||||||
|  |             .leftJoin(creatorSettlementRatio) | ||||||
|  |             .on( | ||||||
|  |                 member.id.eq(creatorSettlementRatio.member.id) | ||||||
|  |                     .and(creatorSettlementRatio.deletedAt.isNull) | ||||||
|  |             ) | ||||||
|  |             .where( | ||||||
|  |                 useCan.isRefund.isFalse | ||||||
|  |                     .and(useCan.canUsage.eq(CanUsage.PAID_COMMUNITY_POST)) | ||||||
|  |                     .and(useCan.createdAt.goe(startDate)) | ||||||
|  |                     .and(useCan.createdAt.loe(endDate)) | ||||||
|  |             ) | ||||||
|  |             .groupBy(member.id, creatorSettlementRatio.communitySettlementRatio) | ||||||
|  |             .orderBy(member.id.desc()) | ||||||
|  |             .offset(offset) | ||||||
|  |             .limit(limit) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,142 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.creator.admin.calculate.GetCreatorCalculateCommunityPostResponse | ||||||
|  | import kr.co.vividnext.sodalive.extensions.convertLocalDateTime | ||||||
|  | import org.springframework.cache.annotation.Cacheable | ||||||
|  | import org.springframework.stereotype.Service | ||||||
|  | import org.springframework.transaction.annotation.Transactional | ||||||
|  |  | ||||||
|  | @Service | ||||||
|  | class AdminCalculateService(private val repository: AdminCalculateQueryRepository) { | ||||||
|  |     @Transactional(readOnly = true) | ||||||
|  |     @Cacheable( | ||||||
|  |         cacheNames = ["cache_ttl_3_hours"], | ||||||
|  |         key = "'calculateLive:' + " + "#startDateStr + ':' + #endDateStr" | ||||||
|  |     ) | ||||||
|  |     fun getCalculateLive(startDateStr: String, endDateStr: String): List<GetCalculateLiveResponse> { | ||||||
|  |         val startDate = startDateStr.convertLocalDateTime() | ||||||
|  |         val endDate = endDateStr.convertLocalDateTime(hour = 23, minute = 59, second = 59) | ||||||
|  |  | ||||||
|  |         return repository | ||||||
|  |             .getCalculateLive(startDate, endDate) | ||||||
|  |             .map { it.toGetCalculateLiveResponse() } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional(readOnly = true) | ||||||
|  |     @Cacheable( | ||||||
|  |         cacheNames = ["cache_ttl_3_hours"], | ||||||
|  |         key = "'calculateContent:' + " + "#startDateStr + ':' + #endDateStr" | ||||||
|  |     ) | ||||||
|  |     fun getCalculateContentList(startDateStr: String, endDateStr: String): List<GetCalculateContentResponse> { | ||||||
|  |         val startDate = startDateStr.convertLocalDateTime() | ||||||
|  |         val endDate = endDateStr.convertLocalDateTime(hour = 23, minute = 59, second = 59) | ||||||
|  |  | ||||||
|  |         return repository | ||||||
|  |             .getCalculateContentList(startDate, endDate) | ||||||
|  |             .map { it.toGetCalculateContentResponse() } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional(readOnly = true) | ||||||
|  |     @Cacheable( | ||||||
|  |         cacheNames = ["cache_ttl_3_hours"], | ||||||
|  |         key = "'cumulativeSalesByContent:' + " + "#offset + ':' + #limit" | ||||||
|  |     ) | ||||||
|  |     fun getCumulativeSalesByContent(offset: Long, limit: Long): GetCumulativeSalesByContentResponse { | ||||||
|  |         val totalCount = repository.getCumulativeSalesByContentTotalCount() | ||||||
|  |         val items = repository | ||||||
|  |             .getCumulativeSalesByContent(offset, limit) | ||||||
|  |             .map { it.toCumulativeSalesByContentItem() } | ||||||
|  |  | ||||||
|  |         return GetCumulativeSalesByContentResponse(totalCount, items) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional(readOnly = true) | ||||||
|  |     @Cacheable( | ||||||
|  |         cacheNames = ["cache_ttl_3_hours"], | ||||||
|  |         key = "'calculateContentDonationList2:' + " + "#startDateStr + ':' + #endDateStr" | ||||||
|  |     ) | ||||||
|  |     fun getCalculateContentDonationList( | ||||||
|  |         startDateStr: String, | ||||||
|  |         endDateStr: String | ||||||
|  |     ): List<GetCalculateContentDonationResponse> { | ||||||
|  |         val startDate = startDateStr.convertLocalDateTime() | ||||||
|  |         val endDate = endDateStr.convertLocalDateTime(hour = 23, minute = 59, second = 59) | ||||||
|  |  | ||||||
|  |         return repository | ||||||
|  |             .getCalculateContentDonationList(startDate, endDate) | ||||||
|  |             .map { it.toGetCalculateContentDonationResponse() } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional(readOnly = true) | ||||||
|  |     @Cacheable( | ||||||
|  |         cacheNames = ["cache_ttl_3_hours"], | ||||||
|  |         key = "'calculateCommunityPost:' + " + "#startDateStr + ':' + #endDateStr + ':' + #offset" | ||||||
|  |     ) | ||||||
|  |     fun getCalculateCommunityPost( | ||||||
|  |         startDateStr: String, | ||||||
|  |         endDateStr: String, | ||||||
|  |         offset: Long, | ||||||
|  |         limit: Long | ||||||
|  |     ): GetCreatorCalculateCommunityPostResponse { | ||||||
|  |         val startDate = startDateStr.convertLocalDateTime() | ||||||
|  |         val endDate = endDateStr.convertLocalDateTime(hour = 23, minute = 59, second = 59) | ||||||
|  |  | ||||||
|  |         val totalCount = repository.getCalculateCommunityPostTotalCount(startDate, endDate) | ||||||
|  |         val items = repository | ||||||
|  |             .getCalculateCommunityPostList(startDate, endDate, offset, limit) | ||||||
|  |             .map { it.toGetCalculateCommunityPostResponse() } | ||||||
|  |  | ||||||
|  |         return GetCreatorCalculateCommunityPostResponse(totalCount, items) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCalculateLiveByCreator( | ||||||
|  |         startDateStr: String, | ||||||
|  |         endDateStr: String, | ||||||
|  |         offset: Long, | ||||||
|  |         limit: Long | ||||||
|  |     ) = run { | ||||||
|  |         val startDate = startDateStr.convertLocalDateTime() | ||||||
|  |         val endDate = endDateStr.convertLocalDateTime(hour = 23, minute = 59, second = 59) | ||||||
|  |  | ||||||
|  |         val totalCount = repository.getCalculateLiveByCreatorTotalCount(startDate, endDate) | ||||||
|  |         val items = repository | ||||||
|  |             .getCalculateLiveByCreator(startDate, endDate, offset, limit) | ||||||
|  |             .map { it.toGetCalculateByCreator() } | ||||||
|  |  | ||||||
|  |         GetCalculateByCreatorResponse(totalCount, items) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCalculateContentByCreator( | ||||||
|  |         startDateStr: String, | ||||||
|  |         endDateStr: String, | ||||||
|  |         offset: Long, | ||||||
|  |         limit: Long | ||||||
|  |     ) = run { | ||||||
|  |         val startDate = startDateStr.convertLocalDateTime() | ||||||
|  |         val endDate = endDateStr.convertLocalDateTime(hour = 23, minute = 59, second = 59) | ||||||
|  |  | ||||||
|  |         val totalCount = repository.getCalculateContentByCreatorTotalCount(startDate, endDate) | ||||||
|  |         val items = repository | ||||||
|  |             .getCalculateContentByCreator(startDate, endDate, offset, limit) | ||||||
|  |             .map { it.toGetCalculateByCreator() } | ||||||
|  |  | ||||||
|  |         GetCalculateByCreatorResponse(totalCount, items) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCalculateCommunityByCreator( | ||||||
|  |         startDateStr: String, | ||||||
|  |         endDateStr: String, | ||||||
|  |         offset: Long, | ||||||
|  |         limit: Long | ||||||
|  |     ) = run { | ||||||
|  |         val startDate = startDateStr.convertLocalDateTime() | ||||||
|  |         val endDate = endDateStr.convertLocalDateTime(hour = 23, minute = 59, second = 59) | ||||||
|  |  | ||||||
|  |         val totalCount = repository.getCalculateCommunityByCreatorTotalCount(startDate, endDate) | ||||||
|  |         val items = repository | ||||||
|  |             .getCalculateCommunityByCreator(startDate, endDate, offset, limit) | ||||||
|  |             .map { it.toGetCalculateByCreator() } | ||||||
|  |  | ||||||
|  |         GetCalculateByCreatorResponse(totalCount, items) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,14 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate | ||||||
|  |  | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  |  | ||||||
|  | data class GetCalculateByCreatorItem( | ||||||
|  |     @JsonProperty("email") val email: String, | ||||||
|  |     @JsonProperty("nickname") val nickname: String, | ||||||
|  |     @JsonProperty("totalCan") val totalCan: Int, | ||||||
|  |     @JsonProperty("totalKrw") val totalKrw: Int, | ||||||
|  |     @JsonProperty("paymentFee") val paymentFee: Int, | ||||||
|  |     @JsonProperty("settlementAmount") val settlementAmount: Int, | ||||||
|  |     @JsonProperty("tax") val tax: Int, | ||||||
|  |     @JsonProperty("depositAmount") val depositAmount: Int | ||||||
|  | ) | ||||||
| @@ -0,0 +1,44 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate | ||||||
|  |  | ||||||
|  | import com.querydsl.core.annotations.QueryProjection | ||||||
|  | import java.math.BigDecimal | ||||||
|  | import java.math.RoundingMode | ||||||
|  |  | ||||||
|  | data class GetCalculateByCreatorQueryData @QueryProjection constructor( | ||||||
|  |     val email: String, | ||||||
|  |     val nickname: String, | ||||||
|  |     val totalCan: Int, | ||||||
|  |     val settlementRatio: Int? | ||||||
|  | ) { | ||||||
|  |     fun toGetCalculateByCreator(): GetCalculateByCreatorItem { | ||||||
|  |         // 원화 = totalCoin * 100 ( 캔 1개 = 100원 ) | ||||||
|  |         val totalKrw = BigDecimal(totalCan).multiply(BigDecimal(100)) | ||||||
|  |  | ||||||
|  |         // 결제수수료 : 6.6% | ||||||
|  |         val paymentFee = totalKrw.multiply(BigDecimal(0.066)) | ||||||
|  |  | ||||||
|  |         // 정산금액 = (원화 - 결제수수료) 의 70% | ||||||
|  |         val settlementAmount = if (settlementRatio != null) { | ||||||
|  |             totalKrw.subtract(paymentFee).multiply(BigDecimal(settlementRatio).divide(BigDecimal(100.0))) | ||||||
|  |         } else { | ||||||
|  |             totalKrw.subtract(paymentFee).multiply(BigDecimal(0.7)) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // 원천세 = 정산금액의 3.3% | ||||||
|  |         val tax = settlementAmount.multiply(BigDecimal(0.033)) | ||||||
|  |  | ||||||
|  |         // 입금액 | ||||||
|  |         val depositAmount = settlementAmount.subtract(tax) | ||||||
|  |  | ||||||
|  |         return GetCalculateByCreatorItem( | ||||||
|  |             email = email, | ||||||
|  |             nickname = nickname, | ||||||
|  |             totalCan = totalCan, | ||||||
|  |             totalKrw = totalKrw.toInt(), | ||||||
|  |             paymentFee = paymentFee.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             settlementAmount = settlementAmount.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             tax = tax.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             depositAmount = depositAmount.setScale(0, RoundingMode.HALF_UP).toInt() | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,6 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate | ||||||
|  |  | ||||||
|  | data class GetCalculateByCreatorResponse( | ||||||
|  |     val totalCount: Int, | ||||||
|  |     val items: List<GetCalculateByCreatorItem> | ||||||
|  | ) | ||||||
| @@ -0,0 +1,50 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate | ||||||
|  |  | ||||||
|  | import com.querydsl.core.annotations.QueryProjection | ||||||
|  | import java.math.BigDecimal | ||||||
|  | import java.math.RoundingMode | ||||||
|  |  | ||||||
|  | data class GetCalculateCommunityPostQueryData @QueryProjection constructor( | ||||||
|  |     val nickname: String, | ||||||
|  |     val title: String, | ||||||
|  |     val date: String, | ||||||
|  |     val can: Int, | ||||||
|  |     val numberOfPurchase: Long, | ||||||
|  |     val totalCan: Int, | ||||||
|  |     val settlementRatio: Int? | ||||||
|  | ) { | ||||||
|  |     fun toGetCalculateCommunityPostResponse(): GetCalculateCommunityPostResponse { | ||||||
|  |         // 원화 = totalCoin * 100 ( 캔 1개 = 100원 ) | ||||||
|  |         val totalKrw = BigDecimal(totalCan).multiply(BigDecimal(100)) | ||||||
|  |  | ||||||
|  |         // 결제수수료 : 6.6% | ||||||
|  |         val paymentFee = totalKrw.multiply(BigDecimal(0.066)) | ||||||
|  |  | ||||||
|  |         // 정산금액 = (원화 - 결제수수료) 의 70% | ||||||
|  |         val settlementAmount = if (settlementRatio != null) { | ||||||
|  |             totalKrw.subtract(paymentFee).multiply(BigDecimal(settlementRatio).divide(BigDecimal(100.0))) | ||||||
|  |         } else { | ||||||
|  |             totalKrw.subtract(paymentFee).multiply(BigDecimal(0.7)) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // 원천세 = 정산금액의 3.3% | ||||||
|  |         val tax = settlementAmount.multiply(BigDecimal(0.033)) | ||||||
|  |  | ||||||
|  |         // 입금액 | ||||||
|  |         val depositAmount = settlementAmount.subtract(tax) | ||||||
|  |  | ||||||
|  |         return GetCalculateCommunityPostResponse( | ||||||
|  |             nickname = nickname, | ||||||
|  |             title = title, | ||||||
|  |             date = date, | ||||||
|  |             can = can, | ||||||
|  |             numberOfPurchase = numberOfPurchase.toInt(), | ||||||
|  |             totalCan = totalCan, | ||||||
|  |             totalKrw = totalKrw.toInt(), | ||||||
|  |             paymentFee = paymentFee.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             settlementAmount = settlementAmount.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             tax = tax.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             depositAmount = depositAmount.setScale(0, RoundingMode.HALF_UP).toInt() | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,17 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate | ||||||
|  |  | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  |  | ||||||
|  | data class GetCalculateCommunityPostResponse( | ||||||
|  |     @JsonProperty("nickname") val nickname: String, | ||||||
|  |     @JsonProperty("title") val title: String, | ||||||
|  |     @JsonProperty("date") val date: String, | ||||||
|  |     @JsonProperty("can") val can: Int, | ||||||
|  |     @JsonProperty("numberOfPurchase") val numberOfPurchase: Int, | ||||||
|  |     @JsonProperty("totalCan") val totalCan: Int, | ||||||
|  |     @JsonProperty("totalKrw") val totalKrw: Int, | ||||||
|  |     @JsonProperty("paymentFee") val paymentFee: Int, | ||||||
|  |     @JsonProperty("settlementAmount") val settlementAmount: Int, | ||||||
|  |     @JsonProperty("tax") val tax: Int, | ||||||
|  |     @JsonProperty("depositAmount") val depositAmount: Int | ||||||
|  | ) | ||||||
| @@ -0,0 +1,66 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate | ||||||
|  |  | ||||||
|  | import com.querydsl.core.annotations.QueryProjection | ||||||
|  | import java.math.BigDecimal | ||||||
|  | import java.math.RoundingMode | ||||||
|  |  | ||||||
|  | data class GetCalculateContentDonationQueryData @QueryProjection constructor( | ||||||
|  |     // 등록 크리에이터 닉네임 | ||||||
|  |     val nickname: String, | ||||||
|  |     // 콘텐츠 제목 | ||||||
|  |     val title: String, | ||||||
|  |     // 콘텐츠 가격 | ||||||
|  |     val price: Int, | ||||||
|  |     // 콘텐츠 등록 날짜 | ||||||
|  |     val registrationDate: String, | ||||||
|  |     // 후원 날짜 | ||||||
|  |     val donationDate: String, | ||||||
|  |     // 인원 | ||||||
|  |     val numberOfDonation: Long, | ||||||
|  |     // 합계 | ||||||
|  |     val totalCan: Int | ||||||
|  | ) { | ||||||
|  |     fun toGetCalculateContentDonationResponse(): GetCalculateContentDonationResponse { | ||||||
|  |         // 원화 = totalCoin * 100 ( 캔 1개 = 100원 ) | ||||||
|  |         val totalKrw = BigDecimal(totalCan).multiply(BigDecimal(100)) | ||||||
|  |  | ||||||
|  |         // 결제수수료 : 6.6% | ||||||
|  |         val paymentFee = totalKrw.multiply(BigDecimal(0.066)) | ||||||
|  |  | ||||||
|  |         // 정산금액 | ||||||
|  |         // 유료콘텐츠 (원화 - 결제수수료) 의 90% | ||||||
|  |         // 무료콘텐츠 (원화 - 결제수수료) 의 70% | ||||||
|  |         val settlementAmount = if (price > 0) { | ||||||
|  |             totalKrw.subtract(paymentFee).multiply(BigDecimal(0.9)) | ||||||
|  |         } else { | ||||||
|  |             totalKrw.subtract(paymentFee).multiply(BigDecimal(0.7)) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // 원천세 = 정산금액의 3.3% | ||||||
|  |         val tax = settlementAmount.multiply(BigDecimal(0.033)) | ||||||
|  |  | ||||||
|  |         // 입금액 | ||||||
|  |         val depositAmount = settlementAmount.subtract(tax) | ||||||
|  |  | ||||||
|  |         val paidOrFree = if (price > 0) { | ||||||
|  |             "유료" | ||||||
|  |         } else { | ||||||
|  |             "무료" | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return GetCalculateContentDonationResponse( | ||||||
|  |             nickname = nickname, | ||||||
|  |             title = title, | ||||||
|  |             paidOrFree = paidOrFree, | ||||||
|  |             registrationDate = registrationDate, | ||||||
|  |             donationDate = donationDate, | ||||||
|  |             numberOfDonation = numberOfDonation.toInt(), | ||||||
|  |             totalCan = totalCan, | ||||||
|  |             totalKrw = totalKrw.toInt(), | ||||||
|  |             paymentFee = paymentFee.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             settlementAmount = settlementAmount.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             tax = tax.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             depositAmount = depositAmount.setScale(0, RoundingMode.HALF_UP).toInt() | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,18 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate | ||||||
|  |  | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  |  | ||||||
|  | data class GetCalculateContentDonationResponse( | ||||||
|  |     @JsonProperty("nickname") val nickname: String, | ||||||
|  |     @JsonProperty("title") val title: String, | ||||||
|  |     @JsonProperty("paidOrFree") val paidOrFree: String, | ||||||
|  |     @JsonProperty("registrationDate") val registrationDate: String, | ||||||
|  |     @JsonProperty("donationDate") val donationDate: String, | ||||||
|  |     @JsonProperty("numberOfDonation") val numberOfDonation: Int, | ||||||
|  |     @JsonProperty("totalCan") val totalCan: Int, | ||||||
|  |     @JsonProperty("totalKrw") val totalKrw: Int, | ||||||
|  |     @JsonProperty("paymentFee") val paymentFee: Int, | ||||||
|  |     @JsonProperty("settlementAmount") val settlementAmount: Int, | ||||||
|  |     @JsonProperty("tax") val tax: Int, | ||||||
|  |     @JsonProperty("depositAmount") val depositAmount: Int | ||||||
|  | ) | ||||||
| @@ -0,0 +1,73 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate | ||||||
|  |  | ||||||
|  | import com.querydsl.core.annotations.QueryProjection | ||||||
|  | import kr.co.vividnext.sodalive.content.order.OrderType | ||||||
|  | import java.math.BigDecimal | ||||||
|  | import java.math.RoundingMode | ||||||
|  |  | ||||||
|  | data class GetCalculateContentQueryData @QueryProjection constructor( | ||||||
|  |     // 등록 크리에이터 닉네임 | ||||||
|  |     val nickname: String, | ||||||
|  |     // 콘텐츠 제목 | ||||||
|  |     val title: String, | ||||||
|  |     // 콘텐츠 등록 날짜 | ||||||
|  |     val registrationDate: String, | ||||||
|  |     // 콘텐츠 판매 날짜 | ||||||
|  |     val saleDate: String, | ||||||
|  |     // 대여/소장 구분 | ||||||
|  |     val orderType: OrderType, | ||||||
|  |     // 판매 금액(캔) | ||||||
|  |     val orderPrice: Int, | ||||||
|  |     // 인원 | ||||||
|  |     val numberOfPeople: Long, | ||||||
|  |     // 합계 | ||||||
|  |     val totalCan: Int, | ||||||
|  |     // 포인트 | ||||||
|  |     val totalPoint: Int, | ||||||
|  |     // 정산비율 | ||||||
|  |     val settlementRatio: Int? | ||||||
|  | ) { | ||||||
|  |     fun toGetCalculateContentResponse(): GetCalculateContentResponse { | ||||||
|  |         val orderTypeStr = if (totalPoint > 0) { | ||||||
|  |             "포인트" | ||||||
|  |         } else if (orderType == OrderType.RENTAL) { | ||||||
|  |             "대여" | ||||||
|  |         } else { | ||||||
|  |             "소장" | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // 원화 = totalCoin * 100 ( 캔 1개 = 100원 ) | ||||||
|  |         val totalKrw = BigDecimal(totalCan).multiply(BigDecimal(100)) | ||||||
|  |  | ||||||
|  |         // 결제수수료 : 6.6% | ||||||
|  |         val paymentFee = totalKrw.multiply(BigDecimal(0.066)) | ||||||
|  |  | ||||||
|  |         // 정산금액 = (원화 - 결제수수료) 의 70% | ||||||
|  |         val settlementAmount = if (settlementRatio != null) { | ||||||
|  |             totalKrw.subtract(paymentFee).multiply(BigDecimal(settlementRatio).divide(BigDecimal(100.0))) | ||||||
|  |         } else { | ||||||
|  |             totalKrw.subtract(paymentFee).multiply(BigDecimal(0.7)) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // 원천세 = 정산금액의 3.3% | ||||||
|  |         val tax = settlementAmount.multiply(BigDecimal(0.033)) | ||||||
|  |  | ||||||
|  |         val depositAmount = settlementAmount.subtract(tax) | ||||||
|  |  | ||||||
|  |         return GetCalculateContentResponse( | ||||||
|  |             nickname = nickname, | ||||||
|  |             title = title, | ||||||
|  |             registrationDate = registrationDate, | ||||||
|  |             saleDate = saleDate, | ||||||
|  |             orderType = orderTypeStr, | ||||||
|  |             orderPrice = orderPrice, | ||||||
|  |             numberOfPeople = numberOfPeople.toInt(), | ||||||
|  |             totalCan = totalCan, | ||||||
|  |             totalKrw = totalKrw.toInt(), | ||||||
|  |             paymentFee = paymentFee.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             settlementAmount = settlementAmount.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             tax = tax.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             depositAmount = depositAmount.setScale(0, RoundingMode.HALF_UP).toInt() | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,19 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate | ||||||
|  |  | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  |  | ||||||
|  | data class GetCalculateContentResponse( | ||||||
|  |     @JsonProperty("nickname") val nickname: String, | ||||||
|  |     @JsonProperty("title") val title: String, | ||||||
|  |     @JsonProperty("registrationDate") val registrationDate: String, | ||||||
|  |     @JsonProperty("saleDate") val saleDate: String, | ||||||
|  |     @JsonProperty("orderType") val orderType: String, | ||||||
|  |     @JsonProperty("orderPrice") val orderPrice: Int, | ||||||
|  |     @JsonProperty("numberOfPeople") val numberOfPeople: Int, | ||||||
|  |     @JsonProperty("totalCan") val totalCan: Int, | ||||||
|  |     @JsonProperty("totalKrw") val totalKrw: Int, | ||||||
|  |     @JsonProperty("paymentFee") val paymentFee: Int, | ||||||
|  |     @JsonProperty("settlementAmount") val settlementAmount: Int, | ||||||
|  |     @JsonProperty("tax") val tax: Int, | ||||||
|  |     @JsonProperty("depositAmount") val depositAmount: Int | ||||||
|  | ) | ||||||
| @@ -0,0 +1,84 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate | ||||||
|  |  | ||||||
|  | import com.querydsl.core.annotations.QueryProjection | ||||||
|  | import kr.co.vividnext.sodalive.can.use.CanUsage | ||||||
|  | import java.math.BigDecimal | ||||||
|  | import java.math.RoundingMode | ||||||
|  |  | ||||||
|  | data class GetCalculateLiveQueryData @QueryProjection constructor( | ||||||
|  |     val email: String, | ||||||
|  |     val nickname: String, | ||||||
|  |     val date: String, | ||||||
|  |     val title: String, | ||||||
|  |     // 유료방 입장 금액 | ||||||
|  |     val entranceFee: Int, | ||||||
|  |     // 코인 사용 구분 | ||||||
|  |     val canUsage: CanUsage, | ||||||
|  |     // 참여인원 | ||||||
|  |     val memberCount: Long, | ||||||
|  |     // 합계 | ||||||
|  |     val totalAmount: Int, | ||||||
|  |     // 정산비율 | ||||||
|  |     val settlementRatio: Int? | ||||||
|  | ) { | ||||||
|  |     fun toGetCalculateLiveResponse(): GetCalculateLiveResponse { | ||||||
|  |         val canUsageStr = when (canUsage) { | ||||||
|  |             CanUsage.LIVE -> { | ||||||
|  |                 "유료" | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             CanUsage.SPIN_ROULETTE -> { | ||||||
|  |                 "룰렛" | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             CanUsage.HEART -> { | ||||||
|  |                 "하트" | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             else -> { | ||||||
|  |                 "후원" | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         val numberOfPeople = if (canUsage == CanUsage.LIVE) { | ||||||
|  |             memberCount.toInt() | ||||||
|  |         } else { | ||||||
|  |             0 | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // 원화 = totalCoin * 100 ( 캔 1개 = 100원 ) | ||||||
|  |         val totalKrw = BigDecimal(totalAmount).multiply(BigDecimal(100)) | ||||||
|  |  | ||||||
|  |         // 결제수수료 : 6.6% | ||||||
|  |         val paymentFee = totalKrw.multiply(BigDecimal(0.066)) | ||||||
|  |  | ||||||
|  |         // 정산금액 = (원화 - 결제수수료) 의 70% | ||||||
|  |         val settlementAmount = if (settlementRatio != null) { | ||||||
|  |             totalKrw.subtract(paymentFee).multiply(BigDecimal(settlementRatio).divide(BigDecimal(100.0))) | ||||||
|  |         } else { | ||||||
|  |             totalKrw.subtract(paymentFee).multiply(BigDecimal(0.7)) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // 원천세 = 정산금액의 3.3% | ||||||
|  |         val tax = settlementAmount.multiply(BigDecimal(0.033)) | ||||||
|  |  | ||||||
|  |         // 입금액 | ||||||
|  |         val depositAmount = settlementAmount.subtract(tax) | ||||||
|  |  | ||||||
|  |         return GetCalculateLiveResponse( | ||||||
|  |             email = email, | ||||||
|  |             nickname = nickname, | ||||||
|  |             date = date, | ||||||
|  |             title = title, | ||||||
|  |             entranceFee = entranceFee, | ||||||
|  |             canUsageStr = canUsageStr, | ||||||
|  |             numberOfPeople = numberOfPeople, | ||||||
|  |             totalAmount = totalAmount, | ||||||
|  |             totalKrw = totalKrw.toInt(), | ||||||
|  |             paymentFee = paymentFee.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             settlementAmount = settlementAmount.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             tax = tax.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             depositAmount = depositAmount.setScale(0, RoundingMode.HALF_UP).toInt() | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,19 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate | ||||||
|  |  | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  |  | ||||||
|  | data class GetCalculateLiveResponse( | ||||||
|  |     @JsonProperty("email") val email: String, | ||||||
|  |     @JsonProperty("nickname") val nickname: String, | ||||||
|  |     @JsonProperty("date") val date: String, | ||||||
|  |     @JsonProperty("title") val title: String, | ||||||
|  |     @JsonProperty("entranceFee") val entranceFee: Int, | ||||||
|  |     @JsonProperty("canUsageStr") val canUsageStr: String, | ||||||
|  |     @JsonProperty("numberOfPeople") val numberOfPeople: Int, | ||||||
|  |     @JsonProperty("totalAmount") val totalAmount: Int, | ||||||
|  |     @JsonProperty("totalKrw") val totalKrw: Int, | ||||||
|  |     @JsonProperty("paymentFee") val paymentFee: Int, | ||||||
|  |     @JsonProperty("settlementAmount") val settlementAmount: Int, | ||||||
|  |     @JsonProperty("tax") val tax: Int, | ||||||
|  |     @JsonProperty("depositAmount") val depositAmount: Int | ||||||
|  | ) | ||||||
| @@ -0,0 +1,92 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate | ||||||
|  |  | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.querydsl.core.annotations.QueryProjection | ||||||
|  | import kr.co.vividnext.sodalive.content.order.OrderType | ||||||
|  | import java.math.BigDecimal | ||||||
|  | import java.math.RoundingMode | ||||||
|  |  | ||||||
|  | data class GetCumulativeSalesByContentQueryData @QueryProjection constructor( | ||||||
|  |     // 등록 크리에이터 닉네임 | ||||||
|  |     val nickname: String, | ||||||
|  |     // 콘텐츠 제목 | ||||||
|  |     val title: String, | ||||||
|  |     // 콘텐츠 등록 날짜 | ||||||
|  |     val registrationDate: String, | ||||||
|  |     // 대여/소장 구분 | ||||||
|  |     val orderType: OrderType, | ||||||
|  |     // 판매 금액(캔) | ||||||
|  |     val orderPrice: Int, | ||||||
|  |     // 인원 | ||||||
|  |     val numberOfPeople: Long, | ||||||
|  |     // 합계 | ||||||
|  |     val totalCan: Int, | ||||||
|  |     // 포인트 | ||||||
|  |     val totalPoint: Int, | ||||||
|  |     // 정산비율 | ||||||
|  |     val settlementRatio: Int? | ||||||
|  | ) { | ||||||
|  |     fun toCumulativeSalesByContentItem(): CumulativeSalesByContentItem { | ||||||
|  |         val orderTypeStr = if (totalPoint > 0) { | ||||||
|  |             "포인트" | ||||||
|  |         } else if (orderType == OrderType.RENTAL) { | ||||||
|  |             "대여" | ||||||
|  |         } else { | ||||||
|  |             "소장" | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // 원화 = totalCoin * 100 ( 캔 1개 = 100원 ) | ||||||
|  |         val totalKrw = BigDecimal(totalCan).multiply(BigDecimal(100)) | ||||||
|  |  | ||||||
|  |         // 결제수수료 : 6.6% | ||||||
|  |         val paymentFee = totalKrw.multiply(BigDecimal(0.066)) | ||||||
|  |  | ||||||
|  |         // 정산금액 = (원화 - 결제수수료) 의 70% | ||||||
|  |         val settlementAmount = if (settlementRatio != null) { | ||||||
|  |             totalKrw.subtract(paymentFee).multiply(BigDecimal(settlementRatio).divide(BigDecimal(100.0))) | ||||||
|  |         } else { | ||||||
|  |             totalKrw.subtract(paymentFee).multiply(BigDecimal(0.7)) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // 원천세 = 정산금액의 3.3% | ||||||
|  |         val tax = settlementAmount.multiply(BigDecimal(0.033)) | ||||||
|  |  | ||||||
|  |         // 입금액 | ||||||
|  |         val depositAmount = settlementAmount.subtract(tax) | ||||||
|  |  | ||||||
|  |         return CumulativeSalesByContentItem( | ||||||
|  |             nickname = nickname, | ||||||
|  |             title = title, | ||||||
|  |             registrationDate = registrationDate, | ||||||
|  |             orderType = orderTypeStr, | ||||||
|  |             orderPrice = orderPrice, | ||||||
|  |             numberOfPeople = numberOfPeople.toInt(), | ||||||
|  |             totalCan = totalCan, | ||||||
|  |             totalKrw = totalKrw.toInt(), | ||||||
|  |             paymentFee = paymentFee.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             settlementAmount = settlementAmount.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             tax = tax.setScale(0, RoundingMode.HALF_UP).toInt(), | ||||||
|  |             depositAmount = depositAmount.setScale(0, RoundingMode.HALF_UP).toInt() | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | data class GetCumulativeSalesByContentResponse( | ||||||
|  |     @JsonProperty("totalCount") val totalCount: Int, | ||||||
|  |     @JsonProperty("items") val items: List<CumulativeSalesByContentItem> | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class CumulativeSalesByContentItem( | ||||||
|  |     @JsonProperty("nickname") val nickname: String, | ||||||
|  |     @JsonProperty("title") val title: String, | ||||||
|  |     @JsonProperty("registrationDate") val registrationDate: String, | ||||||
|  |     @JsonProperty("orderType") val orderType: String, | ||||||
|  |     @JsonProperty("orderPrice") val orderPrice: Int, | ||||||
|  |     @JsonProperty("numberOfPeople") val numberOfPeople: Int, | ||||||
|  |     @JsonProperty("totalCan") val totalCan: Int, | ||||||
|  |     @JsonProperty("totalKrw") val totalKrw: Int, | ||||||
|  |     @JsonProperty("paymentFee") val paymentFee: Int, | ||||||
|  |     @JsonProperty("settlementAmount") val settlementAmount: Int, | ||||||
|  |     @JsonProperty("tax") val tax: Int, | ||||||
|  |     @JsonProperty("depositAmount") val depositAmount: Int | ||||||
|  | ) | ||||||
| @@ -0,0 +1,16 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate.ratio | ||||||
|  |  | ||||||
|  | data class CreateCreatorSettlementRatioRequest( | ||||||
|  |     val memberId: Long, | ||||||
|  |     val subsidy: Int, | ||||||
|  |     val liveSettlementRatio: Int, | ||||||
|  |     val contentSettlementRatio: Int, | ||||||
|  |     val communitySettlementRatio: Int | ||||||
|  | ) { | ||||||
|  |     fun toEntity() = CreatorSettlementRatio( | ||||||
|  |         subsidy = subsidy, | ||||||
|  |         liveSettlementRatio = liveSettlementRatio, | ||||||
|  |         contentSettlementRatio = contentSettlementRatio, | ||||||
|  |         communitySettlementRatio = communitySettlementRatio | ||||||
|  |     ) | ||||||
|  | } | ||||||
| @@ -0,0 +1,38 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate.ratio | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.common.BaseEntity | ||||||
|  | import kr.co.vividnext.sodalive.member.Member | ||||||
|  | import java.time.LocalDateTime | ||||||
|  | import javax.persistence.Entity | ||||||
|  | import javax.persistence.FetchType | ||||||
|  | import javax.persistence.JoinColumn | ||||||
|  | import javax.persistence.OneToOne | ||||||
|  |  | ||||||
|  | @Entity | ||||||
|  | data class CreatorSettlementRatio( | ||||||
|  |     var subsidy: Int, | ||||||
|  |     var liveSettlementRatio: Int, | ||||||
|  |     var contentSettlementRatio: Int, | ||||||
|  |     var communitySettlementRatio: Int | ||||||
|  | ) : BaseEntity() { | ||||||
|  |     @OneToOne(fetch = FetchType.LAZY) | ||||||
|  |     @JoinColumn(name = "member_id", nullable = false) | ||||||
|  |     var member: Member? = null | ||||||
|  |  | ||||||
|  |     var deletedAt: LocalDateTime? = null | ||||||
|  |  | ||||||
|  |     fun softDelete() { | ||||||
|  |         this.deletedAt = LocalDateTime.now() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun restore() { | ||||||
|  |         this.deletedAt = null | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun updateValues(subsidy: Int, live: Int, content: Int, community: Int) { | ||||||
|  |         this.subsidy = subsidy | ||||||
|  |         this.liveSettlementRatio = live | ||||||
|  |         this.contentSettlementRatio = content | ||||||
|  |         this.communitySettlementRatio = community | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,41 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate.ratio | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.common.ApiResponse | ||||||
|  | import org.springframework.data.domain.Pageable | ||||||
|  | import org.springframework.security.access.prepost.PreAuthorize | ||||||
|  | import org.springframework.web.bind.annotation.GetMapping | ||||||
|  | import org.springframework.web.bind.annotation.PathVariable | ||||||
|  | import org.springframework.web.bind.annotation.PostMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestBody | ||||||
|  | import org.springframework.web.bind.annotation.RequestMapping | ||||||
|  | import org.springframework.web.bind.annotation.RestController | ||||||
|  |  | ||||||
|  | @RestController | ||||||
|  | @PreAuthorize("hasRole('ADMIN')") | ||||||
|  | @RequestMapping("/admin/calculate/ratio") | ||||||
|  | class CreatorSettlementRatioController(private val service: CreatorSettlementRatioService) { | ||||||
|  |     @PostMapping | ||||||
|  |     fun createCreatorSettlementRatio( | ||||||
|  |         @RequestBody request: CreateCreatorSettlementRatioRequest | ||||||
|  |     ) = ApiResponse.ok(service.createCreatorSettlementRatio(request)) | ||||||
|  |  | ||||||
|  |     @GetMapping | ||||||
|  |     fun getCreatorSettlementRatio( | ||||||
|  |         pageable: Pageable | ||||||
|  |     ) = ApiResponse.ok( | ||||||
|  |         service.getCreatorSettlementRatio( | ||||||
|  |             offset = pageable.offset, | ||||||
|  |             limit = pageable.pageSize.toLong() | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     @PostMapping("/update") | ||||||
|  |     fun updateCreatorSettlementRatio( | ||||||
|  |         @RequestBody request: CreateCreatorSettlementRatioRequest | ||||||
|  |     ) = ApiResponse.ok(service.updateCreatorSettlementRatio(request)) | ||||||
|  |  | ||||||
|  |     @PostMapping("/delete/{memberId}") | ||||||
|  |     fun deleteCreatorSettlementRatio( | ||||||
|  |         @PathVariable memberId: Long | ||||||
|  |     ) = ApiResponse.ok(service.deleteCreatorSettlementRatio(memberId)) | ||||||
|  | } | ||||||
| @@ -0,0 +1,51 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate.ratio | ||||||
|  |  | ||||||
|  | import com.querydsl.jpa.impl.JPAQueryFactory | ||||||
|  | import kr.co.vividnext.sodalive.admin.calculate.ratio.QCreatorSettlementRatio.creatorSettlementRatio | ||||||
|  | import kr.co.vividnext.sodalive.member.QMember.member | ||||||
|  | import org.springframework.data.jpa.repository.JpaRepository | ||||||
|  |  | ||||||
|  | interface CreatorSettlementRatioRepository : | ||||||
|  |     JpaRepository<CreatorSettlementRatio, Long>, | ||||||
|  |     CreatorSettlementRatioQueryRepository { | ||||||
|  |     fun findByMemberId(memberId: Long): CreatorSettlementRatio? | ||||||
|  | } | ||||||
|  |  | ||||||
|  | interface CreatorSettlementRatioQueryRepository { | ||||||
|  |     fun getCreatorSettlementRatio(offset: Long, limit: Long): List<GetCreatorSettlementRatioItem> | ||||||
|  |     fun getCreatorSettlementRatioTotalCount(): Int | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class CreatorSettlementRatioQueryRepositoryImpl( | ||||||
|  |     private val queryFactory: JPAQueryFactory | ||||||
|  | ) : CreatorSettlementRatioQueryRepository { | ||||||
|  |     override fun getCreatorSettlementRatio(offset: Long, limit: Long): List<GetCreatorSettlementRatioItem> { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetCreatorSettlementRatioItem( | ||||||
|  |                     member.id, | ||||||
|  |                     member.nickname, | ||||||
|  |                     creatorSettlementRatio.subsidy, | ||||||
|  |                     creatorSettlementRatio.liveSettlementRatio, | ||||||
|  |                     creatorSettlementRatio.contentSettlementRatio, | ||||||
|  |                     creatorSettlementRatio.communitySettlementRatio | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(creatorSettlementRatio) | ||||||
|  |             .innerJoin(creatorSettlementRatio.member, member) | ||||||
|  |             .where(creatorSettlementRatio.deletedAt.isNull) | ||||||
|  |             .orderBy(creatorSettlementRatio.id.asc()) | ||||||
|  |             .offset(offset) | ||||||
|  |             .limit(limit) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun getCreatorSettlementRatioTotalCount(): Int { | ||||||
|  |         return queryFactory | ||||||
|  |             .select(creatorSettlementRatio.id) | ||||||
|  |             .from(creatorSettlementRatio) | ||||||
|  |             .where(creatorSettlementRatio.deletedAt.isNull) | ||||||
|  |             .fetch() | ||||||
|  |             .size | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,77 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate.ratio | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.common.SodaException | ||||||
|  | import kr.co.vividnext.sodalive.member.MemberRepository | ||||||
|  | import kr.co.vividnext.sodalive.member.MemberRole | ||||||
|  | import org.springframework.data.repository.findByIdOrNull | ||||||
|  | import org.springframework.stereotype.Service | ||||||
|  | import org.springframework.transaction.annotation.Transactional | ||||||
|  |  | ||||||
|  | @Service | ||||||
|  | class CreatorSettlementRatioService( | ||||||
|  |     private val repository: CreatorSettlementRatioRepository, | ||||||
|  |     private val memberRepository: MemberRepository | ||||||
|  | ) { | ||||||
|  |     @Transactional | ||||||
|  |     fun createCreatorSettlementRatio(request: CreateCreatorSettlementRatioRequest) { | ||||||
|  |         val creator = memberRepository.findByIdOrNull(request.memberId) | ||||||
|  |             ?: throw SodaException("잘못된 크리에이터 입니다.") | ||||||
|  |  | ||||||
|  |         if (creator.role != MemberRole.CREATOR) { | ||||||
|  |             throw SodaException("잘못된 크리에이터 입니다.") | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         val existing = repository.findByMemberId(request.memberId) | ||||||
|  |         if (existing != null) { | ||||||
|  |             // revive if soft-deleted, then update values | ||||||
|  |             existing.restore() | ||||||
|  |             existing.updateValues( | ||||||
|  |                 request.subsidy, | ||||||
|  |                 request.liveSettlementRatio, | ||||||
|  |                 request.contentSettlementRatio, | ||||||
|  |                 request.communitySettlementRatio | ||||||
|  |             ) | ||||||
|  |             repository.save(existing) | ||||||
|  |             return | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         val creatorSettlementRatio = request.toEntity() | ||||||
|  |         creatorSettlementRatio.member = creator | ||||||
|  |         repository.save(creatorSettlementRatio) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional | ||||||
|  |     fun updateCreatorSettlementRatio(request: CreateCreatorSettlementRatioRequest) { | ||||||
|  |         val creator = memberRepository.findByIdOrNull(request.memberId) | ||||||
|  |             ?: throw SodaException("잘못된 크리에이터 입니다.") | ||||||
|  |         if (creator.role != MemberRole.CREATOR) { | ||||||
|  |             throw SodaException("잘못된 크리에이터 입니다.") | ||||||
|  |         } | ||||||
|  |         val existing = repository.findByMemberId(request.memberId) | ||||||
|  |             ?: throw SodaException("해당 크리에이터의 정산 비율 설정이 없습니다.") | ||||||
|  |         existing.restore() | ||||||
|  |         existing.updateValues( | ||||||
|  |             request.subsidy, | ||||||
|  |             request.liveSettlementRatio, | ||||||
|  |             request.contentSettlementRatio, | ||||||
|  |             request.communitySettlementRatio | ||||||
|  |         ) | ||||||
|  |         repository.save(existing) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional | ||||||
|  |     fun deleteCreatorSettlementRatio(memberId: Long) { | ||||||
|  |         val existing = repository.findByMemberId(memberId) | ||||||
|  |             ?: throw SodaException("해당 크리에이터의 정산 비율 설정이 없습니다.") | ||||||
|  |         existing.softDelete() | ||||||
|  |         repository.save(existing) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional(readOnly = true) | ||||||
|  |     fun getCreatorSettlementRatio(offset: Long, limit: Long): GetCreatorSettlementRatioResponse { | ||||||
|  |         val totalCount = repository.getCreatorSettlementRatioTotalCount() | ||||||
|  |         val items = repository.getCreatorSettlementRatio(offset, limit) | ||||||
|  |  | ||||||
|  |         return GetCreatorSettlementRatioResponse(totalCount, items) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,17 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.calculate.ratio | ||||||
|  |  | ||||||
|  | import com.querydsl.core.annotations.QueryProjection | ||||||
|  |  | ||||||
|  | data class GetCreatorSettlementRatioResponse( | ||||||
|  |     val totalCount: Int, | ||||||
|  |     val items: List<GetCreatorSettlementRatioItem> | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class GetCreatorSettlementRatioItem @QueryProjection constructor( | ||||||
|  |     val memberId: Long, | ||||||
|  |     val nickname: String, | ||||||
|  |     val subsidy: Int, | ||||||
|  |     val liveSettlementRatio: Int, | ||||||
|  |     val contentSettlementRatio: Int, | ||||||
|  |     val communitySettlementRatio: Int | ||||||
|  | ) | ||||||
| @@ -1,8 +1,10 @@ | |||||||
| package kr.co.vividnext.sodalive.admin.can | package kr.co.vividnext.sodalive.admin.can | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.can.CanResponse | ||||||
| import kr.co.vividnext.sodalive.common.ApiResponse | import kr.co.vividnext.sodalive.common.ApiResponse | ||||||
| import org.springframework.security.access.prepost.PreAuthorize | import org.springframework.security.access.prepost.PreAuthorize | ||||||
| import org.springframework.web.bind.annotation.DeleteMapping | import org.springframework.web.bind.annotation.DeleteMapping | ||||||
|  | import org.springframework.web.bind.annotation.GetMapping | ||||||
| import org.springframework.web.bind.annotation.PathVariable | import org.springframework.web.bind.annotation.PathVariable | ||||||
| import org.springframework.web.bind.annotation.PostMapping | import org.springframework.web.bind.annotation.PostMapping | ||||||
| import org.springframework.web.bind.annotation.RequestBody | import org.springframework.web.bind.annotation.RequestBody | ||||||
| @@ -13,6 +15,11 @@ import org.springframework.web.bind.annotation.RestController | |||||||
| @RequestMapping("/admin/can") | @RequestMapping("/admin/can") | ||||||
| @PreAuthorize("hasRole('ADMIN')") | @PreAuthorize("hasRole('ADMIN')") | ||||||
| class AdminCanController(private val service: AdminCanService) { | class AdminCanController(private val service: AdminCanService) { | ||||||
|  |     @GetMapping | ||||||
|  |     fun getCans(): ApiResponse<List<CanResponse>> { | ||||||
|  |         return ApiResponse.ok(service.getCans()) | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @PostMapping |     @PostMapping | ||||||
|     fun insertCan(@RequestBody request: AdminCanRequest) = ApiResponse.ok(service.saveCan(request)) |     fun insertCan(@RequestBody request: AdminCanRequest) = ApiResponse.ok(service.saveCan(request)) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,6 +1,38 @@ | |||||||
| package kr.co.vividnext.sodalive.admin.can | package kr.co.vividnext.sodalive.admin.can | ||||||
|  |  | ||||||
|  | import com.querydsl.jpa.impl.JPAQueryFactory | ||||||
| import kr.co.vividnext.sodalive.can.Can | import kr.co.vividnext.sodalive.can.Can | ||||||
|  | import kr.co.vividnext.sodalive.can.CanResponse | ||||||
|  | import kr.co.vividnext.sodalive.can.CanStatus | ||||||
|  | import kr.co.vividnext.sodalive.can.QCan.can1 | ||||||
|  | import kr.co.vividnext.sodalive.can.QCanResponse | ||||||
| import org.springframework.data.jpa.repository.JpaRepository | import org.springframework.data.jpa.repository.JpaRepository | ||||||
|  | import org.springframework.stereotype.Repository | ||||||
|  |  | ||||||
| interface AdminCanRepository : JpaRepository<Can, Long> | interface AdminCanRepository : JpaRepository<Can, Long>, AdminCanQueryRepository | ||||||
|  |  | ||||||
|  | interface AdminCanQueryRepository { | ||||||
|  |     fun findAllByStatus(status: CanStatus): List<CanResponse> | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @Repository | ||||||
|  | class AdminCanQueryRepositoryImpl(private val queryFactory: JPAQueryFactory) : AdminCanQueryRepository { | ||||||
|  |     override fun findAllByStatus(status: CanStatus): List<CanResponse> { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QCanResponse( | ||||||
|  |                     can1.id, | ||||||
|  |                     can1.title, | ||||||
|  |                     can1.can, | ||||||
|  |                     can1.rewardCan, | ||||||
|  |                     can1.price.intValue(), | ||||||
|  |                     can1.currency, | ||||||
|  |                     can1.price.stringValue() | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(can1) | ||||||
|  |             .where(can1.status.eq(status)) | ||||||
|  |             .orderBy(can1.currency.asc(), can1.price.asc()) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  | } | ||||||
|   | |||||||
| @@ -3,11 +3,13 @@ package kr.co.vividnext.sodalive.admin.can | |||||||
| import kr.co.vividnext.sodalive.can.Can | import kr.co.vividnext.sodalive.can.Can | ||||||
| import kr.co.vividnext.sodalive.can.CanStatus | import kr.co.vividnext.sodalive.can.CanStatus | ||||||
| import kr.co.vividnext.sodalive.extensions.moneyFormat | import kr.co.vividnext.sodalive.extensions.moneyFormat | ||||||
|  | import java.math.BigDecimal | ||||||
|  |  | ||||||
| data class AdminCanRequest( | data class AdminCanRequest( | ||||||
|     val can: Int, |     val can: Int, | ||||||
|     val rewardCan: Int, |     val rewardCan: Int, | ||||||
|     val price: Int |     val price: BigDecimal, | ||||||
|  |     val currency: String | ||||||
| ) { | ) { | ||||||
|     fun toEntity(): Can { |     fun toEntity(): Can { | ||||||
|         var title = "${can.moneyFormat()} 캔" |         var title = "${can.moneyFormat()} 캔" | ||||||
| @@ -20,6 +22,7 @@ data class AdminCanRequest( | |||||||
|             can = can, |             can = can, | ||||||
|             rewardCan = rewardCan, |             rewardCan = rewardCan, | ||||||
|             price = price, |             price = price, | ||||||
|  |             currency = currency, | ||||||
|             status = CanStatus.SALE |             status = CanStatus.SALE | ||||||
|         ) |         ) | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| package kr.co.vividnext.sodalive.admin.can | package kr.co.vividnext.sodalive.admin.can | ||||||
|  |  | ||||||
| import kr.co.vividnext.sodalive.admin.member.AdminMemberRepository | import kr.co.vividnext.sodalive.admin.member.AdminMemberRepository | ||||||
|  | import kr.co.vividnext.sodalive.can.CanResponse | ||||||
| import kr.co.vividnext.sodalive.can.CanStatus | import kr.co.vividnext.sodalive.can.CanStatus | ||||||
| import kr.co.vividnext.sodalive.can.charge.Charge | import kr.co.vividnext.sodalive.can.charge.Charge | ||||||
| import kr.co.vividnext.sodalive.can.charge.ChargeRepository | import kr.co.vividnext.sodalive.can.charge.ChargeRepository | ||||||
| @@ -20,6 +21,10 @@ class AdminCanService( | |||||||
|     private val chargeRepository: ChargeRepository, |     private val chargeRepository: ChargeRepository, | ||||||
|     private val memberRepository: AdminMemberRepository |     private val memberRepository: AdminMemberRepository | ||||||
| ) { | ) { | ||||||
|  |     fun getCans(): List<CanResponse> { | ||||||
|  |         return repository.findAllByStatus(status = CanStatus.SALE) | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Transactional |     @Transactional | ||||||
|     fun saveCan(request: AdminCanRequest) { |     fun saveCan(request: AdminCanRequest) { | ||||||
|         repository.save(request.toEntity()) |         repository.save(request.toEntity()) | ||||||
|   | |||||||
| @@ -21,6 +21,7 @@ class AdminChargeStatusController(private val service: AdminChargeStatusService) | |||||||
|     @GetMapping("/detail") |     @GetMapping("/detail") | ||||||
|     fun getChargeStatusDetail( |     fun getChargeStatusDetail( | ||||||
|         @RequestParam startDateStr: String, |         @RequestParam startDateStr: String, | ||||||
|         @RequestParam paymentGateway: PaymentGateway |         @RequestParam paymentGateway: PaymentGateway, | ||||||
|     ) = ApiResponse.ok(service.getChargeStatusDetail(startDateStr, paymentGateway)) |         @RequestParam(value = "currency", required = false) currency: String? = null | ||||||
|  |     ) = ApiResponse.ok(service.getChargeStatusDetail(startDateStr, paymentGateway, currency)) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,10 +1,12 @@ | |||||||
| package kr.co.vividnext.sodalive.admin.charge | package kr.co.vividnext.sodalive.admin.charge | ||||||
|  |  | ||||||
|  | import com.querydsl.core.BooleanBuilder | ||||||
| import com.querydsl.core.types.dsl.Expressions | import com.querydsl.core.types.dsl.Expressions | ||||||
| import com.querydsl.jpa.impl.JPAQueryFactory | import com.querydsl.jpa.impl.JPAQueryFactory | ||||||
| import kr.co.vividnext.sodalive.can.QCan.can1 | import kr.co.vividnext.sodalive.can.QCan.can1 | ||||||
| import kr.co.vividnext.sodalive.can.charge.ChargeStatus | import kr.co.vividnext.sodalive.can.charge.ChargeStatus | ||||||
| import kr.co.vividnext.sodalive.can.charge.QCharge.charge | import kr.co.vividnext.sodalive.can.charge.QCharge.charge | ||||||
|  | import kr.co.vividnext.sodalive.can.payment.PaymentGateway | ||||||
| import kr.co.vividnext.sodalive.can.payment.PaymentStatus | import kr.co.vividnext.sodalive.can.payment.PaymentStatus | ||||||
| import kr.co.vividnext.sodalive.can.payment.QPayment.payment | import kr.co.vividnext.sodalive.can.payment.QPayment.payment | ||||||
| import kr.co.vividnext.sodalive.member.QMember.member | import kr.co.vividnext.sodalive.member.QMember.member | ||||||
| @@ -13,7 +15,7 @@ import java.time.LocalDateTime | |||||||
|  |  | ||||||
| @Repository | @Repository | ||||||
| class AdminChargeStatusQueryRepository(private val queryFactory: JPAQueryFactory) { | class AdminChargeStatusQueryRepository(private val queryFactory: JPAQueryFactory) { | ||||||
|     fun getChargeStatus(startDate: LocalDateTime, endDate: LocalDateTime): List<GetChargeStatusQueryDto> { |     fun getChargeStatus(startDate: LocalDateTime, endDate: LocalDateTime): List<GetChargeStatusResponse> { | ||||||
|         val formattedDate = Expressions.stringTemplate( |         val formattedDate = Expressions.stringTemplate( | ||||||
|             "DATE_FORMAT({0}, {1})", |             "DATE_FORMAT({0}, {1})", | ||||||
|             Expressions.dateTimeTemplate( |             Expressions.dateTimeTemplate( | ||||||
| @@ -25,15 +27,16 @@ class AdminChargeStatusQueryRepository(private val queryFactory: JPAQueryFactory | |||||||
|             ), |             ), | ||||||
|             "%Y-%m-%d" |             "%Y-%m-%d" | ||||||
|         ) |         ) | ||||||
|  |         val currency = Expressions.stringTemplate("substring({0}, length({0}) - 2, 3)", payment.locale) | ||||||
|  |  | ||||||
|         return queryFactory |         return queryFactory | ||||||
|             .select( |             .select( | ||||||
|                 QGetChargeStatusQueryDto( |                 QGetChargeStatusResponse( | ||||||
|                     formattedDate, |                     formattedDate, | ||||||
|                     payment.price.sum(), |                     payment.price.sum(), | ||||||
|                     can1.price.sum(), |  | ||||||
|                     payment.id.count(), |                     payment.id.count(), | ||||||
|                     payment.paymentGateway |                     payment.paymentGateway.stringValue(), | ||||||
|  |                     currency.coalesce("KRW") | ||||||
|                 ) |                 ) | ||||||
|             ) |             ) | ||||||
|             .from(payment) |             .from(payment) | ||||||
| @@ -45,12 +48,47 @@ class AdminChargeStatusQueryRepository(private val queryFactory: JPAQueryFactory | |||||||
|                     .and(charge.status.eq(ChargeStatus.CHARGE)) |                     .and(charge.status.eq(ChargeStatus.CHARGE)) | ||||||
|                     .and(payment.status.eq(PaymentStatus.COMPLETE)) |                     .and(payment.status.eq(PaymentStatus.COMPLETE)) | ||||||
|             ) |             ) | ||||||
|             .groupBy(formattedDate, payment.paymentGateway) |             .groupBy(formattedDate, payment.paymentGateway, currency.coalesce("KRW")) | ||||||
|             .orderBy(formattedDate.desc()) |             .orderBy(formattedDate.desc()) | ||||||
|             .fetch() |             .fetch() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun getChargeStatusDetail(startDate: LocalDateTime, endDate: LocalDateTime): List<GetChargeStatusDetailQueryDto> { |     fun getChargeStatusSummary(startDate: LocalDateTime, endDate: LocalDateTime): List<GetChargeStatusResponse> { | ||||||
|  |         val currency = Expressions.stringTemplate( | ||||||
|  |             "substring({0}, length({0}) - 2, 3)", | ||||||
|  |             payment.locale | ||||||
|  |         ).coalesce("KRW") | ||||||
|  |  | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetChargeStatusResponse( | ||||||
|  |                     Expressions.stringTemplate("'합계'"), // date | ||||||
|  |                     payment.price.sum(), | ||||||
|  |                     payment.id.count(), | ||||||
|  |                     Expressions.stringTemplate("''"), | ||||||
|  |                     currency | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(payment) | ||||||
|  |             .innerJoin(payment.charge, charge) | ||||||
|  |             .leftJoin(charge.can, can1) | ||||||
|  |             .where( | ||||||
|  |                 charge.createdAt.goe(startDate) | ||||||
|  |                     .and(charge.createdAt.loe(endDate)) | ||||||
|  |                     .and(charge.status.eq(ChargeStatus.CHARGE)) | ||||||
|  |                     .and(payment.status.eq(PaymentStatus.COMPLETE)) | ||||||
|  |             ) | ||||||
|  |             .groupBy(currency) | ||||||
|  |             .orderBy(currency.asc()) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getChargeStatusDetail( | ||||||
|  |         startDate: LocalDateTime, | ||||||
|  |         endDate: LocalDateTime, | ||||||
|  |         paymentGateway: PaymentGateway, | ||||||
|  |         currency: String? = null | ||||||
|  |     ): List<GetChargeStatusDetailQueryDto> { | ||||||
|         val formattedDate = Expressions.stringTemplate( |         val formattedDate = Expressions.stringTemplate( | ||||||
|             "DATE_FORMAT({0}, {1})", |             "DATE_FORMAT({0}, {1})", | ||||||
|             Expressions.dateTimeTemplate( |             Expressions.dateTimeTemplate( | ||||||
| @@ -62,6 +100,20 @@ class AdminChargeStatusQueryRepository(private val queryFactory: JPAQueryFactory | |||||||
|             ), |             ), | ||||||
|             "%Y-%m-%d %H:%i:%s" |             "%Y-%m-%d %H:%i:%s" | ||||||
|         ) |         ) | ||||||
|  |         val currencyExpr = Expressions.stringTemplate( | ||||||
|  |             "substring({0}, length({0}) - 2, 3)", | ||||||
|  |             payment.locale | ||||||
|  |         ).coalesce("KRW") | ||||||
|  |         val whereBuilder = BooleanBuilder() | ||||||
|  |         whereBuilder.and(charge.createdAt.goe(startDate)) | ||||||
|  |             .and(charge.createdAt.loe(endDate)) | ||||||
|  |             .and(charge.status.eq(ChargeStatus.CHARGE)) | ||||||
|  |             .and(payment.status.eq(PaymentStatus.COMPLETE)) | ||||||
|  |             .and(payment.paymentGateway.eq(paymentGateway)) | ||||||
|  |  | ||||||
|  |         if (currency != null) { | ||||||
|  |             whereBuilder.and(currencyExpr.eq(currency)) | ||||||
|  |         } | ||||||
|  |  | ||||||
|         return queryFactory |         return queryFactory | ||||||
|             .select( |             .select( | ||||||
| @@ -70,7 +122,7 @@ class AdminChargeStatusQueryRepository(private val queryFactory: JPAQueryFactory | |||||||
|                     member.nickname, |                     member.nickname, | ||||||
|                     payment.method.coalesce(""), |                     payment.method.coalesce(""), | ||||||
|                     payment.price, |                     payment.price, | ||||||
|                     can1.price, |                     currencyExpr, | ||||||
|                     formattedDate |                     formattedDate | ||||||
|                 ) |                 ) | ||||||
|             ) |             ) | ||||||
| @@ -78,12 +130,7 @@ class AdminChargeStatusQueryRepository(private val queryFactory: JPAQueryFactory | |||||||
|             .innerJoin(charge.member, member) |             .innerJoin(charge.member, member) | ||||||
|             .innerJoin(charge.payment, payment) |             .innerJoin(charge.payment, payment) | ||||||
|             .leftJoin(charge.can, can1) |             .leftJoin(charge.can, can1) | ||||||
|             .where( |             .where(whereBuilder) | ||||||
|                 charge.createdAt.goe(startDate) |  | ||||||
|                     .and(charge.createdAt.loe(endDate)) |  | ||||||
|                     .and(charge.status.eq(ChargeStatus.CHARGE)) |  | ||||||
|                     .and(payment.status.eq(PaymentStatus.COMPLETE)) |  | ||||||
|             ) |  | ||||||
|             .orderBy(formattedDate.desc()) |             .orderBy(formattedDate.desc()) | ||||||
|             .fetch() |             .fetch() | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -20,48 +20,17 @@ class AdminChargeStatusService(val repository: AdminChargeStatusQueryRepository) | |||||||
|             .withZoneSameInstant(ZoneId.of("UTC")) |             .withZoneSameInstant(ZoneId.of("UTC")) | ||||||
|             .toLocalDateTime() |             .toLocalDateTime() | ||||||
|  |  | ||||||
|         var totalChargeAmount = 0 |         val summaryRows = repository.getChargeStatusSummary(startDate, endDate) | ||||||
|         var totalChargeCount = 0L |         val chargeStatusList = repository.getChargeStatus(startDate, endDate).toMutableList() | ||||||
|  |         chargeStatusList.addAll(0, summaryRows) | ||||||
|         val chargeStatusList = repository.getChargeStatus(startDate, endDate) |  | ||||||
|             .asSequence() |  | ||||||
|             .map { |  | ||||||
|                 val chargeAmount = if (it.paymentGateWay == PaymentGateway.APPLE_IAP) { |  | ||||||
|                     it.appleChargeAmount.toInt() |  | ||||||
|                 } else { |  | ||||||
|                     it.pgChargeAmount |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 val chargeCount = it.chargeCount |  | ||||||
|  |  | ||||||
|                 totalChargeAmount += chargeAmount |  | ||||||
|                 totalChargeCount += chargeCount |  | ||||||
|  |  | ||||||
|                 GetChargeStatusResponse( |  | ||||||
|                     date = it.date, |  | ||||||
|                     chargeAmount = chargeAmount, |  | ||||||
|                     chargeCount = chargeCount, |  | ||||||
|                     pg = it.paymentGateWay.name |  | ||||||
|                 ) |  | ||||||
|             } |  | ||||||
|             .toMutableList() |  | ||||||
|  |  | ||||||
|         chargeStatusList.add( |  | ||||||
|             0, |  | ||||||
|             GetChargeStatusResponse( |  | ||||||
|                 date = "합계", |  | ||||||
|                 chargeAmount = totalChargeAmount, |  | ||||||
|                 chargeCount = totalChargeCount, |  | ||||||
|                 pg = "" |  | ||||||
|             ) |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         return chargeStatusList.toList() |         return chargeStatusList.toList() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun getChargeStatusDetail( |     fun getChargeStatusDetail( | ||||||
|         startDateStr: String, |         startDateStr: String, | ||||||
|         paymentGateway: PaymentGateway |         paymentGateway: PaymentGateway, | ||||||
|  |         currency: String? = null | ||||||
|     ): List<GetChargeStatusDetailResponse> { |     ): List<GetChargeStatusDetailResponse> { | ||||||
|         val dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd") |         val dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd") | ||||||
|         val startDate = LocalDate.parse(startDateStr, dateTimeFormatter).atTime(0, 0, 0) |         val startDate = LocalDate.parse(startDateStr, dateTimeFormatter).atTime(0, 0, 0) | ||||||
| @@ -74,28 +43,16 @@ class AdminChargeStatusService(val repository: AdminChargeStatusQueryRepository) | |||||||
|             .withZoneSameInstant(ZoneId.of("UTC")) |             .withZoneSameInstant(ZoneId.of("UTC")) | ||||||
|             .toLocalDateTime() |             .toLocalDateTime() | ||||||
|  |  | ||||||
|         return repository.getChargeStatusDetail(startDate, endDate) |         return repository.getChargeStatusDetail(startDate, endDate, paymentGateway, currency) | ||||||
|             .asSequence() |  | ||||||
|             .filter { |  | ||||||
|                 if (paymentGateway == PaymentGateway.APPLE_IAP) { |  | ||||||
|                     it.appleChargeAmount > 0 |  | ||||||
|                 } else { |  | ||||||
|                     it.pgChargeAmount > 0 |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             .map { |             .map { | ||||||
|                 GetChargeStatusDetailResponse( |                 GetChargeStatusDetailResponse( | ||||||
|                     memberId = it.memberId, |                     memberId = it.memberId, | ||||||
|                     nickname = it.nickname, |                     nickname = it.nickname, | ||||||
|                     method = it.method, |                     method = it.method, | ||||||
|                     amount = if (paymentGateway == PaymentGateway.APPLE_IAP) { |                     amount = it.amount, | ||||||
|                         it.appleChargeAmount.toInt() |                     locale = it.locale, | ||||||
|                     } else { |  | ||||||
|                         it.pgChargeAmount |  | ||||||
|                     }, |  | ||||||
|                     datetime = it.datetime |                     datetime = it.datetime | ||||||
|                 ) |                 ) | ||||||
|             } |             } | ||||||
|             .toList() |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,12 +1,13 @@ | |||||||
| package kr.co.vividnext.sodalive.admin.charge | package kr.co.vividnext.sodalive.admin.charge | ||||||
|  |  | ||||||
| import com.querydsl.core.annotations.QueryProjection | import com.querydsl.core.annotations.QueryProjection | ||||||
|  | import java.math.BigDecimal | ||||||
|  |  | ||||||
| data class GetChargeStatusDetailQueryDto @QueryProjection constructor( | data class GetChargeStatusDetailQueryDto @QueryProjection constructor( | ||||||
|     val memberId: Long, |     val memberId: Long, | ||||||
|     val nickname: String, |     val nickname: String, | ||||||
|     val method: String, |     val method: String, | ||||||
|     val appleChargeAmount: Double, |     val amount: BigDecimal, | ||||||
|     val pgChargeAmount: Int, |     val locale: String, | ||||||
|     val datetime: String |     val datetime: String | ||||||
| ) | ) | ||||||
|   | |||||||
| @@ -1,9 +1,12 @@ | |||||||
| package kr.co.vividnext.sodalive.admin.charge | package kr.co.vividnext.sodalive.admin.charge | ||||||
|  |  | ||||||
|  | import java.math.BigDecimal | ||||||
|  |  | ||||||
| data class GetChargeStatusDetailResponse( | data class GetChargeStatusDetailResponse( | ||||||
|     val memberId: Long, |     val memberId: Long, | ||||||
|     val nickname: String, |     val nickname: String, | ||||||
|     val method: String, |     val method: String, | ||||||
|     val amount: Int, |     val amount: BigDecimal, | ||||||
|  |     val locale: String, | ||||||
|     val datetime: String |     val datetime: String | ||||||
| ) | ) | ||||||
|   | |||||||
| @@ -1,12 +0,0 @@ | |||||||
| package kr.co.vividnext.sodalive.admin.charge |  | ||||||
|  |  | ||||||
| import com.querydsl.core.annotations.QueryProjection |  | ||||||
| import kr.co.vividnext.sodalive.can.payment.PaymentGateway |  | ||||||
|  |  | ||||||
| data class GetChargeStatusQueryDto @QueryProjection constructor( |  | ||||||
|     val date: String, |  | ||||||
|     val appleChargeAmount: Double, |  | ||||||
|     val pgChargeAmount: Int, |  | ||||||
|     val chargeCount: Long, |  | ||||||
|     val paymentGateWay: PaymentGateway |  | ||||||
| ) |  | ||||||
| @@ -1,8 +1,12 @@ | |||||||
| package kr.co.vividnext.sodalive.admin.charge | package kr.co.vividnext.sodalive.admin.charge | ||||||
|  |  | ||||||
| data class GetChargeStatusResponse( | import com.querydsl.core.annotations.QueryProjection | ||||||
|  | import java.math.BigDecimal | ||||||
|  |  | ||||||
|  | data class GetChargeStatusResponse @QueryProjection constructor( | ||||||
|     val date: String, |     val date: String, | ||||||
|     val chargeAmount: Int, |     val chargeAmount: BigDecimal, | ||||||
|     val chargeCount: Long, |     val chargeCount: Long, | ||||||
|     val pg: String |     val pg: String, | ||||||
|  |     val currency: String | ||||||
| ) | ) | ||||||
|   | |||||||
| @@ -0,0 +1,229 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat | ||||||
|  |  | ||||||
|  | import com.amazonaws.services.s3.model.ObjectMetadata | ||||||
|  | import com.fasterxml.jackson.databind.ObjectMapper | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.character.dto.ChatCharacterSearchListPageResponse | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.character.service.AdminChatCharacterService | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.dto.ChatCharacterBannerListPageResponse | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.dto.ChatCharacterBannerRegisterRequest | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.dto.ChatCharacterBannerResponse | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.dto.ChatCharacterBannerUpdateRequest | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.dto.UpdateBannerOrdersRequest | ||||||
|  | import kr.co.vividnext.sodalive.aws.s3.S3Uploader | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.service.ChatCharacterBannerService | ||||||
|  | import kr.co.vividnext.sodalive.common.ApiResponse | ||||||
|  | import kr.co.vividnext.sodalive.common.SodaException | ||||||
|  | import kr.co.vividnext.sodalive.utils.generateFileName | ||||||
|  | import org.springframework.beans.factory.annotation.Value | ||||||
|  | import org.springframework.security.access.prepost.PreAuthorize | ||||||
|  | import org.springframework.web.bind.annotation.DeleteMapping | ||||||
|  | import org.springframework.web.bind.annotation.GetMapping | ||||||
|  | import org.springframework.web.bind.annotation.PathVariable | ||||||
|  | import org.springframework.web.bind.annotation.PostMapping | ||||||
|  | import org.springframework.web.bind.annotation.PutMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestBody | ||||||
|  | import org.springframework.web.bind.annotation.RequestMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestParam | ||||||
|  | import org.springframework.web.bind.annotation.RequestPart | ||||||
|  | import org.springframework.web.bind.annotation.RestController | ||||||
|  | import org.springframework.web.multipart.MultipartFile | ||||||
|  |  | ||||||
|  | @RestController | ||||||
|  | @RequestMapping("/admin/chat/banner") | ||||||
|  | @PreAuthorize("hasRole('ADMIN')") | ||||||
|  | class AdminChatBannerController( | ||||||
|  |     private val bannerService: ChatCharacterBannerService, | ||||||
|  |     private val adminCharacterService: AdminChatCharacterService, | ||||||
|  |     private val s3Uploader: S3Uploader, | ||||||
|  |  | ||||||
|  |     @Value("\${cloud.aws.s3.bucket}") | ||||||
|  |     private val s3Bucket: String, | ||||||
|  |  | ||||||
|  |     @Value("\${cloud.aws.cloud-front.host}") | ||||||
|  |     private val imageHost: String | ||||||
|  | ) { | ||||||
|  |     /** | ||||||
|  |      * 활성화된 배너 목록 조회 API | ||||||
|  |      * | ||||||
|  |      * @param page 페이지 번호 (0부터 시작, 기본값 0) | ||||||
|  |      * @param size 페이지 크기 (기본값 20) | ||||||
|  |      * @return 페이징된 배너 목록 | ||||||
|  |      */ | ||||||
|  |     @GetMapping("/list") | ||||||
|  |     fun getBannerList( | ||||||
|  |         @RequestParam(defaultValue = "0") page: Int, | ||||||
|  |         @RequestParam(defaultValue = "20") size: Int | ||||||
|  |     ) = run { | ||||||
|  |         val pageable = adminCharacterService.createDefaultPageRequest(page, size) | ||||||
|  |         val banners = bannerService.getActiveBanners(pageable) | ||||||
|  |         val response = ChatCharacterBannerListPageResponse( | ||||||
|  |             totalCount = banners.totalElements, | ||||||
|  |             content = banners.content.map { ChatCharacterBannerResponse.from(it, imageHost) } | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         ApiResponse.ok(response) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 배너 상세 조회 API | ||||||
|  |      * | ||||||
|  |      * @param bannerId 배너 ID | ||||||
|  |      * @return 배너 상세 정보 | ||||||
|  |      */ | ||||||
|  |     @GetMapping("/{bannerId}") | ||||||
|  |     fun getBannerDetail(@PathVariable bannerId: Long) = run { | ||||||
|  |         val banner = bannerService.getBannerById(bannerId) | ||||||
|  |         val response = ChatCharacterBannerResponse.from(banner, imageHost) | ||||||
|  |  | ||||||
|  |         ApiResponse.ok(response) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 캐릭터 검색 API (배너 등록을 위한) | ||||||
|  |      * | ||||||
|  |      * @param searchTerm 검색어 (이름, 설명, MBTI, 태그) | ||||||
|  |      * @param page 페이지 번호 (0부터 시작, 기본값 0) | ||||||
|  |      * @param size 페이지 크기 (기본값 20) | ||||||
|  |      * @return 검색된 캐릭터 목록 | ||||||
|  |      */ | ||||||
|  |     @GetMapping("/search-character") | ||||||
|  |     fun searchCharacters( | ||||||
|  |         @RequestParam searchTerm: String, | ||||||
|  |         @RequestParam(defaultValue = "0") page: Int, | ||||||
|  |         @RequestParam(defaultValue = "20") size: Int | ||||||
|  |     ) = run { | ||||||
|  |         val pageable = adminCharacterService.createDefaultPageRequest(page, size) | ||||||
|  |         val pageResult = adminCharacterService.searchCharacters(searchTerm, pageable, imageHost) | ||||||
|  |         val response = ChatCharacterSearchListPageResponse( | ||||||
|  |             totalCount = pageResult.totalElements, | ||||||
|  |             content = pageResult.content | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         ApiResponse.ok(response) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 배너 등록 API | ||||||
|  |      * | ||||||
|  |      * @param image 배너 이미지 | ||||||
|  |      * @param requestString 배너 등록 요청 정보 (캐릭터 ID와 선택적으로 정렬 순서 포함) | ||||||
|  |      * @return 등록된 배너 정보 | ||||||
|  |      */ | ||||||
|  |     @PostMapping("/register") | ||||||
|  |     fun registerBanner( | ||||||
|  |         @RequestPart("image") image: MultipartFile, | ||||||
|  |         @RequestPart("request") requestString: String | ||||||
|  |     ) = run { | ||||||
|  |         val objectMapper = ObjectMapper() | ||||||
|  |         val request = objectMapper.readValue( | ||||||
|  |             requestString, | ||||||
|  |             ChatCharacterBannerRegisterRequest::class.java | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         // 1. 먼저 빈 이미지 경로로 배너 등록 (정렬 순서 포함) | ||||||
|  |         val banner = bannerService.registerBanner( | ||||||
|  |             characterId = request.characterId, | ||||||
|  |             imagePath = "" | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         // 2. 배너 ID를 사용하여 이미지 업로드 | ||||||
|  |         val imagePath = saveImage(banner.id!!, image) | ||||||
|  |  | ||||||
|  |         // 3. 이미지 경로로 배너 업데이트 | ||||||
|  |         val updatedBanner = bannerService.updateBanner(banner.id!!, imagePath) | ||||||
|  |  | ||||||
|  |         val response = ChatCharacterBannerResponse.from(updatedBanner, imageHost) | ||||||
|  |  | ||||||
|  |         ApiResponse.ok(response) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 이미지를 S3에 업로드하고 경로를 반환 | ||||||
|  |      * | ||||||
|  |      * @param bannerId 배너 ID (이미지 경로에 사용) | ||||||
|  |      * @param image 업로드할 이미지 파일 | ||||||
|  |      * @return 업로드된 이미지 경로 | ||||||
|  |      */ | ||||||
|  |     private fun saveImage(bannerId: Long, image: MultipartFile): String { | ||||||
|  |         try { | ||||||
|  |             val metadata = ObjectMetadata() | ||||||
|  |             metadata.contentLength = image.size | ||||||
|  |  | ||||||
|  |             val fileName = generateFileName("character-banner") | ||||||
|  |  | ||||||
|  |             // S3에 이미지 업로드 (배너 ID를 경로에 사용) | ||||||
|  |             return s3Uploader.upload( | ||||||
|  |                 inputStream = image.inputStream, | ||||||
|  |                 bucket = s3Bucket, | ||||||
|  |                 filePath = "characters/banners/$bannerId/$fileName", | ||||||
|  |                 metadata = metadata | ||||||
|  |             ) | ||||||
|  |         } catch (e: Exception) { | ||||||
|  |             throw SodaException("이미지 저장에 실패했습니다: ${e.message}") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 배너 수정 API | ||||||
|  |      * | ||||||
|  |      * @param image 배너 이미지 | ||||||
|  |      * @param requestString 배너 수정 요청 정보 (배너 ID와 선택적으로 캐릭터 ID 포함) | ||||||
|  |      * @return 수정된 배너 정보 | ||||||
|  |      */ | ||||||
|  |     @PutMapping("/update") | ||||||
|  |     fun updateBanner( | ||||||
|  |         @RequestPart("image") image: MultipartFile, | ||||||
|  |         @RequestPart("request") requestString: String | ||||||
|  |     ) = run { | ||||||
|  |         val objectMapper = ObjectMapper() | ||||||
|  |         val request = objectMapper.readValue( | ||||||
|  |             requestString, | ||||||
|  |             ChatCharacterBannerUpdateRequest::class.java | ||||||
|  |         ) | ||||||
|  |         // 배너 정보 조회 | ||||||
|  |         bannerService.getBannerById(request.bannerId) | ||||||
|  |  | ||||||
|  |         // 배너 ID를 사용하여 이미지 업로드 | ||||||
|  |         val imagePath = saveImage(request.bannerId, image) | ||||||
|  |  | ||||||
|  |         // 배너 수정 (이미지와 캐릭터 모두 수정 가능) | ||||||
|  |         val updatedBanner = bannerService.updateBanner( | ||||||
|  |             bannerId = request.bannerId, | ||||||
|  |             imagePath = imagePath, | ||||||
|  |             characterId = request.characterId | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         val response = ChatCharacterBannerResponse.from(updatedBanner, imageHost) | ||||||
|  |  | ||||||
|  |         ApiResponse.ok(response) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 배너 삭제 API (소프트 삭제) | ||||||
|  |      * | ||||||
|  |      * @param bannerId 배너 ID | ||||||
|  |      * @return 성공 여부 | ||||||
|  |      */ | ||||||
|  |     @DeleteMapping("/{bannerId}") | ||||||
|  |     fun deleteBanner(@PathVariable bannerId: Long) = run { | ||||||
|  |         bannerService.deleteBanner(bannerId) | ||||||
|  |  | ||||||
|  |         ApiResponse.ok("배너가 성공적으로 삭제되었습니다.") | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 배너 정렬 순서 일괄 변경 API | ||||||
|  |      * ID 목록의 순서대로 정렬 순서를 1부터 순차적으로 설정합니다. | ||||||
|  |      * | ||||||
|  |      * @param request 정렬 순서 일괄 변경 요청 정보 (배너 ID 목록) | ||||||
|  |      * @return 성공 메시지 | ||||||
|  |      */ | ||||||
|  |     @PutMapping("/orders") | ||||||
|  |     fun updateBannerOrders( | ||||||
|  |         @RequestBody request: UpdateBannerOrdersRequest | ||||||
|  |     ) = run { | ||||||
|  |         bannerService.updateBannerOrders(request.ids) | ||||||
|  |  | ||||||
|  |         ApiResponse.ok(null, "배너 정렬 순서가 성공적으로 변경되었습니다.") | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,32 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.calculate | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.common.ApiResponse | ||||||
|  | import org.springframework.data.domain.Pageable | ||||||
|  | import org.springframework.security.access.prepost.PreAuthorize | ||||||
|  | import org.springframework.web.bind.annotation.GetMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestParam | ||||||
|  | import org.springframework.web.bind.annotation.RestController | ||||||
|  |  | ||||||
|  | @RestController | ||||||
|  | @PreAuthorize("hasRole('ADMIN')") | ||||||
|  | @RequestMapping("/admin/chat/calculate") | ||||||
|  | class AdminChatCalculateController( | ||||||
|  |     private val service: AdminChatCalculateService | ||||||
|  | ) { | ||||||
|  |     @GetMapping("/characters") | ||||||
|  |     fun getCharacterCalculate( | ||||||
|  |         @RequestParam startDateStr: String, | ||||||
|  |         @RequestParam endDateStr: String, | ||||||
|  |         @RequestParam(required = false, defaultValue = "TOTAL_SALES_DESC") sort: ChatCharacterCalculateSort, | ||||||
|  |         pageable: Pageable | ||||||
|  |     ) = ApiResponse.ok( | ||||||
|  |         service.getCharacterCalculate( | ||||||
|  |             startDateStr, | ||||||
|  |             endDateStr, | ||||||
|  |             sort, | ||||||
|  |             pageable.offset, | ||||||
|  |             pageable.pageSize | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  | } | ||||||
| @@ -0,0 +1,139 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.calculate | ||||||
|  |  | ||||||
|  | import com.querydsl.core.types.Projections | ||||||
|  | import com.querydsl.core.types.dsl.CaseBuilder | ||||||
|  | import com.querydsl.core.types.dsl.Expressions | ||||||
|  | import com.querydsl.jpa.impl.JPAQueryFactory | ||||||
|  | import kr.co.vividnext.sodalive.can.use.CanUsage | ||||||
|  | import kr.co.vividnext.sodalive.can.use.QUseCan.useCan | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.QChatCharacter | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.image.QCharacterImage.characterImage | ||||||
|  | import org.springframework.beans.factory.annotation.Value | ||||||
|  | import org.springframework.stereotype.Repository | ||||||
|  | import java.time.LocalDateTime | ||||||
|  |  | ||||||
|  | @Repository | ||||||
|  | class AdminChatCalculateQueryRepository( | ||||||
|  |     private val queryFactory: JPAQueryFactory, | ||||||
|  |  | ||||||
|  |     @Value("\${cloud.aws.cloud-front.host}") | ||||||
|  |     private val imageHost: String | ||||||
|  | ) { | ||||||
|  |     fun getCharacterCalculate( | ||||||
|  |         startUtc: LocalDateTime, | ||||||
|  |         endInclusiveUtc: LocalDateTime, | ||||||
|  |         sort: ChatCharacterCalculateSort, | ||||||
|  |         offset: Long, | ||||||
|  |         limit: Long | ||||||
|  |     ): List<ChatCharacterCalculateQueryData> { | ||||||
|  |         val imageCanExpr = CaseBuilder() | ||||||
|  |             .`when`(useCan.canUsage.eq(CanUsage.CHARACTER_IMAGE_PURCHASE)) | ||||||
|  |             .then(useCan.can.add(useCan.rewardCan)) | ||||||
|  |             .otherwise(0) | ||||||
|  |  | ||||||
|  |         val messageCanExpr = CaseBuilder() | ||||||
|  |             .`when`(useCan.canUsage.eq(CanUsage.CHAT_MESSAGE_PURCHASE)) | ||||||
|  |             .then(useCan.can.add(useCan.rewardCan)) | ||||||
|  |             .otherwise(0) | ||||||
|  |  | ||||||
|  |         val quotaCanExpr = CaseBuilder() | ||||||
|  |             .`when`(useCan.canUsage.eq(CanUsage.CHAT_QUOTA_PURCHASE)) | ||||||
|  |             .then(useCan.can.add(useCan.rewardCan)) | ||||||
|  |             .otherwise(0) | ||||||
|  |  | ||||||
|  |         val imageSum = imageCanExpr.sum() | ||||||
|  |         val messageSum = messageCanExpr.sum() | ||||||
|  |         val quotaSum = quotaCanExpr.sum() | ||||||
|  |         val totalSum = imageSum.add(messageSum).add(quotaSum) | ||||||
|  |  | ||||||
|  |         // 캐릭터 조인: 이미지 경로를 통한 캐릭터(c1) + characterId 직접 지정(c2) | ||||||
|  |         val c1 = QChatCharacter("c1") | ||||||
|  |         val c2 = QChatCharacter("c2") | ||||||
|  |  | ||||||
|  |         val characterIdExpr = c1.id.coalesce(c2.id) | ||||||
|  |         val characterNameAgg = Expressions.stringTemplate( | ||||||
|  |             "coalesce(max({0}), max({1}), '')", | ||||||
|  |             c1.name, | ||||||
|  |             c2.name | ||||||
|  |         ) | ||||||
|  |         val characterImagePathAgg = Expressions.stringTemplate( | ||||||
|  |             "coalesce(max({0}), max({1}))", | ||||||
|  |             c1.imagePath, | ||||||
|  |             c2.imagePath | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         val query = queryFactory | ||||||
|  |             .select( | ||||||
|  |                 Projections.constructor( | ||||||
|  |                     ChatCharacterCalculateQueryData::class.java, | ||||||
|  |                     characterIdExpr, | ||||||
|  |                     characterNameAgg, | ||||||
|  |                     characterImagePathAgg.prepend("/").prepend(imageHost), | ||||||
|  |                     imageSum, | ||||||
|  |                     messageSum, | ||||||
|  |                     quotaSum | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(useCan) | ||||||
|  |             .leftJoin(useCan.characterImage, characterImage) | ||||||
|  |             .leftJoin(characterImage.chatCharacter, c1) | ||||||
|  |             .leftJoin(c2).on(c2.id.eq(useCan.characterId)) | ||||||
|  |             .where( | ||||||
|  |                 useCan.isRefund.isFalse | ||||||
|  |                     .and( | ||||||
|  |                         useCan.canUsage.`in`( | ||||||
|  |                             CanUsage.CHARACTER_IMAGE_PURCHASE, | ||||||
|  |                             CanUsage.CHAT_MESSAGE_PURCHASE, | ||||||
|  |                             CanUsage.CHAT_QUOTA_PURCHASE | ||||||
|  |                         ) | ||||||
|  |                     ) | ||||||
|  |                     .and(useCan.createdAt.goe(startUtc)) | ||||||
|  |                     .and(useCan.createdAt.loe(endInclusiveUtc)) | ||||||
|  |             ) | ||||||
|  |             .groupBy(characterIdExpr) | ||||||
|  |  | ||||||
|  |         when (sort) { | ||||||
|  |             ChatCharacterCalculateSort.TOTAL_SALES_DESC -> | ||||||
|  |                 query.orderBy(totalSum.desc(), characterIdExpr.desc()) | ||||||
|  |  | ||||||
|  |             ChatCharacterCalculateSort.LATEST_DESC -> | ||||||
|  |                 query.orderBy(characterIdExpr.desc(), totalSum.desc()) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return query | ||||||
|  |             .offset(offset) | ||||||
|  |             .limit(limit) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCharacterCalculateTotalCount( | ||||||
|  |         startUtc: LocalDateTime, | ||||||
|  |         endInclusiveUtc: LocalDateTime | ||||||
|  |     ): Int { | ||||||
|  |         val c1 = QChatCharacter("c1") | ||||||
|  |         val c2 = QChatCharacter("c2") | ||||||
|  |         val characterIdExpr = c1.id.coalesce(c2.id) | ||||||
|  |  | ||||||
|  |         return queryFactory | ||||||
|  |             .select(characterIdExpr) | ||||||
|  |             .from(useCan) | ||||||
|  |             .leftJoin(useCan.characterImage, characterImage) | ||||||
|  |             .leftJoin(characterImage.chatCharacter, c1) | ||||||
|  |             .leftJoin(c2).on(c2.id.eq(useCan.characterId)) | ||||||
|  |             .where( | ||||||
|  |                 useCan.isRefund.isFalse | ||||||
|  |                     .and( | ||||||
|  |                         useCan.canUsage.`in`( | ||||||
|  |                             CanUsage.CHARACTER_IMAGE_PURCHASE, | ||||||
|  |                             CanUsage.CHAT_MESSAGE_PURCHASE, | ||||||
|  |                             CanUsage.CHAT_QUOTA_PURCHASE | ||||||
|  |                         ) | ||||||
|  |                     ) | ||||||
|  |                     .and(useCan.createdAt.goe(startUtc)) | ||||||
|  |                     .and(useCan.createdAt.loe(endInclusiveUtc)) | ||||||
|  |             ) | ||||||
|  |             .groupBy(characterIdExpr) | ||||||
|  |             .fetch() | ||||||
|  |             .size | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,49 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.calculate | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.common.SodaException | ||||||
|  | import kr.co.vividnext.sodalive.extensions.convertLocalDateTime | ||||||
|  | import org.springframework.stereotype.Service | ||||||
|  | import org.springframework.transaction.annotation.Transactional | ||||||
|  | import java.time.LocalDate | ||||||
|  | import java.time.ZoneId | ||||||
|  | import java.time.format.DateTimeFormatter | ||||||
|  |  | ||||||
|  | @Service | ||||||
|  | class AdminChatCalculateService( | ||||||
|  |     private val repository: AdminChatCalculateQueryRepository | ||||||
|  | ) { | ||||||
|  |     private val dateFormatter: DateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd") | ||||||
|  |     private val kstZone: ZoneId = ZoneId.of("Asia/Seoul") | ||||||
|  |  | ||||||
|  |     @Transactional(readOnly = true) | ||||||
|  |     fun getCharacterCalculate( | ||||||
|  |         startDateStr: String, | ||||||
|  |         endDateStr: String, | ||||||
|  |         sort: ChatCharacterCalculateSort, | ||||||
|  |         offset: Long, | ||||||
|  |         pageSize: Int | ||||||
|  |     ): ChatCharacterCalculateResponse { | ||||||
|  |         // 날짜 유효성 검증 (KST 기준) | ||||||
|  |         val startDate = LocalDate.parse(startDateStr, dateFormatter) | ||||||
|  |         val endDate = LocalDate.parse(endDateStr, dateFormatter) | ||||||
|  |         val todayKst = LocalDate.now(kstZone) | ||||||
|  |  | ||||||
|  |         if (endDate.isAfter(todayKst)) { | ||||||
|  |             throw SodaException("끝 날짜는 오늘 날짜까지만 입력 가능합니다.") | ||||||
|  |         } | ||||||
|  |         if (startDate.isAfter(endDate)) { | ||||||
|  |             throw SodaException("시작 날짜는 끝 날짜보다 이후일 수 없습니다.") | ||||||
|  |         } | ||||||
|  |         if (endDate.isAfter(startDate.plusMonths(6))) { | ||||||
|  |             throw SodaException("조회 가능 기간은 최대 6개월입니다.") | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         val startUtc = startDateStr.convertLocalDateTime() | ||||||
|  |         val endInclusiveUtc = endDateStr.convertLocalDateTime(hour = 23, minute = 59, second = 59) | ||||||
|  |  | ||||||
|  |         val totalCount = repository.getCharacterCalculateTotalCount(startUtc, endInclusiveUtc) | ||||||
|  |         val rows = repository.getCharacterCalculate(startUtc, endInclusiveUtc, sort, offset, pageSize.toLong()) | ||||||
|  |         val items = rows.map { it.toItem() } | ||||||
|  |         return ChatCharacterCalculateResponse(totalCount, items) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,62 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.calculate | ||||||
|  |  | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import com.querydsl.core.annotations.QueryProjection | ||||||
|  | import java.math.BigDecimal | ||||||
|  | import java.math.RoundingMode | ||||||
|  |  | ||||||
|  | // 정렬 옵션 | ||||||
|  | enum class ChatCharacterCalculateSort { | ||||||
|  |     TOTAL_SALES_DESC, | ||||||
|  |     LATEST_DESC | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // QueryDSL 프로젝션용 DTO | ||||||
|  | data class ChatCharacterCalculateQueryData @QueryProjection constructor( | ||||||
|  |     val characterId: Long, | ||||||
|  |     val characterName: String, | ||||||
|  |     val characterImagePath: String?, | ||||||
|  |     val imagePurchaseCan: Int?, | ||||||
|  |     val messagePurchaseCan: Int?, | ||||||
|  |     val quotaPurchaseCan: Int? | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // 응답 DTO (아이템) | ||||||
|  | data class ChatCharacterCalculateItem( | ||||||
|  |     @JsonProperty("characterId") val characterId: Long, | ||||||
|  |     @JsonProperty("characterImage") val characterImage: String?, | ||||||
|  |     @JsonProperty("name") val name: String, | ||||||
|  |     @JsonProperty("imagePurchaseCan") val imagePurchaseCan: Int, | ||||||
|  |     @JsonProperty("messagePurchaseCan") val messagePurchaseCan: Int, | ||||||
|  |     @JsonProperty("quotaPurchaseCan") val quotaPurchaseCan: Int, | ||||||
|  |     @JsonProperty("totalCan") val totalCan: Int, | ||||||
|  |     @JsonProperty("totalKrw") val totalKrw: Int, | ||||||
|  |     @JsonProperty("settlementKrw") val settlementKrw: Int | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // 응답 DTO (전체) | ||||||
|  | data class ChatCharacterCalculateResponse( | ||||||
|  |     @JsonProperty("totalCount") val totalCount: Int, | ||||||
|  |     @JsonProperty("items") val items: List<ChatCharacterCalculateItem> | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | fun ChatCharacterCalculateQueryData.toItem(): ChatCharacterCalculateItem { | ||||||
|  |     val image = imagePurchaseCan ?: 0 | ||||||
|  |     val message = messagePurchaseCan ?: 0 | ||||||
|  |     val quota = quotaPurchaseCan ?: 0 | ||||||
|  |     val total = image + message + quota | ||||||
|  |     val totalKrw = BigDecimal(total).multiply(BigDecimal(100)) | ||||||
|  |     val settlement = totalKrw.multiply(BigDecimal("0.10")).setScale(0, RoundingMode.HALF_UP) | ||||||
|  |  | ||||||
|  |     return ChatCharacterCalculateItem( | ||||||
|  |         characterId = characterId, | ||||||
|  |         characterImage = characterImagePath, | ||||||
|  |         name = characterName, | ||||||
|  |         imagePurchaseCan = image, | ||||||
|  |         messagePurchaseCan = message, | ||||||
|  |         quotaPurchaseCan = quota, | ||||||
|  |         totalCan = total, | ||||||
|  |         totalKrw = totalKrw.toInt(), | ||||||
|  |         settlementKrw = settlement.toInt() | ||||||
|  |     ) | ||||||
|  | } | ||||||
| @@ -0,0 +1,423 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.character | ||||||
|  |  | ||||||
|  | import com.amazonaws.services.s3.model.ObjectMetadata | ||||||
|  | import com.fasterxml.jackson.databind.ObjectMapper | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.character.dto.ChatCharacterRegisterRequest | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.character.dto.ChatCharacterSearchListPageResponse | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.character.dto.ChatCharacterUpdateRequest | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.character.dto.ExternalApiResponse | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.character.service.AdminChatCharacterService | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.original.service.AdminOriginalWorkService | ||||||
|  | import kr.co.vividnext.sodalive.aws.s3.S3Uploader | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.CharacterType | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.service.ChatCharacterService | ||||||
|  | import kr.co.vividnext.sodalive.common.ApiResponse | ||||||
|  | import kr.co.vividnext.sodalive.common.SodaException | ||||||
|  | import kr.co.vividnext.sodalive.utils.generateFileName | ||||||
|  | import org.springframework.beans.factory.annotation.Value | ||||||
|  | import org.springframework.http.HttpEntity | ||||||
|  | import org.springframework.http.HttpHeaders | ||||||
|  | import org.springframework.http.HttpMethod | ||||||
|  | import org.springframework.http.MediaType | ||||||
|  | import org.springframework.http.client.SimpleClientHttpRequestFactory | ||||||
|  | import org.springframework.security.access.prepost.PreAuthorize | ||||||
|  | import org.springframework.web.bind.annotation.GetMapping | ||||||
|  | import org.springframework.web.bind.annotation.PathVariable | ||||||
|  | import org.springframework.web.bind.annotation.PostMapping | ||||||
|  | import org.springframework.web.bind.annotation.PutMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestParam | ||||||
|  | import org.springframework.web.bind.annotation.RequestPart | ||||||
|  | import org.springframework.web.bind.annotation.RestController | ||||||
|  | import org.springframework.web.client.RestTemplate | ||||||
|  | import org.springframework.web.multipart.MultipartFile | ||||||
|  |  | ||||||
|  | @RestController | ||||||
|  | @RequestMapping("/admin/chat/character") | ||||||
|  | @PreAuthorize("hasRole('ADMIN')") | ||||||
|  | class AdminChatCharacterController( | ||||||
|  |     private val service: ChatCharacterService, | ||||||
|  |     private val adminService: AdminChatCharacterService, | ||||||
|  |     private val s3Uploader: S3Uploader, | ||||||
|  |     private val originalWorkService: AdminOriginalWorkService, | ||||||
|  |  | ||||||
|  |     @Value("\${weraser.api-key}") | ||||||
|  |     private val apiKey: String, | ||||||
|  |  | ||||||
|  |     @Value("\${weraser.api-url}") | ||||||
|  |     private val apiUrl: String, | ||||||
|  |  | ||||||
|  |     @Value("\${cloud.aws.s3.bucket}") | ||||||
|  |     private val s3Bucket: String, | ||||||
|  |  | ||||||
|  |     @Value("\${cloud.aws.cloud-front.host}") | ||||||
|  |     private val imageHost: String | ||||||
|  | ) { | ||||||
|  |     /** | ||||||
|  |      * 활성화된 캐릭터 목록 조회 API | ||||||
|  |      * | ||||||
|  |      * @param page 페이지 번호 (0부터 시작, 기본값 0) | ||||||
|  |      * @param size 페이지 크기 (기본값 20) | ||||||
|  |      * @return 페이징된 캐릭터 목록 | ||||||
|  |      */ | ||||||
|  |     @GetMapping("/list") | ||||||
|  |     fun getCharacterList( | ||||||
|  |         @RequestParam(defaultValue = "0") page: Int, | ||||||
|  |         @RequestParam(defaultValue = "20") size: Int | ||||||
|  |     ) = run { | ||||||
|  |         val pageable = adminService.createDefaultPageRequest(page, size) | ||||||
|  |         val response = adminService.getActiveChatCharacters(pageable, imageHost) | ||||||
|  |  | ||||||
|  |         ApiResponse.ok(response) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 캐릭터 검색(관리자) | ||||||
|  |      * - 이름/설명/MBTI/태그 기준 부분 검색, 활성 캐릭터만 대상 | ||||||
|  |      * - 페이징 지원: page, size 파라미터 사용 | ||||||
|  |      */ | ||||||
|  |     @GetMapping("/search") | ||||||
|  |     fun searchCharacters( | ||||||
|  |         @RequestParam("searchTerm") searchTerm: String, | ||||||
|  |         @RequestParam(defaultValue = "0") page: Int, | ||||||
|  |         @RequestParam(defaultValue = "20") size: Int | ||||||
|  |     ) = run { | ||||||
|  |         val pageable = adminService.createDefaultPageRequest(page, size) | ||||||
|  |         val resultPage = adminService.searchCharacters(searchTerm, pageable, imageHost) | ||||||
|  |         val response = ChatCharacterSearchListPageResponse( | ||||||
|  |             totalCount = resultPage.totalElements, | ||||||
|  |             content = resultPage.content | ||||||
|  |         ) | ||||||
|  |         ApiResponse.ok(response) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 캐릭터 상세 정보 조회 API | ||||||
|  |      * | ||||||
|  |      * @param characterId 캐릭터 ID | ||||||
|  |      * @return 캐릭터 상세 정보 | ||||||
|  |      */ | ||||||
|  |     @GetMapping("/{characterId}") | ||||||
|  |     fun getCharacterDetail( | ||||||
|  |         @PathVariable characterId: Long | ||||||
|  |     ) = run { | ||||||
|  |         val response = adminService.getChatCharacterDetail(characterId, imageHost) | ||||||
|  |  | ||||||
|  |         ApiResponse.ok(response) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @PostMapping("/register") | ||||||
|  |     fun registerCharacter( | ||||||
|  |         @RequestPart("image") image: MultipartFile, | ||||||
|  |         @RequestPart("request") requestString: String | ||||||
|  |     ) = run { | ||||||
|  |         // JSON 문자열을 ChatCharacterRegisterRequest 객체로 변환 | ||||||
|  |         val objectMapper = ObjectMapper() | ||||||
|  |         val request = objectMapper.readValue(requestString, ChatCharacterRegisterRequest::class.java) | ||||||
|  |  | ||||||
|  |         // 외부 API 호출 전 DB에 동일한 이름이 있는지 조회 | ||||||
|  |         val existingCharacter = service.findByName(request.name) | ||||||
|  |         if (existingCharacter != null) { | ||||||
|  |             throw SodaException("동일한 이름은 등록이 불가능합니다: ${request.name}") | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // 1. 외부 API 호출 | ||||||
|  |         val characterUUID = callExternalApi(request) | ||||||
|  |  | ||||||
|  |         // 2. ChatCharacter 저장 | ||||||
|  |         val chatCharacter = service.createChatCharacterWithDetails( | ||||||
|  |             characterUUID = characterUUID, | ||||||
|  |             name = request.name, | ||||||
|  |             description = request.description, | ||||||
|  |             systemPrompt = request.systemPrompt, | ||||||
|  |             age = request.age?.toIntOrNull(), | ||||||
|  |             gender = request.gender, | ||||||
|  |             mbti = request.mbti, | ||||||
|  |             speechPattern = request.speechPattern, | ||||||
|  |             speechStyle = request.speechStyle, | ||||||
|  |             appearance = request.appearance, | ||||||
|  |             originalTitle = request.originalTitle, | ||||||
|  |             originalLink = request.originalLink, | ||||||
|  |             characterType = request.characterType?.let { | ||||||
|  |                 runCatching { CharacterType.valueOf(it) } | ||||||
|  |                     .getOrDefault(CharacterType.Character) | ||||||
|  |             } ?: CharacterType.Character, | ||||||
|  |             tags = request.tags, | ||||||
|  |             values = request.values, | ||||||
|  |             hobbies = request.hobbies, | ||||||
|  |             goals = request.goals, | ||||||
|  |             memories = request.memories.map { Triple(it.title, it.content, it.emotion) }, | ||||||
|  |             personalities = request.personalities.map { Pair(it.trait, it.description) }, | ||||||
|  |             backgrounds = request.backgrounds.map { Pair(it.topic, it.description) }, | ||||||
|  |             relationships = request.relationships | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         // 3. 이미지 저장 및 ChatCharacter에 이미지 path 설정 | ||||||
|  |         val imagePath = saveImage( | ||||||
|  |             characterId = chatCharacter.id!!, | ||||||
|  |             image = image | ||||||
|  |         ) | ||||||
|  |         chatCharacter.imagePath = imagePath | ||||||
|  |         service.saveChatCharacter(chatCharacter) | ||||||
|  |  | ||||||
|  |         // 4. 원작 연결: originalWorkId가 있으면 서비스 계층을 통해 배정 | ||||||
|  |         if (request.originalWorkId != null) { | ||||||
|  |             originalWorkService.assignOneCharacter(request.originalWorkId, chatCharacter.id!!) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         ApiResponse.ok(null) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun callExternalApi(request: ChatCharacterRegisterRequest): String { | ||||||
|  |         try { | ||||||
|  |             val factory = SimpleClientHttpRequestFactory() | ||||||
|  |             factory.setConnectTimeout(20000) // 20초 | ||||||
|  |             factory.setReadTimeout(20000) // 20초 | ||||||
|  |  | ||||||
|  |             val restTemplate = RestTemplate(factory) | ||||||
|  |  | ||||||
|  |             val headers = HttpHeaders() | ||||||
|  |             headers.set("x-api-key", apiKey) // 실제 API 키로 대체 필요 | ||||||
|  |             headers.contentType = MediaType.APPLICATION_JSON | ||||||
|  |  | ||||||
|  |             // 외부 API에 전달하지 않을 필드(originalTitle, originalLink, characterType)를 제외하고 바디 구성 | ||||||
|  |             val body = mutableMapOf<String, Any>() | ||||||
|  |             body["name"] = request.name | ||||||
|  |             body["systemPrompt"] = request.systemPrompt | ||||||
|  |             body["description"] = request.description | ||||||
|  |             request.age?.let { body["age"] = it } | ||||||
|  |             request.gender?.let { body["gender"] = it } | ||||||
|  |             request.mbti?.let { body["mbti"] = it } | ||||||
|  |             request.speechPattern?.let { body["speechPattern"] = it } | ||||||
|  |             request.speechStyle?.let { body["speechStyle"] = it } | ||||||
|  |             request.appearance?.let { body["appearance"] = it } | ||||||
|  |             if (request.tags.isNotEmpty()) body["tags"] = request.tags | ||||||
|  |             if (request.hobbies.isNotEmpty()) body["hobbies"] = request.hobbies | ||||||
|  |             if (request.values.isNotEmpty()) body["values"] = request.values | ||||||
|  |             if (request.goals.isNotEmpty()) body["goals"] = request.goals | ||||||
|  |             if (request.relationships.isNotEmpty()) body["relationships"] = request.relationships | ||||||
|  |             if (request.personalities.isNotEmpty()) body["personalities"] = request.personalities | ||||||
|  |             if (request.backgrounds.isNotEmpty()) body["backgrounds"] = request.backgrounds | ||||||
|  |             if (request.memories.isNotEmpty()) body["memories"] = request.memories | ||||||
|  |  | ||||||
|  |             val httpEntity = HttpEntity(body, headers) | ||||||
|  |  | ||||||
|  |             val response = restTemplate.exchange( | ||||||
|  |                 "$apiUrl/api/characters", | ||||||
|  |                 HttpMethod.POST, | ||||||
|  |                 httpEntity, | ||||||
|  |                 String::class.java | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |             // 응답 파싱 | ||||||
|  |             val objectMapper = ObjectMapper() | ||||||
|  |             val apiResponse = objectMapper.readValue(response.body, ExternalApiResponse::class.java) | ||||||
|  |  | ||||||
|  |             // success가 false이면 throw | ||||||
|  |             if (!apiResponse.success) { | ||||||
|  |                 throw SodaException(apiResponse.message ?: "등록에 실패했습니다. 다시 시도해 주세요.") | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // success가 true이면 data.id 반환 | ||||||
|  |             return apiResponse.data?.id ?: throw SodaException("등록에 실패했습니다. 응답에 ID가 없습니다.") | ||||||
|  |         } catch (e: Exception) { | ||||||
|  |             e.printStackTrace() | ||||||
|  |             throw SodaException("${e.message}, 등록에 실패했습니다. 다시 시도해 주세요.") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun saveImage(characterId: Long, image: MultipartFile): String { | ||||||
|  |         try { | ||||||
|  |             val metadata = ObjectMetadata() | ||||||
|  |             metadata.contentLength = image.size | ||||||
|  |  | ||||||
|  |             // S3에 이미지 업로드 | ||||||
|  |             return s3Uploader.upload( | ||||||
|  |                 inputStream = image.inputStream, | ||||||
|  |                 bucket = s3Bucket, | ||||||
|  |                 filePath = "characters/$characterId/${generateFileName(prefix = "character")}", | ||||||
|  |                 metadata = metadata | ||||||
|  |             ) | ||||||
|  |         } catch (e: Exception) { | ||||||
|  |             throw SodaException("이미지 저장에 실패했습니다: ${e.message}") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 캐릭터 수정 API | ||||||
|  |      * 1. JSON 문자열을 ChatCharacterUpdateRequest 객체로 변환 | ||||||
|  |      * 2. ChatCharacterUpdateRequest를 확인해서 변경한 데이터가 있는지 확인 | ||||||
|  |      * 3. 이미지 있는지 확인 | ||||||
|  |      * 4. 2, 3번 중 하나라도 해당 하면 계속 진행 | ||||||
|  |      * 5. 2, 3번에 데이터 없으면 throw SodaException('변경된 데이터가 없습니다.') | ||||||
|  |      * | ||||||
|  |      * @param image 캐릭터 이미지 (선택적) | ||||||
|  |      * @param requestString ChatCharacterUpdateRequest 객체를 JSON 문자열로 변환한 값 | ||||||
|  |      * @return ApiResponse 객체 | ||||||
|  |      * @throws SodaException 변경된 데이터가 없거나 캐릭터를 찾을 수 없는 경우 | ||||||
|  |      */ | ||||||
|  |     @PutMapping("/update") | ||||||
|  |     fun updateCharacter( | ||||||
|  |         @RequestPart(value = "image", required = false) image: MultipartFile?, | ||||||
|  |         @RequestPart("request") requestString: String | ||||||
|  |     ) = run { | ||||||
|  |         // 1. JSON 문자열을 ChatCharacterUpdateRequest 객체로 변환 | ||||||
|  |         val objectMapper = ObjectMapper() | ||||||
|  |         val request = objectMapper.readValue(requestString, ChatCharacterUpdateRequest::class.java) | ||||||
|  |  | ||||||
|  |         // 2. ChatCharacterUpdateRequest를 확인해서 변경한 데이터가 있는지 확인 | ||||||
|  |         val hasChangedData = hasChanges(request) // 외부 API 대상으로의 변경 여부(3가지 필드 제외) | ||||||
|  |  | ||||||
|  |         // 3. 이미지 있는지 확인 | ||||||
|  |         val hasImage = image != null && !image.isEmpty | ||||||
|  |  | ||||||
|  |         // 3가지만 변경된 경우(외부 API 변경은 없지만 DB 변경은 있는 경우)를 허용하기 위해 별도 플래그 계산 | ||||||
|  |         val hasDbOnlyChanges = | ||||||
|  |             request.originalTitle != null || | ||||||
|  |                 request.originalLink != null || | ||||||
|  |                 request.characterType != null || | ||||||
|  |                 request.originalWorkId != null | ||||||
|  |  | ||||||
|  |         if (!hasChangedData && !hasImage && !hasDbOnlyChanges) { | ||||||
|  |             throw SodaException("변경된 데이터가 없습니다.") | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // 외부 API로 전달할 변경이 있을 때만 외부 API 호출(3가지 필드만 변경된 경우는 호출하지 않음) | ||||||
|  |         if (hasChangedData) { | ||||||
|  |             val chatCharacter = service.findById(request.id) | ||||||
|  |                 ?: throw SodaException("해당 ID의 캐릭터를 찾을 수 없습니다: ${request.id}") | ||||||
|  |  | ||||||
|  |             // 이름이 수정된 경우 DB에 동일한 이름이 있는지 확인 | ||||||
|  |             if (request.name != null && request.name != chatCharacter.name) { | ||||||
|  |                 val existingCharacter = service.findByName(request.name) | ||||||
|  |                 if (existingCharacter != null) { | ||||||
|  |                     throw SodaException("동일한 이름은 등록이 불가능합니다: ${request.name}") | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             callExternalApiForUpdate(chatCharacter.characterUUID, request) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // 이미지 경로 변수 초기화 | ||||||
|  |         // 이미지가 있으면 이미지 저장 | ||||||
|  |         val imagePath = if (hasImage) { | ||||||
|  |             saveImage( | ||||||
|  |                 characterId = request.id, | ||||||
|  |                 image = image!! | ||||||
|  |             ) | ||||||
|  |         } else { | ||||||
|  |             null | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // 엔티티 수정 | ||||||
|  |         service.updateChatCharacterWithDetails( | ||||||
|  |             imagePath = imagePath, | ||||||
|  |             request = request | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         // 원작 연결: originalWorkId가 있으면 서비스 계층을 통해 배정 | ||||||
|  |         if (request.originalWorkId != null) { | ||||||
|  |             // 서비스에서 유효성 검증 및 저장까지 처리 | ||||||
|  |             originalWorkService.assignOneCharacter(request.originalWorkId, request.id) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         ApiResponse.ok(null) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 요청에 변경된 데이터가 있는지 확인 | ||||||
|  |      * id를 제외한 모든 필드가 null이면 변경된 데이터가 없는 것으로 판단 | ||||||
|  |      * | ||||||
|  |      * @param request 수정 요청 데이터 | ||||||
|  |      * @return 변경된 데이터가 있으면 true, 없으면 false | ||||||
|  |      */ | ||||||
|  |     private fun hasChanges(request: ChatCharacterUpdateRequest): Boolean { | ||||||
|  |         return request.systemPrompt != null || | ||||||
|  |             request.description != null || | ||||||
|  |             request.age != null || | ||||||
|  |             request.gender != null || | ||||||
|  |             request.mbti != null || | ||||||
|  |             request.speechPattern != null || | ||||||
|  |             request.speechStyle != null || | ||||||
|  |             request.appearance != null || | ||||||
|  |             request.isActive != null || | ||||||
|  |             request.tags != null || | ||||||
|  |             request.hobbies != null || | ||||||
|  |             request.values != null || | ||||||
|  |             request.goals != null || | ||||||
|  |             request.relationships != null || | ||||||
|  |             request.personalities != null || | ||||||
|  |             request.backgrounds != null || | ||||||
|  |             request.memories != null || | ||||||
|  |             request.name != null | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 외부 API 호출 - 수정 API | ||||||
|  |      * 변경된 데이터만 요청에 포함 | ||||||
|  |      * | ||||||
|  |      * @param characterUUID 캐릭터 UUID | ||||||
|  |      * @param request 수정 요청 데이터 | ||||||
|  |      */ | ||||||
|  |     private fun callExternalApiForUpdate(characterUUID: String, request: ChatCharacterUpdateRequest) { | ||||||
|  |         try { | ||||||
|  |             val factory = SimpleClientHttpRequestFactory() | ||||||
|  |             factory.setConnectTimeout(20000) // 20초 | ||||||
|  |             factory.setReadTimeout(20000) // 20초 | ||||||
|  |  | ||||||
|  |             val restTemplate = RestTemplate(factory) | ||||||
|  |  | ||||||
|  |             val headers = HttpHeaders() | ||||||
|  |             headers.set("x-api-key", apiKey) | ||||||
|  |             headers.contentType = MediaType.APPLICATION_JSON | ||||||
|  |  | ||||||
|  |             // 변경된 데이터만 포함하는 맵 생성 | ||||||
|  |             val updateData = mutableMapOf<String, Any>() | ||||||
|  |  | ||||||
|  |             // isActive = false인 경우 처리 | ||||||
|  |             if (request.isActive != null && !request.isActive) { | ||||||
|  |                 val inactiveName = "inactive_${request.name}" | ||||||
|  |                 val randomSuffix = "_" + java.util.UUID.randomUUID().toString().replace("-", "") | ||||||
|  |                 updateData["name"] = inactiveName + randomSuffix | ||||||
|  |             } else { | ||||||
|  |                 request.name?.let { updateData["name"] = it } | ||||||
|  |                 request.systemPrompt?.let { updateData["systemPrompt"] = it } | ||||||
|  |                 request.description?.let { updateData["description"] = it } | ||||||
|  |                 request.age?.let { updateData["age"] = it } | ||||||
|  |                 request.gender?.let { updateData["gender"] = it } | ||||||
|  |                 request.mbti?.let { updateData["mbti"] = it } | ||||||
|  |                 request.speechPattern?.let { updateData["speechPattern"] = it } | ||||||
|  |                 request.speechStyle?.let { updateData["speechStyle"] = it } | ||||||
|  |                 request.appearance?.let { updateData["appearance"] = it } | ||||||
|  |                 request.tags?.let { updateData["tags"] = it } | ||||||
|  |                 request.hobbies?.let { updateData["hobbies"] = it } | ||||||
|  |                 request.values?.let { updateData["values"] = it } | ||||||
|  |                 request.goals?.let { updateData["goals"] = it } | ||||||
|  |                 request.relationships?.let { updateData["relationships"] = it } | ||||||
|  |                 request.personalities?.let { updateData["personalities"] = it } | ||||||
|  |                 request.backgrounds?.let { updateData["backgrounds"] = it } | ||||||
|  |                 request.memories?.let { updateData["memories"] = it } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             val httpEntity = HttpEntity(updateData, headers) | ||||||
|  |             val response = restTemplate.exchange( | ||||||
|  |                 "$apiUrl/api/characters/$characterUUID", | ||||||
|  |                 HttpMethod.PUT, | ||||||
|  |                 httpEntity, | ||||||
|  |                 String::class.java | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |             // 응답 파싱 | ||||||
|  |             val objectMapper = ObjectMapper() | ||||||
|  |             val apiResponse = objectMapper.readValue(response.body, ExternalApiResponse::class.java) | ||||||
|  |  | ||||||
|  |             // success가 false이면 throw | ||||||
|  |             if (!apiResponse.success) { | ||||||
|  |                 throw SodaException(apiResponse.message ?: "수정에 실패했습니다. 다시 시도해 주세요.") | ||||||
|  |             } | ||||||
|  |         } catch (e: Exception) { | ||||||
|  |             e.printStackTrace() | ||||||
|  |             throw SodaException("${e.message} 수정에 실패했습니다. 다시 시도해 주세요.") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,82 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.character.curation | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.common.ApiResponse | ||||||
|  | import kr.co.vividnext.sodalive.common.SodaException | ||||||
|  | import org.springframework.beans.factory.annotation.Value | ||||||
|  | import org.springframework.security.access.prepost.PreAuthorize | ||||||
|  | import org.springframework.web.bind.annotation.DeleteMapping | ||||||
|  | import org.springframework.web.bind.annotation.GetMapping | ||||||
|  | import org.springframework.web.bind.annotation.PathVariable | ||||||
|  | import org.springframework.web.bind.annotation.PostMapping | ||||||
|  | import org.springframework.web.bind.annotation.PutMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestBody | ||||||
|  | import org.springframework.web.bind.annotation.RequestMapping | ||||||
|  | import org.springframework.web.bind.annotation.RestController | ||||||
|  |  | ||||||
|  | @RestController | ||||||
|  | @RequestMapping("/admin/chat/character/curation") | ||||||
|  | @PreAuthorize("hasRole('ADMIN')") | ||||||
|  | class CharacterCurationAdminController( | ||||||
|  |     private val service: CharacterCurationAdminService, | ||||||
|  |     @Value("\${cloud.aws.cloud-front.host}") | ||||||
|  |     private val imageHost: String | ||||||
|  | ) { | ||||||
|  |     @GetMapping("/list") | ||||||
|  |     fun listAll(): ApiResponse<List<CharacterCurationListItemResponse>> = | ||||||
|  |         ApiResponse.ok(service.listAll()) | ||||||
|  |  | ||||||
|  |     @GetMapping("/{curationId}/characters") | ||||||
|  |     fun listCharacters( | ||||||
|  |         @PathVariable curationId: Long | ||||||
|  |     ): ApiResponse<List<CharacterCurationCharacterItemResponse>> { | ||||||
|  |         val characters = service.listCharacters(curationId) | ||||||
|  |         val items = characters.map { | ||||||
|  |             CharacterCurationCharacterItemResponse( | ||||||
|  |                 id = it.id!!, | ||||||
|  |                 name = it.name, | ||||||
|  |                 description = it.description, | ||||||
|  |                 imageUrl = "$imageHost/${it.imagePath ?: "profile/default-profile.png"}" | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |         return ApiResponse.ok(items) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @PostMapping("/register") | ||||||
|  |     fun register(@RequestBody request: CharacterCurationRegisterRequest) = | ||||||
|  |         ApiResponse.ok(service.register(request).id) | ||||||
|  |  | ||||||
|  |     @PutMapping("/update") | ||||||
|  |     fun update(@RequestBody request: CharacterCurationUpdateRequest) = | ||||||
|  |         ApiResponse.ok(service.update(request).id) | ||||||
|  |  | ||||||
|  |     @DeleteMapping("/{curationId}") | ||||||
|  |     fun delete(@PathVariable curationId: Long) = | ||||||
|  |         ApiResponse.ok(service.softDelete(curationId)) | ||||||
|  |  | ||||||
|  |     @PutMapping("/reorder") | ||||||
|  |     fun reorder(@RequestBody request: CharacterCurationOrderUpdateRequest) = | ||||||
|  |         ApiResponse.ok(service.reorder(request.ids)) | ||||||
|  |  | ||||||
|  |     @PostMapping("/{curationId}/characters") | ||||||
|  |     fun addCharacter( | ||||||
|  |         @PathVariable curationId: Long, | ||||||
|  |         @RequestBody request: CharacterCurationAddCharacterRequest | ||||||
|  |     ): ApiResponse<Boolean> { | ||||||
|  |         val ids = request.characterIds.filter { it > 0 }.distinct() | ||||||
|  |         if (ids.isEmpty()) throw SodaException("등록할 캐릭터 ID 리스트가 비어있습니다") | ||||||
|  |         service.addCharacters(curationId, ids) | ||||||
|  |         return ApiResponse.ok(true) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @DeleteMapping("/{curationId}/characters/{characterId}") | ||||||
|  |     fun removeCharacter( | ||||||
|  |         @PathVariable curationId: Long, | ||||||
|  |         @PathVariable characterId: Long | ||||||
|  |     ) = ApiResponse.ok(service.removeCharacter(curationId, characterId)) | ||||||
|  |  | ||||||
|  |     @PutMapping("/{curationId}/characters/reorder") | ||||||
|  |     fun reorderCharacters( | ||||||
|  |         @PathVariable curationId: Long, | ||||||
|  |         @RequestBody request: CharacterCurationReorderCharactersRequest | ||||||
|  |     ) = ApiResponse.ok(service.reorderCharacters(curationId, request.characterIds)) | ||||||
|  | } | ||||||
| @@ -0,0 +1,45 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.character.curation | ||||||
|  |  | ||||||
|  | data class CharacterCurationRegisterRequest( | ||||||
|  |     val title: String, | ||||||
|  |     val isAdult: Boolean = false, | ||||||
|  |     val isActive: Boolean = true | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class CharacterCurationUpdateRequest( | ||||||
|  |     val id: Long, | ||||||
|  |     val title: String? = null, | ||||||
|  |     val isAdult: Boolean? = null, | ||||||
|  |     val isActive: Boolean? = null | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class CharacterCurationOrderUpdateRequest( | ||||||
|  |     val ids: List<Long> | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class CharacterCurationAddCharacterRequest( | ||||||
|  |     val characterIds: List<Long> | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class CharacterCurationReorderCharactersRequest( | ||||||
|  |     val characterIds: List<Long> | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class CharacterCurationListItemResponse( | ||||||
|  |     val id: Long, | ||||||
|  |     val title: String, | ||||||
|  |     val isAdult: Boolean, | ||||||
|  |     val isActive: Boolean, | ||||||
|  |     val characterCount: Int | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // 관리자 큐레이션 상세 - 캐릭터 리스트 항목 응답 DTO | ||||||
|  | // id, name, description, 이미지 URL | ||||||
|  | // 이미지 URL은 컨트롤러에서 cloud-front host + imagePath로 구성 | ||||||
|  |  | ||||||
|  | data class CharacterCurationCharacterItemResponse( | ||||||
|  |     val id: Long, | ||||||
|  |     val name: String, | ||||||
|  |     val description: String, | ||||||
|  |     val imageUrl: String | ||||||
|  | ) | ||||||
| @@ -0,0 +1,153 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.character.curation | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.ChatCharacter | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.curation.CharacterCuration | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.curation.CharacterCurationMapping | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.curation.repository.CharacterCurationMappingRepository | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.curation.repository.CharacterCurationRepository | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.repository.ChatCharacterRepository | ||||||
|  | import kr.co.vividnext.sodalive.common.SodaException | ||||||
|  | import org.springframework.stereotype.Service | ||||||
|  | import org.springframework.transaction.annotation.Transactional | ||||||
|  |  | ||||||
|  | @Service | ||||||
|  | class CharacterCurationAdminService( | ||||||
|  |     private val curationRepository: CharacterCurationRepository, | ||||||
|  |     private val mappingRepository: CharacterCurationMappingRepository, | ||||||
|  |     private val characterRepository: ChatCharacterRepository | ||||||
|  | ) { | ||||||
|  |  | ||||||
|  |     @Transactional | ||||||
|  |     fun register(request: CharacterCurationRegisterRequest): CharacterCuration { | ||||||
|  |         val sortOrder = (curationRepository.findMaxSortOrder() ?: 0) + 1 | ||||||
|  |         val curation = CharacterCuration( | ||||||
|  |             title = request.title, | ||||||
|  |             isAdult = request.isAdult, | ||||||
|  |             isActive = request.isActive, | ||||||
|  |             sortOrder = sortOrder | ||||||
|  |         ) | ||||||
|  |         return curationRepository.save(curation) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional | ||||||
|  |     fun update(request: CharacterCurationUpdateRequest): CharacterCuration { | ||||||
|  |         val curation = curationRepository.findById(request.id) | ||||||
|  |             .orElseThrow { SodaException("큐레이션을 찾을 수 없습니다: ${request.id}") } | ||||||
|  |  | ||||||
|  |         request.title?.let { curation.title = it } | ||||||
|  |         request.isAdult?.let { curation.isAdult = it } | ||||||
|  |         request.isActive?.let { curation.isActive = it } | ||||||
|  |  | ||||||
|  |         return curationRepository.save(curation) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional | ||||||
|  |     fun softDelete(curationId: Long) { | ||||||
|  |         val curation = curationRepository.findById(curationId) | ||||||
|  |             .orElseThrow { SodaException("큐레이션을 찾을 수 없습니다: $curationId") } | ||||||
|  |         curation.isActive = false | ||||||
|  |         curationRepository.save(curation) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional | ||||||
|  |     fun reorder(ids: List<Long>) { | ||||||
|  |         ids.forEachIndexed { index, id -> | ||||||
|  |             val curation = curationRepository.findById(id) | ||||||
|  |                 .orElseThrow { SodaException("큐레이션을 찾을 수 없습니다: $id") } | ||||||
|  |             curation.sortOrder = index + 1 | ||||||
|  |             curationRepository.save(curation) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional | ||||||
|  |     fun addCharacters(curationId: Long, characterIds: List<Long>) { | ||||||
|  |         if (characterIds.isEmpty()) throw SodaException("등록할 캐릭터 ID 리스트가 비어있습니다") | ||||||
|  |  | ||||||
|  |         val curation = curationRepository.findById(curationId) | ||||||
|  |             .orElseThrow { SodaException("큐레이션을 찾을 수 없습니다: $curationId") } | ||||||
|  |         if (!curation.isActive) throw SodaException("비활성화된 큐레이션입니다: $curationId") | ||||||
|  |  | ||||||
|  |         val uniqueIds = characterIds.filter { it > 0 }.distinct() | ||||||
|  |         if (uniqueIds.isEmpty()) throw SodaException("유효한 캐릭터 ID가 없습니다") | ||||||
|  |  | ||||||
|  |         // 활성 캐릭터만 조회 (조회 단계에서 검증 포함) | ||||||
|  |         val characters = characterRepository.findByIdInAndIsActiveTrue(uniqueIds) | ||||||
|  |         val characterMap = characters.associateBy { it.id!! } | ||||||
|  |  | ||||||
|  |         // 조회 결과에 존재하는 캐릭터만 유효 | ||||||
|  |         val validIds = uniqueIds.filter { id -> characterMap.containsKey(id) } | ||||||
|  |  | ||||||
|  |         val existingMappings = mappingRepository.findByCuration(curation) | ||||||
|  |         val existingCharacterIds = existingMappings.mapNotNull { it.chatCharacter.id }.toSet() | ||||||
|  |         var nextOrder = (existingMappings.maxOfOrNull { it.sortOrder } ?: 0) + 1 | ||||||
|  |  | ||||||
|  |         val toSave = mutableListOf<CharacterCurationMapping>() | ||||||
|  |         validIds.forEach { id -> | ||||||
|  |             if (!existingCharacterIds.contains(id)) { | ||||||
|  |                 val character = characterMap[id] ?: return@forEach | ||||||
|  |                 toSave += CharacterCurationMapping( | ||||||
|  |                     curation = curation, | ||||||
|  |                     chatCharacter = character, | ||||||
|  |                     sortOrder = nextOrder++ | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (toSave.isNotEmpty()) { | ||||||
|  |             mappingRepository.saveAll(toSave) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional | ||||||
|  |     fun removeCharacter(curationId: Long, characterId: Long) { | ||||||
|  |         val curation = curationRepository.findById(curationId) | ||||||
|  |             .orElseThrow { SodaException("큐레이션을 찾을 수 없습니다: $curationId") } | ||||||
|  |         val mappings = mappingRepository.findByCuration(curation) | ||||||
|  |         val target = mappings.firstOrNull { it.chatCharacter.id == characterId } | ||||||
|  |             ?: throw SodaException("매핑을 찾을 수 없습니다: curation=$curationId, character=$characterId") | ||||||
|  |         mappingRepository.delete(target) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional | ||||||
|  |     fun reorderCharacters(curationId: Long, characterIds: List<Long>) { | ||||||
|  |         val curation = curationRepository.findById(curationId) | ||||||
|  |             .orElseThrow { SodaException("큐레이션을 찾을 수 없습니다: $curationId") } | ||||||
|  |         val mappings = mappingRepository.findByCuration(curation) | ||||||
|  |         val mappingByCharacterId = mappings.associateBy { it.chatCharacter.id } | ||||||
|  |  | ||||||
|  |         characterIds.forEachIndexed { index, cid -> | ||||||
|  |             val mapping = mappingByCharacterId[cid] | ||||||
|  |                 ?: throw SodaException("큐레이션에 포함되지 않은 캐릭터입니다: $cid") | ||||||
|  |             mapping.sortOrder = index + 1 | ||||||
|  |             mappingRepository.save(mapping) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional(readOnly = true) | ||||||
|  |     fun listAll(): List<CharacterCurationListItemResponse> { | ||||||
|  |         val curations = curationRepository.findByIsActiveTrueOrderBySortOrderAsc() | ||||||
|  |         if (curations.isEmpty()) return emptyList() | ||||||
|  |  | ||||||
|  |         // DB 집계로 활성 캐릭터 수 카운트 | ||||||
|  |         val counts = mappingRepository.countActiveCharactersByCurations(curations) | ||||||
|  |         val countByCurationId: Map<Long, Int> = counts.associate { it.curationId to it.count.toInt() } | ||||||
|  |  | ||||||
|  |         return curations.map { curation -> | ||||||
|  |             CharacterCurationListItemResponse( | ||||||
|  |                 id = curation.id!!, | ||||||
|  |                 title = curation.title, | ||||||
|  |                 isAdult = curation.isAdult, | ||||||
|  |                 isActive = curation.isActive, | ||||||
|  |                 characterCount = countByCurationId[curation.id!!] ?: 0 | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional(readOnly = true) | ||||||
|  |     fun listCharacters(curationId: Long): List<ChatCharacter> { | ||||||
|  |         val curation = curationRepository.findById(curationId) | ||||||
|  |             .orElseThrow { SodaException("큐레이션을 찾을 수 없습니다: $curationId") } | ||||||
|  |         val mappings = mappingRepository.findByCurationWithCharacterOrderBySortOrderAsc(curation) | ||||||
|  |         return mappings.map { it.chatCharacter } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,132 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.character.dto | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.ChatCharacter | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 관리자 캐릭터 상세 응답 DTO | ||||||
|  |  * - 원작이 연결되어 있으면 원작 요약 정보(originalWork)를 함께 반환한다. | ||||||
|  |  */ | ||||||
|  | data class ChatCharacterDetailResponse( | ||||||
|  |     val id: Long, | ||||||
|  |     val characterUUID: String, | ||||||
|  |     val name: String, | ||||||
|  |     val imageUrl: String?, | ||||||
|  |     val description: String, | ||||||
|  |     val systemPrompt: String, | ||||||
|  |     val characterType: String, | ||||||
|  |     val age: Int?, | ||||||
|  |     val gender: String?, | ||||||
|  |     val mbti: String?, | ||||||
|  |     val speechPattern: String?, | ||||||
|  |     val speechStyle: String?, | ||||||
|  |     val appearance: String?, | ||||||
|  |     val isActive: Boolean, | ||||||
|  |     val tags: List<String>, | ||||||
|  |     val hobbies: List<String>, | ||||||
|  |     val values: List<String>, | ||||||
|  |     val goals: List<String>, | ||||||
|  |     val relationships: List<RelationshipResponse>, | ||||||
|  |     val personalities: List<PersonalityResponse>, | ||||||
|  |     val backgrounds: List<BackgroundResponse>, | ||||||
|  |     val memories: List<MemoryResponse>, | ||||||
|  |     val originalWork: OriginalWorkBriefResponse? // 추가: 원작 요약 정보 | ||||||
|  | ) { | ||||||
|  |     companion object { | ||||||
|  |         fun from(chatCharacter: ChatCharacter, imageHost: String = ""): ChatCharacterDetailResponse { | ||||||
|  |             val fullImagePath = if (chatCharacter.imagePath != null && imageHost.isNotEmpty()) { | ||||||
|  |                 "$imageHost/${chatCharacter.imagePath}" | ||||||
|  |             } else { | ||||||
|  |                 chatCharacter.imagePath ?: "" | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             val ow = chatCharacter.originalWork | ||||||
|  |             val originalWorkBrief = ow?.let { | ||||||
|  |                 val owImage = if (it.imagePath != null && imageHost.isNotEmpty()) { | ||||||
|  |                     "$imageHost/${it.imagePath}" | ||||||
|  |                 } else { | ||||||
|  |                     it.imagePath | ||||||
|  |                 } | ||||||
|  |                 OriginalWorkBriefResponse( | ||||||
|  |                     id = it.id!!, | ||||||
|  |                     imageUrl = owImage, | ||||||
|  |                     title = it.title | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             return ChatCharacterDetailResponse( | ||||||
|  |                 id = chatCharacter.id!!, | ||||||
|  |                 characterUUID = chatCharacter.characterUUID, | ||||||
|  |                 name = chatCharacter.name, | ||||||
|  |                 imageUrl = fullImagePath, | ||||||
|  |                 description = chatCharacter.description, | ||||||
|  |                 systemPrompt = chatCharacter.systemPrompt, | ||||||
|  |                 characterType = chatCharacter.characterType.name, | ||||||
|  |                 age = chatCharacter.age, | ||||||
|  |                 gender = chatCharacter.gender, | ||||||
|  |                 mbti = chatCharacter.mbti, | ||||||
|  |                 speechPattern = chatCharacter.speechPattern, | ||||||
|  |                 speechStyle = chatCharacter.speechStyle, | ||||||
|  |                 appearance = chatCharacter.appearance, | ||||||
|  |                 isActive = chatCharacter.isActive, | ||||||
|  |                 tags = chatCharacter.tagMappings.map { it.tag.tag }, | ||||||
|  |                 hobbies = chatCharacter.hobbyMappings.map { it.hobby.hobby }, | ||||||
|  |                 values = chatCharacter.valueMappings.map { it.value.value }, | ||||||
|  |                 goals = chatCharacter.goalMappings.map { it.goal.goal }, | ||||||
|  |                 relationships = chatCharacter.relationships.map { | ||||||
|  |                     RelationshipResponse( | ||||||
|  |                         personName = it.personName, | ||||||
|  |                         relationshipName = it.relationshipName, | ||||||
|  |                         description = it.description, | ||||||
|  |                         importance = it.importance, | ||||||
|  |                         relationshipType = it.relationshipType, | ||||||
|  |                         currentStatus = it.currentStatus | ||||||
|  |                     ) | ||||||
|  |                 }, | ||||||
|  |                 personalities = chatCharacter.personalities.map { | ||||||
|  |                     PersonalityResponse(it.trait, it.description) | ||||||
|  |                 }, | ||||||
|  |                 backgrounds = chatCharacter.backgrounds.map { | ||||||
|  |                     BackgroundResponse(it.topic, it.description) | ||||||
|  |                 }, | ||||||
|  |                 memories = chatCharacter.memories.map { | ||||||
|  |                     MemoryResponse(it.title, it.content, it.emotion) | ||||||
|  |                 }, | ||||||
|  |                 originalWork = originalWorkBrief | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | data class PersonalityResponse( | ||||||
|  |     val trait: String, | ||||||
|  |     val description: String | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class BackgroundResponse( | ||||||
|  |     val topic: String, | ||||||
|  |     val description: String | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class MemoryResponse( | ||||||
|  |     val title: String, | ||||||
|  |     val content: String, | ||||||
|  |     val emotion: String | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class RelationshipResponse( | ||||||
|  |     val personName: String, | ||||||
|  |     val relationshipName: String, | ||||||
|  |     val description: String, | ||||||
|  |     val importance: Int, | ||||||
|  |     val relationshipType: String, | ||||||
|  |     val currentStatus: String | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 원작 요약 응답 DTO(관리자 캐릭터 상세용) | ||||||
|  |  */ | ||||||
|  | data class OriginalWorkBriefResponse( | ||||||
|  |     val id: Long, | ||||||
|  |     val imageUrl: String?, | ||||||
|  |     val title: String | ||||||
|  | ) | ||||||
| @@ -0,0 +1,90 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.character.dto | ||||||
|  |  | ||||||
|  | import com.fasterxml.jackson.annotation.JsonIgnoreProperties | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  |  | ||||||
|  | data class ChatCharacterPersonalityRequest( | ||||||
|  |     @JsonProperty("trait") val trait: String, | ||||||
|  |     @JsonProperty("description") val description: String | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class ChatCharacterBackgroundRequest( | ||||||
|  |     @JsonProperty("topic") val topic: String, | ||||||
|  |     @JsonProperty("description") val description: String | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class ChatCharacterMemoryRequest( | ||||||
|  |     @JsonProperty("title") val title: String, | ||||||
|  |     @JsonProperty("content") val content: String, | ||||||
|  |     @JsonProperty("emotion") val emotion: String | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class ChatCharacterRelationshipRequest( | ||||||
|  |     @JsonProperty("personName") val personName: String, | ||||||
|  |     @JsonProperty("relationshipName") val relationshipName: String, | ||||||
|  |     @JsonProperty("description") val description: String, | ||||||
|  |     @JsonProperty("importance") val importance: Int, | ||||||
|  |     @JsonProperty("relationshipType") val relationshipType: String, | ||||||
|  |     @JsonProperty("currentStatus") val currentStatus: String | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class ChatCharacterRegisterRequest( | ||||||
|  |     @JsonProperty("name") val name: String, | ||||||
|  |     @JsonProperty("systemPrompt") val systemPrompt: String, | ||||||
|  |     @JsonProperty("description") val description: String, | ||||||
|  |     @JsonProperty("age") val age: String?, | ||||||
|  |     @JsonProperty("gender") val gender: String?, | ||||||
|  |     @JsonProperty("mbti") val mbti: String?, | ||||||
|  |     @JsonProperty("speechPattern") val speechPattern: String?, | ||||||
|  |     @JsonProperty("speechStyle") val speechStyle: String?, | ||||||
|  |     @JsonProperty("appearance") val appearance: String?, | ||||||
|  |     @JsonProperty("originalTitle") val originalTitle: String? = null, | ||||||
|  |     @JsonProperty("originalLink") val originalLink: String? = null, | ||||||
|  |     @JsonProperty("originalWorkId") val originalWorkId: Long? = null, | ||||||
|  |     @JsonProperty("characterType") val characterType: String? = null, | ||||||
|  |     @JsonProperty("tags") val tags: List<String> = emptyList(), | ||||||
|  |     @JsonProperty("hobbies") val hobbies: List<String> = emptyList(), | ||||||
|  |     @JsonProperty("values") val values: List<String> = emptyList(), | ||||||
|  |     @JsonProperty("goals") val goals: List<String> = emptyList(), | ||||||
|  |     @JsonProperty("relationships") val relationships: List<ChatCharacterRelationshipRequest> = emptyList(), | ||||||
|  |     @JsonProperty("personalities") val personalities: List<ChatCharacterPersonalityRequest> = emptyList(), | ||||||
|  |     @JsonProperty("backgrounds") val backgrounds: List<ChatCharacterBackgroundRequest> = emptyList(), | ||||||
|  |     @JsonProperty("memories") val memories: List<ChatCharacterMemoryRequest> = emptyList() | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class ExternalApiResponse( | ||||||
|  |     @JsonProperty("success") val success: Boolean, | ||||||
|  |     @JsonProperty("data") val data: ExternalApiData? = null, | ||||||
|  |     @JsonProperty("message") val message: String? = null | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | @JsonIgnoreProperties(ignoreUnknown = true) | ||||||
|  | data class ExternalApiData( | ||||||
|  |     @JsonProperty("id") val id: String | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class ChatCharacterUpdateRequest( | ||||||
|  |     @JsonProperty("id") val id: Long, | ||||||
|  |     @JsonProperty("name") val name: String? = null, | ||||||
|  |     @JsonProperty("systemPrompt") val systemPrompt: String? = null, | ||||||
|  |     @JsonProperty("description") val description: String? = null, | ||||||
|  |     @JsonProperty("age") val age: String? = null, | ||||||
|  |     @JsonProperty("gender") val gender: String? = null, | ||||||
|  |     @JsonProperty("mbti") val mbti: String? = null, | ||||||
|  |     @JsonProperty("speechPattern") val speechPattern: String? = null, | ||||||
|  |     @JsonProperty("speechStyle") val speechStyle: String? = null, | ||||||
|  |     @JsonProperty("appearance") val appearance: String? = null, | ||||||
|  |     @JsonProperty("originalTitle") val originalTitle: String? = null, | ||||||
|  |     @JsonProperty("originalLink") val originalLink: String? = null, | ||||||
|  |     @JsonProperty("originalWorkId") val originalWorkId: Long? = null, | ||||||
|  |     @JsonProperty("characterType") val characterType: String? = null, | ||||||
|  |     @JsonProperty("isActive") val isActive: Boolean? = null, | ||||||
|  |     @JsonProperty("tags") val tags: List<String>? = null, | ||||||
|  |     @JsonProperty("hobbies") val hobbies: List<String>? = null, | ||||||
|  |     @JsonProperty("values") val values: List<String>? = null, | ||||||
|  |     @JsonProperty("goals") val goals: List<String>? = null, | ||||||
|  |     @JsonProperty("relationships") val relationships: List<ChatCharacterRelationshipRequest>? = null, | ||||||
|  |     @JsonProperty("personalities") val personalities: List<ChatCharacterPersonalityRequest>? = null, | ||||||
|  |     @JsonProperty("backgrounds") val backgrounds: List<ChatCharacterBackgroundRequest>? = null, | ||||||
|  |     @JsonProperty("memories") val memories: List<ChatCharacterMemoryRequest>? = null | ||||||
|  | ) | ||||||
| @@ -0,0 +1,62 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.character.dto | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.ChatCharacter | ||||||
|  | import java.time.ZoneId | ||||||
|  | import java.time.format.DateTimeFormatter | ||||||
|  |  | ||||||
|  | data class ChatCharacterListResponse( | ||||||
|  |     val id: Long, | ||||||
|  |     val name: String, | ||||||
|  |     val imageUrl: String?, | ||||||
|  |     val description: String, | ||||||
|  |     val gender: String?, | ||||||
|  |     val age: Int?, | ||||||
|  |     val mbti: String?, | ||||||
|  |     val speechStyle: String?, | ||||||
|  |     val speechPattern: String?, | ||||||
|  |     val tags: List<String>, | ||||||
|  |     val createdAt: String?, | ||||||
|  |     val updatedAt: String? | ||||||
|  | ) { | ||||||
|  |     companion object { | ||||||
|  |         private val formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss") | ||||||
|  |         private val seoulZoneId = ZoneId.of("Asia/Seoul") | ||||||
|  |  | ||||||
|  |         fun from(chatCharacter: ChatCharacter, imageHost: String = ""): ChatCharacterListResponse { | ||||||
|  |             val fullImagePath = if (chatCharacter.imagePath != null && imageHost.isNotEmpty()) { | ||||||
|  |                 "$imageHost/${chatCharacter.imagePath}" | ||||||
|  |             } else { | ||||||
|  |                 chatCharacter.imagePath | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // UTC에서 Asia/Seoul로 시간대 변환 및 문자열 포맷팅 | ||||||
|  |             val createdAtStr = chatCharacter.createdAt?.atZone(ZoneId.of("UTC")) | ||||||
|  |                 ?.withZoneSameInstant(seoulZoneId) | ||||||
|  |                 ?.format(formatter) | ||||||
|  |  | ||||||
|  |             val updatedAtStr = chatCharacter.updatedAt?.atZone(ZoneId.of("UTC")) | ||||||
|  |                 ?.withZoneSameInstant(seoulZoneId) | ||||||
|  |                 ?.format(formatter) | ||||||
|  |  | ||||||
|  |             return ChatCharacterListResponse( | ||||||
|  |                 id = chatCharacter.id!!, | ||||||
|  |                 name = chatCharacter.name, | ||||||
|  |                 imageUrl = fullImagePath, | ||||||
|  |                 description = chatCharacter.description, | ||||||
|  |                 gender = chatCharacter.gender, | ||||||
|  |                 age = chatCharacter.age, | ||||||
|  |                 mbti = chatCharacter.mbti, | ||||||
|  |                 speechStyle = chatCharacter.speechStyle, | ||||||
|  |                 speechPattern = chatCharacter.speechPattern, | ||||||
|  |                 tags = chatCharacter.tagMappings.map { it.tag.tag }, | ||||||
|  |                 createdAt = createdAtStr, | ||||||
|  |                 updatedAt = updatedAtStr | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | data class ChatCharacterListPageResponse( | ||||||
|  |     val totalCount: Long, | ||||||
|  |     val content: List<ChatCharacterListResponse> | ||||||
|  | ) | ||||||
| @@ -0,0 +1,9 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.character.dto | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 캐릭터 검색 결과 페이지 응답 DTO | ||||||
|  |  */ | ||||||
|  | data class ChatCharacterSearchListPageResponse( | ||||||
|  |     val totalCount: Long, | ||||||
|  |     val content: List<ChatCharacterListResponse> | ||||||
|  | ) | ||||||
| @@ -0,0 +1,30 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.character.dto | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.ChatCharacter | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 원작 연결된 캐릭터 결과 응답 DTO | ||||||
|  |  */ | ||||||
|  | data class OriginalWorkChatCharacterResponse( | ||||||
|  |     val id: Long, | ||||||
|  |     val name: String, | ||||||
|  |     val imagePath: String? | ||||||
|  | ) { | ||||||
|  |     companion object { | ||||||
|  |         fun from(character: ChatCharacter, imageHost: String): OriginalWorkChatCharacterResponse { | ||||||
|  |             return OriginalWorkChatCharacterResponse( | ||||||
|  |                 id = character.id!!, | ||||||
|  |                 name = character.name, | ||||||
|  |                 imagePath = character.imagePath?.let { "$imageHost/$it" } | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 원작 연결된 캐릭터 결과 페이지 응답 DTO | ||||||
|  |  */ | ||||||
|  | data class OriginalWorkChatCharacterListPageResponse( | ||||||
|  |     val totalCount: Long, | ||||||
|  |     val content: List<OriginalWorkChatCharacterResponse> | ||||||
|  | ) | ||||||
| @@ -0,0 +1,170 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.character.image | ||||||
|  |  | ||||||
|  | import com.amazonaws.services.s3.model.ObjectMetadata | ||||||
|  | import com.fasterxml.jackson.databind.ObjectMapper | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.character.image.dto.AdminCharacterImageResponse | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.character.image.dto.RegisterCharacterImageRequest | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.character.image.dto.UpdateCharacterImageOrdersRequest | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.character.image.dto.UpdateCharacterImageTriggersRequest | ||||||
|  | import kr.co.vividnext.sodalive.aws.cloudfront.ImageContentCloudFront | ||||||
|  | import kr.co.vividnext.sodalive.aws.s3.S3Uploader | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.image.CharacterImageService | ||||||
|  | import kr.co.vividnext.sodalive.common.ApiResponse | ||||||
|  | import kr.co.vividnext.sodalive.common.SodaException | ||||||
|  | import kr.co.vividnext.sodalive.utils.ImageBlurUtil | ||||||
|  | import kr.co.vividnext.sodalive.utils.generateFileName | ||||||
|  | import org.springframework.beans.factory.annotation.Value | ||||||
|  | import org.springframework.security.access.prepost.PreAuthorize | ||||||
|  | import org.springframework.web.bind.annotation.DeleteMapping | ||||||
|  | import org.springframework.web.bind.annotation.GetMapping | ||||||
|  | import org.springframework.web.bind.annotation.PathVariable | ||||||
|  | import org.springframework.web.bind.annotation.PostMapping | ||||||
|  | import org.springframework.web.bind.annotation.PutMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestBody | ||||||
|  | import org.springframework.web.bind.annotation.RequestMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestParam | ||||||
|  | import org.springframework.web.bind.annotation.RequestPart | ||||||
|  | import org.springframework.web.bind.annotation.RestController | ||||||
|  | import org.springframework.web.multipart.MultipartFile | ||||||
|  |  | ||||||
|  | @RestController | ||||||
|  | @RequestMapping("/admin/chat/character/image") | ||||||
|  | @PreAuthorize("hasRole('ADMIN')") | ||||||
|  | class AdminCharacterImageController( | ||||||
|  |     private val imageService: CharacterImageService, | ||||||
|  |     private val s3Uploader: S3Uploader, | ||||||
|  |     private val imageCloudFront: ImageContentCloudFront, | ||||||
|  |  | ||||||
|  |     @Value("\${cloud.aws.s3.content-bucket}") | ||||||
|  |     private val s3Bucket: String, | ||||||
|  |  | ||||||
|  |     @Value("\${cloud.aws.s3.bucket}") | ||||||
|  |     private val freeBucket: String | ||||||
|  | ) { | ||||||
|  |  | ||||||
|  |     @GetMapping("/list") | ||||||
|  |     fun list(@RequestParam characterId: Long) = run { | ||||||
|  |         val expiration = 5L * 60L * 1000L // 5분 | ||||||
|  |         val list = imageService.listActiveByCharacter(characterId) | ||||||
|  |             .map { img -> | ||||||
|  |                 val signedUrl = imageCloudFront.generateSignedURL(img.imagePath, expiration) | ||||||
|  |                 AdminCharacterImageResponse.fromWithUrl(img, signedUrl) | ||||||
|  |             } | ||||||
|  |         ApiResponse.ok(list) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @GetMapping("/{imageId}") | ||||||
|  |     fun detail(@PathVariable imageId: Long) = run { | ||||||
|  |         val img = imageService.getById(imageId) | ||||||
|  |         val expiration = 5L * 60L * 1000L // 5분 | ||||||
|  |         val signedUrl = imageCloudFront.generateSignedURL(img.imagePath, expiration) | ||||||
|  |         ApiResponse.ok(AdminCharacterImageResponse.fromWithUrl(img, signedUrl)) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @PostMapping("/register") | ||||||
|  |     fun register( | ||||||
|  |         @RequestPart("image") image: MultipartFile, | ||||||
|  |         @RequestPart("request") requestString: String | ||||||
|  |     ) = run { | ||||||
|  |         val objectMapper = ObjectMapper() | ||||||
|  |         val request = objectMapper.readValue(requestString, RegisterCharacterImageRequest::class.java) | ||||||
|  |  | ||||||
|  |         // 업로드 키 생성 | ||||||
|  |         val s3Key = buildS3Key(characterId = request.characterId) | ||||||
|  |  | ||||||
|  |         // 원본 저장 (content-bucket) | ||||||
|  |         val imagePath = saveImageToBucket(s3Key, image, s3Bucket) | ||||||
|  |  | ||||||
|  |         // 블러 생성 및 저장 (무료 이미지 버킷) | ||||||
|  |         val blurImagePath = saveBlurImageToBucket(s3Key, image, freeBucket) | ||||||
|  |  | ||||||
|  |         imageService.registerImage( | ||||||
|  |             characterId = request.characterId, | ||||||
|  |             imagePath = imagePath, | ||||||
|  |             blurImagePath = blurImagePath, | ||||||
|  |             imagePriceCan = request.imagePriceCan, | ||||||
|  |             messagePriceCan = request.messagePriceCan, | ||||||
|  |             isAdult = request.isAdult, | ||||||
|  |             triggers = request.triggers ?: emptyList() | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         ApiResponse.ok(null) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @PutMapping("/{imageId}/triggers") | ||||||
|  |     fun updateTriggers( | ||||||
|  |         @PathVariable imageId: Long, | ||||||
|  |         @RequestBody request: UpdateCharacterImageTriggersRequest | ||||||
|  |     ) = run { | ||||||
|  |         if (!request.triggers.isNullOrEmpty()) { | ||||||
|  |             imageService.updateTriggers(imageId, request.triggers) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         ApiResponse.ok(null) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @DeleteMapping("/{imageId}") | ||||||
|  |     fun delete(@PathVariable imageId: Long) = run { | ||||||
|  |         imageService.deleteImage(imageId) | ||||||
|  |         ApiResponse.ok(null, "이미지가 삭제되었습니다.") | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @PutMapping("/orders") | ||||||
|  |     fun updateOrders(@RequestBody request: UpdateCharacterImageOrdersRequest) = run { | ||||||
|  |         if (request.characterId == null) throw SodaException("characterId는 필수입니다") | ||||||
|  |         imageService.updateOrders(request.characterId, request.ids) | ||||||
|  |         ApiResponse.ok(null, "정렬 순서가 변경되었습니다.") | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun buildS3Key(characterId: Long): String { | ||||||
|  |         val fileName = generateFileName("character-image") | ||||||
|  |         return "characters/$characterId/images/$fileName" | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun saveImageToBucket(filePath: String, image: MultipartFile, bucket: String): String { | ||||||
|  |         try { | ||||||
|  |             val metadata = ObjectMetadata() | ||||||
|  |             metadata.contentLength = image.size | ||||||
|  |             return s3Uploader.upload( | ||||||
|  |                 inputStream = image.inputStream, | ||||||
|  |                 bucket = bucket, | ||||||
|  |                 filePath = filePath, | ||||||
|  |                 metadata = metadata | ||||||
|  |             ) | ||||||
|  |         } catch (e: Exception) { | ||||||
|  |             throw SodaException("이미지 저장에 실패했습니다: ${e.message}") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun saveBlurImageToBucket(filePath: String, image: MultipartFile, bucket: String): String { | ||||||
|  |         try { | ||||||
|  |             // 멀티파트를 BufferedImage로 읽기 | ||||||
|  |             val bytes = image.bytes | ||||||
|  |             val bimg = javax.imageio.ImageIO.read(java.io.ByteArrayInputStream(bytes)) | ||||||
|  |                 ?: throw SodaException("이미지 포맷을 인식할 수 없습니다.") | ||||||
|  |             val blurred = ImageBlurUtil.blurFast(bimg) | ||||||
|  |  | ||||||
|  |             // PNG로 저장(알파 유지), JPEG 업로드가 필요하면 포맷 변경 가능 | ||||||
|  |             val baos = java.io.ByteArrayOutputStream() | ||||||
|  |             val format = when (image.contentType?.lowercase()) { | ||||||
|  |                 "image/png" -> "png" | ||||||
|  |                 else -> "jpg" | ||||||
|  |             } | ||||||
|  |             javax.imageio.ImageIO.write(blurred, format, baos) | ||||||
|  |             val inputStream = java.io.ByteArrayInputStream(baos.toByteArray()) | ||||||
|  |  | ||||||
|  |             val metadata = ObjectMetadata() | ||||||
|  |             metadata.contentLength = baos.size().toLong() | ||||||
|  |             metadata.contentType = image.contentType ?: if (format == "png") "image/png" else "image/jpeg" | ||||||
|  |  | ||||||
|  |             return s3Uploader.upload( | ||||||
|  |                 inputStream = inputStream, | ||||||
|  |                 bucket = bucket, | ||||||
|  |                 filePath = filePath, | ||||||
|  |                 metadata = metadata | ||||||
|  |             ) | ||||||
|  |         } catch (e: Exception) { | ||||||
|  |             throw SodaException("블러 이미지 저장에 실패했습니다: ${e.message}") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,53 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.character.image.dto | ||||||
|  |  | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.image.CharacterImage | ||||||
|  |  | ||||||
|  | // 요청 DTOs | ||||||
|  |  | ||||||
|  | data class RegisterCharacterImageRequest( | ||||||
|  |     @JsonProperty("characterId") val characterId: Long, | ||||||
|  |     @JsonProperty("imagePriceCan") val imagePriceCan: Long, | ||||||
|  |     @JsonProperty("messagePriceCan") val messagePriceCan: Long, | ||||||
|  |     @JsonProperty("isAdult") val isAdult: Boolean = false, | ||||||
|  |     @JsonProperty("triggers") val triggers: List<String>? = null | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class UpdateCharacterImageTriggersRequest( | ||||||
|  |     @JsonProperty("triggers") val triggers: List<String>? = null | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | data class UpdateCharacterImageOrdersRequest( | ||||||
|  |     @JsonProperty("characterId") val characterId: Long?, | ||||||
|  |     @JsonProperty("ids") val ids: List<Long> | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // 응답 DTOs | ||||||
|  |  | ||||||
|  | data class AdminCharacterImageResponse( | ||||||
|  |     val id: Long, | ||||||
|  |     val characterId: Long, | ||||||
|  |     val imagePriceCan: Long, | ||||||
|  |     val messagePriceCan: Long, | ||||||
|  |     val imageUrl: String, | ||||||
|  |     val triggers: List<String>, | ||||||
|  |     val isAdult: Boolean | ||||||
|  | ) { | ||||||
|  |     companion object { | ||||||
|  |         fun fromWithUrl(entity: CharacterImage, signedUrl: String): AdminCharacterImageResponse { | ||||||
|  |             return base(entity, signedUrl) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private fun base(entity: CharacterImage, url: String): AdminCharacterImageResponse { | ||||||
|  |             return AdminCharacterImageResponse( | ||||||
|  |                 id = entity.id!!, | ||||||
|  |                 characterId = entity.chatCharacter.id!!, | ||||||
|  |                 imagePriceCan = entity.imagePriceCan, | ||||||
|  |                 messagePriceCan = entity.messagePriceCan, | ||||||
|  |                 imageUrl = url, | ||||||
|  |                 triggers = entity.triggerMappings.map { it.tag.word }, | ||||||
|  |                 isAdult = entity.isAdult | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,78 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.character.service | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.character.dto.ChatCharacterDetailResponse | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.character.dto.ChatCharacterListPageResponse | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.character.dto.ChatCharacterListResponse | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.repository.ChatCharacterRepository | ||||||
|  | import kr.co.vividnext.sodalive.common.SodaException | ||||||
|  | import org.springframework.data.domain.Page | ||||||
|  | import org.springframework.data.domain.PageRequest | ||||||
|  | import org.springframework.data.domain.Pageable | ||||||
|  | import org.springframework.data.domain.Sort | ||||||
|  | import org.springframework.stereotype.Service | ||||||
|  | import org.springframework.transaction.annotation.Transactional | ||||||
|  |  | ||||||
|  | @Service | ||||||
|  | class AdminChatCharacterService( | ||||||
|  |     private val chatCharacterRepository: ChatCharacterRepository | ||||||
|  | ) { | ||||||
|  |     /** | ||||||
|  |      * 활성화된 캐릭터 목록을 페이징하여 조회 | ||||||
|  |      * | ||||||
|  |      * @param pageable 페이징 정보 | ||||||
|  |      * @return 페이징된 캐릭터 목록 | ||||||
|  |      */ | ||||||
|  |     @Transactional(readOnly = true) | ||||||
|  |     fun getActiveChatCharacters(pageable: Pageable, imageHost: String = ""): ChatCharacterListPageResponse { | ||||||
|  |         // isActive가 true인 캐릭터만 조회 | ||||||
|  |         val page = chatCharacterRepository.findByIsActiveTrue(pageable) | ||||||
|  |  | ||||||
|  |         // 페이지 정보 생성 | ||||||
|  |         val content = page.content.map { ChatCharacterListResponse.from(it, imageHost) } | ||||||
|  |  | ||||||
|  |         return ChatCharacterListPageResponse( | ||||||
|  |             totalCount = page.totalElements, | ||||||
|  |             content = content | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 기본 페이지 요청 생성 | ||||||
|  |      * | ||||||
|  |      * @param page 페이지 번호 (0부터 시작) | ||||||
|  |      * @param size 페이지 크기 | ||||||
|  |      * @return 페이지 요청 객체 | ||||||
|  |      */ | ||||||
|  |     fun createDefaultPageRequest(page: Int = 0, size: Int = 20): PageRequest { | ||||||
|  |         return PageRequest.of(page, size, Sort.by(Sort.Direction.DESC, "createdAt")) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 캐릭터 상세 정보 조회 | ||||||
|  |      * | ||||||
|  |      * @param characterId 캐릭터 ID | ||||||
|  |      * @param imageHost 이미지 호스트 URL | ||||||
|  |      * @return 캐릭터 상세 정보 | ||||||
|  |      * @throws SodaException 캐릭터를 찾을 수 없는 경우 | ||||||
|  |      */ | ||||||
|  |     @Transactional(readOnly = true) | ||||||
|  |     fun getChatCharacterDetail(characterId: Long, imageHost: String = ""): ChatCharacterDetailResponse { | ||||||
|  |         val chatCharacter = chatCharacterRepository.findById(characterId) | ||||||
|  |             .orElseThrow { SodaException("해당 ID의 캐릭터를 찾을 수 없습니다: $characterId") } | ||||||
|  |  | ||||||
|  |         return ChatCharacterDetailResponse.from(chatCharacter, imageHost) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 캐릭터 검색 (이름, 설명, MBTI, 태그 기반) - 페이징 (기존 사용처 호환용) | ||||||
|  |      */ | ||||||
|  |     @Transactional(readOnly = true) | ||||||
|  |     fun searchCharacters( | ||||||
|  |         searchTerm: String, | ||||||
|  |         pageable: Pageable, | ||||||
|  |         imageHost: String = "" | ||||||
|  |     ): Page<ChatCharacterListResponse> { | ||||||
|  |         val characters = chatCharacterRepository.searchCharacters(searchTerm, pageable) | ||||||
|  |         return characters.map { ChatCharacterListResponse.from(it, imageHost) } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,30 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.dto | ||||||
|  |  | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 캐릭터 배너 등록 요청 DTO | ||||||
|  |  */ | ||||||
|  | data class ChatCharacterBannerRegisterRequest( | ||||||
|  |     // 캐릭터 ID | ||||||
|  |     @JsonProperty("characterId") val characterId: Long | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 캐릭터 배너 수정 요청 DTO | ||||||
|  |  */ | ||||||
|  | data class ChatCharacterBannerUpdateRequest( | ||||||
|  |     // 배너 ID | ||||||
|  |     @JsonProperty("bannerId") val bannerId: Long, | ||||||
|  |  | ||||||
|  |     // 캐릭터 ID (변경할 캐릭터) | ||||||
|  |     @JsonProperty("characterId") val characterId: Long? = null | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 캐릭터 배너 정렬 순서 일괄 변경 요청 DTO | ||||||
|  |  */ | ||||||
|  | data class UpdateBannerOrdersRequest( | ||||||
|  |     // 배너 ID 목록 (순서대로 정렬됨) | ||||||
|  |     @JsonProperty("ids") val ids: List<Long> | ||||||
|  | ) | ||||||
| @@ -0,0 +1,32 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.dto | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.ChatCharacterBanner | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 캐릭터 배너 응답 DTO | ||||||
|  |  */ | ||||||
|  | data class ChatCharacterBannerResponse( | ||||||
|  |     val id: Long, | ||||||
|  |     val imagePath: String, | ||||||
|  |     val characterId: Long, | ||||||
|  |     val characterName: String | ||||||
|  | ) { | ||||||
|  |     companion object { | ||||||
|  |         fun from(banner: ChatCharacterBanner, imageHost: String): ChatCharacterBannerResponse { | ||||||
|  |             return ChatCharacterBannerResponse( | ||||||
|  |                 id = banner.id!!, | ||||||
|  |                 imagePath = "$imageHost/${banner.imagePath}", | ||||||
|  |                 characterId = banner.chatCharacter.id!!, | ||||||
|  |                 characterName = banner.chatCharacter.name | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 캐릭터 배너 목록 페이지 응답 DTO | ||||||
|  |  */ | ||||||
|  | data class ChatCharacterBannerListPageResponse( | ||||||
|  |     val totalCount: Long, | ||||||
|  |     val content: List<ChatCharacterBannerResponse> | ||||||
|  | ) | ||||||
| @@ -0,0 +1,199 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.original | ||||||
|  |  | ||||||
|  | import com.amazonaws.services.s3.model.ObjectMetadata | ||||||
|  | import com.fasterxml.jackson.databind.ObjectMapper | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.character.dto.OriginalWorkChatCharacterListPageResponse | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.character.dto.OriginalWorkChatCharacterResponse | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.original.dto.OriginalWorkAssignCharactersRequest | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.original.dto.OriginalWorkPageResponse | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.original.dto.OriginalWorkRegisterRequest | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.original.dto.OriginalWorkResponse | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.original.dto.OriginalWorkUpdateRequest | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.original.service.AdminOriginalWorkService | ||||||
|  | import kr.co.vividnext.sodalive.aws.s3.S3Uploader | ||||||
|  | import kr.co.vividnext.sodalive.common.ApiResponse | ||||||
|  | import kr.co.vividnext.sodalive.common.SodaException | ||||||
|  | import kr.co.vividnext.sodalive.utils.generateFileName | ||||||
|  | import org.springframework.beans.factory.annotation.Value | ||||||
|  | import org.springframework.security.access.prepost.PreAuthorize | ||||||
|  | import org.springframework.web.bind.annotation.DeleteMapping | ||||||
|  | import org.springframework.web.bind.annotation.GetMapping | ||||||
|  | import org.springframework.web.bind.annotation.PathVariable | ||||||
|  | import org.springframework.web.bind.annotation.PostMapping | ||||||
|  | import org.springframework.web.bind.annotation.PutMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestBody | ||||||
|  | import org.springframework.web.bind.annotation.RequestMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestParam | ||||||
|  | import org.springframework.web.bind.annotation.RequestPart | ||||||
|  | import org.springframework.web.bind.annotation.RestController | ||||||
|  | import org.springframework.web.multipart.MultipartFile | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 원작(오리지널 작품) 관리자 API | ||||||
|  |  * - 원작 등록/수정/삭제 | ||||||
|  |  * - 원작과 캐릭터 연결(배정) 및 해제 | ||||||
|  |  */ | ||||||
|  | @RestController | ||||||
|  | @RequestMapping("/admin/chat/original") | ||||||
|  | @PreAuthorize("hasRole('ADMIN')") | ||||||
|  | class AdminOriginalWorkController( | ||||||
|  |     private val originalWorkService: AdminOriginalWorkService, | ||||||
|  |     private val s3Uploader: S3Uploader, | ||||||
|  |     @Value("\${cloud.aws.s3.bucket}") | ||||||
|  |     private val s3Bucket: String, | ||||||
|  |     @Value("\${cloud.aws.cloud-front.host}") | ||||||
|  |     private val imageHost: String | ||||||
|  | ) { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 원작 등록 | ||||||
|  |      * - 이미지 파일과 JSON 요청을 멀티파트로 받는다. | ||||||
|  |      */ | ||||||
|  |     @PostMapping("/register") | ||||||
|  |     fun register( | ||||||
|  |         @RequestPart("image") image: MultipartFile, | ||||||
|  |         @RequestPart("request") requestString: String | ||||||
|  |     ) = run { | ||||||
|  |         val objectMapper = ObjectMapper() | ||||||
|  |         val request = objectMapper.readValue(requestString, OriginalWorkRegisterRequest::class.java) | ||||||
|  |  | ||||||
|  |         // 서비스 계층을 통해 원작을 생성 | ||||||
|  |         val saved = originalWorkService.createOriginalWork(request) | ||||||
|  |  | ||||||
|  |         // 이미지 업로드 후 이미지 경로 업데이트 | ||||||
|  |         val imagePath = uploadImage(saved.id!!, image) | ||||||
|  |         originalWorkService.updateOriginalWorkImage(saved.id!!, imagePath) | ||||||
|  |  | ||||||
|  |         ApiResponse.ok(null) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 원작 수정 | ||||||
|  |      * - 이미지가 있으면 교체, 없으면 유지 | ||||||
|  |      */ | ||||||
|  |     @PutMapping("/update") | ||||||
|  |     fun update( | ||||||
|  |         @RequestPart(value = "image", required = false) image: MultipartFile?, | ||||||
|  |         @RequestPart("request") requestString: String | ||||||
|  |     ) = run { | ||||||
|  |         val objectMapper = ObjectMapper() | ||||||
|  |         val request = objectMapper.readValue(requestString, OriginalWorkUpdateRequest::class.java) | ||||||
|  |  | ||||||
|  |         // 이미지가 전달된 경우 먼저 업로드하여 경로를 생성 | ||||||
|  |         val imagePath = if (image != null && !image.isEmpty) { | ||||||
|  |             uploadImage(request.id, image) | ||||||
|  |         } else { | ||||||
|  |             null | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         originalWorkService.updateOriginalWork(request, imagePath) | ||||||
|  |         ApiResponse.ok(null) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 원작 삭제 | ||||||
|  |      */ | ||||||
|  |     @DeleteMapping("/{id}") | ||||||
|  |     fun delete(@PathVariable id: Long) = run { | ||||||
|  |         originalWorkService.deleteOriginalWork(id) | ||||||
|  |         ApiResponse.ok(null) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 원작 목록(페이징) | ||||||
|  |      */ | ||||||
|  |     @GetMapping("/list") | ||||||
|  |     fun list( | ||||||
|  |         @RequestParam(defaultValue = "0") page: Int, | ||||||
|  |         @RequestParam(defaultValue = "20") size: Int | ||||||
|  |     ) = run { | ||||||
|  |         val pageRes = originalWorkService.getOriginalWorkPage(page, size) | ||||||
|  |         val content = pageRes.content.map { OriginalWorkResponse.from(it, imageHost) } | ||||||
|  |         ApiResponse.ok(OriginalWorkPageResponse(totalCount = pageRes.totalElements, content = content)) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 원작 검색(관리자) | ||||||
|  |      * - 제목/콘텐츠타입/카테고리 기준 부분 검색, 소프트 삭제 제외 | ||||||
|  |      * - 페이징 제거: 전체 목록 반환 | ||||||
|  |      */ | ||||||
|  |     @GetMapping("/search") | ||||||
|  |     fun search( | ||||||
|  |         @RequestParam("searchTerm") searchTerm: String | ||||||
|  |     ) = run { | ||||||
|  |         val list = originalWorkService.searchOriginalWorksAll(searchTerm) | ||||||
|  |         val content = list.map { OriginalWorkResponse.from(it, imageHost) } | ||||||
|  |         ApiResponse.ok(content) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 원작 상세 | ||||||
|  |      */ | ||||||
|  |     @GetMapping("/{id}") | ||||||
|  |     fun detail(@PathVariable id: Long) = run { | ||||||
|  |         ApiResponse.ok(OriginalWorkResponse.from(originalWorkService.getOriginalWork(id), imageHost)) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 원작에 기존 캐릭터들을 배정 | ||||||
|  |      * - 캐릭터는 하나의 원작에만 속하므로, 해당 캐릭터들의 originalWork를 이 원작으로 설정 | ||||||
|  |      */ | ||||||
|  |     @PostMapping("/{id}/assign-characters") | ||||||
|  |     fun assignCharacters( | ||||||
|  |         @PathVariable id: Long, | ||||||
|  |         @RequestBody body: OriginalWorkAssignCharactersRequest | ||||||
|  |     ) = run { | ||||||
|  |         originalWorkService.assignCharacters(id, body.characterIds) | ||||||
|  |         ApiResponse.ok(null) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 원작에서 캐릭터들 해제 | ||||||
|  |      * - 캐릭터들의 originalWork를 null로 설정 | ||||||
|  |      */ | ||||||
|  |     @PostMapping("/{id}/unassign-characters") | ||||||
|  |     fun unassignCharacters( | ||||||
|  |         @PathVariable id: Long, | ||||||
|  |         @RequestBody body: OriginalWorkAssignCharactersRequest | ||||||
|  |     ) = run { | ||||||
|  |         originalWorkService.unassignCharacters(id, body.characterIds) | ||||||
|  |         ApiResponse.ok(null) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 관리자용: 지정 원작에 속한 캐릭터 목록 페이징 조회 | ||||||
|  |      * - 활성 캐릭터만 포함 | ||||||
|  |      * - 응답 항목: 캐릭터 이미지(URL), 이름 | ||||||
|  |      */ | ||||||
|  |     @GetMapping("/{id}/characters") | ||||||
|  |     fun listCharactersOfOriginal( | ||||||
|  |         @PathVariable id: Long, | ||||||
|  |         @RequestParam(defaultValue = "0") page: Int, | ||||||
|  |         @RequestParam(defaultValue = "20") size: Int | ||||||
|  |     ) = run { | ||||||
|  |         val pageRes = originalWorkService.getCharactersOfOriginalWorkPage(id, page, size) | ||||||
|  |         val content = pageRes.content.map { OriginalWorkChatCharacterResponse.from(it, imageHost) } | ||||||
|  |         ApiResponse.ok( | ||||||
|  |             OriginalWorkChatCharacterListPageResponse( | ||||||
|  |                 totalCount = pageRes.totalElements, | ||||||
|  |                 content = content | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** 이미지 업로드 공통 처리 */ | ||||||
|  |     private fun uploadImage(originalWorkId: Long, image: MultipartFile): String { | ||||||
|  |         try { | ||||||
|  |             val metadata = ObjectMetadata() | ||||||
|  |             metadata.contentLength = image.size | ||||||
|  |             return s3Uploader.upload( | ||||||
|  |                 inputStream = image.inputStream, | ||||||
|  |                 bucket = s3Bucket, | ||||||
|  |                 filePath = "originals/$originalWorkId/${generateFileName(prefix = "original")}", | ||||||
|  |                 metadata = metadata | ||||||
|  |             ) | ||||||
|  |         } catch (e: Exception) { | ||||||
|  |             throw SodaException("이미지 저장에 실패했습니다: ${e.message}") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,95 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.original.dto | ||||||
|  |  | ||||||
|  | import com.fasterxml.jackson.annotation.JsonProperty | ||||||
|  | import kr.co.vividnext.sodalive.chat.original.OriginalWork | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 원작 등록 요청 DTO | ||||||
|  |  */ | ||||||
|  | data class OriginalWorkRegisterRequest( | ||||||
|  |     @JsonProperty("title") val title: String, | ||||||
|  |     @JsonProperty("contentType") val contentType: String, | ||||||
|  |     @JsonProperty("category") val category: String, | ||||||
|  |     @JsonProperty("isAdult") val isAdult: Boolean = false, | ||||||
|  |     @JsonProperty("description") val description: String = "", | ||||||
|  |     @JsonProperty("originalWork") val originalWork: String? = null, | ||||||
|  |     @JsonProperty("originalLink") val originalLink: String? = null, | ||||||
|  |     @JsonProperty("writer") val writer: String? = null, | ||||||
|  |     @JsonProperty("studio") val studio: String? = null, | ||||||
|  |     @JsonProperty("originalLinks") val originalLinks: List<String>? = null, | ||||||
|  |     @JsonProperty("tags") val tags: List<String>? = null | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 원작 수정 요청 DTO (부분 수정 가능) | ||||||
|  |  */ | ||||||
|  | data class OriginalWorkUpdateRequest( | ||||||
|  |     @JsonProperty("id") val id: Long, | ||||||
|  |     @JsonProperty("title") val title: String? = null, | ||||||
|  |     @JsonProperty("contentType") val contentType: String? = null, | ||||||
|  |     @JsonProperty("category") val category: String? = null, | ||||||
|  |     @JsonProperty("isAdult") val isAdult: Boolean? = null, | ||||||
|  |     @JsonProperty("description") val description: String? = null, | ||||||
|  |     @JsonProperty("originalWork") val originalWork: String? = null, | ||||||
|  |     @JsonProperty("originalLink") val originalLink: String? = null, | ||||||
|  |     @JsonProperty("writer") val writer: String? = null, | ||||||
|  |     @JsonProperty("studio") val studio: String? = null, | ||||||
|  |     @JsonProperty("originalLinks") val originalLinks: List<String>? = null, | ||||||
|  |     @JsonProperty("tags") val tags: List<String>? = null | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 원작 상세/목록 응답 DTO | ||||||
|  |  */ | ||||||
|  | data class OriginalWorkResponse( | ||||||
|  |     val id: Long, | ||||||
|  |     val title: String, | ||||||
|  |     val contentType: String, | ||||||
|  |     val category: String, | ||||||
|  |     val isAdult: Boolean, | ||||||
|  |     val description: String, | ||||||
|  |     val originalWork: String?, | ||||||
|  |     val originalLink: String?, | ||||||
|  |     val writer: String?, | ||||||
|  |     val studio: String?, | ||||||
|  |     val originalLinks: List<String>, | ||||||
|  |     val tags: List<String>, | ||||||
|  |     val imageUrl: String? | ||||||
|  | ) { | ||||||
|  |     companion object { | ||||||
|  |         fun from(entity: OriginalWork, imageHost: String = ""): OriginalWorkResponse { | ||||||
|  |             val fullImagePath = if (entity.imagePath != null && imageHost.isNotEmpty()) { | ||||||
|  |                 "$imageHost/${entity.imagePath}" | ||||||
|  |             } else { | ||||||
|  |                 entity.imagePath | ||||||
|  |             } | ||||||
|  |             return OriginalWorkResponse( | ||||||
|  |                 id = entity.id!!, | ||||||
|  |                 title = entity.title, | ||||||
|  |                 contentType = entity.contentType, | ||||||
|  |                 category = entity.category, | ||||||
|  |                 isAdult = entity.isAdult, | ||||||
|  |                 description = entity.description, | ||||||
|  |                 originalWork = entity.originalWork, | ||||||
|  |                 originalLink = entity.originalLink, | ||||||
|  |                 writer = entity.writer, | ||||||
|  |                 studio = entity.studio, | ||||||
|  |                 originalLinks = entity.originalLinks.map { it.url }, | ||||||
|  |                 tags = entity.tagMappings.map { it.tag.tag }, | ||||||
|  |                 imageUrl = fullImagePath | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | data class OriginalWorkPageResponse( | ||||||
|  |     val totalCount: Long, | ||||||
|  |     val content: List<OriginalWorkResponse> | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 원작-캐릭터 연결/해제 요청 DTO | ||||||
|  |  */ | ||||||
|  | data class OriginalWorkAssignCharactersRequest( | ||||||
|  |     @JsonProperty("characterIds") val characterIds: List<Long> | ||||||
|  | ) | ||||||
| @@ -0,0 +1,213 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.chat.original.service | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.original.dto.OriginalWorkRegisterRequest | ||||||
|  | import kr.co.vividnext.sodalive.admin.chat.original.dto.OriginalWorkUpdateRequest | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.ChatCharacter | ||||||
|  | import kr.co.vividnext.sodalive.chat.character.repository.ChatCharacterRepository | ||||||
|  | import kr.co.vividnext.sodalive.chat.original.OriginalWork | ||||||
|  | import kr.co.vividnext.sodalive.chat.original.OriginalWorkRepository | ||||||
|  | import kr.co.vividnext.sodalive.chat.original.OriginalWorkTag | ||||||
|  | import kr.co.vividnext.sodalive.chat.original.OriginalWorkTagMapping | ||||||
|  | import kr.co.vividnext.sodalive.chat.original.repository.OriginalWorkTagRepository | ||||||
|  | import kr.co.vividnext.sodalive.common.SodaException | ||||||
|  | import org.springframework.data.domain.Page | ||||||
|  | import org.springframework.data.domain.PageRequest | ||||||
|  | import org.springframework.data.domain.Sort | ||||||
|  | import org.springframework.stereotype.Service | ||||||
|  | import org.springframework.transaction.annotation.Transactional | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 원작(오리지널 작품) 관련 관리자 서비스 | ||||||
|  |  * - 컨트롤러와 레포지토리 사이의 서비스 계층으로 DB 접근을 캡슐화한다. | ||||||
|  |  */ | ||||||
|  | @Service | ||||||
|  | class AdminOriginalWorkService( | ||||||
|  |     private val originalWorkRepository: OriginalWorkRepository, | ||||||
|  |     private val chatCharacterRepository: ChatCharacterRepository, | ||||||
|  |     private val originalWorkTagRepository: OriginalWorkTagRepository | ||||||
|  | ) { | ||||||
|  |  | ||||||
|  |     /** 원작 등록 (중복 제목 방지 포함) */ | ||||||
|  |     @Transactional | ||||||
|  |     fun createOriginalWork(request: OriginalWorkRegisterRequest): OriginalWork { | ||||||
|  |         originalWorkRepository.findByTitleAndIsDeletedFalse(request.title)?.let { | ||||||
|  |             throw SodaException("동일한 제목의 원작이 이미 존재합니다: ${request.title}") | ||||||
|  |         } | ||||||
|  |         val entity = OriginalWork( | ||||||
|  |             title = request.title, | ||||||
|  |             contentType = request.contentType, | ||||||
|  |             category = request.category, | ||||||
|  |             isAdult = request.isAdult, | ||||||
|  |             description = request.description, | ||||||
|  |             originalWork = request.originalWork, | ||||||
|  |             originalLink = request.originalLink, | ||||||
|  |             writer = request.writer, | ||||||
|  |             studio = request.studio | ||||||
|  |         ) | ||||||
|  |         // 링크 리스트 생성 | ||||||
|  |         request.originalLinks?.filter { it.isNotBlank() }?.forEach { link -> | ||||||
|  |             entity.originalLinks.add(kr.co.vividnext.sodalive.chat.original.OriginalWorkLink(url = link, originalWork = entity)) | ||||||
|  |         } | ||||||
|  |         // 태그 매핑 생성 (기존 태그 재사용) | ||||||
|  |         request.tags?.let { tags -> | ||||||
|  |             val normalized = tags.map { it.trim() }.filter { it.isNotBlank() }.toSet() | ||||||
|  |             normalized.forEach { t -> | ||||||
|  |                 val tagEntity = originalWorkTagRepository.findByTag(t) ?: originalWorkTagRepository.save(OriginalWorkTag(t)) | ||||||
|  |                 entity.tagMappings.add(OriginalWorkTagMapping(originalWork = entity, tag = tagEntity)) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return originalWorkRepository.save(entity) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** 원작 수정 (이미지 경로 포함 선택적 변경) */ | ||||||
|  |     @Transactional | ||||||
|  |     fun updateOriginalWork(request: OriginalWorkUpdateRequest, imagePath: String? = null): OriginalWork { | ||||||
|  |         val ow = originalWorkRepository.findByIdAndIsDeletedFalse(request.id) | ||||||
|  |             .orElseThrow { SodaException("해당 원작을 찾을 수 없습니다") } | ||||||
|  |  | ||||||
|  |         request.title?.let { ow.title = it } | ||||||
|  |         request.contentType?.let { ow.contentType = it } | ||||||
|  |         request.category?.let { ow.category = it } | ||||||
|  |         request.isAdult?.let { ow.isAdult = it } | ||||||
|  |         request.description?.let { ow.description = it } | ||||||
|  |         request.originalWork?.let { ow.originalWork = it } | ||||||
|  |         request.originalLink?.let { ow.originalLink = it } | ||||||
|  |         request.writer?.let { ow.writer = it } | ||||||
|  |         request.studio?.let { ow.studio = it } | ||||||
|  |         // 링크 리스트가 전달되면 기존 것을 교체 | ||||||
|  |         request.originalLinks?.let { links -> | ||||||
|  |             ow.originalLinks.clear() | ||||||
|  |             links.filter { it.isNotBlank() }.forEach { link -> | ||||||
|  |                 ow.originalLinks.add(kr.co.vividnext.sodalive.chat.original.OriginalWorkLink(url = link, originalWork = ow)) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         // 태그 변경사항만 반영 (요청이 null이면 변경 없음) | ||||||
|  |         request.tags?.let { tags -> | ||||||
|  |             val normalized = tags.map { it.trim() }.filter { it.isNotBlank() }.toSet() | ||||||
|  |             val current = ow.tagMappings.map { it.tag.tag }.toSet() | ||||||
|  |             val toAdd = normalized.minus(current) | ||||||
|  |             val toRemove = current.minus(normalized) | ||||||
|  |  | ||||||
|  |             if (toRemove.isNotEmpty()) { | ||||||
|  |                 val itr = ow.tagMappings.iterator() | ||||||
|  |                 while (itr.hasNext()) { | ||||||
|  |                     val m = itr.next() | ||||||
|  |                     if (toRemove.contains(m.tag.tag)) { | ||||||
|  |                         itr.remove() // orphanRemoval=true로 매핑 삭제 | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             if (toAdd.isNotEmpty()) { | ||||||
|  |                 toAdd.forEach { t -> | ||||||
|  |                     val tagEntity = originalWorkTagRepository.findByTag(t) ?: originalWorkTagRepository.save(OriginalWorkTag(t)) | ||||||
|  |                     ow.tagMappings.add(OriginalWorkTagMapping(originalWork = ow, tag = tagEntity)) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         if (imagePath != null) { | ||||||
|  |             ow.imagePath = imagePath | ||||||
|  |         } | ||||||
|  |         return originalWorkRepository.save(ow) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** 원작 이미지 경로만 별도 갱신 */ | ||||||
|  |     @Transactional | ||||||
|  |     fun updateOriginalWorkImage(originalWorkId: Long, imagePath: String): OriginalWork { | ||||||
|  |         val ow = originalWorkRepository.findByIdAndIsDeletedFalse(originalWorkId) | ||||||
|  |             .orElseThrow { SodaException("해당 원작을 찾을 수 없습니다") } | ||||||
|  |         ow.imagePath = imagePath | ||||||
|  |         return originalWorkRepository.save(ow) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** 원작 삭제 (소프트 삭제) */ | ||||||
|  |     @Transactional | ||||||
|  |     fun deleteOriginalWork(id: Long) { | ||||||
|  |         val ow = originalWorkRepository.findByIdAndIsDeletedFalse(id) | ||||||
|  |             .orElseThrow { SodaException("해당 원작을 찾을 수 없습니다: $id") } | ||||||
|  |         ow.isDeleted = true | ||||||
|  |         originalWorkRepository.save(ow) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** 원작 상세 조회 (소프트 삭제 제외) */ | ||||||
|  |     @Transactional(readOnly = true) | ||||||
|  |     fun getOriginalWork(id: Long): OriginalWork { | ||||||
|  |         return originalWorkRepository.findByIdAndIsDeletedFalse(id) | ||||||
|  |             .orElseThrow { SodaException("해당 원작을 찾을 수 없습니다") } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** 원작 페이징 조회 */ | ||||||
|  |     @Transactional(readOnly = true) | ||||||
|  |     fun getOriginalWorkPage(page: Int, size: Int): Page<OriginalWork> { | ||||||
|  |         val safePage = if (page < 0) 0 else page | ||||||
|  |         val safeSize = when { | ||||||
|  |             size <= 0 -> 20 | ||||||
|  |             size > 100 -> 100 | ||||||
|  |             else -> size | ||||||
|  |         } | ||||||
|  |         val pageable = PageRequest.of(safePage, safeSize, Sort.by("createdAt").descending()) | ||||||
|  |         return originalWorkRepository.findByIsDeletedFalse(pageable) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** 지정 원작에 속한 활성 캐릭터 페이징 조회 (최신순) */ | ||||||
|  |     @Transactional(readOnly = true) | ||||||
|  |     fun getCharactersOfOriginalWorkPage(originalWorkId: Long, page: Int, size: Int): Page<ChatCharacter> { | ||||||
|  |         // 원작 존재 및 소프트 삭제 여부 확인 | ||||||
|  |         originalWorkRepository.findByIdAndIsDeletedFalse(originalWorkId) | ||||||
|  |             .orElseThrow { SodaException("해당 원작을 찾을 수 없습니다") } | ||||||
|  |  | ||||||
|  |         val safePage = if (page < 0) 0 else page | ||||||
|  |         val safeSize = when { | ||||||
|  |             size <= 0 -> 20 | ||||||
|  |             size > 100 -> 100 | ||||||
|  |             else -> size | ||||||
|  |         } | ||||||
|  |         val pageable = PageRequest.of(safePage, safeSize, Sort.by("createdAt").descending()) | ||||||
|  |         return chatCharacterRepository.findByOriginalWorkIdAndIsActiveTrue(originalWorkId, pageable) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** 원작 검색 (제목/콘텐츠타입/카테고리, 소프트 삭제 제외) - 무페이징 */ | ||||||
|  |     @Transactional(readOnly = true) | ||||||
|  |     fun searchOriginalWorksAll(searchTerm: String): List<OriginalWork> { | ||||||
|  |         return originalWorkRepository.searchNoPaging(searchTerm) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** 원작에 기존 캐릭터들을 배정 */ | ||||||
|  |     @Transactional | ||||||
|  |     fun assignCharacters(originalWorkId: Long, characterIds: List<Long>) { | ||||||
|  |         val ow = originalWorkRepository.findByIdAndIsDeletedFalse(originalWorkId) | ||||||
|  |             .orElseThrow { SodaException("해당 원작을 찾을 수 없습니다") } | ||||||
|  |         if (characterIds.isEmpty()) return | ||||||
|  |         val characters = chatCharacterRepository.findByIdInAndIsActiveTrue(characterIds) | ||||||
|  |         characters.forEach { it.originalWork = ow } | ||||||
|  |         chatCharacterRepository.saveAll(characters) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** 원작에서 캐릭터들 해제 */ | ||||||
|  |     @Transactional | ||||||
|  |     fun unassignCharacters(originalWorkId: Long, characterIds: List<Long>) { | ||||||
|  |         // 원작 존재 확인 (소프트 삭제 제외) | ||||||
|  |         originalWorkRepository.findByIdAndIsDeletedFalse(originalWorkId) | ||||||
|  |             .orElseThrow { SodaException("해당 원작을 찾을 수 없습니다") } | ||||||
|  |         if (characterIds.isEmpty()) return | ||||||
|  |         val characters = chatCharacterRepository.findByIdInAndIsActiveTrue(characterIds) | ||||||
|  |         characters.forEach { it.originalWork = null } | ||||||
|  |         chatCharacterRepository.saveAll(characters) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** 단일 캐릭터를 지정 원작에 배정 */ | ||||||
|  |     @Transactional | ||||||
|  |     fun assignOneCharacter(originalWorkId: Long, characterId: Long) { | ||||||
|  |         val character = chatCharacterRepository.findById(characterId) | ||||||
|  |             .orElseThrow { SodaException("해당 캐릭터를 찾을 수 없습니다") } | ||||||
|  |  | ||||||
|  |         if (originalWorkId == 0L) { | ||||||
|  |             character.originalWork = null | ||||||
|  |         } else { | ||||||
|  |             val ow = originalWorkRepository.findByIdAndIsDeletedFalse(originalWorkId) | ||||||
|  |                 .orElseThrow { SodaException("해당 원작을 찾을 수 없습니다") } | ||||||
|  |             character.originalWork = ow | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         chatCharacterRepository.save(character) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -15,16 +15,42 @@ import org.springframework.web.bind.annotation.RestController | |||||||
| @RequestMapping("/admin/audio-content") | @RequestMapping("/admin/audio-content") | ||||||
| class AdminContentController(private val service: AdminContentService) { | class AdminContentController(private val service: AdminContentService) { | ||||||
|     @GetMapping("/list") |     @GetMapping("/list") | ||||||
|     fun getAudioContentList(pageable: Pageable) = ApiResponse.ok(service.getAudioContentList(pageable)) |     fun getAudioContentList( | ||||||
|  |         @RequestParam(value = "status", required = false) status: ContentReleaseStatus?, | ||||||
|  |         pageable: Pageable | ||||||
|  |     ) = ApiResponse.ok( | ||||||
|  |         service.getAudioContentList( | ||||||
|  |             status = status ?: ContentReleaseStatus.OPEN, | ||||||
|  |             pageable | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|     @GetMapping("/search") |     @GetMapping("/search") | ||||||
|     fun searchAudioContent( |     fun searchAudioContent( | ||||||
|  |         @RequestParam(value = "status", required = false) status: ContentReleaseStatus?, | ||||||
|         @RequestParam(value = "search_word") searchWord: String, |         @RequestParam(value = "search_word") searchWord: String, | ||||||
|         pageable: Pageable |         pageable: Pageable | ||||||
|     ) = ApiResponse.ok(service.searchAudioContent(searchWord, pageable)) |     ) = ApiResponse.ok( | ||||||
|  |         service.searchAudioContent( | ||||||
|  |             status = status ?: ContentReleaseStatus.OPEN, | ||||||
|  |             searchWord, | ||||||
|  |             pageable | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|     @PutMapping |     @PutMapping | ||||||
|     fun modifyAudioContent( |     fun modifyAudioContent( | ||||||
|         @RequestBody request: UpdateAdminContentRequest |         @RequestBody request: UpdateAdminContentRequest | ||||||
|     ) = ApiResponse.ok(service.updateAudioContent(request)) |     ) = ApiResponse.ok(service.updateAudioContent(request)) | ||||||
|  |  | ||||||
|  |     @GetMapping("/main/tab") | ||||||
|  |     fun getContentMainTabList() = ApiResponse.ok(service.getContentMainTabList()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | enum class ContentReleaseStatus { | ||||||
|  |     // 콘텐츠가 공개된 상태 | ||||||
|  |     OPEN, | ||||||
|  |  | ||||||
|  |     // 예약된 콘텐츠, 아직 공개되지 않은 상태 | ||||||
|  |     SCHEDULED | ||||||
| } | } | ||||||
|   | |||||||
| @@ -10,6 +10,7 @@ import kr.co.vividnext.sodalive.content.hashtag.QAudioContentHashTag.audioConten | |||||||
| import kr.co.vividnext.sodalive.content.hashtag.QHashTag.hashTag | import kr.co.vividnext.sodalive.content.hashtag.QHashTag.hashTag | ||||||
| import kr.co.vividnext.sodalive.content.main.curation.QAudioContentCuration.audioContentCuration | import kr.co.vividnext.sodalive.content.main.curation.QAudioContentCuration.audioContentCuration | ||||||
| import kr.co.vividnext.sodalive.content.theme.QAudioContentTheme.audioContentTheme | import kr.co.vividnext.sodalive.content.theme.QAudioContentTheme.audioContentTheme | ||||||
|  | import org.springframework.beans.factory.annotation.Value | ||||||
| import org.springframework.data.jpa.repository.JpaRepository | import org.springframework.data.jpa.repository.JpaRepository | ||||||
| import org.springframework.stereotype.Repository | import org.springframework.stereotype.Repository | ||||||
| import java.time.LocalDateTime | import java.time.LocalDateTime | ||||||
| @@ -18,18 +19,37 @@ import java.time.LocalDateTime | |||||||
| interface AdminContentRepository : JpaRepository<AudioContent, Long>, AdminAudioContentQueryRepository | interface AdminContentRepository : JpaRepository<AudioContent, Long>, AdminAudioContentQueryRepository | ||||||
|  |  | ||||||
| interface AdminAudioContentQueryRepository { | interface AdminAudioContentQueryRepository { | ||||||
|     fun getAudioContentTotalCount(searchWord: String = ""): Int |     fun getAudioContentTotalCount( | ||||||
|     fun getAudioContentList(offset: Long, limit: Long, searchWord: String = ""): List<GetAdminContentListItem> |         searchWord: String = "", | ||||||
|  |         status: ContentReleaseStatus = ContentReleaseStatus.OPEN | ||||||
|  |     ): Int | ||||||
|  |  | ||||||
|  |     fun getAudioContentList( | ||||||
|  |         status: ContentReleaseStatus = ContentReleaseStatus.OPEN, | ||||||
|  |         offset: Long, | ||||||
|  |         limit: Long, | ||||||
|  |         searchWord: String = "" | ||||||
|  |     ): List<GetAdminContentListItem> | ||||||
|  |  | ||||||
|     fun getHashTagList(audioContentId: Long): List<String> |     fun getHashTagList(audioContentId: Long): List<String> | ||||||
|  |     fun findByIdAndActiveTrue(audioContentId: Long): AudioContent? | ||||||
| } | } | ||||||
|  |  | ||||||
| class AdminAudioContentQueryRepositoryImpl( | class AdminAudioContentQueryRepositoryImpl( | ||||||
|     private val queryFactory: JPAQueryFactory |     private val queryFactory: JPAQueryFactory, | ||||||
|  |  | ||||||
|  |     @Value("\${cloud.aws.cloud-front.host}") | ||||||
|  |     private val imageHost: String | ||||||
| ) : AdminAudioContentQueryRepository { | ) : AdminAudioContentQueryRepository { | ||||||
|     override fun getAudioContentTotalCount(searchWord: String): Int { |     override fun getAudioContentTotalCount( | ||||||
|  |         searchWord: String, | ||||||
|  |         status: ContentReleaseStatus | ||||||
|  |     ): Int { | ||||||
|  |         val now = LocalDateTime.now() | ||||||
|  |  | ||||||
|         var where = audioContent.duration.isNotNull |         var where = audioContent.duration.isNotNull | ||||||
|             .and(audioContent.member.isNotNull) |             .and(audioContent.member.isNotNull) | ||||||
|             .and(audioContent.isActive.isTrue) |             .and(audioContent.isActive.isTrue.or(audioContent.releaseDate.isNotNull)) | ||||||
|  |  | ||||||
|         if (searchWord.trim().length > 1) { |         if (searchWord.trim().length > 1) { | ||||||
|             where = where.and( |             where = where.and( | ||||||
| @@ -38,6 +58,12 @@ class AdminAudioContentQueryRepositoryImpl( | |||||||
|             ) |             ) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         where = if (status == ContentReleaseStatus.SCHEDULED) { | ||||||
|  |             where.and(audioContent.releaseDate.after(now)) | ||||||
|  |         } else { | ||||||
|  |             where.and(audioContent.releaseDate.before(now)) | ||||||
|  |         } | ||||||
|  |  | ||||||
|         return queryFactory |         return queryFactory | ||||||
|             .select(audioContent.id) |             .select(audioContent.id) | ||||||
|             .from(audioContent) |             .from(audioContent) | ||||||
| @@ -46,10 +72,17 @@ class AdminAudioContentQueryRepositoryImpl( | |||||||
|             .size |             .size | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun getAudioContentList(offset: Long, limit: Long, searchWord: String): List<GetAdminContentListItem> { |     override fun getAudioContentList( | ||||||
|  |         status: ContentReleaseStatus, | ||||||
|  |         offset: Long, | ||||||
|  |         limit: Long, | ||||||
|  |         searchWord: String | ||||||
|  |     ): List<GetAdminContentListItem> { | ||||||
|  |         val now = LocalDateTime.now() | ||||||
|  |  | ||||||
|         var where = audioContent.duration.isNotNull |         var where = audioContent.duration.isNotNull | ||||||
|             .and(audioContent.member.isNotNull) |             .and(audioContent.member.isNotNull) | ||||||
|             .and(audioContent.isActive.isTrue) |             .and(audioContent.isActive.isTrue.or(audioContent.releaseDate.isNotNull)) | ||||||
|  |  | ||||||
|         if (searchWord.trim().length > 1) { |         if (searchWord.trim().length > 1) { | ||||||
|             where = where.and( |             where = where.and( | ||||||
| @@ -58,6 +91,12 @@ class AdminAudioContentQueryRepositoryImpl( | |||||||
|             ) |             ) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         where = if (status == ContentReleaseStatus.SCHEDULED) { | ||||||
|  |             where.and(audioContent.releaseDate.after(now)) | ||||||
|  |         } else { | ||||||
|  |             where.and(audioContent.releaseDate.before(now)) | ||||||
|  |         } | ||||||
|  |  | ||||||
|         return queryFactory |         return queryFactory | ||||||
|             .select( |             .select( | ||||||
|                 QGetAdminContentListItem( |                 QGetAdminContentListItem( | ||||||
| @@ -66,14 +105,19 @@ class AdminAudioContentQueryRepositoryImpl( | |||||||
|                     audioContent.detail, |                     audioContent.detail, | ||||||
|                     audioContentCuration.title, |                     audioContentCuration.title, | ||||||
|                     audioContentCuration.id.nullif(0), |                     audioContentCuration.id.nullif(0), | ||||||
|                     audioContent.coverImage, |                     audioContent.coverImage.prepend("/").prepend(imageHost), | ||||||
|                     audioContent.member!!.nickname, |                     audioContent.member!!.nickname, | ||||||
|                     audioContentTheme.theme, |                     audioContentTheme.theme, | ||||||
|  |                     audioContentTheme.id, | ||||||
|                     audioContent.price, |                     audioContent.price, | ||||||
|  |                     audioContent.limited, | ||||||
|  |                     audioContent.remaining, | ||||||
|                     audioContent.isAdult, |                     audioContent.isAdult, | ||||||
|                     audioContent.duration, |                     audioContent.duration, | ||||||
|                     audioContent.content, |                     audioContent.content, | ||||||
|                     formattedDateExpression(audioContent.createdAt) |                     audioContent.isCommentAvailable, | ||||||
|  |                     formattedDateExpression(audioContent.createdAt), | ||||||
|  |                     formattedDateExpression(audioContent.releaseDate, "%Y-%m-%d %H:%i") | ||||||
|                 ) |                 ) | ||||||
|             ) |             ) | ||||||
|             .from(audioContent) |             .from(audioContent) | ||||||
| @@ -82,7 +126,7 @@ class AdminAudioContentQueryRepositoryImpl( | |||||||
|             .where(where) |             .where(where) | ||||||
|             .offset(offset) |             .offset(offset) | ||||||
|             .limit(limit) |             .limit(limit) | ||||||
|             .orderBy(audioContent.id.desc()) |             .orderBy(audioContent.releaseDate.desc()) | ||||||
|             .fetch() |             .fetch() | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -96,10 +140,21 @@ class AdminAudioContentQueryRepositoryImpl( | |||||||
|                 audioContent.duration.isNotNull |                 audioContent.duration.isNotNull | ||||||
|                     .and(audioContent.member.isNotNull) |                     .and(audioContent.member.isNotNull) | ||||||
|                     .and(audioContentHashTag.audioContent.id.eq(audioContentId)) |                     .and(audioContentHashTag.audioContent.id.eq(audioContentId)) | ||||||
|  |                     .and(audioContentHashTag.isActive.isTrue) | ||||||
|             ) |             ) | ||||||
|             .fetch() |             .fetch() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     override fun findByIdAndActiveTrue(audioContentId: Long): AudioContent? { | ||||||
|  |         return queryFactory | ||||||
|  |             .selectFrom(audioContent) | ||||||
|  |             .where( | ||||||
|  |                 audioContent.id.eq(audioContentId), | ||||||
|  |                 audioContent.isActive.isTrue | ||||||
|  |             ) | ||||||
|  |             .fetchFirst() | ||||||
|  |     } | ||||||
|  |  | ||||||
|     private fun formattedDateExpression( |     private fun formattedDateExpression( | ||||||
|         dateTime: DateTimePath<LocalDateTime>, |         dateTime: DateTimePath<LocalDateTime>, | ||||||
|         format: String = "%Y-%m-%d" |         format: String = "%Y-%m-%d" | ||||||
|   | |||||||
| @@ -1,9 +1,11 @@ | |||||||
| package kr.co.vividnext.sodalive.admin.content | package kr.co.vividnext.sodalive.admin.content | ||||||
|  |  | ||||||
| import kr.co.vividnext.sodalive.admin.content.curation.AdminContentCurationRepository | import kr.co.vividnext.sodalive.admin.content.curation.AdminContentCurationRepository | ||||||
|  | import kr.co.vividnext.sodalive.admin.content.tab.AdminContentMainTabRepository | ||||||
|  | import kr.co.vividnext.sodalive.admin.content.theme.AdminContentThemeRepository | ||||||
| import kr.co.vividnext.sodalive.aws.cloudfront.AudioContentCloudFront | import kr.co.vividnext.sodalive.aws.cloudfront.AudioContentCloudFront | ||||||
| import kr.co.vividnext.sodalive.common.SodaException | import kr.co.vividnext.sodalive.common.SodaException | ||||||
| import org.springframework.beans.factory.annotation.Value | import kr.co.vividnext.sodalive.content.main.tab.GetContentMainTabItem | ||||||
| import org.springframework.data.domain.Pageable | import org.springframework.data.domain.Pageable | ||||||
| import org.springframework.data.repository.findByIdOrNull | import org.springframework.data.repository.findByIdOrNull | ||||||
| import org.springframework.stereotype.Service | import org.springframework.stereotype.Service | ||||||
| @@ -12,21 +14,20 @@ import org.springframework.transaction.annotation.Transactional | |||||||
| @Service | @Service | ||||||
| class AdminContentService( | class AdminContentService( | ||||||
|     private val repository: AdminContentRepository, |     private val repository: AdminContentRepository, | ||||||
|  |     private val themeRepository: AdminContentThemeRepository, | ||||||
|     private val audioContentCloudFront: AudioContentCloudFront, |     private val audioContentCloudFront: AudioContentCloudFront, | ||||||
|     private val curationRepository: AdminContentCurationRepository, |     private val curationRepository: AdminContentCurationRepository, | ||||||
|  |     private val contentMainTabRepository: AdminContentMainTabRepository | ||||||
|     @Value("\${cloud.aws.cloud-front.host}") |  | ||||||
|     private val coverImageHost: String |  | ||||||
| ) { | ) { | ||||||
|     fun getAudioContentList(pageable: Pageable): GetAdminContentListResponse { |     fun getAudioContentList(status: ContentReleaseStatus, pageable: Pageable): GetAdminContentListResponse { | ||||||
|         val totalCount = repository.getAudioContentTotalCount() |         val totalCount = repository.getAudioContentTotalCount(status = status) | ||||||
|         val audioContentAndThemeList = repository.getAudioContentList( |         val audioContentAndThemeList = repository.getAudioContentList( | ||||||
|  |             status = status, | ||||||
|             offset = pageable.offset, |             offset = pageable.offset, | ||||||
|             limit = pageable.pageSize.toLong() |             limit = pageable.pageSize.toLong() | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         val audioContentList = audioContentAndThemeList |         val audioContentList = audioContentAndThemeList | ||||||
|             .asSequence() |  | ||||||
|             .map { |             .map { | ||||||
|                 val tags = repository |                 val tags = repository | ||||||
|                     .getHashTagList(audioContentId = it.audioContentId) |                     .getHashTagList(audioContentId = it.audioContentId) | ||||||
| @@ -41,26 +42,25 @@ class AdminContentService( | |||||||
|                 ) |                 ) | ||||||
|                 it |                 it | ||||||
|             } |             } | ||||||
|             .map { |  | ||||||
|                 it.coverImageUrl = "$coverImageHost/${it.coverImageUrl}" |  | ||||||
|                 it |  | ||||||
|             } |  | ||||||
|             .toList() |  | ||||||
|  |  | ||||||
|         return GetAdminContentListResponse(totalCount, audioContentList) |         return GetAdminContentListResponse(totalCount, audioContentList) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun searchAudioContent(searchWord: String, pageable: Pageable): GetAdminContentListResponse { |     fun searchAudioContent( | ||||||
|  |         status: ContentReleaseStatus, | ||||||
|  |         searchWord: String, | ||||||
|  |         pageable: Pageable | ||||||
|  |     ): GetAdminContentListResponse { | ||||||
|         if (searchWord.length < 2) throw SodaException("2글자 이상 입력하세요.") |         if (searchWord.length < 2) throw SodaException("2글자 이상 입력하세요.") | ||||||
|         val totalCount = repository.getAudioContentTotalCount(searchWord) |         val totalCount = repository.getAudioContentTotalCount(searchWord, status = status) | ||||||
|         val audioContentAndThemeList = repository.getAudioContentList( |         val audioContentAndThemeList = repository.getAudioContentList( | ||||||
|  |             status = status, | ||||||
|             offset = pageable.offset, |             offset = pageable.offset, | ||||||
|             limit = pageable.pageSize.toLong(), |             limit = pageable.pageSize.toLong(), | ||||||
|             searchWord = searchWord |             searchWord = searchWord | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         val audioContentList = audioContentAndThemeList |         val audioContentList = audioContentAndThemeList | ||||||
|             .asSequence() |  | ||||||
|             .map { |             .map { | ||||||
|                 val tags = repository |                 val tags = repository | ||||||
|                     .getHashTagList(audioContentId = it.audioContentId) |                     .getHashTagList(audioContentId = it.audioContentId) | ||||||
| @@ -75,11 +75,6 @@ class AdminContentService( | |||||||
|                 ) |                 ) | ||||||
|                 it |                 it | ||||||
|             } |             } | ||||||
|             .map { |  | ||||||
|                 it.coverImageUrl = "$coverImageHost/${it.coverImageUrl}" |  | ||||||
|                 it |  | ||||||
|             } |  | ||||||
|             .toList() |  | ||||||
|  |  | ||||||
|         return GetAdminContentListResponse(totalCount, audioContentList) |         return GetAdminContentListResponse(totalCount, audioContentList) | ||||||
|     } |     } | ||||||
| @@ -90,10 +85,13 @@ class AdminContentService( | |||||||
|             ?: throw SodaException("없는 콘텐츠 입니다.") |             ?: throw SodaException("없는 콘텐츠 입니다.") | ||||||
|  |  | ||||||
|         if (request.isDefaultCoverImage) { |         if (request.isDefaultCoverImage) { | ||||||
|             audioContent.coverImage = "profile/default_profile.png" |             audioContent.coverImage = "`profile/default_profile.png`" | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (request.isActive != null) { |         if (request.isActive != null) { | ||||||
|  |             if (!request.isActive) { | ||||||
|  |                 audioContent.releaseDate = null | ||||||
|  |             } | ||||||
|             audioContent.isActive = request.isActive |             audioContent.isActive = request.isActive | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -117,5 +115,14 @@ class AdminContentService( | |||||||
|             val curation = curationRepository.findByIdAndActive(id = request.curationId) |             val curation = curationRepository.findByIdAndActive(id = request.curationId) | ||||||
|             audioContent.curation = curation |             audioContent.curation = curation | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         if (request.themeId != null) { | ||||||
|  |             val theme = themeRepository.findByIdAndActive(id = request.themeId) | ||||||
|  |             audioContent.theme = theme | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getContentMainTabList(): List<GetContentMainTabItem> { | ||||||
|  |         return contentMainTabRepository.findAllByActiveIsTrue() | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -16,11 +16,16 @@ data class GetAdminContentListItem @QueryProjection constructor( | |||||||
|     var coverImageUrl: String, |     var coverImageUrl: String, | ||||||
|     val creatorNickname: String, |     val creatorNickname: String, | ||||||
|     val theme: String, |     val theme: String, | ||||||
|  |     val themeId: Long, | ||||||
|     val price: Int, |     val price: Int, | ||||||
|  |     val totalContentCount: Int?, | ||||||
|  |     val remainingContentCount: Int?, | ||||||
|     val isAdult: Boolean, |     val isAdult: Boolean, | ||||||
|     val remainingTime: String, |     val remainingTime: String, | ||||||
|     var contentUrl: String, |     var contentUrl: String, | ||||||
|     val date: String |     val isCommentAvailable: Boolean, | ||||||
|  |     val date: String, | ||||||
|  |     val releaseDate: String? | ||||||
| ) { | ) { | ||||||
|     var tags: String = "" |     var tags: String = "" | ||||||
| } | } | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ data class UpdateAdminContentRequest( | |||||||
|     val title: String?, |     val title: String?, | ||||||
|     val detail: String?, |     val detail: String?, | ||||||
|     val curationId: Long?, |     val curationId: Long?, | ||||||
|  |     val themeId: Long?, | ||||||
|     val isAdult: Boolean?, |     val isAdult: Boolean?, | ||||||
|     val isActive: Boolean?, |     val isActive: Boolean?, | ||||||
|     val isCommentAvailable: Boolean? |     val isCommentAvailable: Boolean? | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ import org.springframework.web.bind.annotation.PostMapping | |||||||
| import org.springframework.web.bind.annotation.PutMapping | import org.springframework.web.bind.annotation.PutMapping | ||||||
| import org.springframework.web.bind.annotation.RequestBody | import org.springframework.web.bind.annotation.RequestBody | ||||||
| import org.springframework.web.bind.annotation.RequestMapping | import org.springframework.web.bind.annotation.RequestMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestParam | ||||||
| import org.springframework.web.bind.annotation.RequestPart | import org.springframework.web.bind.annotation.RequestPart | ||||||
| import org.springframework.web.bind.annotation.RestController | import org.springframework.web.bind.annotation.RestController | ||||||
| import org.springframework.web.multipart.MultipartFile | import org.springframework.web.multipart.MultipartFile | ||||||
| @@ -33,5 +34,7 @@ class AdminContentBannerController(private val service: AdminContentBannerServic | |||||||
|     ) = ApiResponse.ok(service.updateBannerOrders(request.ids), "수정되었습니다.") |     ) = ApiResponse.ok(service.updateBannerOrders(request.ids), "수정되었습니다.") | ||||||
|  |  | ||||||
|     @GetMapping |     @GetMapping | ||||||
|     fun getAudioContentMainBannerList() = ApiResponse.ok(service.getAudioContentMainBannerList()) |     fun getAudioContentMainBannerList( | ||||||
|  |         @RequestParam(value = "tabId", required = false) tabId: Long? = null | ||||||
|  |     ) = ApiResponse.ok(service.getAudioContentMainBannerList(tabId = tabId)) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -3,6 +3,8 @@ package kr.co.vividnext.sodalive.admin.content.banner | |||||||
| import com.querydsl.jpa.impl.JPAQueryFactory | import com.querydsl.jpa.impl.JPAQueryFactory | ||||||
| import kr.co.vividnext.sodalive.content.main.banner.AudioContentBanner | import kr.co.vividnext.sodalive.content.main.banner.AudioContentBanner | ||||||
| import kr.co.vividnext.sodalive.content.main.banner.QAudioContentBanner.audioContentBanner | import kr.co.vividnext.sodalive.content.main.banner.QAudioContentBanner.audioContentBanner | ||||||
|  | import kr.co.vividnext.sodalive.content.main.tab.QAudioContentMainTab.audioContentMainTab | ||||||
|  | import kr.co.vividnext.sodalive.creator.admin.content.series.QSeries.series | ||||||
| import kr.co.vividnext.sodalive.event.QEvent.event | import kr.co.vividnext.sodalive.event.QEvent.event | ||||||
| import kr.co.vividnext.sodalive.member.QMember.member | import kr.co.vividnext.sodalive.member.QMember.member | ||||||
| import org.springframework.beans.factory.annotation.Value | import org.springframework.beans.factory.annotation.Value | ||||||
| @@ -13,7 +15,7 @@ import org.springframework.stereotype.Repository | |||||||
| interface AdminContentBannerRepository : JpaRepository<AudioContentBanner, Long>, AdminContentBannerQueryRepository | interface AdminContentBannerRepository : JpaRepository<AudioContentBanner, Long>, AdminContentBannerQueryRepository | ||||||
|  |  | ||||||
| interface AdminContentBannerQueryRepository { | interface AdminContentBannerQueryRepository { | ||||||
|     fun getAudioContentMainBannerList(): List<GetAdminContentBannerResponse> |     fun getAudioContentMainBannerList(tabId: Long = 1): List<GetAdminContentBannerResponse> | ||||||
| } | } | ||||||
|  |  | ||||||
| class AdminContentBannerQueryRepositoryImpl( | class AdminContentBannerQueryRepositoryImpl( | ||||||
| @@ -21,17 +23,28 @@ class AdminContentBannerQueryRepositoryImpl( | |||||||
|     @Value("\${cloud.aws.cloud-front.host}") |     @Value("\${cloud.aws.cloud-front.host}") | ||||||
|     private val cloudFrontHost: String |     private val cloudFrontHost: String | ||||||
| ) : AdminContentBannerQueryRepository { | ) : AdminContentBannerQueryRepository { | ||||||
|     override fun getAudioContentMainBannerList(): List<GetAdminContentBannerResponse> { |     override fun getAudioContentMainBannerList(tabId: Long): List<GetAdminContentBannerResponse> { | ||||||
|  |         var where = audioContentBanner.isActive.isTrue | ||||||
|  |  | ||||||
|  |         where = if (tabId <= 1L) { | ||||||
|  |             where.and(audioContentMainTab.id.isNull) | ||||||
|  |         } else { | ||||||
|  |             where.and(audioContentMainTab.id.eq(tabId)) | ||||||
|  |         } | ||||||
|  |  | ||||||
|         return queryFactory |         return queryFactory | ||||||
|             .select( |             .select( | ||||||
|                 QGetAdminContentBannerResponse( |                 QGetAdminContentBannerResponse( | ||||||
|                     audioContentBanner.id, |                     audioContentBanner.id, | ||||||
|  |                     audioContentBanner.tab.id.coalesce(1), | ||||||
|                     audioContentBanner.type, |                     audioContentBanner.type, | ||||||
|                     audioContentBanner.thumbnailImage.prepend("/").prepend(cloudFrontHost), |                     audioContentBanner.thumbnailImage.prepend("/").prepend(cloudFrontHost), | ||||||
|                     audioContentBanner.event.id, |                     audioContentBanner.event.id, | ||||||
|                     audioContentBanner.event.thumbnailImage, |                     audioContentBanner.event.thumbnailImage, | ||||||
|                     audioContentBanner.creator.id, |                     audioContentBanner.creator.id, | ||||||
|                     audioContentBanner.creator.nickname, |                     audioContentBanner.creator.nickname, | ||||||
|  |                     audioContentBanner.series.id, | ||||||
|  |                     audioContentBanner.series.title, | ||||||
|                     audioContentBanner.link, |                     audioContentBanner.link, | ||||||
|                     audioContentBanner.isAdult |                     audioContentBanner.isAdult | ||||||
|                 ) |                 ) | ||||||
| @@ -39,7 +52,9 @@ class AdminContentBannerQueryRepositoryImpl( | |||||||
|             .from(audioContentBanner) |             .from(audioContentBanner) | ||||||
|             .leftJoin(audioContentBanner.event, event) |             .leftJoin(audioContentBanner.event, event) | ||||||
|             .leftJoin(audioContentBanner.creator, member) |             .leftJoin(audioContentBanner.creator, member) | ||||||
|             .where(audioContentBanner.isActive.isTrue) |             .leftJoin(audioContentBanner.series, series) | ||||||
|  |             .leftJoin(audioContentBanner.tab, audioContentMainTab) | ||||||
|  |             .where(where) | ||||||
|             .orderBy(audioContentBanner.orders.asc()) |             .orderBy(audioContentBanner.orders.asc()) | ||||||
|             .fetch() |             .fetch() | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,6 +1,8 @@ | |||||||
| package kr.co.vividnext.sodalive.admin.content.banner | package kr.co.vividnext.sodalive.admin.content.banner | ||||||
|  |  | ||||||
| import com.fasterxml.jackson.databind.ObjectMapper | import com.fasterxml.jackson.databind.ObjectMapper | ||||||
|  | import kr.co.vividnext.sodalive.admin.content.series.AdminContentSeriesRepository | ||||||
|  | import kr.co.vividnext.sodalive.admin.content.tab.AdminContentMainTabRepository | ||||||
| import kr.co.vividnext.sodalive.aws.s3.S3Uploader | import kr.co.vividnext.sodalive.aws.s3.S3Uploader | ||||||
| import kr.co.vividnext.sodalive.common.SodaException | import kr.co.vividnext.sodalive.common.SodaException | ||||||
| import kr.co.vividnext.sodalive.content.main.banner.AudioContentBanner | import kr.co.vividnext.sodalive.content.main.banner.AudioContentBanner | ||||||
| @@ -19,7 +21,9 @@ class AdminContentBannerService( | |||||||
|     private val s3Uploader: S3Uploader, |     private val s3Uploader: S3Uploader, | ||||||
|     private val repository: AdminContentBannerRepository, |     private val repository: AdminContentBannerRepository, | ||||||
|     private val memberRepository: MemberRepository, |     private val memberRepository: MemberRepository, | ||||||
|  |     private val seriesRepository: AdminContentSeriesRepository, | ||||||
|     private val eventRepository: EventRepository, |     private val eventRepository: EventRepository, | ||||||
|  |     private val contentMainTabRepository: AdminContentMainTabRepository, | ||||||
|     private val objectMapper: ObjectMapper, |     private val objectMapper: ObjectMapper, | ||||||
|  |  | ||||||
|     @Value("\${cloud.aws.s3.bucket}") |     @Value("\${cloud.aws.s3.bucket}") | ||||||
| @@ -32,6 +36,10 @@ class AdminContentBannerService( | |||||||
|             throw SodaException("크리에이터를 선택하세요.") |             throw SodaException("크리에이터를 선택하세요.") | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         if (request.type == AudioContentBannerType.SERIES && request.seriesId == null) { | ||||||
|  |             throw SodaException("시리즈를 선택하세요.") | ||||||
|  |         } | ||||||
|  |  | ||||||
|         if (request.type == AudioContentBannerType.LINK && request.link == null) { |         if (request.type == AudioContentBannerType.LINK && request.link == null) { | ||||||
|             throw SodaException("링크 url을 입력하세요.") |             throw SodaException("링크 url을 입력하세요.") | ||||||
|         } |         } | ||||||
| @@ -52,11 +60,25 @@ class AdminContentBannerService( | |||||||
|             null |             null | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         val series = if (request.seriesId != null && request.seriesId > 0) { | ||||||
|  |             seriesRepository.findByIdOrNull(request.seriesId) | ||||||
|  |         } else { | ||||||
|  |             null | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         val tab = if (request.tabId !== null) { | ||||||
|  |             contentMainTabRepository.findByIdOrNull(request.tabId) | ||||||
|  |         } else { | ||||||
|  |             null | ||||||
|  |         } | ||||||
|  |  | ||||||
|         val audioContentBanner = AudioContentBanner(type = request.type) |         val audioContentBanner = AudioContentBanner(type = request.type) | ||||||
|         audioContentBanner.link = request.link |         audioContentBanner.link = request.link | ||||||
|         audioContentBanner.isAdult = request.isAdult |         audioContentBanner.isAdult = request.isAdult | ||||||
|         audioContentBanner.event = event |         audioContentBanner.event = event | ||||||
|         audioContentBanner.creator = creator |         audioContentBanner.creator = creator | ||||||
|  |         audioContentBanner.series = series | ||||||
|  |         audioContentBanner.tab = tab | ||||||
|         repository.save(audioContentBanner) |         repository.save(audioContentBanner) | ||||||
|  |  | ||||||
|         val fileName = generateFileName() |         val fileName = generateFileName() | ||||||
| @@ -96,35 +118,57 @@ class AdminContentBannerService( | |||||||
|             audioContentBanner.creator = null |             audioContentBanner.creator = null | ||||||
|             audioContentBanner.event = null |             audioContentBanner.event = null | ||||||
|             audioContentBanner.link = null |             audioContentBanner.link = null | ||||||
|  |             audioContentBanner.series = null | ||||||
|  |  | ||||||
|             if (request.type == AudioContentBannerType.CREATOR) { |             when (request.type) { | ||||||
|                 if (request.creatorId != null) { |                 AudioContentBannerType.EVENT -> { | ||||||
|                     val creator = memberRepository.findByIdOrNull(request.creatorId) |                     if (request.eventId != null) { | ||||||
|                         ?: throw SodaException("크리에이터를 선택하세요.") |                         val event = eventRepository.findByIdOrNull(request.eventId) | ||||||
|  |                             ?: throw SodaException("이벤트를 선택하세요.") | ||||||
|  |  | ||||||
|                     audioContentBanner.creator = creator |                         audioContentBanner.event = event | ||||||
|                 } else { |                     } else { | ||||||
|                     throw SodaException("크리에이터를 선택하세요.") |                         throw SodaException("이벤트를 선택하세요.") | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|             } else if (request.type == AudioContentBannerType.LINK) { |  | ||||||
|                 if (request.link != null) { |  | ||||||
|                     audioContentBanner.link = request.link |  | ||||||
|                 } else { |  | ||||||
|                     throw SodaException("링크 url을 입력하세요.") |  | ||||||
|                 } |  | ||||||
|             } else if (request.type == AudioContentBannerType.EVENT) { |  | ||||||
|                 if (request.eventId != null) { |  | ||||||
|                     val event = eventRepository.findByIdOrNull(request.eventId) |  | ||||||
|                         ?: throw SodaException("이벤트를 선택하세요.") |  | ||||||
|  |  | ||||||
|                     audioContentBanner.event = event |                 AudioContentBannerType.CREATOR -> { | ||||||
|                 } else { |                     if (request.creatorId != null) { | ||||||
|                     throw SodaException("이벤트를 선택하세요.") |                         val creator = memberRepository.findByIdOrNull(request.creatorId) | ||||||
|  |                             ?: throw SodaException("크리에이터를 선택하세요.") | ||||||
|  |  | ||||||
|  |                         audioContentBanner.creator = creator | ||||||
|  |                     } else { | ||||||
|  |                         throw SodaException("크리에이터를 선택하세요.") | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 AudioContentBannerType.LINK -> { | ||||||
|  |                     if (request.link != null) { | ||||||
|  |                         audioContentBanner.link = request.link | ||||||
|  |                     } else { | ||||||
|  |                         throw SodaException("링크 url을 입력하세요.") | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 AudioContentBannerType.SERIES -> { | ||||||
|  |                     if (request.seriesId != null) { | ||||||
|  |                         val series = seriesRepository.findByIdOrNull(request.seriesId) | ||||||
|  |                             ?: throw SodaException("시리즈를 선택하세요.") | ||||||
|  |  | ||||||
|  |                         audioContentBanner.series = series | ||||||
|  |                     } else { | ||||||
|  |                         throw SodaException("시리즈를 선택하세요.") | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             audioContentBanner.type = request.type |             audioContentBanner.type = request.type | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         if (request.tabId !== null) { | ||||||
|  |             audioContentBanner.tab = contentMainTabRepository.findByIdOrNull(request.tabId) | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Transactional |     @Transactional | ||||||
| @@ -138,7 +182,7 @@ class AdminContentBannerService( | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun getAudioContentMainBannerList(): List<GetAdminContentBannerResponse> { |     fun getAudioContentMainBannerList(tabId: Long?): List<GetAdminContentBannerResponse> { | ||||||
|         return repository.getAudioContentMainBannerList() |         return repository.getAudioContentMainBannerList(tabId = tabId ?: 1) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -4,8 +4,10 @@ import kr.co.vividnext.sodalive.content.main.banner.AudioContentBannerType | |||||||
|  |  | ||||||
| data class CreateContentBannerRequest( | data class CreateContentBannerRequest( | ||||||
|     val type: AudioContentBannerType, |     val type: AudioContentBannerType, | ||||||
|  |     val tabId: Long?, | ||||||
|     val eventId: Long?, |     val eventId: Long?, | ||||||
|     val creatorId: Long?, |     val creatorId: Long?, | ||||||
|  |     val seriesId: Long?, | ||||||
|     val link: String?, |     val link: String?, | ||||||
|     val isAdult: Boolean |     val isAdult: Boolean | ||||||
| ) | ) | ||||||
|   | |||||||
| @@ -5,12 +5,15 @@ import kr.co.vividnext.sodalive.content.main.banner.AudioContentBannerType | |||||||
|  |  | ||||||
| data class GetAdminContentBannerResponse @QueryProjection constructor( | data class GetAdminContentBannerResponse @QueryProjection constructor( | ||||||
|     val id: Long, |     val id: Long, | ||||||
|  |     val tabId: Long?, | ||||||
|     val type: AudioContentBannerType, |     val type: AudioContentBannerType, | ||||||
|     val thumbnailImageUrl: String, |     val thumbnailImageUrl: String, | ||||||
|     val eventId: Long?, |     val eventId: Long?, | ||||||
|     val eventThumbnailImage: String?, |     val eventThumbnailImage: String?, | ||||||
|     val creatorId: Long?, |     val creatorId: Long?, | ||||||
|     val creatorNickname: String?, |     val creatorNickname: String?, | ||||||
|  |     val seriesId: Long?, | ||||||
|  |     val seriesTitle: String?, | ||||||
|     val link: String?, |     val link: String?, | ||||||
|     val isAdult: Boolean |     val isAdult: Boolean | ||||||
| ) | ) | ||||||
|   | |||||||
| @@ -5,8 +5,10 @@ import kr.co.vividnext.sodalive.content.main.banner.AudioContentBannerType | |||||||
| data class UpdateContentBannerRequest( | data class UpdateContentBannerRequest( | ||||||
|     val id: Long, |     val id: Long, | ||||||
|     val type: AudioContentBannerType?, |     val type: AudioContentBannerType?, | ||||||
|  |     val tabId: Long?, | ||||||
|     val eventId: Long?, |     val eventId: Long?, | ||||||
|     val creatorId: Long?, |     val creatorId: Long?, | ||||||
|  |     val seriesId: Long?, | ||||||
|     val link: String?, |     val link: String?, | ||||||
|     val isAdult: Boolean?, |     val isAdult: Boolean?, | ||||||
|     val isActive: Boolean? |     val isActive: Boolean? | ||||||
|   | |||||||
| @@ -0,0 +1,6 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.content.curation | ||||||
|  |  | ||||||
|  | data class AddItemToCurationRequest( | ||||||
|  |     val curationId: Long, | ||||||
|  |     val itemIdList: List<Long> | ||||||
|  | ) | ||||||
| @@ -7,6 +7,7 @@ import org.springframework.web.bind.annotation.PostMapping | |||||||
| import org.springframework.web.bind.annotation.PutMapping | import org.springframework.web.bind.annotation.PutMapping | ||||||
| import org.springframework.web.bind.annotation.RequestBody | import org.springframework.web.bind.annotation.RequestBody | ||||||
| import org.springframework.web.bind.annotation.RequestMapping | import org.springframework.web.bind.annotation.RequestMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestParam | ||||||
| import org.springframework.web.bind.annotation.RestController | import org.springframework.web.bind.annotation.RestController | ||||||
|  |  | ||||||
| @RestController | @RestController | ||||||
| @@ -29,5 +30,39 @@ class AdminContentCurationController(private val service: AdminContentCurationSe | |||||||
|     ) = ApiResponse.ok(service.updateContentCurationOrders(request.ids), "수정되었습니다.") |     ) = ApiResponse.ok(service.updateContentCurationOrders(request.ids), "수정되었습니다.") | ||||||
|  |  | ||||||
|     @GetMapping |     @GetMapping | ||||||
|     fun getContentCurationList() = ApiResponse.ok(service.getContentCurationList()) |     fun getContentCurationList( | ||||||
|  |         @RequestParam tabId: Long | ||||||
|  |     ) = ApiResponse.ok(service.getContentCurationList(tabId = tabId)) | ||||||
|  |  | ||||||
|  |     @GetMapping("/items") | ||||||
|  |     fun getCurationItems( | ||||||
|  |         @RequestParam curationId: Long | ||||||
|  |     ) = ApiResponse.ok(service.getCurationItem(curationId = curationId)) | ||||||
|  |  | ||||||
|  |     @GetMapping("/search/content") | ||||||
|  |     fun searchCurationContentItem( | ||||||
|  |         @RequestParam curationId: Long, | ||||||
|  |         @RequestParam searchWord: String | ||||||
|  |     ) = ApiResponse.ok(service.searchCurationContentItem(curationId, searchWord)) | ||||||
|  |  | ||||||
|  |     @GetMapping("/search/series") | ||||||
|  |     fun searchCurationSeriesItem( | ||||||
|  |         @RequestParam curationId: Long, | ||||||
|  |         @RequestParam searchWord: String | ||||||
|  |     ) = ApiResponse.ok(service.searchCurationSeriesItem(curationId, searchWord)) | ||||||
|  |  | ||||||
|  |     @PostMapping("/add/item") | ||||||
|  |     fun addItemToCuration( | ||||||
|  |         @RequestBody request: AddItemToCurationRequest | ||||||
|  |     ) = ApiResponse.ok(service.addItemToCuration(request), "큐레이션 아이템을 등록했습니다.") | ||||||
|  |  | ||||||
|  |     @PutMapping("/remove/item") | ||||||
|  |     fun removeItemInCuration( | ||||||
|  |         @RequestBody request: RemoveItemInCurationRequest | ||||||
|  |     ) = ApiResponse.ok(service.removeItemInCuration(request), "큐레이션 아이템을 제거했습니다.") | ||||||
|  |  | ||||||
|  |     @PutMapping("/orders/item") | ||||||
|  |     fun updateItemInCurationOrders( | ||||||
|  |         @RequestBody request: UpdateCurationItemOrdersRequest | ||||||
|  |     ) = ApiResponse.ok(service.updateItemInCurationOrders(request), "수정되었습니다.") | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,106 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.content.curation | ||||||
|  |  | ||||||
|  | import com.querydsl.jpa.impl.JPAQueryFactory | ||||||
|  | import kr.co.vividnext.sodalive.content.QAudioContent.audioContent | ||||||
|  | import kr.co.vividnext.sodalive.content.main.curation.AudioContentCurationItem | ||||||
|  | import kr.co.vividnext.sodalive.content.main.curation.QAudioContentCuration.audioContentCuration | ||||||
|  | import kr.co.vividnext.sodalive.content.main.curation.QAudioContentCurationItem.audioContentCurationItem | ||||||
|  | import kr.co.vividnext.sodalive.creator.admin.content.series.QSeries.series | ||||||
|  | import org.springframework.beans.factory.annotation.Value | ||||||
|  | import org.springframework.data.jpa.repository.JpaRepository | ||||||
|  |  | ||||||
|  | interface AdminContentCurationItemRepository : | ||||||
|  |     JpaRepository<AudioContentCurationItem, Long>, | ||||||
|  |     AdminContentCurationItemQueryRepository | ||||||
|  |  | ||||||
|  | interface AdminContentCurationItemQueryRepository { | ||||||
|  |     fun findByCurationIdAndSeriesId(curationId: Long, seriesId: Long?): AudioContentCurationItem? | ||||||
|  |     fun findByCurationIdAndContentId(curationId: Long, contentId: Long?): AudioContentCurationItem? | ||||||
|  |     fun findByCurationIdAndItemId(curationId: Long, itemId: Long): AudioContentCurationItem? | ||||||
|  |     fun getAudioContentCurationItemList(curationId: Long): List<GetCurationItemResponse> | ||||||
|  |     fun getAudioContentCurationSeriesItemList(curationId: Long): List<GetCurationItemResponse> | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class AdminContentCurationItemQueryRepositoryImpl( | ||||||
|  |     val queryFactory: JPAQueryFactory, | ||||||
|  |  | ||||||
|  |     @Value("\${cloud.aws.cloud-front.host}") | ||||||
|  |     private val imageHost: String | ||||||
|  | ) : AdminContentCurationItemQueryRepository { | ||||||
|  |     override fun findByCurationIdAndSeriesId(curationId: Long, seriesId: Long?): AudioContentCurationItem? { | ||||||
|  |         return queryFactory | ||||||
|  |             .selectFrom(audioContentCurationItem) | ||||||
|  |             .innerJoin(audioContentCurationItem.curation, audioContentCuration) | ||||||
|  |             .innerJoin(audioContentCurationItem.series, series) | ||||||
|  |             .where( | ||||||
|  |                 audioContentCurationItem.curation.id.eq(curationId), | ||||||
|  |                 audioContentCurationItem.series.id.eq(seriesId) | ||||||
|  |             ) | ||||||
|  |             .fetchFirst() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun findByCurationIdAndContentId(curationId: Long, contentId: Long?): AudioContentCurationItem? { | ||||||
|  |         return queryFactory | ||||||
|  |             .selectFrom(audioContentCurationItem) | ||||||
|  |             .innerJoin(audioContentCurationItem.curation, audioContentCuration) | ||||||
|  |             .innerJoin(audioContentCurationItem.content, audioContent) | ||||||
|  |             .where( | ||||||
|  |                 audioContentCurationItem.curation.id.eq(curationId), | ||||||
|  |                 audioContentCurationItem.content.id.eq(contentId) | ||||||
|  |             ) | ||||||
|  |             .fetchFirst() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun findByCurationIdAndItemId(curationId: Long, itemId: Long): AudioContentCurationItem? { | ||||||
|  |         return queryFactory.selectFrom(audioContentCurationItem) | ||||||
|  |             .innerJoin(audioContentCurationItem.curation, audioContentCuration) | ||||||
|  |             .where(audioContentCuration.id.eq(curationId), audioContentCurationItem.id.eq(itemId)) | ||||||
|  |             .fetchFirst() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun getAudioContentCurationItemList(curationId: Long): List<GetCurationItemResponse> { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetCurationItemResponse( | ||||||
|  |                     audioContentCurationItem.id, | ||||||
|  |                     audioContent.title, | ||||||
|  |                     audioContent.detail, | ||||||
|  |                     audioContent.coverImage.prepend("/").prepend(imageHost), | ||||||
|  |                     audioContent.member.nickname.coalesce(""), | ||||||
|  |                     audioContent.isAdult | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(audioContentCurationItem) | ||||||
|  |             .innerJoin(audioContentCurationItem.curation, audioContentCuration) | ||||||
|  |             .innerJoin(audioContentCurationItem.content, audioContent) | ||||||
|  |             .where( | ||||||
|  |                 audioContentCuration.id.eq(curationId), | ||||||
|  |                 audioContentCurationItem.isActive.isTrue | ||||||
|  |             ) | ||||||
|  |             .orderBy(audioContentCurationItem.orders.asc()) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun getAudioContentCurationSeriesItemList(curationId: Long): List<GetCurationItemResponse> { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetCurationItemResponse( | ||||||
|  |                     audioContentCurationItem.id, | ||||||
|  |                     series.title, | ||||||
|  |                     series.introduction, | ||||||
|  |                     series.coverImage.prepend("/").prepend(imageHost), | ||||||
|  |                     series.member.nickname.coalesce(""), | ||||||
|  |                     series.isAdult | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(audioContentCurationItem) | ||||||
|  |             .innerJoin(audioContentCurationItem.curation, audioContentCuration) | ||||||
|  |             .innerJoin(audioContentCurationItem.series, series) | ||||||
|  |             .where( | ||||||
|  |                 audioContentCuration.id.eq(curationId), | ||||||
|  |                 audioContentCurationItem.isActive.isTrue | ||||||
|  |             ) | ||||||
|  |             .orderBy(audioContentCurationItem.orders.asc()) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,8 +1,13 @@ | |||||||
| package kr.co.vividnext.sodalive.admin.content.curation | package kr.co.vividnext.sodalive.admin.content.curation | ||||||
|  |  | ||||||
| import com.querydsl.jpa.impl.JPAQueryFactory | import com.querydsl.jpa.impl.JPAQueryFactory | ||||||
|  | import kr.co.vividnext.sodalive.content.QAudioContent.audioContent | ||||||
| import kr.co.vividnext.sodalive.content.main.curation.AudioContentCuration | import kr.co.vividnext.sodalive.content.main.curation.AudioContentCuration | ||||||
| import kr.co.vividnext.sodalive.content.main.curation.QAudioContentCuration.audioContentCuration | import kr.co.vividnext.sodalive.content.main.curation.QAudioContentCuration.audioContentCuration | ||||||
|  | import kr.co.vividnext.sodalive.content.main.curation.QAudioContentCurationItem.audioContentCurationItem | ||||||
|  | import kr.co.vividnext.sodalive.content.main.tab.QAudioContentMainTab.audioContentMainTab | ||||||
|  | import kr.co.vividnext.sodalive.creator.admin.content.series.QSeries.series | ||||||
|  | import org.springframework.beans.factory.annotation.Value | ||||||
| import org.springframework.data.jpa.repository.JpaRepository | import org.springframework.data.jpa.repository.JpaRepository | ||||||
| import org.springframework.stereotype.Repository | import org.springframework.stereotype.Repository | ||||||
|  |  | ||||||
| @@ -12,26 +17,37 @@ interface AdminContentCurationRepository : | |||||||
|     AdminContentCurationQueryRepository |     AdminContentCurationQueryRepository | ||||||
|  |  | ||||||
| interface AdminContentCurationQueryRepository { | interface AdminContentCurationQueryRepository { | ||||||
|     fun getAudioContentCurationList(): List<GetAdminContentCurationResponse> |     fun getAudioContentCurationList(tabId: Long): List<GetAdminContentCurationResponse> | ||||||
|     fun findByIdAndActive(id: Long): AudioContentCuration? |     fun findByIdAndActive(id: Long): AudioContentCuration? | ||||||
|  |     fun searchCurationContentItem(curationId: Long, searchWord: String): List<SearchCurationItemResponse> | ||||||
|  |     fun searchCurationSeriesItem(curationId: Long, searchWord: String): List<SearchCurationItemResponse> | ||||||
| } | } | ||||||
|  |  | ||||||
| @Repository | @Repository | ||||||
| class AdminContentCurationQueryRepositoryImpl( | class AdminContentCurationQueryRepositoryImpl( | ||||||
|     private val queryFactory: JPAQueryFactory |     private val queryFactory: JPAQueryFactory, | ||||||
|  |  | ||||||
|  |     @Value("\${cloud.aws.cloud-front.host}") | ||||||
|  |     private val imageHost: String | ||||||
| ) : AdminContentCurationQueryRepository { | ) : AdminContentCurationQueryRepository { | ||||||
|     override fun getAudioContentCurationList(): List<GetAdminContentCurationResponse> { |     override fun getAudioContentCurationList(tabId: Long): List<GetAdminContentCurationResponse> { | ||||||
|         return queryFactory |         return queryFactory | ||||||
|             .select( |             .select( | ||||||
|                 QGetAdminContentCurationResponse( |                 QGetAdminContentCurationResponse( | ||||||
|                     audioContentCuration.id, |                     audioContentCuration.id, | ||||||
|  |                     audioContentMainTab.id, | ||||||
|                     audioContentCuration.title, |                     audioContentCuration.title, | ||||||
|                     audioContentCuration.description, |                     audioContentCuration.description, | ||||||
|                     audioContentCuration.isAdult |                     audioContentCuration.isAdult, | ||||||
|  |                     audioContentCuration.isSeries | ||||||
|                 ) |                 ) | ||||||
|             ) |             ) | ||||||
|             .from(audioContentCuration) |             .from(audioContentCuration) | ||||||
|             .where(audioContentCuration.isActive.isTrue) |             .innerJoin(audioContentCuration.tab, audioContentMainTab) | ||||||
|  |             .where( | ||||||
|  |                 audioContentCuration.isActive.isTrue, | ||||||
|  |                 audioContentMainTab.id.eq(tabId) | ||||||
|  |             ) | ||||||
|             .orderBy(audioContentCuration.orders.asc()) |             .orderBy(audioContentCuration.orders.asc()) | ||||||
|             .fetch() |             .fetch() | ||||||
|     } |     } | ||||||
| @@ -45,4 +61,62 @@ class AdminContentCurationQueryRepositoryImpl( | |||||||
|             ) |             ) | ||||||
|             .fetchFirst() |             .fetchFirst() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     override fun searchCurationContentItem( | ||||||
|  |         curationId: Long, | ||||||
|  |         searchWord: String | ||||||
|  |     ): List<SearchCurationItemResponse> { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QSearchCurationItemResponse( | ||||||
|  |                     audioContent.id, | ||||||
|  |                     audioContent.title, | ||||||
|  |                     audioContent.coverImage.prepend("/").prepend(imageHost) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(audioContent) | ||||||
|  |             .leftJoin(audioContentCurationItem) | ||||||
|  |             .on( | ||||||
|  |                 audioContent.id.eq(audioContentCurationItem.content.id) | ||||||
|  |                     .and(audioContentCurationItem.curation.id.eq(curationId)) | ||||||
|  |             ) | ||||||
|  |             .where( | ||||||
|  |                 audioContent.duration.isNotNull | ||||||
|  |                     .and(audioContent.member.isNotNull) | ||||||
|  |                     .and(audioContent.isActive.isTrue) | ||||||
|  |                     .and(audioContent.title.contains(searchWord)) | ||||||
|  |                     .and(audioContentCurationItem.id.isNull) | ||||||
|  |             ) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun searchCurationSeriesItem( | ||||||
|  |         curationId: Long, | ||||||
|  |         searchWord: String | ||||||
|  |     ): List<SearchCurationItemResponse> { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QSearchCurationItemResponse( | ||||||
|  |                     series.id, | ||||||
|  |                     series.title, | ||||||
|  |                     series.coverImage.prepend("/").prepend(imageHost) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(series) | ||||||
|  |             .leftJoin(audioContentCurationItem) | ||||||
|  |             .on( | ||||||
|  |                 series.id.eq(audioContentCurationItem.series.id) | ||||||
|  |                     .and(audioContentCurationItem.curation.id.eq(curationId)) | ||||||
|  |             ) | ||||||
|  |             .where( | ||||||
|  |                 series.isActive.isTrue | ||||||
|  |                     .and(series.member.isNotNull) | ||||||
|  |                     .and(series.title.contains(searchWord)) | ||||||
|  |                     .and( | ||||||
|  |                         audioContentCurationItem.id.isNull | ||||||
|  |                             .or(audioContentCurationItem.isActive.isFalse) | ||||||
|  |                     ) | ||||||
|  |             ) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,24 +1,37 @@ | |||||||
| package kr.co.vividnext.sodalive.admin.content.curation | package kr.co.vividnext.sodalive.admin.content.curation | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.admin.content.AdminContentRepository | ||||||
|  | import kr.co.vividnext.sodalive.admin.content.series.AdminContentSeriesRepository | ||||||
|  | import kr.co.vividnext.sodalive.admin.content.tab.AdminContentMainTabRepository | ||||||
| import kr.co.vividnext.sodalive.common.SodaException | import kr.co.vividnext.sodalive.common.SodaException | ||||||
| import kr.co.vividnext.sodalive.content.main.curation.AudioContentCuration | import kr.co.vividnext.sodalive.content.main.curation.AudioContentCuration | ||||||
|  | import kr.co.vividnext.sodalive.content.main.curation.AudioContentCurationItem | ||||||
| import org.springframework.data.repository.findByIdOrNull | import org.springframework.data.repository.findByIdOrNull | ||||||
| import org.springframework.stereotype.Service | import org.springframework.stereotype.Service | ||||||
| import org.springframework.transaction.annotation.Transactional | import org.springframework.transaction.annotation.Transactional | ||||||
|  |  | ||||||
| @Service | @Service | ||||||
| class AdminContentCurationService( | class AdminContentCurationService( | ||||||
|     private val repository: AdminContentCurationRepository |     private val repository: AdminContentCurationRepository, | ||||||
|  |     private val contentMainTabRepository: AdminContentMainTabRepository, | ||||||
|  |     private val seriesRepository: AdminContentSeriesRepository, | ||||||
|  |     private val contentRepository: AdminContentRepository, | ||||||
|  |     private val contentCurationItemRepository: AdminContentCurationItemRepository | ||||||
| ) { | ) { | ||||||
|     @Transactional |     @Transactional | ||||||
|     fun createContentCuration(request: CreateContentCurationRequest) { |     fun createContentCuration(request: CreateContentCurationRequest) { | ||||||
|         repository.save( |         val tab = contentMainTabRepository.findByIdOrNull(request.tabId) | ||||||
|             AudioContentCuration( |             ?: throw SodaException("잘못된 요청입니다.") | ||||||
|                 title = request.title, |  | ||||||
|                 description = request.description, |         val curation = AudioContentCuration( | ||||||
|                 isAdult = request.isAdult |             title = request.title, | ||||||
|             ) |             description = request.description, | ||||||
|  |             isAdult = request.isAdult, | ||||||
|  |             isSeries = request.isSeries | ||||||
|         ) |         ) | ||||||
|  |         curation.tab = tab | ||||||
|  |  | ||||||
|  |         repository.save(curation) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Transactional |     @Transactional | ||||||
| @@ -41,6 +54,18 @@ class AdminContentCurationService( | |||||||
|         if (request.isActive != null) { |         if (request.isActive != null) { | ||||||
|             audioContentCuration.isActive = request.isActive |             audioContentCuration.isActive = request.isActive | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         if (request.isSeries != null) { | ||||||
|  |             audioContentCuration.isSeries = request.isSeries | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (request.tabId != null) { | ||||||
|  |             val tab = contentMainTabRepository.findByIdOrNull(request.tabId) | ||||||
|  |  | ||||||
|  |             if (tab != null) { | ||||||
|  |                 audioContentCuration.tab = tab | ||||||
|  |             } | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Transactional |     @Transactional | ||||||
| @@ -54,7 +79,90 @@ class AdminContentCurationService( | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun getContentCurationList(): List<GetAdminContentCurationResponse> { |     fun getContentCurationList(tabId: Long): List<GetAdminContentCurationResponse> { | ||||||
|         return repository.getAudioContentCurationList() |         return repository.getAudioContentCurationList(tabId = tabId) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCurationItem(curationId: Long): List<GetCurationItemResponse> { | ||||||
|  |         val curation = repository.findByIdOrNull(curationId) | ||||||
|  |             ?: throw SodaException("잘못된 요청입니다.") | ||||||
|  |  | ||||||
|  |         return if (curation.isSeries) { | ||||||
|  |             contentCurationItemRepository.getAudioContentCurationSeriesItemList(curationId) | ||||||
|  |         } else { | ||||||
|  |             contentCurationItemRepository.getAudioContentCurationItemList(curationId) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun searchCurationContentItem(curationId: Long, searchWord: String): List<SearchCurationItemResponse> { | ||||||
|  |         return repository.searchCurationContentItem(curationId, searchWord) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun searchCurationSeriesItem(curationId: Long, searchWord: String): List<SearchCurationItemResponse> { | ||||||
|  |         return repository.searchCurationSeriesItem(curationId, searchWord) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional | ||||||
|  |     fun addItemToCuration(request: AddItemToCurationRequest) { | ||||||
|  |         // 큐레이션 조회 | ||||||
|  |         val audioContentCuration = repository.findByIdOrNull(id = request.curationId) | ||||||
|  |             ?: throw SodaException("잘못된 요청입니다.") | ||||||
|  |  | ||||||
|  |         if (audioContentCuration.isSeries) { | ||||||
|  |             request.itemIdList.forEach { seriesId -> | ||||||
|  |                 val series = seriesRepository.findByIdAndActiveTrue(seriesId) | ||||||
|  |  | ||||||
|  |                 if (series != null) { | ||||||
|  |                     val item = contentCurationItemRepository.findByCurationIdAndSeriesId( | ||||||
|  |                         curationId = request.curationId, | ||||||
|  |                         seriesId = series.id | ||||||
|  |                     ) ?: AudioContentCurationItem() | ||||||
|  |                     item.curation = audioContentCuration | ||||||
|  |                     item.series = series | ||||||
|  |                     item.isActive = true | ||||||
|  |                     contentCurationItemRepository.save(item) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             request.itemIdList.forEach { contentId -> | ||||||
|  |                 val audioContent = contentRepository.findByIdAndActiveTrue(contentId) | ||||||
|  |  | ||||||
|  |                 if (audioContent != null) { | ||||||
|  |                     val item = contentCurationItemRepository.findByCurationIdAndContentId( | ||||||
|  |                         curationId = request.curationId, | ||||||
|  |                         contentId = audioContent.id | ||||||
|  |                     ) ?: AudioContentCurationItem() | ||||||
|  |                     item.curation = audioContentCuration | ||||||
|  |                     item.content = audioContent | ||||||
|  |                     item.isActive = true | ||||||
|  |                     contentCurationItemRepository.save(item) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional | ||||||
|  |     fun removeItemInCuration(request: RemoveItemInCurationRequest) { | ||||||
|  |         val audioContentCurationItem = contentCurationItemRepository.findByCurationIdAndItemId( | ||||||
|  |             curationId = request.curationId, | ||||||
|  |             itemId = request.itemId | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         audioContentCurationItem?.isActive = false | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Transactional | ||||||
|  |     fun updateItemInCurationOrders(request: UpdateCurationItemOrdersRequest) { | ||||||
|  |         val ids = request.itemIds | ||||||
|  |         for (index in ids.indices) { | ||||||
|  |             val item = contentCurationItemRepository.findByCurationIdAndItemId( | ||||||
|  |                 curationId = request.curationId, | ||||||
|  |                 itemId = ids[index] | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |             if (item != null) { | ||||||
|  |                 item.orders = index + 1 | ||||||
|  |             } | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,16 +1,20 @@ | |||||||
| package kr.co.vividnext.sodalive.admin.content.curation | package kr.co.vividnext.sodalive.admin.content.curation | ||||||
|  |  | ||||||
| data class CreateContentCurationRequest( | data class CreateContentCurationRequest( | ||||||
|  |     val tabId: Long, | ||||||
|     val title: String, |     val title: String, | ||||||
|     val description: String, |     val description: String, | ||||||
|     val isAdult: Boolean |     val isAdult: Boolean, | ||||||
|  |     val isSeries: Boolean | ||||||
| ) | ) | ||||||
|  |  | ||||||
| data class UpdateContentCurationRequest( | data class UpdateContentCurationRequest( | ||||||
|     val id: Long, |     val id: Long, | ||||||
|  |     val tabId: Long?, | ||||||
|     val title: String?, |     val title: String?, | ||||||
|     val description: String?, |     val description: String?, | ||||||
|     val isAdult: Boolean?, |     val isAdult: Boolean?, | ||||||
|  |     val isSeries: Boolean?, | ||||||
|     val isActive: Boolean? |     val isActive: Boolean? | ||||||
| ) | ) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -4,7 +4,9 @@ import com.querydsl.core.annotations.QueryProjection | |||||||
|  |  | ||||||
| data class GetAdminContentCurationResponse @QueryProjection constructor( | data class GetAdminContentCurationResponse @QueryProjection constructor( | ||||||
|     val id: Long, |     val id: Long, | ||||||
|  |     val tabId: Long, | ||||||
|     val title: String, |     val title: String, | ||||||
|     val description: String, |     val description: String, | ||||||
|     val isAdult: Boolean |     val isAdult: Boolean, | ||||||
|  |     val isSeries: Boolean | ||||||
| ) | ) | ||||||
|   | |||||||
| @@ -0,0 +1,12 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.content.curation | ||||||
|  |  | ||||||
|  | import com.querydsl.core.annotations.QueryProjection | ||||||
|  |  | ||||||
|  | data class GetCurationItemResponse @QueryProjection constructor( | ||||||
|  |     val id: Long, | ||||||
|  |     val title: String, | ||||||
|  |     val desc: String, | ||||||
|  |     val coverImageUrl: String, | ||||||
|  |     val creatorNickname: String, | ||||||
|  |     val isAdult: Boolean | ||||||
|  | ) | ||||||
| @@ -0,0 +1,6 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.content.curation | ||||||
|  |  | ||||||
|  | data class RemoveItemInCurationRequest( | ||||||
|  |     val curationId: Long, | ||||||
|  |     val itemId: Long | ||||||
|  | ) | ||||||
| @@ -0,0 +1,9 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.content.curation | ||||||
|  |  | ||||||
|  | import com.querydsl.core.annotations.QueryProjection | ||||||
|  |  | ||||||
|  | data class SearchCurationItemResponse @QueryProjection constructor( | ||||||
|  |     val id: Long, | ||||||
|  |     val title: String, | ||||||
|  |     val coverImageUrl: String | ||||||
|  | ) | ||||||
| @@ -0,0 +1,6 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.content.curation | ||||||
|  |  | ||||||
|  | data class UpdateCurationItemOrdersRequest( | ||||||
|  |     val curationId: Long, | ||||||
|  |     val itemIds: List<Long> | ||||||
|  | ) | ||||||
| @@ -0,0 +1,72 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.content.curation.tag | ||||||
|  |  | ||||||
|  | import com.querydsl.jpa.impl.JPAQueryFactory | ||||||
|  | import kr.co.vividnext.sodalive.content.QAudioContent.audioContent | ||||||
|  | import kr.co.vividnext.sodalive.content.main.curation.tag.ContentHashTagCurationItem | ||||||
|  | import kr.co.vividnext.sodalive.content.main.curation.tag.QContentHashTagCuration.contentHashTagCuration | ||||||
|  | import kr.co.vividnext.sodalive.content.main.curation.tag.QContentHashTagCurationItem.contentHashTagCurationItem | ||||||
|  | import org.springframework.beans.factory.annotation.Value | ||||||
|  | import org.springframework.data.jpa.repository.JpaRepository | ||||||
|  |  | ||||||
|  | interface AdminContentHashTagCurationItemRepository : | ||||||
|  |     JpaRepository<ContentHashTagCurationItem, Long>, | ||||||
|  |     AdminContentHashTagCurationItemQueryRepository | ||||||
|  |  | ||||||
|  | interface AdminContentHashTagCurationItemQueryRepository { | ||||||
|  |     fun getContentHashTagCurationItemList(curationId: Long): List<GetAdminHashTagCurationItemResponse> | ||||||
|  |     fun findByCurationIdAndContentId(curationId: Long, contentId: Long?): ContentHashTagCurationItem? | ||||||
|  |     fun findByCurationIdAndItemId(curationId: Long, itemId: Long): ContentHashTagCurationItem? | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class AdminContentHashTagCurationItemQueryRepositoryImpl( | ||||||
|  |     val queryFactory: JPAQueryFactory, | ||||||
|  |  | ||||||
|  |     @Value("\${cloud.aws.cloud-front.host}") | ||||||
|  |     private val imageHost: String | ||||||
|  | ) : AdminContentHashTagCurationItemQueryRepository { | ||||||
|  |     override fun getContentHashTagCurationItemList(curationId: Long): List<GetAdminHashTagCurationItemResponse> { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetAdminHashTagCurationItemResponse( | ||||||
|  |                     contentHashTagCurationItem.id, | ||||||
|  |                     audioContent.title, | ||||||
|  |                     audioContent.detail, | ||||||
|  |                     audioContent.coverImage.prepend("/").prepend(imageHost), | ||||||
|  |                     audioContent.member.nickname.coalesce(""), | ||||||
|  |                     audioContent.isAdult | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(contentHashTagCurationItem) | ||||||
|  |             .innerJoin(contentHashTagCurationItem.curation, contentHashTagCuration) | ||||||
|  |             .innerJoin(contentHashTagCurationItem.content, audioContent) | ||||||
|  |             .where( | ||||||
|  |                 contentHashTagCuration.id.eq(curationId), | ||||||
|  |                 contentHashTagCurationItem.isActive.isTrue, | ||||||
|  |                 audioContent.isActive.isTrue | ||||||
|  |             ) | ||||||
|  |             .orderBy(contentHashTagCurationItem.orders.asc()) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun findByCurationIdAndContentId(curationId: Long, contentId: Long?): ContentHashTagCurationItem? { | ||||||
|  |         return queryFactory | ||||||
|  |             .selectFrom(contentHashTagCurationItem) | ||||||
|  |             .innerJoin(contentHashTagCurationItem.curation, contentHashTagCuration) | ||||||
|  |             .innerJoin(contentHashTagCurationItem.content, audioContent) | ||||||
|  |             .where( | ||||||
|  |                 contentHashTagCuration.id.eq(curationId), | ||||||
|  |                 audioContent.id.eq(contentId) | ||||||
|  |             ) | ||||||
|  |             .fetchFirst() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun findByCurationIdAndItemId(curationId: Long, itemId: Long): ContentHashTagCurationItem? { | ||||||
|  |         return queryFactory.selectFrom(contentHashTagCurationItem) | ||||||
|  |             .innerJoin(contentHashTagCurationItem.curation, contentHashTagCuration) | ||||||
|  |             .where( | ||||||
|  |                 contentHashTagCuration.id.eq(curationId), | ||||||
|  |                 contentHashTagCurationItem.id.eq(itemId) | ||||||
|  |             ) | ||||||
|  |             .fetchFirst() | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,63 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.content.curation.tag | ||||||
|  |  | ||||||
|  | import kr.co.vividnext.sodalive.admin.content.curation.AddItemToCurationRequest | ||||||
|  | import kr.co.vividnext.sodalive.admin.content.curation.RemoveItemInCurationRequest | ||||||
|  | import kr.co.vividnext.sodalive.admin.content.curation.UpdateCurationItemOrdersRequest | ||||||
|  | import kr.co.vividnext.sodalive.common.ApiResponse | ||||||
|  | import org.springframework.security.access.prepost.PreAuthorize | ||||||
|  | import org.springframework.web.bind.annotation.GetMapping | ||||||
|  | import org.springframework.web.bind.annotation.PostMapping | ||||||
|  | import org.springframework.web.bind.annotation.PutMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestBody | ||||||
|  | import org.springframework.web.bind.annotation.RequestMapping | ||||||
|  | import org.springframework.web.bind.annotation.RequestParam | ||||||
|  | import org.springframework.web.bind.annotation.RestController | ||||||
|  |  | ||||||
|  | @RestController | ||||||
|  | @RequestMapping("/admin/audio-content/tag/curation") | ||||||
|  | @PreAuthorize("hasRole('ADMIN')") | ||||||
|  | class AdminHashTagCurationController(private val service: AdminHashTagCurationService) { | ||||||
|  |     @GetMapping | ||||||
|  |     fun getContentHashTagCurationList() = ApiResponse.ok(service.getContentHashTagCurationList()) | ||||||
|  |  | ||||||
|  |     @PostMapping | ||||||
|  |     fun createContentHashTagCuration( | ||||||
|  |         @RequestBody request: CreateContentHashTagCurationRequest | ||||||
|  |     ) = ApiResponse.ok(service.createContentHashTagCuration(request)) | ||||||
|  |  | ||||||
|  |     @PutMapping | ||||||
|  |     fun updateContentHashTagCuration( | ||||||
|  |         @RequestBody request: UpdateContentHashTagCurationRequest | ||||||
|  |     ) = ApiResponse.ok(service.updateContentHashTagCuration(request)) | ||||||
|  |  | ||||||
|  |     @PutMapping("/orders") | ||||||
|  |     fun updateContentHashTagCurationOrders( | ||||||
|  |         @RequestBody request: UpdateContentHashTagCurationOrderRequest | ||||||
|  |     ) = ApiResponse.ok(service.updateContentHashTagCurationOrders(request.ids), "수정되었습니다.") | ||||||
|  |  | ||||||
|  |     @GetMapping("/items") | ||||||
|  |     fun getHashTagCurationItemList( | ||||||
|  |         @RequestParam curationId: Long | ||||||
|  |     ) = ApiResponse.ok(service.getHashTagCurationItemList(curationId = curationId)) | ||||||
|  |  | ||||||
|  |     @GetMapping("/search/content") | ||||||
|  |     fun searchHashTagCurationContentItem( | ||||||
|  |         @RequestParam curationId: Long, | ||||||
|  |         @RequestParam searchWord: String | ||||||
|  |     ) = ApiResponse.ok(service.searchHashTagCurationContentItem(curationId, searchWord)) | ||||||
|  |  | ||||||
|  |     @PostMapping("/add/item") | ||||||
|  |     fun addItemToHashTagCuration( | ||||||
|  |         @RequestBody request: AddItemToCurationRequest | ||||||
|  |     ) = ApiResponse.ok(service.addItemToHashTagCuration(request), "큐레이션 아이템을 등록했습니다.") | ||||||
|  |  | ||||||
|  |     @PutMapping("/remove/item") | ||||||
|  |     fun removeItemInHashTagCuration( | ||||||
|  |         @RequestBody request: RemoveItemInCurationRequest | ||||||
|  |     ) = ApiResponse.ok(service.removeItemInHashTagCuration(request), "큐레이션 아이템을 제거했습니다.") | ||||||
|  |  | ||||||
|  |     @PutMapping("/orders/item") | ||||||
|  |     fun updateItemInHashTagCurationOrders( | ||||||
|  |         @RequestBody request: UpdateCurationItemOrdersRequest | ||||||
|  |     ) = ApiResponse.ok(service.updateItemInHashTagCurationOrders(request), "수정되었습니다.") | ||||||
|  | } | ||||||
| @@ -0,0 +1,87 @@ | |||||||
|  | package kr.co.vividnext.sodalive.admin.content.curation.tag | ||||||
|  |  | ||||||
|  | import com.querydsl.jpa.impl.JPAQueryFactory | ||||||
|  | import kr.co.vividnext.sodalive.admin.content.curation.QSearchCurationItemResponse | ||||||
|  | import kr.co.vividnext.sodalive.admin.content.curation.SearchCurationItemResponse | ||||||
|  | import kr.co.vividnext.sodalive.content.QAudioContent.audioContent | ||||||
|  | import kr.co.vividnext.sodalive.content.main.curation.tag.ContentHashTagCuration | ||||||
|  | import kr.co.vividnext.sodalive.content.main.curation.tag.QContentHashTagCuration.contentHashTagCuration | ||||||
|  | import kr.co.vividnext.sodalive.content.main.curation.tag.QContentHashTagCurationItem.contentHashTagCurationItem | ||||||
|  | import org.springframework.beans.factory.annotation.Value | ||||||
|  | import org.springframework.data.jpa.repository.JpaRepository | ||||||
|  | import org.springframework.stereotype.Repository | ||||||
|  |  | ||||||
|  | interface AdminHashTagCurationRepository : | ||||||
|  |     JpaRepository<ContentHashTagCuration, Long>, | ||||||
|  |     AdminHashTagCurationQueryRepository | ||||||
|  |  | ||||||
|  | interface AdminHashTagCurationQueryRepository { | ||||||
|  |     fun getContentHashTagCurationList(): List<GetAdminContentHashTagCurationResponse> | ||||||
|  |     fun searchHashTagCurationContentItem(curationId: Long, searchWord: String): List<SearchCurationItemResponse> | ||||||
|  |     fun isExistsTag(tag: String): Boolean | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @Repository | ||||||
|  | class AdminHashTagCurationQueryRepositoryImpl( | ||||||
|  |     private val queryFactory: JPAQueryFactory, | ||||||
|  |  | ||||||
|  |     @Value("\${cloud.aws.cloud-front.host}") | ||||||
|  |     private val imageHost: String | ||||||
|  | ) : AdminHashTagCurationQueryRepository { | ||||||
|  |     override fun getContentHashTagCurationList(): List<GetAdminContentHashTagCurationResponse> { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QGetAdminContentHashTagCurationResponse( | ||||||
|  |                     contentHashTagCuration.id, | ||||||
|  |                     contentHashTagCuration.tag, | ||||||
|  |                     contentHashTagCuration.isAdult | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(contentHashTagCuration) | ||||||
|  |             .where(contentHashTagCuration.isActive.isTrue) | ||||||
|  |             .orderBy(contentHashTagCuration.orders.asc()) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun searchHashTagCurationContentItem( | ||||||
|  |         curationId: Long, | ||||||
|  |         searchWord: String | ||||||
|  |     ): List<SearchCurationItemResponse> { | ||||||
|  |         return queryFactory | ||||||
|  |             .select( | ||||||
|  |                 QSearchCurationItemResponse( | ||||||
|  |                     audioContent.id, | ||||||
|  |                     audioContent.title, | ||||||
|  |                     audioContent.coverImage.prepend("/").prepend(imageHost) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             .from(audioContent) | ||||||
|  |             .leftJoin(contentHashTagCurationItem) | ||||||
|  |             .on( | ||||||
|  |                 audioContent.id.eq(contentHashTagCurationItem.content.id) | ||||||
|  |                     .and(contentHashTagCurationItem.curation.id.eq(curationId)) | ||||||
|  |             ) | ||||||
|  |             .where( | ||||||
|  |                 audioContent.duration.isNotNull | ||||||
|  |                     .and(audioContent.member.isNotNull) | ||||||
|  |                     .and(audioContent.isActive.isTrue) | ||||||
|  |                     .and(audioContent.title.contains(searchWord)) | ||||||
|  |                     .and( | ||||||
|  |                         contentHashTagCurationItem.id.isNull | ||||||
|  |                             .or(contentHashTagCurationItem.isActive.isFalse) | ||||||
|  |                     ) | ||||||
|  |             ) | ||||||
|  |             .fetch() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun isExistsTag(tag: String): Boolean { | ||||||
|  |         return queryFactory | ||||||
|  |             .select(contentHashTagCuration.id) | ||||||
|  |             .from(contentHashTagCuration) | ||||||
|  |             .where( | ||||||
|  |                 contentHashTagCuration.tag.eq(tag), | ||||||
|  |                 contentHashTagCuration.isActive.isTrue | ||||||
|  |             ) | ||||||
|  |             .fetch().isNotEmpty() | ||||||
|  |     } | ||||||
|  | } | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user