Fixed cross-seed

This commit is contained in:
rasmus-kirk
2024-03-01 22:49:56 +01:00
parent d396bd3a11
commit cc45972232
13 changed files with 119 additions and 119 deletions
Generated
+3 -17
View File
@@ -90,22 +90,6 @@
"type": "github" "type": "github"
} }
}, },
"nixpkgs_2": {
"locked": {
"lastModified": 1709218635,
"narHash": "sha256-nytX/MkfqeTD4z7bMq4QRXcHxO9B3vRo9tM6fMtPFA8=",
"owner": "NixOS",
"repo": "nixpkgs",
"rev": "068d4db604958d05d0b46c47f79b507d84dbc069",
"type": "github"
},
"original": {
"owner": "NixOS",
"ref": "nixos-23.11",
"repo": "nixpkgs",
"type": "github"
}
},
"root": { "root": {
"inputs": { "inputs": {
"devshell": "devshell", "devshell": "devshell",
@@ -153,7 +137,9 @@
}, },
"vpnconfinement": { "vpnconfinement": {
"inputs": { "inputs": {
"nixpkgs": "nixpkgs_2" "nixpkgs": [
"nixpkgs"
]
}, },
"locked": { "locked": {
"lastModified": 1709159289, "lastModified": 1709159289,
+1 -4
View File
@@ -37,7 +37,6 @@
flake-root.flakeModule flake-root.flakeModule
treefmt-nix.flakeModule treefmt-nix.flakeModule
devshell.flakeModule devshell.flakeModule
#vpnconfinement.nixosModules.default
]; ];
systems = [ systems = [
"x86_64-linux" "x86_64-linux"
@@ -45,10 +44,7 @@
flake = { flake = {
nixosModules = rec { nixosModules = rec {
#vpnconfinement = vpnconfinement.nixosModules.default;
nixarr = (import ./nixarr vpnconfinement); nixarr = (import ./nixarr vpnconfinement);
#imports = [ vpnconfinement.nixosModules.default ];
#nixarr.imports = [ vpnconfinement ];
default = nixarr; default = nixarr;
}; };
}; };
@@ -56,6 +52,7 @@
perSystem = { perSystem = {
config, config,
pkgs, pkgs,
lib,
... ...
}: { }: {
treefmt.config = { treefmt.config = {
+1 -1
View File
@@ -10,7 +10,7 @@
{ {
config._module.check = false; config._module.check = false;
} }
./nixarr ./nixarr/nixarr.nix
]; ];
}; };
optionsDocNixos = nixosOptionsDoc { optionsDocNixos = nixosOptionsDoc {
+4 -3
View File
@@ -1,4 +1,5 @@
vpnconfinement: { vpnconfinement:
{
config, config,
lib, lib,
pkgs, pkgs,
@@ -66,7 +67,7 @@ in {
stateDir = mkOption { stateDir = mkOption {
type = types.path; type = types.path;
default = "/data/.state"; default = "/data/.state/nixarr";
description = '' description = ''
The location of the state directory for the services. The location of the state directory for the services.
''; '';
@@ -223,7 +224,7 @@ in {
./dnsleaktest.sh ./dnsleaktest.sh
'' + (if cfg.vpn.vpnTestService.port != null then '' '' + (if cfg.vpn.vpnTestService.port != null then ''
echo "starting netcat on port ${builtins.toString cfg.vpn.vpnTestService.port}:" echo "starting netcat on port ${builtins.toString cfg.vpn.vpnTestService.port}:"
nc -vnlp ${builtins.toString cfg.vpn.vpnTestService.port} nc -vnlpu ${builtins.toString cfg.vpn.vpnTestService.port}
'' else ""); '' else "");
}; };
in "${vpn-test}/bin/vpn-test"; in "${vpn-test}/bin/vpn-test";
+1 -1
View File
@@ -13,7 +13,7 @@ in with lib; {
stateDir = mkOption { stateDir = mkOption {
type = types.path; type = types.path;
default = "${nixarr.stateDir}/nixarr/jellyfin"; default = "${nixarr.stateDir}/jellyfin";
description = "The state directory for Jellyfin."; description = "The state directory for Jellyfin.";
}; };
+1 -1
View File
@@ -12,7 +12,7 @@ in {
stateDir = mkOption { stateDir = mkOption {
type = types.path; type = types.path;
default = "${nixarr.stateDir}/nixarr/lidarr"; default = "${nixarr.stateDir}/lidarr";
description = "The state directory for Lidarr"; description = "The state directory for Lidarr";
}; };
+1 -3
View File
@@ -18,7 +18,7 @@ in {
stateDir = mkOption { stateDir = mkOption {
type = types.path; type = types.path;
default = "${nixarr.stateDir}/nixarr/prowlarr"; default = "${nixarr.stateDir}/prowlarr";
description = "The state directory for Prowlarr."; description = "The state directory for Prowlarr.";
}; };
@@ -48,8 +48,6 @@ in {
"d '${cfg.stateDir}' 0700 prowlarr root - -" "d '${cfg.stateDir}' 0700 prowlarr root - -"
]; ];
users.groups.prowlarr = {};
util-nixarr.services.prowlarr = { util-nixarr.services.prowlarr = {
enable = true; enable = true;
dataDir = cfg.stateDir; dataDir = cfg.stateDir;
+3 -1
View File
@@ -70,6 +70,8 @@ in {
}; };
}; };
users.groups = mkIf (cfg.group == "prowlarr") {}; users.groups = mkIf (cfg.group == "prowlarr") {
prowlarr = { };
};
}; };
} }
+1 -1
View File
@@ -14,7 +14,7 @@ in {
stateDir = mkOption { stateDir = mkOption {
type = types.path; type = types.path;
default = "${nixarr.stateDir}/nixarr/radarr"; default = "${nixarr.stateDir}/radarr";
description = "The state directory for radarr."; description = "The state directory for radarr.";
}; };
+1 -1
View File
@@ -12,7 +12,7 @@ in {
stateDir = mkOption { stateDir = mkOption {
type = types.path; type = types.path;
default = "${nixarr.stateDir}/nixarr/readarr"; default = "${nixarr.stateDir}/readarr";
description = "The state directory for Readarr"; description = "The state directory for Readarr";
}; };
+51 -5
View File
@@ -8,7 +8,37 @@ with lib; let
cfg = config.util-nixarr.services.cross-seed; cfg = config.util-nixarr.services.cross-seed;
settingsFormat = pkgs.formats.json {}; settingsFormat = pkgs.formats.json {};
settingsFile = settingsFormat.generate "settings.json" cfg.settings; settingsFile = settingsFormat.generate "settings.json" cfg.settings;
cross-seedPkg = import ../../../pkgs/cross-seed { inherit (pkgs) stdenv lib fetchFromGitHub; }; cross-seedPkg = pkgs.callPackage ../../../pkgs/cross-seed {};
configJs = pkgs.writeText "config.js" ''
// Loads a json.config
"use strict";
const fs = require('fs');
const jsonPath = '${cfg.dataDir}/config.json'
// Synchronously read the JSON-configuration file
const configFileContent = fs.readFileSync(jsonPath, { encoding: 'utf8' });
// Parse the JSON content into a JavaScript object
let config = JSON.parse(configFileContent);
// Function to recursively replace null values with undefined
/*
function replaceNullWithUndefined(obj) {
Object.keys(obj).forEach(key => {
if (obj[key] === null) {
obj[key] = undefined;
} else if (typeof obj[key] === 'object') {
replaceNullWithUndefined(obj[key]);
}
});
}
replaceNullWithUndefined(config);
*/
// Export the configuration object
module.exports = config;
'';
in { in {
options = { options = {
util-nixarr.services.cross-seed = { util-nixarr.services.cross-seed = {
@@ -52,10 +82,24 @@ in {
}; };
config = mkIf cfg.enable { config = mkIf cfg.enable {
systemd.tmpfiles.rules = [ assertions = [
"d '${cfg.dataDir}' 0700 ${cfg.user} ${cfg.group} - -" {
assertion = cfg.enable -> cfg.settings.outputDir != null;
message = ''
The settings.outputDir must be set if cross-seed is enabled.
'';
}
]; ];
systemd.tmpfiles.rules = [
"L+ '${cfg.dataDir}'/config.js - - - - ${configJs}"
"d '${cfg.dataDir}' 0700 ${cfg.user} ${cfg.group} - -"
] ++ (
if cfg.settings.outputDir != null then
[ "d '${cfg.settings.outputDir}' 0755 ${cfg.user} ${cfg.group} - -" ]
else []
);
systemd.services.cross-seed = { systemd.services.cross-seed = {
description = "cross-seed"; description = "cross-seed";
after = ["network.target"]; after = ["network.target"];
@@ -73,7 +117,7 @@ in {
Type = "simple"; Type = "simple";
User = cfg.user; User = cfg.user;
Group = cfg.group; Group = cfg.group;
ExecStart = "${getExe cross-seedPkg} daemon"; ExecStart = "${cross-seedPkg}/bin/cross-seed daemon";
Restart = "on-failure"; Restart = "on-failure";
}; };
}; };
@@ -85,6 +129,8 @@ in {
}; };
}; };
users.groups = mkIf (cfg.group == "cross-seed") {}; users.groups = mkIf (cfg.group == "cross-seed") {
cross-seed = { };
};
}; };
} }
+36 -13
View File
@@ -9,12 +9,25 @@ with lib; let
nixarr = config.nixarr; nixarr = config.nixarr;
cfg-cross-seed = config.nixarr.transmission.privateTrackers.cross-seed; cfg-cross-seed = config.nixarr.transmission.privateTrackers.cross-seed;
transmissionCrossSeedScript = with builtins; pkgs.writeShellApplication { transmissionCrossSeedScript = with builtins; pkgs.writeShellApplication {
name = "mk-cross-seed-credentials"; name = "transmission-cross-seed-script";
runtimeInputs = with pkgs; [ curl ]; runtimeInputs = with pkgs; [ curl ];
text = '' text = ''
curl -XPOST http://localhost:2468/api/webhook?apikey=YOUR_API_KEY --data-urlencode "infoHash=$TR_TORRENT_HASH" PROWLARR_API_KEY=$(cat prowlarr-api-key)
curl -XPOST http://localhost:2468/api/webhook?apikey="$PROWLARR_API_KEY" --data-urlencode "infoHash=$TR_TORRENT_HASH"
'';
};
importProwlarrApi = with builtins; pkgs.writeShellApplication {
name = "import-prowlarr-api";
runtimeInputs = with pkgs; [ yq ];
text = ''
touch ${cfg.stateDir}/prowlarr-api-key
chmod 400 ${cfg.stateDir}/prowlarr-api-key
chown torrenter ${cfg.stateDir}/prowlarr-api-key
xq -r '.Config.ApiKey' "${nixarr.prowlarr.stateDir}/config.xml" > "${cfg.stateDir}/prowlarr-api-key"
''; '';
}; };
mkCrossSeedCredentials = with builtins; pkgs.writeShellApplication { mkCrossSeedCredentials = with builtins; pkgs.writeShellApplication {
@@ -24,14 +37,17 @@ with lib; let
text = text =
"INDEX_LINKS=(" "INDEX_LINKS=("
+ strings.concatMapStringsSep " " toString cfg.privateTrackers.cross-seed.indexIds + (strings.concatMapStringsSep " " toString cfg.privateTrackers.cross-seed.indexIds)
+ ")" + ")"
'' + "\n"
+ ''
TMP_JSON=$(mktemp) TMP_JSON=$(mktemp)
CRED_FILE="/run/secrets/cross-seed/credentialsFile.json" CRED_FILE="/run/secrets/cross-seed/credentialsFile.json"
PROWLARR_API_KEY=$(xq '.Config.ApiKey' "${nixarr.prowlarr.stateDir}/config.xml") PROWLARR_API_KEY=$(xq -r '.Config.ApiKey' "${nixarr.prowlarr.stateDir}/config.xml")
CRED_DIR=$(dirname "$filePath") # shellcheck disable=SC2034
CRED_DIR=$(dirname "$CRED_FILE")
mkdir -p "$CRED_DIR"
echo '{}' > "$CRED_FILE" echo '{}' > "$CRED_FILE"
chmod 400 "$CRED_FILE" chmod 400 "$CRED_FILE"
chown "${config.util-nixarr.services.cross-seed.user}" "$CRED_FILE" chown "${config.util-nixarr.services.cross-seed.user}" "$CRED_FILE"
@@ -49,7 +65,7 @@ in {
stateDir = mkOption { stateDir = mkOption {
type = types.path; type = types.path;
default = "${nixarr.stateDir}/nixarr/transmission"; default = "${nixarr.stateDir}/transmission";
description = '' description = ''
The state directory for Transmission. The state directory for Transmission.
''; '';
@@ -103,7 +119,7 @@ in {
stateDir = mkOption { stateDir = mkOption {
type = types.path; type = types.path;
default = "${nixarr.stateDir}/nixarr/cross-seed"; default = "${nixarr.stateDir}/cross-seed";
description = '' description = ''
The state directory for Transmission. The state directory for Transmission.
''; '';
@@ -212,8 +228,8 @@ in {
#group = "media"; #group = "media";
settings = { settings = {
torrentDir = "${nixarr.mediaDir}/torrents"; torrentDir = "${nixarr.mediaDir}/torrents";
outputDir = "${nixarr.mediaDir}/torrents/cross-seed"; outputDir = "${nixarr.mediaDir}/torrents/.cross-seed";
transmissionRpcUrl = "http://transmission:${builtins.toString cfg.uiPort}/transmission/rpc"; transmissionRpcUrl = "http://localhost:${builtins.toString cfg.uiPort}/transmission/rpc";
rssCadence = "20 minutes"; rssCadence = "20 minutes";
# Enable infrequent periodic searches # Enable infrequent periodic searches
@@ -224,11 +240,16 @@ in {
}; };
# Run as root in case that the cfg.credentialsFile is not readable by cross-seed # Run as root in case that the cfg.credentialsFile is not readable by cross-seed
systemd.services.cross-seed.serviceConfig = mkIf cfg-cross-seed.enable { systemd.services.cross-seed.serviceConfig = mkIf cfg-cross-seed.enable {
ExecStartPre = [(mkBefore ExecStartPre = mkBefore [(
("+" + (getExe mkCrossSeedCredentials)) "+" + "${mkCrossSeedCredentials}/bin/mk-cross-seed-credentials"
)]; )];
}; };
systemd.services.transmission.serviceConfig = mkIf cfg-cross-seed.enable {
ExecStartPre = mkBefore [(
"+" + "${importProwlarrApi}/bin/import-prowlarr-api"
)];
};
services.transmission = { services.transmission = {
enable = true; enable = true;
user = "torrenter"; user = "torrenter";
@@ -270,7 +291,9 @@ in {
anti-brute-force-threshold = 10; anti-brute-force-threshold = 10;
script-torrent-done-enabled = cfg-cross-seed.enable; script-torrent-done-enabled = cfg-cross-seed.enable;
script-torrent-done-filename = if cfg-cross-seed.enable then transmissionCrossSeedScript else null; script-torrent-done-filename = if cfg-cross-seed.enable then
"${transmissionCrossSeedScript}/bin/transmission-cross-seed-script"
else null;
message-level = message-level =
if cfg.messageLevel == "none" if cfg.messageLevel == "none"
+15 -68
View File
@@ -1,75 +1,22 @@
{ { lib, buildNpmPackage, fetchFromGitHub }:
config,
pkgs,
lib,
...
}:
with lib; let
cfg = config.util-nixarr.services.prowlarr;
settingsFormat = pkgs.formats.json {};
settingsFile = settingsFormat.generate "settings.json" cfg.settings;
cross-seedPkg = import ../../../pkgs/cross-seed { inherit (pkgs) stdenv lib fetchFromGitHub; };
in {
options = {
util-nixarr.services.prowlarr = {
enable = mkEnableOption "cross-seed";
configFile = mkOption { buildNpmPackage rec {
type = with types; nullOr path; pname = "cross-seed";
default = null; version = "5.9.2";
example = "/var/lib/secrets/cross-seed/settings.json";
description = "";
};
dataDir = mkOption { src = fetchFromGitHub {
type = types.path; owner = "cross-seed";
default = "/var/lib/cross-seed"; repo = pname;
}; rev = "v${version}";
hash = "sha256-E0AlsFV9RP01YVwjw6ZQ8Lf1IVyuudxrb5oJ61EfIyo=";
user = mkOption {
type = types.str;
default = "cross-seed";
description = "User account under which cross-seed runs.";
};
group = mkOption {
type = types.str;
default = "cross-seed";
description = "Group under which cross-seed runs.";
};
};
}; };
config = mkIf cfg.enable { npmDepsHash = "sha256-hZKLv+bzRFiMjNemydCUC1d7xul7Mm+vOPtCUD7p9XQ=";
systemd.tmpfiles.rules = [
"d '${cfg.dataDir}' 0700 ${cfg.user} ${cfg.group} - -"
];
systemd.services.prowlarr = { meta = with lib; {
description = "cross-seed"; description = "cross-seed is an app designed to help you download torrents that you can cross seed based on your existing torrents";
after = ["network.target"]; homepage = "https://www.cross-seed.org";
wantedBy = ["multi-user.target"]; license = licenses.asl20;
maintainers = with maintainers; [ rasmus-kirk ];
environment.CONFIG_DIR = cfg.dataDir;
serviceConfig = {
ExecStartPre = [("+" + pkgs.writeShellScript "transmission-prestart" ''
mv ${cfg.configFile} ${cfg.dataDir}
'')];
Type = "simple";
User = cfg.user;
Group = cfg.group;
ExecStart = "${getExe cross-seedPkg} daemon";
Restart = "on-failure";
};
};
users.users = mkIf (cfg.user == "cross-seed") {
cross-seed = {
group = cfg.group;
};
};
users.groups = mkIf (cfg.group == "cross-seed") {};
}; };
} }