mirror of
https://github.com/CompeyDev/fxtwitter-docker.git
synced 2025-04-06 02:50:54 +01:00
Merge pull request #361 from FixTweet/3rd-party-redirect
Add base redirect (For nitter, etc)
This commit is contained in:
commit
1ba6a707a5
3 changed files with 264 additions and 71 deletions
172
src/server.ts
172
src/server.ts
|
@ -16,6 +16,24 @@ declare const globalThis: {
|
||||||
|
|
||||||
const router = Router();
|
const router = Router();
|
||||||
|
|
||||||
|
const getBaseRedirectUrl = (request: IRequest) => {
|
||||||
|
const baseRedirect = request.headers
|
||||||
|
?.get('cookie')
|
||||||
|
?.match(/(?<=base_redirect=)(.*?)(?=;|$)/)?.[0];
|
||||||
|
|
||||||
|
if (baseRedirect) {
|
||||||
|
console.log('Found base redirect', baseRedirect);
|
||||||
|
try {
|
||||||
|
new URL(baseRedirect);
|
||||||
|
} catch (e) {
|
||||||
|
return Constants.TWITTER_ROOT;
|
||||||
|
}
|
||||||
|
return baseRedirect.endsWith('/') ? baseRedirect.slice(0, -1) : baseRedirect;
|
||||||
|
}
|
||||||
|
|
||||||
|
return Constants.TWITTER_ROOT;
|
||||||
|
};
|
||||||
|
|
||||||
/* Handler for status (Tweet) request */
|
/* Handler for status (Tweet) request */
|
||||||
const statusRequest = async (
|
const statusRequest = async (
|
||||||
request: IRequest,
|
request: IRequest,
|
||||||
|
@ -94,6 +112,8 @@ const statusRequest = async (
|
||||||
console.log(`We're using twitter domain`);
|
console.log(`We're using twitter domain`);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const baseUrl = getBaseRedirectUrl(request);
|
||||||
|
|
||||||
/* Check if request is to api.fxtwitter.com, or the tweet is appended with .json
|
/* Check if request is to api.fxtwitter.com, or the tweet is appended with .json
|
||||||
Note that unlike TwitFix, FixTweet will never generate embeds for .json, and
|
Note that unlike TwitFix, FixTweet will never generate embeds for .json, and
|
||||||
in fact we only support .json because it's what people using TwitFix API would
|
in fact we only support .json because it's what people using TwitFix API would
|
||||||
|
@ -136,13 +156,27 @@ const statusRequest = async (
|
||||||
Since we obviously have no media to give the user, we'll just redirect to the Tweet.
|
Since we obviously have no media to give the user, we'll just redirect to the Tweet.
|
||||||
Embeds will return as usual to bots as if direct media was never specified. */
|
Embeds will return as usual to bots as if direct media was never specified. */
|
||||||
if (!isBotUA) {
|
if (!isBotUA) {
|
||||||
return Response.redirect(`${Constants.TWITTER_ROOT}/${handle}/status/${id}`, 302);
|
const baseUrl = getBaseRedirectUrl(request);
|
||||||
|
/* Do not cache if using a custom redirect */
|
||||||
|
const cacheControl = baseUrl !== Constants.TWITTER_ROOT ? 'max-age=0' : undefined;
|
||||||
|
|
||||||
|
return new Response(null, {
|
||||||
|
status: 302,
|
||||||
|
headers: {
|
||||||
|
'Location': `${baseUrl}/${handle}/status/${id}`,
|
||||||
|
...(cacheControl ? { 'cache-control': cacheControl } : {})
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
let headers = Constants.RESPONSE_HEADERS;
|
let headers = Constants.RESPONSE_HEADERS;
|
||||||
|
|
||||||
if (statusResponse.cacheControl) {
|
if (statusResponse.cacheControl) {
|
||||||
headers = { ...headers, 'cache-control': statusResponse.cacheControl };
|
headers = {
|
||||||
|
...headers,
|
||||||
|
'cache-control':
|
||||||
|
baseUrl !== Constants.TWITTER_ROOT ? 'max-age=0' : statusResponse.cacheControl
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the response containing embed information */
|
/* Return the response containing embed information */
|
||||||
|
@ -162,10 +196,17 @@ const statusRequest = async (
|
||||||
/* A human has clicked a fxtwitter.com/:screen_name/status/:id link!
|
/* A human has clicked a fxtwitter.com/:screen_name/status/:id link!
|
||||||
Obviously we just need to redirect to the Tweet directly.*/
|
Obviously we just need to redirect to the Tweet directly.*/
|
||||||
console.log('Matched human UA', userAgent);
|
console.log('Matched human UA', userAgent);
|
||||||
return Response.redirect(
|
|
||||||
`${Constants.TWITTER_ROOT}/${handle}/status/${id?.match(/\d{2,20}/)?.[0]}`,
|
const cacheControl = baseUrl !== Constants.TWITTER_ROOT ? 'max-age=0' : undefined;
|
||||||
302
|
|
||||||
);
|
return new Response(null, {
|
||||||
|
status: 302,
|
||||||
|
headers: {
|
||||||
|
'Location': `${baseUrl}/${handle}/status/${id?.match(/\d{2,20}/)?.[0]}`,
|
||||||
|
...(cacheControl ? { 'cache-control': cacheControl } : {})
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -221,14 +262,32 @@ const profileRequest = async (
|
||||||
} else if (profileResponse.text) {
|
} else if (profileResponse.text) {
|
||||||
console.log('handleProfile sent embed');
|
console.log('handleProfile sent embed');
|
||||||
/* TODO This check has purpose in the original handleStatus handler, but I'm not sure if this edge case can happen here */
|
/* TODO This check has purpose in the original handleStatus handler, but I'm not sure if this edge case can happen here */
|
||||||
|
const baseUrl = getBaseRedirectUrl(request);
|
||||||
|
/* Check for custom redirect */
|
||||||
|
|
||||||
if (!isBotUA) {
|
if (!isBotUA) {
|
||||||
return Response.redirect(`${Constants.TWITTER_ROOT}/${handle}`, 302);
|
/* Do not cache if using a custom redirect */
|
||||||
|
const cacheControl = baseUrl !== Constants.TWITTER_ROOT ? 'max-age=0' : undefined;
|
||||||
|
|
||||||
|
return new Response(null, {
|
||||||
|
status: 302,
|
||||||
|
headers: {
|
||||||
|
'Location': `${baseUrl}/${handle}`,
|
||||||
|
...(cacheControl ? { 'cache-control': cacheControl } : {})
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
let headers = Constants.RESPONSE_HEADERS;
|
let headers = Constants.RESPONSE_HEADERS;
|
||||||
|
|
||||||
if (profileResponse.cacheControl) {
|
if (profileResponse.cacheControl) {
|
||||||
headers = { ...headers, 'cache-control': profileResponse.cacheControl };
|
headers = {
|
||||||
|
...headers,
|
||||||
|
'cache-control':
|
||||||
|
baseUrl !== Constants.TWITTER_ROOT
|
||||||
|
? 'max-age=0'
|
||||||
|
: profileResponse.cacheControl
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the response containing embed information */
|
/* Return the response containing embed information */
|
||||||
|
@ -239,7 +298,7 @@ const profileRequest = async (
|
||||||
} else {
|
} else {
|
||||||
/* Somehow handleStatus sent us nothing. This should *never* happen, but we have a case for it. */
|
/* Somehow handleStatus sent us nothing. This should *never* happen, but we have a case for it. */
|
||||||
return new Response(Strings.ERROR_UNKNOWN, {
|
return new Response(Strings.ERROR_UNKNOWN, {
|
||||||
headers: Constants.RESPONSE_HEADERS,
|
headers: { ...Constants.RESPONSE_HEADERS, 'cache-control': 'max-age=0' },
|
||||||
status: 500
|
status: 500
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -247,13 +306,34 @@ const profileRequest = async (
|
||||||
/* A human has clicked a fxtwitter.com/:screen_name link!
|
/* A human has clicked a fxtwitter.com/:screen_name link!
|
||||||
Obviously we just need to redirect to the user directly.*/
|
Obviously we just need to redirect to the user directly.*/
|
||||||
console.log('Matched human UA', userAgent);
|
console.log('Matched human UA', userAgent);
|
||||||
return Response.redirect(`${Constants.TWITTER_ROOT}/${handle}`, 302);
|
|
||||||
|
const baseUrl = getBaseRedirectUrl(request);
|
||||||
|
/* Do not cache if using a custom redirect */
|
||||||
|
const cacheControl = baseUrl !== Constants.TWITTER_ROOT ? 'max-age=0' : undefined;
|
||||||
|
|
||||||
|
return new Response(null, {
|
||||||
|
status: 302,
|
||||||
|
headers: {
|
||||||
|
'Location': `${baseUrl}/${handle}`,
|
||||||
|
...(cacheControl ? { 'cache-control': cacheControl } : {})
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const genericTwitterRedirect = async (request: IRequest) => {
|
const genericTwitterRedirect = async (request: IRequest) => {
|
||||||
const url = new URL(request.url);
|
const url = new URL(request.url);
|
||||||
return Response.redirect(`${Constants.TWITTER_ROOT}${url.pathname}`, 302);
|
const baseUrl = getBaseRedirectUrl(request);
|
||||||
|
/* Do not cache if using a custom redirect */
|
||||||
|
const cacheControl = baseUrl !== Constants.TWITTER_ROOT ? 'max-age=0' : undefined;
|
||||||
|
|
||||||
|
return new Response(null, {
|
||||||
|
status: 302,
|
||||||
|
headers: {
|
||||||
|
'Location': `${baseUrl}${url.pathname}`,
|
||||||
|
...(cacheControl ? { 'cache-control': cacheControl } : {})
|
||||||
|
}
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const versionRequest = async (request: IRequest) => {
|
const versionRequest = async (request: IRequest) => {
|
||||||
|
@ -292,6 +372,70 @@ const versionRequest = async (request: IRequest) => {
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const setRedirectRequest = async (request: IRequest) => {
|
||||||
|
/* Query params */
|
||||||
|
const { searchParams } = new URL(request.url);
|
||||||
|
let url = searchParams.get('url');
|
||||||
|
|
||||||
|
if (!url) {
|
||||||
|
/* Remove redirect URL */
|
||||||
|
return new Response(
|
||||||
|
Strings.MESSAGE_HTML.format({
|
||||||
|
message: `Your base redirect has been cleared. To set one, please pass along the <code>url</code> parameter.`
|
||||||
|
}),
|
||||||
|
{
|
||||||
|
headers: {
|
||||||
|
'set-cookie': `base_redirect=; path=/; max-age=0; expires=Thu, 01 Jan 1970 00:00:00 GMT; secure; HttpOnly`,
|
||||||
|
...Constants.RESPONSE_HEADERS
|
||||||
|
},
|
||||||
|
status: 200
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
new URL(url);
|
||||||
|
} catch (e) {
|
||||||
|
try {
|
||||||
|
new URL(`https://${url}`);
|
||||||
|
} catch (e) {
|
||||||
|
/* URL is not well-formed, remove */
|
||||||
|
console.log('Invalid base redirect URL, removing cookie before redirect');
|
||||||
|
|
||||||
|
return new Response(
|
||||||
|
Strings.MESSAGE_HTML.format({
|
||||||
|
message: `Your URL does not appear to be well-formed. Example: ?url=https://nitter.net`
|
||||||
|
}),
|
||||||
|
{
|
||||||
|
headers: {
|
||||||
|
'set-cookie': `base_redirect=; path=/; max-age=0; expires=Thu, 01 Jan 1970 00:00:00 GMT; secure; HttpOnly`,
|
||||||
|
...Constants.RESPONSE_HEADERS
|
||||||
|
},
|
||||||
|
status: 200
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
url = `https://${url}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Set cookie for url */
|
||||||
|
return new Response(
|
||||||
|
Strings.MESSAGE_HTML.format({
|
||||||
|
message: `Successfully set base redirect, you will now be redirected to ${sanitizeText(
|
||||||
|
url
|
||||||
|
)} rather than ${Constants.TWITTER_ROOT}`
|
||||||
|
}),
|
||||||
|
{
|
||||||
|
headers: {
|
||||||
|
'set-cookie': `base_redirect=${url}; path=/; max-age=63072000; secure; HttpOnly`,
|
||||||
|
...Constants.RESPONSE_HEADERS
|
||||||
|
},
|
||||||
|
status: 200
|
||||||
|
}
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
/* TODO: is there any way to consolidate these stupid routes for itty-router?
|
/* TODO: is there any way to consolidate these stupid routes for itty-router?
|
||||||
I couldn't find documentation allowing for regex matching */
|
I couldn't find documentation allowing for regex matching */
|
||||||
router.get('/:prefix?/:handle/status/:id', statusRequest);
|
router.get('/:prefix?/:handle/status/:id', statusRequest);
|
||||||
|
@ -307,6 +451,7 @@ router.get('/:prefix?/:handle/statuses/:id/:language', statusRequest);
|
||||||
router.get('/status/:id', statusRequest);
|
router.get('/status/:id', statusRequest);
|
||||||
router.get('/status/:id/:language', statusRequest);
|
router.get('/status/:id/:language', statusRequest);
|
||||||
router.get('/version', versionRequest);
|
router.get('/version', versionRequest);
|
||||||
|
router.get('/set_base_redirect', setRedirectRequest);
|
||||||
|
|
||||||
/* Oembeds (used by Discord to enhance responses)
|
/* Oembeds (used by Discord to enhance responses)
|
||||||
|
|
||||||
|
@ -424,7 +569,10 @@ export const cacheWrapper = async (
|
||||||
|
|
||||||
switch (request.method) {
|
switch (request.method) {
|
||||||
case 'GET':
|
case 'GET':
|
||||||
if (!Constants.API_HOST_LIST.includes(cacheUrl.hostname)) {
|
if (
|
||||||
|
!Constants.API_HOST_LIST.includes(cacheUrl.hostname) &&
|
||||||
|
!request.headers?.get('Cookie')?.includes('base_redirect')
|
||||||
|
) {
|
||||||
/* cache may be undefined in tests */
|
/* cache may be undefined in tests */
|
||||||
const cachedResponse = await cache.match(cacheKey);
|
const cachedResponse = await cache.match(cacheKey);
|
||||||
|
|
||||||
|
|
150
src/strings.ts
150
src/strings.ts
|
@ -66,64 +66,98 @@ This is caused by Twitter API downtime or a new bug. Try again in a little while
|
||||||
.replace(/( {2})/g, '')
|
.replace(/( {2})/g, '')
|
||||||
.replace(/>\s+</gm, '><'),
|
.replace(/>\s+</gm, '><'),
|
||||||
VERSION_HTML: `<!DOCTYPE html>
|
VERSION_HTML: `<!DOCTYPE html>
|
||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="utf-8">
|
<meta charset="utf-8">
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
<meta content="${BRANDING_NAME}" property="og:title"/>
|
<meta content="${BRANDING_NAME}" property="og:title"/>
|
||||||
<meta content="${BRANDING_NAME}" property="og:site_name"/>
|
<meta content="${BRANDING_NAME}" property="og:site_name"/>
|
||||||
<meta content="https://cdn.discordapp.com/icons/958942151817977906/7a220767640cbedbf780767585eaa10d.png?size=96" property="og:image"/>
|
<meta content="https://cdn.discordapp.com/icons/958942151817977906/7a220767640cbedbf780767585eaa10d.png?size=96" property="og:image"/>
|
||||||
<meta content="https://cdn.discordapp.com/icons/958942151817977906/7a220767640cbedbf780767585eaa10d.png?size=96" property="twitter:image"/>
|
<meta content="https://cdn.discordapp.com/icons/958942151817977906/7a220767640cbedbf780767585eaa10d.png?size=96" property="twitter:image"/>
|
||||||
<meta content="#1E98F0" name="theme-color"/>
|
<meta content="#1E98F0" name="theme-color"/>
|
||||||
<meta content="Worker release: ${RELEASE_NAME}
|
<meta content="Worker release: ${RELEASE_NAME}
|
||||||
|
|
||||||
Stats for nerds:
|
Stats for nerds:
|
||||||
🕵️♂️ {ua}
|
🕵️♂️ {ua}
|
||||||
🌐 {ip}
|
🌐 {ip}
|
||||||
🌎 {city}, {region}, {country}
|
🌎 {city}, {region}, {country}
|
||||||
🛴 {asn}
|
🛴 {asn}
|
||||||
|
|
||||||
Edge Connection:
|
Edge Connection:
|
||||||
{rtt} 📶 {httpversion} 🔒 {tlsversion} ➡ ⛅ {colo}
|
{rtt} 📶 {httpversion} 🔒 {tlsversion} ➡ ⛅ {colo}
|
||||||
" property="og:description"/></head>
|
" property="og:description"/></head>
|
||||||
<title>${BRANDING_NAME}</title>
|
<title>${BRANDING_NAME}</title>
|
||||||
<style>
|
<style>
|
||||||
body {
|
body {
|
||||||
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif;
|
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif;
|
||||||
padding: 0 20px;
|
padding: 0 20px;
|
||||||
}
|
}
|
||||||
h1 {
|
h1 {
|
||||||
font-size: 4em;
|
font-size: 4em;
|
||||||
font-weight: 900;
|
font-weight: 900;
|
||||||
margin-bottom: 0;
|
margin-bottom: 0;
|
||||||
}
|
}
|
||||||
h2 {
|
h2 {
|
||||||
white-space: pre-wrap;
|
white-space: pre-wrap;
|
||||||
}
|
}
|
||||||
p {
|
p {
|
||||||
font-size: 10px;
|
font-size: 10px;
|
||||||
opacity: 0.3;
|
opacity: 0.3;
|
||||||
}
|
}
|
||||||
.cf {
|
.cf {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
height: 48px;
|
height: 48px;
|
||||||
width: 48px;
|
width: 48px;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<h1>${BRANDING_NAME}</h1>
|
<h1>${BRANDING_NAME}</h1>
|
||||||
<h3>A better way to embed X / Twitter posts on Discord, Telegram, and more.</h2>
|
<h3>A better way to embed X / Twitter posts on Discord, Telegram, and more.</h2>
|
||||||
<h2>Worker release: ${RELEASE_NAME}</h2>
|
<h2>Worker release: ${RELEASE_NAME}</h2>
|
||||||
<br>
|
<br>
|
||||||
<h3>Stats for nerds:</h3>
|
<h3>Stats for nerds:</h3>
|
||||||
<h2>Edge Connection:
|
<h2>Edge Connection:
|
||||||
{rtt} 📶 {httpversion} 🔒 {tlsversion} ➡ <img class="cf" referrerpolicy="no-referrer" src="https://cdn.discordapp.com/emojis/988895299693080616.webp?size=96&quality=lossless"> {colo}</h2>
|
{rtt} 📶 {httpversion} 🔒 {tlsversion} ➡ <img class="cf" referrerpolicy="no-referrer" src="https://cdn.discordapp.com/emojis/988895299693080616.webp?size=96&quality=lossless"> {colo}</h2>
|
||||||
<h2>User Agent:
|
<h2>User Agent:
|
||||||
{ua}</h2>
|
{ua}</h2>
|
||||||
</body>
|
</body>
|
||||||
</html>`
|
</html>`
|
||||||
|
.replace(/( {2})/g, '')
|
||||||
|
.replace(/>\s+</gm, '><'),
|
||||||
|
MESSAGE_HTML: `<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<meta content="${BRANDING_NAME}" property="og:title"/>
|
||||||
|
<meta content="${BRANDING_NAME}" property="og:site_name"/>
|
||||||
|
<title>${BRANDING_NAME}</title>
|
||||||
|
<style>
|
||||||
|
body {
|
||||||
|
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif;
|
||||||
|
padding: 0 20px;
|
||||||
|
}
|
||||||
|
h1 {
|
||||||
|
font-size: 4em;
|
||||||
|
font-weight: 900;
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
h2 {
|
||||||
|
white-space: pre-wrap;
|
||||||
|
}
|
||||||
|
p {
|
||||||
|
font-size: 10px;
|
||||||
|
opacity: 0.3;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>${BRANDING_NAME}</h1>
|
||||||
|
<h2>{message}</h2>
|
||||||
|
</body>
|
||||||
|
</html>`
|
||||||
.replace(/( {2})/g, '')
|
.replace(/( {2})/g, '')
|
||||||
.replace(/>\s+</gm, '><'),
|
.replace(/>\s+</gm, '><'),
|
||||||
DEFAULT_AUTHOR_TEXT: 'Twitter',
|
DEFAULT_AUTHOR_TEXT: 'Twitter',
|
||||||
|
|
|
@ -3,7 +3,7 @@ import { cacheWrapper } from '../src/server';
|
||||||
const botHeaders = { 'User-Agent': 'Discordbot/2.0' };
|
const botHeaders = { 'User-Agent': 'Discordbot/2.0' };
|
||||||
const humanHeaders = {
|
const humanHeaders = {
|
||||||
'User-Agent':
|
'User-Agent':
|
||||||
'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/103.0.0.0 Safari/537.36'
|
'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/116.0.0.0 Safari/537.36'
|
||||||
};
|
};
|
||||||
const githubUrl = 'https://github.com/FixTweet/FixTweet';
|
const githubUrl = 'https://github.com/FixTweet/FixTweet';
|
||||||
const twitterBaseUrl = 'https://twitter.com';
|
const twitterBaseUrl = 'https://twitter.com';
|
||||||
|
@ -54,6 +54,17 @@ test('Tweet redirect human', async () => {
|
||||||
expect(result.headers.get('location')).toEqual('https://twitter.com/jack/status/20');
|
expect(result.headers.get('location')).toEqual('https://twitter.com/jack/status/20');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('Tweet redirect human custom base redirect', async () => {
|
||||||
|
const result = await cacheWrapper(
|
||||||
|
new Request('https://fxtwitter.com/jack/status/20', {
|
||||||
|
method: 'GET',
|
||||||
|
headers: { ...humanHeaders, 'Cookie': 'cf_clearance=a; base_redirect=https://nitter.net' }
|
||||||
|
})
|
||||||
|
);
|
||||||
|
expect(result.status).toEqual(302);
|
||||||
|
expect(result.headers.get('location')).toEqual('https://nitter.net/jack/status/20');
|
||||||
|
});
|
||||||
|
|
||||||
test('Twitter moment redirect', async () => {
|
test('Twitter moment redirect', async () => {
|
||||||
const result = await cacheWrapper(
|
const result = await cacheWrapper(
|
||||||
new Request(
|
new Request(
|
||||||
|
|
Loading…
Add table
Reference in a new issue