mirror of
https://github.com/CompeyDev/fxtwitter-docker.git
synced 2025-04-05 18:40:56 +01:00
Bugfixes
This commit is contained in:
parent
01da30b9e8
commit
d06762ffe9
10 changed files with 58 additions and 91 deletions
|
@ -61,7 +61,6 @@ export const Constants = {
|
||||||
'content-type': 'text/html;charset=UTF-8',
|
'content-type': 'text/html;charset=UTF-8',
|
||||||
'x-powered-by': `${RELEASE_NAME}`,
|
'x-powered-by': `${RELEASE_NAME}`,
|
||||||
'x-trans-rights': 'true',
|
'x-trans-rights': 'true',
|
||||||
'cache-control': 'max-age=3600', // Can be overriden in some cases, like unfinished poll tweets
|
|
||||||
'Vary': 'Accept-Encoding, User-Agent'
|
'Vary': 'Accept-Encoding, User-Agent'
|
||||||
},
|
},
|
||||||
API_RESPONSE_HEADERS: {
|
API_RESPONSE_HEADERS: {
|
||||||
|
|
|
@ -77,7 +77,7 @@ export const handleStatus = async (
|
||||||
for (const [header, value] of Object.entries(Constants.API_RESPONSE_HEADERS)) {
|
for (const [header, value] of Object.entries(Constants.API_RESPONSE_HEADERS)) {
|
||||||
c.header(header, value);
|
c.header(header, value);
|
||||||
}
|
}
|
||||||
return c.text(JSON.stringify(api));
|
return c.json(api);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (tweet === null) {
|
if (tweet === null) {
|
||||||
|
@ -423,7 +423,7 @@ export const handleStatus = async (
|
||||||
const lang = tweet.lang === null ? 'en' : tweet.lang || 'en';
|
const lang = tweet.lang === null ? 'en' : tweet.lang || 'en';
|
||||||
|
|
||||||
/* Finally, after all that work we return the response HTML! */
|
/* Finally, after all that work we return the response HTML! */
|
||||||
return c.text(Strings.BASE_HTML.format({
|
return c.html(Strings.BASE_HTML.format({
|
||||||
lang: `lang="${lang}"`,
|
lang: `lang="${lang}"`,
|
||||||
headers: headers.join(''),
|
headers: headers.join(''),
|
||||||
body: ivbody
|
body: ivbody
|
||||||
|
|
|
@ -519,5 +519,5 @@ export const threadAPIProvider = async (c: Context) => {
|
||||||
for (const [header, value] of Object.entries(Constants.API_RESPONSE_HEADERS)) {
|
for (const [header, value] of Object.entries(Constants.API_RESPONSE_HEADERS)) {
|
||||||
c.header(header, value);
|
c.header(header, value);
|
||||||
}
|
}
|
||||||
return c.text(JSON.stringify(processedResponse));
|
return c.json(processedResponse);
|
||||||
};
|
};
|
||||||
|
|
|
@ -6,19 +6,8 @@ import { Strings } from '../../strings';
|
||||||
export const api = new Hono();
|
export const api = new Hono();
|
||||||
|
|
||||||
/* 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('/:handle?/status/:id/:language?', statusRequest);
|
api.get('/status/:id/:language?', statusRequest);
|
||||||
api.get(
|
api.get('/:handle/status/:id/:language?', statusRequest);
|
||||||
'/:handle?/status/:id/:mediaType{(photos?|videos?)}/:mediaNumber{[1-4]}/:language?',
|
|
||||||
statusRequest
|
|
||||||
);
|
|
||||||
|
|
||||||
api.get('/:handle', profileRequest);
|
api.get('/:handle', profileRequest);
|
||||||
|
api.get('/robots.txt', async (c) => c.text(Strings.ROBOTS_TXT));
|
||||||
api.get(
|
|
||||||
'/robots.txt',
|
|
||||||
async (c) => {
|
|
||||||
c.header('cache-control', 'max-age=0, no-cache, no-store, must-revalidate');
|
|
||||||
c.status(200);
|
|
||||||
return c.text(Strings.ROBOTS_TXT);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
|
@ -1,29 +1,20 @@
|
||||||
import { Context } from 'hono';
|
import { Context } from 'hono';
|
||||||
import { Constants } from '../../constants';
|
|
||||||
import { sanitizeText } from '../../helpers/utils';
|
import { sanitizeText } from '../../helpers/utils';
|
||||||
import { Strings } from '../../strings';
|
import { Strings } from '../../strings';
|
||||||
|
|
||||||
export const versionRoute = async (context: Context) => {
|
export const versionRoute = async (c: Context) => {
|
||||||
const request = context.req;
|
c.header('cache-control', 'max-age=0, no-cache, no-store, must-revalidate');
|
||||||
return new Response(
|
const req = c.req;
|
||||||
Strings.VERSION_HTML.format({
|
return c.html(Strings.VERSION_HTML.format({
|
||||||
rtt: request.raw.cf?.clientTcpRtt ? `🏓 ${request.raw.cf.clientTcpRtt} ms RTT` : '',
|
rtt: req.raw.cf?.clientTcpRtt ? `🏓 ${req.raw.cf.clientTcpRtt} ms RTT` : '',
|
||||||
colo: (request.raw.cf?.colo as string) ?? '??',
|
colo: (req.raw.cf?.colo as string) ?? '??',
|
||||||
httpversion: (request.raw.cf?.httpProtocol as string) ?? 'Unknown HTTP Version',
|
httpversion: (req.raw.cf?.httpProtocol as string) ?? 'Unknown HTTP Version',
|
||||||
tlsversion: (request.raw.cf?.tlsVersion as string) ?? 'Unknown TLS Version',
|
tlsversion: (req.raw.cf?.tlsVersion as string) ?? 'Unknown TLS Version',
|
||||||
ip: request.header('x-real-ip') ?? request.header('cf-connecting-ip') ?? 'Unknown IP',
|
ip: req.header('x-real-ip') ?? req.header('cf-connecting-ip') ?? 'Unknown IP',
|
||||||
city: (request.raw.cf?.city as string) ?? 'Unknown City',
|
city: (req.raw.cf?.city as string) ?? 'Unknown City',
|
||||||
region: (request.raw.cf?.region as string) ?? request.raw.cf?.country ?? 'Unknown Region',
|
region: (req.raw.cf?.region as string) ?? req.raw.cf?.country ?? 'Unknown Region',
|
||||||
country: (request.raw.cf?.country as string) ?? 'Unknown Country',
|
country: (req.raw.cf?.country as string) ?? 'Unknown Country',
|
||||||
asn: `AS${request.raw.cf?.asn ?? '??'} (${request.raw.cf?.asOrganization ?? 'Unknown ASN'})`,
|
asn: `AS${req.raw.cf?.asn ?? '??'} (${req.raw.cf?.asOrganization ?? 'Unknown ASN'})`,
|
||||||
ua: sanitizeText(request.header('user-agent') ?? 'Unknown User Agent')
|
ua: sanitizeText(req.header('user-agent') ?? 'Unknown User Agent')
|
||||||
}),
|
}))
|
||||||
{
|
|
||||||
headers: {
|
|
||||||
...Constants.RESPONSE_HEADERS,
|
|
||||||
'cache-control': 'max-age=0, no-cache, no-store, must-revalidate'
|
|
||||||
},
|
|
||||||
status: 200
|
|
||||||
}
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
|
@ -10,37 +10,6 @@ import { oembed } from './routes/oembed';
|
||||||
|
|
||||||
export const twitter = new Hono();
|
export const twitter = new Hono();
|
||||||
|
|
||||||
twitter.get('/status/:id', statusRequest);
|
|
||||||
// twitter.get('/:handle/status/:id', statusRequest);
|
|
||||||
// twitter.get('/:prefix/:handle/status/:id/:language?', statusRequest);
|
|
||||||
// twitter.get(
|
|
||||||
// '/:prefix/:handle/status/:id/:mediaType{(photos?|videos?)}/:mediaNumber{[1-4]}/:language?',
|
|
||||||
// statusRequest
|
|
||||||
// );
|
|
||||||
// twitter.get('/:handle?/:endpoint{status(es)?}/:id/:language?', statusRequest);
|
|
||||||
// twitter.get(
|
|
||||||
// '/:handle?/:endpoint{status(es)?}/:id/:mediaType{(photos?|videos?)}/:mediaNumber{[1-4]}/:language?',
|
|
||||||
// statusRequest
|
|
||||||
// );
|
|
||||||
|
|
||||||
twitter.get('/version', versionRoute);
|
|
||||||
twitter.get('/set_base_redirect', setRedirectRequest);
|
|
||||||
twitter.get('/oembed', oembed);
|
|
||||||
|
|
||||||
twitter.get(
|
|
||||||
'/robots.txt',
|
|
||||||
async (c) => {
|
|
||||||
c.header('cache-control', 'max-age=0, no-cache, no-store, must-revalidate');
|
|
||||||
c.status(200);
|
|
||||||
return c.text(Strings.ROBOTS_TXT);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
twitter.get('/i/events/:id', genericTwitterRedirect);
|
|
||||||
twitter.get('/hashtag/:hashtag', genericTwitterRedirect);
|
|
||||||
|
|
||||||
twitter.get('/:handle', profileRequest);
|
|
||||||
|
|
||||||
export const getBaseRedirectUrl = (c: Context) => {
|
export const getBaseRedirectUrl = (c: Context) => {
|
||||||
const baseRedirect = c.req.header('cookie')?.match(/(?<=base_redirect=)(.*?)(?=;|$)/)?.[0];
|
const baseRedirect = c.req.header('cookie')?.match(/(?<=base_redirect=)(.*?)(?=;|$)/)?.[0];
|
||||||
|
|
||||||
|
@ -56,3 +25,28 @@ export const getBaseRedirectUrl = (c: Context) => {
|
||||||
|
|
||||||
return Constants.TWITTER_ROOT;
|
return Constants.TWITTER_ROOT;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
twitter.get('/status/:id', statusRequest);
|
||||||
|
twitter.get('/:handle/status/:id', statusRequest);
|
||||||
|
twitter.get('/:prefix/:handle/status/:id/:language?', statusRequest);
|
||||||
|
twitter.get(
|
||||||
|
'/:prefix/:handle/status/:id/:mediaType{(photos?|videos?)}/:mediaNumber{[1-4]}/:language?',
|
||||||
|
statusRequest
|
||||||
|
);
|
||||||
|
twitter.get('/:handle?/:endpoint{status(es)?}/:id/:language?', statusRequest);
|
||||||
|
twitter.get(
|
||||||
|
'/:handle?/:endpoint{status(es)?}/:id/:mediaType{(photos?|videos?)}/:mediaNumber{[1-4]}/:language?',
|
||||||
|
statusRequest
|
||||||
|
);
|
||||||
|
|
||||||
|
twitter.get('/version', versionRoute);
|
||||||
|
twitter.get('/set_base_redirect', setRedirectRequest);
|
||||||
|
twitter.get('/oembed', oembed);
|
||||||
|
|
||||||
|
twitter.get('/robots.txt', async (c) => c.text(Strings.ROBOTS_TXT));
|
||||||
|
|
||||||
|
twitter.get('/i/events/:id', genericTwitterRedirect);
|
||||||
|
twitter.get('/hashtag/:hashtag', genericTwitterRedirect);
|
||||||
|
|
||||||
|
twitter.get('/:handle', profileRequest);
|
|
@ -27,7 +27,7 @@ export const setRedirectRequest = async (c: Context) => {
|
||||||
if (origin && !Constants.STANDARD_DOMAIN_LIST.includes(new URL(origin).hostname)) {
|
if (origin && !Constants.STANDARD_DOMAIN_LIST.includes(new URL(origin).hostname)) {
|
||||||
c.status(403);
|
c.status(403);
|
||||||
|
|
||||||
return c.text(Strings.MESSAGE_HTML.format({
|
return c.html(Strings.MESSAGE_HTML.format({
|
||||||
message: `Failed to set base redirect: Your request seems to be originating from another domain, please open this up in a new tab if you are trying to set your base redirect.`
|
message: `Failed to set base redirect: Your request seems to be originating from another domain, please open this up in a new tab if you are trying to set your base redirect.`
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
@ -40,7 +40,7 @@ export const setRedirectRequest = async (c: Context) => {
|
||||||
// eslint-disable-next-line sonarjs/no-duplicate-string
|
// eslint-disable-next-line sonarjs/no-duplicate-string
|
||||||
c.header('content-security-policy', `frame-ancestors ${Constants.STANDARD_DOMAIN_LIST.join(' ')};`);
|
c.header('content-security-policy', `frame-ancestors ${Constants.STANDARD_DOMAIN_LIST.join(' ')};`);
|
||||||
c.status(200);
|
c.status(200);
|
||||||
return c.text(Strings.MESSAGE_HTML.format({
|
return c.html(Strings.MESSAGE_HTML.format({
|
||||||
message: `Your base redirect has been cleared. To set one, please pass along the <code>url</code> parameter.`
|
message: `Your base redirect has been cleared. To set one, please pass along the <code>url</code> parameter.`
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
@ -57,7 +57,7 @@ export const setRedirectRequest = async (c: Context) => {
|
||||||
c.header('set-cookie', `base_redirect=; path=/; max-age=0; expires=Thu, 01 Jan 1970 00:00:00 GMT; secure; HttpOnly`);
|
c.header('set-cookie', `base_redirect=; path=/; max-age=0; expires=Thu, 01 Jan 1970 00:00:00 GMT; secure; HttpOnly`);
|
||||||
c.header('content-security-policy', `frame-ancestors ${Constants.STANDARD_DOMAIN_LIST.join(' ')};`);
|
c.header('content-security-policy', `frame-ancestors ${Constants.STANDARD_DOMAIN_LIST.join(' ')};`);
|
||||||
c.status(200);
|
c.status(200);
|
||||||
return c.text(Strings.MESSAGE_HTML.format({
|
return c.html(Strings.MESSAGE_HTML.format({
|
||||||
message: `Your URL does not appear to be well-formed. Example: ?url=https://nitter.net`
|
message: `Your URL does not appear to be well-formed. Example: ?url=https://nitter.net`
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
@ -68,7 +68,7 @@ export const setRedirectRequest = async (c: Context) => {
|
||||||
c.header('set-cookie', `base_redirect=${url}; path=/; max-age=63072000; secure; HttpOnly`);
|
c.header('set-cookie', `base_redirect=${url}; path=/; max-age=63072000; secure; HttpOnly`);
|
||||||
c.header('content-security-policy', `frame-ancestors ${Constants.STANDARD_DOMAIN_LIST.join(' ')};`);
|
c.header('content-security-policy', `frame-ancestors ${Constants.STANDARD_DOMAIN_LIST.join(' ')};`);
|
||||||
|
|
||||||
return c.text(Strings.MESSAGE_HTML.format({
|
return c.html(Strings.MESSAGE_HTML.format({
|
||||||
message: `Successfully set base redirect, you will now be redirected to ${sanitizeText(url)} rather than ${Constants.TWITTER_ROOT}`
|
message: `Successfully set base redirect, you will now be redirected to ${sanitizeText(url)} rather than ${Constants.TWITTER_ROOT}`
|
||||||
}))
|
}))
|
||||||
};
|
};
|
||||||
|
|
|
@ -6,11 +6,7 @@ import { Strings } from '../../../strings';
|
||||||
|
|
||||||
/* Handler for status (Tweet) request */
|
/* Handler for status (Tweet) request */
|
||||||
export const statusRequest = async (c: Context) => {
|
export const statusRequest = async (c: Context) => {
|
||||||
const handle = c.req.param('handle');
|
const { handle, id, mediaNumber, language, prefix } = c.req.param();
|
||||||
const id = c.req.param('id');
|
|
||||||
const mediaNumber = c.req.param('mediaNumber');
|
|
||||||
const language = c.req.param('language');
|
|
||||||
const prefix = c.req.param('prefix');
|
|
||||||
const url = new URL(c.req.url);
|
const url = new URL(c.req.url);
|
||||||
const flags: InputFlags = {};
|
const flags: InputFlags = {};
|
||||||
|
|
||||||
|
|
|
@ -4,7 +4,7 @@ import { Strings } from './strings';
|
||||||
import { userAPI } from './providers/twitter/profile';
|
import { userAPI } from './providers/twitter/profile';
|
||||||
|
|
||||||
export const returnError = (c: Context, error: string): Response => {
|
export const returnError = (c: Context, error: string): Response => {
|
||||||
return c.text(Strings.BASE_HTML.format({
|
return c.html(Strings.BASE_HTML.format({
|
||||||
lang: '',
|
lang: '',
|
||||||
headers: [
|
headers: [
|
||||||
`<meta property="og:title" content="${Constants.BRANDING_NAME}"/>`,
|
`<meta property="og:title" content="${Constants.BRANDING_NAME}"/>`,
|
||||||
|
@ -32,7 +32,7 @@ export const handleProfile = async (
|
||||||
for (const [header, value] of Object.entries(Constants.API_RESPONSE_HEADERS)) {
|
for (const [header, value] of Object.entries(Constants.API_RESPONSE_HEADERS)) {
|
||||||
c.header(header, value);
|
c.header(header, value);
|
||||||
}
|
}
|
||||||
return c.text(JSON.stringify(api));
|
return c.json(api);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If there was any errors fetching the User, we'll return it */
|
/* If there was any errors fetching the User, we'll return it */
|
||||||
|
@ -51,7 +51,7 @@ export const handleProfile = async (
|
||||||
// TODO Add card creation logic here
|
// TODO Add card creation logic here
|
||||||
|
|
||||||
/* Finally, after all that work we return the response HTML! */
|
/* Finally, after all that work we return the response HTML! */
|
||||||
return c.text(Strings.BASE_HTML.format({
|
return c.html(Strings.BASE_HTML.format({
|
||||||
lang: `lang="en"`,
|
lang: `lang="en"`,
|
||||||
headers: headers.join('')
|
headers: headers.join('')
|
||||||
}));
|
}));
|
||||||
|
|
|
@ -64,13 +64,11 @@ app.onError((err, c) => {
|
||||||
console.error(error.stack);
|
console.error(error.stack);
|
||||||
c.status(200);
|
c.status(200);
|
||||||
c.header('cache-control', noCache);
|
c.header('cache-control', noCache);
|
||||||
c.header('content-type', 'text/html');
|
|
||||||
|
|
||||||
|
return c.html(Strings.ERROR_HTML);
|
||||||
return c.text(Strings.ERROR_HTML);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
app.use('*', logger());
|
// app.use('*', logger());
|
||||||
|
|
||||||
app.use('*', async (c, next) => {
|
app.use('*', async (c, next) => {
|
||||||
console.log(`Hello from ⛅ ${c.req.raw.cf?.colo || 'UNK'}`);
|
console.log(`Hello from ⛅ ${c.req.raw.cf?.colo || 'UNK'}`);
|
||||||
|
@ -81,8 +79,8 @@ app.use('*', async (c, next) => {
|
||||||
app.use('*', cacheMiddleware());
|
app.use('*', cacheMiddleware());
|
||||||
app.use('*', timing({ enabled: false }));
|
app.use('*', timing({ enabled: false }));
|
||||||
|
|
||||||
app.route(`/api`, api);
|
|
||||||
app.route(`/twitter`, twitter);
|
app.route(`/twitter`, twitter);
|
||||||
|
app.route(`/api`, api);
|
||||||
|
|
||||||
app.all(
|
app.all(
|
||||||
'/error',
|
'/error',
|
||||||
|
@ -104,7 +102,7 @@ export default {
|
||||||
return new Response(Strings.ERROR_HTML, {
|
return new Response(Strings.ERROR_HTML, {
|
||||||
headers: {
|
headers: {
|
||||||
...Constants.RESPONSE_HEADERS,
|
...Constants.RESPONSE_HEADERS,
|
||||||
'content-type': 'text/html',
|
'content-type': 'text/html;charset=utf-8',
|
||||||
'cache-control': noCache
|
'cache-control': noCache
|
||||||
},
|
},
|
||||||
status: 200
|
status: 200
|
||||||
|
|
Loading…
Add table
Reference in a new issue