![]() |
SOLVED: MKV files wont play in Android client app when accessed through NGINX reverse proxy - Printable Version +- Jellyfin Forum (https://forum.jellyfin.org) +-- Forum: Support (https://forum.jellyfin.org/f-support) +--- Forum: Troubleshooting (https://forum.jellyfin.org/f-troubleshooting) +--- Thread: SOLVED: MKV files wont play in Android client app when accessed through NGINX reverse proxy (/t-solved-mkv-files-wont-play-in-android-client-app-when-accessed-through-nginx-reverse-proxy) |
MKV files wont play in Android client app when accessed through NGINX reverse proxy - stiw47 - 2025-02-24 TBH, I am not quite sure if this is Jellyfin related, or NGINX related or SSO-Auth plugin related, but I spent lot of time yesterday, was not able to figure it on my own, and decided to ask for help. In short: 1. I am using jellyfin-plugin-sso: https://github.com/9p4/jellyfin-plugin-sso for authentication through Authentik. I already removed all internal users in Jellyfin, and only users from Authentik have access to the Jellyfin. Authentication works without issue. True, as we already know, it is not possible to login to Android Jellyfin client with SSO, but here I am using quick connect, and it is pretty stable, logged in session persist across phone reboot, etc. TBH, I'm not sure if this SSO login is related with the actual issue at all, but since I started using it 2-3 weeks ago, and not sure if issue started at the same time (noticed issue yesterday), I meant maybe is worthy to mention. 2. Jellyfin is running on 192.168.0.21:8096 in local network. It is docker installation on Archlinux host. Docker image is one from linuxserver.io: https://docs.linuxserver.io/images/docker-jellyfin/, and it is almost always on latest version, since I am doing docker compose pull && docker compose up -d almost every day. Anyway, current Jellyfin server version is 10.10.6. Docker compose will be provided later in TL;DR. At this point, everything is ok. If I connect Jellyfin Android client to http://192.168.0.21:8096 in home LAN, and login with quick connect, I can play any media file from Jellyfin libraries. Ok, "any media file" is little overrated, since I have .mkv containers, .webm, and .mp4 containers in my libraries, but ok, it can play both of them.3. Since I enabled SSO and 2FA, I want that my Jellyfin server being publicly accessible on internet. I already have my valid domain name, and valid Let's Encrypt certificate, and as little more background it is not first time to do something like this, I have a lot more services open to internet, and accessed in similar way like this Jellyfin through NGINX reverse proxy. So we coming to the part. Let's say my domain is e.g. example.com ( <- btw, this is Authentik address). And let's say my Jellyfin subdomain configured in NGINX is jellyfin.example.com . As said, it is configured as reverse proxy in NGINX, will provide conf file later in TL;DR, and it is working. I can access Jellyfin, I can browse all libraries, I can play any file .mp4 or .mkv when Jellyfin is accessed via subdomain and from web browser, regardless that web browser is on Windows PC, Linux PC, Android phone, etc.THE ISSUE: I cannot play .mkv files in Android Jellyfin client, when Jellyfin is accessed via subdomain from Android client. The error message in UI/player is pretty generic one, same as when codec is not supported: Playback error Playback failed due to a fatal player error. ![]() Let's just remind on point 2. above, and underline that there is no any issue with mkv file play, when Jellyfin Android client connects to 192.168.0.21:8096, instead to https://jellyfin.example.com Also, .mp4 and .webm files have no issue with playing from Android client and accessed via subdomain. Even more, and just as additional info, there is Android app called Findroid, unofficial Jellyfin client. This app has no issue with playing anything when connected via my subdomain. But it would be another topic why this app does not fit me and I cannot take it as full replacement. TL;DR My Jellyfin is part of big docker compose stack, including a lot of *arr apps and other, so below I will paste Jellyfin only related part from docker-compose.yaml :Code: services: This is my NGINX reverse proxy, /etc/nginx/conf.d/jellyfin.conf . Few stuffs changed yesterday in trials to make it work. Anyway, below is current version:Code: server { Since above jellyfin.conf inherits some settings from main nginx.conf , maybe worthy to mention that part from nginx.conf . It is in http block:Code: # Basic Security Headers Btw, NGINX is running also in docker, and it is pretty basic and standard installation, without additional modules, etc. and from official NGINX alpine image latest. This is what I can see in Jellyfin logs when I start playback of .mkv video ( docker logs -f media-jellyfin ):Code: [11:02:11] [INF] [27] Jellyfin.Plugin.LocalIntros.IntroProvider: Selecting intros based on criteria, 30 intros found I am using Local Intros plugin, and we can see ^ that intro 85.webm is playing successfully before movie (as said before, .webm has not issue). However, when .mkv should be played, it is stopped immediatelly on 0ms, without any further error. This is the specification of the above .mkv video: Code: < 🥩 stiw47@archmedia: Alfa i Omega 🥓 > $ ffprobe -hide_banner -i Alpha.and.Omega.2010.1080p.x264.SR.HR.EN.RU.mkv As we can see, nothing exotic. H264 video stream with 4 aac audio tracks, i.e. something should be able to play anywhere. For test purpose, I copied it to .mp4 container, with keep all codecs same: Code: $ ffmpeg -i Alpha.and.Omega.2010.1080p.x264.SR.HR.EN.RU.mkv -map 0 -c copy Alpha.and.Omega.2010.1080p.x264.SR.HR.EN.RU.mp4 Code: < 😜 stiw47@archmedia: Alfa i Omega 🛂 > $ ffprobe -hide_banner -i Alpha.and.Omega.2010.1080p.x264.SR.HR.EN.RU.mp4 And this .mp4 file play successfully in Android client accessed via subdomain: Code: [11:21:20] [INF] [97] Jellyfin.Plugin.LocalIntros.IntroProvider: Selecting intros based on criteria, 30 intros found ^^ Stopped at 13416 ms <- manually by me. I tried to disable/re-enable HW transcoding - no success (I did not expected, since this is h264 and shouldn't be related) I tried to spin up one more Jellyfin container, with the difference on port 8097 instead of 8096, and created related NGINX reverse proxy with jellyfin2.example.com pointing to it. This container has no SSO, login with internal user, and I'm also not able to play .mkv h264 in android app. I tried to add video/x-matroska mkv; to nginx mime.types file - no success.Tried also to append mkv to existing mime type video/webm webm mkv; - no success.Of course, nginx restarted when mime.types edited. Appreciate any help, and ready to provide any additional info. thanks. RE: MKV files wont play in Android client app when accessed through NGINX reverse proxy - TheDreadPirate - 2025-02-24 Remove these security options. Code: add_header X-XSS-Protection "1; mode=block" always; These are both known to be problematic with some clients. This one MIGHT also be an issue. Code: add_header Referrer-Policy "no-referrer" always; https://jellyfin.org/docs/general/networking/nginx/#nginx-from-a-subdomain-jellyfinexampleorg Also make sure your phone is set to use the integrated player and not the web player. Settings > Client settings. RE: MKV files wont play in Android client app when accessed through NGINX reverse proxy - stiw47 - 2025-02-24 Wov, thank you very much @TheDreadPirate. Since I'm watching some movie now, I done just quick test. Commented out all security headers:
Restarted NGINX container, and it works, in web player. Cannot remember now why I don't like integrated player, and why I prefer web player, but will check this later as well. Btw, I am sure that yesterday I also tried to comment out all above headers and restarted NGINX, but the difference is that I cleared cache and data of Android client today additionally and it started work. Later will check in more detail line by line, what header(s) exactly was(were) troublesome, and mark thread solved. Thx. RE: MKV files wont play in Android client app when accessed through NGINX reverse proxy - stiw47 - 2025-02-24 Ok, after more trial and errors, I found it. Actually, it was not related to any of the: Code: add_header X-XSS-Protection "1; mode=block" always; And all 3 ^ are uncommented now, mkv play is working. It was related to the CSP which I skipped to post here. Yeah, I have it also, and it is (IMHO) little idiotic long, TBH I am still learning about it and trying to find some optimal way.... Anyway, started ADB logcat to live monitor Jellyfin apk log when reproduce mkv error: Code: $ adb logcat | grep -F "`adb shell ps | grep org.jellyfin.mobile | tr -s [:space:] ' ' | cut -d' ' -f2`" Found this in log: Code: 02-24 19:29:24.082 14351 14351 E WebView : Refused to load media from 'blob:https://jellyfin.example.com/e48a257b-0c37-4e73-951f-5d5f9abce829' because it violates the following Content Security Policy directive: "default-src 'self'". Note that 'media-src' was not explicitly set, so 'default-src' is used as a fallback. After several trials and errors where and what should be put, ended with this in NGINX CSP: Code: add_header Content-Security-Policy "default-src 'self'; media-src 'self' blob: https://example.com https://*.example.com; img-src.... etc. etc. insanely long.... Will need to check/monitor now what else will be broken 😂, since I did not used media-src in CSP before 😂.
|