mirror of
				https://github.com/donaldzou/WGDashboard.git
				synced 2025-10-26 04:16:24 +00:00 
			
		
		
		
	Huge update
A welcome session Added Time based One-Time-Passcode (TOTP) UI Update
This commit is contained in:
		| @@ -7,7 +7,7 @@ | ||||
|     <title>Vite App</title> | ||||
|   </head> | ||||
|   <body> | ||||
|     <div id="app" class="vw-100 vh-100"></div> | ||||
|     <div id="app" class="w-100 vh-100"></div> | ||||
|     <script type="module" src="./src/main.js"></script> | ||||
|   </body> | ||||
| </html> | ||||
|   | ||||
							
								
								
									
										296
									
								
								src/static/app/package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										296
									
								
								src/static/app/package-lock.json
									
									
									
										generated
									
									
									
								
							| @@ -11,6 +11,8 @@ | ||||
|         "bootstrap": "^5.3.2", | ||||
|         "bootstrap-icons": "^1.11.2", | ||||
|         "pinia": "^2.1.7", | ||||
|         "qrcode": "^1.5.3", | ||||
|         "uuid": "^9.0.1", | ||||
|         "vue": "^3.3.11", | ||||
|         "vue-router": "^4.2.5" | ||||
|       }, | ||||
| @@ -690,6 +692,28 @@ | ||||
|       "resolved": "https://registry.npmjs.org/@vue/shared/-/shared-3.4.3.tgz", | ||||
|       "integrity": "sha512-rIwlkkP1n4uKrRzivAKPZIEkHiuwY5mmhMJ2nZKCBLz8lTUlE73rQh4n1OnnMurXt1vcUNyH4ZPfdh8QweTjpQ==" | ||||
|     }, | ||||
|     "node_modules/ansi-regex": { | ||||
|       "version": "5.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", | ||||
|       "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/ansi-styles": { | ||||
|       "version": "4.3.0", | ||||
|       "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", | ||||
|       "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", | ||||
|       "dependencies": { | ||||
|         "color-convert": "^2.0.1" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/chalk/ansi-styles?sponsor=1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/bootstrap": { | ||||
|       "version": "5.3.2", | ||||
|       "resolved": "https://registry.npmjs.org/bootstrap/-/bootstrap-5.3.2.tgz", | ||||
| @@ -723,11 +747,68 @@ | ||||
|         } | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/camelcase": { | ||||
|       "version": "5.3.1", | ||||
|       "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", | ||||
|       "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", | ||||
|       "engines": { | ||||
|         "node": ">=6" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/cliui": { | ||||
|       "version": "6.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/cliui/-/cliui-6.0.0.tgz", | ||||
|       "integrity": "sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ==", | ||||
|       "dependencies": { | ||||
|         "string-width": "^4.2.0", | ||||
|         "strip-ansi": "^6.0.0", | ||||
|         "wrap-ansi": "^6.2.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/color-convert": { | ||||
|       "version": "2.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", | ||||
|       "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", | ||||
|       "dependencies": { | ||||
|         "color-name": "~1.1.4" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=7.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/color-name": { | ||||
|       "version": "1.1.4", | ||||
|       "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", | ||||
|       "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" | ||||
|     }, | ||||
|     "node_modules/csstype": { | ||||
|       "version": "3.1.3", | ||||
|       "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.3.tgz", | ||||
|       "integrity": "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==" | ||||
|     }, | ||||
|     "node_modules/decamelize": { | ||||
|       "version": "1.2.0", | ||||
|       "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", | ||||
|       "integrity": "sha512-z2S+W9X73hAUUki+N+9Za2lBlun89zigOyGrsax+KUQ6wKW4ZoWpEYBkGhQjwAjjDCkWxhY0VKEhk8wzY7F5cA==", | ||||
|       "engines": { | ||||
|         "node": ">=0.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/dijkstrajs": { | ||||
|       "version": "1.0.3", | ||||
|       "resolved": "https://registry.npmjs.org/dijkstrajs/-/dijkstrajs-1.0.3.tgz", | ||||
|       "integrity": "sha512-qiSlmBq9+BCdCA/L46dw8Uy93mloxsPSbwnm5yrKn2vMPiy8KyAskTF6zuV/j5BMsmOGZDPs7KjU+mjb670kfA==" | ||||
|     }, | ||||
|     "node_modules/emoji-regex": { | ||||
|       "version": "8.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", | ||||
|       "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" | ||||
|     }, | ||||
|     "node_modules/encode-utf8": { | ||||
|       "version": "1.0.3", | ||||
|       "resolved": "https://registry.npmjs.org/encode-utf8/-/encode-utf8-1.0.3.tgz", | ||||
|       "integrity": "sha512-ucAnuBEhUK4boH2HjVYG5Q2mQyPorvv0u/ocS+zhdw0S8AlHYY+GOFhP1Gio5z4icpP2ivFSvhtFjQi8+T9ppw==" | ||||
|     }, | ||||
|     "node_modules/entities": { | ||||
|       "version": "4.5.0", | ||||
|       "resolved": "https://registry.npmjs.org/entities/-/entities-4.5.0.tgz", | ||||
| @@ -782,6 +863,18 @@ | ||||
|       "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz", | ||||
|       "integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==" | ||||
|     }, | ||||
|     "node_modules/find-up": { | ||||
|       "version": "4.1.0", | ||||
|       "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", | ||||
|       "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", | ||||
|       "dependencies": { | ||||
|         "locate-path": "^5.0.0", | ||||
|         "path-exists": "^4.0.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/fsevents": { | ||||
|       "version": "2.3.3", | ||||
|       "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", | ||||
| @@ -796,6 +889,33 @@ | ||||
|         "node": "^8.16.0 || ^10.6.0 || >=11.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/get-caller-file": { | ||||
|       "version": "2.0.5", | ||||
|       "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", | ||||
|       "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", | ||||
|       "engines": { | ||||
|         "node": "6.* || 8.* || >= 10.*" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/is-fullwidth-code-point": { | ||||
|       "version": "3.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", | ||||
|       "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/locate-path": { | ||||
|       "version": "5.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", | ||||
|       "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", | ||||
|       "dependencies": { | ||||
|         "p-locate": "^4.1.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/magic-string": { | ||||
|       "version": "0.30.5", | ||||
|       "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.5.tgz", | ||||
| @@ -824,6 +944,47 @@ | ||||
|         "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/p-limit": { | ||||
|       "version": "2.3.0", | ||||
|       "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", | ||||
|       "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", | ||||
|       "dependencies": { | ||||
|         "p-try": "^2.0.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=6" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/sponsors/sindresorhus" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/p-locate": { | ||||
|       "version": "4.1.0", | ||||
|       "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", | ||||
|       "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", | ||||
|       "dependencies": { | ||||
|         "p-limit": "^2.2.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/p-try": { | ||||
|       "version": "2.2.0", | ||||
|       "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", | ||||
|       "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", | ||||
|       "engines": { | ||||
|         "node": ">=6" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/path-exists": { | ||||
|       "version": "4.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", | ||||
|       "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/picocolors": { | ||||
|       "version": "1.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.0.tgz", | ||||
| @@ -879,6 +1040,14 @@ | ||||
|         } | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/pngjs": { | ||||
|       "version": "5.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/pngjs/-/pngjs-5.0.0.tgz", | ||||
|       "integrity": "sha512-40QW5YalBNfQo5yRYmiw7Yz6TKKVr3h6970B2YE+3fQpsWcrbj1PzJgxeJ19DRQjhMbKPIuMY8rFaXc8moolVw==", | ||||
|       "engines": { | ||||
|         "node": ">=10.13.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/postcss": { | ||||
|       "version": "8.4.32", | ||||
|       "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.32.tgz", | ||||
| @@ -906,6 +1075,36 @@ | ||||
|         "node": "^10 || ^12 || >=14" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/qrcode": { | ||||
|       "version": "1.5.3", | ||||
|       "resolved": "https://registry.npmjs.org/qrcode/-/qrcode-1.5.3.tgz", | ||||
|       "integrity": "sha512-puyri6ApkEHYiVl4CFzo1tDkAZ+ATcnbJrJ6RiBM1Fhctdn/ix9MTE3hRph33omisEbC/2fcfemsseiKgBPKZg==", | ||||
|       "dependencies": { | ||||
|         "dijkstrajs": "^1.0.1", | ||||
|         "encode-utf8": "^1.0.3", | ||||
|         "pngjs": "^5.0.0", | ||||
|         "yargs": "^15.3.1" | ||||
|       }, | ||||
|       "bin": { | ||||
|         "qrcode": "bin/qrcode" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=10.13.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/require-directory": { | ||||
|       "version": "2.1.1", | ||||
|       "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", | ||||
|       "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", | ||||
|       "engines": { | ||||
|         "node": ">=0.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/require-main-filename": { | ||||
|       "version": "2.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz", | ||||
|       "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==" | ||||
|     }, | ||||
|     "node_modules/rollup": { | ||||
|       "version": "4.9.2", | ||||
|       "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.9.2.tgz", | ||||
| @@ -935,6 +1134,11 @@ | ||||
|         "fsevents": "~2.3.2" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/set-blocking": { | ||||
|       "version": "2.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", | ||||
|       "integrity": "sha512-KiKBS8AnWGEyLzofFfmvKwpdPzqiy16LvQfK3yv/fVH7Bj13/wl3JSR1J+rfgRE9q7xUJK4qvgS8raSOeLUehw==" | ||||
|     }, | ||||
|     "node_modules/source-map-js": { | ||||
|       "version": "1.0.2", | ||||
|       "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.0.2.tgz", | ||||
| @@ -943,6 +1147,42 @@ | ||||
|         "node": ">=0.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/string-width": { | ||||
|       "version": "4.2.3", | ||||
|       "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", | ||||
|       "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", | ||||
|       "dependencies": { | ||||
|         "emoji-regex": "^8.0.0", | ||||
|         "is-fullwidth-code-point": "^3.0.0", | ||||
|         "strip-ansi": "^6.0.1" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/strip-ansi": { | ||||
|       "version": "6.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", | ||||
|       "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", | ||||
|       "dependencies": { | ||||
|         "ansi-regex": "^5.0.1" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/uuid": { | ||||
|       "version": "9.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/uuid/-/uuid-9.0.1.tgz", | ||||
|       "integrity": "sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==", | ||||
|       "funding": [ | ||||
|         "https://github.com/sponsors/broofa", | ||||
|         "https://github.com/sponsors/ctavan" | ||||
|       ], | ||||
|       "bin": { | ||||
|         "uuid": "dist/bin/uuid" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/vite": { | ||||
|       "version": "5.0.10", | ||||
|       "resolved": "https://registry.npmjs.org/vite/-/vite-5.0.10.tgz", | ||||
| @@ -1031,6 +1271,62 @@ | ||||
|       "peerDependencies": { | ||||
|         "vue": "^3.2.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/which-module": { | ||||
|       "version": "2.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.1.tgz", | ||||
|       "integrity": "sha512-iBdZ57RDvnOR9AGBhML2vFZf7h8vmBjhoaZqODJBFWHVtKkDmKuHai3cx5PgVMrX5YDNp27AofYbAwctSS+vhQ==" | ||||
|     }, | ||||
|     "node_modules/wrap-ansi": { | ||||
|       "version": "6.2.0", | ||||
|       "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", | ||||
|       "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", | ||||
|       "dependencies": { | ||||
|         "ansi-styles": "^4.0.0", | ||||
|         "string-width": "^4.1.0", | ||||
|         "strip-ansi": "^6.0.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/y18n": { | ||||
|       "version": "4.0.3", | ||||
|       "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.3.tgz", | ||||
|       "integrity": "sha512-JKhqTOwSrqNA1NY5lSztJ1GrBiUodLMmIZuLiDaMRJ+itFd+ABVE8XBjOvIWL+rSqNDC74LCSFmlb/U4UZ4hJQ==" | ||||
|     }, | ||||
|     "node_modules/yargs": { | ||||
|       "version": "15.4.1", | ||||
|       "resolved": "https://registry.npmjs.org/yargs/-/yargs-15.4.1.tgz", | ||||
|       "integrity": "sha512-aePbxDmcYW++PaqBsJ+HYUFwCdv4LVvdnhBy78E57PIor8/OVvhMrADFFEDh8DHDFRv/O9i3lPhsENjO7QX0+A==", | ||||
|       "dependencies": { | ||||
|         "cliui": "^6.0.0", | ||||
|         "decamelize": "^1.2.0", | ||||
|         "find-up": "^4.1.0", | ||||
|         "get-caller-file": "^2.0.1", | ||||
|         "require-directory": "^2.1.1", | ||||
|         "require-main-filename": "^2.0.0", | ||||
|         "set-blocking": "^2.0.0", | ||||
|         "string-width": "^4.2.0", | ||||
|         "which-module": "^2.0.0", | ||||
|         "y18n": "^4.0.0", | ||||
|         "yargs-parser": "^18.1.2" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/yargs-parser": { | ||||
|       "version": "18.1.3", | ||||
|       "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-18.1.3.tgz", | ||||
|       "integrity": "sha512-o50j0JeToy/4K6OZcaQmW6lyXXKhq7csREXcDwk2omFPJEwUNOVtJKvmDr9EI1fAJZUyZcRF7kxGBWmRXudrCQ==", | ||||
|       "dependencies": { | ||||
|         "camelcase": "^5.0.0", | ||||
|         "decamelize": "^1.2.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=6" | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -12,6 +12,8 @@ | ||||
|     "bootstrap": "^5.3.2", | ||||
|     "bootstrap-icons": "^1.11.2", | ||||
|     "pinia": "^2.1.7", | ||||
|     "qrcode": "^1.5.3", | ||||
|     "uuid": "^9.0.1", | ||||
|     "vue": "^3.3.11", | ||||
|     "vue-router": "^4.2.5" | ||||
|   }, | ||||
|   | ||||
| @@ -17,6 +17,10 @@ body { | ||||
|     font-weight: bold; | ||||
| } | ||||
|  | ||||
| .dashboardLogo{ | ||||
|     background: -webkit-linear-gradient(#178bff, #ff4a00); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Sidebar | ||||
|  */ | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| <script setup> | ||||
| <script setup > | ||||
| import { RouterView } from 'vue-router' | ||||
|  | ||||
| import {DashboardConfigurationStore} from "@/stores/DashboardConfigurationStore.js"; | ||||
| const store = DashboardConfigurationStore(); | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| @@ -9,5 +10,11 @@ import { RouterView } from 'vue-router' | ||||
| 			<span class="navbar-brand mb-0 h1">WGDashboard</span> | ||||
| 		</div> | ||||
| 	</nav> | ||||
| 	<RouterView></RouterView> | ||||
| 	<Suspense> | ||||
| 		<RouterView v-slot="{ Component }"> | ||||
| 			<Transition name="fade" mode="out-in"> | ||||
| 				<Component :is="Component"></Component> | ||||
| 			</Transition> | ||||
| 		</RouterView> | ||||
| 	</Suspense> | ||||
| </template> | ||||
| @@ -1,46 +1,48 @@ | ||||
| <script> | ||||
| import {wgdashboardStore} from "@/stores/wgdashboardStore.js"; | ||||
| import {WireguardConfigurationsStore} from "@/stores/WireguardConfigurationsStore.js"; | ||||
|  | ||||
| export default { | ||||
| 	name: "configurationList", | ||||
| 	async setup(){ | ||||
| 		const store = wgdashboardStore(); | ||||
| 		await store.getWireguardConfigurations(); | ||||
| 		return {store} | ||||
| 		const wireguardConfigurationsStore = WireguardConfigurationsStore(); | ||||
| 		await wireguardConfigurationsStore.getConfigurations(); | ||||
| 		return {wireguardConfigurationsStore} | ||||
| 	} | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| 	<div class="mt-4"> | ||||
| 		<h3 class="mb-3 text-body">Wireguard Configurations</h3> | ||||
| 		<p class="text-muted" v-if="this.store.WireguardConfigurations.length === 0">You don't have any WireGuard configurations yet. Please check the configuration folder or change it in "Settings". By default the folder is "/etc/wireguard".</p> | ||||
| 		 | ||||
| 		<div class="card conf_card rounded-3 shadow" v-else v-for="c in this.store.WireguardConfigurations" :key="c.Name"> | ||||
| 			<div class="card-body"> | ||||
| 				<div class="row"> | ||||
| 					<div class="row"> | ||||
| 						<div class="col card-col"> | ||||
| 							<small class="text-muted"><strong>CONFIGURATION</strong></small> | ||||
| 							<h6 class="card-title" style="margin:0 !important;"><samp>{{c.Name}}</samp></h6> | ||||
| 						</div> | ||||
| 						<div class="col card-col"> | ||||
| 							<small class="text-muted"><strong>STATUS</strong></small> | ||||
| 							<h6><span>{{c.Status ? "Running":"Stopped"}}</span> | ||||
| 								<span class="dot" :class="{active: c.Status}"></span></h6> | ||||
| 						</div> | ||||
| 						<div class="col-sm card-col"> | ||||
| 							<small class="text-muted"><strong>PUBLIC KEY</strong></small> | ||||
| 							<h6 style="margin:0 !important;"><samp>{{c.PublicKey}}</samp></h6> | ||||
| 						</div> | ||||
| 						<div class="col-sm index-switch"> | ||||
| 							<div class="switch-test"> | ||||
| 								<input type="checkbox" class="toggle--switch" checked :id="c.Name + '-switch'"> | ||||
| 								<label :for="c.Name + '-switch'" class="toggleLabel"></label> | ||||
| 							</div> | ||||
| 						</div> | ||||
| 		<div class="container"> | ||||
| 			<div class="d-flex mb-4 "> | ||||
| 				<h3 class="text-body">Wireguard Configurations</h3> | ||||
| 				<RouterLink to="/new_configuration" class="btn btn-dark btn-brand rounded-3 px-3 py-2 shadow ms-auto rounded-5"> | ||||
| 					<i class="bi bi-plus-circle-fill me-2"></i> | ||||
| 					Configuration | ||||
| 				</RouterLink> | ||||
| 			</div> | ||||
| 			<p class="text-muted" v-if="this.wireguardConfigurationsStore.Configurations.length === 0">You don't have any WireGuard configurations yet. Please check the configuration folder or change it in "Settings". By default the folder is "/etc/wireguard".</p> | ||||
|  | ||||
| 			<div class="d-flex gap-3 flex-column" v-else > | ||||
| 				<RouterLink :to="'/configuration/' + c.Name" | ||||
| 				            class="card conf_card rounded-3 shadow text-decoration-none" v-for="c in this.wireguardConfigurationsStore.Configurations" :key="c.Name"> | ||||
| 					<div class="card-body d-flex align-items-center gap-3 flex-wrap"> | ||||
| 						<h6 class="mb-0"><span class="dot" :class="{active: c.Status}"></span></h6> | ||||
| 						<h6 class="card-title mb-0"><samp>{{c.Name}}</samp></h6> | ||||
| 						<h6 class="mb-0 ms-auto"> | ||||
| 							<i class="bi bi-chevron-right"></i> | ||||
| 						</h6> | ||||
| 					</div> | ||||
| 				</div> | ||||
| 					<div class="card-footer"> | ||||
| 						<small class="me-2 text-muted"> | ||||
| 							<strong>PUBLIC KEY</strong> | ||||
| 						</small> | ||||
| 						<small class="mb-0 d-block d-lg-inline-block "> | ||||
| 							<samp style="line-break: anywhere">{{c.PublicKey}}</samp> | ||||
| 						</small> | ||||
| 					</div> | ||||
| 				</RouterLink> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
|   | ||||
| @@ -1,50 +1,59 @@ | ||||
| <script> | ||||
| import {wgdashboardStore} from "@/stores/wgdashboardStore.js"; | ||||
| import {WireguardConfigurationsStore} from "@/stores/WireguardConfigurationsStore.js"; | ||||
| import {DashboardConfigurationStore} from "@/stores/DashboardConfigurationStore.js"; | ||||
|  | ||||
| export default { | ||||
| 	name: "navbar", | ||||
| 	setup(){ | ||||
| 		const store = wgdashboardStore() | ||||
| 		return {store} | ||||
| 		const wireguardConfigurationsStore = WireguardConfigurationsStore(); | ||||
| 		const dashboardConfigurationStore = DashboardConfigurationStore(); | ||||
| 		return {wireguardConfigurationsStore, dashboardConfigurationStore} | ||||
| 	} | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| 	<nav id="sidebarMenu" class="col-md-3 col-lg-2 d-md-block bg-body-tertiary sidebar border border-right p-0"> | ||||
| 		<div class="sidebar-sticky pt-3"> | ||||
| 			<ul class="nav flex-column"> | ||||
| 				<li class="nav-item"> | ||||
| 					<RouterLink class="nav-link" to="/" exact-active-class="active">Home</RouterLink></li> | ||||
| 				<li class="nav-item"> | ||||
| 					<RouterLink class="nav-link" to="/settings" exact-active-class="active">Settings</RouterLink></li> | ||||
| 			</ul> | ||||
| 			<hr> | ||||
| 			<h6 class="sidebar-heading d-flex justify-content-between align-items-center px-3 mt-4 mb-1 text-muted"> | ||||
| 				<span>Configurations</span> | ||||
| 				 | ||||
| 			</h6> | ||||
| 			<ul class="nav flex-column"> | ||||
| 				<li class="nav-item"> | ||||
| 					<RouterLink :to="'/configuration/'+c.Name" class="nav-link nav-conf-link" v-for="c in this.store.WireguardConfigurations"> | ||||
| 						<samp>{{c.Name}}</samp> | ||||
| 					</RouterLink> | ||||
| 				</li> | ||||
| 			</ul> | ||||
| 			<hr> | ||||
| 			<h6 class="sidebar-heading d-flex justify-content-between align-items-center px-3 mt-4 mb-1 text-muted"> | ||||
| 				<span>Tools</span> | ||||
| 			</h6> | ||||
| 			<ul class="nav flex-column"> | ||||
| 				<li class="nav-item"><a class="nav-link" data-toggle="modal" data-target="#ping_modal" href="#">Ping</a></li> | ||||
| 				<li class="nav-item"><a class="nav-link" data-toggle="modal" data-target="#traceroute_modal" href="#">Traceroute</a></li> | ||||
| 			</ul> | ||||
| 			<hr> | ||||
| 			<ul class="nav flex-column"> | ||||
| 				<li class="nav-item"><a href="https://github.com/donaldzou/WGDashboard/releases/tag/"><small class="nav-link text-muted"></small></a></li> | ||||
| 			</ul> | ||||
| 		</div> | ||||
| 	</nav> | ||||
| 	<div class="col-md-3 col-lg-2 d-md-block p-3"> | ||||
| 		<nav id="sidebarMenu" class=" bg-body-tertiary sidebar border h-100 rounded-3 shadow" > | ||||
| 			<div class="sidebar-sticky pt-3"> | ||||
| 				<ul class="nav flex-column"> | ||||
| 					<li class="nav-item"> | ||||
| 						<RouterLink class="nav-link" to="/" exact-active-class="active">Home</RouterLink></li> | ||||
| 					<li class="nav-item"> | ||||
| 						<RouterLink class="nav-link" to="/settings" exact-active-class="active">Settings</RouterLink></li> | ||||
| 				</ul> | ||||
| 				<hr> | ||||
| 				<h6 class="sidebar-heading d-flex justify-content-between align-items-center px-3 mt-4 mb-1 text-muted"> | ||||
| 					<span>Configurations</span> | ||||
| 				</h6> | ||||
| 				<ul class="nav flex-column"> | ||||
| 					<li class="nav-item"> | ||||
| 						<RouterLink :to="'/configuration/'+c.Name" class="nav-link nav-conf-link" v-for="c in this.wireguardConfigurationsStore.Configurations"> | ||||
| 							<samp>{{c.Name}}</samp> | ||||
| 						</RouterLink> | ||||
| 					</li> | ||||
| 				</ul> | ||||
| 				<hr> | ||||
| 				<h6 class="sidebar-heading d-flex justify-content-between align-items-center px-3 mt-4 mb-1 text-muted"> | ||||
| 					<span>Tools</span> | ||||
| 				</h6> | ||||
| 				<ul class="nav flex-column"> | ||||
| 					<li class="nav-item"><a class="nav-link" data-toggle="modal" data-target="#ping_modal" href="#">Ping</a></li> | ||||
| 					<li class="nav-item"><a class="nav-link" data-toggle="modal" data-target="#traceroute_modal" href="#">Traceroute</a></li> | ||||
| 				</ul> | ||||
| 				<hr> | ||||
| 				<ul class="nav flex-column"> | ||||
| 					<li class="nav-item"><a class="nav-link text-danger" @click="this.dashboardConfigurationStore.signOut()" role="button" style="font-weight: bold">Sign Out</a></li> | ||||
| 				</ul> | ||||
| 				<ul class="nav flex-column"> | ||||
| 					<li class="nav-item"><a href="https://github.com/donaldzou/WGDashboard/releases/tag/"><small class="nav-link text-muted"></small></a></li> | ||||
| 				</ul> | ||||
| 			</div> | ||||
| 		</nav> | ||||
| 	</div> | ||||
| 	 | ||||
| 	 | ||||
| </template> | ||||
|  | ||||
| <style scoped> | ||||
|   | ||||
| @@ -0,0 +1,120 @@ | ||||
| <script> | ||||
| import {DashboardConfigurationStore} from "@/stores/DashboardConfigurationStore.js"; | ||||
| import {v4} from "uuid"; | ||||
| import {fetchPost} from "@/utilities/fetch.js"; | ||||
|  | ||||
| export default { | ||||
| 	name: "accountSettingsInputPassword", | ||||
| 	props:{ | ||||
| 		targetData: String, | ||||
| 		warning: false, | ||||
| 		warningText: "" | ||||
| 	}, | ||||
| 	setup(){ | ||||
| 		const store = DashboardConfigurationStore(); | ||||
| 		const uuid = `input_${v4()}`; | ||||
| 		return {store, uuid}; | ||||
| 	}, | ||||
| 	data(){ | ||||
| 		return{ | ||||
| 			value: { | ||||
| 				currentPassword: "", | ||||
| 				newPassword: "", | ||||
| 				repeatNewPassword: "" | ||||
| 			}, | ||||
| 			invalidFeedback: "", | ||||
| 			showInvalidFeedback: false, | ||||
| 			isValid: false, | ||||
| 			timeout: undefined | ||||
| 		} | ||||
| 	}, | ||||
| 	methods:{ | ||||
| 		async useValidation(){ | ||||
| 			if (Object.values(this.value).find(x => x.length === 0) === undefined){ | ||||
| 				if (this.value.newPassword === this.value.repeatNewPassword){ | ||||
| 					await fetchPost("/api/updateDashboardConfigurationItem", { | ||||
| 						section: "Account", | ||||
| 						key: this.targetData, | ||||
| 						value: this.value | ||||
| 					}, (res) => { | ||||
| 						if (res.status){ | ||||
| 							this.isValid = true; | ||||
| 							this.showInvalidFeedback = false; | ||||
| 							this.store.Configuration.Account[this.targetData] = this.value | ||||
| 							clearTimeout(this.timeout) | ||||
| 							this.timeout = setTimeout(() => { | ||||
| 								this.isValid = false; | ||||
| 								this.value = { | ||||
| 									currentPassword: "", | ||||
| 									newPassword: "", | ||||
| 									repeatNewPassword: "" | ||||
| 								} | ||||
| 							}, 5000); | ||||
| 						}else{ | ||||
| 							this.isValid = false; | ||||
| 							this.showInvalidFeedback = true; | ||||
| 							this.invalidFeedback = res.message | ||||
| 						} | ||||
| 					}) | ||||
| 				}else{ | ||||
| 					this.showInvalidFeedback = true; | ||||
| 					this.invalidFeedback = "New passwords does not match" | ||||
| 				} | ||||
| 				 | ||||
| 			}else{ | ||||
| 				this.showInvalidFeedback = true; | ||||
| 				this.invalidFeedback = "Please fill in all required fields." | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| 	<div> | ||||
| 		<div class="row"> | ||||
| 			<div class="col-sm"> | ||||
| 				<div class="form-group mb-2"> | ||||
| 					<label :for="'currentPassword_' + this.uuid" class="text-muted mb-1"> | ||||
| 						<strong><small>Current Password</small></strong> | ||||
| 					</label> | ||||
| 					<input type="password" class="form-control mb-2" | ||||
| 					       :class="{'is-invalid': showInvalidFeedback, 'is-valid': isValid}" | ||||
| 					       v-model="this.value.currentPassword" | ||||
| 					       :id="'currentPassword_' + this.uuid"> | ||||
| 					<div class="invalid-feedback d-block" v-if="showInvalidFeedback">{{this.invalidFeedback}}</div> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 			<div class="col-sm"> | ||||
| 				<div class="form-group mb-2"> | ||||
| 					<label :for="'newPassword_' + this.uuid" class="text-muted mb-1"> | ||||
| 						<strong><small>New Password</small></strong> | ||||
| 					</label> | ||||
| 					<input type="password" class="form-control mb-2" | ||||
| 					       :class="{'is-invalid': showInvalidFeedback, 'is-valid': isValid}" | ||||
| 					       v-model="this.value.newPassword" | ||||
| 					       :id="'newPassword_' + this.uuid"> | ||||
|  | ||||
| 				</div> | ||||
| 			</div> | ||||
| 			<div class="col-sm"> | ||||
| 				<div class="form-group mb-2"> | ||||
| 					<label :for="'repeatNewPassword_' + this.uuid" class="text-muted mb-1"> | ||||
| 						<strong><small>Repeat New Password</small></strong> | ||||
| 					</label> | ||||
| 					<input type="password" class="form-control mb-2" | ||||
| 					       :class="{'is-invalid': showInvalidFeedback, 'is-valid': isValid}" | ||||
| 					       v-model="this.value.repeatNewPassword" | ||||
| 					       :id="'repeatNewPassword_' + this.uuid"> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<button class="btn btn-success btn-sm fw-bold rounded-3" @click="this.useValidation()"> | ||||
| 			<i class="bi bi-key-fill me-2"></i>Update Password | ||||
| 		</button> | ||||
| 	</div> | ||||
| </template> | ||||
|  | ||||
| <style scoped> | ||||
|  | ||||
| </style> | ||||
| @@ -0,0 +1,86 @@ | ||||
| <script> | ||||
| import {DashboardConfigurationStore} from "@/stores/DashboardConfigurationStore.js"; | ||||
| import {v4} from "uuid"; | ||||
| import {fetchPost} from "@/utilities/fetch.js"; | ||||
|  | ||||
| export default { | ||||
| 	name: "accountSettingsInputUsername", | ||||
| 	props:{ | ||||
| 		targetData: String, | ||||
| 		title: String, | ||||
| 		warning: false, | ||||
| 		warningText: "" | ||||
| 	}, | ||||
| 	setup(){ | ||||
| 		const store = DashboardConfigurationStore(); | ||||
| 		const uuid = `input_${v4()}`; | ||||
| 		return {store, uuid}; | ||||
| 	}, | ||||
| 	data(){ | ||||
| 		return{ | ||||
| 			value:"", | ||||
| 			invalidFeedback: "", | ||||
| 			showInvalidFeedback: false, | ||||
| 			isValid: false, | ||||
| 			timeout: undefined, | ||||
| 			changed: false, | ||||
| 			updating: false, | ||||
| 		} | ||||
| 	}, | ||||
| 	mounted() { | ||||
| 		this.value = this.store.Configuration.Account[this.targetData]; | ||||
| 	}, | ||||
| 	methods:{ | ||||
| 		async useValidation(){ | ||||
| 			if (this.changed){ | ||||
| 				this.updating = true | ||||
| 				await fetchPost("/api/updateDashboardConfigurationItem", { | ||||
| 					section: "Account", | ||||
| 					key: this.targetData, | ||||
| 					value: this.value | ||||
| 				}, (res) => { | ||||
| 					if (res.status){ | ||||
| 						this.isValid = true; | ||||
| 						this.showInvalidFeedback = false; | ||||
| 						this.store.Configuration.Account[this.targetData] = this.value | ||||
| 						clearTimeout(this.timeout) | ||||
| 						this.timeout = setTimeout(() => this.isValid = false, 5000); | ||||
| 					}else{ | ||||
| 						this.isValid = false; | ||||
| 						this.showInvalidFeedback = true; | ||||
| 						this.invalidFeedback = res.message | ||||
| 					} | ||||
| 					this.changed = false | ||||
| 					this.updating = false; | ||||
| 				}) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| 	<div class="form-group mb-2"> | ||||
| 		<label :for="this.uuid" class="text-muted mb-1"> | ||||
| 			<strong><small>{{this.title}}</small></strong> | ||||
| 		</label> | ||||
| 		<input type="text" class="form-control" | ||||
| 		       :class="{'is-invalid': showInvalidFeedback, 'is-valid': isValid}" | ||||
| 		       :id="this.uuid" | ||||
| 		       v-model="this.value" | ||||
| 		       @keydown="this.changed = true" | ||||
| 		       @blur="useValidation()" | ||||
| 		       :disabled="this.updating" | ||||
| 		> | ||||
| 		<div class="invalid-feedback">{{this.invalidFeedback}}</div> | ||||
| 		<div class="px-2 py-1 text-warning-emphasis bg-warning-subtle border border-warning-subtle rounded-2 d-inline-block mt-1" | ||||
| 		     v-if="warning" | ||||
| 		> | ||||
| 			<small><i class="bi bi-exclamation-triangle-fill me-2"></i><span v-html="warningText"></span></small> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </template> | ||||
|  | ||||
| <style scoped> | ||||
|  | ||||
| </style> | ||||
| @@ -0,0 +1,91 @@ | ||||
| <script> | ||||
| import {DashboardConfigurationStore} from "@/stores/DashboardConfigurationStore.js"; | ||||
| import {v4} from "uuid"; | ||||
| import {fetchPost} from "@/utilities/fetch.js"; | ||||
|  | ||||
| export default { | ||||
| 	name: "dashboardSettingsInputIPAddressAndPort", | ||||
| 	props:{ | ||||
| 		// targetData: String, | ||||
| 		// title: String, | ||||
| 		// warning: false, | ||||
| 		// warningText: "" | ||||
| 	}, | ||||
| 	setup(){ | ||||
| 		const store = DashboardConfigurationStore(); | ||||
| 		const uuid = `input_${v4()}`; | ||||
| 		return {store, uuid}; | ||||
| 	}, | ||||
| 	data(){ | ||||
| 		return{ | ||||
| 			app_ip:"", | ||||
| 			app_port:"", | ||||
| 			invalidFeedback: "", | ||||
| 			showInvalidFeedback: false, | ||||
| 			isValid: false, | ||||
| 			timeout: undefined, | ||||
| 			changed: false, | ||||
| 			updating: false, | ||||
| 		} | ||||
| 	}, | ||||
| 	mounted() { | ||||
| 		this.app_ip = this.store.Configuration.Server.app_ip; | ||||
| 		this.app_port = this.store.Configuration.Server.app_port; | ||||
| 	}, | ||||
| 	methods:{ | ||||
| 		async useValidation(){ | ||||
| 			if(this.changed){ | ||||
| 				await fetchPost("/api/updateDashboardConfigurationItem", { | ||||
| 					section: "Server", | ||||
| 					key: this.targetData, | ||||
| 					value: this.value | ||||
| 				}, (res) => { | ||||
| 					if (res.status){ | ||||
| 						this.isValid = true; | ||||
| 						this.showInvalidFeedback = false; | ||||
| 						this.store.Configuration.Account[this.targetData] = this.value | ||||
| 						clearTimeout(this.timeout) | ||||
| 						this.timeout = setTimeout(() => this.isValid = false, 5000); | ||||
| 					}else{ | ||||
| 						this.isValid = false; | ||||
| 						this.showInvalidFeedback = true; | ||||
| 						this.invalidFeedback = res.message | ||||
| 					} | ||||
| 				}) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| 	<div> | ||||
| 		<div class="invalid-feedback d-block mt-0">{{this.invalidFeedback}}</div> | ||||
| 		<div class="row"> | ||||
| 			<div class="form-group mb-2 col-sm"> | ||||
| 				<label :for="'app_ip_' + this.uuid" class="text-muted mb-1"> | ||||
| 					<strong><small>Dashboard IP Address</small></strong> | ||||
| 				</label> | ||||
| 				<input type="text" class="form-control mb-2" :id="'app_ip_' + this.uuid" v-model="this.app_ip"> | ||||
| 				<div class="px-2 py-1 text-warning-emphasis bg-warning-subtle border border-warning-subtle rounded-2 d-inline-block"> | ||||
| 					<small><i class="bi bi-exclamation-triangle-fill me-2"></i><code>0.0.0.0</code> means it can be access by anyone with your server | ||||
| 						IP Address.</small> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 			<div class="form-group col-sm"> | ||||
| 				<label :for="'app_port_' + this.uuid" class="text-muted mb-1"> | ||||
| 					<strong><small>Dashboard Port</small></strong> | ||||
| 				</label> | ||||
| 				<input type="text" class="form-control mb-2" :id="'app_port_' + this.uuid" v-model="this.app_port"> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<button class="btn btn-success btn-sm fw-bold rounded-3"> | ||||
| 			<i class="bi bi-floppy-fill me-2"></i>Update Dashboard Settings & Restart | ||||
| 		</button> | ||||
| 	</div> | ||||
| </template> | ||||
|  | ||||
|  | ||||
| <style scoped> | ||||
|  | ||||
| </style> | ||||
| @@ -0,0 +1,86 @@ | ||||
| <script> | ||||
| import {DashboardConfigurationStore} from "@/stores/DashboardConfigurationStore.js"; | ||||
| import {v4} from "uuid"; | ||||
| import {fetchPost} from "@/utilities/fetch.js"; | ||||
|  | ||||
| export default { | ||||
| 	name: "dashboardSettingsInputWireguardConfigurationPath", | ||||
| 	props:{ | ||||
| 		targetData: String, | ||||
| 		title: String, | ||||
| 		warning: false, | ||||
| 		warningText: "" | ||||
| 	}, | ||||
| 	setup(){ | ||||
| 		const store = DashboardConfigurationStore(); | ||||
| 		const uuid = `input_${v4()}`; | ||||
| 		return {store, uuid}; | ||||
| 	}, | ||||
| 	data(){ | ||||
| 		return{ | ||||
| 			value:"", | ||||
| 			invalidFeedback: "", | ||||
| 			showInvalidFeedback: false, | ||||
| 			isValid: false, | ||||
| 			timeout: undefined, | ||||
| 			changed: false, | ||||
| 			updating: false, | ||||
| 		} | ||||
| 	}, | ||||
| 	mounted() { | ||||
| 		this.value = this.store.Configuration.Server[this.targetData]; | ||||
| 	}, | ||||
| 	methods:{ | ||||
| 		async useValidation(){ | ||||
| 			if(this.changed){ | ||||
| 				await fetchPost("/api/updateDashboardConfigurationItem", { | ||||
| 					section: "Server", | ||||
| 					key: this.targetData, | ||||
| 					value: this.value | ||||
| 				}, (res) => { | ||||
| 					if (res.status){ | ||||
| 						this.isValid = true; | ||||
| 						this.showInvalidFeedback = false; | ||||
| 						this.store.Configuration.Account[this.targetData] = this.value | ||||
| 						clearTimeout(this.timeout) | ||||
| 						this.timeout = setTimeout(() => this.isValid = false, 5000); | ||||
| 					}else{ | ||||
| 						this.isValid = false; | ||||
| 						this.showInvalidFeedback = true; | ||||
| 						this.invalidFeedback = res.message | ||||
| 					} | ||||
| 					this.changed = false; | ||||
| 					this.updating = false | ||||
| 				}) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| 	<div class="form-group mb-2"> | ||||
| 		<label :for="this.uuid" class="text-muted mb-1"> | ||||
| 			<strong><small>{{this.title}}</small></strong> | ||||
| 		</label> | ||||
| 		<input type="text" class="form-control" | ||||
| 		       :class="{'is-invalid': this.showInvalidFeedback, 'is-valid': this.isValid}" | ||||
| 		       :id="this.uuid" | ||||
| 		       v-model="this.value" | ||||
| 		       @keydown="this.changed = true" | ||||
| 		       @blur="this.useValidation()" | ||||
| 		       :disabled="this.updating" | ||||
| 		> | ||||
| 		<div class="invalid-feedback">{{this.invalidFeedback}}</div> | ||||
| 		<div class="px-2 py-1 text-warning-emphasis bg-warning-subtle border border-warning-subtle rounded-2 d-inline-block mt-1" | ||||
| 		     v-if="warning" | ||||
| 		> | ||||
| 			<small><i class="bi bi-exclamation-triangle-fill me-2"></i><span v-html="warningText"></span></small> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </template> | ||||
|  | ||||
|  | ||||
| <style scoped> | ||||
|  | ||||
| </style> | ||||
| @@ -0,0 +1,49 @@ | ||||
| <script> | ||||
| import {DashboardConfigurationStore} from "@/stores/DashboardConfigurationStore.js"; | ||||
| import {fetchPost} from "@/utilities/fetch.js"; | ||||
|  | ||||
| export default { | ||||
| 	name: "dashboardTheme", | ||||
| 	setup(){ | ||||
| 		const dashboardConfigurationStore = DashboardConfigurationStore(); | ||||
| 		return {dashboardConfigurationStore} | ||||
| 	}, | ||||
| 	methods: { | ||||
| 		async switchTheme(value){ | ||||
| 			await fetchPost("/api/updateDashboardConfigurationItem", { | ||||
| 				section: "Server", | ||||
| 				key: "dashboard_theme", | ||||
| 				value: value | ||||
| 			}, (res) => { | ||||
| 				if (res.status){ | ||||
| 					this.dashboardConfigurationStore.Configuration.Server.dashboard_theme = value; | ||||
| 				} | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| 	<div class="card mb-4 shadow rounded-3"> | ||||
| 		<p class="card-header">Dashboard Theme</p> | ||||
| 		<div class="card-body d-flex gap-2"> | ||||
| 			<button class="btn btn-outline-primary flex-grow-1" | ||||
| 			        @click="this.switchTheme('light')" | ||||
| 			        :class="{active: this.dashboardConfigurationStore.Configuration.Server.dashboard_theme === 'light'}"> | ||||
| 				<i class="bi bi-sun-fill"></i> | ||||
| 				Light | ||||
| 			</button> | ||||
| 			<button class="btn btn-outline-primary flex-grow-1" | ||||
| 			        @click="this.switchTheme('dark')" | ||||
| 			        :class="{active: this.dashboardConfigurationStore.Configuration.Server.dashboard_theme === 'dark'}"> | ||||
| 				<i class="bi bi-moon-fill"></i> | ||||
| 				Dark | ||||
| 			</button> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </template> | ||||
|  | ||||
| <style scoped> | ||||
|  | ||||
| </style> | ||||
| @@ -0,0 +1,86 @@ | ||||
| <script> | ||||
| import {DashboardConfigurationStore} from "@/stores/DashboardConfigurationStore.js"; | ||||
| import {v4} from "uuid"; | ||||
| import {fetchPost} from "@/utilities/fetch.js"; | ||||
|  | ||||
| export default { | ||||
| 	props:{ | ||||
| 		targetData: String, | ||||
| 		title: String, | ||||
| 		warning: false, | ||||
| 		warningText: "", | ||||
| 		 | ||||
| 	}, | ||||
| 	setup(){ | ||||
| 		const store = DashboardConfigurationStore(); | ||||
| 		const uuid = `input_${v4()}`; | ||||
| 		return {store, uuid}; | ||||
| 	}, | ||||
| 	data(){ | ||||
| 		return{ | ||||
| 			value:"", | ||||
| 			invalidFeedback: "", | ||||
| 			showInvalidFeedback: false, | ||||
| 			isValid: false, | ||||
| 			timeout: undefined, | ||||
| 			changed: false, | ||||
| 			updating: false, | ||||
| 		} | ||||
| 	}, | ||||
| 	mounted() { | ||||
| 		this.value = this.store.Configuration.Peers[this.targetData]; | ||||
| 	}, | ||||
| 	methods:{ | ||||
| 		async useValidation(){ | ||||
| 			if(this.changed){ | ||||
| 				await fetchPost("/api/updateDashboardConfigurationItem", { | ||||
| 					section: "Peers", | ||||
| 					key: this.targetData, | ||||
| 					value: this.value | ||||
| 				}, (res) => { | ||||
| 					if (res.status){ | ||||
| 						this.isValid = true; | ||||
| 						this.showInvalidFeedback = false; | ||||
| 						this.store.Configuration.Peers[this.targetData] = this.value | ||||
| 						clearTimeout(this.timeout) | ||||
| 						this.timeout = setTimeout(() => this.isValid = false, 5000); | ||||
|  | ||||
| 					}else{ | ||||
| 						this.isValid = false; | ||||
| 						this.showInvalidFeedback = true; | ||||
| 						this.invalidFeedback = res.message | ||||
| 					} | ||||
| 					this.changed = false | ||||
| 					this.updating = false; | ||||
| 				}) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| 	<div class="form-group mb-2"> | ||||
| 		<label :for="this.uuid" class="text-muted mb-1"> | ||||
| 			<strong><small>{{this.title}}</small></strong> | ||||
| 		</label> | ||||
| 		<input type="text" class="form-control"  | ||||
| 		       :class="{'is-invalid': showInvalidFeedback, 'is-valid': isValid}" | ||||
| 		       :id="this.uuid" | ||||
| 		       v-model="this.value" | ||||
| 		       @keydown="this.changed = true" | ||||
| 		       @blur="useValidation()" | ||||
| 		       :disabled="this.updating" | ||||
| 		> | ||||
| 		<div class="invalid-feedback">{{this.invalidFeedback}}</div> | ||||
| 		<div class="px-2 py-1 text-warning-emphasis bg-warning-subtle border border-warning-subtle rounded-2 d-inline-block mt-1" | ||||
| 			v-if="warning" | ||||
| 		> | ||||
| 			<small><i class="bi bi-exclamation-triangle-fill me-2"></i><span v-html="warningText"></span></small> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </template> | ||||
|  | ||||
| <style scoped> | ||||
|  | ||||
| </style> | ||||
							
								
								
									
										87
									
								
								src/static/app/src/components/setupComponent/totp.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								src/static/app/src/components/setupComponent/totp.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | ||||
| <script> | ||||
| import {fetchGet, fetchPost} from "@/utilities/fetch.js"; | ||||
| import QRCode from "qrcode"; | ||||
|  | ||||
| export default { | ||||
| 	name: "totp", | ||||
| 	async setup(){ | ||||
| 		let l = "" | ||||
| 		await fetchGet("/api/Welcome_GetTotpLink", {}, (res => { | ||||
| 			if (res.status) l = res.data; | ||||
| 		})); | ||||
| 		return {l} | ||||
| 	}, | ||||
| 	mounted() { | ||||
| 		if (this.l) { | ||||
| 			QRCode.toCanvas(document.getElementById('qrcode'), this.l, function (error) {}) | ||||
| 		} | ||||
| 	}, | ||||
| 	data(){ | ||||
| 		return { | ||||
| 			totp: "", | ||||
| 			totpInvalidMessage: "", | ||||
| 			verified: false | ||||
| 		} | ||||
| 	}, | ||||
| 	methods: { | ||||
| 		validateTotp(){ | ||||
| 			 | ||||
| 		}	 | ||||
| 	}, | ||||
| 	watch: { | ||||
| 		totp(newVal){ | ||||
| 			const input = document.querySelector("#totp"); | ||||
| 			input.classList.remove("is-invalid", "is-valid") | ||||
| 			if (newVal.length === 6){ | ||||
| 				console.log(newVal) | ||||
| 				if (/[0-9]{6}/.test(newVal)){ | ||||
| 					fetchPost("/api/Welcome_VerifyTotpLink", { | ||||
| 						totp: newVal | ||||
| 					}, (res) => { | ||||
| 						if (res.status){ | ||||
| 							this.verified = true; | ||||
| 							input.classList.add("is-valid"); | ||||
| 							this.$emit("verified") | ||||
| 						}else{ | ||||
| 							input.classList.add("is-invalid") | ||||
| 							this.totpInvalidMessage = "TOTP does not match." | ||||
| 						} | ||||
| 					}) | ||||
| 				}else{ | ||||
| 					input.classList.add("is-invalid"); | ||||
| 					this.totpInvalidMessage = "TOTP can only contain numbers" | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| 	<div class="mb-3"> | ||||
| 		<p class="mb-2"><small class="text-muted">1. Please scan the following QR Code to generate TOTP</small></p> | ||||
| 		<canvas id="qrcode" class="rounded-3 mb-2"></canvas> | ||||
| 		<div class="p-3 bg-body-secondary rounded-3 border mb-3"> | ||||
| 			<p class="text-muted mb-0"><small>Or you can click the link below:</small> | ||||
| 			</p><a :href="this.l"><code style="line-break: anywhere">{{this.l}}</code></a> | ||||
| 		</div> | ||||
| 		<label for="totp" class="mb-2"><small class="text-muted">2. Enter the TOTP generated by your authenticator to verify</small></label> | ||||
| 		<div class="form-group"> | ||||
| 			<input class="form-control text-center totp" | ||||
| 			       id="totp" maxlength="6" type="text" inputmode="numeric" autocomplete="one-time-code" | ||||
| 			       v-model="this.totp" | ||||
| 			       :disabled="this.verified" | ||||
| 			> | ||||
| 			<div class="invalid-feedback"> | ||||
| 				{{this.totpInvalidMessage}} | ||||
| 			</div> | ||||
| 			<div class="valid-feedback"> | ||||
| 				TOTP verified! | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </template> | ||||
|  | ||||
| <style scoped> | ||||
| 	 | ||||
| </style> | ||||
| @@ -3,15 +3,22 @@ import 'bootstrap/dist/css/bootstrap.css' | ||||
| import 'bootstrap/dist/js/bootstrap.js' | ||||
| import 'bootstrap-icons/font/bootstrap-icons.css' | ||||
|  | ||||
| import { createApp } from 'vue' | ||||
| import {createApp, markRaw} from 'vue' | ||||
| import { createPinia } from 'pinia' | ||||
|  | ||||
| import App from './App.vue' | ||||
| import router from './router' | ||||
|  | ||||
| const app = createApp(App) | ||||
|  | ||||
| app.use(createPinia()) | ||||
| const app = createApp(App) | ||||
| app.use(router) | ||||
| const pinia = createPinia(); | ||||
|  | ||||
| pinia.use(({ store }) => { | ||||
| 	store.$router = markRaw(router) | ||||
| }) | ||||
|  | ||||
| app.use(pinia) | ||||
|  | ||||
|  | ||||
| app.mount('#app') | ||||
|   | ||||
							
								
								
									
										18
									
								
								src/static/app/src/models/WireguardConfigurations.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								src/static/app/src/models/WireguardConfigurations.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| import {fetchGet} from "@/utilities/fetch.js"; | ||||
|  | ||||
| export class WireguardConfigurations{ | ||||
| 	Configurations = undefined; | ||||
| 	constructor() { | ||||
| 		this.Configurations = undefined | ||||
| 	} | ||||
| 	async initialization(){ | ||||
| 		await this.getConfigurations() | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	async getConfigurations(){ | ||||
| 		await fetchGet("/api/getWireguardConfigurations", {}, (res) => { | ||||
| 			if (res.status)  this.Configurations = res.data | ||||
| 		}); | ||||
| 	} | ||||
| } | ||||
| @@ -6,6 +6,10 @@ import ConfigurationList from "@/components/configurationList.vue"; | ||||
| import {fetchGet} from "@/utilities/fetch.js"; | ||||
| import {wgdashboardStore} from "@/stores/wgdashboardStore.js"; | ||||
| import Settings from "@/views/settings.vue"; | ||||
| import {WireguardConfigurationsStore} from "@/stores/WireguardConfigurationsStore.js"; | ||||
| import {DashboardConfigurationStore} from "@/stores/DashboardConfigurationStore.js"; | ||||
| import Setup from "@/views/setup.vue"; | ||||
| import NewConfiguration from "@/views/newConfiguration.vue"; | ||||
|  | ||||
| const checkAuth = async () => { | ||||
|   let result = false | ||||
| @@ -35,27 +39,36 @@ const router = createRouter({ | ||||
|           name: "Settings", | ||||
|           path: '/settings', | ||||
|           component: Settings | ||||
|         }, | ||||
|         { | ||||
|           name: "New Configuration", | ||||
|           path: '/new_configuration', | ||||
|           component: NewConfiguration | ||||
|         } | ||||
|       ] | ||||
|     }, | ||||
|     { | ||||
|       path: '/signin', component: Signin | ||||
|     }, | ||||
|     { | ||||
|       path: '/welcome', component: Setup, | ||||
|       meta: { | ||||
|         requiresAuth: true | ||||
|       }, | ||||
|     } | ||||
|   ] | ||||
| }); | ||||
|  | ||||
| router.beforeEach(async (to, from, next) => { | ||||
|   const store = wgdashboardStore(); | ||||
|   const wireguardConfigurationsStore = WireguardConfigurationsStore(); | ||||
|   const dashboardConfigurationStore = DashboardConfigurationStore(); | ||||
|    | ||||
|    | ||||
|   if (to.meta.requiresAuth){ | ||||
|     if (cookie.getCookie("authToken") && await checkAuth()){ | ||||
|        | ||||
|       console.log(to.name) | ||||
|       if (!store.DashboardConfiguration){ | ||||
|         await store.getDashboardConfiguration() | ||||
|       } | ||||
|       if (!store.WireguardConfigurations && to.name !== "Configuration List"){ | ||||
|         await store.getWireguardConfigurations() | ||||
|       await dashboardConfigurationStore.getConfiguration() | ||||
|       if (!wireguardConfigurationsStore.Configurations && to.name !== "Configuration List"){ | ||||
|         await wireguardConfigurationsStore.getConfigurations(); | ||||
|       } | ||||
|       next() | ||||
|     }else{ | ||||
|   | ||||
							
								
								
									
										28
									
								
								src/static/app/src/stores/DashboardConfigurationStore.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								src/static/app/src/stores/DashboardConfigurationStore.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | ||||
| import {defineStore} from "pinia"; | ||||
| import {fetchGet, fetchPost} from "@/utilities/fetch.js"; | ||||
| import {cookie} from "@/utilities/cookie.js"; | ||||
|  | ||||
| export const DashboardConfigurationStore = defineStore('DashboardConfigurationStore', { | ||||
| 	state: () => ({ | ||||
| 		Configuration: undefined | ||||
| 	}), | ||||
| 	actions: { | ||||
| 		async getConfiguration(){ | ||||
| 			await fetchGet("/api/getDashboardConfiguration", {}, (res) => { | ||||
| 				if (res.status) this.Configuration = res.data | ||||
| 			}); | ||||
| 		}, | ||||
| 		async updateConfiguration(){ | ||||
| 			await fetchPost("/api/updateDashboardConfiguration", { | ||||
| 				DashboardConfiguration: this.Configuration | ||||
| 			}, (res) => { | ||||
| 				console.log(res) | ||||
| 			}) | ||||
| 		}, | ||||
| 		async signOut(){ | ||||
| 			await fetchGet("/api/signout", {}, (res) => { | ||||
| 				this.$router.go('/signin') | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
							
								
								
									
										15
									
								
								src/static/app/src/stores/WireguardConfigurationsStore.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								src/static/app/src/stores/WireguardConfigurationsStore.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| import {defineStore} from "pinia"; | ||||
| import {fetchGet} from "@/utilities/fetch.js"; | ||||
|  | ||||
| export const WireguardConfigurationsStore = defineStore('WireguardConfigurationsStore', { | ||||
| 	state: () => ({ | ||||
| 		Configurations: undefined | ||||
| 	}), | ||||
| 	actions: { | ||||
| 		async getConfigurations(){ | ||||
| 			await fetchGet("/api/getWireguardConfigurations", {}, (res) => { | ||||
| 				if (res.status)  this.Configurations = res.data | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| @@ -1,12 +0,0 @@ | ||||
| import { ref, computed } from 'vue' | ||||
| import { defineStore } from 'pinia' | ||||
|  | ||||
| export const useCounterStore = defineStore('counter', () => { | ||||
|   const count = ref(0) | ||||
|   const doubleCount = computed(() => count.value * 2) | ||||
|   function increment() { | ||||
|     count.value++ | ||||
|   } | ||||
|  | ||||
|   return { count, doubleCount, increment } | ||||
| }) | ||||
| @@ -1,23 +1,20 @@ | ||||
| import {defineStore} from "pinia"; | ||||
| import {fetchGet} from "@/utilities/fetch.js"; | ||||
|  | ||||
|  | ||||
| export const wgdashboardStore = defineStore('WGDashboardStore', { | ||||
| 	state: () => ({  | ||||
| 	state: () => ({ | ||||
| 		WireguardConfigurations: undefined, | ||||
| 		DashboardConfiguration: undefined | ||||
| 	}), | ||||
| 	actions: { | ||||
| 		async getWireguardConfigurations(){ | ||||
| 			await fetchGet("/api/getWireguardConfigurations", {}, (res) => { | ||||
| 				console.log(res.status) | ||||
| 				if (res.status)  this.WireguardConfigurations = res.data | ||||
| 			}) | ||||
| 		}, | ||||
| 		async getDashboardConfiguration(){ | ||||
| 			await fetchGet("/api/getDashboardConfiguration", {}, (res) => { | ||||
| 				console.log(res.status) | ||||
| 				if (res.status)  this.DashboardConfiguration = res.data | ||||
| 			}) | ||||
| 		} | ||||
| 	}, | ||||
| }) | ||||
| 	} | ||||
| }); | ||||
|  | ||||
|  | ||||
|   | ||||
							
								
								
									
										3
									
								
								src/static/app/src/utilities/ipCheck.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								src/static/app/src/utilities/ipCheck.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| export const ipV46RegexCheck = (input) => { | ||||
|  | ||||
| } | ||||
| @@ -1,19 +1,30 @@ | ||||
| <script> | ||||
| import Navbar from "@/components/navbar.vue"; | ||||
| import {wgdashboardStore} from "@/stores/wgdashboardStore.js"; | ||||
| import {WireguardConfigurations} from "@/models/WireguardConfigurations.js"; | ||||
| import {DashboardConfigurationStore} from "@/stores/DashboardConfigurationStore.js"; | ||||
|  | ||||
| export default { | ||||
| 	name: "index", | ||||
| 	components: {Navbar} | ||||
| 	components: {Navbar}, | ||||
| 	async setup(){ | ||||
| 		const dashboardConfigurationStore = DashboardConfigurationStore() | ||||
| 		return {dashboardConfigurationStore} | ||||
| 	} | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| 	<div class="container-fluid flex-grow-1 main" data-bs-theme="dark"> | ||||
| 	<div class="container-fluid flex-grow-1 main" :data-bs-theme="this.dashboardConfigurationStore.Configuration.Server.dashboard_theme"> | ||||
| 		<div class="row h-100"> | ||||
| 			<Navbar></Navbar> | ||||
| 			<main class="col-md-9 ml-sm-auto col-lg-10 px-md-4 mb-4"> | ||||
| 			<main class="col-md-9 ml-sm-auto col-lg-10 px-md-4 overflow-y-scroll mb-0"  style="height: calc(100vh - 50px)"> | ||||
| 				<Suspense> | ||||
| 					<RouterView></RouterView> | ||||
| 					<RouterView v-slot="{Component}"> | ||||
| 						<Transition name="fade2" mode="out-in"> | ||||
| 							<Component :is="Component"></Component> | ||||
| 						</Transition> | ||||
| 					</RouterView> | ||||
| 				</Suspense> | ||||
| 			</main> | ||||
| 		</div> | ||||
|   | ||||
							
								
								
									
										26
									
								
								src/static/app/src/views/newConfiguration.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								src/static/app/src/views/newConfiguration.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| <script> | ||||
| export default { | ||||
| 	name: "newConfiguration" | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| 	<div class="mt-4"> | ||||
| 		<div class="container"> | ||||
| 			<div class="d-flex align-items-center"> | ||||
| 				<div class="mb-3 d-flex align-items-center gap-4"> | ||||
| 					<RouterLink to="/"> | ||||
| 						<h3 class="mb-0 text-body"> | ||||
| 							<i class="bi bi-chevron-left"></i> | ||||
| 						</h3> | ||||
| 					</RouterLink> | ||||
| 					<h3 class="text-body mb-0">New Configuration</h3> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </template> | ||||
|  | ||||
| <style scoped> | ||||
|  | ||||
| </style> | ||||
| @@ -1,17 +1,83 @@ | ||||
| <script> | ||||
| import {wgdashboardStore} from "@/stores/wgdashboardStore.js"; | ||||
| import {DashboardConfigurationStore} from "@/stores/DashboardConfigurationStore.js"; | ||||
| import PeersDefaultSettingsInput from "@/components/settingsComponent/peersDefaultSettingsInput.vue"; | ||||
| import {ipV46RegexCheck} from "@/utilities/ipCheck.js"; | ||||
| import AccountSettingsInputUsername from "@/components/settingsComponent/accountSettingsInputUsername.vue"; | ||||
| import AccountSettingsInputPassword from "@/components/settingsComponent/accountSettingsInputPassword.vue"; | ||||
| import DashboardSettingsInputWireguardConfigurationPath | ||||
| 	from "@/components/settingsComponent/dashboardSettingsInputWireguardConfigurationPath.vue"; | ||||
| import DashboardTheme from "@/components/settingsComponent/dashboardTheme.vue"; | ||||
| import DashboardSettingsInputIPAddressAndPort | ||||
| 	from "@/components/settingsComponent/dashboardSettingsInputIPAddressAndPort.vue"; | ||||
|  | ||||
| export default { | ||||
| 	name: "settings", | ||||
| 	methods: {ipV46RegexCheck}, | ||||
| 	components: { | ||||
| 		DashboardSettingsInputIPAddressAndPort, | ||||
| 		DashboardTheme, | ||||
| 		DashboardSettingsInputWireguardConfigurationPath, | ||||
| 		AccountSettingsInputPassword, AccountSettingsInputUsername, PeersDefaultSettingsInput}, | ||||
| 	setup(){ | ||||
| 		const store = wgdashboardStore(); | ||||
| 		return {store} | ||||
| 		const dashboardConfigurationStore = DashboardConfigurationStore() | ||||
| 		return {dashboardConfigurationStore} | ||||
| 	}, | ||||
| 	watch: { | ||||
| 		// 'dashboardConfigurationStore.Configuration': { | ||||
| 		// 	deep: true, | ||||
| 		// 	handler(){ | ||||
| 		// 		this.dashboardConfigurationStore.updateConfiguration(); | ||||
| 		// 	} | ||||
| 		// } | ||||
| 	} | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| 	<div class="mt-4"> | ||||
| 		<div class="container"> | ||||
| 			<h3 class="mb-3 text-body">Settings</h3> | ||||
| 			<DashboardTheme></DashboardTheme> | ||||
| 			<div class="card mb-4 shadow rounded-3"> | ||||
| 				<p class="card-header">Peers Default Settings</p> | ||||
| 				<div class="card-body"> | ||||
| 					<PeersDefaultSettingsInput targetData="peer_global_dns" title="DNS"></PeersDefaultSettingsInput> | ||||
| 					<PeersDefaultSettingsInput targetData="peer_endpoint_allowed_ip" title="Peer Endpoint Allowed IPs"></PeersDefaultSettingsInput> | ||||
| 					<PeersDefaultSettingsInput targetData="peer_mtu" title="MTU (Max Transmission Unit)"></PeersDefaultSettingsInput> | ||||
| 					<PeersDefaultSettingsInput targetData="peer_keep_alive" title="Persistent Keepalive"></PeersDefaultSettingsInput> | ||||
| 					<PeersDefaultSettingsInput targetData="remote_endpoint" title="Peer Remote Endpoint" | ||||
| 					                           :warning="true" warningText="This will be change globally, and will be apply to all peer's QR code and configuration file." | ||||
| 					></PeersDefaultSettingsInput> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 			<div class="card mb-4 shadow rounded-3"> | ||||
| 				<p class="card-header">WireGuard Configurations Settings</p> | ||||
| 				<div class="card-body"> | ||||
| 					<DashboardSettingsInputWireguardConfigurationPath | ||||
| 						targetData="wg_conf_path" | ||||
| 						title="Configurations Directory" | ||||
| 						:warning="true" | ||||
| 						warning-text="Remember to remove <code>/</code> at the end of your path. e.g <code>/etc/wireguard</code>" | ||||
| 					> | ||||
|  | ||||
| 					</DashboardSettingsInputWireguardConfigurationPath> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 			<div class="card mb-4 shadow rounded-3"> | ||||
| 				<p class="card-header">Account Settings</p> | ||||
| 				<div class="card-body"> | ||||
| 					<AccountSettingsInputUsername targetData="username" | ||||
| 					                              title="Username" | ||||
| 					></AccountSettingsInputUsername> | ||||
| 					<hr> | ||||
| 					<AccountSettingsInputPassword | ||||
| 						targetData="password"> | ||||
| 					</AccountSettingsInputPassword> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </template> | ||||
|  | ||||
| <style scoped> | ||||
|   | ||||
							
								
								
									
										130
									
								
								src/static/app/src/views/setup.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										130
									
								
								src/static/app/src/views/setup.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,130 @@ | ||||
| <script> | ||||
| import {DashboardConfigurationStore} from "@/stores/DashboardConfigurationStore.js"; | ||||
| import QRCode from 'qrcode' | ||||
| import Totp from "@/components/setupComponent/totp.vue"; | ||||
| import {fetchPost} from "@/utilities/fetch.js"; | ||||
| export default { | ||||
| 	name: "setup", | ||||
| 	components: {Totp}, | ||||
| 	setup(){ | ||||
| 		const store = DashboardConfigurationStore(); | ||||
| 		return {store} | ||||
| 	}, | ||||
| 	data(){ | ||||
| 		return { | ||||
| 			setup: { | ||||
| 				username: "", | ||||
| 				newPassword: "", | ||||
| 				repeatNewPassword: "", | ||||
| 				enable_totp: false, | ||||
| 				verified_totp: false | ||||
| 			}, | ||||
| 			loading: false, | ||||
| 			errorMessage: "", | ||||
| 			done: false | ||||
| 		} | ||||
| 	}, | ||||
| 	computed: { | ||||
| 		goodToSubmit(){ | ||||
| 			return this.setup.username  | ||||
| 				&& this.setup.newPassword.length >= 8 | ||||
| 				&& this.setup.repeatNewPassword.length >= 8 | ||||
| 				&& this.setup.newPassword === this.setup.repeatNewPassword | ||||
| 				&& ((this.setup.enable_totp && this.setup.verified_totp) || !this.setup.enable_totp) | ||||
| 		} | ||||
| 	}, | ||||
| 	methods: { | ||||
| 		submit(){ | ||||
| 			this.loading = true | ||||
| 			fetchPost("/api/Welcome_Finish", this.setup, (res) => { | ||||
| 				if (res.status){ | ||||
| 					this.done = true; | ||||
| 					setTimeout(() => { | ||||
| 						this.$router.push('/') | ||||
| 					}, 500) | ||||
| 				}else{ | ||||
| 					document.querySelectorAll("#createAccount input").forEach(x => x.classList.add("is-invalid")) | ||||
| 					this.errorMessage = res.message; | ||||
| 					document.querySelector(".login-container-fluid") | ||||
| 						.scrollTo({ | ||||
| 							top: 0, | ||||
| 							left: 0, | ||||
| 							behavior: "smooth", | ||||
| 						}) | ||||
| 				} | ||||
| 				this.loading = false | ||||
| 			}) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| 	<div class="container-fluid login-container-fluid d-flex main pt-5 overflow-scroll"  | ||||
| 	     :data-bs-theme="this.store.Configuration.Server.dashboard_theme"> | ||||
| 		<div class="mx-auto text-body" style="width: 500px"> | ||||
| 			<span class="dashboardLogo display-4">Nice to meet you!</span> | ||||
| 			<p class="mb-5">Please fill in the following fields to finish setup 😊</p> | ||||
| 			<div> | ||||
| 				<h3>Create an account</h3> | ||||
| 				<div class="alert alert-danger" v-if="this.errorMessage"> | ||||
| 					{{this.errorMessage}} | ||||
| 				</div> | ||||
| 				<div class="d-flex flex-column gap-3"> | ||||
| 					<div id="createAccount"> | ||||
| 						<div class="form-group text-body"> | ||||
| 							<label for="username" class="mb-1 text-muted"> | ||||
| 								<small>Pick an username you like</small></label> | ||||
| 							<input type="text" | ||||
| 							       v-model="this.setup.username" | ||||
| 							       class="form-control" id="username" name="username" placeholder="Maybe something like 'wiredragon'?" required> | ||||
| 						</div> | ||||
| 						<div class="form-group text-body"> | ||||
| 							<label for="password" class="mb-1 text-muted"> | ||||
| 								<small>Create a password (at least 8 characters)</small></label> | ||||
| 							<input type="password" | ||||
| 							       v-model="this.setup.newPassword" | ||||
| 							       class="form-control" id="password" name="password" placeholder="Make sure is strong enough" required> | ||||
| 						</div> | ||||
| 						<div class="form-group text-body"> | ||||
| 							<label for="confirmPassword" class="mb-1 text-muted"> | ||||
| 								<small>Confirm password</small></label> | ||||
| 							<input type="password" | ||||
| 							       v-model="this.setup.repeatNewPassword" | ||||
| 							       class="form-control" id="confirmPassword" name="confirmPassword" placeholder="and you can remember it :)" required> | ||||
| 						</div> | ||||
| 					</div> | ||||
| 					<hr> | ||||
| 					<div class="form-check form-switch"> | ||||
| 						<input class="form-check-input" type="checkbox" role="switch" id="enable_totp"  | ||||
| 						       v-model="this.setup.enable_totp"> | ||||
| 						<label class="form-check-label"  | ||||
| 						       for="enable_totp">Enable 2 Factor Authentication? <strong>Strongly recommended</strong></label> | ||||
| 					</div> | ||||
| 					<Suspense> | ||||
| 						<Transition name="fade"> | ||||
| 							<Totp v-if="this.setup.enable_totp" @verified="this.setup.verified_totp = true"></Totp> | ||||
| 						</Transition> | ||||
| 					</Suspense> | ||||
| 					 | ||||
| 					<button class="btn btn-dark btn-lg mb-5 d-flex btn-brand shadow align-items-center"  | ||||
| 					        ref="signInBtn" | ||||
| 					        :disabled="!this.goodToSubmit || this.loading || this.done" @click="this.submit()"> | ||||
| 						<span class="d-flex align-items-center w-100" v-if="!this.loading && !this.done"> | ||||
| 							Finish<i class="bi bi-chevron-right ms-auto"></i></span> | ||||
| 						<span class="d-flex align-items-center w-100" v-else-if="this.done"> | ||||
| 							Welcome to WGDashboard!</span> | ||||
| 						<span class="d-flex align-items-center w-100" v-else> | ||||
| 							Saving...<span class="spinner-border ms-auto spinner-border-sm" role="status"> | ||||
| 							  <span class="visually-hidden">Loading...</span> | ||||
| 							</span></span> | ||||
| 					</button> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </template> | ||||
|  | ||||
| <style scoped> | ||||
|  | ||||
| </style> | ||||
| @@ -1,26 +1,48 @@ | ||||
| <script> | ||||
| import {fetchPost} from "../utilities/fetch.js"; | ||||
| import {fetchGet, fetchPost} from "../utilities/fetch.js"; | ||||
| import {DashboardConfigurationStore} from "@/stores/DashboardConfigurationStore.js"; | ||||
|  | ||||
| export default { | ||||
| 	name: "signin", | ||||
| 	async setup(){ | ||||
| 		const store = DashboardConfigurationStore() | ||||
| 		let theme = "" | ||||
| 		let totpEnabled = false; | ||||
| 		await fetchGet("/api/getDashboardTheme", {}, (res) => { | ||||
| 			theme = res.data | ||||
| 		}); | ||||
| 		await fetchGet("/api/isTotpEnabled", {}, (res) => { | ||||
| 			totpEnabled = res.data | ||||
| 		});  | ||||
| 		return {store, theme, totpEnabled} | ||||
| 	}, | ||||
| 	data(){ | ||||
| 		return { | ||||
| 			username: "", | ||||
| 			password: "", | ||||
| 			totp: "", | ||||
| 			loginError: false, | ||||
| 			loginErrorMessage: "" | ||||
| 			loginErrorMessage: "", | ||||
| 			loading: false | ||||
| 		} | ||||
| 	}, | ||||
| 	methods: { | ||||
| 		async auth(){ | ||||
| 			if (this.username && this.password){ | ||||
| 			if (this.username && this.password && ((this.totpEnabled && this.totp) || !this.totpEnabled)){ | ||||
| 				this.loading = true | ||||
| 				await fetchPost("/api/authenticate", { | ||||
| 					username: this.username, | ||||
| 					password: this.password | ||||
| 					password: this.password, | ||||
| 					totp: this.totp | ||||
| 				}, (response) => { | ||||
| 					if (response.status){ | ||||
| 						this.loginError = false; | ||||
| 						this.$router.push('/') | ||||
| 						this.$refs["signInBtn"].classList.add("signedIn") | ||||
| 						if (response.message){ | ||||
| 							this.$router.push('/welcome') | ||||
| 						}else{ | ||||
| 							this.$router.push('/') | ||||
| 						} | ||||
| 					}else{ | ||||
| 						this.loginError = true; | ||||
| 						this.loginErrorMessage = response.message; | ||||
| @@ -28,7 +50,9 @@ export default { | ||||
| 							x.classList.remove("is-valid") | ||||
| 							x.classList.add("is-invalid") | ||||
| 						}); | ||||
| 						this.loading = false | ||||
| 					} | ||||
| 					 | ||||
| 				}) | ||||
| 			}else{ | ||||
| 				document.querySelectorAll("input[required]").forEach(x => { | ||||
| @@ -47,25 +71,49 @@ export default { | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
| 	<div class="container-fluid login-container-fluid h-100 d-flex"> | ||||
| 	<div class="container-fluid login-container-fluid d-flex main" :data-bs-theme="this.theme"> | ||||
| 		<div class="login-box m-auto" style="width: 500px;"> | ||||
| 			<h5 class="text-center">Welcome to</h5> | ||||
| 			<h1 class="text-center">WGDashboard</h1> | ||||
| 			<h4 class="mb-0 text-body">Welcome to</h4> | ||||
| 			<span class="dashboardLogo display-3">WGDashboard</span> | ||||
| 			<div class="m-auto"> | ||||
| 				<div class="alert alert-danger mt-2 mb-0" role="alert" v-if="loginError"> | ||||
| 					{{this.loginErrorMessage}} | ||||
| 				</div> | ||||
| 				<form @submit="(e) => {e.preventDefault(); this.auth();}"> | ||||
| 					<div class="form-group"> | ||||
| 						<label for="username" class="text-left" style="font-size: 1rem"><i class="bi bi-person-circle"></i></label> | ||||
| 						<input type="text" v-model="username" class="form-control" id="username" name="username" placeholder="Username" required> | ||||
| 					<div class="form-group text-body"> | ||||
| 						<label for="username" class="text-left" style="font-size: 1rem"> | ||||
| 							<i class="bi bi-person-circle"></i></label> | ||||
| 						<input type="text" v-model="username" class="form-control" id="username" name="username" | ||||
| 						       autocomplete="on" | ||||
| 						       placeholder="Username" required> | ||||
| 					</div> | ||||
| 					<div class="form-group"> | ||||
| 					<div class="form-group text-body"> | ||||
| 						<label for="password" class="text-left" style="font-size: 1rem"><i class="bi bi-key-fill"></i></label> | ||||
| 						<input type="password" v-model="password" class="form-control" id="password" name="password" placeholder="Password" required> | ||||
| 						<input type="password"  | ||||
| 						       v-model="password" class="form-control" id="password" name="password" | ||||
| 						       autocomplete="on" | ||||
| 						       placeholder="Password" required> | ||||
| 					</div> | ||||
| 					<button class="btn btn-dark ms-auto mt-4 w-100 d-flex"> | ||||
| 						Sign In<i class="ms-auto bi bi-chevron-right"></i></button> | ||||
| 					<div class="form-group text-body" v-if="totpEnabled"> | ||||
| 						<label for="totp" class="text-left" style="font-size: 1rem"><i class="bi bi-lock-fill"></i></label> | ||||
| 						<input class="form-control totp" | ||||
| 						       required | ||||
| 						       id="totp" maxlength="6" type="text" inputmode="numeric" autocomplete="one-time-code" | ||||
| 						       placeholder="OTP from your authenticator" | ||||
| 						       v-model="this.totp" | ||||
| 						> | ||||
| 					</div> | ||||
| 					<button class="btn btn-lg btn-dark ms-auto mt-4 w-100 d-flex btn-brand shadow signInBtn" ref="signInBtn"> | ||||
| 						<span v-if="!this.loading" class="d-flex w-100"> | ||||
| 							Sign In<i class="ms-auto bi bi-chevron-right"></i> | ||||
| 						</span> | ||||
| 						<span v-else class="d-flex w-100 align-items-center"> | ||||
| 							Signing In... | ||||
| 							<span class="spinner-border ms-auto spinner-border-sm" role="status"> | ||||
| 							  <span class="visually-hidden">Loading...</span> | ||||
| 							</span> | ||||
| 						</span> | ||||
| 					</button> | ||||
| 				</form> | ||||
| 			</div> | ||||
| 		</div> | ||||
|   | ||||
| @@ -6,6 +6,7 @@ import vue from '@vitejs/plugin-vue' | ||||
|  | ||||
| // https://vitejs.dev/config/ | ||||
| export default defineConfig({ | ||||
|   base: "/static/app/dist", | ||||
|   plugins: [ | ||||
|     vue(), | ||||
|   ], | ||||
| @@ -18,5 +19,14 @@ export default defineConfig({ | ||||
|     proxy: { | ||||
|       '/api': proxy | ||||
|     } | ||||
|   }, | ||||
|   build: { | ||||
|     rollupOptions: { | ||||
|       output: { | ||||
|         entryFileNames: `assets/[name].js`, | ||||
|         chunkFileNames: `assets/[name].js`, | ||||
|         assetFileNames: `assets/[name].[ext]` | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| }) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user