diff --git a/src/realms/api/hit.ts b/src/realms/api/hit.ts
new file mode 100644
index 0000000..5519c20
--- /dev/null
+++ b/src/realms/api/hit.ts
@@ -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);
+ }
+}
\ No newline at end of file
diff --git a/src/realms/api/router.ts b/src/realms/api/router.ts
index 6816dc5..da08d56 100644
--- a/src/realms/api/router.ts
+++ b/src/realms/api/router.ts
@@ -3,6 +3,7 @@ import { statusRequest } from '../twitter/routes/status';
import { profileRequest } from '../twitter/routes/profile';
import { Strings } from '../../strings';
import { Constants } from '../../constants';
+import { linkHitRequest } from './hit';
export const api = new Hono();
@@ -16,6 +17,9 @@ api.use('*', async (c, 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. */
api.get('/status/:id', statusRequest);
api.get('/status/:id/', statusRequest);
diff --git a/src/render/instantview.ts b/src/render/instantview.ts
index 256a814..09e6969 100644
--- a/src/render/instantview.ts
+++ b/src/render/instantview.ts
@@ -60,7 +60,7 @@ const formatDate = (date: Date): string => {
const htmlifyLinks = (input: string): string => {
const urlPattern = /\bhttps?:\/\/[\w.-]+\.\w+[/\w.-]*\w/g;
return input.replace(urlPattern, url => {
- return `${url}`;
+ return `${url}`;
});
};
@@ -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 { author } = status;
@@ -153,7 +174,7 @@ const generateStatusFooter = (status: APIStatus, isQuote = false): string => {
}'s profile picture" />`,
location: author.location ? `📌 ${author.location}` : '',
website: author.website
- ? `🔗 ${author.website.display_url}`
+ ? `🔗 ${author.website.display_url}`
: '',
joined: author.joined ? `📆 ${formatDate(new Date(author.joined))}` : '',
following: truncateSocialCount(author.following),