fxtwitter-docker/src/worker.ts
2023-11-09 03:53:24 -05:00

114 lines
3 KiB
TypeScript

import { Env, Hono } from 'hono';
import { timing } from 'hono/timing';
import { logger } from 'hono/logger';
import { RewriteFrames } from '@sentry/integrations';
import { sentry } from '@hono/sentry';
import { Strings } from './strings';
import { Constants } from './constants';
import { api } from './realms/api/router';
import { twitter } from './realms/twitter/router';
import { cacheMiddleware } from './caches';
const noCache = 'max-age=0, no-cache, no-store, must-revalidate';
const app = new Hono<{ Bindings: { TwitterProxy: Fetcher; AnalyticsEngine: AnalyticsEngineDataset } }>({
getPath: req => {
let url: URL;
try {
url = new URL(req.url);
} catch (err) {
return '/error';
}
const baseHostName = url.hostname.split('.').slice(-2).join('.');
let realm = 'twitter';
/* Override if in API_HOST_LIST. Note that we have to check full hostname for this. */
if (Constants.API_HOST_LIST.includes(url.hostname)) {
realm = 'api';
} else if (Constants.STANDARD_DOMAIN_LIST.includes(baseHostName)) {
realm = 'twitter';
}
/* Defaults to Twitter realm if unknown domain specified (such as the *.workers.dev hostname or deprecated domain) */
console.log(`/${realm}${url.pathname}`);
return `/${realm}${url.pathname}`;
}
});
app.use(
'*',
sentry({
dsn: SENTRY_DSN,
requestDataOptions: {
allowedHeaders: /(.*)/,
allowedSearchParams: /(.*)/
},
integrations: [new RewriteFrames({ root: '/' })],
release: RELEASE_NAME
})
);
app.use('*', async (c, next) => {
/* Apply all headers from Constants.RESPONSE_HEADERS */
for (const [header, value] of Object.entries(Constants.RESPONSE_HEADERS)) {
c.header(header, value);
}
await next();
});
app.onError((err, c) => {
c.get('sentry').captureException(err);
/* workaround for silly TypeScript things */
const error = err as Error;
console.error(error.stack);
c.status(200);
c.header('cache-control', noCache);
c.header('content-type', 'text/html');
return c.text(Strings.ERROR_HTML);
});
app.use('*', logger());
app.use('*', async (c, next) => {
console.log(`Hello from ⛅ ${c.req.raw.cf?.colo || 'UNK'}`);
console.log('userAgent', c.req.header('user-agent'));
await next();
});
app.use('*', cacheMiddleware());
app.use('*', timing({ enabled: false }));
app.route(`/api`, api);
app.route(`/twitter`, twitter);
app.all(
'/error',
async (c) => {
c.header('cache-control', noCache);
c.status(400);
return c.body('')
}
);
export default {
async fetch(request: Request, env: Env, ctx: ExecutionContext) {
try {
return await app.fetch(request, env, ctx);
} catch (err) {
console.error(err);
console.log('Ouch, that error hurt so much Sentry couldnt catch it');
return new Response(Strings.ERROR_HTML, {
headers: {
...Constants.RESPONSE_HEADERS,
'content-type': 'text/html',
'cache-control': noCache
},
status: 200
});
}
}
};