diff --git a/README.md b/README.md index 1d2c9bc..b3b40c3 100644 --- a/README.md +++ b/README.md @@ -42,34 +42,34 @@ On Discord, we'll also automatically embed videos linked from other platforms, s ## Embed Polls -If you want to share the results of a Twitter poll, you can do so by just linking the Tweet using FixTweet. +If you want to share the results of a Twitter poll, you can do so by just linking the post using FixTweet. ![](https://cdn.discordapp.com/attachments/165560751363325952/1006331192372629544/FixTweet.png) -## Embed Quote Tweets & Media +## Embed Quotes & Media -Quote tweets and their media can provide important context to a Tweet. So we'll automatically add said context, and even media if there isn't already media embedded in the quote. +Quotes and their media can provide important context to a post. So we'll automatically add said context, and even media if there isn't already media embedded in the quote. ![](https://cdn.discordapp.com/attachments/165560751363325952/1006334041202630757/FixTweet.png) -## Translate Tweets +## Translate Posts -You can translate a tweet into any other supported language, with the original and translated text displayed as space allows. +You can translate a post into any other supported language, with the original and translated text displayed as space allows. -Just append a tweet with its 2-letter ISO language code. So for English, add `/en` at the end. +Just append a post with its 2-letter ISO language code. So for English, add `/en` at the end. ![](https://cdn.discordapp.com/attachments/165560751363325952/1006333011136102400/FixTweet.png) ## Direct media links -Want to link directly to a Tweet's media without the embed? You can easily do that using FixTweet. +Want to link directly to a post's media without the embed? You can easily do that using FixTweet. ![Image demonstrating the feature](https://cdn.discordapp.com/attachments/165560751363325952/1006338772192989194/FixTweet.png) There are a few supported ways to do this: - Add `d.` or `dl.` before the domain (so, `d.twittpr.com` or `dl.fxtwitter.com`) -- Add `.mp4` to the end of videos or `.jpg` to the end of images, after the tweet ID +- Add `.mp4` to the end of videos or `.jpg` to the end of images, after the post ID - Add `/dl` or `/dir` between the domain and the username Examples from above: @@ -78,7 +78,7 @@ Examples from above: - `https://fxtwitter.com/dangeredwolf/status/1548117889437208581.jpg` - `https://fxtwitter.com/dl/dangeredwolf/status/1548117889437208581` -Tweets with multiple images are supported, so you can do something like this and it will pick the correct one: +Posts with multiple images are supported, so you can do something like this and it will pick the correct one: `https://d.twittpr.com/dangeredwolf/status/1547514042146865153/photo/3` @@ -92,19 +92,19 @@ The default Twitter embeds include t.co link shorteners, which make it difficult ## Color-matched embeds on Discord -We use Twitter's color data for either the first image/video of the tweet, or the author's profile picture. It makes the embed's appearance more _aesthetic_, as well as in line with the content of the Tweet. +We use Twitter's color data for either the first image/video of the post, or the author's profile picture. It makes the embed's appearance more _aesthetic_, as well as in line with the content of the post. ![](https://cdn.discordapp.com/attachments/165560751363325952/1006370708265386134/fixtweetcolor3.png) ## Built with privacy in mind -FixTweet doesn't save logs of what tweets you're sending, nor do we have a public record of what tweets are being embedded by FixTweet. +FixTweet doesn't save logs of what posts you're sending, nor do we have a public record of what posts are being embedded by FixTweet. -In fact, because our core embedding and API service uses Cloudflare Workers, FixTweet can only run when you send it a request. Its memory doesn't stick around, and it doesn't have a file system or database to read from at all. That is how we keep our privacy promise by building it into the architecture. We use Cloudflare Analytics Engine to aggregate basic, anonymous statistics, which do not include information that could identify individual users or Tweets. My goal is always to provide a good public service, and FixTweet doesn't have any ads or tracking to make money off of, nor do we sell data. +In fact, because our core embedding and API service uses Cloudflare Workers, FixTweet can only run when you send it a request. Its memory doesn't stick around, and it doesn't have a file system or database to read from at all. That is how we keep our privacy promise by building it into the architecture. We use Cloudflare Analytics Engine to aggregate basic, anonymous statistics, which do not include information that could identify individual users or posts. My goal is always to provide a good public service, and FixTweet doesn't have any ads or tracking to make money off of, nor do we sell data. 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 or Twitter API is down) 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 advertising. +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 post as they are only used for Twitter telemetry and advertising. --- @@ -114,14 +114,14 @@ In many ways, FixTweet has richer embeds and does more. Here's a table comparing | | FixTweet | Twitter default | vxTwitter (BetterTwitFix) | Twxtter (sixFix) | | --------------------------------------- | :---------------------------------: | :------------------------------: | :-------------------------------------------------: | :-----------------------------------: | -| Embed Tweets / Images | :heavy_check_mark: | :heavy_check_mark: | :heavy_check_mark: | :heavy_check_mark: | -| Embed profile pictures on text Tweets | :heavy_check_mark: | :x: | :heavy_check_mark: | :heavy_check_mark: | +| Embed Posts / Images | :heavy_check_mark: | :heavy_check_mark: | :heavy_check_mark: | :heavy_check_mark: | +| Embed profile pictures on text posts | :heavy_check_mark: | :x: | :heavy_check_mark: | :heavy_check_mark: | | Embed Twitter Videos | :heavy_check_mark: | :heavy_minus_sign: Discord Only¹ | :heavy_check_mark: | :heavy_check_mark: | | Embed External Videos (YouTube, etc.) | :heavy_check_mark:⁵ | :x: | :x:⁴ | :x: | | Embed Poll results | :heavy_check_mark: | :x: | [:heavy_check_mark:][polladd] | :x: | -| Embed Quote Tweets | :heavy_check_mark: | :x: | :ballot_box_with_check: Without Media | :ballot_box_with_check: Without Media | +| Embed Quotes | :heavy_check_mark: | :x: | :ballot_box_with_check: Without Media | :ballot_box_with_check: Without Media | | Embed Multiple Images | :heavy_check_mark: | :heavy_minus_sign: Discord Only³ | :heavy_check_mark: | :x: | -| Translate Tweets | :heavy_check_mark: | :x: | :x: | :x: | +| Translate Posts | :heavy_check_mark: | :x: | :x: | :x: | | Publicly accessible embed index | :x:² | N/A | :x:² | :heavy_check_mark: | | Replace t.co with original links | :heavy_check_mark: | :x: | :x: | :x: | | Media-based embed colors on Discord | :heavy_check_mark: | :x: | :x: | :x: | @@ -129,7 +129,7 @@ In many ways, FixTweet has richer embeds and does more. Here's a table comparing | Strip Twitter tracking info on redirect | :heavy_check_mark: | :x: | :heavy_check_mark: | :heavy_check_mark: | | Show retweet, like, reply counts | :heavy_check_mark: | :heavy_minus_sign: Discord Only³ | :ballot_box_with_check: No replies | :ballot_box_with_check: No replies | | Discord sed replace (`s/`) friendly | :ballot_box_with_check: twittpr.com | N/A | :x: | :heavy_check_mark: | -| Tweet fetch API for Developers | :heavy_check_mark: | N/A | :x: | :heavy_check_mark: | +| Status fetch API for Developers | :heavy_check_mark: | N/A | :x: | :heavy_check_mark: | | DDoS protection & low latency globally | :heavy_check_mark: | N/A | :x: | :x: | | Last commit | [![][flc]][fc] | N/A | [![][vlc]][vc] | [![][slc]][sc] | @@ -171,7 +171,7 @@ Once you're set up with your worker on `*.workers.dev`, [add your worker to your Populate Sentry details in your `.env` to use Sentry in your product to catch exceptions. -In 2023, Twitter began blocking Tweets with NSFW media from the guest API. We use a service binding code-named [elongator](https://github.com/FixTweet/elongator), which use empty Twitter accounts to make these requests successfully. This is an optional component and is only necessary for those who plan to support embedding NSFW media. This method also means you never have to pay Elon Musk to use Twitter's official API. +In 2023, Twitter began blocking posts with NSFW media from the guest API. We use a service binding code-named [elongator](https://github.com/FixTweet/elongator), which use empty Twitter accounts to make these requests successfully. This is an optional component and is only necessary for those who plan to support embedding NSFW media. This method also means you never have to pay Elon Musk to use Twitter's official API. --- @@ -183,7 +183,7 @@ They all run the exact same worker and function identically... mostly. `fxtwitter.com` is the primary domain these days, with `twittpr.com` as an alternative that allows for quick sed replacement. `pxtwitter.com` was our original domain, but we consider that to be deprecated now. -The way we handle this is that on post-deprecation Tweets linked using `pxtwitter.com`, instead of saying "FixTweet", it will have a notice that we've moved to `fxtwitter.com`. The embeds and redirects will still work, and Tweets posted before deprecation will not be unaffected at all, but it will gently encourage people to migrate by applying the notice to newer Tweets. +The way we handle this is that on post-deprecation posts linked using `pxtwitter.com`, instead of saying "FixTweet", it will have a notice that we've moved to `fxtwitter.com`. The embeds and redirects will still work, and posts posted before deprecation will not be unaffected at all, but it will gently encourage people to migrate by applying the notice to newer posts. `pxtwitter.com` was our original domain for the project, bought the day before we launched FixTweet (then known as pxTwitter). I was trying to find something memorable and `px` kinda sounds like "pix" or can mean "pixels" which is fitting as a service that can embed images, videos, etc. Not long after, I bought `twittpr.com` because it's easier to do sed replacement with on Discord (`s/e/p`), and because it had a `p` in it, it was sorta related to pxTwitter. They have always functioned identically. @@ -193,9 +193,9 @@ A couple weeks later, I acquired the `fxtwitter.com` domain from RobinUniverse a Telegram's embedding servers sometimes never even send us a request to embed a URL, possibly due to their servers being overloaded. If you have a link that is broken you can try one of FixTweet's other domains (`fxtwitter.com`, `pxtwitter.com`, `twittpr.com`) or use [Webpage Bot](https://t.me/WebpageBot) to try to clear the cache of the embed. -### What if I don't want FixTweet to combine my Tweet's images together with multi-image? +### What if I don't want FixTweet to combine my post's images together with multi-image? -No problem! You can pick any specific photo from a Tweet using Twitter's own URL syntax (`/photo/1` is the first photo of a tweet) and we'll render you the full-resolution original image. +No problem! You can pick any specific photo from a post using Twitter's own URL syntax (`/photo/1` is the first photo of a post) and we'll render you the full-resolution original image. --- diff --git a/src/embed/status.ts b/src/embed/status.ts index bdf9e8f..ca55ca0 100644 --- a/src/embed/status.ts +++ b/src/embed/status.ts @@ -35,10 +35,14 @@ export const handleStatus = async ( const api = await statusAPI(status, language, event as FetchEvent, flags); const tweet = api?.tweet as APITweet; - + const isTelegram = (userAgent || '').indexOf('Telegram') > -1; /* Should sensitive posts be allowed Instant View? */ - const useIV = isTelegram /*&& !tweet.possibly_sensitive*/ && !flags?.direct && !flags?.api && (tweet.media?.mosaic || tweet.is_note_tweet); + const useIV = + isTelegram /*&& !tweet.possibly_sensitive*/ && + !flags?.direct && + !flags?.api && + (tweet.media?.mosaic || tweet.is_note_tweet); let ivbody = ''; @@ -297,9 +301,7 @@ export const handleStatus = async ( `` ); } else { - headers.push( - `` - ); + headers.push(``); } } diff --git a/src/render/instantview.ts b/src/render/instantview.ts index b67cc46..db55aa2 100644 --- a/src/render/instantview.ts +++ b/src/render/instantview.ts @@ -1,10 +1,10 @@ -import { Constants } from "../constants"; -import { getSocialTextIV } from "../helpers/author"; -import { sanitizeText } from "../helpers/utils"; +import { Constants } from '../constants'; +import { getSocialTextIV } from '../helpers/author'; +import { sanitizeText } from '../helpers/utils'; const populateUserLinks = (tweet: APITweet, text: string): string => { /* TODO: Maybe we can add username splices to our API so only genuinely valid users are linked? */ - text.match(/@(\w{1,15})/g)?.forEach((match) => { + text.match(/@(\w{1,15})/g)?.forEach(match => { const username = match.replace('@', ''); text = text.replace( match, @@ -12,13 +12,13 @@ const populateUserLinks = (tweet: APITweet, text: string): string => { ); }); return text; -} +}; const generateTweetMedia = (tweet: APITweet): string => { let media = ''; if (tweet.media?.all?.length) { - tweet.media.all.forEach((mediaItem) => { - switch(mediaItem.type) { + tweet.media.all.forEach(mediaItem => { + switch (mediaItem.type) { case 'photo': media += `${tweet.author.name}'s photo`; break; @@ -32,7 +32,7 @@ const generateTweetMedia = (tweet: APITweet): string => { }); } return media; -} +}; // const formatDateTime = (date: Date): string => { // const yyyy = date.getFullYear(); @@ -45,18 +45,18 @@ const generateTweetMedia = (tweet: APITweet): string => { const htmlifyLinks = (input: string): string => { const urlPattern = /\bhttps?:\/\/\S+/g; - return input.replace(urlPattern, (url) => { - return `${url}`; + return input.replace(urlPattern, url => { + return `${url}`; }); -} +}; const htmlifyHashtags = (input: string): string => { const hashtagPattern = /#([a-zA-Z_]\w*)/g; return input.replace(hashtagPattern, (match, hashtag) => { - const encodedHashtag = encodeURIComponent(hashtag); - return `${match}`; + const encodedHashtag = encodeURIComponent(hashtag); + return `${match}`; }); -} +}; export const renderInstantView = (properties: RenderProperties): ResponseInstructions => { console.log('Generating Instant View (placeholder)...'); @@ -81,13 +81,17 @@ export const renderInstantView = (properties: RenderProperties): ResponseInstruc
- If you can see this, your browser is doing something weird with your user agent. View original post + If you can see this, your browser is doing something weird with your user agent. View original post

${tweet.author.name} (@${tweet.author.screen_name})

Instant View (✨ Beta) - View original

- + diff --git a/src/strings.ts b/src/strings.ts index 729f60d..295ba39 100644 --- a/src/strings.ts +++ b/src/strings.ts @@ -114,7 +114,7 @@ This is caused by Twitter API downtime or a new bug. Try again in a little while

${BRANDING_NAME}

-

A better way to embed Tweets on Discord, Telegram, and more.

+

A better way to embed X / Twitter posts on Discord, Telegram, and more.

Worker release: ${RELEASE_NAME}


Stats for nerds:

@@ -146,9 +146,9 @@ This is caused by Twitter API downtime or a new bug. Try again in a little while FINAL_POLL_RESULTS: 'Final results', ERROR_API_FAIL: - "Tweet failed to load due to an API error. This is most common with NSFW Tweets as Twitter / X currently blocks us from fetching them. We're still working on a fix for that.🙏", - ERROR_PRIVATE: `Sorry, we can't embed this Tweet because the user is private or suspended :(`, - ERROR_TWEET_NOT_FOUND: `Sorry, that Tweet doesn't exist :(`, + "Post failed to load due to an API error. This is most common with NSFW Posts as Twitter / X currently blocks us from fetching them. We're still working on a fix for that.🙏", + ERROR_PRIVATE: `Sorry, we can't embed this post because the user is private or suspended :(`, + ERROR_TWEET_NOT_FOUND: `Sorry, that post doesn't exist :(`, ERROR_USER_NOT_FOUND: `Sorry, that user doesn't exist :(`, ERROR_UNKNOWN: `Unknown error occurred, sorry about that :(`, @@ -167,7 +167,7 @@ This is caused by Twitter API downtime or a new bug. Try again in a little while # \\-------------------------------------------/ # Do you breathe air? Are you a human? Do you know how to write code? -# Do you want an easy way to fetch Tweets but Elon Musk wants to charge you $100 per month? +# Do you want an easy way to fetch posts but Elon Musk wants to charge you $100 per month? # Did you know we have a fetch API you can use for free, no API keys required? # Check out the docs at https://${API_HOST_LIST.split(',')[0]} to learn how to use it