Skip to content

Commit

Permalink
remove dataDir and add some missing files
Browse files Browse the repository at this point in the history
  • Loading branch information
coonce committed Dec 21, 2024
1 parent d4e5e68 commit 4b33626
Show file tree
Hide file tree
Showing 3 changed files with 163 additions and 10 deletions.
11 changes: 2 additions & 9 deletions nixos/modules/services/misc/jellyseerr.nix
Original file line number Diff line number Diff line change
Expand Up @@ -21,17 +21,10 @@ in
description = ''The port which the Jellyseerr web UI should listen to.'';
};

dataDir = lib.mkOption {
type = lib.types.path;
default = "/var/lib/jellyseerr";
description = "Base data directory";
};

configDir = lib.mkOption {
type = lib.types.path;
default = "${cfg.dataDir}/config";
defaultText = "\${cfg.dataDir}/config";
description = "Directory containing server configuration files.";
default = "/var/lib/jellyseerr/config";
description = "Config data directory";
};
};

Expand Down
160 changes: 160 additions & 0 deletions nixos/tests/jellyseerr.nix
Original file line number Diff line number Diff line change
@@ -0,0 +1,160 @@
import ./make-test-python.nix (
{ lib, pkgs, ... }:

{
name = "jellyseerr";
meta.maintainers = with lib.maintainers; [ minijackson ];

nodes.machine =
{ ... }:
{
services.jellyseerr.enable = true;
environment.systemPackages = with pkgs; [ ffmpeg ];
};

# Documentation of the Jellyfin API: https://api.jellyfin.org/
# Beware, this link can be resource intensive
testScript =
let
payloads = {
auth = pkgs.writeText "auth.json" (
builtins.toJSON {
Username = "jellyfin";
}
);
empty = pkgs.writeText "empty.json" (builtins.toJSON { });
};
in
''
import json
from urllib.parse import urlencode
machine.wait_for_unit("jellyseerr.service")
machine.wait_for_open_port(5055)
machine.succeed("curl --fail http://localhost:5055/")
machine.succeed("curl --fail http://localhost:5055/public/preview.jpg")
machine.wait_until_succeeds("curl --fail http://localhost:5055/health | grep Healthy")
auth_header = 'MediaBrowser Client="NixOS Integration Tests", DeviceId="1337", Device="Apple II", Version="20.09"'
def api_get(path):
return f"curl --fail 'http://localhost:8096{path}' -H 'X-Emby-Authorization:{auth_header}'"
def api_post(path, json_file=None):
if json_file:
return f"curl --fail -X post 'http://localhost:8096{path}' -d '@{json_file}' -H Content-Type:application/json -H 'X-Emby-Authorization:{auth_header}'"
else:
return f"curl --fail -X post 'http://localhost:8096{path}' -H 'X-Emby-Authorization:{auth_header}'"
with machine.nested("Wizard completes"):
machine.wait_until_succeeds(api_get("/Startup/Configuration"))
machine.succeed(api_get("/Startup/FirstUser"))
machine.succeed(api_post("/Startup/Complete"))
with machine.nested("Can login"):
auth_result_str = machine.succeed(
api_post(
"/Users/AuthenticateByName",
"${payloads.auth}",
)
)
auth_result = json.loads(auth_result_str)
auth_token = auth_result["AccessToken"]
auth_header += f", Token={auth_token}"
sessions_result_str = machine.succeed(api_get("/Sessions"))
sessions_result = json.loads(sessions_result_str)
this_session = [
session for session in sessions_result if session["DeviceId"] == "1337"
]
if len(this_session) != 1:
raise Exception("Session not created")
me_str = machine.succeed(api_get("/Users/Me"))
me = json.loads(me_str)["Id"]
with machine.nested("Can add library"):
tempdir = machine.succeed("mktemp -d -p /var/lib/jellyfin").strip()
machine.succeed(f"chmod 755 '{tempdir}'")
# Generate a dummy video that we can test later
videofile = f"{tempdir}/Big Buck Bunny (2008) [1080p].mkv"
machine.succeed(f"ffmpeg -f lavfi -i testsrc2=duration=5 '{videofile}'")
add_folder_query = urlencode(
{
"name": "My Library",
"collectionType": "Movies",
"paths": tempdir,
"refreshLibrary": "true",
}
)
machine.succeed(
api_post(
f"/Library/VirtualFolders?{add_folder_query}",
"${payloads.empty}",
)
)
def is_refreshed(_):
folders_str = machine.succeed(api_get("/Library/VirtualFolders"))
folders = json.loads(folders_str)
print(folders)
return all(folder["RefreshStatus"] == "Idle" for folder in folders)
retry(is_refreshed)
with machine.nested("Can identify videos"):
items = []
# For some reason, having the folder refreshed doesn't mean the
# movie was scanned
def has_movie(_):
global items
items_str = machine.succeed(
api_get(f"/Users/{me}/Items?IncludeItemTypes=Movie&Recursive=true")
)
items = json.loads(items_str)["Items"]
return len(items) == 1
retry(has_movie)
video = items[0]["Id"]
item_info_str = machine.succeed(api_get(f"/Users/{me}/Items/{video}"))
item_info = json.loads(item_info_str)
if item_info["Name"] != "Big Buck Bunny":
raise Exception("Jellyfin failed to properly identify file")
with machine.nested("Can read videos"):
media_source_id = item_info["MediaSources"][0]["Id"]
machine.succeed(
"ffmpeg"
+ f" -headers 'X-Emby-Authorization:{auth_header}'"
+ f" -i http://localhost:8096/Videos/{video}/master.m3u8?mediaSourceId={media_source_id}"
+ " /tmp/test.mkv"
)
duration = machine.succeed(
"ffprobe /tmp/test.mkv"
+ " -show_entries format=duration"
+ " -of compact=print_section=0:nokey=1"
)
if duration.strip() != "5.000000":
raise Exception("Downloaded video has wrong duration")
'';
}
)
2 changes: 1 addition & 1 deletion pkgs/by-name/je/jellyseerr/package.nix
Original file line number Diff line number Diff line change
Expand Up @@ -53,7 +53,7 @@ stdenv.mkDerivation rec {
installPhase = ''
runHook preInstall
mkdir -p $out/share
cp -r -t $out/share .next node_modules dist config package.json overseerr-api.yml
cp -r -t $out/share .next node_modules dist public package.json overseerr-api.yml
runHook postInstall
'';

Expand Down

0 comments on commit 4b33626

Please sign in to comment.