Compare commits
	
		
			96 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 4da3cea59f | |||
| e06f9253cf | |||
| 93a931830f | |||
| 85249e83b1 | |||
| 4ee8de47cf | |||
| 8abaabbe51 | |||
| 0fa043f0df | |||
| 7211133aba | |||
| d957ca52ea | |||
| 231d8ca7ee | |||
| 4a476e93af | |||
| 3bfcd64589 | |||
| d17825cabf | |||
| 6205151d53 | |||
| 4861ca7f3e | |||
| 575685f24d | |||
| 0e9f7a526c | |||
| efb11f6047 | |||
| e95ca8045a | |||
| 2b93f3ea8a | |||
| 2a7a32fda6 | |||
| d7428af1d8 | |||
| 3ca5f6b3b4 | |||
| eb96408d33 | |||
| 11bd1b4f1f | |||
| e3214084f6 | |||
| 3803ac9197 | |||
| e1621b72ad | |||
| b64a8436e7 | |||
| d8ed9a149f | |||
| 9d7125a311 | |||
| 6d71a3ebf4 | |||
| 75adbf73cf | |||
| e5b8557e4c | |||
| 33d7c94648 | |||
| f6b2ae523d | |||
| 0a500f16cd | |||
| 61da5d8110 | |||
| 773b286216 | |||
| d097eccbd9 | |||
| f15282a3f9 | |||
| 0a8945a294 | |||
| b871e4295d | |||
| 0945060a49 | |||
| f7eae96b8c | |||
| c2a50a1480 | |||
| bb99c2dcd1 | |||
| e8382fb579 | |||
| b5321377bd | |||
| 6e26bbbf5f | |||
| 59e178476e | |||
| 824c109a42 | |||
| f48159b31b | |||
| 6195001d4b | |||
| 0afa80345d | |||
| dbb0d177b8 | |||
| 4ffc06db7b | |||
| 588f3bae89 | |||
| d889432ce8 | |||
| 44830f08bc | |||
| 1cd3ebf8c5 | |||
| 558b69eeaa | |||
| 2b22063a81 | |||
| 48cc380504 | |||
| 75473cabe7 | |||
| 6420ffb055 | |||
| a62ee63c83 | |||
| c89f2a2939 | |||
| d51c58867d | |||
| 2fa520fdde | |||
| 5bb7212420 | |||
| 2226705e3f | |||
| 72932955d1 | |||
| 339e2f39d8 | |||
| 12c6d7e3da | |||
| 1624e666e8 | |||
| 86f4cd00ea | |||
| 518aa3eb08 | |||
| 86985ef735 | |||
| fef7daaf96 | |||
| b14fa05adc | |||
| 7c71e94304 | |||
| 8fa489f2bc | |||
| 512acc0b8c | |||
| 567ac07612 | |||
| fa01864d0d | |||
| 91b25c22ca | |||
| 55d6b91cc2 | |||
| 807eb9c267 | |||
| e02d91e662 | |||
| 8d9cf8d1b3 | |||
| d97450cf53 | |||
| fd0a586ed6 | |||
| af6500ccb4 | |||
| 9a3c55d235 | |||
| 86ef277c90 | 
							
								
								
									
										3
									
								
								.env
									
									
									
									
									
								
							
							
						
						
									
										3
									
								
								.env
									
									
									
									
									
								
							| @@ -1,3 +1,4 @@ | |||||||
