mirror of
https://github.com/CompeyDev/fxtwitter-docker.git
synced 2025-04-06 11:00:54 +01:00
Wrap foreign links (Fix #574?)
This commit is contained in:
parent
349fa39b2c
commit
be165eae17
3 changed files with 42 additions and 2 deletions
15
src/realms/api/hit.ts
Normal file
15
src/realms/api/hit.ts
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
import { Context } from "hono";
|
||||||
|
|
||||||
|
export const linkHitRequest = async (c: Context) => {
|
||||||
|
// eslint-disable-next-line sonarjs/no-duplicate-string
|
||||||
|
const userAgent = c.req.header('User-Agent') || '';
|
||||||
|
|
||||||
|
if (userAgent.includes('Telegram')) {
|
||||||
|
c.status(403);
|
||||||
|
}
|
||||||
|
// If param `url` exists, 302 redirect to it
|
||||||
|
if (typeof c.req.query('url') === 'string') {
|
||||||
|
const url = new URL(c.req.query('url') as string);
|
||||||
|
return c.redirect(url.href, 302);
|
||||||
|
}
|
||||||
|
}
|
|
@ -3,6 +3,7 @@ import { statusRequest } from '../twitter/routes/status';
|
||||||
import { profileRequest } from '../twitter/routes/profile';
|
import { profileRequest } from '../twitter/routes/profile';
|
||||||
import { Strings } from '../../strings';
|
import { Strings } from '../../strings';
|
||||||
import { Constants } from '../../constants';
|
import { Constants } from '../../constants';
|
||||||
|
import { linkHitRequest } from './hit';
|
||||||
|
|
||||||
export const api = new Hono();
|
export const api = new Hono();
|
||||||
|
|
||||||
|
@ -16,6 +17,9 @@ api.use('*', async (c, next) => {
|
||||||
}
|
}
|
||||||
await next();
|
await next();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
api.get('/2/hit', linkHitRequest);
|
||||||
|
|
||||||
/* Current v1 API endpoints. Currently, these still go through the Twitter embed requests. API v2+ won't do this. */
|
/* Current v1 API endpoints. Currently, these still go through the Twitter embed requests. API v2+ won't do this. */
|
||||||
api.get('/status/:id', statusRequest);
|
api.get('/status/:id', statusRequest);
|
||||||
api.get('/status/:id/', statusRequest);
|
api.get('/status/:id/', statusRequest);
|
||||||
|
|
|
@ -60,7 +60,7 @@ const formatDate = (date: Date): string => {
|
||||||
const htmlifyLinks = (input: string): string => {
|
const htmlifyLinks = (input: string): string => {
|
||||||
const urlPattern = /\bhttps?:\/\/[\w.-]+\.\w+[/\w.-]*\w/g;
|
const urlPattern = /\bhttps?:\/\/[\w.-]+\.\w+[/\w.-]*\w/g;
|
||||||
return input.replace(urlPattern, url => {
|
return input.replace(urlPattern, url => {
|
||||||
return `<a href="${url}">${url}</a>`;
|
return `<a href="${wrapForeignLinks(url)}">${url}</a>`;
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -117,6 +117,27 @@ const truncateSocialCount = (count: number): string => {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const wrapForeignLinks = (url: string) => {
|
||||||
|
let unwrap = false;
|
||||||
|
const whitelistedDomains = [
|
||||||
|
'twitter.com',
|
||||||
|
'x.com',
|
||||||
|
't.me',
|
||||||
|
'telegram.me'
|
||||||
|
]
|
||||||
|
try {
|
||||||
|
const urlObj = new URL(url);
|
||||||
|
|
||||||
|
if (!whitelistedDomains.includes(urlObj.hostname)) {
|
||||||
|
unwrap = true;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
unwrap = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return unwrap ? `https://${Constants.API_HOST_LIST[0]}/2/hit?url=${encodeURIComponent(url)}` : url;
|
||||||
|
}
|
||||||
|
|
||||||
const generateStatusFooter = (status: APIStatus, isQuote = false): string => {
|
const generateStatusFooter = (status: APIStatus, isQuote = false): string => {
|
||||||
const { author } = status;
|
const { author } = status;
|
||||||
|
|
||||||
|
@ -153,7 +174,7 @@ const generateStatusFooter = (status: APIStatus, isQuote = false): string => {
|
||||||
}'s profile picture" />`,
|
}'s profile picture" />`,
|
||||||
location: author.location ? `📌 ${author.location}` : '',
|
location: author.location ? `📌 ${author.location}` : '',
|
||||||
website: author.website
|
website: author.website
|
||||||
? `🔗 <a href=${author.website.url}>${author.website.display_url}</a>`
|
? `🔗 <a rel="nofollow" href="${wrapForeignLinks(author.website.url)}">${author.website.display_url}</a>`
|
||||||
: '',
|
: '',
|
||||||
joined: author.joined ? `📆 ${formatDate(new Date(author.joined))}` : '',
|
joined: author.joined ? `📆 ${formatDate(new Date(author.joined))}` : '',
|
||||||
following: truncateSocialCount(author.following),
|
following: truncateSocialCount(author.following),
|
||||||
|
|
Loading…
Add table
Reference in a new issue