{ config, lib, options, pkgs, ... }: with lib; let cfg = config.services.terraria; opt = options.services.terraria; worldSizeMap = { small = 1; medium = 2; large = 3; }; valFlag = name: val: optionalString (val != null) "-${name} \"${escape ["\\" "\""] (toString val)}\""; boolFlag = name: val: optionalString val "-${name}"; flags = [ (valFlag "port" cfg.port) (valFlag "maxPlayers" cfg.maxPlayers) (valFlag "password" cfg.password) (valFlag "motd" cfg.messageOfTheDay) (valFlag "world" cfg.worldPath) (valFlag "autocreate" (builtins.getAttr cfg.autoCreatedWorldSize worldSizeMap)) (valFlag "banlist" cfg.banListPath) (boolFlag "secure" cfg.secure) (boolFlag "noupnp" cfg.noUPnP) ]; tmuxCmd = "${lib.getExe pkgs.tmux} -S ${lib.escapeShellArg cfg.dataDir}/terraria.sock"; stopScript = pkgs.writeShellScript "terraria-stop" '' if ! [ -d "/proc/$1" ]; then exit 0 fi lastline=$(${tmuxCmd} capture-pane -p | grep . | tail -n1) # If the service is not configured to auto-start a world, it will show the world selection prompt # If the last non-empty line on-screen starts with "Choose World", we know the prompt is open if [[ "$lastline" =~ ^'Choose World' ]]; then # In this case, nothing needs to be saved, so we can kill the process ${tmuxCmd} kill-session else # Otherwise, we send the `exit` command ${tmuxCmd} send-keys Enter exit Enter fi # Wait for the process to stop tail --pid="$1" -f /dev/null ''; in { options = { services.terraria = { enable = mkOption { type = types.bool; default = false; description = '' If enabled, starts a Terraria server. The server can be connected to via `tmux -S ''${config.${opt.dataDir}}/terraria.sock attach` for administration by users who are a part of the `terraria` group (use `C-b d` shortcut to detach again). ''; }; port = mkOption { type = types.port; default = 7777; description = '' Specifies the port to listen on. ''; }; maxPlayers = mkOption { type = types.ints.u8; default = 255; description = '' Sets the max number of players (between 1 and 255). ''; }; password = mkOption { type = types.nullOr types.str; default = null; description = '' Sets the server password. Leave `null` for no password. ''; }; messageOfTheDay = mkOption { type = types.nullOr types.str; default = null; description = '' Set the server message of the day text. ''; }; worldPath = mkOption { type = types.nullOr types.path; default = null; description = '' The path to the world file (`.wld`) which should be loaded. If no world exists at this path, one will be created with the size specified by `autoCreatedWorldSize`. ''; }; autoCreatedWorldSize = mkOption { type = types.enum [ "small" "medium" "large" ]; default = "medium"; description = '' Specifies the size of the auto-created world if `worldPath` does not point to an existing world. ''; }; banListPath = mkOption { type = types.nullOr types.path; default = null; description = '' The path to the ban list. ''; }; secure = mkOption { type = types.bool; default = false; description = "Adds additional cheat protection to the server."; }; noUPnP = mkOption { type = types.bool; default = false; description = "Disables automatic Universal Plug and Play."; }; openFirewall = mkOption { type = types.bool; default = false; description = "Whether to open ports in the firewall"; }; dataDir = mkOption { type = types.str; default = "/var/lib/terraria"; example = "/srv/terraria"; description = "Path to variable state data directory for terraria."; }; }; }; config = mkIf cfg.enable { users.users.terraria = { description = "Terraria server service user"; group = "terraria"; home = cfg.dataDir; createHome = true; uid = config.ids.uids.terraria; }; users.groups.terraria = { gid = config.ids.gids.terraria; }; systemd.services.terraria = { description = "Terraria Server Service"; wantedBy = [ "multi-user.target" ]; after = [ "network.target" ]; serviceConfig = { User = "terraria"; Group = "terraria"; Type = "forking"; GuessMainPID = true; UMask = 007; ExecStart = "${tmuxCmd} new -d ${pkgs.terraria-server}/bin/TerrariaServer ${concatStringsSep " " flags}"; ExecStop = "${stopScript} $MAINPID"; }; }; networking.firewall = mkIf cfg.openFirewall { allowedTCPPorts = [ cfg.port ]; allowedUDPPorts = [ cfg.port ]; }; }; }