2020-04-24 23:36:52 +00:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
let
|
|
|
|
cfg = config.services.factorio;
|
|
|
|
name = "Factorio";
|
|
|
|
stateDir = "/var/lib/${cfg.stateDirName}";
|
|
|
|
mkSavePath = name: "${stateDir}/saves/${name}.zip";
|
|
|
|
configFile = pkgs.writeText "factorio.conf" ''
|
|
|
|
use-system-read-write-data-directories=true
|
|
|
|
[path]
|
|
|
|
read-data=${cfg.package}/share/factorio/data
|
|
|
|
write-data=${stateDir}
|
|
|
|
'';
|
|
|
|
serverSettings = {
|
|
|
|
name = cfg.game-name;
|
|
|
|
description = cfg.description;
|
|
|
|
visibility = {
|
|
|
|
public = cfg.public;
|
|
|
|
lan = cfg.lan;
|
|
|
|
};
|
|
|
|
username = cfg.username;
|
|
|
|
password = cfg.password;
|
|
|
|
token = cfg.token;
|
|
|
|
game_password = cfg.game-password;
|
|
|
|
require_user_verification = cfg.requireUserVerification;
|
|
|
|
max_upload_in_kilobytes_per_second = 0;
|
|
|
|
minimum_latency_in_ticks = 0;
|
|
|
|
ignore_player_limit_for_returning_players = false;
|
|
|
|
allow_commands = "admins-only";
|
|
|
|
autosave_interval = cfg.autosave-interval;
|
|
|
|
autosave_slots = 5;
|
|
|
|
afk_autokick_interval = 0;
|
|
|
|
auto_pause = true;
|
|
|
|
only_admins_can_pause_the_game = true;
|
|
|
|
autosave_only_on_server = true;
|
2021-03-15 08:37:03 +00:00
|
|
|
non_blocking_saving = cfg.nonBlockingSaving;
|
2020-04-24 23:36:52 +00:00
|
|
|
} // cfg.extraSettings;
|
2024-09-19 14:19:46 +00:00
|
|
|
serverSettingsString = builtins.toJSON (lib.filterAttrsRecursive (n: v: v != null) serverSettings);
|
2024-01-02 11:29:13 +00:00
|
|
|
serverSettingsFile = pkgs.writeText "server-settings.json" serverSettingsString;
|
2024-10-29 11:11:06 +00:00
|
|
|
playerListOption = name: list:
|
|
|
|
lib.optionalString (list != [])
|
|
|
|
"--${name}=${pkgs.writeText "${name}.json" (builtins.toJSON list)}";
|
2022-12-17 10:02:37 +00:00
|
|
|
modDir = pkgs.factorio-utils.mkModDirDrv cfg.mods cfg.mods-dat;
|
2020-04-24 23:36:52 +00:00
|
|
|
in
|
|
|
|
{
|
|
|
|
options = {
|
|
|
|
services.factorio = {
|
2024-09-19 14:19:46 +00:00
|
|
|
enable = lib.mkEnableOption name;
|
|
|
|
port = lib.mkOption {
|
|
|
|
type = lib.types.port;
|
2020-04-24 23:36:52 +00:00
|
|
|
default = 34197;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
The port to which the service should bind.
|
2020-11-30 08:33:03 +00:00
|
|
|
'';
|
|
|
|
};
|
2021-05-20 23:08:51 +00:00
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
bind = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2022-03-30 09:31:56 +00:00
|
|
|
default = "0.0.0.0";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2022-03-30 09:31:56 +00:00
|
|
|
The address to which the service should bind.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2024-10-29 11:11:06 +00:00
|
|
|
allowedPlayers = lib.mkOption {
|
|
|
|
# I would personally prefer for `allowedPlayers = []` to mean "no-one
|
|
|
|
# can connect" but Factorio seems to ignore empty whitelists (even with
|
|
|
|
# --use-server-whitelist) so we can't implement that behaviour, so we
|
|
|
|
# might as well match theirs.
|
|
|
|
type = lib.types.listOf lib.types.str;
|
|
|
|
default = [];
|
|
|
|
example = [ "Rseding91" "Oxyd" ];
|
|
|
|
description = ''
|
|
|
|
If non-empty, only these player names are allowed to connect. The game
|
|
|
|
will not be able to save any changes made in-game with the /whitelist
|
|
|
|
console command, though they will still take effect until the server
|
|
|
|
is restarted.
|
|
|
|
|
|
|
|
If empty, the whitelist defaults to open, but can be managed with the
|
|
|
|
in-game /whitelist console command (see: /help whitelist), which will
|
|
|
|
cause changes to be saved to the game's state directory (see also:
|
|
|
|
`stateDirName`).
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
# Opting not to include the banlist in addition the the whitelist because:
|
|
|
|
# - banlists are not as often known in advance,
|
|
|
|
# - losing banlist changes on restart seems much more of a headache.
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
admins = lib.mkOption {
|
|
|
|
type = lib.types.listOf lib.types.str;
|
2021-05-20 23:08:51 +00:00
|
|
|
default = [];
|
|
|
|
example = [ "username" ];
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2021-05-20 23:08:51 +00:00
|
|
|
List of player names which will be admin.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
openFirewall = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
2020-11-30 08:33:03 +00:00
|
|
|
default = false;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-11-30 08:33:03 +00:00
|
|
|
Whether to automatically open the specified UDP port in the firewall.
|
2020-04-24 23:36:52 +00:00
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
saveName = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2020-04-24 23:36:52 +00:00
|
|
|
default = "default";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
The name of the savegame that will be used by the server.
|
|
|
|
|
2021-12-06 16:07:01 +00:00
|
|
|
When not present in /var/lib/''${config.services.factorio.stateDirName}/saves,
|
|
|
|
a new map with default settings will be generated before starting the service.
|
2020-04-24 23:36:52 +00:00
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
loadLatestSave = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
2022-05-18 14:49:53 +00:00
|
|
|
default = false;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2022-05-18 14:49:53 +00:00
|
|
|
Load the latest savegame on startup. This overrides saveName, in that the latest
|
|
|
|
save will always be used even if a saved game of the given name exists. It still
|
|
|
|
controls the 'canonical' name of the savegame.
|
|
|
|
|
|
|
|
Set this to true to have the server automatically reload a recent autosave after
|
|
|
|
a crash or desync.
|
|
|
|
'';
|
|
|
|
};
|
2020-04-24 23:36:52 +00:00
|
|
|
# TODO Add more individual settings as nixos-options?
|
|
|
|
# TODO XXX The server tries to copy a newly created config file over the old one
|
|
|
|
# on shutdown, but fails, because it's in the nix store. When is this needed?
|
|
|
|
# Can an admin set options in-game and expect to have them persisted?
|
2024-09-19 14:19:46 +00:00
|
|
|
configFile = lib.mkOption {
|
|
|
|
type = lib.types.path;
|
2020-04-24 23:36:52 +00:00
|
|
|
default = configFile;
|
2024-09-19 14:19:46 +00:00
|
|
|
defaultText = lib.literalExpression "configFile";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
The server's configuration file.
|
|
|
|
|
|
|
|
The default file generated by this module contains lines essential to
|
|
|
|
the server's operation. Use its contents as a basis for any
|
|
|
|
customizations.
|
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
extraSettingsFile = lib.mkOption {
|
|
|
|
type = lib.types.nullOr lib.types.path;
|
2024-01-02 11:29:13 +00:00
|
|
|
default = null;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2024-01-02 11:29:13 +00:00
|
|
|
File, which is dynamically applied to server-settings.json before
|
|
|
|
startup.
|
|
|
|
|
|
|
|
This option should be used for credentials.
|
|
|
|
|
|
|
|
For example a settings file could contain:
|
|
|
|
```json
|
|
|
|
{
|
|
|
|
"game-password": "hunter1"
|
|
|
|
}
|
|
|
|
```
|
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
stateDirName = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2020-04-24 23:36:52 +00:00
|
|
|
default = "factorio";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Name of the directory under /var/lib holding the server's data.
|
|
|
|
|
|
|
|
The configuration and map will be stored here.
|
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
mods = lib.mkOption {
|
|
|
|
type = lib.types.listOf lib.types.package;
|
2020-04-24 23:36:52 +00:00
|
|
|
default = [];
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Mods the server should install and activate.
|
|
|
|
|
|
|
|
The derivations in this list must "build" the mod by simply copying
|
|
|
|
the .zip, named correctly, into the output directory. Eventually,
|
|
|
|
there will be a way to pull in the most up-to-date list of
|
|
|
|
derivations via nixos-channel. Until then, this is for experts only.
|
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
mods-dat = lib.mkOption {
|
|
|
|
type = lib.types.nullOr lib.types.path;
|
2022-12-17 10:02:37 +00:00
|
|
|
default = null;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2022-12-17 10:02:37 +00:00
|
|
|
Mods settings can be changed by specifying a dat file, in the [mod
|
|
|
|
settings file
|
|
|
|
format](https://wiki.factorio.com/Mod_settings_file_format).
|
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
game-name = lib.mkOption {
|
|
|
|
type = lib.types.nullOr lib.types.str;
|
2020-04-24 23:36:52 +00:00
|
|
|
default = "Factorio Game";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Name of the game as it will appear in the game listing.
|
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
description = lib.mkOption {
|
|
|
|
type = lib.types.nullOr lib.types.str;
|
2020-04-24 23:36:52 +00:00
|
|
|
default = "";
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Description of the game that will appear in the listing.
|
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
extraSettings = lib.mkOption {
|
|
|
|
type = lib.types.attrs;
|
2020-04-24 23:36:52 +00:00
|
|
|
default = {};
|
2024-10-29 11:11:06 +00:00
|
|
|
example = { max_players = 64; };
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Extra game configuration that will go into server-settings.json
|
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
public = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
2020-04-24 23:36:52 +00:00
|
|
|
default = false;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Game will be published on the official Factorio matching server.
|
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
lan = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
2020-04-24 23:36:52 +00:00
|
|
|
default = false;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Game will be broadcast on LAN.
|
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
username = lib.mkOption {
|
|
|
|
type = lib.types.nullOr lib.types.str;
|
2020-04-24 23:36:52 +00:00
|
|
|
default = null;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Your factorio.com login credentials. Required for games with visibility public.
|
2024-01-02 11:29:13 +00:00
|
|
|
|
|
|
|
This option is insecure. Use extraSettingsFile instead.
|
2020-04-24 23:36:52 +00:00
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
package = lib.mkPackageOption pkgs "factorio-headless" {
|
2024-01-02 11:29:13 +00:00
|
|
|
example = "factorio-headless-experimental";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
password = lib.mkOption {
|
|
|
|
type = lib.types.nullOr lib.types.str;
|
2020-04-24 23:36:52 +00:00
|
|
|
default = null;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Your factorio.com login credentials. Required for games with visibility public.
|
2024-01-02 11:29:13 +00:00
|
|
|
|
|
|
|
This option is insecure. Use extraSettingsFile instead.
|
2020-04-24 23:36:52 +00:00
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
token = lib.mkOption {
|
|
|
|
type = lib.types.nullOr lib.types.str;
|
2020-04-24 23:36:52 +00:00
|
|
|
default = null;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Authentication token. May be used instead of 'password' above.
|
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
game-password = lib.mkOption {
|
|
|
|
type = lib.types.nullOr lib.types.str;
|
2020-04-24 23:36:52 +00:00
|
|
|
default = null;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Game password.
|
2024-01-02 11:29:13 +00:00
|
|
|
|
|
|
|
This option is insecure. Use extraSettingsFile instead.
|
2020-04-24 23:36:52 +00:00
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
requireUserVerification = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
2020-04-24 23:36:52 +00:00
|
|
|
default = true;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
When set to true, the server will only allow clients that have a valid factorio.com account.
|
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
autosave-interval = lib.mkOption {
|
|
|
|
type = lib.types.nullOr lib.types.int;
|
2020-04-24 23:36:52 +00:00
|
|
|
default = null;
|
|
|
|
example = 10;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Autosave interval in minutes.
|
|
|
|
'';
|
|
|
|
};
|
2024-09-19 14:19:46 +00:00
|
|
|
nonBlockingSaving = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
2021-03-15 08:37:03 +00:00
|
|
|
default = false;
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2021-03-15 08:37:03 +00:00
|
|
|
Highly experimental feature, enable only at your own risk of losing your saves.
|
|
|
|
On UNIX systems, server will fork itself to create an autosave.
|
|
|
|
Autosaving on connected Windows clients will be disabled regardless of autosave_only_on_server option.
|
|
|
|
'';
|
|
|
|
};
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
config = lib.mkIf cfg.enable {
|
2020-04-24 23:36:52 +00:00
|
|
|
systemd.services.factorio = {
|
|
|
|
description = "Factorio headless server";
|
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
after = [ "network.target" ];
|
|
|
|
|
2024-01-02 11:29:13 +00:00
|
|
|
preStart =
|
|
|
|
(toString [
|
|
|
|
"test -e ${stateDir}/saves/${cfg.saveName}.zip"
|
|
|
|
"||"
|
|
|
|
"${cfg.package}/bin/factorio"
|
2020-04-24 23:36:52 +00:00
|
|
|
"--config=${cfg.configFile}"
|
|
|
|
"--create=${mkSavePath cfg.saveName}"
|
2024-09-19 14:19:46 +00:00
|
|
|
(lib.optionalString (cfg.mods != []) "--mod-directory=${modDir}")
|
2024-01-02 11:29:13 +00:00
|
|
|
])
|
2024-09-19 14:19:46 +00:00
|
|
|
+ (lib.optionalString (cfg.extraSettingsFile != null) ("\necho ${lib.strings.escapeShellArg serverSettingsString}"
|
2024-01-02 11:29:13 +00:00
|
|
|
+ " \"$(cat ${cfg.extraSettingsFile})\" | ${lib.getExe pkgs.jq} -s add"
|
|
|
|
+ " > ${stateDir}/server-settings.json"));
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
serviceConfig = {
|
|
|
|
Restart = "always";
|
|
|
|
KillSignal = "SIGINT";
|
|
|
|
DynamicUser = true;
|
|
|
|
StateDirectory = cfg.stateDirName;
|
|
|
|
UMask = "0007";
|
|
|
|
ExecStart = toString [
|
|
|
|
"${cfg.package}/bin/factorio"
|
|
|
|
"--config=${cfg.configFile}"
|
|
|
|
"--port=${toString cfg.port}"
|
2022-03-30 09:31:56 +00:00
|
|
|
"--bind=${cfg.bind}"
|
2024-09-19 14:19:46 +00:00
|
|
|
(lib.optionalString (!cfg.loadLatestSave) "--start-server=${mkSavePath cfg.saveName}")
|
2024-01-02 11:29:13 +00:00
|
|
|
"--server-settings=${
|
|
|
|
if (cfg.extraSettingsFile != null)
|
|
|
|
then "${stateDir}/server-settings.json"
|
|
|
|
else serverSettingsFile
|
|
|
|
}"
|
2024-09-19 14:19:46 +00:00
|
|
|
(lib.optionalString cfg.loadLatestSave "--start-server-load-latest")
|
|
|
|
(lib.optionalString (cfg.mods != []) "--mod-directory=${modDir}")
|
2024-10-29 11:11:06 +00:00
|
|
|
(playerListOption "server-adminlist" cfg.admins)
|
|
|
|
(playerListOption "server-whitelist" cfg.allowedPlayers)
|
|
|
|
(lib.optionalString (cfg.allowedPlayers != []) "--use-server-whitelist")
|
2020-04-24 23:36:52 +00:00
|
|
|
];
|
|
|
|
|
|
|
|
# Sandboxing
|
|
|
|
NoNewPrivileges = true;
|
|
|
|
PrivateTmp = true;
|
|
|
|
PrivateDevices = true;
|
|
|
|
ProtectSystem = "strict";
|
|
|
|
ProtectHome = true;
|
|
|
|
ProtectControlGroups = true;
|
|
|
|
ProtectKernelModules = true;
|
|
|
|
ProtectKernelTunables = true;
|
|
|
|
RestrictAddressFamilies = [ "AF_UNIX" "AF_INET" "AF_INET6" "AF_NETLINK" ];
|
|
|
|
RestrictRealtime = true;
|
|
|
|
RestrictNamespaces = true;
|
|
|
|
MemoryDenyWriteExecute = true;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
networking.firewall.allowedUDPPorts = lib.optional cfg.openFirewall cfg.port;
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
}
|