{ pkgs, nix-packages, config, lib, nixpkgs-libjxl-update, system, ... }: let purge_url_script = pkgs.writeScript "purge-url" '' access_key=$(cat ${config.sops.secrets."services/bunny-key".path}) for url in $@; do url=$(echo $url | ${pkgs.python3}/bin/python3 -c "import sys; import urllib.parse; print(urllib.parse.quote(sys.stdin.read().strip()))") ${pkgs.curl}/bin/curl -H "Authorization: Bearer $access_key" -X POST "https://api.bunny.net/purge?url=$url&async=false" done ''; emoji_set_names = ["volpeon-blobfox-flip" "volpeon-blobfox" "volpeon-bunhd-flip" "volpeon-bunhd" "volpeon-drgn" "volpeon-fox" "volpeon-raccoon" "volpeon-vlpn" "lotte" "caro"]; emoji_sets = builtins.listToAttrs (map (name: { inherit name; value = "${pkgs."emoji-${name}"}"; }) emoji_set_names); copy_emoji_set = name: '' mkdir -p $out/emoji/${name} lndir ${emoji_sets.${name}} $out/emoji/${name} ''; static_dir = pkgs.stdenvNoCC.mkDerivation { name = "akkoma-static"; src = pkgs.emptyDirectory; nativeBuildInputs = with pkgs; [xorg.lndir]; akkoma_fe = nix-packages.packages.${pkgs.system}.pleroma-fe; akkoma_admin_fe = nix-packages.packages.${pkgs.system}.admin-fe; tos = ./terms-of-service.html; dontUnpack = false; installPhase = '' mkdir -p $out/frontends/pleroma-fe/stable lndir $akkoma_fe $out/frontends/pleroma-fe/stable mkdir -p $out/frontends/admin-fe/stable lndir $akkoma_admin_fe $out/frontends/admin-fe/stable ${toString (map copy_emoji_set emoji_set_names)} mkdir $out/emoji/misc ln -s ${./therian.png} $out/emoji/misc/therian.png mkdir $out/static cp $tos $out/static/terms-of-service.html ''; }; ec = pkgs.formats.elixirConf {}; akkconfig = ec.generate "config.exs" (with ec.lib; { ":pleroma" = { "Pleroma.Upload" = { uploader = mkRaw "Pleroma.Uploaders.S3"; filters = map (v: mkRaw ("Pleroma.Upload.Filter." + v)) ["Mogrify" "Dedupe" "AnonymizeFilename"]; base_url = "https://mastodon-assets.chir.rs/"; }; "Pleroma.Uploaders.S3" = { bucket = "mastodon-chir-rs"; truncated_namespace = ""; }; "Pleroma.Upload.Filter.Mogrify" = { args = ["auto-orient" "strip"]; }; ":instance" = { name = "Raccoon Noises"; email = "lotte@chir.rs"; description = "Single User Akkoma Instance"; limit = 58913; description_limit = 58913; upload_limit = 134217728; languages = ["en" "tok"]; registrations_open = false; invites_enabled = true; static_dir = "${static_dir}"; max_pinned_statuses = 10; attachment_links = true; max_report_comment_size = 58913; safe_dm_mentions = true; healthcheck = true; user_bio_length = 58913; user_name_length = 621; max_account_fields = 69; max_remote_account_fields = 621; account_field_name_length = 621; account_field_value_length = 58913; registration_reason_length = 621; external_user_synchronization = true; cleanup_attachments = true; }; ":markup" = { allow_headings = true; allow_tables = true; allow_fonts = true; }; ":frontend_configurations" = { pleroma_fe = mkMap { webPushNotifications = true; }; }; ":activitypub" = { unfollow_blocked = false; outgoing_blocks = false; blockers_visible = false; deny_follow_blocked = true; sign_object_fetches = true; authorized_fetch_mode = true; }; ":mrf_simple" = let processMap = m: map (k: mkTuple [k m.${k}]) (builtins.attrNames m); in { reject = processMap { "qoto.org" = "Freeze Peach; Admin harasses other server admins; sends unsolicited emails"; "poa.st" = "Hosting neonazis"; "kiwifarms.cc" = "Targeted Harassment"; "pmth.us" = "Harassment"; "nicecrew.digital" = "TERF Instance"; "freespeechextremist.com" = "Freeze Peach"; "ryona.agency" = "Freeze Peach"; "howlr.me" = "Run by verified kiwifarms user"; "rdrama.cc" = "smells like Kiwifarms shit"; "xhais.love" = "Zoophile instance"; "beefyboys.win" = "freeze peach; hosts neonazis"; "bae.st" = "freeze peach"; "vivaldi.net" = "Corporate instance; Registers nonconsensual accounts for Vivaldi Sync users"; }; federated_timeline_removal = processMap { "mastodon.social" = "Too large to be moderated well"; "mastodon.online" = "Too large to be moderated well"; "tumblr.com" = "Too large to be moderated well, corporate instance"; }; }; ":mrf" = { policies = map (v: mkRaw ("Pleroma.Web.ActivityPub.MRF." + v)) ["SimplePolicy" "EnsureRePrepended" "MediaProxyWarmingPolicy" "ForceBotUnlistedPolicy" "AntiFollowbotPolicy" "ObjectAgePolicy" "TagPolicy" "RequireImageDescription"]; transparency = true; }; ":http_security" = { enabled = true; sts = true; referrer_policy = "no-referrer"; }; ":frontends" = { primary = mkMap { name = "pleroma-fe"; ref = "stable"; }; admin = mkMap { name = "admin-fe"; ref = "stable"; }; }; ":media_proxy" = { enabled = true; base_url = "https://mediaproxy.chir.rs"; proxy_opts = { redirect_on_failure = true; }; invalidation = { enabled = true; provider = mkRaw "Pleroma.Web.MediaProxy.Invalidation.Script"; }; }; "Pleroma.Web.MediaProxy.Invalidation.Script" = { script_path = "${purge_url_script}"; }; "Pleroma.Repo" = { adapter = mkRaw "Ecto.Adapters.Postgres"; database = "akkoma"; pool_size = 10; socket_dir = "/run/postgresql"; prepare = mkAtom ":named"; parameters.plan_cache_mode = "force_custom_plan"; }; "Pleroma.Web.Endpoint" = { url = { host = "akko.chir.rs"; port = 443; scheme = "https"; }; secure_cookie_flag = true; }; "Pleroma.Emails.Mailer" = { enabled = true; adapter = mkRaw "Swoosh.Adapters.SMTP"; relay = "mail.chir.rs"; username = "akko@chir.rs"; port = "465"; ssl = true; auth = mkAtom ":always"; }; "Pleroma.Emails.NewUsersDigestEmail" = { enabled = true; }; ":database".rum_enabled = true; ":emoji" = { shortcode_globs = ["/emoji/**/*.png"]; groups = { "BlobfoxFlip" = "/emoji/volpeon-blobfox-flip/*.png"; "Blobfox" = "/emoji/volpeon-blobfox/*.png"; "BunhdFlip" = "/emoji/volpeon-bunhd-flip/*.png"; "Bunhd" = "/emoji/volpeon-bunhd/*.png"; "Drgn" = "/emoji/volpeon-drgn/*.png"; "Fox" = "/emoji/volpeon-fox/*.png"; "Raccoon" = "/emoji/volpeon-raccoon/*.png"; "Vlpn" = "/emoji/volpeon-vlpn/*.png"; "Lotte" = "/emoji/lotte/*.png"; "Caroline" = "/emoji/caro/*.png"; "Misc" = "/emoji/misc/*.png"; }; }; }; ":web_push_encryption".":vapid_details".subject = "lotte@chir.rs"; }); in { imports = [ ./mediaproxy.nix ]; services.pleroma = { enable = true; package = nix-packages.packages.${pkgs.system}.akkoma; configs = [(builtins.readFile akkconfig)]; user = "akkoma"; group = "akkoma"; secretConfigFile = config.sops.secrets."services/akkoma.exs".path; }; systemd.services.pleroma.path = with pkgs; [exiftool nixpkgs-libjxl-update.legacyPackages.${system}.imagemagick ffmpeg]; services.postgresql.ensureDatabases = ["akkoma"]; services.postgresql.ensureUsers = [ { name = "akkoma"; ensurePermissions = {"DATABASE akkoma" = "ALL PRIVILEGES";}; } ]; sops.secrets."services/akkoma.exs" = {owner = "akkoma";}; sops.secrets."services/bunny-key".owner = "akkoma"; services.caddy.virtualHosts."akko.chir.rs" = { useACMEHost = "chir.rs"; extraConfig = '' import baseConfig handle /media_attachments/* { redir https://mastodon-assets.chir.rs{uri} permanent } handle /proxy/* { reverse_proxy { to http://127.0.0.1:24154 } } handle { reverse_proxy { to http://127.0.0.1:4000 } } ''; }; services.postgresql.extraPlugins = with pkgs.postgresql_13.pkgs; [rum]; }