feat: add download fallback

This commit is contained in:
2026-02-20 15:08:53 +01:00
parent 36030f9f4c
commit e666e1b4f8
3 changed files with 58 additions and 9 deletions

View File

@@ -101,18 +101,61 @@ func (app *App) DownloadPlaylist(url string, outputFile string, service string,
return nil return nil
} }
func (app *App) GetDownloadUrlOrFallback(askedService string, quality string, songlink SongLinkResponse) (string, error) {
servicesToTry := []string{}
switch askedService {
default:
case "tidal":
servicesToTry = []string{"tidal", "amazon", "qoboz"}
break
case "amazon":
servicesToTry = []string{"amazon", "tidal", "qoboz"}
break
case "quoboz":
servicesToTry = []string{"quoboz", "tidal", "amazon"}
break
}
var downloadUrl string
var lastError error
for _, service := range servicesToTry {
switch service {
case "tidal":
if songlink.LinksByPlatform.Tidal == nil {
continue
}
tidalId, err := app.GetTidalIdFromSonglink(songlink)
if err != nil {
lastError = err
continue
}
downloadUrl, err = app.GetTidalDownloadUrl(tidalId, quality)
if err != nil {
lastError = err
continue
}
break
}
}
if lastError != nil || downloadUrl == "" {
return "", errors.New("Unable to download from any source.")
}
return downloadUrl, nil
}
func (app *App) DownloadTrack(url string, outputFile string, service string, quality string, downloadInFolder bool) error { func (app *App) DownloadTrack(url string, outputFile string, service string, quality string, downloadInFolder bool) error {
songlink, err := app.ConvertSongUrl(url) songlink, err := app.ConvertSongUrl(url)
if err != nil { if err != nil {
return err return err
} }
tidalId, err := app.GetTidalIdFromSonglink(songlink) downloadUrl, err := app.GetDownloadUrlOrFallback(service, quality, songlink)
if err != nil {
return err
}
downloadUrl, err := app.GetTidalDownloadUrl(tidalId, quality)
if err != nil { if err != nil {
return err return err
} }

View File

@@ -20,8 +20,8 @@ type SongLinkResponse struct {
} }
type LinksByPlatform struct { type LinksByPlatform struct {
Deezer LinkByPlatform `json:"deezer"` Deezer *LinkByPlatform `json:"deezer,omitempty"`
Tidal LinkByPlatform `json:"tidal"` Tidal *LinkByPlatform `json:"tidal,omitempty"`
} }
type LinkByPlatform struct { type LinkByPlatform struct {

View File

@@ -10,6 +10,8 @@ import (
"strings" "strings"
) )
var ErrTidalUrlNotFound = errors.New("Tidal URL not found.")
func (app *App) LoadTidalApis() error { func (app *App) LoadTidalApis() error {
var found bool var found bool
@@ -119,11 +121,15 @@ func (app *App) ParseTidalManifestFromBase64(manifestBase64 string) (TidalManife
err = json.Unmarshal(manifestDecoded, &result) err = json.Unmarshal(manifestDecoded, &result)
if err != nil { if err != nil {
return result, err return result, err
} }
return result, nil return result, nil
} }
func (app *App) GetTidalIdFromSonglink(songlink SongLinkResponse) (string, error) { func (app *App) GetTidalIdFromSonglink(songlink SongLinkResponse) (string, error) {
if songlink.LinksByPlatform.Tidal == nil {
return "", ErrTidalUrlNotFound
}
return ParseTrackId(songlink.LinksByPlatform.Tidal.Url) return ParseTrackId(songlink.LinksByPlatform.Tidal.Url)
} }