Just thought I’d share this since it’s working for me at my home instance of federate.cc, even though it’s not documented in the Lemmy hosting guide.

The image server used by Lemmy, pict-rs, recently added support for object storage like Amazon S3, instead of serving images directly off the disk. This is potentially interesting to you because object storage is orders of magnitude cheaper than disk storage with a VM.

By way of example, I’m hosting my setup on Vultr, but this applies to say Digital Ocean or AWS as well. Going from a 50GB to a 100GB VM instance on Vultr will take you from $12 to $24/month. Up to 180GB, $48/month. Of course these include CPU and RAM step-ups too, but I’m focusing only on disk space for now.

Vultr’s object storage by comparison is $5/month for 1TB of storage and includes a separate 1TB of bandwidth that doesn’t count against your main VM, plus this content is served off of Vultr’s CDN instead of your instance, meaning even less CPU load for you.

This is pretty easy to do. What we’ll be doing is diverging slightly from the official Lemmy ansible setup to add some different environment variables to pict-rs.

After step 5, before running the ansible playbook, we’re going to modify the ansible template slightly:

cd templates/

cp docker-compose.yml docker-compose.yml.original

Now we’re going to edit the docker-compose.yml with your favourite text editor, personally I like micro but vim, emacs, nano or whatever will do…

favourite-editor docker-compose.yml

Down around line 67 begins the section for pictrs, you’ll notice under the environment section there are a bunch of things that the Lemmy guys predefined. We’re going to add some here to take advantage of the new support for object storage in pict-rs 0.4+:

At the bottom of the environment section we’ll add these new vars:

  - PICTRS__STORE__TYPE=object_storage
  - PICTRS__STORE__ENDPOINT=Your Object Store Endpoint
  - PICTRS__STORE__BUCKET_NAME=Your Bucket Name
  - PICTRS__STORE__REGION=Your Bucket Region
  - PICTRS__STORE__USE_PATH_STYLE=false
  - PICTRS__STORE__ACCESS_KEY=Your Access Key
  - PICTRS__STORE__SECRET_KEY=Your Secret Key

So your whole pictrs section looks something like this: https://pastebin.com/X1dP1jew

The actual bucket name, region, access key and secret key will come from your provider. If you’re using Vultr like me then they are under the details after you’ve created your object store, under Overview -> S3 Credentials. On Vultr your endpoint will be something like sjc1.vultrobjects.com, and your region is the domain prefix, so in this case sjc1.

Now you can install as usual. If you have an existing instance already deployed, there is an additional migration command you have to run to move your on-disk images into the object storage.

You’re now good to go and things should pretty much behave like before, except pict-rs will be saving images to your designated cloud/object store, and when serving images it will instead redirect clients to pull directly from the object store, saving you a lot of storage, cpu use and bandwidth, and therefore money.

Hope this helps someone, I am not an expert in either Lemmy administration nor Linux sysadmin stuff, but I can say I’ve done this on my own instance at federate.cc and so far I can’t see any ill effects.

Happy Lemmy-ing!

  • Toribor@corndog.uk
    link
    fedilink
    English
    arrow-up
    2
    ·
    2 years ago

    Thank you for sharing this. I’m going to try to go through this migration shortly.

    Right now I’m running my instance on a fairly lean VPS so being able to lighten the CPU load and not have to pre-allocate disk space is super useful.

    • Toribor@corndog.uk
      link
      fedilink
      English
      arrow-up
      1
      ·
      edit-2
      2 years ago

      Replying to confirm that this works and went very smoothly! If you can see my profile picture, it’s on S3 instead of disk now.

      I’m using pure ansible to deploy my containers (instead of docker compose) so I had to figure out how to start the pictrs container without actually starting pictrs so that I could run the migration. I ended up stopping the container and then running this to perform the migration:

      docker run --name pictrs-migration \
        --user 991:991 \
        -v /my-pictrs-path/:/mnt \
        --rm \
        asonix/pictrs:0.4.0-rc.14 \
        pict-rs \
          migrate-store \
          filesystem \
          object-storage \
              -e https://my-s3-endpoint \
              -b my-s3-bucket-name \
              -r my-region \
              -a my-key-id \
              -s my-key-secret
      

      Then I used ansible to redeploy the container with volume mount removed and the new s3 environment variables.

      Super easy!

  • hoodlem@hoodlem.me
    link
    fedilink
    English
    arrow-up
    2
    ·
    1 year ago

    Thank you for this write-up. Your post is the only place I can find on the internet on making the transition to object storage specifically with Lemmy.

  • Lodion 🇦🇺@aussie.zone
    link
    fedilink
    English
    arrow-up
    1
    ·
    2 years ago

    I’m using S3FS to achieve the same thing, but without modifying the ansible config or using native object storage within pict-rs.

  • poVoq@slrpnk.net
    link
    fedilink
    English
    arrow-up
    1
    ·
    2 years ago

    plus this content is served off of Vultr’s CDN instead of your instance, meaning even less CPU load for you.

    Currently the Lemmy backend proxies all image requests, so this isn’t true (for now).

    • sparky@lemmy.federate.cc@lemmy.federate.ccOP
      link
      fedilink
      English
      arrow-up
      1
      ·
      2 years ago

      Ah! Noted. That said, it is definitely storing the bytes on the object store. I imagine someone clever with nginx or such could set up some rewrite rules to bypass pictrs entirely for GET requests, but unfortunately that’s beyond my pay grade here.

  • Adam@lemmy.adambowl.es
    link
    fedilink
    English
    arrow-up
    1
    ·
    edit-2
    2 years ago

    I tried this with a brand new lemmy ansible setup using Vultr object storage, and my media upload requests respond with a timeout

    Request error: error sending request for url (http://pictrs:8080/image): error trying to connect: dns error: failed to lookup address information: Try again