-
Notifications
You must be signed in to change notification settings - Fork 7
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Synced lyrics support #87
base: main
Are you sure you want to change the base?
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This fixes a bug in the playlist fetching code, where the response can sometimes be nil by the time this is updated -- heavy network congestion, mostly. It doesn't belong in this PR, per se, but if this PR is merged, it addresses an issue. The change should be made regardless.
…n aborted start at support. Signed-off-by: Sean E. Russell <ser@ser1.net>
adds: ability to also write logs to a log file Signed-off-by: Sean E. Russell <ser@ser1.net>
… that makes up for the inaccuracy of mpv Signed-off-by: Sean E. Russell <ser@ser1.net>
Signed-off-by: Sean E. Russell <ser@ser1.net>
I finally set up commit signing. ¯\_(ツ)_/¯ |
…ren't updating from song to song unless the album art also changed.
One thing this is missing is checking if the server actually supports lyrics. Lyrics are an extension, not baked into the base API as specified here. A situation like it is now for gonic where the view doesn't exist is still a "valid" implementation of the spec. I haven't gotten around to testing gonic with lyrics, but without it, browsing the queue gets sluggish (latency dependent). Every movement already triggers a cover art fetch (dealt with by the concurrent album art branch and caching), and also a lyrics fetch with these changes now. Gonic (currently on main, at least), reports an error 70: view not found for lyrics. Because that response is invalid, nothing gets cached and stmps keeps retrying on each re-selection. This will be fixed when that gonic PR gets merged, but any servers running prior versions, and other servers that don't support the extension will have this issue. |
…ntains the OpenSubsonic lyrics extension
I've pushed changes that account for this into the xxxserxxx branch, and backported them to this branch in fa85bb0. One thing to note is that just because a server reports that it has the endpoints doesn't mean that the endpoints work: Navidrome claims to support lyrics, and it has the endpoints, but it does not return
Even without the concurrent branch, every movement shouldn't cause a fetch; it fetches only if the album art hasn't already been fetched. What the concurrent branch does is fetch the art in the background, so that browsing doesn't affect the latency of moving through the list. Yes, if the art isn't loaded, there might be a lag, but if you're browsing an entire album, there should be no lag after the first. This is around (depending on your branch) In any case, cover art issues are unrelated to lyrics support, right? EditIncidentally, it occurs to me that it may be useful to include in the issue reporting the network layout of the reporter. As I've mentioned elsewhere, I'm running both gonic and Navidrome on a several-year-old weenie AMD A10 Micro-6700T with a bunch of other self-hosted services (mpd, MyMPD, Home Assistant, snapcast server, Jellyfin, a mosquitto server, caddy), and the lag I see on the lyrics branch with cover art and lyrics is a slight hesitation if I hold down the "down" arrow when browsing a large queue. I'm thinking that maybe the difference is that my server is in the basement, and I'm ethernetted (over a coaxial converter, and I get ca. 8Mbps R/W between desktop & server). You (@mahmed2000) and @spezifisch are both reporting much more lag, and I wonder if that's entirely network related or some other issue. In any case, the issue template might include a "profile" attribute, where the reporter says whether the server is accessed through a LAN or WAN. |
That would be strange. afaik, subsonic isn't supposed to return lrc files. I checked navidrome's source, and that endpoint should be making the structured response. Is the server returning no lyrics in the StructuredLyrics array specifically, or literally just an empty json object at the lyricsList level?
Right, I'm realizing I could have phrased that a lot better. By fetch I meant just calling GetCoverArt here, so fetching is both from the server or from the cache. There's no lag caused by the cover art after the first time it gets cached. The only issue with cover art is when there's 50 songs from different albums. That's solved by the concurrent album branch, and has nothing to do with lyrics. I mentioned covers more so as "empty lyrics can fail to cache, unlike cover art".
That would the case, as far as I can tell. I access mine over 2 locations. One is when I'm accessing it over LAN over wifi. The other is over a VPN tunnel with 200-300 ms of latency, and spotty bandwidth anywhere in 100-1k kbps. Over LAN there's basically zero issues. Over the tunnel, its a hitch-fest. |
According to what? The only defined metadata tag that stores lyrics, in any format, stores unstructured lyrics. Depending on who and what you read; there are several conversations debating what could go where, but Navidrome actively strips sync data out of synced lyrics stored in the lyrics tag, and it ignores lrc files. It's going to get worse before it gets better, because there's an emerging spec where synchronization is at the word level, while According to the spec,
SYLT/USLT are not widely supported by encoders -- I've never encountered them in the wild -- and most tools for downloading synced lyrics do so to To answer your question, Navidrome just returns an empty structure. If there is a lyrics tag, Navidrome
Regardless, I don't find the source claiming that "Subsonic isn't supposed to return lrc files". Where did you find that? According to the OpenSubsonic specification for getLyricsForSongId,
For the purposes of stmps, it doesn't matter where the server gets lyrics from, so long as To summarize:
|
The fact that any server will never return "text/plain" data on that endpoint, and instead returns the structured xml/json object which contains lyrics as an xml list of tags/json array, as per the response object in that spec page. I think there's been a misunderstanding somewhere? Wherever the server gets its lyrics data from, it cajoles it into the lyrics object (at which point it would stop being a .lrc file). As far as any client is concerned, there need not even be an lrc file, since that data may be from any potential format, or even inside file metadata, or from some third-party service. Only the server's internals need to care and the raw source is never exposed to the API. On Navidrome, there's 2 lyric endpoints. The original (unused by this PR) /getLyrics, and /getLyricsBySongId (the opensubsonic extension). Navidrome's implementation of the second is here. Its making the Subsonic object, but if the lyrics themselves are empty, that's a different issue that isn't anything stmps needs to worry about: Back to the problem this was about, by "returns nothing", do you mean specifically the array of lyrics in the object is empty, or that the server literally just returns "{}"? Because the former would also be valid for just instrumentals or can be chalked up to the server not having any lyrics associated with said song, (unsupported lrc or whatever the case may be). That's still "valid", even if it is technically incorrect. But relying on the server for lyrics means that if the server reports no lyrics, the client must consider that to be "true". |
Navidrome currently only reads lyrics from tags, sync and unsync, from both Vorbis and ID3 tags. If the Vorbis LYRICS tag is synced, Navidrome will return it with timestamps in the As for supporting .lrc files, we have an open PR to read lyrics from files and from lrclib API: navidrome/navidrome#2897. It will (most probably) be merged for the next release. |
Thank you for providing more detail about this.
In no spec I could find were SYLT or USLT referenced for Vorbis, and what information I found about LYRICS was that it was specifically for non-synced lyrics.
Good to know, thanks. I can at least create an audio file for testing.
Outstanding! Given the really poor state of lyrics tag behavior in every container specification, supporting external lrc files is great for users. |
Are you assuming that because The patch in any case makes no assumption about where the data comes from. It requests and expects the response to be in json format, but how it gets to be that way is irrelevant. The patch does not use
On gonic, too.
Yes.
No argument. Which is why the patch updated two weeks ago checks for the extension and disables lyrics if the server doesn't have the lyrics. If it does claim to support the extension, then it should return a structure even if the contents are empty, which is how Navidrome is behaving (currently, with |
I feel like this is more me being overly pedantic and just arguing semantics, sorry about that. I'm not assuming the json/xml response is unstructured, I'm assuming an .lrc contains structured lyrics, but not all structured lyrics are .lrc. My contention isn't with returning unstructured lyrics, but with returning lrc files. We're both trying to say the same thing, I think. I'm assuming you mean to say that if the server returns the contents of an .lrc, it is returning an .lrc for all intents and purposes. To me, .lrc is just a container so a server cannot return an .lrc unless its literally text data of "[timestamp] lyrics\n[timestamp] lyrics". This all started because I considered this incorrect:
All cases where the server reports a valid object that can be traversed down to the structuredLyrics array are valid. No lyrics is still the endpoint working properly. The server failing to read lyrics from .lrc files is still the endpoint working properly. The endpoint not working would have been what current gonic does or a malformed response object, not an empty lyric list. Queue the rambling above. Apologies for the confusion. |
Thank you for the clarification. To clarify, the patch makes no assumption about where the lyrics come from, only that they come in the correct format for the API call; and with the patch I made after you pointed out that the endpoint is an extension, the patch should also no longer generate errors if the server doesn't support the endpoint. I only want to be clear that many user will not see lyrics, even with this patch, because (a) synced lyrics support is spotty among servers, and (b) even if the server does support it, currently users have to specifically provide the server with lyrics in a format the server is looking for. Base gonic doesn't provide them at all; the patched gonic gets synced lyrics from LRC files, and Navidrome -- as far as I can tell -- will provide synced lyrics from the LYRICS tag (even though specs I can find say that this data is explicitly unsynced; details are up a few comments). So users' mileage may vary, and they need to pay attention to what their particular server is doing -- lyrics not showing up in stmps with this patch doesn't necessarily means its a bug in this code :-) |
Implements #49
This was rebased onto main.
This adds support for synchronized lyrics.
It's functional, if not beautiful. mpv gives us ticks every second, on the second, and synchronzed lyrcs rarely line up with the second marks. I've found that a sweet spot is changing lyrics about a half-second in advance, as it nearly always means that at least the right line is highlighted when the vocals happen.
Navidrome doesn't have synchronized lyrics support, that I can tell; @danielepintore has a PR for gonic in the queue waiting for @sentriz to pull it -- that's the fork I'm working and testing with.
As usual, here's the teaser; it's long, but near the end shows me jumping around and having the lyrics match up:
Edit sigh jump forward at the start to 15s; it was apparently recording while waiting for me to respond to a prompt.