Skip to content

teamreflex/fab-fetch

Repository files navigation

fab-fetch

Fetch and archive content from Fab.

A Dockerized version converted to Postgres and Prisma can be found on the 3.0 branch.

eat shit blockberry

Requirements

  • Only tested on Node 14
  • A Fab account, either email & password or an already generated access token.
  • Python, as the sqlite3 Node binding needs it to compile

Setup

$ git clone [email protected]:teamreflex/fab-fetch.git
$ cd fab-fetch
$ npm install
$ cp .env.example .env
$ nano .env # Fill in your details here, and update the user-agent

Running

$ npm start

Environment Variables

TIMEOUT (integer)

Performs message fetching every x milliseconds (60000 equals 1 minute).

FAB_USER_ID (integer)

This is your user ID from the API. You may need to start the bot once using FAB_EMAIL and FAB_PASSWORD to read your user ID. This is not your email or username, it's a unique number assigned to your account and is required for multiple endpoints.

FAB_VERSION (string)

Should be the most up to date version of the app, fetch it from the Google Play Store. The bot will crash when the API responds with a version mismatch.

FAB_USER_AGENT (string)

Keep this as the default unless you know what you're doing. The %version% template can be used to insert the version number set in FAB_VERSION. Neowiz could easily block any requests from Android Emulator user agents, which is why this is configurable.

API_URL (string)

Don't touch this.

ENVIRONMENT (string: prod | dev)

Toggles whether or not to run once or to run on an interval as set in the env. The run dev commands have been removed.

TWITTER_ENABLED (boolean)

Toggles posting for both the archives and profiles accounts. The -without-posting commands have been removed.

DECRYPT_ALL (boolean)

Toggles whether or not the bot just outright pays for every post and decrypts the URLs. false continues using the bruteforce method. Members using Android phones (only HaSeul currently) still pay for the posts and decrypt.

PAY_FOR_USER_IDS (string)

Provide a comma delimited string of user IDs to pay for. Defaults to "4,7" which is HaSeul and Kim Lip (as of 24/9/22).

VOICE_COMMENTS_ENABLED (boolean)

Saves voice comment audio files or not. These get placed in the voice_comments/ folder, and is affected by the MONTHLY_FOLDERS option.

IMAGE_COMMENTS_ENABLED (boolean)

Saves image comment files or not. These get placed in the image_comments/ folder, and is affected by the MONTHLY_FOLDERS option.

MONTHLY_FOLDERS (boolean)

Changes the folder structure. Enabled saves images to a monthly folder structure such as images/JinSoul/2022-06/, disables continues using daily folders like images/JinSoul/220613/.

PAY_ON_FALLBACK (boolean)

If a message is found with no media attached, enabling this option forces the bot to pay for the message as a fallback measure. Disabled means it just skips and saves to the database.

Encryption ~ June 2022

Neowiz has started to encrypt image URLs for thumbnails, letter images, postcard videos and postcard images.

It's using AES256 CBC PKCS5, and the secret key is built up from several parts:

  • A hardcoded IV string
  • Two strings pulled from the same "seed" map. Index 0-9 corresponds to unique strings.
    • The first takes the last digit from the user's userID and plugs that into the map to get a string.
    • The second takes the last digit from the letter/postcard updatedAt timestamp (converted to seconds first) and plugs that into the map.
  • A hardcoded string
  • The last three digits of the letter/postcard updatedAt (converted to seconds first)

This encryption ("password key" internally) key is concatenated like so:

const passwordKey = `${seed1}_${seed2}${hardcode}${lastThree}`;

The relevant strings can be found in src/encryption.ts. A caveat to this is that the IV string, hardcoded password key string as well as all 10 seed strings are hardcoded within the application, and requires decompilation to retreive. Chances are each new application update will require these to be retrived.

The API also prefixes and suffixes the URLs with a 6 digit long random number, this results in the hash changing upon every request, likely a obfuscation technique to throw people off.

How?

Because all content is just on CloudFront, and each message has either the first image (letter), or the thumbnail (postcard) attached. Using these, we can bruteforce the rest of the message.

The way this works is that for letters (standard image post), the URL is structured like so:

https://dnkvjm1f8biz3.cloudfront.net/images/letter/197/1645623283_20220223223440_1_f.jpg
https://dnkvjm1f8biz3.cloudfront.net/images/letter/LETTERID/UNIXTIMESTAMP_DATETIME_IMAGENUMBER_f.jpg

In most cases, all you need to do is increment IMAGENUMBER and you'll get the next image. In some cases, you will need to increment UNIXTIMESTAMP, as obviously things overlap two milliseconds from time to time. I'm yet to find a post that requires a DATETIME increment, but that would be trivial to add.

As for postcards (video posts), this is a little different. The thumbnail URL is structured like so:

https://dnkvjm1f8biz3.cloudfront.net/images/postcard/27/1645498037_20220222114713_b.jpg
https://dnkvjm1f8biz3.cloudfront.net/images/postcard/POSTCARDID/UNIXTIMESTAMP_DATETIME_b.jpg

