diff --git a/README.md b/README.md index 6d5d2da..8a81835 100644 --- a/README.md +++ b/README.md @@ -80,7 +80,7 @@ FixTweet doesn't save logs of what tweets you're sending, nor do we have a publi In fact, because our core embedding and API service uses Cloudflare Workers, where FixTweet can only run when you send it a request and its memory doesn't stick around, and it doesn't have a file system or database to access at all. That is how we keep our privacy promise by building it into the architecture. My goal is always to provide a good public service, and FixTweet doesn't make any money. -Note: We use Cloudflare to cache FixTweet responses to make repeat access faster, which have a maximum TTL of 1 hour. Temporary real-time logging in the terminal (specifically `wrangler tail`) may be used only by the developer while the Worker is being serviced or debugged (to make sure things work as they should), however these logs are only shown in the terminal and are never saved or used for any other purpose. +Note: We use Cloudflare to cache FixTweet responses to make repeat access faster, which have a maximum TTL of 1 hour. Temporary real-time logging in the terminal (specifically `wrangler tail`) may be used only by the developer while the Worker is being serviced or debugged (to make sure things work as they should), however these logs are only shown in the terminal and are never saved or used for any other purpose. URLs that cause runtime errors in the script (aka Exceptions, usually exceedingly rare unless there was a faulty update pushed out) may be logged for a developer to diagnose the issue that is preventing your embed from working. On a different note, if the person who posted a FixTweet link forgot to strip tracking parameters (like `?s` and `&t`), we strip it upon redirecting to the Tweet as they are only used for Twitter Telemetry and Marketing. @@ -139,6 +139,8 @@ Clone the repo, install [Node.js](https://nodejs.org/) and run `npm install` in Once you're set up with your worker on `*.workers.dev`, [add your worker to your custom domain](https://developers.cloudflare.com/workers/platform/routing/custom-domains/). +Optional: Set the `EXCEPTION_DISCORD_WEBHOOK` secret to a Discord webhook URL to log exceptions to a Discord channel. At this time, we have not integrated an error handling SDK to keep the script small and execution times to a minimum, but it is something that may be explored in the future. + --- **Licensed under the permissive MIT license. Feel free to send a pull request!** diff --git a/src/env.d.ts b/src/env.d.ts index 93fcfcb..48128d6 100644 --- a/src/env.d.ts +++ b/src/env.d.ts @@ -5,3 +5,5 @@ declare const HOST_URL: string; declare const REDIRECT_URL: string; declare const MOSAIC_DOMAIN_LIST: string; declare const API_HOST: string; + +declare const EXCEPTION_DISCORD_WEBHOOK: string; diff --git a/src/mosaic.ts b/src/mosaic.ts index 542c66e..9583cae 100644 --- a/src/mosaic.ts +++ b/src/mosaic.ts @@ -44,7 +44,9 @@ export const handleMosaic = async ( path += `/${mosaicMedia[3]}`; } - const size = calcSize(mediaList.map(i => ({ width: i.width, height: i.height } as Size))); + const size = calcSize( + mediaList.map(i => ({ width: i.width, height: i.height } as Size)) + ); return { height: size.height, width: size.width, @@ -87,7 +89,7 @@ const calcSize = (images: Size[]): Size => { const bottom = calcHorizontalSize(images[2], images[3]); const all = calcVerticalSize(top, bottom); - const sizeMult = (all.width > 2000 || all.height > 2000) ? BIG_IMAGE_MULTIPLIER : 1; + const sizeMult = all.width > 2000 || all.height > 2000 ? BIG_IMAGE_MULTIPLIER : 1; return { width: all.width * sizeMult, height: all.height * sizeMult @@ -126,11 +128,11 @@ const calcVerticalSize = (first: Size, second: Size): VerticalSize => { swapped = true; } - const smallHeight = Math.round(big.width / small.width * small.height); + const smallHeight = Math.round((big.width / small.width) * small.height); return { width: big.width, height: smallHeight + SPACING_SIZE + big.height, firstHeight: swapped ? smallHeight : big.height, secondHeight: swapped ? big.height : smallHeight } as VerticalSize; -}; \ No newline at end of file +}; diff --git a/src/server.ts b/src/server.ts index 196bc92..e1e05c2 100644 --- a/src/server.ts +++ b/src/server.ts @@ -217,6 +217,31 @@ const cacheWrapper = async (event: FetchEvent): Promise => { /* Event to receive web requests on Cloudflare Worker */ -addEventListener('fetch', (event: FetchEvent) => { - event.respondWith(cacheWrapper(event)); +addEventListener('fetch', async (event: FetchEvent) => { + try { + event.respondWith(cacheWrapper(event)); + } catch (e: unknown) { + let error = e as Error; + if (typeof EXCEPTION_DISCORD_WEBHOOK !== 'undefined') { + try { + const a = await fetch(EXCEPTION_DISCORD_WEBHOOK, { + method: 'POST', + headers: { + 'Content-Type': 'application/json', + 'User-Agent': `${Constants.BRANDING_NAME}` + }, + body: JSON.stringify({ + embeds: [ + { + title: `Exception in ${Constants.BRANDING_NAME}`, + description: `${error} - occurred while processing ${event.request.url}`, + } + ] + }) + }); + } catch (e) { + console.log('Failed to send caught exception to Discord', e); + } + } + } }); diff --git a/src/tweetTypes.ts b/src/tweetTypes.ts index 43e4abc..b9e38bb 100644 --- a/src/tweetTypes.ts +++ b/src/tweetTypes.ts @@ -121,7 +121,7 @@ type TweetCardBindingValues = { }; type TweetCard = { - binding_values: TweetCardBindingValues; + binding_values: TweetCardBindingValues; name: string; };