516 lines
15 KiB
Go
516 lines
15 KiB
Go
package backend
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"math/rand"
|
|
"net/http"
|
|
"os"
|
|
"path/filepath"
|
|
"regexp"
|
|
"strings"
|
|
"time"
|
|
)
|
|
|
|
type QobuzDownloader struct {
|
|
client *http.Client
|
|
appID string
|
|
}
|
|
|
|
type QobuzSearchResponse struct {
|
|
Query string `json:"query"`
|
|
Tracks struct {
|
|
Limit int `json:"limit"`
|
|
Offset int `json:"offset"`
|
|
Total int `json:"total"`
|
|
Items []QobuzTrack `json:"items"`
|
|
} `json:"tracks"`
|
|
}
|
|
|
|
type QobuzTrack struct {
|
|
ID int64 `json:"id"`
|
|
Title string `json:"title"`
|
|
Version string `json:"version"`
|
|
Duration int `json:"duration"`
|
|
TrackNumber int `json:"track_number"`
|
|
MediaNumber int `json:"media_number"`
|
|
ISRC string `json:"isrc"`
|
|
Copyright string `json:"copyright"`
|
|
MaximumBitDepth int `json:"maximum_bit_depth"`
|
|
MaximumSamplingRate float64 `json:"maximum_sampling_rate"`
|
|
Hires bool `json:"hires"`
|
|
HiresStreamable bool `json:"hires_streamable"`
|
|
ReleaseDateOriginal string `json:"release_date_original"`
|
|
Performer struct {
|
|
Name string `json:"name"`
|
|
ID int64 `json:"id"`
|
|
} `json:"performer"`
|
|
Album struct {
|
|
Title string `json:"title"`
|
|
ID string `json:"id"`
|
|
Image struct {
|
|
Small string `json:"small"`
|
|
Thumbnail string `json:"thumbnail"`
|
|
Large string `json:"large"`
|
|
} `json:"image"`
|
|
Artist struct {
|
|
Name string `json:"name"`
|
|
ID int64 `json:"id"`
|
|
} `json:"artist"`
|
|
Label struct {
|
|
Name string `json:"name"`
|
|
} `json:"label"`
|
|
} `json:"album"`
|
|
}
|
|
|
|
type QobuzStreamResponse struct {
|
|
URL string `json:"url"`
|
|
}
|
|
|
|
func NewQobuzDownloader() *QobuzDownloader {
|
|
return &QobuzDownloader{
|
|
client: &http.Client{
|
|
Timeout: 60 * time.Second,
|
|
},
|
|
appID: "798273057",
|
|
}
|
|
}
|
|
|
|
func (q *QobuzDownloader) searchByISRC(isrc string) (*QobuzTrack, error) {
|
|
apiBase := "https://www.qobuz.com/api.json/0.2/track/search?query="
|
|
url := fmt.Sprintf("%s%s&limit=1&app_id=%s", apiBase, isrc, q.appID)
|
|
|
|
resp, err := q.client.Get(url)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to search track: %w", err)
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode != 200 {
|
|
return nil, fmt.Errorf("API returned status %d", resp.StatusCode)
|
|
}
|
|
|
|
var searchResp QobuzSearchResponse
|
|
|
|
body, err := io.ReadAll(resp.Body)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to read response body: %w", err)
|
|
}
|
|
|
|
if len(body) == 0 {
|
|
return nil, fmt.Errorf("API returned empty response")
|
|
}
|
|
|
|
if err := json.Unmarshal(body, &searchResp); err != nil {
|
|
|
|
bodyStr := string(body)
|
|
if len(bodyStr) > 200 {
|
|
bodyStr = bodyStr[:200] + "..."
|
|
}
|
|
return nil, fmt.Errorf("failed to decode response: %w (response: %s)", err, bodyStr)
|
|
}
|
|
|
|
if len(searchResp.Tracks.Items) == 0 {
|
|
return nil, fmt.Errorf("track not found for ISRC: %s", isrc)
|
|
}
|
|
|
|
return &searchResp.Tracks.Items[0], nil
|
|
}
|
|
|
|
func buildQobuzAPIURL(apiBase string, trackID int64, quality string) string {
|
|
if strings.Contains(apiBase, "qbz.afkarxyz.fun") {
|
|
return fmt.Sprintf("%s%d?quality=%s", apiBase, trackID, quality)
|
|
}
|
|
return fmt.Sprintf("%s%d&quality=%s", apiBase, trackID, quality)
|
|
}
|
|
|
|
func (q *QobuzDownloader) DownloadFromStandard(apiBase string, trackID int64, quality string) (string, error) {
|
|
apiURL := buildQobuzAPIURL(apiBase, trackID, quality)
|
|
resp, err := q.client.Get(apiURL)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode != 200 {
|
|
return "", fmt.Errorf("status %d", resp.StatusCode)
|
|
}
|
|
|
|
body, err := io.ReadAll(resp.Body)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
if len(body) == 0 {
|
|
return "", fmt.Errorf("empty body")
|
|
}
|
|
|
|
var streamResp QobuzStreamResponse
|
|
if err := json.Unmarshal(body, &streamResp); err == nil && streamResp.URL != "" {
|
|
return streamResp.URL, nil
|
|
}
|
|
|
|
var nestedResp struct {
|
|
Data struct {
|
|
URL string `json:"url"`
|
|
} `json:"data"`
|
|
}
|
|
if err := json.Unmarshal(body, &nestedResp); err == nil && nestedResp.Data.URL != "" {
|
|
return nestedResp.Data.URL, nil
|
|
}
|
|
|
|
return "", fmt.Errorf("invalid response")
|
|
}
|
|
|
|
func (q *QobuzDownloader) GetDownloadURL(trackID int64, quality string, allowFallback bool) (string, error) {
|
|
qualityCode := quality
|
|
if qualityCode == "" || qualityCode == "5" {
|
|
qualityCode = "6"
|
|
}
|
|
|
|
fmt.Printf("Getting download URL for track ID: %d with requested quality: %s\n", trackID, qualityCode)
|
|
|
|
standardAPIs := []string{
|
|
"https://dab.yeet.su/api/stream?trackId=",
|
|
"https://dabmusic.xyz/api/stream?trackId=",
|
|
"https://qbz.afkarxyz.fun/api/track/",
|
|
}
|
|
|
|
downloadFunc := func(qual string) (string, error) {
|
|
type Provider struct {
|
|
Name string
|
|
Func func() (string, error)
|
|
}
|
|
|
|
var providers []Provider
|
|
|
|
for _, api := range standardAPIs {
|
|
currentAPI := api
|
|
providers = append(providers, Provider{
|
|
Name: "Standard(" + currentAPI + ")",
|
|
Func: func() (string, error) {
|
|
return q.DownloadFromStandard(currentAPI, trackID, qual)
|
|
},
|
|
})
|
|
}
|
|
|
|
rand.Seed(time.Now().UnixNano())
|
|
rand.Shuffle(len(providers), func(i, j int) { providers[i], providers[j] = providers[j], providers[i] })
|
|
|
|
var lastErr error
|
|
for _, p := range providers {
|
|
|
|
fmt.Printf("Trying Provider: %s (Quality: %s)...\n", p.Name, qual)
|
|
|
|
url, err := p.Func()
|
|
if err == nil {
|
|
fmt.Printf("✓ Success\n")
|
|
return url, nil
|
|
}
|
|
|
|
fmt.Printf("Provider failed: %v\n", err)
|
|
lastErr = err
|
|
}
|
|
return "", lastErr
|
|
}
|
|
|
|
url, err := downloadFunc(qualityCode)
|
|
if err == nil {
|
|
return url, nil
|
|
}
|
|
|
|
currentQuality := qualityCode
|
|
|
|
if currentQuality == "27" && allowFallback {
|
|
fmt.Printf("⚠ Download with quality 27 failed, trying fallback to 7 (24-bit Standard)...\n")
|
|
url, err := downloadFunc("7")
|
|
if err == nil {
|
|
fmt.Println("✓ Success with fallback quality 7")
|
|
return url, nil
|
|
}
|
|
|
|
currentQuality = "7"
|
|
}
|
|
|
|
if currentQuality == "7" && allowFallback {
|
|
fmt.Printf("⚠ Download with quality 7 failed, trying fallback to 6 (16-bit Lossless)...\n")
|
|
url, err := downloadFunc("6")
|
|
if err == nil {
|
|
fmt.Println("✓ Success with fallback quality 6")
|
|
return url, nil
|
|
}
|
|
}
|
|
|
|
return "", fmt.Errorf("all APIs and fallbacks failed. Last error: %v", err)
|
|
}
|
|
|
|
func (q *QobuzDownloader) DownloadFile(url, filepath string) error {
|
|
fmt.Println("Starting file download...")
|
|
|
|
downloadClient := &http.Client{
|
|
Timeout: 5 * time.Minute,
|
|
}
|
|
|
|
resp, err := downloadClient.Get(url)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to download file: %w", err)
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode != 200 {
|
|
return fmt.Errorf("download failed with status %d", resp.StatusCode)
|
|
}
|
|
|
|
fmt.Printf("Creating file: %s\n", filepath)
|
|
out, err := os.Create(filepath)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to create file: %w", err)
|
|
}
|
|
defer out.Close()
|
|
|
|
fmt.Println("Downloading...")
|
|
|
|
pw := NewProgressWriter(out)
|
|
_, err = io.Copy(pw, resp.Body)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to write file: %w", err)
|
|
}
|
|
|
|
fmt.Printf("\rDownloaded: %.2f MB (Complete)\n", float64(pw.GetTotal())/(1024*1024))
|
|
return nil
|
|
}
|
|
|
|
func (q *QobuzDownloader) DownloadCoverArt(coverURL, filepath string) error {
|
|
if coverURL == "" {
|
|
return fmt.Errorf("no cover URL provided")
|
|
}
|
|
|
|
resp, err := q.client.Get(coverURL)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to download cover: %w", err)
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode != 200 {
|
|
return fmt.Errorf("cover download failed with status %d", resp.StatusCode)
|
|
}
|
|
|
|
out, err := os.Create(filepath)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to create cover file: %w", err)
|
|
}
|
|
defer out.Close()
|
|
|
|
_, err = io.Copy(out, resp.Body)
|
|
return err
|
|
}
|
|
|
|
func buildQobuzFilename(title, artist, album, albumArtist, releaseDate string, trackNumber, discNumber int, format string, includeTrackNumber bool, position int, useAlbumTrackNumber bool) string {
|
|
var filename string
|
|
|
|
numberToUse := position
|
|
if useAlbumTrackNumber && trackNumber > 0 {
|
|
numberToUse = trackNumber
|
|
}
|
|
|
|
year := ""
|
|
if len(releaseDate) >= 4 {
|
|
year = releaseDate[:4]
|
|
}
|
|
|
|
if strings.Contains(format, "{") {
|
|
filename = format
|
|
filename = strings.ReplaceAll(filename, "{title}", title)
|
|
filename = strings.ReplaceAll(filename, "{artist}", artist)
|
|
filename = strings.ReplaceAll(filename, "{album}", album)
|
|
filename = strings.ReplaceAll(filename, "{album_artist}", albumArtist)
|
|
filename = strings.ReplaceAll(filename, "{year}", year)
|
|
filename = strings.ReplaceAll(filename, "{date}", SanitizeFilename(releaseDate))
|
|
|
|
if discNumber > 0 {
|
|
filename = strings.ReplaceAll(filename, "{disc}", fmt.Sprintf("%d", discNumber))
|
|
} else {
|
|
filename = strings.ReplaceAll(filename, "{disc}", "")
|
|
}
|
|
|
|
if numberToUse > 0 {
|
|
filename = strings.ReplaceAll(filename, "{track}", fmt.Sprintf("%02d", numberToUse))
|
|
} else {
|
|
|
|
filename = regexp.MustCompile(`\{track\}\.\s*`).ReplaceAllString(filename, "")
|
|
filename = regexp.MustCompile(`\{track\}\s*-\s*`).ReplaceAllString(filename, "")
|
|
filename = regexp.MustCompile(`\{track\}\s*`).ReplaceAllString(filename, "")
|
|
}
|
|
} else {
|
|
|
|
switch format {
|
|
case "artist-title":
|
|
filename = fmt.Sprintf("%s - %s", artist, title)
|
|
case "title":
|
|
filename = title
|
|
default:
|
|
filename = fmt.Sprintf("%s - %s", title, artist)
|
|
}
|
|
|
|
if includeTrackNumber && position > 0 {
|
|
filename = fmt.Sprintf("%02d. %s", numberToUse, filename)
|
|
}
|
|
}
|
|
|
|
return filename + ".flac"
|
|
}
|
|
|
|
func (q *QobuzDownloader) DownloadTrack(spotifyID, outputDir, quality, filenameFormat string, includeTrackNumber bool, position int, spotifyTrackName, spotifyArtistName, spotifyAlbumName, spotifyAlbumArtist, spotifyReleaseDate string, useAlbumTrackNumber bool, spotifyCoverURL string, embedMaxQualityCover bool, spotifyTrackNumber, spotifyDiscNumber, spotifyTotalTracks int, spotifyTotalDiscs int, spotifyCopyright, spotifyPublisher, spotifyURL string, allowFallback bool, useFirstArtistOnly bool, useSingleGenre bool, embedGenre bool) (string, error) {
|
|
var deezerISRC string
|
|
if spotifyID != "" {
|
|
songlinkClient := NewSongLinkClient()
|
|
isrc, err := songlinkClient.GetISRC(spotifyID)
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to get ISRC: %v", err)
|
|
}
|
|
deezerISRC = isrc
|
|
} else {
|
|
return "", fmt.Errorf("spotify ID is required for Qobuz download")
|
|
}
|
|
|
|
return q.DownloadTrackWithISRC(deezerISRC, spotifyID, outputDir, quality, filenameFormat, includeTrackNumber, position, spotifyTrackName, spotifyArtistName, spotifyAlbumName, spotifyAlbumArtist, spotifyReleaseDate, useAlbumTrackNumber, spotifyCoverURL, embedMaxQualityCover, spotifyTrackNumber, spotifyDiscNumber, spotifyTotalTracks, spotifyTotalDiscs, spotifyCopyright, spotifyPublisher, spotifyURL, allowFallback, useFirstArtistOnly, useSingleGenre, embedGenre)
|
|
}
|
|
|
|
func (q *QobuzDownloader) DownloadTrackWithISRC(deezerISRC, spotifyID, outputDir, quality, filenameFormat string, includeTrackNumber bool, position int, spotifyTrackName, spotifyArtistName, spotifyAlbumName, spotifyAlbumArtist, spotifyReleaseDate string, useAlbumTrackNumber bool, spotifyCoverURL string, embedMaxQualityCover bool, spotifyTrackNumber, spotifyDiscNumber, spotifyTotalTracks int, spotifyTotalDiscs int, spotifyCopyright, spotifyPublisher, spotifyURL string, allowFallback bool, useFirstArtistOnly bool, useSingleGenre bool, embedGenre bool) (string, error) {
|
|
fmt.Printf("Fetching track info for ISRC: %s\n", deezerISRC)
|
|
|
|
metaChan := make(chan Metadata, 1)
|
|
if embedGenre && deezerISRC != "" {
|
|
go func() {
|
|
fmt.Println("Fetching MusicBrainz metadata...")
|
|
if fetchedMeta, err := FetchMusicBrainzMetadata(deezerISRC, spotifyTrackName, spotifyArtistName, spotifyAlbumName, useSingleGenre, embedGenre); err == nil {
|
|
fmt.Println("✓ MusicBrainz metadata fetched")
|
|
metaChan <- fetchedMeta
|
|
} else {
|
|
fmt.Printf("Warning: Failed to fetch MusicBrainz metadata: %v\n", err)
|
|
metaChan <- Metadata{}
|
|
}
|
|
}()
|
|
} else {
|
|
close(metaChan)
|
|
}
|
|
|
|
if outputDir != "." {
|
|
if err := os.MkdirAll(outputDir, 0755); err != nil {
|
|
return "", fmt.Errorf("failed to create output directory: %w", err)
|
|
}
|
|
}
|
|
|
|
track, err := q.searchByISRC(deezerISRC)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
artists := spotifyArtistName
|
|
trackTitle := spotifyTrackName
|
|
albumTitle := spotifyAlbumName
|
|
|
|
fmt.Printf("Found track: %s - %s\n", artists, trackTitle)
|
|
fmt.Printf("Album: %s\n", albumTitle)
|
|
|
|
qualityInfo := "Standard"
|
|
if track.Hires {
|
|
qualityInfo = fmt.Sprintf("Hi-Res (%d-bit / %.1f kHz)", track.MaximumBitDepth, track.MaximumSamplingRate)
|
|
}
|
|
fmt.Printf("Quality: %s\n", qualityInfo)
|
|
|
|
fmt.Println("Getting download URL...")
|
|
downloadURL, err := q.GetDownloadURL(track.ID, quality, allowFallback)
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to get download URL: %w", err)
|
|
}
|
|
|
|
if downloadURL == "" {
|
|
return "", fmt.Errorf("received empty download URL")
|
|
}
|
|
|
|
urlPreview := downloadURL
|
|
if len(downloadURL) > 60 {
|
|
urlPreview = downloadURL[:60] + "..."
|
|
}
|
|
fmt.Printf("Download URL obtained: %s\n", urlPreview)
|
|
|
|
safeArtist := sanitizeFilename(artists)
|
|
safeAlbumArtist := sanitizeFilename(spotifyAlbumArtist)
|
|
|
|
if useFirstArtistOnly {
|
|
safeArtist = sanitizeFilename(GetFirstArtist(artists))
|
|
safeAlbumArtist = sanitizeFilename(GetFirstArtist(spotifyAlbumArtist))
|
|
}
|
|
|
|
safeTitle := sanitizeFilename(trackTitle)
|
|
safeAlbum := sanitizeFilename(albumTitle)
|
|
|
|
filename := buildQobuzFilename(safeTitle, safeArtist, safeAlbum, safeAlbumArtist, spotifyReleaseDate, spotifyTrackNumber, spotifyDiscNumber, filenameFormat, includeTrackNumber, position, useAlbumTrackNumber)
|
|
filepath := filepath.Join(outputDir, filename)
|
|
|
|
if fileInfo, err := os.Stat(filepath); err == nil && fileInfo.Size() > 0 {
|
|
fmt.Printf("File already exists: %s (%.2f MB)\n", filepath, float64(fileInfo.Size())/(1024*1024))
|
|
return "EXISTS:" + filepath, nil
|
|
}
|
|
|
|
fmt.Printf("Downloading FLAC file to: %s\n", filepath)
|
|
if err := q.DownloadFile(downloadURL, filepath); err != nil {
|
|
return "", fmt.Errorf("failed to download file: %w", err)
|
|
}
|
|
|
|
fmt.Printf("Downloaded: %s\n", filepath)
|
|
|
|
coverPath := ""
|
|
|
|
if spotifyCoverURL != "" {
|
|
coverPath = filepath + ".cover.jpg"
|
|
coverClient := NewCoverClient()
|
|
if err := coverClient.DownloadCoverToPath(spotifyCoverURL, coverPath, embedMaxQualityCover); err != nil {
|
|
fmt.Printf("Warning: Failed to download Spotify cover: %v\n", err)
|
|
coverPath = ""
|
|
} else {
|
|
defer os.Remove(coverPath)
|
|
fmt.Println("Spotify cover downloaded")
|
|
}
|
|
}
|
|
|
|
var mbMeta Metadata
|
|
if deezerISRC != "" {
|
|
mbMeta = <-metaChan
|
|
}
|
|
|
|
fmt.Println("Embedding metadata and cover art...")
|
|
|
|
trackNumberToEmbed := spotifyTrackNumber
|
|
if trackNumberToEmbed == 0 {
|
|
trackNumberToEmbed = 1
|
|
}
|
|
|
|
metadata := Metadata{
|
|
Title: trackTitle,
|
|
Artist: artists,
|
|
Album: albumTitle,
|
|
AlbumArtist: spotifyAlbumArtist,
|
|
Date: spotifyReleaseDate,
|
|
TrackNumber: trackNumberToEmbed,
|
|
TotalTracks: spotifyTotalTracks,
|
|
DiscNumber: spotifyDiscNumber,
|
|
TotalDiscs: spotifyTotalDiscs,
|
|
URL: spotifyURL,
|
|
Copyright: spotifyCopyright,
|
|
Publisher: spotifyPublisher,
|
|
Description: "https://github.com/afkarxyz/SpotiFLAC",
|
|
ISRC: deezerISRC,
|
|
Genre: mbMeta.Genre,
|
|
}
|
|
|
|
if err := EmbedMetadata(filepath, metadata, coverPath); err != nil {
|
|
return "", fmt.Errorf("failed to embed metadata: %w", err)
|
|
}
|
|
|
|
fmt.Println("Metadata embedded successfully!")
|
|
return filepath, nil
|
|
}
|