Because postcards only have one piece of content, we can also bruteforce the .mp4 based on the thumbnail .jpg. This is the .mp4 URL for the thumbnail .jpg above:

https://dnkvjm1f8biz3.cloudfront.net/images/postcard/27/1645498036_20220222114713_f.mp4

Naturally, thumbnails can only be generated after the video has been uploaded, which means in some cases, we have to decrement UNIXTIMESTAMP rather than increase it, as well as updating the extension from _b.jpg to _f.mp4.

Unfortunately, this only applies for members using iPhones. Posts from Androids are not predictable:

https://dnkvjm1f8biz3.cloudfront.net/images/letter/207/1645746549_IMAGE_20220225_084902_2362453240698742359844.jpg

It starts off much the same until the final segment. I have no idea what it could possibly correspond to, so for now I've made the bot just check for any URL including _IMAGE_, and it will just pay for that post. Thankfully it's just HaSeul and Yves for now, but Neowiz could start randomizing images whenever they wanted, and it would be trivial to do.

How? (update: 16/03/22)

On March 15th, Neowiz updated the API so messages now have a thumbnail property.

Pre-update messages have had their thumbnail properties set to the first image URL, and the bot will default to checking for this. Post-update messages for some reason have no thumbnail and an empty images array. This is assumingly a bug, as it results in no thumbnails in the app.

To compensate for this, the bot will derive valid (or close enough) image URLs from the message createdAt timestamp and the message letterId. An example of how this works can be found in tools.js, run like so:

$ node tools.js 1647325657000 347
> https://dnkvjm1f8biz3.cloudfront.net/images/letter/347/1647325657_20220315152737_1_f.jpg

This is the exact code that makes this happen:

const time = DateTime.fromMillis(Number(timestamp), { zone: "Asia/Seoul" });
console.log(
  `https://dnkvjm1f8biz3.cloudfront.net/images/letter/${letter}/${time.toFormat(
    "X"
  )}_${time.toFormat("yyyyMMddHHmmss")}_1_f.jpg`
);

This URL acts as the starting point for the bruteforce function, and it suffers from the same drawbacks as previously, including a new one:

  • The IMAGENUMBER must still be incremented for multi-image messages.
  • The UNIXTIMESTAMP must still increment in some cases.
  • Now the DATETIME must often be decremented, sometimes by 1, sometimes by 2. The bot allows a maximum of 2 retries (a decrement of 2) before it fails.

Because there's no longer any URLs by default and it's unable to check the string for markers of an Android post, the bot checks for a userId of 4 (HaSeul) and defaults to paying for those posts. Yves recently swapped back to an iPhone, so for now it's hardcoded. In the event Yves swaps back, the bot will simply skip anything with no media, much like it does already for the one message that was sent with just text.

Caveats & Considerations

  • There is no need to go manually view a post before the bot reads it.
    • The "this message requires points to open" is client side.
    • As soon as a post is retreived from the API, their backend deducts points.
    • Because of this, anything you want to do with the API can be entirely automated.
  • The user-agent requires the latest Android app version number. They have a little bit of leeway in terms of enforcing switching over during update rollouts, but requests will fail once they fully switch everything over.
    • The bot will scrape the Fab Play Store page for the latest version, and insert it into the user-agent on every request.
    • It only does this in production/non-dev mode, so make sure FAB_VERSION is set to the latest while in dev.
    • Support for fetching the latest version number from the Play Store has been removed. As of late May, it's now hidden behind a modal that can't be scraped. FAB_VERSION is now required.
  • As of 2.0, it no longer judges whether or not to download/post to Twitter on if the image has been downloaded, but stores everything in a sqlite database.
    • It also pulls posts in by fetching each artist's messages endpoint, then skipping any messages that exist in the database, thus cutting down on the amount of image bruteforcing going on.
    • Due to using the artist endpoint, it may find in-review messages. Neowiz were recently alerted to this leaking posts, but I'm not 100% sure if that was entirely fixed.
  • There's little error handling. I run it using pm2 so it'll just restart upon an error crash, like during maintenance periods.
  • Twitter rate limiting when enabling Twitter posting, while having more than 10~ posts to update will also crash it.
  • I have no idea if they have rate limiting or account banning or what. I've been running a couple bots using the same account credentials since the weekend the app launched and everything has been fine, but Neowiz could quite easily block Android emulator usage/user-agents, accounts querying on intervals etc.
    • Because of the uncertainty, just use common sense. Personally, I have my access token in the environment, as if it gets in a crash loop and keeps spamming requests, at least it'll just look like I'm spamming refresh on the home screen rather than spamming login requests.
  • Neowiz have made some REST API 101 rookie mistakes already, so if you find anything suspicious then just report it. They're quick to respond.

License

I don't really know what MIT permits, it's just always been my default. Don't really care what you do, just give credit I guess.

Contact

About

Fetch and archive content from Fab.

Resources

License

Stars

Watchers

Forks