mirror of
				https://github.com/abhinavxd/libredesk.git
				synced 2025-10-24 16:43:40 +00:00 
			
		
		
		
	Compare commits
	
		
			525 Commits
		
	
	
		
			v0.1.0-alp
			...
			refactor-a
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 98492a1869 | ||
|  | 18b50b11c8 | ||
|  | 5a1628f710 | ||
|  | 12ebe32ba3 | ||
|  | fce2587a9d | ||
|  | 7d92ac9cce | ||
|  | 3ce3c5e0ee | ||
|  | 35ad00ec51 | ||
|  | 9ec96be959 | ||
|  | 6ca36d611f | ||
|  | 5a87d24d72 | ||
|  | 7d4e7e68c3 | ||
|  | 5b941fd993 | ||
|  | 63e348e512 | ||
|  | 10a845dc81 | ||
|  | 0228989202 | ||
|  | 3f7d151d33 | ||
|  | a516773b14 | ||
|  | f6d3bd543f | ||
|  | c1c14f7f54 | ||
|  | 634fc66e9f | ||
|  | 0dec822c1c | ||
|  | 958f5e38c0 | ||
|  | 550a3fa801 | ||
|  | 6bbfbe8cf6 | ||
|  | f9ed326d72 | ||
|  | e0dc0285a4 | ||
|  | b971619ea6 | ||
|  | 69accaebef | ||
|  | 27de73536e | ||
|  | df108a3363 | ||
|  | 266c3dab72 | ||
|  | bf2c1fff6f | ||
|  | 2930af0c4f | ||
|  | 389c4e3dd3 | ||
|  | 9a119e6dc3 | ||
|  | ee178d383d | ||
|  | fc4db676d9 | ||
|  | 70cb3d0f80 | ||
|  | c9920c3377 | ||
|  | 6d62c3a4ba | ||
|  | d9b5fb8f0f | ||
|  | 3de320f1fb | ||
|  | be977dcff2 | ||
|  | 5e19f13e18 | ||
|  | ccc5940dd9 | ||
|  | 4203b82e90 | ||
|  | ba07e224c2 | ||
|  | 3fff65150f | ||
|  | c4fcf6bd91 | ||
|  | 5ea1b9e84c | ||
|  | 5b522888bc | ||
|  | dc2250ce50 | ||
|  | 839a06f0d2 | ||
|  | d2e5d85e3a | ||
|  | 0737d22374 | ||
|  | d6af9d10ea | ||
|  | 6381fc23c2 | ||
|  | 6bb5728665 | ||
|  | 2322ec33b0 | ||
|  | 9132e11458 | ||
|  | e70f92d377 | ||
|  | 591108f094 | ||
|  | 1b2a5e4f36 | ||
|  | f613cc237b | ||
|  | c37258fccb | ||
|  | 1879d9d22b | ||
|  | b369e2f56a | ||
|  | ef56f1a74e | ||
|  | d274adb19b | ||
|  | d31fcb00b6 | ||
|  | 88d719ec4f | ||
|  | 147180a536 | ||
|  | faa195f0a6 | ||
|  | 4b0422d904 | ||
|  | 9303997cea | ||
|  | aba07b3096 | ||
|  | 27aac88f53 | ||
|  | cb6b0e420b | ||
|  | e004afd7d1 | ||
|  | 6a77d346dc | ||
|  | 60c89cb617 | ||
|  | b7d4b187e8 | ||
|  | 2bf45f32de | ||
|  | 981372ab86 | ||
|  | 803196985d | ||
|  | ebf6a980e8 | ||
|  | 813ef91964 | ||
|  | 3b9fb7a08d | ||
|  | 7fb86f140c | ||
|  | aa8d326fa1 | ||
|  | ca9a0a5892 | ||
|  | 73e2950174 | ||
|  | e7b8e5c4bb | ||
|  | 582c906440 | ||
|  | f3881ee0aa | ||
|  | b557c2ca4b | ||
|  | 30884d3536 | ||
|  | bce0d1d12f | ||
|  | 67a4f6a162 | ||
|  | ec28ac8f3a | ||
|  | bc71fcfdc1 | ||
|  | bc0bee8f6a | ||
|  | 499fc0dad1 | ||
|  | 03b932c1c0 | ||
|  | 012de059e7 | ||
|  | 6357faf6c8 | ||
|  | f7a12cffd3 | ||
|  | 6487bf9a0a | ||
|  | 53d5715429 | ||
|  | b561e79440 | ||
|  | e567acbe59 | ||
|  | 57d0e90b5f | ||
|  | 5a0e3a8072 | ||
|  | d95a5f40cf | ||
|  | 6981a0790d | ||
|  | 55bc9bfc91 | ||
|  | 67db2e5ff2 | ||
|  | 64304c2384 | ||
|  | c5fe6aaadd | ||
|  | fea7eef658 | ||
|  | 475e400810 | ||
|  | 641ae0540e | ||
|  | dc6fede081 | ||
|  | 28dcd6cb2f | ||
|  | ade833fb7b | ||
|  | 5bcb0a2ad9 | ||
|  | ad2f685fec | ||
|  | 26c7df538c | ||
|  | 625a08d0aa | ||
|  | bf1510b9c3 | ||
|  | bae896d38d | ||
|  | 37b7c05b30 | ||
|  | eb05368f18 | ||
|  | 7ef510894b | ||
|  | 69268a3a84 | ||
|  | fcd3462d25 | ||
|  | fbf502451a | ||
|  | dc909ceb4f | ||
|  | cc1432b3e4 | ||
|  | d532a99771 | ||
|  | 50baa3f38e | ||
|  | 63a8f04408 | ||
|  | ea0b7d6d52 | ||
|  | 5d6897a960 | ||
|  | c4a95672fe | ||
|  | 2efd07b405 | ||
|  | 0b9cf38826 | ||
|  | b44c314299 | ||
|  | 2e1188e443 | ||
|  | afeec39b59 | ||
|  | fb2a08ec1a | ||
|  | 7f2df0082c | ||
|  | 6c523ac447 | ||
|  | 02fc57c35a | ||
|  | cd0a357695 | ||
|  | 2dc751e602 | ||
|  | 8bc0cce993 | ||
|  | f6e2fc1956 | ||
|  | 5fe5ac5882 | ||
|  | 975577555d | ||
|  | f43acb77a1 | ||
|  | 331c84fa56 | ||
|  | 9314efb9d9 | ||
|  | 5c8481af97 | ||
|  | d9bc4d1c0d | ||
|  | 087c8ad491 | ||
|  | 65cac843cb | ||
|  | 23b0481f24 | ||
|  | 9a651702ce | ||
|  | a0203f882e | ||
|  | 75425ca0dd | ||
|  | c2849fa63d | ||
|  | b20c7845ac | ||
|  | 38a5b25b1f | ||
|  | 9dce155ebc | ||
|  | 314341b40d | ||
|  | 1f6e3322aa | ||
|  | 102ba99b3c | ||
|  | 8285575f1c | ||
|  | 01d3b590a9 | ||
|  | 210e0de1ae | ||
|  | 1f8fdf2ef6 | ||
|  | 696e4780ac | ||
|  | 3998798e54 | ||
|  | 70b5da29e1 | ||
|  | 88ef5d26db | ||
|  | 54bad59392 | ||
|  | 506bb91e20 | ||
|  | d1478e1971 | ||
|  | 5583b472f7 | ||
|  | b715483260 | ||
|  | 8ce0464603 | ||
|  | a84ed1ed32 | ||
|  | 7426a09478 | ||
|  | 8ad2f078ac | ||
|  | 9226063db3 | ||
|  | a9fd4fe2b6 | ||
|  | 7e8c9962c3 | ||
|  | cf20142e40 | ||
|  | 8654a04dcf | ||
|  | 4c766d8ccb | ||
|  | cb1ec7eb8e | ||
|  | a89c3dbe04 | ||
|  | e2319714ca | ||
|  | 172f78262e | ||
|  | f53d5f188f | ||
|  | 55ec962003 | ||
|  | d3b1955cb2 | ||
|  | fac496fef2 | ||
|  | c36a425a1e | ||
|  | f43ab5041e | ||
|  | cd0ff1b67d | ||
|  | 5bc065469d | ||
|  | 77be86b1f4 | ||
|  | dde84c65b0 | ||
|  | f2d4969733 | ||
|  | aeececd001 | ||
|  | fdeeda8bca | ||
|  | 45bae57183 | ||
|  | a345b2e322 | ||
|  | 490aaedb48 | ||
|  | 87361e5cda | ||
|  | c039d5a20f | ||
|  | 53f15a3a7e | ||
|  | a397d3d3ea | ||
|  | 4ca123e6a1 | ||
|  | 7dd5abdda6 | ||
|  | c16144a2bf | ||
|  | 7f1c2c2f11 | ||
|  | d8a681d17e | ||
|  | f657a873bc | ||
|  | 88e07c324d | ||
|  | 6c9eca3d81 | ||
|  | 07b185050e | ||
|  | 66886c34e5 | ||
|  | 0af7265178 | ||
|  | f722de2fe4 | ||
|  | 6b2be57049 | ||
|  | e1b2ec8a4b | ||
|  | 8d47a7456d | ||
|  | 62023695a5 | ||
|  | a212ed4afb | ||
|  | 8e6bea09fe | ||
|  | 71e2e3cd8a | ||
|  | 59f5084bec | ||
|  | 87e1477811 | ||
|  | 10d3da608c | ||
|  | 0de7c91641 | ||
|  | 61ec075bd6 | ||
|  | 0b2c607cd3 | ||
|  | 0556318714 | ||
|  | 7b35cf0abf | ||
|  | 8619aa8e17 | ||
|  | 25db57805e | ||
|  | 3b2d0d049f | ||
|  | 1c6d03a4c2 | ||
|  | 062e0c39da | ||
|  | 67090fb052 | ||
|  | c434de130b | ||
|  | 4e4f07f2e8 | ||
|  | 19a507c88f | ||
|  | ac61d43688 | ||
|  | 7f8e3ccbbc | ||
|  | facce8bdad | ||
|  | 8acad27b75 | ||
|  | 24fbe14804 | ||
|  | 061677f2b0 | ||
|  | 450b609d47 | ||
|  | 971a433f3d | ||
|  | 220321bb8c | ||
|  | d5ba70667d | ||
|  | a9f9d368b9 | ||
|  | 2fc642c34e | ||
|  | 488f14e87c | ||
|  | 3702a61d74 | ||
|  | b01f6f812d | ||
|  | a0c77bc12e | ||
|  | 8bc511509c | ||
|  | 0254bab266 | ||
|  | 91372f5339 | ||
|  | d69a8c58d1 | ||
|  | 4e893ef876 | ||
|  | 5770188e4d | ||
|  | 8bd7895ccf | ||
|  | e10bb45582 | ||
|  | a397bc059b | ||
|  | 4a305ff889 | ||
|  | 616410c0a9 | ||
|  | 408e1fc142 | ||
|  | bc586fe775 | ||
|  | a49038f965 | ||
|  | 4cfe0ccbd9 | ||
|  | acbb94447c | ||
|  | cd429b9751 | ||
|  | 78d073c499 | ||
|  | 8083ad93b4 | ||
|  | ad99dee544 | ||
|  | a5eeb03f0d | ||
|  | c81f6496ea | ||
|  | 143a12e3c3 | ||
|  | e2d6a214c4 | ||
|  | 4a3afc83a5 | ||
|  | bb512d5ecd | ||
|  | 7957dbbd4a | ||
|  | 199778e771 | ||
|  | b2a53b18d5 | ||
|  | 576c678403 | ||
|  | 9bfe014d1e | ||
|  | 1b536bdc69 | ||
|  | c02339f311 | ||
|  | 1e7ab144b6 | ||
|  | e998529827 | ||
|  | 0a57a2724e | ||
|  | d2248d34c5 | ||
|  | 33f2f67ba8 | ||
|  | 7075ca214c | ||
|  | e68325d609 | ||
|  | 2499df866f | ||
|  | be5779e201 | ||
|  | 2d868b7df1 | ||
|  | 374aabcb10 | ||
|  | e69b1c3e6d | ||
|  | 1821647695 | ||
|  | b4f2186150 | ||
|  | 6d588f7a4e | ||
|  | 2a382d6036 | ||
|  | c639bfba40 | ||
|  | 82aac02a97 | ||
|  | c348a5c9b7 | ||
|  | 008f71d7b4 | ||
|  | 9b41aa0e9a | ||
|  | c60a0788d9 | ||
|  | 013b5bf37e | ||
|  | df0dfb480f | ||
|  | 2daefccd79 | ||
|  | f69e8dd4f8 | ||
|  | d171958223 | ||
|  | 3b7550fcf3 | ||
|  | 0de712762c | ||
|  | 6b6549cb03 | ||
|  | cd4b9a9c23 | ||
|  | e19f817c5f | ||
|  | 5ce8ed72ba | ||
|  | 4ec564ee2e | ||
|  | 19f08ec76a | ||
|  | dd8053b2bb | ||
|  | 72b92d6c66 | ||
|  | 497b54fc49 | ||
|  | 9d18d3d08d | ||
|  | 6bea14e7a9 | ||
|  | 25f23735d5 | ||
|  | 3888793450 | ||
|  | 88e4a55952 | ||
|  | 9aa9a5e1b2 | ||
|  | a3098a1dbd | ||
|  | 76a24467e7 | ||
|  | 4361250c73 | ||
|  | 7d9650be2e | ||
|  | eb707fd8de | ||
|  | 36077b1837 | ||
|  | d5499229b5 | ||
|  | 5e90dfee5a | ||
|  | 1875a62e00 | ||
|  | f60c4e8cb6 | ||
|  | 495ff02067 | ||
|  | 5afec04c07 | ||
|  | 56f00e791e | ||
|  | dcede8a461 | ||
|  | 39fd5c9165 | ||
|  | 4b8a954043 | ||
|  | 6ac9f28a32 | ||
|  | 8101c202fa | ||
|  | 09746fb365 | ||
|  | f59ea59a2e | ||
|  | a2cdd728c0 | ||
|  | ac59a5defc | ||
|  | 05fbe39315 | ||
|  | c7c65a3d83 | ||
|  | 5bf6b7df47 | ||
|  | c034c21fa5 | ||
|  | 4ed241a03d | ||
|  | 6b00f70c37 | ||
|  | c51073d289 | ||
|  | d03d4477de | ||
|  | 3b211dc372 | ||
|  | 6b4f243b74 | ||
|  | 9ff5a53ebb | ||
|  | 9b9282dfd9 | ||
|  | 698e2d960e | ||
|  | a8db8f64b5 | ||
|  | a5a9d1304c | ||
|  | f688be1c88 | ||
|  | d3eb3499df | ||
|  | 721f7c811c | ||
|  | a33e1453a8 | ||
|  | b6ce6975c9 | ||
|  | 860b216e2b | ||
|  | eaa2b1ddcf | ||
|  | 0f12b2a3f3 | ||
|  | def0bb8e4c | ||
|  | a41c360cdb | ||
|  | 159cca6866 | ||
|  | 83f553227a | ||
|  | 28a6a3d246 | ||
|  | 7e16cc1a74 | ||
|  | aeef7d4ad7 | ||
|  | f0358f67f0 | ||
|  | 12f2453f5a | ||
|  | 2742be5619 | ||
|  | d837defbc9 | ||
|  | 5cc849e7eb | ||
|  | 729faf980c | ||
|  | a36c81141b | ||
|  | 756147a2c9 | ||
|  | 88a641fe09 | ||
|  | 785da6715c | ||
|  | 32401fa231 | ||
|  | 83b891c92a | ||
|  | f277f76a0a | ||
|  | 5f1a40acba | ||
|  | d90b9c2be7 | ||
|  | 43184ec2f3 | ||
|  | 2fdcf68a22 | ||
|  | 4bef3e80a2 | ||
|  | 09703c1090 | ||
|  | 45541c221a | ||
|  | fc0e0a8fff | ||
|  | d1f931106d | ||
|  | 227aa26c35 | ||
|  | 79a3f0ff70 | ||
|  | eefacdbda2 | ||
|  | 3783cce1be | ||
|  | a4cb373f32 | ||
|  | 99e8949be6 | ||
|  | 1240051825 | ||
|  | 5398d4ec41 | ||
|  | fd4e47dc68 | ||
|  | 1ff7317c4d | ||
|  | d6449b9336 | ||
|  | 580fb76a39 | ||
|  | 91889423a2 | ||
|  | f12efe5511 | ||
|  | 56187ddc46 | ||
|  | 47af51d0dd | ||
|  | 47a3985a51 | ||
|  | 3f11af13b8 | ||
|  | da629c864c | ||
|  | 6fb35b90b3 | ||
|  | 9892f9dae7 | ||
|  | 277586f025 | ||
|  | f3070e13a7 | ||
|  | 8ed29df11c | ||
|  | 36d91de8f7 | ||
|  | 57c1948379 | ||
|  | 772152c40c | ||
|  | 8e15d733ea | ||
|  | fc47e65fcb | ||
|  | 760be37eda | ||
|  | d1f08ce035 | ||
|  | 8551b65a27 | ||
|  | eb499f64d0 | ||
|  | 494bc15b0a | ||
|  | 360557c58f | ||
|  | 8d8f08e1d2 | ||
|  | 10b4f9d08c | ||
|  | 79f74363da | ||
|  | 8f6295542e | ||
|  | 8e286e2273 | ||
|  | 3aad69fc52 | ||
|  | 58825c3de9 | ||
|  | 03c68afc4c | ||
|  | 15b9caaaed | ||
|  | b0d3dcb5dd | ||
|  | 96ef62b509 | ||
|  | 79c3f5a60c | ||
|  | 70bef7b3ab | ||
|  | b1e1dff3eb | ||
|  | 9b34c2737d | ||
|  | 1b63f03bb1 | ||
|  | 26d76c966f | ||
|  | 1ff335f772 | ||
|  | 5836ee8d90 | ||
|  | 98534f3c5a | ||
|  | 59951f0829 | ||
|  | 461ae3cf22 | ||
|  | da5dfdbcde | ||
|  | 9c67c02b08 | ||
|  | 15b200b0db | ||
|  | f4617c599c | ||
|  | 341d0b7e47 | ||
|  | 78b8c508d8 | ||
|  | f17d96f96f | ||
|  | c75c117a4d | ||
|  | 873d26ccb2 | ||
|  | 71601364ae | ||
|  | 44723fb70d | ||
|  | 67e1230485 | ||
|  | d58898c60f | ||
|  | a8dc0a6242 | ||
|  | 3aa144f703 | ||
|  | fcbd16f042 | ||
|  | e8f3f24422 | ||
|  | 425bb4ed04 | ||
|  | 0c3da82250 | ||
|  | 8649826a89 | ||
|  | d427dfd20c | ||
|  | afb54c371b | ||
|  | 46459599c7 | ||
|  | 63a6aedfd0 | ||
|  | ffbf613e68 | ||
|  | 88f82fe80b | ||
|  | 914b6371b6 | ||
|  | 89eb05f337 | ||
|  | 71a3588855 | ||
|  | c6baf3f9bf | ||
|  | 368ec3c82b | ||
|  | 4cc40ec5d5 | ||
|  | 171e404e6f | ||
|  | 28f4fda274 | ||
|  | 00ded9c19b | ||
|  | 17efaf0f2c | ||
|  | b44290a6f0 | ||
|  | 1a7ee4d8c6 | ||
|  | ab56d01e22 | 
							
								
								
									
										1
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| VERSION export-subst | ||||
							
								
								
									
										16
									
								
								.github/ISSUE_TEMPLATE/confirmed-bug.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								.github/ISSUE_TEMPLATE/confirmed-bug.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| --- | ||||
| name: Confirmed Bug Report | ||||
| about: Report a confirmed bug in Libredesk | ||||
| title: "[Bug] <brief summary>" | ||||
| labels: bug | ||||
| assignees: "" | ||||
| --- | ||||
|  | ||||
| **Version:** | ||||
| - libredesk: [eg: v0.7.0] | ||||
|  | ||||
| **Description of the bug and steps to reproduce:** | ||||
| A clear and concise description of what the bug is. | ||||
|  | ||||
| **Logs / Screenshots:** | ||||
| Attach any relevant logs or screenshots to help diagnose the issue. | ||||
							
								
								
									
										16
									
								
								.github/ISSUE_TEMPLATE/possible-bug.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								.github/ISSUE_TEMPLATE/possible-bug.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| --- | ||||
| name: Possible Bug Report | ||||
| about: Something in Libredesk might be broken but needs confirmation | ||||
| title: "[Possible Bug] <brief summary>" | ||||
| labels: bug, needs-investigation | ||||
| assignees: "" | ||||
| --- | ||||
|  | ||||
| **Version:** | ||||
|  - libredesk: [eg: v0.7.0] | ||||
|   | ||||
| **Description of the bug and steps to reproduce:** | ||||
| A clear and concise description of what the bug is. | ||||
|  | ||||
| **Logs / Screenshots:** | ||||
| Attach any relevant logs or screenshots to help diagnose the issue. | ||||
							
								
								
									
										47
									
								
								.github/workflows/crowdin.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								.github/workflows/crowdin.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,47 @@ | ||||
| name: Crowdin | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     paths: | ||||
|       # Only trigger a Crowdin update when the source localization file is | ||||
|       # updated. | ||||
|       - 'i18n/en.json' | ||||
|     # Only watches for changes happening on "main" branch. | ||||
|     branches: [ main ] | ||||
|  | ||||
| jobs: | ||||
|   crowdin: | ||||
|     runs-on: ubuntu-latest | ||||
|     # Only run on the original repository, not forks | ||||
|     if: github.event.repository.fork == false | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v4 | ||||
|  | ||||
|       - name: Crowdin push | ||||
|         uses: crowdin/github-action@v2 | ||||
|         with: | ||||
|           # Send source (english) strings to Crowdin. | ||||
|           upload_sources: true | ||||
|           # See: https://crowdin.github.io/crowdin-cli/commands | ||||
|           # /crowdin-upload#options | ||||
|           upload_sources_args: '--preserve-hierarchy --delete-obsolete' | ||||
|           # Don't upload or download translations. | ||||
|           upload_translations: false | ||||
|           download_translations: false | ||||
|           # Source language file. | ||||
|           source: 'i18n/en.json' | ||||
|           # Translations files. | ||||
|           translation: 'i18n/%two_letters_code%.json' | ||||
|         env: | ||||
|           # Crowdin.com > Project > Tools > API > Project ID. | ||||
|           CROWDIN_PROJECT_ID: ${{ secrets.CROWDIN_PROJECT_ID }} | ||||
|           # When creating a personal token in Crowdin, you'll be asked to select | ||||
|           # the necessary scopes. The basic Crowdin Personal Token scopes are | ||||
|           # the following: | ||||
|           #  - Projects (List, Get, Create, Edit) -> Read | ||||
|           #  - Translation Status -> Read Only | ||||
|           #  - Source files & strings -> Read and Write | ||||
|           #  - Translations -> Read and Write | ||||
|           CROWDIN_PERSONAL_TOKEN: ${{ secrets.CROWDIN_PERSONAL_TOKEN }} | ||||
|  | ||||
							
								
								
									
										71
									
								
								.github/workflows/frontend-ci.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								.github/workflows/frontend-ci.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | ||||
| name: CI | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: [main] | ||||
|   pull_request: | ||||
|     branches: [main] | ||||
|  | ||||
| jobs: | ||||
|   build-and-test: | ||||
|     runs-on: ubuntu-latest | ||||
|     services: | ||||
|       db: | ||||
|         image: postgres:17-alpine | ||||
|         ports: | ||||
|           - 5432:5432 | ||||
|         env: | ||||
|           POSTGRES_USER: libredesk | ||||
|           POSTGRES_PASSWORD: libredesk | ||||
|           POSTGRES_DB: libredesk | ||||
|         options: >- | ||||
|           --health-cmd="pg_isready -U libredesk" | ||||
|           --health-interval=10s | ||||
|           --health-timeout=5s | ||||
|           --health-retries=5 | ||||
|       redis: | ||||
|         image: redis | ||||
|         ports: | ||||
|           - 6379:6379 | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v3 | ||||
|  | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v3 | ||||
|         with: | ||||
|           go-version: "1.24.3" | ||||
|  | ||||
|       - name: Set up Node.js | ||||
|         uses: actions/setup-node@v3 | ||||
|         with: | ||||
|           node-version: "20" | ||||
|  | ||||
|       - name: Install pnpm | ||||
|         run: npm install -g pnpm | ||||
|  | ||||
|       - name: Install cypress deps | ||||
|         run: sudo apt-get update && sudo apt-get install -y libgtk2.0-0 libgtk-3-0 libgbm-dev libnotify-dev libnss3 libxss1 libasound2t64 libxtst6 xauth xvfb | ||||
|  | ||||
|       - name: Build binary and frontend | ||||
|         run: make build | ||||
|  | ||||
|       - name: Configure app | ||||
|         run: | | ||||
|           cp config.sample.toml config.toml | ||||
|           sed -i 's/host = "db"/host = "127.0.0.1"/' config.toml | ||||
|           sed -i 's/address = "redis:6379"/address = "localhost:6379"/' config.toml | ||||
|  | ||||
|       - name: Run unit tests for frontend | ||||
|         run: cd frontend && pnpm test:run | ||||
|  | ||||
|       - name: Install db schema and run tests | ||||
|         env: | ||||
|           LIBREDESK_SYSTEM_USER_PASSWORD: "StrongPass!123" | ||||
|         run: | | ||||
|           ./libredesk --install --idempotent-install --yes --config ./config.toml | ||||
|           ./libredesk --upgrade --yes --config ./config.toml | ||||
|           ./libredesk --config ./config.toml & | ||||
|           sleep 10 | ||||
|           cd frontend | ||||
|           pnpm run test:e2e:ci | ||||
							
								
								
									
										31
									
								
								.github/workflows/github-pages.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								.github/workflows/github-pages.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | ||||
| name: Deploy MkDocs | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - main | ||||
|  | ||||
| jobs: | ||||
|   deploy: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v3 | ||||
|  | ||||
|       - uses: actions/setup-python@v4 | ||||
|         with: | ||||
|           python-version: 3.x | ||||
|  | ||||
|       - run: pip install mkdocs-material | ||||
|  | ||||
|       - run: | | ||||
|           if [ -f requirements.txt ]; then | ||||
|             pip install -r requirements.txt; | ||||
|           fi | ||||
|  | ||||
|       - run: cd docs && mkdocs build | ||||
|  | ||||
|       - name: Deploy to GitHub Pages | ||||
|         uses: peaceiris/actions-gh-pages@v3 | ||||
|         with: | ||||
|           github_token: ${{ secrets.GITHUB_TOKEN }} | ||||
|           publish_dir: ./docs/site | ||||
							
								
								
									
										22
									
								
								.github/workflows/go.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								.github/workflows/go.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| name: Go | ||||
|  | ||||
| on: [push] | ||||
|  | ||||
| jobs: | ||||
|   test: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|  | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v5 | ||||
|         with: | ||||
|           go-version: "1.24.3" | ||||
|  | ||||
|       - name: Install dependencies | ||||
|         run: go get -v ./... | ||||
|  | ||||
|       - name: Run tests | ||||
|         run: make test | ||||
							
								
								
									
										2
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -40,7 +40,7 @@ jobs: | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v5 | ||||
|         with: | ||||
|           go-version: "1.21" | ||||
|           go-version: "1.24.3" | ||||
|           cache: true | ||||
|  | ||||
|       - name: Set up Node.js | ||||
|   | ||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -6,4 +6,5 @@ libredesk | ||||
| libredesk.exe | ||||
| uploads | ||||
| .env | ||||
| dist/ | ||||
| dist/ | ||||
| .vscode/ | ||||
|   | ||||
| @@ -10,7 +10,7 @@ before: | ||||
|     - make frontend-build | ||||
|  | ||||
| builds: | ||||
|   - id: "standard" | ||||
|   - id: "universal" | ||||
|     main: ./cmd | ||||
|     env: | ||||
|       - CGO_ENABLED=0 | ||||
| @@ -24,29 +24,13 @@ builds: | ||||
|     goarch: | ||||
|       - amd64 | ||||
|       - arm64 | ||||
|     binary: 'libredesk{{ if eq .Os "windows" }}.exe{{ end }}' | ||||
|     ldflags: | ||||
|       - -s -w -X main.version={{.Version}} -X main.commit={{.Commit}} -X main.date={{.Date}} | ||||
|     hooks: | ||||
|       post: make stuff BIN={{ .Path }} | ||||
|  | ||||
|   - id: "arm" | ||||
|     main: ./cmd | ||||
|     env: | ||||
|       - CGO_ENABLED=0 | ||||
|     goos: | ||||
|       - freebsd | ||||
|       - linux | ||||
|       - netbsd | ||||
|       - openbsd | ||||
|     goarch: | ||||
|       - arm | ||||
|     goarm: | ||||
|       - 6 | ||||
|       - 7 | ||||
|     binary: 'libredesk{{ if eq .Os "windows" }}.exe{{ end }}' | ||||
|     ldflags: | ||||
|       - -s -w -X main.version={{.Version}} -X main.commit={{.Commit}} -X main.date={{.Date}} | ||||
|       - -s -w -X "main.buildString={{ .Tag }} ({{ .ShortCommit }} {{ .Date }}, {{ .Os }}/{{ .Arch }})" -X "main.versionString={{ .Tag }}" | ||||
|     hooks: | ||||
|       post: make stuff BIN={{ .Path }} | ||||
|  | ||||
| @@ -70,7 +54,7 @@ dockers: | ||||
|     goos: linux | ||||
|     goarch: amd64 | ||||
|     ids: | ||||
|       - standard | ||||
|       - universal | ||||
|     image_templates: | ||||
|       - "{{ .Env.DOCKER_ORG }}/{{ .ProjectName }}:latest-amd64" | ||||
|       - "{{ .Env.DOCKER_ORG }}/{{ .ProjectName }}:{{ .Tag }}-amd64" | ||||
| @@ -94,7 +78,7 @@ dockers: | ||||
|     goos: linux | ||||
|     goarch: arm64 | ||||
|     ids: | ||||
|       - standard | ||||
|       - universal | ||||
|     image_templates: | ||||
|       - "{{ .Env.DOCKER_ORG }}/{{ .ProjectName }}:latest-arm64" | ||||
|       - "{{ .Env.DOCKER_ORG }}/{{ .ProjectName }}:{{ .Tag }}-arm64" | ||||
| @@ -119,7 +103,7 @@ dockers: | ||||
|     goarch: arm | ||||
|     goarm: 6 | ||||
|     ids: | ||||
|       - arm | ||||
|       - universal | ||||
|     image_templates: | ||||
|       - "{{ .Env.DOCKER_ORG }}/{{ .ProjectName }}:latest-armv6" | ||||
|       - "{{ .Env.DOCKER_ORG }}/{{ .ProjectName }}:{{ .Tag }}-armv6" | ||||
| @@ -144,7 +128,7 @@ dockers: | ||||
|     goarch: arm | ||||
|     goarm: 7 | ||||
|     ids: | ||||
|       - arm | ||||
|       - universal | ||||
|     image_templates: | ||||
|       - "{{ .Env.DOCKER_ORG }}/{{ .ProjectName }}:latest-armv7" | ||||
|       - "{{ .Env.DOCKER_ORG }}/{{ .ProjectName }}:{{ .Tag }}-armv7" | ||||
| @@ -195,4 +179,4 @@ release: | ||||
|     owner: abhinavxd | ||||
|     name: libredesk | ||||
|   prerelease: auto | ||||
|   draft: true | ||||
|   draft: true | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
| FROM alpine:latest | ||||
|  | ||||
| # Install necessary packages | ||||
| RUN apk --no-cache add ca-certificates | ||||
| RUN apk --no-cache add ca-certificates tzdata | ||||
|  | ||||
| # Set the working directory to /libredesk | ||||
| WORKDIR /libredesk | ||||
|   | ||||
							
								
								
									
										36
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								Makefile
									
									
									
									
									
								
							| @@ -1,8 +1,10 @@ | ||||
| # Build variables | ||||
| LAST_COMMIT := $(shell git rev-parse --short HEAD) | ||||
| LAST_COMMIT_DATE := $(shell git show -s --format=%ci ${LAST_COMMIT}) | ||||
| VERSION := $(shell git describe --tags)  | ||||
| BUILDSTR := ${VERSION} (Commit: ${LAST_COMMIT_DATE} (${LAST_COMMIT}), Build: $(shell date +"%Y-%m-%d %H:%M:%S %z")) | ||||
| # Try to get the commit hash from 1) git 2) the VERSION file 3) fallback. | ||||
| LAST_COMMIT := $(or $(shell git rev-parse --short HEAD 2> /dev/null),$(shell head -n 1 VERSION | grep -oP -m 1 "^[a-z0-9]+$$"), "") | ||||
|  | ||||
| # Try to get the semver from 1) git 2) the VERSION file 3) fallback. | ||||
| VERSION := $(or $(LIBREDESK_VERSION),$(shell git describe --tags --abbrev=0 2> /dev/null),$(shell grep -oP 'tag: \Kv\d+\.\d+\.\d+(-[a-zA-Z0-9.-]+)?' VERSION),"v0.0.0") | ||||
|  | ||||
| BUILDSTR := ${VERSION} (\#${LAST_COMMIT} $(shell date -u +"%Y-%m-%dT%H:%M:%S%z")) | ||||
|  | ||||
| # Binary names and paths | ||||
| BIN := libredesk | ||||
| @@ -30,13 +32,13 @@ install-deps: $(STUFFBIN) | ||||
| .PHONY: frontend-build | ||||
| frontend-build: install-deps | ||||
| 	@echo "→ Building frontend for production..." | ||||
| 	@cd ${FRONTEND_DIR} && pnpm build | ||||
| 	@export VITE_APP_VERSION="${VERSION}" && cd ${FRONTEND_DIR} && pnpm build | ||||
|  | ||||
| # Run the Go backend server in development mode. | ||||
| .PHONY: run-backend | ||||
| run-backend: | ||||
| 	@echo "→ Running backend..." | ||||
| 	@go run cmd/*.go | ||||
| 	CGO_ENABLED=0 go run -ldflags="-s -w -X 'main.buildString=${BUILDSTR}' -X 'main.versionString=${VERSION}' -X 'github.com/abhinavxd/libredesk/internal/version.Version=${VERSION}' -X 'main.frontendDir=frontend/dist'" cmd/*.go | ||||
|  | ||||
| # Run the JS frontend server in development mode. | ||||
| .PHONY: run-frontend | ||||
| @@ -44,19 +46,19 @@ run-frontend: | ||||
| 	@echo "→ Installing frontend dependencies (if not already installed)..." | ||||
| 	@cd ${FRONTEND_DIR} && pnpm install | ||||
| 	@echo "→ Running frontend..." | ||||
| 	@export VUE_APP_VERSION="${VERSION}" && cd ${FRONTEND_DIR} && pnpm dev | ||||
| 	@export VITE_APP_VERSION="${VERSION}" && cd ${FRONTEND_DIR} && pnpm dev | ||||
|  | ||||
| # Build the backend binary. | ||||
| .PHONY: backend-build | ||||
| backend-build: $(STUFFBIN) | ||||
| .PHONY: build-backend | ||||
| build-backend: $(STUFFBIN) | ||||
| 	@echo "→ Building backend..." | ||||
| 	@CGO_ENABLED=0 go build -a\ | ||||
| 		-ldflags="-X 'main.buildString=${BUILDSTR}' -X 'main.buildDate=${LAST_COMMIT_DATE}' -s -w" \ | ||||
| 	@CGO_ENABLED=0 go build -a \ | ||||
| 		-ldflags="-X 'main.buildString=${BUILDSTR}' -X 'main.versionString=${VERSION}' -X 'github.com/abhinavxd/libredesk/internal/version.Version=${VERSION}' -s -w" \ | ||||
| 		-o ${BIN} cmd/*.go | ||||
|  | ||||
| # Main build target: builds both frontend and backend, then stuffs static assets into the binary. | ||||
| .PHONY: build | ||||
| build: frontend-build backend-build stuff | ||||
| build: frontend-build build-backend stuff | ||||
| 	@echo "→ Build successful. Current version: $(VERSION)" | ||||
|  | ||||
| # Stuff static assets into the binary using stuffbin. | ||||
| @@ -69,4 +71,10 @@ stuff: $(STUFFBIN) | ||||
| .PHONY: demo-build | ||||
| demo-build: | ||||
| 	@echo "→ Building in demo mode..." | ||||
| 	@export VITE_DEMO_BUILD="true" && $(MAKE) build | ||||
| 	@export VITE_DEMO_BUILD="true" && $(MAKE) build | ||||
|  | ||||
| # Run tests. | ||||
| .PHONY: test | ||||
| test: | ||||
| 	@echo "→ Running tests..." | ||||
| 	go test -count=1 ./... | ||||
|   | ||||
							
								
								
									
										109
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										109
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,46 +1,95 @@ | ||||
| <a href="https://zerodha.tech"><img src="https://zerodha.tech/static/images/github-badge.svg" align="right" /></a> | ||||
| <a href="https://zerodha.tech"><img src="https://zerodha.tech/static/images/github-badge.svg" align="right" alt="Zerodha Tech Badge" /></a> | ||||
|  | ||||
|  | ||||
| # Libredesk | ||||
|  | ||||
| Open source, self-hosted customer support desk. Single binary app. | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| Visit [libredesk.io](https://libredesk.io) for more info. Check out the [**Live demo**](https://demo.libredesk.io/). | ||||
|  | ||||
|  | ||||
| > **CAUTION:** This project is currently in **alpha**. Features and APIs may change and are not yet fully tested. | ||||
|  | ||||
| ## Features | ||||
|  | ||||
| - **Multi Shared Inbox**   | ||||
|   Libredesk supports multiple shared inboxes, letting you manage conversations across teams effortlessly. | ||||
| - **Granular Permissions**   | ||||
|   Create custom roles with granular permissions for teams and individual agents. | ||||
| - **Smart Automation**   | ||||
|   Eliminate repetitive tasks with powerful automation rules. Auto-tag, assign, and route conversations based on custom conditions. | ||||
| - **CSAT Surveys**   | ||||
|   Measure customer satisfaction with automated surveys. | ||||
| - **Macros**   | ||||
|   Save frequently sent messages as templates. With one click, send saved responses, set tags, and more. | ||||
| - **Smart Organization**   | ||||
|   Keep conversations organized with tags, custom statuses for conversations, and snoozing. Find any conversation instantly from the search bar. | ||||
| - **Auto Assignment**   | ||||
|   Distribute workload with auto assignment rules. Auto-assign conversations based on agent capacity or custom criteria. | ||||
| - **SLA Management**   | ||||
|   Set and track response time targets. Get notified when conversations are at risk of breaching SLA commitments. | ||||
| - **Custom attributes**   | ||||
|   Create custom attributes for contacts or conversations such as the subscription plan or the date of their first purchase.  | ||||
| - **AI-Assist**   | ||||
|   Instantly rewrite responses with AI to make them more friendly, professional, or polished. | ||||
| - **Activity logs**   | ||||
|   Track all actions performed by agents and admins—updates and key events across the system—for auditing and accountability. | ||||
| - **Webhooks**   | ||||
|   Integrate with external systems using real-time HTTP notifications for conversation and message events. | ||||
| - **Command Bar**   | ||||
|   Opens with a simple shortcut (CTRL+K) and lets you quickly perform actions on conversations. | ||||
|  | ||||
| And more checkout - [libredesk.io](https://libredesk.io) | ||||
|  | ||||
|  | ||||
| > [!CAUTION] | ||||
| > This project is currently in **alpha**. Features and APIs may change and are not yet fully tested. | ||||
| ## Installation | ||||
|  | ||||
| ### Docker | ||||
|  | ||||
| The latest image is available on DockerHub at [`libredesk/libredesk:latest`](https://hub.docker.com/r/libredesk/libredesk/tags?page=1&ordering=last_updated&name=latest) | ||||
|  | ||||
| ```shell | ||||
| # Download the compose file and sample config file in the current directory. | ||||
| curl -LO https://github.com/abhinavxd/libredesk/raw/main/docker-compose.yml | ||||
| curl -LO https://github.com/abhinavxd/libredesk/raw/main/config.sample.toml | ||||
|  | ||||
| # Copy the config.sample.toml to config.toml and edit it as needed. | ||||
| cp config.sample.toml config.toml | ||||
|  | ||||
| # Run the services in the background. | ||||
| docker compose up -d | ||||
|  | ||||
| # Setting System user password. | ||||
| docker exec -it libredesk_app ./libredesk --set-system-user-password | ||||
| ``` | ||||
|  | ||||
| Go to `http://localhost:9000` and login with username `System` and the password you set using the `--set-system-user-password` command. | ||||
|  | ||||
| See [installation docs](https://libredesk.io/docs/installation/) | ||||
|  | ||||
| __________________ | ||||
|  | ||||
| ### Binary | ||||
| - Download the [latest release](https://github.com/abhinavxd/libredesk/releases) and extract the libredesk binary. | ||||
| - Copy config.sample.toml to config.toml and edit as needed. | ||||
| - `./libredesk --install` to setup the Postgres DB (or `--upgrade` to upgrade an existing DB. Upgrades are idempotent and running them multiple times have no side effects). | ||||
| - Run `./libredesk --set-system-user-password` to set the password for the System user. | ||||
| - Run `./libredesk` and visit `http://localhost:9000` and login with username `System` and the password you set using the --set-system-user-password command. | ||||
|  | ||||
| See [installation docs](https://libredesk.io/docs/installation) | ||||
| __________________ | ||||
|  | ||||
|  | ||||
| ## Developer Setup | ||||
| ## Developers | ||||
| If you are interested in contributing, refer to the [developer setup](https://libredesk.io/docs/developer-setup/). The backend is written in Go and the frontend is Vue js 3 with Shadcn for UI components. | ||||
|  | ||||
| #### Prerequisites | ||||
| ## Development Status | ||||
|  | ||||
| - **go** | ||||
| - **pnpm** | ||||
| - **postgreSQL >= 13** | ||||
| - **redis** | ||||
| Libredesk is under active development.   | ||||
| Track roadmap and progress on the GitHub Project Board:   [https://github.com/users/abhinavxd/projects/1](https://github.com/users/abhinavxd/projects/1) | ||||
|  | ||||
| 1. **Clone the repository**: | ||||
|  | ||||
|    ```bash | ||||
|    git clone https://github.com/abhinavxd/libredesk.git | ||||
|    cd libredesk | ||||
|    ``` | ||||
|  | ||||
| 2. **Create config file**: | ||||
|  | ||||
|    - Copy the sample configuration file `config.toml.sample` to `config.toml`: | ||||
|      | ||||
|        ```bash | ||||
|        cp config.toml.sample config.toml | ||||
|        ``` | ||||
|    - Edit the `config.toml` file to configure your postgres and redis connection settings. | ||||
|  | ||||
| 3. **Run in development mode**: | ||||
|  | ||||
|    - Backend: `make run-backend` | ||||
|    - Frontend: `make run-frontend` | ||||
|  | ||||
| ## Translators | ||||
| You can help translate Libredesk into your language on [Crowdin](https://crowdin.com/project/libredesk).   | ||||
|   | ||||
							
								
								
									
										36
									
								
								cmd/actvity_log.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								cmd/actvity_log.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"strconv" | ||||
|  | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| // handleGetActivityLogs returns activity logs from the database. | ||||
| func handleGetActivityLogs(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app         = r.Context.(*App) | ||||
| 		order       = string(r.RequestCtx.QueryArgs().Peek("order")) | ||||
| 		orderBy     = string(r.RequestCtx.QueryArgs().Peek("order_by")) | ||||
| 		filters     = string(r.RequestCtx.QueryArgs().Peek("filters")) | ||||
| 		page, _     = strconv.Atoi(string(r.RequestCtx.QueryArgs().Peek("page"))) | ||||
| 		pageSize, _ = strconv.Atoi(string(r.RequestCtx.QueryArgs().Peek("page_size"))) | ||||
| 		total       = 0 | ||||
| 	) | ||||
| 	logs, err := app.activityLog.GetAll(order, orderBy, filters, page, pageSize) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	if len(logs) > 0 { | ||||
| 		total = logs[0].Total | ||||
| 	} | ||||
| 	return r.SendEnvelope(envelope.PageResults{ | ||||
| 		Results:    logs, | ||||
| 		Total:      total, | ||||
| 		PerPage:    pageSize, | ||||
| 		TotalPages: (total + pageSize - 1) / pageSize, | ||||
| 		Page:       page, | ||||
| 	}) | ||||
|  | ||||
| } | ||||
							
								
								
									
										42
									
								
								cmd/ai.go
									
									
									
									
									
								
							
							
						
						
									
										42
									
								
								cmd/ai.go
									
									
									
									
									
								
							| @@ -1,15 +1,32 @@ | ||||
| package main | ||||
|  | ||||
| import "github.com/zerodha/fastglue" | ||||
| import ( | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| type aiCompletionReq struct { | ||||
| 	PromptKey string `json:"prompt_key"` | ||||
| 	Content   string `json:"content"` | ||||
| } | ||||
|  | ||||
| type providerUpdateReq struct { | ||||
| 	Provider string `json:"provider"` | ||||
| 	APIKey   string `json:"api_key"` | ||||
| } | ||||
|  | ||||
| // handleAICompletion handles AI completion requests | ||||
| func handleAICompletion(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app       = r.Context.(*App) | ||||
| 		promptKey = string(r.RequestCtx.PostArgs().Peek("prompt_key")) | ||||
| 		content   = string(r.RequestCtx.PostArgs().Peek("content")) | ||||
| 		app = r.Context.(*App) | ||||
| 		req = aiCompletionReq{} | ||||
| 	) | ||||
| 	resp, err := app.ai.Completion(promptKey, content) | ||||
|  | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil)) | ||||
| 	} | ||||
|  | ||||
| 	resp, err := app.ai.Completion(req.PromptKey, req.Content) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| @@ -27,3 +44,18 @@ func handleGetAIPrompts(r *fastglue.Request) error { | ||||
| 	} | ||||
| 	return r.SendEnvelope(resp) | ||||
| } | ||||
|  | ||||
| // handleUpdateAIProvider updates the AI provider | ||||
| func handleUpdateAIProvider(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 		req providerUpdateReq | ||||
| 	) | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil)) | ||||
| 	} | ||||
| 	if err := app.ai.UpdateProvider(req.Provider, req.APIKey); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("Provider updated successfully") | ||||
| } | ||||
|   | ||||
							
								
								
									
										33
									
								
								cmd/auth.go
									
									
									
									
									
								
							
							
						
						
									
										33
									
								
								cmd/auth.go
									
									
									
									
									
								
							| @@ -6,6 +6,7 @@ import ( | ||||
| 	amodels "github.com/abhinavxd/libredesk/internal/auth/models" | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| 	"github.com/abhinavxd/libredesk/internal/stringutil" | ||||
| 	realip "github.com/ferluci/fast-realip" | ||||
| 	"github.com/valyala/fasthttp" | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
| @@ -22,20 +23,21 @@ func handleOIDCLogin(r *fastglue.Request) error { | ||||
| 	) | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error parsing provider id", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error parsing provider id.", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	// Set a state and save it in the session, to prevent CSRF attacks. | ||||
| 	state, err := stringutil.RandomAlphanumeric(32) | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error generating state", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error generating state.", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorGenerating", "name", "state"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	if err = app.auth.SetSessionValues(r, map[string]interface{}{ | ||||
| 		oidcStateSessKey: state, | ||||
| 	}); err != nil { | ||||
| 		app.lo.Error("error saving state in session", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error saving state in session.", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorSaving", "name", "{globals.terms.session}"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	authURL, err := app.auth.LoginURL(providerID, state) | ||||
| @@ -52,30 +54,32 @@ func handleOIDCCallback(r *fastglue.Request) error { | ||||
| 		code            = string(r.RequestCtx.QueryArgs().Peek("code")) | ||||
| 		state           = string(r.RequestCtx.QueryArgs().Peek("state")) | ||||
| 		providerID, err = strconv.Atoi(string(r.RequestCtx.UserValue("id").(string))) | ||||
| 		ip              = realip.FromRequest(r.RequestCtx) | ||||
| 	) | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error parsing provider id", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error parsing provider id.", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	// Compare the state from the session with the state from the query. | ||||
| 	sessionState, err := app.auth.GetSessionValue(r, oidcStateSessKey) | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error getting state from session", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error getting state from session.", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorFetching", "name", "{globals.terms.session}"), nil, envelope.GeneralError) | ||||
| 	} | ||||
| 	if state != sessionState { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusForbidden, "Invalid state.", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusForbidden, app.i18n.Ts("globals.messages.mismatch", "name", "{globals.terms.state}"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	_, claims, err := app.auth.ExchangeOIDCToken(r.RequestCtx, providerID, code) | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error exchanging oidc token", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error exchanging OIDC token.", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, | ||||
| 			app.i18n.T("globals.messages.errorExchangingToken"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	// Lookup the user by email and set the session. | ||||
| 	user, err := app.user.GetByEmail(claims.Email) | ||||
| 	user, err := app.user.GetAgent(0, claims.Email) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| @@ -86,7 +90,18 @@ func handleOIDCCallback(r *fastglue.Request) error { | ||||
| 		FirstName: user.FirstName, | ||||
| 		LastName:  user.LastName, | ||||
| 	}, r); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error saving session.", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, | ||||
| 			app.i18n.Ts("globals.messages.errorSaving", "name", "{globals.terms.session}"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	// Update last login time. | ||||
| 	if err := app.user.UpdateLastLoginAt(user.ID); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Insert activity log. | ||||
| 	if err := app.activityLog.Login(user.ID, user.Email.String, ip); err != nil { | ||||
| 		app.lo.Error("error creating login activity log", "error", err) | ||||
| 	} | ||||
|  | ||||
| 	return r.Redirect("/", fasthttp.StatusFound, nil, "") | ||||
|   | ||||
| @@ -9,6 +9,10 @@ import ( | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| type updateAutomationRuleExecutionModeReq struct { | ||||
| 	Mode string `json:"mode"` | ||||
| } | ||||
|  | ||||
| // handleGetAutomationRules gets all automation rules | ||||
| func handleGetAutomationRules(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| @@ -41,10 +45,11 @@ func handleToggleAutomationRule(r *fastglue.Request) error { | ||||
| 		app   = r.Context.(*App) | ||||
| 		id, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
| 	if err := app.automation.ToggleRule(id); err != nil { | ||||
| 	toggledRule, err := app.automation.ToggleRule(id) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("Rule toggled successfully") | ||||
| 	return r.SendEnvelope(toggledRule) | ||||
| } | ||||
|  | ||||
| // handleUpdateAutomationRule updates an automation rule | ||||
| @@ -55,18 +60,18 @@ func handleUpdateAutomationRule(r *fastglue.Request) error { | ||||
| 		id, err = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid rule `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := r.Decode(&rule, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err = app.automation.UpdateRule(id, rule);err != nil { | ||||
| 	updatedRule, err := app.automation.UpdateRule(id, rule) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("Rule updated successfully") | ||||
| 	return r.SendEnvelope(updatedRule) | ||||
| } | ||||
|  | ||||
| // handleCreateAutomationRule creates a new automation rule | ||||
| @@ -76,12 +81,13 @@ func handleCreateAutomationRule(r *fastglue.Request) error { | ||||
| 		rule = amodels.RuleRecord{} | ||||
| 	) | ||||
| 	if err := r.Decode(&rule, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	if err := app.automation.CreateRule(rule); err != nil { | ||||
| 	createdRule, err := app.automation.CreateRule(rule) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("Rule created successfully") | ||||
| 	return r.SendEnvelope(createdRule) | ||||
| } | ||||
|  | ||||
| // handleDeleteAutomationRule deletes an automation rule | ||||
| @@ -92,15 +98,12 @@ func handleDeleteAutomationRule(r *fastglue.Request) error { | ||||
| 		id, err = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid rule `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	err = app.automation.DeleteRule(id) | ||||
| 	if err != nil { | ||||
| 	if err = app.automation.DeleteRule(id); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("Rule deleted successfully") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleUpdateAutomationRuleWeights updates the weights of the automation rules | ||||
| @@ -110,27 +113,33 @@ func handleUpdateAutomationRuleWeights(r *fastglue.Request) error { | ||||
| 		weights = make(map[int]int) | ||||
| 	) | ||||
| 	if err := r.Decode(&weights, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	err := app.automation.UpdateRuleWeights(weights) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("Weights updated successfully") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleUpdateAutomationRuleExecutionMode updates the execution mode of the automation rules for a given type | ||||
| func handleUpdateAutomationRuleExecutionMode(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app  = r.Context.(*App) | ||||
| 		mode = string(r.RequestCtx.PostArgs().Peek("mode")) | ||||
| 		app = r.Context.(*App) | ||||
| 		req = updateAutomationRuleExecutionModeReq{} | ||||
| 	) | ||||
| 	if mode != amodels.ExecutionModeAll && mode != amodels.ExecutionModeFirstMatch { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid execution mode", nil, envelope.InputError) | ||||
|  | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil)) | ||||
| 	} | ||||
|  | ||||
| 	if req.Mode != amodels.ExecutionModeAll && req.Mode != amodels.ExecutionModeFirstMatch { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.T("automation.invalidRuleExecutionMode"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Only new conversation rules can be updated as they are the only ones that have execution mode. | ||||
| 	if err := app.automation.UpdateRuleExecutionMode(amodels.RuleTypeNewConversation, mode); err != nil { | ||||
| 	if err := app.automation.UpdateRuleExecutionMode(amodels.RuleTypeNewConversation, req.Mode); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("Execution mode updated successfully") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|   | ||||
| @@ -29,14 +29,14 @@ func handleGetBusinessHour(r *fastglue.Request) error { | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid business hour `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	businessHour, err := app.businessHours.Get(id) | ||||
| 	if err != nil { | ||||
| 		if err == businessHours.ErrBusinessHoursNotFound { | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusNotFound, err.Error(), nil, envelope.NotFoundError) | ||||
| 		} | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error fetching business hour", nil, "") | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorFetching", "name", "{globals.terms.businessHour}"), nil, "") | ||||
| 	} | ||||
| 	return r.SendEnvelope(businessHour) | ||||
| } | ||||
| @@ -48,18 +48,19 @@ func handleCreateBusinessHours(r *fastglue.Request) error { | ||||
| 		businessHours = models.BusinessHours{} | ||||
| 	) | ||||
| 	if err := r.Decode(&businessHours, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", err.Error(), envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if businessHours.Name == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty business hour `Name`", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`name`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := app.businessHours.Create(businessHours.Name, businessHours.Description, businessHours.IsAlwaysOpen, businessHours.Hours, businessHours.Holidays); err != nil { | ||||
| 	createdBusinessHours, err := app.businessHours.Create(businessHours.Name, businessHours.Description, businessHours.IsAlwaysOpen, businessHours.Hours, businessHours.Holidays) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| 	return r.SendEnvelope(createdBusinessHours) | ||||
| } | ||||
|  | ||||
| // handleDeleteBusinessHour deletes the business hour with the given id. | ||||
| @@ -69,14 +70,11 @@ func handleDeleteBusinessHour(r *fastglue.Request) error { | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid business hour `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	err = app.businessHours.Delete(id) | ||||
| 	if err != nil { | ||||
| 	if err = app.businessHours.Delete(id); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| @@ -88,20 +86,17 @@ func handleUpdateBusinessHours(r *fastglue.Request) error { | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid business hour `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := r.Decode(&businessHours, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", err.Error(), envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if businessHours.Name == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty business hour `Name`", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`name`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := app.businessHours.Update(id, businessHours.Name, businessHours.Description, businessHours.IsAlwaysOpen, businessHours.Hours, businessHours.Holidays); err != nil { | ||||
| 	updatedBusinessHours, err := app.businessHours.Update(id, businessHours.Name, businessHours.Description, businessHours.IsAlwaysOpen, businessHours.Hours, businessHours.Holidays) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| 	return r.SendEnvelope(updatedBusinessHours) | ||||
| } | ||||
|   | ||||
							
								
								
									
										63
									
								
								cmd/config.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								cmd/config.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
|  | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| // handleGetConfig returns the public configuration needed for app initialization, this includes minimal app settings and enabled SSO providers (without secrets). | ||||
| func handleGetConfig(r *fastglue.Request) error { | ||||
| 	var app = r.Context.(*App) | ||||
|  | ||||
| 	// Get app settings | ||||
| 	settingsJSON, err := app.setting.GetByPrefix("app") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Unmarshal settings | ||||
| 	var settings map[string]any | ||||
| 	if err := json.Unmarshal(settingsJSON, &settings); err != nil { | ||||
| 		app.lo.Error("error unmarshalling settings", "err", err) | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorFetching", "name", app.i18n.T("globals.terms.setting")), nil)) | ||||
| 	} | ||||
|  | ||||
| 	// Filter to only include public fields needed for initial app load | ||||
| 	publicSettings := map[string]any{ | ||||
| 		"app.lang":        settings["app.lang"], | ||||
| 		"app.favicon_url": settings["app.favicon_url"], | ||||
| 		"app.logo_url":    settings["app.logo_url"], | ||||
| 		"app.site_name":   settings["app.site_name"], | ||||
| 	} | ||||
|  | ||||
| 	// Get all OIDC providers | ||||
| 	oidcProviders, err := app.oidc.GetAll() | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Filter for enabled providers and remove client_secret | ||||
| 	enabledProviders := make([]map[string]any, 0) | ||||
| 	for _, provider := range oidcProviders { | ||||
| 		if provider.Enabled { | ||||
| 			providerMap := map[string]any{ | ||||
| 				"id":           provider.ID, | ||||
| 				"name":         provider.Name, | ||||
| 				"provider":     provider.Provider, | ||||
| 				"provider_url": provider.ProviderURL, | ||||
| 				"client_id":    provider.ClientID, | ||||
| 				"logo_url":     provider.ProviderLogoURL, | ||||
| 				"enabled":      provider.Enabled, | ||||
| 				"redirect_uri": provider.RedirectURI, | ||||
| 			} | ||||
| 			enabledProviders = append(enabledProviders, providerMap) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Add SSO providers to the response | ||||
| 	publicSettings["app.sso_providers"] = enabledProviders | ||||
|  | ||||
| 	return r.SendEnvelope(publicSettings) | ||||
| } | ||||
							
								
								
									
										288
									
								
								cmd/contacts.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										288
									
								
								cmd/contacts.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,288 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"path/filepath" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
|  | ||||
| 	amodels "github.com/abhinavxd/libredesk/internal/auth/models" | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| 	"github.com/abhinavxd/libredesk/internal/stringutil" | ||||
| 	"github.com/abhinavxd/libredesk/internal/user/models" | ||||
| 	"github.com/valyala/fasthttp" | ||||
| 	"github.com/volatiletech/null/v9" | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| type createContactNoteReq struct { | ||||
| 	Note string `json:"note"` | ||||
| } | ||||
|  | ||||
| type blockContactReq struct { | ||||
| 	Enabled bool `json:"enabled"` | ||||
| } | ||||
|  | ||||
| // handleGetContacts returns a list of contacts from the database. | ||||
| func handleGetContacts(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app         = r.Context.(*App) | ||||
| 		order       = string(r.RequestCtx.QueryArgs().Peek("order")) | ||||
| 		orderBy     = string(r.RequestCtx.QueryArgs().Peek("order_by")) | ||||
| 		filters     = string(r.RequestCtx.QueryArgs().Peek("filters")) | ||||
| 		page, _     = strconv.Atoi(string(r.RequestCtx.QueryArgs().Peek("page"))) | ||||
| 		pageSize, _ = strconv.Atoi(string(r.RequestCtx.QueryArgs().Peek("page_size"))) | ||||
| 		total       = 0 | ||||
| 	) | ||||
| 	contacts, err := app.user.GetContacts(page, pageSize, order, orderBy, filters) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	if len(contacts) > 0 { | ||||
| 		total = contacts[0].Total | ||||
| 	} | ||||
| 	return r.SendEnvelope(envelope.PageResults{ | ||||
| 		Results:    contacts, | ||||
| 		Total:      total, | ||||
| 		PerPage:    pageSize, | ||||
| 		TotalPages: (total + pageSize - 1) / pageSize, | ||||
| 		Page:       page, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // handleGetTags returns a contact from the database. | ||||
| func handleGetContact(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| 		id, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
| 	if id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	c, err := app.user.GetContact(id, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(c) | ||||
| } | ||||
|  | ||||
| // handleUpdateContact updates a contact in the database. | ||||
| func handleUpdateContact(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| 		id, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
| 	if id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	contact, err := app.user.GetContact(id, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	form, err := r.RequestCtx.MultipartForm() | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error parsing form data", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	// Parse form data | ||||
| 	firstName := "" | ||||
| 	if v, ok := form.Value["first_name"]; ok && len(v) > 0 { | ||||
| 		firstName = string(v[0]) | ||||
| 	} | ||||
| 	lastName := "" | ||||
| 	if v, ok := form.Value["last_name"]; ok && len(v) > 0 { | ||||
| 		lastName = string(v[0]) | ||||
| 	} | ||||
| 	email := "" | ||||
| 	if v, ok := form.Value["email"]; ok && len(v) > 0 { | ||||
| 		email = strings.TrimSpace(string(v[0])) | ||||
| 	} | ||||
| 	phoneNumber := "" | ||||
| 	if v, ok := form.Value["phone_number"]; ok && len(v) > 0 { | ||||
| 		phoneNumber = string(v[0]) | ||||
| 	} | ||||
| 	phoneNumberCallingCode := "" | ||||
| 	if v, ok := form.Value["phone_number_calling_code"]; ok && len(v) > 0 { | ||||
| 		phoneNumberCallingCode = string(v[0]) | ||||
| 	} | ||||
| 	avatarURL := "" | ||||
| 	if v, ok := form.Value["avatar_url"]; ok && len(v) > 0 { | ||||
| 		avatarURL = string(v[0]) | ||||
| 	} | ||||
|  | ||||
| 	// Set nulls to empty strings. | ||||
| 	if avatarURL == "null" { | ||||
| 		avatarURL = "" | ||||
| 	} | ||||
| 	if phoneNumberCallingCode == "null" { | ||||
| 		phoneNumberCallingCode = "" | ||||
| 	} | ||||
| 	if phoneNumber == "null" { | ||||
| 		phoneNumber = "" | ||||
| 	} | ||||
|  | ||||
| 	// Validate mandatory fields. | ||||
| 	if email == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "email"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	if !stringutil.ValidEmail(email) { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "email"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	if firstName == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "first_name"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Another contact with same new email? | ||||
| 	existingContact, _ := app.user.GetContact(0, email) | ||||
| 	if existingContact.ID > 0 && existingContact.ID != id { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.T("contact.alreadyExistsWithEmail"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	contactToUpdate := models.User{ | ||||
| 		FirstName:              firstName, | ||||
| 		LastName:               lastName, | ||||
| 		Email:                  null.StringFrom(email), | ||||
| 		AvatarURL:              null.NewString(avatarURL, avatarURL != ""), | ||||
| 		PhoneNumber:            null.NewString(phoneNumber, phoneNumber != ""), | ||||
| 		PhoneNumberCallingCode: null.NewString(phoneNumberCallingCode, phoneNumberCallingCode != ""), | ||||
| 	} | ||||
|  | ||||
| 	if err := app.user.UpdateContact(id, contactToUpdate); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Delete avatar? | ||||
| 	if avatarURL == "" && contact.AvatarURL.Valid { | ||||
| 		fileName := filepath.Base(contact.AvatarURL.String) | ||||
| 		app.media.Delete(fileName) | ||||
| 		contact.AvatarURL.Valid = false | ||||
| 		contact.AvatarURL.String = "" | ||||
| 	} | ||||
|  | ||||
| 	// Upload avatar? | ||||
| 	files, ok := form.File["files"] | ||||
| 	if ok && len(files) > 0 { | ||||
| 		if err := uploadUserAvatar(r, contact, files); err != nil { | ||||
| 			return sendErrorEnvelope(r, err) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Refetch contact and return it | ||||
| 	contact, err = app.user.GetContact(id, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(contact) | ||||
| } | ||||
|  | ||||
| // handleGetContactNotes returns all notes for a contact. | ||||
| func handleGetContactNotes(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app          = r.Context.(*App) | ||||
| 		contactID, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
| 	if contactID <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	notes, err := app.user.GetNotes(contactID) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(notes) | ||||
| } | ||||
|  | ||||
| // handleCreateContactNote creates a note for a contact. | ||||
| func handleCreateContactNote(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app          = r.Context.(*App) | ||||
| 		contactID, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 		auser        = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		req          = createContactNoteReq{} | ||||
| 	) | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil)) | ||||
| 	} | ||||
| 	if len(req.Note) == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "note"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	n, err := app.user.CreateNote(contactID, auser.ID, req.Note) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	n, err = app.user.GetNote(n.ID) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(n) | ||||
| } | ||||
|  | ||||
| // handleDeleteContactNote deletes a note for a contact. | ||||
| func handleDeleteContactNote(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app          = r.Context.(*App) | ||||
| 		contactID, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 		noteID, _    = strconv.Atoi(r.RequestCtx.UserValue("note_id").(string)) | ||||
| 		auser        = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 	) | ||||
| 	if contactID <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	if noteID <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`note_id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	agent, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Allow deletion of only own notes and not those created by others, but also allow `Admin` to delete any note. | ||||
| 	if !agent.HasAdminRole() { | ||||
| 		note, err := app.user.GetNote(noteID) | ||||
| 		if err != nil { | ||||
| 			return sendErrorEnvelope(r, err) | ||||
| 		} | ||||
| 		if note.UserID != auser.ID { | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusForbidden, app.i18n.Ts("globals.messages.canOnlyDeleteOwn", "name", "{globals.terms.note}"), nil, envelope.InputError) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	app.lo.Info("deleting contact note", "note_id", noteID, "contact_id", contactID, "actor_id", auser.ID) | ||||
|  | ||||
| 	if err := app.user.DeleteNote(noteID, contactID); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleBlockContact blocks a contact. | ||||
| func handleBlockContact(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app          = r.Context.(*App) | ||||
| 		contactID, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 		auser        = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		req          = blockContactReq{} | ||||
| 	) | ||||
|  | ||||
| 	if contactID <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil)) | ||||
| 	} | ||||
|  | ||||
| 	app.lo.Info("setting contact block status", "contact_id", contactID, "enabled", req.Enabled, "actor_id", auser.ID) | ||||
|  | ||||
| 	if err := app.user.ToggleEnabled(contactID, models.UserTypeContact, req.Enabled); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	contact, err := app.user.GetContact(contactID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(contact) | ||||
| } | ||||
| @@ -1,7 +1,6 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"strconv" | ||||
| 	"time" | ||||
|  | ||||
| @@ -10,12 +9,49 @@ import ( | ||||
| 	"github.com/abhinavxd/libredesk/internal/automation/models" | ||||
| 	cmodels "github.com/abhinavxd/libredesk/internal/conversation/models" | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| 	medModels "github.com/abhinavxd/libredesk/internal/media/models" | ||||
| 	"github.com/abhinavxd/libredesk/internal/stringutil" | ||||
| 	umodels "github.com/abhinavxd/libredesk/internal/user/models" | ||||
| 	wmodels "github.com/abhinavxd/libredesk/internal/webhook/models" | ||||
| 	"github.com/valyala/fasthttp" | ||||
| 	"github.com/volatiletech/null/v9" | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| type assigneeChangeReq struct { | ||||
| 	AssigneeID int `json:"assignee_id"` | ||||
| } | ||||
|  | ||||
| type teamAssigneeChangeReq struct { | ||||
| 	AssigneeID int `json:"assignee_id"` | ||||
| } | ||||
|  | ||||
| type priorityUpdateReq struct { | ||||
| 	Priority string `json:"priority"` | ||||
| } | ||||
|  | ||||
| type statusUpdateReq struct { | ||||
| 	Status       string `json:"status"` | ||||
| 	SnoozedUntil string `json:"snoozed_until,omitempty"` | ||||
| } | ||||
|  | ||||
| type tagsUpdateReq struct { | ||||
| 	Tags []string `json:"tags"` | ||||
| } | ||||
|  | ||||
| type createConversationRequest struct { | ||||
| 	InboxID         int    `json:"inbox_id"` | ||||
| 	AssignedAgentID int    `json:"agent_id"` | ||||
| 	AssignedTeamID  int    `json:"team_id"` | ||||
| 	Email           string `json:"contact_email"` | ||||
| 	FirstName       string `json:"first_name"` | ||||
| 	LastName        string `json:"last_name"` | ||||
| 	Subject         string `json:"subject"` | ||||
| 	Content         string `json:"content"` | ||||
| 	Attachments     []int  `json:"attachments"` | ||||
| 	Initiator       string `json:"initiator"` // "contact" | "agent" | ||||
| } | ||||
|  | ||||
| // handleGetAllConversations retrieves all conversations. | ||||
| func handleGetAllConversations(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| @@ -37,14 +73,6 @@ func handleGetAllConversations(r *fastglue.Request) error { | ||||
| 		total = conversations[0].Total | ||||
| 	} | ||||
|  | ||||
| 	// Set deadlines for SLA if conversation has a policy | ||||
| 	for i := range conversations { | ||||
| 		if conversations[i].SLAPolicyID.Int != 0 { | ||||
| 			setSLADeadlines(app, &conversations[i]) | ||||
| 		} | ||||
| 		conversations[i].ID = 0 | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(envelope.PageResults{ | ||||
| 		Results:    conversations, | ||||
| 		Total:      total, | ||||
| @@ -68,20 +96,12 @@ func handleGetAssignedConversations(r *fastglue.Request) error { | ||||
| 	) | ||||
| 	conversations, err := app.conversation.GetAssignedConversationsList(user.ID, order, orderBy, filters, page, pageSize) | ||||
| 	if err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, err.Error(), nil, "") | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	if len(conversations) > 0 { | ||||
| 		total = conversations[0].Total | ||||
| 	} | ||||
|  | ||||
| 	// Set deadlines for SLA if conversation has a policy | ||||
| 	for i := range conversations { | ||||
| 		if conversations[i].SLAPolicyID.Int != 0 { | ||||
| 			setSLADeadlines(app, &conversations[i]) | ||||
| 		} | ||||
| 		conversations[i].ID = 0 | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(envelope.PageResults{ | ||||
| 		Results:    conversations, | ||||
| 		Total:      total, | ||||
| @@ -105,20 +125,12 @@ func handleGetUnassignedConversations(r *fastglue.Request) error { | ||||
|  | ||||
| 	conversations, err := app.conversation.GetUnassignedConversationsList(order, orderBy, filters, page, pageSize) | ||||
| 	if err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, err.Error(), nil, "") | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	if len(conversations) > 0 { | ||||
| 		total = conversations[0].Total | ||||
| 	} | ||||
|  | ||||
| 	// Set deadlines for SLA if conversation has a policy | ||||
| 	for i := range conversations { | ||||
| 		if conversations[i].SLAPolicyID.Int != 0 { | ||||
| 			setSLADeadlines(app, &conversations[i]) | ||||
| 		} | ||||
| 		conversations[i].ID = 0 | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(envelope.PageResults{ | ||||
| 		Results:    conversations, | ||||
| 		Total:      total, | ||||
| @@ -141,7 +153,7 @@ func handleGetViewConversations(r *fastglue.Request) error { | ||||
| 		total       = 0 | ||||
| 	) | ||||
| 	if viewID < 1 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid `view_id`", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`view_id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Check if user has access to the view. | ||||
| @@ -150,15 +162,15 @@ func handleGetViewConversations(r *fastglue.Request) error { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	if view.UserID != auser.ID { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusForbidden, "You don't have access to this view.", nil, envelope.PermissionError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusForbidden, app.i18n.T("conversation.viewPermissionDenied"), nil, envelope.PermissionError) | ||||
| 	} | ||||
|  | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Prepare lists user has access to based on user permissions, internally this affects the SQL query. | ||||
| 	// Prepare lists user has access to based on user permissions, internally this prepares the SQL query. | ||||
| 	lists := []string{} | ||||
| 	for _, perm := range user.Permissions { | ||||
| 		if perm == authzModels.PermConversationsReadAll { | ||||
| @@ -179,7 +191,7 @@ func handleGetViewConversations(r *fastglue.Request) error { | ||||
|  | ||||
| 	// No lists found, user doesn't have access to any conversations. | ||||
| 	if len(lists) == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusForbidden, "Permission denied", nil, envelope.PermissionError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusForbidden, app.i18n.Ts("globals.messages.denied", "name", "{globals.terms.permission}"), nil, envelope.PermissionError) | ||||
| 	} | ||||
|  | ||||
| 	conversations, err := app.conversation.GetViewConversationsList(user.ID, user.Teams.IDs(), lists, order, orderBy, string(view.Filters), page, pageSize) | ||||
| @@ -190,14 +202,6 @@ func handleGetViewConversations(r *fastglue.Request) error { | ||||
| 		total = conversations[0].Total | ||||
| 	} | ||||
|  | ||||
| 	// Set deadlines for SLA if conversation has a policy | ||||
| 	for i := range conversations { | ||||
| 		if conversations[i].SLAPolicyID.Int != 0 { | ||||
| 			setSLADeadlines(app, &conversations[i]) | ||||
| 		} | ||||
| 		conversations[i].ID = 0 | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(envelope.PageResults{ | ||||
| 		Results:    conversations, | ||||
| 		Total:      total, | ||||
| @@ -222,7 +226,7 @@ func handleGetTeamUnassignedConversations(r *fastglue.Request) error { | ||||
| 	) | ||||
| 	teamID, _ := strconv.Atoi(teamIDStr) | ||||
| 	if teamID < 1 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid `team_id`", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`team_id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Check if user belongs to the team. | ||||
| @@ -232,7 +236,7 @@ func handleGetTeamUnassignedConversations(r *fastglue.Request) error { | ||||
| 	} | ||||
|  | ||||
| 	if !exists { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.PermissionError, "You're not a member of this team, Please refresh the page and try again.", nil)) | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.PermissionError, app.i18n.T("conversation.notMemberOfTeam"), nil)) | ||||
| 	} | ||||
|  | ||||
| 	conversations, err := app.conversation.GetTeamUnassignedConversationsList(teamID, order, orderBy, filters, page, pageSize) | ||||
| @@ -243,14 +247,6 @@ func handleGetTeamUnassignedConversations(r *fastglue.Request) error { | ||||
| 		total = conversations[0].Total | ||||
| 	} | ||||
|  | ||||
| 	// Set deadlines for SLA if conversation has a policy | ||||
| 	for i := range conversations { | ||||
| 		if conversations[i].SLAPolicyID.Int != 0 { | ||||
| 			setSLADeadlines(app, &conversations[i]) | ||||
| 		} | ||||
| 		conversations[i].ID = 0 | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(envelope.PageResults{ | ||||
| 		Results:    conversations, | ||||
| 		Total:      total, | ||||
| @@ -268,7 +264,7 @@ func handleGetConversation(r *fastglue.Request) error { | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 	) | ||||
|  | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| @@ -278,13 +274,8 @@ func handleGetConversation(r *fastglue.Request) error { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	if conv.SLAPolicyID.Int != 0 { | ||||
| 		setSLADeadlines(app, conv) | ||||
| 	} | ||||
|  | ||||
| 	prev, _ := app.conversation.GetContactConversations(conv.ContactID) | ||||
| 	conv.PreviousConversations = filterCurrentConv(prev, conv.UUID) | ||||
| 	conv.ID = 0 | ||||
| 	prev, _ := app.conversation.GetContactPreviousConversations(conv.ContactID, 10) | ||||
| 	conv.PreviousConversations = filterCurrentPreviousConv(prev, conv.UUID) | ||||
| 	return r.SendEnvelope(conv) | ||||
| } | ||||
|  | ||||
| @@ -295,7 +286,7 @@ func handleUpdateConversationAssigneeLastSeen(r *fastglue.Request) error { | ||||
| 		uuid  = r.RequestCtx.UserValue("uuid").(string) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 	) | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| @@ -306,7 +297,7 @@ func handleUpdateConversationAssigneeLastSeen(r *fastglue.Request) error { | ||||
| 	if err = app.conversation.UpdateConversationAssigneeLastSeen(uuid); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("Last seen updated successfully") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleGetConversationParticipants retrieves participants of a conversation. | ||||
| @@ -316,7 +307,7 @@ func handleGetConversationParticipants(r *fastglue.Request) error { | ||||
| 		uuid  = r.RequestCtx.UserValue("uuid").(string) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 	) | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| @@ -334,33 +325,37 @@ func handleGetConversationParticipants(r *fastglue.Request) error { | ||||
| // handleUpdateUserAssignee updates the user assigned to a conversation. | ||||
| func handleUpdateUserAssignee(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app        = r.Context.(*App) | ||||
| 		uuid       = r.RequestCtx.UserValue("uuid").(string) | ||||
| 		auser      = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		assigneeID = r.RequestCtx.PostArgs().GetUintOrZero("assignee_id") | ||||
| 		app   = r.Context.(*App) | ||||
| 		uuid  = r.RequestCtx.UserValue("uuid").(string) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		req   = assigneeChangeReq{} | ||||
| 	) | ||||
| 	if assigneeID == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid `assignee_id`", nil, envelope.InputError) | ||||
|  | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		app.lo.Error("error decoding assignee change request", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	_, err = enforceConversationAccess(app, uuid, user) | ||||
| 	conversation, err := enforceConversationAccess(app, uuid, user) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	if err := app.conversation.UpdateConversationUserAssignee(uuid, assigneeID, user); err != nil { | ||||
| 	// Already assigned? | ||||
| 	if conversation.AssignedUserID.Int == req.AssigneeID { | ||||
| 		return r.SendEnvelope(true) | ||||
| 	} | ||||
|  | ||||
| 	if err := app.conversation.UpdateConversationUserAssignee(uuid, req.AssigneeID, user); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Evaluate automation rules. | ||||
| 	app.automation.EvaluateConversationUpdateRules(uuid, models.EventConversationUserAssigned) | ||||
|  | ||||
| 	return r.SendEnvelope("User assigned successfully") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleUpdateTeamAssignee updates the team assigned to a conversation. | ||||
| @@ -369,13 +364,17 @@ func handleUpdateTeamAssignee(r *fastglue.Request) error { | ||||
| 		app   = r.Context.(*App) | ||||
| 		uuid  = r.RequestCtx.UserValue("uuid").(string) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		req   = teamAssigneeChangeReq{} | ||||
| 	) | ||||
| 	assigneeID, err := r.RequestCtx.PostArgs().GetUint("assignee_id") | ||||
| 	if err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid assignee `id`.", nil, envelope.InputError) | ||||
|  | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		app.lo.Error("error decoding team assignee change request", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	assigneeID := req.AssigneeID | ||||
|  | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| @@ -389,89 +388,85 @@ func handleUpdateTeamAssignee(r *fastglue.Request) error { | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Already assigned? | ||||
| 	if conversation.AssignedTeamID.Int == assigneeID { | ||||
| 		return r.SendEnvelope(true) | ||||
| 	} | ||||
| 	if err := app.conversation.UpdateConversationTeamAssignee(uuid, assigneeID, user); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Evaluate automation rules on team assignment. | ||||
| 	app.automation.EvaluateConversationUpdateRules(uuid, models.EventConversationTeamAssigned) | ||||
|  | ||||
| 	// Apply SLA policy if team has changed and the new team has an SLA policy. | ||||
| 	if conversation.AssignedTeamID.Int != assigneeID && assigneeID != 0 { | ||||
| 		team, err := app.team.Get(assigneeID) | ||||
| 		if err != nil { | ||||
| 			return sendErrorEnvelope(r, err) | ||||
| 		} | ||||
| 		if team.SLAPolicyID.Int != 0 { | ||||
| 			if err := app.conversation.ApplySLA(*conversation, team.SLAPolicyID.Int, user); err != nil { | ||||
| 				return sendErrorEnvelope(r, err) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return r.SendEnvelope("Team assigned successfully") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleUpdateConversationPriority updates the priority of a conversation. | ||||
| func handleUpdateConversationPriority(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app      = r.Context.(*App) | ||||
| 		uuid     = r.RequestCtx.UserValue("uuid").(string) | ||||
| 		auser    = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		priority = string(r.RequestCtx.PostArgs().Peek("priority")) | ||||
| 		app   = r.Context.(*App) | ||||
| 		uuid  = r.RequestCtx.UserValue("uuid").(string) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		req   = priorityUpdateReq{} | ||||
| 	) | ||||
|  | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		app.lo.Error("error decoding priority update request", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	priority := req.Priority | ||||
| 	if priority == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid `priority`", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`priority`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	conversation, err := app.conversation.GetConversation(0, uuid) | ||||
|  | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	_, err = enforceConversationAccess(app, uuid, user) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	allowed, err := app.authz.EnforceConversationAccess(user, conversation) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	if !allowed { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.PermissionError, "Permission denied", nil)) | ||||
| 	} | ||||
| 	if err := app.conversation.UpdateConversationPriority(uuid, 0 /**priority_id**/, priority, user); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Evaluate automation rules. | ||||
| 	app.automation.EvaluateConversationUpdateRules(uuid, models.EventConversationPriorityChange) | ||||
| 	return r.SendEnvelope("Priority updated successfully") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleUpdateConversationStatus updates the status of a conversation. | ||||
| func handleUpdateConversationStatus(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app          = r.Context.(*App) | ||||
| 		status       = string(r.RequestCtx.PostArgs().Peek("status")) | ||||
| 		snoozedUntil = string(r.RequestCtx.PostArgs().Peek("snoozed_until")) | ||||
| 		uuid         = r.RequestCtx.UserValue("uuid").(string) | ||||
| 		auser        = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		app   = r.Context.(*App) | ||||
| 		uuid  = r.RequestCtx.UserValue("uuid").(string) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		req   = statusUpdateReq{} | ||||
| 	) | ||||
|  | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		app.lo.Error("error decoding status update request", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	status := req.Status | ||||
| 	snoozedUntil := req.SnoozedUntil | ||||
|  | ||||
| 	// Validate inputs | ||||
| 	if status == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid `status`", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`status`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	if snoozedUntil == "" && status == cmodels.StatusSnoozed { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid `snoozed_until`", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`snoozed_until`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	if status == cmodels.StatusSnoozed { | ||||
| 		_, err := time.ParseDuration(snoozedUntil) | ||||
| 		if err != nil { | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid `snoozed_until`", nil, envelope.InputError) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`snoozed_until`"), nil, envelope.InputError) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Enforce conversation access. | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| @@ -482,7 +477,7 @@ func handleUpdateConversationStatus(r *fastglue.Request) error { | ||||
|  | ||||
| 	// Make sure a user is assigned before resolving conversation. | ||||
| 	if status == cmodels.StatusResolved && conversation.AssignedUserID.Int == 0 { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.InputError, "Cannot resolve the conversation without an assigned user, Please assign a user before attempting to resolve.", nil)) | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.InputError, app.i18n.T("conversation.resolveWithoutAssignee"), nil)) | ||||
| 	} | ||||
|  | ||||
| 	// Update conversation status. | ||||
| @@ -490,9 +485,6 @@ func handleUpdateConversationStatus(r *fastglue.Request) error { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Evaluate automation rules. | ||||
| 	app.automation.EvaluateConversationUpdateRules(uuid, models.EventConversationStatusChange) | ||||
|  | ||||
| 	// If status is `Resolved`, send CSAT survey if enabled on inbox. | ||||
| 	if status == cmodels.StatusResolved { | ||||
| 		// Check if CSAT is enabled on the inbox and send CSAT survey message. | ||||
| @@ -506,67 +498,98 @@ func handleUpdateConversationStatus(r *fastglue.Request) error { | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return r.SendEnvelope("Status updated successfully") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleUpdateConversationtags updates conversation tags. | ||||
| func handleUpdateConversationtags(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app      = r.Context.(*App) | ||||
| 		tagNames = []string{} | ||||
| 		tagJSON  = r.RequestCtx.PostArgs().Peek("tags") | ||||
| 		auser    = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		uuid     = r.RequestCtx.UserValue("uuid").(string) | ||||
| 		app   = r.Context.(*App) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		uuid  = r.RequestCtx.UserValue("uuid").(string) | ||||
| 		req   = tagsUpdateReq{} | ||||
| 	) | ||||
|  | ||||
| 	if err := json.Unmarshal(tagJSON, &tagNames); err != nil { | ||||
| 		app.lo.Error("error unmarshalling tags JSON", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error unmarshalling tags JSON", nil, envelope.GeneralError) | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		app.lo.Error("error decoding tags update request", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	conversation, err := app.conversation.GetConversation(0, uuid) | ||||
|  | ||||
| 	tagNames := req.Tags | ||||
|  | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	_, err = enforceConversationAccess(app, uuid, user) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	if err != nil { | ||||
| 	if err := app.conversation.SetConversationTags(uuid, models.ActionSetTags, tagNames, user); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	if allowed, err := app.authz.EnforceConversationAccess(user, conversation); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} else if !allowed { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.PermissionError, "Permission denied", nil)) | ||||
| 	} | ||||
|  | ||||
| 	if err := app.conversation.UpsertConversationTags(uuid, tagNames, user); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("Tags added successfully") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleDashboardCounts retrieves general dashboard counts for all users. | ||||
| func handleDashboardCounts(r *fastglue.Request) error { | ||||
| // handleUpdateConversationCustomAttributes updates custom attributes of a conversation. | ||||
| func handleUpdateConversationCustomAttributes(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 		app        = r.Context.(*App) | ||||
| 		attributes = map[string]any{} | ||||
| 		auser      = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		uuid       = r.RequestCtx.UserValue("uuid").(string) | ||||
| 	) | ||||
| 	counts, err := app.conversation.GetDashboardCounts(0, 0) | ||||
| 	if err := r.Decode(&attributes, ""); err != nil { | ||||
| 		app.lo.Error("error unmarshalling custom attributes JSON", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Enforce conversation access. | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(counts) | ||||
| 	_, err = enforceConversationAccess(app, uuid, user) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Update custom attributes. | ||||
| 	if err := app.conversation.UpdateConversationCustomAttributes(uuid, attributes); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleDashboardCharts retrieves general dashboard chart data. | ||||
| func handleDashboardCharts(r *fastglue.Request) error { | ||||
| // handleUpdateContactCustomAttributes updates custom attributes of a contact. | ||||
| func handleUpdateContactCustomAttributes(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 		app        = r.Context.(*App) | ||||
| 		attributes = map[string]any{} | ||||
| 		auser      = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		uuid       = r.RequestCtx.UserValue("uuid").(string) | ||||
| 	) | ||||
| 	charts, err := app.conversation.GetDashboardChart(0, 0) | ||||
| 	if err := r.Decode(&attributes, ""); err != nil { | ||||
| 		app.lo.Error("error unmarshalling custom attributes JSON", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Enforce conversation access. | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(charts) | ||||
| 	conversation, err := enforceConversationAccess(app, uuid, user) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	if err := app.user.UpdateCustomAttributes(conversation.ContactID, attributes); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	// Broadcast update. | ||||
| 	app.conversation.BroadcastConversationUpdate(conversation.UUID, "contact.custom_attributes", attributes) | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // enforceConversationAccess fetches the conversation and checks if the user has access to it. | ||||
| @@ -577,7 +600,7 @@ func enforceConversationAccess(app *App, uuid string, user umodels.User) (*cmode | ||||
| 	} | ||||
| 	allowed, err := app.authz.EnforceConversationAccess(user, conversation) | ||||
| 	if err != nil { | ||||
| 		return nil, envelope.NewError(envelope.GeneralError, "Error checking permissions", nil) | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if !allowed { | ||||
| 		return nil, envelope.NewError(envelope.PermissionError, "Permission denied", nil) | ||||
| @@ -585,21 +608,6 @@ func enforceConversationAccess(app *App, uuid string, user umodels.User) (*cmode | ||||
| 	return &conversation, nil | ||||
| } | ||||
|  | ||||
| // setSLADeadlines gets the latest SLA deadlines for a conversation and sets them. | ||||
| func setSLADeadlines(app *App, conversation *cmodels.Conversation) error { | ||||
| 	if conversation.ID < 1 { | ||||
| 		return nil | ||||
| 	} | ||||
| 	first, resolution, err := app.sla.GetLatestDeadlines(conversation.ID) | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error getting SLA deadlines", "id", conversation.ID, "error", err) | ||||
| 		return err | ||||
| 	} | ||||
| 	conversation.FirstResponseDueAt = null.NewTime(first, first != time.Time{}) | ||||
| 	conversation.ResolutionDueAt = null.NewTime(resolution, resolution != time.Time{}) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // handleRemoveUserAssignee removes the user assigned to a conversation. | ||||
| func handleRemoveUserAssignee(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| @@ -607,7 +615,7 @@ func handleRemoveUserAssignee(r *fastglue.Request) error { | ||||
| 		uuid  = r.RequestCtx.UserValue("uuid").(string) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 	) | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| @@ -615,7 +623,7 @@ func handleRemoveUserAssignee(r *fastglue.Request) error { | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	if err = app.conversation.RemoveConversationAssignee(uuid, "user"); err != nil { | ||||
| 	if err = app.conversation.RemoveConversationAssignee(uuid, "user", user); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(true) | ||||
| @@ -628,7 +636,7 @@ func handleRemoveTeamAssignee(r *fastglue.Request) error { | ||||
| 		uuid  = r.RequestCtx.UserValue("uuid").(string) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 	) | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| @@ -636,18 +644,155 @@ func handleRemoveTeamAssignee(r *fastglue.Request) error { | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	if err = app.conversation.RemoveConversationAssignee(uuid, "team"); err != nil { | ||||
| 	if err = app.conversation.RemoveConversationAssignee(uuid, "team", user); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // filterCurrentConv removes the current conversation from the list of conversations. | ||||
| func filterCurrentConv(convs []cmodels.Conversation, uuid string) []cmodels.Conversation { | ||||
| // filterCurrentPreviousConv removes the current conversation from the list of previous conversations. | ||||
| func filterCurrentPreviousConv(convs []cmodels.PreviousConversation, uuid string) []cmodels.PreviousConversation { | ||||
| 	for i, c := range convs { | ||||
| 		if c.UUID == uuid { | ||||
| 			return append(convs[:i], convs[i+1:]...) | ||||
| 		} | ||||
| 	} | ||||
| 	return []cmodels.Conversation{} | ||||
| 	return []cmodels.PreviousConversation{} | ||||
| } | ||||
|  | ||||
| // handleCreateConversation creates a new conversation and sends a message to it. | ||||
| func handleCreateConversation(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		req   = createConversationRequest{} | ||||
| 	) | ||||
|  | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		app.lo.Error("error decoding create conversation request", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Validate the request | ||||
| 	if err := validateCreateConversationRequest(req, app); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	to := []string{req.Email} | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Find or create contact. | ||||
| 	contact := umodels.User{ | ||||
| 		Email:           null.StringFrom(req.Email), | ||||
| 		SourceChannelID: null.StringFrom(req.Email), | ||||
| 		FirstName:       req.FirstName, | ||||
| 		LastName:        req.LastName, | ||||
| 		InboxID:         req.InboxID, | ||||
| 	} | ||||
| 	if err := app.user.CreateContact(&contact); err != nil { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorCreating", "name", "{globals.terms.contact}"), nil)) | ||||
| 	} | ||||
|  | ||||
| 	// Create conversation first. | ||||
| 	conversationID, conversationUUID, err := app.conversation.CreateConversation( | ||||
| 		contact.ID, | ||||
| 		contact.ContactChannelID, | ||||
| 		req.InboxID, | ||||
| 		"",         /** last_message **/ | ||||
| 		time.Now(), /** last_message_at **/ | ||||
| 		req.Subject, | ||||
| 		true, /** append reference number to subject? **/ | ||||
| 	) | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error creating conversation", "error", err) | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorCreating", "name", "{globals.terms.conversation}"), nil)) | ||||
| 	} | ||||
|  | ||||
| 	// Get media for the attachment ids. | ||||
| 	var media = make([]medModels.Media, 0, len(req.Attachments)) | ||||
| 	for _, id := range req.Attachments { | ||||
| 		m, err := app.media.Get(id, "") | ||||
| 		if err != nil { | ||||
| 			app.lo.Error("error fetching media", "error", err) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorFetching", "name", "{globals.terms.media}"), nil, envelope.GeneralError) | ||||
| 		} | ||||
| 		media = append(media, m) | ||||
| 	} | ||||
|  | ||||
| 	// Send initial message based on the initiator of conversation. | ||||
| 	switch req.Initiator { | ||||
| 	case umodels.UserTypeAgent: | ||||
| 		// Queue reply. | ||||
| 		if _, err := app.conversation.QueueReply(media, req.InboxID, auser.ID /**sender_id**/, conversationUUID, req.Content, to, nil /**cc**/, nil /**bcc**/, map[string]any{} /**meta**/); err != nil { | ||||
| 			// Delete the conversation if msg queue fails. | ||||
| 			if err := app.conversation.DeleteConversation(conversationUUID); err != nil { | ||||
| 				app.lo.Error("error deleting conversation", "error", err) | ||||
| 			} | ||||
| 			return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorSending", "name", "{globals.terms.message}"), nil)) | ||||
| 		} | ||||
| 	case umodels.UserTypeContact: | ||||
| 		// Create message on behalf of contact. | ||||
| 		if _, err := app.conversation.CreateContactMessage(media, contact.ID, conversationUUID, req.Content, cmodels.ContentTypeHTML); err != nil { | ||||
| 			// Delete the conversation if message creation fails. | ||||
| 			if err := app.conversation.DeleteConversation(conversationUUID); err != nil { | ||||
| 				app.lo.Error("error deleting conversation", "error", err) | ||||
| 			} | ||||
| 			return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorSending", "name", "{globals.terms.message}"), nil)) | ||||
| 		} | ||||
| 	default: | ||||
| 		// Guard anyway. | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`initiator`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Assign the conversation to the agent or team. | ||||
| 	if req.AssignedAgentID > 0 { | ||||
| 		app.conversation.UpdateConversationUserAssignee(conversationUUID, req.AssignedAgentID, user) | ||||
| 	} | ||||
| 	if req.AssignedTeamID > 0 { | ||||
| 		app.conversation.UpdateConversationTeamAssignee(conversationUUID, req.AssignedTeamID, user) | ||||
| 	} | ||||
|  | ||||
| 	// Trigger webhook event for conversation created. | ||||
| 	conversation, err := app.conversation.GetConversation(conversationID, "") | ||||
| 	if err == nil { | ||||
| 		app.webhook.TriggerEvent(wmodels.EventConversationCreated, conversation) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(conversation) | ||||
| } | ||||
|  | ||||
| // validateCreateConversationRequest validates the create conversation request fields. | ||||
| func validateCreateConversationRequest(req createConversationRequest, app *App) error { | ||||
| 	if req.InboxID <= 0 { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.required", "name", "`inbox_id`"), nil) | ||||
| 	} | ||||
| 	if req.Content == "" { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.required", "name", "`content`"), nil) | ||||
| 	} | ||||
| 	if req.Email == "" { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.required", "name", "`contact_email`"), nil) | ||||
| 	} | ||||
| 	if req.FirstName == "" { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.required", "name", "`first_name`"), nil) | ||||
| 	} | ||||
| 	if !stringutil.ValidEmail(req.Email) { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.invalid", "name", "`contact_email`"), nil) | ||||
| 	} | ||||
| 	if req.Initiator != umodels.UserTypeContact && req.Initiator != umodels.UserTypeAgent { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.invalid", "name", "`initiator`"), nil) | ||||
| 	} | ||||
|  | ||||
| 	// Check if inbox exists and is enabled. | ||||
| 	inbox, err := app.inbox.GetDBRecord(req.InboxID) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	if !inbox.Enabled { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.disabled", "name", "inbox"), nil) | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|   | ||||
							
								
								
									
										20
									
								
								cmd/csat.go
									
									
									
									
									
								
							
							
						
						
									
										20
									
								
								cmd/csat.go
									
									
									
									
									
								
							| @@ -17,7 +17,7 @@ func handleShowCSAT(r *fastglue.Request) error { | ||||
| 	if err != nil { | ||||
| 		return app.tmpl.RenderWebPage(r.RequestCtx, "error", map[string]interface{}{ | ||||
| 			"Data": map[string]interface{}{ | ||||
| 				"ErrorMessage": "Page not found", | ||||
| 				"ErrorMessage": app.i18n.T("globals.messages.pageNotFound"), | ||||
| 			}, | ||||
| 		}) | ||||
| 	} | ||||
| @@ -25,8 +25,8 @@ func handleShowCSAT(r *fastglue.Request) error { | ||||
| 	if csat.ResponseTimestamp.Valid { | ||||
| 		return app.tmpl.RenderWebPage(r.RequestCtx, "info", map[string]interface{}{ | ||||
| 			"Data": map[string]interface{}{ | ||||
| 				"Title":   "Thank you!", | ||||
| 				"Message": "We appreciate you taking the time to submit your feedback.", | ||||
| 				"Title":   app.i18n.T("globals.messages.thankYou"), | ||||
| 				"Message": app.i18n.T("csat.thankYouMessage"), | ||||
| 			}, | ||||
| 		}) | ||||
| 	} | ||||
| @@ -35,14 +35,14 @@ func handleShowCSAT(r *fastglue.Request) error { | ||||
| 	if err != nil { | ||||
| 		return app.tmpl.RenderWebPage(r.RequestCtx, "error", map[string]interface{}{ | ||||
| 			"Data": map[string]interface{}{ | ||||
| 				"ErrorMessage": "Page not found", | ||||
| 				"ErrorMessage": app.i18n.T("globals.messages.pageNotFound"), | ||||
| 			}, | ||||
| 		}) | ||||
| 	} | ||||
|  | ||||
| 	return app.tmpl.RenderWebPage(r.RequestCtx, "csat", map[string]interface{}{ | ||||
| 		"Data": map[string]interface{}{ | ||||
| 			"Title":    "Rate your interaction with us", | ||||
| 			"Title": app.i18n.T("csat.pageTitle"), | ||||
| 			"CSAT": map[string]interface{}{ | ||||
| 				"UUID": csat.UUID, | ||||
| 			}, | ||||
| @@ -67,7 +67,7 @@ func handleUpdateCSATResponse(r *fastglue.Request) error { | ||||
| 	if err != nil { | ||||
| 		return app.tmpl.RenderWebPage(r.RequestCtx, "error", map[string]interface{}{ | ||||
| 			"Data": map[string]interface{}{ | ||||
| 				"ErrorMessage": "Invalid `rating`", | ||||
| 				"ErrorMessage": app.i18n.T("globals.messages.somethingWentWrong"), | ||||
| 			}, | ||||
| 		}) | ||||
| 	} | ||||
| @@ -75,7 +75,7 @@ func handleUpdateCSATResponse(r *fastglue.Request) error { | ||||
| 	if ratingI < 1 || ratingI > 5 { | ||||
| 		return app.tmpl.RenderWebPage(r.RequestCtx, "error", map[string]interface{}{ | ||||
| 			"Data": map[string]interface{}{ | ||||
| 				"ErrorMessage": "Invalid `rating`", | ||||
| 				"ErrorMessage": app.i18n.T("globals.messages.somethingWentWrong"), | ||||
| 			}, | ||||
| 		}) | ||||
| 	} | ||||
| @@ -83,7 +83,7 @@ func handleUpdateCSATResponse(r *fastglue.Request) error { | ||||
| 	if uuid == "" { | ||||
| 		return app.tmpl.RenderWebPage(r.RequestCtx, "error", map[string]interface{}{ | ||||
| 			"Data": map[string]interface{}{ | ||||
| 				"ErrorMessage": "Invalid `uuid`", | ||||
| 				"ErrorMessage": app.i18n.T("globals.messages.somethingWentWrong"), | ||||
| 			}, | ||||
| 		}) | ||||
| 	} | ||||
| @@ -98,8 +98,8 @@ func handleUpdateCSATResponse(r *fastglue.Request) error { | ||||
|  | ||||
| 	return app.tmpl.RenderWebPage(r.RequestCtx, "info", map[string]interface{}{ | ||||
| 		"Data": map[string]interface{}{ | ||||
| 			"Title":   "Thank you!", | ||||
| 			"Message": "We appreciate you taking the time to submit your feedback.", | ||||
| 			"Title":   app.i18n.T("globals.messages.thankYou"), | ||||
| 			"Message": app.i18n.T("csat.thankYouMessage"), | ||||
| 		}, | ||||
| 	}) | ||||
| } | ||||
|   | ||||
							
								
								
									
										139
									
								
								cmd/custom_attributes.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										139
									
								
								cmd/custom_attributes.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,139 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"slices" | ||||
| 	"strconv" | ||||
|  | ||||
| 	cmodels "github.com/abhinavxd/libredesk/internal/custom_attribute/models" | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| 	"github.com/valyala/fasthttp" | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| 	// disallowedKeys contains keys that are not allowed for custom attributes as they're the default fields. | ||||
| 	disallowedKeys = []string{ | ||||
| 		"contact_email", | ||||
| 		"content", | ||||
| 		"subject", | ||||
| 		"status", | ||||
| 		"priority", | ||||
| 		"assigned_team", | ||||
| 		"assigned_user", | ||||
| 		"hours_since_created", | ||||
| 		"hours_since_first_reply", | ||||
| 		"hours_since_last_reply", | ||||
| 		"hours_since_resolved", | ||||
| 		"inbox", | ||||
| 	} | ||||
| ) | ||||
|  | ||||
| // handleGetCustomAttribute retrieves a custom attribute by its ID. | ||||
| func handleGetCustomAttribute(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	attribute, err := app.customAttribute.Get(id) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(attribute) | ||||
| } | ||||
|  | ||||
| // handleGetCustomAttributes retrieves all custom attributes from the database. | ||||
| func handleGetCustomAttributes(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app       = r.Context.(*App) | ||||
| 		appliesTo = string(r.RequestCtx.QueryArgs().Peek("applies_to")) | ||||
| 	) | ||||
| 	attributes, err := app.customAttribute.GetAll(appliesTo) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(attributes) | ||||
| } | ||||
|  | ||||
| // handleCreateCustomAttribute creates a new custom attribute in the database. | ||||
| func handleCreateCustomAttribute(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app       = r.Context.(*App) | ||||
| 		attribute = cmodels.CustomAttribute{} | ||||
| 	) | ||||
| 	if err := r.Decode(&attribute, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
| 	if err := validateCustomAttribute(app, attribute); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	createdAttr, err := app.customAttribute.Create(attribute) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(createdAttr) | ||||
| } | ||||
|  | ||||
| // handleUpdateCustomAttribute updates an existing custom attribute in the database. | ||||
| func handleUpdateCustomAttribute(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app       = r.Context.(*App) | ||||
| 		attribute = cmodels.CustomAttribute{} | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	if err := r.Decode(&attribute, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
| 	if err := validateCustomAttribute(app, attribute); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	updatedAttr, err := app.customAttribute.Update(id, attribute) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(updatedAttr) | ||||
| } | ||||
|  | ||||
| // handleDeleteCustomAttribute deletes a custom attribute from the database. | ||||
| func handleDeleteCustomAttribute(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	if err = app.customAttribute.Delete(id); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // validateCustomAttribute validates a custom attribute. | ||||
| func validateCustomAttribute(app *App, attribute cmodels.CustomAttribute) error { | ||||
| 	if attribute.Name == "" { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "`name`"), nil) | ||||
| 	} | ||||
| 	if attribute.AppliesTo == "" { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "`applies_to`"), nil) | ||||
| 	} | ||||
| 	if attribute.DataType == "" { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "`type`"), nil) | ||||
| 	} | ||||
| 	if attribute.Description == "" { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "`description`"), nil) | ||||
| 	} | ||||
| 	if attribute.Key == "" { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "`key`"), nil) | ||||
| 	} | ||||
| 	if slices.Contains(disallowedKeys, attribute.Key) { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.T("admin.customAttributes.keyNotAllowed"), nil) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
							
								
								
									
										140
									
								
								cmd/handlers.go
									
									
									
									
									
								
							
							
						
						
									
										140
									
								
								cmd/handlers.go
									
									
									
									
									
								
							| @@ -12,33 +12,33 @@ import ( | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| 	slaReqFields = map[string][2]int{"name": {1, 255}, "description": {1, 255}, "first_response_time": {1, 255}, "resolution_time": {1, 255}} | ||||
| ) | ||||
|  | ||||
| // initHandlers initializes the HTTP routes and handlers for the application. | ||||
| func initHandlers(g *fastglue.Fastglue, hub *ws.Hub) { | ||||
| 	// Authentication. | ||||
| 	g.POST("/api/v1/login", handleLogin) | ||||
| 	g.GET("/logout", handleLogout) | ||||
| 	g.POST("/api/v1/auth/login", handleLogin) | ||||
| 	g.GET("/logout", auth(handleLogout)) | ||||
| 	g.GET("/api/v1/oidc/{id}/login", handleOIDCLogin) | ||||
| 	g.GET("/api/v1/oidc/{id}/finish", handleOIDCCallback) | ||||
|  | ||||
| 	// i18n. | ||||
| 	g.GET("/api/v1/lang/{lang}", handleGetI18nLang) | ||||
|  | ||||
| 	// Public config for app initialization. | ||||
| 	g.GET("/api/v1/config", handleGetConfig) | ||||
|  | ||||
| 	// Media. | ||||
| 	g.GET("/uploads/{uuid}", auth(handleServeMedia)) | ||||
| 	g.POST("/api/v1/media", auth(handleMediaUpload)) | ||||
|  | ||||
| 	// Settings. | ||||
| 	g.GET("/api/v1/settings/general", handleGetGeneralSettings) | ||||
| 	g.GET("/api/v1/settings/general", auth(handleGetGeneralSettings)) | ||||
| 	g.PUT("/api/v1/settings/general", perm(handleUpdateGeneralSettings, "general_settings:manage")) | ||||
| 	g.GET("/api/v1/settings/notifications/email", perm(handleGetEmailNotificationSettings, "notification_settings:manage")) | ||||
| 	g.PUT("/api/v1/settings/notifications/email", perm(handleUpdateEmailNotificationSettings, "notification_settings:manage")) | ||||
|  | ||||
| 	// OpenID connect single sign-on. | ||||
| 	g.GET("/api/v1/oidc/enabled", handleGetAllEnabledOIDC) | ||||
| 	g.GET("/api/v1/oidc", perm(handleGetAllOIDC, "oidc:manage")) | ||||
| 	g.POST("/api/v1/oidc", perm(handleCreateOIDC, "oidc:manage")) | ||||
| 	g.POST("/api/v1/oidc/test", perm(handleTestOIDC, "oidc:manage")) | ||||
| 	g.GET("/api/v1/oidc/{id}", perm(handleGetOIDC, "oidc:manage")) | ||||
| 	g.PUT("/api/v1/oidc/{id}", perm(handleUpdateOIDC, "oidc:manage")) | ||||
| 	g.DELETE("/api/v1/oidc/{id}", perm(handleDeleteOIDC, "oidc:manage")) | ||||
| @@ -63,10 +63,14 @@ func initHandlers(g *fastglue.Fastglue, hub *ws.Hub) { | ||||
| 	g.GET("/api/v1/conversations/{uuid}/messages", perm(handleGetMessages, "messages:read")) | ||||
| 	g.POST("/api/v1/conversations/{cuuid}/messages", perm(handleSendMessage, "messages:write")) | ||||
| 	g.PUT("/api/v1/conversations/{cuuid}/messages/{uuid}/retry", perm(handleRetryMessage, "messages:write")) | ||||
| 	g.POST("/api/v1/conversations", perm(handleCreateConversation, "conversations:write")) | ||||
| 	g.PUT("/api/v1/conversations/{uuid}/custom-attributes", auth(handleUpdateConversationCustomAttributes)) | ||||
| 	g.PUT("/api/v1/conversations/{uuid}/contacts/custom-attributes", auth(handleUpdateContactCustomAttributes)) | ||||
|  | ||||
| 	// Search. | ||||
| 	g.GET("/api/v1/conversations/search", perm(handleSearchConversations, "conversations:read")) | ||||
| 	g.GET("/api/v1/messages/search", perm(handleSearchMessages, "messages:read")) | ||||
| 	g.GET("/api/v1/contacts/search", perm(handleSearchContacts, "contacts:read")) | ||||
|  | ||||
| 	// Views. | ||||
| 	g.GET("/api/v1/views/me", perm(handleGetUserViews, "view:manage")) | ||||
| @@ -81,7 +85,7 @@ func initHandlers(g *fastglue.Fastglue, hub *ws.Hub) { | ||||
| 	g.DELETE("/api/v1/statuses/{id}", perm(handleDeleteStatus, "status:manage")) | ||||
| 	g.GET("/api/v1/priorities", auth(handleGetPriorities)) | ||||
|  | ||||
| 	// Tag. | ||||
| 	// Tags. | ||||
| 	g.GET("/api/v1/tags", auth(handleGetTags)) | ||||
| 	g.POST("/api/v1/tags", perm(handleCreateTag, "tags:manage")) | ||||
| 	g.PUT("/api/v1/tags/{id}", perm(handleUpdateTag, "tags:manage")) | ||||
| @@ -95,21 +99,36 @@ func initHandlers(g *fastglue.Fastglue, hub *ws.Hub) { | ||||
| 	g.DELETE("/api/v1/macros/{id}", perm(handleDeleteMacro, "macros:manage")) | ||||
| 	g.POST("/api/v1/conversations/{uuid}/macros/{id}/apply", auth(handleApplyMacro)) | ||||
|  | ||||
| 	// User. | ||||
| 	g.GET("/api/v1/users/me", auth(handleGetCurrentUser)) | ||||
| 	g.PUT("/api/v1/users/me", auth(handleUpdateCurrentUser)) | ||||
| 	g.GET("/api/v1/users/me/teams", auth(handleGetCurrentUserTeams)) | ||||
| 	g.DELETE("/api/v1/users/me/avatar", auth(handleDeleteAvatar)) | ||||
| 	g.GET("/api/v1/users/compact", auth(handleGetUsersCompact)) | ||||
| 	g.GET("/api/v1/users", perm(handleGetUsers, "users:manage")) | ||||
| 	g.GET("/api/v1/users/{id}", perm(handleGetUser, "users:manage")) | ||||
| 	g.POST("/api/v1/users", perm(handleCreateUser, "users:manage")) | ||||
| 	g.PUT("/api/v1/users/{id}", perm(handleUpdateUser, "users:manage")) | ||||
| 	g.DELETE("/api/v1/users/{id}", perm(handleDeleteUser, "users:manage")) | ||||
| 	g.POST("/api/v1/users/reset-password", tryAuth(handleResetPassword)) | ||||
| 	g.POST("/api/v1/users/set-password", tryAuth(handleSetPassword)) | ||||
| 	// Agents. | ||||
| 	g.GET("/api/v1/agents/me", auth(handleGetCurrentAgent)) | ||||
| 	g.PUT("/api/v1/agents/me", auth(handleUpdateCurrentAgent)) | ||||
| 	g.GET("/api/v1/agents/me/teams", auth(handleGetCurrentAgentTeams)) | ||||
| 	g.PUT("/api/v1/agents/me/availability", auth(handleUpdateAgentAvailability)) | ||||
| 	g.DELETE("/api/v1/agents/me/avatar", auth(handleDeleteCurrentAgentAvatar)) | ||||
|  | ||||
| 	// Team. | ||||
| 	g.GET("/api/v1/agents/compact", auth(handleGetAgentsCompact)) | ||||
| 	g.GET("/api/v1/agents", perm(handleGetAgents, "users:manage")) | ||||
| 	g.GET("/api/v1/agents/{id}", perm(handleGetAgent, "users:manage")) | ||||
| 	g.POST("/api/v1/agents", perm(handleCreateAgent, "users:manage")) | ||||
| 	g.PUT("/api/v1/agents/{id}", perm(handleUpdateAgent, "users:manage")) | ||||
| 	g.DELETE("/api/v1/agents/{id}", perm(handleDeleteAgent, "users:manage")) | ||||
| 	g.POST("/api/v1/agents/{id}/api-key", perm(handleGenerateAPIKey, "users:manage")) | ||||
| 	g.DELETE("/api/v1/agents/{id}/api-key", perm(handleRevokeAPIKey, "users:manage")) | ||||
| 	g.POST("/api/v1/agents/reset-password", tryAuth(handleResetPassword)) | ||||
| 	g.POST("/api/v1/agents/set-password", tryAuth(handleSetPassword)) | ||||
|  | ||||
| 	// Contacts. | ||||
| 	g.GET("/api/v1/contacts", perm(handleGetContacts, "contacts:read_all")) | ||||
| 	g.GET("/api/v1/contacts/{id}", perm(handleGetContact, "contacts:read")) | ||||
| 	g.PUT("/api/v1/contacts/{id}", perm(handleUpdateContact, "contacts:write")) | ||||
| 	g.PUT("/api/v1/contacts/{id}/block", perm(handleBlockContact, "contacts:block")) | ||||
|  | ||||
| 	// Contact notes. | ||||
| 	g.GET("/api/v1/contacts/{id}/notes", perm(handleGetContactNotes, "contact_notes:read")) | ||||
| 	g.POST("/api/v1/contacts/{id}/notes", perm(handleCreateContactNote, "contact_notes:write")) | ||||
| 	g.DELETE("/api/v1/contacts/{id}/notes/{note_id}", perm(handleDeleteContactNote, "contact_notes:delete")) | ||||
|  | ||||
| 	// Teams. | ||||
| 	g.GET("/api/v1/teams/compact", auth(handleGetTeamsCompact)) | ||||
| 	g.GET("/api/v1/teams", perm(handleGetTeams, "teams:manage")) | ||||
| 	g.GET("/api/v1/teams/{id}", perm(handleGetTeam, "teams:manage")) | ||||
| @@ -117,20 +136,17 @@ func initHandlers(g *fastglue.Fastglue, hub *ws.Hub) { | ||||
| 	g.PUT("/api/v1/teams/{id}", perm(handleUpdateTeam, "teams:manage")) | ||||
| 	g.DELETE("/api/v1/teams/{id}", perm(handleDeleteTeam, "teams:manage")) | ||||
|  | ||||
| 	// i18n. | ||||
| 	g.GET("/api/v1/lang/{lang}", handleGetI18nLang) | ||||
| 	// Automations. | ||||
| 	g.GET("/api/v1/automations/rules", perm(handleGetAutomationRules, "automations:manage")) | ||||
| 	g.GET("/api/v1/automations/rules/{id}", perm(handleGetAutomationRule, "automations:manage")) | ||||
| 	g.POST("/api/v1/automations/rules", perm(handleCreateAutomationRule, "automations:manage")) | ||||
| 	g.PUT("/api/v1/automations/rules/{id}/toggle", perm(handleToggleAutomationRule, "automations:manage")) | ||||
| 	g.PUT("/api/v1/automations/rules/{id}", perm(handleUpdateAutomationRule, "automations:manage")) | ||||
| 	g.PUT("/api/v1/automations/rules/weights", perm(handleUpdateAutomationRuleWeights, "automations:manage")) | ||||
| 	g.PUT("/api/v1/automations/rules/execution-mode", perm(handleUpdateAutomationRuleExecutionMode, "automations:manage")) | ||||
| 	g.DELETE("/api/v1/automations/rules/{id}", perm(handleDeleteAutomationRule, "automations:manage")) | ||||
|  | ||||
| 	// Automation. | ||||
| 	g.GET("/api/v1/automation/rules", perm(handleGetAutomationRules, "automations:manage")) | ||||
| 	g.GET("/api/v1/automation/rules/{id}", perm(handleGetAutomationRule, "automations:manage")) | ||||
| 	g.POST("/api/v1/automation/rules", perm(handleCreateAutomationRule, "automations:manage")) | ||||
| 	g.PUT("/api/v1/automation/rules/{id}/toggle", perm(handleToggleAutomationRule, "automations:manage")) | ||||
| 	g.PUT("/api/v1/automation/rules/{id}", perm(handleUpdateAutomationRule, "automations:manage")) | ||||
| 	g.PUT("/api/v1/automation/rules/weights", perm(handleUpdateAutomationRuleWeights, "automations:manage")) | ||||
| 	g.PUT("/api/v1/automation/rules/execution-mode", perm(handleUpdateAutomationRuleExecutionMode, "automations:manage")) | ||||
| 	g.DELETE("/api/v1/automation/rules/{id}", perm(handleDeleteAutomationRule, "automations:manage")) | ||||
|  | ||||
| 	// Inbox. | ||||
| 	// Inboxes. | ||||
| 	g.GET("/api/v1/inboxes", auth(handleGetInboxes)) | ||||
| 	g.GET("/api/v1/inboxes/{id}", perm(handleGetInbox, "inboxes:manage")) | ||||
| 	g.POST("/api/v1/inboxes", perm(handleCreateInbox, "inboxes:manage")) | ||||
| @@ -138,18 +154,28 @@ func initHandlers(g *fastglue.Fastglue, hub *ws.Hub) { | ||||
| 	g.PUT("/api/v1/inboxes/{id}", perm(handleUpdateInbox, "inboxes:manage")) | ||||
| 	g.DELETE("/api/v1/inboxes/{id}", perm(handleDeleteInbox, "inboxes:manage")) | ||||
|  | ||||
| 	// Role. | ||||
| 	// Roles. | ||||
| 	g.GET("/api/v1/roles", perm(handleGetRoles, "roles:manage")) | ||||
| 	g.GET("/api/v1/roles/{id}", perm(handleGetRole, "roles:manage")) | ||||
| 	g.POST("/api/v1/roles", perm(handleCreateRole, "roles:manage")) | ||||
| 	g.PUT("/api/v1/roles/{id}", perm(handleUpdateRole, "roles:manage")) | ||||
| 	g.DELETE("/api/v1/roles/{id}", perm(handleDeleteRole, "roles:manage")) | ||||
|  | ||||
| 	// Dashboard. | ||||
| 	g.GET("/api/v1/reports/overview/counts", perm(handleDashboardCounts, "reports:manage")) | ||||
| 	g.GET("/api/v1/reports/overview/charts", perm(handleDashboardCharts, "reports:manage")) | ||||
| 	// Webhooks. | ||||
| 	g.GET("/api/v1/webhooks", perm(handleGetWebhooks, "webhooks:manage")) | ||||
| 	g.GET("/api/v1/webhooks/{id}", perm(handleGetWebhook, "webhooks:manage")) | ||||
| 	g.POST("/api/v1/webhooks", perm(handleCreateWebhook, "webhooks:manage")) | ||||
| 	g.PUT("/api/v1/webhooks/{id}", perm(handleUpdateWebhook, "webhooks:manage")) | ||||
| 	g.DELETE("/api/v1/webhooks/{id}", perm(handleDeleteWebhook, "webhooks:manage")) | ||||
| 	g.PUT("/api/v1/webhooks/{id}/toggle", perm(handleToggleWebhook, "webhooks:manage")) | ||||
| 	g.POST("/api/v1/webhooks/{id}/test", perm(handleTestWebhook, "webhooks:manage")) | ||||
|  | ||||
| 	// Template. | ||||
| 	// Reports. | ||||
| 	g.GET("/api/v1/reports/overview/sla", perm(handleOverviewSLA, "reports:manage")) | ||||
| 	g.GET("/api/v1/reports/overview/counts", perm(handleOverviewCounts, "reports:manage")) | ||||
| 	g.GET("/api/v1/reports/overview/charts", perm(handleOverviewCharts, "reports:manage")) | ||||
|  | ||||
| 	// Templates. | ||||
| 	g.GET("/api/v1/templates", perm(handleGetTemplates, "templates:manage")) | ||||
| 	g.GET("/api/v1/templates/{id}", perm(handleGetTemplate, "templates:manage")) | ||||
| 	g.POST("/api/v1/templates", perm(handleCreateTemplate, "templates:manage")) | ||||
| @@ -157,22 +183,33 @@ func initHandlers(g *fastglue.Fastglue, hub *ws.Hub) { | ||||
| 	g.DELETE("/api/v1/templates/{id}", perm(handleDeleteTemplate, "templates:manage")) | ||||
|  | ||||
| 	// Business hours. | ||||
| 	g.GET("/api/v1/business-hours", perm(handleGetBusinessHours, "business_hours:manage")) | ||||
| 	g.GET("/api/v1/business-hours", auth(handleGetBusinessHours)) | ||||
| 	g.GET("/api/v1/business-hours/{id}", perm(handleGetBusinessHour, "business_hours:manage")) | ||||
| 	g.POST("/api/v1/business-hours", perm(handleCreateBusinessHours, "business_hours:manage")) | ||||
| 	g.PUT("/api/v1/business-hours/{id}", perm(handleUpdateBusinessHours, "business_hours:manage")) | ||||
| 	g.DELETE("/api/v1/business-hours/{id}", perm(handleDeleteBusinessHour, "business_hours:manage")) | ||||
|  | ||||
| 	// SLA. | ||||
| 	// SLAs. | ||||
| 	g.GET("/api/v1/sla", perm(handleGetSLAs, "sla:manage")) | ||||
| 	g.GET("/api/v1/sla/{id}", perm(handleGetSLA, "sla:manage")) | ||||
| 	g.POST("/api/v1/sla", perm(fastglue.ReqLenRangeParams(handleCreateSLA, slaReqFields), "sla:manage")) | ||||
| 	g.PUT("/api/v1/sla/{id}", perm(fastglue.ReqLenRangeParams(handleUpdateSLA, slaReqFields), "sla:manage")) | ||||
| 	g.POST("/api/v1/sla", perm(handleCreateSLA, "sla:manage")) | ||||
| 	g.PUT("/api/v1/sla/{id}", perm(handleUpdateSLA, "sla:manage")) | ||||
| 	g.DELETE("/api/v1/sla/{id}", perm(handleDeleteSLA, "sla:manage")) | ||||
|  | ||||
| 	// AI completion. | ||||
| 	// AI completions. | ||||
| 	g.GET("/api/v1/ai/prompts", auth(handleGetAIPrompts)) | ||||
| 	g.POST("/api/v1/ai/completion", auth(handleAICompletion)) | ||||
| 	g.PUT("/api/v1/ai/provider", perm(handleUpdateAIProvider, "ai:manage")) | ||||
|  | ||||
| 	// Custom attributes. | ||||
| 	g.GET("/api/v1/custom-attributes", auth(handleGetCustomAttributes)) | ||||
| 	g.POST("/api/v1/custom-attributes", perm(handleCreateCustomAttribute, "custom_attributes:manage")) | ||||
| 	g.GET("/api/v1/custom-attributes/{id}", perm(handleGetCustomAttribute, "custom_attributes:manage")) | ||||
| 	g.PUT("/api/v1/custom-attributes/{id}", perm(handleUpdateCustomAttribute, "custom_attributes:manage")) | ||||
| 	g.DELETE("/api/v1/custom-attributes/{id}", perm(handleDeleteCustomAttribute, "custom_attributes:manage")) | ||||
|  | ||||
| 	// Actvity logs. | ||||
| 	g.GET("/api/v1/activity-logs", perm(handleGetActivityLogs, "activity_logs:manage")) | ||||
|  | ||||
| 	// WebSocket. | ||||
| 	g.GET("/ws", auth(func(r *fastglue.Request) error { | ||||
| @@ -185,6 +222,7 @@ func initHandlers(g *fastglue.Fastglue, hub *ws.Hub) { | ||||
| 	g.GET("/teams/{all:*}", authPage(serveIndexPage)) | ||||
| 	g.GET("/views/{all:*}", authPage(serveIndexPage)) | ||||
| 	g.GET("/admin/{all:*}", authPage(serveIndexPage)) | ||||
| 	g.GET("/contacts/{all:*}", authPage(serveIndexPage)) | ||||
| 	g.GET("/reports/{all:*}", authPage(serveIndexPage)) | ||||
| 	g.GET("/account/{all:*}", authPage(serveIndexPage)) | ||||
| 	g.GET("/reset-password", notAuthPage(serveIndexPage)) | ||||
| @@ -214,7 +252,7 @@ func serveIndexPage(r *fastglue.Request) error { | ||||
| 	// Serve the index.html file from the embedded filesystem. | ||||
| 	file, err := app.fs.Get(path.Join(frontendDir, "index.html")) | ||||
| 	if err != nil { | ||||
| 		return r.SendErrorEnvelope(http.StatusNotFound, "Page not found", nil, envelope.NotFoundError) | ||||
| 		return r.SendErrorEnvelope(http.StatusNotFound, app.i18n.Ts("globals.messages.notFound", "name", "{globals.terms.file}"), nil, envelope.NotFoundError) | ||||
| 	} | ||||
| 	r.RequestCtx.Response.Header.Set("Content-Type", "text/html") | ||||
| 	r.RequestCtx.SetBody(file.ReadBytes()) | ||||
| @@ -222,7 +260,7 @@ func serveIndexPage(r *fastglue.Request) error { | ||||
| 	// Set CSRF cookie if not already set. | ||||
| 	if err := app.auth.SetCSRFCookie(r); err != nil { | ||||
| 		app.lo.Error("error setting csrf cookie", "error", err) | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.T("user.errorAcquiringSession"), nil)) | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorSaving", "name", "{globals.terms.session}"), nil)) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| @@ -236,7 +274,7 @@ func serveStaticFiles(r *fastglue.Request) error { | ||||
|  | ||||
| 	file, err := app.fs.Get(filePath) | ||||
| 	if err != nil { | ||||
| 		return r.SendErrorEnvelope(http.StatusNotFound, "File not found", nil, envelope.NotFoundError) | ||||
| 		return r.SendErrorEnvelope(http.StatusNotFound, app.i18n.Ts("globals.messages.notFound", "name", "{globals.terms.file}"), nil, envelope.NotFoundError) | ||||
| 	} | ||||
|  | ||||
| 	// Set the appropriate Content-Type based on the file extension. | ||||
| @@ -261,7 +299,7 @@ func serveFrontendStaticFiles(r *fastglue.Request) error { | ||||
| 	finalPath := filepath.Join(frontendDir, filePath) | ||||
| 	file, err := app.fs.Get(finalPath) | ||||
| 	if err != nil { | ||||
| 		return r.SendErrorEnvelope(http.StatusNotFound, "File not found", nil, envelope.NotFoundError) | ||||
| 		return r.SendErrorEnvelope(http.StatusNotFound, app.i18n.Ts("globals.messages.notFound", "name", "{globals.terms.file}"), nil, envelope.NotFoundError) | ||||
| 	} | ||||
|  | ||||
| 	// Set the appropriate Content-Type based on the file extension. | ||||
|   | ||||
| @@ -27,6 +27,7 @@ func handleGetI18nLang(r *fastglue.Request) error { | ||||
| 	return r.SendBytes(http.StatusOK, "application/json", i.JSON()) | ||||
| } | ||||
|  | ||||
| // loadI18nLang loads the i18n language pack for the given language code. | ||||
| func loadI18nLang(lang string, fs stuffbin.FileSystem) (*i18n.I18n, error) { | ||||
| 	// Helper function to read and initialize i18n language. | ||||
| 	readLang := func(lang string) ([]byte, error) { | ||||
|   | ||||
							
								
								
									
										107
									
								
								cmd/inboxes.go
									
									
									
									
									
								
							
							
						
						
									
										107
									
								
								cmd/inboxes.go
									
									
									
									
									
								
							| @@ -1,6 +1,7 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"net/mail" | ||||
| 	"strconv" | ||||
|  | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| @@ -9,15 +10,23 @@ import ( | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| // handleGetInboxes returns all inboxes | ||||
| func handleGetInboxes(r *fastglue.Request) error { | ||||
| 	var app = r.Context.(*App) | ||||
| 	inboxes, err := app.inbox.GetAll() | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	for i := range inboxes { | ||||
| 		if err := inboxes[i].ClearPasswords(); err != nil { | ||||
| 			app.lo.Error("error clearing inbox passwords from response", "error", err) | ||||
| 			return envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorFetching", "name", "{globals.terms.inbox}"), nil) | ||||
| 		} | ||||
| 	} | ||||
| 	return r.SendEnvelope(inboxes) | ||||
| } | ||||
|  | ||||
| // handleGetInbox returns an inbox by ID | ||||
| func handleGetInbox(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| @@ -25,33 +34,45 @@ func handleGetInbox(r *fastglue.Request) error { | ||||
| 	) | ||||
| 	inbox, err := app.inbox.GetDBRecord(id) | ||||
| 	if err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error fetching inbox", nil, envelope.GeneralError) | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	if err := inbox.ClearPasswords(); err != nil { | ||||
| 		app.lo.Error("error clearing out passwords", "error", err) | ||||
| 		return envelope.NewError(envelope.GeneralError, "Error fetching inbox", nil) | ||||
| 		app.lo.Error("error clearing inbox passwords from response", "error", err) | ||||
| 		return envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorFetching", "name", "{globals.terms.inbox}"), nil) | ||||
| 	} | ||||
| 	return r.SendEnvelope(inbox) | ||||
| } | ||||
|  | ||||
| // handleCreateInbox creates a new inbox | ||||
| func handleCreateInbox(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 		inb = imodels.Inbox{} | ||||
| 		app   = r.Context.(*App) | ||||
| 		inbox = imodels.Inbox{} | ||||
| 	) | ||||
| 	if err := r.Decode(&inb, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", err.Error(), envelope.InputError) | ||||
| 	if err := r.Decode(&inbox, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
| 	err := app.inbox.Create(inb) | ||||
|  | ||||
| 	createdInbox, err := app.inbox.Create(inbox) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	if err := reloadInboxes(app); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error reloading inboxes", nil, envelope.GeneralError) | ||||
| 	if err := validateInbox(app, createdInbox); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| 	if err := reloadInboxes(app); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.couldNotReload", "name", "{globals.terms.inbox}"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	// Clear passwords before returning. | ||||
| 	if err := createdInbox.ClearPasswords(); err != nil { | ||||
| 		app.lo.Error("error clearing inbox passwords from response", "error", err) | ||||
| 		return envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorFetching", "name", "{globals.terms.inbox}"), nil) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(createdInbox) | ||||
| } | ||||
|  | ||||
| // handleUpdateInbox updates an inbox | ||||
| @@ -63,24 +84,36 @@ func handleUpdateInbox(r *fastglue.Request) error { | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid inbox `id`.", nil, envelope.InputError) | ||||
| 			app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := r.Decode(&inbox, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", err.Error(), envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
| 	err = app.inbox.Update(id, inbox) | ||||
|  | ||||
| 	if err := validateInbox(app, inbox); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	updatedInbox, err := app.inbox.Update(id, inbox) | ||||
| 	if err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Could not update inbox.", nil, envelope.GeneralError) | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	if err := reloadInboxes(app); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error reloading inboxes, Please restart the app if the issue persists", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.couldNotReload", "name", "{globals.terms.inbox}"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(inbox) | ||||
| 	// Clear passwords before returning. | ||||
| 	if err := updatedInbox.ClearPasswords(); err != nil { | ||||
| 		app.lo.Error("error clearing inbox passwords from response", "error", err) | ||||
| 		return envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorFetching", "name", "{globals.terms.inbox}"), nil) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(updatedInbox) | ||||
| } | ||||
|  | ||||
| // handleToggleInbox toggles an inbox | ||||
| func handleToggleInbox(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| @@ -88,20 +121,28 @@ func handleToggleInbox(r *fastglue.Request) error { | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid inbox `id`.", nil, envelope.InputError) | ||||
| 			app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err = app.inbox.Toggle(id); err != nil { | ||||
| 	toggledInbox, err := app.inbox.Toggle(id) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if err := reloadInboxes(app); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error reloading inboxes", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.couldNotReload", "name", "{globals.terms.inbox}"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| 	// Clear passwords before returning | ||||
| 	if err := toggledInbox.ClearPasswords(); err != nil { | ||||
| 		app.lo.Error("error clearing inbox passwords from response", "error", err) | ||||
| 		return envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorFetching", "name", "{globals.terms.inbox}"), nil) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(toggledInbox) | ||||
| } | ||||
|  | ||||
| // handleDeleteInbox deletes an inbox | ||||
| func handleDeleteInbox(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| @@ -109,12 +150,28 @@ func handleDeleteInbox(r *fastglue.Request) error { | ||||
| 	) | ||||
| 	err := app.inbox.SoftDelete(id) | ||||
| 	if err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Could not update inbox.", nil, envelope.GeneralError) | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	if err := reloadInboxes(app); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error reloading inboxes", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.couldNotReload", "name", "{globals.terms.inbox}"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // validateInbox validates the inbox | ||||
| func validateInbox(app *App, inbox imodels.Inbox) error { | ||||
| 	// Validate from address. | ||||
| 	if _, err := mail.ParseAddress(inbox.From); err != nil { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.invalidFromAddress"), nil) | ||||
| 	} | ||||
| 	if len(inbox.Config) == 0 { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "config"), nil) | ||||
| 	} | ||||
| 	if inbox.Name == "" { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "name"), nil) | ||||
| 	} | ||||
| 	if inbox.Channel == "" { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "channel"), nil) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|   | ||||
							
								
								
									
										230
									
								
								cmd/init.go
									
									
									
									
									
								
							
							
						
						
									
										230
									
								
								cmd/init.go
									
									
									
									
									
								
							| @@ -12,6 +12,7 @@ import ( | ||||
|  | ||||
| 	"html/template" | ||||
|  | ||||
| 	activitylog "github.com/abhinavxd/libredesk/internal/activity_log" | ||||
| 	"github.com/abhinavxd/libredesk/internal/ai" | ||||
| 	auth_ "github.com/abhinavxd/libredesk/internal/auth" | ||||
| 	"github.com/abhinavxd/libredesk/internal/authz" | ||||
| @@ -23,6 +24,7 @@ import ( | ||||
| 	"github.com/abhinavxd/libredesk/internal/conversation/priority" | ||||
| 	"github.com/abhinavxd/libredesk/internal/conversation/status" | ||||
| 	"github.com/abhinavxd/libredesk/internal/csat" | ||||
| 	customAttribute "github.com/abhinavxd/libredesk/internal/custom_attribute" | ||||
| 	"github.com/abhinavxd/libredesk/internal/inbox" | ||||
| 	"github.com/abhinavxd/libredesk/internal/inbox/channel/email" | ||||
| 	imodels "github.com/abhinavxd/libredesk/internal/inbox/models" | ||||
| @@ -33,6 +35,7 @@ import ( | ||||
| 	notifier "github.com/abhinavxd/libredesk/internal/notification" | ||||
| 	emailnotifier "github.com/abhinavxd/libredesk/internal/notification/providers/email" | ||||
| 	"github.com/abhinavxd/libredesk/internal/oidc" | ||||
| 	"github.com/abhinavxd/libredesk/internal/report" | ||||
| 	"github.com/abhinavxd/libredesk/internal/role" | ||||
| 	"github.com/abhinavxd/libredesk/internal/search" | ||||
| 	"github.com/abhinavxd/libredesk/internal/setting" | ||||
| @@ -42,6 +45,7 @@ import ( | ||||
| 	tmpl "github.com/abhinavxd/libredesk/internal/template" | ||||
| 	"github.com/abhinavxd/libredesk/internal/user" | ||||
| 	"github.com/abhinavxd/libredesk/internal/view" | ||||
| 	"github.com/abhinavxd/libredesk/internal/webhook" | ||||
| 	"github.com/abhinavxd/libredesk/internal/ws" | ||||
| 	"github.com/jmoiron/sqlx" | ||||
| 	"github.com/knadh/go-i18n" | ||||
| @@ -217,8 +221,9 @@ func initConversations( | ||||
| 	csat *csat.Manager, | ||||
| 	automationEngine *automation.Engine, | ||||
| 	template *tmpl.Manager, | ||||
| 	webhook *webhook.Manager, | ||||
| ) *conversation.Manager { | ||||
| 	c, err := conversation.New(hub, i18n, notif, sla, status, priority, inboxStore, userStore, teamStore, mediaStore, settings, csat, automationEngine, template, conversation.Opts{ | ||||
| 	c, err := conversation.New(hub, i18n, notif, sla, status, priority, inboxStore, userStore, teamStore, mediaStore, settings, csat, automationEngine, template, webhook, conversation.Opts{ | ||||
| 		DB:                       db, | ||||
| 		Lo:                       initLogger("conversation_manager"), | ||||
| 		OutgoingMessageQueueSize: ko.MustInt("message.outgoing_queue_size"), | ||||
| @@ -231,11 +236,12 @@ func initConversations( | ||||
| } | ||||
|  | ||||
| // initTag inits tag manager. | ||||
| func initTag(db *sqlx.DB) *tag.Manager { | ||||
| func initTag(db *sqlx.DB, i18n *i18n.I18n) *tag.Manager { | ||||
| 	var lo = initLogger("tag_manager") | ||||
| 	mgr, err := tag.New(tag.Opts{ | ||||
| 		DB: db, | ||||
| 		Lo: lo, | ||||
| 		DB:   db, | ||||
| 		Lo:   lo, | ||||
| 		I18n: i18n, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing tags: %v", err) | ||||
| @@ -244,11 +250,12 @@ func initTag(db *sqlx.DB) *tag.Manager { | ||||
| } | ||||
|  | ||||
| // initViews inits view manager. | ||||
| func initView(db *sqlx.DB) *view.Manager { | ||||
| func initView(db *sqlx.DB, i18n *i18n.I18n) *view.Manager { | ||||
| 	var lo = initLogger("view_manager") | ||||
| 	m, err := view.New(view.Opts{ | ||||
| 		DB: db, | ||||
| 		Lo: lo, | ||||
| 		DB:   db, | ||||
| 		Lo:   lo, | ||||
| 		I18n: i18n, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing view manager: %v", err) | ||||
| @@ -257,11 +264,12 @@ func initView(db *sqlx.DB) *view.Manager { | ||||
| } | ||||
|  | ||||
| // initMacro inits macro manager. | ||||
| func initMacro(db *sqlx.DB) *macro.Manager { | ||||
| func initMacro(db *sqlx.DB, i18n *i18n.I18n) *macro.Manager { | ||||
| 	var lo = initLogger("macro") | ||||
| 	m, err := macro.New(macro.Opts{ | ||||
| 		DB: db, | ||||
| 		Lo: lo, | ||||
| 		DB:   db, | ||||
| 		Lo:   lo, | ||||
| 		I18n: i18n, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing macro manager: %v", err) | ||||
| @@ -270,11 +278,12 @@ func initMacro(db *sqlx.DB) *macro.Manager { | ||||
| } | ||||
|  | ||||
| // initBusinessHours inits business hours manager. | ||||
| func initBusinessHours(db *sqlx.DB) *businesshours.Manager { | ||||
| func initBusinessHours(db *sqlx.DB, i18n *i18n.I18n) *businesshours.Manager { | ||||
| 	var lo = initLogger("business-hours") | ||||
| 	m, err := businesshours.New(businesshours.Opts{ | ||||
| 		DB: db, | ||||
| 		Lo: lo, | ||||
| 		DB:   db, | ||||
| 		Lo:   lo, | ||||
| 		I18n: i18n, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing business hours manager: %v", err) | ||||
| @@ -283,12 +292,13 @@ func initBusinessHours(db *sqlx.DB) *businesshours.Manager { | ||||
| } | ||||
|  | ||||
| // initSLA inits SLA manager. | ||||
| func initSLA(db *sqlx.DB, teamManager *team.Manager, settings *setting.Manager, businessHours *businesshours.Manager) *sla.Manager { | ||||
| func initSLA(db *sqlx.DB, teamManager *team.Manager, settings *setting.Manager, businessHours *businesshours.Manager, notifier *notifier.Service, template *tmpl.Manager, userManager *user.Manager, i18n *i18n.I18n) *sla.Manager { | ||||
| 	var lo = initLogger("sla") | ||||
| 	m, err := sla.New(sla.Opts{ | ||||
| 		DB: db, | ||||
| 		Lo: lo, | ||||
| 	}, teamManager, settings, businessHours) | ||||
| 		DB:   db, | ||||
| 		Lo:   lo, | ||||
| 		I18n: i18n, | ||||
| 	}, teamManager, settings, businessHours, notifier, template, userManager) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing SLA manager: %v", err) | ||||
| 	} | ||||
| @@ -296,11 +306,12 @@ func initSLA(db *sqlx.DB, teamManager *team.Manager, settings *setting.Manager, | ||||
| } | ||||
|  | ||||
| // initCSAT inits CSAT manager. | ||||
| func initCSAT(db *sqlx.DB) *csat.Manager { | ||||
| func initCSAT(db *sqlx.DB, i18n *i18n.I18n) *csat.Manager { | ||||
| 	var lo = initLogger("csat") | ||||
| 	m, err := csat.New(csat.Opts{ | ||||
| 		DB: db, | ||||
| 		Lo: lo, | ||||
| 		DB:   db, | ||||
| 		Lo:   lo, | ||||
| 		I18n: i18n, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing CSAT manager: %v", err) | ||||
| @@ -308,11 +319,16 @@ func initCSAT(db *sqlx.DB) *csat.Manager { | ||||
| 	return m | ||||
| } | ||||
|  | ||||
| // initWS inits websocket hub. | ||||
| func initWS(user *user.Manager) *ws.Hub { | ||||
| 	return ws.NewHub(user) | ||||
| } | ||||
|  | ||||
| // initTemplates inits template manager. | ||||
| func initTemplate(db *sqlx.DB, fs stuffbin.FileSystem, consts *constants) *tmpl.Manager { | ||||
| func initTemplate(db *sqlx.DB, fs stuffbin.FileSystem, consts *constants, i18n *i18n.I18n) *tmpl.Manager { | ||||
| 	var ( | ||||
| 		lo      = initLogger("template") | ||||
| 		funcMap = getTmplFuncs(consts) | ||||
| 		funcMap = getTmplFuncs(consts, i18n) | ||||
| 	) | ||||
| 	tpls, err := stuffbin.ParseTemplatesGlob(funcMap, fs, "/static/email-templates/*.html") | ||||
| 	if err != nil { | ||||
| @@ -322,7 +338,7 @@ func initTemplate(db *sqlx.DB, fs stuffbin.FileSystem, consts *constants) *tmpl. | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error parsing web templates: %v", err) | ||||
| 	} | ||||
| 	m, err := tmpl.New(lo, db, webTpls, tpls, funcMap) | ||||
| 	m, err := tmpl.New(lo, db, webTpls, tpls, funcMap, i18n) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing template manager: %v", err) | ||||
| 	} | ||||
| @@ -330,7 +346,7 @@ func initTemplate(db *sqlx.DB, fs stuffbin.FileSystem, consts *constants) *tmpl. | ||||
| } | ||||
|  | ||||
| // getTmplFuncs returns the template functions. | ||||
| func getTmplFuncs(consts *constants) template.FuncMap { | ||||
| func getTmplFuncs(consts *constants, i18n *i18n.I18n) template.FuncMap { | ||||
| 	return template.FuncMap{ | ||||
| 		"RootURL": func() string { | ||||
| 			return consts.AppBaseURL | ||||
| @@ -350,6 +366,9 @@ func getTmplFuncs(consts *constants) template.FuncMap { | ||||
| 		"SiteName": func() string { | ||||
| 			return consts.SiteName | ||||
| 		}, | ||||
| 		"L": func() interface{} { | ||||
| 			return i18n | ||||
| 		}, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -366,7 +385,10 @@ func reloadSettings(app *App) error { | ||||
| 		app.lo.Error("error unmarshalling settings from DB", "error", err) | ||||
| 		return err | ||||
| 	} | ||||
| 	if err := ko.Load(confmap.Provider(out, "."), nil); err != nil { | ||||
| 	app.Lock() | ||||
| 	err = ko.Load(confmap.Provider(out, "."), nil) | ||||
| 	app.Unlock() | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error loading settings into koanf", "error", err) | ||||
| 		return err | ||||
| 	} | ||||
| @@ -378,7 +400,7 @@ func reloadSettings(app *App) error { | ||||
| // reloadTemplates reloads the templates from the filesystem. | ||||
| func reloadTemplates(app *App) error { | ||||
| 	app.lo.Info("reloading templates") | ||||
| 	funcMap := getTmplFuncs(app.consts.Load().(*constants)) | ||||
| 	funcMap := getTmplFuncs(app.consts.Load().(*constants), app.i18n) | ||||
| 	tpls, err := stuffbin.ParseTemplatesGlob(funcMap, app.fs, "/static/email-templates/*.html") | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error parsing email templates", "error", err) | ||||
| @@ -393,11 +415,12 @@ func reloadTemplates(app *App) error { | ||||
| } | ||||
|  | ||||
| // initTeam inits team manager. | ||||
| func initTeam(db *sqlx.DB) *team.Manager { | ||||
| func initTeam(db *sqlx.DB, i18n *i18n.I18n) *team.Manager { | ||||
| 	var lo = initLogger("team-manager") | ||||
| 	mgr, err := team.New(team.Opts{ | ||||
| 		DB: db, | ||||
| 		Lo: lo, | ||||
| 		DB:   db, | ||||
| 		Lo:   lo, | ||||
| 		I18n: i18n, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing team manager: %v", err) | ||||
| @@ -406,7 +429,7 @@ func initTeam(db *sqlx.DB) *team.Manager { | ||||
| } | ||||
|  | ||||
| // initMedia inits media manager. | ||||
| func initMedia(db *sqlx.DB) *media.Manager { | ||||
| func initMedia(db *sqlx.DB, i18n *i18n.I18n) *media.Manager { | ||||
| 	var ( | ||||
| 		store      media.Store | ||||
| 		err        error | ||||
| @@ -447,6 +470,7 @@ func initMedia(db *sqlx.DB) *media.Manager { | ||||
| 		Store: store, | ||||
| 		Lo:    lo, | ||||
| 		DB:    db, | ||||
| 		I18n:  i18n, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing media: %v", err) | ||||
| @@ -455,9 +479,9 @@ func initMedia(db *sqlx.DB) *media.Manager { | ||||
| } | ||||
|  | ||||
| // initInbox initializes the inbox manager without registering inboxes. | ||||
| func initInbox(db *sqlx.DB) *inbox.Manager { | ||||
| func initInbox(db *sqlx.DB, i18n *i18n.I18n) *inbox.Manager { | ||||
| 	var lo = initLogger("inbox-manager") | ||||
| 	mgr, err := inbox.New(lo, db) | ||||
| 	mgr, err := inbox.New(lo, db, i18n) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing inbox manager: %v", err) | ||||
| 	} | ||||
| @@ -465,11 +489,12 @@ func initInbox(db *sqlx.DB) *inbox.Manager { | ||||
| } | ||||
|  | ||||
| // initAutomationEngine initializes the automation engine. | ||||
| func initAutomationEngine(db *sqlx.DB) *automation.Engine { | ||||
| func initAutomationEngine(db *sqlx.DB, i18n *i18n.I18n) *automation.Engine { | ||||
| 	var lo = initLogger("automation_engine") | ||||
| 	engine, err := automation.New(automation.Opts{ | ||||
| 		DB: db, | ||||
| 		Lo: lo, | ||||
| 		DB:   db, | ||||
| 		Lo:   lo, | ||||
| 		I18n: i18n, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing automation engine: %v", err) | ||||
| @@ -491,13 +516,13 @@ func initAutoAssigner(teamManager *team.Manager, userManager *user.Manager, conv | ||||
| } | ||||
|  | ||||
| // initNotifier initializes the notifier service with available providers. | ||||
| func initNotifier(userStore notifier.UserStore) *notifier.Service { | ||||
| func initNotifier() *notifier.Service { | ||||
| 	smtpCfg := email.SMTPConfig{} | ||||
| 	if err := ko.UnmarshalWithConf("notification.email", &smtpCfg, koanf.UnmarshalConf{Tag: "json"}); err != nil { | ||||
| 		log.Fatalf("error unmarshalling email notification provider config: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	emailNotifier, err := emailnotifier.New([]email.SMTPConfig{smtpCfg}, userStore, emailnotifier.Opts{ | ||||
| 	emailNotifier, err := emailnotifier.New([]email.SMTPConfig{smtpCfg}, emailnotifier.Opts{ | ||||
| 		Lo:        initLogger("email-notifier"), | ||||
| 		FromEmail: ko.String("notification.email.email_address"), | ||||
| 	}) | ||||
| @@ -513,7 +538,7 @@ func initNotifier(userStore notifier.UserStore) *notifier.Service { | ||||
| } | ||||
|  | ||||
| // initEmailInbox initializes the email inbox. | ||||
| func initEmailInbox(inboxRecord imodels.Inbox, store inbox.MessageStore) (inbox.Inbox, error) { | ||||
| func initEmailInbox(inboxRecord imodels.Inbox, msgStore inbox.MessageStore, usrStore inbox.UserStore) (inbox.Inbox, error) { | ||||
| 	var config email.Config | ||||
|  | ||||
| 	// Load JSON data into Koanf. | ||||
| @@ -539,7 +564,7 @@ func initEmailInbox(inboxRecord imodels.Inbox, store inbox.MessageStore) (inbox. | ||||
| 		log.Printf("WARNING: No `from` email address set for `%s` inbox: Name: `%s`", inboxRecord.Channel, inboxRecord.Name) | ||||
| 	} | ||||
|  | ||||
| 	inbox, err := email.New(store, email.Opts{ | ||||
| 	inbox, err := email.New(msgStore, usrStore, email.Opts{ | ||||
| 		ID:     inboxRecord.ID, | ||||
| 		Config: config, | ||||
| 		Lo:     initLogger("email_inbox"), | ||||
| @@ -549,16 +574,16 @@ func initEmailInbox(inboxRecord imodels.Inbox, store inbox.MessageStore) (inbox. | ||||
| 		return nil, fmt.Errorf("initializing `%s` inbox: `%s` error : %w", inboxRecord.Channel, inboxRecord.Name, err) | ||||
| 	} | ||||
|  | ||||
| 	log.Printf("`%s` inbox successfully initialized. %d SMTP servers. %d IMAP clients.", inboxRecord.Name, len(config.SMTP), len(config.IMAP)) | ||||
| 	log.Printf("`%s` inbox successfully initialized", inboxRecord.Name) | ||||
|  | ||||
| 	return inbox, nil | ||||
| } | ||||
|  | ||||
| // initializeInboxes handles inbox initialization. | ||||
| func initializeInboxes(inboxR imodels.Inbox, store inbox.MessageStore) (inbox.Inbox, error) { | ||||
| func initializeInboxes(inboxR imodels.Inbox, msgStore inbox.MessageStore, usrStore inbox.UserStore) (inbox.Inbox, error) { | ||||
| 	switch inboxR.Channel { | ||||
| 	case "email": | ||||
| 		return initEmailInbox(inboxR, store) | ||||
| 		return initEmailInbox(inboxR, msgStore, usrStore) | ||||
| 	default: | ||||
| 		return nil, fmt.Errorf("unknown inbox channel: %s", inboxR.Channel) | ||||
| 	} | ||||
| @@ -571,8 +596,9 @@ func reloadInboxes(app *App) error { | ||||
| } | ||||
|  | ||||
| // startInboxes registers the active inboxes and starts receiver for each. | ||||
| func startInboxes(ctx context.Context, mgr *inbox.Manager, store inbox.MessageStore) { | ||||
| 	mgr.SetMessageStore(store) | ||||
| func startInboxes(ctx context.Context, mgr *inbox.Manager, msgStore inbox.MessageStore, usrStore inbox.UserStore) { | ||||
| 	mgr.SetMessageStore(msgStore) | ||||
| 	mgr.SetUserStore(usrStore) | ||||
|  | ||||
| 	if err := mgr.InitInboxes(initializeInboxes); err != nil { | ||||
| 		log.Fatalf("error initializing inboxes: %v", err) | ||||
| @@ -584,8 +610,8 @@ func startInboxes(ctx context.Context, mgr *inbox.Manager, store inbox.MessageSt | ||||
| } | ||||
|  | ||||
| // initAuthz initializes authorization enforcer. | ||||
| func initAuthz() *authz.Enforcer { | ||||
| 	enforcer, err := authz.NewEnforcer(initLogger("authz")) | ||||
| func initAuthz(i18n *i18n.I18n) *authz.Enforcer { | ||||
| 	enforcer, err := authz.NewEnforcer(initLogger("authz"), i18n) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing authz: %v", err) | ||||
| 	} | ||||
| @@ -593,7 +619,7 @@ func initAuthz() *authz.Enforcer { | ||||
| } | ||||
|  | ||||
| // initAuth initializes the authentication manager. | ||||
| func initAuth(o *oidc.Manager, rd *redis.Client) *auth_.Auth { | ||||
| func initAuth(o *oidc.Manager, rd *redis.Client, i18n *i18n.I18n) *auth_.Auth { | ||||
| 	lo := initLogger("auth") | ||||
|  | ||||
| 	providers, err := buildProviders(o) | ||||
| @@ -601,7 +627,8 @@ func initAuth(o *oidc.Manager, rd *redis.Client) *auth_.Auth { | ||||
| 		log.Fatalf("error initializing auth: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	auth, err := auth_.New(auth_.Config{Providers: providers}, rd, lo) | ||||
| 	secure := !ko.Bool("app.server.disable_secure_cookies") | ||||
| 	auth, err := auth_.New(auth_.Config{Providers: providers, SecureCookies: secure}, i18n, rd, lo) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing auth: %v", err) | ||||
| 	} | ||||
| @@ -648,11 +675,12 @@ func buildProviders(o *oidc.Manager) ([]auth_.Provider, error) { | ||||
| } | ||||
|  | ||||
| // initOIDC initializes open id connect config manager. | ||||
| func initOIDC(db *sqlx.DB, settings *setting.Manager) *oidc.Manager { | ||||
| func initOIDC(db *sqlx.DB, settings *setting.Manager, i18n *i18n.I18n) *oidc.Manager { | ||||
| 	lo := initLogger("oidc") | ||||
| 	o, err := oidc.New(oidc.Opts{ | ||||
| 		DB: db, | ||||
| 		Lo: lo, | ||||
| 		DB:   db, | ||||
| 		Lo:   lo, | ||||
| 		I18n: i18n, | ||||
| 	}, settings) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing oidc: %v", err) | ||||
| @@ -662,9 +690,11 @@ func initOIDC(db *sqlx.DB, settings *setting.Manager) *oidc.Manager { | ||||
|  | ||||
| // initI18n inits i18n. | ||||
| func initI18n(fs stuffbin.FileSystem) *i18n.I18n { | ||||
| 	file, err := fs.Get("i18n/" + cmp.Or(ko.String("app.lang"), defLang) + ".json") | ||||
| 	fileName := cmp.Or(ko.String("app.lang"), defLang) | ||||
| 	log.Printf("loading i18n language file: %s", fileName) | ||||
| 	file, err := fs.Get("i18n/" + fileName + ".json") | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error reading i18n language file") | ||||
| 		log.Fatalf("error reading i18n language file `%s` : %v", fileName, err) | ||||
| 	} | ||||
| 	i18n, err := i18n.New(file.ReadBytes()) | ||||
| 	if err != nil { | ||||
| @@ -708,11 +738,12 @@ func initDB() *sqlx.DB { | ||||
| } | ||||
|  | ||||
| // initRedis inits role manager. | ||||
| func initRole(db *sqlx.DB) *role.Manager { | ||||
| func initRole(db *sqlx.DB, i18n *i18n.I18n) *role.Manager { | ||||
| 	var lo = initLogger("role_manager") | ||||
| 	r, err := role.New(role.Opts{ | ||||
| 		DB: db, | ||||
| 		Lo: lo, | ||||
| 		DB:   db, | ||||
| 		Lo:   lo, | ||||
| 		I18n: i18n, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing role manager: %v", err) | ||||
| @@ -721,10 +752,11 @@ func initRole(db *sqlx.DB) *role.Manager { | ||||
| } | ||||
|  | ||||
| // initStatus inits conversation status manager. | ||||
| func initStatus(db *sqlx.DB) *status.Manager { | ||||
| func initStatus(db *sqlx.DB, i18n *i18n.I18n) *status.Manager { | ||||
| 	manager, err := status.New(status.Opts{ | ||||
| 		DB: db, | ||||
| 		Lo: initLogger("status-manager"), | ||||
| 		DB:   db, | ||||
| 		Lo:   initLogger("status-manager"), | ||||
| 		I18n: i18n, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing status manager: %v", err) | ||||
| @@ -733,10 +765,11 @@ func initStatus(db *sqlx.DB) *status.Manager { | ||||
| } | ||||
|  | ||||
| // initPriority inits conversation priority manager. | ||||
| func initPriority(db *sqlx.DB) *priority.Manager { | ||||
| func initPriority(db *sqlx.DB, i18n *i18n.I18n) *priority.Manager { | ||||
| 	manager, err := priority.New(priority.Opts{ | ||||
| 		DB: db, | ||||
| 		Lo: initLogger("priority-manager"), | ||||
| 		DB:   db, | ||||
| 		Lo:   initLogger("priority-manager"), | ||||
| 		I18n: i18n, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing priority manager: %v", err) | ||||
| @@ -745,11 +778,12 @@ func initPriority(db *sqlx.DB) *priority.Manager { | ||||
| } | ||||
|  | ||||
| // initAI inits AI manager. | ||||
| func initAI(db *sqlx.DB) *ai.Manager { | ||||
| func initAI(db *sqlx.DB, i18n *i18n.I18n) *ai.Manager { | ||||
| 	lo := initLogger("ai") | ||||
| 	m, err := ai.New(ai.Opts{ | ||||
| 		DB: db, | ||||
| 		Lo: lo, | ||||
| 		DB:   db, | ||||
| 		Lo:   lo, | ||||
| 		I18n: i18n, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing AI manager: %v", err) | ||||
| @@ -758,11 +792,12 @@ func initAI(db *sqlx.DB) *ai.Manager { | ||||
| } | ||||
|  | ||||
| // initSearch inits search manager. | ||||
| func initSearch(db *sqlx.DB) *search.Manager { | ||||
| func initSearch(db *sqlx.DB, i18n *i18n.I18n) *search.Manager { | ||||
| 	lo := initLogger("search") | ||||
| 	m, err := search.New(search.Opts{ | ||||
| 		DB: db, | ||||
| 		Lo: lo, | ||||
| 		DB:   db, | ||||
| 		Lo:   lo, | ||||
| 		I18n: i18n, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing search manager: %v", err) | ||||
| @@ -770,6 +805,65 @@ func initSearch(db *sqlx.DB) *search.Manager { | ||||
| 	return m | ||||
| } | ||||
|  | ||||
| // initCustomAttribute inits custom attribute manager. | ||||
| func initCustomAttribute(db *sqlx.DB, i18n *i18n.I18n) *customAttribute.Manager { | ||||
| 	lo := initLogger("custom-attribute") | ||||
| 	m, err := customAttribute.New(customAttribute.Opts{ | ||||
| 		DB:   db, | ||||
| 		Lo:   lo, | ||||
| 		I18n: i18n, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing custom attribute manager: %v", err) | ||||
| 	} | ||||
| 	return m | ||||
| } | ||||
|  | ||||
| // initActivityLog inits activity log manager. | ||||
| func initActivityLog(db *sqlx.DB, i18n *i18n.I18n) *activitylog.Manager { | ||||
| 	lo := initLogger("activity-log") | ||||
| 	m, err := activitylog.New(activitylog.Opts{ | ||||
| 		DB:   db, | ||||
| 		Lo:   lo, | ||||
| 		I18n: i18n, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing activity log manager: %v", err) | ||||
| 	} | ||||
| 	return m | ||||
| } | ||||
|  | ||||
| // initReport inits report manager. | ||||
| func initReport(db *sqlx.DB, i18n *i18n.I18n) *report.Manager { | ||||
| 	lo := initLogger("report") | ||||
| 	m, err := report.New(report.Opts{ | ||||
| 		DB:   db, | ||||
| 		Lo:   lo, | ||||
| 		I18n: i18n, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing report manager: %v", err) | ||||
| 	} | ||||
| 	return m | ||||
| } | ||||
|  | ||||
| // initWebhook inits webhook manager. | ||||
| func initWebhook(db *sqlx.DB, i18n *i18n.I18n) *webhook.Manager { | ||||
| 	var lo = initLogger("webhook") | ||||
| 	m, err := webhook.New(webhook.Opts{ | ||||
| 		DB:        db, | ||||
| 		Lo:        lo, | ||||
| 		I18n:      i18n, | ||||
| 		Workers:   ko.MustInt("webhook.workers"), | ||||
| 		QueueSize: ko.MustInt("webhook.queue_size"), | ||||
| 		Timeout:   ko.MustDuration("webhook.timeout"), | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error initializing webhook manager: %v", err) | ||||
| 	} | ||||
| 	return m | ||||
| } | ||||
|  | ||||
| // initLogger initializes a logf logger. | ||||
| func initLogger(src string) *logf.Logger { | ||||
| 	lvl, env := ko.MustString("app.log_level"), ko.MustString("app.env") | ||||
|   | ||||
| @@ -9,10 +9,10 @@ import ( | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/abhinavxd/libredesk/internal/colorlog" | ||||
| 	"github.com/abhinavxd/libredesk/internal/dbutil" | ||||
| 	"github.com/abhinavxd/libredesk/internal/user" | ||||
| 	"github.com/jmoiron/sqlx" | ||||
| 	"github.com/knadh/stuffbin" | ||||
| 	"github.com/lib/pq" | ||||
| ) | ||||
|  | ||||
| // Install checks if the schema is already installed, prompts for confirmation, and installs the schema if needed. | ||||
| @@ -24,9 +24,9 @@ func install(ctx context.Context, db *sqlx.DB, fs stuffbin.FileSystem, idempoten | ||||
| 	} | ||||
|  | ||||
| 	// Make sure the system user password is strong enough. | ||||
| 	password := strings.TrimSpace(os.Getenv("LIBREDESK_SYSTEM_USER_PASSWORD")) | ||||
| 	if password != "" && !user.IsStrongSystemUserPassword(password) && !schemaInstalled { | ||||
| 		log.Fatalf("system user password is not strong, %s", user.SystemUserPasswordHint) | ||||
| 	password := os.Getenv("LIBREDESK_SYSTEM_USER_PASSWORD") | ||||
| 	if password != "" && !user.IsStrongPassword(password) && !schemaInstalled { | ||||
| 		log.Fatalf("system user password is not strong, %s", user.PasswordHint) | ||||
| 	} | ||||
|  | ||||
| 	if !idempotentInstall { | ||||
| @@ -49,11 +49,10 @@ func install(ctx context.Context, db *sqlx.DB, fs stuffbin.FileSystem, idempoten | ||||
| 			os.Exit(0) | ||||
| 		} | ||||
| 	} else { | ||||
| 		log.Println("installing database schema...") | ||||
| 		time.Sleep(5 * time.Second) | ||||
| 	} | ||||
|  | ||||
| 	log.Println("installing database schema...") | ||||
|  | ||||
| 	// Install schema. | ||||
| 	if err := installSchema(db, fs); err != nil { | ||||
| 		log.Fatalf("error installing schema: %v", err) | ||||
| @@ -76,7 +75,7 @@ func setSystemUserPass(ctx context.Context, db *sqlx.DB) { | ||||
| // checkSchema verifies if the DB schema is already installed by querying a table. | ||||
| func checkSchema(db *sqlx.DB) (bool, error) { | ||||
| 	if _, err := db.Exec(`SELECT * FROM settings LIMIT 1`); err != nil { | ||||
| 		if pqErr, ok := err.(*pq.Error); ok && pqErr.Code == "42P01" { | ||||
| 		if dbutil.IsTableNotExistError(err) { | ||||
| 			return false, nil | ||||
| 		} | ||||
| 		return false, err | ||||
|   | ||||
							
								
								
									
										60
									
								
								cmd/login.go
									
									
									
									
									
								
							
							
						
						
									
										60
									
								
								cmd/login.go
									
									
									
									
									
								
							| @@ -3,22 +3,44 @@ package main | ||||
| import ( | ||||
| 	amodels "github.com/abhinavxd/libredesk/internal/auth/models" | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| 	realip "github.com/ferluci/fast-realip" | ||||
| 	"github.com/valyala/fasthttp" | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| // handleLogin logs a user in. | ||||
| type loginRequest struct { | ||||
| 	Email    string `json:"email"` | ||||
| 	Password string `json:"password"` | ||||
| } | ||||
|  | ||||
| // handleLogin logs in the user and returns the user. | ||||
| func handleLogin(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app      = r.Context.(*App) | ||||
| 		p        = r.RequestCtx.PostArgs() | ||||
| 		email    = string(p.Peek("email")) | ||||
| 		password = p.Peek("password") | ||||
| 		ip       = realip.FromRequest(r.RequestCtx) | ||||
| 		loginReq loginRequest | ||||
| 	) | ||||
| 	user, err := app.user.VerifyPassword(email, password) | ||||
|  | ||||
| 	// Decode JSON request. | ||||
| 	if err := r.Decode(&loginReq, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if loginReq.Email == "" || loginReq.Password == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.T("globals.messages.badRequest"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Verify email and password. | ||||
| 	user, err := app.user.VerifyPassword(loginReq.Email, []byte(loginReq.Password)) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Check if user is enabled. | ||||
| 	if !user.Enabled { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.T("user.accountDisabled"), nil)) | ||||
| 	} | ||||
|  | ||||
| 	if err := app.auth.SaveSession(amodels.User{ | ||||
| 		ID:        user.ID, | ||||
| 		Email:     user.Email.String, | ||||
| @@ -26,25 +48,43 @@ func handleLogin(r *fastglue.Request) error { | ||||
| 		LastName:  user.LastName, | ||||
| 	}, r); err != nil { | ||||
| 		app.lo.Error("error saving session", "error", err) | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.T("user.errorAcquiringSession"), nil)) | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorSaving", "name", "{globals.terms.session}"), nil)) | ||||
| 	} | ||||
| 	// Set CSRF cookie if not already set. | ||||
| 	if err := app.auth.SetCSRFCookie(r); err != nil { | ||||
| 		app.lo.Error("error setting csrf cookie", "error", err) | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.T("user.errorAcquiringSession"), nil)) | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorSaving", "name", "{globals.terms.session}"), nil)) | ||||
| 	} | ||||
|  | ||||
| 	// Update last login time. | ||||
| 	if err := app.user.UpdateLastLoginAt(user.ID); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Insert activity log. | ||||
| 	if err := app.activityLog.Login(user.ID, user.Email.String, ip); err != nil { | ||||
| 		app.lo.Error("error creating login activity log", "error", err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(user) | ||||
| } | ||||
|  | ||||
| // handleLogout logs out the user and redirects to the dashboard. | ||||
| func handleLogout(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 		app   = r.Context.(*App) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		ip    = realip.FromRequest(r.RequestCtx) | ||||
| 	) | ||||
| 	if err := app.auth.DestroySession(r); err != nil { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.T("user.errorAcquiringSession"), nil)) | ||||
|  | ||||
| 	// Insert activity log. | ||||
| 	if err := app.activityLog.Logout(auser.ID, auser.Email, ip); err != nil { | ||||
| 		app.lo.Error("error creating logout activity log", "error", err) | ||||
| 	} | ||||
|  | ||||
| 	if err := app.auth.DestroySession(r); err != nil { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorDestroying", "name", "{globals.terms.session}"), nil)) | ||||
| 	} | ||||
| 	// Add no-cache headers. | ||||
| 	r.RequestCtx.Response.Header.Add("Cache-Control", | ||||
| 		"no-store, no-cache, must-revalidate, post-check=0, pre-check=0") | ||||
|   | ||||
							
								
								
									
										69
									
								
								cmd/macro.go
									
									
									
									
									
								
							
							
						
						
									
										69
									
								
								cmd/macro.go
									
									
									
									
									
								
							| @@ -2,7 +2,6 @@ package main | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"slices" | ||||
| 	"strconv" | ||||
|  | ||||
| @@ -24,14 +23,14 @@ func handleGetMacros(r *fastglue.Request) error { | ||||
| 	for i, m := range macros { | ||||
| 		var actions []autoModels.RuleAction | ||||
| 		if err := json.Unmarshal(m.Actions, &actions); err != nil { | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error unmarshalling macro actions", nil, envelope.GeneralError) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.macroAction}"), nil, envelope.GeneralError) | ||||
| 		} | ||||
| 		// Set display values for actions as the value field can contain DB IDs | ||||
| 		if err := setDisplayValues(app, actions); err != nil { | ||||
| 			app.lo.Warn("error setting display values", "error", err) | ||||
| 		} | ||||
| 		if macros[i].Actions, err = json.Marshal(actions); err != nil { | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "marshal failed", nil, envelope.GeneralError) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.macroAction}"), nil, envelope.GeneralError) | ||||
| 		} | ||||
| 	} | ||||
| 	return r.SendEnvelope(macros) | ||||
| @@ -44,8 +43,7 @@ func handleGetMacro(r *fastglue.Request) error { | ||||
| 		id, err = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid macro `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	macro, err := app.macro.Get(id) | ||||
| @@ -55,14 +53,14 @@ func handleGetMacro(r *fastglue.Request) error { | ||||
|  | ||||
| 	var actions []autoModels.RuleAction | ||||
| 	if err := json.Unmarshal(macro.Actions, &actions); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error unmarshalling macro actions", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.macroAction}"), nil, envelope.GeneralError) | ||||
| 	} | ||||
| 	// Set display values for actions as the value field can contain DB IDs | ||||
| 	if err := setDisplayValues(app, actions); err != nil { | ||||
| 		app.lo.Warn("error setting display values", "error", err) | ||||
| 	} | ||||
| 	if macro.Actions, err = json.Marshal(actions); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "marshal failed", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.macroAction}"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(macro) | ||||
| @@ -76,19 +74,19 @@ func handleCreateMacro(r *fastglue.Request) error { | ||||
| 	) | ||||
|  | ||||
| 	if err := r.Decode(¯o, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", err.Error(), envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := validateMacro(macro); err != nil { | ||||
| 	if err := validateMacro(app, macro); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	err := app.macro.Create(macro.Name, macro.MessageContent, macro.UserID, macro.TeamID, macro.Visibility, macro.Actions) | ||||
| 	createdMacro, err := app.macro.Create(macro.Name, macro.MessageContent, macro.UserID, macro.TeamID, macro.Visibility, macro.VisibleWhen, macro.Actions) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(macro) | ||||
| 	return r.SendEnvelope(createdMacro) | ||||
| } | ||||
|  | ||||
| // handleUpdateMacro updates a macro. | ||||
| @@ -108,32 +106,29 @@ func handleUpdateMacro(r *fastglue.Request) error { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", err.Error(), envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := validateMacro(macro); err != nil { | ||||
| 	if err := validateMacro(app, macro); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	if err = app.macro.Update(id, macro.Name, macro.MessageContent, macro.UserID, macro.TeamID, macro.Visibility, macro.Actions); err != nil { | ||||
| 	updatedMacro, err := app.macro.Update(id, macro.Name, macro.MessageContent, macro.UserID, macro.TeamID, macro.Visibility, macro.VisibleWhen, macro.Actions) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(macro) | ||||
| 	return r.SendEnvelope(updatedMacro) | ||||
| } | ||||
|  | ||||
| // handleDeleteMacro deletes macro. | ||||
| func handleDeleteMacro(r *fastglue.Request) error { | ||||
| 	var app = r.Context.(*App) | ||||
|  | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid macro `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := app.macro.Delete(id); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope("Macro deleted successfully") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleApplyMacro applies macro actions to a conversation. | ||||
| @@ -145,7 +140,7 @@ func handleApplyMacro(r *fastglue.Request) error { | ||||
| 		id, _            = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 		incomingActions  = []autoModels.RuleAction{} | ||||
| 	) | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| @@ -156,7 +151,7 @@ func handleApplyMacro(r *fastglue.Request) error { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	if allowed, err := app.authz.EnforceConversationAccess(user, conversation); err != nil || !allowed { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.PermissionError, "Permission denied", nil)) | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.PermissionError, app.i18n.Ts("globals.messages.denied", "name", "{globals.terms.permission}"), nil)) | ||||
| 	} | ||||
|  | ||||
| 	macro, err := app.macro.Get(id) | ||||
| @@ -167,7 +162,7 @@ func handleApplyMacro(r *fastglue.Request) error { | ||||
| 	// Decode incoming actions. | ||||
| 	if err := r.Decode(&incomingActions, "json"); err != nil { | ||||
| 		app.lo.Error("error unmashalling incoming actions", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Failed to decode incoming actions", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.macroAction}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Make sure no duplicate action types are present. | ||||
| @@ -175,7 +170,7 @@ func handleApplyMacro(r *fastglue.Request) error { | ||||
| 	for _, act := range incomingActions { | ||||
| 		if actionTypes[act.Type] { | ||||
| 			app.lo.Warn("duplicate action types found in macro apply apply request", "action", act.Type, "user_id", user.ID) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Duplicate actions are not allowed", nil, envelope.InputError) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.T("macro.duplicateActionsNotAllowed"), nil, envelope.InputError) | ||||
| 		} | ||||
| 		actionTypes[act.Type] = true | ||||
| 	} | ||||
| @@ -184,11 +179,11 @@ func handleApplyMacro(r *fastglue.Request) error { | ||||
| 	for _, act := range incomingActions { | ||||
| 		if !isMacroActionAllowed(act.Type) { | ||||
| 			app.lo.Warn("action not allowed in macro", "action", act.Type, "user_id", user.ID) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusForbidden, "Action not allowed in macro", nil, envelope.PermissionError) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusForbidden, app.i18n.Ts("macro.actionNotAllowed", "name", act.Type), nil, envelope.PermissionError) | ||||
| 		} | ||||
| 		if !hasActionPermission(act.Type, user.Permissions) { | ||||
| 			app.lo.Warn("no permission to execute macro action", "action", act.Type, "user_id", user.ID) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusForbidden, "No permission to execute this macro", nil, envelope.PermissionError) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusForbidden, app.i18n.T("macro.permissionDenied"), nil, envelope.PermissionError) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| @@ -201,7 +196,7 @@ func handleApplyMacro(r *fastglue.Request) error { | ||||
| 	} | ||||
|  | ||||
| 	if successCount == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Failed to apply macro", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.T("macro.couldNotApply"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	// Increment usage count. | ||||
| @@ -209,12 +204,12 @@ func handleApplyMacro(r *fastglue.Request) error { | ||||
|  | ||||
| 	if successCount < len(incomingActions) { | ||||
| 		return r.SendJSON(fasthttp.StatusMultiStatus, map[string]interface{}{ | ||||
| 			"message": fmt.Sprintf("Macro executed with errors. %d actions succeeded out of %d", successCount, len(incomingActions)), | ||||
| 			"message": app.i18n.T("macro.partiallyApplied"), | ||||
| 		}) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendJSON(fasthttp.StatusOK, map[string]interface{}{ | ||||
| 		"message": "Macro applied successfully", | ||||
| 		"message": app.i18n.T("macro.applied"), | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| @@ -239,7 +234,7 @@ func setDisplayValues(app *App, actions []autoModels.RuleAction) error { | ||||
| 			return t.Name, nil | ||||
| 		}, | ||||
| 		autoModels.ActionAssignUser: func(id int) (string, error) { | ||||
| 			u, err := app.user.Get(id) | ||||
| 			u, err := app.user.GetAgent(id, "") | ||||
| 			if err != nil { | ||||
| 				app.lo.Warn("user not found for macro action", "user_id", id) | ||||
| 				return "", err | ||||
| @@ -276,18 +271,22 @@ func setDisplayValues(app *App, actions []autoModels.RuleAction) error { | ||||
| } | ||||
|  | ||||
| // validateMacro validates an incoming macro. | ||||
| func validateMacro(macro models.Macro) error { | ||||
| func validateMacro(app *App, macro models.Macro) error { | ||||
| 	if macro.Name == "" { | ||||
| 		return envelope.NewError(envelope.InputError, "Empty macro `name`", nil) | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "`name`"), nil) | ||||
| 	} | ||||
|  | ||||
| 	if len(macro.VisibleWhen) == 0 { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "`visible_when`"), nil) | ||||
| 	} | ||||
|  | ||||
| 	var act []autoModels.RuleAction | ||||
| 	if err := json.Unmarshal(macro.Actions, &act); err != nil { | ||||
| 		return envelope.NewError(envelope.InputError, "Could not parse macro actions", nil) | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.macroAction}"), nil) | ||||
| 	} | ||||
| 	for _, a := range act { | ||||
| 		if len(a.Value) == 0 { | ||||
| 			return envelope.NewError(envelope.InputError, fmt.Sprintf("Empty value for action: %s", a.Type), nil) | ||||
| 			return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", a.Type), nil) | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| @@ -298,7 +297,7 @@ func isMacroActionAllowed(action string) bool { | ||||
| 	switch action { | ||||
| 	case autoModels.ActionSendPrivateNote, autoModels.ActionReply: | ||||
| 		return false | ||||
| 	case autoModels.ActionAssignTeam, autoModels.ActionAssignUser, autoModels.ActionSetStatus, autoModels.ActionSetPriority, autoModels.ActionSetTags: | ||||
| 	case autoModels.ActionAssignTeam, autoModels.ActionAssignUser, autoModels.ActionSetStatus, autoModels.ActionSetPriority, autoModels.ActionAddTags, autoModels.ActionSetTags, autoModels.ActionRemoveTags: | ||||
| 		return true | ||||
| 	default: | ||||
| 		return false | ||||
|   | ||||
							
								
								
									
										200
									
								
								cmd/main.go
									
									
									
									
									
								
							
							
						
						
									
										200
									
								
								cmd/main.go
									
									
									
									
									
								
							| @@ -6,17 +6,24 @@ import ( | ||||
| 	"log" | ||||
| 	"os" | ||||
| 	"os/signal" | ||||
| 	"sync" | ||||
| 	"sync/atomic" | ||||
| 	"syscall" | ||||
| 	"time" | ||||
|  | ||||
| 	_ "time/tzdata" | ||||
|  | ||||
| 	activitylog "github.com/abhinavxd/libredesk/internal/activity_log" | ||||
| 	"github.com/abhinavxd/libredesk/internal/ai" | ||||
| 	auth_ "github.com/abhinavxd/libredesk/internal/auth" | ||||
| 	"github.com/abhinavxd/libredesk/internal/authz" | ||||
| 	businesshours "github.com/abhinavxd/libredesk/internal/business_hours" | ||||
| 	"github.com/abhinavxd/libredesk/internal/colorlog" | ||||
| 	"github.com/abhinavxd/libredesk/internal/csat" | ||||
| 	customAttribute "github.com/abhinavxd/libredesk/internal/custom_attribute" | ||||
| 	"github.com/abhinavxd/libredesk/internal/macro" | ||||
| 	notifier "github.com/abhinavxd/libredesk/internal/notification" | ||||
| 	"github.com/abhinavxd/libredesk/internal/report" | ||||
| 	"github.com/abhinavxd/libredesk/internal/search" | ||||
| 	"github.com/abhinavxd/libredesk/internal/sla" | ||||
| 	"github.com/abhinavxd/libredesk/internal/view" | ||||
| @@ -34,7 +41,7 @@ import ( | ||||
| 	"github.com/abhinavxd/libredesk/internal/team" | ||||
| 	"github.com/abhinavxd/libredesk/internal/template" | ||||
| 	"github.com/abhinavxd/libredesk/internal/user" | ||||
| 	"github.com/abhinavxd/libredesk/internal/ws" | ||||
| 	"github.com/abhinavxd/libredesk/internal/webhook" | ||||
| 	"github.com/knadh/go-i18n" | ||||
| 	"github.com/knadh/koanf/v2" | ||||
| 	"github.com/knadh/stuffbin" | ||||
| @@ -50,38 +57,49 @@ var ( | ||||
| 	frontendDir = "frontend/dist" | ||||
|  | ||||
| 	// Injected at build time. | ||||
| 	buildString = "" | ||||
| 	buildString   string | ||||
| 	versionString string | ||||
| ) | ||||
|  | ||||
| // App is the global app context which is passed and injected in the http handlers. | ||||
| type App struct { | ||||
| 	fs            stuffbin.FileSystem | ||||
| 	consts        atomic.Value | ||||
| 	auth          *auth_.Auth | ||||
| 	authz         *authz.Enforcer | ||||
| 	i18n          *i18n.I18n | ||||
| 	lo            *logf.Logger | ||||
| 	oidc          *oidc.Manager | ||||
| 	media         *media.Manager | ||||
| 	setting       *setting.Manager | ||||
| 	role          *role.Manager | ||||
| 	user          *user.Manager | ||||
| 	team          *team.Manager | ||||
| 	status        *status.Manager | ||||
| 	priority      *priority.Manager | ||||
| 	tag           *tag.Manager | ||||
| 	inbox         *inbox.Manager | ||||
| 	tmpl          *template.Manager | ||||
| 	macro         *macro.Manager | ||||
| 	conversation  *conversation.Manager | ||||
| 	automation    *automation.Engine | ||||
| 	businessHours *businesshours.Manager | ||||
| 	sla           *sla.Manager | ||||
| 	csat          *csat.Manager | ||||
| 	view          *view.Manager | ||||
| 	ai            *ai.Manager | ||||
| 	search        *search.Manager | ||||
| 	notifier      *notifier.Service | ||||
| 	fs              stuffbin.FileSystem | ||||
| 	consts          atomic.Value | ||||
| 	auth            *auth_.Auth | ||||
| 	authz           *authz.Enforcer | ||||
| 	i18n            *i18n.I18n | ||||
| 	lo              *logf.Logger | ||||
| 	oidc            *oidc.Manager | ||||
| 	media           *media.Manager | ||||
| 	setting         *setting.Manager | ||||
| 	role            *role.Manager | ||||
| 	user            *user.Manager | ||||
| 	team            *team.Manager | ||||
| 	status          *status.Manager | ||||
| 	priority        *priority.Manager | ||||
| 	tag             *tag.Manager | ||||
| 	inbox           *inbox.Manager | ||||
| 	tmpl            *template.Manager | ||||
| 	macro           *macro.Manager | ||||
| 	conversation    *conversation.Manager | ||||
| 	automation      *automation.Engine | ||||
| 	businessHours   *businesshours.Manager | ||||
| 	sla             *sla.Manager | ||||
| 	csat            *csat.Manager | ||||
| 	view            *view.Manager | ||||
| 	ai              *ai.Manager | ||||
| 	search          *search.Manager | ||||
| 	activityLog     *activitylog.Manager | ||||
| 	notifier        *notifier.Service | ||||
| 	customAttribute *customAttribute.Manager | ||||
| 	report          *report.Manager | ||||
| 	webhook         *webhook.Manager | ||||
|  | ||||
| 	// Global state that stores data on an available app update. | ||||
| 	update *AppUpdate | ||||
| 	// Flag to indicate if app restart is required for settings to take effect. | ||||
| 	restartRequired bool | ||||
| 	sync.Mutex | ||||
| } | ||||
|  | ||||
| func main() { | ||||
| @@ -99,9 +117,7 @@ func main() { | ||||
| 	} | ||||
|  | ||||
| 	// Build string injected at build time. | ||||
| 	if buildString != "" { | ||||
| 		colorlog.Green("Build: %s", buildString) | ||||
| 	} | ||||
| 	colorlog.Green("Build: %s", buildString) | ||||
|  | ||||
| 	// Load the config files into Koanf. | ||||
| 	initConfig(ko) | ||||
| @@ -136,83 +152,104 @@ func main() { | ||||
|  | ||||
| 	// Upgrade. | ||||
| 	if ko.Bool("upgrade") { | ||||
| 		log.Println("no upgrades available") | ||||
| 		upgrade(db, fs, !ko.Bool("yes")) | ||||
| 		os.Exit(0) | ||||
| 	} | ||||
|  | ||||
| 	// Check for pending upgrade. | ||||
| 	checkPendingUpgrade(db) | ||||
|  | ||||
| 	// Load app settings from DB into the Koanf instance. | ||||
| 	settings := initSettings(db) | ||||
| 	loadSettings(settings) | ||||
|  | ||||
| 	// Fallback for config typo. Logs a warning but continues to work with the incorrect key. | ||||
| 	// Uses 'message.message_outgoing_scan_interval' (correct key) as default key, falls back to the common typo. | ||||
| 	msgOutgoingScanIntervalKey := "message.message_outgoing_scan_interval" | ||||
| 	if ko.String(msgOutgoingScanIntervalKey) == "" { | ||||
| 		if ko.String("message.message_outoing_scan_interval") != "" { | ||||
| 			colorlog.Red("WARNING: typo in config key 'message.message_outoing_scan_interval' detected. Use 'message.message_outgoing_scan_interval' instead in your config.toml file. Support for this incorrect key will be removed in a future release.") | ||||
| 			msgOutgoingScanIntervalKey = "message.message_outoing_scan_interval" | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var ( | ||||
| 		autoAssignInterval          = ko.MustDuration("autoassigner.autoassign_interval") | ||||
| 		unsnoozeInterval            = ko.MustDuration("conversation.unsnooze_interval") | ||||
| 		automationWorkers           = ko.MustInt("automation.worker_count") | ||||
| 		messageOutgoingQWorkers     = ko.MustDuration("message.outgoing_queue_workers") | ||||
| 		messageIncomingQWorkers     = ko.MustDuration("message.incoming_queue_workers") | ||||
| 		messageOutgoingScanInterval = ko.MustDuration("message.message_outoing_scan_interval") | ||||
| 		messageOutgoingScanInterval = ko.MustDuration(msgOutgoingScanIntervalKey) | ||||
| 		slaEvaluationInterval       = ko.MustDuration("sla.evaluation_interval") | ||||
| 		lo                          = initLogger(appName) | ||||
| 		wsHub                       = ws.NewHub() | ||||
| 		rdb                         = initRedis() | ||||
| 		constants                   = initConstants() | ||||
| 		i18n                        = initI18n(fs) | ||||
| 		csat                        = initCSAT(db) | ||||
| 		oidc                        = initOIDC(db, settings) | ||||
| 		status                      = initStatus(db) | ||||
| 		priority                    = initPriority(db) | ||||
| 		auth                        = initAuth(oidc, rdb) | ||||
| 		template                    = initTemplate(db, fs, constants) | ||||
| 		media                       = initMedia(db) | ||||
| 		inbox                       = initInbox(db) | ||||
| 		team                        = initTeam(db) | ||||
| 		businessHours               = initBusinessHours(db) | ||||
| 		csat                        = initCSAT(db, i18n) | ||||
| 		oidc                        = initOIDC(db, settings, i18n) | ||||
| 		status                      = initStatus(db, i18n) | ||||
| 		priority                    = initPriority(db, i18n) | ||||
| 		auth                        = initAuth(oidc, rdb, i18n) | ||||
| 		template                    = initTemplate(db, fs, constants, i18n) | ||||
| 		media                       = initMedia(db, i18n) | ||||
| 		inbox                       = initInbox(db, i18n) | ||||
| 		team                        = initTeam(db, i18n) | ||||
| 		businessHours               = initBusinessHours(db, i18n) | ||||
| 		webhook                     = initWebhook(db, i18n) | ||||
| 		user                        = initUser(i18n, db) | ||||
| 		notifier                    = initNotifier(user) | ||||
| 		automation                  = initAutomationEngine(db) | ||||
| 		sla                         = initSLA(db, team, settings, businessHours) | ||||
| 		conversation                = initConversations(i18n, sla, status, priority, wsHub, notifier, db, inbox, user, team, media, settings, csat, automation, template) | ||||
| 		wsHub                       = initWS(user) | ||||
| 		notifier                    = initNotifier() | ||||
| 		automation                  = initAutomationEngine(db, i18n) | ||||
| 		sla                         = initSLA(db, team, settings, businessHours, notifier, template, user, i18n) | ||||
| 		conversation                = initConversations(i18n, sla, status, priority, wsHub, notifier, db, inbox, user, team, media, settings, csat, automation, template, webhook) | ||||
| 		autoassigner                = initAutoAssigner(team, user, conversation) | ||||
| 	) | ||||
| 	automation.SetConversationStore(conversation) | ||||
|  | ||||
| 	startInboxes(ctx, inbox, conversation) | ||||
| 	startInboxes(ctx, inbox, conversation, user) | ||||
| 	go automation.Run(ctx, automationWorkers) | ||||
| 	go autoassigner.Run(ctx, autoAssignInterval) | ||||
| 	go conversation.Run(ctx, messageIncomingQWorkers, messageOutgoingQWorkers, messageOutgoingScanInterval) | ||||
| 	go conversation.RunUnsnoozer(ctx, unsnoozeInterval) | ||||
| 	go webhook.Run(ctx) | ||||
| 	go notifier.Run(ctx) | ||||
| 	go sla.Run(ctx, slaEvaluationInterval) | ||||
| 	go sla.SendNotifications(ctx) | ||||
| 	go media.DeleteUnlinkedMedia(ctx) | ||||
| 	go user.MonitorAgentAvailability(ctx) | ||||
|  | ||||
| 	var app = &App{ | ||||
| 		lo:            lo, | ||||
| 		fs:            fs, | ||||
| 		sla:           sla, | ||||
| 		oidc:          oidc, | ||||
| 		i18n:          i18n, | ||||
| 		auth:          auth, | ||||
| 		media:         media, | ||||
| 		setting:       settings, | ||||
| 		inbox:         inbox, | ||||
| 		user:          user, | ||||
| 		team:          team, | ||||
| 		status:        status, | ||||
| 		priority:      priority, | ||||
| 		tmpl:          template, | ||||
| 		notifier:      notifier, | ||||
| 		consts:        atomic.Value{}, | ||||
| 		conversation:  conversation, | ||||
| 		automation:    automation, | ||||
| 		businessHours: businessHours, | ||||
| 		authz:         initAuthz(), | ||||
| 		view:          initView(db), | ||||
| 		csat:          initCSAT(db), | ||||
| 		search:        initSearch(db), | ||||
| 		role:          initRole(db), | ||||
| 		tag:           initTag(db), | ||||
| 		macro:         initMacro(db), | ||||
| 		ai:            initAI(db), | ||||
| 		lo:              lo, | ||||
| 		fs:              fs, | ||||
| 		sla:             sla, | ||||
| 		oidc:            oidc, | ||||
| 		i18n:            i18n, | ||||
| 		auth:            auth, | ||||
| 		media:           media, | ||||
| 		setting:         settings, | ||||
| 		inbox:           inbox, | ||||
| 		user:            user, | ||||
| 		team:            team, | ||||
| 		status:          status, | ||||
| 		priority:        priority, | ||||
| 		tmpl:            template, | ||||
| 		notifier:        notifier, | ||||
| 		consts:          atomic.Value{}, | ||||
| 		conversation:    conversation, | ||||
| 		automation:      automation, | ||||
| 		businessHours:   businessHours, | ||||
| 		activityLog:     initActivityLog(db, i18n), | ||||
| 		customAttribute: initCustomAttribute(db, i18n), | ||||
| 		authz:           initAuthz(i18n), | ||||
| 		view:            initView(db, i18n), | ||||
| 		report:          initReport(db, i18n), | ||||
| 		csat:            initCSAT(db, i18n), | ||||
| 		search:          initSearch(db, i18n), | ||||
| 		role:            initRole(db, i18n), | ||||
| 		tag:             initTag(db, i18n), | ||||
| 		macro:           initMacro(db, i18n), | ||||
| 		ai:              initAI(db, i18n), | ||||
| 		webhook:         webhook, | ||||
| 	} | ||||
| 	app.consts.Store(constants) | ||||
|  | ||||
| @@ -226,7 +263,7 @@ func main() { | ||||
| 		WriteTimeout:         ko.MustDuration("app.server.write_timeout"), | ||||
| 		MaxRequestBodySize:   ko.MustInt("app.server.max_body_size"), | ||||
| 		MaxKeepaliveDuration: ko.MustDuration("app.server.keepalive_timeout"), | ||||
| 		ReadBufferSize:       ko.MustInt("app.server.max_body_size"), | ||||
| 		ReadBufferSize:       ko.Int("app.server.read_buffer_size"), | ||||
| 	} | ||||
|  | ||||
| 	go func() { | ||||
| @@ -239,6 +276,11 @@ func main() { | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	// Start the app update checker. | ||||
| 	if ko.Bool("app.check_updates") { | ||||
| 		go checkUpdates(versionString, time.Hour*1, app) | ||||
| 	} | ||||
|  | ||||
| 	// Wait for shutdown signal. | ||||
| 	<-ctx.Done() | ||||
| 	colorlog.Red("Shutting down HTTP server...") | ||||
| @@ -251,6 +293,8 @@ func main() { | ||||
| 	autoassigner.Close() | ||||
| 	colorlog.Red("Shutting down notifier...") | ||||
| 	notifier.Close() | ||||
| 	colorlog.Red("Shutting down webhook...") | ||||
| 	webhook.Close() | ||||
| 	colorlog.Red("Shutting down conversation...") | ||||
| 	conversation.Close() | ||||
| 	colorlog.Red("Shutting down SLA...") | ||||
|   | ||||
							
								
								
									
										30
									
								
								cmd/media.go
									
									
									
									
									
								
							
							
						
						
									
										30
									
								
								cmd/media.go
									
									
									
									
									
								
							| @@ -24,6 +24,7 @@ const ( | ||||
| 	thumbPrefix = "thumb_" | ||||
| ) | ||||
|  | ||||
| // handleMediaUpload handles media uploads. | ||||
| func handleMediaUpload(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app     = r.Context.(*App) | ||||
| @@ -33,19 +34,19 @@ func handleMediaUpload(r *fastglue.Request) error { | ||||
| 	form, err := r.RequestCtx.MultipartForm() | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error parsing form data.", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error parsing data", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	files, ok := form.File["files"] | ||||
| 	if !ok || len(files) == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "File not found", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.notFound", "name", "{globals.terms.file}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	fileHeader := files[0] | ||||
| 	file, err := fileHeader.Open() | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error reading uploaded file", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error reading file", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorReading", "name", "{globals.terms.file}"), nil, envelope.GeneralError) | ||||
| 	} | ||||
| 	defer file.Close() | ||||
|  | ||||
| @@ -74,15 +75,15 @@ func handleMediaUpload(r *fastglue.Request) error { | ||||
| 	if bytesToMegabytes(srcFileSize) > float64(consts.MaxFileUploadSizeMB) { | ||||
| 		app.lo.Error("error: uploaded file size is larger than max allowed", "size", bytesToMegabytes(srcFileSize), "max_allowed", consts.MaxFileUploadSizeMB) | ||||
| 		return r.SendErrorEnvelope( | ||||
| 			http.StatusRequestEntityTooLarge, | ||||
| 			fmt.Sprintf("File size is too large. Please upload file lesser than %d MB", consts.MaxFileUploadSizeMB), | ||||
| 			fasthttp.StatusRequestEntityTooLarge, | ||||
| 			app.i18n.Ts("media.fileSizeTooLarge", "size", fmt.Sprintf("%dMB", consts.MaxFileUploadSizeMB)), | ||||
| 			nil, | ||||
| 			envelope.GeneralError, | ||||
| 		) | ||||
| 	} | ||||
|  | ||||
| 	if !slices.Contains(consts.AllowedUploadFileExtensions, "*") && !slices.Contains(consts.AllowedUploadFileExtensions, srcExt) { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "File type not allowed", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.T("media.fileTypeNotAllowed"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Delete files on any error. | ||||
| @@ -102,11 +103,10 @@ func handleMediaUpload(r *fastglue.Request) error { | ||||
| 		thumbFile, err := image.CreateThumb(image.DefThumbSize, file) | ||||
| 		if err != nil { | ||||
| 			app.lo.Error("error creating thumb image", "error", err) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error creating image thumbnail", nil, envelope.GeneralError) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorCreating", "name", "{globals.terms.thumbnail}"), nil, envelope.GeneralError) | ||||
| 		} | ||||
| 		thumbName, err = app.media.Upload(thumbName, srcContentType, thumbFile) | ||||
| 		if err != nil { | ||||
| 			app.lo.Error("error uploading thumbnail", "error", err) | ||||
| 			return sendErrorEnvelope(r, err) | ||||
| 		} | ||||
|  | ||||
| @@ -116,7 +116,7 @@ func handleMediaUpload(r *fastglue.Request) error { | ||||
| 		if err != nil { | ||||
| 			cleanUp = true | ||||
| 			app.lo.Error("error getting image dimensions", "error", err) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error uploading file", nil, envelope.GeneralError) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorUploading", "name", "{globals.terms.media}"), nil, envelope.GeneralError) | ||||
| 		} | ||||
| 		meta, _ = json.Marshal(map[string]interface{}{ | ||||
| 			"width":  width, | ||||
| @@ -129,7 +129,7 @@ func handleMediaUpload(r *fastglue.Request) error { | ||||
| 	if err != nil { | ||||
| 		cleanUp = true | ||||
| 		app.lo.Error("error uploading file", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error uploading file", nil, envelope.GeneralError) | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Insert in DB. | ||||
| @@ -137,7 +137,7 @@ func handleMediaUpload(r *fastglue.Request) error { | ||||
| 	if err != nil { | ||||
| 		cleanUp = true | ||||
| 		app.lo.Error("error inserting metadata into database", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error inserting media", nil, envelope.GeneralError) | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(media) | ||||
| } | ||||
| @@ -150,13 +150,13 @@ func handleServeMedia(r *fastglue.Request) error { | ||||
| 		uuid  = r.RequestCtx.UserValue("uuid").(string) | ||||
| 	) | ||||
|  | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Fetch media from DB. | ||||
| 	media, err := app.media.GetByUUID(strings.TrimPrefix(uuid, thumbPrefix)) | ||||
| 	media, err := app.media.Get(0, strings.TrimPrefix(uuid, thumbPrefix)) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| @@ -164,7 +164,6 @@ func handleServeMedia(r *fastglue.Request) error { | ||||
| 	// Check if the user has permission to access the linked model. | ||||
| 	allowed, err := app.authz.EnforceMediaAccess(user, media.Model.String) | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error checking media permission", "error", err, "model", media.Model.String, "model_id", media.ModelID) | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| @@ -181,7 +180,7 @@ func handleServeMedia(r *fastglue.Request) error { | ||||
| 	} | ||||
|  | ||||
| 	if !allowed { | ||||
| 		return r.SendErrorEnvelope(http.StatusUnauthorized, "Permission denied", nil, envelope.PermissionError) | ||||
| 		return r.SendErrorEnvelope(http.StatusUnauthorized, app.i18n.Ts("globals.messages.denied", "name", "{globals.terms.permission}"), nil, envelope.UnauthorizedError) | ||||
| 	} | ||||
| 	consts := app.consts.Load().(*constants) | ||||
| 	switch consts.UploadProvider { | ||||
| @@ -193,6 +192,7 @@ func handleServeMedia(r *fastglue.Request) error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // bytesToMegabytes converts bytes to megabytes. | ||||
| func bytesToMegabytes(bytes int64) float64 { | ||||
| 	return float64(bytes) / 1024 / 1024 | ||||
| } | ||||
|   | ||||
| @@ -4,7 +4,6 @@ import ( | ||||
| 	"strconv" | ||||
|  | ||||
| 	amodels "github.com/abhinavxd/libredesk/internal/auth/models" | ||||
| 	"github.com/abhinavxd/libredesk/internal/automation/models" | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| 	medModels "github.com/abhinavxd/libredesk/internal/media/models" | ||||
| 	"github.com/valyala/fasthttp" | ||||
| @@ -15,6 +14,7 @@ type messageReq struct { | ||||
| 	Attachments []int    `json:"attachments"` | ||||
| 	Message     string   `json:"message"` | ||||
| 	Private     bool     `json:"private"` | ||||
| 	To          []string `json:"to"` | ||||
| 	CC          []string `json:"cc"` | ||||
| 	BCC         []string `json:"bcc"` | ||||
| } | ||||
| @@ -30,7 +30,7 @@ func handleGetMessages(r *fastglue.Request) error { | ||||
| 		total       = 0 | ||||
| 	) | ||||
|  | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| @@ -48,11 +48,14 @@ func handleGetMessages(r *fastglue.Request) error { | ||||
|  | ||||
| 	for i := range messages { | ||||
| 		total = messages[i].Total | ||||
| 		// Populate attachment URLs | ||||
| 		for j := range messages[i].Attachments { | ||||
| 			messages[i].Attachments[j].URL = app.media.GetURL(messages[i].Attachments[j].UUID) | ||||
| 		} | ||||
| 		// Redact CSAT survey link | ||||
| 		messages[i].CensorCSATContent() | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(envelope.PageResults{ | ||||
| 		Total:      total, | ||||
| 		Results:    messages, | ||||
| @@ -70,7 +73,7 @@ func handleGetMessage(r *fastglue.Request) error { | ||||
| 		cuuid = r.RequestCtx.UserValue("cuuid").(string) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 	) | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| @@ -96,7 +99,7 @@ func handleGetMessage(r *fastglue.Request) error { | ||||
| 	return r.SendEnvelope(message) | ||||
| } | ||||
|  | ||||
| // handleRetryMessage changes message status so it can be retried for sending. | ||||
| // handleRetryMessage changes message status to `pending`, so it's enqueued for sending. | ||||
| func handleRetryMessage(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| @@ -105,7 +108,7 @@ func handleRetryMessage(r *fastglue.Request) error { | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 	) | ||||
|  | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| @@ -116,8 +119,7 @@ func handleRetryMessage(r *fastglue.Request) error { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	err = app.conversation.MarkMessageAsPending(uuid) | ||||
| 	if err != nil { | ||||
| 	if err = app.conversation.MarkMessageAsPending(uuid); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(true) | ||||
| @@ -129,51 +131,46 @@ func handleSendMessage(r *fastglue.Request) error { | ||||
| 		app   = r.Context.(*App) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		cuuid = r.RequestCtx.UserValue("cuuid").(string) | ||||
| 		media = []medModels.Media{} | ||||
| 		req   = messageReq{} | ||||
| 	) | ||||
|  | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Check permission | ||||
| 	_, err = enforceConversationAccess(app, cuuid, user) | ||||
| 	// Check access to conversation. | ||||
| 	conv, err := enforceConversationAccess(app, cuuid, user) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		app.lo.Error("error unmarshalling message request", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "error decoding request", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Prepare attachments. | ||||
| 	var media = make([]medModels.Media, 0, len(req.Attachments)) | ||||
| 	for _, id := range req.Attachments { | ||||
| 		m, err := app.media.Get(id) | ||||
| 		m, err := app.media.Get(id, "") | ||||
| 		if err != nil { | ||||
| 			app.lo.Error("error fetching media", "error", err) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error fetching media", nil, envelope.GeneralError) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorFetching", "name", "{globals.terms.media}"), nil, envelope.GeneralError) | ||||
| 		} | ||||
| 		media = append(media, m) | ||||
| 	} | ||||
|  | ||||
| 	if req.Private { | ||||
| 		if err := app.conversation.SendPrivateNote(media, user.ID, cuuid, req.Message); err != nil { | ||||
| 		message, err := app.conversation.SendPrivateNote(media, user.ID, cuuid, req.Message) | ||||
| 		if err != nil { | ||||
| 			return sendErrorEnvelope(r, err) | ||||
| 		} | ||||
| 	} else { | ||||
| 		if err := app.conversation.SendReply(media, user.ID, cuuid, req.Message, req.CC, req.BCC, map[string]interface{}{}); err != nil { | ||||
| 			return sendErrorEnvelope(r, err) | ||||
| 		} | ||||
| 		// Evaluate automation rules. | ||||
| 		app.automation.EvaluateConversationUpdateRules(cuuid, models.EventConversationMessageOutgoing) | ||||
| 		return r.SendEnvelope(message) | ||||
| 	} | ||||
|  | ||||
| 	// Reopen if snoozed/closed/resolved regardless of automation rules - this is the default behavior | ||||
| 	if err := app.conversation.ReOpenConversation(cuuid, user); err != nil { | ||||
| 	message, err := app.conversation.QueueReply(media, conv.InboxID, user.ID, cuuid, req.Message, req.To, req.CC, req.BCC, map[string]any{} /**meta**/) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope("Message sent successfully") | ||||
| 	return r.SendEnvelope(message) | ||||
| } | ||||
|   | ||||
| @@ -6,29 +6,80 @@ import ( | ||||
|  | ||||
| 	amodels "github.com/abhinavxd/libredesk/internal/auth/models" | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| 	"github.com/abhinavxd/libredesk/internal/user/models" | ||||
| 	"github.com/valyala/fasthttp" | ||||
| 	"github.com/zerodha/fastglue" | ||||
| 	"github.com/zerodha/simplesessions/v3" | ||||
| ) | ||||
|  | ||||
| // tryAuth is a middleware that attempts to authenticate the user and add them to the context | ||||
| // but doesn't enforce authentication. Handlers can check if user exists in context optionally. | ||||
| // authenticateUser handles both API key and session-based authentication | ||||
| // Returns the authenticated user or an error | ||||
| // For session-based auth, CSRF is checked for POST/PUT/DELETE requests | ||||
| func authenticateUser(r *fastglue.Request, app *App) (models.User, error) { | ||||
| 	var user models.User | ||||
|  | ||||
| 	// Check for Authorization header first (API key authentication) | ||||
| 	apiKey, apiSecret, err := r.ParseAuthHeader(fastglue.AuthBasic | fastglue.AuthToken) | ||||
| 	if err == nil && len(apiKey) > 0 && len(apiSecret) > 0 { | ||||
| 		user, err = app.user.ValidateAPIKey(string(apiKey), string(apiSecret)) | ||||
| 		if err != nil { | ||||
| 			return user, err | ||||
| 		} | ||||
| 		return user, nil | ||||
| 	} | ||||
|  | ||||
| 	// Session-based authentication - Check CSRF first. | ||||
| 	method := string(r.RequestCtx.Method()) | ||||
| 	if method == "POST" || method == "PUT" || method == "DELETE" { | ||||
| 		cookieToken := string(r.RequestCtx.Request.Header.Cookie("csrf_token")) | ||||
| 		hdrToken := string(r.RequestCtx.Request.Header.Peek("X-CSRFTOKEN")) | ||||
|  | ||||
| 		// Match CSRF token from cookie and header. | ||||
| 		if cookieToken == "" || hdrToken == "" || cookieToken != hdrToken { | ||||
| 			app.lo.Error("csrf token mismatch", "method", method, "cookie_token", cookieToken, "header_token", hdrToken) | ||||
| 			return user, envelope.NewError(envelope.PermissionError, app.i18n.T("auth.csrfTokenMismatch"), nil) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Validate session and fetch user. | ||||
| 	sessUser, err := app.auth.ValidateSession(r) | ||||
| 	if err != nil || sessUser.ID <= 0 { | ||||
| 		app.lo.Error("error validating session", "error", err) | ||||
| 		return user, envelope.NewError(envelope.GeneralError, app.i18n.T("auth.invalidOrExpiredSession"), nil) | ||||
| 	} | ||||
|  | ||||
| 	// Get agent user from cache or load it. | ||||
| 	user, err = app.user.GetAgentCachedOrLoad(sessUser.ID) | ||||
| 	if err != nil { | ||||
| 		return user, err | ||||
| 	} | ||||
|  | ||||
| 	// Destroy session if user is disabled. | ||||
| 	if !user.Enabled { | ||||
| 		if err := app.auth.DestroySession(r); err != nil { | ||||
| 			app.lo.Error("error destroying session", "error", err) | ||||
| 		} | ||||
| 		return user, envelope.NewError(envelope.PermissionError, app.i18n.T("user.accountDisabled"), nil) | ||||
| 	} | ||||
|  | ||||
| 	return user, nil | ||||
| } | ||||
|  | ||||
| // tryAuth attempts to authenticate the user and add them to the context but doesn't enforce authentication. | ||||
| // Handlers can check if user exists in context optionally. | ||||
| // Supports both API key authentication (Authorization header) and session-based authentication. | ||||
| func tryAuth(handler fastglue.FastRequestHandler) fastglue.FastRequestHandler { | ||||
| 	return func(r *fastglue.Request) error { | ||||
| 		app := r.Context.(*App) | ||||
|  | ||||
| 		// Try to validate session without returning error. | ||||
| 		userSession, err := app.auth.ValidateSession(r) | ||||
| 		if err != nil || userSession.ID <= 0 { | ||||
| 			return handler(r) | ||||
| 		} | ||||
|  | ||||
| 		// Try to get user. | ||||
| 		user, err := app.user.Get(userSession.ID) | ||||
| 		// Try to authenticate user using shared authentication logic, but don't return errors | ||||
| 		user, err := authenticateUser(r, app) | ||||
| 		if err != nil { | ||||
| 			// Authentication failed, but this is optional, so continue without user | ||||
| 			return handler(r) | ||||
| 		} | ||||
|  | ||||
| 		// Set user in context if found. | ||||
| 		// Set user in context if authentication succeeded. | ||||
| 		r.RequestCtx.SetUserValue("user", amodels.User{ | ||||
| 			ID:        user.ID, | ||||
| 			Email:     user.Email.String, | ||||
| @@ -40,25 +91,25 @@ func tryAuth(handler fastglue.FastRequestHandler) fastglue.FastRequestHandler { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // auth makes sure the user is logged in. | ||||
| // auth validates the session or API key and adds the user to the request context. | ||||
| // Supports both API key authentication (Authorization header) and session-based authentication. | ||||
| func auth(handler fastglue.FastRequestHandler) fastglue.FastRequestHandler { | ||||
| 	return func(r *fastglue.Request) error { | ||||
| 		var ( | ||||
| 			app = r.Context.(*App) | ||||
| 		) | ||||
| 		var app = r.Context.(*App) | ||||
|  | ||||
| 		// Validate session and fetch user. | ||||
| 		userSession, err := app.auth.ValidateSession(r) | ||||
| 		if err != nil || userSession.ID <= 0 { | ||||
| 			app.lo.Error("error validating session", "error", err) | ||||
| 			return r.SendErrorEnvelope(http.StatusUnauthorized, "Invalid or expired session", nil, envelope.PermissionError) | ||||
| 		// Authenticate user using shared authentication logic | ||||
| 		user, err := authenticateUser(r, app) | ||||
| 		if err != nil { | ||||
| 			if envErr, ok := err.(envelope.Error); ok { | ||||
| 				if envErr.ErrorType == envelope.PermissionError { | ||||
| 					return r.SendErrorEnvelope(http.StatusForbidden, envErr.Message, nil, envelope.PermissionError) | ||||
| 				} | ||||
| 				return r.SendErrorEnvelope(http.StatusUnauthorized, envErr.Message, nil, envelope.GeneralError) | ||||
| 			} | ||||
| 			return sendErrorEnvelope(r, err) | ||||
| 		} | ||||
|  | ||||
| 		// Set user in the request context. | ||||
| 		user, err := app.user.Get(userSession.ID) | ||||
| 		if err != nil { | ||||
| 			return sendErrorEnvelope(r, err) | ||||
| 		} | ||||
| 		r.RequestCtx.SetUserValue("user", amodels.User{ | ||||
| 			ID:        user.ID, | ||||
| 			Email:     user.Email.String, | ||||
| @@ -70,45 +121,36 @@ func auth(handler fastglue.FastRequestHandler) fastglue.FastRequestHandler { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // perm does session validation, CSRF, and permission enforcement. | ||||
| // perm checks if the user has the required permission to access the endpoint. | ||||
| // Supports both API key authentication (Authorization header) and session-based authentication. | ||||
| func perm(handler fastglue.FastRequestHandler, perm string) fastglue.FastRequestHandler { | ||||
| 	return func(r *fastglue.Request) error { | ||||
| 		var ( | ||||
| 			app         = r.Context.(*App) | ||||
| 			cookieToken = string(r.RequestCtx.Request.Header.Cookie("csrf_token")) | ||||
| 			hdrToken    = string(r.RequestCtx.Request.Header.Peek("X-CSRFTOKEN")) | ||||
| 		) | ||||
| 		var app = r.Context.(*App) | ||||
|  | ||||
| 		if cookieToken == "" || hdrToken == "" || cookieToken != hdrToken { | ||||
| 			app.lo.Error("csrf token mismatch", "cookie_token", cookieToken, "header_token", hdrToken) | ||||
| 			return r.SendErrorEnvelope(http.StatusForbidden, "Invalid CSRF token", nil, envelope.PermissionError) | ||||
| 		} | ||||
|  | ||||
| 		// Validate session and fetch user. | ||||
| 		sessUser, err := app.auth.ValidateSession(r) | ||||
| 		if err != nil || sessUser.ID <= 0 { | ||||
| 			app.lo.Error("error validating session", "error", err) | ||||
| 			return r.SendErrorEnvelope(http.StatusUnauthorized, "Invalid or expired session", nil, envelope.PermissionError) | ||||
| 		} | ||||
|  | ||||
| 		// Get user from DB. | ||||
| 		user, err := app.user.Get(sessUser.ID) | ||||
| 		// Authenticate user using shared authentication logic | ||||
| 		user, err := authenticateUser(r, app) | ||||
| 		if err != nil { | ||||
| 			if envErr, ok := err.(envelope.Error); ok { | ||||
| 				if envErr.ErrorType == envelope.PermissionError { | ||||
| 					return r.SendErrorEnvelope(http.StatusForbidden, envErr.Message, nil, envelope.PermissionError) | ||||
| 				} | ||||
| 				return r.SendErrorEnvelope(http.StatusUnauthorized, envErr.Message, nil, envelope.GeneralError) | ||||
| 			} | ||||
| 			return sendErrorEnvelope(r, err) | ||||
| 		} | ||||
|  | ||||
| 		// Split the permission string into object and action and enforce it. | ||||
| 		parts := strings.Split(perm, ":") | ||||
| 		if len(parts) != 2 { | ||||
| 			return r.SendErrorEnvelope(http.StatusInternalServerError, "Invalid permission format", nil, envelope.GeneralError) | ||||
| 			return r.SendErrorEnvelope(http.StatusInternalServerError, app.i18n.Ts("globals.messages.invalid", "name", "{globals.terms.permission}"), nil, envelope.GeneralError) | ||||
| 		} | ||||
| 		object, action := parts[0], parts[1] | ||||
| 		ok, err := app.authz.Enforce(user, object, action) | ||||
| 		if err != nil { | ||||
| 			return r.SendErrorEnvelope(http.StatusInternalServerError, "Error checking permissions", nil, envelope.GeneralError) | ||||
| 			return r.SendErrorEnvelope(http.StatusInternalServerError, app.i18n.Ts("globals.messages.errorChecking", "name", "{globals.terms.permission}"), nil, envelope.GeneralError) | ||||
| 		} | ||||
| 		if !ok { | ||||
| 			return r.SendErrorEnvelope(http.StatusForbidden, "Permission denied", nil, envelope.PermissionError) | ||||
| 			return r.SendErrorEnvelope(http.StatusForbidden, app.i18n.Ts("globals.messages.denied", "name", "{globals.terms.permission}"), nil, envelope.PermissionError) | ||||
| 		} | ||||
|  | ||||
| 		// Set user in the request context. | ||||
| @@ -131,9 +173,17 @@ func authPage(handler fastglue.FastRequestHandler) fastglue.FastRequestHandler { | ||||
| 		// Validate session. | ||||
| 		user, err := app.auth.ValidateSession(r) | ||||
| 		if err != nil { | ||||
| 			app.lo.Error("error validating session", "error", err) | ||||
| 			return r.SendErrorEnvelope(http.StatusUnauthorized, "Invalid or expired session", nil, envelope.PermissionError) | ||||
| 			// Session is not valid, destroy it and redirect to login. | ||||
| 			if err != simplesessions.ErrInvalidSession { | ||||
| 				app.lo.Error("error validating session", "error", err) | ||||
| 				return r.SendErrorEnvelope(http.StatusUnauthorized, app.i18n.Ts("globals.messages.errorValidating", "name", "{globals.terms.session}"), nil, envelope.GeneralError) | ||||
| 			} | ||||
| 			if err := app.auth.DestroySession(r); err != nil { | ||||
| 				app.lo.Error("error destroying session", "error", err) | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		// User is authenticated. | ||||
| 		if user.ID > 0 { | ||||
| 			return handler(r) | ||||
| 		} | ||||
| @@ -142,7 +192,7 @@ func authPage(handler fastglue.FastRequestHandler) fastglue.FastRequestHandler { | ||||
| 		if len(nextURI) == 0 { | ||||
| 			nextURI = r.RequestCtx.RequestURI() | ||||
| 		} | ||||
| 		return r.RedirectURI("/", fasthttp.StatusFound, map[string]interface{}{ | ||||
| 		return r.RedirectURI("/", fasthttp.StatusFound, map[string]any{ | ||||
| 			"next": string(nextURI), | ||||
| 		}, "") | ||||
| 	} | ||||
| @@ -157,7 +207,7 @@ func notAuthPage(handler fastglue.FastRequestHandler) fastglue.FastRequestHandle | ||||
| 		user, err := app.auth.ValidateSession(r) | ||||
| 		if err != nil { | ||||
| 			app.lo.Error("error validating session", "error", err) | ||||
| 			return r.SendErrorEnvelope(http.StatusUnauthorized, "Invalid or expired session", nil, envelope.PermissionError) | ||||
| 			return r.SendErrorEnvelope(http.StatusUnauthorized, app.i18n.T("auth.invalidOrExpiredSessionClearCookie"), nil, envelope.GeneralError) | ||||
| 		} | ||||
|  | ||||
| 		if user.ID != 0 { | ||||
|   | ||||
							
								
								
									
										74
									
								
								cmd/oidc.go
									
									
									
									
									
								
							
							
						
						
									
										74
									
								
								cmd/oidc.go
									
									
									
									
									
								
							| @@ -2,23 +2,15 @@ package main | ||||
|  | ||||
| import ( | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| 	"github.com/abhinavxd/libredesk/internal/oidc/models" | ||||
| 	"github.com/abhinavxd/libredesk/internal/stringutil" | ||||
| 	"github.com/valyala/fasthttp" | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| // handleGetAllEnabledOIDC returns all enabled OIDC records | ||||
| func handleGetAllEnabledOIDC(r *fastglue.Request) error { | ||||
| 	app := r.Context.(*App) | ||||
| 	out, err := app.oidc.GetAllEnabled() | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(out) | ||||
| } | ||||
|  | ||||
| // handleGetAllOIDC returns all OIDC records | ||||
| func handleGetAllOIDC(r *fastglue.Request) error { | ||||
| 	app := r.Context.(*App) | ||||
| @@ -26,6 +18,10 @@ func handleGetAllOIDC(r *fastglue.Request) error { | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	// Replace secrets with dummy values. | ||||
| 	for i := range out { | ||||
| 		out[i].ClientSecret = strings.Repeat(stringutil.PasswordDummy, 10) | ||||
| 	} | ||||
| 	return r.SendEnvelope(out) | ||||
| } | ||||
|  | ||||
| @@ -35,7 +31,7 @@ func handleGetOIDC(r *fastglue.Request) error { | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid OIDC `id`", nil, envelope.InputError) | ||||
| 			app.i18n.Ts("globals.messages.invalid", "name", "OIDC `id`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	o, err := app.oidc.Get(id, false) | ||||
| 	if err != nil { | ||||
| @@ -44,18 +40,6 @@ func handleGetOIDC(r *fastglue.Request) error { | ||||
| 	return r.SendEnvelope(o) | ||||
| } | ||||
|  | ||||
| // handleTestOIDC tests an OIDC provider URL by doing a discovery on the provider URL. | ||||
| func handleTestOIDC(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app         = r.Context.(*App) | ||||
| 		providerURL = string(r.RequestCtx.PostArgs().Peek("provider_url")) | ||||
| 	) | ||||
| 	if err := app.auth.TestProvider(providerURL); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("OIDC provider discovered successfully") | ||||
| } | ||||
|  | ||||
| // handleCreateOIDC creates a new OIDC record. | ||||
| func handleCreateOIDC(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| @@ -63,18 +47,28 @@ func handleCreateOIDC(r *fastglue.Request) error { | ||||
| 		req = models.OIDC{} | ||||
| 	) | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Bad request", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	if err := app.oidc.Create(req); err != nil { | ||||
| 	// Test OIDC provider URL by performing a discovery. | ||||
| 	if err := app.auth.TestProvider(req.ProviderURL); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	createdOIDC, err := app.oidc.Create(req) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Reload the auth manager to update the OIDC providers. | ||||
| 	if err := reloadAuth(app); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, err.Error(), nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.couldNotReload", "name", "OIDC"), nil, envelope.GeneralError) | ||||
| 	} | ||||
| 	return r.SendEnvelope("OIDC created successfully") | ||||
|  | ||||
| 	// Clear client secret before returning | ||||
| 	createdOIDC.ClientSecret = strings.Repeat(stringutil.PasswordDummy, 10) | ||||
|  | ||||
| 	return r.SendEnvelope(createdOIDC) | ||||
| } | ||||
|  | ||||
| // handleUpdateOIDC updates an OIDC record. | ||||
| @@ -85,23 +79,32 @@ func handleUpdateOIDC(r *fastglue.Request) error { | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid oidc `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "OIDC `id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Bad request", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	if err = app.oidc.Update(id, req); err != nil { | ||||
| 	// Test OIDC provider URL by performing a discovery. | ||||
| 	if err := app.auth.TestProvider(req.ProviderURL); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	updatedOIDC, err := app.oidc.Update(id, req) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Reload the auth manager to update the OIDC providers. | ||||
| 	if err := reloadAuth(app); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, err.Error(), nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.couldNotReload", "name", "OIDC"), nil, envelope.GeneralError) | ||||
| 	} | ||||
| 	return r.SendEnvelope("OIDC updated successfully") | ||||
|  | ||||
| 	// Clear client secret before returning | ||||
| 	updatedOIDC.ClientSecret = strings.Repeat(stringutil.PasswordDummy, 10) | ||||
|  | ||||
| 	return r.SendEnvelope(updatedOIDC) | ||||
| } | ||||
|  | ||||
| // handleDeleteOIDC deletes an OIDC record. | ||||
| @@ -109,11 +112,10 @@ func handleDeleteOIDC(r *fastglue.Request) error { | ||||
| 	var app = r.Context.(*App) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid oidc `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "OIDC `id`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	if err = app.oidc.Delete(id); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("OIDC deleted successfully") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|   | ||||
							
								
								
									
										45
									
								
								cmd/report.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								cmd/report.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"strconv" | ||||
|  | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| // handleOverviewCounts retrieves general dashboard counts for all users. | ||||
| func handleOverviewCounts(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 	) | ||||
| 	counts, err := app.report.GetOverViewCounts() | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(counts) | ||||
| } | ||||
|  | ||||
| // handleOverviewCharts retrieves general dashboard chart data. | ||||
| func handleOverviewCharts(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app     = r.Context.(*App) | ||||
| 		days, _ = strconv.Atoi(string(r.RequestCtx.QueryArgs().Peek("days"))) | ||||
| 	) | ||||
| 	charts, err := app.report.GetOverviewChart(days) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(charts) | ||||
| } | ||||
|  | ||||
| // handleOverviewSLA retrieves SLA data for the dashboard. | ||||
| func handleOverviewSLA(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app     = r.Context.(*App) | ||||
| 		days, _ = strconv.Atoi(string(r.RequestCtx.QueryArgs().Peek("days"))) | ||||
| 	) | ||||
| 	sla, err := app.report.GetOverviewSLA(days) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(sla) | ||||
| } | ||||
							
								
								
									
										24
									
								
								cmd/roles.go
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								cmd/roles.go
									
									
									
									
									
								
							| @@ -9,17 +9,19 @@ import ( | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| // handleGetRoles returns all roles | ||||
| func handleGetRoles(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 	) | ||||
| 	agents, err := app.role.GetAll() | ||||
| 	roles, err := app.role.GetAll() | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(agents) | ||||
| 	return r.SendEnvelope(roles) | ||||
| } | ||||
|  | ||||
| // handleGetRole returns a single role | ||||
| func handleGetRole(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| @@ -32,33 +34,35 @@ func handleGetRole(r *fastglue.Request) error { | ||||
| 	return r.SendEnvelope(role) | ||||
| } | ||||
|  | ||||
| // handleDeleteRole deletes a role | ||||
| func handleDeleteRole(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| 		id, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
| 	err := app.role.Delete(id) | ||||
| 	if err != nil { | ||||
| 	if err := app.role.Delete(id); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleCreateRole creates a new role | ||||
| func handleCreateRole(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 		req = models.Role{} | ||||
| 	) | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", err.Error(), envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	err := app.role.Create(req) | ||||
| 	createdRole, err := app.role.Create(req) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(true) | ||||
| 	return r.SendEnvelope(createdRole) | ||||
| } | ||||
|  | ||||
| // handleUpdateRole updates a role | ||||
| func handleUpdateRole(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| @@ -66,11 +70,11 @@ func handleUpdateRole(r *fastglue.Request) error { | ||||
| 		req   = models.Role{} | ||||
| 	) | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", err.Error(), envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	err := app.role.Update(id, req) | ||||
| 	updatedRole, err := app.role.Update(id, req) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(true) | ||||
| 	return r.SendEnvelope(updatedRole) | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,8 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
|  | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
| @@ -11,36 +13,45 @@ const ( | ||||
|  | ||||
| // handleSearchConversations searches conversations based on the query. | ||||
| func handleSearchConversations(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 		q   = string(r.RequestCtx.QueryArgs().Peek("query")) | ||||
| 	) | ||||
|  | ||||
| 	if len(q) < minSearchQueryLength { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.InputError, "Query length should be at least 3 characters", nil)) | ||||
| 	app := r.Context.(*App) | ||||
| 	wrapper := func(query string) (interface{}, error) { | ||||
| 		return app.search.Conversations(query) | ||||
| 	} | ||||
|  | ||||
| 	conversations, err := app.search.Conversations(q) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(conversations) | ||||
| 	return handleSearch(r, wrapper) | ||||
| } | ||||
|  | ||||
| // handleSearchMessages searches messages based on the query. | ||||
| func handleSearchMessages(r *fastglue.Request) error { | ||||
| 	app := r.Context.(*App) | ||||
| 	wrapper := func(query string) (interface{}, error) { | ||||
| 		return app.search.Messages(query) | ||||
| 	} | ||||
| 	return handleSearch(r, wrapper) | ||||
| } | ||||
|  | ||||
| // handleSearchContacts searches contacts based on the query. | ||||
| func handleSearchContacts(r *fastglue.Request) error { | ||||
| 	app := r.Context.(*App) | ||||
| 	wrapper := func(query string) (interface{}, error) { | ||||
| 		return app.search.Contacts(query) | ||||
| 	} | ||||
| 	return handleSearch(r, wrapper) | ||||
| } | ||||
|  | ||||
| // handleSearch searches for the given query using the provided search function. | ||||
| func handleSearch(r *fastglue.Request, searchFunc func(string) (interface{}, error)) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 		q   = string(r.RequestCtx.QueryArgs().Peek("query")) | ||||
| 	) | ||||
|  | ||||
| 	if len(q) < minSearchQueryLength { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.InputError, "Query length should be at least 3 characters", nil)) | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.InputError, app.i18n.Ts("search.minQueryLength", "length", fmt.Sprintf("%d", minSearchQueryLength)), nil)) | ||||
| 	} | ||||
|  | ||||
| 	messages, err := app.search.Messages(q) | ||||
| 	results, err := searchFunc(q) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(messages) | ||||
| 	return r.SendEnvelope(results) | ||||
| } | ||||
|   | ||||
| @@ -2,6 +2,7 @@ package main | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"net/mail" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| @@ -11,7 +12,7 @@ import ( | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| // handleGetGeneralSettings fetches general settings. | ||||
| // handleGetGeneralSettings fetches general settings, this endpoint is not behind auth as it has no sensitive data and is required for the app to function. | ||||
| func handleGetGeneralSettings(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| @@ -20,7 +21,19 @@ func handleGetGeneralSettings(r *fastglue.Request) error { | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(out) | ||||
| 	// Unmarshal to set the app.update to the settings, so the frontend can show that an update is available. | ||||
| 	var settings map[string]interface{} | ||||
| 	if err := json.Unmarshal(out, &settings); err != nil { | ||||
| 		app.lo.Error("error unmarshalling settings", "err", err) | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorFetching", "name", app.i18n.T("globals.terms.setting")), nil)) | ||||
| 	} | ||||
| 	// Set the app.update to the settings, adding `app` prefix to the key to match the settings structure in db. | ||||
| 	settings["app.update"] = app.update | ||||
| 	// Set app version. | ||||
| 	settings["app.version"] = versionString | ||||
| 	// Set restart required flag. | ||||
| 	settings["app.restart_required"] = app.restartRequired | ||||
| 	return r.SendEnvelope(settings) | ||||
| } | ||||
|  | ||||
| // handleUpdateGeneralSettings updates general settings. | ||||
| @@ -31,20 +44,39 @@ func handleUpdateGeneralSettings(r *fastglue.Request) error { | ||||
| 	) | ||||
|  | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Bad request", nil, "") | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.T("globals.messages.badRequest"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Get current language before update. | ||||
| 	app.Lock() | ||||
| 	oldLang := ko.String("app.lang") | ||||
| 	app.Unlock() | ||||
|  | ||||
| 	// Remove any trailing slash `/` from the root url. | ||||
| 	req.RootURL = strings.TrimRight(req.RootURL, "/") | ||||
|  | ||||
| 	if err := app.setting.Update(req); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	// Reload the settings and templates. | ||||
| 	if err := reloadSettings(app); err != nil { | ||||
| 		return envelope.NewError(envelope.GeneralError, "Could not reload settings, Please restart the app.", nil) | ||||
| 		return envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.couldNotReload", "name", app.i18n.T("globals.terms.setting")), nil) | ||||
| 	} | ||||
|  | ||||
| 	// Check if language changed and reload i18n if needed. | ||||
| 	app.Lock() | ||||
| 	newLang := ko.String("app.lang") | ||||
| 	if oldLang != newLang { | ||||
| 		app.lo.Info("language changed, reloading i18n", "old_lang", oldLang, "new_lang", newLang) | ||||
| 		app.i18n = initI18n(app.fs) | ||||
| 		app.lo.Info("reloaded i18n", "old_lang", oldLang, "new_lang", newLang) | ||||
| 	} | ||||
| 	app.Unlock() | ||||
|  | ||||
| 	if err := reloadTemplates(app); err != nil { | ||||
| 		return envelope.NewError(envelope.GeneralError, "Could not reload settings, Please restart the app.", nil) | ||||
| 		return envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.couldNotReload", "name", app.i18n.T("globals.terms.setting")), nil) | ||||
| 	} | ||||
| 	return r.SendEnvelope("Settings updated successfully") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleGetEmailNotificationSettings fetches email notification settings. | ||||
| @@ -61,7 +93,7 @@ func handleGetEmailNotificationSettings(r *fastglue.Request) error { | ||||
|  | ||||
| 	// Unmarshal and filter out password. | ||||
| 	if err := json.Unmarshal(out, ¬if); err != nil { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, "Error fetching settings", nil)) | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorFetching", "name", app.i18n.T("globals.terms.setting")), nil)) | ||||
| 	} | ||||
| 	if notif.Password != "" { | ||||
| 		notif.Password = strings.Repeat(stringutil.PasswordDummy, 10) | ||||
| @@ -78,7 +110,7 @@ func handleUpdateEmailNotificationSettings(r *fastglue.Request) error { | ||||
| 	) | ||||
|  | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Bad request", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.T("globals.messages.badRequest"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	out, err := app.setting.GetByPrefix("notification.email") | ||||
| @@ -87,9 +119,15 @@ func handleUpdateEmailNotificationSettings(r *fastglue.Request) error { | ||||
| 	} | ||||
|  | ||||
| 	if err := json.Unmarshal(out, &cur); err != nil { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, "Error updating settings", nil)) | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorUpdating", "name", app.i18n.T("globals.terms.setting")), nil)) | ||||
| 	} | ||||
|  | ||||
| 	// Make sure it's a valid from email address. | ||||
| 	if _, err := mail.ParseAddress(req.EmailAddress); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.T("globals.messages.invalidFromAddress"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// If empty then retain previous password. | ||||
| 	if req.Password == "" { | ||||
| 		req.Password = cur.Password | ||||
| 	} | ||||
| @@ -97,5 +135,11 @@ func handleUpdateEmailNotificationSettings(r *fastglue.Request) error { | ||||
| 	if err := app.setting.Update(req); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("Settings updated successfully, Please restart the app for changes to take effect.") | ||||
|  | ||||
| 	// Email notification settings require app restart to take effect. | ||||
| 	app.Lock() | ||||
| 	app.restartRequired = true | ||||
| 	app.Unlock() | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|   | ||||
							
								
								
									
										158
									
								
								cmd/sla.go
									
									
									
									
									
								
							
							
						
						
									
										158
									
								
								cmd/sla.go
									
									
									
									
									
								
							| @@ -5,10 +5,12 @@ import ( | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| 	smodels "github.com/abhinavxd/libredesk/internal/sla/models" | ||||
| 	"github.com/valyala/fasthttp" | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| // handleGetSLAs returns all SLAs. | ||||
| func handleGetSLAs(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| @@ -20,50 +22,82 @@ func handleGetSLAs(r *fastglue.Request) error { | ||||
| 	return r.SendEnvelope(slas) | ||||
| } | ||||
|  | ||||
| // handleGetSLA returns the SLA with the given ID. | ||||
| func handleGetSLA(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid SLA `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	sla, err := app.sla.Get(id) | ||||
| 	if err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, err.Error(), nil, "") | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(sla) | ||||
| } | ||||
|  | ||||
| // handleCreateSLA creates a new SLA. | ||||
| func handleCreateSLA(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app           = r.Context.(*App) | ||||
| 		name          = string(r.RequestCtx.PostArgs().Peek("name")) | ||||
| 		desc          = string(r.RequestCtx.PostArgs().Peek("description")) | ||||
| 		firstRespTime = string(r.RequestCtx.PostArgs().Peek("first_response_time")) | ||||
| 		resTime       = string(r.RequestCtx.PostArgs().Peek("resolution_time")) | ||||
| 		app = r.Context.(*App) | ||||
| 		sla smodels.SLAPolicy | ||||
| 	) | ||||
| 	// Validate time duration strings | ||||
| 	if _, err := time.ParseDuration(firstRespTime); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid `first_response_time` duration.", nil, envelope.InputError) | ||||
|  | ||||
| 	if err := r.Decode(&sla, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
| 	if _, err := time.ParseDuration(resTime); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid `resolution_time` duration.", nil, envelope.InputError) | ||||
| 	} | ||||
| 	if err := app.sla.Create(name, desc, firstRespTime, resTime); err != nil { | ||||
|  | ||||
| 	if err := validateSLA(app, &sla); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("SLA created successfully.") | ||||
|  | ||||
| 	createdSLA, err := app.sla.Create(sla.Name, sla.Description, sla.FirstResponseTime, sla.ResolutionTime, sla.NextResponseTime, sla.Notifications) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(createdSLA) | ||||
| } | ||||
|  | ||||
| // handleUpdateSLA updates the SLA with the given ID. | ||||
| func handleUpdateSLA(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 		sla smodels.SLAPolicy | ||||
| 	) | ||||
|  | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := r.Decode(&sla, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := validateSLA(app, &sla); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	updatedSLA, err := app.sla.Update(id, sla.Name, sla.Description, sla.FirstResponseTime, sla.ResolutionTime, sla.NextResponseTime, sla.Notifications) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(updatedSLA) | ||||
| } | ||||
|  | ||||
| // handleDeleteSLA deletes the SLA with the given ID. | ||||
| func handleDeleteSLA(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid SLA `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err = app.sla.Delete(id); err != nil { | ||||
| @@ -73,31 +107,83 @@ func handleDeleteSLA(r *fastglue.Request) error { | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| func handleUpdateSLA(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app           = r.Context.(*App) | ||||
| 		name          = string(r.RequestCtx.PostArgs().Peek("name")) | ||||
| 		desc          = string(r.RequestCtx.PostArgs().Peek("description")) | ||||
| 		firstRespTime = string(r.RequestCtx.PostArgs().Peek("first_response_time")) | ||||
| 		resTime       = string(r.RequestCtx.PostArgs().Peek("resolution_time")) | ||||
| 	) | ||||
|  | ||||
| 	// Validate time duration strings | ||||
| 	if _, err := time.ParseDuration(firstRespTime); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid `first_response_time` duration.", nil, envelope.InputError) | ||||
| // validateSLA validates the SLA policy and returns an envelope.Error if any validation fails. | ||||
| func validateSLA(app *App, sla *smodels.SLAPolicy) error { | ||||
| 	if sla.Name == "" { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "`name`"), nil) | ||||
| 	} | ||||
| 	if _, err := time.ParseDuration(resTime); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid `resolution_time` duration.", nil, envelope.InputError) | ||||
| 	if sla.FirstResponseTime.String == "" && sla.NextResponseTime.String == "" && sla.ResolutionTime.String == "" { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "At least one of `first_response_time`, `next_response_time`, or `resolution_time` must be provided."), nil) | ||||
| 	} | ||||
|  | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid SLA `id`.", nil, envelope.InputError) | ||||
| 	// Validate notifications if any. | ||||
| 	for _, n := range sla.Notifications { | ||||
| 		if n.Type == "" { | ||||
| 			return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "`type`"), nil) | ||||
| 		} | ||||
| 		if n.TimeDelayType == "" { | ||||
| 			return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "`time_delay_type`"), nil) | ||||
| 		} | ||||
| 		if n.Metric == "" { | ||||
| 			return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "`metric`"), nil) | ||||
| 		} | ||||
| 		if n.TimeDelayType != "immediately" { | ||||
| 			if n.TimeDelay == "" { | ||||
| 				return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "`time_delay`"), nil) | ||||
| 			} | ||||
| 			// Validate time delay duration. | ||||
| 			td, err := time.ParseDuration(n.TimeDelay) | ||||
| 			if err != nil { | ||||
| 				return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.invalid", "name", "`time_delay`"), nil) | ||||
| 			} | ||||
| 			if td.Minutes() < 1 { | ||||
| 				return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.invalid", "name", "`time_delay`"), nil) | ||||
| 			} | ||||
| 		} | ||||
| 		if len(n.Recipients) == 0 { | ||||
| 			return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "`recipients`"), nil) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if err := app.sla.Update(id, name, desc, firstRespTime, resTime); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	// Validate first response time duration string if not empty. | ||||
| 	if sla.FirstResponseTime.String != "" { | ||||
| 		frt, err := time.ParseDuration(sla.FirstResponseTime.String) | ||||
| 		if err != nil { | ||||
| 			return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.invalid", "name", "`first_response_time`"), nil) | ||||
| 		} | ||||
| 		if frt.Minutes() < 1 { | ||||
| 			return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.invalid", "name", "`first_response_time`"), nil) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| 	// Validate resolution time duration string if not empty. | ||||
| 	if sla.ResolutionTime.String != "" { | ||||
| 		rt, err := time.ParseDuration(sla.ResolutionTime.String) | ||||
| 		if err != nil { | ||||
| 			return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.invalid", "name", "`resolution_time`"), nil) | ||||
| 		} | ||||
| 		if rt.Minutes() < 1 { | ||||
| 			return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.invalid", "name", "`resolution_time`"), nil) | ||||
| 		} | ||||
| 		// Compare with first response time if both are present. | ||||
| 		if sla.FirstResponseTime.String != "" { | ||||
| 			frt, _ := time.ParseDuration(sla.FirstResponseTime.String) | ||||
| 			if frt > rt { | ||||
| 				return envelope.NewError(envelope.InputError, app.i18n.T("sla.firstResponseTimeAfterResolution"), nil) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Validate next response time duration string if not empty. | ||||
| 	if sla.NextResponseTime.String != "" { | ||||
| 		nrt, err := time.ParseDuration(sla.NextResponseTime.String) | ||||
| 		if err != nil { | ||||
| 			return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.invalid", "name", "`next_response_time`"), nil) | ||||
| 		} | ||||
| 		if nrt.Minutes() < 1 { | ||||
| 			return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.invalid", "name", "`next_response_time`"), nil) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|   | ||||
| @@ -26,19 +26,19 @@ func handleCreateStatus(r *fastglue.Request) error { | ||||
| 		status = cmodels.Status{} | ||||
| 	) | ||||
| 	if err := r.Decode(&status, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", err.Error(), envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if status.Name == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty status `Name`", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`name`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	err := app.status.Create(status.Name) | ||||
| 	createdStatus, err := app.status.Create(status.Name) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| 	return r.SendEnvelope(createdStatus) | ||||
| } | ||||
|  | ||||
| func handleDeleteStatus(r *fastglue.Request) error { | ||||
| @@ -46,20 +46,13 @@ func handleDeleteStatus(r *fastglue.Request) error { | ||||
| 		app = r.Context.(*App) | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid status `id`.", nil, envelope.InputError) | ||||
| 	if err != nil || id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty status `ID`", nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	err = app.status.Delete(id) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| @@ -70,22 +63,21 @@ func handleUpdateStatus(r *fastglue.Request) error { | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid status `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := r.Decode(&status, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", err.Error(), envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if status.Name == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty status `Name`", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`name`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	err = app.status.Update(id, status.Name) | ||||
| 	updatedStatus, err := app.status.Update(id, status.Name) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| 	return r.SendEnvelope(updatedStatus) | ||||
| } | ||||
|   | ||||
							
								
								
									
										39
									
								
								cmd/tags.go
									
									
									
									
									
								
							
							
						
						
									
										39
									
								
								cmd/tags.go
									
									
									
									
									
								
							| @@ -9,83 +9,80 @@ import ( | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| // handleGetTags returns all tags from the database. | ||||
| func handleGetTags(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 	) | ||||
| 	t, err := app.tag.GetAll() | ||||
| 	if err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, err.Error(), nil, "") | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(t) | ||||
| } | ||||
|  | ||||
| // handleCreateTag creates a new tag in the database. | ||||
| func handleCreateTag(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 		tag = tmodels.Tag{} | ||||
| 	) | ||||
| 	if err := r.Decode(&tag, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", err.Error(), envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if tag.Name == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty tag `Name`", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`name`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	err := app.tag.Create(tag.Name) | ||||
| 	createdTag, err := app.tag.Create(tag.Name) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| 	return r.SendEnvelope(createdTag) | ||||
| } | ||||
|  | ||||
| // handleDeleteTag deletes a tag from the database. | ||||
| func handleDeleteTag(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid tag `id`.", nil, envelope.InputError) | ||||
| 	if err != nil || id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty tag `ID`", nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	err = app.tag.Delete(id) | ||||
| 	if err != nil { | ||||
| 	if err = app.tag.Delete(id); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleUpdateTag updates an existing tag in the database. | ||||
| func handleUpdateTag(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 		tag = tmodels.Tag{} | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid tag `id`.", nil, envelope.InputError) | ||||
| 	if err != nil || id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := r.Decode(&tag, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", err.Error(), envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if tag.Name == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty tag `Name`", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`name`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	err = app.tag.Update(id, tag.Name) | ||||
| 	updatedTag, err := app.tag.Update(id, tag.Name) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| 	return r.SendEnvelope(updatedTag) | ||||
| } | ||||
|   | ||||
							
								
								
									
										54
									
								
								cmd/teams.go
									
									
									
									
									
								
							
							
						
						
									
										54
									
								
								cmd/teams.go
									
									
									
									
									
								
							| @@ -4,8 +4,8 @@ import ( | ||||
| 	"strconv" | ||||
|  | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| 	"github.com/abhinavxd/libredesk/internal/team/models" | ||||
| 	"github.com/valyala/fasthttp" | ||||
| 	"github.com/volatiletech/null/v9" | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| @@ -40,7 +40,7 @@ func handleGetTeam(r *fastglue.Request) error { | ||||
| 		id, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
| 	if id < 1 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid team `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	team, err := app.team.Get(id) | ||||
| 	if err != nil { | ||||
| @@ -52,41 +52,42 @@ func handleGetTeam(r *fastglue.Request) error { | ||||
| // handleCreateTeam creates a new team. | ||||
| func handleCreateTeam(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app                             = r.Context.(*App) | ||||
| 		name                            = string(r.RequestCtx.PostArgs().Peek("name")) | ||||
| 		timezone                        = string(r.RequestCtx.PostArgs().Peek("timezone")) | ||||
| 		emoji                           = string(r.RequestCtx.PostArgs().Peek("emoji")) | ||||
| 		conversationAssignmentType      = string(r.RequestCtx.PostArgs().Peek("conversation_assignment_type")) | ||||
| 		businessHrsID, _                = strconv.Atoi(string(r.RequestCtx.PostArgs().Peek("business_hours_id"))) | ||||
| 		slaPolicyID, _                  = strconv.Atoi(string(r.RequestCtx.PostArgs().Peek("sla_policy_id"))) | ||||
| 		maxAutoAssignedConversations, _ = strconv.Atoi(string(r.RequestCtx.PostArgs().Peek("max_auto_assigned_conversations"))) | ||||
| 		app = r.Context.(*App) | ||||
| 		req = models.Team{} | ||||
| 	) | ||||
| 	if err := app.team.Create(name, timezone, conversationAssignmentType, null.NewInt(businessHrsID, businessHrsID != 0), null.NewInt(slaPolicyID, slaPolicyID != 0), emoji, maxAutoAssignedConversations); err != nil { | ||||
|  | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil)) | ||||
| 	} | ||||
|  | ||||
| 	createdTeam, err := app.team.Create(req.Name, req.Timezone, req.ConversationAssignmentType, req.BusinessHoursID, req.SLAPolicyID, req.Emoji.String, req.MaxAutoAssignedConversations) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("Team created successfully.") | ||||
| 	return r.SendEnvelope(createdTeam) | ||||
| } | ||||
|  | ||||
| // handleUpdateTeam updates an existing team. | ||||
| func handleUpdateTeam(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app                             = r.Context.(*App) | ||||
| 		name                            = string(r.RequestCtx.PostArgs().Peek("name")) | ||||
| 		timezone                        = string(r.RequestCtx.PostArgs().Peek("timezone")) | ||||
| 		emoji                           = string(r.RequestCtx.PostArgs().Peek("emoji")) | ||||
| 		conversationAssignmentType      = string(r.RequestCtx.PostArgs().Peek("conversation_assignment_type")) | ||||
| 		id, _                           = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 		businessHrsID, _                = strconv.Atoi(string(r.RequestCtx.PostArgs().Peek("business_hours_id"))) | ||||
| 		slaPolicyID, _                  = strconv.Atoi(string(r.RequestCtx.PostArgs().Peek("sla_policy_id"))) | ||||
| 		maxAutoAssignedConversations, _ = strconv.Atoi(string(r.RequestCtx.PostArgs().Peek("max_auto_assigned_conversations"))) | ||||
| 		app   = r.Context.(*App) | ||||
| 		id, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 		req   = models.Team{} | ||||
| 	) | ||||
|  | ||||
| 	if id < 1 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid team `id`", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	if err := app.team.Update(id, name, timezone, conversationAssignmentType, null.NewInt(businessHrsID, businessHrsID != 0), null.NewInt(slaPolicyID, slaPolicyID != 0), emoji, maxAutoAssignedConversations); err != nil { | ||||
|  | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil)) | ||||
| 	} | ||||
|  | ||||
| 	updatedTeam, err := app.team.Update(id, req.Name, req.Timezone, req.ConversationAssignmentType, req.BusinessHoursID, req.SLAPolicyID, req.Emoji.String, req.MaxAutoAssignedConversations) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("Team updated successfully.") | ||||
| 	return r.SendEnvelope(updatedTeam) | ||||
| } | ||||
|  | ||||
| // handleDeleteTeam deletes a team | ||||
| @@ -96,12 +97,11 @@ func handleDeleteTeam(r *fastglue.Request) error { | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid team `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	err = app.team.Delete(id) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("Team deleted successfully.") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|   | ||||
| @@ -16,7 +16,7 @@ func handleGetTemplates(r *fastglue.Request) error { | ||||
| 		typ = string(r.RequestCtx.QueryArgs().Peek("type")) | ||||
| 	) | ||||
| 	if typ == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Invalid `type`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`type`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	t, err := app.tmpl.GetAll(typ) | ||||
| 	if err != nil { | ||||
| @@ -32,8 +32,7 @@ func handleGetTemplate(r *fastglue.Request) error { | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid template `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	t, err := app.tmpl.Get(id) | ||||
| 	if err != nil { | ||||
| @@ -49,12 +48,16 @@ func handleCreateTemplate(r *fastglue.Request) error { | ||||
| 		req = models.Template{} | ||||
| 	) | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Bad request", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	if err := app.tmpl.Create(req); err != nil { | ||||
| 	if req.Name == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`name`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	template, err := app.tmpl.Create(req) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(true) | ||||
| 	return r.SendEnvelope(template) | ||||
| } | ||||
|  | ||||
| // handleUpdateTemplate updates a template. | ||||
| @@ -69,12 +72,16 @@ func handleUpdateTemplate(r *fastglue.Request) error { | ||||
| 			"Invalid template `id`.", nil, envelope.InputError) | ||||
| 	} | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Bad request", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	if err = app.tmpl.Update(id, req); err != nil { | ||||
| 	if req.Name == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`name`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	updatedTemplate, err := app.tmpl.Update(id, req) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(true) | ||||
| 	return r.SendEnvelope(updatedTemplate) | ||||
| } | ||||
|  | ||||
| // handleDeleteTemplate deletes a template. | ||||
| @@ -89,7 +96,7 @@ func handleDeleteTemplate(r *fastglue.Request) error { | ||||
| 			"Invalid template `id`.", nil, envelope.InputError) | ||||
| 	} | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Bad request", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	if err = app.tmpl.Delete(id); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
|   | ||||
							
								
								
									
										98
									
								
								cmd/updates.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								cmd/updates.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,98 @@ | ||||
| // Copyright Kailash Nadh (https://github.com/knadh/listmonk) | ||||
| // SPDX-License-Identifier: AGPL-3.0 | ||||
| // Adapted from listmonk for Libredesk. | ||||
|  | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"regexp" | ||||
| 	"time" | ||||
|  | ||||
| 	"golang.org/x/mod/semver" | ||||
| ) | ||||
|  | ||||
| const updateCheckURL = "https://updates.libredesk.io/updates.json" | ||||
|  | ||||
| type AppUpdate struct { | ||||
| 	Update struct { | ||||
| 		ReleaseVersion string `json:"release_version"` | ||||
| 		ReleaseDate    string `json:"release_date"` | ||||
| 		URL            string `json:"url"` | ||||
| 		Description    string `json:"description"` | ||||
|  | ||||
| 		// This is computed and set locally based on the local version. | ||||
| 		IsNew bool `json:"is_new"` | ||||
| 	} `json:"update"` | ||||
| 	Messages []struct { | ||||
| 		Date        string `json:"date"` | ||||
| 		Title       string `json:"title"` | ||||
| 		Description string `json:"description"` | ||||
| 		URL         string `json:"url"` | ||||
| 		Priority    string `json:"priority"` | ||||
| 	} `json:"messages"` | ||||
| } | ||||
|  | ||||
| var reSemver = regexp.MustCompile(`-(.*)`) | ||||
|  | ||||
| // checkUpdates is a blocking function that checks for updates to the app | ||||
| // at the given intervals. On detecting a new update (new semver), it | ||||
| // sets the global update status that renders a prompt on the UI. | ||||
| func checkUpdates(curVersion string, interval time.Duration, app *App) { | ||||
| 	// Strip -* suffix. | ||||
| 	curVersion = reSemver.ReplaceAllString(curVersion, "") | ||||
|  | ||||
| 	fnCheck := func() { | ||||
| 		resp, err := http.Get(updateCheckURL) | ||||
| 		if err != nil { | ||||
| 			app.lo.Error("error checking for app updates", "err", err) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		if resp.StatusCode != 200 { | ||||
| 			app.lo.Error("non-ok status code checking for app updates", "status", resp.StatusCode) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		b, err := io.ReadAll(resp.Body) | ||||
| 		if err != nil { | ||||
| 			app.lo.Error("error reading response body", "err", err) | ||||
| 			return | ||||
| 		} | ||||
| 		resp.Body.Close() | ||||
|  | ||||
| 		var out AppUpdate | ||||
| 		if err := json.Unmarshal(b, &out); err != nil { | ||||
| 			app.lo.Error("error unmarshalling response body", "err", err) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		// There is an update. Set it on the global app state. | ||||
| 		if semver.IsValid(out.Update.ReleaseVersion) { | ||||
| 			v := reSemver.ReplaceAllString(out.Update.ReleaseVersion, "") | ||||
| 			if semver.Compare(v, curVersion) > 0 { | ||||
| 				out.Update.IsNew = true | ||||
| 				app.lo.Info("new update available", "version", out.Update.ReleaseVersion) | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		app.Lock() | ||||
| 		app.update = &out | ||||
| 		app.Unlock() | ||||
| 	} | ||||
|  | ||||
| 	// Give a 5 minute buffer after app start in case the admin wants to disable | ||||
| 	// update checks entirely and not make a request to upstream. | ||||
| 	time.Sleep(time.Minute * 5) | ||||
| 	fnCheck() | ||||
|  | ||||
| 	// Thereafter, check every $interval. | ||||
| 	ticker := time.NewTicker(interval) | ||||
| 	defer ticker.Stop() | ||||
|  | ||||
| 	for range ticker.C { | ||||
| 		fnCheck() | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										152
									
								
								cmd/upgrade.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										152
									
								
								cmd/upgrade.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,152 @@ | ||||
| // Copyright Kailash Nadh (https://github.com/knadh/listmonk) | ||||
| // SPDX-License-Identifier: AGPL-3.0 | ||||
| // Adapted from listmonk for Libredesk. | ||||
|  | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"log" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/abhinavxd/libredesk/internal/dbutil" | ||||
| 	"github.com/abhinavxd/libredesk/internal/migrations" | ||||
| 	"github.com/jmoiron/sqlx" | ||||
| 	"github.com/knadh/koanf/v2" | ||||
| 	"github.com/knadh/stuffbin" | ||||
| 	"golang.org/x/mod/semver" | ||||
| ) | ||||
|  | ||||
| // migFunc represents a migration function for a particular version. | ||||
| // fn (generally) executes database migrations and additionally | ||||
| // takes the filesystem and config objects in case there are additional bits | ||||
| // of logic to be performed before executing upgrades. fn is idempotent. | ||||
| type migFunc struct { | ||||
| 	version string | ||||
| 	fn      func(*sqlx.DB, stuffbin.FileSystem, *koanf.Koanf) error | ||||
| } | ||||
|  | ||||
| // migList is the list of available migList ordered by the semver. | ||||
| // Each migration is a Go file in internal/migrations named after the semver. | ||||
| // The functions are named as: v0.7.0 => migrations.V0_7_0() and are idempotent. | ||||
| var migList = []migFunc{ | ||||
| 	{"v0.3.0", migrations.V0_3_0}, | ||||
| 	{"v0.4.0", migrations.V0_4_0}, | ||||
| 	{"v0.5.0", migrations.V0_5_0}, | ||||
| 	{"v0.6.0", migrations.V0_6_0}, | ||||
| 	{"v0.7.0", migrations.V0_7_0}, | ||||
| } | ||||
|  | ||||
| // upgrade upgrades the database to the current version by running SQL migration files | ||||
| // for all version from the last known version to the current one. | ||||
| func upgrade(db *sqlx.DB, fs stuffbin.FileSystem, prompt bool) { | ||||
| 	if prompt { | ||||
| 		var ok string | ||||
| 		fmt.Printf("** IMPORTANT: Take a backup of the database before upgrading.\n") | ||||
| 		fmt.Print("continue (y/n)?  ") | ||||
| 		if _, err := fmt.Scanf("%s", &ok); err != nil { | ||||
| 			log.Fatalf("error reading value from terminal: %v", err) | ||||
| 		} | ||||
| 		if !strings.EqualFold(ok, "y") { | ||||
| 			fmt.Println("upgrade cancelled") | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	_, toRun, err := getPendingMigrations(db) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error checking migrations: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	// No migrations to run. | ||||
| 	if len(toRun) == 0 { | ||||
| 		log.Printf("no upgrades to run. Database is up to date.") | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	// Execute migrations in succession. | ||||
| 	for _, m := range toRun { | ||||
| 		log.Printf("running migration %s", m.version) | ||||
| 		if err := m.fn(db, fs, ko); err != nil { | ||||
| 			log.Fatalf("error running migration %s: %v", m.version, err) | ||||
| 		} | ||||
|  | ||||
| 		// Record the migration version in the settings table. There was no | ||||
| 		// settings table until v0.7.0, so ignore the no-table errors. | ||||
| 		if err := recordMigrationVersion(m.version, db); err != nil { | ||||
| 			if dbutil.IsTableNotExistError(err) { | ||||
| 				continue | ||||
| 			} | ||||
| 			log.Fatalf("error recording migration version %s: %v", m.version, err) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	log.Printf("upgrade complete") | ||||
| } | ||||
|  | ||||
| // getPendingMigrations gets the pending migrations by comparing the last | ||||
| // recorded migration in the DB against all migrations listed in `migrations`. | ||||
| func getPendingMigrations(db *sqlx.DB) (string, []migFunc, error) { | ||||
| 	lastVer, err := getLastMigrationVersion(db) | ||||
| 	if err != nil { | ||||
| 		return "", nil, err | ||||
| 	} | ||||
|  | ||||
| 	// Iterate through the migration versions and get everything above the last | ||||
| 	// upgraded semver. | ||||
| 	var toRun []migFunc | ||||
| 	for i, m := range migList { | ||||
| 		if semver.Compare(m.version, lastVer) > 0 { | ||||
| 			toRun = migList[i:] | ||||
| 			break | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return lastVer, toRun, nil | ||||
| } | ||||
|  | ||||
| // getLastMigrationVersion returns the last migration semver recorded in the DB. | ||||
| // If there isn't any, `v0.0.0` is returned. | ||||
| func getLastMigrationVersion(db *sqlx.DB) (string, error) { | ||||
| 	var v string | ||||
| 	if err := db.Get(&v, ` | ||||
| 		SELECT COALESCE( | ||||
| 			(SELECT value->>-1 FROM settings WHERE key='migrations'), | ||||
| 		'v0.0.0')`); err != nil { | ||||
| 		if dbutil.IsTableNotExistError(err) { | ||||
| 			return "v0.0.0", nil | ||||
| 		} | ||||
| 		return v, err | ||||
| 	} | ||||
| 	return v, nil | ||||
| } | ||||
|  | ||||
| // recordMigrationVersion inserts the given version (of DB migration) into the | ||||
| // `migrations` array in the settings table. | ||||
| func recordMigrationVersion(ver string, db *sqlx.DB) error { | ||||
| 	_, err := db.Exec(fmt.Sprintf(`INSERT INTO settings (key, value) | ||||
| 	VALUES('migrations', '["%s"]'::JSONB) | ||||
| 	ON CONFLICT (key) DO UPDATE SET value = settings.value || EXCLUDED.value`, ver)) | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| // checkPendingUpgrade checks if the current database schema matches the expected binary version. | ||||
| func checkPendingUpgrade(db *sqlx.DB) { | ||||
| 	lastVer, toRun, err := getPendingMigrations(db) | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("error checking migrations: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	// No migrations to run. | ||||
| 	if len(toRun) == 0 { | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	var vers []string | ||||
| 	for _, m := range toRun { | ||||
| 		vers = append(vers, m.version) | ||||
| 	} | ||||
|  | ||||
| 	log.Fatalf(`there are %d pending database upgrade(s): %v. The last upgrade was %s. Backup the database and run libredesk --upgrade`, | ||||
| 		len(toRun), vers, lastVer) | ||||
| } | ||||
							
								
								
									
										611
									
								
								cmd/users.go
									
									
									
									
									
								
							
							
						
						
									
										611
									
								
								cmd/users.go
									
									
									
									
									
								
							| @@ -2,7 +2,7 @@ package main | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"mime/multipart" | ||||
| 	"path/filepath" | ||||
| 	"slices" | ||||
| 	"strconv" | ||||
| @@ -16,278 +16,308 @@ import ( | ||||
| 	"github.com/abhinavxd/libredesk/internal/stringutil" | ||||
| 	tmpl "github.com/abhinavxd/libredesk/internal/template" | ||||
| 	"github.com/abhinavxd/libredesk/internal/user/models" | ||||
| 	realip "github.com/ferluci/fast-realip" | ||||
| 	"github.com/valyala/fasthttp" | ||||
| 	"github.com/volatiletech/null/v9" | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	maxAvatarSizeMB = 5 | ||||
| 	maxAvatarSizeMB = 2 | ||||
| ) | ||||
|  | ||||
| // handleGetUsers returns all users. | ||||
| func handleGetUsers(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 	) | ||||
| 	agents, err := app.user.GetAll() | ||||
| 	if err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, err.Error(), nil, "") | ||||
| 	} | ||||
| 	return r.SendEnvelope(agents) | ||||
| type updateAvailabilityRequest struct { | ||||
| 	Status string `json:"status"` | ||||
| } | ||||
|  | ||||
| // handleGetUsersCompact returns all users in a compact format. | ||||
| func handleGetUsersCompact(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 	) | ||||
| 	agents, err := app.user.GetAllCompact() | ||||
| 	if err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, err.Error(), nil, "") | ||||
| 	} | ||||
| 	return r.SendEnvelope(agents) | ||||
| type resetPasswordRequest struct { | ||||
| 	Email string `json:"email"` | ||||
| } | ||||
|  | ||||
| // handleGetUser returns a user. | ||||
| func handleGetUser(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid user `id`.", nil, envelope.InputError) | ||||
| 	} | ||||
| 	user, err := app.user.Get(id) | ||||
| type setPasswordRequest struct { | ||||
| 	Token    string `json:"token"` | ||||
| 	Password string `json:"password"` | ||||
| } | ||||
|  | ||||
| type availabilityRequest struct { | ||||
| 	Status string `json:"status"` | ||||
| } | ||||
|  | ||||
| type agentReq struct { | ||||
| 	FirstName          string   `json:"first_name"` | ||||
| 	LastName           string   `json:"last_name"` | ||||
| 	Email              string   `json:"email"` | ||||
| 	SendWelcomeEmail   bool     `json:"send_welcome_email"` | ||||
| 	Teams              []string `json:"teams"` | ||||
| 	Roles              []string `json:"roles"` | ||||
| 	Enabled            bool     `json:"enabled"` | ||||
| 	AvailabilityStatus string   `json:"availability_status"` | ||||
| 	NewPassword        string   `json:"new_password,omitempty"` | ||||
| } | ||||
|  | ||||
| // handleGetAgents returns all agents. | ||||
| func handleGetAgents(r *fastglue.Request) error { | ||||
| 	var app = r.Context.(*App) | ||||
| 	agents, err := app.user.GetAgents() | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(user) | ||||
| 	return r.SendEnvelope(agents) | ||||
| } | ||||
|  | ||||
| // handleGetCurrentUserTeams returns the teams of a user. | ||||
| func handleGetCurrentUserTeams(r *fastglue.Request) error { | ||||
| // handleGetAgentsCompact returns all agents in a compact format. | ||||
| func handleGetAgentsCompact(r *fastglue.Request) error { | ||||
| 	var app = r.Context.(*App) | ||||
| 	agents, err := app.user.GetAgentsCompact() | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(agents) | ||||
| } | ||||
|  | ||||
| // handleGetAgent returns an agent. | ||||
| func handleGetAgent(r *fastglue.Request) error { | ||||
| 	var app = r.Context.(*App) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
| 	agent, err := app.user.GetAgent(id, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(agent) | ||||
| } | ||||
|  | ||||
| // handleUpdateAgentAvailability updates the current agent availability. | ||||
| func handleUpdateAgentAvailability(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app      = r.Context.(*App) | ||||
| 		auser    = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		ip       = realip.FromRequest(r.RequestCtx) | ||||
| 		availReq availabilityRequest | ||||
| 	) | ||||
|  | ||||
| 	// Decode JSON request | ||||
| 	if err := r.Decode(&availReq, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Fetch entire agent | ||||
| 	agent, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Same status? | ||||
| 	if agent.AvailabilityStatus == availReq.Status { | ||||
| 		return r.SendEnvelope(agent) | ||||
| 	} | ||||
|  | ||||
| 	// Update availability status | ||||
| 	if err := app.user.UpdateAvailability(auser.ID, availReq.Status); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Skip activity log if agent returns online from away (to avoid spam). | ||||
| 	if !(agent.AvailabilityStatus == models.Away && availReq.Status == models.Online) { | ||||
| 		if err := app.activityLog.UserAvailability(auser.ID, auser.Email, availReq.Status, ip, "", 0); err != nil { | ||||
| 			app.lo.Error("error creating activity log", "error", err) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Fetch updated agent and return | ||||
| 	agent, err = app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(agent) | ||||
| } | ||||
|  | ||||
| // handleGetCurrentAgentTeams returns the teams of current agent. | ||||
| func handleGetCurrentAgentTeams(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 	) | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	teams, err := app.team.GetUserTeams(user.ID) | ||||
| 	teams, err := app.team.GetUserTeams(auser.ID) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(teams) | ||||
| } | ||||
|  | ||||
| // handleUpdateCurrentUser updates the current user. | ||||
| func handleUpdateCurrentUser(r *fastglue.Request) error { | ||||
| // handleUpdateCurrentAgent updates the current agent. | ||||
| func handleUpdateCurrentAgent(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 	) | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Get current user. | ||||
| 	currentUser, err := app.user.Get(user.ID) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	form, err := r.RequestCtx.MultipartForm() | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error parsing form data", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error parsing data", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	files, ok := form.File["files"] | ||||
|  | ||||
| 	// Upload avatar? | ||||
| 	if ok && len(files) > 0 { | ||||
| 		fileHeader := files[0] | ||||
| 		file, err := fileHeader.Open() | ||||
| 		agent, err := app.user.GetAgent(auser.ID, "") | ||||
| 		if err != nil { | ||||
| 			app.lo.Error("error reading uploaded", "error", err) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error reading file", nil, envelope.GeneralError) | ||||
| 			return sendErrorEnvelope(r, err) | ||||
| 		} | ||||
| 		defer file.Close() | ||||
|  | ||||
| 		// Sanitize filename. | ||||
| 		srcFileName := stringutil.SanitizeFilename(fileHeader.Filename) | ||||
| 		srcContentType := fileHeader.Header.Get("Content-Type") | ||||
| 		srcFileSize := fileHeader.Size | ||||
| 		srcExt := strings.TrimPrefix(strings.ToLower(filepath.Ext(srcFileName)), ".") | ||||
|  | ||||
| 		if !slices.Contains(image.Exts, srcExt) { | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "File type is not an image", nil, envelope.InputError) | ||||
| 		} | ||||
|  | ||||
| 		// Check file size | ||||
| 		if bytesToMegabytes(srcFileSize) > maxAvatarSizeMB { | ||||
| 			app.lo.Error("error uploaded file size is larger than max allowed", "size", bytesToMegabytes(srcFileSize), "max_allowed", maxAvatarSizeMB) | ||||
| 			return r.SendErrorEnvelope( | ||||
| 				http.StatusRequestEntityTooLarge, | ||||
| 				fmt.Sprintf("File size is too large. Please upload file lesser than %d MB", maxAvatarSizeMB), | ||||
| 				nil, | ||||
| 				envelope.GeneralError, | ||||
| 			) | ||||
| 		} | ||||
|  | ||||
| 		// Reset ptr. | ||||
| 		file.Seek(0, 0) | ||||
| 		linkedModel := null.StringFrom(mmodels.ModelUser) | ||||
| 		linkedID := null.IntFrom(user.ID) | ||||
| 		disposition := null.NewString("", false) | ||||
| 		contentID := "" | ||||
| 		meta := []byte("{}") | ||||
| 		media, err := app.media.UploadAndInsert(srcFileName, srcContentType, contentID, linkedModel, linkedID, file, int(srcFileSize), disposition, meta) | ||||
| 		if err != nil { | ||||
| 			app.lo.Error("error uploading file", "error", err) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error uploading file", nil, envelope.GeneralError) | ||||
| 		} | ||||
|  | ||||
| 		// Delete current avatar. | ||||
| 		if currentUser.AvatarURL.Valid { | ||||
| 			fileName := filepath.Base(currentUser.AvatarURL.String) | ||||
| 			app.media.Delete(fileName) | ||||
| 		} | ||||
|  | ||||
| 		// Save file path. | ||||
| 		path, err := stringutil.GetPathFromURL(media.URL) | ||||
| 		if err != nil { | ||||
| 			app.lo.Debug("error getting path from URL", "url", media.URL, "error", err) | ||||
| 			return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error uploading file", nil, envelope.GeneralError) | ||||
| 		} | ||||
| 		if err := app.user.UpdateAvatar(user.ID, path); err != nil { | ||||
| 		if err := uploadUserAvatar(r, agent, files); err != nil { | ||||
| 			return sendErrorEnvelope(r, err) | ||||
| 		} | ||||
| 	} | ||||
| 	return r.SendEnvelope("User updated successfully.") | ||||
|  | ||||
| 	// Fetch updated agent and return. | ||||
| 	agent, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(agent) | ||||
| } | ||||
|  | ||||
| // handleCreateUser creates a new user. | ||||
| func handleCreateUser(r *fastglue.Request) error { | ||||
| // handleCreateAgent creates a new agent. | ||||
| func handleCreateAgent(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app  = r.Context.(*App) | ||||
| 		user = models.User{} | ||||
| 		app = r.Context.(*App) | ||||
| 		req = agentReq{} | ||||
| 	) | ||||
| 	if err := r.Decode(&user, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", err.Error(), envelope.InputError) | ||||
|  | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if user.Email.String == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty `email`", nil, envelope.InputError) | ||||
| 	// Validate agent request | ||||
| 	if err := validateAgentRequest(r, &req); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if user.Roles == nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Please select at least one role", nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if user.FirstName == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty `first_name`", nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Right now, only agents can be created. | ||||
| 	if err := app.user.CreateAgent(&user); err != nil { | ||||
| 	agent, err := app.user.CreateAgent(req.FirstName, req.LastName, req.Email, req.Roles) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Upsert user teams. | ||||
| 	if err := app.team.UpsertUserTeams(user.ID, user.Teams.Names()); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	if len(req.Teams) > 0 { | ||||
| 		app.team.UpsertUserTeams(agent.ID, req.Teams) | ||||
| 	} | ||||
|  | ||||
| 	if user.SendWelcomeEmail { | ||||
| 	if req.SendWelcomeEmail { | ||||
| 		// Generate reset token. | ||||
| 		resetToken, err := app.user.SetResetPasswordToken(user.ID) | ||||
| 		resetToken, err := app.user.SetResetPasswordToken(agent.ID) | ||||
| 		if err != nil { | ||||
| 			return sendErrorEnvelope(r, err) | ||||
| 		} | ||||
|  | ||||
| 		// Render template and send email. | ||||
| 		content, err := app.tmpl.RenderTemplate(tmpl.TmplWelcome, map[string]interface{}{ | ||||
| 		content, err := app.tmpl.RenderInMemoryTemplate(tmpl.TmplWelcome, map[string]any{ | ||||
| 			"ResetToken": resetToken, | ||||
| 			"Email":      user.Email, | ||||
| 			"Email":      req.Email, | ||||
| 		}) | ||||
| 		if err != nil { | ||||
| 			app.lo.Error("error rendering template", "error", err) | ||||
| 			return r.SendEnvelope("User created successfully, but error rendering welcome email.") | ||||
| 		} | ||||
|  | ||||
| 		if err := app.notifier.Send(notifier.Message{ | ||||
| 			UserIDs:  []int{user.ID}, | ||||
| 			Subject:  "Welcome", | ||||
| 			Content:  content, | ||||
| 			Provider: notifier.ProviderEmail, | ||||
| 			RecipientEmails: []string{req.Email}, | ||||
| 			Subject:         app.i18n.T("globals.messages.welcomeToLibredesk"), | ||||
| 			Content:         content, | ||||
| 			Provider:        notifier.ProviderEmail, | ||||
| 		}); err != nil { | ||||
| 			app.lo.Error("error sending notification message", "error", err) | ||||
| 			return r.SendEnvelope("User created successfully, but error sending welcome email.") | ||||
| 		} | ||||
| 	} | ||||
| 	return r.SendEnvelope("User created successfully.") | ||||
|  | ||||
| 	// Refetch agent as other details might've changed. | ||||
| 	agent, err = app.user.GetAgent(agent.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(agent) | ||||
| } | ||||
|  | ||||
| // handleUpdateUser updates a user. | ||||
| func handleUpdateUser(r *fastglue.Request) error { | ||||
| // handleUpdateAgent updates an agent. | ||||
| func handleUpdateAgent(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app  = r.Context.(*App) | ||||
| 		user = models.User{} | ||||
| 		app   = r.Context.(*App) | ||||
| 		req   = agentReq{} | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		ip    = realip.FromRequest(r.RequestCtx) | ||||
| 		id, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid user `id`.", nil, envelope.InputError) | ||||
| 	if id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := r.Decode(&user, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", err.Error(), envelope.InputError) | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if user.Email.String == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty `email`", nil, envelope.InputError) | ||||
| 	// Validate agent request | ||||
| 	if err := validateAgentRequest(r, &req); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if user.Roles == nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Please select at least one role", nil, envelope.InputError) | ||||
| 	agent, err := app.user.GetAgent(id, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	oldAvailabilityStatus := agent.AvailabilityStatus | ||||
|  | ||||
| 	if user.FirstName == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty `first_name`", nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Update user. | ||||
| 	if err = app.user.Update(id, user); err != nil { | ||||
| 	// Update agent with individual fields | ||||
| 	if err = app.user.UpdateAgent(id, req.FirstName, req.LastName, req.Email, req.Roles, req.Enabled, req.AvailabilityStatus, req.NewPassword); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Upsert user teams. | ||||
| 	if err := app.team.UpsertUserTeams(id, user.Teams.Names()); err != nil { | ||||
| 	// Invalidate authz cache. | ||||
| 	defer app.authz.InvalidateUserCache(id) | ||||
|  | ||||
| 	// Create activity log if user availability status changed. | ||||
| 	if oldAvailabilityStatus != req.AvailabilityStatus { | ||||
| 		if err := app.activityLog.UserAvailability(auser.ID, auser.Email, req.AvailabilityStatus, ip, req.Email, id); err != nil { | ||||
| 			app.lo.Error("error creating activity log", "error", err) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Upsert agent teams. | ||||
| 	if err := app.team.UpsertUserTeams(id, req.Teams); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope("User updated successfully.") | ||||
| 	// Refetch agent and return. | ||||
| 	agent, err = app.user.GetAgent(id, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(agent) | ||||
| } | ||||
|  | ||||
| // handleDeleteUser soft deletes a user. | ||||
| func handleDeleteUser(r *fastglue.Request) error { | ||||
| // handleDeleteAgent soft deletes an agent. | ||||
| func handleDeleteAgent(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app     = r.Context.(*App) | ||||
| 		id, err = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 		auser   = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 	) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid user `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "{globals.terms.user} `id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Disallow if self-deleting. | ||||
| 	if id == auser.ID { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.T("user.userCannotDeleteSelf"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Soft delete user. | ||||
| 	if err = app.user.SoftDelete(id); err != nil { | ||||
| 	if err = app.user.SoftDeleteAgent(id); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| @@ -296,123 +326,278 @@ func handleDeleteUser(r *fastglue.Request) error { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope("User deleted successfully.") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleGetCurrentUser returns the current logged in user. | ||||
| func handleGetCurrentUser(r *fastglue.Request) error { | ||||
| // handleGetCurrentAgent returns the current logged in agent. | ||||
| func handleGetCurrentAgent(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 	) | ||||
| 	u, err := app.user.Get(auser.ID) | ||||
| 	u, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope(u) | ||||
| } | ||||
|  | ||||
| // handleDeleteAvatar deletes a user avatar. | ||||
| func handleDeleteAvatar(r *fastglue.Request) error { | ||||
| // handleDeleteCurrentAgentAvatar deletes the current agent's avatar. | ||||
| func handleDeleteCurrentAgentAvatar(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 	) | ||||
|  | ||||
| 	// Get user | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	agent, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Valid str? | ||||
| 	if user.AvatarURL.String == "" { | ||||
| 	if agent.AvatarURL.String == "" { | ||||
| 		return r.SendEnvelope(true) | ||||
| 	} | ||||
|  | ||||
| 	fileName := filepath.Base(user.AvatarURL.String) | ||||
| 	fileName := filepath.Base(agent.AvatarURL.String) | ||||
|  | ||||
| 	// Delete file from the store. | ||||
| 	if err := app.media.Delete(fileName); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	err = app.user.UpdateAvatar(user.ID, "") | ||||
| 	if err != nil { | ||||
|  | ||||
| 	if err = app.user.UpdateAvatar(agent.ID, ""); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("Avatar deleted successfully.") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleResetPassword generates a reset password token and sends an email to the user. | ||||
| // handleResetPassword generates a reset password token and sends an email to the agent. | ||||
| func handleResetPassword(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app       = r.Context.(*App) | ||||
| 		p         = r.RequestCtx.PostArgs() | ||||
| 		auser, ok = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		email     = string(p.Peek("email")) | ||||
| 		resetReq  resetPasswordRequest | ||||
| 	) | ||||
| 	if ok && auser.ID > 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "User is already logged in", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.T("user.userAlreadyLoggedIn"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if email == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty `email`", nil, envelope.InputError) | ||||
| 	// Decode JSON request | ||||
| 	if err := r.Decode(&resetReq, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	user, err := app.user.GetByEmail(email) | ||||
| 	if resetReq.Email == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`email`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	agent, err := app.user.GetAgent(0, resetReq.Email) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 		// Send 200 even if user not found, to prevent email enumeration. | ||||
| 		return r.SendEnvelope(true) | ||||
| 	} | ||||
|  | ||||
| 	token, err := app.user.SetResetPasswordToken(user.ID) | ||||
| 	token, err := app.user.SetResetPasswordToken(agent.ID) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Send email. | ||||
| 	content, err := app.tmpl.RenderTemplate(tmpl.TmplResetPassword, | ||||
| 		map[string]string{ | ||||
| 			"ResetToken": token, | ||||
| 		}) | ||||
| 	content, err := app.tmpl.RenderInMemoryTemplate(tmpl.TmplResetPassword, map[string]string{ | ||||
| 		"ResetToken": token, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error rendering template", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error rendering template", nil, envelope.GeneralError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.T("globals.messages.errorSendingPasswordResetEmail"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	if err := app.notifier.Send(notifier.Message{ | ||||
| 		UserIDs:  []int{user.ID}, | ||||
| 		Subject:  "Reset Password", | ||||
| 		Content:  content, | ||||
| 		Provider: notifier.ProviderEmail, | ||||
| 		RecipientEmails: []string{agent.Email.String}, | ||||
| 		Subject:         "Reset Password", | ||||
| 		Content:         content, | ||||
| 		Provider:        notifier.ProviderEmail, | ||||
| 	}); err != nil { | ||||
| 		app.lo.Error("error sending notification message", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, "Error sending notification message", nil, envelope.GeneralError) | ||||
| 		app.lo.Error("error sending password reset email", "error", err) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusInternalServerError, app.i18n.T("globals.messages.errorSendingPasswordResetEmail"), nil, envelope.GeneralError) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope("Reset password email sent successfully.") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleSetPassword resets the password with the provided token. | ||||
| func handleSetPassword(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app      = r.Context.(*App) | ||||
| 		user, ok = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		p        = r.RequestCtx.PostArgs() | ||||
| 		password = string(p.Peek("password")) | ||||
| 		token    = string(p.Peek("token")) | ||||
| 		app       = r.Context.(*App) | ||||
| 		agent, ok = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 		req       setPasswordRequest | ||||
| 	) | ||||
|  | ||||
| 	if ok && user.ID > 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "User is already logged in", nil, envelope.InputError) | ||||
| 	if ok && agent.ID > 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.T("user.userAlreadyLoggedIn"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if password == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty `password`", nil, envelope.InputError) | ||||
| 	if err := r.Decode(&req, "json"); err != nil { | ||||
| 		return sendErrorEnvelope(r, envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), nil)) | ||||
| 	} | ||||
|  | ||||
| 	if err := app.user.ResetPassword(token, password); err != nil { | ||||
| 	if req.Password == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "{globals.terms.password}"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := app.user.ResetPassword(req.Token, req.Password); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope("Password reset successfully.") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // uploadUserAvatar uploads the user avatar. | ||||
| func uploadUserAvatar(r *fastglue.Request, user models.User, files []*multipart.FileHeader) error { | ||||
| 	var app = r.Context.(*App) | ||||
|  | ||||
| 	fileHeader := files[0] | ||||
| 	file, err := fileHeader.Open() | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error opening uploaded file", "user_id", user.ID, "error", err) | ||||
| 		return envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorReading", "name", "{globals.terms.file}"), nil) | ||||
| 	} | ||||
| 	defer file.Close() | ||||
|  | ||||
| 	// Sanitize filename. | ||||
| 	srcFileName := stringutil.SanitizeFilename(fileHeader.Filename) | ||||
| 	srcContentType := fileHeader.Header.Get("Content-Type") | ||||
| 	srcFileSize := fileHeader.Size | ||||
| 	srcExt := strings.TrimPrefix(strings.ToLower(filepath.Ext(srcFileName)), ".") | ||||
|  | ||||
| 	if !slices.Contains(image.Exts, srcExt) { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.T("globals.messages.fileTypeisNotAnImage"), nil) | ||||
| 	} | ||||
|  | ||||
| 	// Check file size | ||||
| 	if bytesToMegabytes(srcFileSize) > maxAvatarSizeMB { | ||||
| 		app.lo.Error("error uploaded file size is larger than max allowed", "user_id", user.ID, "size", bytesToMegabytes(srcFileSize), "max_allowed", maxAvatarSizeMB) | ||||
| 		return envelope.NewError( | ||||
| 			envelope.InputError, | ||||
| 			app.i18n.Ts("media.fileSizeTooLarge", "size", fmt.Sprintf("%dMB", maxAvatarSizeMB)), | ||||
| 			nil, | ||||
| 		) | ||||
| 	} | ||||
|  | ||||
| 	// Reset ptr. | ||||
| 	file.Seek(0, 0) | ||||
| 	linkedModel := null.StringFrom(mmodels.ModelUser) | ||||
| 	linkedID := null.IntFrom(user.ID) | ||||
| 	disposition := null.NewString("", false) | ||||
| 	contentID := "" | ||||
| 	meta := []byte("{}") | ||||
| 	media, err := app.media.UploadAndInsert(srcFileName, srcContentType, contentID, linkedModel, linkedID, file, int(srcFileSize), disposition, meta) | ||||
| 	if err != nil { | ||||
| 		app.lo.Error("error uploading file", "user_id", user.ID, "error", err) | ||||
| 		return envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorUploading", "name", "{globals.terms.file}"), nil) | ||||
| 	} | ||||
|  | ||||
| 	// Delete current avatar. | ||||
| 	if user.AvatarURL.Valid { | ||||
| 		fileName := filepath.Base(user.AvatarURL.String) | ||||
| 		if err := app.media.Delete(fileName); err != nil { | ||||
| 			app.lo.Error("error deleting user avatar", "user_id", user.ID, "error", err) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Save file path. | ||||
| 	path, err := stringutil.GetPathFromURL(media.URL) | ||||
| 	if err != nil { | ||||
| 		app.lo.Debug("error getting path from URL", "user_id", user.ID, "url", media.URL, "error", err) | ||||
| 		return envelope.NewError(envelope.GeneralError, app.i18n.Ts("globals.messages.errorUploading", "name", "{globals.terms.file}"), nil) | ||||
| 	} | ||||
|  | ||||
| 	if err := app.user.UpdateAvatar(user.ID, path); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // handleGenerateAPIKey generates a new API key for a user | ||||
| func handleGenerateAPIKey(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| 		id, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
| 	if id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Check if user exists | ||||
| 	user, err := app.user.GetAgent(id, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Generate API key and secret | ||||
| 	apiKey, apiSecret, err := app.user.GenerateAPIKey(user.ID) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Return the API key and secret (only shown once) | ||||
| 	response := struct { | ||||
| 		APIKey    string `json:"api_key"` | ||||
| 		APISecret string `json:"api_secret"` | ||||
| 	}{ | ||||
| 		APIKey:    apiKey, | ||||
| 		APISecret: apiSecret, | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(response) | ||||
| } | ||||
|  | ||||
| // handleRevokeAPIKey revokes a user's API key | ||||
| func handleRevokeAPIKey(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| 		id, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
| 	if id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Check if user exists | ||||
| 	_, err := app.user.GetAgent(id, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Revoke API key | ||||
| 	if err := app.user.RevokeAPIKey(id); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // validateAgentRequest validates common agent request fields and normalizes the email | ||||
| func validateAgentRequest(r *fastglue.Request, req *agentReq) error { | ||||
| 	var app = r.Context.(*App) | ||||
|  | ||||
| 	// Normalize email | ||||
| 	req.Email = strings.TrimSpace(strings.ToLower(req.Email)) | ||||
| 	if req.Email == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`email`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if !stringutil.ValidEmail(req.Email) { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`email`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if req.Roles == nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`role`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if req.FirstName == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`first_name`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|   | ||||
							
								
								
									
										65
									
								
								cmd/views.go
									
									
									
									
									
								
							
							
						
						
									
										65
									
								
								cmd/views.go
									
									
									
									
									
								
							| @@ -16,7 +16,7 @@ func handleGetUserViews(r *fastglue.Request) error { | ||||
| 		app   = r.Context.(*App) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 	) | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| @@ -35,61 +35,50 @@ func handleCreateUserView(r *fastglue.Request) error { | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 	) | ||||
| 	if err := r.Decode(&view, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", err.Error(), envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	if view.Name == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty view `Name`", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`Name`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if string(view.Filters) == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty view `Filter`", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`Filters`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := app.view.Create(view.Name, view.Filters, user.ID); err != nil { | ||||
| 	createdView, err := app.view.Create(view.Name, view.Filters, user.ID) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	return r.SendEnvelope("View created successfully") | ||||
| 	return r.SendEnvelope(createdView) | ||||
| } | ||||
|  | ||||
| // handleGetUserView deletes a view for a user. | ||||
| // handleDeleteUserView deletes a view for a user. | ||||
| func handleDeleteUserView(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| 		auser = r.RequestCtx.UserValue("user").(amodels.User) | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid view `id`.", nil, envelope.InputError) | ||||
| 	if err != nil || id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty view `ID`", nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	view, err := app.view.Get(id) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	if view.UserID != user.ID { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusForbidden, "Forbidden", nil, envelope.PermissionError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusForbidden, app.i18n.Ts("globals.messages.denied", "name", "{globals.terms.permission}"), nil, envelope.PermissionError) | ||||
| 	} | ||||
|  | ||||
| 	if err = app.view.Delete(id); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope("View deleted successfully") | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleUpdateUserView updates a view for a user. | ||||
| @@ -101,39 +90,31 @@ func handleUpdateUserView(r *fastglue.Request) error { | ||||
| 	) | ||||
| 	id, err := strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	if err != nil || id == 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, | ||||
| 			"Invalid view `id`.", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := r.Decode(&view, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "decode failed", err.Error(), envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	user, err := app.user.Get(auser.ID) | ||||
| 	user, err := app.user.GetAgent(auser.ID, "") | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	if view.Name == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty view `Name`", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`name`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if string(view.Filters) == "" { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, "Empty view `Filter`", nil, envelope.InputError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.empty", "name", "`filters`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	v, err := app.view.Get(id) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	if v.UserID != user.ID { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusForbidden, "Forbidden", nil, envelope.PermissionError) | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusForbidden, app.i18n.Ts("globals.messages.denied", "name", "{globals.terms.permission}"), nil, envelope.PermissionError) | ||||
| 	} | ||||
|  | ||||
| 	if err = app.view.Update(id, view.Name, view.Filters); err != nil { | ||||
| 	updatedView, err := app.view.Update(id, view.Name, view.Filters) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| 	return r.SendEnvelope(updatedView) | ||||
| } | ||||
|   | ||||
							
								
								
									
										191
									
								
								cmd/webhooks.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										191
									
								
								cmd/webhooks.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,191 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/abhinavxd/libredesk/internal/envelope" | ||||
| 	"github.com/abhinavxd/libredesk/internal/stringutil" | ||||
| 	"github.com/abhinavxd/libredesk/internal/webhook/models" | ||||
| 	"github.com/valyala/fasthttp" | ||||
| 	"github.com/zerodha/fastglue" | ||||
| ) | ||||
|  | ||||
| // handleGetWebhooks returns all webhooks from the database. | ||||
| func handleGetWebhooks(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app = r.Context.(*App) | ||||
| 	) | ||||
| 	webhooks, err := app.webhook.GetAll() | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
| 	// Hide secrets. | ||||
| 	for i := range webhooks { | ||||
| 		if webhooks[i].Secret != "" { | ||||
| 			webhooks[i].Secret = strings.Repeat(stringutil.PasswordDummy, 10) | ||||
| 		} | ||||
| 	} | ||||
| 	return r.SendEnvelope(webhooks) | ||||
| } | ||||
|  | ||||
| // handleGetWebhook returns a specific webhook by ID. | ||||
| func handleGetWebhook(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| 		id, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
| 	if id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	webhook, err := app.webhook.Get(id) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Hide secret in the response. | ||||
| 	if webhook.Secret != "" { | ||||
| 		webhook.Secret = strings.Repeat(stringutil.PasswordDummy, 10) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(webhook) | ||||
| } | ||||
|  | ||||
| // handleCreateWebhook creates a new webhook in the database. | ||||
| func handleCreateWebhook(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app     = r.Context.(*App) | ||||
| 		webhook = models.Webhook{} | ||||
| 	) | ||||
| 	if err := r.Decode(&webhook, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Validate webhook fields | ||||
| 	if err := validateWebhook(app, webhook); err != nil { | ||||
| 		return r.SendEnvelope(err) | ||||
| 	} | ||||
|  | ||||
| 	webhook, err := app.webhook.Create(webhook) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Clear secret before returning | ||||
| 	webhook.Secret = strings.Repeat(stringutil.PasswordDummy, 10) | ||||
|  | ||||
| 	return r.SendEnvelope(webhook) | ||||
| } | ||||
|  | ||||
| // handleUpdateWebhook updates an existing webhook in the database. | ||||
| func handleUpdateWebhook(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app     = r.Context.(*App) | ||||
| 		webhook = models.Webhook{} | ||||
| 		id, _   = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
|  | ||||
| 	if id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := r.Decode(&webhook, "json"); err != nil { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.errorParsing", "name", "{globals.terms.request}"), err.Error(), envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	// Validate webhook fields | ||||
| 	if err := validateWebhook(app, webhook); err != nil { | ||||
| 		return r.SendEnvelope(err) | ||||
| 	} | ||||
|  | ||||
| 	// If secret is empty or contains dummy characters, fetch existing webhook and preserve the secret | ||||
| 	if webhook.Secret == "" || strings.Contains(webhook.Secret, stringutil.PasswordDummy) { | ||||
| 		existingWebhook, err := app.webhook.Get(id) | ||||
| 		if err != nil { | ||||
| 			return sendErrorEnvelope(r, err) | ||||
| 		} | ||||
| 		webhook.Secret = existingWebhook.Secret | ||||
| 	} | ||||
|  | ||||
| 	updatedWebhook, err := app.webhook.Update(id, webhook) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Clear secret before returning | ||||
| 	updatedWebhook.Secret = strings.Repeat(stringutil.PasswordDummy, 10) | ||||
|  | ||||
| 	return r.SendEnvelope(updatedWebhook) | ||||
| } | ||||
|  | ||||
| // handleDeleteWebhook deletes a webhook from the database. | ||||
| func handleDeleteWebhook(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| 		id, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
| 	if id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := app.webhook.Delete(id); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // handleToggleWebhook toggles the active status of a webhook. | ||||
| func handleToggleWebhook(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| 		id, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
|  | ||||
| 	if id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	toggledWebhook, err := app.webhook.Toggle(id) | ||||
| 	if err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	// Clear secret before returning | ||||
| 	toggledWebhook.Secret = strings.Repeat(stringutil.PasswordDummy, 10) | ||||
|  | ||||
| 	return r.SendEnvelope(toggledWebhook) | ||||
| } | ||||
|  | ||||
| // handleTestWebhook sends a test payload to a webhook. | ||||
| func handleTestWebhook(r *fastglue.Request) error { | ||||
| 	var ( | ||||
| 		app   = r.Context.(*App) | ||||
| 		id, _ = strconv.Atoi(r.RequestCtx.UserValue("id").(string)) | ||||
| 	) | ||||
|  | ||||
| 	if id <= 0 { | ||||
| 		return r.SendErrorEnvelope(fasthttp.StatusBadRequest, app.i18n.Ts("globals.messages.invalid", "name", "`id`"), nil, envelope.InputError) | ||||
| 	} | ||||
|  | ||||
| 	if err := app.webhook.SendTestWebhook(id); err != nil { | ||||
| 		return sendErrorEnvelope(r, err) | ||||
| 	} | ||||
|  | ||||
| 	return r.SendEnvelope(true) | ||||
| } | ||||
|  | ||||
| // validateWebhook validates the webhook data. | ||||
| func validateWebhook(app *App, webhook models.Webhook) error { | ||||
| 	if webhook.Name == "" { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "`name`"), nil) | ||||
| 	} | ||||
| 	if webhook.URL == "" { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "`url`"), nil) | ||||
| 	} | ||||
| 	if len(webhook.Events) == 0 { | ||||
| 		return envelope.NewError(envelope.InputError, app.i18n.Ts("globals.messages.empty", "name", "`events`"), nil) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| @@ -1,74 +1,124 @@ | ||||
| # App. | ||||
| [app] | ||||
| # Log level: info, debug, warn, error, fatal | ||||
| log_level = "debug" | ||||
| # Environment: dev, prod. | ||||
| # Setting to "dev" will enable color logging in terminal. | ||||
| env = "dev" | ||||
| # Whether to automatically check for application updates on start up, app updates are shown as a banner in the admin panel. | ||||
| check_updates = true | ||||
|  | ||||
| # HTTP server. | ||||
| [app.server] | ||||
| # Address to bind the HTTP server to. | ||||
| address = "0.0.0.0:9000" | ||||
| # Unix socket path (leave empty to use TCP address instead) | ||||
| socket = "" | ||||
| # Do NOT disable secure cookies in production environment if you don't know exactly what you're doing! | ||||
| disable_secure_cookies = false | ||||
| # Request read and write timeouts. | ||||
| read_timeout = "5s" | ||||
| write_timeout = "5s" | ||||
| max_body_size = 500000000 | ||||
| # Maximum request body size in bytes (100MB) | ||||
| # If you are using proxy, you may need to configure them to allow larger request bodies. | ||||
| max_body_size = 104857600 | ||||
| # Size of the read buffer for incoming requests | ||||
| read_buffer_size = 4096 | ||||
| # Keepalive settings. | ||||
| keepalive_timeout = "10s" | ||||
|  | ||||
| # File upload provider to use. | ||||
| # File upload provider to use, either `fs` or `s3`. | ||||
| [upload] | ||||
| provider = "fs" | ||||
|  | ||||
| # Filesytem provider. | ||||
| # Filesystem provider. | ||||
| [upload.fs] | ||||
| # Directory where uploaded files are stored, make sure this directory exists and is writable by the application. | ||||
| upload_path = 'uploads' | ||||
|  | ||||
| # S3 provider. | ||||
| [upload.s3] | ||||
| # S3 endpoint URL (required only for non-AWS S3-compatible providers like MinIO). | ||||
| # Leave empty to use default AWS endpoints. | ||||
| url = "" | ||||
|  | ||||
| # AWS S3 credentials, keep empty to use attached IAM roles. | ||||
| access_key = "" | ||||
| secret_key = "" | ||||
|  | ||||
| # AWS region, e.g., "us-east-1", "eu-west-1", etc. | ||||
| region = "ap-south-1" | ||||
| bucket = "bucket" | ||||
| # S3 bucket name where files will be stored. | ||||
| bucket = "bucket-name" | ||||
| # Optional prefix path within the S3 bucket where files will be stored. | ||||
| # Example, if set to "uploads/media", files will be stored under that path. | ||||
| # Useful for organizing files inside a shared bucket. | ||||
| bucket_path = "" | ||||
| expiry = "6h" | ||||
| # S3 signed URL expiry duration (e.g., "30m", "1h") | ||||
| expiry = "30m" | ||||
|  | ||||
| # Postgres. | ||||
| [db] | ||||
| # If using docker compose, use the service name as the host. | ||||
| host = "127.0.0.1" | ||||
| # If running locally, use `localhost`. | ||||
| host = "db" | ||||
| # Database port, default is 5432. | ||||
| port = 5432 | ||||
| user = "postgres" | ||||
| password = "postgres" | ||||
| # Update the following values with your database credentials. | ||||
| user = "libredesk" | ||||
| password = "libredesk" | ||||
| database = "libredesk" | ||||
| ssl_mode = "disable" | ||||
| # Maximum number of open database connections | ||||
| max_open = 30 | ||||
| # Maximum number of idle connections in the pool | ||||
| max_idle = 30 | ||||
| # Maximum time a connection can be reused before being closed | ||||
| max_lifetime = "300s" | ||||
|  | ||||
| # Redis. | ||||
| [redis] | ||||
| # If using docker compose, use the service name as the host. | ||||
| address = "127.0.0.1:6379" | ||||
| # If running locally, use `localhost:6379`. | ||||
| address = "redis:6379" | ||||
| password = "" | ||||
| db = 0 | ||||
|  | ||||
| [message] | ||||
| # Number of workers processing outgoing message queue | ||||
| outgoing_queue_workers = 10 | ||||
| # Number of workers processing incoming message queue | ||||
| incoming_queue_workers = 10 | ||||
| message_outoing_scan_interval = "50ms" | ||||
| # How often to scan for outgoing messages to process, keep it low to process messages quickly. | ||||
| message_outgoing_scan_interval = "50ms" | ||||
| # Maximum number of messages that can be queued for incoming processing | ||||
| incoming_queue_size = 5000 | ||||
| # Maximum number of messages that can be queued for outgoing processing | ||||
| outgoing_queue_size = 5000 | ||||
|  | ||||
| [notification] | ||||
| # Number of concurrent notification workers | ||||
| concurrency = 2 | ||||
| # Maximum number of notifications that can be queued | ||||
| queue_size = 2000 | ||||
|  | ||||
| [automation] | ||||
| # Number of workers processing automation rules | ||||
| worker_count = 10 | ||||
|  | ||||
| [autoassigner] | ||||
| # How often to run automatic conversation assignment | ||||
| autoassign_interval = "5m" | ||||
|  | ||||
| [webhook] | ||||
| # Number of webhook delivery workers | ||||
| workers = 5 | ||||
| # Maximum number of webhook deliveries that can be queued | ||||
| queue_size = 10000 | ||||
| # HTTP timeout for webhook requests | ||||
| timeout = "15s" | ||||
|  | ||||
| [conversation] | ||||
| # How often to check for conversations to unsnooze | ||||
| unsnooze_interval = "5m" | ||||
|  | ||||
| [sla] | ||||
| evaluation_interval = "5m" | ||||
| # How often to evaluate SLA compliance for conversations | ||||
| evaluation_interval = "5m" | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| services: | ||||
|   # Libredesk app | ||||
|   app: | ||||
|     image: libredesk:latest | ||||
|     image: libredesk/libredesk:latest | ||||
|     container_name: libredesk_app | ||||
|     restart: unless-stopped | ||||
|     ports: | ||||
| @@ -28,14 +28,15 @@ services: | ||||
|     networks: | ||||
|       - libredesk | ||||
|     ports: | ||||
|       - "5432:5432" | ||||
|       # Only bind on the local interface. To connect to Postgres externally, change this to 0.0.0.0 | ||||
|       - "127.0.0.1:5432:5432" | ||||
|     environment: | ||||
|       # Set these environment variables to configure the database, defaults to libredesk. | ||||
|       POSTGRES_USER: ${POSTGRES_USER:-libredesk} | ||||
|       POSTGRES_PASSWORD: ${POSTGRES_PASSWORD:-libredesk}  | ||||
|       POSTGRES_DB: ${POSTGRES_DB:-libredesk} | ||||
|     healthcheck: | ||||
|       test: ["CMD-SHELL", "pg_isready -U libredesk"] | ||||
|       test: ["CMD-SHELL", "pg_isready -U ${POSTGRES_USER:-libredesk} -d ${POSTGRES_DB:-libredesk}"] | ||||
|       interval: 10s | ||||
|       timeout: 5s | ||||
|       retries: 6 | ||||
| @@ -48,7 +49,8 @@ services: | ||||
|     container_name: libredesk_redis | ||||
|     restart: unless-stopped | ||||
|     ports: | ||||
|       - "6379:6379" | ||||
|       # Only bind on the local interface. | ||||
|       - "127.0.0.1:6379:6379" | ||||
|     networks: | ||||
|       - libredesk | ||||
|     volumes: | ||||
| @@ -59,4 +61,4 @@ networks: | ||||
|  | ||||
| volumes: | ||||
|   postgres-data: | ||||
|   redis-data: | ||||
|   redis-data: | ||||
|   | ||||
							
								
								
									
										30
									
								
								docs/docs/api-getting-started.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								docs/docs/api-getting-started.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| # API getting started | ||||
|  | ||||
| You can access the Libredesk API to interact with your instance programmatically. | ||||
|  | ||||
| ## Generating API keys | ||||
|  | ||||
| 1. **Edit agent**: Go to Admin → Teammate → Agent → Edit | ||||
| 2. **Generate new API key**: An API Key and API Secret will be generated for the agent | ||||
| 3. **Save the credentials**: Keep both the API Key and API Secret secure | ||||
| 4. **Key management**: You can revoke / regenerate API keys at any time from the same page | ||||
|  | ||||
| ## Using the API | ||||
|  | ||||
| LibreDesk supports two authentication schemes: | ||||
|  | ||||
| ### Basic authentication | ||||
| ```bash | ||||
| curl -X GET "https://your-libredesk-instance.com/api/endpoint" \ | ||||
|   -H "Authorization: Basic <base64_encoded_key:secret>" | ||||
| ``` | ||||
|  | ||||
| ### Token authentication | ||||
| ```bash | ||||
| curl -X GET "https://your-libredesk-instance.com/api/endpoint" \ | ||||
|   -H "Authorization: token your_api_key:your_api_secret" | ||||
| ``` | ||||
|  | ||||
| ## API Documentation | ||||
|  | ||||
| Complete API documentation with available endpoints and examples coming soon. | ||||
							
								
								
									
										32
									
								
								docs/docs/developer-setup.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								docs/docs/developer-setup.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| # Developer Setup | ||||
|  | ||||
| Libredesk is a monorepo with a Go backend and a Vue.js frontend. The frontend uses Shadcn for UI components. | ||||
|  | ||||
| ### Pre-requisites | ||||
|  | ||||
| - go | ||||
| - nodejs (if you are working on the frontend) and `pnpm` | ||||
| - redis | ||||
| - postgres database (>= 13) | ||||
|  | ||||
| ### First time setup | ||||
|  | ||||
| Clone the repository: | ||||
|  | ||||
| ```sh | ||||
| git clone https://github.com/abhinavxd/libredesk.git | ||||
| ``` | ||||
|  | ||||
| 1. Copy `config.toml.sample` as `config.toml` and add your config. | ||||
| 2. Run `make` to build the libredesk binary. Once the binary is built, run `./libredesk --install` to run the DB setup and set the System user password. | ||||
|  | ||||
| ### Running the Dev Environment | ||||
|  | ||||
| 1. Run `make run-backend` to start the libredesk backend dev server on `:9000`. | ||||
| 2. Run `make run-frontend` to start the Vue frontend in dev mode using pnpm on `:8000`. Requests are proxied to the backend running on `:9000` check `vite.config.js` for the proxy config. | ||||
|  | ||||
| --- | ||||
|  | ||||
| # Production Build | ||||
|  | ||||
| Run `make` to build the Go binary, build the Javascript frontend, and embed the static assets producing a single self-contained binary, `libredesk`. | ||||
							
								
								
									
										
											BIN
										
									
								
								docs/docs/images/hero.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								docs/docs/images/hero.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 298 KiB | 
							
								
								
									
										17
									
								
								docs/docs/index.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								docs/docs/index.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| # Introduction | ||||
|  | ||||
| Libredesk is an open-source, self-hosted customer support desk — single binary app. | ||||
|  | ||||
| <div style="border: 1px solid #ccc; padding: 2px; border-radius: 6px; box-shadow: 0 2px 6px rgba(0, 0, 0, 0.1); background-color: #f9f9f9;"> | ||||
|   <a href="https://libredesk.io"> | ||||
|     <img src="images/hero.png" alt="libredesk UI screenshot" style="display: block; margin: 0 auto; max-width: 100%; border-radius: 4px;" /> | ||||
|   </a> | ||||
| </div> | ||||
|  | ||||
| ## Developers | ||||
|  | ||||
| Libredesk is licensed under AGPLv3. Contributions are welcome. | ||||
|  | ||||
| - Source code: [GitHub](https://github.com/abhinavxd/libredesk) | ||||
| - Setup guide: [Developer setup](developer-setup.md) | ||||
| - Stack: Go backend, Vue 3 frontend (Shadcn UI) | ||||
							
								
								
									
										65
									
								
								docs/docs/installation.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								docs/docs/installation.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,65 @@ | ||||
| # Installation | ||||
|  | ||||
| Libredesk is a single binary application that requires postgres and redis to run. You can install it using the binary or docker. | ||||
|  | ||||
| ## Binary | ||||
|  | ||||
| 1. Download the [latest release](https://github.com/abhinavxd/libredesk/releases) and extract the libredesk binary. | ||||
| 2. `./libredesk --install` to install the tables in the Postgres DB (⩾ 13) and set the System user password. | ||||
| 3. Run `./libredesk` and visit `http://localhost:9000` and login with the email `System` and the password you set during installation. | ||||
|  | ||||
| !!! Tip | ||||
|     To set the System user password during installation, set the environment variables: | ||||
|     `LIBREDESK_SYSTEM_USER_PASSWORD=xxxxxxxxxxx ./libredesk --install` | ||||
|  | ||||
|  | ||||
| ## Docker | ||||
|  | ||||
| The latest image is available on DockerHub at `libredesk/libredesk:latest` | ||||
|  | ||||
| The recommended method is to download the [docker-compose.yml](https://github.com/abhinavxd/libredesk/blob/main/docker-compose.yml) file, customize it for your environment and then to simply run `docker compose up -d`. | ||||
|  | ||||
| ```shell | ||||
| # Download the compose file and the sample config file in the current directory. | ||||
| curl -LO https://github.com/abhinavxd/libredesk/raw/main/docker-compose.yml | ||||
| curl -LO https://github.com/abhinavxd/libredesk/raw/main/config.sample.toml | ||||
|  | ||||
| # Copy the config.sample.toml to config.toml and edit it as needed. | ||||
| cp config.sample.toml config.toml | ||||
|  | ||||
| # Run the services in the background. | ||||
| docker compose up -d | ||||
|  | ||||
| # Setting System user password. | ||||
| docker exec -it libredesk_app ./libredesk --set-system-user-password | ||||
| ``` | ||||
|  | ||||
| Go to `http://localhost:9000` and login with the email `System` and the password you set using the `--set-system-user-password` command. | ||||
|  | ||||
|  | ||||
| ## Compiling from source | ||||
|  | ||||
| To compile the latest unreleased version (`main` branch): | ||||
|  | ||||
| 1. Make sure `go`, `nodejs`, and `pnpm` are installed on your system. | ||||
| 2. `git clone git@github.com:abhinavxd/libredesk.git` | ||||
| 3. `cd libredesk && make`. This will generate the `libredesk` binary. | ||||
|  | ||||
|  | ||||
| ## Nginx | ||||
|  | ||||
| Libredesk uses websockets for real-time updates. If you are using Nginx, you need to add the following (or similar) configuration to your Nginx configuration file. | ||||
|  | ||||
| ```nginx | ||||
| client_max_body_size 100M; | ||||
| location / { | ||||
|     proxy_pass http://localhost:9000; | ||||
|     proxy_http_version 1.1; | ||||
|     proxy_set_header Upgrade $http_upgrade; | ||||
|     proxy_set_header Connection 'upgrade'; | ||||
|     proxy_set_header Host $host; | ||||
|     proxy_set_header X-Real-IP $remote_addr; | ||||
|     proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; | ||||
|     proxy_cache_bypass $http_upgrade; | ||||
| } | ||||
| ``` | ||||
							
								
								
									
										57
									
								
								docs/docs/sso.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								docs/docs/sso.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,57 @@ | ||||
| # Setting up SSO | ||||
|  | ||||
| Libredesk supports external OpenID Connect providers (e.g., Google, Keycloak) for signing in users. | ||||
|  | ||||
| !!! note | ||||
|     User accounts must be created in Libredesk manually; signup is not supported. | ||||
|  | ||||
| ## Generic Configuration Steps | ||||
|  | ||||
| Since each provider’s configuration might differ, consult your provider’s documentation for any additional or divergent settings. | ||||
|  | ||||
| 1. Provider setup:   | ||||
|    In your provider’s admin console, create a new OpenID Connect application/client. Retrieve: | ||||
|       - Client ID | ||||
|       - Client Secret | ||||
|  | ||||
| 2. Libredesk configuration:  | ||||
|    In Libredesk, navigate to Security > SSO and click New SSO and enter the following details: | ||||
|       - Provider URL (e.g., the URL of your OpenID provider) | ||||
|       - Client ID | ||||
|       - Client Secret | ||||
|       - A descriptive name for the connection | ||||
|  | ||||
| 3. Redirect URL:   | ||||
|    After saving, copy the generated Callback URL from Libredesk and add it as a valid redirect URI in your provider’s client settings. | ||||
|     | ||||
| ## Provider Examples | ||||
|  | ||||
| #### Keycloak | ||||
|  | ||||
| 1. Log in to your Keycloak Admin Console. | ||||
|  | ||||
| 2. In Keycloak, navigate to Clients and click Create: | ||||
|  | ||||
|       - Client ID (e.g., `libredesk-app`) | ||||
|       - Client Protocol: `openid-connect` | ||||
|       - Root URL and Web Origins: your app domain (e.g., `https://ticket.example.com`) | ||||
|       - Under Authentication flow, uncheck everything except the standard flow | ||||
|       - Click save | ||||
|  | ||||
| 3. Go to the credentials tab: | ||||
|       - Ensure client authenticator is set to `Client Id and Secret` | ||||
|       - Note down the generated client secret | ||||
|  | ||||
| 4. In Libredesk, go to Admin > Security > SSO and click New SSO: | ||||
|       - Provider URL (e.g., `https://keycloak.example.com/realms/yourrealm`) | ||||
|       - Name (e.g., `Keycloak`) | ||||
|       - Client ID | ||||
|       - Client secret | ||||
|       - Click save | ||||
|  | ||||
| 5. After saving, click on the three dots and choose Edit to open the new SSO entry. | ||||
|  | ||||
| 6. Copy the generated Callback URL from Libredesk. | ||||
|  | ||||
| 7. Back in Keycloak, edit the client and add the Callback URL to Valid Redirect URIs: | ||||
|       - e.g., `https://ticket.example.com/api/v1/oidc/1/finish` | ||||
							
								
								
									
										60
									
								
								docs/docs/templating.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								docs/docs/templating.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| # Templating | ||||
|  | ||||
| Templating in outgoing emails allows you to personalize content by embedding dynamic expressions like `{{ .Recipient.FullName }}`. These expressions reference fields from the conversation, contact, recipient, and author objects. | ||||
|  | ||||
| ## Outgoing Email Template Expressions | ||||
|  | ||||
| If you want to customize the look of outgoing emails, you can do so in the Admin > Templates -> Outgoing Email Templates section. This template will be used for all outgoing emails including replies to conversations, notifications, and other system-generated emails. | ||||
|  | ||||
| ### Conversation Variables | ||||
|  | ||||
| | Variable | Value | | ||||
| |---------------------------------|--------------------------------------------------------| | ||||
| | {{ .Conversation.ReferenceNumber }} | The unique reference number of the conversation | | ||||
| | {{ .Conversation.Subject }} | The subject of the conversation | | ||||
| | {{ .Conversation.Priority }} | The priority level of the conversation | | ||||
| | {{ .Conversation.UUID }} | The unique identifier of the conversation | | ||||
|  | ||||
| ### Contact Variables | ||||
|  | ||||
| | Variable | Value | | ||||
| |------------------------------|------------------------------------| | ||||
| | {{ .Contact.FirstName }} | First name of the contact/customer | | ||||
| | {{ .Contact.LastName }} | Last name of the contact/customer | | ||||
| | {{ .Contact.FullName }} | Full name of the contact/customer | | ||||
| | {{ .Contact.Email }} | Email address of the contact/customer | | ||||
|  | ||||
| ### Recipient Variables | ||||
|  | ||||
| | Variable | Value | | ||||
| |--------------------------------|-----------------------------------| | ||||
| | {{ .Recipient.FirstName }} | First name of the recipient | | ||||
| | {{ .Recipient.LastName }} | Last name of the recipient | | ||||
| | {{ .Recipient.FullName }} | Full name of the recipient | | ||||
| | {{ .Recipient.Email }} | Email address of the recipient | | ||||
|  | ||||
| ### Author Variables | ||||
|  | ||||
| | Variable | Value | | ||||
| |------------------------------|-----------------------------------| | ||||
| | {{ .Author.FirstName }} | First name of the message author | | ||||
| | {{ .Author.LastName }} | Last name of the message author | | ||||
| | {{ .Author.FullName }} | Full name of the message author | | ||||
| | {{ .Author.Email }} | Email address of the message author | | ||||
|  | ||||
| ### Example outgoing email template | ||||
|  | ||||
| ```html | ||||
| Dear {{ .Recipient.FirstName }}, | ||||
|  | ||||
| {{ template "content" . }} | ||||
|  | ||||
| Best regards, | ||||
| {{ .Author.FullName }} | ||||
| --- | ||||
| Reference: {{ .Conversation.ReferenceNumber }} | ||||
| ``` | ||||
|  | ||||
| Here, the `{{ template "content" . }}` serves as a placeholder for the body of the outgoing email. It will be replaced with the actual email content at the time of sending. | ||||
|  | ||||
| Similarly, the `{{ .Recipient.FirstName }}` expression will dynamically insert the recipient's first name when the email is sent. | ||||
							
								
								
									
										3
									
								
								docs/docs/translations.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								docs/docs/translations.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| # Translations / Internationalization | ||||
|  | ||||
| You can help translate libreDesk into different languages by contributing here: [Libredesk Translation Project](https://crowdin.com/project/libredesk) | ||||
							
								
								
									
										18
									
								
								docs/docs/upgrade.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								docs/docs/upgrade.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| # Upgrade | ||||
|  | ||||
| !!! warning "Warning" | ||||
|     Always take a backup of the Postgres database before upgrading Libredesk. | ||||
|  | ||||
| ## Binary | ||||
| - Stop running libredesk binary. | ||||
| - Download the [latest release](https://github.com/abhinavxd/libredesk/releases) and extract the libredesk binary and overwrite the previous version. | ||||
| - `./libredesk --upgrade` to upgrade an existing database schema. Upgrades are idempotent and running them multiple times have no side effects. | ||||
| - Run `./libredesk` again. | ||||
|  | ||||
| ## Docker | ||||
|  | ||||
| ```shell | ||||
| docker compose down app | ||||
| docker compose pull | ||||
| docker compose up app -d | ||||
| ``` | ||||
							
								
								
									
										222
									
								
								docs/docs/webhooks.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										222
									
								
								docs/docs/webhooks.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,222 @@ | ||||
| # Webhooks | ||||
|  | ||||
| Webhooks allow you to receive real-time HTTP notifications when specific events occur in your Libredesk instance. This enables you to integrate Libredesk with external systems and automate workflows based on conversation and message events. | ||||
|  | ||||
| ## Overview | ||||
|  | ||||
| When a configured event occurs in Libredesk, a HTTP POST request is sent to the webhook URL you specify. The request contains a JSON payload with event details and relevant data. | ||||
|  | ||||
| ## Webhook Configuration | ||||
|  | ||||
| 1. Navigate to **Admin > Integrations > Webhooks** in your Libredesk dashboard | ||||
| 2. Click **Create Webhook** | ||||
| 3. Configure the following: | ||||
|    - **Name**: A descriptive name for your webhook | ||||
|    - **URL**: The endpoint URL where webhook payloads will be sent | ||||
|    - **Events**: Select which events you want to subscribe to | ||||
|    - **Secret**: Optional secret key for signature verification | ||||
|    - **Status**: Enable or disable the webhook | ||||
|  | ||||
| ## Security | ||||
|  | ||||
| ### Signature Verification | ||||
|  | ||||
| If you provide a secret key, webhook payloads will be signed using HMAC-SHA256. The signature is included in the `X-Signature-256` header in the format `sha256=<signature>`. | ||||
|  | ||||
| To verify the signature: | ||||
|  | ||||
| ```python | ||||
| import hmac | ||||
| import hashlib | ||||
|  | ||||
| def verify_signature(payload, signature, secret): | ||||
|     expected_signature = hmac.new( | ||||
|         secret.encode('utf-8'), | ||||
|         payload, | ||||
|         hashlib.sha256 | ||||
|     ).hexdigest() | ||||
|     return hmac.compare_digest(f"sha256={expected_signature}", signature) | ||||
| ``` | ||||
|  | ||||
| ### Headers | ||||
|  | ||||
| Each webhook request includes the following headers: | ||||
|  | ||||
| - `Content-Type`: `application/json` | ||||
| - `User-Agent`: `Libredesk-Webhook/<libredesk_version_here>` | ||||
| - `X-Signature-256`: HMAC signature (if secret is configured) | ||||
|  | ||||
| ## Available Events | ||||
|  | ||||
| ### Conversation Events | ||||
|  | ||||
| #### `conversation.created` | ||||
| Triggered when a new conversation is created. | ||||
|  | ||||
| **Sample Payload:** | ||||
| ```json | ||||
| { | ||||
|   "event": "conversation.created", | ||||
|   "timestamp": "2025-06-15T10:30:00Z", | ||||
|   "payload": { | ||||
|     "id": 123, | ||||
|     "created_at": "2025-06-15T10:30:00Z", | ||||
|     "updated_at": "2025-06-15T10:30:00Z", | ||||
|     "uuid": "550e8400-e29b-41d4-a716-446655440000", | ||||
|     "contact_id": 456, | ||||
|     "inbox_id": 1, | ||||
|     "reference_number": "100", | ||||
|     "priority": "Medium", | ||||
|     "priority_id": 2, | ||||
|     "status": "Open", | ||||
|     "status_id": 1, | ||||
|     "subject": "Help with account setup", | ||||
|     "inbox_name": "Support", | ||||
|     "inbox_channel": "email", | ||||
|     "contact": { | ||||
|       "id": 456, | ||||
|       "first_name": "John", | ||||
|       "last_name": "Doe", | ||||
|       "email": "john.doe@example.com", | ||||
|       "type": "contact" | ||||
|     }, | ||||
|     "custom_attributes": {}, | ||||
|     "tags": [] | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| #### `conversation.status_changed` | ||||
| Triggered when a conversation's status is updated. | ||||
|  | ||||
| **Sample Payload:** | ||||
| ```json | ||||
| { | ||||
|   "event": "conversation.status_changed", | ||||
|   "timestamp": "2025-06-15T10:35:00Z", | ||||
|   "payload": { | ||||
|     "conversation_uuid": "550e8400-e29b-41d4-a716-446655440000", | ||||
|     "previous_status": "Open", | ||||
|     "new_status": "Resolved", | ||||
|     "snooze_until": "", | ||||
|     "actor_id": 789 | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| #### `conversation.assigned` | ||||
| Triggered when a conversation is assigned to a user. | ||||
|  | ||||
| **Sample Payload:** | ||||
| ```json | ||||
| { | ||||
|   "event": "conversation.assigned", | ||||
|   "timestamp": "2025-06-15T10:32:00Z", | ||||
|   "payload": { | ||||
|     "conversation_uuid": "550e8400-e29b-41d4-a716-446655440000", | ||||
|     "assigned_to": 789, | ||||
|     "actor_id": 789 | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| #### `conversation.unassigned` | ||||
| Triggered when a conversation is unassigned from a user. | ||||
|  | ||||
| **Sample Payload:** | ||||
| ```json | ||||
| { | ||||
|   "event": "conversation.unassigned", | ||||
|   "timestamp": "2025-06-15T10:40:00Z", | ||||
|   "payload": { | ||||
|     "conversation_uuid": "550e8400-e29b-41d4-a716-446655440000", | ||||
|     "actor_id": 789 | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| #### `conversation.tags_changed` | ||||
| Triggered when tags are added or removed from a conversation. | ||||
|  | ||||
| **Sample Payload:** | ||||
| ```json | ||||
| { | ||||
|   "event": "conversation.tags_changed", | ||||
|   "timestamp": "2025-06-15T10:45:00Z", | ||||
|   "payload": { | ||||
|     "conversation_uuid": "550e8400-e29b-41d4-a716-446655440000", | ||||
|     "previous_tags": ["bug", "priority"], | ||||
|     "new_tags": ["bug", "priority", "resolved"], | ||||
|     "actor_id": 789 | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ### Message Events | ||||
|  | ||||
| #### `message.created` | ||||
| Triggered when a new message is created in a conversation. | ||||
|  | ||||
| **Sample Payload:** | ||||
| ```json | ||||
| { | ||||
|   "event": "message.created", | ||||
|   "timestamp": "2025-06-15T10:33:00Z", | ||||
|   "payload": { | ||||
|     "id": 987, | ||||
|     "created_at": "2025-06-15T10:33:00Z", | ||||
|     "updated_at": "2025-06-15T10:33:00Z", | ||||
|     "uuid": "123e4567-e89b-12d3-a456-426614174000", | ||||
|     "type": "outgoing", | ||||
|     "status": "sent", | ||||
|     "conversation_id": 123, | ||||
|     "content": "<p>Hello! How can I help you today?</p>", | ||||
|     "text_content": "Hello! How can I help you today?", | ||||
|     "content_type": "html", | ||||
|     "private": false, | ||||
|     "sender_id": 789, | ||||
|     "sender_type": "agent", | ||||
|     "attachments": [] | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| #### `message.updated` | ||||
| Triggered when an existing message is updated. | ||||
|  | ||||
| **Sample Payload:** | ||||
| ```json | ||||
| { | ||||
|   "event": "message.updated", | ||||
|   "timestamp": "2025-06-15T10:34:00Z", | ||||
|   "payload": { | ||||
|     "id": 987, | ||||
|     "created_at": "2025-06-15T10:33:00Z", | ||||
|     "updated_at": "2025-06-15T10:34:00Z", | ||||
|     "uuid": "123e4567-e89b-12d3-a456-426614174000", | ||||
|     "type": "outgoing", | ||||
|     "status": "sent", | ||||
|     "conversation_id": 123, | ||||
|     "content": "<p>Hello! How can I help you today? (Updated)</p>", | ||||
|     "text_content": "Hello! How can I help you today? (Updated)", | ||||
|     "content_type": "html", | ||||
|     "private": false, | ||||
|     "sender_id": 789, | ||||
|     "sender_type": "agent", | ||||
|     "attachments": [] | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ## Delivery and Retries | ||||
|  | ||||
| - Webhooks requests timeout can be configured in the `config.toml` file | ||||
| - Failed deliveries are not automatically retried | ||||
| - Webhook delivery runs in a background worker pool for better performance | ||||
| - If the webhook queue is full (configurable in config.toml file), new events may be dropped | ||||
|  | ||||
| ## Testing Webhooks | ||||
|  | ||||
| You can test your webhook configuration using tools like: | ||||
|  | ||||
| - [Webhook.site](https://webhook.site) - Generate a temporary URL to inspect webhook payloads | ||||
							
								
								
									
										38
									
								
								docs/mkdocs.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								docs/mkdocs.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | ||||
| site_name: Libredesk Docs | ||||
| theme: | ||||
|   name: material | ||||
|   language: en | ||||
|   font: | ||||
|     text: Source Sans Pro | ||||
|     code: Roboto Mono | ||||
|     weights: [400, 700] | ||||
|   direction: ltr | ||||
|   palette: | ||||
|     primary: white | ||||
|     accent: red | ||||
|   features: | ||||
|     - navigation.indexes | ||||
|     - navigation.sections | ||||
|     - content.code.copy | ||||
| extra: | ||||
|   search: | ||||
|     language: en | ||||
|  | ||||
| markdown_extensions: | ||||
|   - admonition | ||||
|   - codehilite | ||||
|   - toc: | ||||
|       permalink: true | ||||
|  | ||||
| nav: | ||||
|   - Introduction: index.md | ||||
|   - Getting Started: | ||||
|       - Installation: installation.md | ||||
|       - Upgrade Guide: upgrade.md | ||||
|       - Email Templates: templating.md | ||||
|       - SSO Setup: sso.md | ||||
|       - Webhooks: webhooks.md | ||||
|       - API Getting Started: api-getting-started.md | ||||
|   - Contributions: | ||||
|       - Developer Setup: developer-setup.md | ||||
|       - Translate Libredesk: translations.md | ||||
							
								
								
									
										8
									
								
								frontend/.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								frontend/.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							| @@ -1,8 +0,0 @@ | ||||
| { | ||||
|   "recommendations": [ | ||||
|     "Vue.volar", | ||||
|     "Vue.vscode-typescript-vue-plugin", | ||||
|     "dbaeumer.vscode-eslint", | ||||
|     "esbenp.prettier-vscode" | ||||
|   ] | ||||
| } | ||||
| @@ -8,7 +8,6 @@ | ||||
|     "baseColor": "gray", | ||||
|     "cssVariables": true | ||||
|   }, | ||||
|   "framework": "vite", | ||||
|   "aliases": { | ||||
|     "components": "@/components", | ||||
|     "utils": "@/lib/utils" | ||||
|   | ||||
| @@ -3,7 +3,7 @@ import { defineConfig } from 'cypress' | ||||
| export default defineConfig({ | ||||
|   e2e: { | ||||
|     specPattern: 'cypress/e2e/**/*.{cy,spec}.{js,jsx,ts,tsx}', | ||||
|     baseUrl: 'http://localhost:4173' | ||||
|     baseUrl: 'http://localhost:9000' | ||||
|   }, | ||||
|   component: { | ||||
|     specPattern: 'src/**/__tests__/*.{cy,spec}.{js,ts,jsx,tsx}', | ||||
|   | ||||
| @@ -1,8 +0,0 @@ | ||||
| // https://on.cypress.io/api | ||||
|  | ||||
| describe('My First Test', () => { | ||||
|   it('visits the app root url', () => { | ||||
|     cy.visit('/') | ||||
|     cy.contains('h1', 'You did it!') | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										150
									
								
								frontend/cypress/e2e/testLogin.cy.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										150
									
								
								frontend/cypress/e2e/testLogin.cy.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,150 @@ | ||||
| // cypress/e2e/login.cy.js | ||||
|  | ||||
| describe('Login Component', () => { | ||||
|     beforeEach(() => { | ||||
|         // Mock the API response for OIDC providers | ||||
|         cy.intercept('GET', '**/api/v1/config', { | ||||
|             statusCode: 200, | ||||
|             body: { | ||||
|                 data: { | ||||
|                     "app.favicon_url": "http://localhost:9000/favicon.ico", | ||||
|                     "app.lang": "en", | ||||
|                     "app.logo_url": "http://localhost:9000/logo.png", | ||||
|                     "app.site_name": "Libredesk", | ||||
|                     "app.sso_providers": [ | ||||
|                         { | ||||
|                             "client_id": "xx", | ||||
|                             "enabled": true, | ||||
|                             "id": 1, | ||||
|                             "logo_url": "/images/google-logo.png", | ||||
|                             "name": "Google", | ||||
|                             "provider": "Google", | ||||
|                             "provider_url": "https://accounts.google.com", | ||||
|                             "redirect_uri": "http://localhost:9000/api/v1/oidc/1/finish" | ||||
|                         } | ||||
|                     ] | ||||
|                 } | ||||
|             } | ||||
|         }).as('getOIDCProviders') | ||||
|  | ||||
|         // Visit the login page | ||||
|         cy.visit('/') | ||||
|     }) | ||||
|  | ||||
|     it('should display login form', () => { | ||||
|         cy.contains('h3', 'Libredesk').should('be.visible') | ||||
|         cy.contains('p', 'Sign in to your account').should('be.visible') | ||||
|         cy.get('#email').should('be.visible') | ||||
|         cy.get('#password').should('be.visible') | ||||
|         cy.contains('a', 'Forgot password?').should('be.visible') | ||||
|         cy.contains('button', 'Sign in').should('be.visible') | ||||
|     }) | ||||
|  | ||||
|     it('should display OIDC providers when loaded', () => { | ||||
|         cy.wait('@getOIDCProviders') | ||||
|         cy.contains('button', 'Google').should('be.visible') | ||||
|         cy.contains('div', 'Or continue with').should('be.visible') | ||||
|     }) | ||||
|  | ||||
|     it('should show error for invalid login attempt', () => { | ||||
|         // Mock failed login API call | ||||
|         cy.intercept('POST', '**/api/v1/auth/login', { | ||||
|             statusCode: 401, | ||||
|             body: { | ||||
|                 message: 'Invalid credentials' | ||||
|             } | ||||
|         }).as('loginFailure') | ||||
|  | ||||
|         // Enter System username and wrong password | ||||
|         cy.get('#email').type('System') | ||||
|         cy.get('#password').type('WrongPassword') | ||||
|  | ||||
|         // Submit form | ||||
|         cy.contains('button', 'Sign in').click() | ||||
|  | ||||
|         // Wait for API call | ||||
|         cy.wait('@loginFailure') | ||||
|  | ||||
|         // Verify error message appears | ||||
|         cy.contains('Invalid credentials').should('be.visible') | ||||
|     }) | ||||
|  | ||||
|     it('should login successfully with correct credentials', () => { | ||||
|         // Mock successful login API call | ||||
|         cy.intercept('POST', '**/api/v1/auth/login', { | ||||
|             statusCode: 200, | ||||
|             body: { | ||||
|                 data: { | ||||
|                     id: 1, | ||||
|                     email: 'System', | ||||
|                     name: 'System User' | ||||
|                 } | ||||
|             } | ||||
|         }).as('loginSuccess') | ||||
|  | ||||
|         // Enter System username and correct password | ||||
|         cy.get('#email').type('System') | ||||
|         cy.get('#password').type('StrongPass!123') | ||||
|  | ||||
|         // Submit form | ||||
|         cy.contains('button', 'Sign in').click() | ||||
|  | ||||
|         // Wait for API call | ||||
|         cy.wait('@loginSuccess') | ||||
|  | ||||
|         // Verify redirection to inboxes page | ||||
|         cy.url().should('include', '/inboxes/assigned') | ||||
|     }) | ||||
|  | ||||
|     it('should validate email format', () => { | ||||
|         // Enter invalid email and a password | ||||
|         cy.get('#email').type('invalid-email') | ||||
|         cy.get('#password').type('password') | ||||
|  | ||||
|         // Submit form | ||||
|         cy.contains('button', 'Sign in').click() | ||||
|  | ||||
|         // Check for validation error (matching the error message with a trailing period) | ||||
|         cy.contains('Invalid email address').should('be.visible') | ||||
|     }) | ||||
|  | ||||
|     it('should validate empty password', () => { | ||||
|         // Enter email but no password | ||||
|         cy.get('#email').type('valid@example.com') | ||||
|  | ||||
|         // Submit form | ||||
|         cy.contains('button', 'Sign in').click() | ||||
|  | ||||
|         // Check for validation error (matching the error message with a trailing period) | ||||
|         cy.contains('Password cannot be empty').should('be.visible') | ||||
|     }) | ||||
|  | ||||
|     it('should show loading state during login', () => { | ||||
|         // Mock slow API response | ||||
|         cy.intercept('POST', '**/api/v1/auth/login', { | ||||
|             statusCode: 200, | ||||
|             body: { | ||||
|                 data: { | ||||
|                     id: 1, | ||||
|                     email: 'System', | ||||
|                     name: 'System User' | ||||
|                 } | ||||
|             }, | ||||
|             delay: 1000 | ||||
|         }).as('slowLogin') | ||||
|  | ||||
|         // Enter credentials | ||||
|         cy.get('#email').type('System') | ||||
|         cy.get('#password').type('StrongPass!123') | ||||
|  | ||||
|         // Submit form | ||||
|         cy.contains('button', 'Sign in').click() | ||||
|  | ||||
|         // Check if loading state is shown | ||||
|         cy.contains('Logging in...').should('be.visible') | ||||
|         cy.get('.animate-spin').should('be.visible') | ||||
|  | ||||
|         // Wait for API call to finish | ||||
|         cy.wait('@slowLogin') | ||||
|     }) | ||||
| }) | ||||
| @@ -6,8 +6,7 @@ | ||||
|   <meta name="viewport" content="width=device-width, initial-scale=1.0" /> | ||||
|   <link rel="preconnect" href="https://fonts.googleapis.com"> | ||||
|   <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin> | ||||
|   <link | ||||
|     href="https://fonts.googleapis.com/css2?family=Inter:ital,opsz,wght@0,14..32,100..900;1,14..32,100..900&family=Plus+Jakarta+Sans:ital,wght@0,200..800;1,200..800&family=Poppins:ital,wght@0,100;0,200;0,300;0,400;0,500;0,600;0,700;0,800;0,900;1,100;1,200;1,300;1,400;1,500;1,600;1,700;1,800;1,900&display=swap" | ||||
|   <link href="https://fonts.googleapis.com/css2?family=Plus+Jakarta+Sans:ital,wght@0,200..800;1,200..800&display=swap" | ||||
|     rel="stylesheet"> | ||||
| </head> | ||||
|  | ||||
|   | ||||
| @@ -1,13 +1,16 @@ | ||||
| { | ||||
|   "name": "libredesk", | ||||
|   "version": "0.0.0", | ||||
|   "version": "0.6.0-alpha", | ||||
|   "private": true, | ||||
|   "type": "module", | ||||
|   "scripts": { | ||||
|     "dev": "pnpm exec vite", | ||||
|     "build": "vite build", | ||||
|     "preview": "vite preview", | ||||
|     "test": "vitest", | ||||
|     "test:run": "vitest run", | ||||
|     "test:e2e": "start-server-and-test preview http://localhost:4173 'cypress run --e2e'", | ||||
|     "test:e2e:ci": "cypress run --e2e --headless", | ||||
|     "test:e2e:dev": "start-server-and-test 'vite dev --port 4173' http://localhost:4173 'cypress open --e2e'", | ||||
|     "test:unit": "cypress run --component", | ||||
|     "test:unit:dev": "cypress open --component", | ||||
| @@ -15,46 +18,42 @@ | ||||
|     "format": "prettier --write src/" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "@codemirror/lang-html": "^6.4.9", | ||||
|     "@codemirror/theme-one-dark": "^6.1.3", | ||||
|     "@formkit/auto-animate": "^0.8.2", | ||||
|     "@internationalized/date": "^3.5.5", | ||||
|     "@radix-icons/vue": "^1.0.0", | ||||
|     "@tailwindcss/typography": "^0.5.10", | ||||
|     "@tailwindcss/typography": "^0.5.16", | ||||
|     "@tanstack/vue-table": "^8.19.2", | ||||
|     "@tiptap/extension-image": "^2.5.9", | ||||
|     "@tiptap/extension-link": "^2.9.1", | ||||
|     "@tiptap/extension-ordered-list": "^2.4.0", | ||||
|     "@tiptap/extension-link": "^2.11.2", | ||||
|     "@tiptap/extension-placeholder": "^2.4.0", | ||||
|     "@tiptap/extension-table": "^2.11.5", | ||||
|     "@tiptap/extension-table-cell": "^2.11.5", | ||||
|     "@tiptap/extension-table-header": "^2.11.5", | ||||
|     "@tiptap/extension-table-row": "^2.11.5", | ||||
|     "@tiptap/pm": "^2.4.0", | ||||
|     "@tiptap/starter-kit": "^2.4.0", | ||||
|     "@tiptap/suggestion": "^2.4.0", | ||||
|     "@tiptap/vue-3": "^2.4.0", | ||||
|     "@unovis/ts": "^1.4.4", | ||||
|     "@unovis/vue": "^1.4.4", | ||||
|     "@vee-validate/zod": "^4.13.2", | ||||
|     "@vue/reactivity": "^3.4.15", | ||||
|     "@vue/runtime-core": "^3.4.15", | ||||
|     "@vueup/vue-quill": "^1.2.0", | ||||
|     "@vee-validate/zod": "^4.15.0", | ||||
|     "@vueuse/core": "^12.4.0", | ||||
|     "add": "^2.0.6", | ||||
|     "axios": "^1.7.9", | ||||
|     "axios": "^1.8.2", | ||||
|     "class-variance-authority": "^0.7.0", | ||||
|     "clsx": "^2.1.1", | ||||
|     "codeflask": "^1.4.1", | ||||
|     "codemirror": "^6.0.2", | ||||
|     "date-fns": "^3.6.0", | ||||
|     "install": "^0.13.0", | ||||
|     "lucide-vue-next": "^0.378.0", | ||||
|     "mitt": "^3.0.1", | ||||
|     "npm": "^10.4.0", | ||||
|     "npx": "^10.2.2", | ||||
|     "pinia": "^2.1.7", | ||||
|     "qs": "^6.12.1", | ||||
|     "radix-vue": "latest", | ||||
|     "shadcn-vue": "latest", | ||||
|     "radix-vue": "^1.9.17", | ||||
|     "reka-ui": "^2.2.0", | ||||
|     "tailwind-merge": "^2.3.0", | ||||
|     "tailwindcss-animate": "^1.0.7", | ||||
|     "textarea": "^0.3.0", | ||||
|     "vee-validate": "^4.13.2", | ||||
|     "vee-validate": "^4.15.0", | ||||
|     "vue": "^3.4.37", | ||||
|     "vue-dompurify-html": "^5.2.0", | ||||
|     "vue-i18n": "9", | ||||
|     "vue-letter": "^0.2.0", | ||||
|     "vue-picture-cropper": "^0.7.0", | ||||
| @@ -62,13 +61,13 @@ | ||||
|     "vue-sonner": "^1.3.0", | ||||
|     "vue3-emoji-picker": "^1.1.8", | ||||
|     "vuedraggable": "^4.1.0", | ||||
|     "zod": "^3.23.8" | ||||
|     "zod": "^3.24.1" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@rushstack/eslint-patch": "^1.3.3", | ||||
|     "@vitejs/plugin-vue": "^5.0.3", | ||||
|     "@vue/eslint-config-prettier": "^8.0.0", | ||||
|     "autoprefixer": "latest", | ||||
|     "autoprefixer": "^10.4.20", | ||||
|     "cypress": "^13.6.3", | ||||
|     "eslint": "^8.49.0", | ||||
|     "eslint-plugin-cypress": "^2.15.1", | ||||
| @@ -77,8 +76,10 @@ | ||||
|     "prettier": "^3.0.3", | ||||
|     "sass": "^1.70.0", | ||||
|     "start-server-and-test": "^2.0.3", | ||||
|     "tailwindcss": "latest", | ||||
|     "vite": "^5.4.9" | ||||
|     "tailwindcss": "^3.4.17", | ||||
|     "tailwindcss-animate": "^1.0.7", | ||||
|     "vite": "^5.4.19", | ||||
|     "vitest": "^3.2.2" | ||||
|   }, | ||||
|   "packageManager": "pnpm@9.15.3+sha512.1f79bc245a66eb0b07c5d4d83131240774642caaa86ef7d0434ab47c0d16f66b04e21e0c086eb61e62c77efc4d7f7ec071afad3796af64892fae66509173893a" | ||||
| } | ||||
| } | ||||
							
								
								
									
										2459
									
								
								frontend/pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										2459
									
								
								frontend/pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,5 +1,5 @@ | ||||
| <template> | ||||
|   <div class="flex w-full h-screen"> | ||||
|   <div class="flex w-full h-screen text-foreground"> | ||||
|     <!-- Icon sidebar always visible --> | ||||
|     <SidebarProvider style="--sidebar-width: 3rem" class="w-auto z-50"> | ||||
|       <ShadcnSidebar collapsible="none" class="border-r"> | ||||
| @@ -8,25 +8,64 @@ | ||||
|             <SidebarGroupContent> | ||||
|               <SidebarMenu> | ||||
|                 <SidebarMenuItem> | ||||
|                   <SidebarMenuButton asChild :isActive="route.path.startsWith('/inboxes')"> | ||||
|                     <router-link :to="{ name: 'inboxes' }"> | ||||
|                       <Inbox /> | ||||
|                     </router-link> | ||||
|                   </SidebarMenuButton> | ||||
|                   <Tooltip> | ||||
|                     <TooltipTrigger as-child> | ||||
|                       <SidebarMenuButton asChild :isActive="route.path.startsWith('/inboxes')"> | ||||
|                         <router-link :to="{ name: 'inboxes' }"> | ||||
|                           <Inbox /> | ||||
|                         </router-link> | ||||
|                       </SidebarMenuButton> | ||||
|                     </TooltipTrigger> | ||||
|                     <TooltipContent side="right"> | ||||
|                       <p>{{ t('globals.terms.inbox', 2) }}</p> | ||||
|                     </TooltipContent> | ||||
|                   </Tooltip> | ||||
|                 </SidebarMenuItem> | ||||
|                 <SidebarMenuItem v-if="userStore.hasAdminTabPermissions"> | ||||
|                   <SidebarMenuButton asChild :isActive="route.path.startsWith('/admin')"> | ||||
|                     <router-link :to="{ name: 'admin' }"> | ||||
|                       <Shield /> | ||||
|                     </router-link> | ||||
|                   </SidebarMenuButton> | ||||
|                 <SidebarMenuItem v-if="userStore.can('contacts:read_all')"> | ||||
|                   <Tooltip> | ||||
|                     <TooltipTrigger as-child> | ||||
|                       <SidebarMenuButton asChild :isActive="route.path.startsWith('/contacts')"> | ||||
|                         <router-link :to="{ name: 'contacts' }"> | ||||
|                           <BookUser /> | ||||
|                         </router-link> | ||||
|                       </SidebarMenuButton> | ||||
|                     </TooltipTrigger> | ||||
|                     <TooltipContent side="right"> | ||||
|                       <p>{{ t('globals.terms.contact', 2) }}</p> | ||||
|                     </TooltipContent> | ||||
|                   </Tooltip> | ||||
|                 </SidebarMenuItem> | ||||
|                 <SidebarMenuItem v-if="userStore.hasReportTabPermissions"> | ||||
|                   <SidebarMenuButton asChild :isActive="route.path.startsWith('/reports')"> | ||||
|                     <router-link :to="{ name: 'reports' }"> | ||||
|                       <FileLineChart /> | ||||
|                     </router-link> | ||||
|                   </SidebarMenuButton> | ||||
|                   <Tooltip> | ||||
|                     <TooltipTrigger as-child> | ||||
|                       <SidebarMenuButton asChild :isActive="route.path.startsWith('/reports')"> | ||||
|                         <router-link :to="{ name: 'reports' }"> | ||||
|                           <FileLineChart /> | ||||
|                         </router-link> | ||||
|                       </SidebarMenuButton> | ||||
|                     </TooltipTrigger> | ||||
|                     <TooltipContent side="right"> | ||||
|                       <p>{{ t('globals.terms.report', 2) }}</p> | ||||
|                     </TooltipContent> | ||||
|                   </Tooltip> | ||||
|                 </SidebarMenuItem> | ||||
|                 <SidebarMenuItem v-if="userStore.hasAdminTabPermissions"> | ||||
|                   <Tooltip> | ||||
|                     <TooltipTrigger as-child> | ||||
|                       <SidebarMenuButton asChild :isActive="route.path.startsWith('/admin')"> | ||||
|                         <router-link | ||||
|                           :to="{ | ||||
|                             name: userStore.can('general_settings:manage') ? 'general' : 'admin' | ||||
|                           }" | ||||
|                         > | ||||
|                           <Shield /> | ||||
|                         </router-link> | ||||
|                       </SidebarMenuButton> | ||||
|                     </TooltipTrigger> | ||||
|                     <TooltipContent side="right"> | ||||
|                       <p>{{ t('globals.terms.admin') }}</p> | ||||
|                     </TooltipContent> | ||||
|                   </Tooltip> | ||||
|                 </SidebarMenuItem> | ||||
|               </SidebarMenu> | ||||
|             </SidebarGroupContent> | ||||
| @@ -46,9 +85,16 @@ | ||||
|         @create-view="openCreateViewForm = true" | ||||
|         @edit-view="editView" | ||||
|         @delete-view="deleteView" | ||||
|         @create-conversation="() => (openCreateConversationDialog = true)" | ||||
|       > | ||||
|         <div class="flex flex-col h-screen"> | ||||
|           <!-- Show admin banner only in admin routes --> | ||||
|           <AdminBanner v-if="route.path.startsWith('/admin')" /> | ||||
|  | ||||
|           <!-- Common header for all pages --> | ||||
|           <PageHeader /> | ||||
|  | ||||
|           <!-- Main content --> | ||||
|           <RouterView class="flex-grow" /> | ||||
|         </div> | ||||
|         <ViewForm v-model:openDialog="openCreateViewForm" v-model:view="view" /> | ||||
| @@ -58,6 +104,9 @@ | ||||
|  | ||||
|   <!-- Command box --> | ||||
|   <Command /> | ||||
|  | ||||
|   <!-- Create conversation dialog --> | ||||
|   <CreateConversation v-model="openCreateConversationDialog" v-if="openCreateConversationDialog" /> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| @@ -75,13 +124,18 @@ import { useTeamStore } from '@/stores/team' | ||||
| import { useSlaStore } from '@/stores/sla' | ||||
| import { useMacroStore } from '@/stores/macro' | ||||
| import { useTagStore } from '@/stores/tag' | ||||
| import { useCustomAttributeStore } from '@/stores/customAttributes' | ||||
| import { useIdleDetection } from '@/composables/useIdleDetection' | ||||
| import PageHeader from './components/layout/PageHeader.vue' | ||||
| import ViewForm from '@/features/view/ViewForm.vue' | ||||
| import AdminBanner from '@/components/banner/AdminBanner.vue' | ||||
| import api from '@/api' | ||||
| import { toast as sooner } from 'vue-sonner' | ||||
| import Sidebar from '@/components/sidebar/Sidebar.vue' | ||||
| import Command from '@/features/command/CommandBox.vue' | ||||
| import { Inbox, Shield, FileLineChart } from 'lucide-vue-next' | ||||
| import CreateConversation from '@/features/conversation/CreateConversation.vue' | ||||
| import { Inbox, Shield, FileLineChart, BookUser } from 'lucide-vue-next' | ||||
| import { useI18n } from 'vue-i18n' | ||||
| import { useRoute } from 'vue-router' | ||||
| import { | ||||
|   Sidebar as ShadcnSidebar, | ||||
| @@ -94,6 +148,7 @@ import { | ||||
|   SidebarMenuItem, | ||||
|   SidebarProvider | ||||
| } from '@/components/ui/sidebar' | ||||
| import { Tooltip, TooltipContent, TooltipTrigger } from '@/components/ui/tooltip' | ||||
| import SidebarNavUser from '@/components/sidebar/SidebarNavUser.vue' | ||||
|  | ||||
| const route = useRoute() | ||||
| @@ -106,21 +161,28 @@ const inboxStore = useInboxStore() | ||||
| const slaStore = useSlaStore() | ||||
| const macroStore = useMacroStore() | ||||
| const tagStore = useTagStore() | ||||
| const customAttributeStore = useCustomAttributeStore() | ||||
| const userViews = ref([]) | ||||
| const view = ref({}) | ||||
| const openCreateViewForm = ref(false) | ||||
| const openCreateConversationDialog = ref(false) | ||||
| const { t } = useI18n() | ||||
|  | ||||
| initWS() | ||||
| useIdleDetection() | ||||
|  | ||||
| onMounted(() => { | ||||
|   initToaster() | ||||
|   listenViewRefresh() | ||||
|   initStores() | ||||
| }) | ||||
|  | ||||
| // initialize data stores | ||||
| // Initialize data stores | ||||
| const initStores = async () => { | ||||
|   if (!userStore.userID) { | ||||
|     await userStore.getCurrentUser() | ||||
|   } | ||||
|   await Promise.allSettled([ | ||||
|     userStore.getCurrentUser(), | ||||
|     getUserViews(), | ||||
|     conversationStore.fetchStatuses(), | ||||
|     conversationStore.fetchPriorities(), | ||||
| @@ -129,7 +191,8 @@ const initStores = async () => { | ||||
|     inboxStore.fetchInboxes(), | ||||
|     slaStore.fetchSlas(), | ||||
|     macroStore.loadMacros(), | ||||
|     tagStore.fetchTags() | ||||
|     tagStore.fetchTags(), | ||||
|     customAttributeStore.fetchCustomAttributes() | ||||
|   ]) | ||||
| } | ||||
|  | ||||
| @@ -143,12 +206,12 @@ const deleteView = async (view) => { | ||||
|     await api.deleteView(view.id) | ||||
|     emitter.emit(EMITTER_EVENTS.REFRESH_LIST, { model: 'view' }) | ||||
|     emitter.emit(EMITTER_EVENTS.SHOW_TOAST, { | ||||
|       title: 'Success', | ||||
|       description: 'View deleted successfully' | ||||
|       description: t('globals.messages.deletedSuccessfully', { | ||||
|         name: t('globals.terms.view') | ||||
|       }) | ||||
|     }) | ||||
|   } catch (err) { | ||||
|     emitter.emit(EMITTER_EVENTS.SHOW_TOAST, { | ||||
|       title: 'Error', | ||||
|       variant: 'destructive', | ||||
|       description: handleHTTPError(err).message | ||||
|     }) | ||||
| @@ -161,7 +224,6 @@ const getUserViews = async () => { | ||||
|     userViews.value = response.data.data | ||||
|   } catch (err) { | ||||
|     emitter.emit(EMITTER_EVENTS.SHOW_TOAST, { | ||||
|       title: 'Error', | ||||
|       variant: 'destructive', | ||||
|       description: handleHTTPError(err).message | ||||
|     }) | ||||
|   | ||||
| @@ -1,7 +1,27 @@ | ||||
| <template> | ||||
|     <RouterView /> | ||||
|   <RouterView /> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { onMounted } from 'vue' | ||||
| import { RouterView } from 'vue-router' | ||||
| </script> | ||||
| import { EMITTER_EVENTS } from '@/constants/emitterEvents.js' | ||||
| import { useEmitter } from '@/composables/useEmitter' | ||||
| import { toast as sooner } from 'vue-sonner' | ||||
|  | ||||
| const emitter = useEmitter() | ||||
|  | ||||
| onMounted(() => { | ||||
|   initToaster() | ||||
| }) | ||||
|  | ||||
| const initToaster = () => { | ||||
|   emitter.on(EMITTER_EVENTS.SHOW_TOAST, (message) => { | ||||
|     if (message.variant === 'destructive') { | ||||
|       sooner.error(message.description) | ||||
|     } else { | ||||
|       sooner.success(message.description) | ||||
|     } | ||||
|   }) | ||||
| } | ||||
| </script> | ||||
|   | ||||
| @@ -1,9 +1,7 @@ | ||||
| <template> | ||||
|   <TooltipProvider :delay-duration="150"> | ||||
|     <div class="!font-jakarta"> | ||||
|       <Toaster class="pointer-events-auto" position="top-center" richColors /> | ||||
|       <RouterView /> | ||||
|     </div> | ||||
|     <Toaster class="pointer-events-auto" position="top-center" richColors /> | ||||
|     <RouterView /> | ||||
|   </TooltipProvider> | ||||
| </template> | ||||
|  | ||||
|   | ||||
| @@ -7,15 +7,15 @@ const http = axios.create({ | ||||
| }) | ||||
|  | ||||
| function getCSRFToken () { | ||||
|   const name = 'csrf_token='; | ||||
|   const cookies = document.cookie.split(';'); | ||||
|   const name = 'csrf_token=' | ||||
|   const cookies = document.cookie.split(';') | ||||
|   for (let i = 0; i < cookies.length; i++) { | ||||
|     let c = cookies[i].trim(); | ||||
|     let c = cookies[i].trim() | ||||
|     if (c.indexOf(name) === 0) { | ||||
|       return c.substring(name.length, c.length); | ||||
|       return c.substring(name.length, c.length) | ||||
|     } | ||||
|   } | ||||
|   return ''; | ||||
|   return '' | ||||
| } | ||||
|  | ||||
| // Request interceptor. | ||||
| @@ -27,19 +27,40 @@ http.interceptors.request.use((request) => { | ||||
|  | ||||
|   // Set content type for POST/PUT requests if the content type is not set. | ||||
|   if ((request.method === 'post' || request.method === 'put') && !request.headers['Content-Type']) { | ||||
|     request.headers['Content-Type'] = 'application/x-www-form-urlencoded' | ||||
|     request.headers['Content-Type'] = 'application/json' | ||||
|   } | ||||
|    | ||||
|   if (request.headers['Content-Type'] === 'application/x-www-form-urlencoded') { | ||||
|     request.data = qs.stringify(request.data) | ||||
|   } | ||||
|    | ||||
|   return request | ||||
| }) | ||||
|  | ||||
| const getCustomAttributes = (appliesTo) => | ||||
|   http.get('/api/v1/custom-attributes', { | ||||
|     params: { applies_to: appliesTo } | ||||
|   }) | ||||
| const createCustomAttribute = (data) => | ||||
|   http.post('/api/v1/custom-attributes', data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const getCustomAttribute = (id) => http.get(`/api/v1/custom-attributes/${id}`) | ||||
| const updateCustomAttribute = (id, data) => | ||||
|   http.put(`/api/v1/custom-attributes/${id}`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const deleteCustomAttribute = (id) => http.delete(`/api/v1/custom-attributes/${id}`) | ||||
| const searchConversations = (params) => http.get('/api/v1/conversations/search', { params }) | ||||
| const searchMessages = (params) => http.get('/api/v1/messages/search', { params }) | ||||
| const resetPassword = (data) => http.post('/api/v1/users/reset-password', data) | ||||
| const setPassword = (data) => http.post('/api/v1/users/set-password', data) | ||||
| const deleteUser = (id) => http.delete(`/api/v1/users/${id}`) | ||||
| const searchContacts = (params) => http.get('/api/v1/contacts/search', { params }) | ||||
| const getEmailNotificationSettings = () => http.get('/api/v1/settings/notifications/email') | ||||
| const updateEmailNotificationSettings = (data) => http.put('/api/v1/settings/notifications/email', data) | ||||
| const updateEmailNotificationSettings = (data) => | ||||
|   http.put('/api/v1/settings/notifications/email', data) | ||||
| const getPriorities = () => http.get('/api/v1/priorities') | ||||
| const getStatuses = () => http.get('/api/v1/statuses') | ||||
| const createStatus = (data) => http.post('/api/v1/statuses', data) | ||||
| @@ -66,11 +87,12 @@ const updateTemplate = (id, data) => | ||||
|  | ||||
| const getAllBusinessHours = () => http.get('/api/v1/business-hours') | ||||
| const getBusinessHours = (id) => http.get(`/api/v1/business-hours/${id}`) | ||||
| const createBusinessHours = (data) => http.post('/api/v1/business-hours', data, { | ||||
|   headers: { | ||||
|     'Content-Type': 'application/json' | ||||
|   } | ||||
| }) | ||||
| const createBusinessHours = (data) => | ||||
|   http.post('/api/v1/business-hours', data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const updateBusinessHours = (id, data) => | ||||
|   http.put(`/api/v1/business-hours/${id}`, data, { | ||||
|     headers: { | ||||
| @@ -81,8 +103,18 @@ const deleteBusinessHours = (id) => http.delete(`/api/v1/business-hours/${id}`) | ||||
|  | ||||
| const getAllSLAs = () => http.get('/api/v1/sla') | ||||
| const getSLA = (id) => http.get(`/api/v1/sla/${id}`) | ||||
| const createSLA = (data) => http.post('/api/v1/sla', data) | ||||
| const updateSLA = (id, data) => http.put(`/api/v1/sla/${id}`, data) | ||||
| const createSLA = (data) => | ||||
|   http.post('/api/v1/sla', data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const updateSLA = (id, data) => | ||||
|   http.put(`/api/v1/sla/${id}`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const deleteSLA = (id) => http.delete(`/api/v1/sla/${id}`) | ||||
| const createOIDC = (data) => | ||||
|   http.post('/api/v1/oidc', data, { | ||||
| @@ -90,8 +122,7 @@ const createOIDC = (data) => | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const testOIDC = (data) => http.post('/api/v1/oidc/test', data) | ||||
| const getAllEnabledOIDC = () => http.get('/api/v1/oidc/enabled') | ||||
| const getConfig = () => http.get('/api/v1/config') | ||||
| const getAllOIDC = () => http.get('/api/v1/oidc') | ||||
| const getOIDC = (id) => http.get(`/api/v1/oidc/${id}`) | ||||
| const updateOIDC = (id, data) => | ||||
| @@ -108,33 +139,42 @@ const updateSettings = (key, data) => | ||||
|     } | ||||
|   }) | ||||
| const getSettings = (key) => http.get(`/api/v1/settings/${key}`) | ||||
| const login = (data) => http.post(`/api/v1/login`, data) | ||||
| const login = (data) => http.post(`/api/v1/auth/login`, data, { | ||||
|   headers: { | ||||
|     'Content-Type': 'application/json' | ||||
|   } | ||||
| }) | ||||
| const getAutomationRules = (type) => | ||||
|   http.get(`/api/v1/automation/rules`, { | ||||
|   http.get(`/api/v1/automations/rules`, { | ||||
|     params: { type: type } | ||||
|   }) | ||||
| const toggleAutomationRule = (id) => http.put(`/api/v1/automation/rules/${id}/toggle`) | ||||
| const getAutomationRule = (id) => http.get(`/api/v1/automation/rules/${id}`) | ||||
| const toggleAutomationRule = (id) => http.put(`/api/v1/automations/rules/${id}/toggle`) | ||||
| const getAutomationRule = (id) => http.get(`/api/v1/automations/rules/${id}`) | ||||
| const updateAutomationRule = (id, data) => | ||||
|   http.put(`/api/v1/automation/rules/${id}`, data, { | ||||
|   http.put(`/api/v1/automations/rules/${id}`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const createAutomationRule = (data) => | ||||
|   http.post(`/api/v1/automation/rules`, data, { | ||||
|   http.post(`/api/v1/automations/rules`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const deleteAutomationRule = (id) => http.delete(`/api/v1/automation/rules/${id}`) | ||||
| const deleteAutomationRule = (id) => http.delete(`/api/v1/automations/rules/${id}`) | ||||
| const updateAutomationRuleWeights = (data) => | ||||
|   http.put(`/api/v1/automation/rules/weights`, data, { | ||||
|   http.put(`/api/v1/automations/rules/weights`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const updateAutomationRulesExecutionMode = (data) => | ||||
|   http.put(`/api/v1/automations/rules/execution-mode`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const updateAutomationRulesExecutionMode = (data) => http.put(`/api/v1/automation/rules/execution-mode`, data) | ||||
| const getRoles = () => http.get('/api/v1/roles') | ||||
| const getRole = (id) => http.get(`/api/v1/roles/${id}`) | ||||
| const createRole = (data) => | ||||
| @@ -150,35 +190,124 @@ const updateRole = (id, data) => | ||||
|     } | ||||
|   }) | ||||
| const deleteRole = (id) => http.delete(`/api/v1/roles/${id}`) | ||||
| const getUser = (id) => http.get(`/api/v1/users/${id}`) | ||||
| const getTeam = (id) => http.get(`/api/v1/teams/${id}`) | ||||
| const getTeams = () => http.get('/api/v1/teams') | ||||
| const updateTeam = (id, data) => http.put(`/api/v1/teams/${id}`, data) | ||||
| const createTeam = (data) => http.post('/api/v1/teams', data) | ||||
| const getTeamsCompact = () => http.get('/api/v1/teams/compact') | ||||
| const deleteTeam = (id) => http.delete(`/api/v1/teams/${id}`) | ||||
|  | ||||
| const getUsers = () => http.get('/api/v1/users') | ||||
| const getUsersCompact = () => http.get('/api/v1/users/compact') | ||||
| const updateCurrentUser = (data) => | ||||
|   http.put('/api/v1/users/me', data, { | ||||
| const getContacts = (params) => http.get('/api/v1/contacts', { params }) | ||||
| const getContact = (id) => http.get(`/api/v1/contacts/${id}`) | ||||
| const updateContact = (id, data) => | ||||
|   http.put(`/api/v1/contacts/${id}`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'multipart/form-data' | ||||
|     } | ||||
|   }) | ||||
| const deleteUserAvatar = () => http.delete('/api/v1/users/me/avatar') | ||||
| const getCurrentUser = () => http.get('/api/v1/users/me') | ||||
| const getCurrentUserTeams = () => http.get('/api/v1/users/me/teams') | ||||
| const blockContact = (id, data) => http.put(`/api/v1/contacts/${id}/block`, data, { | ||||
|   headers: { | ||||
|     'Content-Type': 'application/json' | ||||
|   } | ||||
| }) | ||||
| const getTeam = (id) => http.get(`/api/v1/teams/${id}`) | ||||
| const getTeams = () => http.get('/api/v1/teams') | ||||
| const updateTeam = (id, data) => http.put(`/api/v1/teams/${id}`, data, { | ||||
|   headers: { | ||||
|     'Content-Type': 'application/json' | ||||
|   } | ||||
| }) | ||||
| const createTeam = (data) => http.post('/api/v1/teams', data, { | ||||
|   headers: { | ||||
|     'Content-Type': 'application/json' | ||||
|   } | ||||
| }) | ||||
| const getTeamsCompact = () => http.get('/api/v1/teams/compact') | ||||
| const deleteTeam = (id) => http.delete(`/api/v1/teams/${id}`) | ||||
| const updateUser = (id, data) => | ||||
|   http.put(`/api/v1/agents/${id}`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const getUsers = () => http.get('/api/v1/agents') | ||||
| const getUsersCompact = () => http.get('/api/v1/agents/compact') | ||||
| const updateCurrentUser = (data) => | ||||
|   http.put('/api/v1/agents/me', data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'multipart/form-data' | ||||
|     } | ||||
|   }) | ||||
| const getUser = (id) => http.get(`/api/v1/agents/${id}`) | ||||
| const deleteUserAvatar = () => http.delete('/api/v1/agents/me/avatar') | ||||
| const getCurrentUser = () => http.get('/api/v1/agents/me') | ||||
| const getCurrentUserTeams = () => http.get('/api/v1/agents/me/teams') | ||||
| const updateCurrentUserAvailability = (data) => http.put('/api/v1/agents/me/availability', data, { | ||||
|   headers: { | ||||
|     'Content-Type': 'application/json' | ||||
|   } | ||||
| }) | ||||
| const resetPassword = (data) => http.post('/api/v1/agents/reset-password', data, { | ||||
|   headers: { | ||||
|     'Content-Type': 'application/json' | ||||
|   } | ||||
| }) | ||||
| const setPassword = (data) => http.post('/api/v1/agents/set-password', data, { | ||||
|   headers: { | ||||
|     'Content-Type': 'application/json' | ||||
|   } | ||||
| }) | ||||
| const deleteUser = (id) => http.delete(`/api/v1/agents/${id}`) | ||||
| const createUser = (data) => | ||||
|   http.post('/api/v1/agents', data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const getTags = () => http.get('/api/v1/tags') | ||||
| const upsertTags = (uuid, data) => http.post(`/api/v1/conversations/${uuid}/tags`, data) | ||||
| const updateAssignee = (uuid, assignee_type, data) => http.put(`/api/v1/conversations/${uuid}/assignee/${assignee_type}`, data) | ||||
| const removeAssignee = (uuid, assignee_type) => http.put(`/api/v1/conversations/${uuid}/assignee/${assignee_type}/remove`) | ||||
| const updateConversationStatus = (uuid, data) => http.put(`/api/v1/conversations/${uuid}/status`, data) | ||||
| const updateConversationPriority = (uuid, data) => http.put(`/api/v1/conversations/${uuid}/priority`, data) | ||||
| const upsertTags = (uuid, data) => http.post(`/api/v1/conversations/${uuid}/tags`, data, { | ||||
|   headers: { | ||||
|     'Content-Type': 'application/json' | ||||
|   } | ||||
| }) | ||||
| const updateAssignee = (uuid, assignee_type, data) => | ||||
|   http.put(`/api/v1/conversations/${uuid}/assignee/${assignee_type}`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const removeAssignee = (uuid, assignee_type) => | ||||
|   http.put(`/api/v1/conversations/${uuid}/assignee/${assignee_type}/remove`) | ||||
| const updateContactCustomAttribute = (uuid, data) => | ||||
|   http.put(`/api/v1/conversations/${uuid}/contacts/custom-attributes`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const updateConversationCustomAttribute = (uuid, data) => | ||||
|   http.put(`/api/v1/conversations/${uuid}/custom-attributes`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const createConversation = (data) => | ||||
|   http.post('/api/v1/conversations', data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const updateConversationStatus = (uuid, data) => | ||||
|   http.put(`/api/v1/conversations/${uuid}/status`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const updateConversationPriority = (uuid, data) => | ||||
|   http.put(`/api/v1/conversations/${uuid}/priority`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const updateAssigneeLastSeen = (uuid) => http.put(`/api/v1/conversations/${uuid}/last-seen`) | ||||
| const getConversationMessage = (cuuid, uuid) => http.get(`/api/v1/conversations/${cuuid}/messages/${uuid}`) | ||||
| const retryMessage = (cuuid, uuid) => http.put(`/api/v1/conversations/${cuuid}/messages/${uuid}/retry`) | ||||
| const getConversationMessages = (uuid, params) => http.get(`/api/v1/conversations/${uuid}/messages`, { params }) | ||||
| const getConversationMessage = (cuuid, uuid) => | ||||
|   http.get(`/api/v1/conversations/${cuuid}/messages/${uuid}`) | ||||
| const retryMessage = (cuuid, uuid) => | ||||
|   http.put(`/api/v1/conversations/${cuuid}/messages/${uuid}/retry`) | ||||
| const getConversationMessages = (uuid, params) => | ||||
|   http.get(`/api/v1/conversations/${uuid}/messages`, { params }) | ||||
| const sendMessage = (uuid, data) => | ||||
|   http.post(`/api/v1/conversations/${uuid}/messages`, data, { | ||||
|     headers: { | ||||
| @@ -189,28 +318,33 @@ const getConversation = (uuid) => http.get(`/api/v1/conversations/${uuid}`) | ||||
| const getConversationParticipants = (uuid) => http.get(`/api/v1/conversations/${uuid}/participants`) | ||||
| const getAllMacros = () => http.get('/api/v1/macros') | ||||
| const getMacro = (id) => http.get(`/api/v1/macros/${id}`) | ||||
| const createMacro = (data) => http.post('/api/v1/macros', data, { | ||||
|   headers: { | ||||
|     'Content-Type': 'application/json' | ||||
|   } | ||||
| }) | ||||
| const updateMacro = (id, data) => http.put(`/api/v1/macros/${id}`, data, { | ||||
|   headers: { | ||||
|     'Content-Type': 'application/json' | ||||
|   } | ||||
| }) | ||||
| const createMacro = (data) => | ||||
|   http.post('/api/v1/macros', data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const updateMacro = (id, data) => | ||||
|   http.put(`/api/v1/macros/${id}`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const deleteMacro = (id) => http.delete(`/api/v1/macros/${id}`) | ||||
| const applyMacro = (uuid, id, data) => http.post(`/api/v1/conversations/${uuid}/macros/${id}/apply`, data, { | ||||
|   headers: { | ||||
|     'Content-Type': 'application/json' | ||||
|   } | ||||
| }) | ||||
| const applyMacro = (uuid, id, data) => | ||||
|   http.post(`/api/v1/conversations/${uuid}/macros/${id}/apply`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const getTeamUnassignedConversations = (teamID, params) => | ||||
|   http.get(`/api/v1/teams/${teamID}/conversations/unassigned`, { params }) | ||||
| const getAssignedConversations = (params) => http.get('/api/v1/conversations/assigned', { params }) | ||||
| const getUnassignedConversations = (params) => http.get('/api/v1/conversations/unassigned', { params }) | ||||
| const getUnassignedConversations = (params) => | ||||
|   http.get('/api/v1/conversations/unassigned', { params }) | ||||
| const getAllConversations = (params) => http.get('/api/v1/conversations/all', { params }) | ||||
| const getViewConversations = (id, params) => http.get(`/api/v1/views/${id}/conversations`, { params }) | ||||
| const getViewConversations = (id, params) => | ||||
|   http.get(`/api/v1/views/${id}/conversations`, { params }) | ||||
| const uploadMedia = (data) => | ||||
|   http.post('/api/v1/media', data, { | ||||
|     headers: { | ||||
| @@ -218,20 +352,9 @@ const uploadMedia = (data) => | ||||
|     } | ||||
|   }) | ||||
| const getOverviewCounts = () => http.get('/api/v1/reports/overview/counts') | ||||
| const getOverviewCharts = () => http.get('/api/v1/reports/overview/charts') | ||||
| const getOverviewCharts = (params) => http.get('/api/v1/reports/overview/charts', { params }) | ||||
| const getOverviewSLA = (params) => http.get('/api/v1/reports/overview/sla', { params }) | ||||
| const getLanguage = (lang) => http.get(`/api/v1/lang/${lang}`) | ||||
| const createUser = (data) => | ||||
|   http.post('/api/v1/users', data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const updateUser = (id, data) => | ||||
|   http.put(`/api/v1/users/${id}`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const createInbox = (data) => | ||||
|   http.post('/api/v1/inboxes', data, { | ||||
|     headers: { | ||||
| @@ -263,7 +386,50 @@ const updateView = (id, data) => | ||||
|   }) | ||||
| const deleteView = (id) => http.delete(`/api/v1/views/me/${id}`) | ||||
| const getAiPrompts = () => http.get('/api/v1/ai/prompts') | ||||
| const aiCompletion = (data) => http.post('/api/v1/ai/completion', data) | ||||
| const aiCompletion = (data) => http.post('/api/v1/ai/completion', data, { | ||||
|   headers: { | ||||
|     'Content-Type': 'application/json' | ||||
|   } | ||||
| }) | ||||
| const updateAIProvider = (data) => http.put('/api/v1/ai/provider', data, { | ||||
|   headers: { | ||||
|     'Content-Type': 'application/json' | ||||
|   } | ||||
| }) | ||||
| const getContactNotes = (id) => http.get(`/api/v1/contacts/${id}/notes`) | ||||
| const createContactNote = (id, data) => http.post(`/api/v1/contacts/${id}/notes`, data, { | ||||
|   headers: { | ||||
|     'Content-Type': 'application/json' | ||||
|   } | ||||
| }) | ||||
| const deleteContactNote = (id, noteId) => http.delete(`/api/v1/contacts/${id}/notes/${noteId}`) | ||||
| const getActivityLogs = (params) => http.get('/api/v1/activity-logs', { params }) | ||||
| const getWebhooks = () => http.get('/api/v1/webhooks') | ||||
| const getWebhook = (id) => http.get(`/api/v1/webhooks/${id}`) | ||||
| const createWebhook = (data) => | ||||
|   http.post('/api/v1/webhooks', data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const updateWebhook = (id, data) => | ||||
|   http.put(`/api/v1/webhooks/${id}`, data, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
| const deleteWebhook = (id) => http.delete(`/api/v1/webhooks/${id}`) | ||||
| const toggleWebhook = (id) => http.put(`/api/v1/webhooks/${id}/toggle`) | ||||
| const testWebhook = (id) => http.post(`/api/v1/webhooks/${id}/test`) | ||||
|  | ||||
| const generateAPIKey = (id) =>  | ||||
|   http.post(`/api/v1/agents/${id}/api-key`, {}, { | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json' | ||||
|     } | ||||
|   }) | ||||
|  | ||||
| const revokeAPIKey = (id) => http.delete(`/api/v1/agents/${id}/api-key`) | ||||
|  | ||||
| export default { | ||||
|   login, | ||||
| @@ -304,6 +470,7 @@ export default { | ||||
|   getViewConversations, | ||||
|   getOverviewCharts, | ||||
|   getOverviewCounts, | ||||
|   getOverviewSLA, | ||||
|   getConversationParticipants, | ||||
|   getConversationMessage, | ||||
|   getConversationMessages, | ||||
| @@ -320,15 +487,20 @@ export default { | ||||
|   updateConversationStatus, | ||||
|   updateConversationPriority, | ||||
|   upsertTags, | ||||
|   updateConversationCustomAttribute, | ||||
|   updateContactCustomAttribute, | ||||
|   uploadMedia, | ||||
|   updateAssigneeLastSeen, | ||||
|   updateUser, | ||||
|   updateCurrentUserAvailability, | ||||
|   updateAutomationRule, | ||||
|   updateAutomationRuleWeights, | ||||
|   updateAutomationRulesExecutionMode, | ||||
|   updateAIProvider, | ||||
|   createAutomationRule, | ||||
|   toggleAutomationRule, | ||||
|   deleteAutomationRule, | ||||
|   createConversation, | ||||
|   sendMessage, | ||||
|   retryMessage, | ||||
|   createUser, | ||||
| @@ -342,10 +514,9 @@ export default { | ||||
|   updateSettings, | ||||
|   createOIDC, | ||||
|   getAllOIDC, | ||||
|   getAllEnabledOIDC, | ||||
|   getConfig, | ||||
|   getOIDC, | ||||
|   updateOIDC, | ||||
|   testOIDC, | ||||
|   deleteOIDC, | ||||
|   getTemplate, | ||||
|   getTemplates, | ||||
| @@ -373,5 +544,28 @@ export default { | ||||
|   aiCompletion, | ||||
|   searchConversations, | ||||
|   searchMessages, | ||||
|   searchContacts, | ||||
|   removeAssignee, | ||||
|   getContacts, | ||||
|   getContact, | ||||
|   updateContact, | ||||
|   blockContact, | ||||
|   getCustomAttributes, | ||||
|   createCustomAttribute, | ||||
|   updateCustomAttribute, | ||||
|   deleteCustomAttribute, | ||||
|   getCustomAttribute, | ||||
|   getContactNotes, | ||||
|   createContactNote, | ||||
|   deleteContactNote, | ||||
|   getActivityLogs, | ||||
|   getWebhooks, | ||||
|   getWebhook, | ||||
|   createWebhook, | ||||
|   updateWebhook, | ||||
|   deleteWebhook, | ||||
|   toggleWebhook, | ||||
|   testWebhook, | ||||
|   generateAPIKey, | ||||
|   revokeAPIKey | ||||
| } | ||||
|   | ||||
| @@ -13,15 +13,95 @@ | ||||
|     min-height: 100%; | ||||
|     overflow: hidden; | ||||
|     margin: 0; | ||||
|     @apply bg-background text-foreground; | ||||
|   } | ||||
|  | ||||
|     @media (max-width: 768px) { | ||||
|   @media (max-width: 768px) { | ||||
|     html, | ||||
|     body { | ||||
|       overflow-x: auto; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| // Theme. | ||||
| @layer base { | ||||
|   * { | ||||
|     @apply border-border; | ||||
|   } | ||||
|  | ||||
|   .native-html { | ||||
|     p { | ||||
|       margin-bottom: 0.5rem; | ||||
|     } | ||||
|  | ||||
|     ul { | ||||
|       list-style-type: disc; | ||||
|       margin-left: 1.5rem; | ||||
|       margin-top: 0.5rem; | ||||
|       margin-bottom: 0.5rem; | ||||
|     } | ||||
|  | ||||
|     ol { | ||||
|       list-style-type: decimal; | ||||
|       margin-left: 1.5rem; | ||||
|       margin-top: 0.5rem; | ||||
|       margin-bottom: 0.5rem; | ||||
|     } | ||||
|  | ||||
|     li { | ||||
|       padding-left: 0.25rem; | ||||
|     } | ||||
|  | ||||
|     h1, | ||||
|     h2, | ||||
|     h3, | ||||
|     h4, | ||||
|     h5, | ||||
|     h6 { | ||||
|       font-size: 1.25rem; | ||||
|       font-weight: 700; | ||||
|     } | ||||
|  | ||||
|     a { | ||||
|       color: #0066cc; | ||||
|       cursor: pointer; | ||||
|  | ||||
|       &:hover { | ||||
|         color: #003d7a; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   :root { | ||||
|     --sidebar-background: 0 0% 100%; | ||||
|     --sidebar-foreground: 240 5.9% 10%; | ||||
|     --sidebar-primary: 240 5.9% 10%; | ||||
|     --sidebar-primary-foreground: 0 0% 98%; | ||||
|     --sidebar-accent: 240 4.8% 95.9%; | ||||
|     --sidebar-accent-foreground: 240 5.9% 10%; | ||||
|     --sidebar-border: 220 13% 91%; | ||||
|     --sidebar-ring: 217.2 91.2% 59.8%; | ||||
|   } | ||||
|   .dark { | ||||
|     --sidebar-background: 240 5.9% 10%; | ||||
|     --sidebar-foreground: 240 4.8% 95.9%; | ||||
|     --sidebar-primary: 224.3 76.3% 48%; | ||||
|     --sidebar-primary-foreground: 0 0% 100%; | ||||
|     --sidebar-accent: 240 3.7% 15.9%; | ||||
|     --sidebar-accent-foreground: 240 4.8% 95.9%; | ||||
|     --sidebar-border: 240 3.7% 15.9%; | ||||
|     --sidebar-ring: 217.2 91.2% 59.8%; | ||||
|   } | ||||
|  | ||||
|   :root { | ||||
|     --vis-tooltip-background-color: none !important; | ||||
|     --vis-tooltip-border-color: none !important; | ||||
|     --vis-tooltip-text-color: none !important; | ||||
|     --vis-tooltip-shadow-color: none !important; | ||||
|     --vis-tooltip-backdrop-filter: none !important; | ||||
|     --vis-tooltip-padding: none !important; | ||||
|     --vis-primary-color: var(--primary); | ||||
|     --vis-secondary-color: 160 81% 40%; | ||||
|     --vis-text-color: var(--muted-foreground); | ||||
|   } | ||||
|  | ||||
|   :root { | ||||
|     --background: 0 0% 100%; | ||||
|     --foreground: 240 10% 3.9%; | ||||
| @@ -50,11 +130,11 @@ | ||||
|     --border: 240 5.9% 90%; | ||||
|     --input: 240 5.9% 90%; | ||||
|     --ring: 240 5.9% 10%; | ||||
|     --radius: 0.75rem; | ||||
|     --radius: 0.5rem; | ||||
|   } | ||||
|  | ||||
|   .dark { | ||||
|     --background: 240 10% 3.9%; | ||||
|     --background: 240 5.9% 10%; | ||||
|     --foreground: 0 0% 98%; | ||||
|  | ||||
|     --card: 240 10% 3.9%; | ||||
| @@ -84,72 +164,8 @@ | ||||
|   } | ||||
| } | ||||
|  | ||||
| @layer base { | ||||
|   :root { | ||||
|     --vis-tooltip-background-color: none !important; | ||||
|     --vis-tooltip-border-color: none !important; | ||||
|     --vis-tooltip-text-color: none !important; | ||||
|     --vis-tooltip-shadow-color: none !important; | ||||
|     --vis-tooltip-backdrop-filter: none !important; | ||||
|     --vis-tooltip-padding: none !important; | ||||
|     --vis-primary-color: var(--primary); | ||||
|     --vis-secondary-color: 160 81% 40%; | ||||
|     --vis-text-color: var(--muted-foreground); | ||||
|   } | ||||
| } | ||||
|  | ||||
| // Shake animation | ||||
| @keyframes shake { | ||||
|   0% { | ||||
|     transform: translateX(0); | ||||
|   } | ||||
|   15% { | ||||
|     transform: translateX(-5px); | ||||
|   } | ||||
|   25% { | ||||
|     transform: translateX(5px); | ||||
|   } | ||||
|   35% { | ||||
|     transform: translateX(-5px); | ||||
|   } | ||||
|   45% { | ||||
|     transform: translateX(5px); | ||||
|   } | ||||
|   55% { | ||||
|     transform: translateX(-5px); | ||||
|   } | ||||
|   65% { | ||||
|     transform: translateX(5px); | ||||
|   } | ||||
|   75% { | ||||
|     transform: translateX(-5px); | ||||
|   } | ||||
|   85% { | ||||
|     transform: translateX(5px); | ||||
|   } | ||||
|   95% { | ||||
|     transform: translateX(-5px); | ||||
|   } | ||||
|   100% { | ||||
|     transform: translateX(0); | ||||
|   } | ||||
| } | ||||
|  | ||||
| .animate-shake { | ||||
|   animation: shake 0.5s infinite; | ||||
| } | ||||
|  | ||||
| .message-bubble { | ||||
|   @apply flex | ||||
|   flex-col | ||||
|   px-4 | ||||
|   pt-2 | ||||
|   pb-3 | ||||
|   min-w-[30%] max-w-[70%] | ||||
|   border | ||||
|   overflow-x-auto | ||||
|   rounded-xl; | ||||
|   box-shadow: 0 1px 2px rgba(0, 0, 0, 0.05); | ||||
|   @apply flex flex-col px-4 pt-2 pb-3 w-fit min-w-[30%] max-w-full border overflow-x-auto rounded shadow-sm; | ||||
|   table { | ||||
|     width: 100% !important; | ||||
|     table-layout: fixed !important; | ||||
| @@ -165,7 +181,7 @@ | ||||
| } | ||||
|  | ||||
| .box { | ||||
|   @apply border shadow rounded-lg; | ||||
|   @apply border shadow rounded; | ||||
| } | ||||
|  | ||||
| // Scrollbar start | ||||
| @@ -191,85 +207,6 @@ | ||||
| } | ||||
| // End Scrollbar | ||||
|  | ||||
| .code-editor { | ||||
|   @apply rounded-md border shadow h-[65vh] min-h-[250px] w-full relative; | ||||
| } | ||||
|  | ||||
| .ql-container { | ||||
|   margin: 0 !important; | ||||
| } | ||||
|  | ||||
| .ql-container .ql-editor { | ||||
|   height: 300px !important; | ||||
|   border-radius: var(--radius) !important; | ||||
|   @apply rounded-lg rounded-t-none; | ||||
| } | ||||
|  | ||||
| .ql-toolbar { | ||||
|   @apply rounded-t-lg; | ||||
| } | ||||
|  | ||||
| .blinking-dot { | ||||
|   display: inline-block; | ||||
|   width: 8px; | ||||
|   height: 8px; | ||||
|   background-color: red; | ||||
|   border-radius: 50%; | ||||
|   animation: blink 2s infinite; | ||||
| } | ||||
|  | ||||
| @keyframes blink { | ||||
|   0%, | ||||
|   100% { | ||||
|     opacity: 1; | ||||
|   } | ||||
|   50% { | ||||
|     opacity: 0; | ||||
|   } | ||||
| } | ||||
|  | ||||
| // Sidebar start | ||||
| @layer base { | ||||
|   :root { | ||||
|     --sidebar-background: 0 0% 96%; | ||||
|     --sidebar-foreground: 240 5.3% 26.1%; | ||||
|     --sidebar-primary: 240 5.9% 10%; | ||||
|     --sidebar-primary-foreground: 0 0% 98%; | ||||
|     --sidebar-accent: 240 4.8% 95.9%; | ||||
|     --sidebar-accent-foreground: 240 5.9% 10%; | ||||
|     --sidebar-border: 220 13% 91%; | ||||
|     --sidebar-ring: 217.2 91.2% 59.8%; | ||||
|   } | ||||
|  | ||||
|   .dark { | ||||
|     --sidebar-background: 240 5.9% 10%; | ||||
|     --sidebar-foreground: 240 4.8% 95.9%; | ||||
|     --sidebar-primary: 224.3 76.3% 48%; | ||||
|     --sidebar-primary-foreground: 0 0% 100%; | ||||
|     --sidebar-accent: 240 3.7% 15.9%; | ||||
|     --sidebar-accent-foreground: 240 4.8% 95.9%; | ||||
|     --sidebar-border: 240 3.7% 15.9%; | ||||
|     --sidebar-ring: 217.2 91.2% 59.8%; | ||||
|   } | ||||
| } | ||||
| a[data-active='true'] { | ||||
|   background-color: hsl(var(--sidebar-background)) !important; | ||||
|   color: hsl(var(--sidebar-accent-foreground)) !important; | ||||
|   font-weight: 500; | ||||
|   transition: | ||||
|     background-color 0.2s, | ||||
|     color 0.2s; | ||||
| } | ||||
| a[data-active='false']:hover { | ||||
|   background-color: hsl(var(--sidebar-accent)) !important; | ||||
|   color: hsl(var(--sidebar-accent-foreground)) !important; | ||||
|   font-weight: 500; | ||||
|   transition: | ||||
|     background-color 0.2s, | ||||
|     color 0.2s; | ||||
| } | ||||
| // Sidebar end | ||||
|  | ||||
| .show-quoted-text { | ||||
|   blockquote { | ||||
|     @apply block; | ||||
| @@ -282,33 +219,13 @@ a[data-active='false']:hover { | ||||
|   } | ||||
| } | ||||
|  | ||||
| .dot-loader { | ||||
|   display: inline-flex; | ||||
|   align-items: center; | ||||
| [data-radix-popper-content-wrapper] { | ||||
|   z-index: 9999 !important; | ||||
| } | ||||
|  | ||||
| .dot { | ||||
|   width: 4px; | ||||
|   height: 4px; | ||||
|   border-radius: 50%; | ||||
|   background-color: currentColor; | ||||
|   margin: 0 2px; | ||||
|   animation: dot-flashing 1s infinite linear alternate; | ||||
| } | ||||
|  | ||||
| .dot:nth-child(2) { | ||||
|   animation-delay: 0.2s; | ||||
| } | ||||
|  | ||||
| .dot:nth-child(3) { | ||||
|   animation-delay: 0.4s; | ||||
| } | ||||
|  | ||||
| @keyframes dot-flashing { | ||||
|   0% { | ||||
|     opacity: 0.2; | ||||
|   } | ||||
|   100% { | ||||
|     opacity: 1; | ||||
| // Components | ||||
| @layer components { | ||||
|   .link-style { | ||||
|     @apply text-blue-500 hover:underline; | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										63
									
								
								frontend/src/components/banner/AdminBanner.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								frontend/src/components/banner/AdminBanner.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | ||||
| <template> | ||||
|   <div class="border-b"> | ||||
|     <!-- Update notification --> | ||||
|     <div | ||||
|       v-if="appSettingsStore.settings['app.update']?.update?.is_new" | ||||
|       class="px-4 py-2.5 border-b border-border/50 last:border-b-0" | ||||
|     > | ||||
|       <div class="flex items-center gap-3"> | ||||
|         <div class="flex-shrink-0"> | ||||
|           <Download class="w-5 h-5 text-primary" /> | ||||
|         </div> | ||||
|         <div class="min-w-0 flex-1"> | ||||
|           <div class="flex items-center gap-2 text-sm text-foreground"> | ||||
|             <span>{{ $t('update.newUpdateAvailable') }}</span> | ||||
|             <a | ||||
|               :href="appSettingsStore.settings['app.update'].update.url" | ||||
|               target="_blank" | ||||
|               rel="nofollow noreferrer" | ||||
|               class="font-semibold text-primary hover:text-primary/80 underline transition-colors focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-1" | ||||
|             > | ||||
|               {{ appSettingsStore.settings['app.update'].update.release_version }} | ||||
|             </a> | ||||
|             <span class="text-muted-foreground">•</span> | ||||
|             <span class="text-muted-foreground"> | ||||
|               {{ appSettingsStore.settings['app.update'].update.release_date }} | ||||
|             </span> | ||||
|           </div> | ||||
|  | ||||
|           <!-- Update description --> | ||||
|           <div | ||||
|             v-if="appSettingsStore.settings['app.update'].update.description" | ||||
|             class="mt-2 text-xs text-muted-foreground" | ||||
|           > | ||||
|             {{ appSettingsStore.settings['app.update'].update.description }} | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|  | ||||
|     <!-- Restart required notification --> | ||||
|     <div | ||||
|       v-if="appSettingsStore.settings['app.restart_required']" | ||||
|       class="px-4 py-2.5 border-b border-border/50 last:border-b-0" | ||||
|     > | ||||
|       <div class="flex items-center gap-3"> | ||||
|         <div class="flex-shrink-0"> | ||||
|           <Info class="w-5 h-5 text-primary" /> | ||||
|         </div> | ||||
|         <div class="min-w-0 flex-1"> | ||||
|           <div class="text-sm text-foreground"> | ||||
|             {{ $t('admin.banner.restartMessage') }} | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { Download, Info } from 'lucide-vue-next' | ||||
| import { useAppSettingsStore } from '@/stores/appSettings' | ||||
| const appSettingsStore = useAppSettingsStore() | ||||
| </script> | ||||
							
								
								
									
										24
									
								
								frontend/src/components/button/CloseButton.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								frontend/src/components/button/CloseButton.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| <template> | ||||
|   <Button | ||||
|     variant="ghost" | ||||
|     @click.prevent="onClose" | ||||
|     size="xs" | ||||
|     class="text-gray-400 dark:text-gray-500 hover:text-gray-600 hover:bg-gray-100 dark:hover:bg-gray-800 w-6 h-6 p-0" | ||||
|   > | ||||
|     <slot> | ||||
|       <X size="16" /> | ||||
|     </slot> | ||||
|   </Button> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { Button } from '@/components/ui/button' | ||||
| import { X } from 'lucide-vue-next' | ||||
|  | ||||
| defineProps({ | ||||
|   onClose: { | ||||
|     type: Function, | ||||
|     required: true | ||||
|   } | ||||
| }) | ||||
| </script> | ||||
							
								
								
									
										61
									
								
								frontend/src/components/combobox/SelectCombobox.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								frontend/src/components/combobox/SelectCombobox.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | ||||
| <template> | ||||
|   <ComboBox | ||||
|     :model-value="normalizedValue" | ||||
|     @update:model-value="$emit('update:modelValue', $event)" | ||||
|     :items="items" | ||||
|     :placeholder="placeholder" | ||||
|   > | ||||
|     <!-- Items --> | ||||
|     <template #item="{ item }"> | ||||
|       <div class="flex items-center gap-2"> | ||||
|         <!--USER --> | ||||
|         <Avatar v-if="type === 'user'" class="w-7 h-7"> | ||||
|           <AvatarImage :src="item.avatar_url || ''" :alt="item.label.slice(0, 2)" /> | ||||
|           <AvatarFallback>{{ item.label.slice(0, 2).toUpperCase() }}</AvatarFallback> | ||||
|         </Avatar> | ||||
|  | ||||
|         <!-- Others --> | ||||
|         <span v-else-if="item.emoji">{{ item.emoji }}</span> | ||||
|         <span>{{ item.label }}</span> | ||||
|       </div> | ||||
|     </template> | ||||
|  | ||||
|     <!-- Selected --> | ||||
|     <template #selected="{ selected }"> | ||||
|       <div class="flex items-center gap-2"> | ||||
|         <div v-if="selected" class="flex items-center gap-2"> | ||||
|           <!--USER --> | ||||
|           <Avatar v-if="type === 'user'" class="w-7 h-7"> | ||||
|             <AvatarImage :src="selected.avatar_url || ''" :alt="selected.label.slice(0, 2)" /> | ||||
|             <AvatarFallback>{{ selected.label.slice(0, 2).toUpperCase() }}</AvatarFallback> | ||||
|           </Avatar> | ||||
|  | ||||
|           <!-- Others --> | ||||
|           <span v-else-if="selected.emoji">{{ selected.emoji }}</span> | ||||
|           <span>{{ selected.label }}</span> | ||||
|         </div> | ||||
|         <span v-else>{{ placeholder }}</span> | ||||
|       </div> | ||||
|     </template> | ||||
|   </ComboBox> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { computed } from 'vue' | ||||
| import { Avatar, AvatarImage, AvatarFallback } from '@/components/ui/avatar' | ||||
| import ComboBox from '@/components/ui/combobox/ComboBox.vue' | ||||
|  | ||||
| const props = defineProps({ | ||||
|   modelValue: [String, Number, Object], | ||||
|   placeholder: String, | ||||
|   items: Array, | ||||
|   type: { | ||||
|     type: String | ||||
|   } | ||||
| }) | ||||
|  | ||||
| // Convert to str. | ||||
| const normalizedValue = computed(() => String(props.modelValue || '')) | ||||
|  | ||||
| defineEmits(['update:modelValue']) | ||||
| </script> | ||||
| @@ -1,19 +1,26 @@ | ||||
| <template> | ||||
|   <div class="w-full"> | ||||
|     <div class="rounded-md border shadow"> | ||||
|     <div class="rounded border shadow"> | ||||
|       <Table> | ||||
|         <TableHeader> | ||||
|           <TableRow v-for="headerGroup in table.getHeaderGroups()" :key="headerGroup.id"> | ||||
|             <TableHead v-for="header in headerGroup.headers" :key="header.id" class="font-semibold"> | ||||
|               <FlexRender v-if="!header.isPlaceholder" :render="header.column.columnDef.header" | ||||
|                 :props="header.getContext()" /> | ||||
|               <FlexRender | ||||
|                 v-if="!header.isPlaceholder" | ||||
|                 :render="header.column.columnDef.header" | ||||
|                 :props="header.getContext()" | ||||
|               /> | ||||
|             </TableHead> | ||||
|           </TableRow> | ||||
|         </TableHeader> | ||||
|         <TableBody> | ||||
|           <template v-if="table.getRowModel().rows?.length"> | ||||
|             <TableRow v-for="row in table.getRowModel().rows" :key="row.id" | ||||
|               :data-state="row.getIsSelected() ? 'selected' : undefined" class="hover:bg-muted/50"> | ||||
|             <TableRow | ||||
|               v-for="row in table.getRowModel().rows" | ||||
|               :key="row.id" | ||||
|               :data-state="row.getIsSelected() ? 'selected' : undefined" | ||||
|               class="hover:bg-muted/50" | ||||
|             > | ||||
|               <TableCell v-for="cell in row.getVisibleCells()" :key="cell.id"> | ||||
|                 <FlexRender :render="cell.column.columnDef.cell" :props="cell.getContext()" /> | ||||
|               </TableCell> | ||||
| @@ -32,9 +39,10 @@ | ||||
|   </div> | ||||
| </template> | ||||
|  | ||||
|  | ||||
| <script setup> | ||||
| import { FlexRender, getCoreRowModel, useVueTable } from '@tanstack/vue-table' | ||||
| import { useI18n } from 'vue-i18n' | ||||
| import { computed } from 'vue' | ||||
|  | ||||
| import { | ||||
|   Table, | ||||
| @@ -45,20 +53,30 @@ import { | ||||
|   TableRow | ||||
| } from '@/components/ui/table' | ||||
|  | ||||
| const { t } = useI18n() | ||||
| const props = defineProps({ | ||||
|   columns: Array, | ||||
|   data: Array, | ||||
|   emptyText: { | ||||
|     type: String, | ||||
|     default: 'No results.' | ||||
|     default: '' | ||||
|   } | ||||
| }) | ||||
|  | ||||
| // Set the default value for emptyText if it's empty | ||||
| const emptyText = computed( | ||||
|   () => | ||||
|     props.emptyText || | ||||
|     t('globals.messages.noResults', { | ||||
|       name: t('globals.terms.result', 2).toLowerCase() | ||||
|     }) | ||||
| ) | ||||
|  | ||||
| const table = useVueTable({ | ||||
|   get data () { | ||||
|   get data() { | ||||
|     return props.data | ||||
|   }, | ||||
|   get columns () { | ||||
|   get columns() { | ||||
|     return props.columns | ||||
|   }, | ||||
|   getCoreRowModel: getCoreRowModel() | ||||
|   | ||||
| @@ -1,10 +1,13 @@ | ||||
| <template> | ||||
|     <div ref="codeEditor" id="code-editor" class="code-editor" /> | ||||
|     <div ref="codeEditor" @click="editorView?.focus()" class="w-full h-[28rem] border rounded-md" /> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { ref, onMounted, watch, nextTick } from 'vue' | ||||
| import CodeFlask from 'codeflask' | ||||
| import { ref, onMounted, watch, nextTick, useTemplateRef } from 'vue' | ||||
| import { EditorView, basicSetup } from 'codemirror' | ||||
| import { html } from '@codemirror/lang-html' | ||||
| import { oneDark } from '@codemirror/theme-one-dark' | ||||
| import { useColorMode } from '@vueuse/core' | ||||
|  | ||||
| const props = defineProps({ | ||||
|     modelValue: { type: String, default: '' }, | ||||
| @@ -13,45 +16,38 @@ const props = defineProps({ | ||||
| }) | ||||
|  | ||||
| const emit = defineEmits(['update:modelValue']) | ||||
| const codeEditor = ref(null) | ||||
| const data = ref('') | ||||
| const flask = ref(null) | ||||
| let editorView = null  | ||||
| const codeEditor = useTemplateRef('codeEditor') | ||||
|  | ||||
| const initCodeEditor = (body) => { | ||||
|     const el = document.createElement('code-flask') | ||||
|     el.attachShadow({ mode: 'open' }) | ||||
|     el.shadowRoot.innerHTML = ` | ||||
|       <style> | ||||
|         .codeflask .codeflask__flatten { | ||||
|           font-size: 15px; | ||||
|           white-space: pre-wrap; | ||||
|           word-break: break-word; | ||||
|         } | ||||
|         .codeflask .codeflask__lines { background: #fafafa; z-index: 10; } | ||||
|         .codeflask .token.tag { font-weight: bold; } | ||||
|         .codeflask .token.attr-name { color: #111; } | ||||
|         .codeflask .token.attr-value { color: #000 !important; } | ||||
|       </style> | ||||
|       <div id="area"></div> | ||||
|     ` | ||||
|     codeEditor.value.appendChild(el) | ||||
|     const isDark = useColorMode().value === 'dark' | ||||
|  | ||||
|     flask.value = new CodeFlask(el.shadowRoot.getElementById('area'), { | ||||
|         language: props.language, | ||||
|         lineNumbers: false, | ||||
|         styleParent: el.shadowRoot, | ||||
|         readonly: props.disabled | ||||
|     editorView = new EditorView({ | ||||
|         doc: body, | ||||
|         extensions: [ | ||||
|             basicSetup, | ||||
|             html(), | ||||
|             ...(isDark ? [oneDark] : []), | ||||
|             EditorView.editable.of(!props.disabled), | ||||
|             EditorView.theme({ | ||||
|                 '&': { height: '100%' }, | ||||
|                 '.cm-editor': { height: '100%' }, | ||||
|                 '.cm-scroller': { overflow: 'auto' } | ||||
|             }), | ||||
|             EditorView.updateListener.of((update) => { | ||||
|                 if (!update.docChanged) return | ||||
|                 const v = update.state.doc.toString() | ||||
|                 emit('update:modelValue', v) | ||||
|                 data.value = v | ||||
|                  | ||||
|             }) | ||||
|         ], | ||||
|         parent: codeEditor.value | ||||
|     }) | ||||
|  | ||||
|     flask.value.onUpdate((v) => { | ||||
|         emit('update:modelValue', v) | ||||
|         data.value = v | ||||
|     }) | ||||
|  | ||||
|     flask.value.updateCode(body) | ||||
|  | ||||
|     nextTick(() => { | ||||
|         document.querySelector('code-flask').shadowRoot.querySelector('textarea').focus() | ||||
|         editorView?.focus() | ||||
|     }) | ||||
| } | ||||
|  | ||||
| @@ -61,7 +57,9 @@ onMounted(() => { | ||||
|  | ||||
| watch(() => props.modelValue, (newVal) => { | ||||
|     if (newVal !== data.value) { | ||||
|         flask.value.updateCode(newVal) | ||||
|         editorView?.dispatch({ | ||||
|             changes: { from: 0, to: editorView.state.doc.length, insert: newVal } | ||||
|         }) | ||||
|     } | ||||
| }) | ||||
| </script> | ||||
							
								
								
									
										309
									
								
								frontend/src/components/editor/TextEditor.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										309
									
								
								frontend/src/components/editor/TextEditor.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,309 @@ | ||||
| <template> | ||||
|   <div class="editor-wrapper h-full overflow-y-auto"> | ||||
|     <BubbleMenu | ||||
|       :editor="editor" | ||||
|       :tippy-options="{ duration: 100 }" | ||||
|       v-if="editor" | ||||
|       class="bg-background p-1 box will-change-transform" | ||||
|     > | ||||
|       <div class="flex space-x-1 items-center"> | ||||
|         <DropdownMenu v-if="aiPrompts.length > 0"> | ||||
|           <DropdownMenuTrigger> | ||||
|             <Button size="sm" variant="ghost" class="flex items-center justify-center"> | ||||
|               <span class="flex items-center"> | ||||
|                 <span class="text-medium">AI</span> | ||||
|                 <Bot size="14" class="ml-1" /> | ||||
|                 <ChevronDown class="w-4 h-4 ml-2" /> | ||||
|               </span> | ||||
|             </Button> | ||||
|           </DropdownMenuTrigger> | ||||
|           <DropdownMenuContent> | ||||
|             <DropdownMenuItem | ||||
|               v-for="prompt in aiPrompts" | ||||
|               :key="prompt.key" | ||||
|               @select="emitPrompt(prompt.key)" | ||||
|             > | ||||
|               {{ prompt.title }} | ||||
|             </DropdownMenuItem> | ||||
|           </DropdownMenuContent> | ||||
|         </DropdownMenu> | ||||
|         <Button | ||||
|           size="sm" | ||||
|           variant="ghost" | ||||
|           @click.prevent="editor?.chain().focus().toggleBold().run()" | ||||
|           :class="{ 'bg-gray-200 dark:bg-secondary': editor?.isActive('bold') }" | ||||
|         > | ||||
|           <Bold size="14" /> | ||||
|         </Button> | ||||
|         <Button | ||||
|           size="sm" | ||||
|           variant="ghost" | ||||
|           @click.prevent="editor?.chain().focus().toggleItalic().run()" | ||||
|           :class="{ 'bg-gray-200 dark:bg-secondary': editor?.isActive('italic') }" | ||||
|         > | ||||
|           <Italic size="14" /> | ||||
|         </Button> | ||||
|         <Button | ||||
|           size="sm" | ||||
|           variant="ghost" | ||||
|           @click.prevent="editor?.chain().focus().toggleBulletList().run()" | ||||
|           :class="{ 'bg-gray-200 dark:bg-secondary': editor?.isActive('bulletList') }" | ||||
|         > | ||||
|           <List size="14" /> | ||||
|         </Button> | ||||
|  | ||||
|         <Button | ||||
|           size="sm" | ||||
|           variant="ghost" | ||||
|           @click.prevent="editor?.chain().focus().toggleOrderedList().run()" | ||||
|           :class="{ 'bg-gray-200 dark:bg-secondary': editor?.isActive('orderedList') }" | ||||
|         > | ||||
|           <ListOrdered size="14" /> | ||||
|         </Button> | ||||
|         <Button | ||||
|           size="sm" | ||||
|           variant="ghost" | ||||
|           @click.prevent="openLinkModal" | ||||
|           :class="{ 'bg-gray-200 dark:bg-secondary': editor?.isActive('link') }" | ||||
|         > | ||||
|           <LinkIcon size="14" /> | ||||
|         </Button> | ||||
|         <div v-if="showLinkInput" class="flex space-x-2 p-2 bg-background border rounded"> | ||||
|           <Input | ||||
|             v-model="linkUrl" | ||||
|             type="text" | ||||
|             placeholder="Enter link URL" | ||||
|             class="border p-1 text-sm w-[200px]" | ||||
|           /> | ||||
|           <Button size="sm" @click="setLink"> | ||||
|             <Check size="14" /> | ||||
|           </Button> | ||||
|           <Button size="sm" @click="unsetLink"> | ||||
|             <X size="14" /> | ||||
|           </Button> | ||||
|         </div> | ||||
|       </div> | ||||
|     </BubbleMenu> | ||||
|     <EditorContent :editor="editor" class="native-html" /> | ||||
|   </div> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { ref, watch, onUnmounted } from 'vue' | ||||
| import { useEditor, EditorContent, BubbleMenu } from '@tiptap/vue-3' | ||||
| import { | ||||
|   ChevronDown, | ||||
|   Bold, | ||||
|   Italic, | ||||
|   Bot, | ||||
|   List, | ||||
|   ListOrdered, | ||||
|   Link as LinkIcon, | ||||
|   Check, | ||||
|   X | ||||
| } from 'lucide-vue-next' | ||||
| import { Button } from '@/components/ui/button' | ||||
| import { | ||||
|   DropdownMenu, | ||||
|   DropdownMenuContent, | ||||
|   DropdownMenuItem, | ||||
|   DropdownMenuTrigger | ||||
| } from '@/components/ui/dropdown-menu' | ||||
| import { Input } from '@/components/ui/input' | ||||
| import Placeholder from '@tiptap/extension-placeholder' | ||||
| import Image from '@tiptap/extension-image' | ||||
| import StarterKit from '@tiptap/starter-kit' | ||||
| import Link from '@tiptap/extension-link' | ||||
| import Table from '@tiptap/extension-table' | ||||
| import TableRow from '@tiptap/extension-table-row' | ||||
| import TableCell from '@tiptap/extension-table-cell' | ||||
| import TableHeader from '@tiptap/extension-table-header' | ||||
|  | ||||
| const textContent = defineModel('textContent', { default: '' }) | ||||
| const htmlContent = defineModel('htmlContent', { default: '' }) | ||||
| const showLinkInput = ref(false) | ||||
| const linkUrl = ref('') | ||||
|  | ||||
| const props = defineProps({ | ||||
|   placeholder: String, | ||||
|   insertContent: String, | ||||
|   autoFocus: { | ||||
|     type: Boolean, | ||||
|     default: true | ||||
|   }, | ||||
|   aiPrompts: { | ||||
|     type: Array, | ||||
|     default: () => [] | ||||
|   } | ||||
| }) | ||||
|  | ||||
| const emit = defineEmits(['send', 'aiPromptSelected']) | ||||
|  | ||||
| const emitPrompt = (key) => emit('aiPromptSelected', key) | ||||
|  | ||||
| // To preseve the table styling in emails, need to set the table style inline. | ||||
| // Created these custom extensions to set the table style inline. | ||||
| const CustomTable = Table.extend({ | ||||
|   addAttributes() { | ||||
|     return { | ||||
|       ...this.parent?.(), | ||||
|       style: { | ||||
|         parseHTML: (element) => | ||||
|           (element.getAttribute('style') || '') + '; border: 1px solid #dee2e6 !important; width: 100%; margin:0; table-layout: fixed; border-collapse: collapse; position:relative; border-radius: 0.25rem;' | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| }) | ||||
|  | ||||
| const CustomTableCell = TableCell.extend({ | ||||
|   addAttributes() { | ||||
|     return { | ||||
|       ...this.parent?.(), | ||||
|       style: { | ||||
|         parseHTML: (element) => | ||||
|           (element.getAttribute('style') || '') + | ||||
|           '; border: 1px solid #dee2e6 !important; box-sizing: border-box !important; min-width: 1em !important; padding: 6px 8px !important; vertical-align: top !important;' | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| }) | ||||
|  | ||||
| const CustomTableHeader = TableHeader.extend({ | ||||
|   addAttributes() { | ||||
|     return { | ||||
|       ...this.parent?.(), | ||||
|       style: { | ||||
|         parseHTML: (element) => | ||||
|           (element.getAttribute('style') || '') + | ||||
|           '; background-color: #f8f9fa !important; color: #212529 !important; font-weight: bold !important; text-align: left !important; border: 1px solid #dee2e6 !important; padding: 6px 8px !important;' | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| }) | ||||
|  | ||||
| const isInternalUpdate = ref(false) | ||||
|  | ||||
| const editor = useEditor({ | ||||
|   extensions: [ | ||||
|     StarterKit.configure(), | ||||
|     Image.configure({ HTMLAttributes: { class: 'inline-image' } }), | ||||
|     Placeholder.configure({ placeholder: () => props.placeholder }), | ||||
|     Link, | ||||
|     CustomTable.configure({ resizable: false }), | ||||
|     TableRow, | ||||
|     CustomTableCell, | ||||
|     CustomTableHeader | ||||
|   ], | ||||
|   autofocus: props.autoFocus, | ||||
|   content: htmlContent.value, | ||||
|   editorProps: { | ||||
|     attributes: { class: 'outline-none' }, | ||||
|     handleKeyDown: (view, event) => { | ||||
|       if (event.ctrlKey && event.key === 'Enter') { | ||||
|         emit('send') | ||||
|         return true | ||||
|       } | ||||
|     } | ||||
|   }, | ||||
|   // To update state when user types. | ||||
|   onUpdate: ({ editor }) => { | ||||
|     isInternalUpdate.value = true | ||||
|     htmlContent.value = editor.getHTML() | ||||
|     textContent.value = editor.getText() | ||||
|     isInternalUpdate.value = false | ||||
|   } | ||||
| }) | ||||
|  | ||||
| watch( | ||||
|   htmlContent, | ||||
|   (newContent) => { | ||||
|     if (!isInternalUpdate.value && editor.value && newContent !== editor.value.getHTML()) { | ||||
|       editor.value.commands.setContent(newContent || '', false) | ||||
|       textContent.value = editor.value.getText() | ||||
|       editor.value.commands.focus() | ||||
|     } | ||||
|   }, | ||||
|   { immediate: true } | ||||
| ) | ||||
|  | ||||
| // Insert content at cursor position when insertContent prop changes. | ||||
| watch( | ||||
|   () => props.insertContent, | ||||
|   (val) => { | ||||
|     if (val) editor.value?.commands.insertContent(val) | ||||
|   } | ||||
| ) | ||||
|  | ||||
| onUnmounted(() => { | ||||
|   editor.value?.destroy() | ||||
| }) | ||||
|  | ||||
| const openLinkModal = () => { | ||||
|   if (editor.value?.isActive('link')) { | ||||
|     linkUrl.value = editor.value.getAttributes('link').href | ||||
|   } else { | ||||
|     linkUrl.value = '' | ||||
|   } | ||||
|   showLinkInput.value = true | ||||
| } | ||||
|  | ||||
| const setLink = () => { | ||||
|   if (linkUrl.value) { | ||||
|     editor.value?.chain().focus().extendMarkRange('link').setLink({ href: linkUrl.value }).run() | ||||
|   } | ||||
|   showLinkInput.value = false | ||||
| } | ||||
|  | ||||
| const unsetLink = () => { | ||||
|   editor.value?.chain().focus().unsetLink().run() | ||||
|   showLinkInput.value = false | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <style lang="scss"> | ||||
| // Moving placeholder to the top. | ||||
| .tiptap p.is-editor-empty:first-child::before { | ||||
|   content: attr(data-placeholder); | ||||
|   float: left; | ||||
|   color: #adb5bd; | ||||
|   pointer-events: none; | ||||
|   height: 0; | ||||
| } | ||||
|  | ||||
| // Ensure the parent div has a proper height | ||||
| .editor-wrapper div[aria-expanded='false'] { | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
|   height: 100%; | ||||
| } | ||||
|  | ||||
| // Ensure the editor content has a proper height and breaks words | ||||
| .tiptap.ProseMirror { | ||||
|   flex: 1; | ||||
|   min-height: 70px; | ||||
|   overflow-y: auto; | ||||
|   word-wrap: break-word !important; | ||||
|   overflow-wrap: break-word !important; | ||||
|   word-break: break-word; | ||||
|   white-space: pre-wrap; | ||||
|   max-width: 100%; | ||||
| } | ||||
|  | ||||
| .tiptap { | ||||
|   // Table styling | ||||
|   .tableWrapper { | ||||
|     margin: 1.5rem 0; | ||||
|     overflow-x: auto; | ||||
|   } | ||||
|  | ||||
|   // Anchor tag styling | ||||
|   a { | ||||
|     color: #0066cc; | ||||
|     cursor: pointer; | ||||
|  | ||||
|     &:hover { | ||||
|       color: #003d7a; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </style> | ||||
							
								
								
									
										212
									
								
								frontend/src/components/filter/FilterBuilder.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										212
									
								
								frontend/src/components/filter/FilterBuilder.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,212 @@ | ||||
| <template> | ||||
|   <div class="space-y-4"> | ||||
|     <div class="w-[27rem]" v-if="modelValue.length === 0"></div> | ||||
|  | ||||
|     <div | ||||
|       v-for="(modelFilter, index) in modelValue" | ||||
|       :key="index" | ||||
|       class="group flex items-center gap-3" | ||||
|     > | ||||
|       <div class="flex gap-2 w-full"> | ||||
|         <!-- Field --> | ||||
|         <div class="flex-1"> | ||||
|           <Select v-model="modelFilter.field"> | ||||
|             <SelectTrigger> | ||||
|               <SelectValue | ||||
|                 :placeholder=" | ||||
|                   t('globals.messages.select', { name: t('globals.terms.field').toLowerCase() }) | ||||
|                 " | ||||
|               /> | ||||
|             </SelectTrigger> | ||||
|             <SelectContent> | ||||
|               <SelectGroup> | ||||
|                 <SelectItem v-for="field in fields" :key="field.field" :value="field.field"> | ||||
|                   {{ field.label }} | ||||
|                 </SelectItem> | ||||
|               </SelectGroup> | ||||
|             </SelectContent> | ||||
|           </Select> | ||||
|         </div> | ||||
|  | ||||
|         <!-- Operator --> | ||||
|         <div class="flex-1"> | ||||
|           <Select v-model="modelFilter.operator" v-if="modelFilter.field"> | ||||
|             <SelectTrigger> | ||||
|               <SelectValue | ||||
|                 :placeholder=" | ||||
|                   t('globals.messages.select', { name: t('globals.terms.operator').toLowerCase() }) | ||||
|                 " | ||||
|               /> | ||||
|             </SelectTrigger> | ||||
|             <SelectContent> | ||||
|               <SelectGroup> | ||||
|                 <SelectItem v-for="op in getFieldOperators(modelFilter)" :key="op" :value="op"> | ||||
|                   {{ op }} | ||||
|                 </SelectItem> | ||||
|               </SelectGroup> | ||||
|             </SelectContent> | ||||
|           </Select> | ||||
|         </div> | ||||
|  | ||||
|         <!-- Value --> | ||||
|         <div class="flex-1"> | ||||
|           <div v-if="modelFilter.field && modelFilter.operator"> | ||||
|             <template v-if="modelFilter.operator !== 'set' && modelFilter.operator !== 'not set'"> | ||||
|               <SelectComboBox | ||||
|                 v-if=" | ||||
|                   getFieldOptions(modelFilter).length > 0 && | ||||
|                   modelFilter.field === 'assigned_user_id' | ||||
|                 " | ||||
|                 v-model="modelFilter.value" | ||||
|                 :items="getFieldOptions(modelFilter)" | ||||
|                 :placeholder="t('globals.messages.select', { name: '' })" | ||||
|                 type="user" | ||||
|               /> | ||||
|  | ||||
|               <SelectComboBox | ||||
|                 v-else-if=" | ||||
|                   getFieldOptions(modelFilter).length > 0 && | ||||
|                   modelFilter.field === 'assigned_team_id' | ||||
|                 " | ||||
|                 v-model="modelFilter.value" | ||||
|                 :items="getFieldOptions(modelFilter)" | ||||
|                 :placeholder="t('globals.messages.select', { name: '' })" | ||||
|                 type="team" | ||||
|               /> | ||||
|  | ||||
|               <SelectComboBox | ||||
|                 v-else-if="getFieldOptions(modelFilter).length > 0" | ||||
|                 v-model="modelFilter.value" | ||||
|                 :items="getFieldOptions(modelFilter)" | ||||
|                 :placeholder="t('globals.messages.select', { name: '' })" | ||||
|               /> | ||||
|  | ||||
|               <Input | ||||
|                 v-else | ||||
|                 v-model="modelFilter.value" | ||||
|                 :placeholder="t('globals.terms.value')" | ||||
|                 type="text" | ||||
|               /> | ||||
|             </template> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|       <CloseButton :onClose="() => removeFilter(index)" /> | ||||
|     </div> | ||||
|  | ||||
|     <div class="flex items-center justify-between pt-3"> | ||||
|       <Button variant="ghost" size="sm" @click="addFilter" class="text-slate-600"> | ||||
|         <Plus class="w-3 h-3 mr-1" /> | ||||
|         {{ | ||||
|           $t('globals.messages.add', { | ||||
|             name: $t('globals.terms.filter') | ||||
|           }) | ||||
|         }} | ||||
|       </Button> | ||||
|       <div class="flex gap-2" v-if="showButtons"> | ||||
|         <Button variant="ghost" @click="clearFilters">{{ $t('globals.messages.reset') }}</Button> | ||||
|         <Button @click="applyFilters">{{ $t('globals.messages.apply') }}</Button> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { computed, onMounted, watch } from 'vue' | ||||
| import { | ||||
|   Select, | ||||
|   SelectContent, | ||||
|   SelectGroup, | ||||
|   SelectItem, | ||||
|   SelectTrigger, | ||||
|   SelectValue | ||||
| } from '@/components/ui/select' | ||||
| import { Plus } from 'lucide-vue-next' | ||||
| import { Button } from '@/components/ui/button' | ||||
| import { Input } from '@/components/ui/input' | ||||
| import { useI18n } from 'vue-i18n' | ||||
| import CloseButton from '@/components/button/CloseButton.vue' | ||||
| import SelectComboBox from '@/components/combobox/SelectCombobox.vue' | ||||
|  | ||||
| const props = defineProps({ | ||||
|   fields: { | ||||
|     type: Array, | ||||
|     required: true | ||||
|   }, | ||||
|   showButtons: { | ||||
|     type: Boolean, | ||||
|     default: true | ||||
|   } | ||||
| }) | ||||
| const { t } = useI18n() | ||||
| const emit = defineEmits(['apply', 'clear']) | ||||
| const modelValue = defineModel('modelValue', { required: false, default: () => [] }) | ||||
|  | ||||
| const createFilter = () => ({ field: '', operator: '', value: '' }) | ||||
|  | ||||
| onMounted(() => { | ||||
|   if (modelValue.value.length === 0) { | ||||
|     modelValue.value = [createFilter()] | ||||
|   } | ||||
| }) | ||||
|  | ||||
| const getModel = (field) => { | ||||
|   const fieldConfig = props.fields.find((f) => f.field === field) | ||||
|   return fieldConfig?.model || '' | ||||
| } | ||||
|  | ||||
| // Set model for each filter | ||||
| watch( | ||||
|   () => modelValue.value, | ||||
|   (filters) => { | ||||
|     filters.forEach((filter) => { | ||||
|       if (filter.field && !filter.model) { | ||||
|         filter.model = getModel(filter.field) | ||||
|       } | ||||
|     }) | ||||
|   }, | ||||
|   { deep: true } | ||||
| ) | ||||
|  | ||||
| // Reset operator and value when field changes for a filter at a given index | ||||
| watch( | ||||
|   () => modelValue.value.map((f) => f.field), | ||||
|   (newFields, oldFields) => { | ||||
|     newFields.forEach((field, index) => { | ||||
|       if (field !== oldFields[index]) { | ||||
|         modelValue.value[index].operator = '' | ||||
|         modelValue.value[index].value = '' | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| ) | ||||
|  | ||||
| const addFilter = () => { | ||||
|   modelValue.value = [...modelValue.value, createFilter()] | ||||
| } | ||||
| const removeFilter = (index) => { | ||||
|   modelValue.value = modelValue.value.filter((_, i) => i !== index) | ||||
| } | ||||
| const applyFilters = () => { | ||||
|   modelValue.value = validFilters.value | ||||
|   emit('apply', modelValue.value) | ||||
| } | ||||
| const clearFilters = () => { | ||||
|   modelValue.value = [] | ||||
|   emit('clear') | ||||
| } | ||||
|  | ||||
| const validFilters = computed(() => { | ||||
|   return modelValue.value.filter((filter) => filter.field && filter.operator && filter.value) | ||||
| }) | ||||
|  | ||||
| const getFieldOptions = (fieldValue) => { | ||||
|   const field = props.fields.find((f) => f.field === fieldValue.field) | ||||
|   return field?.options || [] | ||||
| } | ||||
|  | ||||
| const getFieldOperators = (modelFilter) => { | ||||
|   const field = props.fields.find((f) => f.field === modelFilter.field) | ||||
|   return field?.operators || [] | ||||
| } | ||||
| </script> | ||||
| @@ -1,17 +1,17 @@ | ||||
| <template> | ||||
|   <div | ||||
|     class="flex flex-col p-4 border rounded-lg shadow-sm hover:shadow transition-colors cursor-pointer max-w-xs" | ||||
|     class="flex flex-col p-4 border rounded shadow-sm hover:shadow transition-colors cursor-pointer max-w-xs" | ||||
|     @click="handleClick"> | ||||
|     <div class="flex items-center mb-2"> | ||||
|       <component :is="icon" size="24" class="mr-2 text-primary" /> | ||||
|       <h3 class="text-lg font-medium text-gray-800">{{ title }}</h3> | ||||
|       <h3 class="text-lg font-medium">{{ title }}</h3> | ||||
|     </div> | ||||
|     <p class="text-sm text-gray-600">{{ subTitle }}</p> | ||||
|     <p class="text-sm text-gray-600 dark:text-gray-400">{{ subTitle }}</p> | ||||
|   </div> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { defineProps, defineEmits } from 'vue' | ||||
| import { defineEmits } from 'vue' | ||||
|  | ||||
| const props = defineProps({ | ||||
|   title: String, | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| <template> | ||||
|   <div v-if="!isHidden"> | ||||
|     <div class="flex items-center space-x-4 h-12 px-2"> | ||||
|       <SidebarTrigger class="cursor-pointer w-4 h-4" /> | ||||
|       <span class="text-xl font-semibold text-gray-800"> | ||||
|       <SidebarTrigger class="cursor-pointer" /> | ||||
|       <span class="text-xl font-semibold"> | ||||
|         {{ title }} | ||||
|       </span> | ||||
|     </div> | ||||
|   | ||||
| @@ -1,6 +1,11 @@ | ||||
| <script setup> | ||||
| import { adminNavItems, reportsNavItems, accountNavItems } from '@/constants/navigation' | ||||
| import { RouterLink, useRoute } from 'vue-router' | ||||
| import { | ||||
|   adminNavItems, | ||||
|   reportsNavItems, | ||||
|   accountNavItems, | ||||
|   contactNavItems | ||||
| } from '@/constants/navigation' | ||||
| import { RouterLink, useRoute, useRouter } from 'vue-router' | ||||
| import { Collapsible, CollapsibleContent, CollapsibleTrigger } from '@/components/ui/collapsible' | ||||
| import { | ||||
|   Sidebar, | ||||
| @@ -9,7 +14,6 @@ import { | ||||
|   SidebarHeader, | ||||
|   SidebarInset, | ||||
|   SidebarMenu, | ||||
|   SidebarSeparator, | ||||
|   SidebarMenuAction, | ||||
|   SidebarMenuButton, | ||||
|   SidebarMenuItem, | ||||
| @@ -18,14 +22,15 @@ import { | ||||
|   SidebarProvider, | ||||
|   SidebarRail | ||||
| } from '@/components/ui/sidebar' | ||||
| import { useAppSettingsStore } from '@/stores/appSettings' | ||||
| import { | ||||
|   ChevronRight, | ||||
|   EllipsisVertical, | ||||
|   User, | ||||
|   Search, | ||||
|   Plus, | ||||
|   CircleUserRound, | ||||
|   UserSearch, | ||||
|   UsersRound, | ||||
|   Search | ||||
|   CircleDashed, | ||||
|   List | ||||
| } from 'lucide-vue-next' | ||||
| import { | ||||
|   DropdownMenu, | ||||
| @@ -35,16 +40,30 @@ import { | ||||
| } from '@/components/ui/dropdown-menu' | ||||
| import { filterNavItems } from '@/utils/nav-permissions' | ||||
| import { useStorage } from '@vueuse/core' | ||||
| import { computed } from 'vue' | ||||
| import { computed, ref, watch } from 'vue' | ||||
| import { useI18n } from 'vue-i18n' | ||||
| import { useUserStore } from '@/stores/user' | ||||
| import { useConversationStore } from '@/stores/conversation' | ||||
|  | ||||
| defineProps({ | ||||
|   userTeams: { type: Array, default: () => [] }, | ||||
|   userViews: { type: Array, default: () => [] } | ||||
| }) | ||||
| const userStore = useUserStore() | ||||
| const conversationStore = useConversationStore() | ||||
| const settingsStore = useAppSettingsStore() | ||||
| const route = useRoute() | ||||
| const emit = defineEmits(['createView', 'editView', 'deleteView']) | ||||
| const router = useRouter() | ||||
| const { t } = useI18n() | ||||
| const emit = defineEmits(['createView', 'editView', 'deleteView', 'createConversation']) | ||||
|  | ||||
| const isActiveParent = (parentHref) => { | ||||
|   return route.path.startsWith(parentHref) | ||||
| } | ||||
|  | ||||
| const isInboxRoute = (path) => { | ||||
|   return path.startsWith('/inboxes') | ||||
| } | ||||
|  | ||||
| const openCreateViewDialog = () => { | ||||
|   emit('createView') | ||||
| @@ -58,18 +77,86 @@ const deleteView = (view) => { | ||||
|   emit('deleteView', view) | ||||
| } | ||||
|  | ||||
| // Navigation methods with conversation retention | ||||
| const navigateToInbox = (type) => { | ||||
|   if (conversationStore.hasConversationOpen && conversationStore.conversation.data?.uuid) { | ||||
|     router.push({ | ||||
|       name: 'inbox-conversation', | ||||
|       params: { | ||||
|         type, | ||||
|         uuid: conversationStore.conversation.data.uuid | ||||
|       } | ||||
|     }) | ||||
|   } else { | ||||
|     router.push({ | ||||
|       name: 'inbox', | ||||
|       params: { type } | ||||
|     }) | ||||
|   } | ||||
| } | ||||
|  | ||||
| const navigateToTeamInbox = (teamID) => { | ||||
|   if (conversationStore.hasConversationOpen && conversationStore.conversation.data?.uuid) { | ||||
|     router.push({ | ||||
|       name: 'team-inbox-conversation', | ||||
|       params: { | ||||
|         teamID, | ||||
|         uuid: conversationStore.conversation.data.uuid | ||||
|       } | ||||
|     }) | ||||
|   } else { | ||||
|     router.push({ | ||||
|       name: 'team-inbox', | ||||
|       params: { teamID } | ||||
|     }) | ||||
|   } | ||||
| } | ||||
|  | ||||
| const navigateToViewInbox = (viewID) => { | ||||
|   if (conversationStore.hasConversationOpen && conversationStore.conversation.data?.uuid) { | ||||
|     router.push({ | ||||
|       name: 'view-inbox-conversation', | ||||
|       params: { | ||||
|         viewID, | ||||
|         uuid: conversationStore.conversation.data.uuid | ||||
|       } | ||||
|     }) | ||||
|   } else { | ||||
|     router.push({ | ||||
|       name: 'view-inbox', | ||||
|       params: { viewID } | ||||
|     }) | ||||
|   } | ||||
| } | ||||
|  | ||||
| const filteredAdminNavItems = computed(() => filterNavItems(adminNavItems, userStore.can)) | ||||
| const filteredReportsNavItems = computed(() => filterNavItems(reportsNavItems, userStore.can)) | ||||
| const filteredContactsNavItems = computed(() => filterNavItems(contactNavItems, userStore.can)) | ||||
|  | ||||
| const isActiveParent = (parentHref) => { | ||||
|   return route.path.startsWith(parentHref) | ||||
| } | ||||
|  | ||||
| const isInboxRoute = (path) => { | ||||
|   return path.startsWith('/inboxes') | ||||
| // For auto opening admin collapsibles when a child route is active | ||||
| const openAdminCollapsible = ref(null) | ||||
| const toggleAdminCollapsible = (titleKey) => { | ||||
|   openAdminCollapsible.value = openAdminCollapsible.value === titleKey ? null : titleKey | ||||
| } | ||||
| // Watch for route changes and update the active collapsible | ||||
| watch( | ||||
|   [() => route.path, filteredAdminNavItems], | ||||
|   () => { | ||||
|     const activeItem = filteredAdminNavItems.value.find((item) => { | ||||
|       if (!item.children) return isActiveParent(item.href) | ||||
|       return item.children.some((child) => isActiveParent(child.href)) | ||||
|     }) | ||||
|     if (activeItem) { | ||||
|       openAdminCollapsible.value = activeItem.titleKey | ||||
|     } | ||||
|   }, | ||||
|   { immediate: true } | ||||
| ) | ||||
|  | ||||
| // Sidebar open state in local storage | ||||
| const sidebarOpen = useStorage('mainSidebarOpen', true) | ||||
| const teamInboxOpen = useStorage('teamInboxOpen', true) | ||||
| const viewInboxOpen = useStorage('viewInboxOpen', true) | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| @@ -78,6 +165,43 @@ const sidebarOpen = useStorage('mainSidebarOpen', true) | ||||
|     :default-open="sidebarOpen" | ||||
|     v-on:update:open="sidebarOpen = $event" | ||||
|   > | ||||
|     <!-- Contacts sidebar --> | ||||
|     <template | ||||
|       v-if="route.matched.some((record) => record.name && record.name.startsWith('contact'))" | ||||
|     > | ||||
|       <Sidebar collapsible="offcanvas" class="border-r ml-12"> | ||||
|         <SidebarHeader> | ||||
|           <SidebarMenu> | ||||
|             <SidebarMenuItem> | ||||
|               <div class="px-1"> | ||||
|                 <span class="font-semibold text-xl"> | ||||
|                   {{ t('globals.terms.contact', 2) }} | ||||
|                 </span> | ||||
|               </div> | ||||
|             </SidebarMenuItem> | ||||
|           </SidebarMenu> | ||||
|         </SidebarHeader> | ||||
|         <SidebarContent> | ||||
|           <SidebarGroup> | ||||
|             <SidebarMenu> | ||||
|               <SidebarMenuItem v-for="item in filteredContactsNavItems" :key="item.titleKey"> | ||||
|                 <SidebarMenuButton :isActive="isActiveParent(item.href)" asChild> | ||||
|                   <router-link :to="item.href"> | ||||
|                     <span>{{ | ||||
|                       t('globals.messages.all', { | ||||
|                         name: t(item.titleKey, 2).toLowerCase() | ||||
|                       }) | ||||
|                     }}</span> | ||||
|                   </router-link> | ||||
|                 </SidebarMenuButton> | ||||
|               </SidebarMenuItem> | ||||
|             </SidebarMenu> | ||||
|           </SidebarGroup> | ||||
|         </SidebarContent> | ||||
|         <SidebarRail /> | ||||
|       </Sidebar> | ||||
|     </template> | ||||
|  | ||||
|     <!-- Reports sidebar --> | ||||
|     <template | ||||
|       v-if=" | ||||
| @@ -89,22 +213,21 @@ const sidebarOpen = useStorage('mainSidebarOpen', true) | ||||
|         <SidebarHeader> | ||||
|           <SidebarMenu> | ||||
|             <SidebarMenuItem> | ||||
|               <SidebarMenuButton :isActive="isActiveParent('/reports/overview')" asChild> | ||||
|                 <div> | ||||
|                   <span class="font-semibold text-xl">Reports</span> | ||||
|                 </div> | ||||
|               </SidebarMenuButton> | ||||
|               <div class="px-1"> | ||||
|                 <span class="font-semibold text-xl"> | ||||
|                   {{ t('globals.terms.report', 2) }} | ||||
|                 </span> | ||||
|               </div> | ||||
|             </SidebarMenuItem> | ||||
|           </SidebarMenu> | ||||
|         </SidebarHeader> | ||||
|         <SidebarSeparator /> | ||||
|         <SidebarContent> | ||||
|           <SidebarGroup> | ||||
|             <SidebarMenu> | ||||
|               <SidebarMenuItem v-for="item in filteredReportsNavItems" :key="item.title"> | ||||
|               <SidebarMenuItem v-for="item in filteredReportsNavItems" :key="item.titleKey"> | ||||
|                 <SidebarMenuButton :isActive="isActiveParent(item.href)" asChild> | ||||
|                   <router-link :to="item.href"> | ||||
|                     <span>{{ item.title }}</span> | ||||
|                     <span>{{ t(item.titleKey) }}</span> | ||||
|                   </router-link> | ||||
|                 </SidebarMenuButton> | ||||
|               </SidebarMenuItem> | ||||
| @@ -121,37 +244,41 @@ const sidebarOpen = useStorage('mainSidebarOpen', true) | ||||
|         <SidebarHeader> | ||||
|           <SidebarMenu> | ||||
|             <SidebarMenuItem> | ||||
|               <SidebarMenuButton :isActive="isActiveParent('/admin')" asChild> | ||||
|                 <div> | ||||
|                   <span class="font-semibold text-xl">Admin</span> | ||||
|               <div class="flex flex-col items-start justify-between w-full px-1"> | ||||
|                 <span class="font-semibold text-xl"> | ||||
|                   {{ t('globals.terms.admin') }} | ||||
|                 </span> | ||||
|                 <!-- App version --> | ||||
|                 <div class="text-xs text-muted-foreground"> | ||||
|                   ({{ settingsStore.settings['app.version'] }}) | ||||
|                 </div> | ||||
|               </SidebarMenuButton> | ||||
|               </div> | ||||
|             </SidebarMenuItem> | ||||
|           </SidebarMenu> | ||||
|         </SidebarHeader> | ||||
|         <SidebarSeparator /> | ||||
|         <SidebarContent> | ||||
|           <SidebarGroup> | ||||
|             <SidebarMenu> | ||||
|               <SidebarMenuItem v-for="item in filteredAdminNavItems" :key="item.title"> | ||||
|               <SidebarMenuItem v-for="item in filteredAdminNavItems" :key="item.titleKey"> | ||||
|                 <SidebarMenuButton | ||||
|                   v-if="!item.children" | ||||
|                   :isActive="isActiveParent(item.href)" | ||||
|                   asChild | ||||
|                 > | ||||
|                   <router-link :to="item.href"> | ||||
|                     <span>{{ item.title }}</span> | ||||
|                     <span>{{ t(item.titleKey) }}</span> | ||||
|                   </router-link> | ||||
|                 </SidebarMenuButton> | ||||
|  | ||||
|                 <Collapsible | ||||
|                   v-else | ||||
|                   class="group/collapsible" | ||||
|                   :default-open="isActiveParent(item.href)" | ||||
|                   :open="openAdminCollapsible === item.titleKey" | ||||
|                   @update:open="toggleAdminCollapsible(item.titleKey)" | ||||
|                 > | ||||
|                   <CollapsibleTrigger as-child> | ||||
|                     <SidebarMenuButton :isActive="isActiveParent(item.href)"> | ||||
|                       <span>{{ item.title }}</span> | ||||
|                       <span>{{ t(item.titleKey, item.isTitleKeyPlural === true ? 2 : 1) }}</span> | ||||
|                       <ChevronRight | ||||
|                         class="ml-auto transition-transform duration-200 group-data-[state=open]/collapsible:rotate-90" | ||||
|                       /> | ||||
| @@ -159,10 +286,10 @@ const sidebarOpen = useStorage('mainSidebarOpen', true) | ||||
|                   </CollapsibleTrigger> | ||||
|                   <CollapsibleContent> | ||||
|                     <SidebarMenuSub> | ||||
|                       <SidebarMenuSubItem v-for="child in item.children" :key="child.title"> | ||||
|                       <SidebarMenuSubItem v-for="child in item.children" :key="child.titleKey"> | ||||
|                         <SidebarMenuButton size="sm" :isActive="isActiveParent(child.href)" asChild> | ||||
|                           <router-link :to="child.href"> | ||||
|                             <span>{{ child.title }}</span> | ||||
|                             <span>{{ t(child.titleKey) }}</span> | ||||
|                           </router-link> | ||||
|                         </SidebarMenuButton> | ||||
|                       </SidebarMenuSubItem> | ||||
| @@ -183,22 +310,21 @@ const sidebarOpen = useStorage('mainSidebarOpen', true) | ||||
|         <SidebarHeader> | ||||
|           <SidebarMenu> | ||||
|             <SidebarMenuItem> | ||||
|               <SidebarMenuButton :isActive="isActiveParent('/account/profile')" asChild> | ||||
|                 <div> | ||||
|                   <span class="font-semibold text-xl">Account</span> | ||||
|                 </div> | ||||
|               </SidebarMenuButton> | ||||
|               <div class="px-1"> | ||||
|                 <span class="font-semibold text-xl"> | ||||
|                   {{ t('globals.terms.account') }} | ||||
|                 </span> | ||||
|               </div> | ||||
|             </SidebarMenuItem> | ||||
|           </SidebarMenu> | ||||
|         </SidebarHeader> | ||||
|         <SidebarSeparator /> | ||||
|         <SidebarContent> | ||||
|           <SidebarGroup> | ||||
|             <SidebarMenu> | ||||
|               <SidebarMenuItem v-for="item in accountNavItems" :key="item.title"> | ||||
|               <SidebarMenuItem v-for="item in accountNavItems" :key="item.titleKey"> | ||||
|                 <SidebarMenuButton :isActive="isActiveParent(item.href)" asChild> | ||||
|                   <router-link :to="item.href"> | ||||
|                     <span>{{ item.title }}</span> | ||||
|                     <span>{{ t(item.titleKey) }}</span> | ||||
|                   </router-link> | ||||
|                 </SidebarMenuButton> | ||||
|                 <SidebarMenuAction> | ||||
| @@ -218,64 +344,83 @@ const sidebarOpen = useStorage('mainSidebarOpen', true) | ||||
|         <SidebarHeader> | ||||
|           <SidebarMenu> | ||||
|             <SidebarMenuItem> | ||||
|               <SidebarMenuButton asChild> | ||||
|                 <div class="flex items-center justify-between w-full"> | ||||
|                   <div class="font-semibold text-xl">Inbox</div> | ||||
|                   <div class="ml-auto"> | ||||
|                     <router-link :to="{ name: 'search' }"> | ||||
|                       <div class="flex items-center bg-accent p-2 rounded-full"> | ||||
|                         <Search | ||||
|                           class="transition-transform duration-200 hover:scale-110 cursor-pointer" | ||||
|                           size="15" | ||||
|                           stroke-width="2.5" | ||||
|                         /> | ||||
|                       </div> | ||||
|                     </router-link> | ||||
|                   </div> | ||||
|               <div class="flex items-center justify-between w-full px-1"> | ||||
|                 <div class="font-semibold text-xl"> | ||||
|                   <span>{{ t('globals.terms.inbox') }}</span> | ||||
|                 </div> | ||||
|               </SidebarMenuButton> | ||||
|                 <div class="mr-1 mt-1 hover:scale-110 transition-transform"> | ||||
|                   <router-link :to="{ name: 'search' }"> | ||||
|                     <Search size="18" stroke-width="2.5" /> | ||||
|                   </router-link> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </SidebarMenuItem> | ||||
|           </SidebarMenu> | ||||
|         </SidebarHeader> | ||||
|         <SidebarSeparator /> | ||||
|  | ||||
|         <SidebarContent> | ||||
|           <SidebarGroup> | ||||
|             <SidebarMenu> | ||||
|               <SidebarMenuItem> | ||||
|                 <SidebarMenuButton asChild> | ||||
|                   <a href="#" @click="emit('createConversation')"> | ||||
|                     <Plus /> | ||||
|                     <span | ||||
|                       >{{ | ||||
|                         t('globals.messages.new', { | ||||
|                           name: t('globals.terms.conversation').toLowerCase() | ||||
|                         }) | ||||
|                       }} | ||||
|                     </span> | ||||
|                   </a> | ||||
|                 </SidebarMenuButton> | ||||
|               </SidebarMenuItem> | ||||
|               <SidebarMenuItem> | ||||
|                 <SidebarMenuButton asChild :isActive="isActiveParent('/inboxes/assigned')"> | ||||
|                   <router-link :to="{ name: 'inbox', params: { type: 'assigned' } }"> | ||||
|                     <CircleUserRound /> | ||||
|                     <span>My inbox</span> | ||||
|                   </router-link> | ||||
|                   <a href="#" @click.prevent="navigateToInbox('assigned')"> | ||||
|                     <User /> | ||||
|                     <span>{{ t('globals.terms.myInbox') }}</span> | ||||
|                   </a> | ||||
|                 </SidebarMenuButton> | ||||
|               </SidebarMenuItem> | ||||
|  | ||||
|               <SidebarMenuItem> | ||||
|                 <SidebarMenuButton asChild :isActive="isActiveParent('/inboxes/unassigned')"> | ||||
|                   <router-link :to="{ name: 'inbox', params: { type: 'unassigned' } }"> | ||||
|                     <UserSearch /> | ||||
|                     <span>Unassigned</span> | ||||
|                   </router-link> | ||||
|                   <a href="#" @click.prevent="navigateToInbox('unassigned')"> | ||||
|                     <CircleDashed /> | ||||
|                     <span> | ||||
|                       {{ t('globals.terms.unassigned') }} | ||||
|                     </span> | ||||
|                   </a> | ||||
|                 </SidebarMenuButton> | ||||
|               </SidebarMenuItem> | ||||
|  | ||||
|               <SidebarMenuItem> | ||||
|                 <SidebarMenuButton asChild :isActive="isActiveParent('/inboxes/all')"> | ||||
|                   <router-link :to="{ name: 'inbox', params: { type: 'all' } }"> | ||||
|                     <UsersRound /> | ||||
|                     <span>All</span> | ||||
|                   </router-link> | ||||
|                   <a href="#" @click.prevent="navigateToInbox('all')"> | ||||
|                     <List /> | ||||
|                     <span> | ||||
|                       {{ t('globals.messages.all') }} | ||||
|                     </span> | ||||
|                   </a> | ||||
|                 </SidebarMenuButton> | ||||
|               </SidebarMenuItem> | ||||
|  | ||||
|               <!-- Team Inboxes --> | ||||
|               <Collapsible defaultOpen class="group/collapsible" v-if="userTeams.length"> | ||||
|               <Collapsible | ||||
|                 defaultOpen | ||||
|                 class="group/collapsible" | ||||
|                 v-if="userTeams.length" | ||||
|                 v-model:open="teamInboxOpen" | ||||
|               > | ||||
|                 <SidebarMenuItem> | ||||
|                   <CollapsibleTrigger as-child> | ||||
|                     <SidebarMenuButton asChild> | ||||
|                       <router-link to="#"> | ||||
|                         <!-- <Users /> --> | ||||
|                         <span>Team inboxes</span> | ||||
|                         <span> | ||||
|                           {{ t('globals.terms.teamInbox', 2) }} | ||||
|                         </span> | ||||
|                         <ChevronRight | ||||
|                           class="ml-auto transition-transform duration-200 group-data-[state=open]/collapsible:rotate-90" | ||||
|                         /> | ||||
| @@ -290,9 +435,9 @@ const sidebarOpen = useStorage('mainSidebarOpen', true) | ||||
|                           :is-active="route.params.teamID == team.id" | ||||
|                           asChild | ||||
|                         > | ||||
|                           <router-link :to="{ name: 'team-inbox', params: { teamID: team.id } }"> | ||||
|                           <a href="#" @click.prevent="navigateToTeamInbox(team.id)"> | ||||
|                             {{ team.emoji }}<span>{{ team.name }}</span> | ||||
|                           </router-link> | ||||
|                           </a> | ||||
|                         </SidebarMenuButton> | ||||
|                       </SidebarMenuSubItem> | ||||
|                     </SidebarMenuSub> | ||||
| @@ -301,31 +446,30 @@ const sidebarOpen = useStorage('mainSidebarOpen', true) | ||||
|               </Collapsible> | ||||
|  | ||||
|               <!-- Views --> | ||||
|               <Collapsible class="group/collapsible" defaultOpen> | ||||
|               <Collapsible class="group/collapsible" defaultOpen v-model:open="viewInboxOpen"> | ||||
|                 <SidebarMenuItem> | ||||
|                   <CollapsibleTrigger as-child> | ||||
|                   <CollapsibleTrigger asChild> | ||||
|                     <SidebarMenuButton asChild> | ||||
|                       <router-link to="#"> | ||||
|                       <router-link to="#" class="group/item !p-2"> | ||||
|                         <!-- <SlidersHorizontal /> --> | ||||
|                         <span>Views</span> | ||||
|                         <span> | ||||
|                           {{ t('globals.terms.view', 2) }} | ||||
|                         </span> | ||||
|                         <div> | ||||
|                           <Plus | ||||
|                             size="18" | ||||
|                             @click.stop="openCreateViewDialog" | ||||
|                             class="rounded-lg cursor-pointer opacity-0 transition-all duration-200 group-hover:opacity-100 hover:bg-gray-200 hover:shadow-sm text-gray-600 hover:text-gray-800 transform hover:scale-105 active:scale-100 p-1" | ||||
|                             class="rounded cursor-pointer opacity-0 transition-all duration-200 group-hover/item:opacity-100 hover:bg-gray-200 hover:shadow-sm text-gray-600 hover:text-gray-800 transform hover:scale-105 active:scale-100 p-1" | ||||
|                           /> | ||||
|                         </div> | ||||
|                         <ChevronRight | ||||
|                           class="ml-auto transition-transform duration-200 group-data-[state=open]/collapsible:rotate-90" | ||||
|                           v-if="userViews.length" | ||||
|                         /> | ||||
|                       </router-link> | ||||
|                     </SidebarMenuButton> | ||||
|                   </CollapsibleTrigger> | ||||
|  | ||||
|                   <SidebarMenuAction> | ||||
|                     <ChevronRight | ||||
|                       class="transition-transform duration-200 group-data-[state=open]/collapsible:rotate-90" | ||||
|                       v-if="userViews.length" | ||||
|                     /> | ||||
|                   </SidebarMenuAction> | ||||
|  | ||||
|                   <CollapsibleContent> | ||||
|                     <SidebarMenuSub v-for="view in userViews" :key="view.id"> | ||||
|                       <SidebarMenuSubItem> | ||||
| @@ -334,26 +478,25 @@ const sidebarOpen = useStorage('mainSidebarOpen', true) | ||||
|                           :isActive="route.params.viewID == view.id" | ||||
|                           asChild | ||||
|                         > | ||||
|                           <router-link :to="{ name: 'view-inbox', params: { viewID: view.id } }"> | ||||
|                             <span class="break-all w-24">{{ view.name }}</span> | ||||
|                           </router-link> | ||||
|                           <a href="#" @click.prevent="navigateToViewInbox(view.id)"> | ||||
|                             <span class="break-words w-32 truncate">{{ view.name }}</span> | ||||
|                             <SidebarMenuAction :showOnHover="true" class="mr-3"> | ||||
|                               <DropdownMenu> | ||||
|                                 <DropdownMenuTrigger asChild> | ||||
|                                   <EllipsisVertical /> | ||||
|                                 </DropdownMenuTrigger> | ||||
|                                 <DropdownMenuContent> | ||||
|                                   <DropdownMenuItem @click="() => editView(view)"> | ||||
|                                     <span>{{ t('globals.messages.edit') }}</span> | ||||
|                                   </DropdownMenuItem> | ||||
|                                   <DropdownMenuItem @click="() => deleteView(view)"> | ||||
|                                     <span>{{ t('globals.messages.delete') }}</span> | ||||
|                                   </DropdownMenuItem> | ||||
|                                 </DropdownMenuContent> | ||||
|                               </DropdownMenu> | ||||
|                             </SidebarMenuAction> | ||||
|                           </a> | ||||
|                         </SidebarMenuButton> | ||||
|  | ||||
|                         <SidebarMenuAction> | ||||
|                           <DropdownMenu> | ||||
|                             <DropdownMenuTrigger asChild> | ||||
|                               <EllipsisVertical /> | ||||
|                             </DropdownMenuTrigger> | ||||
|                             <DropdownMenuContent> | ||||
|                               <DropdownMenuItem @click="() => editView(view)"> | ||||
|                                 <span>Edit</span> | ||||
|                               </DropdownMenuItem> | ||||
|                               <DropdownMenuItem @click="() => deleteView(view)"> | ||||
|                                 <span>Delete</span> | ||||
|                               </DropdownMenuItem> | ||||
|                             </DropdownMenuContent> | ||||
|                           </DropdownMenu> | ||||
|                         </SidebarMenuAction> | ||||
|                       </SidebarMenuSubItem> | ||||
|                     </SidebarMenuSub> | ||||
|                   </CollapsibleContent> | ||||
|   | ||||
| @@ -1,82 +1,139 @@ | ||||
| <template> | ||||
|     <DropdownMenu> | ||||
|         <DropdownMenuTrigger as-child> | ||||
|             <SidebarMenuButton size="lg" | ||||
|                 class="data-[state=open]:bg-sidebar-accent data-[state=open]:text-sidebar-accent-foreground p-0"> | ||||
|                 <Avatar class="h-8 w-8 rounded-lg"> | ||||
|                     <AvatarImage :src="userStore.avatar" alt="Abhinav" /> | ||||
|                     <AvatarFallback class="rounded-lg"> | ||||
|                         {{ userStore.getInitials }} | ||||
|                     </AvatarFallback> | ||||
|                 </Avatar> | ||||
|                 <div class="grid flex-1 text-left text-sm leading-tight"> | ||||
|                     <span class="truncate font-semibold">{{ userStore.getFullName }}</span> | ||||
|                     <span class="truncate text-xs">{{ userStore.email }}</span> | ||||
|                 </div> | ||||
|                 <ChevronsUpDown class="ml-auto size-4" /> | ||||
|             </SidebarMenuButton> | ||||
|         </DropdownMenuTrigger> | ||||
|         <DropdownMenuContent class="w-[--radix-dropdown-menu-trigger-width] min-w-56 rounded-lg" side="bottom" | ||||
|             :side-offset="4"> | ||||
|             <DropdownMenuLabel class="p-0 font-normal"> | ||||
|                 <div class="flex items-center gap-2 px-1 py-1.5 text-left text-sm"> | ||||
|                     <Avatar class="h-8 w-8 rounded-lg"> | ||||
|                         <AvatarImage :src="userStore.avatar" alt="Abhinav" /> | ||||
|                         <AvatarFallback class="rounded-lg"> | ||||
|                             {{ userStore.getInitials }} | ||||
|                         </AvatarFallback> | ||||
|                     </Avatar> | ||||
|                     <div class="grid flex-1 text-left text-sm leading-tight"> | ||||
|                         <span class="truncate font-semibold">{{ userStore.getFullName }}</span> | ||||
|                         <span class="truncate text-xs">{{ userStore.email }}</span> | ||||
|                     </div> | ||||
|                 </div> | ||||
|             </DropdownMenuLabel> | ||||
|             <DropdownMenuSeparator /> | ||||
|             <DropdownMenuGroup> | ||||
|                 <DropdownMenuItem> | ||||
|                     <router-link to="/account" class="flex items-center"> | ||||
|                         <CircleUserRound size="18" class="mr-2" /> | ||||
|                         Account | ||||
|                     </router-link> | ||||
|                 </DropdownMenuItem> | ||||
|             </DropdownMenuGroup> | ||||
|             <DropdownMenuSeparator /> | ||||
|             <DropdownMenuItem @click="logout"> | ||||
|                 <LogOut size="18" class="mr-2" /> | ||||
|                 Log out | ||||
|             </DropdownMenuItem> | ||||
|         </DropdownMenuContent> | ||||
|     </DropdownMenu> | ||||
|   <DropdownMenu> | ||||
|     <DropdownMenuTrigger as-child> | ||||
|       <SidebarMenuButton | ||||
|         size="md" | ||||
|         class="p-0" | ||||
|       > | ||||
|         <Avatar class="h-8 w-8 rounded relative overflow-visible"> | ||||
|           <AvatarImage :src="userStore.avatar" alt="U" class="rounded" /> | ||||
|           <AvatarFallback class="rounded"> | ||||
|             {{ userStore.getInitials }} | ||||
|           </AvatarFallback> | ||||
|           <div | ||||
|             class="absolute bottom-0 right-0 h-2.5 w-2.5 rounded-full border border-background" | ||||
|             :class="{ | ||||
|               'bg-green-500': userStore.user.availability_status === 'online', | ||||
|               'bg-amber-500': | ||||
|                 userStore.user.availability_status === 'away' || | ||||
|                 userStore.user.availability_status === 'away_manual' || | ||||
|                 userStore.user.availability_status === 'away_and_reassigning', | ||||
|               'bg-gray-400': userStore.user.availability_status === 'offline' | ||||
|             }" | ||||
|           ></div> | ||||
|         </Avatar> | ||||
|         <div class="grid flex-1 text-left text-sm leading-tight"> | ||||
|           <span class="truncate font-semibold">{{ userStore.getFullName }}</span> | ||||
|           <span class="truncate text-xs">{{ userStore.email }}</span> | ||||
|         </div> | ||||
|         <ChevronsUpDown class="ml-auto size-4" /> | ||||
|       </SidebarMenuButton> | ||||
|     </DropdownMenuTrigger> | ||||
|     <DropdownMenuContent | ||||
|       class="w-[--radix-dropdown-menu-trigger-width] min-w-56" | ||||
|       side="bottom" | ||||
|       :side-offset="4" | ||||
|     > | ||||
|       <DropdownMenuLabel class="font-normal space-y-2 px-2"> | ||||
|         <!-- User header --> | ||||
|         <div class="flex items-center gap-2 py-1.5 text-left text-sm"> | ||||
|           <Avatar class="h-8 w-8 rounded"> | ||||
|             <AvatarImage :src="userStore.avatar" alt="U" /> | ||||
|             <AvatarFallback class="rounded"> | ||||
|               {{ userStore.getInitials }} | ||||
|             </AvatarFallback> | ||||
|           </Avatar> | ||||
|           <div class="flex-1 flex flex-col leading-tight"> | ||||
|             <span class="truncate font-semibold">{{ userStore.getFullName }}</span> | ||||
|             <span class="truncate text-xs text-muted-foreground">{{ userStore.email }}</span> | ||||
|           </div> | ||||
|         </div> | ||||
|  | ||||
|         <div class="space-y-2"> | ||||
|           <!-- Dark-mode toggle --> | ||||
|           <div class="flex items-center justify-between text-sm"> | ||||
|             <div class="flex items-center gap-2"> | ||||
|               <Moon v-if="mode === 'dark'" size="16" class="text-muted-foreground" /> | ||||
|               <Sun v-else size="16" class="text-muted-foreground" /> | ||||
|               <span class="text-muted-foreground">{{ t('navigation.darkMode') }}</span> | ||||
|             </div> | ||||
|             <Switch | ||||
|               :checked="mode === 'dark'" | ||||
|               @update:checked="(val) => (mode = val ? 'dark' : 'light')" | ||||
|             /> | ||||
|           </div> | ||||
|  | ||||
|           <div class="border-t border-gray-200 dark:border-gray-700 pt-3 space-y-3"> | ||||
|             <!-- Away toggle --> | ||||
|             <div class="flex items-center justify-between text-sm"> | ||||
|               <span class="text-muted-foreground">{{ t('navigation.away') }}</span> | ||||
|               <Switch | ||||
|                 :checked=" | ||||
|                   ['away_manual', 'away_and_reassigning'].includes( | ||||
|                     userStore.user.availability_status | ||||
|                   ) | ||||
|                 " | ||||
|                 @update:checked=" | ||||
|                   (val) => userStore.updateUserAvailability(val ? 'away_manual' : 'online') | ||||
|                 " | ||||
|               /> | ||||
|             </div> | ||||
|             <!-- Reassign toggle --> | ||||
|             <div class="flex items-center justify-between text-sm"> | ||||
|               <span class="text-muted-foreground">{{ t('navigation.reassignReplies') }}</span> | ||||
|               <Switch | ||||
|                 :checked="userStore.user.availability_status === 'away_and_reassigning'" | ||||
|                 @update:checked=" | ||||
|                   (val) => | ||||
|                     userStore.updateUserAvailability(val ? 'away_and_reassigning' : 'away_manual') | ||||
|                 " | ||||
|               /> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </DropdownMenuLabel> | ||||
|       <DropdownMenuSeparator /> | ||||
|       <DropdownMenuGroup> | ||||
|         <DropdownMenuItem @click.prevent="router.push({ name: 'account' })"> | ||||
|           <CircleUserRound size="18" class="mr-2" /> | ||||
|           {{ t('globals.terms.account') }} | ||||
|         </DropdownMenuItem> | ||||
|       </DropdownMenuGroup> | ||||
|       <DropdownMenuSeparator /> | ||||
|       <DropdownMenuItem @click="logout"> | ||||
|         <LogOut size="18" class="mr-2" /> | ||||
|         {{ t('navigation.logout') }} | ||||
|       </DropdownMenuItem> | ||||
|     </DropdownMenuContent> | ||||
|   </DropdownMenu> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { useI18n } from 'vue-i18n' | ||||
| import { | ||||
|     DropdownMenu, | ||||
|     DropdownMenuContent, | ||||
|     DropdownMenuGroup, | ||||
|     DropdownMenuItem, | ||||
|     DropdownMenuLabel, | ||||
|     DropdownMenuSeparator, | ||||
|     DropdownMenuTrigger, | ||||
|   DropdownMenu, | ||||
|   DropdownMenuContent, | ||||
|   DropdownMenuGroup, | ||||
|   DropdownMenuItem, | ||||
|   DropdownMenuLabel, | ||||
|   DropdownMenuSeparator, | ||||
|   DropdownMenuTrigger | ||||
| } from '@/components/ui/dropdown-menu' | ||||
| import { | ||||
|     SidebarMenuButton, | ||||
| } from '@/components/ui/sidebar' | ||||
| import { | ||||
|     Avatar, | ||||
|     AvatarFallback, | ||||
|     AvatarImage, | ||||
| } from '@/components/ui/avatar' | ||||
| import { | ||||
|     ChevronsUpDown, | ||||
|     CircleUserRound, | ||||
|     LogOut, | ||||
| } from 'lucide-vue-next' | ||||
| import { SidebarMenuButton } from '@/components/ui/sidebar' | ||||
| import { Avatar, AvatarFallback, AvatarImage } from '@/components/ui/avatar' | ||||
| import { Switch } from '@/components/ui/switch' | ||||
| import { ChevronsUpDown, CircleUserRound, LogOut, Moon, Sun } from 'lucide-vue-next' | ||||
| import { useUserStore } from '@/stores/user' | ||||
| import { useRouter } from 'vue-router' | ||||
|  | ||||
| import { useColorMode } from '@vueuse/core' | ||||
|  | ||||
| const mode = useColorMode() | ||||
| const userStore = useUserStore() | ||||
| const router = useRouter() | ||||
| const { t } = useI18n() | ||||
|  | ||||
| const logout = () => { | ||||
|     window.location.href = '/logout' | ||||
|   window.location.href = '/logout' | ||||
| } | ||||
| </script> | ||||
| </script> | ||||
|   | ||||
| @@ -1,53 +1,112 @@ | ||||
| <template> | ||||
|     <table class="min-w-full divide-y divide-gray-200"> | ||||
|         <thead class="bg-gray-50"> | ||||
|             <tr> | ||||
|                 <th v-for="(header, index) in headers" :key="index" scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                     {{ header }} | ||||
|                 </th> | ||||
|                 <th scope="col" class="relative px-6 py-3"></th> | ||||
|             </tr> | ||||
|         </thead> | ||||
|         <tbody class="bg-white divide-y divide-gray-200"> | ||||
|             <tr v-for="(item, index) in data" :key="index"> | ||||
|                 <td v-for="key in keys" :key="key" class="px-6 py-4 whitespace-nowrap text-sm font-medium text-gray-900"> | ||||
|                     {{ item[key] }} | ||||
|                 </td> | ||||
|                 <td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500"> | ||||
|                     <Button size="xs" variant="ghost" @click.prevent="deleteItem(item)"> | ||||
|                         <Trash2 class="h-4 w-4" /> | ||||
|                     </Button> | ||||
|                 </td> | ||||
|             </tr> | ||||
|         </tbody> | ||||
|     </table> | ||||
|   <table class="min-w-full table-fixed divide-y divide-border"> | ||||
|     <thead class="bg-muted"> | ||||
|       <tr> | ||||
|         <th | ||||
|           v-for="(header, index) in headers" | ||||
|           :key="index" | ||||
|           scope="col" | ||||
|           class="px-6 py-3 text-left text-xs font-medium text-muted-foreground uppercase tracking-wider" | ||||
|         > | ||||
|           {{ header }} | ||||
|         </th> | ||||
|         <th v-if="showDelete" scope="col" class="relative px-6 py-3"></th> | ||||
|       </tr> | ||||
|     </thead> | ||||
|     <tbody class="bg-background divide-y divide-border"> | ||||
|       <!-- Loading State --> | ||||
|       <template v-if="loading"> | ||||
|         <tr v-for="i in skeletonRows" :key="`skeleton-${i}`" class="hover:bg-accent"> | ||||
|           <td | ||||
|             v-for="(header, index) in headers" | ||||
|             :key="`skeleton-cell-${index}`" | ||||
|             class="px-6 py-3 text-sm font-medium text-foreground whitespace-normal break-words" | ||||
|           > | ||||
|             <Skeleton class="h-4 w-[85%]" /> | ||||
|           </td> | ||||
|           <td v-if="showDelete" class="px-6 py-4 text-sm text-muted-foreground"> | ||||
|             <Skeleton class="h-8 w-8 rounded" /> | ||||
|           </td> | ||||
|         </tr> | ||||
|       </template> | ||||
|  | ||||
|       <!-- No Results State --> | ||||
|       <template v-else-if="data.length === 0"> | ||||
|         <tr> | ||||
|           <td :colspan="headers.length + (showDelete ? 1 : 0)" class="px-6 py-12 text-center"> | ||||
|             <div class="flex flex-col items-center space-y-4"> | ||||
|               <span class="text-md text-muted-foreground"> | ||||
|                 {{ | ||||
|                   $t('globals.messages.noResults', { | ||||
|                     name: $t('globals.terms.result', 2).toLowerCase() | ||||
|                   }) | ||||
|                 }} | ||||
|               </span> | ||||
|             </div> | ||||
|           </td> | ||||
|         </tr> | ||||
|       </template> | ||||
|  | ||||
|       <!-- Data Rows --> | ||||
|       <template v-else> | ||||
|         <tr v-for="(item, index) in data" :key="index" class="hover:bg-accent"> | ||||
|           <td | ||||
|             v-for="key in keys" | ||||
|             :key="key" | ||||
|             class="px-6 py-4 text-sm font-medium text-foreground whitespace-normal break-words" | ||||
|           > | ||||
|             {{ item[key] }} | ||||
|           </td> | ||||
|           <td v-if="showDelete" class="px-6 py-4 text-sm text-muted-foreground"> | ||||
|             <Button size="xs" variant="ghost" @click.prevent="deleteItem(item)"> | ||||
|               <Trash2 class="h-4 w-4" /> | ||||
|             </Button> | ||||
|           </td> | ||||
|         </tr> | ||||
|       </template> | ||||
|     </tbody> | ||||
|   </table> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { Trash2 } from 'lucide-vue-next'; | ||||
| import { defineProps, defineEmits } from 'vue'; | ||||
| import { Trash2 } from 'lucide-vue-next' | ||||
| import { defineEmits } from 'vue' | ||||
| import { Button } from '@/components/ui/button' | ||||
| import { Skeleton } from '@/components/ui/skeleton' | ||||
|  | ||||
| defineProps({ | ||||
|     headers: { | ||||
|         type: Array, | ||||
|         required: true, | ||||
|         default: () => [] | ||||
|     }, | ||||
|     keys: { | ||||
|         type: Array, | ||||
|         required: true, | ||||
|         default: () => [] | ||||
|     }, | ||||
|     data: { | ||||
|         type: Array, | ||||
|         required: true, | ||||
|         default: () => [] | ||||
|     } | ||||
| }); | ||||
|   headers: { | ||||
|     type: Array, | ||||
|     required: true, | ||||
|     default: () => [] | ||||
|   }, | ||||
|   keys: { | ||||
|     type: Array, | ||||
|     required: true, | ||||
|     default: () => [] | ||||
|   }, | ||||
|   data: { | ||||
|     type: Array, | ||||
|     required: true, | ||||
|     default: () => [] | ||||
|   }, | ||||
|   showDelete: { | ||||
|     type: Boolean, | ||||
|     default: true | ||||
|   }, | ||||
|   loading: { | ||||
|     type: Boolean, | ||||
|     default: false | ||||
|   }, | ||||
|   skeletonRows: { | ||||
|     type: Number, | ||||
|     default: 5 | ||||
|   } | ||||
| }) | ||||
|  | ||||
| const emit = defineEmits(['deleteItem']); | ||||
| const emit = defineEmits(['deleteItem']) | ||||
|  | ||||
| function deleteItem(item) { | ||||
|     emit('deleteItem', item); | ||||
|   emit('deleteItem', item) | ||||
| } | ||||
| </script> | ||||
| </script> | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
| import { AvatarImage } from 'radix-vue' | ||||
|  | ||||
| const props = defineProps({ | ||||
|   src: { type: String, required: true }, | ||||
|   src: { type: String, required: false, default: '' }, | ||||
|   asChild: { type: Boolean, required: false }, | ||||
|   as: { type: null, required: false } | ||||
| }) | ||||
|   | ||||
							
								
								
									
										59
									
								
								frontend/src/components/ui/avatar/AvatarUpload.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								frontend/src/components/ui/avatar/AvatarUpload.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | ||||
| <template> | ||||
|   <div class="relative group w-28 h-28 cursor-pointer" @click="triggerFileInput"> | ||||
|     <Avatar class="size-28"> | ||||
|       <AvatarImage :src="src || ''" /> | ||||
|       <AvatarFallback>{{ initials }}</AvatarFallback> | ||||
|     </Avatar> | ||||
|  | ||||
|     <!-- Hover Overlay --> | ||||
|     <div | ||||
|       class="absolute inset-0 bg-black bg-opacity-50 flex items-center justify-center opacity-0 group-hover:opacity-100 transition-opacity cursor-pointer rounded-full" | ||||
|     > | ||||
|       <span class="text-white font-semibold">{{ label }}</span> | ||||
|     </div> | ||||
|  | ||||
|     <!-- Delete Icon --> | ||||
|     <X | ||||
|       class="absolute top-1 right-1 rounded-full p-1 shadow-md z-10 opacity-0 group-hover:opacity-100 transition-opacity" | ||||
|       size="20" | ||||
|       @click.stop="emit('remove')" | ||||
|       v-if="src" | ||||
|     /> | ||||
|  | ||||
|     <!-- File Input --> | ||||
|     <input | ||||
|       ref="fileInput" | ||||
|       type="file" | ||||
|       class="hidden" | ||||
|       accept="image/png,image/jpeg,image/jpg" | ||||
|       @change="handleChange" | ||||
|     /> | ||||
|   </div> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { ref } from 'vue' | ||||
| import { Avatar, AvatarImage, AvatarFallback } from '@/components/ui/avatar' | ||||
| import { X } from 'lucide-vue-next' | ||||
|  | ||||
| defineProps({ | ||||
|   src: String, | ||||
|   initials: String, | ||||
|   label: { | ||||
|     type: String, | ||||
|     default: 'Upload' | ||||
|   } | ||||
| }) | ||||
|  | ||||
| const emit = defineEmits(['upload', 'remove']) | ||||
| const fileInput = ref(null) | ||||
|  | ||||
| function triggerFileInput() { | ||||
|   fileInput.value?.click() | ||||
| } | ||||
|  | ||||
| function handleChange(e) { | ||||
|   const file = e.target.files[0] | ||||
|   if (file) emit('upload', file) | ||||
| } | ||||
| </script> | ||||
| @@ -3,6 +3,7 @@ import { cva } from 'class-variance-authority' | ||||
| export { default as Avatar } from './Avatar.vue' | ||||
| export { default as AvatarImage } from './AvatarImage.vue' | ||||
| export { default as AvatarFallback } from './AvatarFallback.vue' | ||||
| export { default as AvatarUpload } from './AvatarUpload.vue' | ||||
|  | ||||
| export const avatarVariant = cva( | ||||
|   'inline-flex items-center justify-center font-normal text-foreground select-none shrink-0 bg-secondary overflow-hidden', | ||||
|   | ||||
| @@ -1,24 +1,16 @@ | ||||
| <script setup> | ||||
| import { Primitive } from 'radix-vue' | ||||
| import { buttonVariants } from '.' | ||||
| import { Primitive } from 'reka-ui' | ||||
| import { cn } from '@/lib/utils' | ||||
| import { ref, computed } from 'vue' | ||||
|  | ||||
| import { buttonVariants } from '.' | ||||
| import { Loader2 } from 'lucide-vue-next' | ||||
| const props = defineProps({ | ||||
|   variant: { type: null, required: false }, | ||||
|   size: { type: null, required: false }, | ||||
|   class: { type: null, required: false }, | ||||
|   asChild: { type: Boolean, required: false }, | ||||
|   as: { type: null, required: false, default: 'button' }, | ||||
|   isLoading: { type: Boolean, required: false, default: false } | ||||
| }) | ||||
|  | ||||
| const isDisabled = ref(false) | ||||
|  | ||||
| const computedClass = computed(() => { | ||||
|   return cn(buttonVariants({ variant: props.variant, size: props.size }), props.class, { | ||||
|     'cursor-not-allowed opacity-50': props.isLoading || isDisabled.value | ||||
|   }) | ||||
|   isLoading: { type: Boolean, required: false, default: false }, | ||||
|   disabled: { type: Boolean, required: false, default: false } | ||||
| }) | ||||
| </script> | ||||
|  | ||||
| @@ -26,14 +18,22 @@ const computedClass = computed(() => { | ||||
|   <Primitive | ||||
|     :as="as" | ||||
|     :as-child="asChild" | ||||
|     :class="computedClass" | ||||
|     :disabled="isLoading || isDisabled" | ||||
|     :class=" | ||||
|       cn( | ||||
|         buttonVariants({ variant, size }), | ||||
|         'relative', | ||||
|         { 'text-transparent': isLoading }, | ||||
|         props.class | ||||
|       ) | ||||
|     " | ||||
|     :disabled="isLoading || disabled" | ||||
|   > | ||||
|     <span v-if="isLoading" class="dot-loader"> | ||||
|       <span class="dot"></span> | ||||
|       <span class="dot"></span> | ||||
|       <span class="dot"></span> | ||||
|     <slot /> | ||||
|     <span | ||||
|       v-if="isLoading" | ||||
|       class="absolute inset-0 flex items-center justify-center pointer-events-none text-background" | ||||
|     > | ||||
|       <Loader2 class="h-5 w-5 animate-spin" /> | ||||
|     </span> | ||||
|     <slot v-else /> | ||||
|   </Primitive> | ||||
| </template> | ||||
|   | ||||
| @@ -1,31 +1,34 @@ | ||||
| import { cva } from 'class-variance-authority' | ||||
| import { cva } from 'class-variance-authority'; | ||||
|  | ||||
| export { default as Button } from './Button.vue' | ||||
| export { default as Button } from './Button.vue'; | ||||
|  | ||||
| export const buttonVariants = cva( | ||||
|   'inline-flex items-center justify-center whitespace-nowrap rounded-md text-sm font-medium transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50', | ||||
|   'inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:size-4 [&_svg]:shrink-0', | ||||
|   { | ||||
|     variants: { | ||||
|       variant: { | ||||
|         default: 'bg-primary text-primary-foreground shadow hover:bg-primary/90', | ||||
|         destructive: 'bg-destructive text-destructive-foreground shadow-sm hover:bg-destructive/90', | ||||
|         default: | ||||
|           'bg-primary text-primary-foreground shadow hover:bg-primary/90', | ||||
|         destructive: | ||||
|           'bg-destructive text-destructive-foreground shadow-sm hover:bg-destructive/90', | ||||
|         outline: | ||||
|           'border border-input bg-background shadow-sm hover:bg-accent hover:text-accent-foreground', | ||||
|         secondary: 'bg-secondary text-secondary-foreground shadow-sm hover:bg-secondary/80', | ||||
|         secondary: | ||||
|           'bg-secondary text-secondary-foreground shadow-sm hover:bg-secondary/80', | ||||
|         ghost: 'hover:bg-accent hover:text-accent-foreground', | ||||
|         link: 'text-primary underline-offset-4 hover:underline' | ||||
|         link: 'text-primary underline-offset-4 hover:underline', | ||||
|       }, | ||||
|       size: { | ||||
|         default: 'h-9 px-4 py-2', | ||||
|         xs: 'h-7 rounded px-2', | ||||
|         sm: 'h-8 rounded-md px-3 text-xs', | ||||
|         lg: 'h-10 rounded-md px-8', | ||||
|         icon: 'h-9 w-9' | ||||
|       } | ||||
|         icon: 'h-9 w-9', | ||||
|       }, | ||||
|     }, | ||||
|     defaultVariants: { | ||||
|       variant: 'default', | ||||
|       size: 'default' | ||||
|     } | ||||
|   } | ||||
| ) | ||||
|       size: 'default', | ||||
|     }, | ||||
|   }, | ||||
| ); | ||||
|   | ||||
| @@ -1,94 +0,0 @@ | ||||
| <script setup> | ||||
| import { VisDonut, VisSingleContainer } from '@unovis/vue' | ||||
| import { Donut } from '@unovis/ts' | ||||
| import { computed, ref } from 'vue' | ||||
| import { useMounted } from '@vueuse/core' | ||||
| import { ChartSingleTooltip, defaultColors } from '@/components/ui/chart' | ||||
| import { cn } from '@/lib/utils' | ||||
|  | ||||
| const props = defineProps({ | ||||
|   data: { type: Array, required: true }, | ||||
|   colors: { type: Array, required: false }, | ||||
|   index: { type: null, required: true }, | ||||
|   margin: { | ||||
|     type: null, | ||||
|     required: false, | ||||
|     default: () => ({ top: 0, bottom: 0, left: 0, right: 0 }) | ||||
|   }, | ||||
|   showLegend: { type: Boolean, required: false, default: true }, | ||||
|   showTooltip: { type: Boolean, required: false, default: true }, | ||||
|   filterOpacity: { type: Number, required: false, default: 0.2 }, | ||||
|   category: { type: String, required: true }, | ||||
|   type: { type: String, required: false, default: 'donut' }, | ||||
|   sortFunction: { type: Function, required: false, default: () => undefined }, | ||||
|   valueFormatter: { type: Function, required: false, default: (tick) => `${tick}` }, | ||||
|   customTooltip: { type: null, required: false } | ||||
| }) | ||||
|  | ||||
| const category = computed(() => props.category) | ||||
| const index = computed(() => props.index) | ||||
|  | ||||
| const isMounted = useMounted() | ||||
| const activeSegmentKey = ref() | ||||
| const colors = computed(() => | ||||
|   props.colors?.length | ||||
|     ? props.colors | ||||
|     : defaultColors(props.data.filter((d) => d[props.category]).filter(Boolean).length) | ||||
| ) | ||||
| const legendItems = computed(() => | ||||
|   props.data.map((item, i) => ({ | ||||
|     name: item[props.index], | ||||
|     color: colors.value[i], | ||||
|     inactive: false | ||||
|   })) | ||||
| ) | ||||
|  | ||||
| const totalValue = computed(() => | ||||
|   props.data.reduce((prev, curr) => { | ||||
|     return prev + curr[props.category] | ||||
|   }, 0) | ||||
| ) | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
|   <div :class="cn('w-full h-48 flex flex-col items-end', $attrs.class ?? '')"> | ||||
|     <VisSingleContainer | ||||
|       :style="{ height: isMounted ? '100%' : 'auto' }" | ||||
|       :margin="{ left: 20, right: 20 }" | ||||
|       :data="data" | ||||
|     > | ||||
|       <ChartSingleTooltip | ||||
|         :selector="Donut.selectors.segment" | ||||
|         :index="category" | ||||
|         :items="legendItems" | ||||
|         :value-formatter="valueFormatter" | ||||
|         :custom-tooltip="customTooltip" | ||||
|       /> | ||||
|  | ||||
|       <VisDonut | ||||
|         :value="(d) => d[category]" | ||||
|         :sort-function="sortFunction" | ||||
|         :color="colors" | ||||
|         :arc-width="type === 'donut' ? 20 : 0" | ||||
|         :show-background="false" | ||||
|         :central-label="type === 'donut' ? valueFormatter(totalValue) : ''" | ||||
|         :events="{ | ||||
|           [Donut.selectors.segment]: { | ||||
|             click: (d, ev, i, elements) => { | ||||
|               if (d?.data?.[index] === activeSegmentKey) { | ||||
|                 activeSegmentKey = undefined | ||||
|                 elements.forEach((el) => (el.style.opacity = '1')) | ||||
|               } else { | ||||
|                 activeSegmentKey = d?.data?.[index] | ||||
|                 elements.forEach((el) => (el.style.opacity = `${filterOpacity}`)) | ||||
|                 elements[i].style.opacity = '1' | ||||
|               } | ||||
|             } | ||||
|           } | ||||
|         }" | ||||
|       /> | ||||
|  | ||||
|       <slot /> | ||||
|     </VisSingleContainer> | ||||
|   </div> | ||||
| </template> | ||||
| @@ -1 +0,0 @@ | ||||
| export { default as DonutChart } from './DonutChart.vue' | ||||
| @@ -5,7 +5,7 @@ | ||||
|         variant="outline" | ||||
|         role="combobox" | ||||
|         :aria-expanded="open" | ||||
|         class="w-full justify-between" | ||||
|         :class="['w-full justify-between', buttonClass]" | ||||
|       > | ||||
|         <slot name="selected" :selected="selectedItem">{{ selectedLabel }}</slot> | ||||
|         <CaretSortIcon class="ml-2 h-4 w-4 shrink-0 opacity-50" /> | ||||
| @@ -58,7 +58,11 @@ const props = defineProps({ | ||||
|     required: true | ||||
|   }, | ||||
|   placeholder: String, | ||||
|   defaultLabel: String | ||||
|   defaultLabel: String, | ||||
|   buttonClass: { | ||||
|     type: String, | ||||
|     default: '' | ||||
|   } | ||||
| }) | ||||
|  | ||||
| const emit = defineEmits(['select']) | ||||
|   | ||||
							
								
								
									
										116
									
								
								frontend/src/components/ui/date-filter/DateFilter.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										116
									
								
								frontend/src/components/ui/date-filter/DateFilter.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,116 @@ | ||||
| <template> | ||||
|   <div class="flex items-center gap-2"> | ||||
|     <Select v-model="selectedDays" @update:model-value="handleFilterChange"> | ||||
|       <SelectTrigger class="w-[140px] h-8 text-xs"> | ||||
|         <SelectValue | ||||
|           :placeholder=" | ||||
|             t('globals.messages.select', { | ||||
|               name: t('globals.terms.day', 2) | ||||
|             }) | ||||
|           " | ||||
|         /> | ||||
|       </SelectTrigger> | ||||
|       <SelectContent class="text-xs"> | ||||
|         <SelectItem value="0">{{ $t('globals.terms.today') }}</SelectItem> | ||||
|         <SelectItem value="1"> | ||||
|           {{ | ||||
|             $t('globals.messages.lastNItems', { | ||||
|               n: 1, | ||||
|               name: t('globals.terms.day', 1).toLowerCase() | ||||
|             }) | ||||
|           }} | ||||
|         </SelectItem> | ||||
|         <SelectItem value="2"> | ||||
|           {{ | ||||
|             $t('globals.messages.lastNItems', { | ||||
|               n: 2, | ||||
|               name: t('globals.terms.day', 2).toLowerCase() | ||||
|             }) | ||||
|           }} | ||||
|         </SelectItem> | ||||
|         <SelectItem value="7"> | ||||
|           {{ | ||||
|             $t('globals.messages.lastNItems', { | ||||
|               n: 7, | ||||
|               name: t('globals.terms.day', 2).toLowerCase() | ||||
|             }) | ||||
|           }} | ||||
|         </SelectItem> | ||||
|         <SelectItem value="30"> | ||||
|           {{ | ||||
|             $t('globals.messages.lastNItems', { | ||||
|               n: 30, | ||||
|               name: t('globals.terms.day', 2).toLowerCase() | ||||
|             }) | ||||
|           }} | ||||
|         </SelectItem> | ||||
|         <SelectItem value="90"> | ||||
|           {{ | ||||
|             $t('globals.messages.lastNItems', { | ||||
|               n: 90, | ||||
|               name: t('globals.terms.day', 2).toLowerCase() | ||||
|             }) | ||||
|           }} | ||||
|         </SelectItem> | ||||
|         <SelectItem value="custom"> | ||||
|           {{ | ||||
|             $t('globals.messages.custom', { | ||||
|               name: t('globals.terms.day', 2).toLowerCase() | ||||
|             }) | ||||
|           }} | ||||
|         </SelectItem> | ||||
|       </SelectContent> | ||||
|     </Select> | ||||
|     <div v-if="selectedDays === 'custom'" class="flex items-center gap-2"> | ||||
|       <Input | ||||
|         v-model="customDaysInput" | ||||
|         type="number" | ||||
|         min="1" | ||||
|         max="365" | ||||
|         class="w-20 h-8" | ||||
|         @blur="handleCustomDaysChange" | ||||
|         @keyup.enter="handleCustomDaysChange" | ||||
|       /> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
|  | ||||
| <script setup> | ||||
| import { ref } from 'vue' | ||||
| import { useI18n } from 'vue-i18n' | ||||
| import { | ||||
|   Select, | ||||
|   SelectContent, | ||||
|   SelectItem, | ||||
|   SelectTrigger, | ||||
|   SelectValue | ||||
| } from '@/components/ui/select' | ||||
| import { Input } from '@/components/ui/input' | ||||
|  | ||||
| const { t } = useI18n() | ||||
|  | ||||
| const emit = defineEmits(['filterChange']) | ||||
| const selectedDays = ref('30') | ||||
| const customDaysInput = ref('') | ||||
|  | ||||
| const handleFilterChange = (value) => { | ||||
|   if (value === 'custom') { | ||||
|     customDaysInput.value = '30' | ||||
|     emit('filterChange', 30) | ||||
|   } else { | ||||
|     emit('filterChange', parseInt(value)) | ||||
|   } | ||||
| } | ||||
|  | ||||
| const handleCustomDaysChange = () => { | ||||
|   const days = parseInt(customDaysInput.value) | ||||
|   if (days && days > 0 && days <= 365) { | ||||
|     emit('filterChange', days) | ||||
|   } else { | ||||
|     customDaysInput.value = '30' | ||||
|     emit('filterChange', 30) | ||||
|   } | ||||
| } | ||||
|  | ||||
| handleFilterChange(selectedDays.value) | ||||
| </script> | ||||
							
								
								
									
										1
									
								
								frontend/src/components/ui/date-filter/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								frontend/src/components/ui/date-filter/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| export { default as DateFilter } from './DateFilter.vue' | ||||
| @@ -1,19 +1,19 @@ | ||||
| <script setup> | ||||
| import { useVModel } from '@vueuse/core' | ||||
| import { cn } from '@/lib/utils' | ||||
| import { useVModel } from '@vueuse/core'; | ||||
| import { cn } from '@/lib/utils'; | ||||
|  | ||||
| const props = defineProps({ | ||||
|   defaultValue: { type: [String, Number], required: false }, | ||||
|   modelValue: { type: [String, Number], required: false }, | ||||
|   class: { type: null, required: false } | ||||
| }) | ||||
|   class: { type: null, required: false }, | ||||
| }); | ||||
|  | ||||
| const emits = defineEmits(['update:modelValue']) | ||||
| const emits = defineEmits(['update:modelValue']); | ||||
|  | ||||
| const modelValue = useVModel(props, 'modelValue', emits, { | ||||
|   passive: true, | ||||
|   defaultValue: props.defaultValue | ||||
| }) | ||||
|   defaultValue: props.defaultValue, | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| @@ -22,7 +22,7 @@ const modelValue = useVModel(props, 'modelValue', emits, { | ||||
|     :class=" | ||||
|       cn( | ||||
|         'flex h-9 w-full rounded-md border border-input bg-transparent px-3 py-1 text-sm shadow-sm transition-colors file:border-0 file:bg-transparent file:text-sm file:font-medium placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:cursor-not-allowed disabled:opacity-50', | ||||
|         props.class | ||||
|         props.class, | ||||
|       ) | ||||
|     " | ||||
|   /> | ||||
|   | ||||
| @@ -1 +1 @@ | ||||
| export { default as Input } from './Input.vue' | ||||
| export { default as Input } from './Input.vue'; | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user