Self-hosting a Bluesky PDS means running your own Personal Data Server that is capable of federating with the wider ATProto network. - https://atproto.com/guides/self-hosting


Set up your personal ATProto PDS with NixOs

This guide walks you through the process to host your own PDS instance using NixOs, agenix for secure environment management, and Caddy as a reverse proxy for traffic encryption.

Thanks to the Nixpkgs contributors for making this possible!

You need to have a server running NixOs. Using nixos-anywhere makes it easy to install on a vps. The server configuration should be in a NixOs flake.

Retrieve the SSH key from the host:

ssh-keyscan your.pds.domain

Add the ssh-rsa line to your flake’s secrets folder in the file secrets/secrets.nix:


let
  vps_ssh = "ssh-rsa ...";
in
{
  "pds-env.age".publicKeys = [ vps_ssh ];
}

Generate PDS secrets environment variables:

{
  # Generate JWT secret
  JWT_SECRET=$(openssl rand --hex 16)
  echo "PDS_JWT_SECRET=$JWT_SECRET"

  # Generate admin password
  ADMIN_PASSWORD=$(openssl rand --hex 16)
  echo "PDS_ADMIN_PASSWORD=$ADMIN_PASSWORD"

  # Generate PLC rotation key
  PLC_KEY=$(openssl ecparam --name secp256k1 --genkey --noout --outform DER | tail --bytes=+8 | head --bytes=32 | xxd --plain --cols 32)
  echo "PDS_PLC_ROTATION_KEY_K256_PRIVATE_KEY_HEX=$PLC_KEY"
} > /tmp/pds-secrets.env

use agenix to encrypt the pds-env file according to the secrets.nix config:

cd secrets/
agenix -e pds-env.age < /tmp/pds-env.env

Then you can delete /tmp/pds-env.env after keeping its secrets safe in a password manager.

Modify the configuration.nix of the flake to enable the PDS service and use Caddy as a proxy to handle encryption:

{
  config,
  pkgs,
  ...
}: {

  networking.firewall.allowedTCPPorts = [
    80
    443
  ];

  age.identityPaths = [ "/etc/ssh/ssh_host_rsa_key" ];

  age.secrets.pds-env = {
    file = ./secrets/pds-env.age;
    owner = "pds";
    group = "pds";
  };

  services.pds = {
    enable = true;
    settings = {
      PDS_HOSTNAME = "your.pds.domain";
      PDS_PORT = 3000;
      PDS_HOST = "127.0.0.1";
    };
    environmentFiles = [
      config.age.secrets.pds-env.path
    ];
  };


  services.caddy = {
    enable = true;
    configFile = pkgs.writeText "Caddyfile" ''
      your.pds.domain {
        reverse_proxy http://127.0.0.1:3000
      }
    '';
  };
}

Now, it is possible to test the flake for the ovh configuration (if you followed the note):

nixos-rebuild test --flake .#ovh-vps

Deploy the new configuration:

nixos-rebuild switch --flake .#ovh-vps --target-host "root@your.pds.domain"

The new personal PDS should be up and running at the chosen hostname:

curl https://your.pds.domain
         __                         __
        /\ \__                     /\ \__
    __  \ \ ,_\  _____   _ __   ___\ \ ,_\   ___
  /'__'\ \ \ \/ /\ '__'\/\''__\/ __'\ \ \/  / __'\
 /\ \L\.\_\ \ \_\ \ \L\ \ \ \//\ \L\ \ \ \_/\ \L\ \
 \ \__/.\_\\ \__\\ \ ,__/\ \_\\ \____/\ \__\ \____/
  \/__/\/_/ \/__/ \ \ \/  \/_/ \/___/  \/__/\/___/
                   \ \_\
                    \/_/


This is an AT Protocol Personal Data Server (aka, an atproto PDS)

Most API routes are under /xrpc/

      Code: https://github.com/bluesky-social/atproto
 Self-Host: https://github.com/bluesky-social/pds
  Protocol: https://atproto.com

To migrate an account to the new PDS, follow this guide: Migrating PDS account with goat.