43 Commits

Author SHA1 Message Date
4ee8de47cf 🚀RELEASE 0.7.0
All checks were successful
continuous-integration/drone/push Build is passing
2022-01-12 18:04:27 +01:00
8abaabbe51 Added the new providers to the readme 2022-01-12 18:04:14 +01:00
0fa043f0df Now recognizing r/ pattern
All checks were successful
continuous-integration/drone/push Build is passing
2022-01-12 17:54:50 +01:00
7211133aba Implemented reddit name recognition 2022-01-12 17:51:21 +01:00
d957ca52ea Removed yarn lock copy
All checks were successful
continuous-integration/drone/push Build is passing
2022-01-12 17:40:29 +01:00
231d8ca7ee Push w/o lockfile
Some checks failed
continuous-integration/drone/push Build is failing
2022-01-12 17:33:48 +01:00
4a476e93af 🚀RELEASE 0.6.1
Some checks failed
continuous-integration/drone/push Build is failing
2022-01-12 17:31:58 +01:00
3bfcd64589 Merge pull request 'Update dependency fastify-jwt to v4' (#12) from renovate/fastify-jwt-4.x into main
All checks were successful
continuous-integration/drone/push Build is passing
Reviewed-on: #12
2022-01-12 16:30:00 +00:00
d17825cabf 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
All checks were successful
continuous-integration/drone/push Build is passing
Reviewed-on: #11
2022-01-12 16:29:36 +00:00
6205151d53 Merge pull request 'Update dependency dotenv to v11' (#13) from renovate/dotenv-11.x into main
All checks were successful
continuous-integration/drone/push Build is passing
Reviewed-on: #13
2022-01-12 16:29:24 +00:00
4861ca7f3e Update dependency dotenv to v11
All checks were successful
continuous-integration/drone/push Build is passing
continuous-integration/drone/pr Build is passing
2022-01-11 07:02:02 +00:00
575685f24d Update dependency fastify-jwt to v4
All checks were successful
continuous-integration/drone/push Build is passing
continuous-integration/drone/pr Build is passing
2021-11-25 09:00:46 +00:00
0e9f7a526c Update registry.odit.services/hub/library/node Docker tag to v17
All checks were successful
continuous-integration/drone/push Build is passing
continuous-integration/drone/pr Build is passing
2021-11-18 19:00:32 +00:00
efb11f6047 Enabled dep dashboard
All checks were successful
continuous-integration/drone/push Build is passing
2021-10-15 16:23:21 +02:00
e95ca8045a Merge pull request 'Update dependency knex to ^0.95.0' (#5) from renovate/knex-0.x into main
All checks were successful
continuous-integration/drone/push Build is passing
Reviewed-on: #5
2021-10-15 14:22:03 +00:00
2b93f3ea8a Update dependency knex to ^0.95.0
All checks were successful
continuous-integration/drone/push Build is passing
continuous-integration/drone/pr Build is passing
2021-10-15 14:18:41 +00:00
2a7a32fda6 Merge pull request 'Configure Renovate' (#4) from renovate/configure into main
All checks were successful
continuous-integration/drone/push Build is passing
Reviewed-on: #4
2021-10-15 14:17:08 +00:00
d7428af1d8 Add renovate.json
All checks were successful
continuous-integration/drone/push Build is passing
continuous-integration/drone/pr Build is passing
2021-10-15 14:16:47 +00:00
3ca5f6b3b4 Preleminary emoji support
All checks were successful
continuous-integration/drone/push Build is passing
2021-09-25 19:29:23 +02:00
eb96408d33 Working clientside stuff
All checks were successful
continuous-integration/drone/push Build is passing
closes #3
2021-09-25 18:45:52 +02:00
11bd1b4f1f Added 'clientside' flag to getters and setters 2021-09-25 18:28:49 +02:00
e3214084f6 Added migration for clientside redirects
ref #3
2021-09-25 18:25:53 +02:00
3803ac9197 🚀RELEASE 0.6.0
All checks were successful
continuous-integration/drone/push Build is passing
2021-09-25 18:03:09 +02:00
e1621b72ad Added rudementary page content
All checks were successful
continuous-integration/drone/push Build is passing
2021-09-25 17:44:34 +02:00
b64a8436e7 Updated opengraph type
ref #2
2021-09-25 17:42:44 +02:00
d8ed9a149f Switched docker base images to odit mirror 2021-09-25 17:38:11 +02:00
9d7125a311 Now with custom opengraph (tm)
All checks were successful
continuous-integration/drone/push Build is passing
ref #2
2021-09-25 17:35:52 +02:00
6d71a3ebf4 Removed unused log
All checks were successful
continuous-integration/drone/push Build is passing
2021-09-25 17:27:59 +02:00
75adbf73cf Now with working bad bot detection
ref #2
2021-09-25 17:27:41 +02:00
e5b8557e4c Getters now return the no_preview status 2021-09-25 17:21:30 +02:00
33d7c94648 New urls can now be created with disabled preview
ref #2
2021-09-25 17:19:36 +02:00
f6b2ae523d Added bot check for native short urls
ref #2
2021-09-25 17:17:42 +02:00
0a500f16cd Added migration for disallowing bot previews 2021-09-25 17:16:38 +02:00
61da5d8110 Added basic bot checking 2021-09-25 17:14:34 +02:00
773b286216 Added package for bot recognition 2021-09-25 17:12:52 +02:00
d097eccbd9 removed logging 2021-09-25 17:12:22 +02:00
f15282a3f9 Log user agents
All checks were successful
continuous-integration/drone/push Build is passing
2021-09-25 17:09:19 +02:00
0a8945a294 Added noindex header to all shorturl routes
All checks were successful
continuous-integration/drone/push Build is passing
ref #2
2021-09-25 16:59:28 +02:00
b871e4295d 🚀RELEASE 0.5.1
All checks were successful
continuous-integration/drone/push Build is passing
2021-08-21 10:03:16 +02:00
0945060a49 Added optional filtering by provider 2021-08-21 10:02:52 +02:00
f7eae96b8c Shortened return to avoid variable memory asignment 2021-08-21 10:01:24 +02:00
c2a50a1480 Changed object property order (just for us pesky humans to improve readability) 2021-08-21 09:59:57 +02:00
bb99c2dcd1 Added endpoint to query over all visited urls 2021-08-21 09:59:14 +02:00
10 changed files with 1282 additions and 4293 deletions

View File

@@ -2,8 +2,71 @@
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.0](https://git.odit.services/kauft.es/linkylinky/compare/0.6.1...0.7.0)
- Push w/o lockfile [`231d8ca`](https://git.odit.services/kauft.es/linkylinky/commit/231d8ca7ee0da402322afd3ac8ddce58e14943c1)
- 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) #### [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) - 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) - 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) - Reverted last change since we can defer it from the provider [`b532137`](https://git.odit.services/kauft.es/linkylinky/commit/b5321377bde9bd7de13e2eee4dd38a072cf0d4a3)

View File

@@ -1,7 +1,7 @@
FROM node:16.6.2-alpine3.14 FROM registry.odit.services/hub/library/node:17.0.1-alpine3.14
WORKDIR /app WORKDIR /app
COPY package.json . COPY package.json .
COPY yarn.lock . # COPY yarn.lock .
RUN yarn --production --frozen-lockfile RUN yarn --production --frozen-lockfile
COPY migrations ./migrations COPY migrations ./migrations
COPY src ./src COPY src ./src
@@ -9,7 +9,7 @@ COPY knexfile.js ./
RUN mkdir db RUN mkdir db
# #
# FROM astefanutti/scratch-node:16.0.0 # FROM astefanutti/scratch-node:16.0.0
FROM node:16.6.2-alpine3.14 FROM registry.odit.services/hub/library/node:17.0.1-alpine3.14
WORKDIR /app WORKDIR /app
COPY --from=0 /app /app COPY --from=0 /app /app
ENV NODE_ENV production ENV NODE_ENV production

View File

@@ -4,6 +4,20 @@
<h3 align="center">A small url shortener, originaly developed for kauft.es</h3> <h3 align="center">A small url shortener, originaly developed for kauft.es</h3>
</p> </p>
## Features
* Url shortinging to 18-byte hex ids
* Special shortening "providers" for well-used services
### Provider patterns
| Provider | Pattern | Resolves to | Notes |
| - | - | - | - |
| YouTube (Video) | `/yt/id` | `https://youtu.be/id` | None |
| YouTube (Playlist) | `/ytpl/id` | `https://youtube.com/playlist?list=id` | Remember: Private playlists will result in user-side errors |
| Amazon | `/a/id` | `https://amazon.de/dp/id` | Recognizes all kind of cursed amazon urls (+smile and others) |
| eBay | `/e/id` | `https://ebay.de/itm/` | Only tested with German eBay |
| Reddit | `/r/id` | `https://redd.it/` | Powered by the awesome work of u/TheAppleFreak https://kauf.es/r/4vaqiw |
## Dev Setup 🛠 ## Dev Setup 🛠
> Runs on port 3000 > Runs on port 3000

View 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) {
};

View 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) {
};

View File

@@ -1,6 +1,6 @@
{ {
"name": "@odit/shortener-backend", "name": "@odit/shortener-backend",
"version": "0.5.0", "version": "0.7.0",
"main": "index.js", "main": "index.js",
"license": "MIT", "license": "MIT",
"private": false, "private": false,
@@ -25,13 +25,14 @@
}, },
"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-auth": "^1.1.0",
"fastify-basic-auth": "^2.1.0", "fastify-basic-auth": "^2.1.0",
"fastify-cors": "^6.0.2", "fastify-cors": "^6.0.2",
"fastify-jwt": "^3.0.1", "fastify-jwt": "^4.0.0",
"knex": "^0.21.21", "isbot": "^3.3.3",
"knex": "^0.95.0",
"sqlite3": "^5.0.2", "sqlite3": "^5.0.2",
"uniqid": "^5.3.0" "uniqid": "^5.3.0"
}, },