| SSL=false | SSL=true | ||||||
|  | DOMAIN=kauft.es | ||||||
| DISABLE_PROVIDERS=false | DISABLE_PROVIDERS=false | ||||||
| ENABLE_REGISTER=true | ENABLE_REGISTER=true | ||||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -58,7 +58,6 @@ jspm_packages/ | |||||||
| .yarn/build-state.yml | .yarn/build-state.yml | ||||||
| .yarn/install-state.gz | .yarn/install-state.gz | ||||||
| .pnp.* | .pnp.* | ||||||
| yarn.lock |  | ||||||
| package-lock.json | package-lock.json | ||||||
|  |  | ||||||
| *.sqlite* | *.sqlite* | ||||||
							
								
								
									
										178
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										178
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -2,11 +2,189 @@ | |||||||
|  |  | ||||||
| All notable changes to this project will be documented in this file. Dates are displayed in UTC. | All notable changes to this project will be documented in this file. Dates are displayed in UTC. | ||||||
|  |  | ||||||
|  | #### [0.7.1](https://git.odit.services/kauft.es/linkylinky/compare/0.7.0...0.7.1) | ||||||
|  |  | ||||||
|  | - Added examples to readme [`93a9318`](https://git.odit.services/kauft.es/linkylinky/commit/93a931830f853cea4502b9c74e0d74202efb23ce) | ||||||
|  | - Updated readme w/ native provider [`85249e8`](https://git.odit.services/kauft.es/linkylinky/commit/85249e83b1ec349d740d72394dfbda4ab4d5984e) | ||||||
|  | - Smol bugfix [`e06f925`](https://git.odit.services/kauft.es/linkylinky/commit/e06f9253cf39cd76205da448e6758ca3b51130d6) | ||||||
|  |  | ||||||
|  | #### [0.7.0](https://git.odit.services/kauft.es/linkylinky/compare/0.6.1...0.7.0) | ||||||
|  |  | ||||||
|  | > 12 January 2022 | ||||||
|  |  | ||||||
|  | - Push w/o lockfile [`231d8ca`](https://git.odit.services/kauft.es/linkylinky/commit/231d8ca7ee0da402322afd3ac8ddce58e14943c1) | ||||||
|  | - 🚀RELEASE 0.7.0 [`4ee8de4`](https://git.odit.services/kauft.es/linkylinky/commit/4ee8de47cf227a58e8d2a2eedfc06575a1e5a1c0) | ||||||
|  | - Added the new providers to the readme [`8abaabb`](https://git.odit.services/kauft.es/linkylinky/commit/8abaabbe5170372ca35ff469cee25c65a8f3d729) | ||||||
|  | - Implemented reddit name recognition [`7211133`](https://git.odit.services/kauft.es/linkylinky/commit/7211133aba22315919443d5314594b7ec027c02b) | ||||||
|  | - Now recognizing r/ pattern [`0fa043f`](https://git.odit.services/kauft.es/linkylinky/commit/0fa043f0dfbae8b4095375b39c220a901cc6051f) | ||||||
|  | - Removed yarn lock copy [`d957ca5`](https://git.odit.services/kauft.es/linkylinky/commit/d957ca52ea2332f6cd1c5608467be6245f5082b8) | ||||||
|  |  | ||||||
|  | #### [0.6.1](https://git.odit.services/kauft.es/linkylinky/compare/0.6.0...0.6.1) | ||||||
|  |  | ||||||
|  | > 12 January 2022 | ||||||
|  |  | ||||||
|  | - Working clientside stuff [`#3`](https://git.odit.services/kauft.es/linkylinky/issues/3) | ||||||
|  | - 🚀RELEASE 0.6.1 [`4a476e9`](https://git.odit.services/kauft.es/linkylinky/commit/4a476e93af43c8920448fcfeff3f13edc0bc0681) | ||||||
|  | - Merge pull request 'Update dependency fastify-jwt to v4' (#12) from renovate/fastify-jwt-4.x into main [`3bfcd64`](https://git.odit.services/kauft.es/linkylinky/commit/3bfcd645890253813cc2e4c740f67d577682934b) | ||||||
|  | - Merge pull request 'Update registry.odit.services/hub/library/node Docker tag to v17' (#11) from renovate/registry.odit.services-hub-library-node-17.x into main [`d17825c`](https://git.odit.services/kauft.es/linkylinky/commit/d17825cabf2d46d040bd33a3d1ad5a9d0bccfbe5) | ||||||
|  | - Merge pull request 'Update dependency dotenv to v11' (#13) from renovate/dotenv-11.x into main [`6205151`](https://git.odit.services/kauft.es/linkylinky/commit/6205151d53b374d890276d294d0abc6e6114abe5) | ||||||
|  | - Update dependency knex to ^0.95.0 [`2b93f3e`](https://git.odit.services/kauft.es/linkylinky/commit/2b93f3ea8a9b2f4be73829fb1041e6233d81fa91) | ||||||
|  | - Preleminary emoji support [`3ca5f6b`](https://git.odit.services/kauft.es/linkylinky/commit/3ca5f6b3b468ca739bc77e1d9b751625e03f300d) | ||||||
|  | - Update dependency fastify-jwt to v4 [`575685f`](https://git.odit.services/kauft.es/linkylinky/commit/575685f24d1d0a2c0574741e540f448642990cd5) | ||||||
|  | - Update dependency dotenv to v11 [`4861ca7`](https://git.odit.services/kauft.es/linkylinky/commit/4861ca7f3e3ed3e62ee774d59cb13a38d5e5e1e1) | ||||||
|  | - Update registry.odit.services/hub/library/node Docker tag to v17 [`0e9f7a5`](https://git.odit.services/kauft.es/linkylinky/commit/0e9f7a526ce5188e964e7beb78a5fb63870b473f) | ||||||
|  | - Enabled dep dashboard [`efb11f6`](https://git.odit.services/kauft.es/linkylinky/commit/efb11f6047e37901e20c613393d435f869e8a1f9) | ||||||
|  | - Merge pull request 'Update dependency knex to ^0.95.0' (#5) from renovate/knex-0.x into main [`e95ca80`](https://git.odit.services/kauft.es/linkylinky/commit/e95ca8045af6afcbc9dfcc9976d107a231d12953) | ||||||
|  | - Merge pull request 'Configure Renovate' (#4) from renovate/configure into main [`2a7a32f`](https://git.odit.services/kauft.es/linkylinky/commit/2a7a32fda615fa01751deffadafa6df1a5d0fede) | ||||||
|  | - Added 'clientside' flag to getters and setters [`11bd1b4`](https://git.odit.services/kauft.es/linkylinky/commit/11bd1b4f1f4d079b3ca047472da9da34fb64732d) | ||||||
|  | - Added migration for clientside redirects [`e321408`](https://git.odit.services/kauft.es/linkylinky/commit/e3214084f6dbf3e1595c54a9594e6d5f12323991) | ||||||
|  | - Add renovate.json [`d7428af`](https://git.odit.services/kauft.es/linkylinky/commit/d7428af1d81c820fdca7220959485f361956b86e) | ||||||
|  |  | ||||||
|  | #### [0.6.0](https://git.odit.services/kauft.es/linkylinky/compare/0.5.1...0.6.0) | ||||||
|  |  | ||||||
|  | > 25 September 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE 0.6.0 [`3803ac9`](https://git.odit.services/kauft.es/linkylinky/commit/3803ac9197a1d4c6535606cde43b1e04b3056f9c) | ||||||
|  | - Now with custom opengraph (tm) [`9d7125a`](https://git.odit.services/kauft.es/linkylinky/commit/9d7125a31186f3b7f33df4875db12c67cf91f536) | ||||||
|  | - Added rudementary page content [`e1621b7`](https://git.odit.services/kauft.es/linkylinky/commit/e1621b72ade1fce381f30bcf310a6dcf8d1c65ea) | ||||||
|  | - Added migration for disallowing bot previews [`0a500f1`](https://git.odit.services/kauft.es/linkylinky/commit/0a500f16cde0b0e38a115dd4e76185a2a45dda49) | ||||||
|  | - Added basic bot checking [`61da5d8`](https://git.odit.services/kauft.es/linkylinky/commit/61da5d81108a4b86f92e1c05ca99372c6b188347) | ||||||
|  | - Now with working bad bot detection [`75adbf7`](https://git.odit.services/kauft.es/linkylinky/commit/75adbf73cfbf4fadf2cd632ecb687eff95749f52) | ||||||
|  | - New urls can now be created with disabled preview [`33d7c94`](https://git.odit.services/kauft.es/linkylinky/commit/33d7c94648d062ece3fa437e71ce1ded70324cd8) | ||||||
|  | - Added bot check for native short urls [`f6b2ae5`](https://git.odit.services/kauft.es/linkylinky/commit/f6b2ae523da149b72f2be6d9171584cdc0be6e99) | ||||||
|  | - Added package for bot recognition [`773b286`](https://git.odit.services/kauft.es/linkylinky/commit/773b286216c9e530ccdf3fbfcfe95fa938097be5) | ||||||
|  | - Getters now return the no_preview status [`e5b8557`](https://git.odit.services/kauft.es/linkylinky/commit/e5b8557e4ccae90981e675c2adae81236aa9144e) | ||||||
|  | - Added noindex header to all shorturl routes [`0a8945a`](https://git.odit.services/kauft.es/linkylinky/commit/0a8945a294de2bc5bce9410f3ebf20b8f5abd0a5) | ||||||
|  | - Switched docker base images to odit mirror [`d8ed9a1`](https://git.odit.services/kauft.es/linkylinky/commit/d8ed9a149fddae8e8cc61b78cbc940a9d7ade8f2) | ||||||
|  | - Updated opengraph type [`b64a843`](https://git.odit.services/kauft.es/linkylinky/commit/b64a8436e745d176cc49ef72434dce0f8f1bcc12) | ||||||
|  | - Log user agents [`f15282a`](https://git.odit.services/kauft.es/linkylinky/commit/f15282a3f9bdaaf09a36d8662b64a5bc2740633f) | ||||||
|  | - Removed unused log [`6d71a3e`](https://git.odit.services/kauft.es/linkylinky/commit/6d71a3ebf4d880b8291cacd620f86416cff0d744) | ||||||
|  | - removed logging [`d097ecc`](https://git.odit.services/kauft.es/linkylinky/commit/d097eccbd925e3627fa710d7891d269b372d174c) | ||||||
|  |  | ||||||
|  | #### [0.5.1](https://git.odit.services/kauft.es/linkylinky/compare/0.5.0...0.5.1) | ||||||
|  |  | ||||||
|  | > 21 August 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE 0.5.1 [`b871e42`](https://git.odit.services/kauft.es/linkylinky/commit/b871e4295dc4b067f4f4dd28c3f940c8aa9624e0) | ||||||
|  | - Added endpoint to query over all visited urls [`bb99c2d`](https://git.odit.services/kauft.es/linkylinky/commit/bb99c2dcd1ce89af1bc27a035be5265345d2a05f) | ||||||
|  | - Added optional filtering by provider [`0945060`](https://git.odit.services/kauft.es/linkylinky/commit/0945060a49ed35c038df2c04bbd07174bc5654ca) | ||||||
|  | - Shortened return to avoid variable memory asignment [`f7eae96`](https://git.odit.services/kauft.es/linkylinky/commit/f7eae96b8c3479d333fc00bc36bce451733700db) | ||||||
|  | - Changed object property order (just for us pesky humans to improve readability) [`c2a50a1`](https://git.odit.services/kauft.es/linkylinky/commit/c2a50a14807e223987464c63fac6d67ed98b1a93) | ||||||
|  |  | ||||||
|  | #### [0.5.0](https://git.odit.services/kauft.es/linkylinky/compare/0.4.3...0.5.0) | ||||||
|  |  | ||||||
|  | > 21 August 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE 0.5.0 [`e8382fb`](https://git.odit.services/kauft.es/linkylinky/commit/e8382fb579f769364693974cbe74ce06e9810a9a) | ||||||
|  | - Removed visits 404 resolution [`59e1784`](https://git.odit.services/kauft.es/linkylinky/commit/59e178476e8f7e64de92ae23e859b87680e7af64) | ||||||
|  | - Added migration for visits provider [`f48159b`](https://git.odit.services/kauft.es/linkylinky/commit/f48159b31bcaf7c7449c89fb1e3851bcb04f5e3c) | ||||||
|  | - Reverted last change since we can defer it from the provider [`b532137`](https://git.odit.services/kauft.es/linkylinky/commit/b5321377bde9bd7de13e2eee4dd38a072cf0d4a3) | ||||||
|  | - Changed the way visits entrys get compiled to enable provider visits count searches over the default api [`6e26bbb`](https://git.odit.services/kauft.es/linkylinky/commit/6e26bbbf5f29e4c1eaa56c5dc41ab77c192d0cfb) | ||||||
|  | - Server now inserts provider on visit [`824c109`](https://git.odit.services/kauft.es/linkylinky/commit/824c109a420efd0e1e6de01d93827020f3fe5a5f) | ||||||
|  | - Added package script for migration creation [`6195001`](https://git.odit.services/kauft.es/linkylinky/commit/6195001d4b701d39470ff4be0e265f9afb288e78) | ||||||
|  |  | ||||||
|  | #### [0.4.3](https://git.odit.services/kauft.es/linkylinky/compare/0.4.2...0.4.3) | ||||||
|  |  | ||||||
|  | > 21 August 2021 | ||||||
|  |  | ||||||
|  | - Fixed auth error crashing the entire server thanks to fastify handling stuff not the same way that they do in the docs..... [`#1`](https://git.odit.services/kauft.es/linkylinky/issues/1) | ||||||
|  | - 🚀RELEASE 0.4.3 [`0afa803`](https://git.odit.services/kauft.es/linkylinky/commit/0afa80345d47e09e20d4365634f8532248b2044c) | ||||||
|  |  | ||||||
|  | #### [0.4.2](https://git.odit.services/kauft.es/linkylinky/compare/0.4.1...0.4.2) | ||||||
|  |  | ||||||
|  | > 18 August 2021 | ||||||
|  |  | ||||||
|  | - Changed register api route and added user deletion route [`588f3ba`](https://git.odit.services/kauft.es/linkylinky/commit/588f3bae8980f76461d20e15475ec797078b0b54) | ||||||
|  | - 🚀RELEASE 0.4.2 [`4ffc06d`](https://git.odit.services/kauft.es/linkylinky/commit/4ffc06db7bb84bc7bfc9c57a80927f7201185274) | ||||||
|  |  | ||||||
|  | #### [0.4.1](https://git.odit.services/kauft.es/linkylinky/compare/0.4.0...0.4.1) | ||||||
|  |  | ||||||
|  | > 18 August 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE 0.4.1 [`d889432`](https://git.odit.services/kauft.es/linkylinky/commit/d889432ce8a403f6a609423eaf458a5904dc5b98) | ||||||
|  | - Fixed jwtcount not being recognized [`44830f0`](https://git.odit.services/kauft.es/linkylinky/commit/44830f08bc212f8079b5ac2da3d51eedbe6d5c41) | ||||||
|  |  | ||||||
|  | #### [0.4.0](https://git.odit.services/kauft.es/linkylinky/compare/0.3.0...0.4.0) | ||||||
|  |  | ||||||
|  | > 18 August 2021 | ||||||
|  |  | ||||||
|  | - Basic jwt implementation :party: [`75473ca`](https://git.odit.services/kauft.es/linkylinky/commit/75473cabe79975296e473002e16d3abafbd2635e) | ||||||
|  | - Implemented jwtcount basics [`48cc380`](https://git.odit.services/kauft.es/linkylinky/commit/48cc380504206ea08b3a5082f19ad10bdd7cf773) | ||||||
|  | - Implemented jwt count validation and update on logout [`558b69e`](https://git.odit.services/kauft.es/linkylinky/commit/558b69eeaa78ea015473c674d5f919d64128a930) | ||||||
|  | - Switched to fastify-auth to support multiple auth providers [`6420ffb`](https://git.odit.services/kauft.es/linkylinky/commit/6420ffb055f08348c54cd08a193aba5fe5ebc13a) | ||||||
|  | - 🚀RELEASE 0.4.0 [`1cd3ebf`](https://git.odit.services/kauft.es/linkylinky/commit/1cd3ebf8c5a9413b93ab49c8813dad5c5c547cb2) | ||||||
|  | - All authenticated entpoints now accept jwtauth [`2b22063`](https://git.odit.services/kauft.es/linkylinky/commit/2b22063a81193c3d698525a050ef300e542c1f05) | ||||||
|  |  | ||||||
|  | #### [0.3.0](https://git.odit.services/kauft.es/linkylinky/compare/0.2.0...0.3.0) | ||||||
|  |  | ||||||
|  | > 18 August 2021 | ||||||
|  |  | ||||||
|  | - Added license [`5bb7212`](https://git.odit.services/kauft.es/linkylinky/commit/5bb7212420ba102e743d62a47074191cdb264d2a) | ||||||
|  | - 🚀RELEASE 0.3.0 [`a62ee63`](https://git.odit.services/kauft.es/linkylinky/commit/a62ee63c838a64d7fd28ae110cad474e2c531995) | ||||||
|  | - Added stats api route [`c89f2a2`](https://git.odit.services/kauft.es/linkylinky/commit/c89f2a2939b658ffc688646ccce32fc0e1530583) | ||||||
|  | - Added author info [`2fa520f`](https://git.odit.services/kauft.es/linkylinky/commit/2fa520fdde1d55b2766b4b03e394e21a4f6cbae5) | ||||||
|  | - Updated readme [`2226705`](https://git.odit.services/kauft.es/linkylinky/commit/2226705e3f57d7d2f3ef5a79947c3ab44ec62c38) | ||||||
|  | - Added comments [`d51c588`](https://git.odit.services/kauft.es/linkylinky/commit/d51c58867d7508c84f26c236e38567bc6a4adbed) | ||||||
|  |  | ||||||
|  | #### [0.2.0](https://git.odit.services/kauft.es/linkylinky/compare/0.1.4...0.2.0) | ||||||
|  |  | ||||||
|  | > 14 August 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE 0.2.0 [`7293295`](https://git.odit.services/kauft.es/linkylinky/commit/72932955d15976947dd553c5deba51cbf541b215) | ||||||
|  | - Added ebay provider recognition [`12c6d7e`](https://git.odit.services/kauft.es/linkylinky/commit/12c6d7e3da5cb4de6a597b4639f313b8e4319646) | ||||||
|  | - Ebay provider resolution [`339e2f3`](https://git.odit.services/kauft.es/linkylinky/commit/339e2f39d88d42a961e1e495f319dc0663cdc0a6) | ||||||
|  | - Now recognizing with and without protocol [`1624e66`](https://git.odit.services/kauft.es/linkylinky/commit/1624e666e83e0afe346bcacb105ea1a1535f0690) | ||||||
|  |  | ||||||
|  | #### [0.1.4](https://git.odit.services/kauft.es/linkylinky/compare/0.1.3...0.1.4) | ||||||
|  |  | ||||||
|  | > 14 August 2021 | ||||||
|  |  | ||||||
|  | - Added cors [`518aa3e`](https://git.odit.services/kauft.es/linkylinky/commit/518aa3eb08cb72854812130d45b3b89afb074693) | ||||||
|  | - 🚀RELEASE 0.1.4 [`86f4cd0`](https://git.odit.services/kauft.es/linkylinky/commit/86f4cd00ea517f7e2cacbf69f2163eae597610ee) | ||||||
|  |  | ||||||
|  | #### [0.1.3](https://git.odit.services/kauft.es/linkylinky/compare/0.1.2...0.1.3) | ||||||
|  |  | ||||||
|  | > 14 August 2021 | ||||||
|  |  | ||||||
|  | - Weired knex env fix [`fef7daa`](https://git.odit.services/kauft.es/linkylinky/commit/fef7daaf961e9112aa00f2ad22c3b0518ad258a7) | ||||||
|  | - 🚀RELEASE 0.1.3 [`86985ef`](https://git.odit.services/kauft.es/linkylinky/commit/86985ef7355c2db09dcd6c8e52ee7ebde3ed4128) | ||||||
|  |  | ||||||
|  | #### [0.1.2](https://git.odit.services/kauft.es/linkylinky/compare/0.1.1...0.1.2) | ||||||
|  |  | ||||||
|  | > 14 August 2021 | ||||||
|  |  | ||||||
|  | - API Endpoint to get all short urls [`fa01864`](https://git.odit.services/kauft.es/linkylinky/commit/fa01864d0de7921e741d63c3f5795161279ea478) | ||||||
|  | - 🚀RELEASE 0.1.2 [`b14fa05`](https://git.odit.services/kauft.es/linkylinky/commit/b14fa05adc1ef8e34fd73bf5f495505df7d44b33) | ||||||
|  | - Changed the way that the migration get's triggered [`8fa489f`](https://git.odit.services/kauft.es/linkylinky/commit/8fa489f2bcd9d1ad63ab4581b7f3b27eee3d9038) | ||||||
|  | - Get all api endpoints [`91b25c2`](https://git.odit.services/kauft.es/linkylinky/commit/91b25c22ca035aec5cc2fdc80ca4fbb56a2f8b65) | ||||||
|  | - Added optional db volume to docker-compose [`7c71e94`](https://git.odit.services/kauft.es/linkylinky/commit/7c71e943041d8e3ac9dcf95cad27f2f0bc0b024c) | ||||||
|  | - Prod db now resides in db folder [`512acc0`](https://git.odit.services/kauft.es/linkylinky/commit/512acc0b8c15d4cfbafef21ec20707bac4629ea4) | ||||||
|  |  | ||||||
|  | #### [0.1.1](https://git.odit.services/kauft.es/linkylinky/compare/0.1.0...0.1.1) | ||||||
|  |  | ||||||
|  | > 14 August 2021 | ||||||
|  |  | ||||||
|  | - 🚀RELEASE 0.1.1 [`55d6b91`](https://git.odit.services/kauft.es/linkylinky/commit/55d6b91cc252ee6676d13df9abd8b61283a07f0f) | ||||||
|  | - fix: Dockerfile [`807eb9c`](https://git.odit.services/kauft.es/linkylinky/commit/807eb9c267377df0b5f30f81eb380ef8a1c2add2) | ||||||
|  |  | ||||||
|  | #### [0.1.0](https://git.odit.services/kauft.es/linkylinky/compare/0.0.2...0.1.0) | ||||||
|  |  | ||||||
|  | > 14 August 2021 | ||||||
|  |  | ||||||
|  | - add lockfile [skip-ci] [`d97450c`](https://git.odit.services/kauft.es/linkylinky/commit/d97450cf5389965d8183b5719aad7e9e6f0181ae) | ||||||
|  | - 🚀RELEASE 0.1.0 [`e02d91e`](https://git.odit.services/kauft.es/linkylinky/commit/e02d91e662e090b79b03ec77896100923464aa6b) | ||||||
|  | - smaller Dockerfile [skip-ci] [`86ef277`](https://git.odit.services/kauft.es/linkylinky/commit/86ef277c902aabb52acb5f846cb55406dc4d1095) | ||||||
|  | - Added tag commit args [`fd0a586`](https://git.odit.services/kauft.es/linkylinky/commit/fd0a586ed6e365801979c16d809a8254f64049be) | ||||||
|  | - Now signing release commits [`af6500c`](https://git.odit.services/kauft.es/linkylinky/commit/af6500ccb4792616f28bec77eda5c5e116422604) | ||||||
|  |  | ||||||
| #### 0.0.2 | #### 0.0.2 | ||||||
|  |  | ||||||
|  | > 14 August 2021 | ||||||
|  |  | ||||||
| - Migrated to knex [`823b211`](https://git.odit.services/kauft.es/linkylinky/commit/823b211c8c4fd4de3a6043696ca070101173aa1b) | - Migrated to knex [`823b211`](https://git.odit.services/kauft.es/linkylinky/commit/823b211c8c4fd4de3a6043696ca070101173aa1b) | ||||||
| - Added basics for auth [`6d417fa`](https://git.odit.services/kauft.es/linkylinky/commit/6d417fac1b4d77ab5552b75483cfc787d466e699) | - Added basics for auth [`6d417fa`](https://git.odit.services/kauft.es/linkylinky/commit/6d417fac1b4d77ab5552b75483cfc787d466e699) | ||||||
| - Initial [`ab17f91`](https://git.odit.services/kauft.es/linkylinky/commit/ab17f9148af6811a53bf323172d045cc71ecfb6d) | - Initial [`ab17f91`](https://git.odit.services/kauft.es/linkylinky/commit/ab17f9148af6811a53bf323172d045cc71ecfb6d) | ||||||
|  | - 🚀RELEASE 0.0.2 [`a4bdb4d`](https://git.odit.services/kauft.es/linkylinky/commit/a4bdb4dca4aaba397688d43f1ec3a26cb2143dea) | ||||||
| - Added image builds by drone [`1a8410c`](https://git.odit.services/kauft.es/linkylinky/commit/1a8410c421e30001ae9623c19a8e72e0632ed920) | - Added image builds by drone [`1a8410c`](https://git.odit.services/kauft.es/linkylinky/commit/1a8410c421e30001ae9623c19a8e72e0632ed920) | ||||||
| - Working registration/auth [`81b314a`](https://git.odit.services/kauft.es/linkylinky/commit/81b314ac55e9ab0cd603dda48473c0fe3db6c42e) | - Working registration/auth [`81b314a`](https://git.odit.services/kauft.es/linkylinky/commit/81b314ac55e9ab0cd603dda48473c0fe3db6c42e) | ||||||
| - Implemented short code generation via nanoid [`1550860`](https://git.odit.services/kauft.es/linkylinky/commit/15508606a41057857cce0d2ae16e389432c76960) | - Implemented short code generation via nanoid [`1550860`](https://git.odit.services/kauft.es/linkylinky/commit/15508606a41057857cce0d2ae16e389432c76960) | ||||||
|   | |||||||
							
								
								
									
										26
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										26
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,15 +1,17 @@ | |||||||
| FROM node:16-alpine as run | FROM registry.odit.services/hub/library/node:17.0.1-alpine3.14 | ||||||
|  | WORKDIR /app | ||||||
| COPY package.json . | COPY package.json . | ||||||
| RUN yarn --prod | # COPY yarn.lock . | ||||||
|  | RUN yarn --production --frozen-lockfile | ||||||
|  | COPY migrations ./migrations | ||||||
|  | COPY src ./src | ||||||
|  | COPY knexfile.js ./ | ||||||
|  | RUN mkdir db | ||||||
|  | #  | ||||||
|  | # FROM astefanutti/scratch-node:16.0.0 | ||||||
|  | FROM registry.odit.services/hub/library/node:17.0.1-alpine3.14 | ||||||
|  | WORKDIR /app | ||||||
|  | COPY --from=0 /app /app | ||||||
| ENV NODE_ENV production | ENV NODE_ENV production | ||||||
| EXPOSE 3000 | EXPOSE 3000 | ||||||
|  | ENTRYPOINT ["/bin/sh", "-c", "yarn migrate && node ./src/server.js"] | ||||||
| WORKDIR /app |  | ||||||
|  |  | ||||||
| COPY migrations ./migrations |  | ||||||
| COPY src/server.js ./ |  | ||||||
| COPY knexfile.js ./ |  | ||||||
|  |  | ||||||
| CMD ["node", "./server.js"] |  | ||||||
							
								
								
									
										24
									
								
								LICENSE
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								LICENSE
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | MIT License | ||||||
|  | ----------- | ||||||
|  |  | ||||||
|  | Copyright (c) 2021 ODIT.Services (https://odit.services) | ||||||
|  | Permission is hereby granted, free of charge, to any person | ||||||
|  | obtaining a copy of this software and associated documentation | ||||||
|  | files (the "Software"), to deal in the Software without | ||||||
|  | restriction, including without limitation the rights to use, | ||||||
|  | copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||||
|  | copies of the Software, and to permit persons to whom the | ||||||
|  | Software is furnished to do so, subject to the following | ||||||
|  | conditions: | ||||||
|  |  | ||||||
|  | The above copyright notice and this permission notice shall be | ||||||
|  | included in all copies or substantial portions of the Software. | ||||||
|  |  | ||||||
|  | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||||||
|  | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES | ||||||
|  | OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||||||
|  | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT | ||||||
|  | HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, | ||||||
|  | WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING | ||||||
|  | FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR | ||||||
|  | OTHER DEALINGS IN THE SOFTWARE. | ||||||
							
								
								
									
										23
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										23
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,5 +1,24 @@ | |||||||
| # LinkyLinky 🔗 | <p align="center"> | ||||||
| > A small url shortener, originaly developed for kauft.es |   <img height="150" src="https://git.odit.services/user/avatar/kauft.es/140"> | ||||||
|  |   <h1 align="center">LinkyLinky 🔗</h1> | ||||||
|  |   <h3 align="center">A small url shortener, originaly developed for kauft.es</h3> | ||||||
|  | </p> | ||||||
|  |  | ||||||
|  | ## Features | ||||||
|  | * Url shortinging to 18-byte hex ids | ||||||
|  | * Special shortening "providers" for well-used services | ||||||
|  |  | ||||||
|  | ### Provider patterns | ||||||
|  | > The base pattern is always baseurl/providerOrShortCode | ||||||
|  |  | ||||||
|  | | Provider | Pattern | Resolves to | Example | Notes | | ||||||
|  | | - | - | - | - | - | | ||||||
|  | | Native (Shortcode) | `/id` | Whatever the database entry points to | https://kauft.es/den-hut | None | | ||||||
|  | | YouTube (Video) | `/yt/id` | `https://youtu.be/id` | https://kauft.es/yt/dQw4w9WgXcQ | None | | ||||||
|  | | YouTube (Playlist) | `/ytpl/id` | `https://youtube.com/playlist?list=id` | https://kauft.es/ytpl/PLKIxB9vhdS_3x0A5za3mmu1wdoolgRQ65 | Remember: Private playlists will result in user-side errors | | ||||||
|  | | Amazon | `/a/id` | `https://amazon.de/dp/id` | https://kauft.es/a/B08Z2TXCPY | Recognizes all kind of cursed amazon urls (+smile and others) | | ||||||
|  | | eBay | `/e/id` | `https://ebay.de/itm/id` | https://kauft.es/e/373831556670 | Only tested with German eBay | | ||||||
|  | | Reddit | `/r/id` | `https://redd.it/id` | https://kauft.es/r/4vapin | Powered by the awesome work of u/TheAppleFreak https://kauft.es/r/4vapin | | ||||||
|  |  | ||||||
| ## Dev Setup 🛠 | ## Dev Setup 🛠 | ||||||
| > Runs on port 3000 | > Runs on port 3000 | ||||||
|   | |||||||
| @@ -5,3 +5,5 @@ services: | |||||||
|     #image: registry.odit.services/kauft.es/linkylinky:latest |     #image: registry.odit.services/kauft.es/linkylinky:latest | ||||||
|     ports: |     ports: | ||||||
|       - 3000:3000 |       - 3000:3000 | ||||||
|  |     #volumes: | ||||||
|  |     #  - ./db:/app/db | ||||||
|   | |||||||
| @@ -12,7 +12,7 @@ module.exports = { | |||||||
|   production: { |   production: { | ||||||
|     client: 'sqlite3', |     client: 'sqlite3', | ||||||
|     connection: { |     connection: { | ||||||
|       filename: './db.sqlite3' |       filename: './db/db.sqlite3' | ||||||
|     }, |     }, | ||||||
|     migrations: { |     migrations: { | ||||||
|       tableName: 'knex_migrations' |       tableName: 'knex_migrations' | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								migrations/20210818160424_jwtcount.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								migrations/20210818160424_jwtcount.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  |  | ||||||
|  | exports.up = function(knex) { | ||||||
|  |     return knex.schema.table('users', function (table) { | ||||||
|  |         table.integer("jwtcount").defaultTo(0); | ||||||
|  |     }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | exports.down = function(knex) { | ||||||
|  |    | ||||||
|  | }; | ||||||
							
								
								
									
										10
									
								
								migrations/20210821094219_visits_providers.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								migrations/20210821094219_visits_providers.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  |  | ||||||
|  | exports.up = function(knex) { | ||||||
|  |     return knex.schema.table('visits', function (table) { | ||||||
|  |         table.text('provider').defaultTo("N/A"); | ||||||
|  |     }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | exports.down = function(knex) { | ||||||
|  |    | ||||||
|  | }; | ||||||
							
								
								
									
										10
									
								
								migrations/20210925171533_bot_preview.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								migrations/20210925171533_bot_preview.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  |  | ||||||
|  | exports.up = function(knex) { | ||||||
|  |     return knex.schema.table('urls', function (table) { | ||||||
|  |         table.boolean('no_preview').defaultTo(false); | ||||||
|  |     }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | exports.down = function(knex) { | ||||||
|  |    | ||||||
|  | }; | ||||||
							
								
								
									
										9
									
								
								migrations/20210925182457_clientside_redirect.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								migrations/20210925182457_clientside_redirect.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | exports.up = function(knex) { | ||||||
|  |     return knex.schema.table('urls', function (table) { | ||||||
|  |         table.boolean('clientside').defaultTo(false); | ||||||
|  |     }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | exports.down = function(knex) { | ||||||
|  |    | ||||||
|  | }; | ||||||
							
								
								
									
										31
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										31
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,21 +1,38 @@ | |||||||
| { | { | ||||||
|   "name": "@odit/shortener-backend", |   "name": "@odit/shortener-backend", | ||||||
|   "version": "0.0.2", |   "version": "0.7.1", | ||||||
|   "main": "index.js", |   "main": "index.js", | ||||||
|   "license": "MIT", |   "license": "MIT", | ||||||
|   "private": false, |   "private": false, | ||||||
|  |   "author": { | ||||||
|  |     "name": "ODIT.Services", | ||||||
|  |     "email": "info@odit.services", | ||||||
|  |     "url": "https://odit.services" | ||||||
|  |   }, | ||||||
|  |   "contributors": [ | ||||||
|  |     { | ||||||
|  |       "name": "Nicolai Ort", | ||||||
|  |       "email": "info@nicolai-ort.com", | ||||||
|  |       "url": "https://nicolai-ort.com" | ||||||
|  |     } | ||||||
|  |   ], | ||||||
|   "scripts": { |   "scripts": { | ||||||
|     "dev": "nodemon src/server.js", |     "dev": "nodemon src/server.js", | ||||||
|     "start": "node src/server.js", |     "start": "node src/server.js", | ||||||
|     "migrate": "knex migrate:latest", |     "migrate": "knex migrate:latest", | ||||||
|     "release": "release-it" |     "release": "release-it", | ||||||
|  |     "create:migration": "knex migrate:make" | ||||||
|   }, |   }, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "argon2": "^0.28.2", |     "argon2": "^0.28.2", | ||||||
|     "dotenv": "^10.0.0", |     "dotenv": "^11.0.0", | ||||||
|     "fastify": "^3.20.1", |     "fastify": "^3.20.1", | ||||||
|  |     "fastify-auth": "^1.1.0", | ||||||
|     "fastify-basic-auth": "^2.1.0", |     "fastify-basic-auth": "^2.1.0", | ||||||
|     "knex": "^0.21.21", |     "fastify-cors": "^6.0.2", | ||||||
|  |     "fastify-jwt": "^4.0.0", | ||||||
|  |     "isbot": "^3.3.3", | ||||||
|  |     "knex": "^0.95.0", | ||||||
|     "sqlite3": "^5.0.2", |     "sqlite3": "^5.0.2", | ||||||
|     "uniqid": "^5.3.0" |     "uniqid": "^5.3.0" | ||||||
|   }, |   }, | ||||||
| @@ -28,6 +45,12 @@ | |||||||
|       "commit": true, |       "commit": true, | ||||||
|       "requireCleanWorkingDir": false, |       "requireCleanWorkingDir": false, | ||||||
|       "commitMessage": "🚀RELEASE ${version}", |       "commitMessage": "🚀RELEASE ${version}", | ||||||
|  |       "git.commitArgs": [ | ||||||
|  |         "-S" | ||||||
|  |       ], | ||||||
|  |       "git.tagArgs": [ | ||||||
|  |         "-S" | ||||||
|  |       ], | ||||||
|       "push": true, |       "push": true, | ||||||
|       "tag": true, |       "tag": true, | ||||||
|       "tagName": null, |       "tagName": null, | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								renovate.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								renovate.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | { | ||||||
|  |   "$schema": "https://docs.renovatebot.com/renovate-schema.json", | ||||||
|  |   "dependencyDashboard": true | ||||||
|  | } | ||||||
							
								
								
									
										522
									
								
								src/server.js
									
									
									
									
									
								
							
							
						
						
									
										522
									
								
								src/server.js
									
									
									
									
									
								
							| @@ -2,12 +2,15 @@ const fastify = require('fastify')({ logger: true }) | |||||||
| var uniqid = require('uniqid'); | var uniqid = require('uniqid'); | ||||||
| require('dotenv').config(); | require('dotenv').config(); | ||||||
| const argon2 = require('argon2'); | const argon2 = require('argon2'); | ||||||
|  | const isBot = require('isbot') | ||||||
|  |  | ||||||
| let config = { | let config = { | ||||||
|     domain: process.env.DOMAIN || "localhost:3000", |     domain: process.env.DOMAIN || "localhost:3000", | ||||||
|     https: (process.env.SSL === 'true') || false, |     https: (process.env.SSL === 'true') || false, | ||||||
|  |     env: process.env.NODE_ENV || 'development', | ||||||
|     recognizeProviders: !(process.env.DISABLE_PROVIDERS === 'true'), |     recognizeProviders: !(process.env.DISABLE_PROVIDERS === 'true'), | ||||||
|     registrationEnabled: (process.env.ENABLE_REGISTER === 'true'), |     registrationEnabled: (process.env.ENABLE_REGISTER === 'true'), | ||||||
|  |     jwt_secret: process.env.JWT_SECRET || "pleaseneverusethisdefaultsecret", | ||||||
|     getBaseUrl() { |     getBaseUrl() { | ||||||
|         if (config.https) { |         if (config.https) { | ||||||
|             return `https://${config.domain}`; |             return `https://${config.domain}`; | ||||||
| @@ -15,29 +18,88 @@ let config = { | |||||||
|         return `http://${config.domain}`; |         return `http://${config.domain}`; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | const knexConfiguration = require('../knexfile')[config.env]; | ||||||
| const knex = require('knex')({ | const knex = require('knex')(knexConfiguration); | ||||||
|     client: 'sqlite3', |  | ||||||
|     connection: { |  | ||||||
|         filename: "./dev.sqlite3" |  | ||||||
|     } |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| const authenticate = { realm: 'Short' } | const authenticate = { realm: 'Short' } | ||||||
|  | fastify.register(require('fastify-auth')) | ||||||
| fastify.register(require('fastify-basic-auth'), { validate, authenticate }); | fastify.register(require('fastify-basic-auth'), { validate, authenticate }); | ||||||
|  | fastify.register(require('fastify-jwt'), { | ||||||
|  |     secret: config.jwt_secret | ||||||
|  | }); | ||||||
|  | fastify.register(require('fastify-cors'), { | ||||||
|  |     origin: true, | ||||||
|  |     preflight: true, | ||||||
|  |     preflightContinue: true | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | fastify.decorate('verifyJWT', function async(request, reply, done) { | ||||||
|  |     let token = request.headers.authorization; | ||||||
|  |     if (!token || token == "" || token == "Bearer") { | ||||||
|  |         done(new Error("No jwt provided")); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (token.startsWith("Bearer")) { | ||||||
|  |         token = token.replace("Bearer ", ""); | ||||||
|  |         fastify.log.info("Detected bearer and replaced it") | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fastify.jwt.verify(token, async (err, decoded) => { | ||||||
|  |         if (err) { | ||||||
|  |             fastify.log.error("JWT validation failed:") | ||||||
|  |             done(new Error("JWT Validation failed")); | ||||||
|  |         } | ||||||
|  |         else { | ||||||
|  |             if (!decoded.payload) { | ||||||
|  |                 done(new Error("JWT is empty")); | ||||||
|  |             } | ||||||
|  |             fastify.log.info(`Token verified. User is ${decoded.payload.user}`); | ||||||
|  |  | ||||||
|  |             const jwtcount = (await knex.select('jwtcount') | ||||||
|  |                 .from('users') | ||||||
|  |                 .where('username', '=', decoded.payload.user) | ||||||
|  |                 .limit(1))[0].jwtcount; | ||||||
|  |  | ||||||
|  |             if (decoded.payload.jwtcount < jwtcount) { | ||||||
|  |                 fastify.log.error("Auth ended at jwtcount") | ||||||
|  |                 done(new Error("JWT in no longer valid")) | ||||||
|  |             } | ||||||
|  |             else { | ||||||
|  |                 fastify.log.info(`JWT count verified`); | ||||||
|  |                 request.user = decoded.payload.user; | ||||||
|  |                 done() | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     }) | ||||||
|  | }) | ||||||
|  |  | ||||||
| //Automagic Amazn redirects on /a/ | //Automagic Amazn redirects on /a/ | ||||||
| fastify.get('/a/:id', async (req, res) => { | fastify.get('/a/:id', async (req, res) => { | ||||||
|     res.redirect(302, `https://amazon.de/dp/${req.params.id}`) |     res.redirect(302, `https://amazon.de/dp/${req.params.id}`) | ||||||
|  |     await knex('visits').insert({ shortcode: req.params.id, provider: 'a' }); | ||||||
| }) | }) | ||||||
|  |  | ||||||
| //Automagic Youtube redirects on /yt/ | //Automagic Youtube redirects on /yt/ | ||||||
| fastify.get('/yt/:id', async (req, res) => { | fastify.get('/yt/:id', async (req, res) => { | ||||||
|     res.redirect(302, `https://youtu.be/${req.params.id}`) |     res.redirect(302, `https://youtu.be/${req.params.id}`) | ||||||
|  |     await knex('visits').insert({ shortcode: req.params.id, provider: 'yt' }); | ||||||
| }) | }) | ||||||
| //Automagic Youtube Playlist redirects on /ytpl/ | //Automagic Youtube Playlist redirects on /ytpl/ | ||||||
| fastify.get('/ytpl/:id', async (req, res) => { | fastify.get('/ytpl/:id', async (req, res) => { | ||||||
|     res.redirect(302, `https://youtube.com/playlist?list=${req.params.id}`) |     res.redirect(302, `https://youtube.com/playlist?list=${req.params.id}`) | ||||||
|  |     await knex('visits').insert({ shortcode: req.params.id, provider: 'ytpl' }); | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | //Automagic ebay item redirects on /e/ | ||||||
|  | fastify.get('/e/:id', async (req, res) => { | ||||||
|  |     res.redirect(302, `https://ebay.de/itm/${req.params.id}`) | ||||||
|  |     await knex('visits').insert({ shortcode: req.params.id, provider: 'e' }); | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | //Automagic reddit redirects on /r/ | ||||||
|  | fastify.get('/r/:id', async (req, res) => { | ||||||
|  |     res.redirect(302, `https://redd.it/${req.params.id}`) | ||||||
|  |     await knex('visits').insert({ shortcode: req.params.id, provider: 'r' }); | ||||||
| }) | }) | ||||||
|  |  | ||||||
| //Normal shorturls | //Normal shorturls | ||||||
| @@ -48,15 +110,26 @@ fastify.get('/:shortcode', async (req, res) => { | |||||||
|     if (!shortcode) { |     if (!shortcode) { | ||||||
|         return 404; |         return 404; | ||||||
|     } |     } | ||||||
|     const target = await knex.select('target') |     const target = await knex.select('target', 'no_preview', 'clientside') | ||||||
|         .from('urls') |         .from('urls') | ||||||
|         .where('shortcode', '=', shortcode) |         .where('shortcode', '=', shortcode) | ||||||
|         .limit(1); |         .limit(1); | ||||||
|     if (!target[0]) { |     if (!target[0]) { | ||||||
|         return 404 |         return 404 | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     if (isBot(req.headers['user-agent']) && target[0].no_preview) { | ||||||
|  |         res.type("text/html"); | ||||||
|  |         return bot_html; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (target[0].clientside) { | ||||||
|  |         res.type("text/html"); | ||||||
|  |         return clientside_html.replace("{{targeturl}}", target[0].target) | ||||||
|  |     } | ||||||
|  |  | ||||||
|     res.redirect(302, target[0].target); |     res.redirect(302, target[0].target); | ||||||
|     await knex('visits').insert({ shortcode }); |     await knex('visits').insert({ shortcode, provider: 'native' }); | ||||||
| }) | }) | ||||||
|  |  | ||||||
| //Create new url schema | //Create new url schema | ||||||
| @@ -66,6 +139,8 @@ const newUrlSchema = { | |||||||
|         properties: { |         properties: { | ||||||
|             target: { type: 'string' }, |             target: { type: 'string' }, | ||||||
|             shortcode: { type: 'string' }, |             shortcode: { type: 'string' }, | ||||||
|  |             no_preview: { type: 'boolean' }, | ||||||
|  |             clientside: { type: 'boolean' } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| }; | }; | ||||||
| @@ -74,6 +149,8 @@ const newUrlSchema = { | |||||||
| fastify.post('/api', { newUrlSchema }, async (req, res) => { | fastify.post('/api', { newUrlSchema }, async (req, res) => { | ||||||
|     const target = req.body?.target; |     const target = req.body?.target; | ||||||
|     let shortcode = req.body?.shortcode; |     let shortcode = req.body?.shortcode; | ||||||
|  |     let no_preview = req.body?.no_preview || false; | ||||||
|  |     let clientside = req.body?.clientside || false; | ||||||
|  |  | ||||||
|     //Check if the user provided a target |     //Check if the user provided a target | ||||||
|     if (!target) { |     if (!target) { | ||||||
| @@ -93,7 +170,7 @@ fastify.post('/api', { newUrlSchema }, async (req, res) => { | |||||||
|                 return response; |                 return response; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         const exists = await knex.select('shortcode') |         const exists = await knex.select('shortcode', 'no_preview', 'clientside') | ||||||
|             .from('urls') |             .from('urls') | ||||||
|             .where('target', '=', target) |             .where('target', '=', target) | ||||||
|             .limit(1); |             .limit(1); | ||||||
| @@ -102,7 +179,9 @@ fastify.post('/api', { newUrlSchema }, async (req, res) => { | |||||||
|             return { |             return { | ||||||
|                 url: `${config.getBaseUrl()}/${shortcode}`, |                 url: `${config.getBaseUrl()}/${shortcode}`, | ||||||
|                 shortcode, |                 shortcode, | ||||||
|                 target |                 target, | ||||||
|  |                 no_preview: exists[0].no_preview, | ||||||
|  |                 clientside: exists[0].clientside | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         shortcode = uniqid(); |         shortcode = uniqid(); | ||||||
| @@ -124,12 +203,28 @@ fastify.post('/api', { newUrlSchema }, async (req, res) => { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     //Create a new db entry |     //Create a new db entry | ||||||
|     await knex('urls').insert({ target, shortcode }); |     await knex('urls').insert({ target, shortcode, no_preview, clientside }); | ||||||
|  |  | ||||||
|     return { |     return { | ||||||
|         url: `${config.getBaseUrl()}/${shortcode}`, |         url: `${config.getBaseUrl()}/${shortcode}`, | ||||||
|         shortcode, |         shortcode, | ||||||
|         target |         target, | ||||||
|  |         no_preview, | ||||||
|  |         clientside | ||||||
|  |     } | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | //Get stats api route | ||||||
|  | fastify.get('/api/stats', async (req, res) => { | ||||||
|  |     const urls = await knex.select('shortcode') | ||||||
|  |         .from('urls'); | ||||||
|  |  | ||||||
|  |     const visits = await knex.select('timestamp') | ||||||
|  |         .from('visits'); | ||||||
|  |  | ||||||
|  |     return { | ||||||
|  |         urls: urls.length, | ||||||
|  |         visits: visits.length, | ||||||
|     } |     } | ||||||
| }); | }); | ||||||
|  |  | ||||||
| @@ -142,7 +237,7 @@ fastify.get('/api/:shortcode', async (req, res) => { | |||||||
|         return 404; |         return 404; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const exists = await knex.select('shortcode', 'target') |     const exists = await knex.select('shortcode', 'target', 'no_preview', 'clientside') | ||||||
|         .from('urls') |         .from('urls') | ||||||
|         .where('shortcode', '=', shortcode) |         .where('shortcode', '=', shortcode) | ||||||
|         .limit(1); |         .limit(1); | ||||||
| @@ -158,12 +253,15 @@ fastify.get('/api/:shortcode', async (req, res) => { | |||||||
|         url: `${config.getBaseUrl()}/${exists[0].shortcode}`, |         url: `${config.getBaseUrl()}/${exists[0].shortcode}`, | ||||||
|         shortcode: exists[0].shortcode, |         shortcode: exists[0].shortcode, | ||||||
|         target: exists[0].target, |         target: exists[0].target, | ||||||
|  |         no_preview: exists[0].no_preview, | ||||||
|  |         clientside: exists[0].clientside, | ||||||
|         visits: visits.length |         visits: visits.length | ||||||
|     } |     } | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  |  | ||||||
| //User registration | //User registration | ||||||
| fastify.post('/api/register', async (req, res) => { | fastify.post('/api/auth/register', async (req, res) => { | ||||||
|     if (!config.registrationEnabled) { |     if (!config.registrationEnabled) { | ||||||
|         res.statusCode = 400; |         res.statusCode = 400; | ||||||
|         return "Registration was disabled by your admin"; |         return "Registration was disabled by your admin"; | ||||||
| @@ -195,9 +293,10 @@ fastify.post('/api/register', async (req, res) => { | |||||||
|     return "Done!" |     return "Done!" | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  | //Anything in here has some kind of auth | ||||||
| fastify.after(() => { | fastify.after(() => { | ||||||
|     //Get url api route |     //Get url visits api route | ||||||
|     fastify.get('/api/:shortcode/visits', { onRequest: fastify.basicAuth }, async (req, res) => { |     fastify.get('/api/:shortcode/visits', { onRequest: fastify.auth([fastify.basicAuth, fastify.verifyJWT]) }, async (req, res) => { | ||||||
|         const shortcode = req.params.shortcode; |         const shortcode = req.params.shortcode; | ||||||
|  |  | ||||||
|         //This should never happen but better safe than 500 |         //This should never happen but better safe than 500 | ||||||
| @@ -205,23 +304,26 @@ fastify.after(() => { | |||||||
|             return 404; |             return 404; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         const exists = await knex.select('shortcode', 'target') |         const visits = await knex.select('timestamp', 'provider') | ||||||
|             .from('urls') |  | ||||||
|             .where('shortcode', '=', shortcode) |  | ||||||
|             .limit(1); |  | ||||||
|         if (exists.length == 0) { |  | ||||||
|             return 404; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         const visits = await knex.select('timestamp') |  | ||||||
|             .from('visits') |             .from('visits') | ||||||
|             .where('shortcode', '=', shortcode); |             .where('shortcode', '=', shortcode); | ||||||
|  |  | ||||||
|         return visits; |         return visits; | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|  |     //Get all visits api route | ||||||
|  |     fastify.get('/api/visits', { onRequest: fastify.auth([fastify.basicAuth, fastify.verifyJWT]) }, async (req, res) => { | ||||||
|  |         if (req.query.provider) { | ||||||
|  |             return await knex.select('shortcode', 'provider', 'timestamp') | ||||||
|  |                 .from('visits') | ||||||
|  |                 .where("provider", "=", req.query.provider); | ||||||
|  |         } | ||||||
|  |         return await knex.select('shortcode', 'provider', 'timestamp') | ||||||
|  |             .from('visits'); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|     //Get url api route |     //Get url api route | ||||||
|     fastify.delete('/api/:shortcode', async (req, res) => { |     fastify.delete('/api/:shortcode', { onRequest: fastify.auth([fastify.basicAuth, fastify.verifyJWT]) }, async (req, res) => { | ||||||
|         const shortcode = req.params.shortcode; |         const shortcode = req.params.shortcode; | ||||||
|  |  | ||||||
|         //This should never happen but better safe than 500 |         //This should never happen but better safe than 500 | ||||||
| @@ -237,6 +339,63 @@ fastify.after(() => { | |||||||
|         return true; |         return true; | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|  |     //Get all urls api route | ||||||
|  |     fastify.get('/api', { onRequest: fastify.auth([fastify.basicAuth, fastify.verifyJWT]) }, async (req, res) => { | ||||||
|  |         urls = await knex.select('target', 'shortcode', 'no_preview', 'clientside') | ||||||
|  |             .from('urls'); | ||||||
|  |  | ||||||
|  |         for (let url of urls) { | ||||||
|  |             url.url = `${config.getBaseUrl()}/${url.shortcode}` | ||||||
|  |             if (req.query.showVisits) { | ||||||
|  |                 url.visits = (await knex.select('timestamp') | ||||||
|  |                     .from('visits') | ||||||
|  |                     .where('shortcode', '=', url.shortcode)).length; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return urls; | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     fastify.post('/api/auth/login', { onRequest: fastify.auth([fastify.basicAuth]) }, async (req, reply) => { | ||||||
|  |         const jwtcount = (await knex.select('jwtcount') | ||||||
|  |             .from('users') | ||||||
|  |             .where('username', '=', req.user) | ||||||
|  |             .limit(1))[0].jwtcount; | ||||||
|  |         const payload = { | ||||||
|  |             user: req.user, | ||||||
|  |             jwtcount | ||||||
|  |         }; | ||||||
|  |         const token = fastify.jwt.sign({ payload }) | ||||||
|  |         reply.send({ token }); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     fastify.post('/api/auth/check', { onRequest: fastify.auth([fastify.basicAuth, fastify.verifyJWT]) }, (req, reply) => { | ||||||
|  |         return "logged in"; | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     fastify.post('/api/auth/logout', { onRequest: fastify.auth([fastify.basicAuth, fastify.verifyJWT]) }, async (req, reply) => { | ||||||
|  |         let jwtcount = (await knex.select('jwtcount') | ||||||
|  |             .from('users') | ||||||
|  |             .where('username', '=', req.user) | ||||||
|  |             .limit(1))[0].jwtcount; | ||||||
|  |         jwtcount += 1; | ||||||
|  |         await knex('users') | ||||||
|  |             .where('username', '=', req.user) | ||||||
|  |             .update({ | ||||||
|  |                 jwtcount | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |         return "Done!"; | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     fastify.post('/api/auth/deleteme', { onRequest: fastify.auth([fastify.basicAuth, fastify.verifyJWT]) }, async (req, reply) => { | ||||||
|  |         await knex('users') | ||||||
|  |             .where('username', '=', req.user) | ||||||
|  |             .delete(); | ||||||
|  |  | ||||||
|  |         return "Done!"; | ||||||
|  |     }); | ||||||
|  |  | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -266,7 +425,7 @@ function checkKnownProviders(target) { | |||||||
|             target |             target | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     const amazonID = target.match(/https?:\/\/(www|smile|)\.?(amazon|smile)\.(de)(?:(?:\/.*\/|\/)(?:dp|gp))(\/product\/|\/)([A-Z0-9]+)/); |     const amazonID = target.match(/(?:https?:\/\/|)(www|smile|)\.?(amazon|smile)\.(de)(?:(?:\/.*\/|\/)(?:dp|gp))(\/product\/|\/)([A-Z0-9]+)/); | ||||||
|     if (amazonID) { |     if (amazonID) { | ||||||
|         const shortcode = `a/${amazonID[5]}` |         const shortcode = `a/${amazonID[5]}` | ||||||
|         return { |         return { | ||||||
| @@ -275,6 +434,25 @@ function checkKnownProviders(target) { | |||||||
|             target |             target | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |     const ebayID = target.match(/(?:[ebay]*(?:[\/]|[itm=])|^)([0-9]{9,12})/); | ||||||
|  |     if (ebayID) { | ||||||
|  |         const shortcode = `e/${ebayID[1]}` | ||||||
|  |         return { | ||||||
|  |             url: `${config.getBaseUrl()}/${shortcode}`, | ||||||
|  |             shortcode, | ||||||
|  |             target | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const redditID = target.match(/(((((?:https?:)?\/\/)((?!about\.)[\w-]+?\.)?([rc]edd(?:it\.com|\.it)))(?!\/(?:blog|about|code|advertising|jobs|rules|wiki|contact|buttons|gold|page|help|prefs|user|message|widget)\b)((?:\/r\/[\w-]+\b(?<!\/pcmasterrace))|(?:\/tb))?(\/comments)??(\/\w{2,7}\b(?<!\/46ijrl)(?<!\/wiki))((?:(?!\))\S)*)))/); | ||||||
|  |     if (redditID) { | ||||||
|  |         const shortcode = `r${redditID[9]}` | ||||||
|  |         return { | ||||||
|  |             url: `${config.getBaseUrl()}/${shortcode}`, | ||||||
|  |             shortcode, | ||||||
|  |             target | ||||||
|  |         } | ||||||
|  |     } | ||||||
|     return null; |     return null; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -294,17 +472,291 @@ async function validate(username, password, req, reply) { | |||||||
|     if (!(await argon2.verify(user[0].password, password))) { |     if (!(await argon2.verify(user[0].password, password))) { | ||||||
|         return new Error('Wrong credentials'); |         return new Error('Wrong credentials'); | ||||||
|     } |     } | ||||||
|  |     req.user = username; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | const bot_html = `<!DOCTYPE html> | ||||||
|  |         <html lang="en"> | ||||||
|  |         <head> | ||||||
|  |             <meta charset="UTF-8"> | ||||||
|  |             <meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||||
|  |             <meta name="viewport" content="width=device-width, initial-scale=1.0"> | ||||||
|  |             <meta property="og:title" content="LinkyLinky"> | ||||||
|  |             <meta property="og:site_name" content="LinkyLinky by Kauft.es"> | ||||||
|  |             <meta property="og:url" content="https://kauft.es/"> | ||||||
|  |             <meta property="og:description" content="LinkyLinky by Kauft.es is a custom url shortener. You're reading this, b/c someone doesn't want their shorturl to be indexed by bots/crawlers/spiders."> | ||||||
|  |             <meta property="og:type" content="article"> | ||||||
|  |             <meta property="og:image" content="https://kauft.es/dashboard/icon_128.png"> | ||||||
|  |  | ||||||
|  |             <title>LinkyLinky</title> | ||||||
|  |         </head> | ||||||
|  |         <body> | ||||||
|  |         <p align="center"> | ||||||
|  |         <img height="150" src="https://kauft.es/dashboard/icon_128.png"> | ||||||
|  |         <h1 align="center">LinkyLinky 🔗</h1> | ||||||
|  |         <h3 align="center">A small url shortener, originaly developed for kauft.es</h3> | ||||||
|  |         <p>LinkyLinky by Kauft.es is a custom url shortener.<br> | ||||||
|  |         You're reading this, b/c someone doesn't want their shorturl to be indexed by bots/crawlers/spiders.</p> | ||||||
|  |         </p>         | ||||||
|  |         </body> | ||||||
|  |         </html> | ||||||
|  |         `; | ||||||
|  |  | ||||||
|  | const clientside_html = `<!DOCTYPE html> | ||||||
|  |     <html lang="en"> | ||||||
|  |  | ||||||
|  |     <head> | ||||||
|  |         <meta charset="UTF-8"> | ||||||
|  |         <meta name="viewport" content="width=device-width, initial-scale=1.0"> | ||||||
|  |         <meta http-equiv="X-UA-Compatible" content="ie=edge"> | ||||||
|  |         <meta name="robot" content="no-index"> | ||||||
|  |     </head> | ||||||
|  |  | ||||||
|  |     <body> | ||||||
|  |     <style> | ||||||
|  |     body { | ||||||
|  |         background: black; | ||||||
|  |         overflow: hidden; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .containCube { | ||||||
|  |         position: relative; | ||||||
|  |         height: 100vh; | ||||||
|  |         width: 100%; | ||||||
|  |         perspective: 800px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .containCube .cube { | ||||||
|  |         position: absolute; | ||||||
|  |         height: 300px; | ||||||
|  |         width: 300px; | ||||||
|  |         top: 0; | ||||||
|  |         right: 0; | ||||||
|  |         bottom: 0; | ||||||
|  |         left: 0; | ||||||
|  |         margin: auto; | ||||||
|  |         box-sizing: border-box; | ||||||
|  |         transform-style: preserve-3d; | ||||||
|  |         transform-origin: 50% 50%; | ||||||
|  |         -webkit-animation: rotate 20s ease-in-out infinite alternate; | ||||||
|  |         animation: rotate 20s ease-in-out infinite alternate; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .containCube .cubeGroup { | ||||||
|  |         position: absolute; | ||||||
|  |         display: grid; | ||||||
|  |         box-sizing: border-box; | ||||||
|  |         height: 100%; | ||||||
|  |         width: 100%; | ||||||
|  |         color: white; | ||||||
|  |         text-shadow: 0 0 1px black; | ||||||
|  |         border: 3px dashed white; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .containCube .cubeGroup h1 { | ||||||
|  |         margin: auto; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .containCube .cube-front { | ||||||
|  |         transform: translatez(150px); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .containCube .cube-rear { | ||||||
|  |         transform: translatez(-150px) rotatey(180deg); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .containCube .cube-right { | ||||||
|  |         transform-origin: 100%; | ||||||
|  |         transform: rotatey(90deg) translatex(150px); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .containCube .cube-left { | ||||||
|  |         transform-origin: 0%; | ||||||
|  |         transform: rotatey(-90deg) translatex(-150px); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .containCube .cube-bottom { | ||||||
|  |         transform-origin: 50% 100%; | ||||||
|  |         transform: rotatex(-90deg) translatey(150px); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .containCube .cube-top { | ||||||
|  |         transform-origin: 50% 0%; | ||||||
|  |         transform: rotatex(90deg) translatey(-150px); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .containCube .cube-1 { | ||||||
|  |         background: red; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .containCube .cube-2 { | ||||||
|  |         background: red; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .containCube .cube-3 { | ||||||
|  |         background: red; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .containCube .cube-4 { | ||||||
|  |         background: red; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .containCube .cube-5 { | ||||||
|  |         background: red; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .containCube .cube-6 { | ||||||
|  |         background: red; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @-webkit-keyframes rotate { | ||||||
|  |         10% { | ||||||
|  |             transform: rotate3d(1, 1, 0, 320deg); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         20% { | ||||||
|  |             transform: rotate3d(1, 0, 0, -90deg); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         30% { | ||||||
|  |             transform: rotate3d(1, 1, 0, 440deg); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         40% { | ||||||
|  |             transform: rotate3d(1, 0, 0, -180deg); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         50% { | ||||||
|  |             transform: rotate3d(1, 1, 0, 460deg); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         60% { | ||||||
|  |             transform: rotate3d(0, 1, 0, -195deg); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         70% { | ||||||
|  |             transform: rotate3d(1, 1, 0, 172deg); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         80% { | ||||||
|  |             transform: rotate3d(0, 1, 0, -360deg); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         90% { | ||||||
|  |             transform: rotate3d(1, 1, 0, 280deg); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @keyframes rotate { | ||||||
|  |         10% { | ||||||
|  |             transform: rotate3d(1, 1, 0, 320deg); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         20% { | ||||||
|  |             transform: rotate3d(1, 0, 0, -90deg); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         30% { | ||||||
|  |             transform: rotate3d(1, 1, 0, 440deg); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         40% { | ||||||
|  |             transform: rotate3d(1, 0, 0, -180deg); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         50% { | ||||||
|  |             transform: rotate3d(1, 1, 0, 460deg); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         60% { | ||||||
|  |             transform: rotate3d(0, 1, 0, -195deg); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         70% { | ||||||
|  |             transform: rotate3d(1, 1, 0, 172deg); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         80% { | ||||||
|  |             transform: rotate3d(0, 1, 0, -360deg); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         90% { | ||||||
|  |             transform: rotate3d(1, 1, 0, 280deg); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @-webkit-keyframes rotateZed { | ||||||
|  |         20% { | ||||||
|  |             transform: translatez(100px); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         40% { | ||||||
|  |             transform: translatez(-100px); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         60% { | ||||||
|  |             transform: translatez(100px); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         80% { | ||||||
|  |             transform: translatez(-100px); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @keyframes rotateZed { | ||||||
|  |         20% { | ||||||
|  |             transform: translatez(100px); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         40% { | ||||||
|  |             transform: translatez(-100px); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         60% { | ||||||
|  |             transform: translatez(100px); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         80% { | ||||||
|  |             transform: translatez(-100px); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | </style> | ||||||
|  |  | ||||||
|  |         <div class="containCube"> | ||||||
|  |             <div class="cube"> | ||||||
|  |                 <div class="cubeGroup cube-front cube-1"> | ||||||
|  |                     <h1>kauft.es</h1> | ||||||
|  |                 </div> | ||||||
|  |                 <div class="cubeGroup cube-top cube-2"> | ||||||
|  |                     <h1>kauft.es</h1> | ||||||
|  |                 </div> | ||||||
|  |                 <div class="cubeGroup cube-left cube-3"> | ||||||
|  |                     <h1>kauft.es</h1> | ||||||
|  |                 </div> | ||||||
|  |                 <div class="cubeGroup cube-right cube-4"> | ||||||
|  |                     <h1>kauft.es</h1> | ||||||
|  |                 </div> | ||||||
|  |                 <div class="cubeGroup cube-rear cube-5"> | ||||||
|  |                     <h1>kauft.es</h1> | ||||||
|  |                 </div> | ||||||
|  |                 <div class="cubeGroup cube-bottom cube-6"> | ||||||
|  |                     <h1>kauft.es</h1> | ||||||
|  |                 </div> | ||||||
|  |             </div> | ||||||
|  |         </div> | ||||||
|  |  | ||||||
|  |         <script> | ||||||
|  |             setTimeout(function () { | ||||||
|  |                 location.replace("{{targeturl}}"); | ||||||
|  |             }, 3000);//Delay 3 seconds | ||||||
|  |         </script> | ||||||
|  |     </body> | ||||||
|  |  | ||||||
|  |     </html> | ||||||
|  | `; | ||||||
|  |  | ||||||
| // Run the server! | // Run the server! | ||||||
| const start = async () => { | const start = async () => { | ||||||
|     try { |  | ||||||
|         await knex.migrate.latest() |  | ||||||
|     } catch (err) { |  | ||||||
|         fastify.log.error(err) |  | ||||||
|         process.exit(1) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     try { |     try { | ||||||
|         await fastify.listen(3000, '0.0.0.0') |         await fastify.listen(3000, '0.0.0.0') | ||||||
|     } catch (err) { |     } catch (err) { | ||||||
|   | |||||||
							
								
								
									
										837
									
								
								src/🥵🥵🥵.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										837
									
								
								src/🥵🥵🥵.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,837 @@ | |||||||
|  | const emojis = [ | ||||||
|  |   "😀", | ||||||
|  |   "😃", | ||||||
|  |   "😄", | ||||||
|  |   "😁", | ||||||
|  |   "😆", | ||||||
|  |   "😅", | ||||||
|  |   "😂", | ||||||
|  |   "🤣", | ||||||
|  |   "😊", | ||||||
|  |   "😇", | ||||||
|  |   "🙂", | ||||||
|  |   "🙃", | ||||||
|  |   "😉", | ||||||
|  |   "😌", | ||||||
|  |   "😍", | ||||||
|  |   "😘", | ||||||
|  |   "😗", | ||||||
|  |   "😙", | ||||||
|  |   "😚", | ||||||
|  |   "😋", | ||||||
|  |   "😛", | ||||||
|  |   "😝", | ||||||
|  |   "😜", | ||||||
|  |   "🤪", | ||||||
|  |   "🤨", | ||||||
|  |   "🧐", | ||||||
|  |   "🤓", | ||||||
|  |   "😎", | ||||||
|  |   "🤩", | ||||||
|  |   "😏", | ||||||
|  |   "🤯", | ||||||
|  |   "😳", | ||||||
|  |   "🤗", | ||||||
|  |   "🤔", | ||||||
|  |   "🤭", | ||||||
|  |   "🤫", | ||||||
|  |   "🤥", | ||||||
|  |   "😶", | ||||||
|  |   "😯", | ||||||
|  |   "😮", | ||||||
|  |   "😲", | ||||||
|  |   "😴", | ||||||
|  |   "🤤", | ||||||
|  |   "🤐", | ||||||
|  |   "🤧", | ||||||
|  |   "😷", | ||||||
|  |   "🤒", | ||||||
|  |   "🤑", | ||||||
|  |   "🤠", | ||||||
|  |   "😈", | ||||||
|  |   "👿", | ||||||
|  |   "👹", | ||||||
|  |   "👺", | ||||||
|  |   "🤡", | ||||||
|  |   "💩", | ||||||
|  |   "👻", | ||||||
|  |   "👽", | ||||||
|  |   "👾", | ||||||
|  |   "🤖", | ||||||
|  |   "🎃", | ||||||
|  |   "😺", | ||||||
|  |   "😸", | ||||||
|  |   "😹", | ||||||
|  |   "😻", | ||||||
|  |   "😼", | ||||||
|  |   "😽", | ||||||
|  |   "🙀", | ||||||
|  |   "😿", | ||||||
|  |   "😾", | ||||||
|  |   "🤲", | ||||||
|  |   "👐", | ||||||
|  |   "🙌", | ||||||
|  |   "👏", | ||||||
|  |   "🤝", | ||||||
|  |   "👍", | ||||||
|  |   "👊", | ||||||
|  |   "✊", | ||||||
|  |   "🤛", | ||||||
|  |   "🤜", | ||||||
|  |   "🤞", | ||||||
|  |   "✌️", | ||||||
|  |   "🤟", | ||||||
|  |   "🤘", | ||||||
|  |   "👈", | ||||||
|  |   "👉", | ||||||
|  |   "👆", | ||||||
|  |   "👇", | ||||||
|  |   "☝️", | ||||||
|  |   "✋", | ||||||
|  |   "🤚", | ||||||
|  |   "🖐", | ||||||
|  |   "🖖", | ||||||
|  |   "👋", | ||||||
|  |   "🤙", | ||||||
|  |   "💪", | ||||||
|  |   "✍️", | ||||||
|  |   "💁", | ||||||
|  |   "🙅", | ||||||
|  |   "🙋", | ||||||
|  |   "🤷", | ||||||
|  |   "💃", | ||||||
|  |   "🕺🏻", | ||||||
|  |   "🧥", | ||||||
|  |   "👚", | ||||||
|  |   "👕", | ||||||
|  |   "👖", | ||||||
|  |   "👔", | ||||||
|  |   "👗", | ||||||
|  |   "👙", | ||||||
|  |   "👘", | ||||||
|  |   "👠", | ||||||
|  |   "👡", | ||||||
|  |   "👢", | ||||||
|  |   "👞", | ||||||
|  |   "👟", | ||||||
|  |   "🧦", | ||||||
|  |   "🧤", | ||||||
|  |   "🧣", | ||||||
|  |   "🎩", | ||||||
|  |   "🧢", | ||||||
|  |   "👒", | ||||||
|  |   "🎓", | ||||||
|  |   "⛑", | ||||||
|  |   "👑", | ||||||
|  |   "💍", | ||||||
|  |   "👝", | ||||||
|  |   "👛", | ||||||
|  |   "👜", | ||||||
|  |   "💼", | ||||||
|  |   "🎒", | ||||||
|  |   "👓", | ||||||
|  |   "🕶", | ||||||
|  |   "🌂", | ||||||
|  |   "💢", | ||||||
|  |   "♨️", | ||||||
|  |   "❗️", | ||||||
|  |   "❕", | ||||||
|  |   "❓", | ||||||
|  |   "❔", | ||||||
|  |   "⁉️", | ||||||
|  |   "🔅", | ||||||
|  |   "🔆", | ||||||
|  |   "🚸", | ||||||
|  |   "🔰", | ||||||
|  |   "♻️", | ||||||
|  |   "✅", | ||||||
|  |   "🌀", | ||||||
|  |   "💤", | ||||||
|  |   "🚮", | ||||||
|  |   "🆗", | ||||||
|  |   "🔣", | ||||||
|  |   "🆒", | ||||||
|  |   "🔜", | ||||||
|  |   "✔️", | ||||||
|  |   "🔈", | ||||||
|  |   "🔉", | ||||||
|  |   "🔊", | ||||||
|  |   "📢", | ||||||
|  |   "💬", | ||||||
|  |   "💭", | ||||||
|  |   "🗯", | ||||||
|  |   "🀄️", | ||||||
|  |   "🃏", | ||||||
|  |   "🎴", | ||||||
|  |   "🐶", | ||||||
|  |   "🐱", | ||||||
|  |   "🐭", | ||||||
|  |   "🐹", | ||||||
|  |   "🐰", | ||||||
|  |   "🦊", | ||||||
|  |   "🐻", | ||||||
|  |   "🐼", | ||||||
|  |   "🐨", | ||||||
|  |   "🐯", | ||||||
|  |   "🦁", | ||||||
|  |   "🐮", | ||||||
|  |   "🐷", | ||||||
|  |   "🐽", | ||||||
|  |   "🐸", | ||||||
|  |   "🐵", | ||||||
|  |   "🙈", | ||||||
|  |   "🙉", | ||||||
|  |   "🙊", | ||||||
|  |   "🐒", | ||||||
|  |   "🐔", | ||||||
|  |   "🐧", | ||||||
|  |   "🐦", | ||||||
|  |   "🐤", | ||||||
|  |   "🐣", | ||||||
|  |   "🐥", | ||||||
|  |   "🦆", | ||||||
|  |   "🦅", | ||||||
|  |   "🦉", | ||||||
|  |   "🦇", | ||||||
|  |   "🐺", | ||||||
|  |   "🐗", | ||||||
|  |   "🐴", | ||||||
|  |   "🦄", | ||||||
|  |   "🐝", | ||||||
|  |   "🐌", | ||||||
|  |   "🐞", | ||||||
|  |   "🐜", | ||||||
|  |   "🦗", | ||||||
|  |   "🕷", | ||||||
|  |   "🕸", | ||||||
|  |   "🦂", | ||||||
|  |   "🐢", | ||||||
|  |   "🐍", | ||||||
|  |   "🦎", | ||||||
|  |   "🦖", | ||||||
|  |   "🦕", | ||||||
|  |   "🐙", | ||||||
|  |   "🦑", | ||||||
|  |   "🦐", | ||||||
|  |   "🦀", | ||||||
|  |   "🐡", | ||||||
|  |   "🐠", | ||||||
|  |   "🐟", | ||||||
|  |   "🐬", | ||||||
|  |   "🐳", | ||||||
|  |   "🐋", | ||||||
|  |   "🦈", | ||||||
|  |   "🐊", | ||||||
|  |   "🐅", | ||||||
|  |   "🐆", | ||||||
|  |   "🦓", | ||||||
|  |   "🦍", | ||||||
|  |   "🐘", | ||||||
|  |   "🦏", | ||||||
|  |   "🐪", | ||||||
|  |   "🐫", | ||||||
|  |   "🦒", | ||||||
|  |   "🐃", | ||||||
|  |   "🐂", | ||||||
|  |   "🐄", | ||||||
|  |   "🐎", | ||||||
|  |   "🐖", | ||||||
|  |   "🐏", | ||||||
|  |   "🐑", | ||||||
|  |   "🐐", | ||||||
|  |   "🦌", | ||||||
|  |   "🐕", | ||||||
|  |   "🐩", | ||||||
|  |   "🐈", | ||||||
|  |   "🐓", | ||||||
|  |   "🦃", | ||||||
|  |   "🕊", | ||||||
|  |   "🐇", | ||||||
|  |   "🐁", | ||||||
|  |   "🐀", | ||||||
|  |   "🐿", | ||||||
|  |   "🦔", | ||||||
|  |   "🐾", | ||||||
|  |   "🐉", | ||||||
|  |   "🐲", | ||||||
|  |   "🌵", | ||||||
|  |   "🎄", | ||||||
|  |   "🌲", | ||||||
|  |   "🌳", | ||||||
|  |   "🌴", | ||||||
|  |   "🌱", | ||||||
|  |   "🌿", | ||||||
|  |   "☘️", | ||||||
|  |   "🍀", | ||||||
|  |   "🎍", | ||||||
|  |   "🎋", | ||||||
|  |   "🍃", | ||||||
|  |   "🍂", | ||||||
|  |   "🍁", | ||||||
|  |   "🍄", | ||||||
|  |   "🐚", | ||||||
|  |   "🌾", | ||||||
|  |   "💐", | ||||||
|  |   "🌷", | ||||||
|  |   "🌹", | ||||||
|  |   "🥀", | ||||||
|  |   "🌺", | ||||||
|  |   "🌸", | ||||||
|  |   "🌼", | ||||||
|  |   "🌻", | ||||||
|  |   "🌞", | ||||||
|  |   "🌝", | ||||||
|  |   "🌛", | ||||||
|  |   "🌜", | ||||||
|  |   "🌚", | ||||||
|  |   "🌕", | ||||||
|  |   "🌖", | ||||||
|  |   "🌗", | ||||||
|  |   "🌘", | ||||||
|  |   "🌑", | ||||||
|  |   "🌒", | ||||||
|  |   "🌓", | ||||||
|  |   "🌔", | ||||||
|  |   "🌙", | ||||||
|  |   "🌎", | ||||||
|  |   "🌏", | ||||||
|  |   "💫", | ||||||
|  |   "🌟", | ||||||
|  |   "⚡️", | ||||||
|  |   "☄️", | ||||||
|  |   "🌪", | ||||||
|  |   "☀️", | ||||||
|  |   "🌤", | ||||||
|  |   "⛅️", | ||||||
|  |   "🌥", | ||||||
|  |   "☁️", | ||||||
|  |   "🌦", | ||||||
|  |   "🌧", | ||||||
|  |   "⛈", | ||||||
|  |   "🌩", | ||||||
|  |   "🌨", | ||||||
|  |   "❄️", | ||||||
|  |   "☃️", | ||||||
|  |   "⛄️", | ||||||
|  |   "🌬", | ||||||
|  |   "💨", | ||||||
|  |   "☔️", | ||||||
|  |   "☂️", | ||||||
|  |   "🌊", | ||||||
|  |   "🌫", | ||||||
|  |   "🕐", | ||||||
|  |   "🕑", | ||||||
|  |   "🕒", | ||||||
|  |   "🕓", | ||||||
|  |   "🕔", | ||||||
|  |   "🕕", | ||||||
|  |   "🕖", | ||||||
|  |   "🕗", | ||||||
|  |   "🕘", | ||||||
|  |   "🕙", | ||||||
|  |   "🕚", | ||||||
|  |   "🕛", | ||||||
|  |   "🕜", | ||||||
|  |   "🕝", | ||||||
|  |   "🕞", | ||||||
|  |   "🕟", | ||||||
|  |   "🕠", | ||||||
|  |   "🕡", | ||||||
|  |   "🕢", | ||||||
|  |   "🕣", | ||||||
|  |   "🕤", | ||||||
|  |   "🕥", | ||||||
|  |   "🕦", | ||||||
|  |   "🕧", | ||||||
|  |   "🍏", | ||||||
|  |   "🍎", | ||||||
|  |   "🍐", | ||||||
|  |   "🍊", | ||||||
|  |   "🍋", | ||||||
|  |   "🍌", | ||||||
|  |   "🍉", | ||||||
|  |   "🍇", | ||||||
|  |   "🍓", | ||||||
|  |   "🍈", | ||||||
|  |   "🍒", | ||||||
|  |   "🍍", | ||||||
|  |   "🥥", | ||||||
|  |   "🥝", | ||||||
|  |   "🍅", | ||||||
|  |   "🥑", | ||||||
|  |   "🥦", | ||||||
|  |   "🥒", | ||||||
|  |   "🌶", | ||||||
|  |   "🌽", | ||||||
|  |   "🥕", | ||||||
|  |   "🥔", | ||||||
|  |   "🍠", | ||||||
|  |   "🥐", | ||||||
|  |   "🍞", | ||||||
|  |   "🥖", | ||||||
|  |   "🥨", | ||||||
|  |   "🧀", | ||||||
|  |   "🥚", | ||||||
|  |   "🍳", | ||||||
|  |   "🥞", | ||||||
|  |   "🥓", | ||||||
|  |   "🥩", | ||||||
|  |   "🍗", | ||||||
|  |   "🍖", | ||||||
|  |   "🌭", | ||||||
|  |   "🍔", | ||||||
|  |   "🍟", | ||||||
|  |   "🍕", | ||||||
|  |   "🥪", | ||||||
|  |   "🥙", | ||||||
|  |   "🌮", | ||||||
|  |   "🌯", | ||||||
|  |   "🥗", | ||||||
|  |   "🥘", | ||||||
|  |   "🥫", | ||||||
|  |   "🍝", | ||||||
|  |   "🍲", | ||||||
|  |   "🍛", | ||||||
|  |   "🍣", | ||||||
|  |   "🍱", | ||||||
|  |   "🥟", | ||||||
|  |   "🍤", | ||||||
|  |   "🍙", | ||||||
|  |   "🍚", | ||||||
|  |   "🍘", | ||||||
|  |   "🍥", | ||||||
|  |   "🥠", | ||||||
|  |   "🍢", | ||||||
|  |   "🍡", | ||||||
|  |   "🍧", | ||||||
|  |   "🍨", | ||||||
|  |   "🍦", | ||||||
|  |   "🥧", | ||||||
|  |   "🍰", | ||||||
|  |   "🎂", | ||||||
|  |   "🍮", | ||||||
|  |   "🍭", | ||||||
|  |   "🍬", | ||||||
|  |   "🍫", | ||||||
|  |   "🍿", | ||||||
|  |   "🍩", | ||||||
|  |   "🍪", | ||||||
|  |   "🌰", | ||||||
|  |   "🥜", | ||||||
|  |   "🍯", | ||||||
|  |   "🥛", | ||||||
|  |   "🍼", | ||||||
|  |   "☕️", | ||||||
|  |   "🍵", | ||||||
|  |   "🥤", | ||||||
|  |   "🍶", | ||||||
|  |   "🍺", | ||||||
|  |   "🍻", | ||||||
|  |   "🥂", | ||||||
|  |   "🍷", | ||||||
|  |   "🥃", | ||||||
|  |   "🍸", | ||||||
|  |   "🍹", | ||||||
|  |   "🍾", | ||||||
|  |   "🥄", | ||||||
|  |   "🍴", | ||||||
|  |   "🍽", | ||||||
|  |   "🥣", | ||||||
|  |   "🥡", | ||||||
|  |   "🥢", | ||||||
|  |   "⚽️", | ||||||
|  |   "🏀", | ||||||
|  |   "🏈", | ||||||
|  |   "⚾", | ||||||
|  |   "️🎾", | ||||||
|  |   "🏐", | ||||||
|  |   "🏉", | ||||||
|  |   "🎱", | ||||||
|  |   "🏓", | ||||||
|  |   "🏸", | ||||||
|  |   "🏑", | ||||||
|  |   "🏒", | ||||||
|  |   "🏏", | ||||||
|  |   "🥅", | ||||||
|  |   "⛳️", | ||||||
|  |   "🏹", | ||||||
|  |   "🎣", | ||||||
|  |   "🥊", | ||||||
|  |   "🥋", | ||||||
|  |   "🎽", | ||||||
|  |   "🛷", | ||||||
|  |   "⛸", | ||||||
|  |   "🥌", | ||||||
|  |   "🎿", | ||||||
|  |   "⛷", | ||||||
|  |   "🏂", | ||||||
|  |   "🏆", | ||||||
|  |   "🥇", | ||||||
|  |   "🥈", | ||||||
|  |   "🥉", | ||||||
|  |   "🏅", | ||||||
|  |   "🎖", | ||||||
|  |   "🏵", | ||||||
|  |   "🎗", | ||||||
|  |   "🎫", | ||||||
|  |   "🎟", | ||||||
|  |   "🎪", | ||||||
|  |   "🎭", | ||||||
|  |   "🎨", | ||||||
|  |   "🎬", | ||||||
|  |   "🎤", | ||||||
|  |   "🎧", | ||||||
|  |   "🎼", | ||||||
|  |   "🎹", | ||||||
|  |   "🥁", | ||||||
|  |   "🎷", | ||||||
|  |   "🎺", | ||||||
|  |   "🎸", | ||||||
|  |   "🎻", | ||||||
|  |   "🎲", | ||||||
|  |   "🎳", | ||||||
|  |   "🎮", | ||||||
|  |   "🎰", | ||||||
|  |   "🚗", | ||||||
|  |   "🚕", | ||||||
|  |   "🚙", | ||||||
|  |   "🚌", | ||||||
|  |   "🚎", | ||||||
|  |   "🏎", | ||||||
|  |   "🚓", | ||||||
|  |   "🚑", | ||||||
|  |   "🚒", | ||||||
|  |   "🚐", | ||||||
|  |   "🚚", | ||||||
|  |   "🚛", | ||||||
|  |   "🚜", | ||||||
|  |   "🛴", | ||||||
|  |   "🚲", | ||||||
|  |   "🛵", | ||||||
|  |   "🏍", | ||||||
|  |   "🚔", | ||||||
|  |   "🚍", | ||||||
|  |   "🚘", | ||||||
|  |   "🚖", | ||||||
|  |   "🚡", | ||||||
|  |   "🚠", | ||||||
|  |   "🚟", | ||||||
|  |   "🚃", | ||||||
|  |   "🚋", | ||||||
|  |   "🚞", | ||||||
|  |   "🚝", | ||||||
|  |   "🚄", | ||||||
|  |   "🚅", | ||||||
|  |   "🚈", | ||||||
|  |   "🚂", | ||||||
|  |   "🚆", | ||||||
|  |   "🚇", | ||||||
|  |   "🚊", | ||||||
|  |   "🚉", | ||||||
|  |   "✈️", | ||||||
|  |   "🛫", | ||||||
|  |   "🛬", | ||||||
|  |   "🛩", | ||||||
|  |   "💺", | ||||||
|  |   "🛰", | ||||||
|  |   "🚀", | ||||||
|  |   "🛸", | ||||||
|  |   "🚁", | ||||||
|  |   "🛶", | ||||||
|  |   "⛵️", | ||||||
|  |   "🚤", | ||||||
|  |   "🛥", | ||||||
|  |   "🛳", | ||||||
|  |   "⛴", | ||||||
|  |   "🚢", | ||||||
|  |   "⚓️", | ||||||
|  |   "⛽️", | ||||||
|  |   "🚧", | ||||||
|  |   "🚦", | ||||||
|  |   "🚥", | ||||||
|  |   "🚏", | ||||||
|  |   "🗺", | ||||||
|  |   "🗿", | ||||||
|  |   "🗽", | ||||||
|  |   "🗼", | ||||||
|  |   "🏰", | ||||||
|  |   "🏯", | ||||||
|  |   "🏟", | ||||||
|  |   "🎡", | ||||||
|  |   "🎢", | ||||||
|  |   "🎠", | ||||||
|  |   "⛲️", | ||||||
|  |   "⛱", | ||||||
|  |   "🏖", | ||||||
|  |   "🏝", | ||||||
|  |   "🏜", | ||||||
|  |   "🌋", | ||||||
|  |   "⛰", | ||||||
|  |   "🏔", | ||||||
|  |   "🗻", | ||||||
|  |   "🏕", | ||||||
|  |   "⛺️", | ||||||
|  |   "🏠", | ||||||
|  |   "🏡", | ||||||
|  |   "🏘", | ||||||
|  |   "🏚", | ||||||
|  |   "🏗", | ||||||
|  |   "🏭", | ||||||
|  |   "🏢", | ||||||
|  |   "🏬", | ||||||
|  |   "🏣", | ||||||
|  |   "🏤", | ||||||
|  |   "🏥", | ||||||
|  |   "🏦", | ||||||
|  |   "🏨", | ||||||
|  |   "🏪", | ||||||
|  |   "🏫", | ||||||
|  |   "🏩", | ||||||
|  |   "💒", | ||||||
|  |   "🏛", | ||||||
|  |   "⛪️", | ||||||
|  |   "🕌", | ||||||
|  |   "🕍", | ||||||
|  |   "🕋", | ||||||
|  |   "⛩", | ||||||
|  |   "🛤", | ||||||
|  |   "🛣", | ||||||
|  |   "🗾", | ||||||
|  |   "🎑", | ||||||
|  |   "🏞", | ||||||
|  |   "🌅", | ||||||
|  |   "🌄", | ||||||
|  |   "🌠", | ||||||
|  |   "🎇", | ||||||
|  |   "🎆", | ||||||
|  |   "🌇", | ||||||
|  |   "🏙", | ||||||
|  |   "🌃", | ||||||
|  |   "🌌", | ||||||
|  |   "🌉", | ||||||
|  |   "🌁", | ||||||
|  |   "⌚️", | ||||||
|  |   "📱", | ||||||
|  |   "📲", | ||||||
|  |   "💻", | ||||||
|  |   "⌨️", | ||||||
|  |   "🖥", | ||||||
|  |   "🖨", | ||||||
|  |   "🖱", | ||||||
|  |   "🖲", | ||||||
|  |   "🕹", | ||||||
|  |   "🗜", | ||||||
|  |   "💽", | ||||||
|  |   "💾", | ||||||
|  |   "💿", | ||||||
|  |   "📀", | ||||||
|  |   "📼", | ||||||
|  |   "📷", | ||||||
|  |   "📸", | ||||||
|  |   "📹", | ||||||
|  |   "🎥", | ||||||
|  |   "📽", | ||||||
|  |   "🎞", | ||||||
|  |   "📞", | ||||||
|  |   "☎️", | ||||||
|  |   "📟", | ||||||
|  |   "📠", | ||||||
|  |   "📺", | ||||||
|  |   "📻", | ||||||
|  |   "🎙", | ||||||
|  |   "🎚", | ||||||
|  |   "🎛", | ||||||
|  |   "⏱", | ||||||
|  |   "⏲", | ||||||
|  |   "⏰", | ||||||
|  |   "🕰", | ||||||
|  |   "⏳", | ||||||
|  |   "📡", | ||||||
|  |   "🔋", | ||||||
|  |   "🔌", | ||||||
|  |   "🔦", | ||||||
|  |   "🕯", | ||||||
|  |   "🛢", | ||||||
|  |   "💵", | ||||||
|  |   "💴", | ||||||
|  |   "💶", | ||||||
|  |   "💷", | ||||||
|  |   "💰", | ||||||
|  |   "💳", | ||||||
|  |   "💎", | ||||||
|  |   "⚖️", | ||||||
|  |   "🔧", | ||||||
|  |   "🔨", | ||||||
|  |   "🛠", | ||||||
|  |   "⛏", | ||||||
|  |   "🔩", | ||||||
|  |   "⚙️", | ||||||
|  |   "⛓", | ||||||
|  |   "🔫", | ||||||
|  |   "💣", | ||||||
|  |   "🔪", | ||||||
|  |   "🗡", | ||||||
|  |   "⚔️", | ||||||
|  |   "🛡", | ||||||
|  |   "🏺", | ||||||
|  |   "📿", | ||||||
|  |   "💈", | ||||||
|  |   "⚗️", | ||||||
|  |   "🔭", | ||||||
|  |   "🔬", | ||||||
|  |   "🕳", | ||||||
|  |   "💊", | ||||||
|  |   "💉", | ||||||
|  |   "🌡", | ||||||
|  |   "🚽", | ||||||
|  |   "🚰", | ||||||
|  |   "🚿", | ||||||
|  |   "🛁", | ||||||
|  |   "🛎", | ||||||
|  |   "🔑", | ||||||
|  |   "🗝", | ||||||
|  |   "🚪", | ||||||
|  |   "🛋", | ||||||
|  |   "🛏", | ||||||
|  |   "🛌", | ||||||
|  |   "🖼", | ||||||
|  |   "🛍", | ||||||
|  |   "🛒", | ||||||
|  |   "🎏", | ||||||
|  |   "🎀", | ||||||
|  |   "🎎", | ||||||
|  |   "🏮", | ||||||
|  |   "🎐", | ||||||
|  |   "✉️", | ||||||
|  |   "📩", | ||||||
|  |   "📨", | ||||||
|  |   "📧", | ||||||
|  |   "💌", | ||||||
|  |   "📥", | ||||||
|  |   "📤", | ||||||
|  |   "📦", | ||||||
|  |   "🏷", | ||||||
|  |   "📪", | ||||||
|  |   "📫", | ||||||
|  |   "📬", | ||||||
|  |   "📭", | ||||||
|  |   "📮", | ||||||
|  |   "📯", | ||||||
|  |   "📜", | ||||||
|  |   "📃", | ||||||
|  |   "📄", | ||||||
|  |   "📑", | ||||||
|  |   "📊", | ||||||
|  |   "📈", | ||||||
|  |   "📉", | ||||||
|  |   "🗒", | ||||||
|  |   "🗓", | ||||||
|  |   "📆", | ||||||
|  |   "📅", | ||||||
|  |   "🗑", | ||||||
|  |   "📇", | ||||||
|  |   "🗃", | ||||||
|  |   "🗳", | ||||||
|  |   "🗄", | ||||||
|  |   "📋", | ||||||
|  |   "📁", | ||||||
|  |   "📂", | ||||||
|  |   "🗂", | ||||||
|  |   "🗞", | ||||||
|  |   "📰", | ||||||
|  |   "📓", | ||||||
|  |   "📔", | ||||||
|  |   "📒", | ||||||
|  |   "📕", | ||||||
|  |   "📗", | ||||||
|  |   "📘", | ||||||
|  |   "📙", | ||||||
|  |   "📖", | ||||||
|  |   "🔖", | ||||||
|  |   "🔗", | ||||||
|  |   "📎", | ||||||
|  |   "🖇", | ||||||
|  |   "📐", | ||||||
|  |   "📏", | ||||||
|  |   "📌", | ||||||
|  |   "📍", | ||||||
|  |   "✂️", | ||||||
|  |   "🖊", | ||||||
|  |   "🖋", | ||||||
|  |   "✒️", | ||||||
|  |   "🖌", | ||||||
|  |   "🖍", | ||||||
|  |   "📝", | ||||||
|  |   "✏️", | ||||||
|  |   "🔍", | ||||||
|  |   "🔎", | ||||||
|  |   "🔏", | ||||||
|  |   "🔐", | ||||||
|  |   "🔒", | ||||||
|  |   "🔓", | ||||||
|  |   "🏳️", | ||||||
|  |   "🏁", | ||||||
|  |   "🚩", | ||||||
|  |   "❤️", | ||||||
|  |   "🧡", | ||||||
|  |   "💛", | ||||||
|  |   "💚", | ||||||
|  |   "💙", | ||||||
|  |   "💜", | ||||||
|  |   "🖤", | ||||||
|  |   "💔", | ||||||
|  |   "❣️", | ||||||
|  |   "💕", | ||||||
|  |   "💞", | ||||||
|  |   "💓", | ||||||
|  |   "💗", | ||||||
|  |   "💖", | ||||||
|  |   "💘", | ||||||
|  |   "💝", | ||||||
|  |   "✨", | ||||||
|  |   "💥", | ||||||
|  |   "🔥", | ||||||
|  |   "🌈", | ||||||
|  |   "💧", | ||||||
|  |   "💦", | ||||||
|  |   "🍑", | ||||||
|  |   "🍆", | ||||||
|  |   "🍜", | ||||||
|  |   "🎯", | ||||||
|  |   "🚨", | ||||||
|  |   "💡", | ||||||
|  |   "💸", | ||||||
|  |   "🔮", | ||||||
|  |   "🎁", | ||||||
|  |   "🎈", | ||||||
|  |   "🎊", | ||||||
|  |   "🎉", | ||||||
|  |   "📚", | ||||||
|  |   "🔔", | ||||||
|  |   "📣", | ||||||
|  |   "💅", | ||||||
|  |   "👀", | ||||||
|  |   "🧠", | ||||||
|  |   "🏳️🌈", | ||||||
|  |   "🐛", | ||||||
|  |   "🦋", | ||||||
|  |   "💄", | ||||||
|  |   "💋", | ||||||
|  |   "👄", | ||||||
|  |   "💯", | ||||||
|  |   "🎵", | ||||||
|  |   "🎶", | ||||||
|  |   "🙏", | ||||||
|  |   "⭐️" | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | const getEmoji = (number = 10) => { | ||||||
|  |   let id = ""; | ||||||
|  |   while (number > 0) { | ||||||
|  |     id += emojis[Math.floor(Math.random() * emojis.length)]; | ||||||
|  |     number--; | ||||||
|  |   } | ||||||
|  |   return id; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | module.exports = getEmoji; | ||||||
		Reference in New Issue
	
	Block a user