4
renovate.json Normal file
View File

@@ -0,0 +1,4 @@
{
"$schema": "https://docs.renovatebot.com/renovate-schema.json",
"dependencyDashboard": true
}

View File

@@ -2,6 +2,7 @@ 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",
@@ -95,6 +96,12 @@ fastify.get('/e/:id', async (req, res) => {
await knex('visits').insert({ shortcode: req.params.id, provider: 'e' }); 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
fastify.get('/:shortcode', async (req, res) => { fastify.get('/:shortcode', async (req, res) => {
const shortcode = req.params.shortcode; const shortcode = req.params.shortcode;
@@ -103,13 +110,24 @@ 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, provider: 'native' }); await knex('visits').insert({ shortcode, provider: 'native' });
}) })
@@ -121,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' }
} }
} }
}; };
@@ -129,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) {
@@ -148,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);
@@ -157,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();
@@ -179,12 +203,14 @@ 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
} }
}); });
@@ -211,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);
@@ -227,6 +253,8 @@ 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
} }
}); });
@@ -267,7 +295,7 @@ fastify.post('/api/auth/register', async (req, res) => {
//Anything in here has some kind of auth //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.auth([fastify.basicAuth, fastify.verifyJWT]) }, 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;
@@ -283,6 +311,17 @@ fastify.after(() => {
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', { onRequest: fastify.auth([fastify.basicAuth, fastify.verifyJWT]) }, 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;
@@ -302,7 +341,7 @@ fastify.after(() => {
//Get all urls api route //Get all urls api route
fastify.get('/api', { onRequest: fastify.auth([fastify.basicAuth, fastify.verifyJWT]) }, async (req, res) => { fastify.get('/api', { onRequest: fastify.auth([fastify.basicAuth, fastify.verifyJWT]) }, async (req, res) => {
urls = await knex.select('target', 'shortcode') urls = await knex.select('target', 'shortcode', 'no_preview', 'clientside')
.from('urls'); .from('urls');
for (let url of urls) { for (let url of urls) {
@@ -404,6 +443,16 @@ function checkKnownProviders(target) {
target 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[9]) {
const shortcode = `r${redditID[9]}`
return {
url: `${config.getBaseUrl()}/${shortcode}`,
shortcode,
target
}
}
return null; return null;
} }
@@ -426,6 +475,286 @@ async function validate(username, password, req, reply) {
req.user = username; 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 { try {

837
src/🥵🥵🥵.js Normal file
View 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;

4278
yarn.lock

File diff suppressed because it is too large Load Diff