mirror of
https://github.com/zarzet/SpotiFLAC-Mobile.git
synced 2026-07-04 11:48:00 +02:00
Compare commits
29 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 4e530ffbc3 | |||
| 14f6776fdc | |||
| da1c6e9171 | |||
| 9c3e934395 | |||
| 15d2c3b465 | |||
| 8aaa6d5cbe | |||
| 9158d0228d | |||
| 2bbcda3320 | |||
| a7622676dd | |||
| 5779f910a2 | |||
| 030f44a444 | |||
| 1248270fb4 | |||
| 413e3b0686 | |||
| ac711efadc | |||
| 59f2fe880a | |||
| 355f2eba2a | |||
| f2f45fa31d | |||
| 042937a8ed | |||
| 674e9af3d0 | |||
| 76d50fab3a | |||
| 81e25d7dab | |||
| 26f26f792a | |||
| 4dfa76b49e | |||
| f511f30ad0 | |||
| a1aa1319ce | |||
| c936bd7dd0 | |||
| 3a60ea2f4e | |||
| 7dba938299 | |||
| 93e77aeb84 |
@@ -4,5 +4,5 @@ contact_links:
|
|||||||
url: https://github.com/zarzet/SpotiFLAC-Mobile#readme
|
url: https://github.com/zarzet/SpotiFLAC-Mobile#readme
|
||||||
about: Check the README for setup instructions and FAQ
|
about: Check the README for setup instructions and FAQ
|
||||||
- name: Extension Development Guide
|
- name: Extension Development Guide
|
||||||
url: https://zarz.moe/docs
|
url: https://spotiflac.zarz.moe/docs
|
||||||
about: Documentation for building SpotiFLAC extensions
|
about: Documentation for building SpotiFLAC extensions
|
||||||
|
|||||||
@@ -71,7 +71,7 @@ jobs:
|
|||||||
- name: Setup Go
|
- name: Setup Go
|
||||||
uses: actions/setup-go@v6
|
uses: actions/setup-go@v6
|
||||||
with:
|
with:
|
||||||
go-version: "1.25.7"
|
go-version: "1.25.8"
|
||||||
cache-dependency-path: go_backend/go.sum
|
cache-dependency-path: go_backend/go.sum
|
||||||
|
|
||||||
# Cache Gradle for faster builds
|
# Cache Gradle for faster builds
|
||||||
@@ -93,12 +93,12 @@ jobs:
|
|||||||
# Accept licenses
|
# Accept licenses
|
||||||
yes | $ANDROID_HOME/cmdline-tools/latest/bin/sdkmanager --licenses || true
|
yes | $ANDROID_HOME/cmdline-tools/latest/bin/sdkmanager --licenses || true
|
||||||
|
|
||||||
# Install NDK r27d LTS (required for 16KB page size support on Android 15+)
|
# Install NDK r29 (supports 16KB page size for Android 15+)
|
||||||
# Platform android-36 and build-tools 36.0.0 for targetSdk 36 (Android 16)
|
# Platform android-36 and build-tools 36.0.0 for targetSdk 36 (Android 16)
|
||||||
$ANDROID_HOME/cmdline-tools/latest/bin/sdkmanager "ndk;27.3.13750724" "platforms;android-36" "build-tools;36.0.0"
|
$ANDROID_HOME/cmdline-tools/latest/bin/sdkmanager "ndk;29.0.14206865" "platforms;android-36" "build-tools;36.0.0"
|
||||||
|
|
||||||
# Set NDK path
|
# Set NDK path
|
||||||
echo "ANDROID_NDK_HOME=$ANDROID_HOME/ndk/27.3.13750724" >> $GITHUB_ENV
|
echo "ANDROID_NDK_HOME=$ANDROID_HOME/ndk/29.0.14206865" >> $GITHUB_ENV
|
||||||
|
|
||||||
- name: Install gomobile
|
- name: Install gomobile
|
||||||
run: |
|
run: |
|
||||||
@@ -164,17 +164,22 @@ jobs:
|
|||||||
path: build/app/outputs/flutter-apk/SpotiFLAC-*.apk
|
path: build/app/outputs/flutter-apk/SpotiFLAC-*.apk
|
||||||
|
|
||||||
build-ios:
|
build-ios:
|
||||||
runs-on: macos-latest
|
runs-on: macos-15
|
||||||
needs: get-version # Only depends on version, NOT android build!
|
needs: get-version # Only depends on version, NOT android build!
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v6
|
uses: actions/checkout@v6
|
||||||
|
|
||||||
|
- name: Select Xcode 26.1.1
|
||||||
|
run: |
|
||||||
|
sudo xcode-select -s /Applications/Xcode_26.1.1.app
|
||||||
|
xcodebuild -version
|
||||||
|
|
||||||
- name: Setup Go
|
- name: Setup Go
|
||||||
uses: actions/setup-go@v6
|
uses: actions/setup-go@v6
|
||||||
with:
|
with:
|
||||||
go-version: "1.25.7"
|
go-version: "1.25.8"
|
||||||
cache-dependency-path: go_backend/go.sum
|
cache-dependency-path: go_backend/go.sum
|
||||||
|
|
||||||
# Cache CocoaPods
|
# Cache CocoaPods
|
||||||
|
|||||||
@@ -77,6 +77,7 @@ flutter_*.log
|
|||||||
# Development tools
|
# Development tools
|
||||||
tool/
|
tool/
|
||||||
.claude/settings.local.json
|
.claude/settings.local.json
|
||||||
|
.playwright-mcp/
|
||||||
|
|
||||||
# FVM Version Cache
|
# FVM Version Cache
|
||||||
.fvm/
|
.fvm/
|
||||||
|
|||||||
@@ -170,5 +170,18 @@ Interested in contributing? Check out the [Contributing Guide](CONTRIBUTING.md)
|
|||||||
| [dabmusic.xyz](https://dabmusic.xyz) | [AfkarXYZ](https://github.com/afkarxyz) | [LRCLib](https://lrclib.net) | [Paxsenix](https://lyrics.paxsenix.org) | [Cobalt](https://cobalt.tools) |
|
| [dabmusic.xyz](https://dabmusic.xyz) | [AfkarXYZ](https://github.com/afkarxyz) | [LRCLib](https://lrclib.net) | [Paxsenix](https://lyrics.paxsenix.org) | [Cobalt](https://cobalt.tools) |
|
||||||
| [qwkuns.me](https://qwkuns.me) | [SpotubeDL](https://spotubedl.com) | [Song.link](https://song.link) | [IDHS](https://github.com/sjdonado/idonthavespotify) | |
|
| [qwkuns.me](https://qwkuns.me) | [SpotubeDL](https://spotubedl.com) | [Song.link](https://song.link) | [IDHS](https://github.com/sjdonado/idonthavespotify) | |
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Disclaimer
|
||||||
|
|
||||||
|
This repository and its contents are provided strictly for educational and research purposes. The software is provided "as-is" without warranty of any kind, express or implied, as stated in the [MIT License](LICENSE).
|
||||||
|
|
||||||
|
- No copyrighted content is hosted, stored, mirrored, or distributed by this repository.
|
||||||
|
- Users must ensure that their use of this software is properly authorized and complies with all applicable laws, regulations, and third-party terms of service.
|
||||||
|
- This software is provided free of charge by the maintainer. If you paid a third party for access to this software in its original form from this repository, you may have been misled or scammed. Any redistribution or commercial use by third parties must comply with the terms of the repository license. No affiliation, endorsement, or support by the maintainer is implied unless explicitly stated in writing.
|
||||||
|
- SpotiFLAC Mobile is an independent project. It is not affiliated with, endorsed by, or connected to any other project or version on other platforms that may share a similar name. The maintainer of this repository has no control over or responsibility for third-party projects.
|
||||||
|
- The author(s) disclaim all liability for any direct, indirect, incidental, or consequential damages arising from the use or misuse of this software. Users assume all risk associated with its use.
|
||||||
|
- If you are a copyright holder or authorized representative and believe this repository infringes upon your rights, please contact the maintainer with sufficient detail (including relevant URLs and proof of ownership). The matter will be promptly investigated and appropriate action will be taken, which may include removal of the referenced material.
|
||||||
|
|
||||||
> [!TIP]
|
> [!TIP]
|
||||||
> **Star the repo** to get notified about all new releases directly from GitHub.
|
> **Star the repo** to get notified about all new releases directly from GitHub.
|
||||||
|
|||||||
@@ -57,6 +57,18 @@ android {
|
|||||||
}
|
}
|
||||||
|
|
||||||
buildTypes {
|
buildTypes {
|
||||||
|
getByName("debug") {
|
||||||
|
ndk {
|
||||||
|
debugSymbolLevel = "FULL"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
getByName("profile") {
|
||||||
|
ndk {
|
||||||
|
debugSymbolLevel = "FULL"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
release {
|
release {
|
||||||
// For local builds: use release signing if key.properties exists
|
// For local builds: use release signing if key.properties exists
|
||||||
// For CI builds: APK is signed by GitHub Action after build
|
// For CI builds: APK is signed by GitHub Action after build
|
||||||
@@ -71,6 +83,9 @@ android {
|
|||||||
getDefaultProguardFile("proguard-android-optimize.txt"),
|
getDefaultProguardFile("proguard-android-optimize.txt"),
|
||||||
"proguard-rules.pro"
|
"proguard-rules.pro"
|
||||||
)
|
)
|
||||||
|
ndk {
|
||||||
|
debugSymbolLevel = "FULL"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -94,24 +94,6 @@
|
|||||||
android:exported="false"
|
android:exported="false"
|
||||||
android:foregroundServiceType="dataSync" />
|
android:foregroundServiceType="dataSync" />
|
||||||
|
|
||||||
<!-- Audio playback service for media notification / background audio -->
|
|
||||||
<service
|
|
||||||
android:name="com.ryanheise.audioservice.AudioService"
|
|
||||||
android:exported="true"
|
|
||||||
android:foregroundServiceType="mediaPlayback">
|
|
||||||
<intent-filter>
|
|
||||||
<action android:name="android.media.browse.MediaBrowserService" />
|
|
||||||
</intent-filter>
|
|
||||||
</service>
|
|
||||||
|
|
||||||
<receiver
|
|
||||||
android:name="com.ryanheise.audioservice.MediaButtonReceiver"
|
|
||||||
android:exported="true">
|
|
||||||
<intent-filter>
|
|
||||||
<action android:name="android.intent.action.MEDIA_BUTTON" />
|
|
||||||
</intent-filter>
|
|
||||||
</receiver>
|
|
||||||
|
|
||||||
<!-- flutter_local_notifications receivers -->
|
<!-- flutter_local_notifications receivers -->
|
||||||
<receiver android:exported="false" android:name="com.dexterous.flutterlocalnotifications.ScheduledNotificationReceiver" />
|
<receiver android:exported="false" android:name="com.dexterous.flutterlocalnotifications.ScheduledNotificationReceiver" />
|
||||||
<receiver android:exported="false" android:name="com.dexterous.flutterlocalnotifications.ScheduledNotificationBootReceiver">
|
<receiver android:exported="false" android:name="com.dexterous.flutterlocalnotifications.ScheduledNotificationBootReceiver">
|
||||||
|
|||||||
@@ -163,10 +163,6 @@ class MainActivity: FlutterFragmentActivity() {
|
|||||||
"sm-t225",
|
"sm-t225",
|
||||||
"hammerhead",
|
"hammerhead",
|
||||||
)
|
)
|
||||||
/**
|
|
||||||
* Check if device should use Skia instead of Impeller.
|
|
||||||
* Returns true for devices with old/problematic GPUs or old Android versions.
|
|
||||||
*/
|
|
||||||
private fun shouldDisableImpeller(): Boolean {
|
private fun shouldDisableImpeller(): Boolean {
|
||||||
val hardware = Build.HARDWARE.lowercase(Locale.ROOT)
|
val hardware = Build.HARDWARE.lowercase(Locale.ROOT)
|
||||||
val board = Build.BOARD.lowercase(Locale.ROOT)
|
val board = Build.BOARD.lowercase(Locale.ROOT)
|
||||||
@@ -215,7 +211,6 @@ class MainActivity: FlutterFragmentActivity() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Try to get GPU renderer string.
|
|
||||||
* Note: This may return empty on some devices before OpenGL context is created.
|
* Note: This may return empty on some devices before OpenGL context is created.
|
||||||
*/
|
*/
|
||||||
private fun getGpuRenderer(): String {
|
private fun getGpuRenderer(): String {
|
||||||
@@ -2384,6 +2379,41 @@ class MainActivity: FlutterFragmentActivity() {
|
|||||||
}
|
}
|
||||||
result.success(response)
|
result.success(response)
|
||||||
}
|
}
|
||||||
|
"rewriteSplitArtistTags" -> {
|
||||||
|
val filePath = call.argument<String>("file_path") ?: ""
|
||||||
|
val artist = call.argument<String>("artist") ?: ""
|
||||||
|
val albumArtist = call.argument<String>("album_artist") ?: ""
|
||||||
|
val response = withContext(Dispatchers.IO) {
|
||||||
|
if (filePath.startsWith("content://")) {
|
||||||
|
val uri = Uri.parse(filePath)
|
||||||
|
val tempPath = copyUriToTemp(uri, ".flac")
|
||||||
|
?: return@withContext errorJson("Failed to copy SAF file to temp")
|
||||||
|
try {
|
||||||
|
val raw = Gobackend.rewriteSplitArtistTagsExport(tempPath, artist, albumArtist)
|
||||||
|
val obj = JSONObject(raw)
|
||||||
|
if (!obj.optBoolean("success", false)) {
|
||||||
|
return@withContext raw
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!writeUriFromPath(uri, tempPath)) {
|
||||||
|
return@withContext errorJson("Failed to write rewritten tags back to SAF file")
|
||||||
|
}
|
||||||
|
|
||||||
|
obj.put("file_path", filePath)
|
||||||
|
obj.toString()
|
||||||
|
} catch (e: Exception) {
|
||||||
|
errorJson("Failed to rewrite split artist tags in SAF file: ${e.message}")
|
||||||
|
} finally {
|
||||||
|
try {
|
||||||
|
File(tempPath).delete()
|
||||||
|
} catch (_: Exception) {}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Gobackend.rewriteSplitArtistTagsExport(filePath, artist, albumArtist)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
result.success(response)
|
||||||
|
}
|
||||||
"cleanupConnections" -> {
|
"cleanupConnections" -> {
|
||||||
withContext(Dispatchers.IO) {
|
withContext(Dispatchers.IO) {
|
||||||
Gobackend.cleanupConnections()
|
Gobackend.cleanupConnections()
|
||||||
@@ -2516,12 +2546,27 @@ class MainActivity: FlutterFragmentActivity() {
|
|||||||
val spotifyId = call.argument<String>("spotify_id") ?: ""
|
val spotifyId = call.argument<String>("spotify_id") ?: ""
|
||||||
val durationMs = call.argument<Number>("duration_ms")?.toLong() ?: 0L
|
val durationMs = call.argument<Number>("duration_ms")?.toLong() ?: 0L
|
||||||
val outputPath = call.argument<String>("output_path") ?: ""
|
val outputPath = call.argument<String>("output_path") ?: ""
|
||||||
|
val rawAudioFilePath = call.argument<String>("audio_file_path") ?: ""
|
||||||
val response = withContext(Dispatchers.IO) {
|
val response = withContext(Dispatchers.IO) {
|
||||||
|
var safAudioTemp: String? = null
|
||||||
try {
|
try {
|
||||||
Gobackend.fetchAndSaveLyrics(trackName, artistName, spotifyId, durationMs, outputPath)
|
// Resolve SAF content:// URI to a temp file the Go backend can read
|
||||||
|
val audioFilePath = if (rawAudioFilePath.startsWith("content://")) {
|
||||||
|
val uri = Uri.parse(rawAudioFilePath)
|
||||||
|
val tempPath = copyUriToTemp(uri)
|
||||||
|
safAudioTemp = tempPath
|
||||||
|
tempPath ?: ""
|
||||||
|
} else {
|
||||||
|
rawAudioFilePath
|
||||||
|
}
|
||||||
|
Gobackend.fetchAndSaveLyrics(trackName, artistName, spotifyId, durationMs, outputPath, audioFilePath)
|
||||||
"""{"success":true}"""
|
"""{"success":true}"""
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
"""{"success":false,"error":"${e.message?.replace("\"", "'")}"}"""
|
"""{"success":false,"error":"${e.message?.replace("\"", "'")}"}"""
|
||||||
|
} finally {
|
||||||
|
if (safAudioTemp != null) {
|
||||||
|
try { File(safAudioTemp).delete() } catch (_: Exception) {}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
result.success(response)
|
result.success(response)
|
||||||
|
|||||||
+1
-1
@@ -2,4 +2,4 @@ distributionBase=GRADLE_USER_HOME
|
|||||||
distributionPath=wrapper/dists
|
distributionPath=wrapper/dists
|
||||||
zipStoreBase=GRADLE_USER_HOME
|
zipStoreBase=GRADLE_USER_HOME
|
||||||
zipStorePath=wrapper/dists
|
zipStorePath=wrapper/dists
|
||||||
distributionUrl=https\://services.gradle.org/distributions/gradle-9.3.1-all.zip
|
distributionUrl=https\://services.gradle.org/distributions/gradle-9.4.1-all.zip
|
||||||
|
|||||||
@@ -20,7 +20,7 @@ pluginManagement {
|
|||||||
plugins {
|
plugins {
|
||||||
id("dev.flutter.flutter-plugin-loader") version "1.0.0"
|
id("dev.flutter.flutter-plugin-loader") version "1.0.0"
|
||||||
id("com.android.application") version "8.13.2" apply false
|
id("com.android.application") version "8.13.2" apply false
|
||||||
id("org.jetbrains.kotlin.android") version "2.2.21" apply false
|
id("org.jetbrains.kotlin.android") version "2.3.20" apply false
|
||||||
}
|
}
|
||||||
|
|
||||||
include(":app")
|
include(":app")
|
||||||
|
|||||||
@@ -0,0 +1,611 @@
|
|||||||
|
package gobackend
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/binary"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// APEv2 tag format constants.
|
||||||
|
const (
|
||||||
|
apeTagPreamble = "APETAGEX"
|
||||||
|
apeTagHeaderSize = 32
|
||||||
|
apeTagVersion2 = 2000
|
||||||
|
apeTagFlagHeader = 1 << 29 // bit 29: this is the header, not the footer
|
||||||
|
apeTagFlagReadOnly = 1 << 0
|
||||||
|
// Item flags: bits 1-2 encode content type
|
||||||
|
apeItemFlagUTF8 = 0 << 1 // 00: UTF-8 text
|
||||||
|
apeItemFlagBinary = 1 << 1 // 01: binary data
|
||||||
|
apeItemFlagLink = 2 << 1 // 10: external link
|
||||||
|
)
|
||||||
|
|
||||||
|
// APETagItem represents a single key-value item in an APEv2 tag.
|
||||||
|
type APETagItem struct {
|
||||||
|
Key string
|
||||||
|
Value string
|
||||||
|
Flags uint32
|
||||||
|
}
|
||||||
|
|
||||||
|
// APETag represents a complete APEv2 tag block.
|
||||||
|
type APETag struct {
|
||||||
|
Version uint32
|
||||||
|
Items []APETagItem
|
||||||
|
ReadOnly bool
|
||||||
|
}
|
||||||
|
|
||||||
|
// ReadAPETags reads APEv2 tags from a file.
|
||||||
|
// APEv2 tags are typically appended at the end of the file.
|
||||||
|
// The layout is: [audio data] [APEv2 header (optional)] [items...] [APEv2 footer]
|
||||||
|
// We locate the footer first (last 32 bytes), then read the tag block.
|
||||||
|
func ReadAPETags(filePath string) (*APETag, error) {
|
||||||
|
f, err := os.Open(filePath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to open file: %w", err)
|
||||||
|
}
|
||||||
|
defer f.Close()
|
||||||
|
|
||||||
|
fi, err := f.Stat()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to stat file: %w", err)
|
||||||
|
}
|
||||||
|
fileSize := fi.Size()
|
||||||
|
|
||||||
|
if fileSize < apeTagHeaderSize {
|
||||||
|
return nil, fmt.Errorf("file too small for APE tag")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Try to find APE tag footer at the end of file.
|
||||||
|
// The footer is the last 32 bytes before any ID3v1 tag (128 bytes).
|
||||||
|
tag, err := readAPETagAtOffset(f, fileSize, fileSize-apeTagHeaderSize)
|
||||||
|
if err == nil {
|
||||||
|
return tag, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Retry: skip ID3v1 tag (128 bytes) if present
|
||||||
|
if fileSize > apeTagHeaderSize+128 {
|
||||||
|
tag, err = readAPETagAtOffset(f, fileSize, fileSize-apeTagHeaderSize-128)
|
||||||
|
if err == nil {
|
||||||
|
return tag, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, fmt.Errorf("no APEv2 tag found")
|
||||||
|
}
|
||||||
|
|
||||||
|
func readAPETagAtOffset(f *os.File, fileSize, footerOffset int64) (*APETag, error) {
|
||||||
|
if footerOffset < 0 || footerOffset+apeTagHeaderSize > fileSize {
|
||||||
|
return nil, fmt.Errorf("invalid footer offset")
|
||||||
|
}
|
||||||
|
|
||||||
|
footer := make([]byte, apeTagHeaderSize)
|
||||||
|
if _, err := f.ReadAt(footer, footerOffset); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to read APE footer: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if string(footer[0:8]) != apeTagPreamble {
|
||||||
|
return nil, fmt.Errorf("APE preamble not found")
|
||||||
|
}
|
||||||
|
|
||||||
|
version := binary.LittleEndian.Uint32(footer[8:12])
|
||||||
|
tagSize := binary.LittleEndian.Uint32(footer[12:16]) // size of items + footer (32 bytes)
|
||||||
|
itemCount := binary.LittleEndian.Uint32(footer[16:20])
|
||||||
|
flags := binary.LittleEndian.Uint32(footer[20:24])
|
||||||
|
|
||||||
|
if version != apeTagVersion2 && version != 1000 {
|
||||||
|
return nil, fmt.Errorf("unsupported APE tag version: %d", version)
|
||||||
|
}
|
||||||
|
if tagSize < apeTagHeaderSize {
|
||||||
|
return nil, fmt.Errorf("APE tag size too small: %d", tagSize)
|
||||||
|
}
|
||||||
|
if itemCount > 1000 {
|
||||||
|
return nil, fmt.Errorf("APE tag item count too large: %d", itemCount)
|
||||||
|
}
|
||||||
|
|
||||||
|
// This should be the footer (bit 29 clear)
|
||||||
|
isHeader := (flags & apeTagFlagHeader) != 0
|
||||||
|
if isHeader {
|
||||||
|
return nil, fmt.Errorf("expected APE footer but found header")
|
||||||
|
}
|
||||||
|
|
||||||
|
// tagSize includes items + footer (32 bytes), but NOT the header.
|
||||||
|
itemsSize := int64(tagSize) - apeTagHeaderSize
|
||||||
|
if itemsSize < 0 {
|
||||||
|
return nil, fmt.Errorf("invalid APE tag: items size negative")
|
||||||
|
}
|
||||||
|
|
||||||
|
itemsOffset := footerOffset - itemsSize
|
||||||
|
if itemsOffset < 0 {
|
||||||
|
return nil, fmt.Errorf("APE tag items extend before file start")
|
||||||
|
}
|
||||||
|
|
||||||
|
itemsData := make([]byte, itemsSize)
|
||||||
|
if _, err := f.ReadAt(itemsData, itemsOffset); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to read APE items: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
items, err := parseAPEItems(itemsData, int(itemCount))
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to parse APE items: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return &APETag{
|
||||||
|
Version: version,
|
||||||
|
Items: items,
|
||||||
|
ReadOnly: (flags & apeTagFlagReadOnly) != 0,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseAPEItems(data []byte, count int) ([]APETagItem, error) {
|
||||||
|
items := make([]APETagItem, 0, count)
|
||||||
|
pos := 0
|
||||||
|
|
||||||
|
for i := 0; i < count && pos < len(data); i++ {
|
||||||
|
if pos+8 > len(data) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
valueSize := int(binary.LittleEndian.Uint32(data[pos : pos+4]))
|
||||||
|
itemFlags := binary.LittleEndian.Uint32(data[pos+4 : pos+8])
|
||||||
|
pos += 8
|
||||||
|
|
||||||
|
// Key is null-terminated ASCII (2-255 bytes, case-insensitive)
|
||||||
|
keyEnd := pos
|
||||||
|
for keyEnd < len(data) && data[keyEnd] != 0 {
|
||||||
|
keyEnd++
|
||||||
|
}
|
||||||
|
if keyEnd >= len(data) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
key := string(data[pos:keyEnd])
|
||||||
|
pos = keyEnd + 1
|
||||||
|
|
||||||
|
if pos+valueSize > len(data) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
value := string(data[pos : pos+valueSize])
|
||||||
|
pos += valueSize
|
||||||
|
|
||||||
|
items = append(items, APETagItem{
|
||||||
|
Key: key,
|
||||||
|
Value: value,
|
||||||
|
Flags: itemFlags,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return items, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// WriteAPETags writes APEv2 tags to the end of a file.
|
||||||
|
// If the file already has APEv2 tags, they are replaced.
|
||||||
|
// The tag is written with both header and footer.
|
||||||
|
func WriteAPETags(filePath string, tag *APETag) error {
|
||||||
|
existingSize, err := findExistingAPETagSize(filePath)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to check existing APE tag: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
tagData, err := marshalAPETag(tag)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to marshal APE tag: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if existingSize > 0 {
|
||||||
|
fi, err := os.Stat(filePath)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to stat file: %w", err)
|
||||||
|
}
|
||||||
|
newSize := fi.Size() - int64(existingSize)
|
||||||
|
if err := os.Truncate(filePath, newSize); err != nil {
|
||||||
|
return fmt.Errorf("failed to truncate existing APE tag: %w", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
f, err := os.OpenFile(filePath, os.O_WRONLY|os.O_APPEND, 0644)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to open file for writing: %w", err)
|
||||||
|
}
|
||||||
|
defer f.Close()
|
||||||
|
|
||||||
|
if _, err := f.Write(tagData); err != nil {
|
||||||
|
return fmt.Errorf("failed to write APE tag: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// findExistingAPETagSize returns the total size of an existing APE tag
|
||||||
|
// (header + items + footer) at the end of the file, or 0 if none exists.
|
||||||
|
func findExistingAPETagSize(filePath string) (int64, error) {
|
||||||
|
f, err := os.Open(filePath)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
defer f.Close()
|
||||||
|
|
||||||
|
fi, err := f.Stat()
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
fileSize := fi.Size()
|
||||||
|
|
||||||
|
offsets := []int64{fileSize - apeTagHeaderSize}
|
||||||
|
if fileSize > apeTagHeaderSize+128 {
|
||||||
|
offsets = append(offsets, fileSize-apeTagHeaderSize-128)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, offset := range offsets {
|
||||||
|
if offset < 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
footer := make([]byte, apeTagHeaderSize)
|
||||||
|
if _, err := f.ReadAt(footer, offset); err != nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if string(footer[0:8]) != apeTagPreamble {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
flags := binary.LittleEndian.Uint32(footer[20:24])
|
||||||
|
if (flags & apeTagFlagHeader) != 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
tagSize := int64(binary.LittleEndian.Uint32(footer[12:16]))
|
||||||
|
|
||||||
|
// Check if there's also a header (tagSize only covers items + footer)
|
||||||
|
hasHeader := (flags & (1 << 31)) != 0 // bit 31 = tag contains header
|
||||||
|
totalSize := tagSize
|
||||||
|
if hasHeader {
|
||||||
|
totalSize += apeTagHeaderSize
|
||||||
|
}
|
||||||
|
|
||||||
|
// Include any trailing data after the footer (e.g. ID3v1 128-byte tag).
|
||||||
|
// When truncating, we must remove the APE tag AND everything after it.
|
||||||
|
trailingBytes := fileSize - (offset + apeTagHeaderSize)
|
||||||
|
totalSize += trailingBytes
|
||||||
|
|
||||||
|
return totalSize, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// marshalAPETag serializes an APETag into bytes (header + items + footer).
|
||||||
|
func marshalAPETag(tag *APETag) ([]byte, error) {
|
||||||
|
if tag == nil || len(tag.Items) == 0 {
|
||||||
|
return nil, fmt.Errorf("empty APE tag")
|
||||||
|
}
|
||||||
|
|
||||||
|
var itemsData []byte
|
||||||
|
for _, item := range tag.Items {
|
||||||
|
keyBytes := []byte(item.Key)
|
||||||
|
valueBytes := []byte(item.Value)
|
||||||
|
|
||||||
|
// 4 bytes: value size (LE)
|
||||||
|
sizeBuf := make([]byte, 4)
|
||||||
|
binary.LittleEndian.PutUint32(sizeBuf, uint32(len(valueBytes)))
|
||||||
|
|
||||||
|
// 4 bytes: item flags (LE)
|
||||||
|
flagsBuf := make([]byte, 4)
|
||||||
|
binary.LittleEndian.PutUint32(flagsBuf, item.Flags)
|
||||||
|
|
||||||
|
itemsData = append(itemsData, sizeBuf...)
|
||||||
|
itemsData = append(itemsData, flagsBuf...)
|
||||||
|
itemsData = append(itemsData, keyBytes...)
|
||||||
|
itemsData = append(itemsData, 0)
|
||||||
|
itemsData = append(itemsData, valueBytes...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// tagSize = items data + footer (32 bytes)
|
||||||
|
tagSize := uint32(len(itemsData) + apeTagHeaderSize)
|
||||||
|
itemCount := uint32(len(tag.Items))
|
||||||
|
|
||||||
|
version := uint32(apeTagVersion2)
|
||||||
|
if tag.Version != 0 {
|
||||||
|
version = tag.Version
|
||||||
|
}
|
||||||
|
|
||||||
|
// flags: bit 29 = 1 (is header), bit 31 = 1 (contains header)
|
||||||
|
headerFlags := uint32(apeTagFlagHeader | (1 << 31))
|
||||||
|
header := buildAPEHeaderFooter(version, tagSize, itemCount, headerFlags)
|
||||||
|
|
||||||
|
// flags: bit 29 = 0 (is footer), bit 31 = 1 (contains header)
|
||||||
|
footerFlags := uint32(1 << 31)
|
||||||
|
footer := buildAPEHeaderFooter(version, tagSize, itemCount, footerFlags)
|
||||||
|
|
||||||
|
// Final layout: header + items + footer
|
||||||
|
result := make([]byte, 0, len(header)+len(itemsData)+len(footer))
|
||||||
|
result = append(result, header...)
|
||||||
|
result = append(result, itemsData...)
|
||||||
|
result = append(result, footer...)
|
||||||
|
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func buildAPEHeaderFooter(version, tagSize, itemCount, flags uint32) []byte {
|
||||||
|
buf := make([]byte, apeTagHeaderSize)
|
||||||
|
copy(buf[0:8], apeTagPreamble)
|
||||||
|
binary.LittleEndian.PutUint32(buf[8:12], version)
|
||||||
|
binary.LittleEndian.PutUint32(buf[12:16], tagSize)
|
||||||
|
binary.LittleEndian.PutUint32(buf[16:20], itemCount)
|
||||||
|
binary.LittleEndian.PutUint32(buf[20:24], flags)
|
||||||
|
// bytes 24-31 are reserved (zeros)
|
||||||
|
return buf
|
||||||
|
}
|
||||||
|
|
||||||
|
// APETagToAudioMetadata converts an APETag to our unified AudioMetadata struct.
|
||||||
|
func APETagToAudioMetadata(tag *APETag) *AudioMetadata {
|
||||||
|
if tag == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
metadata := &AudioMetadata{}
|
||||||
|
for _, item := range tag.Items {
|
||||||
|
key := strings.ToUpper(strings.TrimSpace(item.Key))
|
||||||
|
value := strings.TrimSpace(item.Value)
|
||||||
|
if value == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
switch key {
|
||||||
|
case "TITLE":
|
||||||
|
metadata.Title = value
|
||||||
|
case "ARTIST":
|
||||||
|
metadata.Artist = value
|
||||||
|
case "ALBUM":
|
||||||
|
metadata.Album = value
|
||||||
|
case "ALBUMARTIST", "ALBUM ARTIST":
|
||||||
|
metadata.AlbumArtist = value
|
||||||
|
case "GENRE":
|
||||||
|
metadata.Genre = value
|
||||||
|
case "YEAR":
|
||||||
|
metadata.Year = value
|
||||||
|
case "DATE":
|
||||||
|
metadata.Date = value
|
||||||
|
case "TRACK", "TRACKNUMBER":
|
||||||
|
metadata.TrackNumber, metadata.TotalTracks = parseIndexPair(value)
|
||||||
|
case "DISC", "DISCNUMBER":
|
||||||
|
metadata.DiscNumber, metadata.TotalDiscs = parseIndexPair(value)
|
||||||
|
case "ISRC":
|
||||||
|
metadata.ISRC = value
|
||||||
|
case "LYRICS", "UNSYNCEDLYRICS":
|
||||||
|
if metadata.Lyrics == "" {
|
||||||
|
metadata.Lyrics = value
|
||||||
|
}
|
||||||
|
case "LABEL", "PUBLISHER":
|
||||||
|
metadata.Label = value
|
||||||
|
case "COPYRIGHT":
|
||||||
|
metadata.Copyright = value
|
||||||
|
case "COMPOSER":
|
||||||
|
metadata.Composer = value
|
||||||
|
case "COMMENT":
|
||||||
|
metadata.Comment = value
|
||||||
|
case "REPLAYGAIN_TRACK_GAIN":
|
||||||
|
metadata.ReplayGainTrackGain = value
|
||||||
|
case "REPLAYGAIN_TRACK_PEAK":
|
||||||
|
metadata.ReplayGainTrackPeak = value
|
||||||
|
case "REPLAYGAIN_ALBUM_GAIN":
|
||||||
|
metadata.ReplayGainAlbumGain = value
|
||||||
|
case "REPLAYGAIN_ALBUM_PEAK":
|
||||||
|
metadata.ReplayGainAlbumPeak = value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return metadata
|
||||||
|
}
|
||||||
|
|
||||||
|
// AudioMetadataToAPEItems converts metadata fields to APE tag items.
|
||||||
|
func AudioMetadataToAPEItems(metadata *AudioMetadata) []APETagItem {
|
||||||
|
if metadata == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var items []APETagItem
|
||||||
|
addItem := func(key, value string) {
|
||||||
|
if value != "" {
|
||||||
|
items = append(items, APETagItem{Key: key, Value: value})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
addItem("Title", metadata.Title)
|
||||||
|
addItem("Artist", metadata.Artist)
|
||||||
|
addItem("Album", metadata.Album)
|
||||||
|
addItem("Album Artist", metadata.AlbumArtist)
|
||||||
|
addItem("Genre", metadata.Genre)
|
||||||
|
if metadata.Date != "" {
|
||||||
|
addItem("Year", metadata.Date)
|
||||||
|
} else if metadata.Year != "" {
|
||||||
|
addItem("Year", metadata.Year)
|
||||||
|
}
|
||||||
|
if metadata.TrackNumber > 0 {
|
||||||
|
addItem("Track", formatIndexValue(metadata.TrackNumber, metadata.TotalTracks))
|
||||||
|
}
|
||||||
|
if metadata.DiscNumber > 0 {
|
||||||
|
addItem("Disc", formatIndexValue(metadata.DiscNumber, metadata.TotalDiscs))
|
||||||
|
}
|
||||||
|
addItem("ISRC", metadata.ISRC)
|
||||||
|
addItem("Lyrics", metadata.Lyrics)
|
||||||
|
addItem("Label", metadata.Label)
|
||||||
|
addItem("Copyright", metadata.Copyright)
|
||||||
|
addItem("Composer", metadata.Composer)
|
||||||
|
addItem("Comment", metadata.Comment)
|
||||||
|
addItem("REPLAYGAIN_TRACK_GAIN", metadata.ReplayGainTrackGain)
|
||||||
|
addItem("REPLAYGAIN_TRACK_PEAK", metadata.ReplayGainTrackPeak)
|
||||||
|
addItem("REPLAYGAIN_ALBUM_GAIN", metadata.ReplayGainAlbumGain)
|
||||||
|
addItem("REPLAYGAIN_ALBUM_PEAK", metadata.ReplayGainAlbumPeak)
|
||||||
|
|
||||||
|
return items
|
||||||
|
}
|
||||||
|
|
||||||
|
// apeKeysFromFields builds a set of upper-case APE tag keys corresponding to
|
||||||
|
// the metadata fields map sent by the editor. This is used during merge to
|
||||||
|
// ensure that even empty (cleared) fields override old values.
|
||||||
|
func apeKeysFromFields(fields map[string]string) map[string]struct{} {
|
||||||
|
mapping := map[string]string{
|
||||||
|
"title": "TITLE",
|
||||||
|
"artist": "ARTIST",
|
||||||
|
"album": "ALBUM",
|
||||||
|
"album_artist": "ALBUM ARTIST",
|
||||||
|
"date": "DATE",
|
||||||
|
"genre": "GENRE",
|
||||||
|
"track_number": "TRACK",
|
||||||
|
"disc_number": "DISC",
|
||||||
|
"isrc": "ISRC",
|
||||||
|
"lyrics": "LYRICS",
|
||||||
|
"label": "LABEL",
|
||||||
|
"copyright": "COPYRIGHT",
|
||||||
|
"composer": "COMPOSER",
|
||||||
|
"comment": "COMMENT",
|
||||||
|
"replaygain_track_gain": "REPLAYGAIN_TRACK_GAIN",
|
||||||
|
"replaygain_track_peak": "REPLAYGAIN_TRACK_PEAK",
|
||||||
|
"replaygain_album_gain": "REPLAYGAIN_ALBUM_GAIN",
|
||||||
|
"replaygain_album_peak": "REPLAYGAIN_ALBUM_PEAK",
|
||||||
|
}
|
||||||
|
result := make(map[string]struct{})
|
||||||
|
for fk, apeKey := range mapping {
|
||||||
|
if _, present := fields[fk]; present {
|
||||||
|
result[strings.ToUpper(apeKey)] = struct{}{}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Some fields have reader aliases that must also be cleared when the
|
||||||
|
// canonical key is updated (e.g. DATE writer ↔ DATE/YEAR reader,
|
||||||
|
// DISC ↔ DISCNUMBER, TRACK ↔ TRACKNUMBER, "ALBUM ARTIST" ↔ ALBUMARTIST,
|
||||||
|
// LABEL ↔ PUBLISHER, LYRICS ↔ UNSYNCEDLYRICS).
|
||||||
|
if _, present := fields["date"]; present {
|
||||||
|
result["DATE"] = struct{}{}
|
||||||
|
}
|
||||||
|
if _, present := fields["disc_number"]; present {
|
||||||
|
result["DISCNUMBER"] = struct{}{}
|
||||||
|
}
|
||||||
|
if _, present := fields["disc_total"]; present {
|
||||||
|
result["DISCNUMBER"] = struct{}{}
|
||||||
|
}
|
||||||
|
if _, present := fields["track_number"]; present {
|
||||||
|
result["TRACKNUMBER"] = struct{}{}
|
||||||
|
}
|
||||||
|
if _, present := fields["track_total"]; present {
|
||||||
|
result["TRACKNUMBER"] = struct{}{}
|
||||||
|
}
|
||||||
|
if _, present := fields["album_artist"]; present {
|
||||||
|
result["ALBUMARTIST"] = struct{}{}
|
||||||
|
}
|
||||||
|
if _, present := fields["label"]; present {
|
||||||
|
result["PUBLISHER"] = struct{}{}
|
||||||
|
}
|
||||||
|
if _, present := fields["lyrics"]; present {
|
||||||
|
result["UNSYNCEDLYRICS"] = struct{}{}
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
// MergeAPEItems overlays newItems on top of existing items.
|
||||||
|
// For each new item, if a matching key exists (case-insensitive) in existing,
|
||||||
|
// it is replaced. New keys are appended. Existing items whose keys are NOT
|
||||||
|
// in newItems are preserved (cover art, ReplayGain, custom tags, etc.).
|
||||||
|
//
|
||||||
|
// overrideKeys is an optional set of upper-case keys that should be removed
|
||||||
|
// from existing even if they do not appear in newItems. This handles field
|
||||||
|
// deletion: the caller sends an empty value which is not serialized into
|
||||||
|
// newItems, but the old value must still be dropped.
|
||||||
|
func MergeAPEItems(existing, newItems []APETagItem, overrideKeys map[string]struct{}) []APETagItem {
|
||||||
|
// Build a set of keys being updated (upper-case for case-insensitive match)
|
||||||
|
combined := make(map[string]struct{}, len(newItems)+len(overrideKeys))
|
||||||
|
for k := range overrideKeys {
|
||||||
|
combined[strings.ToUpper(k)] = struct{}{}
|
||||||
|
}
|
||||||
|
for _, item := range newItems {
|
||||||
|
combined[strings.ToUpper(item.Key)] = struct{}{}
|
||||||
|
}
|
||||||
|
|
||||||
|
var merged []APETagItem
|
||||||
|
for _, item := range existing {
|
||||||
|
if _, overwritten := combined[strings.ToUpper(item.Key)]; !overwritten {
|
||||||
|
merged = append(merged, item)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
merged = append(merged, newItems...)
|
||||||
|
|
||||||
|
return merged
|
||||||
|
}
|
||||||
|
|
||||||
|
// ReadAPETagsFromReader reads APEv2 tags from an io.ReaderAt + size.
|
||||||
|
// This is useful for reading APE tags from files opened via SAF or other abstractions.
|
||||||
|
func ReadAPETagsFromReader(r io.ReaderAt, fileSize int64) (*APETag, error) {
|
||||||
|
if fileSize < apeTagHeaderSize {
|
||||||
|
return nil, fmt.Errorf("file too small for APE tag")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Try footer at end of file
|
||||||
|
footer := make([]byte, apeTagHeaderSize)
|
||||||
|
if _, err := r.ReadAt(footer, fileSize-apeTagHeaderSize); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to read APE footer: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if string(footer[0:8]) == apeTagPreamble {
|
||||||
|
tag, err := parseAPETagFromFooter(r, fileSize, fileSize-apeTagHeaderSize, footer)
|
||||||
|
if err == nil {
|
||||||
|
return tag, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Retry: skip ID3v1 tag (128 bytes)
|
||||||
|
if fileSize > apeTagHeaderSize+128 {
|
||||||
|
offset := fileSize - apeTagHeaderSize - 128
|
||||||
|
if _, err := r.ReadAt(footer, offset); err == nil {
|
||||||
|
if string(footer[0:8]) == apeTagPreamble {
|
||||||
|
tag, err := parseAPETagFromFooter(r, fileSize, offset, footer)
|
||||||
|
if err == nil {
|
||||||
|
return tag, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, fmt.Errorf("no APEv2 tag found")
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseAPETagFromFooter(r io.ReaderAt, fileSize, footerOffset int64, footer []byte) (*APETag, error) {
|
||||||
|
version := binary.LittleEndian.Uint32(footer[8:12])
|
||||||
|
tagSize := binary.LittleEndian.Uint32(footer[12:16])
|
||||||
|
itemCount := binary.LittleEndian.Uint32(footer[16:20])
|
||||||
|
flags := binary.LittleEndian.Uint32(footer[20:24])
|
||||||
|
|
||||||
|
if version != apeTagVersion2 && version != 1000 {
|
||||||
|
return nil, fmt.Errorf("unsupported APE tag version: %d", version)
|
||||||
|
}
|
||||||
|
if tagSize < apeTagHeaderSize {
|
||||||
|
return nil, fmt.Errorf("APE tag size too small: %d", tagSize)
|
||||||
|
}
|
||||||
|
if itemCount > 1000 {
|
||||||
|
return nil, fmt.Errorf("APE tag item count too large: %d", itemCount)
|
||||||
|
}
|
||||||
|
if (flags & apeTagFlagHeader) != 0 {
|
||||||
|
return nil, fmt.Errorf("expected footer, found header")
|
||||||
|
}
|
||||||
|
|
||||||
|
itemsSize := int64(tagSize) - apeTagHeaderSize
|
||||||
|
itemsOffset := footerOffset - itemsSize
|
||||||
|
if itemsOffset < 0 {
|
||||||
|
return nil, fmt.Errorf("APE items extend before file start")
|
||||||
|
}
|
||||||
|
|
||||||
|
itemsData := make([]byte, itemsSize)
|
||||||
|
if _, err := r.ReadAt(itemsData, itemsOffset); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to read APE items: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
items, err := parseAPEItems(itemsData, int(itemCount))
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to parse APE items: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return &APETag{
|
||||||
|
Version: version,
|
||||||
|
Items: items,
|
||||||
|
ReadOnly: (flags & apeTagFlagReadOnly) != 0,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
@@ -21,13 +21,20 @@ type AudioMetadata struct {
|
|||||||
Year string
|
Year string
|
||||||
Date string
|
Date string
|
||||||
TrackNumber int
|
TrackNumber int
|
||||||
|
TotalTracks int
|
||||||
DiscNumber int
|
DiscNumber int
|
||||||
|
TotalDiscs int
|
||||||
ISRC string
|
ISRC string
|
||||||
Lyrics string
|
Lyrics string
|
||||||
Label string
|
Label string
|
||||||
Copyright string
|
Copyright string
|
||||||
Composer string
|
Composer string
|
||||||
Comment string
|
Comment string
|
||||||
|
// ReplayGain fields (text values, e.g. "-6.50 dB", "0.988831")
|
||||||
|
ReplayGainTrackGain string
|
||||||
|
ReplayGainTrackPeak string
|
||||||
|
ReplayGainAlbumGain string
|
||||||
|
ReplayGainAlbumPeak string
|
||||||
}
|
}
|
||||||
|
|
||||||
type MP3Quality struct {
|
type MP3Quality struct {
|
||||||
@@ -168,9 +175,9 @@ func parseID3v22Frames(data []byte, metadata *AudioMetadata, tagUnsync bool) {
|
|||||||
case "TCO":
|
case "TCO":
|
||||||
metadata.Genre = cleanGenre(value)
|
metadata.Genre = cleanGenre(value)
|
||||||
case "TRK":
|
case "TRK":
|
||||||
metadata.TrackNumber = parseTrackNumber(value)
|
metadata.TrackNumber, metadata.TotalTracks = parseIndexPair(value)
|
||||||
case "TPA":
|
case "TPA":
|
||||||
metadata.DiscNumber = parseTrackNumber(value)
|
metadata.DiscNumber, metadata.TotalDiscs = parseIndexPair(value)
|
||||||
case "TCM":
|
case "TCM":
|
||||||
metadata.Composer = value
|
metadata.Composer = value
|
||||||
case "TPB":
|
case "TPB":
|
||||||
@@ -287,9 +294,9 @@ func parseID3v23Frames(data []byte, metadata *AudioMetadata, version byte, tagUn
|
|||||||
case "TCON":
|
case "TCON":
|
||||||
metadata.Genre = cleanGenre(value)
|
metadata.Genre = cleanGenre(value)
|
||||||
case "TRCK":
|
case "TRCK":
|
||||||
metadata.TrackNumber = parseTrackNumber(value)
|
metadata.TrackNumber, metadata.TotalTracks = parseIndexPair(value)
|
||||||
case "TPOS":
|
case "TPOS":
|
||||||
metadata.DiscNumber = parseTrackNumber(value)
|
metadata.DiscNumber, metadata.TotalDiscs = parseIndexPair(value)
|
||||||
case "TSRC":
|
case "TSRC":
|
||||||
metadata.ISRC = value
|
metadata.ISRC = value
|
||||||
case "TCOM":
|
case "TCOM":
|
||||||
@@ -311,6 +318,17 @@ func parseID3v23Frames(data []byte, metadata *AudioMetadata, version byte, tagUn
|
|||||||
if isLyricsDescription(desc) && userValue != "" && metadata.Lyrics == "" {
|
if isLyricsDescription(desc) && userValue != "" && metadata.Lyrics == "" {
|
||||||
metadata.Lyrics = userValue
|
metadata.Lyrics = userValue
|
||||||
}
|
}
|
||||||
|
upperDesc := strings.ToUpper(desc)
|
||||||
|
switch upperDesc {
|
||||||
|
case "REPLAYGAIN_TRACK_GAIN":
|
||||||
|
metadata.ReplayGainTrackGain = userValue
|
||||||
|
case "REPLAYGAIN_TRACK_PEAK":
|
||||||
|
metadata.ReplayGainTrackPeak = userValue
|
||||||
|
case "REPLAYGAIN_ALBUM_GAIN":
|
||||||
|
metadata.ReplayGainAlbumGain = userValue
|
||||||
|
case "REPLAYGAIN_ALBUM_PEAK":
|
||||||
|
metadata.ReplayGainAlbumPeak = userValue
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pos += 10 + frameSize
|
pos += 10 + frameSize
|
||||||
@@ -338,7 +356,6 @@ func readID3v1(file *os.File) (*AudioMetadata, error) {
|
|||||||
Year: strings.TrimRight(string(tag[93:97]), " \x00"),
|
Year: strings.TrimRight(string(tag[93:97]), " \x00"),
|
||||||
}
|
}
|
||||||
|
|
||||||
// ID3v1.1 track number (if byte 125 is 0 and byte 126 is not)
|
|
||||||
if tag[125] == 0 && tag[126] != 0 {
|
if tag[125] == 0 && tag[126] != 0 {
|
||||||
metadata.TrackNumber = int(tag[126])
|
metadata.TrackNumber = int(tag[126])
|
||||||
}
|
}
|
||||||
@@ -373,27 +390,23 @@ func extractTextFrame(data []byte) string {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// extractCommentFrame parses an ID3v2 COMM frame.
|
|
||||||
// Format: encoding(1) + language(3) + description(null-terminated) + text
|
|
||||||
func extractCommentFrame(data []byte) string {
|
func extractCommentFrame(data []byte) string {
|
||||||
if len(data) < 5 {
|
if len(data) < 5 {
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
encoding := data[0]
|
encoding := data[0]
|
||||||
// skip 3-byte language code
|
|
||||||
rest := data[4:]
|
rest := data[4:]
|
||||||
|
|
||||||
// find null terminator separating description from text
|
|
||||||
var text []byte
|
var text []byte
|
||||||
switch encoding {
|
switch encoding {
|
||||||
case 1, 2: // UTF-16 variants use double-null terminator
|
case 1, 2:
|
||||||
for i := 0; i+1 < len(rest); i += 2 {
|
for i := 0; i+1 < len(rest); i += 2 {
|
||||||
if rest[i] == 0 && rest[i+1] == 0 {
|
if rest[i] == 0 && rest[i+1] == 0 {
|
||||||
text = rest[i+2:]
|
text = rest[i+2:]
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
default: // ISO-8859-1 or UTF-8
|
default:
|
||||||
idx := bytes.IndexByte(rest, 0)
|
idx := bytes.IndexByte(rest, 0)
|
||||||
if idx >= 0 && idx+1 < len(rest) {
|
if idx >= 0 && idx+1 < len(rest) {
|
||||||
text = rest[idx+1:]
|
text = rest[idx+1:]
|
||||||
@@ -406,33 +419,30 @@ func extractCommentFrame(data []byte) string {
|
|||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
// re-prepend encoding byte so extractTextFrame can decode properly
|
|
||||||
framed := make([]byte, 1+len(text))
|
framed := make([]byte, 1+len(text))
|
||||||
framed[0] = encoding
|
framed[0] = encoding
|
||||||
copy(framed[1:], text)
|
copy(framed[1:], text)
|
||||||
return extractTextFrame(framed)
|
return extractTextFrame(framed)
|
||||||
}
|
}
|
||||||
|
|
||||||
// extractLyricsFrame parses ID3 unsynchronized lyrics frames (USLT/ULT).
|
|
||||||
// Format: encoding(1) + language(3) + description(null-terminated) + lyrics text.
|
|
||||||
func extractLyricsFrame(data []byte) string {
|
func extractLyricsFrame(data []byte) string {
|
||||||
if len(data) < 5 {
|
if len(data) < 5 {
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
encoding := data[0]
|
encoding := data[0]
|
||||||
rest := data[4:] // skip 3-byte language code
|
rest := data[4:]
|
||||||
|
|
||||||
var text []byte
|
var text []byte
|
||||||
switch encoding {
|
switch encoding {
|
||||||
case 1, 2: // UTF-16 variants use double-null terminator
|
case 1, 2:
|
||||||
for i := 0; i+1 < len(rest); i += 2 {
|
for i := 0; i+1 < len(rest); i += 2 {
|
||||||
if rest[i] == 0 && rest[i+1] == 0 {
|
if rest[i] == 0 && rest[i+1] == 0 {
|
||||||
text = rest[i+2:]
|
text = rest[i+2:]
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
default: // ISO-8859-1 or UTF-8
|
default:
|
||||||
idx := bytes.IndexByte(rest, 0)
|
idx := bytes.IndexByte(rest, 0)
|
||||||
if idx >= 0 && idx+1 < len(rest) {
|
if idx >= 0 && idx+1 < len(rest) {
|
||||||
text = rest[idx+1:]
|
text = rest[idx+1:]
|
||||||
@@ -451,8 +461,6 @@ func extractLyricsFrame(data []byte) string {
|
|||||||
return extractTextFrame(framed)
|
return extractTextFrame(framed)
|
||||||
}
|
}
|
||||||
|
|
||||||
// extractUserTextFrame parses ID3 TXXX/TXX user text frame:
|
|
||||||
// encoding(1) + description + separator + value.
|
|
||||||
func extractUserTextFrame(data []byte) (string, string) {
|
func extractUserTextFrame(data []byte) (string, string) {
|
||||||
if len(data) < 2 {
|
if len(data) < 2 {
|
||||||
return "", ""
|
return "", ""
|
||||||
@@ -463,7 +471,7 @@ func extractUserTextFrame(data []byte) (string, string) {
|
|||||||
|
|
||||||
var descRaw, valueRaw []byte
|
var descRaw, valueRaw []byte
|
||||||
switch encoding {
|
switch encoding {
|
||||||
case 1, 2: // UTF-16 variants
|
case 1, 2:
|
||||||
for i := 0; i+1 < len(payload); i += 2 {
|
for i := 0; i+1 < len(payload); i += 2 {
|
||||||
if payload[i] == 0 && payload[i+1] == 0 {
|
if payload[i] == 0 && payload[i+1] == 0 {
|
||||||
descRaw = payload[:i]
|
descRaw = payload[:i]
|
||||||
@@ -471,7 +479,7 @@ func extractUserTextFrame(data []byte) (string, string) {
|
|||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
default: // ISO-8859-1 or UTF-8
|
default:
|
||||||
idx := bytes.IndexByte(payload, 0)
|
idx := bytes.IndexByte(payload, 0)
|
||||||
if idx >= 0 {
|
if idx >= 0 {
|
||||||
descRaw = payload[:idx]
|
descRaw = payload[:idx]
|
||||||
@@ -574,14 +582,28 @@ func cleanGenre(genre string) string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func parseTrackNumber(s string) int {
|
func parseTrackNumber(s string) int {
|
||||||
s = strings.TrimSpace(s)
|
num, _ := parseIndexPair(s)
|
||||||
if idx := strings.Index(s, "/"); idx > 0 {
|
|
||||||
s = s[:idx]
|
|
||||||
}
|
|
||||||
num, _ := strconv.Atoi(s)
|
|
||||||
return num
|
return num
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func parseIndexPair(s string) (int, int) {
|
||||||
|
s = strings.TrimSpace(s)
|
||||||
|
if s == "" {
|
||||||
|
return 0, 0
|
||||||
|
}
|
||||||
|
|
||||||
|
first := s
|
||||||
|
second := ""
|
||||||
|
if idx := strings.Index(s, "/"); idx > 0 {
|
||||||
|
first = s[:idx]
|
||||||
|
second = s[idx+1:]
|
||||||
|
}
|
||||||
|
|
||||||
|
num, _ := strconv.Atoi(strings.TrimSpace(first))
|
||||||
|
total, _ := strconv.Atoi(strings.TrimSpace(second))
|
||||||
|
return num, total
|
||||||
|
}
|
||||||
|
|
||||||
func removeUnsync(data []byte) []byte {
|
func removeUnsync(data []byte) []byte {
|
||||||
if len(data) == 0 {
|
if len(data) == 0 {
|
||||||
return data
|
return data
|
||||||
@@ -665,7 +687,6 @@ func GetMP3Quality(filePath string) (*MP3Quality, error) {
|
|||||||
|
|
||||||
file.Seek(audioStart, io.SeekStart)
|
file.Seek(audioStart, io.SeekStart)
|
||||||
|
|
||||||
// Find first valid MP3 frame sync
|
|
||||||
frameHeader := make([]byte, 4)
|
frameHeader := make([]byte, 4)
|
||||||
var frameStart int64 = -1
|
var frameStart int64 = -1
|
||||||
for i := 0; i < 10000; i++ {
|
for i := 0; i < 10000; i++ {
|
||||||
@@ -692,8 +713,6 @@ func GetMP3Quality(filePath string) (*MP3Quality, error) {
|
|||||||
sampleRateIdx := (frameHeader[2] >> 2) & 0x03
|
sampleRateIdx := (frameHeader[2] >> 2) & 0x03
|
||||||
channelMode := (frameHeader[3] >> 6) & 0x03
|
channelMode := (frameHeader[3] >> 6) & 0x03
|
||||||
|
|
||||||
// Sample rate tables: [version][index]
|
|
||||||
// version: 0=MPEG2.5, 1=reserved, 2=MPEG2, 3=MPEG1
|
|
||||||
sampleRates := [][]int{
|
sampleRates := [][]int{
|
||||||
{11025, 12000, 8000},
|
{11025, 12000, 8000},
|
||||||
{0, 0, 0},
|
{0, 0, 0},
|
||||||
@@ -704,15 +723,12 @@ func GetMP3Quality(filePath string) (*MP3Quality, error) {
|
|||||||
quality.SampleRate = sampleRates[version][sampleRateIdx]
|
quality.SampleRate = sampleRates[version][sampleRateIdx]
|
||||||
}
|
}
|
||||||
|
|
||||||
// Bitrate tables for all MPEG versions and layers
|
|
||||||
// MPEG1 Layer III
|
|
||||||
if version == 3 && layer == 1 {
|
if version == 3 && layer == 1 {
|
||||||
bitrates := []int{0, 32, 40, 48, 56, 64, 80, 96, 112, 128, 160, 192, 224, 256, 320, 0}
|
bitrates := []int{0, 32, 40, 48, 56, 64, 80, 96, 112, 128, 160, 192, 224, 256, 320, 0}
|
||||||
if bitrateIdx < 16 {
|
if bitrateIdx < 16 {
|
||||||
quality.Bitrate = bitrates[bitrateIdx] * 1000
|
quality.Bitrate = bitrates[bitrateIdx] * 1000
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// MPEG2/2.5 Layer III
|
|
||||||
if (version == 0 || version == 2) && layer == 1 {
|
if (version == 0 || version == 2) && layer == 1 {
|
||||||
bitrates := []int{0, 8, 16, 24, 32, 40, 48, 56, 64, 80, 96, 112, 128, 144, 160, 0}
|
bitrates := []int{0, 8, 16, 24, 32, 40, 48, 56, 64, 80, 96, 112, 128, 144, 160, 0}
|
||||||
if bitrateIdx < 16 {
|
if bitrateIdx < 16 {
|
||||||
@@ -720,14 +736,11 @@ func GetMP3Quality(filePath string) (*MP3Quality, error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Determine samples per frame for duration calculation
|
|
||||||
samplesPerFrame := 1152 // MPEG1 Layer III
|
samplesPerFrame := 1152 // MPEG1 Layer III
|
||||||
if version == 0 || version == 2 {
|
if version == 0 || version == 2 {
|
||||||
samplesPerFrame = 576 // MPEG2/2.5 Layer III
|
samplesPerFrame = 576 // MPEG2/2.5 Layer III
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try to read Xing/VBRI header from the first frame for VBR info
|
|
||||||
// Xing header offset depends on MPEG version and channel mode
|
|
||||||
var xingOffset int
|
var xingOffset int
|
||||||
if version == 3 { // MPEG1
|
if version == 3 { // MPEG1
|
||||||
if channelMode == 3 { // Mono
|
if channelMode == 3 { // Mono
|
||||||
@@ -743,7 +756,6 @@ func GetMP3Quality(filePath string) (*MP3Quality, error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Read enough of the first frame to find Xing/VBRI header
|
|
||||||
xingBuf := make([]byte, 200)
|
xingBuf := make([]byte, 200)
|
||||||
file.Seek(frameStart+4, io.SeekStart)
|
file.Seek(frameStart+4, io.SeekStart)
|
||||||
n, _ := io.ReadFull(file, xingBuf)
|
n, _ := io.ReadFull(file, xingBuf)
|
||||||
@@ -753,7 +765,6 @@ func GetMP3Quality(filePath string) (*MP3Quality, error) {
|
|||||||
vbrBytes := int64(0)
|
vbrBytes := int64(0)
|
||||||
isVBR := false
|
isVBR := false
|
||||||
|
|
||||||
// Check for Xing/Info header
|
|
||||||
if xingOffset+8 <= n {
|
if xingOffset+8 <= n {
|
||||||
tag := string(xingBuf[xingOffset : xingOffset+4])
|
tag := string(xingBuf[xingOffset : xingOffset+4])
|
||||||
if tag == "Xing" || tag == "Info" {
|
if tag == "Xing" || tag == "Info" {
|
||||||
@@ -772,7 +783,6 @@ func GetMP3Quality(filePath string) (*MP3Quality, error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check for VBRI header (always at offset 32 from frame start + 4)
|
|
||||||
if !isVBR && 36+26 <= n {
|
if !isVBR && 36+26 <= n {
|
||||||
if string(xingBuf[32:36]) == "VBRI" {
|
if string(xingBuf[32:36]) == "VBRI" {
|
||||||
vbrBytes = int64(binary.BigEndian.Uint32(xingBuf[36+6 : 36+10]))
|
vbrBytes = int64(binary.BigEndian.Uint32(xingBuf[36+6 : 36+10]))
|
||||||
@@ -784,11 +794,9 @@ func GetMP3Quality(filePath string) (*MP3Quality, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if isVBR && vbrFrames > 0 && quality.SampleRate > 0 {
|
if isVBR && vbrFrames > 0 && quality.SampleRate > 0 {
|
||||||
// Accurate duration from total frames
|
|
||||||
totalSamples := int64(vbrFrames) * int64(samplesPerFrame)
|
totalSamples := int64(vbrFrames) * int64(samplesPerFrame)
|
||||||
quality.Duration = int(totalSamples / int64(quality.SampleRate))
|
quality.Duration = int(totalSamples / int64(quality.SampleRate))
|
||||||
|
|
||||||
// Accurate average bitrate
|
|
||||||
if vbrBytes > 0 && quality.Duration > 0 {
|
if vbrBytes > 0 && quality.Duration > 0 {
|
||||||
quality.Bitrate = int(vbrBytes * 8 / int64(quality.Duration))
|
quality.Bitrate = int(vbrBytes * 8 / int64(quality.Duration))
|
||||||
} else if quality.Duration > 0 {
|
} else if quality.Duration > 0 {
|
||||||
@@ -796,7 +804,6 @@ func GetMP3Quality(filePath string) (*MP3Quality, error) {
|
|||||||
quality.Bitrate = int(audioSize * 8 / int64(quality.Duration))
|
quality.Bitrate = int(audioSize * 8 / int64(quality.Duration))
|
||||||
}
|
}
|
||||||
} else if quality.Bitrate > 0 {
|
} else if quality.Bitrate > 0 {
|
||||||
// CBR fallback: estimate duration from file size and frame bitrate
|
|
||||||
audioSize := fileSize - audioStart - 128 // subtract possible ID3v1 tag
|
audioSize := fileSize - audioStart - 128 // subtract possible ID3v1 tag
|
||||||
if audioSize > 0 {
|
if audioSize > 0 {
|
||||||
quality.Duration = int(audioSize * 8 / int64(quality.Bitrate))
|
quality.Duration = int(audioSize * 8 / int64(quality.Bitrate))
|
||||||
@@ -983,7 +990,6 @@ func parseVorbisComments(data []byte, metadata *AudioMetadata) {
|
|||||||
artistValues := make([]string, 0, 1)
|
artistValues := make([]string, 0, 1)
|
||||||
albumArtistValues := make([]string, 0, 1)
|
albumArtistValues := make([]string, 0, 1)
|
||||||
|
|
||||||
// Read vendor string length
|
|
||||||
var vendorLen uint32
|
var vendorLen uint32
|
||||||
if err := binary.Read(reader, binary.LittleEndian, &vendorLen); err != nil {
|
if err := binary.Read(reader, binary.LittleEndian, &vendorLen); err != nil {
|
||||||
return
|
return
|
||||||
@@ -1012,8 +1018,6 @@ func parseVorbisComments(data []byte, metadata *AudioMetadata) {
|
|||||||
if commentLen > remaining {
|
if commentLen > remaining {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
// Large comment entries are typically METADATA_BLOCK_PICTURE.
|
|
||||||
// Skip them so we can continue parsing normal text tags after/before.
|
|
||||||
if commentLen > 512*1024 {
|
if commentLen > 512*1024 {
|
||||||
reader.Seek(int64(commentLen), io.SeekCurrent)
|
reader.Seek(int64(commentLen), io.SeekCurrent)
|
||||||
continue
|
continue
|
||||||
@@ -1049,9 +1053,9 @@ func parseVorbisComments(data []byte, metadata *AudioMetadata) {
|
|||||||
case "GENRE":
|
case "GENRE":
|
||||||
metadata.Genre = value
|
metadata.Genre = value
|
||||||
case "TRACKNUMBER", "TRACK":
|
case "TRACKNUMBER", "TRACK":
|
||||||
metadata.TrackNumber = parseTrackNumber(value)
|
metadata.TrackNumber, metadata.TotalTracks = parseIndexPair(value)
|
||||||
case "DISCNUMBER", "DISC":
|
case "DISCNUMBER", "DISC":
|
||||||
metadata.DiscNumber = parseTrackNumber(value)
|
metadata.DiscNumber, metadata.TotalDiscs = parseIndexPair(value)
|
||||||
case "ISRC":
|
case "ISRC":
|
||||||
metadata.ISRC = value
|
metadata.ISRC = value
|
||||||
case "COMPOSER":
|
case "COMPOSER":
|
||||||
@@ -1066,6 +1070,14 @@ func parseVorbisComments(data []byte, metadata *AudioMetadata) {
|
|||||||
metadata.Label = value
|
metadata.Label = value
|
||||||
case "COPYRIGHT":
|
case "COPYRIGHT":
|
||||||
metadata.Copyright = value
|
metadata.Copyright = value
|
||||||
|
case "REPLAYGAIN_TRACK_GAIN":
|
||||||
|
metadata.ReplayGainTrackGain = value
|
||||||
|
case "REPLAYGAIN_TRACK_PEAK":
|
||||||
|
metadata.ReplayGainTrackPeak = value
|
||||||
|
case "REPLAYGAIN_ALBUM_GAIN":
|
||||||
|
metadata.ReplayGainAlbumGain = value
|
||||||
|
case "REPLAYGAIN_ALBUM_PEAK":
|
||||||
|
metadata.ReplayGainAlbumPeak = value
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1123,7 +1135,6 @@ func GetOggQuality(filePath string) (*OggQuality, error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Read granule position from the last Ogg page for accurate duration
|
|
||||||
stat, err := file.Stat()
|
stat, err := file.Stat()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return quality, nil
|
return quality, nil
|
||||||
@@ -1133,7 +1144,6 @@ func GetOggQuality(filePath string) (*OggQuality, error) {
|
|||||||
granule := readLastOggGranulePosition(file, fileSize)
|
granule := readLastOggGranulePosition(file, fileSize)
|
||||||
if granule > 0 {
|
if granule > 0 {
|
||||||
if isOpus {
|
if isOpus {
|
||||||
// Opus always uses 48kHz granule position internally
|
|
||||||
totalSamples := granule - int64(preSkip)
|
totalSamples := granule - int64(preSkip)
|
||||||
if totalSamples > 0 {
|
if totalSamples > 0 {
|
||||||
durationSec := float64(totalSamples) / 48000.0
|
durationSec := float64(totalSamples) / 48000.0
|
||||||
@@ -1151,11 +1161,9 @@ func GetOggQuality(filePath string) (*OggQuality, error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fallback bitrate estimate if duration exists but bitrate couldn't be derived.
|
|
||||||
if quality.Bitrate <= 0 && quality.Duration > 0 {
|
if quality.Bitrate <= 0 && quality.Duration > 0 {
|
||||||
quality.Bitrate = int(fileSize * 8 / int64(quality.Duration))
|
quality.Bitrate = int(fileSize * 8 / int64(quality.Duration))
|
||||||
}
|
}
|
||||||
// Guard against obviously invalid values from corrupted/unreliable granule reads.
|
|
||||||
if quality.Duration > 24*60*60 {
|
if quality.Duration > 24*60*60 {
|
||||||
quality.Duration = 0
|
quality.Duration = 0
|
||||||
quality.Bitrate = 0
|
quality.Bitrate = 0
|
||||||
@@ -1167,10 +1175,7 @@ func GetOggQuality(filePath string) (*OggQuality, error) {
|
|||||||
return quality, nil
|
return quality, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// readLastOggGranulePosition seeks to the end of the file and scans backwards
|
|
||||||
// to find the last Ogg page, then reads its granule position (bytes 6-13).
|
|
||||||
func readLastOggGranulePosition(file *os.File, fileSize int64) int64 {
|
func readLastOggGranulePosition(file *os.File, fileSize int64) int64 {
|
||||||
// Read the last chunk of the file to find the last OggS sync
|
|
||||||
searchSize := int64(65536)
|
searchSize := int64(65536)
|
||||||
if searchSize > fileSize {
|
if searchSize > fileSize {
|
||||||
searchSize = fileSize
|
searchSize = fileSize
|
||||||
@@ -1194,7 +1199,6 @@ func readLastOggGranulePosition(file *os.File, fileSize int64) int64 {
|
|||||||
if i+27 > n {
|
if i+27 > n {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
// Validate minimal header fields to avoid false positives inside payload bytes.
|
|
||||||
version := buf[i+4]
|
version := buf[i+4]
|
||||||
headerType := buf[i+5]
|
headerType := buf[i+5]
|
||||||
if version != 0 || headerType > 0x07 {
|
if version != 0 || headerType > 0x07 {
|
||||||
@@ -1212,7 +1216,6 @@ func readLastOggGranulePosition(file *os.File, fileSize int64) int64 {
|
|||||||
if i+headerLen+payloadLen > n {
|
if i+headerLen+payloadLen > n {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
// Granule position is at bytes 6-13 of the Ogg page header (little-endian int64).
|
|
||||||
return int64(binary.LittleEndian.Uint64(buf[i+6 : i+14]))
|
return int64(binary.LittleEndian.Uint64(buf[i+6 : i+14]))
|
||||||
}
|
}
|
||||||
return 0
|
return 0
|
||||||
@@ -1272,7 +1275,6 @@ func extractMP3CoverArt(filePath string) ([]byte, string, error) {
|
|||||||
return nil, "", err
|
return nil, "", err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Parse frames looking for APIC (Attached Picture)
|
|
||||||
pos := 0
|
pos := 0
|
||||||
var frameIDLen, headerLen int
|
var frameIDLen, headerLen int
|
||||||
if majorVersion == 2 {
|
if majorVersion == 2 {
|
||||||
@@ -1303,7 +1305,6 @@ func extractMP3CoverArt(filePath string) ([]byte, string, error) {
|
|||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check for APIC (ID3v2.3/2.4) or PIC (ID3v2.2)
|
|
||||||
if (frameIDLen == 4 && frameID == "APIC") || (frameIDLen == 3 && frameID == "PIC") {
|
if (frameIDLen == 4 && frameID == "APIC") || (frameIDLen == 3 && frameID == "PIC") {
|
||||||
frameData := tagData[pos+headerLen : pos+headerLen+frameSize]
|
frameData := tagData[pos+headerLen : pos+headerLen+frameSize]
|
||||||
imageData, mimeType := parseAPICFrame(frameData, majorVersion)
|
imageData, mimeType := parseAPICFrame(frameData, majorVersion)
|
||||||
|
|||||||
@@ -11,7 +11,6 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
// CueSheet represents a parsed .cue file
|
|
||||||
type CueSheet struct {
|
type CueSheet struct {
|
||||||
Performer string `json:"performer"`
|
Performer string `json:"performer"`
|
||||||
Title string `json:"title"`
|
Title string `json:"title"`
|
||||||
@@ -24,7 +23,6 @@ type CueSheet struct {
|
|||||||
Tracks []CueTrack `json:"tracks"`
|
Tracks []CueTrack `json:"tracks"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// CueTrack represents a single track in a cue sheet
|
|
||||||
type CueTrack struct {
|
type CueTrack struct {
|
||||||
Number int `json:"number"`
|
Number int `json:"number"`
|
||||||
Title string `json:"title"`
|
Title string `json:"title"`
|
||||||
@@ -35,7 +33,6 @@ type CueTrack struct {
|
|||||||
PreGap float64 `json:"pre_gap"` // INDEX 00 in seconds (or -1 if not present)
|
PreGap float64 `json:"pre_gap"` // INDEX 00 in seconds (or -1 if not present)
|
||||||
}
|
}
|
||||||
|
|
||||||
// CueSplitInfo represents the information needed to split a CUE+audio file
|
|
||||||
type CueSplitInfo struct {
|
type CueSplitInfo struct {
|
||||||
CuePath string `json:"cue_path"`
|
CuePath string `json:"cue_path"`
|
||||||
AudioPath string `json:"audio_path"`
|
AudioPath string `json:"audio_path"`
|
||||||
@@ -46,7 +43,6 @@ type CueSplitInfo struct {
|
|||||||
Tracks []CueSplitTrack `json:"tracks"`
|
Tracks []CueSplitTrack `json:"tracks"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// CueSplitTrack has the FFmpeg split parameters for a single track
|
|
||||||
type CueSplitTrack struct {
|
type CueSplitTrack struct {
|
||||||
Number int `json:"number"`
|
Number int `json:"number"`
|
||||||
Title string `json:"title"`
|
Title string `json:"title"`
|
||||||
@@ -62,7 +58,6 @@ var (
|
|||||||
reQuoted = regexp.MustCompile(`"([^"]*)"`)
|
reQuoted = regexp.MustCompile(`"([^"]*)"`)
|
||||||
)
|
)
|
||||||
|
|
||||||
// ParseCueFile parses a .cue file and returns a CueSheet
|
|
||||||
func ParseCueFile(cuePath string) (*CueSheet, error) {
|
func ParseCueFile(cuePath string) (*CueSheet, error) {
|
||||||
f, err := os.Open(cuePath)
|
f, err := os.Open(cuePath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -202,7 +197,6 @@ func ParseCueFile(cuePath string) (*CueSheet, error) {
|
|||||||
return sheet, nil
|
return sheet, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// parseCueTimestamp converts MM:SS:FF (frames at 75fps) to seconds
|
|
||||||
func parseCueTimestamp(ts string) float64 {
|
func parseCueTimestamp(ts string) float64 {
|
||||||
parts := strings.Split(ts, ":")
|
parts := strings.Split(ts, ":")
|
||||||
if len(parts) != 3 {
|
if len(parts) != 3 {
|
||||||
@@ -216,7 +210,6 @@ func parseCueTimestamp(ts string) float64 {
|
|||||||
return float64(minutes)*60 + float64(seconds) + float64(frames)/75.0
|
return float64(minutes)*60 + float64(seconds) + float64(frames)/75.0
|
||||||
}
|
}
|
||||||
|
|
||||||
// formatCueTimestamp converts seconds to HH:MM:SS.mmm format for FFmpeg
|
|
||||||
func formatCueTimestamp(seconds float64) string {
|
func formatCueTimestamp(seconds float64) string {
|
||||||
if seconds < 0 {
|
if seconds < 0 {
|
||||||
return "0"
|
return "0"
|
||||||
@@ -227,7 +220,6 @@ func formatCueTimestamp(seconds float64) string {
|
|||||||
return fmt.Sprintf("%02d:%02d:%06.3f", hours, mins, secs)
|
return fmt.Sprintf("%02d:%02d:%06.3f", hours, mins, secs)
|
||||||
}
|
}
|
||||||
|
|
||||||
// unquoteCue removes surrounding quotes from a CUE value
|
|
||||||
func unquoteCue(s string) string {
|
func unquoteCue(s string) string {
|
||||||
s = strings.TrimSpace(s)
|
s = strings.TrimSpace(s)
|
||||||
if matches := reQuoted.FindStringSubmatch(s); len(matches) == 2 {
|
if matches := reQuoted.FindStringSubmatch(s); len(matches) == 2 {
|
||||||
@@ -236,14 +228,12 @@ func unquoteCue(s string) string {
|
|||||||
return s
|
return s
|
||||||
}
|
}
|
||||||
|
|
||||||
// parseCueFileLine parses the FILE command's filename and type
|
|
||||||
func parseCueFileLine(rest string) (string, string) {
|
func parseCueFileLine(rest string) (string, string) {
|
||||||
rest = strings.TrimSpace(rest)
|
rest = strings.TrimSpace(rest)
|
||||||
|
|
||||||
var filename, ftype string
|
var filename, ftype string
|
||||||
|
|
||||||
if strings.HasPrefix(rest, "\"") {
|
if strings.HasPrefix(rest, "\"") {
|
||||||
// Quoted filename
|
|
||||||
endQuote := strings.Index(rest[1:], "\"")
|
endQuote := strings.Index(rest[1:], "\"")
|
||||||
if endQuote >= 0 {
|
if endQuote >= 0 {
|
||||||
filename = rest[1 : endQuote+1]
|
filename = rest[1 : endQuote+1]
|
||||||
@@ -253,7 +243,6 @@ func parseCueFileLine(rest string) (string, string) {
|
|||||||
filename = rest
|
filename = rest
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Unquoted filename - last word is the type
|
|
||||||
parts := strings.Fields(rest)
|
parts := strings.Fields(rest)
|
||||||
if len(parts) >= 2 {
|
if len(parts) >= 2 {
|
||||||
ftype = parts[len(parts)-1]
|
ftype = parts[len(parts)-1]
|
||||||
@@ -266,18 +255,14 @@ func parseCueFileLine(rest string) (string, string) {
|
|||||||
return filename, strings.TrimSpace(ftype)
|
return filename, strings.TrimSpace(ftype)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ResolveCueAudioPath finds the actual audio file referenced by a .cue sheet.
|
|
||||||
// It checks relative to the cue file's directory.
|
|
||||||
func ResolveCueAudioPath(cuePath string, cueFileName string) string {
|
func ResolveCueAudioPath(cuePath string, cueFileName string) string {
|
||||||
cueDir := filepath.Dir(cuePath)
|
cueDir := filepath.Dir(cuePath)
|
||||||
|
|
||||||
// 1. Try the exact filename from the .cue
|
|
||||||
candidate := filepath.Join(cueDir, cueFileName)
|
candidate := filepath.Join(cueDir, cueFileName)
|
||||||
if _, err := os.Stat(candidate); err == nil {
|
if _, err := os.Stat(candidate); err == nil {
|
||||||
return candidate
|
return candidate
|
||||||
}
|
}
|
||||||
|
|
||||||
// 2. Try common case variations
|
|
||||||
baseName := strings.TrimSuffix(cueFileName, filepath.Ext(cueFileName))
|
baseName := strings.TrimSuffix(cueFileName, filepath.Ext(cueFileName))
|
||||||
commonExts := []string{".flac", ".wav", ".ape", ".mp3", ".ogg", ".wv", ".m4a"}
|
commonExts := []string{".flac", ".wav", ".ape", ".mp3", ".ogg", ".wv", ".m4a"}
|
||||||
for _, ext := range commonExts {
|
for _, ext := range commonExts {
|
||||||
@@ -285,14 +270,12 @@ func ResolveCueAudioPath(cuePath string, cueFileName string) string {
|
|||||||
if _, err := os.Stat(candidate); err == nil {
|
if _, err := os.Stat(candidate); err == nil {
|
||||||
return candidate
|
return candidate
|
||||||
}
|
}
|
||||||
// Try uppercase ext
|
|
||||||
candidate = filepath.Join(cueDir, baseName+strings.ToUpper(ext))
|
candidate = filepath.Join(cueDir, baseName+strings.ToUpper(ext))
|
||||||
if _, err := os.Stat(candidate); err == nil {
|
if _, err := os.Stat(candidate); err == nil {
|
||||||
return candidate
|
return candidate
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3. Try to find any audio file with the same base name as the .cue file
|
|
||||||
cueBase := strings.TrimSuffix(filepath.Base(cuePath), filepath.Ext(cuePath))
|
cueBase := strings.TrimSuffix(filepath.Base(cuePath), filepath.Ext(cuePath))
|
||||||
for _, ext := range commonExts {
|
for _, ext := range commonExts {
|
||||||
candidate = filepath.Join(cueDir, cueBase+ext)
|
candidate = filepath.Join(cueDir, cueBase+ext)
|
||||||
@@ -301,7 +284,6 @@ func ResolveCueAudioPath(cuePath string, cueFileName string) string {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 4. If there's only one audio file in the directory, use that
|
|
||||||
entries, err := os.ReadDir(cueDir)
|
entries, err := os.ReadDir(cueDir)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
audioExts := map[string]bool{
|
audioExts := map[string]bool{
|
||||||
@@ -326,13 +308,9 @@ func ResolveCueAudioPath(cuePath string, cueFileName string) string {
|
|||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
// BuildCueSplitInfo creates the split information from a parsed CUE sheet.
|
|
||||||
// This is returned to the Dart side so FFmpeg can perform the splitting.
|
|
||||||
// audioDir, if non-empty, overrides the directory for audio file resolution.
|
|
||||||
func BuildCueSplitInfo(cuePath string, sheet *CueSheet, audioDir string) (*CueSplitInfo, error) {
|
func BuildCueSplitInfo(cuePath string, sheet *CueSheet, audioDir string) (*CueSplitInfo, error) {
|
||||||
resolveDir := cuePath
|
resolveDir := cuePath
|
||||||
if audioDir != "" {
|
if audioDir != "" {
|
||||||
// Create a virtual path in audioDir so ResolveCueAudioPath looks there
|
|
||||||
resolveDir = filepath.Join(audioDir, filepath.Base(cuePath))
|
resolveDir = filepath.Join(audioDir, filepath.Base(cuePath))
|
||||||
}
|
}
|
||||||
audioPath := ResolveCueAudioPath(resolveDir, sheet.FileName)
|
audioPath := ResolveCueAudioPath(resolveDir, sheet.FileName)
|
||||||
@@ -360,11 +338,9 @@ func BuildCueSplitInfo(cuePath string, sheet *CueSheet, audioDir string) (*CueSp
|
|||||||
composer = sheet.Composer
|
composer = sheet.Composer
|
||||||
}
|
}
|
||||||
|
|
||||||
// End time is the start of the next track, or -1 for the last track
|
|
||||||
endSec := float64(-1)
|
endSec := float64(-1)
|
||||||
if i+1 < len(sheet.Tracks) {
|
if i+1 < len(sheet.Tracks) {
|
||||||
nextTrack := sheet.Tracks[i+1]
|
nextTrack := sheet.Tracks[i+1]
|
||||||
// Use pre-gap of next track if available, otherwise its start time
|
|
||||||
if nextTrack.PreGap >= 0 {
|
if nextTrack.PreGap >= 0 {
|
||||||
endSec = nextTrack.PreGap
|
endSec = nextTrack.PreGap
|
||||||
} else {
|
} else {
|
||||||
@@ -386,11 +362,6 @@ func BuildCueSplitInfo(cuePath string, sheet *CueSheet, audioDir string) (*CueSp
|
|||||||
return info, nil
|
return info, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// ParseCueFileJSON parses a .cue file and returns JSON with split info.
|
|
||||||
// This is the main entry point called from Dart via the platform bridge.
|
|
||||||
// audioDir, if non-empty, overrides the directory used for resolving the
|
|
||||||
// referenced audio file (useful when the .cue was copied to a temp dir
|
|
||||||
// but the audio still lives in the original location, e.g. SAF).
|
|
||||||
func ParseCueFileJSON(cuePath string, audioDir string) (string, error) {
|
func ParseCueFileJSON(cuePath string, audioDir string) (string, error) {
|
||||||
sheet, err := ParseCueFile(cuePath)
|
sheet, err := ParseCueFile(cuePath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -410,9 +381,6 @@ func ParseCueFileJSON(cuePath string, audioDir string) (string, error) {
|
|||||||
return string(jsonBytes), nil
|
return string(jsonBytes), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// ScanCueFileForLibrary parses a .cue file and returns multiple LibraryScanResult
|
|
||||||
// entries, one per track. This is used by the library scanner to populate the
|
|
||||||
// library with individual track entries from a single CUE+FLAC album.
|
|
||||||
func ScanCueFileForLibrary(cuePath string, scanTime string) ([]LibraryScanResult, error) {
|
func ScanCueFileForLibrary(cuePath string, scanTime string) ([]LibraryScanResult, error) {
|
||||||
sheet, err := ParseCueFile(cuePath)
|
sheet, err := ParseCueFile(cuePath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -425,13 +393,6 @@ func ScanCueFileForLibrary(cuePath string, scanTime string) ([]LibraryScanResult
|
|||||||
return scanCueSheetForLibrary(cuePath, sheet, audioPath, "", 0, "", scanTime)
|
return scanCueSheetForLibrary(cuePath, sheet, audioPath, "", 0, "", scanTime)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ScanCueFileForLibraryExt is like ScanCueFileForLibrary but with extra parameters
|
|
||||||
// for SAF (Storage Access Framework) scenarios:
|
|
||||||
// - audioDir: if non-empty, overrides the directory used to find the audio file
|
|
||||||
// - virtualPathPrefix: if non-empty, used instead of cuePath as the base for
|
|
||||||
// virtual file paths (e.g. a content:// URI). IDs are also based on this.
|
|
||||||
// - fileModTime: if > 0, used as the FileModTime for all results instead of
|
|
||||||
// stat-ing the cuePath on disk (useful when the real file lives behind SAF)
|
|
||||||
func ScanCueFileForLibraryExt(cuePath, audioDir, virtualPathPrefix string, fileModTime int64, scanTime string) ([]LibraryScanResult, error) {
|
func ScanCueFileForLibraryExt(cuePath, audioDir, virtualPathPrefix string, fileModTime int64, scanTime string) ([]LibraryScanResult, error) {
|
||||||
return ScanCueFileForLibraryExtWithCoverCacheKey(
|
return ScanCueFileForLibraryExtWithCoverCacheKey(
|
||||||
cuePath,
|
cuePath,
|
||||||
@@ -483,7 +444,6 @@ func scanCueSheetForLibrary(cuePath string, sheet *CueSheet, audioPath, virtualP
|
|||||||
return nil, fmt.Errorf("cue sheet is nil for %s", cuePath)
|
return nil, fmt.Errorf("cue sheet is nil for %s", cuePath)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try to get quality info from the audio file
|
|
||||||
var bitDepth, sampleRate int
|
var bitDepth, sampleRate int
|
||||||
var totalDurationSec float64
|
var totalDurationSec float64
|
||||||
audioExt := strings.ToLower(filepath.Ext(audioPath))
|
audioExt := strings.ToLower(filepath.Ext(audioPath))
|
||||||
@@ -505,7 +465,6 @@ func scanCueSheetForLibrary(cuePath string, sheet *CueSheet, audioPath, virtualP
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Extract cover from audio file for all tracks
|
|
||||||
var coverPath string
|
var coverPath string
|
||||||
libraryCoverCacheMu.RLock()
|
libraryCoverCacheMu.RLock()
|
||||||
coverCacheDir := libraryCoverCacheDir
|
coverCacheDir := libraryCoverCacheDir
|
||||||
@@ -522,13 +481,11 @@ func scanCueSheetForLibrary(cuePath string, sheet *CueSheet, audioPath, virtualP
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Determine the base path for virtual paths and IDs
|
|
||||||
pathBase := cuePath
|
pathBase := cuePath
|
||||||
if virtualPathPrefix != "" {
|
if virtualPathPrefix != "" {
|
||||||
pathBase = virtualPathPrefix
|
pathBase = virtualPathPrefix
|
||||||
}
|
}
|
||||||
|
|
||||||
// Determine fileModTime
|
|
||||||
modTime := fileModTime
|
modTime := fileModTime
|
||||||
if modTime <= 0 {
|
if modTime <= 0 {
|
||||||
if info, err := os.Stat(cuePath); err == nil {
|
if info, err := os.Stat(cuePath); err == nil {
|
||||||
@@ -556,7 +513,11 @@ func scanCueSheetForLibrary(cuePath string, sheet *CueSheet, audioPath, virtualP
|
|||||||
album = "Unknown Album"
|
album = "Unknown Album"
|
||||||
}
|
}
|
||||||
|
|
||||||
// Calculate duration for this track
|
composer := track.Composer
|
||||||
|
if composer == "" {
|
||||||
|
composer = sheet.Composer
|
||||||
|
}
|
||||||
|
|
||||||
var duration int
|
var duration int
|
||||||
if i+1 < len(sheet.Tracks) {
|
if i+1 < len(sheet.Tracks) {
|
||||||
nextStart := sheet.Tracks[i+1].StartTime
|
nextStart := sheet.Tracks[i+1].StartTime
|
||||||
@@ -570,9 +531,6 @@ func scanCueSheetForLibrary(cuePath string, sheet *CueSheet, audioPath, virtualP
|
|||||||
|
|
||||||
id := generateLibraryID(fmt.Sprintf("%s#track%d", pathBase, track.Number))
|
id := generateLibraryID(fmt.Sprintf("%s#track%d", pathBase, track.Number))
|
||||||
|
|
||||||
// Use a virtual file path that includes the track number to ensure
|
|
||||||
// uniqueness in the database (file_path has a UNIQUE constraint).
|
|
||||||
// Format: /path/to/album.cue#track01 or content://...album.cue#track01
|
|
||||||
virtualFilePath := fmt.Sprintf("%s#track%02d", pathBase, track.Number)
|
virtualFilePath := fmt.Sprintf("%s#track%02d", pathBase, track.Number)
|
||||||
|
|
||||||
result := LibraryScanResult{
|
result := LibraryScanResult{
|
||||||
@@ -586,12 +544,15 @@ func scanCueSheetForLibrary(cuePath string, sheet *CueSheet, audioPath, virtualP
|
|||||||
ScannedAt: scanTime,
|
ScannedAt: scanTime,
|
||||||
ISRC: track.ISRC,
|
ISRC: track.ISRC,
|
||||||
TrackNumber: track.Number,
|
TrackNumber: track.Number,
|
||||||
|
TotalTracks: len(sheet.Tracks),
|
||||||
DiscNumber: 1,
|
DiscNumber: 1,
|
||||||
|
TotalDiscs: 1,
|
||||||
Duration: duration,
|
Duration: duration,
|
||||||
ReleaseDate: sheet.Date,
|
ReleaseDate: sheet.Date,
|
||||||
BitDepth: bitDepth,
|
BitDepth: bitDepth,
|
||||||
SampleRate: sampleRate,
|
SampleRate: sampleRate,
|
||||||
Genre: sheet.Genre,
|
Genre: sheet.Genre,
|
||||||
|
Composer: composer,
|
||||||
Format: "cue+" + strings.TrimPrefix(audioExt, "."),
|
Format: "cue+" + strings.TrimPrefix(audioExt, "."),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
+80
-5
@@ -196,15 +196,22 @@ type deezerAlbumSimple struct {
|
|||||||
RecordType string `json:"record_type"`
|
RecordType string `json:"record_type"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *DeezerClient) convertTrack(track deezerTrack) TrackMetadata {
|
// deezerTrackArtistDisplay returns the display artist string for a track,
|
||||||
artistName := track.Artist.Name
|
// preferring the Contributors list (comma-joined) when available, falling
|
||||||
|
// back to the primary Artist.Name.
|
||||||
|
func deezerTrackArtistDisplay(track deezerTrack) string {
|
||||||
if len(track.Contributors) > 0 {
|
if len(track.Contributors) > 0 {
|
||||||
names := make([]string, len(track.Contributors))
|
names := make([]string, len(track.Contributors))
|
||||||
for i, a := range track.Contributors {
|
for i, a := range track.Contributors {
|
||||||
names[i] = a.Name
|
names[i] = a.Name
|
||||||
}
|
}
|
||||||
artistName = strings.Join(names, ", ")
|
return strings.Join(names, ", ")
|
||||||
}
|
}
|
||||||
|
return track.Artist.Name
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *DeezerClient) convertTrack(track deezerTrack) TrackMetadata {
|
||||||
|
artistName := deezerTrackArtistDisplay(track)
|
||||||
|
|
||||||
albumImage := track.Album.CoverXL
|
albumImage := track.Album.CoverXL
|
||||||
if albumImage == "" {
|
if albumImage == "" {
|
||||||
@@ -623,6 +630,12 @@ func (c *DeezerClient) GetAlbum(ctx context.Context, albumID string) (*AlbumResp
|
|||||||
}
|
}
|
||||||
|
|
||||||
isrcMap := c.fetchISRCsParallel(ctx, allTracks)
|
isrcMap := c.fetchISRCsParallel(ctx, allTracks)
|
||||||
|
totalDiscs := 0
|
||||||
|
for _, track := range allTracks {
|
||||||
|
if track.DiskNumber > totalDiscs {
|
||||||
|
totalDiscs = track.DiskNumber
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
tracks := make([]AlbumTrackMetadata, 0, len(allTracks))
|
tracks := make([]AlbumTrackMetadata, 0, len(allTracks))
|
||||||
albumType := album.RecordType
|
albumType := album.RecordType
|
||||||
@@ -641,7 +654,7 @@ func (c *DeezerClient) GetAlbum(ctx context.Context, albumID string) (*AlbumResp
|
|||||||
|
|
||||||
tracks = append(tracks, AlbumTrackMetadata{
|
tracks = append(tracks, AlbumTrackMetadata{
|
||||||
SpotifyID: fmt.Sprintf("deezer:%d", track.ID),
|
SpotifyID: fmt.Sprintf("deezer:%d", track.ID),
|
||||||
Artists: track.Artist.Name,
|
Artists: deezerTrackArtistDisplay(track),
|
||||||
Name: track.Title,
|
Name: track.Title,
|
||||||
AlbumName: album.Title,
|
AlbumName: album.Title,
|
||||||
AlbumArtist: artistName,
|
AlbumArtist: artistName,
|
||||||
@@ -651,6 +664,7 @@ func (c *DeezerClient) GetAlbum(ctx context.Context, albumID string) (*AlbumResp
|
|||||||
TrackNumber: trackNum,
|
TrackNumber: trackNum,
|
||||||
TotalTracks: album.NbTracks,
|
TotalTracks: album.NbTracks,
|
||||||
DiscNumber: track.DiskNumber,
|
DiscNumber: track.DiskNumber,
|
||||||
|
TotalDiscs: totalDiscs,
|
||||||
ExternalURL: track.Link,
|
ExternalURL: track.Link,
|
||||||
ISRC: isrc,
|
ISRC: isrc,
|
||||||
AlbumID: fmt.Sprintf("deezer:%d", album.ID),
|
AlbumID: fmt.Sprintf("deezer:%d", album.ID),
|
||||||
@@ -741,6 +755,10 @@ func (c *DeezerClient) GetArtist(ctx context.Context, artistID string) (*ArtistR
|
|||||||
Artists: artist.Name,
|
Artists: artist.Name,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// The Deezer /artist/{id}/albums endpoint does not return nb_tracks.
|
||||||
|
// Fetch track counts in parallel from individual /album/{id} endpoints.
|
||||||
|
c.fetchAlbumTrackCounts(ctx, albums)
|
||||||
}
|
}
|
||||||
|
|
||||||
result := &ArtistResponsePayload{
|
result := &ArtistResponsePayload{
|
||||||
@@ -760,6 +778,63 @@ func (c *DeezerClient) GetArtist(ctx context.Context, artistID string) (*ArtistR
|
|||||||
return result, nil
|
return result, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// fetchAlbumTrackCounts fetches nb_tracks for each album in parallel using
|
||||||
|
// individual /album/{id} calls, since the /artist/{id}/albums endpoint does
|
||||||
|
// not include this field. Albums whose track count is already known (non-zero)
|
||||||
|
// are skipped.
|
||||||
|
func (c *DeezerClient) fetchAlbumTrackCounts(ctx context.Context, albums []ArtistAlbumMetadata) {
|
||||||
|
// Find albums that need track counts
|
||||||
|
type indexedID struct {
|
||||||
|
idx int
|
||||||
|
albumID string
|
||||||
|
}
|
||||||
|
var toFetch []indexedID
|
||||||
|
for i, a := range albums {
|
||||||
|
if a.TotalTracks == 0 {
|
||||||
|
rawID := strings.TrimPrefix(a.ID, "deezer:")
|
||||||
|
if rawID != "" {
|
||||||
|
toFetch = append(toFetch, indexedID{idx: i, albumID: rawID})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if len(toFetch) == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const maxParallel = 10
|
||||||
|
sem := make(chan struct{}, maxParallel)
|
||||||
|
var mu sync.Mutex
|
||||||
|
var wg sync.WaitGroup
|
||||||
|
|
||||||
|
for _, item := range toFetch {
|
||||||
|
wg.Add(1)
|
||||||
|
go func(it indexedID) {
|
||||||
|
defer wg.Done()
|
||||||
|
|
||||||
|
select {
|
||||||
|
case sem <- struct{}{}:
|
||||||
|
defer func() { <-sem }()
|
||||||
|
case <-ctx.Done():
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
albumURL := fmt.Sprintf(deezerAlbumURL, it.albumID)
|
||||||
|
var resp struct {
|
||||||
|
NbTracks int `json:"nb_tracks"`
|
||||||
|
}
|
||||||
|
if err := c.getJSON(ctx, albumURL, &resp); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
mu.Lock()
|
||||||
|
albums[it.idx].TotalTracks = resp.NbTracks
|
||||||
|
mu.Unlock()
|
||||||
|
}(item)
|
||||||
|
}
|
||||||
|
|
||||||
|
wg.Wait()
|
||||||
|
}
|
||||||
|
|
||||||
func (c *DeezerClient) GetRelatedArtists(ctx context.Context, artistID string, limit int) ([]SearchArtistResult, error) {
|
func (c *DeezerClient) GetRelatedArtists(ctx context.Context, artistID string, limit int) ([]SearchArtistResult, error) {
|
||||||
normalizedArtistID := strings.TrimSpace(strings.TrimPrefix(artistID, "deezer:"))
|
normalizedArtistID := strings.TrimSpace(strings.TrimPrefix(artistID, "deezer:"))
|
||||||
if normalizedArtistID == "" {
|
if normalizedArtistID == "" {
|
||||||
@@ -892,7 +967,7 @@ func (c *DeezerClient) GetPlaylist(ctx context.Context, playlistID string) (*Pla
|
|||||||
|
|
||||||
tracks = append(tracks, AlbumTrackMetadata{
|
tracks = append(tracks, AlbumTrackMetadata{
|
||||||
SpotifyID: fmt.Sprintf("deezer:%d", track.ID),
|
SpotifyID: fmt.Sprintf("deezer:%d", track.ID),
|
||||||
Artists: track.Artist.Name,
|
Artists: deezerTrackArtistDisplay(track),
|
||||||
Name: track.Title,
|
Name: track.Title,
|
||||||
AlbumName: track.Album.Title,
|
AlbumName: track.Album.Title,
|
||||||
AlbumArtist: track.Artist.Name,
|
AlbumArtist: track.Artist.Name,
|
||||||
|
|||||||
+22
-184
@@ -14,14 +14,7 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
const deezerYoinkifyURL = "https://yoinkify.lol/api/download"
|
const deezerMusicDLURL = "https://api.zarz.moe/v1/dzr"
|
||||||
const deezerMusicDLURL = "https://www.musicdl.me/api/download"
|
|
||||||
|
|
||||||
type YoinkifyRequest struct {
|
|
||||||
URL string `json:"url"`
|
|
||||||
Format string `json:"format"`
|
|
||||||
GenreSource string `json:"genreSource"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type DeezerDownloadResult struct {
|
type DeezerDownloadResult struct {
|
||||||
FilePath string
|
FilePath string
|
||||||
@@ -37,41 +30,6 @@ type DeezerDownloadResult struct {
|
|||||||
LyricsLRC string
|
LyricsLRC string
|
||||||
}
|
}
|
||||||
|
|
||||||
func resolveSpotifyURLForYoinkify(req DownloadRequest) (string, error) {
|
|
||||||
rawSpotify := strings.TrimSpace(req.SpotifyID)
|
|
||||||
if rawSpotify != "" {
|
|
||||||
if isLikelySpotifyTrackID(rawSpotify) {
|
|
||||||
return fmt.Sprintf("https://open.spotify.com/track/%s", rawSpotify), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if parsed, err := parseSpotifyURI(rawSpotify); err == nil && parsed.Type == "track" && parsed.ID != "" {
|
|
||||||
return fmt.Sprintf("https://open.spotify.com/track/%s", parsed.ID), nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
deezerID := strings.TrimSpace(req.DeezerID)
|
|
||||||
if deezerID == "" {
|
|
||||||
if prefixed, found := strings.CutPrefix(rawSpotify, "deezer:"); found {
|
|
||||||
deezerID = strings.TrimSpace(prefixed)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if deezerID != "" {
|
|
||||||
songlink := NewSongLinkClient()
|
|
||||||
spotifyID, err := songlink.GetSpotifyIDFromDeezer(deezerID)
|
|
||||||
if err != nil {
|
|
||||||
return "", fmt.Errorf("failed to map deezer:%s to Spotify ID: %w", deezerID, err)
|
|
||||||
}
|
|
||||||
spotifyID = strings.TrimSpace(spotifyID)
|
|
||||||
if spotifyID == "" {
|
|
||||||
return "", fmt.Errorf("SongLink returned empty Spotify ID for deezer:%s", deezerID)
|
|
||||||
}
|
|
||||||
return fmt.Sprintf("https://open.spotify.com/track/%s", spotifyID), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
return "", fmt.Errorf("missing Spotify track ID for Deezer Yoinkify")
|
|
||||||
}
|
|
||||||
|
|
||||||
func isLikelySpotifyTrackID(value string) bool {
|
func isLikelySpotifyTrackID(value string) bool {
|
||||||
if len(value) != 22 {
|
if len(value) != 22 {
|
||||||
return false
|
return false
|
||||||
@@ -88,113 +46,6 @@ func isLikelySpotifyTrackID(value string) bool {
|
|||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *DeezerClient) DownloadFromYoinkify(spotifyURL, outputPath string, outputFD int, itemID string) error {
|
|
||||||
payload := YoinkifyRequest{
|
|
||||||
URL: spotifyURL,
|
|
||||||
Format: "flac",
|
|
||||||
GenreSource: "spotify",
|
|
||||||
}
|
|
||||||
|
|
||||||
jsonData, err := json.Marshal(payload)
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("failed to encode Yoinkify request: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx := context.Background()
|
|
||||||
if itemID != "" {
|
|
||||||
StartItemProgress(itemID)
|
|
||||||
defer CompleteItemProgress(itemID)
|
|
||||||
ctx = initDownloadCancel(itemID)
|
|
||||||
defer clearDownloadCancel(itemID)
|
|
||||||
}
|
|
||||||
|
|
||||||
if isDownloadCancelled(itemID) {
|
|
||||||
return ErrDownloadCancelled
|
|
||||||
}
|
|
||||||
|
|
||||||
req, err := http.NewRequestWithContext(ctx, http.MethodPost, deezerYoinkifyURL, bytes.NewBuffer(jsonData))
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("failed to create Yoinkify request: %w", err)
|
|
||||||
}
|
|
||||||
req.Header.Set("Content-Type", "application/json")
|
|
||||||
req.Header.Set("Accept", "*/*")
|
|
||||||
req.Header.Set("User-Agent", getRandomUserAgent())
|
|
||||||
|
|
||||||
resp, err := GetDownloadClient().Do(req)
|
|
||||||
if err != nil {
|
|
||||||
if isDownloadCancelled(itemID) {
|
|
||||||
return ErrDownloadCancelled
|
|
||||||
}
|
|
||||||
return fmt.Errorf("failed to call Yoinkify: %w", err)
|
|
||||||
}
|
|
||||||
defer resp.Body.Close()
|
|
||||||
|
|
||||||
contentType := strings.ToLower(strings.TrimSpace(resp.Header.Get("Content-Type")))
|
|
||||||
if resp.StatusCode != http.StatusOK {
|
|
||||||
bodyBytes, _ := io.ReadAll(io.LimitReader(resp.Body, 4096))
|
|
||||||
bodyText := strings.TrimSpace(string(bodyBytes))
|
|
||||||
if bodyText != "" {
|
|
||||||
return fmt.Errorf("Yoinkify returned status %d: %s", resp.StatusCode, bodyText)
|
|
||||||
}
|
|
||||||
return fmt.Errorf("Yoinkify returned status %d", resp.StatusCode)
|
|
||||||
}
|
|
||||||
|
|
||||||
if strings.Contains(contentType, "application/json") {
|
|
||||||
bodyBytes, _ := io.ReadAll(io.LimitReader(resp.Body, 4096))
|
|
||||||
bodyText := strings.TrimSpace(string(bodyBytes))
|
|
||||||
if bodyText == "" {
|
|
||||||
bodyText = "empty JSON payload"
|
|
||||||
}
|
|
||||||
return fmt.Errorf("Yoinkify returned JSON instead of audio: %s", bodyText)
|
|
||||||
}
|
|
||||||
|
|
||||||
expectedSize := resp.ContentLength
|
|
||||||
if expectedSize > 0 && itemID != "" {
|
|
||||||
SetItemBytesTotal(itemID, expectedSize)
|
|
||||||
}
|
|
||||||
|
|
||||||
out, err := openOutputForWrite(outputPath, outputFD)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
bufWriter := bufio.NewWriterSize(out, 256*1024)
|
|
||||||
var written int64
|
|
||||||
if itemID != "" {
|
|
||||||
pw := NewItemProgressWriter(bufWriter, itemID)
|
|
||||||
written, err = io.Copy(pw, resp.Body)
|
|
||||||
} else {
|
|
||||||
written, err = io.Copy(bufWriter, resp.Body)
|
|
||||||
}
|
|
||||||
|
|
||||||
flushErr := bufWriter.Flush()
|
|
||||||
closeErr := out.Close()
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
cleanupOutputOnError(outputPath, outputFD)
|
|
||||||
if isDownloadCancelled(itemID) {
|
|
||||||
return ErrDownloadCancelled
|
|
||||||
}
|
|
||||||
return fmt.Errorf("download interrupted: %w", err)
|
|
||||||
}
|
|
||||||
if flushErr != nil {
|
|
||||||
cleanupOutputOnError(outputPath, outputFD)
|
|
||||||
return fmt.Errorf("failed to flush output: %w", flushErr)
|
|
||||||
}
|
|
||||||
if closeErr != nil {
|
|
||||||
cleanupOutputOnError(outputPath, outputFD)
|
|
||||||
return fmt.Errorf("failed to close output: %w", closeErr)
|
|
||||||
}
|
|
||||||
|
|
||||||
if expectedSize > 0 && written != expectedSize {
|
|
||||||
cleanupOutputOnError(outputPath, outputFD)
|
|
||||||
return fmt.Errorf("incomplete download: expected %d bytes, got %d bytes", expectedSize, written)
|
|
||||||
}
|
|
||||||
|
|
||||||
GoLog("[Deezer] Downloaded via Yoinkify: %.2f MB\n", float64(written)/(1024*1024))
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func resolveDeezerTrackURL(req DownloadRequest) (string, error) {
|
func resolveDeezerTrackURL(req DownloadRequest) (string, error) {
|
||||||
deezerID := strings.TrimSpace(req.DeezerID)
|
deezerID := strings.TrimSpace(req.DeezerID)
|
||||||
if deezerID == "" {
|
if deezerID == "" {
|
||||||
@@ -211,7 +62,6 @@ func resolveDeezerTrackURL(req DownloadRequest) (string, error) {
|
|||||||
return trackURL, nil
|
return trackURL, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try SongLink
|
|
||||||
spotifyID := strings.TrimSpace(req.SpotifyID)
|
spotifyID := strings.TrimSpace(req.SpotifyID)
|
||||||
if spotifyID != "" && isLikelySpotifyTrackID(spotifyID) {
|
if spotifyID != "" && isLikelySpotifyTrackID(spotifyID) {
|
||||||
songlink := NewSongLinkClient()
|
songlink := NewSongLinkClient()
|
||||||
@@ -231,7 +81,6 @@ func resolveDeezerTrackURL(req DownloadRequest) (string, error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try ISRC
|
|
||||||
isrc := strings.TrimSpace(req.ISRC)
|
isrc := strings.TrimSpace(req.ISRC)
|
||||||
if isrc != "" {
|
if isrc != "" {
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), SongLinkTimeout)
|
ctx, cancel := context.WithTimeout(context.Background(), SongLinkTimeout)
|
||||||
@@ -294,7 +143,6 @@ func (c *DeezerClient) GetMusicDLDownloadURL(deezerTrackURL string) (string, err
|
|||||||
return "", fmt.Errorf("failed to create MusicDL request: %w", err)
|
return "", fmt.Errorf("failed to create MusicDL request: %w", err)
|
||||||
}
|
}
|
||||||
req.Header.Set("Content-Type", "application/json")
|
req.Header.Set("Content-Type", "application/json")
|
||||||
req.Header.Set("X-Debug-Key", getQobuzDebugKey())
|
|
||||||
req.Header.Set("User-Agent", getRandomUserAgent())
|
req.Header.Set("User-Agent", getRandomUserAgent())
|
||||||
|
|
||||||
resp, err := c.httpClient.Do(req)
|
resp, err := c.httpClient.Do(req)
|
||||||
@@ -479,41 +327,29 @@ func downloadFromDeezer(req DownloadRequest) (DeezerDownloadResult, error) {
|
|||||||
)
|
)
|
||||||
}()
|
}()
|
||||||
|
|
||||||
// Try MusicDL first (better quality), fallback to Yoinkify
|
|
||||||
var downloadErr error
|
|
||||||
deezerTrackURL, deezerURLErr := resolveDeezerTrackURL(req)
|
deezerTrackURL, deezerURLErr := resolveDeezerTrackURL(req)
|
||||||
if deezerURLErr == nil {
|
if deezerURLErr != nil {
|
||||||
GoLog("[Deezer] Trying MusicDL for: %s\n", deezerTrackURL)
|
return DeezerDownloadResult{}, fmt.Errorf(
|
||||||
downloadErr = deezerClient.DownloadFromMusicDL(deezerTrackURL, outputPath, req.OutputFD, req.ItemID)
|
"deezer download failed: could not resolve Deezer URL: %w",
|
||||||
if downloadErr != nil {
|
deezerURLErr,
|
||||||
if errors.Is(downloadErr, ErrDownloadCancelled) {
|
)
|
||||||
return DeezerDownloadResult{}, ErrDownloadCancelled
|
|
||||||
}
|
|
||||||
GoLog("[Deezer] MusicDL failed: %v, falling back to Yoinkify\n", downloadErr)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
GoLog("[Deezer] Could not resolve Deezer URL: %v, using Yoinkify directly\n", deezerURLErr)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if downloadErr != nil || deezerURLErr != nil {
|
GoLog("[Deezer] Trying MusicDL for: %s\n", deezerTrackURL)
|
||||||
spotifyURL, err := resolveSpotifyURLForYoinkify(req)
|
downloadErr := deezerClient.DownloadFromMusicDL(
|
||||||
if err != nil {
|
deezerTrackURL,
|
||||||
if deezerURLErr != nil {
|
outputPath,
|
||||||
return DeezerDownloadResult{}, fmt.Errorf(
|
req.OutputFD,
|
||||||
"deezer download failed: direct Deezer resolution error: %v; Yoinkify fallback error: %w",
|
req.ItemID,
|
||||||
deezerURLErr,
|
)
|
||||||
err,
|
if downloadErr != nil {
|
||||||
)
|
if errors.Is(downloadErr, ErrDownloadCancelled) {
|
||||||
}
|
return DeezerDownloadResult{}, ErrDownloadCancelled
|
||||||
return DeezerDownloadResult{}, err
|
|
||||||
}
|
|
||||||
downloadErr = deezerClient.DownloadFromYoinkify(spotifyURL, outputPath, req.OutputFD, req.ItemID)
|
|
||||||
if downloadErr != nil {
|
|
||||||
if errors.Is(downloadErr, ErrDownloadCancelled) {
|
|
||||||
return DeezerDownloadResult{}, ErrDownloadCancelled
|
|
||||||
}
|
|
||||||
return DeezerDownloadResult{}, fmt.Errorf("deezer download failed (MusicDL + Yoinkify): %w", downloadErr)
|
|
||||||
}
|
}
|
||||||
|
return DeezerDownloadResult{}, fmt.Errorf(
|
||||||
|
"deezer download failed via MusicDL: %w",
|
||||||
|
downloadErr,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
<-parallelDone
|
<-parallelDone
|
||||||
@@ -533,10 +369,12 @@ func downloadFromDeezer(req DownloadRequest) (DeezerDownloadResult, error) {
|
|||||||
TrackNumber: req.TrackNumber,
|
TrackNumber: req.TrackNumber,
|
||||||
TotalTracks: req.TotalTracks,
|
TotalTracks: req.TotalTracks,
|
||||||
DiscNumber: req.DiscNumber,
|
DiscNumber: req.DiscNumber,
|
||||||
|
TotalDiscs: req.TotalDiscs,
|
||||||
ISRC: req.ISRC,
|
ISRC: req.ISRC,
|
||||||
Genre: req.Genre,
|
Genre: req.Genre,
|
||||||
Label: req.Label,
|
Label: req.Label,
|
||||||
Copyright: req.Copyright,
|
Copyright: req.Copyright,
|
||||||
|
Composer: req.Composer,
|
||||||
}
|
}
|
||||||
|
|
||||||
var coverData []byte
|
var coverData []byte
|
||||||
|
|||||||
@@ -25,7 +25,6 @@ var (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func GetISRCIndex(outputDir string) *ISRCIndex {
|
func GetISRCIndex(outputDir string) *ISRCIndex {
|
||||||
// Fast path: check cache first
|
|
||||||
isrcIndexCacheMu.RLock()
|
isrcIndexCacheMu.RLock()
|
||||||
idx, exists := isrcIndexCache[outputDir]
|
idx, exists := isrcIndexCache[outputDir]
|
||||||
isrcIndexCacheMu.RUnlock()
|
isrcIndexCacheMu.RUnlock()
|
||||||
@@ -34,13 +33,11 @@ func GetISRCIndex(outputDir string) *ISRCIndex {
|
|||||||
return idx
|
return idx
|
||||||
}
|
}
|
||||||
|
|
||||||
// Use per-directory mutex to prevent multiple goroutines from building simultaneously
|
|
||||||
buildLock, _ := isrcBuildingMu.LoadOrStore(outputDir, &sync.Mutex{})
|
buildLock, _ := isrcBuildingMu.LoadOrStore(outputDir, &sync.Mutex{})
|
||||||
mu := buildLock.(*sync.Mutex)
|
mu := buildLock.(*sync.Mutex)
|
||||||
mu.Lock()
|
mu.Lock()
|
||||||
defer mu.Unlock()
|
defer mu.Unlock()
|
||||||
|
|
||||||
// Double-check cache after acquiring lock (another goroutine may have built it)
|
|
||||||
isrcIndexCacheMu.RLock()
|
isrcIndexCacheMu.RLock()
|
||||||
idx, exists = isrcIndexCache[outputDir]
|
idx, exists = isrcIndexCache[outputDir]
|
||||||
isrcIndexCacheMu.RUnlock()
|
isrcIndexCacheMu.RUnlock()
|
||||||
|
|||||||
+476
-183
File diff suppressed because it is too large
Load Diff
@@ -193,13 +193,15 @@ func TestBuildReEnrichFFmpegMetadataOmitsEmptyFields(t *testing.T) {
|
|||||||
Copyright: "",
|
Copyright: "",
|
||||||
}
|
}
|
||||||
|
|
||||||
metadata := buildReEnrichFFmpegMetadata(req, "")
|
metadata := buildReEnrichFFmpegMetadata(&req, "")
|
||||||
|
|
||||||
if metadata["TITLE"] != "Song" {
|
// Title and Artist are never written by re-enrich (they are search keys
|
||||||
t.Fatalf("title = %q", metadata["TITLE"])
|
// preserved as-is from the file).
|
||||||
|
if _, exists := metadata["TITLE"]; exists {
|
||||||
|
t.Fatalf("TITLE should not be in metadata: %#v", metadata)
|
||||||
}
|
}
|
||||||
if metadata["ARTIST"] != "Artist" {
|
if _, exists := metadata["ARTIST"]; exists {
|
||||||
t.Fatalf("artist = %q", metadata["ARTIST"])
|
t.Fatalf("ARTIST should not be in metadata: %#v", metadata)
|
||||||
}
|
}
|
||||||
if metadata["ALBUM"] != "Album" {
|
if metadata["ALBUM"] != "Album" {
|
||||||
t.Fatalf("album = %q", metadata["ALBUM"])
|
t.Fatalf("album = %q", metadata["ALBUM"])
|
||||||
@@ -222,3 +224,47 @@ func TestBuildReEnrichFFmpegMetadataOmitsEmptyFields(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestApplyReEnrichTrackMetadataCopiesComposerAndTotals(t *testing.T) {
|
||||||
|
req := reEnrichRequest{}
|
||||||
|
|
||||||
|
applyReEnrichTrackMetadata(&req, ExtTrackMetadata{
|
||||||
|
TrackNumber: 7,
|
||||||
|
TotalTracks: 12,
|
||||||
|
DiscNumber: 2,
|
||||||
|
TotalDiscs: 3,
|
||||||
|
Composer: "Composer",
|
||||||
|
})
|
||||||
|
|
||||||
|
if req.TrackNumber != 7 || req.TotalTracks != 12 {
|
||||||
|
t.Fatalf("track metadata = %d/%d", req.TrackNumber, req.TotalTracks)
|
||||||
|
}
|
||||||
|
if req.DiscNumber != 2 || req.TotalDiscs != 3 {
|
||||||
|
t.Fatalf("disc metadata = %d/%d", req.DiscNumber, req.TotalDiscs)
|
||||||
|
}
|
||||||
|
if req.Composer != "Composer" {
|
||||||
|
t.Fatalf("composer = %q", req.Composer)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestBuildReEnrichFFmpegMetadataFormatsTotalsAndComposer(t *testing.T) {
|
||||||
|
req := reEnrichRequest{
|
||||||
|
TrackNumber: 7,
|
||||||
|
TotalTracks: 12,
|
||||||
|
DiscNumber: 2,
|
||||||
|
TotalDiscs: 3,
|
||||||
|
Composer: "Composer",
|
||||||
|
}
|
||||||
|
|
||||||
|
metadata := buildReEnrichFFmpegMetadata(&req, "")
|
||||||
|
|
||||||
|
if metadata["TRACKNUMBER"] != "7/12" {
|
||||||
|
t.Fatalf("TRACKNUMBER = %q", metadata["TRACKNUMBER"])
|
||||||
|
}
|
||||||
|
if metadata["DISCNUMBER"] != "2/3" {
|
||||||
|
t.Fatalf("DISCNUMBER = %q", metadata["DISCNUMBER"])
|
||||||
|
}
|
||||||
|
if metadata["COMPOSER"] != "Composer" {
|
||||||
|
t.Fatalf("COMPOSER = %q", metadata["COMPOSER"])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -43,12 +43,12 @@ func compareVersions(v1, v2 string) int {
|
|||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
type LoadedExtension struct {
|
type loadedExtension struct {
|
||||||
ID string `json:"id"`
|
ID string `json:"id"`
|
||||||
Manifest *ExtensionManifest `json:"manifest"`
|
Manifest *ExtensionManifest `json:"manifest"`
|
||||||
VM *goja.Runtime `json:"-"`
|
VM *goja.Runtime `json:"-"`
|
||||||
VMMu sync.Mutex `json:"-"`
|
VMMu sync.Mutex `json:"-"`
|
||||||
runtime *ExtensionRuntime
|
runtime *extensionRuntime
|
||||||
initialized bool
|
initialized bool
|
||||||
Enabled bool `json:"enabled"`
|
Enabled bool `json:"enabled"`
|
||||||
Error string `json:"error,omitempty"`
|
Error string `json:"error,omitempty"`
|
||||||
@@ -73,7 +73,7 @@ func getExtensionInitSettings(extensionID string) map[string]interface{} {
|
|||||||
return filtered
|
return filtered
|
||||||
}
|
}
|
||||||
|
|
||||||
func ensureRuntimeReadyLocked(ext *LoadedExtension, applyStoredSettings bool) error {
|
func ensureRuntimeReadyLocked(ext *loadedExtension, applyStoredSettings bool) error {
|
||||||
if ext.VM == nil || ext.runtime == nil {
|
if ext.VM == nil || ext.runtime == nil {
|
||||||
if err := initializeVMLocked(ext); err != nil {
|
if err := initializeVMLocked(ext); err != nil {
|
||||||
ext.Error = err.Error()
|
ext.Error = err.Error()
|
||||||
@@ -100,14 +100,14 @@ func ensureRuntimeReadyLocked(ext *LoadedExtension, applyStoredSettings bool) er
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ext *LoadedExtension) ensureRuntimeReady() error {
|
func (ext *loadedExtension) ensureRuntimeReady() error {
|
||||||
ext.VMMu.Lock()
|
ext.VMMu.Lock()
|
||||||
defer ext.VMMu.Unlock()
|
defer ext.VMMu.Unlock()
|
||||||
|
|
||||||
return ensureRuntimeReadyLocked(ext, true)
|
return ensureRuntimeReadyLocked(ext, true)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ext *LoadedExtension) lockReadyVM() (*goja.Runtime, error) {
|
func (ext *loadedExtension) lockReadyVM() (*goja.Runtime, error) {
|
||||||
ext.VMMu.Lock()
|
ext.VMMu.Lock()
|
||||||
if err := ensureRuntimeReadyLocked(ext, true); err != nil {
|
if err := ensureRuntimeReadyLocked(ext, true); err != nil {
|
||||||
ext.VMMu.Unlock()
|
ext.VMMu.Unlock()
|
||||||
@@ -116,28 +116,28 @@ func (ext *LoadedExtension) lockReadyVM() (*goja.Runtime, error) {
|
|||||||
return ext.VM, nil
|
return ext.VM, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
type ExtensionManager struct {
|
type extensionManager struct {
|
||||||
mu sync.RWMutex
|
mu sync.RWMutex
|
||||||
extensions map[string]*LoadedExtension
|
extensions map[string]*loadedExtension
|
||||||
extensionsDir string
|
extensionsDir string
|
||||||
dataDir string
|
dataDir string
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
globalExtManager *ExtensionManager
|
globalExtManager *extensionManager
|
||||||
globalExtManagerOnce sync.Once
|
globalExtManagerOnce sync.Once
|
||||||
)
|
)
|
||||||
|
|
||||||
func GetExtensionManager() *ExtensionManager {
|
func getExtensionManager() *extensionManager {
|
||||||
globalExtManagerOnce.Do(func() {
|
globalExtManagerOnce.Do(func() {
|
||||||
globalExtManager = &ExtensionManager{
|
globalExtManager = &extensionManager{
|
||||||
extensions: make(map[string]*LoadedExtension),
|
extensions: make(map[string]*loadedExtension),
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
return globalExtManager
|
return globalExtManager
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) SetDirectories(extensionsDir, dataDir string) error {
|
func (m *extensionManager) SetDirectories(extensionsDir, dataDir string) error {
|
||||||
m.mu.Lock()
|
m.mu.Lock()
|
||||||
defer m.mu.Unlock()
|
defer m.mu.Unlock()
|
||||||
|
|
||||||
@@ -154,7 +154,7 @@ func (m *ExtensionManager) SetDirectories(extensionsDir, dataDir string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) LoadExtensionFromFile(filePath string) (*LoadedExtension, error) {
|
func (m *extensionManager) LoadExtensionFromFile(filePath string) (*loadedExtension, error) {
|
||||||
if !strings.HasSuffix(strings.ToLower(filePath), ".spotiflac-ext") {
|
if !strings.HasSuffix(strings.ToLower(filePath), ".spotiflac-ext") {
|
||||||
return nil, fmt.Errorf("Invalid file format. Please select a .spotiflac-ext file")
|
return nil, fmt.Errorf("Invalid file format. Please select a .spotiflac-ext file")
|
||||||
}
|
}
|
||||||
@@ -272,7 +272,7 @@ func (m *ExtensionManager) LoadExtensionFromFile(filePath string) (*LoadedExtens
|
|||||||
return nil, fmt.Errorf("failed to create extension data directory: %w", err)
|
return nil, fmt.Errorf("failed to create extension data directory: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
ext := &LoadedExtension{
|
ext := &loadedExtension{
|
||||||
ID: manifest.Name,
|
ID: manifest.Name,
|
||||||
Manifest: manifest,
|
Manifest: manifest,
|
||||||
Enabled: false, // New extensions start disabled
|
Enabled: false, // New extensions start disabled
|
||||||
@@ -292,7 +292,7 @@ func (m *ExtensionManager) LoadExtensionFromFile(filePath string) (*LoadedExtens
|
|||||||
return ext, nil
|
return ext, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func initializeVMLocked(ext *LoadedExtension) error {
|
func initializeVMLocked(ext *loadedExtension) error {
|
||||||
ext.VM = nil
|
ext.VM = nil
|
||||||
ext.runtime = nil
|
ext.runtime = nil
|
||||||
ext.initialized = false
|
ext.initialized = false
|
||||||
@@ -305,7 +305,7 @@ func initializeVMLocked(ext *LoadedExtension) error {
|
|||||||
return fmt.Errorf("failed to read index.js: %w", err)
|
return fmt.Errorf("failed to read index.js: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
runtime := NewExtensionRuntime(ext)
|
runtime := newExtensionRuntime(ext)
|
||||||
ext.runtime = runtime
|
ext.runtime = runtime
|
||||||
runtime.RegisterAPIs(vm)
|
runtime.RegisterAPIs(vm)
|
||||||
runtime.RegisterGoBackendAPIs(vm)
|
runtime.RegisterGoBackendAPIs(vm)
|
||||||
@@ -342,14 +342,14 @@ func initializeVMLocked(ext *LoadedExtension) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) initializeVM(ext *LoadedExtension) error {
|
func (m *extensionManager) initializeVM(ext *loadedExtension) error {
|
||||||
ext.VMMu.Lock()
|
ext.VMMu.Lock()
|
||||||
defer ext.VMMu.Unlock()
|
defer ext.VMMu.Unlock()
|
||||||
return initializeVMLocked(ext)
|
return initializeVMLocked(ext)
|
||||||
}
|
}
|
||||||
|
|
||||||
func initializeExtensionWithSettingsLocked(
|
func initializeExtensionWithSettingsLocked(
|
||||||
ext *LoadedExtension,
|
ext *loadedExtension,
|
||||||
settings map[string]interface{},
|
settings map[string]interface{},
|
||||||
) error {
|
) error {
|
||||||
if ext.VM == nil {
|
if ext.VM == nil {
|
||||||
@@ -405,7 +405,7 @@ func initializeExtensionWithSettingsLocked(
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func runCleanupLocked(ext *LoadedExtension) error {
|
func runCleanupLocked(ext *loadedExtension) error {
|
||||||
if ext.VM != nil {
|
if ext.VM != nil {
|
||||||
script := `
|
script := `
|
||||||
(function() {
|
(function() {
|
||||||
@@ -446,7 +446,7 @@ func runCleanupLocked(ext *LoadedExtension) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func teardownVMLocked(ext *LoadedExtension) {
|
func teardownVMLocked(ext *loadedExtension) {
|
||||||
if err := runCleanupLocked(ext); err != nil {
|
if err := runCleanupLocked(ext); err != nil {
|
||||||
GoLog("[Extension] Error calling cleanup for %s: %v\n", ext.ID, err)
|
GoLog("[Extension] Error calling cleanup for %s: %v\n", ext.ID, err)
|
||||||
}
|
}
|
||||||
@@ -461,7 +461,7 @@ func teardownVMLocked(ext *LoadedExtension) {
|
|||||||
ext.initialized = false
|
ext.initialized = false
|
||||||
}
|
}
|
||||||
|
|
||||||
func validateExtensionLoad(ext *LoadedExtension) error {
|
func validateExtensionLoad(ext *loadedExtension) error {
|
||||||
ext.VMMu.Lock()
|
ext.VMMu.Lock()
|
||||||
defer ext.VMMu.Unlock()
|
defer ext.VMMu.Unlock()
|
||||||
|
|
||||||
@@ -472,7 +472,7 @@ func validateExtensionLoad(ext *LoadedExtension) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) UnloadExtension(extensionID string) error {
|
func (m *extensionManager) UnloadExtension(extensionID string) error {
|
||||||
m.mu.Lock()
|
m.mu.Lock()
|
||||||
defer m.mu.Unlock()
|
defer m.mu.Unlock()
|
||||||
|
|
||||||
@@ -491,7 +491,7 @@ func (m *ExtensionManager) UnloadExtension(extensionID string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) GetExtension(extensionID string) (*LoadedExtension, error) {
|
func (m *extensionManager) GetExtension(extensionID string) (*loadedExtension, error) {
|
||||||
m.mu.RLock()
|
m.mu.RLock()
|
||||||
defer m.mu.RUnlock()
|
defer m.mu.RUnlock()
|
||||||
|
|
||||||
@@ -502,18 +502,18 @@ func (m *ExtensionManager) GetExtension(extensionID string) (*LoadedExtension, e
|
|||||||
return ext, nil
|
return ext, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) GetAllExtensions() []*LoadedExtension {
|
func (m *extensionManager) GetAllExtensions() []*loadedExtension {
|
||||||
m.mu.RLock()
|
m.mu.RLock()
|
||||||
defer m.mu.RUnlock()
|
defer m.mu.RUnlock()
|
||||||
|
|
||||||
result := make([]*LoadedExtension, 0, len(m.extensions))
|
result := make([]*loadedExtension, 0, len(m.extensions))
|
||||||
for _, ext := range m.extensions {
|
for _, ext := range m.extensions {
|
||||||
result = append(result, ext)
|
result = append(result, ext)
|
||||||
}
|
}
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) SetExtensionEnabled(extensionID string, enabled bool) error {
|
func (m *extensionManager) SetExtensionEnabled(extensionID string, enabled bool) error {
|
||||||
m.mu.Lock()
|
m.mu.Lock()
|
||||||
defer m.mu.Unlock()
|
defer m.mu.Unlock()
|
||||||
|
|
||||||
@@ -547,7 +547,7 @@ func (m *ExtensionManager) SetExtensionEnabled(extensionID string, enabled bool)
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) LoadExtensionsFromDirectory(dirPath string) ([]string, []error) {
|
func (m *extensionManager) LoadExtensionsFromDirectory(dirPath string) ([]string, []error) {
|
||||||
var loaded []string
|
var loaded []string
|
||||||
var errors []error
|
var errors []error
|
||||||
|
|
||||||
@@ -585,7 +585,7 @@ func (m *ExtensionManager) LoadExtensionsFromDirectory(dirPath string) ([]string
|
|||||||
return loaded, errors
|
return loaded, errors
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) loadExtensionFromDirectory(dirPath string) (*LoadedExtension, error) {
|
func (m *extensionManager) loadExtensionFromDirectory(dirPath string) (*loadedExtension, error) {
|
||||||
m.mu.Lock()
|
m.mu.Lock()
|
||||||
defer m.mu.Unlock()
|
defer m.mu.Unlock()
|
||||||
|
|
||||||
@@ -615,7 +615,7 @@ func (m *ExtensionManager) loadExtensionFromDirectory(dirPath string) (*LoadedEx
|
|||||||
return nil, fmt.Errorf("failed to create extension data directory: %w", err)
|
return nil, fmt.Errorf("failed to create extension data directory: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
ext := &LoadedExtension{
|
ext := &loadedExtension{
|
||||||
ID: manifest.Name,
|
ID: manifest.Name,
|
||||||
Manifest: manifest,
|
Manifest: manifest,
|
||||||
Enabled: false, // Will be restored from settings store
|
Enabled: false, // Will be restored from settings store
|
||||||
@@ -643,7 +643,7 @@ func (m *ExtensionManager) loadExtensionFromDirectory(dirPath string) (*LoadedEx
|
|||||||
return ext, nil
|
return ext, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) RemoveExtension(extensionID string) error {
|
func (m *extensionManager) RemoveExtension(extensionID string) error {
|
||||||
ext, err := m.GetExtension(extensionID)
|
ext, err := m.GetExtension(extensionID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@@ -663,7 +663,7 @@ func (m *ExtensionManager) RemoveExtension(extensionID string) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Only allows upgrades (new version > current version), not downgrades
|
// Only allows upgrades (new version > current version), not downgrades
|
||||||
func (m *ExtensionManager) UpgradeExtension(filePath string) (*LoadedExtension, error) {
|
func (m *extensionManager) UpgradeExtension(filePath string) (*loadedExtension, error) {
|
||||||
if !strings.HasSuffix(strings.ToLower(filePath), ".spotiflac-ext") {
|
if !strings.HasSuffix(strings.ToLower(filePath), ".spotiflac-ext") {
|
||||||
return nil, fmt.Errorf("Invalid file format. Please select a .spotiflac-ext file")
|
return nil, fmt.Errorf("Invalid file format. Please select a .spotiflac-ext file")
|
||||||
}
|
}
|
||||||
@@ -777,7 +777,7 @@ func (m *ExtensionManager) UpgradeExtension(filePath string) (*LoadedExtension,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ext := &LoadedExtension{
|
ext := &loadedExtension{
|
||||||
ID: newManifest.Name,
|
ID: newManifest.Name,
|
||||||
Manifest: newManifest,
|
Manifest: newManifest,
|
||||||
Enabled: wasEnabled, // Preserve enabled state from before upgrade
|
Enabled: wasEnabled, // Preserve enabled state from before upgrade
|
||||||
@@ -812,7 +812,7 @@ type ExtensionUpgradeInfo struct {
|
|||||||
IsInstalled bool `json:"is_installed"`
|
IsInstalled bool `json:"is_installed"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) checkExtensionUpgradeInternal(filePath string) (*ExtensionUpgradeInfo, error) {
|
func (m *extensionManager) checkExtensionUpgradeInternal(filePath string) (*ExtensionUpgradeInfo, error) {
|
||||||
if !strings.HasSuffix(strings.ToLower(filePath), ".spotiflac-ext") {
|
if !strings.HasSuffix(strings.ToLower(filePath), ".spotiflac-ext") {
|
||||||
return nil, fmt.Errorf("Invalid file format. Please select a .spotiflac-ext file")
|
return nil, fmt.Errorf("Invalid file format. Please select a .spotiflac-ext file")
|
||||||
}
|
}
|
||||||
@@ -871,7 +871,7 @@ func (m *ExtensionManager) checkExtensionUpgradeInternal(filePath string) (*Exte
|
|||||||
return info, nil
|
return info, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) CheckExtensionUpgradeJSON(filePath string) (string, error) {
|
func (m *extensionManager) CheckExtensionUpgradeJSON(filePath string) (string, error) {
|
||||||
info, err := m.checkExtensionUpgradeInternal(filePath)
|
info, err := m.checkExtensionUpgradeInternal(filePath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
@@ -885,7 +885,7 @@ func (m *ExtensionManager) CheckExtensionUpgradeJSON(filePath string) (string, e
|
|||||||
return string(jsonBytes), nil
|
return string(jsonBytes), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) GetInstalledExtensionsJSON() (string, error) {
|
func (m *extensionManager) GetInstalledExtensionsJSON() (string, error) {
|
||||||
extensions := m.GetAllExtensions()
|
extensions := m.GetAllExtensions()
|
||||||
|
|
||||||
type ExtensionInfo struct {
|
type ExtensionInfo struct {
|
||||||
@@ -908,6 +908,7 @@ func (m *ExtensionManager) GetInstalledExtensionsJSON() (string, error) {
|
|||||||
HasDownloadProvider bool `json:"has_download_provider"`
|
HasDownloadProvider bool `json:"has_download_provider"`
|
||||||
HasLyricsProvider bool `json:"has_lyrics_provider"`
|
HasLyricsProvider bool `json:"has_lyrics_provider"`
|
||||||
SkipMetadataEnrichment bool `json:"skip_metadata_enrichment"`
|
SkipMetadataEnrichment bool `json:"skip_metadata_enrichment"`
|
||||||
|
SkipLyrics bool `json:"skip_lyrics"`
|
||||||
SearchBehavior *SearchBehaviorConfig `json:"search_behavior,omitempty"`
|
SearchBehavior *SearchBehaviorConfig `json:"search_behavior,omitempty"`
|
||||||
TrackMatching *TrackMatchingConfig `json:"track_matching,omitempty"`
|
TrackMatching *TrackMatchingConfig `json:"track_matching,omitempty"`
|
||||||
PostProcessing *PostProcessingConfig `json:"post_processing,omitempty"`
|
PostProcessing *PostProcessingConfig `json:"post_processing,omitempty"`
|
||||||
@@ -965,6 +966,7 @@ func (m *ExtensionManager) GetInstalledExtensionsJSON() (string, error) {
|
|||||||
HasDownloadProvider: ext.Manifest.IsDownloadProvider(),
|
HasDownloadProvider: ext.Manifest.IsDownloadProvider(),
|
||||||
HasLyricsProvider: ext.Manifest.IsLyricsProvider(),
|
HasLyricsProvider: ext.Manifest.IsLyricsProvider(),
|
||||||
SkipMetadataEnrichment: ext.Manifest.SkipMetadataEnrichment,
|
SkipMetadataEnrichment: ext.Manifest.SkipMetadataEnrichment,
|
||||||
|
SkipLyrics: ext.Manifest.SkipLyrics,
|
||||||
SearchBehavior: ext.Manifest.SearchBehavior,
|
SearchBehavior: ext.Manifest.SearchBehavior,
|
||||||
TrackMatching: ext.Manifest.TrackMatching,
|
TrackMatching: ext.Manifest.TrackMatching,
|
||||||
PostProcessing: ext.Manifest.PostProcessing,
|
PostProcessing: ext.Manifest.PostProcessing,
|
||||||
@@ -980,7 +982,7 @@ func (m *ExtensionManager) GetInstalledExtensionsJSON() (string, error) {
|
|||||||
return string(jsonBytes), nil
|
return string(jsonBytes), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) InitializeExtension(extensionID string, settings map[string]interface{}) error {
|
func (m *extensionManager) InitializeExtension(extensionID string, settings map[string]interface{}) error {
|
||||||
m.mu.Lock()
|
m.mu.Lock()
|
||||||
defer m.mu.Unlock()
|
defer m.mu.Unlock()
|
||||||
|
|
||||||
@@ -998,7 +1000,7 @@ func (m *ExtensionManager) InitializeExtension(extensionID string, settings map[
|
|||||||
return initializeExtensionWithSettingsLocked(ext, settings)
|
return initializeExtensionWithSettingsLocked(ext, settings)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) CleanupExtension(extensionID string) error {
|
func (m *extensionManager) CleanupExtension(extensionID string) error {
|
||||||
m.mu.Lock()
|
m.mu.Lock()
|
||||||
defer m.mu.Unlock()
|
defer m.mu.Unlock()
|
||||||
|
|
||||||
@@ -1020,7 +1022,7 @@ func (m *ExtensionManager) CleanupExtension(extensionID string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) UnloadAllExtensions() {
|
func (m *extensionManager) UnloadAllExtensions() {
|
||||||
m.mu.Lock()
|
m.mu.Lock()
|
||||||
extensionIDs := make([]string, 0, len(m.extensions))
|
extensionIDs := make([]string, 0, len(m.extensions))
|
||||||
for id := range m.extensions {
|
for id := range m.extensions {
|
||||||
@@ -1035,7 +1037,7 @@ func (m *ExtensionManager) UnloadAllExtensions() {
|
|||||||
GoLog("[Extension] All extensions unloaded\n")
|
GoLog("[Extension] All extensions unloaded\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) InvokeAction(extensionID string, actionName string) (map[string]interface{}, error) {
|
func (m *extensionManager) InvokeAction(extensionID string, actionName string) (map[string]interface{}, error) {
|
||||||
m.mu.Lock()
|
m.mu.Lock()
|
||||||
defer m.mu.Unlock()
|
defer m.mu.Unlock()
|
||||||
|
|
||||||
@@ -1044,13 +1046,14 @@ func (m *ExtensionManager) InvokeAction(extensionID string, actionName string) (
|
|||||||
return nil, fmt.Errorf("extension not found: %s", extensionID)
|
return nil, fmt.Errorf("extension not found: %s", extensionID)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := ext.ensureRuntimeReady(); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if !ext.Enabled {
|
if !ext.Enabled {
|
||||||
return nil, fmt.Errorf("extension is disabled")
|
return nil, fmt.Errorf("extension is disabled")
|
||||||
}
|
}
|
||||||
|
vm, err := ext.lockReadyVM()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer ext.VMMu.Unlock()
|
||||||
|
|
||||||
script := fmt.Sprintf(`
|
script := fmt.Sprintf(`
|
||||||
(function() {
|
(function() {
|
||||||
@@ -1070,7 +1073,7 @@ func (m *ExtensionManager) InvokeAction(extensionID string, actionName string) (
|
|||||||
})()
|
})()
|
||||||
`, actionName, actionName, actionName)
|
`, actionName, actionName, actionName)
|
||||||
|
|
||||||
result, err := RunWithTimeoutAndRecover(ext.VM, script, DefaultJSTimeout)
|
result, err := RunWithTimeoutAndRecover(vm, script, DefaultJSTimeout)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
GoLog("[Extension] InvokeAction error for %s.%s: %v\n", extensionID, actionName, err)
|
GoLog("[Extension] InvokeAction error for %s.%s: %v\n", extensionID, actionName, err)
|
||||||
return nil, fmt.Errorf("action failed: %v", err)
|
return nil, fmt.Errorf("action failed: %v", err)
|
||||||
|
|||||||
@@ -115,6 +115,7 @@ type ExtensionManifest struct {
|
|||||||
QualityOptions []QualityOption `json:"qualityOptions,omitempty"`
|
QualityOptions []QualityOption `json:"qualityOptions,omitempty"`
|
||||||
MinAppVersion string `json:"minAppVersion,omitempty"`
|
MinAppVersion string `json:"minAppVersion,omitempty"`
|
||||||
SkipMetadataEnrichment bool `json:"skipMetadataEnrichment,omitempty"`
|
SkipMetadataEnrichment bool `json:"skipMetadataEnrichment,omitempty"`
|
||||||
|
SkipLyrics bool `json:"skipLyrics,omitempty"`
|
||||||
SkipBuiltInFallback bool `json:"skipBuiltInFallback,omitempty"`
|
SkipBuiltInFallback bool `json:"skipBuiltInFallback,omitempty"`
|
||||||
SearchBehavior *SearchBehaviorConfig `json:"searchBehavior,omitempty"`
|
SearchBehavior *SearchBehaviorConfig `json:"searchBehavior,omitempty"`
|
||||||
URLHandler *URLHandlerConfig `json:"urlHandler,omitempty"`
|
URLHandler *URLHandlerConfig `json:"urlHandler,omitempty"`
|
||||||
|
|||||||
@@ -26,7 +26,9 @@ type ExtTrackMetadata struct {
|
|||||||
Images string `json:"images,omitempty"`
|
Images string `json:"images,omitempty"`
|
||||||
ReleaseDate string `json:"release_date,omitempty"`
|
ReleaseDate string `json:"release_date,omitempty"`
|
||||||
TrackNumber int `json:"track_number,omitempty"`
|
TrackNumber int `json:"track_number,omitempty"`
|
||||||
|
TotalTracks int `json:"total_tracks,omitempty"`
|
||||||
DiscNumber int `json:"disc_number,omitempty"`
|
DiscNumber int `json:"disc_number,omitempty"`
|
||||||
|
TotalDiscs int `json:"total_discs,omitempty"`
|
||||||
ISRC string `json:"isrc,omitempty"`
|
ISRC string `json:"isrc,omitempty"`
|
||||||
ProviderID string `json:"provider_id"`
|
ProviderID string `json:"provider_id"`
|
||||||
ItemType string `json:"item_type,omitempty"`
|
ItemType string `json:"item_type,omitempty"`
|
||||||
@@ -41,6 +43,7 @@ type ExtTrackMetadata struct {
|
|||||||
Label string `json:"label,omitempty"`
|
Label string `json:"label,omitempty"`
|
||||||
Copyright string `json:"copyright,omitempty"`
|
Copyright string `json:"copyright,omitempty"`
|
||||||
Genre string `json:"genre,omitempty"`
|
Genre string `json:"genre,omitempty"`
|
||||||
|
Composer string `json:"composer,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *ExtTrackMetadata) ResolvedCoverURL() string {
|
func (t *ExtTrackMetadata) ResolvedCoverURL() string {
|
||||||
@@ -113,19 +116,19 @@ type ExtDownloadResult struct {
|
|||||||
DecryptionKey string `json:"decryption_key,omitempty"`
|
DecryptionKey string `json:"decryption_key,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type ExtensionProviderWrapper struct {
|
type extensionProviderWrapper struct {
|
||||||
extension *LoadedExtension
|
extension *loadedExtension
|
||||||
vm *goja.Runtime
|
vm *goja.Runtime
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewExtensionProviderWrapper(ext *LoadedExtension) *ExtensionProviderWrapper {
|
func newExtensionProviderWrapper(ext *loadedExtension) *extensionProviderWrapper {
|
||||||
return &ExtensionProviderWrapper{
|
return &extensionProviderWrapper{
|
||||||
extension: ext,
|
extension: ext,
|
||||||
vm: ext.VM,
|
vm: ext.VM,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ExtensionProviderWrapper) lockReadyVM() error {
|
func (p *extensionProviderWrapper) lockReadyVM() error {
|
||||||
vm, err := p.extension.lockReadyVM()
|
vm, err := p.extension.lockReadyVM()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@@ -134,7 +137,7 @@ func (p *ExtensionProviderWrapper) lockReadyVM() error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ExtensionProviderWrapper) SearchTracks(query string, limit int) (*ExtSearchResult, error) {
|
func (p *extensionProviderWrapper) SearchTracks(query string, limit int) (*ExtSearchResult, error) {
|
||||||
if !p.extension.Manifest.IsMetadataProvider() {
|
if !p.extension.Manifest.IsMetadataProvider() {
|
||||||
return nil, fmt.Errorf("extension '%s' is not a metadata provider", p.extension.ID)
|
return nil, fmt.Errorf("extension '%s' is not a metadata provider", p.extension.ID)
|
||||||
}
|
}
|
||||||
@@ -194,7 +197,7 @@ func (p *ExtensionProviderWrapper) SearchTracks(query string, limit int) (*ExtSe
|
|||||||
return &searchResult, nil
|
return &searchResult, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ExtensionProviderWrapper) GetTrack(trackID string) (*ExtTrackMetadata, error) {
|
func (p *extensionProviderWrapper) GetTrack(trackID string) (*ExtTrackMetadata, error) {
|
||||||
if !p.extension.Manifest.IsMetadataProvider() {
|
if !p.extension.Manifest.IsMetadataProvider() {
|
||||||
return nil, fmt.Errorf("extension '%s' is not a metadata provider", p.extension.ID)
|
return nil, fmt.Errorf("extension '%s' is not a metadata provider", p.extension.ID)
|
||||||
}
|
}
|
||||||
@@ -243,7 +246,7 @@ func (p *ExtensionProviderWrapper) GetTrack(trackID string) (*ExtTrackMetadata,
|
|||||||
return &track, nil
|
return &track, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ExtensionProviderWrapper) GetAlbum(albumID string) (*ExtAlbumMetadata, error) {
|
func (p *extensionProviderWrapper) GetAlbum(albumID string) (*ExtAlbumMetadata, error) {
|
||||||
if !p.extension.Manifest.IsMetadataProvider() {
|
if !p.extension.Manifest.IsMetadataProvider() {
|
||||||
return nil, fmt.Errorf("extension '%s' is not a metadata provider", p.extension.ID)
|
return nil, fmt.Errorf("extension '%s' is not a metadata provider", p.extension.ID)
|
||||||
}
|
}
|
||||||
@@ -295,7 +298,7 @@ func (p *ExtensionProviderWrapper) GetAlbum(albumID string) (*ExtAlbumMetadata,
|
|||||||
return &album, nil
|
return &album, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ExtensionProviderWrapper) GetArtist(artistID string) (*ExtArtistMetadata, error) {
|
func (p *extensionProviderWrapper) GetArtist(artistID string) (*ExtArtistMetadata, error) {
|
||||||
if !p.extension.Manifest.IsMetadataProvider() {
|
if !p.extension.Manifest.IsMetadataProvider() {
|
||||||
return nil, fmt.Errorf("extension '%s' is not a metadata provider", p.extension.ID)
|
return nil, fmt.Errorf("extension '%s' is not a metadata provider", p.extension.ID)
|
||||||
}
|
}
|
||||||
@@ -350,7 +353,7 @@ func (p *ExtensionProviderWrapper) GetArtist(artistID string) (*ExtArtistMetadat
|
|||||||
return &artist, nil
|
return &artist, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ExtensionProviderWrapper) EnrichTrack(track *ExtTrackMetadata) (*ExtTrackMetadata, error) {
|
func (p *extensionProviderWrapper) EnrichTrack(track *ExtTrackMetadata) (*ExtTrackMetadata, error) {
|
||||||
if !p.extension.Manifest.IsMetadataProvider() {
|
if !p.extension.Manifest.IsMetadataProvider() {
|
||||||
return track, nil
|
return track, nil
|
||||||
}
|
}
|
||||||
@@ -412,7 +415,7 @@ func (p *ExtensionProviderWrapper) EnrichTrack(track *ExtTrackMetadata) (*ExtTra
|
|||||||
return &enrichedTrack, nil
|
return &enrichedTrack, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ExtensionProviderWrapper) CheckAvailability(isrc, trackName, artistName, spotifyID, deezerID string) (*ExtAvailabilityResult, error) {
|
func (p *extensionProviderWrapper) CheckAvailability(isrc, trackName, artistName, spotifyID, deezerID string) (*ExtAvailabilityResult, error) {
|
||||||
if !p.extension.Manifest.IsDownloadProvider() {
|
if !p.extension.Manifest.IsDownloadProvider() {
|
||||||
return nil, fmt.Errorf("extension '%s' is not a download provider", p.extension.ID)
|
return nil, fmt.Errorf("extension '%s' is not a download provider", p.extension.ID)
|
||||||
}
|
}
|
||||||
@@ -460,7 +463,7 @@ func (p *ExtensionProviderWrapper) CheckAvailability(isrc, trackName, artistName
|
|||||||
return &availability, nil
|
return &availability, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ExtensionProviderWrapper) GetDownloadURL(trackID, quality string) (*ExtDownloadURLResult, error) {
|
func (p *extensionProviderWrapper) GetDownloadURL(trackID, quality string) (*ExtDownloadURLResult, error) {
|
||||||
if !p.extension.Manifest.IsDownloadProvider() {
|
if !p.extension.Manifest.IsDownloadProvider() {
|
||||||
return nil, fmt.Errorf("extension '%s' is not a download provider", p.extension.ID)
|
return nil, fmt.Errorf("extension '%s' is not a download provider", p.extension.ID)
|
||||||
}
|
}
|
||||||
@@ -510,7 +513,7 @@ func (p *ExtensionProviderWrapper) GetDownloadURL(trackID, quality string) (*Ext
|
|||||||
|
|
||||||
const ExtDownloadTimeout = DownloadTimeout
|
const ExtDownloadTimeout = DownloadTimeout
|
||||||
|
|
||||||
func (p *ExtensionProviderWrapper) Download(trackID, quality, outputPath, itemID string, onProgress func(percent int)) (*ExtDownloadResult, error) {
|
func (p *extensionProviderWrapper) Download(trackID, quality, outputPath, itemID string, onProgress func(percent int)) (*ExtDownloadResult, error) {
|
||||||
if !p.extension.Manifest.IsDownloadProvider() {
|
if !p.extension.Manifest.IsDownloadProvider() {
|
||||||
return nil, fmt.Errorf("extension '%s' is not a download provider", p.extension.ID)
|
return nil, fmt.Errorf("extension '%s' is not a download provider", p.extension.ID)
|
||||||
}
|
}
|
||||||
@@ -601,40 +604,40 @@ func (p *ExtensionProviderWrapper) Download(trackID, quality, outputPath, itemID
|
|||||||
return &downloadResult, nil
|
return &downloadResult, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) GetMetadataProviders() []*ExtensionProviderWrapper {
|
func (m *extensionManager) GetMetadataProviders() []*extensionProviderWrapper {
|
||||||
m.mu.RLock()
|
m.mu.RLock()
|
||||||
defer m.mu.RUnlock()
|
defer m.mu.RUnlock()
|
||||||
|
|
||||||
var providers []*ExtensionProviderWrapper
|
var providers []*extensionProviderWrapper
|
||||||
for _, ext := range m.extensions {
|
for _, ext := range m.extensions {
|
||||||
if ext.Enabled && ext.Manifest.IsMetadataProvider() && ext.Error == "" {
|
if ext.Enabled && ext.Manifest.IsMetadataProvider() && ext.Error == "" {
|
||||||
providers = append(providers, NewExtensionProviderWrapper(ext))
|
providers = append(providers, newExtensionProviderWrapper(ext))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return providers
|
return providers
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) GetDownloadProviders() []*ExtensionProviderWrapper {
|
func (m *extensionManager) GetDownloadProviders() []*extensionProviderWrapper {
|
||||||
m.mu.RLock()
|
m.mu.RLock()
|
||||||
defer m.mu.RUnlock()
|
defer m.mu.RUnlock()
|
||||||
|
|
||||||
var providers []*ExtensionProviderWrapper
|
var providers []*extensionProviderWrapper
|
||||||
for _, ext := range m.extensions {
|
for _, ext := range m.extensions {
|
||||||
if ext.Enabled && ext.Manifest.IsDownloadProvider() && ext.Error == "" {
|
if ext.Enabled && ext.Manifest.IsDownloadProvider() && ext.Error == "" {
|
||||||
providers = append(providers, NewExtensionProviderWrapper(ext))
|
providers = append(providers, newExtensionProviderWrapper(ext))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return providers
|
return providers
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) SearchTracksWithExtensions(query string, limit int) ([]ExtTrackMetadata, error) {
|
func (m *extensionManager) SearchTracksWithExtensions(query string, limit int) ([]ExtTrackMetadata, error) {
|
||||||
providers := m.GetMetadataProviders()
|
providers := m.GetMetadataProviders()
|
||||||
if len(providers) == 0 {
|
if len(providers) == 0 {
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
providerByID := make(map[string]*ExtensionProviderWrapper, len(providers))
|
providerByID := make(map[string]*extensionProviderWrapper, len(providers))
|
||||||
orderedProviders := make([]*ExtensionProviderWrapper, 0, len(providers))
|
orderedProviders := make([]*extensionProviderWrapper, 0, len(providers))
|
||||||
for _, provider := range providers {
|
for _, provider := range providers {
|
||||||
providerByID[provider.extension.ID] = provider
|
providerByID[provider.extension.ID] = provider
|
||||||
}
|
}
|
||||||
@@ -775,7 +778,9 @@ func normalizeBuiltInMetadataTrack(track TrackMetadata, providerID string) ExtTr
|
|||||||
Images: track.Images,
|
Images: track.Images,
|
||||||
ReleaseDate: track.ReleaseDate,
|
ReleaseDate: track.ReleaseDate,
|
||||||
TrackNumber: track.TrackNumber,
|
TrackNumber: track.TrackNumber,
|
||||||
|
TotalTracks: track.TotalTracks,
|
||||||
DiscNumber: track.DiscNumber,
|
DiscNumber: track.DiscNumber,
|
||||||
|
TotalDiscs: track.TotalDiscs,
|
||||||
ISRC: track.ISRC,
|
ISRC: track.ISRC,
|
||||||
ProviderID: providerID,
|
ProviderID: providerID,
|
||||||
SpotifyID: prefixedID,
|
SpotifyID: prefixedID,
|
||||||
@@ -783,6 +788,7 @@ func normalizeBuiltInMetadataTrack(track TrackMetadata, providerID string) ExtTr
|
|||||||
TidalID: tidalID,
|
TidalID: tidalID,
|
||||||
QobuzID: qobuzID,
|
QobuzID: qobuzID,
|
||||||
AlbumType: track.AlbumType,
|
AlbumType: track.AlbumType,
|
||||||
|
Composer: track.Composer,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -824,13 +830,13 @@ func searchBuiltInMetadataTracks(providerID, query string, limit int) ([]ExtTrac
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) SearchTracksWithMetadataProviders(query string, limit int, includeExtensions bool) ([]ExtTrackMetadata, error) {
|
func (m *extensionManager) SearchTracksWithMetadataProviders(query string, limit int, includeExtensions bool) ([]ExtTrackMetadata, error) {
|
||||||
priority := GetMetadataProviderPriority()
|
priority := GetMetadataProviderPriority()
|
||||||
if limit <= 0 {
|
if limit <= 0 {
|
||||||
limit = 20
|
limit = 20
|
||||||
}
|
}
|
||||||
|
|
||||||
extensionProviders := make(map[string]*ExtensionProviderWrapper)
|
extensionProviders := make(map[string]*extensionProviderWrapper)
|
||||||
if includeExtensions {
|
if includeExtensions {
|
||||||
for _, provider := range m.GetMetadataProviders() {
|
for _, provider := range m.GetMetadataProviders() {
|
||||||
extensionProviders[provider.extension.ID] = provider
|
extensionProviders[provider.extension.ID] = provider
|
||||||
@@ -910,7 +916,7 @@ func (m *ExtensionManager) SearchTracksWithMetadataProviders(query string, limit
|
|||||||
|
|
||||||
func DownloadWithExtensionFallback(req DownloadRequest) (*DownloadResponse, error) {
|
func DownloadWithExtensionFallback(req DownloadRequest) (*DownloadResponse, error) {
|
||||||
priority := GetProviderPriority()
|
priority := GetProviderPriority()
|
||||||
extManager := GetExtensionManager()
|
extManager := getExtensionManager()
|
||||||
strictMode := !req.UseFallback
|
strictMode := !req.UseFallback
|
||||||
selectedProvider := strings.TrimSpace(req.Service)
|
selectedProvider := strings.TrimSpace(req.Service)
|
||||||
|
|
||||||
@@ -965,7 +971,7 @@ func DownloadWithExtensionFallback(req DownloadRequest) (*DownloadResponse, erro
|
|||||||
if err == nil && ext.Enabled && ext.Error == "" && ext.Manifest.IsMetadataProvider() {
|
if err == nil && ext.Enabled && ext.Error == "" && ext.Manifest.IsMetadataProvider() {
|
||||||
GoLog("[DownloadWithExtensionFallback] Enriching track from extension '%s'...\n", req.Source)
|
GoLog("[DownloadWithExtensionFallback] Enriching track from extension '%s'...\n", req.Source)
|
||||||
|
|
||||||
provider := NewExtensionProviderWrapper(ext)
|
provider := newExtensionProviderWrapper(ext)
|
||||||
trackMeta := &ExtTrackMetadata{
|
trackMeta := &ExtTrackMetadata{
|
||||||
ID: req.SpotifyID,
|
ID: req.SpotifyID,
|
||||||
Name: req.TrackName,
|
Name: req.TrackName,
|
||||||
@@ -975,8 +981,11 @@ func DownloadWithExtensionFallback(req DownloadRequest) (*DownloadResponse, erro
|
|||||||
ISRC: req.ISRC,
|
ISRC: req.ISRC,
|
||||||
ReleaseDate: req.ReleaseDate,
|
ReleaseDate: req.ReleaseDate,
|
||||||
TrackNumber: req.TrackNumber,
|
TrackNumber: req.TrackNumber,
|
||||||
|
TotalTracks: req.TotalTracks,
|
||||||
DiscNumber: req.DiscNumber,
|
DiscNumber: req.DiscNumber,
|
||||||
|
TotalDiscs: req.TotalDiscs,
|
||||||
ProviderID: req.Source,
|
ProviderID: req.Source,
|
||||||
|
Composer: req.Composer,
|
||||||
}
|
}
|
||||||
|
|
||||||
enrichedTrack, err := provider.EnrichTrack(trackMeta)
|
enrichedTrack, err := provider.EnrichTrack(trackMeta)
|
||||||
@@ -1037,13 +1046,30 @@ func DownloadWithExtensionFallback(req DownloadRequest) (*DownloadResponse, erro
|
|||||||
GoLog("[DownloadWithExtensionFallback] ReleaseDate from enrichment: %s\n", enrichedTrack.ReleaseDate)
|
GoLog("[DownloadWithExtensionFallback] ReleaseDate from enrichment: %s\n", enrichedTrack.ReleaseDate)
|
||||||
req.ReleaseDate = enrichedTrack.ReleaseDate
|
req.ReleaseDate = enrichedTrack.ReleaseDate
|
||||||
}
|
}
|
||||||
|
if enrichedTrack.TrackNumber > 0 && req.TrackNumber == 0 {
|
||||||
|
GoLog("[DownloadWithExtensionFallback] TrackNumber from enrichment: %d\n", enrichedTrack.TrackNumber)
|
||||||
|
req.TrackNumber = enrichedTrack.TrackNumber
|
||||||
|
}
|
||||||
|
if enrichedTrack.TotalTracks > 0 && req.TotalTracks == 0 {
|
||||||
|
GoLog("[DownloadWithExtensionFallback] TotalTracks from enrichment: %d\n", enrichedTrack.TotalTracks)
|
||||||
|
req.TotalTracks = enrichedTrack.TotalTracks
|
||||||
|
}
|
||||||
|
if enrichedTrack.DiscNumber > 0 && req.DiscNumber == 0 {
|
||||||
|
GoLog("[DownloadWithExtensionFallback] DiscNumber from enrichment: %d\n", enrichedTrack.DiscNumber)
|
||||||
|
req.DiscNumber = enrichedTrack.DiscNumber
|
||||||
|
}
|
||||||
|
if enrichedTrack.TotalDiscs > 0 && req.TotalDiscs == 0 {
|
||||||
|
GoLog("[DownloadWithExtensionFallback] TotalDiscs from enrichment: %d\n", enrichedTrack.TotalDiscs)
|
||||||
|
req.TotalDiscs = enrichedTrack.TotalDiscs
|
||||||
|
}
|
||||||
|
if enrichedTrack.Composer != "" && req.Composer == "" {
|
||||||
|
GoLog("[DownloadWithExtensionFallback] Composer from enrichment: %s\n", enrichedTrack.Composer)
|
||||||
|
req.Composer = enrichedTrack.Composer
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// If key metadata is still missing after extension enrichment, search
|
|
||||||
// configured metadata providers (Spotify/Deezer/Tidal/Qobuz) — same
|
|
||||||
// logic that ReEnrichFile uses.
|
|
||||||
if req.Source != "" && !isBuiltInProvider(strings.ToLower(req.Source)) &&
|
if req.Source != "" && !isBuiltInProvider(strings.ToLower(req.Source)) &&
|
||||||
req.TrackName != "" && req.ArtistName != "" &&
|
req.TrackName != "" && req.ArtistName != "" &&
|
||||||
(req.AlbumName == "" || req.ReleaseDate == "" || req.ISRC == "") {
|
(req.AlbumName == "" || req.ReleaseDate == "" || req.ISRC == "") {
|
||||||
@@ -1072,9 +1098,18 @@ func DownloadWithExtensionFallback(req DownloadRequest) (*DownloadResponse, erro
|
|||||||
if track.TrackNumber > 0 && req.TrackNumber == 0 {
|
if track.TrackNumber > 0 && req.TrackNumber == 0 {
|
||||||
req.TrackNumber = track.TrackNumber
|
req.TrackNumber = track.TrackNumber
|
||||||
}
|
}
|
||||||
|
if track.TotalTracks > 0 && req.TotalTracks == 0 {
|
||||||
|
req.TotalTracks = track.TotalTracks
|
||||||
|
}
|
||||||
if track.DiscNumber > 0 && req.DiscNumber == 0 {
|
if track.DiscNumber > 0 && req.DiscNumber == 0 {
|
||||||
req.DiscNumber = track.DiscNumber
|
req.DiscNumber = track.DiscNumber
|
||||||
}
|
}
|
||||||
|
if track.TotalDiscs > 0 && req.TotalDiscs == 0 {
|
||||||
|
req.TotalDiscs = track.TotalDiscs
|
||||||
|
}
|
||||||
|
if track.Composer != "" && req.Composer == "" {
|
||||||
|
req.Composer = track.Composer
|
||||||
|
}
|
||||||
if track.CoverURL != "" && req.CoverURL == "" {
|
if track.CoverURL != "" && req.CoverURL == "" {
|
||||||
req.CoverURL = track.CoverURL
|
req.CoverURL = track.CoverURL
|
||||||
}
|
}
|
||||||
@@ -1091,7 +1126,6 @@ func DownloadWithExtensionFallback(req DownloadRequest) (*DownloadResponse, erro
|
|||||||
GoLog("[DownloadWithExtensionFallback] Metadata provider search failed (non-fatal): %v\n", searchErr)
|
GoLog("[DownloadWithExtensionFallback] Metadata provider search failed (non-fatal): %v\n", searchErr)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try Deezer extended metadata if we have ISRC
|
|
||||||
if req.ISRC != "" &&
|
if req.ISRC != "" &&
|
||||||
(req.Genre == "" || req.Label == "" || req.Copyright == "") {
|
(req.Genre == "" || req.Label == "" || req.Copyright == "") {
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
|
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
|
||||||
@@ -1121,7 +1155,7 @@ func DownloadWithExtensionFallback(req DownloadRequest) (*DownloadResponse, erro
|
|||||||
if err == nil && ext.Enabled && ext.Error == "" && ext.Manifest.IsDownloadProvider() {
|
if err == nil && ext.Enabled && ext.Error == "" && ext.Manifest.IsDownloadProvider() {
|
||||||
skipBuiltIn = ext.Manifest.SkipBuiltInFallback
|
skipBuiltIn = ext.Manifest.SkipBuiltInFallback
|
||||||
|
|
||||||
provider := NewExtensionProviderWrapper(ext)
|
provider := newExtensionProviderWrapper(ext)
|
||||||
|
|
||||||
trackID := req.SpotifyID
|
trackID := req.SpotifyID
|
||||||
|
|
||||||
@@ -1205,8 +1239,6 @@ func DownloadWithExtensionFallback(req DownloadRequest) (*DownloadResponse, erro
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Always pass enriched metadata from req so Flutter can
|
|
||||||
// embed it — fills gaps from metadata provider search.
|
|
||||||
if req.AlbumName != "" && resp.Album == "" {
|
if req.AlbumName != "" && resp.Album == "" {
|
||||||
resp.Album = req.AlbumName
|
resp.Album = req.AlbumName
|
||||||
}
|
}
|
||||||
@@ -1344,7 +1376,7 @@ func DownloadWithExtensionFallback(req DownloadRequest) (*DownloadResponse, erro
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
provider := NewExtensionProviderWrapper(ext)
|
provider := newExtensionProviderWrapper(ext)
|
||||||
|
|
||||||
availability, err := provider.CheckAvailability(req.ISRC, req.TrackName, req.ArtistName, req.SpotifyID, req.DeezerID)
|
availability, err := provider.CheckAvailability(req.ISRC, req.TrackName, req.ArtistName, req.SpotifyID, req.DeezerID)
|
||||||
if err != nil || !availability.Available {
|
if err != nil || !availability.Available {
|
||||||
@@ -1433,6 +1465,28 @@ func DownloadWithExtensionFallback(req DownloadRequest) (*DownloadResponse, erro
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if req.AlbumName != "" && resp.Album == "" {
|
||||||
|
resp.Album = req.AlbumName
|
||||||
|
}
|
||||||
|
if req.AlbumArtist != "" && resp.AlbumArtist == "" {
|
||||||
|
resp.AlbumArtist = req.AlbumArtist
|
||||||
|
}
|
||||||
|
if req.ReleaseDate != "" && resp.ReleaseDate == "" {
|
||||||
|
resp.ReleaseDate = req.ReleaseDate
|
||||||
|
}
|
||||||
|
if req.ISRC != "" && resp.ISRC == "" {
|
||||||
|
resp.ISRC = req.ISRC
|
||||||
|
}
|
||||||
|
if req.TrackNumber > 0 && resp.TrackNumber == 0 {
|
||||||
|
resp.TrackNumber = req.TrackNumber
|
||||||
|
}
|
||||||
|
if req.DiscNumber > 0 && resp.DiscNumber == 0 {
|
||||||
|
resp.DiscNumber = req.DiscNumber
|
||||||
|
}
|
||||||
|
if req.CoverURL != "" && resp.CoverURL == "" {
|
||||||
|
resp.CoverURL = req.CoverURL
|
||||||
|
}
|
||||||
|
|
||||||
return resp, nil
|
return resp, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1570,12 +1624,15 @@ func buildOutputPath(req DownloadRequest) string {
|
|||||||
"album_artist": req.AlbumArtist,
|
"album_artist": req.AlbumArtist,
|
||||||
"track": req.TrackNumber,
|
"track": req.TrackNumber,
|
||||||
"track_number": req.TrackNumber,
|
"track_number": req.TrackNumber,
|
||||||
|
"total_tracks": req.TotalTracks,
|
||||||
"disc": req.DiscNumber,
|
"disc": req.DiscNumber,
|
||||||
"disc_number": req.DiscNumber,
|
"disc_number": req.DiscNumber,
|
||||||
|
"total_discs": req.TotalDiscs,
|
||||||
"year": extractYear(req.ReleaseDate),
|
"year": extractYear(req.ReleaseDate),
|
||||||
"date": req.ReleaseDate,
|
"date": req.ReleaseDate,
|
||||||
"release_date": req.ReleaseDate,
|
"release_date": req.ReleaseDate,
|
||||||
"isrc": req.ISRC,
|
"isrc": req.ISRC,
|
||||||
|
"composer": req.Composer,
|
||||||
}
|
}
|
||||||
|
|
||||||
filename := buildFilenameFromTemplate(req.FilenameFormat, metadata)
|
filename := buildFilenameFromTemplate(req.FilenameFormat, metadata)
|
||||||
@@ -1600,7 +1657,7 @@ func buildOutputPath(req DownloadRequest) string {
|
|||||||
return filepath.Join(outputDir, filename+ext)
|
return filepath.Join(outputDir, filename+ext)
|
||||||
}
|
}
|
||||||
|
|
||||||
func buildOutputPathForExtension(req DownloadRequest, ext *LoadedExtension) string {
|
func buildOutputPathForExtension(req DownloadRequest, ext *loadedExtension) string {
|
||||||
if strings.TrimSpace(req.OutputPath) != "" {
|
if strings.TrimSpace(req.OutputPath) != "" {
|
||||||
return strings.TrimSpace(req.OutputPath)
|
return strings.TrimSpace(req.OutputPath)
|
||||||
}
|
}
|
||||||
@@ -1609,7 +1666,6 @@ func buildOutputPathForExtension(req DownloadRequest, ext *LoadedExtension) stri
|
|||||||
return buildOutputPath(req)
|
return buildOutputPath(req)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SAF mode: use extension's data dir as writable temp location
|
|
||||||
tempDir := filepath.Join(ext.DataDir, "downloads")
|
tempDir := filepath.Join(ext.DataDir, "downloads")
|
||||||
os.MkdirAll(tempDir, 0755)
|
os.MkdirAll(tempDir, 0755)
|
||||||
AddAllowedDownloadDir(tempDir)
|
AddAllowedDownloadDir(tempDir)
|
||||||
@@ -1621,12 +1677,15 @@ func buildOutputPathForExtension(req DownloadRequest, ext *LoadedExtension) stri
|
|||||||
"album_artist": req.AlbumArtist,
|
"album_artist": req.AlbumArtist,
|
||||||
"track": req.TrackNumber,
|
"track": req.TrackNumber,
|
||||||
"track_number": req.TrackNumber,
|
"track_number": req.TrackNumber,
|
||||||
|
"total_tracks": req.TotalTracks,
|
||||||
"disc": req.DiscNumber,
|
"disc": req.DiscNumber,
|
||||||
"disc_number": req.DiscNumber,
|
"disc_number": req.DiscNumber,
|
||||||
|
"total_discs": req.TotalDiscs,
|
||||||
"year": extractYear(req.ReleaseDate),
|
"year": extractYear(req.ReleaseDate),
|
||||||
"date": req.ReleaseDate,
|
"date": req.ReleaseDate,
|
||||||
"release_date": req.ReleaseDate,
|
"release_date": req.ReleaseDate,
|
||||||
"isrc": req.ISRC,
|
"isrc": req.ISRC,
|
||||||
|
"composer": req.Composer,
|
||||||
}
|
}
|
||||||
|
|
||||||
filename := buildFilenameFromTemplate(req.FilenameFormat, metadata)
|
filename := buildFilenameFromTemplate(req.FilenameFormat, metadata)
|
||||||
@@ -1644,7 +1703,7 @@ func buildOutputPathForExtension(req DownloadRequest, ext *LoadedExtension) stri
|
|||||||
return filepath.Join(tempDir, filename+outputExt)
|
return filepath.Join(tempDir, filename+outputExt)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ExtensionProviderWrapper) CustomSearch(query string, options map[string]interface{}) ([]ExtTrackMetadata, error) {
|
func (p *extensionProviderWrapper) CustomSearch(query string, options map[string]interface{}) ([]ExtTrackMetadata, error) {
|
||||||
if !p.extension.Manifest.HasCustomSearch() {
|
if !p.extension.Manifest.HasCustomSearch() {
|
||||||
return nil, fmt.Errorf("extension '%s' does not support custom search", p.extension.ID)
|
return nil, fmt.Errorf("extension '%s' does not support custom search", p.extension.ID)
|
||||||
}
|
}
|
||||||
@@ -1726,7 +1785,7 @@ type ExtURLHandleResult struct {
|
|||||||
CoverURL string `json:"cover_url,omitempty"`
|
CoverURL string `json:"cover_url,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ExtensionProviderWrapper) HandleURL(url string) (*ExtURLHandleResult, error) {
|
func (p *extensionProviderWrapper) HandleURL(url string) (*ExtURLHandleResult, error) {
|
||||||
if !p.extension.Manifest.HasURLHandler() {
|
if !p.extension.Manifest.HasURLHandler() {
|
||||||
return nil, fmt.Errorf("extension '%s' does not support URL handling", p.extension.ID)
|
return nil, fmt.Errorf("extension '%s' does not support URL handling", p.extension.ID)
|
||||||
}
|
}
|
||||||
@@ -1812,7 +1871,7 @@ type MatchTrackResult struct {
|
|||||||
Reason string `json:"reason,omitempty"`
|
Reason string `json:"reason,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ExtensionProviderWrapper) MatchTrack(sourceTrack map[string]interface{}, candidates []map[string]interface{}) (*MatchTrackResult, error) {
|
func (p *extensionProviderWrapper) MatchTrack(sourceTrack map[string]interface{}, candidates []map[string]interface{}) (*MatchTrackResult, error) {
|
||||||
if !p.extension.Manifest.HasCustomMatching() {
|
if !p.extension.Manifest.HasCustomMatching() {
|
||||||
return nil, fmt.Errorf("extension '%s' does not support custom matching", p.extension.ID)
|
return nil, fmt.Errorf("extension '%s' does not support custom matching", p.extension.ID)
|
||||||
}
|
}
|
||||||
@@ -1883,7 +1942,7 @@ type PostProcessInput struct {
|
|||||||
|
|
||||||
const PostProcessTimeout = 2 * time.Minute
|
const PostProcessTimeout = 2 * time.Minute
|
||||||
|
|
||||||
func (p *ExtensionProviderWrapper) PostProcess(filePath string, metadata map[string]interface{}, hookID string) (*PostProcessResult, error) {
|
func (p *extensionProviderWrapper) PostProcess(filePath string, metadata map[string]interface{}, hookID string) (*PostProcessResult, error) {
|
||||||
if !p.extension.Manifest.HasPostProcessing() {
|
if !p.extension.Manifest.HasPostProcessing() {
|
||||||
return nil, fmt.Errorf("extension '%s' does not support post-processing", p.extension.ID)
|
return nil, fmt.Errorf("extension '%s' does not support post-processing", p.extension.ID)
|
||||||
}
|
}
|
||||||
@@ -1946,7 +2005,7 @@ func (p *ExtensionProviderWrapper) PostProcess(filePath string, metadata map[str
|
|||||||
return &postResult, nil
|
return &postResult, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ExtensionProviderWrapper) PostProcessV2(input PostProcessInput, metadata map[string]interface{}, hookID string) (*PostProcessResult, error) {
|
func (p *extensionProviderWrapper) PostProcessV2(input PostProcessInput, metadata map[string]interface{}, hookID string) (*PostProcessResult, error) {
|
||||||
if !p.extension.Manifest.HasPostProcessing() {
|
if !p.extension.Manifest.HasPostProcessing() {
|
||||||
return nil, fmt.Errorf("extension '%s' does not support post-processing", p.extension.ID)
|
return nil, fmt.Errorf("extension '%s' does not support post-processing", p.extension.ID)
|
||||||
}
|
}
|
||||||
@@ -2016,39 +2075,39 @@ func (p *ExtensionProviderWrapper) PostProcessV2(input PostProcessInput, metadat
|
|||||||
return &postResult, nil
|
return &postResult, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) GetSearchProviders() []*ExtensionProviderWrapper {
|
func (m *extensionManager) GetSearchProviders() []*extensionProviderWrapper {
|
||||||
m.mu.RLock()
|
m.mu.RLock()
|
||||||
defer m.mu.RUnlock()
|
defer m.mu.RUnlock()
|
||||||
|
|
||||||
var providers []*ExtensionProviderWrapper
|
var providers []*extensionProviderWrapper
|
||||||
for _, ext := range m.extensions {
|
for _, ext := range m.extensions {
|
||||||
if ext.Enabled && ext.Manifest.HasCustomSearch() && ext.Error == "" {
|
if ext.Enabled && ext.Manifest.HasCustomSearch() && ext.Error == "" {
|
||||||
providers = append(providers, NewExtensionProviderWrapper(ext))
|
providers = append(providers, newExtensionProviderWrapper(ext))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return providers
|
return providers
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) GetURLHandlers() []*ExtensionProviderWrapper {
|
func (m *extensionManager) GetURLHandlers() []*extensionProviderWrapper {
|
||||||
m.mu.RLock()
|
m.mu.RLock()
|
||||||
defer m.mu.RUnlock()
|
defer m.mu.RUnlock()
|
||||||
|
|
||||||
var providers []*ExtensionProviderWrapper
|
var providers []*extensionProviderWrapper
|
||||||
for _, ext := range m.extensions {
|
for _, ext := range m.extensions {
|
||||||
if ext.Enabled && ext.Manifest.HasURLHandler() && ext.Error == "" {
|
if ext.Enabled && ext.Manifest.HasURLHandler() && ext.Error == "" {
|
||||||
providers = append(providers, NewExtensionProviderWrapper(ext))
|
providers = append(providers, newExtensionProviderWrapper(ext))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return providers
|
return providers
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) FindURLHandler(url string) *ExtensionProviderWrapper {
|
func (m *extensionManager) FindURLHandler(url string) *extensionProviderWrapper {
|
||||||
m.mu.RLock()
|
m.mu.RLock()
|
||||||
defer m.mu.RUnlock()
|
defer m.mu.RUnlock()
|
||||||
|
|
||||||
for _, ext := range m.extensions {
|
for _, ext := range m.extensions {
|
||||||
if ext.Enabled && ext.Manifest.MatchesURL(url) && ext.Error == "" {
|
if ext.Enabled && ext.Manifest.MatchesURL(url) && ext.Error == "" {
|
||||||
return NewExtensionProviderWrapper(ext)
|
return newExtensionProviderWrapper(ext)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
@@ -2059,7 +2118,7 @@ type ExtURLHandleResultWithExtID struct {
|
|||||||
ExtensionID string
|
ExtensionID string
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) HandleURLWithExtension(url string) (*ExtURLHandleResultWithExtID, error) {
|
func (m *extensionManager) HandleURLWithExtension(url string) (*ExtURLHandleResultWithExtID, error) {
|
||||||
handler := m.FindURLHandler(url)
|
handler := m.FindURLHandler(url)
|
||||||
if handler == nil {
|
if handler == nil {
|
||||||
return nil, fmt.Errorf("no extension found to handle URL: %s", url)
|
return nil, fmt.Errorf("no extension found to handle URL: %s", url)
|
||||||
@@ -2079,20 +2138,20 @@ func (m *ExtensionManager) HandleURLWithExtension(url string) (*ExtURLHandleResu
|
|||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) GetPostProcessingProviders() []*ExtensionProviderWrapper {
|
func (m *extensionManager) GetPostProcessingProviders() []*extensionProviderWrapper {
|
||||||
m.mu.RLock()
|
m.mu.RLock()
|
||||||
defer m.mu.RUnlock()
|
defer m.mu.RUnlock()
|
||||||
|
|
||||||
var providers []*ExtensionProviderWrapper
|
var providers []*extensionProviderWrapper
|
||||||
for _, ext := range m.extensions {
|
for _, ext := range m.extensions {
|
||||||
if ext.Enabled && ext.Manifest.HasPostProcessing() && ext.Error == "" {
|
if ext.Enabled && ext.Manifest.HasPostProcessing() && ext.Error == "" {
|
||||||
providers = append(providers, NewExtensionProviderWrapper(ext))
|
providers = append(providers, newExtensionProviderWrapper(ext))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return providers
|
return providers
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) RunPostProcessing(filePath string, metadata map[string]interface{}) (*PostProcessResult, error) {
|
func (m *extensionManager) RunPostProcessing(filePath string, metadata map[string]interface{}) (*PostProcessResult, error) {
|
||||||
providers := m.GetPostProcessingProviders()
|
providers := m.GetPostProcessingProviders()
|
||||||
if len(providers) == 0 {
|
if len(providers) == 0 {
|
||||||
return &PostProcessResult{Success: true, NewFilePath: filePath}, nil
|
return &PostProcessResult{Success: true, NewFilePath: filePath}, nil
|
||||||
@@ -2137,7 +2196,7 @@ func (m *ExtensionManager) RunPostProcessing(filePath string, metadata map[strin
|
|||||||
return &PostProcessResult{Success: true, NewFilePath: currentPath}, nil
|
return &PostProcessResult{Success: true, NewFilePath: currentPath}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) RunPostProcessingV2(input PostProcessInput, metadata map[string]interface{}) (*PostProcessResult, error) {
|
func (m *extensionManager) RunPostProcessingV2(input PostProcessInput, metadata map[string]interface{}) (*PostProcessResult, error) {
|
||||||
providers := m.GetPostProcessingProviders()
|
providers := m.GetPostProcessingProviders()
|
||||||
if len(providers) == 0 {
|
if len(providers) == 0 {
|
||||||
return &PostProcessResult{Success: true, NewFilePath: input.Path, NewFileURI: input.URI}, nil
|
return &PostProcessResult{Success: true, NewFilePath: input.Path, NewFileURI: input.URI}, nil
|
||||||
@@ -2205,7 +2264,7 @@ type ExtLyricsLine struct {
|
|||||||
EndTimeMs int64 `json:"endTimeMs"`
|
EndTimeMs int64 `json:"endTimeMs"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ExtensionProviderWrapper) FetchLyrics(trackName, artistName, albumName string, durationSec float64) (*LyricsResponse, error) {
|
func (p *extensionProviderWrapper) FetchLyrics(trackName, artistName, albumName string, durationSec float64) (*LyricsResponse, error) {
|
||||||
if !p.extension.Manifest.IsLyricsProvider() {
|
if !p.extension.Manifest.IsLyricsProvider() {
|
||||||
return nil, fmt.Errorf("extension '%s' is not a lyrics provider", p.extension.ID)
|
return nil, fmt.Errorf("extension '%s' is not a lyrics provider", p.extension.ID)
|
||||||
}
|
}
|
||||||
@@ -2267,7 +2326,6 @@ func (p *ExtensionProviderWrapper) FetchLyrics(trackName, artistName, albumName
|
|||||||
return nil, fmt.Errorf("failed to parse lyrics result: %w", err)
|
return nil, fmt.Errorf("failed to parse lyrics result: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Convert ExtLyricsResult to LyricsResponse
|
|
||||||
response := &LyricsResponse{
|
response := &LyricsResponse{
|
||||||
SyncType: extResult.SyncType,
|
SyncType: extResult.SyncType,
|
||||||
Instrumental: extResult.Instrumental,
|
Instrumental: extResult.Instrumental,
|
||||||
@@ -2288,7 +2346,6 @@ func (p *ExtensionProviderWrapper) FetchLyrics(trackName, artistName, albumName
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the extension provided plainLyrics but no lines, parse them as unsynced
|
|
||||||
if len(response.Lines) == 0 && response.PlainLyrics != "" && !response.Instrumental {
|
if len(response.Lines) == 0 && response.PlainLyrics != "" && !response.Instrumental {
|
||||||
response.SyncType = "UNSYNCED"
|
response.SyncType = "UNSYNCED"
|
||||||
for _, line := range strings.Split(response.PlainLyrics, "\n") {
|
for _, line := range strings.Split(response.PlainLyrics, "\n") {
|
||||||
@@ -2305,18 +2362,17 @@ func (p *ExtensionProviderWrapper) FetchLyrics(trackName, artistName, albumName
|
|||||||
return response, nil
|
return response, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ExtensionManager) GetLyricsProviders() []*ExtensionProviderWrapper {
|
func (m *extensionManager) GetLyricsProviders() []*extensionProviderWrapper {
|
||||||
m.mu.RLock()
|
m.mu.RLock()
|
||||||
defer m.mu.RUnlock()
|
defer m.mu.RUnlock()
|
||||||
|
|
||||||
var providers []*ExtensionProviderWrapper
|
var providers []*extensionProviderWrapper
|
||||||
for _, ext := range m.extensions {
|
for _, ext := range m.extensions {
|
||||||
if ext.Enabled && ext.Manifest.IsLyricsProvider() && ext.Error == "" {
|
if ext.Enabled && ext.Manifest.IsLyricsProvider() && ext.Error == "" {
|
||||||
providers = append(providers, NewExtensionProviderWrapper(ext))
|
providers = append(providers, newExtensionProviderWrapper(ext))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Keep a deterministic order so provider selection is stable across runs.
|
|
||||||
sort.Slice(providers, func(i, j int) bool {
|
sort.Slice(providers, func(i, j int) bool {
|
||||||
return providers[i].extension.ID < providers[j].extension.ID
|
return providers[i].extension.ID < providers[j].extension.ID
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -51,7 +51,7 @@ func TestSearchTracksWithMetadataProvidersUsesPriorityAndDedupes(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
manager := GetExtensionManager()
|
manager := getExtensionManager()
|
||||||
tracks, err := manager.SearchTracksWithMetadataProviders("query", 3, false)
|
tracks, err := manager.SearchTracksWithMetadataProviders("query", 3, false)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("SearchTracksWithMetadataProviders returned error: %v", err)
|
t.Fatalf("SearchTracksWithMetadataProviders returned error: %v", err)
|
||||||
|
|||||||
@@ -80,7 +80,7 @@ func SetExtensionTokens(extensionID string, accessToken, refreshToken string, ex
|
|||||||
state.IsAuthenticated = accessToken != ""
|
state.IsAuthenticated = accessToken != ""
|
||||||
}
|
}
|
||||||
|
|
||||||
type ExtensionRuntime struct {
|
type extensionRuntime struct {
|
||||||
extensionID string
|
extensionID string
|
||||||
manifest *ExtensionManifest
|
manifest *ExtensionManifest
|
||||||
settings map[string]interface{}
|
settings map[string]interface{}
|
||||||
@@ -123,10 +123,10 @@ var (
|
|||||||
privateIPCacheMu sync.RWMutex
|
privateIPCacheMu sync.RWMutex
|
||||||
)
|
)
|
||||||
|
|
||||||
func NewExtensionRuntime(ext *LoadedExtension) *ExtensionRuntime {
|
func newExtensionRuntime(ext *loadedExtension) *extensionRuntime {
|
||||||
jar, _ := newSimpleCookieJar()
|
jar, _ := newSimpleCookieJar()
|
||||||
|
|
||||||
runtime := &ExtensionRuntime{
|
runtime := &extensionRuntime{
|
||||||
extensionID: ext.ID,
|
extensionID: ext.ID,
|
||||||
manifest: ext.Manifest,
|
manifest: ext.Manifest,
|
||||||
settings: make(map[string]interface{}),
|
settings: make(map[string]interface{}),
|
||||||
@@ -142,25 +142,25 @@ func NewExtensionRuntime(ext *LoadedExtension) *ExtensionRuntime {
|
|||||||
return runtime
|
return runtime
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) setActiveDownloadItemID(itemID string) {
|
func (r *extensionRuntime) setActiveDownloadItemID(itemID string) {
|
||||||
r.activeDownloadMu.Lock()
|
r.activeDownloadMu.Lock()
|
||||||
defer r.activeDownloadMu.Unlock()
|
defer r.activeDownloadMu.Unlock()
|
||||||
r.activeDownloadItemID = strings.TrimSpace(itemID)
|
r.activeDownloadItemID = strings.TrimSpace(itemID)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) clearActiveDownloadItemID() {
|
func (r *extensionRuntime) clearActiveDownloadItemID() {
|
||||||
r.activeDownloadMu.Lock()
|
r.activeDownloadMu.Lock()
|
||||||
defer r.activeDownloadMu.Unlock()
|
defer r.activeDownloadMu.Unlock()
|
||||||
r.activeDownloadItemID = ""
|
r.activeDownloadItemID = ""
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) getActiveDownloadItemID() string {
|
func (r *extensionRuntime) getActiveDownloadItemID() string {
|
||||||
r.activeDownloadMu.RLock()
|
r.activeDownloadMu.RLock()
|
||||||
defer r.activeDownloadMu.RUnlock()
|
defer r.activeDownloadMu.RUnlock()
|
||||||
return r.activeDownloadItemID
|
return r.activeDownloadItemID
|
||||||
}
|
}
|
||||||
|
|
||||||
func newExtensionHTTPClient(ext *LoadedExtension, jar http.CookieJar, timeout time.Duration) *http.Client {
|
func newExtensionHTTPClient(ext *loadedExtension, jar http.CookieJar, timeout time.Duration) *http.Client {
|
||||||
// Extension sandbox enforces HTTPS-only domains. Do not apply global
|
// Extension sandbox enforces HTTPS-only domains. Do not apply global
|
||||||
// allow_http scheme downgrade here, because some extension APIs (e.g.
|
// allow_http scheme downgrade here, because some extension APIs (e.g.
|
||||||
// spotify-web) will redirect http -> https and can end up in 301 loops.
|
// spotify-web) will redirect http -> https and can end up in 301 loops.
|
||||||
@@ -329,11 +329,11 @@ func (j *simpleCookieJar) Cookies(u *url.URL) []*http.Cookie {
|
|||||||
return j.cookies[u.Host]
|
return j.cookies[u.Host]
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) SetSettings(settings map[string]interface{}) {
|
func (r *extensionRuntime) SetSettings(settings map[string]interface{}) {
|
||||||
r.settings = settings
|
r.settings = settings
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) RegisterAPIs(vm *goja.Runtime) {
|
func (r *extensionRuntime) RegisterAPIs(vm *goja.Runtime) {
|
||||||
r.vm = vm
|
r.vm = vm
|
||||||
|
|
||||||
httpObj := vm.NewObject()
|
httpObj := vm.NewObject()
|
||||||
|
|||||||
@@ -52,7 +52,7 @@ func summarizeURLForLog(urlStr string) string {
|
|||||||
return fmt.Sprintf("%s://%s%s", parsed.Scheme, parsed.Host, parsed.Path)
|
return fmt.Sprintf("%s://%s%s", parsed.Scheme, parsed.Host, parsed.Path)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) authOpenUrl(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) authOpenUrl(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"success": false,
|
"success": false,
|
||||||
@@ -99,7 +99,7 @@ func (r *ExtensionRuntime) authOpenUrl(call goja.FunctionCall) goja.Value {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) authGetCode(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) authGetCode(call goja.FunctionCall) goja.Value {
|
||||||
extensionAuthStateMu.RLock()
|
extensionAuthStateMu.RLock()
|
||||||
defer extensionAuthStateMu.RUnlock()
|
defer extensionAuthStateMu.RUnlock()
|
||||||
|
|
||||||
@@ -111,7 +111,7 @@ func (r *ExtensionRuntime) authGetCode(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(state.AuthCode)
|
return r.vm.ToValue(state.AuthCode)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) authSetCode(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) authSetCode(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(false)
|
return r.vm.ToValue(false)
|
||||||
}
|
}
|
||||||
@@ -149,7 +149,7 @@ func (r *ExtensionRuntime) authSetCode(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(true)
|
return r.vm.ToValue(true)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) authClear(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) authClear(call goja.FunctionCall) goja.Value {
|
||||||
extensionAuthStateMu.Lock()
|
extensionAuthStateMu.Lock()
|
||||||
delete(extensionAuthState, r.extensionID)
|
delete(extensionAuthState, r.extensionID)
|
||||||
extensionAuthStateMu.Unlock()
|
extensionAuthStateMu.Unlock()
|
||||||
@@ -162,7 +162,7 @@ func (r *ExtensionRuntime) authClear(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(true)
|
return r.vm.ToValue(true)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) authIsAuthenticated(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) authIsAuthenticated(call goja.FunctionCall) goja.Value {
|
||||||
extensionAuthStateMu.RLock()
|
extensionAuthStateMu.RLock()
|
||||||
defer extensionAuthStateMu.RUnlock()
|
defer extensionAuthStateMu.RUnlock()
|
||||||
|
|
||||||
@@ -178,7 +178,7 @@ func (r *ExtensionRuntime) authIsAuthenticated(call goja.FunctionCall) goja.Valu
|
|||||||
return r.vm.ToValue(state.IsAuthenticated)
|
return r.vm.ToValue(state.IsAuthenticated)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) authGetTokens(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) authGetTokens(call goja.FunctionCall) goja.Value {
|
||||||
extensionAuthStateMu.RLock()
|
extensionAuthStateMu.RLock()
|
||||||
defer extensionAuthStateMu.RUnlock()
|
defer extensionAuthStateMu.RUnlock()
|
||||||
|
|
||||||
@@ -201,7 +201,6 @@ func (r *ExtensionRuntime) authGetTokens(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(result)
|
return r.vm.ToValue(result)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Length should be between 43-128 characters (RFC 7636)
|
|
||||||
func generatePKCEVerifier(length int) (string, error) {
|
func generatePKCEVerifier(length int) (string, error) {
|
||||||
if length < 43 {
|
if length < 43 {
|
||||||
length = 43
|
length = 43
|
||||||
@@ -226,11 +225,10 @@ func generatePKCEVerifier(length int) (string, error) {
|
|||||||
|
|
||||||
func generatePKCEChallenge(verifier string) string {
|
func generatePKCEChallenge(verifier string) string {
|
||||||
hash := sha256.Sum256([]byte(verifier))
|
hash := sha256.Sum256([]byte(verifier))
|
||||||
// Base64url encode without padding (RFC 7636)
|
|
||||||
return base64.RawURLEncoding.EncodeToString(hash[:])
|
return base64.RawURLEncoding.EncodeToString(hash[:])
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) authGeneratePKCE(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) authGeneratePKCE(call goja.FunctionCall) goja.Value {
|
||||||
length := 64
|
length := 64
|
||||||
if len(call.Arguments) > 0 && !goja.IsUndefined(call.Arguments[0]) {
|
if len(call.Arguments) > 0 && !goja.IsUndefined(call.Arguments[0]) {
|
||||||
if l, ok := call.Arguments[0].Export().(float64); ok && l >= 43 && l <= 128 {
|
if l, ok := call.Arguments[0].Export().(float64); ok && l >= 43 && l <= 128 {
|
||||||
@@ -267,7 +265,7 @@ func (r *ExtensionRuntime) authGeneratePKCE(call goja.FunctionCall) goja.Value {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) authGetPKCE(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) authGetPKCE(call goja.FunctionCall) goja.Value {
|
||||||
extensionAuthStateMu.RLock()
|
extensionAuthStateMu.RLock()
|
||||||
defer extensionAuthStateMu.RUnlock()
|
defer extensionAuthStateMu.RUnlock()
|
||||||
|
|
||||||
@@ -283,8 +281,7 @@ func (r *ExtensionRuntime) authGetPKCE(call goja.FunctionCall) goja.Value {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// config: { authUrl, clientId, redirectUri, scope, extraParams }
|
func (r *extensionRuntime) authStartOAuthWithPKCE(call goja.FunctionCall) goja.Value {
|
||||||
func (r *ExtensionRuntime) authStartOAuthWithPKCE(call goja.FunctionCall) goja.Value {
|
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"success": false,
|
"success": false,
|
||||||
@@ -388,8 +385,7 @@ func (r *ExtensionRuntime) authStartOAuthWithPKCE(call goja.FunctionCall) goja.V
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// config: { tokenUrl, clientId, redirectUri, code, extraParams }
|
func (r *extensionRuntime) authExchangeCodeWithPKCE(call goja.FunctionCall) goja.Value {
|
||||||
func (r *ExtensionRuntime) authExchangeCodeWithPKCE(call goja.FunctionCall) goja.Value {
|
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"success": false,
|
"success": false,
|
||||||
|
|||||||
@@ -50,7 +50,7 @@ func ClearFFmpegCommand(commandID string) {
|
|||||||
delete(ffmpegCommands, commandID)
|
delete(ffmpegCommands, commandID)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) ffmpegExecute(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) ffmpegExecute(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"success": false,
|
"success": false,
|
||||||
@@ -107,7 +107,7 @@ func (r *ExtensionRuntime) ffmpegExecute(call goja.FunctionCall) goja.Value {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) ffmpegGetInfo(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) ffmpegGetInfo(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"success": false,
|
"success": false,
|
||||||
@@ -134,7 +134,7 @@ func (r *ExtensionRuntime) ffmpegGetInfo(call goja.FunctionCall) goja.Value {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) ffmpegConvert(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) ffmpegConvert(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 2 {
|
if len(call.Arguments) < 2 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"success": false,
|
"success": false,
|
||||||
|
|||||||
@@ -71,7 +71,7 @@ func isPathWithinBase(baseDir, targetPath string) bool {
|
|||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) validatePath(path string) (string, error) {
|
func (r *extensionRuntime) validatePath(path string) (string, error) {
|
||||||
if !r.manifest.Permissions.File {
|
if !r.manifest.Permissions.File {
|
||||||
return "", fmt.Errorf("file access denied: extension does not have 'file' permission")
|
return "", fmt.Errorf("file access denied: extension does not have 'file' permission")
|
||||||
}
|
}
|
||||||
@@ -106,7 +106,7 @@ func (r *ExtensionRuntime) validatePath(path string) (string, error) {
|
|||||||
return absPath, nil
|
return absPath, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) fileDownload(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) fileDownload(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 2 {
|
if len(call.Arguments) < 2 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"success": false,
|
"success": false,
|
||||||
@@ -271,7 +271,7 @@ func (r *ExtensionRuntime) fileDownload(call goja.FunctionCall) goja.Value {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) fileExists(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) fileExists(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(false)
|
return r.vm.ToValue(false)
|
||||||
}
|
}
|
||||||
@@ -286,7 +286,7 @@ func (r *ExtensionRuntime) fileExists(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(err == nil)
|
return r.vm.ToValue(err == nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) fileDelete(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) fileDelete(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"success": false,
|
"success": false,
|
||||||
@@ -315,7 +315,7 @@ func (r *ExtensionRuntime) fileDelete(call goja.FunctionCall) goja.Value {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) fileRead(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) fileRead(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"success": false,
|
"success": false,
|
||||||
@@ -346,7 +346,7 @@ func (r *ExtensionRuntime) fileRead(call goja.FunctionCall) goja.Value {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) fileWrite(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) fileWrite(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 2 {
|
if len(call.Arguments) < 2 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"success": false,
|
"success": false,
|
||||||
@@ -386,7 +386,7 @@ func (r *ExtensionRuntime) fileWrite(call goja.FunctionCall) goja.Value {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) fileCopy(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) fileCopy(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 2 {
|
if len(call.Arguments) < 2 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"success": false,
|
"success": false,
|
||||||
@@ -459,7 +459,7 @@ func (r *ExtensionRuntime) fileCopy(call goja.FunctionCall) goja.Value {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) fileMove(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) fileMove(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 2 {
|
if len(call.Arguments) < 2 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"success": false,
|
"success": false,
|
||||||
@@ -507,7 +507,7 @@ func (r *ExtensionRuntime) fileMove(call goja.FunctionCall) goja.Value {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) fileGetSize(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) fileGetSize(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"success": false,
|
"success": false,
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ type HTTPResponse struct {
|
|||||||
Headers map[string]string `json:"headers"`
|
Headers map[string]string `json:"headers"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) validateDomain(urlStr string) error {
|
func (r *extensionRuntime) validateDomain(urlStr string) error {
|
||||||
parsed, err := url.Parse(urlStr)
|
parsed, err := url.Parse(urlStr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("invalid URL: %w", err)
|
return fmt.Errorf("invalid URL: %w", err)
|
||||||
@@ -49,7 +49,7 @@ func (r *ExtensionRuntime) validateDomain(urlStr string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) httpGet(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) httpGet(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"error": "URL is required",
|
"error": "URL is required",
|
||||||
@@ -118,12 +118,13 @@ func (r *ExtensionRuntime) httpGet(call goja.FunctionCall) goja.Value {
|
|||||||
"statusCode": resp.StatusCode,
|
"statusCode": resp.StatusCode,
|
||||||
"status": resp.StatusCode,
|
"status": resp.StatusCode,
|
||||||
"ok": resp.StatusCode >= 200 && resp.StatusCode < 300,
|
"ok": resp.StatusCode >= 200 && resp.StatusCode < 300,
|
||||||
|
"url": resp.Request.URL.String(),
|
||||||
"body": string(body),
|
"body": string(body),
|
||||||
"headers": respHeaders,
|
"headers": respHeaders,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) httpPost(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) httpPost(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"error": "URL is required",
|
"error": "URL is required",
|
||||||
@@ -214,12 +215,13 @@ func (r *ExtensionRuntime) httpPost(call goja.FunctionCall) goja.Value {
|
|||||||
"statusCode": resp.StatusCode,
|
"statusCode": resp.StatusCode,
|
||||||
"status": resp.StatusCode,
|
"status": resp.StatusCode,
|
||||||
"ok": resp.StatusCode >= 200 && resp.StatusCode < 300,
|
"ok": resp.StatusCode >= 200 && resp.StatusCode < 300,
|
||||||
|
"url": resp.Request.URL.String(),
|
||||||
"body": string(body),
|
"body": string(body),
|
||||||
"headers": respHeaders,
|
"headers": respHeaders,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) httpRequest(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) httpRequest(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"error": "URL is required",
|
"error": "URL is required",
|
||||||
@@ -322,24 +324,25 @@ func (r *ExtensionRuntime) httpRequest(call goja.FunctionCall) goja.Value {
|
|||||||
"statusCode": resp.StatusCode,
|
"statusCode": resp.StatusCode,
|
||||||
"status": resp.StatusCode,
|
"status": resp.StatusCode,
|
||||||
"ok": resp.StatusCode >= 200 && resp.StatusCode < 300,
|
"ok": resp.StatusCode >= 200 && resp.StatusCode < 300,
|
||||||
|
"url": resp.Request.URL.String(),
|
||||||
"body": string(body),
|
"body": string(body),
|
||||||
"headers": respHeaders,
|
"headers": respHeaders,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) httpPut(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) httpPut(call goja.FunctionCall) goja.Value {
|
||||||
return r.httpMethodShortcut("PUT", call)
|
return r.httpMethodShortcut("PUT", call)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) httpDelete(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) httpDelete(call goja.FunctionCall) goja.Value {
|
||||||
return r.httpMethodShortcut("DELETE", call)
|
return r.httpMethodShortcut("DELETE", call)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) httpPatch(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) httpPatch(call goja.FunctionCall) goja.Value {
|
||||||
return r.httpMethodShortcut("PATCH", call)
|
return r.httpMethodShortcut("PATCH", call)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) httpMethodShortcut(method string, call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) httpMethodShortcut(method string, call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"error": "URL is required",
|
"error": "URL is required",
|
||||||
@@ -446,12 +449,13 @@ func (r *ExtensionRuntime) httpMethodShortcut(method string, call goja.FunctionC
|
|||||||
"statusCode": resp.StatusCode,
|
"statusCode": resp.StatusCode,
|
||||||
"status": resp.StatusCode,
|
"status": resp.StatusCode,
|
||||||
"ok": resp.StatusCode >= 200 && resp.StatusCode < 300,
|
"ok": resp.StatusCode >= 200 && resp.StatusCode < 300,
|
||||||
|
"url": resp.Request.URL.String(),
|
||||||
"body": string(body),
|
"body": string(body),
|
||||||
"headers": respHeaders,
|
"headers": respHeaders,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) httpClearCookies(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) httpClearCookies(call goja.FunctionCall) goja.Value {
|
||||||
if jar, ok := r.cookieJar.(*simpleCookieJar); ok {
|
if jar, ok := r.cookieJar.(*simpleCookieJar); ok {
|
||||||
jar.mu.Lock()
|
jar.mu.Lock()
|
||||||
jar.cookies = make(map[string][]*http.Cookie)
|
jar.cookies = make(map[string][]*http.Cookie)
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ import (
|
|||||||
"github.com/dop251/goja"
|
"github.com/dop251/goja"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (r *ExtensionRuntime) matchingCompareStrings(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) matchingCompareStrings(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 2 {
|
if len(call.Arguments) < 2 {
|
||||||
return r.vm.ToValue(0.0)
|
return r.vm.ToValue(0.0)
|
||||||
}
|
}
|
||||||
@@ -22,7 +22,7 @@ func (r *ExtensionRuntime) matchingCompareStrings(call goja.FunctionCall) goja.V
|
|||||||
return r.vm.ToValue(similarity)
|
return r.vm.ToValue(similarity)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) matchingCompareDuration(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) matchingCompareDuration(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 2 {
|
if len(call.Arguments) < 2 {
|
||||||
return r.vm.ToValue(false)
|
return r.vm.ToValue(false)
|
||||||
}
|
}
|
||||||
@@ -43,7 +43,7 @@ func (r *ExtensionRuntime) matchingCompareDuration(call goja.FunctionCall) goja.
|
|||||||
return r.vm.ToValue(diff <= tolerance)
|
return r.vm.ToValue(diff <= tolerance)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) matchingNormalizeString(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) matchingNormalizeString(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue("")
|
return r.vm.ToValue("")
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,11 +12,7 @@ import (
|
|||||||
"github.com/dop251/goja"
|
"github.com/dop251/goja"
|
||||||
)
|
)
|
||||||
|
|
||||||
// These polyfills make porting browser/Node.js libraries easier
|
func (r *extensionRuntime) fetchPolyfill(call goja.FunctionCall) goja.Value {
|
||||||
// without compromising sandbox security.
|
|
||||||
|
|
||||||
// Returns a Promise-like object with json(), text() methods.
|
|
||||||
func (r *ExtensionRuntime) fetchPolyfill(call goja.FunctionCall) goja.Value {
|
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.createFetchError("URL is required")
|
return r.createFetchError("URL is required")
|
||||||
}
|
}
|
||||||
@@ -38,7 +34,6 @@ func (r *ExtensionRuntime) fetchPolyfill(call goja.FunctionCall) goja.Value {
|
|||||||
method = strings.ToUpper(m)
|
method = strings.ToUpper(m)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Body - support string, object (auto-stringify), or nil
|
|
||||||
if bodyArg, ok := opts["body"]; ok && bodyArg != nil {
|
if bodyArg, ok := opts["body"]; ok && bodyArg != nil {
|
||||||
switch v := bodyArg.(type) {
|
switch v := bodyArg.(type) {
|
||||||
case string:
|
case string:
|
||||||
@@ -110,7 +105,7 @@ func (r *ExtensionRuntime) fetchPolyfill(call goja.FunctionCall) goja.Value {
|
|||||||
responseObj.Set("status", resp.StatusCode)
|
responseObj.Set("status", resp.StatusCode)
|
||||||
responseObj.Set("statusText", http.StatusText(resp.StatusCode))
|
responseObj.Set("statusText", http.StatusText(resp.StatusCode))
|
||||||
responseObj.Set("headers", respHeaders)
|
responseObj.Set("headers", respHeaders)
|
||||||
responseObj.Set("url", urlStr)
|
responseObj.Set("url", resp.Request.URL.String())
|
||||||
|
|
||||||
bodyString := string(body)
|
bodyString := string(body)
|
||||||
|
|
||||||
@@ -138,7 +133,7 @@ func (r *ExtensionRuntime) fetchPolyfill(call goja.FunctionCall) goja.Value {
|
|||||||
return responseObj
|
return responseObj
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) createFetchError(message string) goja.Value {
|
func (r *extensionRuntime) createFetchError(message string) goja.Value {
|
||||||
errorObj := r.vm.NewObject()
|
errorObj := r.vm.NewObject()
|
||||||
errorObj.Set("ok", false)
|
errorObj.Set("ok", false)
|
||||||
errorObj.Set("status", 0)
|
errorObj.Set("status", 0)
|
||||||
@@ -153,7 +148,7 @@ func (r *ExtensionRuntime) createFetchError(message string) goja.Value {
|
|||||||
return errorObj
|
return errorObj
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) atobPolyfill(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) atobPolyfill(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue("")
|
return r.vm.ToValue("")
|
||||||
}
|
}
|
||||||
@@ -169,7 +164,7 @@ func (r *ExtensionRuntime) atobPolyfill(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(string(decoded))
|
return r.vm.ToValue(string(decoded))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) btoaPolyfill(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) btoaPolyfill(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue("")
|
return r.vm.ToValue("")
|
||||||
}
|
}
|
||||||
@@ -177,7 +172,7 @@ func (r *ExtensionRuntime) btoaPolyfill(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(base64.StdEncoding.EncodeToString([]byte(input)))
|
return r.vm.ToValue(base64.StdEncoding.EncodeToString([]byte(input)))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) registerTextEncoderDecoder(vm *goja.Runtime) {
|
func (r *extensionRuntime) registerTextEncoderDecoder(vm *goja.Runtime) {
|
||||||
vm.Set("TextEncoder", func(call goja.ConstructorCall) *goja.Object {
|
vm.Set("TextEncoder", func(call goja.ConstructorCall) *goja.Object {
|
||||||
encoder := call.This
|
encoder := call.This
|
||||||
encoder.Set("encoding", "utf-8")
|
encoder.Set("encoding", "utf-8")
|
||||||
@@ -197,7 +192,6 @@ func (r *ExtensionRuntime) registerTextEncoderDecoder(vm *goja.Runtime) {
|
|||||||
})
|
})
|
||||||
|
|
||||||
encoder.Set("encodeInto", func(call goja.FunctionCall) goja.Value {
|
encoder.Set("encodeInto", func(call goja.FunctionCall) goja.Value {
|
||||||
// Simplified implementation
|
|
||||||
if len(call.Arguments) < 2 {
|
if len(call.Arguments) < 2 {
|
||||||
return vm.ToValue(map[string]interface{}{"read": 0, "written": 0})
|
return vm.ToValue(map[string]interface{}{"read": 0, "written": 0})
|
||||||
}
|
}
|
||||||
@@ -258,7 +252,7 @@ func (r *ExtensionRuntime) registerTextEncoderDecoder(vm *goja.Runtime) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) registerURLClass(vm *goja.Runtime) {
|
func (r *extensionRuntime) registerURLClass(vm *goja.Runtime) {
|
||||||
vm.Set("URL", func(call goja.ConstructorCall) *goja.Object {
|
vm.Set("URL", func(call goja.ConstructorCall) *goja.Object {
|
||||||
urlObj := call.This
|
urlObj := call.This
|
||||||
|
|
||||||
@@ -422,8 +416,7 @@ func (r *ExtensionRuntime) registerURLClass(vm *goja.Runtime) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// JSON is already built-in to Goja; this ensures a fallback exists.
|
func (r *extensionRuntime) registerJSONGlobal(vm *goja.Runtime) {
|
||||||
func (r *ExtensionRuntime) registerJSONGlobal(vm *goja.Runtime) {
|
|
||||||
jsonScript := `
|
jsonScript := `
|
||||||
if (typeof JSON === 'undefined') {
|
if (typeof JSON === 'undefined') {
|
||||||
var JSON = {
|
var JSON = {
|
||||||
|
|||||||
@@ -21,7 +21,7 @@ const (
|
|||||||
storageFlushRetryDelay = 2 * time.Second
|
storageFlushRetryDelay = 2 * time.Second
|
||||||
)
|
)
|
||||||
|
|
||||||
func (r *ExtensionRuntime) getStoragePath() string {
|
func (r *extensionRuntime) getStoragePath() string {
|
||||||
return filepath.Join(r.dataDir, "storage.json")
|
return filepath.Join(r.dataDir, "storage.json")
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -36,7 +36,7 @@ func cloneInterfaceMap(src map[string]interface{}) map[string]interface{} {
|
|||||||
return dst
|
return dst
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) ensureStorageLoaded() error {
|
func (r *extensionRuntime) ensureStorageLoaded() error {
|
||||||
r.storageMu.RLock()
|
r.storageMu.RLock()
|
||||||
if r.storageLoaded {
|
if r.storageLoaded {
|
||||||
r.storageMu.RUnlock()
|
r.storageMu.RUnlock()
|
||||||
@@ -74,7 +74,7 @@ func (r *ExtensionRuntime) ensureStorageLoaded() error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) loadStorage() (map[string]interface{}, error) {
|
func (r *extensionRuntime) loadStorage() (map[string]interface{}, error) {
|
||||||
if err := r.ensureStorageLoaded(); err != nil {
|
if err := r.ensureStorageLoaded(); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -84,7 +84,7 @@ func (r *ExtensionRuntime) loadStorage() (map[string]interface{}, error) {
|
|||||||
return cloneInterfaceMap(r.storageCache), nil
|
return cloneInterfaceMap(r.storageCache), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) queueStorageFlushLocked(delay time.Duration) {
|
func (r *extensionRuntime) queueStorageFlushLocked(delay time.Duration) {
|
||||||
if r.storageClosed {
|
if r.storageClosed {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -94,7 +94,7 @@ func (r *ExtensionRuntime) queueStorageFlushLocked(delay time.Duration) {
|
|||||||
r.storageTimer = time.AfterFunc(delay, r.flushStorageDirtyAsync)
|
r.storageTimer = time.AfterFunc(delay, r.flushStorageDirtyAsync)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) persistStorageSnapshot(storage map[string]interface{}) error {
|
func (r *extensionRuntime) persistStorageSnapshot(storage map[string]interface{}) error {
|
||||||
data, err := json.Marshal(storage)
|
data, err := json.Marshal(storage)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@@ -106,13 +106,13 @@ func (r *ExtensionRuntime) persistStorageSnapshot(storage map[string]interface{}
|
|||||||
return os.WriteFile(r.getStoragePath(), data, 0600)
|
return os.WriteFile(r.getStoragePath(), data, 0600)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) flushStorageDirtyAsync() {
|
func (r *extensionRuntime) flushStorageDirtyAsync() {
|
||||||
if err := r.flushStorageDirty(); err != nil {
|
if err := r.flushStorageDirty(); err != nil {
|
||||||
GoLog("[Extension:%s] Storage flush error: %v\n", r.extensionID, err)
|
GoLog("[Extension:%s] Storage flush error: %v\n", r.extensionID, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) flushStorageDirty() error {
|
func (r *extensionRuntime) flushStorageDirty() error {
|
||||||
r.storageMu.Lock()
|
r.storageMu.Lock()
|
||||||
if r.storageClosed {
|
if r.storageClosed {
|
||||||
r.storageTimer = nil
|
r.storageTimer = nil
|
||||||
@@ -140,7 +140,7 @@ func (r *ExtensionRuntime) flushStorageDirty() error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) flushStorageNow() error {
|
func (r *extensionRuntime) flushStorageNow() error {
|
||||||
r.storageMu.Lock()
|
r.storageMu.Lock()
|
||||||
if r.storageTimer != nil {
|
if r.storageTimer != nil {
|
||||||
r.storageTimer.Stop()
|
r.storageTimer.Stop()
|
||||||
@@ -157,7 +157,7 @@ func (r *ExtensionRuntime) flushStorageNow() error {
|
|||||||
return r.persistStorageSnapshot(snapshot)
|
return r.persistStorageSnapshot(snapshot)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) closeStorageFlusher() {
|
func (r *extensionRuntime) closeStorageFlusher() {
|
||||||
r.storageMu.Lock()
|
r.storageMu.Lock()
|
||||||
r.storageClosed = true
|
r.storageClosed = true
|
||||||
r.storageDirty = false
|
r.storageDirty = false
|
||||||
@@ -168,7 +168,7 @@ func (r *ExtensionRuntime) closeStorageFlusher() {
|
|||||||
r.storageMu.Unlock()
|
r.storageMu.Unlock()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) storageGet(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) storageGet(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return goja.Undefined()
|
return goja.Undefined()
|
||||||
}
|
}
|
||||||
@@ -193,7 +193,7 @@ func (r *ExtensionRuntime) storageGet(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(value)
|
return r.vm.ToValue(value)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) storageSet(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) storageSet(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 2 {
|
if len(call.Arguments) < 2 {
|
||||||
return r.vm.ToValue(false)
|
return r.vm.ToValue(false)
|
||||||
}
|
}
|
||||||
@@ -225,7 +225,7 @@ func (r *ExtensionRuntime) storageSet(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(true)
|
return r.vm.ToValue(true)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) storageRemove(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) storageRemove(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(false)
|
return r.vm.ToValue(false)
|
||||||
}
|
}
|
||||||
@@ -254,15 +254,15 @@ func (r *ExtensionRuntime) storageRemove(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(true)
|
return r.vm.ToValue(true)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) getCredentialsPath() string {
|
func (r *extensionRuntime) getCredentialsPath() string {
|
||||||
return filepath.Join(r.dataDir, ".credentials.enc")
|
return filepath.Join(r.dataDir, ".credentials.enc")
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) getSaltPath() string {
|
func (r *extensionRuntime) getSaltPath() string {
|
||||||
return filepath.Join(r.dataDir, ".cred_salt")
|
return filepath.Join(r.dataDir, ".cred_salt")
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) getOrCreateSalt() ([]byte, error) {
|
func (r *extensionRuntime) getOrCreateSalt() ([]byte, error) {
|
||||||
saltPath := r.getSaltPath()
|
saltPath := r.getSaltPath()
|
||||||
|
|
||||||
salt, err := os.ReadFile(saltPath)
|
salt, err := os.ReadFile(saltPath)
|
||||||
@@ -282,7 +282,7 @@ func (r *ExtensionRuntime) getOrCreateSalt() ([]byte, error) {
|
|||||||
return salt, nil
|
return salt, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) getEncryptionKey() ([]byte, error) {
|
func (r *extensionRuntime) getEncryptionKey() ([]byte, error) {
|
||||||
salt, err := r.getOrCreateSalt()
|
salt, err := r.getOrCreateSalt()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@@ -293,7 +293,7 @@ func (r *ExtensionRuntime) getEncryptionKey() ([]byte, error) {
|
|||||||
return hash[:], nil
|
return hash[:], nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) ensureCredentialsLoaded() error {
|
func (r *extensionRuntime) ensureCredentialsLoaded() error {
|
||||||
r.credentialsMu.RLock()
|
r.credentialsMu.RLock()
|
||||||
if r.credentialsLoaded {
|
if r.credentialsLoaded {
|
||||||
r.credentialsMu.RUnlock()
|
r.credentialsMu.RUnlock()
|
||||||
@@ -340,7 +340,7 @@ func (r *ExtensionRuntime) ensureCredentialsLoaded() error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) loadCredentials() (map[string]interface{}, error) {
|
func (r *extensionRuntime) loadCredentials() (map[string]interface{}, error) {
|
||||||
if err := r.ensureCredentialsLoaded(); err != nil {
|
if err := r.ensureCredentialsLoaded(); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -350,7 +350,7 @@ func (r *ExtensionRuntime) loadCredentials() (map[string]interface{}, error) {
|
|||||||
return cloneInterfaceMap(r.credentialsCache), nil
|
return cloneInterfaceMap(r.credentialsCache), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) saveCredentials(creds map[string]interface{}) error {
|
func (r *extensionRuntime) saveCredentials(creds map[string]interface{}) error {
|
||||||
data, err := json.Marshal(creds)
|
data, err := json.Marshal(creds)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@@ -377,7 +377,7 @@ func (r *ExtensionRuntime) saveCredentials(creds map[string]interface{}) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) credentialsStore(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) credentialsStore(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 2 {
|
if len(call.Arguments) < 2 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"success": false,
|
"success": false,
|
||||||
@@ -414,7 +414,7 @@ func (r *ExtensionRuntime) credentialsStore(call goja.FunctionCall) goja.Value {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) credentialsGet(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) credentialsGet(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return goja.Undefined()
|
return goja.Undefined()
|
||||||
}
|
}
|
||||||
@@ -439,7 +439,7 @@ func (r *ExtensionRuntime) credentialsGet(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(value)
|
return r.vm.ToValue(value)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) credentialsRemove(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) credentialsRemove(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(false)
|
return r.vm.ToValue(false)
|
||||||
}
|
}
|
||||||
@@ -464,7 +464,7 @@ func (r *ExtensionRuntime) credentialsRemove(call goja.FunctionCall) goja.Value
|
|||||||
return r.vm.ToValue(true)
|
return r.vm.ToValue(true)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) credentialsHas(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) credentialsHas(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue(false)
|
return r.vm.ToValue(false)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ import (
|
|||||||
"github.com/dop251/goja"
|
"github.com/dop251/goja"
|
||||||
)
|
)
|
||||||
|
|
||||||
func setStorageValue(t *testing.T, runtime *ExtensionRuntime, key string, value interface{}) {
|
func setStorageValue(t *testing.T, runtime *extensionRuntime, key string, value interface{}) {
|
||||||
t.Helper()
|
t.Helper()
|
||||||
result := runtime.storageSet(goja.FunctionCall{
|
result := runtime.storageSet(goja.FunctionCall{
|
||||||
Arguments: []goja.Value{
|
Arguments: []goja.Value{
|
||||||
@@ -39,7 +39,7 @@ func readStorageMap(t *testing.T, storagePath string) map[string]interface{} {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestExtensionRuntimeStorage_DebouncedWriteCompactJSON(t *testing.T) {
|
func TestExtensionRuntimeStorage_DebouncedWriteCompactJSON(t *testing.T) {
|
||||||
ext := &LoadedExtension{
|
ext := &loadedExtension{
|
||||||
ID: "storage-test",
|
ID: "storage-test",
|
||||||
Manifest: &ExtensionManifest{
|
Manifest: &ExtensionManifest{
|
||||||
Name: "storage-test",
|
Name: "storage-test",
|
||||||
@@ -47,7 +47,7 @@ func TestExtensionRuntimeStorage_DebouncedWriteCompactJSON(t *testing.T) {
|
|||||||
DataDir: t.TempDir(),
|
DataDir: t.TempDir(),
|
||||||
}
|
}
|
||||||
|
|
||||||
runtime := NewExtensionRuntime(ext)
|
runtime := newExtensionRuntime(ext)
|
||||||
runtime.storageFlushDelay = 25 * time.Millisecond
|
runtime.storageFlushDelay = 25 * time.Millisecond
|
||||||
runtime.RegisterAPIs(goja.New())
|
runtime.RegisterAPIs(goja.New())
|
||||||
|
|
||||||
@@ -86,7 +86,7 @@ func TestExtensionRuntimeStorage_DebouncedWriteCompactJSON(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestUnloadExtension_FlushesPendingStorage(t *testing.T) {
|
func TestUnloadExtension_FlushesPendingStorage(t *testing.T) {
|
||||||
ext := &LoadedExtension{
|
ext := &loadedExtension{
|
||||||
ID: "unload-storage-test",
|
ID: "unload-storage-test",
|
||||||
Manifest: &ExtensionManifest{
|
Manifest: &ExtensionManifest{
|
||||||
Name: "unload-storage-test",
|
Name: "unload-storage-test",
|
||||||
@@ -95,13 +95,13 @@ func TestUnloadExtension_FlushesPendingStorage(t *testing.T) {
|
|||||||
VM: goja.New(),
|
VM: goja.New(),
|
||||||
}
|
}
|
||||||
|
|
||||||
runtime := NewExtensionRuntime(ext)
|
runtime := newExtensionRuntime(ext)
|
||||||
runtime.storageFlushDelay = time.Hour
|
runtime.storageFlushDelay = time.Hour
|
||||||
runtime.RegisterAPIs(ext.VM)
|
runtime.RegisterAPIs(ext.VM)
|
||||||
ext.runtime = runtime
|
ext.runtime = runtime
|
||||||
|
|
||||||
manager := &ExtensionManager{
|
manager := &extensionManager{
|
||||||
extensions: map[string]*LoadedExtension{
|
extensions: map[string]*loadedExtension{
|
||||||
ext.ID: ext,
|
ext.ID: ext,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,7 +16,7 @@ import (
|
|||||||
"github.com/dop251/goja"
|
"github.com/dop251/goja"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (r *ExtensionRuntime) base64Encode(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) base64Encode(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue("")
|
return r.vm.ToValue("")
|
||||||
}
|
}
|
||||||
@@ -24,7 +24,7 @@ func (r *ExtensionRuntime) base64Encode(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(base64.StdEncoding.EncodeToString([]byte(input)))
|
return r.vm.ToValue(base64.StdEncoding.EncodeToString([]byte(input)))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) base64Decode(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) base64Decode(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue("")
|
return r.vm.ToValue("")
|
||||||
}
|
}
|
||||||
@@ -36,7 +36,7 @@ func (r *ExtensionRuntime) base64Decode(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(string(decoded))
|
return r.vm.ToValue(string(decoded))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) md5Hash(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) md5Hash(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue("")
|
return r.vm.ToValue("")
|
||||||
}
|
}
|
||||||
@@ -45,7 +45,7 @@ func (r *ExtensionRuntime) md5Hash(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(hex.EncodeToString(hash[:]))
|
return r.vm.ToValue(hex.EncodeToString(hash[:]))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) sha256Hash(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) sha256Hash(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue("")
|
return r.vm.ToValue("")
|
||||||
}
|
}
|
||||||
@@ -54,7 +54,7 @@ func (r *ExtensionRuntime) sha256Hash(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(hex.EncodeToString(hash[:]))
|
return r.vm.ToValue(hex.EncodeToString(hash[:]))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) hmacSHA256(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) hmacSHA256(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 2 {
|
if len(call.Arguments) < 2 {
|
||||||
return r.vm.ToValue("")
|
return r.vm.ToValue("")
|
||||||
}
|
}
|
||||||
@@ -66,7 +66,7 @@ func (r *ExtensionRuntime) hmacSHA256(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(hex.EncodeToString(mac.Sum(nil)))
|
return r.vm.ToValue(hex.EncodeToString(mac.Sum(nil)))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) hmacSHA256Base64(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) hmacSHA256Base64(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 2 {
|
if len(call.Arguments) < 2 {
|
||||||
return r.vm.ToValue("")
|
return r.vm.ToValue("")
|
||||||
}
|
}
|
||||||
@@ -78,7 +78,7 @@ func (r *ExtensionRuntime) hmacSHA256Base64(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(base64.StdEncoding.EncodeToString(mac.Sum(nil)))
|
return r.vm.ToValue(base64.StdEncoding.EncodeToString(mac.Sum(nil)))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) hmacSHA1(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) hmacSHA1(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 2 {
|
if len(call.Arguments) < 2 {
|
||||||
return r.vm.ToValue([]byte{})
|
return r.vm.ToValue([]byte{})
|
||||||
}
|
}
|
||||||
@@ -130,7 +130,7 @@ func (r *ExtensionRuntime) hmacSHA1(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(jsArray)
|
return r.vm.ToValue(jsArray)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) parseJSON(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) parseJSON(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return goja.Undefined()
|
return goja.Undefined()
|
||||||
}
|
}
|
||||||
@@ -145,7 +145,7 @@ func (r *ExtensionRuntime) parseJSON(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(result)
|
return r.vm.ToValue(result)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) stringifyJSON(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) stringifyJSON(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue("")
|
return r.vm.ToValue("")
|
||||||
}
|
}
|
||||||
@@ -160,7 +160,7 @@ func (r *ExtensionRuntime) stringifyJSON(call goja.FunctionCall) goja.Value {
|
|||||||
return r.vm.ToValue(string(data))
|
return r.vm.ToValue(string(data))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) cryptoEncrypt(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) cryptoEncrypt(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 2 {
|
if len(call.Arguments) < 2 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"success": false,
|
"success": false,
|
||||||
@@ -187,7 +187,7 @@ func (r *ExtensionRuntime) cryptoEncrypt(call goja.FunctionCall) goja.Value {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) cryptoDecrypt(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) cryptoDecrypt(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 2 {
|
if len(call.Arguments) < 2 {
|
||||||
return r.vm.ToValue(map[string]interface{}{
|
return r.vm.ToValue(map[string]interface{}{
|
||||||
"success": false,
|
"success": false,
|
||||||
@@ -222,7 +222,7 @@ func (r *ExtensionRuntime) cryptoDecrypt(call goja.FunctionCall) goja.Value {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) cryptoGenerateKey(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) cryptoGenerateKey(call goja.FunctionCall) goja.Value {
|
||||||
length := 32
|
length := 32
|
||||||
if len(call.Arguments) > 0 && !goja.IsUndefined(call.Arguments[0]) {
|
if len(call.Arguments) > 0 && !goja.IsUndefined(call.Arguments[0]) {
|
||||||
if l, ok := call.Arguments[0].Export().(float64); ok {
|
if l, ok := call.Arguments[0].Export().(float64); ok {
|
||||||
@@ -245,35 +245,35 @@ func (r *ExtensionRuntime) cryptoGenerateKey(call goja.FunctionCall) goja.Value
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) randomUserAgent(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) randomUserAgent(call goja.FunctionCall) goja.Value {
|
||||||
return r.vm.ToValue(getRandomUserAgent())
|
return r.vm.ToValue(getRandomUserAgent())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) logDebug(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) logDebug(call goja.FunctionCall) goja.Value {
|
||||||
msg := r.formatLogArgs(call.Arguments)
|
msg := r.formatLogArgs(call.Arguments)
|
||||||
GoLog("[Extension:%s:DEBUG] %s\n", r.extensionID, msg)
|
GoLog("[Extension:%s:DEBUG] %s\n", r.extensionID, msg)
|
||||||
return goja.Undefined()
|
return goja.Undefined()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) logInfo(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) logInfo(call goja.FunctionCall) goja.Value {
|
||||||
msg := r.formatLogArgs(call.Arguments)
|
msg := r.formatLogArgs(call.Arguments)
|
||||||
GoLog("[Extension:%s:INFO] %s\n", r.extensionID, msg)
|
GoLog("[Extension:%s:INFO] %s\n", r.extensionID, msg)
|
||||||
return goja.Undefined()
|
return goja.Undefined()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) logWarn(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) logWarn(call goja.FunctionCall) goja.Value {
|
||||||
msg := r.formatLogArgs(call.Arguments)
|
msg := r.formatLogArgs(call.Arguments)
|
||||||
GoLog("[Extension:%s:WARN] %s\n", r.extensionID, msg)
|
GoLog("[Extension:%s:WARN] %s\n", r.extensionID, msg)
|
||||||
return goja.Undefined()
|
return goja.Undefined()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) logError(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) logError(call goja.FunctionCall) goja.Value {
|
||||||
msg := r.formatLogArgs(call.Arguments)
|
msg := r.formatLogArgs(call.Arguments)
|
||||||
GoLog("[Extension:%s:ERROR] %s\n", r.extensionID, msg)
|
GoLog("[Extension:%s:ERROR] %s\n", r.extensionID, msg)
|
||||||
return goja.Undefined()
|
return goja.Undefined()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) formatLogArgs(args []goja.Value) string {
|
func (r *extensionRuntime) formatLogArgs(args []goja.Value) string {
|
||||||
parts := make([]string, len(args))
|
parts := make([]string, len(args))
|
||||||
for i, arg := range args {
|
for i, arg := range args {
|
||||||
parts[i] = fmt.Sprintf("%v", arg.Export())
|
parts[i] = fmt.Sprintf("%v", arg.Export())
|
||||||
@@ -281,7 +281,7 @@ func (r *ExtensionRuntime) formatLogArgs(args []goja.Value) string {
|
|||||||
return strings.Join(parts, " ")
|
return strings.Join(parts, " ")
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) sanitizeFilenameWrapper(call goja.FunctionCall) goja.Value {
|
func (r *extensionRuntime) sanitizeFilenameWrapper(call goja.FunctionCall) goja.Value {
|
||||||
if len(call.Arguments) < 1 {
|
if len(call.Arguments) < 1 {
|
||||||
return r.vm.ToValue("")
|
return r.vm.ToValue("")
|
||||||
}
|
}
|
||||||
@@ -289,7 +289,7 @@ func (r *ExtensionRuntime) sanitizeFilenameWrapper(call goja.FunctionCall) goja.
|
|||||||
return r.vm.ToValue(sanitizeFilename(input))
|
return r.vm.ToValue(sanitizeFilename(input))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ExtensionRuntime) RegisterGoBackendAPIs(vm *goja.Runtime) {
|
func (r *extensionRuntime) RegisterGoBackendAPIs(vm *goja.Runtime) {
|
||||||
gobackendObj := vm.Get("gobackend")
|
gobackendObj := vm.Get("gobackend")
|
||||||
if gobackendObj == nil || goja.IsUndefined(gobackendObj) {
|
if gobackendObj == nil || goja.IsUndefined(gobackendObj) {
|
||||||
gobackendObj = vm.NewObject()
|
gobackendObj = vm.NewObject()
|
||||||
|
|||||||
@@ -145,7 +145,7 @@ func initExtensionStore(cacheDir string) *extensionStore {
|
|||||||
|
|
||||||
if globalExtensionStore == nil {
|
if globalExtensionStore == nil {
|
||||||
globalExtensionStore = &extensionStore{
|
globalExtensionStore = &extensionStore{
|
||||||
registryURL: "", // No default - user must provide a registry URL
|
registryURL: "",
|
||||||
cacheDir: cacheDir,
|
cacheDir: cacheDir,
|
||||||
cacheTTL: cacheTTL,
|
cacheTTL: cacheTTL,
|
||||||
}
|
}
|
||||||
@@ -154,8 +154,6 @@ func initExtensionStore(cacheDir string) *extensionStore {
|
|||||||
return globalExtensionStore
|
return globalExtensionStore
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetRegistryURL updates the registry URL and clears the in-memory cache
|
|
||||||
// so the next fetch will use the new URL. Disk cache is also cleared.
|
|
||||||
func (s *extensionStore) setRegistryURL(registryURL string) {
|
func (s *extensionStore) setRegistryURL(registryURL string) {
|
||||||
s.cacheMu.Lock()
|
s.cacheMu.Lock()
|
||||||
defer s.cacheMu.Unlock()
|
defer s.cacheMu.Unlock()
|
||||||
@@ -168,7 +166,6 @@ func (s *extensionStore) setRegistryURL(registryURL string) {
|
|||||||
s.cache = nil
|
s.cache = nil
|
||||||
s.cacheTime = time.Time{}
|
s.cacheTime = time.Time{}
|
||||||
|
|
||||||
// Clear disk cache since it's from a different registry
|
|
||||||
if s.cacheDir != "" {
|
if s.cacheDir != "" {
|
||||||
cachePath := filepath.Join(s.cacheDir, cacheFileName)
|
cachePath := filepath.Join(s.cacheDir, cacheFileName)
|
||||||
os.Remove(cachePath)
|
os.Remove(cachePath)
|
||||||
@@ -177,7 +174,6 @@ func (s *extensionStore) setRegistryURL(registryURL string) {
|
|||||||
LogInfo("ExtensionStore", "Registry URL updated to: %s", registryURL)
|
LogInfo("ExtensionStore", "Registry URL updated to: %s", registryURL)
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetRegistryURL returns the currently configured registry URL.
|
|
||||||
func (s *extensionStore) getRegistryURL() string {
|
func (s *extensionStore) getRegistryURL() string {
|
||||||
s.cacheMu.RLock()
|
s.cacheMu.RLock()
|
||||||
defer s.cacheMu.RUnlock()
|
defer s.cacheMu.RUnlock()
|
||||||
@@ -299,7 +295,7 @@ func (s *extensionStore) getExtensionsWithStatus(forceRefresh bool) ([]storeExte
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
manager := GetExtensionManager()
|
manager := getExtensionManager()
|
||||||
installed := make(map[string]string) // id -> version
|
installed := make(map[string]string) // id -> version
|
||||||
|
|
||||||
if manager != nil {
|
if manager != nil {
|
||||||
@@ -378,32 +374,22 @@ func (s *extensionStore) downloadExtension(extensionID string, destPath string)
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// ResolveRegistryURL normalises a user-supplied URL into a direct registry.json URL.
|
|
||||||
//
|
|
||||||
// Accepted formats:
|
|
||||||
// - https://raw.githubusercontent.com/owner/repo/<branch>/registry.json → returned as-is
|
|
||||||
// - https://github.com/owner/repo (with optional trailing path / .git) → resolved via
|
|
||||||
// the GitHub API to discover the default branch, then converted to the raw URL
|
|
||||||
// - Any other HTTPS URL → returned as-is (assumed to be a direct link)
|
|
||||||
func resolveRegistryURL(input string) (string, error) {
|
func resolveRegistryURL(input string) (string, error) {
|
||||||
input = strings.TrimSpace(input)
|
input = strings.TrimSpace(input)
|
||||||
if input == "" {
|
if input == "" {
|
||||||
return "", fmt.Errorf("registry URL is empty")
|
return "", fmt.Errorf("registry URL is empty")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Already a fully-qualified raw URL – keep it.
|
|
||||||
if strings.Contains(input, "raw.githubusercontent.com") {
|
if strings.Contains(input, "raw.githubusercontent.com") {
|
||||||
return input, nil
|
return input, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
const ghPrefix = "https://github.com/"
|
const ghPrefix = "https://github.com/"
|
||||||
if !strings.HasPrefix(input, ghPrefix) {
|
if !strings.HasPrefix(input, ghPrefix) {
|
||||||
// Also accept http:// and upgrade silently.
|
|
||||||
const ghPrefixHTTP = "http://github.com/"
|
const ghPrefixHTTP = "http://github.com/"
|
||||||
if strings.HasPrefix(input, ghPrefixHTTP) {
|
if strings.HasPrefix(input, ghPrefixHTTP) {
|
||||||
input = "https://github.com/" + input[len(ghPrefixHTTP):]
|
input = "https://github.com/" + input[len(ghPrefixHTTP):]
|
||||||
} else {
|
} else {
|
||||||
// Not a GitHub URL – return as-is.
|
|
||||||
return input, nil
|
return input, nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -423,8 +409,6 @@ func resolveRegistryURL(input string) (string, error) {
|
|||||||
return resolved, nil
|
return resolved, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// resolveGitHubDefaultBranch calls the GitHub API to discover the repository's
|
|
||||||
// default branch. Falls back to "main" on any error.
|
|
||||||
func resolveGitHubDefaultBranch(owner, repo string) string {
|
func resolveGitHubDefaultBranch(owner, repo string) string {
|
||||||
apiURL := fmt.Sprintf("https://api.github.com/repos/%s/%s", owner, repo)
|
apiURL := fmt.Sprintf("https://api.github.com/repos/%s/%s", owner, repo)
|
||||||
client := NewHTTPClientWithTimeout(10 * time.Second)
|
client := NewHTTPClientWithTimeout(10 * time.Second)
|
||||||
|
|||||||
@@ -99,7 +99,7 @@ func TestIsDomainAllowed(t *testing.T) {
|
|||||||
|
|
||||||
func TestExtensionRuntime_NetworkSandbox(t *testing.T) {
|
func TestExtensionRuntime_NetworkSandbox(t *testing.T) {
|
||||||
// Create a mock extension with limited network permissions
|
// Create a mock extension with limited network permissions
|
||||||
ext := &LoadedExtension{
|
ext := &loadedExtension{
|
||||||
ID: "test-ext",
|
ID: "test-ext",
|
||||||
Manifest: &ExtensionManifest{
|
Manifest: &ExtensionManifest{
|
||||||
Name: "test-ext",
|
Name: "test-ext",
|
||||||
@@ -110,7 +110,7 @@ func TestExtensionRuntime_NetworkSandbox(t *testing.T) {
|
|||||||
DataDir: t.TempDir(),
|
DataDir: t.TempDir(),
|
||||||
}
|
}
|
||||||
|
|
||||||
runtime := NewExtensionRuntime(ext)
|
runtime := newExtensionRuntime(ext)
|
||||||
|
|
||||||
if err := runtime.validateDomain("https://api.allowed.com/path"); err != nil {
|
if err := runtime.validateDomain("https://api.allowed.com/path"); err != nil {
|
||||||
t.Errorf("Expected api.allowed.com to be allowed, got error: %v", err)
|
t.Errorf("Expected api.allowed.com to be allowed, got error: %v", err)
|
||||||
@@ -132,7 +132,7 @@ func TestExtensionRuntime_NetworkSandbox(t *testing.T) {
|
|||||||
func TestExtensionRuntime_FileSandbox(t *testing.T) {
|
func TestExtensionRuntime_FileSandbox(t *testing.T) {
|
||||||
tempDir := t.TempDir()
|
tempDir := t.TempDir()
|
||||||
|
|
||||||
ext := &LoadedExtension{
|
ext := &loadedExtension{
|
||||||
ID: "test-ext",
|
ID: "test-ext",
|
||||||
Manifest: &ExtensionManifest{
|
Manifest: &ExtensionManifest{
|
||||||
Name: "test-ext",
|
Name: "test-ext",
|
||||||
@@ -143,7 +143,7 @@ func TestExtensionRuntime_FileSandbox(t *testing.T) {
|
|||||||
DataDir: tempDir,
|
DataDir: tempDir,
|
||||||
}
|
}
|
||||||
|
|
||||||
runtime := NewExtensionRuntime(ext)
|
runtime := newExtensionRuntime(ext)
|
||||||
|
|
||||||
validPath, err := runtime.validatePath("test.txt")
|
validPath, err := runtime.validatePath("test.txt")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -177,7 +177,7 @@ func TestExtensionRuntime_FileSandbox(t *testing.T) {
|
|||||||
t.Error("Expected absolute path to be blocked")
|
t.Error("Expected absolute path to be blocked")
|
||||||
}
|
}
|
||||||
|
|
||||||
extNoFile := &LoadedExtension{
|
extNoFile := &loadedExtension{
|
||||||
ID: "test-ext-no-file",
|
ID: "test-ext-no-file",
|
||||||
Manifest: &ExtensionManifest{
|
Manifest: &ExtensionManifest{
|
||||||
Name: "test-ext-no-file",
|
Name: "test-ext-no-file",
|
||||||
@@ -187,7 +187,7 @@ func TestExtensionRuntime_FileSandbox(t *testing.T) {
|
|||||||
},
|
},
|
||||||
DataDir: tempDir,
|
DataDir: tempDir,
|
||||||
}
|
}
|
||||||
runtimeNoFile := NewExtensionRuntime(extNoFile)
|
runtimeNoFile := newExtensionRuntime(extNoFile)
|
||||||
_, err = runtimeNoFile.validatePath("test.txt")
|
_, err = runtimeNoFile.validatePath("test.txt")
|
||||||
if err == nil {
|
if err == nil {
|
||||||
t.Error("Expected file access to be denied without file permission")
|
t.Error("Expected file access to be denied without file permission")
|
||||||
@@ -195,7 +195,7 @@ func TestExtensionRuntime_FileSandbox(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestExtensionRuntime_UtilityFunctions(t *testing.T) {
|
func TestExtensionRuntime_UtilityFunctions(t *testing.T) {
|
||||||
ext := &LoadedExtension{
|
ext := &loadedExtension{
|
||||||
ID: "test-ext",
|
ID: "test-ext",
|
||||||
Manifest: &ExtensionManifest{
|
Manifest: &ExtensionManifest{
|
||||||
Name: "test-ext",
|
Name: "test-ext",
|
||||||
@@ -203,7 +203,7 @@ func TestExtensionRuntime_UtilityFunctions(t *testing.T) {
|
|||||||
DataDir: t.TempDir(),
|
DataDir: t.TempDir(),
|
||||||
}
|
}
|
||||||
|
|
||||||
runtime := NewExtensionRuntime(ext)
|
runtime := newExtensionRuntime(ext)
|
||||||
vm := goja.New()
|
vm := goja.New()
|
||||||
runtime.RegisterAPIs(vm)
|
runtime.RegisterAPIs(vm)
|
||||||
|
|
||||||
@@ -243,7 +243,7 @@ func TestExtensionRuntime_UtilityFunctions(t *testing.T) {
|
|||||||
|
|
||||||
func TestExtensionRuntime_SSRFProtection(t *testing.T) {
|
func TestExtensionRuntime_SSRFProtection(t *testing.T) {
|
||||||
// Create extension with limited network permissions
|
// Create extension with limited network permissions
|
||||||
ext := &LoadedExtension{
|
ext := &loadedExtension{
|
||||||
ID: "test-ext",
|
ID: "test-ext",
|
||||||
Manifest: &ExtensionManifest{
|
Manifest: &ExtensionManifest{
|
||||||
Name: "test-ext",
|
Name: "test-ext",
|
||||||
@@ -254,7 +254,7 @@ func TestExtensionRuntime_SSRFProtection(t *testing.T) {
|
|||||||
DataDir: t.TempDir(),
|
DataDir: t.TempDir(),
|
||||||
}
|
}
|
||||||
|
|
||||||
runtime := NewExtensionRuntime(ext)
|
runtime := newExtensionRuntime(ext)
|
||||||
|
|
||||||
privateIPs := []string{
|
privateIPs := []string{
|
||||||
"http://localhost/admin",
|
"http://localhost/admin",
|
||||||
|
|||||||
@@ -20,6 +20,10 @@ func (e *JSExecutionError) Error() string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func RunWithTimeout(vm *goja.Runtime, script string, timeout time.Duration) (goja.Value, error) {
|
func RunWithTimeout(vm *goja.Runtime, script string, timeout time.Duration) (goja.Value, error) {
|
||||||
|
if vm == nil {
|
||||||
|
return nil, fmt.Errorf("extension runtime unavailable")
|
||||||
|
}
|
||||||
|
|
||||||
if timeout <= 0 {
|
if timeout <= 0 {
|
||||||
timeout = DefaultJSTimeout
|
timeout = DefaultJSTimeout
|
||||||
}
|
}
|
||||||
@@ -49,7 +53,7 @@ func RunWithTimeout(vm *goja.Runtime, script string, timeout time.Duration) (goj
|
|||||||
IsTimeout: true,
|
IsTimeout: true,
|
||||||
}}
|
}}
|
||||||
} else {
|
} else {
|
||||||
GoLog("[ExtensionRuntime] panic during JS execution: %v\n%s\n", r, string(debug.Stack()))
|
GoLog("[extensionRuntime] panic during JS execution: %v\n%s\n", r, string(debug.Stack()))
|
||||||
resultCh <- result{nil, fmt.Errorf("panic during execution: %v", r)}
|
resultCh <- result{nil, fmt.Errorf("panic during execution: %v", r)}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -69,6 +73,11 @@ func RunWithTimeout(vm *goja.Runtime, script string, timeout time.Duration) (goj
|
|||||||
|
|
||||||
vm.Interrupt("execution timeout")
|
vm.Interrupt("execution timeout")
|
||||||
|
|
||||||
|
// MUST wait for the goroutine to finish before returning.
|
||||||
|
// The Goja VM is NOT thread-safe — if we return while the goroutine
|
||||||
|
// is still executing JS (e.g. blocked on an HTTP call), the next
|
||||||
|
// caller will access the VM concurrently and crash with a nil
|
||||||
|
// pointer dereference.
|
||||||
select {
|
select {
|
||||||
case res := <-resultCh:
|
case res := <-resultCh:
|
||||||
if res.err != nil {
|
if res.err != nil {
|
||||||
@@ -78,7 +87,10 @@ func RunWithTimeout(vm *goja.Runtime, script string, timeout time.Duration) (goj
|
|||||||
Message: "execution timeout exceeded",
|
Message: "execution timeout exceeded",
|
||||||
IsTimeout: true,
|
IsTimeout: true,
|
||||||
}
|
}
|
||||||
case <-time.After(1 * time.Second):
|
case <-time.After(60 * time.Second):
|
||||||
|
// Goroutine is truly stuck (e.g. HTTP read with no timeout).
|
||||||
|
// Log a warning — the VM should NOT be reused after this.
|
||||||
|
GoLog("[extensionRuntime] WARNING: JS goroutine did not exit within 60s after interrupt, VM may be unsafe\n")
|
||||||
return nil, &JSExecutionError{
|
return nil, &JSExecutionError{
|
||||||
Message: "execution timeout exceeded (force)",
|
Message: "execution timeout exceeded (force)",
|
||||||
IsTimeout: true,
|
IsTimeout: true,
|
||||||
@@ -92,8 +104,9 @@ func RunWithTimeout(vm *goja.Runtime, script string, timeout time.Duration) (goj
|
|||||||
func RunWithTimeoutAndRecover(vm *goja.Runtime, script string, timeout time.Duration) (goja.Value, error) {
|
func RunWithTimeoutAndRecover(vm *goja.Runtime, script string, timeout time.Duration) (goja.Value, error) {
|
||||||
result, err := RunWithTimeout(vm, script, timeout)
|
result, err := RunWithTimeout(vm, script, timeout)
|
||||||
|
|
||||||
// Clear any interrupt state so VM can be reused
|
if vm != nil {
|
||||||
vm.ClearInterrupt()
|
vm.ClearInterrupt()
|
||||||
|
}
|
||||||
|
|
||||||
return result, err
|
return result, err
|
||||||
}
|
}
|
||||||
|
|||||||
+15
-15
@@ -2,28 +2,28 @@ module github.com/zarz/spotiflac_android/go_backend
|
|||||||
|
|
||||||
go 1.25.0
|
go 1.25.0
|
||||||
|
|
||||||
toolchain go1.25.7
|
toolchain go1.25.8
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/dop251/goja v0.0.0-20260216154549-8b74ce4618c5
|
github.com/dop251/goja v0.0.0-20260311135729-065cd970411c
|
||||||
github.com/go-flac/flacpicture/v2 v2.0.2
|
github.com/go-flac/flacpicture/v2 v2.0.2
|
||||||
github.com/go-flac/flacvorbis/v2 v2.0.2
|
github.com/go-flac/flacvorbis/v2 v2.0.2
|
||||||
github.com/go-flac/go-flac/v2 v2.0.4
|
github.com/go-flac/go-flac/v2 v2.0.4
|
||||||
github.com/refraction-networking/utls v1.8.2
|
github.com/refraction-networking/utls v1.8.2
|
||||||
golang.org/x/mobile v0.0.0-20260211191516-dcd2a3258864
|
golang.org/x/mobile v0.0.0-20260312152759-81488f6aeb60
|
||||||
golang.org/x/net v0.50.0
|
golang.org/x/net v0.52.0
|
||||||
golang.org/x/text v0.34.0
|
golang.org/x/text v0.35.0
|
||||||
)
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/andybalholm/brotli v1.0.6 // indirect
|
github.com/andybalholm/brotli v1.2.0 // indirect
|
||||||
github.com/dlclark/regexp2 v1.11.4 // indirect
|
github.com/dlclark/regexp2 v1.11.5 // indirect
|
||||||
github.com/go-sourcemap/sourcemap v2.1.3+incompatible // indirect
|
github.com/go-sourcemap/sourcemap v2.1.4+incompatible // indirect
|
||||||
github.com/google/pprof v0.0.0-20230207041349-798e818bf904 // indirect
|
github.com/google/pprof v0.0.0-20260302011040-a15ffb7f9dcc // indirect
|
||||||
github.com/klauspost/compress v1.17.4 // indirect
|
github.com/klauspost/compress v1.18.5 // indirect
|
||||||
golang.org/x/crypto v0.48.0 // indirect
|
golang.org/x/crypto v0.49.0 // indirect
|
||||||
golang.org/x/mod v0.33.0 // indirect
|
golang.org/x/mod v0.34.0 // indirect
|
||||||
golang.org/x/sync v0.19.0 // indirect
|
golang.org/x/sync v0.20.0 // indirect
|
||||||
golang.org/x/sys v0.41.0 // indirect
|
golang.org/x/sys v0.42.0 // indirect
|
||||||
golang.org/x/tools v0.42.0 // indirect
|
golang.org/x/tools v0.43.0 // indirect
|
||||||
)
|
)
|
||||||
|
|||||||
+30
-28
@@ -1,49 +1,51 @@
|
|||||||
github.com/Masterminds/semver/v3 v3.2.1 h1:RN9w6+7QoMeJVGyfmbcgs28Br8cvmnucEXnY0rYXWg0=
|
github.com/Masterminds/semver/v3 v3.2.1 h1:RN9w6+7QoMeJVGyfmbcgs28Br8cvmnucEXnY0rYXWg0=
|
||||||
github.com/Masterminds/semver/v3 v3.2.1/go.mod h1:qvl/7zhW3nngYb5+80sSMF+FG2BjYrf8m9wsX0PNOMQ=
|
github.com/Masterminds/semver/v3 v3.2.1/go.mod h1:qvl/7zhW3nngYb5+80sSMF+FG2BjYrf8m9wsX0PNOMQ=
|
||||||
github.com/andybalholm/brotli v1.0.6 h1:Yf9fFpf49Zrxb9NlQaluyE92/+X7UVHlhMNJN2sxfOI=
|
github.com/andybalholm/brotli v1.2.0 h1:ukwgCxwYrmACq68yiUqwIWnGY0cTPox/M94sVwToPjQ=
|
||||||
github.com/andybalholm/brotli v1.0.6/go.mod h1:fO7iG3H7G2nSZ7m0zPUDn85XEX2GTukHGRSepvi9Eig=
|
github.com/andybalholm/brotli v1.2.0/go.mod h1:rzTDkvFWvIrjDXZHkuS16NPggd91W3kUSvPlQ1pLaKY=
|
||||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
||||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/dlclark/regexp2 v1.11.4 h1:rPYF9/LECdNymJufQKmri9gV604RvvABwgOA8un7yAo=
|
github.com/dlclark/regexp2 v1.11.5 h1:Q/sSnsKerHeCkc/jSTNq1oCm7KiVgUMZRDUoRu0JQZQ=
|
||||||
github.com/dlclark/regexp2 v1.11.4/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8=
|
github.com/dlclark/regexp2 v1.11.5/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8=
|
||||||
github.com/dop251/goja v0.0.0-20260216154549-8b74ce4618c5 h1:QckvTXtu55YMopmVeDrPQ/r+T6xjw8KMCmE3UgUldkw=
|
github.com/dop251/goja v0.0.0-20260311135729-065cd970411c h1:OcLmPfx1T1RmZVHHFwWMPaZDdRf0DBMZOFMVWJa7Pdk=
|
||||||
github.com/dop251/goja v0.0.0-20260216154549-8b74ce4618c5/go.mod h1:MxLav0peU43GgvwVgNbLAj1s/bSGboKkhuULvq/7hx4=
|
github.com/dop251/goja v0.0.0-20260311135729-065cd970411c/go.mod h1:MxLav0peU43GgvwVgNbLAj1s/bSGboKkhuULvq/7hx4=
|
||||||
github.com/go-flac/flacpicture/v2 v2.0.2 h1:HCaJIVZpxnpdWs6G3ECEVRelzqS5xOi1Ba1AGmtXbzE=
|
github.com/go-flac/flacpicture/v2 v2.0.2 h1:HCaJIVZpxnpdWs6G3ECEVRelzqS5xOi1Ba1AGmtXbzE=
|
||||||
github.com/go-flac/flacpicture/v2 v2.0.2/go.mod h1:DMZBPWPAmdLqNhqFSy5ZBs9wyBzOekXutGfP7/TFCuo=
|
github.com/go-flac/flacpicture/v2 v2.0.2/go.mod h1:DMZBPWPAmdLqNhqFSy5ZBs9wyBzOekXutGfP7/TFCuo=
|
||||||
github.com/go-flac/flacvorbis/v2 v2.0.2 h1:xCL3OhxrxWkHrbWUBvGNe+6FQ03yLmBbz0v5z4V2PoQ=
|
github.com/go-flac/flacvorbis/v2 v2.0.2 h1:xCL3OhxrxWkHrbWUBvGNe+6FQ03yLmBbz0v5z4V2PoQ=
|
||||||
github.com/go-flac/flacvorbis/v2 v2.0.2/go.mod h1:SwTB5gs13VaM/N7rstwPoUsPibiMKklgwybYP9dYo2g=
|
github.com/go-flac/flacvorbis/v2 v2.0.2/go.mod h1:SwTB5gs13VaM/N7rstwPoUsPibiMKklgwybYP9dYo2g=
|
||||||
github.com/go-flac/go-flac/v2 v2.0.4 h1:atf/kFa8U9idtkA//NO22XGr+MzQLeXZecnmP9sYBf0=
|
github.com/go-flac/go-flac/v2 v2.0.4 h1:atf/kFa8U9idtkA//NO22XGr+MzQLeXZecnmP9sYBf0=
|
||||||
github.com/go-flac/go-flac/v2 v2.0.4/go.mod h1:sYOlTKxutMW0RDYF+KlD6Zn+VOCZlIFQG/r/usPveCs=
|
github.com/go-flac/go-flac/v2 v2.0.4/go.mod h1:sYOlTKxutMW0RDYF+KlD6Zn+VOCZlIFQG/r/usPveCs=
|
||||||
github.com/go-sourcemap/sourcemap v2.1.3+incompatible h1:W1iEw64niKVGogNgBN3ePyLFfuisuzeidWPMPWmECqU=
|
github.com/go-sourcemap/sourcemap v2.1.4+incompatible h1:a+iTbH5auLKxaNwQFg0B+TCYl6lbukKPc7b5x0n1s6Q=
|
||||||
github.com/go-sourcemap/sourcemap v2.1.3+incompatible/go.mod h1:F8jJfvm2KbVjc5NqelyYJmf/v5J0dwNLS2mL4sNA1Jg=
|
github.com/go-sourcemap/sourcemap v2.1.4+incompatible/go.mod h1:F8jJfvm2KbVjc5NqelyYJmf/v5J0dwNLS2mL4sNA1Jg=
|
||||||
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
|
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
|
||||||
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
|
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
|
||||||
github.com/google/pprof v0.0.0-20230207041349-798e818bf904 h1:4/hN5RUoecvl+RmJRE2YxKWtnnQls6rQjjW5oV7qg2U=
|
github.com/google/pprof v0.0.0-20260302011040-a15ffb7f9dcc h1:VBbFa1lDYWEeV5FZKUiYKYT0VxCp9twUmmaq9eb8sXw=
|
||||||
github.com/google/pprof v0.0.0-20230207041349-798e818bf904/go.mod h1:uglQLonpP8qtYCYyzA+8c/9qtqgA3qsXGYqCPKARAFg=
|
github.com/google/pprof v0.0.0-20260302011040-a15ffb7f9dcc/go.mod h1:MxpfABSjhmINe3F1It9d+8exIHFvUqtLIRCdOGNXqiI=
|
||||||
github.com/klauspost/compress v1.17.4 h1:Ej5ixsIri7BrIjBkRZLTo6ghwrEtHFk7ijlczPW4fZ4=
|
github.com/klauspost/compress v1.18.5 h1:/h1gH5Ce+VWNLSWqPzOVn6XBO+vJbCNGvjoaGBFW2IE=
|
||||||
github.com/klauspost/compress v1.17.4/go.mod h1:/dCuZOvVtNoHsyb+cuJD3itjs3NbnF6KH9zAO4BDxPM=
|
github.com/klauspost/compress v1.18.5/go.mod h1:cwPg85FWrGar70rWktvGQj8/hthj3wpl0PGDogxkrSQ=
|
||||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||||
github.com/refraction-networking/utls v1.8.2 h1:j4Q1gJj0xngdeH+Ox/qND11aEfhpgoEvV+S9iJ2IdQo=
|
github.com/refraction-networking/utls v1.8.2 h1:j4Q1gJj0xngdeH+Ox/qND11aEfhpgoEvV+S9iJ2IdQo=
|
||||||
github.com/refraction-networking/utls v1.8.2/go.mod h1:jkSOEkLqn+S/jtpEHPOsVv/4V4EVnelwbMQl4vCWXAM=
|
github.com/refraction-networking/utls v1.8.2/go.mod h1:jkSOEkLqn+S/jtpEHPOsVv/4V4EVnelwbMQl4vCWXAM=
|
||||||
github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA=
|
github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA=
|
||||||
github.com/stretchr/testify v1.10.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
|
github.com/stretchr/testify v1.10.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
|
||||||
golang.org/x/crypto v0.48.0 h1:/VRzVqiRSggnhY7gNRxPauEQ5Drw9haKdM0jqfcCFts=
|
github.com/xyproto/randomstring v1.0.5 h1:YtlWPoRdgMu3NZtP45drfy1GKoojuR7hmRcnhZqKjWU=
|
||||||
golang.org/x/crypto v0.48.0/go.mod h1:r0kV5h3qnFPlQnBSrULhlsRfryS2pmewsg+XfMgkVos=
|
github.com/xyproto/randomstring v1.0.5/go.mod h1:rgmS5DeNXLivK7YprL0pY+lTuhNQW3iGxZ18UQApw/E=
|
||||||
golang.org/x/mobile v0.0.0-20260211191516-dcd2a3258864 h1:cTVynMSsMYgbUrtia2HB1jrhdUwQNtQti91vUCyjMp4=
|
golang.org/x/crypto v0.49.0 h1:+Ng2ULVvLHnJ/ZFEq4KdcDd/cfjrrjjNSXNzxg0Y4U4=
|
||||||
golang.org/x/mobile v0.0.0-20260211191516-dcd2a3258864/go.mod h1:4OGHIUSBiIqyFAQDaX1tpY0BVnO20DvNDeATBu8aeFQ=
|
golang.org/x/crypto v0.49.0/go.mod h1:ErX4dUh2UM+CFYiXZRTcMpEcN8b/1gxEuv3nODoYtCA=
|
||||||
golang.org/x/mod v0.33.0 h1:tHFzIWbBifEmbwtGz65eaWyGiGZatSrT9prnU8DbVL8=
|
golang.org/x/mobile v0.0.0-20260312152759-81488f6aeb60 h1:MOzyaj0wu2xneBkzkg9LHNYjDBB4W5vP043A2SYQRPA=
|
||||||
golang.org/x/mod v0.33.0/go.mod h1:swjeQEj+6r7fODbD2cqrnje9PnziFuw4bmLbBZFrQ5w=
|
golang.org/x/mobile v0.0.0-20260312152759-81488f6aeb60/go.mod h1:th6VJvzjMbrYF8SduQY5rpD0HG0GleGxjadkqSxFs3k=
|
||||||
golang.org/x/net v0.50.0 h1:ucWh9eiCGyDR3vtzso0WMQinm2Dnt8cFMuQa9K33J60=
|
golang.org/x/mod v0.34.0 h1:xIHgNUUnW6sYkcM5Jleh05DvLOtwc6RitGHbDk4akRI=
|
||||||
golang.org/x/net v0.50.0/go.mod h1:UgoSli3F/pBgdJBHCTc+tp3gmrU4XswgGRgtnwWTfyM=
|
golang.org/x/mod v0.34.0/go.mod h1:ykgH52iCZe79kzLLMhyCUzhMci+nQj+0XkbXpNYtVjY=
|
||||||
golang.org/x/sync v0.19.0 h1:vV+1eWNmZ5geRlYjzm2adRgW2/mcpevXNg50YZtPCE4=
|
golang.org/x/net v0.52.0 h1:He/TN1l0e4mmR3QqHMT2Xab3Aj3L9qjbhRm78/6jrW0=
|
||||||
golang.org/x/sync v0.19.0/go.mod h1:9KTHXmSnoGruLpwFjVSX0lNNA75CykiMECbovNTZqGI=
|
golang.org/x/net v0.52.0/go.mod h1:R1MAz7uMZxVMualyPXb+VaqGSa3LIaUqk0eEt3w36Sw=
|
||||||
golang.org/x/sys v0.41.0 h1:Ivj+2Cp/ylzLiEU89QhWblYnOE9zerudt9Ftecq2C6k=
|
golang.org/x/sync v0.20.0 h1:e0PTpb7pjO8GAtTs2dQ6jYa5BWYlMuX047Dco/pItO4=
|
||||||
golang.org/x/sys v0.41.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks=
|
golang.org/x/sync v0.20.0/go.mod h1:9xrNwdLfx4jkKbNva9FpL6vEN7evnE43NNNJQ2LF3+0=
|
||||||
golang.org/x/text v0.34.0 h1:oL/Qq0Kdaqxa1KbNeMKwQq0reLCCaFtqu2eNuSeNHbk=
|
golang.org/x/sys v0.42.0 h1:omrd2nAlyT5ESRdCLYdm3+fMfNFE/+Rf4bDIQImRJeo=
|
||||||
golang.org/x/text v0.34.0/go.mod h1:homfLqTYRFyVYemLBFl5GgL/DWEiH5wcsQ5gSh1yziA=
|
golang.org/x/sys v0.42.0/go.mod h1:4GL1E5IUh+htKOUEOaiffhrAeqysfVGipDYzABqnCmw=
|
||||||
golang.org/x/tools v0.42.0 h1:uNgphsn75Tdz5Ji2q36v/nsFSfR/9BRFvqhGBaJGd5k=
|
golang.org/x/text v0.35.0 h1:JOVx6vVDFokkpaq1AEptVzLTpDe9KGpj5tR4/X+ybL8=
|
||||||
golang.org/x/tools v0.42.0/go.mod h1:Ma6lCIwGZvHK6XtgbswSoWroEkhugApmsXyrUmBhfr0=
|
golang.org/x/text v0.35.0/go.mod h1:khi/HExzZJ2pGnjenulevKNX1W67CUy0AsXcNubPGCA=
|
||||||
|
golang.org/x/tools v0.43.0 h1:12BdW9CeB3Z+J/I/wj34VMl8X+fEXBxVR90JeMX5E7s=
|
||||||
|
golang.org/x/tools v0.43.0/go.mod h1:uHkMso649BX2cZK6+RpuIPXS3ho2hZo4FVwfoy1vIk0=
|
||||||
gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=
|
gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=
|
||||||
gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ=
|
gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ=
|
||||||
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
||||||
|
|||||||
@@ -66,9 +66,6 @@ var sharedTransport = &http.Transport{
|
|||||||
DisableCompression: true,
|
DisableCompression: true,
|
||||||
}
|
}
|
||||||
|
|
||||||
// metadataTransport is a separate transport for metadata API calls (Deezer, Spotify, SongLink).
|
|
||||||
// Isolated from download traffic so that download failures cannot poison
|
|
||||||
// the connection pool used by metadata enrichment.
|
|
||||||
var metadataTransport = &http.Transport{
|
var metadataTransport = &http.Transport{
|
||||||
DialContext: (&net.Dialer{
|
DialContext: (&net.Dialer{
|
||||||
Timeout: 30 * time.Second,
|
Timeout: 30 * time.Second,
|
||||||
@@ -104,8 +101,6 @@ func NewHTTPClientWithTimeout(timeout time.Duration) *http.Client {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewMetadataHTTPClient creates an HTTP client using the isolated metadata transport.
|
|
||||||
// Use this for API calls that should not be affected by download traffic.
|
|
||||||
func NewMetadataHTTPClient(timeout time.Duration) *http.Client {
|
func NewMetadataHTTPClient(timeout time.Duration) *http.Client {
|
||||||
return &http.Client{
|
return &http.Client{
|
||||||
Transport: newCompatibilityTransport(metadataTransport),
|
Transport: newCompatibilityTransport(metadataTransport),
|
||||||
@@ -229,7 +224,6 @@ func cloneRequestWithHTTPScheme(req *http.Request, scheme string) (*http.Request
|
|||||||
return reqCopy, nil
|
return reqCopy, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Also checks for ISP blocking on errors
|
|
||||||
func DoRequestWithUserAgent(client *http.Client, req *http.Request) (*http.Response, error) {
|
func DoRequestWithUserAgent(client *http.Client, req *http.Request) (*http.Response, error) {
|
||||||
req.Header.Set("User-Agent", getRandomUserAgent())
|
req.Header.Set("User-Agent", getRandomUserAgent())
|
||||||
resp, err := client.Do(req)
|
resp, err := client.Do(req)
|
||||||
@@ -239,7 +233,6 @@ func DoRequestWithUserAgent(client *http.Client, req *http.Request) (*http.Respo
|
|||||||
return resp, err
|
return resp, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// RetryConfig holds configuration for retry logic
|
|
||||||
type RetryConfig struct {
|
type RetryConfig struct {
|
||||||
MaxRetries int
|
MaxRetries int
|
||||||
InitialDelay time.Duration
|
InitialDelay time.Duration
|
||||||
|
|||||||
@@ -6,17 +6,10 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
)
|
)
|
||||||
|
|
||||||
// iOS version: uTLS is not supported on iOS due to cgo DNS resolver issues
|
|
||||||
// Fall back to standard HTTP client
|
|
||||||
|
|
||||||
// GetCloudflareBypassClient returns the standard HTTP client on iOS
|
|
||||||
// uTLS is not available on iOS due to cgo DNS resolver compatibility issues
|
|
||||||
func GetCloudflareBypassClient() *http.Client {
|
func GetCloudflareBypassClient() *http.Client {
|
||||||
return sharedClient
|
return sharedClient
|
||||||
}
|
}
|
||||||
|
|
||||||
// DoRequestWithCloudflareBypass on iOS just uses the standard client
|
|
||||||
// uTLS Chrome fingerprint bypass is not available on iOS
|
|
||||||
func DoRequestWithCloudflareBypass(req *http.Request) (*http.Response, error) {
|
func DoRequestWithCloudflareBypass(req *http.Request) (*http.Response, error) {
|
||||||
req.Header.Set("User-Agent", getRandomUserAgent())
|
req.Header.Set("User-Agent", getRandomUserAgent())
|
||||||
resp, err := sharedClient.Do(req)
|
resp, err := sharedClient.Do(req)
|
||||||
|
|||||||
@@ -16,8 +16,6 @@ import (
|
|||||||
"golang.org/x/net/http2"
|
"golang.org/x/net/http2"
|
||||||
)
|
)
|
||||||
|
|
||||||
// uTLS transport that mimics Chrome's TLS fingerprint to bypass Cloudflare
|
|
||||||
// Uses HTTP/2 for optimal performance as uTLS works best with HTTP/2
|
|
||||||
type utlsTransport struct {
|
type utlsTransport struct {
|
||||||
dialer *net.Dialer
|
dialer *net.Dialer
|
||||||
mu sync.Mutex
|
mu sync.Mutex
|
||||||
@@ -98,15 +96,10 @@ var cloudflareBypassClient = &http.Client{
|
|||||||
Timeout: DefaultTimeout,
|
Timeout: DefaultTimeout,
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetCloudflareBypassClient returns an HTTP client that mimics Chrome's TLS fingerprint
|
|
||||||
// Use this when requests are blocked by Cloudflare (common when using VPN)
|
|
||||||
func GetCloudflareBypassClient() *http.Client {
|
func GetCloudflareBypassClient() *http.Client {
|
||||||
return cloudflareBypassClient
|
return cloudflareBypassClient
|
||||||
}
|
}
|
||||||
|
|
||||||
// DoRequestWithCloudflareBypass attempts request with standard client first,
|
|
||||||
// then retries with uTLS Chrome fingerprint if Cloudflare blocks it.
|
|
||||||
// This is useful when using VPN as Cloudflare detects Go's default TLS fingerprint.
|
|
||||||
func DoRequestWithCloudflareBypass(req *http.Request) (*http.Response, error) {
|
func DoRequestWithCloudflareBypass(req *http.Request) (*http.Response, error) {
|
||||||
req.Header.Set("User-Agent", getRandomUserAgent())
|
req.Header.Set("User-Agent", getRandomUserAgent())
|
||||||
|
|
||||||
@@ -142,7 +135,6 @@ func DoRequestWithCloudflareBypass(req *http.Request) (*http.Response, error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Not Cloudflare, return original response (recreate body)
|
|
||||||
return &http.Response{
|
return &http.Response{
|
||||||
Status: resp.Status,
|
Status: resp.Status,
|
||||||
StatusCode: resp.StatusCode,
|
StatusCode: resp.StatusCode,
|
||||||
|
|||||||
@@ -10,8 +10,6 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
// IDHSClient is a client for I Don't Have Spotify API
|
|
||||||
// Used as fallback when SongLink fails or is rate limited
|
|
||||||
type IDHSClient struct {
|
type IDHSClient struct {
|
||||||
client *http.Client
|
client *http.Client
|
||||||
}
|
}
|
||||||
@@ -55,7 +53,6 @@ func NewIDHSClient() *IDHSClient {
|
|||||||
return globalIDHSClient
|
return globalIDHSClient
|
||||||
}
|
}
|
||||||
|
|
||||||
// Search converts a music link to links on other platforms
|
|
||||||
func (c *IDHSClient) Search(link string, adapters []string) (*IDHSSearchResponse, error) {
|
func (c *IDHSClient) Search(link string, adapters []string) (*IDHSSearchResponse, error) {
|
||||||
idhsRateLimiter.WaitForSlot()
|
idhsRateLimiter.WaitForSlot()
|
||||||
|
|
||||||
@@ -109,7 +106,6 @@ func (c *IDHSClient) Search(link string, adapters []string) (*IDHSSearchResponse
|
|||||||
return &result, nil
|
return &result, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetAvailabilityFromSpotify checks track availability using IDHS as fallback
|
|
||||||
func (c *IDHSClient) GetAvailabilityFromSpotify(spotifyTrackID string) (*TrackAvailability, error) {
|
func (c *IDHSClient) GetAvailabilityFromSpotify(spotifyTrackID string) (*TrackAvailability, error) {
|
||||||
spotifyURL := fmt.Sprintf("https://open.spotify.com/track/%s", spotifyTrackID)
|
spotifyURL := fmt.Sprintf("https://open.spotify.com/track/%s", spotifyTrackID)
|
||||||
|
|
||||||
|
|||||||
@@ -24,13 +24,18 @@ type LibraryScanResult struct {
|
|||||||
FileModTime int64 `json:"fileModTime,omitempty"` // Unix timestamp in milliseconds
|
FileModTime int64 `json:"fileModTime,omitempty"` // Unix timestamp in milliseconds
|
||||||
ISRC string `json:"isrc,omitempty"`
|
ISRC string `json:"isrc,omitempty"`
|
||||||
TrackNumber int `json:"trackNumber,omitempty"`
|
TrackNumber int `json:"trackNumber,omitempty"`
|
||||||
|
TotalTracks int `json:"totalTracks,omitempty"`
|
||||||
DiscNumber int `json:"discNumber,omitempty"`
|
DiscNumber int `json:"discNumber,omitempty"`
|
||||||
|
TotalDiscs int `json:"totalDiscs,omitempty"`
|
||||||
Duration int `json:"duration,omitempty"`
|
Duration int `json:"duration,omitempty"`
|
||||||
ReleaseDate string `json:"releaseDate,omitempty"`
|
ReleaseDate string `json:"releaseDate,omitempty"`
|
||||||
BitDepth int `json:"bitDepth,omitempty"`
|
BitDepth int `json:"bitDepth,omitempty"`
|
||||||
SampleRate int `json:"sampleRate,omitempty"`
|
SampleRate int `json:"sampleRate,omitempty"`
|
||||||
Bitrate int `json:"bitrate,omitempty"` // kbps, for lossy formats (MP3, Opus, Vorbis)
|
Bitrate int `json:"bitrate,omitempty"` // kbps, for lossy formats (MP3, Opus, Vorbis)
|
||||||
Genre string `json:"genre,omitempty"`
|
Genre string `json:"genre,omitempty"`
|
||||||
|
Composer string `json:"composer,omitempty"`
|
||||||
|
Label string `json:"label,omitempty"`
|
||||||
|
Copyright string `json:"copyright,omitempty"`
|
||||||
Format string `json:"format,omitempty"`
|
Format string `json:"format,omitempty"`
|
||||||
MetadataFromFilename bool `json:"metadataFromFilename,omitempty"`
|
MetadataFromFilename bool `json:"metadataFromFilename,omitempty"`
|
||||||
}
|
}
|
||||||
@@ -66,6 +71,9 @@ var supportedAudioFormats = map[string]bool{
|
|||||||
".mp3": true,
|
".mp3": true,
|
||||||
".opus": true,
|
".opus": true,
|
||||||
".ogg": true,
|
".ogg": true,
|
||||||
|
".ape": true,
|
||||||
|
".wv": true,
|
||||||
|
".mpc": true,
|
||||||
".cue": true,
|
".cue": true,
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -170,11 +178,9 @@ func ScanLibraryFolder(folderPath string) (string, error) {
|
|||||||
scanTime := time.Now().UTC().Format(time.RFC3339)
|
scanTime := time.Now().UTC().Format(time.RFC3339)
|
||||||
errorCount := 0
|
errorCount := 0
|
||||||
|
|
||||||
// Track audio files referenced by .cue sheets to avoid duplicates
|
|
||||||
cueReferencedAudioFiles := make(map[string]bool)
|
cueReferencedAudioFiles := make(map[string]bool)
|
||||||
parsedCueFiles := make(map[string]scannedCueFileInfo)
|
parsedCueFiles := make(map[string]scannedCueFileInfo)
|
||||||
|
|
||||||
// First pass: scan .cue files to collect referenced audio paths
|
|
||||||
for _, fileInfo := range audioFileInfos {
|
for _, fileInfo := range audioFileInfos {
|
||||||
filePath := fileInfo.path
|
filePath := fileInfo.path
|
||||||
ext := strings.ToLower(filepath.Ext(filePath))
|
ext := strings.ToLower(filepath.Ext(filePath))
|
||||||
@@ -209,7 +215,6 @@ func ScanLibraryFolder(folderPath string) (string, error) {
|
|||||||
|
|
||||||
ext := strings.ToLower(filepath.Ext(filePath))
|
ext := strings.ToLower(filepath.Ext(filePath))
|
||||||
|
|
||||||
// Handle .cue files: produce multiple track results
|
|
||||||
if ext == ".cue" {
|
if ext == ".cue" {
|
||||||
var cueResults []LibraryScanResult
|
var cueResults []LibraryScanResult
|
||||||
cueInfo, ok := parsedCueFiles[filePath]
|
cueInfo, ok := parsedCueFiles[filePath]
|
||||||
@@ -318,6 +323,8 @@ func scanAudioFileWithKnownModTimeAndDisplayNameAndCoverCacheKey(filePath, displ
|
|||||||
return scanMP3File(filePath, result, displayNameHint)
|
return scanMP3File(filePath, result, displayNameHint)
|
||||||
case ".opus", ".ogg":
|
case ".opus", ".ogg":
|
||||||
return scanOggFile(filePath, result, displayNameHint)
|
return scanOggFile(filePath, result, displayNameHint)
|
||||||
|
case ".ape", ".wv", ".mpc":
|
||||||
|
return scanAPEFile(filePath, result, displayNameHint)
|
||||||
default:
|
default:
|
||||||
return scanFromFilename(filePath, displayNameHint, result)
|
return scanFromFilename(filePath, displayNameHint, result)
|
||||||
}
|
}
|
||||||
@@ -363,9 +370,14 @@ func scanFLACFile(filePath string, result *LibraryScanResult, displayNameHint st
|
|||||||
result.AlbumArtist = metadata.AlbumArtist
|
result.AlbumArtist = metadata.AlbumArtist
|
||||||
result.ISRC = metadata.ISRC
|
result.ISRC = metadata.ISRC
|
||||||
result.TrackNumber = metadata.TrackNumber
|
result.TrackNumber = metadata.TrackNumber
|
||||||
|
result.TotalTracks = metadata.TotalTracks
|
||||||
result.DiscNumber = metadata.DiscNumber
|
result.DiscNumber = metadata.DiscNumber
|
||||||
|
result.TotalDiscs = metadata.TotalDiscs
|
||||||
result.ReleaseDate = metadata.Date
|
result.ReleaseDate = metadata.Date
|
||||||
result.Genre = metadata.Genre
|
result.Genre = metadata.Genre
|
||||||
|
result.Composer = metadata.Composer
|
||||||
|
result.Label = metadata.Label
|
||||||
|
result.Copyright = metadata.Copyright
|
||||||
|
|
||||||
quality, err := GetAudioQuality(filePath)
|
quality, err := GetAudioQuality(filePath)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
@@ -395,12 +407,17 @@ func scanM4AFile(filePath string, result *LibraryScanResult, displayNameHint str
|
|||||||
result.AlbumArtist = metadata.AlbumArtist
|
result.AlbumArtist = metadata.AlbumArtist
|
||||||
result.ISRC = metadata.ISRC
|
result.ISRC = metadata.ISRC
|
||||||
result.TrackNumber = metadata.TrackNumber
|
result.TrackNumber = metadata.TrackNumber
|
||||||
|
result.TotalTracks = metadata.TotalTracks
|
||||||
result.DiscNumber = metadata.DiscNumber
|
result.DiscNumber = metadata.DiscNumber
|
||||||
|
result.TotalDiscs = metadata.TotalDiscs
|
||||||
result.ReleaseDate = metadata.Date
|
result.ReleaseDate = metadata.Date
|
||||||
if result.ReleaseDate == "" {
|
if result.ReleaseDate == "" {
|
||||||
result.ReleaseDate = metadata.Year
|
result.ReleaseDate = metadata.Year
|
||||||
}
|
}
|
||||||
result.Genre = metadata.Genre
|
result.Genre = metadata.Genre
|
||||||
|
result.Composer = metadata.Composer
|
||||||
|
result.Label = metadata.Label
|
||||||
|
result.Copyright = metadata.Copyright
|
||||||
}
|
}
|
||||||
|
|
||||||
quality, err := GetM4AQuality(filePath)
|
quality, err := GetM4AQuality(filePath)
|
||||||
@@ -425,7 +442,9 @@ func scanMP3File(filePath string, result *LibraryScanResult, displayNameHint str
|
|||||||
result.AlbumName = metadata.Album
|
result.AlbumName = metadata.Album
|
||||||
result.AlbumArtist = metadata.AlbumArtist
|
result.AlbumArtist = metadata.AlbumArtist
|
||||||
result.TrackNumber = metadata.TrackNumber
|
result.TrackNumber = metadata.TrackNumber
|
||||||
|
result.TotalTracks = metadata.TotalTracks
|
||||||
result.DiscNumber = metadata.DiscNumber
|
result.DiscNumber = metadata.DiscNumber
|
||||||
|
result.TotalDiscs = metadata.TotalDiscs
|
||||||
result.Genre = metadata.Genre
|
result.Genre = metadata.Genre
|
||||||
if metadata.Date != "" {
|
if metadata.Date != "" {
|
||||||
result.ReleaseDate = metadata.Date
|
result.ReleaseDate = metadata.Date
|
||||||
@@ -433,6 +452,9 @@ func scanMP3File(filePath string, result *LibraryScanResult, displayNameHint str
|
|||||||
result.ReleaseDate = metadata.Year
|
result.ReleaseDate = metadata.Year
|
||||||
}
|
}
|
||||||
result.ISRC = metadata.ISRC
|
result.ISRC = metadata.ISRC
|
||||||
|
result.Composer = metadata.Composer
|
||||||
|
result.Label = metadata.Label
|
||||||
|
result.Copyright = metadata.Copyright
|
||||||
|
|
||||||
quality, err := GetMP3Quality(filePath)
|
quality, err := GetMP3Quality(filePath)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
@@ -462,9 +484,14 @@ func scanOggFile(filePath string, result *LibraryScanResult, displayNameHint str
|
|||||||
result.AlbumArtist = metadata.AlbumArtist
|
result.AlbumArtist = metadata.AlbumArtist
|
||||||
result.ISRC = metadata.ISRC
|
result.ISRC = metadata.ISRC
|
||||||
result.TrackNumber = metadata.TrackNumber
|
result.TrackNumber = metadata.TrackNumber
|
||||||
|
result.TotalTracks = metadata.TotalTracks
|
||||||
result.DiscNumber = metadata.DiscNumber
|
result.DiscNumber = metadata.DiscNumber
|
||||||
|
result.TotalDiscs = metadata.TotalDiscs
|
||||||
result.Genre = metadata.Genre
|
result.Genre = metadata.Genre
|
||||||
result.ReleaseDate = metadata.Date
|
result.ReleaseDate = metadata.Date
|
||||||
|
result.Composer = metadata.Composer
|
||||||
|
result.Label = metadata.Label
|
||||||
|
result.Copyright = metadata.Copyright
|
||||||
|
|
||||||
quality, err := GetOggQuality(filePath)
|
quality, err := GetOggQuality(filePath)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
@@ -481,6 +508,42 @@ func scanOggFile(filePath string, result *LibraryScanResult, displayNameHint str
|
|||||||
return result, nil
|
return result, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func scanAPEFile(filePath string, result *LibraryScanResult, displayNameHint string) (*LibraryScanResult, error) {
|
||||||
|
tag, err := ReadAPETags(filePath)
|
||||||
|
if err != nil {
|
||||||
|
GoLog("[LibraryScan] APE tag read error for %s: %v\n", filePath, err)
|
||||||
|
return scanFromFilename(filePath, displayNameHint, result)
|
||||||
|
}
|
||||||
|
|
||||||
|
metadata := APETagToAudioMetadata(tag)
|
||||||
|
if metadata == nil {
|
||||||
|
return scanFromFilename(filePath, displayNameHint, result)
|
||||||
|
}
|
||||||
|
|
||||||
|
result.TrackName = metadata.Title
|
||||||
|
result.ArtistName = metadata.Artist
|
||||||
|
result.AlbumName = metadata.Album
|
||||||
|
result.AlbumArtist = metadata.AlbumArtist
|
||||||
|
result.ISRC = metadata.ISRC
|
||||||
|
result.TrackNumber = metadata.TrackNumber
|
||||||
|
result.TotalTracks = metadata.TotalTracks
|
||||||
|
result.DiscNumber = metadata.DiscNumber
|
||||||
|
result.TotalDiscs = metadata.TotalDiscs
|
||||||
|
result.Genre = metadata.Genre
|
||||||
|
if metadata.Date != "" {
|
||||||
|
result.ReleaseDate = metadata.Date
|
||||||
|
} else {
|
||||||
|
result.ReleaseDate = metadata.Year
|
||||||
|
}
|
||||||
|
result.Composer = metadata.Composer
|
||||||
|
result.Label = metadata.Label
|
||||||
|
result.Copyright = metadata.Copyright
|
||||||
|
|
||||||
|
applyDefaultLibraryMetadata(filePath, displayNameHint, result)
|
||||||
|
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
||||||
func scanFromFilename(filePath, displayNameHint string, result *LibraryScanResult) (*LibraryScanResult, error) {
|
func scanFromFilename(filePath, displayNameHint string, result *LibraryScanResult) (*LibraryScanResult, error) {
|
||||||
result.MetadataFromFilename = true
|
result.MetadataFromFilename = true
|
||||||
nameSource := libraryDisplayNameOrPath(filePath, displayNameHint)
|
nameSource := libraryDisplayNameOrPath(filePath, displayNameHint)
|
||||||
@@ -827,9 +890,6 @@ func scanLibraryFolderIncrementalWithExistingFiles(folderPath string, existingFi
|
|||||||
return string(jsonBytes), nil
|
return string(jsonBytes), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// ScanLibraryFolderIncremental performs an incremental scan of the library folder
|
|
||||||
// existingFilesJSON is a JSON object mapping filePath -> modTime (unix millis)
|
|
||||||
// Only files that are new or have changed modification time will be scanned
|
|
||||||
func ScanLibraryFolderIncremental(folderPath, existingFilesJSON string) (string, error) {
|
func ScanLibraryFolderIncremental(folderPath, existingFilesJSON string) (string, error) {
|
||||||
existingFiles := make(map[string]int64)
|
existingFiles := make(map[string]int64)
|
||||||
if existingFilesJSON != "" && existingFilesJSON != "{}" {
|
if existingFilesJSON != "" && existingFilesJSON != "{}" {
|
||||||
|
|||||||
@@ -51,7 +51,7 @@ func GetLogBuffer() *LogBuffer {
|
|||||||
globalLogBuffer = &LogBuffer{
|
globalLogBuffer = &LogBuffer{
|
||||||
entries: make([]LogEntry, 0, defaultLogBufferSize),
|
entries: make([]LogEntry, 0, defaultLogBufferSize),
|
||||||
maxSize: defaultLogBufferSize,
|
maxSize: defaultLogBufferSize,
|
||||||
loggingEnabled: false, // Default: disabled for performance (user can enable in settings)
|
loggingEnabled: false,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
return globalLogBuffer
|
return globalLogBuffer
|
||||||
@@ -145,13 +145,10 @@ func LogError(tag, format string, args ...interface{}) {
|
|||||||
GetLogBuffer().Add("ERROR", tag, fmt.Sprintf(format, args...))
|
GetLogBuffer().Add("ERROR", tag, fmt.Sprintf(format, args...))
|
||||||
}
|
}
|
||||||
|
|
||||||
// GoLog is a drop-in replacement for fmt.Printf that also logs to buffer
|
|
||||||
// It parses the tag from the format string if it starts with [Tag]
|
|
||||||
func GoLog(format string, args ...interface{}) {
|
func GoLog(format string, args ...interface{}) {
|
||||||
message := fmt.Sprintf(format, args...)
|
message := fmt.Sprintf(format, args...)
|
||||||
message = strings.TrimSuffix(message, "\n")
|
message = strings.TrimSuffix(message, "\n")
|
||||||
|
|
||||||
// Extract tag from message if present (e.g., "[Tidal] message")
|
|
||||||
tag := "Go"
|
tag := "Go"
|
||||||
level := "INFO"
|
level := "INFO"
|
||||||
|
|
||||||
@@ -163,7 +160,6 @@ func GoLog(format string, args ...interface{}) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Determine level from message content
|
|
||||||
msgLower := strings.ToLower(message)
|
msgLower := strings.ToLower(message)
|
||||||
if strings.Contains(msgLower, "error") || strings.Contains(msgLower, "failed") {
|
if strings.Contains(msgLower, "error") || strings.Contains(msgLower, "failed") {
|
||||||
level = "ERROR"
|
level = "ERROR"
|
||||||
|
|||||||
+14
-251
@@ -3,7 +3,6 @@ package gobackend
|
|||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
|
||||||
"math"
|
"math"
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
@@ -21,9 +20,7 @@ const (
|
|||||||
durationToleranceSec = 10.0
|
durationToleranceSec = 10.0
|
||||||
)
|
)
|
||||||
|
|
||||||
// Lyrics provider names (used in settings and cascade ordering)
|
|
||||||
const (
|
const (
|
||||||
LyricsProviderSpotifyAPI = "spotify_api"
|
|
||||||
LyricsProviderLRCLIB = "lrclib"
|
LyricsProviderLRCLIB = "lrclib"
|
||||||
LyricsProviderNetease = "netease"
|
LyricsProviderNetease = "netease"
|
||||||
LyricsProviderMusixmatch = "musixmatch"
|
LyricsProviderMusixmatch = "musixmatch"
|
||||||
@@ -31,11 +28,8 @@ const (
|
|||||||
LyricsProviderQQMusic = "qqmusic"
|
LyricsProviderQQMusic = "qqmusic"
|
||||||
)
|
)
|
||||||
|
|
||||||
// DefaultLyricsProviders is the default cascade order for lyrics fetching.
|
|
||||||
// LRCLIB first (no proxy dependency), then the others.
|
|
||||||
var DefaultLyricsProviders = []string{
|
var DefaultLyricsProviders = []string{
|
||||||
LyricsProviderLRCLIB,
|
LyricsProviderLRCLIB,
|
||||||
LyricsProviderSpotifyAPI,
|
|
||||||
LyricsProviderMusixmatch,
|
LyricsProviderMusixmatch,
|
||||||
LyricsProviderNetease,
|
LyricsProviderNetease,
|
||||||
LyricsProviderAppleMusic,
|
LyricsProviderAppleMusic,
|
||||||
@@ -47,12 +41,6 @@ var (
|
|||||||
lyricsProviders []string // ordered list of enabled providers
|
lyricsProviders []string // ordered list of enabled providers
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
|
||||||
spotifyLyricsRateLimitMu sync.RWMutex
|
|
||||||
spotifyLyricsRateLimitedTil time.Time
|
|
||||||
)
|
|
||||||
|
|
||||||
// LyricsFetchOptions controls optional provider-specific enhancements.
|
|
||||||
type LyricsFetchOptions struct {
|
type LyricsFetchOptions struct {
|
||||||
IncludeTranslationNetease bool `json:"include_translation_netease"`
|
IncludeTranslationNetease bool `json:"include_translation_netease"`
|
||||||
IncludeRomanizationNetease bool `json:"include_romanization_netease"`
|
IncludeRomanizationNetease bool `json:"include_romanization_netease"`
|
||||||
@@ -72,8 +60,6 @@ var (
|
|||||||
lyricsFetchOptions = defaultLyricsFetchOptions
|
lyricsFetchOptions = defaultLyricsFetchOptions
|
||||||
)
|
)
|
||||||
|
|
||||||
// SetLyricsProviderOrder sets the ordered list of lyrics providers to try.
|
|
||||||
// Providers not in the list are disabled. An empty list resets to defaults.
|
|
||||||
func SetLyricsProviderOrder(providers []string) {
|
func SetLyricsProviderOrder(providers []string) {
|
||||||
lyricsProvidersMu.Lock()
|
lyricsProvidersMu.Lock()
|
||||||
defer lyricsProvidersMu.Unlock()
|
defer lyricsProvidersMu.Unlock()
|
||||||
@@ -84,7 +70,6 @@ func SetLyricsProviderOrder(providers []string) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
validNames := map[string]bool{
|
validNames := map[string]bool{
|
||||||
LyricsProviderSpotifyAPI: true,
|
|
||||||
LyricsProviderLRCLIB: true,
|
LyricsProviderLRCLIB: true,
|
||||||
LyricsProviderNetease: true,
|
LyricsProviderNetease: true,
|
||||||
LyricsProviderMusixmatch: true,
|
LyricsProviderMusixmatch: true,
|
||||||
@@ -119,7 +104,6 @@ func GetLyricsProviderOrder() []string {
|
|||||||
|
|
||||||
func GetAvailableLyricsProviders() []map[string]interface{} {
|
func GetAvailableLyricsProviders() []map[string]interface{} {
|
||||||
return []map[string]interface{}{
|
return []map[string]interface{}{
|
||||||
{"id": LyricsProviderSpotifyAPI, "name": "Spotify Lyrics API", "has_proxy_dependency": true, "description": "Spotify-sourced lyrics via Paxsenix"},
|
|
||||||
{"id": LyricsProviderLRCLIB, "name": "LRCLIB", "has_proxy_dependency": false, "description": "Open-source synced lyrics database"},
|
{"id": LyricsProviderLRCLIB, "name": "LRCLIB", "has_proxy_dependency": false, "description": "Open-source synced lyrics database"},
|
||||||
{"id": LyricsProviderNetease, "name": "Netease", "has_proxy_dependency": true, "description": "NetEase Cloud Music lyrics via Paxsenix"},
|
{"id": LyricsProviderNetease, "name": "Netease", "has_proxy_dependency": true, "description": "NetEase Cloud Music lyrics via Paxsenix"},
|
||||||
{"id": LyricsProviderMusixmatch, "name": "Musixmatch", "has_proxy_dependency": true, "description": "Musixmatch lyrics via Paxsenix"},
|
{"id": LyricsProviderMusixmatch, "name": "Musixmatch", "has_proxy_dependency": true, "description": "Musixmatch lyrics via Paxsenix"},
|
||||||
@@ -249,18 +233,6 @@ type LRCLibResponse struct {
|
|||||||
SyncedLyrics string `json:"syncedLyrics"`
|
SyncedLyrics string `json:"syncedLyrics"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type SpotifyLyricsLine struct {
|
|
||||||
TimeTag string `json:"timeTag"`
|
|
||||||
Words string `json:"words"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type SpotifyLyricsAPIResponse struct {
|
|
||||||
Error bool `json:"error"`
|
|
||||||
Message string `json:"message"`
|
|
||||||
SyncType string `json:"syncType"`
|
|
||||||
Lines []SpotifyLyricsLine `json:"lines"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type LyricsLine struct {
|
type LyricsLine struct {
|
||||||
StartTimeMs int64 `json:"startTimeMs"`
|
StartTimeMs int64 `json:"startTimeMs"`
|
||||||
Words string `json:"words"`
|
Words string `json:"words"`
|
||||||
@@ -368,214 +340,6 @@ func (c *LyricsClient) FetchLyricsFromLRCLibSearch(query string, durationSec flo
|
|||||||
return c.parseLRCLibResponse(&results[0]), nil
|
return c.parseLRCLibResponse(&results[0]), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func parseSpotifyLyricsTimeTagToMs(tag string) int64 {
|
|
||||||
raw := strings.TrimSpace(tag)
|
|
||||||
raw = strings.TrimPrefix(raw, "[")
|
|
||||||
raw = strings.TrimSuffix(raw, "]")
|
|
||||||
if raw == "" {
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
if ms, err := strconv.ParseInt(raw, 10, 64); err == nil {
|
|
||||||
return ms
|
|
||||||
}
|
|
||||||
|
|
||||||
re := regexp.MustCompile(`^(\d{1,2}):(\d{2})\.(\d{1,3})$`)
|
|
||||||
matches := re.FindStringSubmatch(raw)
|
|
||||||
if len(matches) != 4 {
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
minutes, _ := strconv.ParseInt(matches[1], 10, 64)
|
|
||||||
seconds, _ := strconv.ParseInt(matches[2], 10, 64)
|
|
||||||
fraction := matches[3]
|
|
||||||
fractionInt, _ := strconv.ParseInt(fraction, 10, 64)
|
|
||||||
if len(fraction) == 2 {
|
|
||||||
fractionInt *= 10
|
|
||||||
} else if len(fraction) == 1 {
|
|
||||||
fractionInt *= 100
|
|
||||||
}
|
|
||||||
return minutes*60*1000 + seconds*1000 + fractionInt
|
|
||||||
}
|
|
||||||
|
|
||||||
func getSpotifyLyricsRateLimitUntil() time.Time {
|
|
||||||
spotifyLyricsRateLimitMu.RLock()
|
|
||||||
defer spotifyLyricsRateLimitMu.RUnlock()
|
|
||||||
return spotifyLyricsRateLimitedTil
|
|
||||||
}
|
|
||||||
|
|
||||||
func setSpotifyLyricsRateLimitUntil(until time.Time) {
|
|
||||||
spotifyLyricsRateLimitMu.Lock()
|
|
||||||
spotifyLyricsRateLimitedTil = until
|
|
||||||
spotifyLyricsRateLimitMu.Unlock()
|
|
||||||
}
|
|
||||||
|
|
||||||
func parseSpotifyRetryAfter(retryAfter string, now time.Time) time.Time {
|
|
||||||
raw := strings.TrimSpace(retryAfter)
|
|
||||||
if raw == "" {
|
|
||||||
return now.Add(10 * time.Minute)
|
|
||||||
}
|
|
||||||
|
|
||||||
if sec, err := strconv.Atoi(raw); err == nil && sec > 0 {
|
|
||||||
return now.Add(time.Duration(sec) * time.Second)
|
|
||||||
}
|
|
||||||
|
|
||||||
if when, err := http.ParseTime(raw); err == nil && when.After(now) {
|
|
||||||
return when
|
|
||||||
}
|
|
||||||
|
|
||||||
return now.Add(10 * time.Minute)
|
|
||||||
}
|
|
||||||
|
|
||||||
func buildSpotifyLyricsResponse(lines []LyricsLine, syncType, plainLyrics string) (*LyricsResponse, error) {
|
|
||||||
if len(lines) == 0 {
|
|
||||||
return nil, fmt.Errorf("Spotify Lyrics API returned empty lines")
|
|
||||||
}
|
|
||||||
if syncType == "" {
|
|
||||||
if len(lines) > 0 && lines[0].StartTimeMs > 0 {
|
|
||||||
syncType = "LINE_SYNCED"
|
|
||||||
} else {
|
|
||||||
syncType = "UNSYNCED"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return &LyricsResponse{
|
|
||||||
Lines: lines,
|
|
||||||
SyncType: syncType,
|
|
||||||
Instrumental: false,
|
|
||||||
PlainLyrics: plainLyrics,
|
|
||||||
Provider: "Spotify Lyrics API",
|
|
||||||
Source: "Spotify Lyrics API",
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func plainLyricsFromTimedLines(lines []LyricsLine) string {
|
|
||||||
parts := make([]string, 0, len(lines))
|
|
||||||
for _, line := range lines {
|
|
||||||
words := strings.TrimSpace(line.Words)
|
|
||||||
if words == "" {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
parts = append(parts, words)
|
|
||||||
}
|
|
||||||
return strings.Join(parts, "\n")
|
|
||||||
}
|
|
||||||
|
|
||||||
func parseSpotifyLyricsResponseBody(body []byte) (*LyricsResponse, error) {
|
|
||||||
var lrcPayload string
|
|
||||||
if err := json.Unmarshal(body, &lrcPayload); err == nil {
|
|
||||||
trimmed := strings.TrimSpace(lrcPayload)
|
|
||||||
if trimmed == "" {
|
|
||||||
return nil, fmt.Errorf("Spotify Lyrics API returned empty payload")
|
|
||||||
}
|
|
||||||
|
|
||||||
lines := parseSyncedLyrics(trimmed)
|
|
||||||
if len(lines) > 0 {
|
|
||||||
return buildSpotifyLyricsResponse(lines, "LINE_SYNCED", plainLyricsFromTimedLines(lines))
|
|
||||||
}
|
|
||||||
|
|
||||||
plainLines := plainTextLyricsLines(trimmed)
|
|
||||||
return buildSpotifyLyricsResponse(plainLines, "UNSYNCED", trimmed)
|
|
||||||
}
|
|
||||||
|
|
||||||
var apiResp SpotifyLyricsAPIResponse
|
|
||||||
if err := json.Unmarshal(body, &apiResp); err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to parse Spotify Lyrics API response: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
if apiResp.Error {
|
|
||||||
msg := strings.TrimSpace(apiResp.Message)
|
|
||||||
if msg == "" {
|
|
||||||
msg = "Spotify Lyrics API returned error"
|
|
||||||
}
|
|
||||||
return nil, fmt.Errorf("%s", msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
lines := make([]LyricsLine, 0, len(apiResp.Lines))
|
|
||||||
for _, line := range apiResp.Lines {
|
|
||||||
words := strings.TrimSpace(line.Words)
|
|
||||||
if words == "" {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
startMs := parseSpotifyLyricsTimeTagToMs(line.TimeTag)
|
|
||||||
lines = append(lines, LyricsLine{
|
|
||||||
StartTimeMs: startMs,
|
|
||||||
Words: words,
|
|
||||||
EndTimeMs: 0,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
for i := 0; i < len(lines)-1; i++ {
|
|
||||||
nextStart := lines[i+1].StartTimeMs
|
|
||||||
if nextStart > lines[i].StartTimeMs {
|
|
||||||
lines[i].EndTimeMs = nextStart
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if len(lines) > 0 {
|
|
||||||
last := len(lines) - 1
|
|
||||||
if lines[last].EndTimeMs == 0 {
|
|
||||||
lines[last].EndTimeMs = lines[last].StartTimeMs + 5000
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return buildSpotifyLyricsResponse(lines, apiResp.SyncType, plainLyricsFromTimedLines(lines))
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *LyricsClient) FetchLyricsFromSpotifyAPI(spotifyID string) (*LyricsResponse, error) {
|
|
||||||
now := time.Now()
|
|
||||||
if limitedUntil := getSpotifyLyricsRateLimitUntil(); limitedUntil.After(now) {
|
|
||||||
waitFor := int(math.Ceil(limitedUntil.Sub(now).Seconds()))
|
|
||||||
return nil, fmt.Errorf(
|
|
||||||
"Spotify Lyrics API cooldown active (%ds remaining after previous 429)",
|
|
||||||
waitFor,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
spotifyID = strings.TrimSpace(spotifyID)
|
|
||||||
if spotifyID == "" {
|
|
||||||
return nil, fmt.Errorf("spotify ID is empty")
|
|
||||||
}
|
|
||||||
if parsed, err := parseSpotifyURI(spotifyID); err == nil && parsed.Type == "track" && parsed.ID != "" {
|
|
||||||
spotifyID = parsed.ID
|
|
||||||
}
|
|
||||||
|
|
||||||
apiURL := fmt.Sprintf("https://lyrics.paxsenix.org/spotify/lyrics?id=%s", url.QueryEscape(spotifyID))
|
|
||||||
req, err := http.NewRequest("GET", apiURL, nil)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to create request: %w", err)
|
|
||||||
}
|
|
||||||
req.Header.Set("User-Agent", getRandomUserAgent())
|
|
||||||
|
|
||||||
resp, err := c.httpClient.Do(req)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to fetch from Spotify Lyrics API: %w", err)
|
|
||||||
}
|
|
||||||
defer resp.Body.Close()
|
|
||||||
|
|
||||||
bodyBytes, err := io.ReadAll(resp.Body)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to read Spotify Lyrics API response: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
if resp.StatusCode != 200 {
|
|
||||||
if resp.StatusCode == http.StatusTooManyRequests {
|
|
||||||
retryUntil := parseSpotifyRetryAfter(resp.Header.Get("Retry-After"), now)
|
|
||||||
setSpotifyLyricsRateLimitUntil(retryUntil)
|
|
||||||
}
|
|
||||||
var payload map[string]interface{}
|
|
||||||
if err := json.Unmarshal(bodyBytes, &payload); err == nil {
|
|
||||||
if msg, ok := payload["message"].(string); ok && strings.TrimSpace(msg) != "" {
|
|
||||||
return nil, fmt.Errorf("Spotify Lyrics API returned status %d: %s", resp.StatusCode, strings.TrimSpace(msg))
|
|
||||||
}
|
|
||||||
if msg, ok := payload["error"].(string); ok && strings.TrimSpace(msg) != "" {
|
|
||||||
return nil, fmt.Errorf("Spotify Lyrics API returned status %d: %s", resp.StatusCode, strings.TrimSpace(msg))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil, fmt.Errorf("Spotify Lyrics API returned status %d", resp.StatusCode)
|
|
||||||
}
|
|
||||||
|
|
||||||
return parseSpotifyLyricsResponseBody(bodyBytes)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *LyricsClient) findBestMatch(results []LRCLibResponse, targetDurationSec float64) *LRCLibResponse {
|
func (c *LyricsClient) findBestMatch(results []LRCLibResponse, targetDurationSec float64) *LRCLibResponse {
|
||||||
var bestSynced *LRCLibResponse
|
var bestSynced *LRCLibResponse
|
||||||
var bestPlain *LRCLibResponse
|
var bestPlain *LRCLibResponse
|
||||||
@@ -600,6 +364,18 @@ func (c *LyricsClient) findBestMatch(results []LRCLibResponse, targetDurationSec
|
|||||||
return bestPlain
|
return bestPlain
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func plainLyricsFromTimedLines(lines []LyricsLine) string {
|
||||||
|
parts := make([]string, 0, len(lines))
|
||||||
|
for _, line := range lines {
|
||||||
|
words := strings.TrimSpace(line.Words)
|
||||||
|
if words == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
parts = append(parts, words)
|
||||||
|
}
|
||||||
|
return strings.Join(parts, "\n")
|
||||||
|
}
|
||||||
|
|
||||||
func (c *LyricsClient) durationMatches(lrcDuration, targetDuration float64) bool {
|
func (c *LyricsClient) durationMatches(lrcDuration, targetDuration float64) bool {
|
||||||
diff := math.Abs(lrcDuration - targetDuration)
|
diff := math.Abs(lrcDuration - targetDuration)
|
||||||
return diff <= durationToleranceSec
|
return diff <= durationToleranceSec
|
||||||
@@ -609,8 +385,8 @@ func (c *LyricsClient) FetchLyricsAllSources(spotifyID, trackName, artistName st
|
|||||||
primaryArtist := normalizeArtistName(artistName)
|
primaryArtist := normalizeArtistName(artistName)
|
||||||
fetchOptions := GetLyricsFetchOptions()
|
fetchOptions := GetLyricsFetchOptions()
|
||||||
|
|
||||||
extManager := GetExtensionManager()
|
extManager := getExtensionManager()
|
||||||
var extensionProviders []*ExtensionProviderWrapper
|
var extensionProviders []*extensionProviderWrapper
|
||||||
if extManager != nil {
|
if extManager != nil {
|
||||||
extensionProviders = extManager.GetLyricsProviders()
|
extensionProviders = extManager.GetLyricsProviders()
|
||||||
}
|
}
|
||||||
@@ -669,9 +445,6 @@ func (c *LyricsClient) FetchLyricsAllSources(spotifyID, trackName, artistName st
|
|||||||
var err error
|
var err error
|
||||||
|
|
||||||
switch providerName {
|
switch providerName {
|
||||||
case LyricsProviderSpotifyAPI:
|
|
||||||
lyrics, err = c.FetchLyricsFromSpotifyAPI(spotifyID)
|
|
||||||
|
|
||||||
case LyricsProviderLRCLIB:
|
case LyricsProviderLRCLIB:
|
||||||
lyrics, err = c.tryLRCLIB(primaryArtist, artistName, trackName, simplifiedTrack, durationSec)
|
lyrics, err = c.tryLRCLIB(primaryArtist, artistName, trackName, simplifiedTrack, durationSec)
|
||||||
|
|
||||||
@@ -753,19 +526,16 @@ func (c *LyricsClient) FetchLyricsAllSources(spotifyID, trackName, artistName st
|
|||||||
return nil, fmt.Errorf("lyrics not found from any source")
|
return nil, fmt.Errorf("lyrics not found from any source")
|
||||||
}
|
}
|
||||||
|
|
||||||
// tryLRCLIB attempts all LRCLIB search strategies (exact match, simplified, search).
|
|
||||||
func (c *LyricsClient) tryLRCLIB(primaryArtist, artistName, trackName, simplifiedTrack string, durationSec float64) (*LyricsResponse, error) {
|
func (c *LyricsClient) tryLRCLIB(primaryArtist, artistName, trackName, simplifiedTrack string, durationSec float64) (*LyricsResponse, error) {
|
||||||
var lyrics *LyricsResponse
|
var lyrics *LyricsResponse
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
// 1. Exact match with primary artist
|
|
||||||
lyrics, err = c.FetchLyricsWithMetadata(primaryArtist, trackName)
|
lyrics, err = c.FetchLyricsWithMetadata(primaryArtist, trackName)
|
||||||
if err == nil && lyrics != nil && (len(lyrics.Lines) > 0 || lyrics.Instrumental) {
|
if err == nil && lyrics != nil && (len(lyrics.Lines) > 0 || lyrics.Instrumental) {
|
||||||
lyrics.Source = "LRCLIB"
|
lyrics.Source = "LRCLIB"
|
||||||
return lyrics, nil
|
return lyrics, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// 2. Exact match with full artist name
|
|
||||||
if primaryArtist != artistName {
|
if primaryArtist != artistName {
|
||||||
lyrics, err = c.FetchLyricsWithMetadata(artistName, trackName)
|
lyrics, err = c.FetchLyricsWithMetadata(artistName, trackName)
|
||||||
if err == nil && lyrics != nil && (len(lyrics.Lines) > 0 || lyrics.Instrumental) {
|
if err == nil && lyrics != nil && (len(lyrics.Lines) > 0 || lyrics.Instrumental) {
|
||||||
@@ -774,7 +544,6 @@ func (c *LyricsClient) tryLRCLIB(primaryArtist, artistName, trackName, simplifie
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3. Simplified track name
|
|
||||||
if simplifiedTrack != trackName {
|
if simplifiedTrack != trackName {
|
||||||
lyrics, err = c.FetchLyricsWithMetadata(primaryArtist, simplifiedTrack)
|
lyrics, err = c.FetchLyricsWithMetadata(primaryArtist, simplifiedTrack)
|
||||||
if err == nil && lyrics != nil && (len(lyrics.Lines) > 0 || lyrics.Instrumental) {
|
if err == nil && lyrics != nil && (len(lyrics.Lines) > 0 || lyrics.Instrumental) {
|
||||||
@@ -783,7 +552,6 @@ func (c *LyricsClient) tryLRCLIB(primaryArtist, artistName, trackName, simplifie
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 4. Search by query
|
|
||||||
query := primaryArtist + " " + trackName
|
query := primaryArtist + " " + trackName
|
||||||
lyrics, err = c.FetchLyricsFromLRCLibSearch(query, durationSec)
|
lyrics, err = c.FetchLyricsFromLRCLibSearch(query, durationSec)
|
||||||
if err == nil && lyrics != nil && (len(lyrics.Lines) > 0 || lyrics.Instrumental) {
|
if err == nil && lyrics != nil && (len(lyrics.Lines) > 0 || lyrics.Instrumental) {
|
||||||
@@ -791,7 +559,6 @@ func (c *LyricsClient) tryLRCLIB(primaryArtist, artistName, trackName, simplifie
|
|||||||
return lyrics, nil
|
return lyrics, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// 5. Search with simplified track name
|
|
||||||
if simplifiedTrack != trackName {
|
if simplifiedTrack != trackName {
|
||||||
query = primaryArtist + " " + simplifiedTrack
|
query = primaryArtist + " " + simplifiedTrack
|
||||||
lyrics, err = c.FetchLyricsFromLRCLibSearch(query, durationSec)
|
lyrics, err = c.FetchLyricsFromLRCLibSearch(query, durationSec)
|
||||||
@@ -909,8 +676,6 @@ func lyricsHasUsableText(lyrics *LyricsResponse) bool {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
// detectLyricsErrorPayload extracts human-readable error messages from
|
|
||||||
// JSON payloads returned by lyrics proxies when no lyric is available.
|
|
||||||
func detectLyricsErrorPayload(raw string) (string, bool) {
|
func detectLyricsErrorPayload(raw string) (string, bool) {
|
||||||
trimmed := strings.TrimSpace(raw)
|
trimmed := strings.TrimSpace(raw)
|
||||||
if trimmed == "" || !strings.HasPrefix(trimmed, "{") {
|
if trimmed == "" || !strings.HasPrefix(trimmed, "{") {
|
||||||
@@ -1035,8 +800,6 @@ func simplifyTrackName(name string) string {
|
|||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add a loose fallback form for provider queries where punctuation
|
|
||||||
// and separators differ (e.g. "/" vs "_" vs spaces).
|
|
||||||
if loose := normalizeLooseTitle(result); loose != "" {
|
if loose := normalizeLooseTitle(result); loose != "" {
|
||||||
return loose
|
return loose
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,8 +11,6 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
// AppleMusicClient fetches lyrics from Apple Music.
|
|
||||||
// Uses Paxsenix endpoints for search and lyrics.
|
|
||||||
type AppleMusicClient struct {
|
type AppleMusicClient struct {
|
||||||
httpClient *http.Client
|
httpClient *http.Client
|
||||||
}
|
}
|
||||||
@@ -25,7 +23,6 @@ type appleMusicSearchResult struct {
|
|||||||
Duration int `json:"duration"`
|
Duration int `json:"duration"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// PaxResponse represents the lyrics proxy response for word-by-word / line lyrics
|
|
||||||
type paxResponse struct {
|
type paxResponse struct {
|
||||||
Type string `json:"type"` // "Syllable" or "Line"
|
Type string `json:"type"` // "Syllable" or "Line"
|
||||||
Content []paxLyrics `json:"content"` // List of lyric lines
|
Content []paxLyrics `json:"content"` // List of lyric lines
|
||||||
@@ -103,7 +100,6 @@ func selectBestAppleMusicSearchResult(results []appleMusicSearchResult, trackNam
|
|||||||
return &results[bestIndex]
|
return &results[bestIndex]
|
||||||
}
|
}
|
||||||
|
|
||||||
// SearchSong searches for a song on Apple Music and returns its ID.
|
|
||||||
func (c *AppleMusicClient) SearchSong(trackName, artistName string, durationSec float64) (string, error) {
|
func (c *AppleMusicClient) SearchSong(trackName, artistName string, durationSec float64) (string, error) {
|
||||||
query := trackName + " " + artistName
|
query := trackName + " " + artistName
|
||||||
if strings.TrimSpace(query) == "" {
|
if strings.TrimSpace(query) == "" {
|
||||||
@@ -144,7 +140,6 @@ func (c *AppleMusicClient) SearchSong(trackName, artistName string, durationSec
|
|||||||
return strings.TrimSpace(best.ID), nil
|
return strings.TrimSpace(best.ID), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// FetchLyricsByID fetches lyrics from the paxsenix proxy using Apple Music song ID.
|
|
||||||
func (c *AppleMusicClient) FetchLyricsByID(songID string) (string, error) {
|
func (c *AppleMusicClient) FetchLyricsByID(songID string) (string, error) {
|
||||||
lyricsURL := fmt.Sprintf("https://lyrics.paxsenix.org/apple-music/lyrics?id=%s", songID)
|
lyricsURL := fmt.Sprintf("https://lyrics.paxsenix.org/apple-music/lyrics?id=%s", songID)
|
||||||
|
|
||||||
@@ -252,7 +247,6 @@ func formatPaxContent(lyricsType string, content []paxLyrics, multiPersonWordByW
|
|||||||
return strings.TrimSpace(sb.String())
|
return strings.TrimSpace(sb.String())
|
||||||
}
|
}
|
||||||
|
|
||||||
// FetchLyrics searches Apple Music and returns parsed LyricsResponse.
|
|
||||||
func (c *AppleMusicClient) FetchLyrics(
|
func (c *AppleMusicClient) FetchLyrics(
|
||||||
trackName,
|
trackName,
|
||||||
artistName string,
|
artistName string,
|
||||||
@@ -272,10 +266,8 @@ func (c *AppleMusicClient) FetchLyrics(
|
|||||||
return nil, fmt.Errorf("apple music proxy returned non-lyric payload: %s", errMsg)
|
return nil, fmt.Errorf("apple music proxy returned non-lyric payload: %s", errMsg)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try to parse as pax format (word-by-word or line)
|
|
||||||
lrcText, err := formatPaxLyricsToLRC(rawLyrics, multiPersonWordByWord)
|
lrcText, err := formatPaxLyricsToLRC(rawLyrics, multiPersonWordByWord)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// If pax parsing fails, try to parse as direct LRC text
|
|
||||||
lrcText = rawLyrics
|
lrcText = rawLyrics
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -289,7 +281,6 @@ func (c *AppleMusicClient) FetchLyrics(
|
|||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fall back to plain text if no timestamps found
|
|
||||||
resultLines := plainTextLyricsLines(lrcText)
|
resultLines := plainTextLyricsLines(lrcText)
|
||||||
|
|
||||||
if len(resultLines) > 0 {
|
if len(resultLines) > 0 {
|
||||||
|
|||||||
@@ -11,8 +11,6 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
// MusixmatchClient fetches lyrics from Musixmatch via a proxy server.
|
|
||||||
// The proxy handles Musixmatch authentication internally.
|
|
||||||
type MusixmatchClient struct {
|
type MusixmatchClient struct {
|
||||||
httpClient *http.Client
|
httpClient *http.Client
|
||||||
baseURL string
|
baseURL string
|
||||||
@@ -114,7 +112,6 @@ func (c *MusixmatchClient) fetchLyricsPayload(trackName, artistName string, dura
|
|||||||
return "", fmt.Errorf("failed to decode musixmatch response")
|
return "", fmt.Errorf("failed to decode musixmatch response")
|
||||||
}
|
}
|
||||||
|
|
||||||
// FetchLyricsInLanguage retrieves lyrics from Musixmatch for a specific language code.
|
|
||||||
func (c *MusixmatchClient) FetchLyricsInLanguage(trackName, artistName string, durationSec float64, language string) (*LyricsResponse, error) {
|
func (c *MusixmatchClient) FetchLyricsInLanguage(trackName, artistName string, durationSec float64, language string) (*LyricsResponse, error) {
|
||||||
lang := strings.ToLower(strings.TrimSpace(language))
|
lang := strings.ToLower(strings.TrimSpace(language))
|
||||||
if lang == "" {
|
if lang == "" {
|
||||||
@@ -151,7 +148,6 @@ func (c *MusixmatchClient) FetchLyricsInLanguage(trackName, artistName string, d
|
|||||||
return nil, fmt.Errorf("no lyrics found on musixmatch for language %s", lang)
|
return nil, fmt.Errorf("no lyrics found on musixmatch for language %s", lang)
|
||||||
}
|
}
|
||||||
|
|
||||||
// FetchLyrics searches Musixmatch and returns parsed LyricsResponse.
|
|
||||||
func (c *MusixmatchClient) FetchLyrics(trackName, artistName string, durationSec float64, preferredLanguage string) (*LyricsResponse, error) {
|
func (c *MusixmatchClient) FetchLyrics(trackName, artistName string, durationSec float64, preferredLanguage string) (*LyricsResponse, error) {
|
||||||
if preferred := strings.ToLower(strings.TrimSpace(preferredLanguage)); preferred != "" {
|
if preferred := strings.ToLower(strings.TrimSpace(preferredLanguage)); preferred != "" {
|
||||||
localized, localizedErr := c.FetchLyricsInLanguage(trackName, artistName, durationSec, preferred)
|
localized, localizedErr := c.FetchLyricsInLanguage(trackName, artistName, durationSec, preferred)
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
// NeteaseClient fetches lyrics through Paxsenix's NetEase endpoints.
|
|
||||||
type NeteaseClient struct {
|
type NeteaseClient struct {
|
||||||
httpClient *http.Client
|
httpClient *http.Client
|
||||||
}
|
}
|
||||||
@@ -51,7 +50,6 @@ func NewNeteaseClient() *NeteaseClient {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// SearchSong searches for a song on Netease and returns the song ID.
|
|
||||||
func (c *NeteaseClient) SearchSong(trackName, artistName string) (int64, error) {
|
func (c *NeteaseClient) SearchSong(trackName, artistName string) (int64, error) {
|
||||||
query := trackName + " " + artistName
|
query := trackName + " " + artistName
|
||||||
if strings.TrimSpace(query) == "" {
|
if strings.TrimSpace(query) == "" {
|
||||||
@@ -96,7 +94,6 @@ func (c *NeteaseClient) SearchSong(trackName, artistName string) (int64, error)
|
|||||||
return searchResp.Result.Songs[0].ID, nil
|
return searchResp.Result.Songs[0].ID, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// FetchLyricsByID fetches synced lyrics for a given Netease song ID.
|
|
||||||
func (c *NeteaseClient) FetchLyricsByID(songID int64, includeTranslation, includeRomanization bool) (string, error) {
|
func (c *NeteaseClient) FetchLyricsByID(songID int64, includeTranslation, includeRomanization bool) (string, error) {
|
||||||
lyricsURL := "https://lyrics.paxsenix.org/netease/lyrics"
|
lyricsURL := "https://lyrics.paxsenix.org/netease/lyrics"
|
||||||
params := url.Values{}
|
params := url.Values{}
|
||||||
@@ -146,7 +143,6 @@ func (c *NeteaseClient) FetchLyricsByID(songID int64, includeTranslation, includ
|
|||||||
return lyric, nil
|
return lyric, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// FetchLyrics searches for a track and returns parsed LyricsResponse.
|
|
||||||
func (c *NeteaseClient) FetchLyrics(
|
func (c *NeteaseClient) FetchLyrics(
|
||||||
trackName,
|
trackName,
|
||||||
artistName string,
|
artistName string,
|
||||||
@@ -166,7 +162,6 @@ func (c *NeteaseClient) FetchLyrics(
|
|||||||
|
|
||||||
lines := parseSyncedLyrics(lrcText)
|
lines := parseSyncedLyrics(lrcText)
|
||||||
if len(lines) == 0 {
|
if len(lines) == 0 {
|
||||||
// May be plain text lyrics without timestamps
|
|
||||||
plainLines := strings.Split(lrcText, "\n")
|
plainLines := strings.Split(lrcText, "\n")
|
||||||
for _, line := range plainLines {
|
for _, line := range plainLines {
|
||||||
trimmed := strings.TrimSpace(line)
|
trimmed := strings.TrimSpace(line)
|
||||||
|
|||||||
@@ -10,8 +10,6 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
// QQMusicClient fetches lyrics from QQ Music.
|
|
||||||
// Uses Paxsenix metadata lookup for lyrics.
|
|
||||||
type QQMusicClient struct {
|
type QQMusicClient struct {
|
||||||
httpClient *http.Client
|
httpClient *http.Client
|
||||||
}
|
}
|
||||||
@@ -34,7 +32,6 @@ func NewQQMusicClient() *QQMusicClient {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// fetchLyricsByMetadata asks Paxsenix to resolve and return QQ lyrics using track metadata.
|
|
||||||
func (c *QQMusicClient) fetchLyricsByMetadata(trackName, artistName string, durationSec float64) (string, error) {
|
func (c *QQMusicClient) fetchLyricsByMetadata(trackName, artistName string, durationSec float64) (string, error) {
|
||||||
payload := qqLyricsMetadataRequest{
|
payload := qqLyricsMetadataRequest{
|
||||||
Artist: []string{artistName},
|
Artist: []string{artistName},
|
||||||
@@ -93,7 +90,6 @@ func formatQQLyricsMetadataToLRC(rawJSON string, multiPersonWordByWord bool) (st
|
|||||||
return formatPaxContent("Syllable", response.Lyrics, multiPersonWordByWord), nil
|
return formatPaxContent("Syllable", response.Lyrics, multiPersonWordByWord), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// FetchLyrics searches QQ Music and returns parsed LyricsResponse.
|
|
||||||
func (c *QQMusicClient) FetchLyrics(
|
func (c *QQMusicClient) FetchLyrics(
|
||||||
trackName,
|
trackName,
|
||||||
artistName string,
|
artistName string,
|
||||||
|
|||||||
+375
-116
@@ -110,6 +110,7 @@ type Metadata struct {
|
|||||||
TrackNumber int
|
TrackNumber int
|
||||||
TotalTracks int
|
TotalTracks int
|
||||||
DiscNumber int
|
DiscNumber int
|
||||||
|
TotalDiscs int
|
||||||
ISRC string
|
ISRC string
|
||||||
Description string
|
Description string
|
||||||
Lyrics string
|
Lyrics string
|
||||||
@@ -118,6 +119,12 @@ type Metadata struct {
|
|||||||
Copyright string
|
Copyright string
|
||||||
Composer string
|
Composer string
|
||||||
Comment string
|
Comment string
|
||||||
|
|
||||||
|
// ReplayGain fields (stored as Vorbis Comments in FLAC)
|
||||||
|
ReplayGainTrackGain string // e.g. "-6.50 dB"
|
||||||
|
ReplayGainTrackPeak string // e.g. "0.988831"
|
||||||
|
ReplayGainAlbumGain string // e.g. "-7.20 dB"
|
||||||
|
ReplayGainAlbumPeak string // e.g. "1.000000"
|
||||||
}
|
}
|
||||||
|
|
||||||
func EmbedMetadata(filePath string, metadata Metadata, coverPath string) error {
|
func EmbedMetadata(filePath string, metadata Metadata, coverPath string) error {
|
||||||
@@ -144,61 +151,7 @@ func EmbedMetadata(filePath string, metadata Metadata, coverPath string) error {
|
|||||||
cmt = flacvorbis.New()
|
cmt = flacvorbis.New()
|
||||||
}
|
}
|
||||||
|
|
||||||
setComment(cmt, "TITLE", metadata.Title)
|
writeVorbisMetadata(cmt, metadata)
|
||||||
setArtistComments(cmt, "ARTIST", metadata.Artist, metadata.ArtistTagMode)
|
|
||||||
setComment(cmt, "ALBUM", metadata.Album)
|
|
||||||
setArtistComments(
|
|
||||||
cmt,
|
|
||||||
"ALBUMARTIST",
|
|
||||||
metadata.AlbumArtist,
|
|
||||||
metadata.ArtistTagMode,
|
|
||||||
)
|
|
||||||
setComment(cmt, "DATE", metadata.Date)
|
|
||||||
|
|
||||||
if metadata.TrackNumber > 0 {
|
|
||||||
if metadata.TotalTracks > 0 {
|
|
||||||
setComment(cmt, "TRACKNUMBER", fmt.Sprintf("%d/%d", metadata.TrackNumber, metadata.TotalTracks))
|
|
||||||
} else {
|
|
||||||
setComment(cmt, "TRACKNUMBER", strconv.Itoa(metadata.TrackNumber))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.DiscNumber > 0 {
|
|
||||||
setComment(cmt, "DISCNUMBER", strconv.Itoa(metadata.DiscNumber))
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.ISRC != "" {
|
|
||||||
setComment(cmt, "ISRC", metadata.ISRC)
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.Description != "" {
|
|
||||||
setComment(cmt, "DESCRIPTION", metadata.Description)
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.Lyrics != "" {
|
|
||||||
setComment(cmt, "LYRICS", metadata.Lyrics)
|
|
||||||
setComment(cmt, "UNSYNCEDLYRICS", metadata.Lyrics)
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.Genre != "" {
|
|
||||||
setComment(cmt, "GENRE", metadata.Genre)
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.Label != "" {
|
|
||||||
setComment(cmt, "ORGANIZATION", metadata.Label)
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.Copyright != "" {
|
|
||||||
setComment(cmt, "COPYRIGHT", metadata.Copyright)
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.Composer != "" {
|
|
||||||
setComment(cmt, "COMPOSER", metadata.Composer)
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.Comment != "" {
|
|
||||||
setComment(cmt, "COMMENT", metadata.Comment)
|
|
||||||
}
|
|
||||||
|
|
||||||
cmtBlock := cmt.Marshal()
|
cmtBlock := cmt.Marshal()
|
||||||
if cmtIdx >= 0 {
|
if cmtIdx >= 0 {
|
||||||
@@ -258,61 +211,7 @@ func EmbedMetadataWithCoverData(filePath string, metadata Metadata, coverData []
|
|||||||
cmt = flacvorbis.New()
|
cmt = flacvorbis.New()
|
||||||
}
|
}
|
||||||
|
|
||||||
setComment(cmt, "TITLE", metadata.Title)
|
writeVorbisMetadata(cmt, metadata)
|
||||||
setArtistComments(cmt, "ARTIST", metadata.Artist, metadata.ArtistTagMode)
|
|
||||||
setComment(cmt, "ALBUM", metadata.Album)
|
|
||||||
setArtistComments(
|
|
||||||
cmt,
|
|
||||||
"ALBUMARTIST",
|
|
||||||
metadata.AlbumArtist,
|
|
||||||
metadata.ArtistTagMode,
|
|
||||||
)
|
|
||||||
setComment(cmt, "DATE", metadata.Date)
|
|
||||||
|
|
||||||
if metadata.TrackNumber > 0 {
|
|
||||||
if metadata.TotalTracks > 0 {
|
|
||||||
setComment(cmt, "TRACKNUMBER", fmt.Sprintf("%d/%d", metadata.TrackNumber, metadata.TotalTracks))
|
|
||||||
} else {
|
|
||||||
setComment(cmt, "TRACKNUMBER", strconv.Itoa(metadata.TrackNumber))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.DiscNumber > 0 {
|
|
||||||
setComment(cmt, "DISCNUMBER", strconv.Itoa(metadata.DiscNumber))
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.ISRC != "" {
|
|
||||||
setComment(cmt, "ISRC", metadata.ISRC)
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.Description != "" {
|
|
||||||
setComment(cmt, "DESCRIPTION", metadata.Description)
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.Lyrics != "" {
|
|
||||||
setComment(cmt, "LYRICS", metadata.Lyrics)
|
|
||||||
setComment(cmt, "UNSYNCEDLYRICS", metadata.Lyrics)
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.Genre != "" {
|
|
||||||
setComment(cmt, "GENRE", metadata.Genre)
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.Label != "" {
|
|
||||||
setComment(cmt, "ORGANIZATION", metadata.Label)
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.Copyright != "" {
|
|
||||||
setComment(cmt, "COPYRIGHT", metadata.Copyright)
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.Composer != "" {
|
|
||||||
setComment(cmt, "COMPOSER", metadata.Composer)
|
|
||||||
}
|
|
||||||
|
|
||||||
if metadata.Comment != "" {
|
|
||||||
setComment(cmt, "COMMENT", metadata.Comment)
|
|
||||||
}
|
|
||||||
|
|
||||||
cmtBlock := cmt.Marshal()
|
cmtBlock := cmt.Marshal()
|
||||||
if cmtIdx >= 0 {
|
if cmtIdx >= 0 {
|
||||||
@@ -358,6 +257,12 @@ func ReadMetadata(filePath string) (*Metadata, error) {
|
|||||||
metadata.Artist = getJoinedComment(cmt, "ARTIST")
|
metadata.Artist = getJoinedComment(cmt, "ARTIST")
|
||||||
metadata.Album = getComment(cmt, "ALBUM")
|
metadata.Album = getComment(cmt, "ALBUM")
|
||||||
metadata.AlbumArtist = getJoinedComment(cmt, "ALBUMARTIST")
|
metadata.AlbumArtist = getJoinedComment(cmt, "ALBUMARTIST")
|
||||||
|
if metadata.AlbumArtist == "" {
|
||||||
|
metadata.AlbumArtist = getJoinedComment(cmt, "ALBUM ARTIST")
|
||||||
|
}
|
||||||
|
if metadata.AlbumArtist == "" {
|
||||||
|
metadata.AlbumArtist = getJoinedComment(cmt, "ALBUM_ARTIST")
|
||||||
|
}
|
||||||
metadata.Date = getComment(cmt, "DATE")
|
metadata.Date = getComment(cmt, "DATE")
|
||||||
metadata.ISRC = getComment(cmt, "ISRC")
|
metadata.ISRC = getComment(cmt, "ISRC")
|
||||||
metadata.Description = getComment(cmt, "DESCRIPTION")
|
metadata.Description = getComment(cmt, "DESCRIPTION")
|
||||||
@@ -369,23 +274,23 @@ func ReadMetadata(filePath string) (*Metadata, error) {
|
|||||||
|
|
||||||
trackNum := getComment(cmt, "TRACKNUMBER")
|
trackNum := getComment(cmt, "TRACKNUMBER")
|
||||||
if trackNum != "" {
|
if trackNum != "" {
|
||||||
fmt.Sscanf(trackNum, "%d", &metadata.TrackNumber)
|
metadata.TrackNumber, metadata.TotalTracks = parseIndexPair(trackNum)
|
||||||
}
|
}
|
||||||
if metadata.TrackNumber == 0 {
|
if metadata.TrackNumber == 0 {
|
||||||
trackNum = getComment(cmt, "TRACK")
|
trackNum = getComment(cmt, "TRACK")
|
||||||
if trackNum != "" {
|
if trackNum != "" {
|
||||||
fmt.Sscanf(trackNum, "%d", &metadata.TrackNumber)
|
metadata.TrackNumber, metadata.TotalTracks = parseIndexPair(trackNum)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
discNum := getComment(cmt, "DISCNUMBER")
|
discNum := getComment(cmt, "DISCNUMBER")
|
||||||
if discNum != "" {
|
if discNum != "" {
|
||||||
fmt.Sscanf(discNum, "%d", &metadata.DiscNumber)
|
metadata.DiscNumber, metadata.TotalDiscs = parseIndexPair(discNum)
|
||||||
}
|
}
|
||||||
if metadata.DiscNumber == 0 {
|
if metadata.DiscNumber == 0 {
|
||||||
discNum = getComment(cmt, "DISC")
|
discNum = getComment(cmt, "DISC")
|
||||||
if discNum != "" {
|
if discNum != "" {
|
||||||
fmt.Sscanf(discNum, "%d", &metadata.DiscNumber)
|
metadata.DiscNumber, metadata.TotalDiscs = parseIndexPair(discNum)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -395,10 +300,21 @@ func ReadMetadata(filePath string) (*Metadata, error) {
|
|||||||
|
|
||||||
metadata.Genre = getComment(cmt, "GENRE")
|
metadata.Genre = getComment(cmt, "GENRE")
|
||||||
metadata.Label = getComment(cmt, "ORGANIZATION")
|
metadata.Label = getComment(cmt, "ORGANIZATION")
|
||||||
|
if metadata.Label == "" {
|
||||||
|
metadata.Label = getComment(cmt, "LABEL")
|
||||||
|
}
|
||||||
|
if metadata.Label == "" {
|
||||||
|
metadata.Label = getComment(cmt, "PUBLISHER")
|
||||||
|
}
|
||||||
metadata.Copyright = getComment(cmt, "COPYRIGHT")
|
metadata.Copyright = getComment(cmt, "COPYRIGHT")
|
||||||
metadata.Composer = getComment(cmt, "COMPOSER")
|
metadata.Composer = getComment(cmt, "COMPOSER")
|
||||||
metadata.Comment = getComment(cmt, "COMMENT")
|
metadata.Comment = getComment(cmt, "COMMENT")
|
||||||
|
|
||||||
|
metadata.ReplayGainTrackGain = getComment(cmt, "REPLAYGAIN_TRACK_GAIN")
|
||||||
|
metadata.ReplayGainTrackPeak = getComment(cmt, "REPLAYGAIN_TRACK_PEAK")
|
||||||
|
metadata.ReplayGainAlbumGain = getComment(cmt, "REPLAYGAIN_ALBUM_GAIN")
|
||||||
|
metadata.ReplayGainAlbumPeak = getComment(cmt, "REPLAYGAIN_ALBUM_PEAK")
|
||||||
|
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -406,6 +322,222 @@ func ReadMetadata(filePath string) (*Metadata, error) {
|
|||||||
return metadata, nil
|
return metadata, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// EditFlacFields opens a FLAC file and updates only the Vorbis Comment keys
|
||||||
|
// that are explicitly present in the fields map. Keys present with a non-empty
|
||||||
|
// value are set; keys present with an empty value are removed (cleared). Keys
|
||||||
|
// absent from the map are left untouched. This is the correct function for
|
||||||
|
// partial edits (e.g. writing only ReplayGain tags) and full editor saves alike.
|
||||||
|
func EditFlacFields(filePath string, fields map[string]string) error {
|
||||||
|
f, err := flac.ParseFile(filePath)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to parse FLAC file: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var cmtIdx int = -1
|
||||||
|
var cmt *flacvorbis.MetaDataBlockVorbisComment
|
||||||
|
|
||||||
|
for idx, meta := range f.Meta {
|
||||||
|
if meta.Type == flac.VorbisComment {
|
||||||
|
cmtIdx = idx
|
||||||
|
cmt, err = flacvorbis.ParseFromMetaDataBlock(*meta)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to parse vorbis comment: %w", err)
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if cmt == nil {
|
||||||
|
cmt = flacvorbis.New()
|
||||||
|
}
|
||||||
|
|
||||||
|
artistMode := fields["artist_tag_mode"]
|
||||||
|
|
||||||
|
// Mapping from fields-map key → one or more Vorbis Comment keys.
|
||||||
|
// Each entry is handled with set-or-clear semantics.
|
||||||
|
simpleKeys := map[string]string{
|
||||||
|
"title": "TITLE",
|
||||||
|
"album": "ALBUM",
|
||||||
|
"date": "DATE",
|
||||||
|
"isrc": "ISRC",
|
||||||
|
"genre": "GENRE",
|
||||||
|
"label": "ORGANIZATION",
|
||||||
|
"copyright": "COPYRIGHT",
|
||||||
|
"composer": "COMPOSER",
|
||||||
|
"comment": "COMMENT",
|
||||||
|
"replaygain_track_gain": "REPLAYGAIN_TRACK_GAIN",
|
||||||
|
"replaygain_track_peak": "REPLAYGAIN_TRACK_PEAK",
|
||||||
|
"replaygain_album_gain": "REPLAYGAIN_ALBUM_GAIN",
|
||||||
|
"replaygain_album_peak": "REPLAYGAIN_ALBUM_PEAK",
|
||||||
|
}
|
||||||
|
|
||||||
|
for fieldKey, vorbisKey := range simpleKeys {
|
||||||
|
if v, ok := fields[fieldKey]; ok {
|
||||||
|
setOrClearComment(cmt, vorbisKey, v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove known aliases for fields that were just written/cleared, so that
|
||||||
|
// tags from other taggers (e.g. LABEL, PUBLISHER, ALBUM ARTIST) don't
|
||||||
|
// conflict with the canonical keys we use.
|
||||||
|
aliasCleanup := map[string][]string{
|
||||||
|
"label": {"LABEL", "PUBLISHER"}, // canonical: ORGANIZATION
|
||||||
|
"date": {"YEAR"}, // canonical: DATE
|
||||||
|
"genre": {}, // no common aliases
|
||||||
|
"copyright": {},
|
||||||
|
}
|
||||||
|
for fieldKey, aliases := range aliasCleanup {
|
||||||
|
if _, ok := fields[fieldKey]; ok {
|
||||||
|
for _, alias := range aliases {
|
||||||
|
removeCommentKey(cmt, alias)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Artist fields: use split-artist logic when mode is set.
|
||||||
|
if v, ok := fields["artist"]; ok {
|
||||||
|
setOrClearArtistComments(cmt, "ARTIST", v, artistMode)
|
||||||
|
}
|
||||||
|
if v, ok := fields["album_artist"]; ok {
|
||||||
|
setOrClearArtistComments(cmt, "ALBUMARTIST", v, artistMode)
|
||||||
|
// Remove aliases from other taggers.
|
||||||
|
removeCommentKey(cmt, "ALBUM ARTIST")
|
||||||
|
removeCommentKey(cmt, "ALBUM_ARTIST")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Track/disc numbers: present + empty → clear; when only totals are edited,
|
||||||
|
// preserve the current index number and rewrite the combined value.
|
||||||
|
if _, ok := fields["track_number"]; ok || fields["track_total"] != "" || hasMapKey(fields, "track_total") {
|
||||||
|
currentTrackNum, currentTotalTracks := parseIndexPair(getComment(cmt, "TRACKNUMBER"))
|
||||||
|
if currentTrackNum == 0 && currentTotalTracks == 0 {
|
||||||
|
currentTrackNum, currentTotalTracks = parseIndexPair(getComment(cmt, "TRACK"))
|
||||||
|
}
|
||||||
|
if v, ok := fields["track_number"]; ok {
|
||||||
|
currentTrackNum = parsePositiveInt(v)
|
||||||
|
}
|
||||||
|
if v, ok := fields["track_total"]; ok {
|
||||||
|
currentTotalTracks = parsePositiveInt(v)
|
||||||
|
}
|
||||||
|
if currentTrackNum > 0 {
|
||||||
|
setOrClearComment(cmt, "TRACKNUMBER", formatIndexValue(currentTrackNum, currentTotalTracks))
|
||||||
|
} else {
|
||||||
|
removeCommentKey(cmt, "TRACKNUMBER")
|
||||||
|
}
|
||||||
|
removeCommentKey(cmt, "TRACK") // alias
|
||||||
|
}
|
||||||
|
if _, ok := fields["disc_number"]; ok || fields["disc_total"] != "" || hasMapKey(fields, "disc_total") {
|
||||||
|
currentDiscNum, currentTotalDiscs := parseIndexPair(getComment(cmt, "DISCNUMBER"))
|
||||||
|
if currentDiscNum == 0 && currentTotalDiscs == 0 {
|
||||||
|
currentDiscNum, currentTotalDiscs = parseIndexPair(getComment(cmt, "DISC"))
|
||||||
|
}
|
||||||
|
if v, ok := fields["disc_number"]; ok {
|
||||||
|
currentDiscNum = parsePositiveInt(v)
|
||||||
|
}
|
||||||
|
if v, ok := fields["disc_total"]; ok {
|
||||||
|
currentTotalDiscs = parsePositiveInt(v)
|
||||||
|
}
|
||||||
|
if currentDiscNum > 0 {
|
||||||
|
setOrClearComment(cmt, "DISCNUMBER", formatIndexValue(currentDiscNum, currentTotalDiscs))
|
||||||
|
} else {
|
||||||
|
removeCommentKey(cmt, "DISCNUMBER")
|
||||||
|
}
|
||||||
|
removeCommentKey(cmt, "DISC") // alias
|
||||||
|
}
|
||||||
|
|
||||||
|
// Lyrics: set both LYRICS + UNSYNCEDLYRICS, or clear both.
|
||||||
|
if v, ok := fields["lyrics"]; ok {
|
||||||
|
if v != "" {
|
||||||
|
setOrClearComment(cmt, "LYRICS", v)
|
||||||
|
setOrClearComment(cmt, "UNSYNCEDLYRICS", v)
|
||||||
|
} else {
|
||||||
|
removeCommentKey(cmt, "LYRICS")
|
||||||
|
removeCommentKey(cmt, "UNSYNCEDLYRICS")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cmtBlock := cmt.Marshal()
|
||||||
|
if cmtIdx >= 0 {
|
||||||
|
f.Meta[cmtIdx] = &cmtBlock
|
||||||
|
} else {
|
||||||
|
f.Meta = append(f.Meta, &cmtBlock)
|
||||||
|
}
|
||||||
|
|
||||||
|
coverPath := strings.TrimSpace(fields["cover_path"])
|
||||||
|
if coverPath != "" && fileExists(coverPath) {
|
||||||
|
coverData, err := os.ReadFile(coverPath)
|
||||||
|
if err == nil && len(coverData) > 0 {
|
||||||
|
for i := len(f.Meta) - 1; i >= 0; i-- {
|
||||||
|
if f.Meta[i].Type == flac.Picture {
|
||||||
|
f.Meta = append(f.Meta[:i], f.Meta[i+1:]...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
picBlock, err := buildPictureBlock("", coverData)
|
||||||
|
if err == nil {
|
||||||
|
f.Meta = append(f.Meta, &picBlock)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return f.Save(filePath)
|
||||||
|
}
|
||||||
|
|
||||||
|
// writeVorbisMetadata writes all metadata fields to a Vorbis Comment block.
|
||||||
|
// Empty/zero values are simply skipped (not written, not cleared). This is
|
||||||
|
// used by the download embedding path where absent fields should preserve any
|
||||||
|
// existing values. The editor path uses EditFlacFields() instead.
|
||||||
|
func writeVorbisMetadata(cmt *flacvorbis.MetaDataBlockVorbisComment, metadata Metadata) {
|
||||||
|
setComment(cmt, "TITLE", metadata.Title)
|
||||||
|
setArtistComments(cmt, "ARTIST", metadata.Artist, metadata.ArtistTagMode)
|
||||||
|
setComment(cmt, "ALBUM", metadata.Album)
|
||||||
|
setArtistComments(cmt, "ALBUMARTIST", metadata.AlbumArtist, metadata.ArtistTagMode)
|
||||||
|
setComment(cmt, "DATE", metadata.Date)
|
||||||
|
|
||||||
|
if metadata.TrackNumber > 0 {
|
||||||
|
setComment(cmt, "TRACKNUMBER", formatIndexValue(metadata.TrackNumber, metadata.TotalTracks))
|
||||||
|
}
|
||||||
|
|
||||||
|
if metadata.DiscNumber > 0 {
|
||||||
|
setComment(cmt, "DISCNUMBER", formatIndexValue(metadata.DiscNumber, metadata.TotalDiscs))
|
||||||
|
}
|
||||||
|
|
||||||
|
if metadata.ISRC != "" {
|
||||||
|
setComment(cmt, "ISRC", metadata.ISRC)
|
||||||
|
}
|
||||||
|
|
||||||
|
if metadata.Description != "" {
|
||||||
|
setComment(cmt, "DESCRIPTION", metadata.Description)
|
||||||
|
}
|
||||||
|
|
||||||
|
if metadata.Lyrics != "" {
|
||||||
|
setComment(cmt, "LYRICS", metadata.Lyrics)
|
||||||
|
setComment(cmt, "UNSYNCEDLYRICS", metadata.Lyrics)
|
||||||
|
}
|
||||||
|
|
||||||
|
if metadata.Genre != "" {
|
||||||
|
setComment(cmt, "GENRE", metadata.Genre)
|
||||||
|
}
|
||||||
|
|
||||||
|
if metadata.Label != "" {
|
||||||
|
setComment(cmt, "ORGANIZATION", metadata.Label)
|
||||||
|
}
|
||||||
|
|
||||||
|
if metadata.Copyright != "" {
|
||||||
|
setComment(cmt, "COPYRIGHT", metadata.Copyright)
|
||||||
|
}
|
||||||
|
|
||||||
|
if metadata.Composer != "" {
|
||||||
|
setComment(cmt, "COMPOSER", metadata.Composer)
|
||||||
|
}
|
||||||
|
|
||||||
|
if metadata.Comment != "" {
|
||||||
|
setComment(cmt, "COMMENT", metadata.Comment)
|
||||||
|
}
|
||||||
|
|
||||||
|
setComment(cmt, "REPLAYGAIN_TRACK_GAIN", metadata.ReplayGainTrackGain)
|
||||||
|
setComment(cmt, "REPLAYGAIN_TRACK_PEAK", metadata.ReplayGainTrackPeak)
|
||||||
|
setComment(cmt, "REPLAYGAIN_ALBUM_GAIN", metadata.ReplayGainAlbumGain)
|
||||||
|
setComment(cmt, "REPLAYGAIN_ALBUM_PEAK", metadata.ReplayGainAlbumPeak)
|
||||||
|
}
|
||||||
|
|
||||||
func setComment(cmt *flacvorbis.MetaDataBlockVorbisComment, key, value string) {
|
func setComment(cmt *flacvorbis.MetaDataBlockVorbisComment, key, value string) {
|
||||||
if value == "" {
|
if value == "" {
|
||||||
return
|
return
|
||||||
@@ -414,7 +546,21 @@ func setComment(cmt *flacvorbis.MetaDataBlockVorbisComment, key, value string) {
|
|||||||
cmt.Comments = append(cmt.Comments, key+"="+value)
|
cmt.Comments = append(cmt.Comments, key+"="+value)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// setOrClearComment writes a Vorbis Comment, or removes the key if value is
|
||||||
|
// empty. Used by the metadata editor path where empty means "delete this tag".
|
||||||
|
func setOrClearComment(cmt *flacvorbis.MetaDataBlockVorbisComment, key, value string) {
|
||||||
|
if value == "" {
|
||||||
|
removeCommentKey(cmt, key)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
removeCommentKey(cmt, key)
|
||||||
|
cmt.Comments = append(cmt.Comments, key+"="+value)
|
||||||
|
}
|
||||||
|
|
||||||
func setArtistComments(cmt *flacvorbis.MetaDataBlockVorbisComment, key, value, mode string) {
|
func setArtistComments(cmt *flacvorbis.MetaDataBlockVorbisComment, key, value, mode string) {
|
||||||
|
if value == "" {
|
||||||
|
return
|
||||||
|
}
|
||||||
values := []string{value}
|
values := []string{value}
|
||||||
if shouldSplitVorbisArtistTags(mode) {
|
if shouldSplitVorbisArtistTags(mode) {
|
||||||
values = splitArtistTagValues(value)
|
values = splitArtistTagValues(value)
|
||||||
@@ -431,6 +577,76 @@ func setArtistComments(cmt *flacvorbis.MetaDataBlockVorbisComment, key, value, m
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// setOrClearArtistComments writes artist Vorbis Comments, or removes the key
|
||||||
|
// if value is empty. Used by the metadata editor path.
|
||||||
|
func setOrClearArtistComments(cmt *flacvorbis.MetaDataBlockVorbisComment, key, value, mode string) {
|
||||||
|
if value == "" {
|
||||||
|
removeCommentKey(cmt, key)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
values := []string{value}
|
||||||
|
if shouldSplitVorbisArtistTags(mode) {
|
||||||
|
values = splitArtistTagValues(value)
|
||||||
|
}
|
||||||
|
if len(values) == 0 {
|
||||||
|
removeCommentKey(cmt, key)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
removeCommentKey(cmt, key)
|
||||||
|
for _, artist := range values {
|
||||||
|
if strings.TrimSpace(artist) == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
cmt.Comments = append(cmt.Comments, key+"="+artist)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// RewriteSplitArtistTags opens a FLAC file and rewrites the ARTIST and
|
||||||
|
// ALBUMARTIST Vorbis comments as multiple separate entries (one per artist).
|
||||||
|
// This is needed because FFmpeg's -metadata flag deduplicates keys, so only
|
||||||
|
// the last value survives when multiple -metadata ARTIST=X flags are used.
|
||||||
|
// The native go-flac writer correctly handles multiple Vorbis comments.
|
||||||
|
func RewriteSplitArtistTags(filePath, artist, albumArtist string) error {
|
||||||
|
if !shouldSplitVorbisArtistTags(artistTagModeSplitVorbis) {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
f, err := flac.ParseFile(filePath)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to parse FLAC file: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var cmtIdx int = -1
|
||||||
|
var cmt *flacvorbis.MetaDataBlockVorbisComment
|
||||||
|
|
||||||
|
for idx, meta := range f.Meta {
|
||||||
|
if meta.Type == flac.VorbisComment {
|
||||||
|
cmtIdx = idx
|
||||||
|
cmt, err = flacvorbis.ParseFromMetaDataBlock(*meta)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to parse vorbis comment: %w", err)
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if cmt == nil {
|
||||||
|
cmt = flacvorbis.New()
|
||||||
|
}
|
||||||
|
|
||||||
|
setArtistComments(cmt, "ARTIST", artist, artistTagModeSplitVorbis)
|
||||||
|
setArtistComments(cmt, "ALBUMARTIST", albumArtist, artistTagModeSplitVorbis)
|
||||||
|
|
||||||
|
cmtMeta := cmt.Marshal()
|
||||||
|
if cmtIdx >= 0 {
|
||||||
|
f.Meta[cmtIdx] = &cmtMeta
|
||||||
|
} else {
|
||||||
|
f.Meta = append(f.Meta, &cmtMeta)
|
||||||
|
}
|
||||||
|
|
||||||
|
return f.Save(filePath)
|
||||||
|
}
|
||||||
|
|
||||||
func removeCommentKey(cmt *flacvorbis.MetaDataBlockVorbisComment, key string) {
|
func removeCommentKey(cmt *flacvorbis.MetaDataBlockVorbisComment, key string) {
|
||||||
keyUpper := strings.ToUpper(key)
|
keyUpper := strings.ToUpper(key)
|
||||||
for i := len(cmt.Comments) - 1; i >= 0; i-- {
|
for i := len(cmt.Comments) - 1; i >= 0; i-- {
|
||||||
@@ -747,9 +963,9 @@ func ReadM4ATags(filePath string) (*AudioMetadata, error) {
|
|||||||
case "\xa9lyr":
|
case "\xa9lyr":
|
||||||
metadata.Lyrics, _ = readM4ATextValue(f, header, fi.Size())
|
metadata.Lyrics, _ = readM4ATextValue(f, header, fi.Size())
|
||||||
case "trkn":
|
case "trkn":
|
||||||
metadata.TrackNumber, _ = readM4AIndexValue(f, header, fi.Size())
|
metadata.TrackNumber, metadata.TotalTracks, _ = readM4AIndexPair(f, header, fi.Size())
|
||||||
case "disk":
|
case "disk":
|
||||||
metadata.DiscNumber, _ = readM4AIndexValue(f, header, fi.Size())
|
metadata.DiscNumber, metadata.TotalDiscs, _ = readM4AIndexPair(f, header, fi.Size())
|
||||||
case "----":
|
case "----":
|
||||||
name, value, freeformErr := readM4AFreeformValue(f, header, fi.Size())
|
name, value, freeformErr := readM4AFreeformValue(f, header, fi.Size())
|
||||||
if freeformErr == nil {
|
if freeformErr == nil {
|
||||||
@@ -774,6 +990,14 @@ func ReadM4ATags(filePath string) (*AudioMetadata, error) {
|
|||||||
if metadata.Lyrics == "" {
|
if metadata.Lyrics == "" {
|
||||||
metadata.Lyrics = value
|
metadata.Lyrics = value
|
||||||
}
|
}
|
||||||
|
case "REPLAYGAIN_TRACK_GAIN":
|
||||||
|
metadata.ReplayGainTrackGain = value
|
||||||
|
case "REPLAYGAIN_TRACK_PEAK":
|
||||||
|
metadata.ReplayGainTrackPeak = value
|
||||||
|
case "REPLAYGAIN_ALBUM_GAIN":
|
||||||
|
metadata.ReplayGainAlbumGain = value
|
||||||
|
case "REPLAYGAIN_ALBUM_PEAK":
|
||||||
|
metadata.ReplayGainAlbumPeak = value
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -936,6 +1160,41 @@ func readM4AIndexValue(f *os.File, parent atomHeader, fileSize int64) (int, erro
|
|||||||
return int(binary.BigEndian.Uint16(payload[2:4])), nil
|
return int(binary.BigEndian.Uint16(payload[2:4])), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func readM4AIndexPair(f *os.File, parent atomHeader, fileSize int64) (int, int, error) {
|
||||||
|
payload, err := readM4ADataPayload(f, parent, fileSize)
|
||||||
|
if err != nil {
|
||||||
|
return 0, 0, err
|
||||||
|
}
|
||||||
|
if len(payload) < 6 {
|
||||||
|
return 0, 0, fmt.Errorf("index payload too short in %s", parent.typ)
|
||||||
|
}
|
||||||
|
return int(binary.BigEndian.Uint16(payload[2:4])), int(binary.BigEndian.Uint16(payload[4:6])), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func parsePositiveInt(value string) int {
|
||||||
|
value = strings.TrimSpace(value)
|
||||||
|
if value == "" {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
n, _ := strconv.Atoi(value)
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func formatIndexValue(number, total int) string {
|
||||||
|
if number <= 0 {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
if total > 0 {
|
||||||
|
return fmt.Sprintf("%d/%d", number, total)
|
||||||
|
}
|
||||||
|
return strconv.Itoa(number)
|
||||||
|
}
|
||||||
|
|
||||||
|
func hasMapKey(fields map[string]string, key string) bool {
|
||||||
|
_, ok := fields[key]
|
||||||
|
return ok
|
||||||
|
}
|
||||||
|
|
||||||
func readM4AFreeformValue(f *os.File, parent atomHeader, fileSize int64) (string, string, error) {
|
func readM4AFreeformValue(f *os.File, parent atomHeader, fileSize int64) (string, string, error) {
|
||||||
start := parent.offset + parent.headerSize
|
start := parent.offset + parent.headerSize
|
||||||
end := parent.offset + parent.size
|
end := parent.offset + parent.size
|
||||||
|
|||||||
@@ -0,0 +1,149 @@
|
|||||||
|
package gobackend
|
||||||
|
|
||||||
|
import "time"
|
||||||
|
|
||||||
|
type cacheEntry struct {
|
||||||
|
data interface{}
|
||||||
|
expiresAt time.Time
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *cacheEntry) isExpired() bool {
|
||||||
|
return time.Now().After(e.expiresAt)
|
||||||
|
}
|
||||||
|
|
||||||
|
type TrackMetadata struct {
|
||||||
|
SpotifyID string `json:"spotify_id,omitempty"`
|
||||||
|
Artists string `json:"artists"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
AlbumName string `json:"album_name"`
|
||||||
|
AlbumArtist string `json:"album_artist,omitempty"`
|
||||||
|
DurationMS int `json:"duration_ms"`
|
||||||
|
Images string `json:"images"`
|
||||||
|
ReleaseDate string `json:"release_date"`
|
||||||
|
TrackNumber int `json:"track_number"`
|
||||||
|
TotalTracks int `json:"total_tracks,omitempty"`
|
||||||
|
DiscNumber int `json:"disc_number,omitempty"`
|
||||||
|
TotalDiscs int `json:"total_discs,omitempty"`
|
||||||
|
ExternalURL string `json:"external_urls"`
|
||||||
|
ISRC string `json:"isrc"`
|
||||||
|
AlbumID string `json:"album_id,omitempty"`
|
||||||
|
ArtistID string `json:"artist_id,omitempty"`
|
||||||
|
AlbumType string `json:"album_type,omitempty"`
|
||||||
|
Composer string `json:"composer,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type AlbumTrackMetadata struct {
|
||||||
|
SpotifyID string `json:"spotify_id,omitempty"`
|
||||||
|
Artists string `json:"artists"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
AlbumName string `json:"album_name"`
|
||||||
|
AlbumArtist string `json:"album_artist,omitempty"`
|
||||||
|
DurationMS int `json:"duration_ms"`
|
||||||
|
Images string `json:"images"`
|
||||||
|
ReleaseDate string `json:"release_date"`
|
||||||
|
TrackNumber int `json:"track_number"`
|
||||||
|
TotalTracks int `json:"total_tracks,omitempty"`
|
||||||
|
DiscNumber int `json:"disc_number,omitempty"`
|
||||||
|
TotalDiscs int `json:"total_discs,omitempty"`
|
||||||
|
ExternalURL string `json:"external_urls"`
|
||||||
|
ISRC string `json:"isrc"`
|
||||||
|
AlbumID string `json:"album_id,omitempty"`
|
||||||
|
AlbumURL string `json:"album_url,omitempty"`
|
||||||
|
AlbumType string `json:"album_type,omitempty"`
|
||||||
|
Composer string `json:"composer,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type AlbumInfoMetadata struct {
|
||||||
|
TotalTracks int `json:"total_tracks"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
ReleaseDate string `json:"release_date"`
|
||||||
|
Artists string `json:"artists"`
|
||||||
|
ArtistId string `json:"artist_id,omitempty"`
|
||||||
|
Images string `json:"images"`
|
||||||
|
Genre string `json:"genre,omitempty"`
|
||||||
|
Label string `json:"label,omitempty"`
|
||||||
|
Copyright string `json:"copyright,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type AlbumResponsePayload struct {
|
||||||
|
AlbumInfo AlbumInfoMetadata `json:"album_info"`
|
||||||
|
TrackList []AlbumTrackMetadata `json:"track_list"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type PlaylistInfoMetadata struct {
|
||||||
|
Name string `json:"name,omitempty"`
|
||||||
|
Images string `json:"images,omitempty"`
|
||||||
|
Tracks struct {
|
||||||
|
Total int `json:"total"`
|
||||||
|
} `json:"tracks"`
|
||||||
|
Owner struct {
|
||||||
|
DisplayName string `json:"display_name"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Images string `json:"images"`
|
||||||
|
} `json:"owner"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type PlaylistResponsePayload struct {
|
||||||
|
PlaylistInfo PlaylistInfoMetadata `json:"playlist_info"`
|
||||||
|
TrackList []AlbumTrackMetadata `json:"track_list"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ArtistInfoMetadata struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Images string `json:"images"`
|
||||||
|
Followers int `json:"followers"`
|
||||||
|
Popularity int `json:"popularity"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ArtistAlbumMetadata struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
ReleaseDate string `json:"release_date"`
|
||||||
|
TotalTracks int `json:"total_tracks"`
|
||||||
|
Images string `json:"images"`
|
||||||
|
AlbumType string `json:"album_type"`
|
||||||
|
Artists string `json:"artists"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ArtistResponsePayload struct {
|
||||||
|
ArtistInfo ArtistInfoMetadata `json:"artist_info"`
|
||||||
|
Albums []ArtistAlbumMetadata `json:"albums"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type TrackResponse struct {
|
||||||
|
Track TrackMetadata `json:"track"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SearchArtistResult struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Images string `json:"images"`
|
||||||
|
Followers int `json:"followers"`
|
||||||
|
Popularity int `json:"popularity"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SearchAlbumResult struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Artists string `json:"artists"`
|
||||||
|
Images string `json:"images"`
|
||||||
|
ReleaseDate string `json:"release_date"`
|
||||||
|
TotalTracks int `json:"total_tracks"`
|
||||||
|
AlbumType string `json:"album_type"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SearchPlaylistResult struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Owner string `json:"owner"`
|
||||||
|
Images string `json:"images"`
|
||||||
|
TotalTracks int `json:"total_tracks"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SearchAllResult struct {
|
||||||
|
Tracks []TrackMetadata `json:"tracks"`
|
||||||
|
Artists []SearchArtistResult `json:"artists"`
|
||||||
|
Albums []SearchAlbumResult `json:"albums"`
|
||||||
|
Playlists []SearchPlaylistResult `json:"playlists"`
|
||||||
|
}
|
||||||
+240
-32
@@ -44,11 +44,12 @@ var (
|
|||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
qobuzTrackGetBaseURL = "https://www.qobuz.com/api.json/0.2/track/get?track_id="
|
qobuzAPIBaseURL = "https://api.zarz.moe/v1/qbz/"
|
||||||
qobuzTrackSearchBaseURL = "https://www.qobuz.com/api.json/0.2/track/search?query="
|
qobuzTrackGetBaseURL = qobuzAPIBaseURL + "track/get?track_id="
|
||||||
qobuzAlbumGetBaseURL = "https://www.qobuz.com/api.json/0.2/album/get?album_id="
|
qobuzTrackSearchBaseURL = qobuzAPIBaseURL + "track/search?query="
|
||||||
qobuzArtistGetBaseURL = "https://www.qobuz.com/api.json/0.2/artist/get?artist_id="
|
qobuzAlbumGetBaseURL = qobuzAPIBaseURL + "album/get?album_id="
|
||||||
qobuzPlaylistGetBaseURL = "https://www.qobuz.com/api.json/0.2/playlist/get?playlist_id="
|
qobuzArtistGetBaseURL = qobuzAPIBaseURL + "artist/get?artist_id="
|
||||||
|
qobuzPlaylistGetBaseURL = qobuzAPIBaseURL + "playlist/get?playlist_id="
|
||||||
qobuzStoreSearchBaseURL = "https://www.qobuz.com/us-en/search/tracks/"
|
qobuzStoreSearchBaseURL = "https://www.qobuz.com/us-en/search/tracks/"
|
||||||
qobuzTrackOpenBaseURL = "https://open.qobuz.com/track/"
|
qobuzTrackOpenBaseURL = "https://open.qobuz.com/track/"
|
||||||
qobuzTrackPlayBaseURL = "https://play.qobuz.com/track/"
|
qobuzTrackPlayBaseURL = "https://play.qobuz.com/track/"
|
||||||
@@ -58,21 +59,19 @@ const (
|
|||||||
qobuzDeebAPIURL = "https://dab.yeet.su/api/stream?trackId="
|
qobuzDeebAPIURL = "https://dab.yeet.su/api/stream?trackId="
|
||||||
qobuzAfkarAPIURL = "https://qbz.afkarxyz.qzz.io/api/track/"
|
qobuzAfkarAPIURL = "https://qbz.afkarxyz.qzz.io/api/track/"
|
||||||
qobuzSquidAPIURL = "https://qobuz.squid.wtf/api/download-music?country=US&track_id="
|
qobuzSquidAPIURL = "https://qobuz.squid.wtf/api/download-music?country=US&track_id="
|
||||||
qobuzDebugKeyXORMask = byte(0x5A)
|
|
||||||
|
qobuzFallbackAPIBaseURL = "https://api.zarz.moe/v1/qbz2/"
|
||||||
|
qobuzFallbackTrackGetBaseURL = qobuzFallbackAPIBaseURL + "track/get?track_id="
|
||||||
|
qobuzFallbackTrackSearchBaseURL = qobuzFallbackAPIBaseURL + "track/search?query="
|
||||||
|
qobuzFallbackAlbumGetBaseURL = qobuzFallbackAPIBaseURL + "album/get?album_id="
|
||||||
|
qobuzFallbackArtistGetBaseURL = qobuzFallbackAPIBaseURL + "artist/get?artist_id="
|
||||||
|
qobuzFallbackPlaylistGetBaseURL = qobuzFallbackAPIBaseURL + "playlist/get?playlist_id="
|
||||||
)
|
)
|
||||||
|
|
||||||
var qobuzStoreTrackIDRegex = regexp.MustCompile(`/v4/ajax/popin-add-cart/track/([0-9]+)`)
|
var qobuzStoreTrackIDRegex = regexp.MustCompile(`/v4/ajax/popin-add-cart/track/([0-9]+)`)
|
||||||
var qobuzArtistAlbumIDRegex = regexp.MustCompile(`data-itemtype="album"\s+data-itemId="([A-Za-z0-9]+)"`)
|
var qobuzArtistAlbumIDRegex = regexp.MustCompile(`data-itemtype="album"\s+data-itemId="([A-Za-z0-9]+)"`)
|
||||||
var qobuzLocaleSegmentRegex = regexp.MustCompile(`^[a-z]{2}-[a-z]{2}$`)
|
var qobuzLocaleSegmentRegex = regexp.MustCompile(`^[a-z]{2}-[a-z]{2}$`)
|
||||||
|
|
||||||
var qobuzDebugKeyObfuscated = []byte{
|
|
||||||
0x69, 0x3b, 0x38, 0x3e, 0x36, 0x37, 0x35, 0x2f, 0x36, 0x3b,
|
|
||||||
0x33, 0x29, 0x2e, 0x32, 0x3f, 0x3d, 0x35, 0x3b, 0x2e, 0x3b,
|
|
||||||
0x34, 0x3e, 0x34, 0x35, 0x35, 0x34, 0x3f, 0x39, 0x35, 0x37,
|
|
||||||
0x3f, 0x29, 0x3f, 0x2c, 0x3f, 0x34, 0x39, 0x36, 0x35, 0x29,
|
|
||||||
0x3f,
|
|
||||||
}
|
|
||||||
|
|
||||||
type QobuzTrack struct {
|
type QobuzTrack struct {
|
||||||
ID int64 `json:"id"`
|
ID int64 `json:"id"`
|
||||||
Title string `json:"title"`
|
Title string `json:"title"`
|
||||||
@@ -786,12 +785,21 @@ func (q *QobuzDownloader) GetTrackByID(trackID int64) (*QobuzTrack, error) {
|
|||||||
|
|
||||||
resp, err := DoRequestWithUserAgent(q.client, req)
|
resp, err := DoRequestWithUserAgent(q.client, req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
if isQobuzPrimaryUnavailable(err) {
|
||||||
|
GoLog("[Qobuz] Primary API unavailable for track %d, trying qbz2 fallback: %v\n", trackID, err)
|
||||||
|
return q.getTrackByIDViaMusicDL(trackID)
|
||||||
|
}
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
defer resp.Body.Close()
|
defer resp.Body.Close()
|
||||||
|
|
||||||
if resp.StatusCode != 200 {
|
if resp.StatusCode != 200 {
|
||||||
return nil, fmt.Errorf("get track failed: HTTP %d", resp.StatusCode)
|
primaryErr := fmt.Errorf("get track failed: HTTP %d", resp.StatusCode)
|
||||||
|
if isQobuzPrimaryUnavailable(primaryErr) {
|
||||||
|
GoLog("[Qobuz] Primary API unavailable for track %d, trying qbz2 fallback: %v\n", trackID, primaryErr)
|
||||||
|
return q.getTrackByIDViaMusicDL(trackID)
|
||||||
|
}
|
||||||
|
return nil, primaryErr
|
||||||
}
|
}
|
||||||
|
|
||||||
var track QobuzTrack
|
var track QobuzTrack
|
||||||
@@ -802,6 +810,16 @@ func (q *QobuzDownloader) GetTrackByID(trackID int64) (*QobuzTrack, error) {
|
|||||||
return &track, nil
|
return &track, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (q *QobuzDownloader) getTrackByIDViaMusicDL(trackID int64) (*QobuzTrack, error) {
|
||||||
|
requestURL := fmt.Sprintf("%s%d", qobuzFallbackTrackGetBaseURL, trackID)
|
||||||
|
var track QobuzTrack
|
||||||
|
if err := q.getQobuzJSON(requestURL, &track); err != nil {
|
||||||
|
return nil, fmt.Errorf("qbz2 fallback also failed for track %d: %w", trackID, err)
|
||||||
|
}
|
||||||
|
GoLog("[Qobuz] qbz2 fallback succeeded for track %d\n", trackID)
|
||||||
|
return &track, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (q *QobuzDownloader) getQobuzJSON(requestURL string, target interface{}) error {
|
func (q *QobuzDownloader) getQobuzJSON(requestURL string, target interface{}) error {
|
||||||
req, err := http.NewRequest("GET", requestURL, nil)
|
req, err := http.NewRequest("GET", requestURL, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -842,6 +860,25 @@ func (q *QobuzDownloader) getQobuzBody(requestURL string) ([]byte, error) {
|
|||||||
return io.ReadAll(resp.Body)
|
return io.ReadAll(resp.Body)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func isQobuzPrimaryUnavailable(err error) bool {
|
||||||
|
if err == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
errStr := err.Error()
|
||||||
|
return strings.Contains(errStr, "HTTP 429") ||
|
||||||
|
strings.Contains(errStr, "HTTP 5") ||
|
||||||
|
strings.Contains(errStr, "rate limit") ||
|
||||||
|
strings.Contains(errStr, "connection refused") ||
|
||||||
|
strings.Contains(errStr, "no such host") ||
|
||||||
|
strings.Contains(errStr, "i/o timeout") ||
|
||||||
|
strings.Contains(errStr, "deadline exceeded") ||
|
||||||
|
strings.Contains(errStr, "EOF") ||
|
||||||
|
strings.Contains(errStr, "connection reset") ||
|
||||||
|
strings.Contains(errStr, "TLS handshake") ||
|
||||||
|
strings.Contains(errStr, "server misbehaving") ||
|
||||||
|
strings.Contains(errStr, "network is unreachable")
|
||||||
|
}
|
||||||
|
|
||||||
func extractQobuzAlbumIDsFromArtistHTML(body []byte) []string {
|
func extractQobuzAlbumIDsFromArtistHTML(body []byte) []string {
|
||||||
matches := qobuzArtistAlbumIDRegex.FindAllSubmatch(body, -1)
|
matches := qobuzArtistAlbumIDRegex.FindAllSubmatch(body, -1)
|
||||||
if len(matches) == 0 {
|
if len(matches) == 0 {
|
||||||
@@ -871,20 +908,48 @@ func (q *QobuzDownloader) getAlbumDetails(albumID string) (*qobuzAlbumDetails, e
|
|||||||
requestURL := fmt.Sprintf("%s%s&app_id=%s", qobuzAlbumGetBaseURL, url.QueryEscape(strings.TrimSpace(albumID)), q.appID)
|
requestURL := fmt.Sprintf("%s%s&app_id=%s", qobuzAlbumGetBaseURL, url.QueryEscape(strings.TrimSpace(albumID)), q.appID)
|
||||||
var album qobuzAlbumDetails
|
var album qobuzAlbumDetails
|
||||||
if err := q.getQobuzJSON(requestURL, &album); err != nil {
|
if err := q.getQobuzJSON(requestURL, &album); err != nil {
|
||||||
|
if isQobuzPrimaryUnavailable(err) {
|
||||||
|
GoLog("[Qobuz] Primary API unavailable for album %s, trying qbz2 fallback: %v\n", albumID, err)
|
||||||
|
return q.getAlbumDetailsViaMusicDL(albumID)
|
||||||
|
}
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return &album, nil
|
return &album, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (q *QobuzDownloader) getAlbumDetailsViaMusicDL(albumID string) (*qobuzAlbumDetails, error) {
|
||||||
|
requestURL := fmt.Sprintf("%s%s", qobuzFallbackAlbumGetBaseURL, url.QueryEscape(strings.TrimSpace(albumID)))
|
||||||
|
var album qobuzAlbumDetails
|
||||||
|
if err := q.getQobuzJSON(requestURL, &album); err != nil {
|
||||||
|
return nil, fmt.Errorf("qbz2 fallback also failed for album %s: %w", albumID, err)
|
||||||
|
}
|
||||||
|
GoLog("[Qobuz] qbz2 fallback succeeded for album %s\n", albumID)
|
||||||
|
return &album, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (q *QobuzDownloader) getArtistDetails(artistID string) (*qobuzArtistDetails, error) {
|
func (q *QobuzDownloader) getArtistDetails(artistID string) (*qobuzArtistDetails, error) {
|
||||||
requestURL := fmt.Sprintf("%s%s&app_id=%s", qobuzArtistGetBaseURL, url.QueryEscape(strings.TrimSpace(artistID)), q.appID)
|
requestURL := fmt.Sprintf("%s%s&app_id=%s", qobuzArtistGetBaseURL, url.QueryEscape(strings.TrimSpace(artistID)), q.appID)
|
||||||
var artist qobuzArtistDetails
|
var artist qobuzArtistDetails
|
||||||
if err := q.getQobuzJSON(requestURL, &artist); err != nil {
|
if err := q.getQobuzJSON(requestURL, &artist); err != nil {
|
||||||
|
if isQobuzPrimaryUnavailable(err) {
|
||||||
|
GoLog("[Qobuz] Primary API unavailable for artist %s, trying qbz2 fallback: %v\n", artistID, err)
|
||||||
|
return q.getArtistDetailsViaMusicDL(artistID)
|
||||||
|
}
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return &artist, nil
|
return &artist, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (q *QobuzDownloader) getArtistDetailsViaMusicDL(artistID string) (*qobuzArtistDetails, error) {
|
||||||
|
requestURL := fmt.Sprintf("%s%s", qobuzFallbackArtistGetBaseURL, url.QueryEscape(strings.TrimSpace(artistID)))
|
||||||
|
var artist qobuzArtistDetails
|
||||||
|
if err := q.getQobuzJSON(requestURL, &artist); err != nil {
|
||||||
|
return nil, fmt.Errorf("qbz2 fallback also failed for artist %s: %w", artistID, err)
|
||||||
|
}
|
||||||
|
GoLog("[Qobuz] qbz2 fallback succeeded for artist %s\n", artistID)
|
||||||
|
return &artist, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (q *QobuzDownloader) getPlaylistDetailsPage(playlistID string, limit, offset int) (*qobuzPlaylistDetails, error) {
|
func (q *QobuzDownloader) getPlaylistDetailsPage(playlistID string, limit, offset int) (*qobuzPlaylistDetails, error) {
|
||||||
requestURL := fmt.Sprintf(
|
requestURL := fmt.Sprintf(
|
||||||
"%s%s&extra=tracks&limit=%d&offset=%d&app_id=%s",
|
"%s%s&extra=tracks&limit=%d&offset=%d&app_id=%s",
|
||||||
@@ -896,11 +961,31 @@ func (q *QobuzDownloader) getPlaylistDetailsPage(playlistID string, limit, offse
|
|||||||
)
|
)
|
||||||
var playlist qobuzPlaylistDetails
|
var playlist qobuzPlaylistDetails
|
||||||
if err := q.getQobuzJSON(requestURL, &playlist); err != nil {
|
if err := q.getQobuzJSON(requestURL, &playlist); err != nil {
|
||||||
|
if isQobuzPrimaryUnavailable(err) {
|
||||||
|
GoLog("[Qobuz] Primary API unavailable for playlist %s, trying qbz2 fallback: %v\n", playlistID, err)
|
||||||
|
return q.getPlaylistDetailsPageViaMusicDL(playlistID, limit, offset)
|
||||||
|
}
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return &playlist, nil
|
return &playlist, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (q *QobuzDownloader) getPlaylistDetailsPageViaMusicDL(playlistID string, limit, offset int) (*qobuzPlaylistDetails, error) {
|
||||||
|
requestURL := fmt.Sprintf(
|
||||||
|
"%s%s&limit=%d&offset=%d",
|
||||||
|
qobuzFallbackPlaylistGetBaseURL,
|
||||||
|
url.QueryEscape(strings.TrimSpace(playlistID)),
|
||||||
|
limit,
|
||||||
|
offset,
|
||||||
|
)
|
||||||
|
var playlist qobuzPlaylistDetails
|
||||||
|
if err := q.getQobuzJSON(requestURL, &playlist); err != nil {
|
||||||
|
return nil, fmt.Errorf("qbz2 fallback also failed for playlist %s: %w", playlistID, err)
|
||||||
|
}
|
||||||
|
GoLog("[Qobuz] qbz2 fallback succeeded for playlist %s (offset=%d)\n", playlistID, offset)
|
||||||
|
return &playlist, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (q *QobuzDownloader) getArtistAlbumIDs(artistID string) ([]string, error) {
|
func (q *QobuzDownloader) getArtistAlbumIDs(artistID string) ([]string, error) {
|
||||||
artist, err := q.getArtistDetails(artistID)
|
artist, err := q.getArtistDetails(artistID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -945,6 +1030,7 @@ func (q *QobuzDownloader) GetAlbumMetadata(resourceID string) (*AlbumResponsePay
|
|||||||
}
|
}
|
||||||
|
|
||||||
tracks := make([]AlbumTrackMetadata, 0, len(album.Tracks.Items))
|
tracks := make([]AlbumTrackMetadata, 0, len(album.Tracks.Items))
|
||||||
|
totalDiscs := 0
|
||||||
for i := range album.Tracks.Items {
|
for i := range album.Tracks.Items {
|
||||||
track := &album.Tracks.Items[i]
|
track := &album.Tracks.Items[i]
|
||||||
track.Album.ID = album.ID
|
track.Album.ID = album.ID
|
||||||
@@ -956,8 +1042,14 @@ func (q *QobuzDownloader) GetAlbumMetadata(resourceID string) (*AlbumResponsePay
|
|||||||
Large: album.Image.Large,
|
Large: album.Image.Large,
|
||||||
}
|
}
|
||||||
track.Album.TracksCount = album.TracksCount
|
track.Album.TracksCount = album.TracksCount
|
||||||
|
if track.MediaNumber > totalDiscs {
|
||||||
|
totalDiscs = track.MediaNumber
|
||||||
|
}
|
||||||
tracks = append(tracks, qobuzTrackToAlbumTrackMetadata(track))
|
tracks = append(tracks, qobuzTrackToAlbumTrackMetadata(track))
|
||||||
}
|
}
|
||||||
|
for i := range tracks {
|
||||||
|
tracks[i].TotalDiscs = totalDiscs
|
||||||
|
}
|
||||||
|
|
||||||
return &AlbumResponsePayload{
|
return &AlbumResponsePayload{
|
||||||
AlbumInfo: qobuzAlbumToAlbumInfo(album),
|
AlbumInfo: qobuzAlbumToAlbumInfo(album),
|
||||||
@@ -1063,9 +1155,7 @@ func (q *QobuzDownloader) GetAvailableProviders() []qobuzAPIProvider {
|
|||||||
return []qobuzAPIProvider{
|
return []qobuzAPIProvider{
|
||||||
{Name: "musicdl", URL: qobuzDownloadAPIURL, Kind: qobuzAPIKindMusicDL},
|
{Name: "musicdl", URL: qobuzDownloadAPIURL, Kind: qobuzAPIKindMusicDL},
|
||||||
{Name: "dabmusic", URL: qobuzDabMusicAPIURL, Kind: qobuzAPIKindStandard},
|
{Name: "dabmusic", URL: qobuzDabMusicAPIURL, Kind: qobuzAPIKindStandard},
|
||||||
// "deeb" is mapped from the legacy reference fallback endpoint.
|
|
||||||
{Name: "deeb", URL: qobuzDeebAPIURL, Kind: qobuzAPIKindStandard},
|
{Name: "deeb", URL: qobuzDeebAPIURL, Kind: qobuzAPIKindStandard},
|
||||||
// "qbz" comes from the desktop reference app and uses /api/track/{id}?quality=...
|
|
||||||
{Name: "qbz", URL: qobuzAfkarAPIURL, Kind: qobuzAPIKindStandard},
|
{Name: "qbz", URL: qobuzAfkarAPIURL, Kind: qobuzAPIKindStandard},
|
||||||
{Name: "squid", URL: qobuzSquidAPIURL, Kind: qobuzAPIKindStandard},
|
{Name: "squid", URL: qobuzSquidAPIURL, Kind: qobuzAPIKindStandard},
|
||||||
}
|
}
|
||||||
@@ -1216,14 +1306,6 @@ func mapQobuzQualityCodeToAPI(qualityCode string) string {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func getQobuzDebugKey() string {
|
|
||||||
decoded := make([]byte, len(qobuzDebugKeyObfuscated))
|
|
||||||
for i, b := range qobuzDebugKeyObfuscated {
|
|
||||||
decoded[i] = b ^ qobuzDebugKeyXORMask
|
|
||||||
}
|
|
||||||
return string(decoded)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (q *QobuzDownloader) SearchTrackByISRC(isrc string) (*QobuzTrack, error) {
|
func (q *QobuzDownloader) SearchTrackByISRC(isrc string) (*QobuzTrack, error) {
|
||||||
candidates, err := q.searchQobuzTracksWithFallback(isrc, 50)
|
candidates, err := q.searchQobuzTracksWithFallback(isrc, 50)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -1376,9 +1458,10 @@ func (q *QobuzDownloader) SearchAll(query string, trackLimit, artistLimit int, f
|
|||||||
}
|
}
|
||||||
|
|
||||||
if artistLimit > 0 {
|
if artistLimit > 0 {
|
||||||
searchURL := fmt.Sprintf("https://www.qobuz.com/api.json/0.2/artist/search?query=%s&limit=%d&app_id=%s",
|
searchURL := fmt.Sprintf("%sartist/search?query=%s&limit=%d&app_id=%s",
|
||||||
url.QueryEscape(cleanQuery), artistLimit, q.appID)
|
qobuzAPIBaseURL, url.QueryEscape(cleanQuery), artistLimit, q.appID)
|
||||||
req, err := http.NewRequest("GET", searchURL, nil)
|
req, err := http.NewRequest("GET", searchURL, nil)
|
||||||
|
artistSearchDone := false
|
||||||
if err == nil {
|
if err == nil {
|
||||||
resp, reqErr := DoRequestWithUserAgent(q.client, req)
|
resp, reqErr := DoRequestWithUserAgent(q.client, req)
|
||||||
if reqErr == nil {
|
if reqErr == nil {
|
||||||
@@ -1403,20 +1486,30 @@ func (q *QobuzDownloader) SearchAll(query string, trackLimit, artistLimit int, f
|
|||||||
Images: imageURL,
|
Images: imageURL,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
artistSearchDone = true
|
||||||
} else {
|
} else {
|
||||||
GoLog("[Qobuz] Artist search decode failed: %v\n", decErr)
|
GoLog("[Qobuz] Artist search decode failed: %v\n", decErr)
|
||||||
}
|
}
|
||||||
|
} else if isQobuzPrimaryUnavailable(fmt.Errorf("HTTP %d", resp.StatusCode)) {
|
||||||
|
GoLog("[Qobuz] Artist search primary API returned HTTP %d, will try qbz2 fallback\n", resp.StatusCode)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
GoLog("[Qobuz] Artist search request failed: %v\n", reqErr)
|
GoLog("[Qobuz] Artist search request failed: %v\n", reqErr)
|
||||||
|
if isQobuzPrimaryUnavailable(reqErr) {
|
||||||
|
GoLog("[Qobuz] Primary API unavailable for artist search, will try qbz2 fallback\n")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if !artistSearchDone {
|
||||||
|
q.searchAllArtistsViaMusicDL(cleanQuery, artistLimit, result)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if albumLimit > 0 {
|
if albumLimit > 0 {
|
||||||
searchURL := fmt.Sprintf("https://www.qobuz.com/api.json/0.2/album/search?query=%s&limit=%d&app_id=%s",
|
searchURL := fmt.Sprintf("%salbum/search?query=%s&limit=%d&app_id=%s",
|
||||||
url.QueryEscape(cleanQuery), albumLimit, q.appID)
|
qobuzAPIBaseURL, url.QueryEscape(cleanQuery), albumLimit, q.appID)
|
||||||
req, err := http.NewRequest("GET", searchURL, nil)
|
req, err := http.NewRequest("GET", searchURL, nil)
|
||||||
|
albumSearchDone := false
|
||||||
if err == nil {
|
if err == nil {
|
||||||
resp, reqErr := DoRequestWithUserAgent(q.client, req)
|
resp, reqErr := DoRequestWithUserAgent(q.client, req)
|
||||||
if reqErr == nil {
|
if reqErr == nil {
|
||||||
@@ -1441,20 +1534,81 @@ func (q *QobuzDownloader) SearchAll(query string, trackLimit, artistLimit int, f
|
|||||||
AlbumType: qobuzNormalizeAlbumType(album.ReleaseType, album.ProductType, album.TracksCount),
|
AlbumType: qobuzNormalizeAlbumType(album.ReleaseType, album.ProductType, album.TracksCount),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
albumSearchDone = true
|
||||||
} else {
|
} else {
|
||||||
GoLog("[Qobuz] Album search decode failed: %v\n", decErr)
|
GoLog("[Qobuz] Album search decode failed: %v\n", decErr)
|
||||||
}
|
}
|
||||||
|
} else if isQobuzPrimaryUnavailable(fmt.Errorf("HTTP %d", resp.StatusCode)) {
|
||||||
|
GoLog("[Qobuz] Album search primary API returned HTTP %d, will try qbz2 fallback\n", resp.StatusCode)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
GoLog("[Qobuz] Album search request failed: %v\n", reqErr)
|
GoLog("[Qobuz] Album search request failed: %v\n", reqErr)
|
||||||
|
if isQobuzPrimaryUnavailable(reqErr) {
|
||||||
|
GoLog("[Qobuz] Primary API unavailable for album search, will try qbz2 fallback\n")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if !albumSearchDone {
|
||||||
|
q.searchAllAlbumsViaMusicDL(cleanQuery, albumLimit, result)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
GoLog("[Qobuz] SearchAll complete: %d tracks, %d artists, %d albums\n", len(result.Tracks), len(result.Artists), len(result.Albums))
|
GoLog("[Qobuz] SearchAll complete: %d tracks, %d artists, %d albums\n", len(result.Tracks), len(result.Artists), len(result.Albums))
|
||||||
return result, nil
|
return result, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (q *QobuzDownloader) searchAllArtistsViaMusicDL(query string, limit int, result *SearchAllResult) {
|
||||||
|
requestURL := fmt.Sprintf("%sartist/search?query=%s&limit=%d", qobuzFallbackAPIBaseURL, url.QueryEscape(query), limit)
|
||||||
|
var searchResp struct {
|
||||||
|
Artists struct {
|
||||||
|
Items []struct {
|
||||||
|
ID int64 `json:"id"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Image qobuzImageSet `json:"image"`
|
||||||
|
} `json:"items"`
|
||||||
|
} `json:"artists"`
|
||||||
|
}
|
||||||
|
if err := q.getQobuzJSON(requestURL, &searchResp); err != nil {
|
||||||
|
GoLog("[Qobuz] qbz2 fallback artist search also failed: %v\n", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
GoLog("[Qobuz] qbz2 fallback artist search succeeded: %d artists\n", len(searchResp.Artists.Items))
|
||||||
|
for _, artist := range searchResp.Artists.Items {
|
||||||
|
imageURL := qobuzFirstNonEmpty(artist.Image.Large, artist.Image.Small, artist.Image.Thumbnail)
|
||||||
|
result.Artists = append(result.Artists, SearchArtistResult{
|
||||||
|
ID: qobuzPrefixedNumericID(artist.ID),
|
||||||
|
Name: strings.TrimSpace(artist.Name),
|
||||||
|
Images: imageURL,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (q *QobuzDownloader) searchAllAlbumsViaMusicDL(query string, limit int, result *SearchAllResult) {
|
||||||
|
requestURL := fmt.Sprintf("%salbum/search?query=%s&limit=%d", qobuzFallbackAPIBaseURL, url.QueryEscape(query), limit)
|
||||||
|
var searchResp struct {
|
||||||
|
Albums struct {
|
||||||
|
Items []qobuzAlbumDetails `json:"items"`
|
||||||
|
} `json:"albums"`
|
||||||
|
}
|
||||||
|
if err := q.getQobuzJSON(requestURL, &searchResp); err != nil {
|
||||||
|
GoLog("[Qobuz] qbz2 fallback album search also failed: %v\n", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
GoLog("[Qobuz] qbz2 fallback album search succeeded: %d albums\n", len(searchResp.Albums.Items))
|
||||||
|
for i := range searchResp.Albums.Items {
|
||||||
|
album := &searchResp.Albums.Items[i]
|
||||||
|
result.Albums = append(result.Albums, SearchAlbumResult{
|
||||||
|
ID: qobuzPrefixedID(album.ID),
|
||||||
|
Name: strings.TrimSpace(album.Title),
|
||||||
|
Artists: qobuzArtistsDisplayName(album.Artists, album.Artist.Name),
|
||||||
|
Images: qobuzAlbumImage(album),
|
||||||
|
ReleaseDate: qobuzNormalizeReleaseDate(album.ReleaseDateOriginal),
|
||||||
|
TotalTracks: album.TracksCount,
|
||||||
|
AlbumType: qobuzNormalizeAlbumType(album.ReleaseType, album.ProductType, album.TracksCount),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (q *QobuzDownloader) SearchTrackByMetadataWithDuration(trackName, artistName string, expectedDurationSec int) (*QobuzTrack, error) {
|
func (q *QobuzDownloader) SearchTrackByMetadataWithDuration(trackName, artistName string, expectedDurationSec int) (*QobuzTrack, error) {
|
||||||
queries := []string{}
|
queries := []string{}
|
||||||
|
|
||||||
@@ -1646,13 +1800,22 @@ func (q *QobuzDownloader) searchQobuzTracksViaAPI(query string, limit int) ([]Qo
|
|||||||
|
|
||||||
resp, err := DoRequestWithUserAgent(q.client, req)
|
resp, err := DoRequestWithUserAgent(q.client, req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
if isQobuzPrimaryUnavailable(err) {
|
||||||
|
GoLog("[Qobuz] Primary API unavailable for track search, trying qbz2 fallback: %v\n", err)
|
||||||
|
return q.searchQobuzTracksViaMusicDL(query, limit)
|
||||||
|
}
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
defer resp.Body.Close()
|
defer resp.Body.Close()
|
||||||
|
|
||||||
if resp.StatusCode != 200 {
|
if resp.StatusCode != 200 {
|
||||||
body, _ := io.ReadAll(io.LimitReader(resp.Body, 512))
|
body, _ := io.ReadAll(io.LimitReader(resp.Body, 512))
|
||||||
return nil, fmt.Errorf("search failed: HTTP %d (%s)", resp.StatusCode, strings.TrimSpace(string(body)))
|
primaryErr := fmt.Errorf("search failed: HTTP %d (%s)", resp.StatusCode, strings.TrimSpace(string(body)))
|
||||||
|
if isQobuzPrimaryUnavailable(primaryErr) {
|
||||||
|
GoLog("[Qobuz] Primary API unavailable for track search, trying qbz2 fallback: %v\n", primaryErr)
|
||||||
|
return q.searchQobuzTracksViaMusicDL(query, limit)
|
||||||
|
}
|
||||||
|
return nil, primaryErr
|
||||||
}
|
}
|
||||||
|
|
||||||
var result struct {
|
var result struct {
|
||||||
@@ -1666,6 +1829,20 @@ func (q *QobuzDownloader) searchQobuzTracksViaAPI(query string, limit int) ([]Qo
|
|||||||
return result.Tracks.Items, nil
|
return result.Tracks.Items, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (q *QobuzDownloader) searchQobuzTracksViaMusicDL(query string, limit int) ([]QobuzTrack, error) {
|
||||||
|
requestURL := fmt.Sprintf("%s%s&limit=%d", qobuzFallbackTrackSearchBaseURL, url.QueryEscape(query), limit)
|
||||||
|
var result struct {
|
||||||
|
Tracks struct {
|
||||||
|
Items []QobuzTrack `json:"items"`
|
||||||
|
} `json:"tracks"`
|
||||||
|
}
|
||||||
|
if err := q.getQobuzJSON(requestURL, &result); err != nil {
|
||||||
|
return nil, fmt.Errorf("qbz2 fallback search also failed: %w", err)
|
||||||
|
}
|
||||||
|
GoLog("[Qobuz] qbz2 fallback search succeeded: %d tracks for '%s'\n", len(result.Tracks.Items), query)
|
||||||
|
return result.Tracks.Items, nil
|
||||||
|
}
|
||||||
|
|
||||||
type qobuzTrackSearchCandidate struct {
|
type qobuzTrackSearchCandidate struct {
|
||||||
score int
|
score int
|
||||||
track QobuzTrack
|
track QobuzTrack
|
||||||
@@ -1855,7 +2032,8 @@ func (q *QobuzDownloader) searchQobuzTracksViaAlbumSearch(query string, limit in
|
|||||||
}
|
}
|
||||||
|
|
||||||
searchURL := fmt.Sprintf(
|
searchURL := fmt.Sprintf(
|
||||||
"https://www.qobuz.com/api.json/0.2/album/search?query=%s&limit=%d&app_id=%s",
|
"%salbum/search?query=%s&limit=%d&app_id=%s",
|
||||||
|
qobuzAPIBaseURL,
|
||||||
url.QueryEscape(strings.TrimSpace(query)),
|
url.QueryEscape(strings.TrimSpace(query)),
|
||||||
albumLimit,
|
albumLimit,
|
||||||
q.appID,
|
q.appID,
|
||||||
@@ -1868,13 +2046,22 @@ func (q *QobuzDownloader) searchQobuzTracksViaAlbumSearch(query string, limit in
|
|||||||
|
|
||||||
resp, err := DoRequestWithUserAgent(q.client, req)
|
resp, err := DoRequestWithUserAgent(q.client, req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
if isQobuzPrimaryUnavailable(err) {
|
||||||
|
GoLog("[Qobuz] Primary API unavailable for album search fallback, trying qbz2: %v\n", err)
|
||||||
|
return q.searchQobuzTracksViaAlbumSearchMusicDL(query, limit, albumLimit)
|
||||||
|
}
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
defer resp.Body.Close()
|
defer resp.Body.Close()
|
||||||
|
|
||||||
if resp.StatusCode != 200 {
|
if resp.StatusCode != 200 {
|
||||||
body, _ := io.ReadAll(io.LimitReader(resp.Body, 512))
|
body, _ := io.ReadAll(io.LimitReader(resp.Body, 512))
|
||||||
return nil, fmt.Errorf("album search failed: HTTP %d (%s)", resp.StatusCode, strings.TrimSpace(string(body)))
|
primaryErr := fmt.Errorf("album search failed: HTTP %d (%s)", resp.StatusCode, strings.TrimSpace(string(body)))
|
||||||
|
if isQobuzPrimaryUnavailable(primaryErr) {
|
||||||
|
GoLog("[Qobuz] Primary API unavailable for album search fallback, trying qbz2: %v\n", primaryErr)
|
||||||
|
return q.searchQobuzTracksViaAlbumSearchMusicDL(query, limit, albumLimit)
|
||||||
|
}
|
||||||
|
return nil, primaryErr
|
||||||
}
|
}
|
||||||
|
|
||||||
var albumResp struct {
|
var albumResp struct {
|
||||||
@@ -1894,6 +2081,25 @@ func (q *QobuzDownloader) searchQobuzTracksViaAlbumSearch(query string, limit in
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (q *QobuzDownloader) searchQobuzTracksViaAlbumSearchMusicDL(query string, limit, albumLimit int) ([]QobuzTrack, error) {
|
||||||
|
requestURL := fmt.Sprintf("%salbum/search?query=%s&limit=%d", qobuzFallbackAPIBaseURL, url.QueryEscape(strings.TrimSpace(query)), albumLimit)
|
||||||
|
var searchResp struct {
|
||||||
|
Albums struct {
|
||||||
|
Items []qobuzAlbumDetails `json:"items"`
|
||||||
|
} `json:"albums"`
|
||||||
|
}
|
||||||
|
if err := q.getQobuzJSON(requestURL, &searchResp); err != nil {
|
||||||
|
return nil, fmt.Errorf("qbz2 fallback album search also failed: %w", err)
|
||||||
|
}
|
||||||
|
GoLog("[Qobuz] qbz2 fallback album search returned %d albums\n", len(searchResp.Albums.Items))
|
||||||
|
return selectQobuzTracksFromAlbumSearchResults(
|
||||||
|
query,
|
||||||
|
limit,
|
||||||
|
searchResp.Albums.Items,
|
||||||
|
q.getAlbumDetails,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
func extractQobuzTrackIDsFromStoreSearchHTML(body []byte) []int64 {
|
func extractQobuzTrackIDsFromStoreSearchHTML(body []byte) []int64 {
|
||||||
matches := qobuzStoreTrackIDRegex.FindAllSubmatch(body, -1)
|
matches := qobuzStoreTrackIDRegex.FindAllSubmatch(body, -1)
|
||||||
if len(matches) == 0 {
|
if len(matches) == 0 {
|
||||||
@@ -2594,10 +2800,12 @@ func downloadFromQobuz(req DownloadRequest) (QobuzDownloadResult, error) {
|
|||||||
TrackNumber: actualTrackNumber,
|
TrackNumber: actualTrackNumber,
|
||||||
TotalTracks: req.TotalTracks,
|
TotalTracks: req.TotalTracks,
|
||||||
DiscNumber: req.DiscNumber,
|
DiscNumber: req.DiscNumber,
|
||||||
|
TotalDiscs: req.TotalDiscs,
|
||||||
ISRC: track.ISRC,
|
ISRC: track.ISRC,
|
||||||
Genre: req.Genre,
|
Genre: req.Genre,
|
||||||
Label: req.Label,
|
Label: req.Label,
|
||||||
Copyright: req.Copyright,
|
Copyright: req.Copyright,
|
||||||
|
Composer: req.Composer,
|
||||||
}
|
}
|
||||||
|
|
||||||
var coverData []byte
|
var coverData []byte
|
||||||
|
|||||||
@@ -201,18 +201,6 @@ func TestNormalizeQobuzQualityCode(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestGetQobuzDebugKey(t *testing.T) {
|
|
||||||
got := getQobuzDebugKey()
|
|
||||||
if len(got) != len(qobuzDebugKeyObfuscated) {
|
|
||||||
t.Fatalf("unexpected debug key length: %d", len(got))
|
|
||||||
}
|
|
||||||
for i := range got {
|
|
||||||
if got[i]^qobuzDebugKeyXORMask != qobuzDebugKeyObfuscated[i] {
|
|
||||||
t.Fatalf("unexpected debug key reconstruction at index %d", i)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestBuildQobuzMusicDLPayloadUsesOpenTrackURL(t *testing.T) {
|
func TestBuildQobuzMusicDLPayloadUsesOpenTrackURL(t *testing.T) {
|
||||||
payloadBytes, err := buildQobuzMusicDLPayload(374610875, "7")
|
payloadBytes, err := buildQobuzMusicDLPayload(374610875, "7")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|||||||
+4
-17
@@ -16,16 +16,13 @@ var hiraganaToRomaji = map[rune]string{
|
|||||||
'や': "ya", 'ゆ': "yu", 'よ': "yo",
|
'や': "ya", 'ゆ': "yu", 'よ': "yo",
|
||||||
'ら': "ra", 'り': "ri", 'る': "ru", 'れ': "re", 'ろ': "ro",
|
'ら': "ra", 'り': "ri", 'る': "ru", 'れ': "re", 'ろ': "ro",
|
||||||
'わ': "wa", 'を': "wo", 'ん': "n",
|
'わ': "wa", 'を': "wo", 'ん': "n",
|
||||||
// Dakuten (voiced)
|
|
||||||
'が': "ga", 'ぎ': "gi", 'ぐ': "gu", 'げ': "ge", 'ご': "go",
|
'が': "ga", 'ぎ': "gi", 'ぐ': "gu", 'げ': "ge", 'ご': "go",
|
||||||
'ざ': "za", 'じ': "ji", 'ず': "zu", 'ぜ': "ze", 'ぞ': "zo",
|
'ざ': "za", 'じ': "ji", 'ず': "zu", 'ぜ': "ze", 'ぞ': "zo",
|
||||||
'だ': "da", 'ぢ': "ji", 'づ': "zu", 'で': "de", 'ど': "do",
|
'だ': "da", 'ぢ': "ji", 'づ': "zu", 'で': "de", 'ど': "do",
|
||||||
'ば': "ba", 'び': "bi", 'ぶ': "bu", 'べ': "be", 'ぼ': "bo",
|
'ば': "ba", 'び': "bi", 'ぶ': "bu", 'べ': "be", 'ぼ': "bo",
|
||||||
// Handakuten (semi-voiced)
|
|
||||||
'ぱ': "pa", 'ぴ': "pi", 'ぷ': "pu", 'ぺ': "pe", 'ぽ': "po",
|
'ぱ': "pa", 'ぴ': "pi", 'ぷ': "pu", 'ぺ': "pe", 'ぽ': "po",
|
||||||
// Small characters
|
|
||||||
'ゃ': "ya", 'ゅ': "yu", 'ょ': "yo",
|
'ゃ': "ya", 'ゅ': "yu", 'ょ': "yo",
|
||||||
'っ': "", // Double consonant marker
|
'っ': "",
|
||||||
'ぁ': "a", 'ぃ': "i", 'ぅ': "u", 'ぇ': "e", 'ぉ': "o",
|
'ぁ': "a", 'ぃ': "i", 'ぅ': "u", 'ぇ': "e", 'ぉ': "o",
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -40,19 +37,15 @@ var katakanaToRomaji = map[rune]string{
|
|||||||
'ヤ': "ya", 'ユ': "yu", 'ヨ': "yo",
|
'ヤ': "ya", 'ユ': "yu", 'ヨ': "yo",
|
||||||
'ラ': "ra", 'リ': "ri", 'ル': "ru", 'レ': "re", 'ロ': "ro",
|
'ラ': "ra", 'リ': "ri", 'ル': "ru", 'レ': "re", 'ロ': "ro",
|
||||||
'ワ': "wa", 'ヲ': "wo", 'ン': "n",
|
'ワ': "wa", 'ヲ': "wo", 'ン': "n",
|
||||||
// Dakuten (voiced)
|
|
||||||
'ガ': "ga", 'ギ': "gi", 'グ': "gu", 'ゲ': "ge", 'ゴ': "go",
|
'ガ': "ga", 'ギ': "gi", 'グ': "gu", 'ゲ': "ge", 'ゴ': "go",
|
||||||
'ザ': "za", 'ジ': "ji", 'ズ': "zu", 'ゼ': "ze", 'ゾ': "zo",
|
'ザ': "za", 'ジ': "ji", 'ズ': "zu", 'ゼ': "ze", 'ゾ': "zo",
|
||||||
'ダ': "da", 'ヂ': "ji", 'ヅ': "zu", 'デ': "de", 'ド': "do",
|
'ダ': "da", 'ヂ': "ji", 'ヅ': "zu", 'デ': "de", 'ド': "do",
|
||||||
'バ': "ba", 'ビ': "bi", 'ブ': "bu", 'ベ': "be", 'ボ': "bo",
|
'バ': "ba", 'ビ': "bi", 'ブ': "bu", 'ベ': "be", 'ボ': "bo",
|
||||||
// Handakuten (semi-voiced)
|
|
||||||
'パ': "pa", 'ピ': "pi", 'プ': "pu", 'ペ': "pe", 'ポ': "po",
|
'パ': "pa", 'ピ': "pi", 'プ': "pu", 'ペ': "pe", 'ポ': "po",
|
||||||
// Small characters
|
|
||||||
'ャ': "ya", 'ュ': "yu", 'ョ': "yo",
|
'ャ': "ya", 'ュ': "yu", 'ョ': "yo",
|
||||||
'ッ': "", // Double consonant marker
|
'ッ': "",
|
||||||
'ァ': "a", 'ィ': "i", 'ゥ': "u", 'ェ': "e", 'ォ': "o",
|
'ァ': "a", 'ィ': "i", 'ゥ': "u", 'ェ': "e", 'ォ': "o",
|
||||||
// Extended katakana
|
'ー': "",
|
||||||
'ー': "", // Long vowel mark
|
|
||||||
'ヴ': "vu",
|
'ヴ': "vu",
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -82,7 +75,6 @@ var combinationKatakana = map[string]string{
|
|||||||
"ジャ": "ja", "ジュ": "ju", "ジョ": "jo",
|
"ジャ": "ja", "ジュ": "ju", "ジョ": "jo",
|
||||||
"ビャ": "bya", "ビュ": "byu", "ビョ": "byo",
|
"ビャ": "bya", "ビュ": "byu", "ビョ": "byo",
|
||||||
"ピャ": "pya", "ピュ": "pyu", "ピョ": "pyo",
|
"ピャ": "pya", "ピュ": "pyu", "ピョ": "pyo",
|
||||||
// Extended combinations
|
|
||||||
"ティ": "ti", "ディ": "di", "トゥ": "tu", "ドゥ": "du",
|
"ティ": "ti", "ディ": "di", "トゥ": "tu", "ドゥ": "du",
|
||||||
"ファ": "fa", "フィ": "fi", "フェ": "fe", "フォ": "fo",
|
"ファ": "fa", "フィ": "fi", "フェ": "fe", "フォ": "fo",
|
||||||
"ウィ": "wi", "ウェ": "we", "ウォ": "wo",
|
"ウィ": "wi", "ウェ": "we", "ウォ": "wo",
|
||||||
@@ -120,7 +112,6 @@ func JapaneseToRomaji(text string) string {
|
|||||||
i := 0
|
i := 0
|
||||||
|
|
||||||
for i < len(runes) {
|
for i < len(runes) {
|
||||||
// Check for っ/ッ (double consonant)
|
|
||||||
if i < len(runes)-1 && (runes[i] == 'っ' || runes[i] == 'ッ') {
|
if i < len(runes)-1 && (runes[i] == 'っ' || runes[i] == 'ッ') {
|
||||||
nextRomaji := ""
|
nextRomaji := ""
|
||||||
if romaji, ok := hiraganaToRomaji[runes[i+1]]; ok {
|
if romaji, ok := hiraganaToRomaji[runes[i+1]]; ok {
|
||||||
@@ -129,13 +120,12 @@ func JapaneseToRomaji(text string) string {
|
|||||||
nextRomaji = romaji
|
nextRomaji = romaji
|
||||||
}
|
}
|
||||||
if len(nextRomaji) > 0 {
|
if len(nextRomaji) > 0 {
|
||||||
result.WriteByte(nextRomaji[0]) // Double the first consonant
|
result.WriteByte(nextRomaji[0])
|
||||||
}
|
}
|
||||||
i++
|
i++
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check for two-character combinations
|
|
||||||
if i < len(runes)-1 {
|
if i < len(runes)-1 {
|
||||||
combo := string(runes[i : i+2])
|
combo := string(runes[i : i+2])
|
||||||
if romaji, ok := combinationHiragana[combo]; ok {
|
if romaji, ok := combinationHiragana[combo]; ok {
|
||||||
@@ -150,17 +140,14 @@ func JapaneseToRomaji(text string) string {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Single character conversion
|
|
||||||
r := runes[i]
|
r := runes[i]
|
||||||
if romaji, ok := hiraganaToRomaji[r]; ok {
|
if romaji, ok := hiraganaToRomaji[r]; ok {
|
||||||
result.WriteString(romaji)
|
result.WriteString(romaji)
|
||||||
} else if romaji, ok := katakanaToRomaji[r]; ok {
|
} else if romaji, ok := katakanaToRomaji[r]; ok {
|
||||||
result.WriteString(romaji)
|
result.WriteString(romaji)
|
||||||
} else if isKanji(r) {
|
} else if isKanji(r) {
|
||||||
// Keep kanji as-is (would need dictionary for proper conversion)
|
|
||||||
result.WriteRune(r)
|
result.WriteRune(r)
|
||||||
} else {
|
} else {
|
||||||
// Keep other characters (punctuation, spaces, etc.)
|
|
||||||
result.WriteRune(r)
|
result.WriteRune(r)
|
||||||
}
|
}
|
||||||
i++
|
i++
|
||||||
|
|||||||
+212
-444
@@ -87,38 +87,210 @@ func GetSongLinkRegion() string {
|
|||||||
return region
|
return region
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const resolveAPIURL = "https://api.zarz.moe/v1/resolve"
|
||||||
|
|
||||||
func songLinkBaseURL() string {
|
func songLinkBaseURL() string {
|
||||||
opts := GetNetworkCompatibilityOptions()
|
|
||||||
if opts.AllowHTTP {
|
|
||||||
return "http://api.song.link/v1-alpha.1/links"
|
|
||||||
}
|
|
||||||
return "https://api.song.link/v1-alpha.1/links"
|
return "https://api.song.link/v1-alpha.1/links"
|
||||||
}
|
}
|
||||||
|
|
||||||
func buildSongLinkURLFromTarget(targetURL string, userCountry string) string {
|
// resolveTrackPlatforms resolves a music URL to all platforms.
|
||||||
if userCountry == "" {
|
// Spotify URLs use the resolve API; if that fails, falls back to SongLink.
|
||||||
userCountry = GetSongLinkRegion()
|
// All other URLs go directly to SongLink.
|
||||||
|
func (s *SongLinkClient) resolveTrackPlatforms(inputURL string) (map[string]songLinkPlatformLink, error) {
|
||||||
|
if isSpotifyURL(inputURL) {
|
||||||
|
payload, err := json.Marshal(map[string]string{"url": inputURL})
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to encode resolve request: %w", err)
|
||||||
|
}
|
||||||
|
links, err := s.doResolveRequest(payload)
|
||||||
|
if err == nil {
|
||||||
|
return links, nil
|
||||||
|
}
|
||||||
|
GoLog("[SongLink] Resolve proxy failed for %s: %v, falling back to SongLink", inputURL, err)
|
||||||
|
return s.songLinkByTargetURL(inputURL)
|
||||||
}
|
}
|
||||||
apiURL := fmt.Sprintf("%s?url=%s", songLinkBaseURL(), url.QueryEscape(targetURL))
|
return s.songLinkByTargetURL(inputURL)
|
||||||
if userCountry != "" {
|
|
||||||
apiURL = fmt.Sprintf("%s&userCountry=%s", apiURL, url.QueryEscape(userCountry))
|
|
||||||
}
|
|
||||||
return apiURL
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func buildSongLinkURLByPlatform(platform, entityType, entityID, userCountry string) string {
|
// resolveTrackPlatformsByPlatform resolves using platform + type + id.
|
||||||
if userCountry == "" {
|
// Spotify uses the resolve API with SongLink fallback; all other platforms use SongLink directly.
|
||||||
userCountry = GetSongLinkRegion()
|
func (s *SongLinkClient) resolveTrackPlatformsByPlatform(platform, entityType, entityID string) (map[string]songLinkPlatformLink, error) {
|
||||||
|
if strings.EqualFold(platform, "spotify") {
|
||||||
|
payload, err := json.Marshal(map[string]string{
|
||||||
|
"platform": platform,
|
||||||
|
"type": entityType,
|
||||||
|
"id": entityID,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to encode resolve request: %w", err)
|
||||||
|
}
|
||||||
|
links, err := s.doResolveRequest(payload)
|
||||||
|
if err == nil {
|
||||||
|
return links, nil
|
||||||
|
}
|
||||||
|
GoLog("[SongLink] Resolve proxy failed for %s/%s/%s: %v, falling back to SongLink", platform, entityType, entityID, err)
|
||||||
|
return s.songLinkByPlatform(platform, entityType, entityID)
|
||||||
}
|
}
|
||||||
apiURL := fmt.Sprintf("%s?platform=%s&type=%s&id=%s",
|
return s.songLinkByPlatform(platform, entityType, entityID)
|
||||||
|
}
|
||||||
|
|
||||||
|
func isSpotifyURL(u string) bool {
|
||||||
|
lower := strings.ToLower(u)
|
||||||
|
return strings.Contains(lower, "spotify.com/") || strings.Contains(lower, "spotify:")
|
||||||
|
}
|
||||||
|
|
||||||
|
// doResolveRequest sends a JSON payload to the resolve API (api.zarz.moe)
|
||||||
|
// and parses the response into a platform link map.
|
||||||
|
func (s *SongLinkClient) doResolveRequest(payload []byte) (map[string]songLinkPlatformLink, error) {
|
||||||
|
req, err := http.NewRequest("POST", resolveAPIURL, bytes.NewReader(payload))
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to create resolve request: %w", err)
|
||||||
|
}
|
||||||
|
req.Header.Set("Content-Type", "application/json")
|
||||||
|
|
||||||
|
resp, err := s.client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("resolve API request failed: %w", err)
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
if resp.StatusCode != 200 {
|
||||||
|
return nil, fmt.Errorf("resolve API returned status %d", resp.StatusCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
body, err := ReadResponseBody(resp)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to read resolve response: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var resolveResp struct {
|
||||||
|
Success bool `json:"success"`
|
||||||
|
ISRC string `json:"isrc"`
|
||||||
|
SongUrls map[string]json.RawMessage `json:"songUrls"`
|
||||||
|
}
|
||||||
|
if err := json.Unmarshal(body, &resolveResp); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to decode resolve response: %w", err)
|
||||||
|
}
|
||||||
|
if !resolveResp.Success {
|
||||||
|
return nil, fmt.Errorf("resolve API returned success=false")
|
||||||
|
}
|
||||||
|
|
||||||
|
keyMap := map[string]string{
|
||||||
|
"Spotify": "spotify",
|
||||||
|
"Deezer": "deezer",
|
||||||
|
"Tidal": "tidal",
|
||||||
|
"YouTubeMusic": "youtubeMusic",
|
||||||
|
"YouTube": "youtube",
|
||||||
|
"AmazonMusic": "amazonMusic",
|
||||||
|
"Qobuz": "qobuz",
|
||||||
|
"AppleMusic": "appleMusic",
|
||||||
|
}
|
||||||
|
|
||||||
|
links := make(map[string]songLinkPlatformLink)
|
||||||
|
for resolveKey, platformKey := range keyMap {
|
||||||
|
rawValue, ok := resolveResp.SongUrls[resolveKey]
|
||||||
|
if !ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if u := extractResolveURLValue(rawValue); u != "" {
|
||||||
|
links[platformKey] = songLinkPlatformLink{URL: u}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(links) == 0 {
|
||||||
|
return nil, fmt.Errorf("resolve API returned no platform links")
|
||||||
|
}
|
||||||
|
|
||||||
|
return links, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func extractResolveURLValue(raw json.RawMessage) string {
|
||||||
|
trimmed := bytes.TrimSpace(raw)
|
||||||
|
if len(trimmed) == 0 || bytes.Equal(trimmed, []byte("null")) {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
var direct string
|
||||||
|
if err := json.Unmarshal(trimmed, &direct); err == nil {
|
||||||
|
return strings.TrimSpace(direct)
|
||||||
|
}
|
||||||
|
|
||||||
|
var list []string
|
||||||
|
if err := json.Unmarshal(trimmed, &list); err == nil {
|
||||||
|
for _, candidate := range list {
|
||||||
|
if cleaned := strings.TrimSpace(candidate); cleaned != "" {
|
||||||
|
return cleaned
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// songLinkByTargetURL calls the SongLink API with a target URL (for non-Spotify URLs).
|
||||||
|
func (s *SongLinkClient) songLinkByTargetURL(targetURL string) (map[string]songLinkPlatformLink, error) {
|
||||||
|
songLinkRateLimiter.WaitForSlot()
|
||||||
|
|
||||||
|
apiURL := fmt.Sprintf("%s?url=%s&userCountry=%s",
|
||||||
|
songLinkBaseURL(),
|
||||||
|
url.QueryEscape(targetURL),
|
||||||
|
url.QueryEscape(GetSongLinkRegion()))
|
||||||
|
|
||||||
|
return s.doSongLinkRequest(apiURL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// songLinkByPlatform calls the SongLink API with platform + type + id (for non-Spotify platforms).
|
||||||
|
func (s *SongLinkClient) songLinkByPlatform(platform, entityType, entityID string) (map[string]songLinkPlatformLink, error) {
|
||||||
|
songLinkRateLimiter.WaitForSlot()
|
||||||
|
|
||||||
|
apiURL := fmt.Sprintf("%s?platform=%s&type=%s&id=%s&userCountry=%s",
|
||||||
songLinkBaseURL(),
|
songLinkBaseURL(),
|
||||||
url.QueryEscape(platform),
|
url.QueryEscape(platform),
|
||||||
url.QueryEscape(entityType),
|
url.QueryEscape(entityType),
|
||||||
url.QueryEscape(entityID))
|
url.QueryEscape(entityID),
|
||||||
if userCountry != "" {
|
url.QueryEscape(GetSongLinkRegion()))
|
||||||
apiURL = fmt.Sprintf("%s&userCountry=%s", apiURL, url.QueryEscape(userCountry))
|
|
||||||
|
return s.doSongLinkRequest(apiURL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// doSongLinkRequest calls the SongLink API and parses the response.
|
||||||
|
func (s *SongLinkClient) doSongLinkRequest(apiURL string) (map[string]songLinkPlatformLink, error) {
|
||||||
|
req, err := http.NewRequest("GET", apiURL, nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to create SongLink request: %w", err)
|
||||||
}
|
}
|
||||||
return apiURL
|
|
||||||
|
retryConfig := songLinkRetryConfig()
|
||||||
|
resp, err := DoRequestWithRetry(s.client, req, retryConfig)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("SongLink request failed: %w", err)
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
if resp.StatusCode == 429 {
|
||||||
|
return nil, fmt.Errorf("SongLink rate limit exceeded")
|
||||||
|
}
|
||||||
|
if resp.StatusCode != 200 {
|
||||||
|
return nil, fmt.Errorf("SongLink returned status %d", resp.StatusCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
body, err := ReadResponseBody(resp)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to read SongLink response: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var songLinkResp struct {
|
||||||
|
LinksByPlatform map[string]songLinkPlatformLink `json:"linksByPlatform"`
|
||||||
|
}
|
||||||
|
if err := json.Unmarshal(body, &songLinkResp); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to decode SongLink response: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(songLinkResp.LinksByPlatform) == 0 {
|
||||||
|
return nil, fmt.Errorf("SongLink returned no platform links")
|
||||||
|
}
|
||||||
|
|
||||||
|
return songLinkResp.LinksByPlatform, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *SongLinkClient) CheckTrackAvailability(spotifyTrackID string, isrc string) (*TrackAvailability, error) {
|
func (s *SongLinkClient) CheckTrackAvailability(spotifyTrackID string, isrc string) (*TrackAvailability, error) {
|
||||||
@@ -136,145 +308,12 @@ func (s *SongLinkClient) CheckTrackAvailability(spotifyTrackID string, isrc stri
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (s *SongLinkClient) checkTrackAvailabilityFromSpotify(spotifyTrackID string) (*TrackAvailability, error) {
|
func (s *SongLinkClient) checkTrackAvailabilityFromSpotify(spotifyTrackID string) (*TrackAvailability, error) {
|
||||||
availability, pageErr := s.checkTrackAvailabilityFromSpotifyPage(spotifyTrackID)
|
|
||||||
if pageErr == nil {
|
|
||||||
return availability, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if !songLinkRateLimiter.TryAcquire() {
|
|
||||||
return nil, fmt.Errorf("song.link page lookup failed: %w (SongLink local rate limit exceeded)", pageErr)
|
|
||||||
}
|
|
||||||
|
|
||||||
spotifyURL := fmt.Sprintf("https://open.spotify.com/track/%s", spotifyTrackID)
|
spotifyURL := fmt.Sprintf("https://open.spotify.com/track/%s", spotifyTrackID)
|
||||||
apiURL := buildSongLinkURLFromTarget(spotifyURL, "")
|
links, err := s.resolveTrackPlatforms(spotifyURL)
|
||||||
|
|
||||||
req, err := http.NewRequest("GET", apiURL, nil)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("failed to create request: %w", err)
|
return nil, fmt.Errorf("resolve proxy failed for Spotify %s: %w", spotifyTrackID, err)
|
||||||
}
|
}
|
||||||
|
return buildTrackAvailabilityFromSongLinkLinks(spotifyTrackID, links), nil
|
||||||
retryConfig := songLinkRetryConfig()
|
|
||||||
resp, err := DoRequestWithRetry(s.client, req, retryConfig)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("song.link page lookup failed: %w; SongLink API lookup failed: %w", pageErr, err)
|
|
||||||
}
|
|
||||||
defer resp.Body.Close()
|
|
||||||
|
|
||||||
if resp.StatusCode == 400 {
|
|
||||||
return nil, fmt.Errorf("track not found on SongLink (invalid Spotify ID or track unavailable)")
|
|
||||||
}
|
|
||||||
if resp.StatusCode == 404 {
|
|
||||||
return nil, fmt.Errorf("track not found on any streaming platform")
|
|
||||||
}
|
|
||||||
if resp.StatusCode == 429 {
|
|
||||||
return nil, fmt.Errorf("song.link page lookup failed: %w; SongLink API rate limit exceeded", pageErr)
|
|
||||||
}
|
|
||||||
if resp.StatusCode != 200 {
|
|
||||||
return nil, fmt.Errorf("song.link page lookup failed: %w; SongLink API returned status %d", pageErr, resp.StatusCode)
|
|
||||||
}
|
|
||||||
|
|
||||||
body, err := ReadResponseBody(resp)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to read response: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
var songLinkResp struct {
|
|
||||||
LinksByPlatform map[string]songLinkPlatformLink `json:"linksByPlatform"`
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := json.Unmarshal(body, &songLinkResp); err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to decode response: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
LogWarn("SongLink", "Spotify %s resolved via SongLink API after song.link page failure: %v", spotifyTrackID, pageErr)
|
|
||||||
return buildTrackAvailabilityFromSongLinkLinks(spotifyTrackID, songLinkResp.LinksByPlatform), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *SongLinkClient) checkTrackAvailabilityFromSpotifyPage(spotifyTrackID string) (*TrackAvailability, error) {
|
|
||||||
pageURL := fmt.Sprintf("https://song.link/s/%s", spotifyTrackID)
|
|
||||||
req, err := http.NewRequest("GET", pageURL, nil)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to create song.link page request: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
req.Header.Set("Accept", "text/html,application/xhtml+xml")
|
|
||||||
req.Header.Set("User-Agent", getRandomUserAgent())
|
|
||||||
|
|
||||||
resp, err := s.client.Do(req)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to fetch song.link page: %w", err)
|
|
||||||
}
|
|
||||||
defer resp.Body.Close()
|
|
||||||
|
|
||||||
if resp.StatusCode == 404 {
|
|
||||||
return nil, fmt.Errorf("track not found on song.link page")
|
|
||||||
}
|
|
||||||
if resp.StatusCode != 200 {
|
|
||||||
return nil, fmt.Errorf("song.link page returned status %d", resp.StatusCode)
|
|
||||||
}
|
|
||||||
|
|
||||||
body, err := ReadResponseBody(resp)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to read song.link page: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
nextDataJSON, err := extractSongLinkNextDataJSON(body)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var pageData struct {
|
|
||||||
Props struct {
|
|
||||||
PageProps struct {
|
|
||||||
PageData struct {
|
|
||||||
Sections []struct {
|
|
||||||
Links []struct {
|
|
||||||
Platform string `json:"platform"`
|
|
||||||
URL string `json:"url"`
|
|
||||||
Show bool `json:"show"`
|
|
||||||
} `json:"links"`
|
|
||||||
} `json:"sections"`
|
|
||||||
} `json:"pageData"`
|
|
||||||
} `json:"pageProps"`
|
|
||||||
} `json:"props"`
|
|
||||||
}
|
|
||||||
if err := json.Unmarshal(nextDataJSON, &pageData); err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to decode song.link page data: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
linksByPlatform := make(map[string]songLinkPlatformLink)
|
|
||||||
for _, section := range pageData.Props.PageProps.PageData.Sections {
|
|
||||||
for _, link := range section.Links {
|
|
||||||
if !link.Show || strings.TrimSpace(link.URL) == "" {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
linksByPlatform[link.Platform] = songLinkPlatformLink{URL: link.URL}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(linksByPlatform) == 0 {
|
|
||||||
return nil, fmt.Errorf("song.link page contained no usable platform links")
|
|
||||||
}
|
|
||||||
|
|
||||||
return buildTrackAvailabilityFromSongLinkLinks(spotifyTrackID, linksByPlatform), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func extractSongLinkNextDataJSON(body []byte) ([]byte, error) {
|
|
||||||
const startMarker = `<script id="__NEXT_DATA__" type="application/json">`
|
|
||||||
const endMarker = `</script>`
|
|
||||||
|
|
||||||
start := bytes.Index(body, []byte(startMarker))
|
|
||||||
if start < 0 {
|
|
||||||
return nil, fmt.Errorf("song.link page missing __NEXT_DATA__")
|
|
||||||
}
|
|
||||||
start += len(startMarker)
|
|
||||||
|
|
||||||
end := bytes.Index(body[start:], []byte(endMarker))
|
|
||||||
if end < 0 {
|
|
||||||
return nil, fmt.Errorf("song.link page has unterminated __NEXT_DATA__")
|
|
||||||
}
|
|
||||||
|
|
||||||
return body[start : start+end], nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *SongLinkClient) checkTrackAvailabilityFromISRC(isrc string) (*TrackAvailability, error) {
|
func (s *SongLinkClient) checkTrackAvailabilityFromISRC(isrc string) (*TrackAvailability, error) {
|
||||||
@@ -469,8 +508,6 @@ func extractYouTubeIDFromURL(youtubeURL string) string {
|
|||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
// isNumeric is defined in library_scan.go
|
|
||||||
|
|
||||||
func (s *SongLinkClient) GetDeezerIDFromSpotify(spotifyTrackID string) (string, error) {
|
func (s *SongLinkClient) GetDeezerIDFromSpotify(spotifyTrackID string) (string, error) {
|
||||||
availability, err := s.CheckTrackAvailability(spotifyTrackID, "")
|
availability, err := s.CheckTrackAvailability(spotifyTrackID, "")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -505,47 +542,17 @@ type AlbumAvailability struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (s *SongLinkClient) CheckAlbumAvailability(spotifyAlbumID string) (*AlbumAvailability, error) {
|
func (s *SongLinkClient) CheckAlbumAvailability(spotifyAlbumID string) (*AlbumAvailability, error) {
|
||||||
songLinkRateLimiter.WaitForSlot()
|
|
||||||
|
|
||||||
spotifyURL := fmt.Sprintf("https://open.spotify.com/album/%s", spotifyAlbumID)
|
spotifyURL := fmt.Sprintf("https://open.spotify.com/album/%s", spotifyAlbumID)
|
||||||
apiURL := buildSongLinkURLFromTarget(spotifyURL, "")
|
links, err := s.resolveTrackPlatforms(spotifyURL)
|
||||||
|
|
||||||
req, err := http.NewRequest("GET", apiURL, nil)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("failed to create request: %w", err)
|
return nil, fmt.Errorf("resolve proxy failed for album %s: %w", spotifyAlbumID, err)
|
||||||
}
|
|
||||||
|
|
||||||
retryConfig := songLinkRetryConfig()
|
|
||||||
resp, err := DoRequestWithRetry(s.client, req, retryConfig)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to check album availability: %w", err)
|
|
||||||
}
|
|
||||||
defer resp.Body.Close()
|
|
||||||
|
|
||||||
if resp.StatusCode != 200 {
|
|
||||||
return nil, fmt.Errorf("API returned status %d", resp.StatusCode)
|
|
||||||
}
|
|
||||||
|
|
||||||
body, err := ReadResponseBody(resp)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to read response: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
var songLinkResp struct {
|
|
||||||
LinksByPlatform map[string]struct {
|
|
||||||
URL string `json:"url"`
|
|
||||||
} `json:"linksByPlatform"`
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := json.Unmarshal(body, &songLinkResp); err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to decode response: %w", err)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
availability := &AlbumAvailability{
|
availability := &AlbumAvailability{
|
||||||
SpotifyID: spotifyAlbumID,
|
SpotifyID: spotifyAlbumID,
|
||||||
}
|
}
|
||||||
|
|
||||||
if deezerLink, ok := songLinkResp.LinksByPlatform["deezer"]; ok && deezerLink.URL != "" {
|
if deezerLink, ok := links["deezer"]; ok && deezerLink.URL != "" {
|
||||||
availability.Deezer = true
|
availability.Deezer = true
|
||||||
availability.DeezerURL = deezerLink.URL
|
availability.DeezerURL = deezerLink.URL
|
||||||
availability.DeezerID = extractDeezerIDFromURL(deezerLink.URL)
|
availability.DeezerID = extractDeezerIDFromURL(deezerLink.URL)
|
||||||
@@ -588,101 +595,19 @@ func (s *SongLinkClient) CheckAvailabilityFromDeezer(deezerTrackID string) (*Tra
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (s *SongLinkClient) checkAvailabilityFromDeezerSongLink(deezerTrackID string) (*TrackAvailability, error) {
|
func (s *SongLinkClient) checkAvailabilityFromDeezerSongLink(deezerTrackID string) (*TrackAvailability, error) {
|
||||||
songLinkRateLimiter.WaitForSlot()
|
|
||||||
|
|
||||||
deezerURL := fmt.Sprintf("https://www.deezer.com/track/%s", deezerTrackID)
|
deezerURL := fmt.Sprintf("https://www.deezer.com/track/%s", deezerTrackID)
|
||||||
apiURL := buildSongLinkURLFromTarget(deezerURL, "")
|
links, err := s.resolveTrackPlatforms(deezerURL)
|
||||||
|
|
||||||
req, err := http.NewRequest("GET", apiURL, nil)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("failed to create request: %w", err)
|
return nil, fmt.Errorf("resolve failed for Deezer %s: %w", deezerTrackID, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
retryConfig := songLinkRetryConfig()
|
availability := buildTrackAvailabilityFromSongLinkLinks("", links)
|
||||||
resp, err := DoRequestWithRetry(s.client, req, retryConfig)
|
// Ensure Deezer is always marked available since we started from a Deezer URL
|
||||||
if err != nil {
|
availability.Deezer = true
|
||||||
return nil, fmt.Errorf("failed to check availability: %w", err)
|
availability.DeezerID = deezerTrackID
|
||||||
|
if availability.DeezerURL == "" {
|
||||||
|
availability.DeezerURL = deezerURL
|
||||||
}
|
}
|
||||||
defer resp.Body.Close()
|
|
||||||
|
|
||||||
if resp.StatusCode == 400 {
|
|
||||||
return nil, fmt.Errorf("track not found on SongLink (invalid Deezer ID)")
|
|
||||||
}
|
|
||||||
if resp.StatusCode == 404 {
|
|
||||||
return nil, fmt.Errorf("track not found on any streaming platform")
|
|
||||||
}
|
|
||||||
if resp.StatusCode == 429 {
|
|
||||||
return nil, fmt.Errorf("SongLink rate limit exceeded")
|
|
||||||
}
|
|
||||||
if resp.StatusCode != 200 {
|
|
||||||
return nil, fmt.Errorf("SongLink API returned status %d", resp.StatusCode)
|
|
||||||
}
|
|
||||||
|
|
||||||
body, err := ReadResponseBody(resp)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to read response: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
var songLinkResp struct {
|
|
||||||
LinksByPlatform map[string]struct {
|
|
||||||
URL string `json:"url"`
|
|
||||||
} `json:"linksByPlatform"`
|
|
||||||
EntitiesByUniqueId map[string]struct {
|
|
||||||
ID string `json:"id"`
|
|
||||||
Type string `json:"type"`
|
|
||||||
Title string `json:"title"`
|
|
||||||
ArtistName string `json:"artistName"`
|
|
||||||
} `json:"entitiesByUniqueId"`
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := json.Unmarshal(body, &songLinkResp); err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to decode response: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
availability := &TrackAvailability{
|
|
||||||
Deezer: true,
|
|
||||||
DeezerID: deezerTrackID,
|
|
||||||
}
|
|
||||||
|
|
||||||
if spotifyLink, ok := songLinkResp.LinksByPlatform["spotify"]; ok && spotifyLink.URL != "" {
|
|
||||||
availability.SpotifyID = extractSpotifyIDFromURL(spotifyLink.URL)
|
|
||||||
}
|
|
||||||
|
|
||||||
if tidalLink, ok := songLinkResp.LinksByPlatform["tidal"]; ok && tidalLink.URL != "" {
|
|
||||||
availability.Tidal = true
|
|
||||||
availability.TidalURL = tidalLink.URL
|
|
||||||
availability.TidalID = extractTidalIDFromURL(tidalLink.URL)
|
|
||||||
}
|
|
||||||
|
|
||||||
if amazonLink, ok := songLinkResp.LinksByPlatform["amazonMusic"]; ok && amazonLink.URL != "" {
|
|
||||||
availability.Amazon = true
|
|
||||||
availability.AmazonURL = amazonLink.URL
|
|
||||||
}
|
|
||||||
|
|
||||||
if qobuzLink, ok := songLinkResp.LinksByPlatform["qobuz"]; ok && qobuzLink.URL != "" {
|
|
||||||
availability.Qobuz = true
|
|
||||||
availability.QobuzURL = qobuzLink.URL
|
|
||||||
availability.QobuzID = extractQobuzIDFromURL(qobuzLink.URL)
|
|
||||||
}
|
|
||||||
|
|
||||||
if deezerLink, ok := songLinkResp.LinksByPlatform["deezer"]; ok && deezerLink.URL != "" {
|
|
||||||
availability.DeezerURL = deezerLink.URL
|
|
||||||
}
|
|
||||||
|
|
||||||
// Prefer youtubeMusic URLs — they are usually closer to music catalog matches.
|
|
||||||
if ytMusicLink, ok := songLinkResp.LinksByPlatform["youtubeMusic"]; ok && ytMusicLink.URL != "" {
|
|
||||||
availability.YouTube = true
|
|
||||||
availability.YouTubeURL = ytMusicLink.URL
|
|
||||||
availability.YouTubeID = extractYouTubeIDFromURL(ytMusicLink.URL)
|
|
||||||
}
|
|
||||||
if !availability.YouTube {
|
|
||||||
if youtubeLink, ok := songLinkResp.LinksByPlatform["youtube"]; ok && youtubeLink.URL != "" {
|
|
||||||
availability.YouTube = true
|
|
||||||
availability.YouTubeURL = youtubeLink.URL
|
|
||||||
availability.YouTubeID = extractYouTubeIDFromURL(youtubeLink.URL)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return availability, nil
|
return availability, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -694,94 +619,12 @@ func (s *SongLinkClient) CheckAvailabilityByPlatform(platform, entityType, entit
|
|||||||
return nil, fmt.Errorf("%s ID is empty", platform)
|
return nil, fmt.Errorf("%s ID is empty", platform)
|
||||||
}
|
}
|
||||||
|
|
||||||
songLinkRateLimiter.WaitForSlot()
|
links, err := s.resolveTrackPlatformsByPlatform(platform, entityType, entityID)
|
||||||
|
|
||||||
apiURL := buildSongLinkURLByPlatform(platform, entityType, entityID, "")
|
|
||||||
|
|
||||||
req, err := http.NewRequest("GET", apiURL, nil)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("failed to create request: %w", err)
|
return nil, fmt.Errorf("resolve failed for %s %s: %w", platform, entityID, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
retryConfig := songLinkRetryConfig()
|
return buildTrackAvailabilityFromSongLinkLinks("", links), nil
|
||||||
resp, err := DoRequestWithRetry(s.client, req, retryConfig)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to check availability: %w", err)
|
|
||||||
}
|
|
||||||
defer resp.Body.Close()
|
|
||||||
|
|
||||||
if resp.StatusCode == 400 {
|
|
||||||
return nil, fmt.Errorf("track not found on SongLink (invalid %s ID)", platform)
|
|
||||||
}
|
|
||||||
if resp.StatusCode == 404 {
|
|
||||||
return nil, fmt.Errorf("track not found on any streaming platform")
|
|
||||||
}
|
|
||||||
if resp.StatusCode == 429 {
|
|
||||||
return nil, fmt.Errorf("SongLink rate limit exceeded")
|
|
||||||
}
|
|
||||||
if resp.StatusCode != 200 {
|
|
||||||
return nil, fmt.Errorf("SongLink API returned status %d", resp.StatusCode)
|
|
||||||
}
|
|
||||||
|
|
||||||
body, err := ReadResponseBody(resp)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to read response: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
var songLinkResp struct {
|
|
||||||
LinksByPlatform map[string]struct {
|
|
||||||
URL string `json:"url"`
|
|
||||||
} `json:"linksByPlatform"`
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := json.Unmarshal(body, &songLinkResp); err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to decode response: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
availability := &TrackAvailability{}
|
|
||||||
|
|
||||||
if spotifyLink, ok := songLinkResp.LinksByPlatform["spotify"]; ok && spotifyLink.URL != "" {
|
|
||||||
availability.SpotifyID = extractSpotifyIDFromURL(spotifyLink.URL)
|
|
||||||
}
|
|
||||||
|
|
||||||
if tidalLink, ok := songLinkResp.LinksByPlatform["tidal"]; ok && tidalLink.URL != "" {
|
|
||||||
availability.Tidal = true
|
|
||||||
availability.TidalURL = tidalLink.URL
|
|
||||||
availability.TidalID = extractTidalIDFromURL(tidalLink.URL)
|
|
||||||
}
|
|
||||||
|
|
||||||
if amazonLink, ok := songLinkResp.LinksByPlatform["amazonMusic"]; ok && amazonLink.URL != "" {
|
|
||||||
availability.Amazon = true
|
|
||||||
availability.AmazonURL = amazonLink.URL
|
|
||||||
}
|
|
||||||
|
|
||||||
if qobuzLink, ok := songLinkResp.LinksByPlatform["qobuz"]; ok && qobuzLink.URL != "" {
|
|
||||||
availability.Qobuz = true
|
|
||||||
availability.QobuzURL = qobuzLink.URL
|
|
||||||
availability.QobuzID = extractQobuzIDFromURL(qobuzLink.URL)
|
|
||||||
}
|
|
||||||
|
|
||||||
if deezerLink, ok := songLinkResp.LinksByPlatform["deezer"]; ok && deezerLink.URL != "" {
|
|
||||||
availability.Deezer = true
|
|
||||||
availability.DeezerURL = deezerLink.URL
|
|
||||||
availability.DeezerID = extractDeezerIDFromURL(deezerLink.URL)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Prefer youtubeMusic URLs — they are usually closer to music catalog matches.
|
|
||||||
if ytMusicLink, ok := songLinkResp.LinksByPlatform["youtubeMusic"]; ok && ytMusicLink.URL != "" {
|
|
||||||
availability.YouTube = true
|
|
||||||
availability.YouTubeURL = ytMusicLink.URL
|
|
||||||
availability.YouTubeID = extractYouTubeIDFromURL(ytMusicLink.URL)
|
|
||||||
}
|
|
||||||
if !availability.YouTube {
|
|
||||||
if youtubeLink, ok := songLinkResp.LinksByPlatform["youtube"]; ok && youtubeLink.URL != "" {
|
|
||||||
availability.YouTube = true
|
|
||||||
availability.YouTubeURL = youtubeLink.URL
|
|
||||||
availability.YouTubeID = extractYouTubeIDFromURL(youtubeLink.URL)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return availability, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func buildTrackAvailabilityFromSongLinkLinks(spotifyTrackID string, links map[string]songLinkPlatformLink) *TrackAvailability {
|
func buildTrackAvailabilityFromSongLinkLinks(spotifyTrackID string, links map[string]songLinkPlatformLink) *TrackAvailability {
|
||||||
@@ -894,85 +737,10 @@ func (s *SongLinkClient) GetYouTubeURLFromDeezer(deezerTrackID string) (string,
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (s *SongLinkClient) CheckAvailabilityFromURL(inputURL string) (*TrackAvailability, error) {
|
func (s *SongLinkClient) CheckAvailabilityFromURL(inputURL string) (*TrackAvailability, error) {
|
||||||
songLinkRateLimiter.WaitForSlot()
|
links, err := s.resolveTrackPlatforms(inputURL)
|
||||||
|
|
||||||
apiURL := buildSongLinkURLFromTarget(inputURL, "")
|
|
||||||
|
|
||||||
req, err := http.NewRequest("GET", apiURL, nil)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("failed to create request: %w", err)
|
return nil, fmt.Errorf("resolve failed for URL %s: %w", inputURL, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
retryConfig := songLinkRetryConfig()
|
return buildTrackAvailabilityFromSongLinkLinks("", links), nil
|
||||||
resp, err := DoRequestWithRetry(s.client, req, retryConfig)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to check availability: %w", err)
|
|
||||||
}
|
|
||||||
defer resp.Body.Close()
|
|
||||||
|
|
||||||
if resp.StatusCode == 400 || resp.StatusCode == 404 {
|
|
||||||
return nil, fmt.Errorf("track not found on SongLink")
|
|
||||||
}
|
|
||||||
if resp.StatusCode == 429 {
|
|
||||||
return nil, fmt.Errorf("SongLink rate limit exceeded")
|
|
||||||
}
|
|
||||||
if resp.StatusCode != 200 {
|
|
||||||
return nil, fmt.Errorf("SongLink API returned status %d", resp.StatusCode)
|
|
||||||
}
|
|
||||||
|
|
||||||
body, err := ReadResponseBody(resp)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to read response: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
var songLinkResp struct {
|
|
||||||
LinksByPlatform map[string]struct {
|
|
||||||
URL string `json:"url"`
|
|
||||||
EntityID string `json:"entityUniqueId"`
|
|
||||||
} `json:"linksByPlatform"`
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := json.Unmarshal(body, &songLinkResp); err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to decode response: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
availability := &TrackAvailability{}
|
|
||||||
|
|
||||||
if spotifyLink, ok := songLinkResp.LinksByPlatform["spotify"]; ok && spotifyLink.URL != "" {
|
|
||||||
availability.SpotifyID = extractSpotifyIDFromURL(spotifyLink.URL)
|
|
||||||
}
|
|
||||||
if tidalLink, ok := songLinkResp.LinksByPlatform["tidal"]; ok && tidalLink.URL != "" {
|
|
||||||
availability.Tidal = true
|
|
||||||
availability.TidalURL = tidalLink.URL
|
|
||||||
availability.TidalID = extractTidalIDFromURL(tidalLink.URL)
|
|
||||||
}
|
|
||||||
if amazonLink, ok := songLinkResp.LinksByPlatform["amazonMusic"]; ok && amazonLink.URL != "" {
|
|
||||||
availability.Amazon = true
|
|
||||||
availability.AmazonURL = amazonLink.URL
|
|
||||||
}
|
|
||||||
if qobuzLink, ok := songLinkResp.LinksByPlatform["qobuz"]; ok && qobuzLink.URL != "" {
|
|
||||||
availability.Qobuz = true
|
|
||||||
availability.QobuzURL = qobuzLink.URL
|
|
||||||
availability.QobuzID = extractQobuzIDFromURL(qobuzLink.URL)
|
|
||||||
}
|
|
||||||
if deezerLink, ok := songLinkResp.LinksByPlatform["deezer"]; ok && deezerLink.URL != "" {
|
|
||||||
availability.Deezer = true
|
|
||||||
availability.DeezerURL = deezerLink.URL
|
|
||||||
availability.DeezerID = extractDeezerIDFromURL(deezerLink.URL)
|
|
||||||
}
|
|
||||||
// Prefer youtubeMusic URLs — they are usually closer to music catalog matches.
|
|
||||||
if ytMusicLink, ok := songLinkResp.LinksByPlatform["youtubeMusic"]; ok && ytMusicLink.URL != "" {
|
|
||||||
availability.YouTube = true
|
|
||||||
availability.YouTubeURL = ytMusicLink.URL
|
|
||||||
availability.YouTubeID = extractYouTubeIDFromURL(ytMusicLink.URL)
|
|
||||||
}
|
|
||||||
if !availability.YouTube {
|
|
||||||
if youtubeLink, ok := songLinkResp.LinksByPlatform["youtube"]; ok && youtubeLink.URL != "" {
|
|
||||||
availability.YouTube = true
|
|
||||||
availability.YouTubeURL = youtubeLink.URL
|
|
||||||
availability.YouTubeID = extractYouTubeIDFromURL(youtubeLink.URL)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return availability, nil
|
|
||||||
}
|
}
|
||||||
|
|||||||
+108
-36
@@ -23,26 +23,24 @@ func TestGetRetryAfterDurationMissingHeaderReturnsZero(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestCheckTrackAvailabilityFromSpotifyPrefersSongLinkPage(t *testing.T) {
|
func TestCheckTrackAvailabilityFromSpotifyViaResolveAPI(t *testing.T) {
|
||||||
|
origRetryConfig := songLinkRetryConfig
|
||||||
|
defer func() { songLinkRetryConfig = origRetryConfig }()
|
||||||
|
|
||||||
client := &SongLinkClient{
|
client := &SongLinkClient{
|
||||||
client: &http.Client{
|
client: &http.Client{
|
||||||
Transport: roundTripFunc(func(req *http.Request) (*http.Response, error) {
|
Transport: roundTripFunc(func(req *http.Request) (*http.Response, error) {
|
||||||
switch {
|
if req.URL.Host == "api.zarz.moe" && req.URL.Path == "/v1/resolve" && req.Method == "POST" {
|
||||||
case req.URL.Host == "api.song.link":
|
body := `{"success":true,"isrc":"USRC12345678","songUrls":{"Spotify":"https://open.spotify.com/track/testspotifyid","Deezer":"https://www.deezer.com/track/908604612","AmazonMusic":"https://music.amazon.com/albums/B086Q2QNLH?trackAsin=B086Q41M9C","Tidal":"https://listen.tidal.com/track/134858527","Qobuz":"https://open.qobuz.com/track/195125822","YouTubeMusic":"https://music.youtube.com/watch?v=testvideoid1"}}`
|
||||||
t.Fatalf("api.song.link should not be called when song.link page succeeds")
|
|
||||||
return nil, nil
|
|
||||||
case req.URL.Host == "song.link" && req.URL.Path == "/s/testspotifyid":
|
|
||||||
body := `<!DOCTYPE html><html><body><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{"pageData":{"sections":[{"displayName":"Listen","links":[{"platform":"spotify","url":"https://open.spotify.com/track/testspotifyid","show":true},{"platform":"deezer","url":"https://www.deezer.com/track/908604612","show":true},{"platform":"amazonMusic","url":"https://music.amazon.com/albums/B086Q2QNLH?trackAsin=B086Q41M9C","show":true},{"platform":"tidal","url":"https://listen.tidal.com/track/134858527","show":true},{"platform":"qobuz","url":"https://open.qobuz.com/track/195125822","show":true},{"platform":"youtubeMusic","url":"https://music.youtube.com/watch?v=testvideoid1","show":true}]}]}}}}</script></body></html>`
|
|
||||||
return &http.Response{
|
return &http.Response{
|
||||||
StatusCode: 200,
|
StatusCode: 200,
|
||||||
Header: make(http.Header),
|
Header: make(http.Header),
|
||||||
Body: io.NopCloser(strings.NewReader(body)),
|
Body: io.NopCloser(strings.NewReader(body)),
|
||||||
Request: req,
|
Request: req,
|
||||||
}, nil
|
}, nil
|
||||||
default:
|
|
||||||
t.Fatalf("unexpected request: %s", req.URL.String())
|
|
||||||
return nil, nil
|
|
||||||
}
|
}
|
||||||
|
t.Fatalf("unexpected request: %s %s", req.Method, req.URL.String())
|
||||||
|
return nil, nil
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@@ -66,62 +64,136 @@ func TestCheckTrackAvailabilityFromSpotifyPrefersSongLinkPage(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestCheckTrackAvailabilityFromSpotifyFallsBackToAPIWhenPageFails(t *testing.T) {
|
func TestCheckTrackAvailabilityFromSpotifyResolveAPIFailure(t *testing.T) {
|
||||||
origRetryConfig := songLinkRetryConfig
|
origRetryConfig := songLinkRetryConfig
|
||||||
songLinkRetryConfig = func() RetryConfig {
|
songLinkRetryConfig = func() RetryConfig {
|
||||||
return RetryConfig{
|
return RetryConfig{MaxRetries: 0, InitialDelay: 0, MaxDelay: 0, BackoffFactor: 1}
|
||||||
MaxRetries: 0,
|
|
||||||
InitialDelay: 0,
|
|
||||||
MaxDelay: 0,
|
|
||||||
BackoffFactor: 1,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
defer func() {
|
defer func() { songLinkRetryConfig = origRetryConfig }()
|
||||||
songLinkRetryConfig = origRetryConfig
|
|
||||||
}()
|
var hitSongLink bool
|
||||||
|
|
||||||
client := &SongLinkClient{
|
client := &SongLinkClient{
|
||||||
client: &http.Client{
|
client: &http.Client{
|
||||||
Transport: roundTripFunc(func(req *http.Request) (*http.Response, error) {
|
Transport: roundTripFunc(func(req *http.Request) (*http.Response, error) {
|
||||||
switch {
|
// Resolve proxy returns 500
|
||||||
case req.URL.Host == "song.link" && req.URL.Path == "/s/testspotifyid":
|
if req.URL.Host == "api.zarz.moe" && req.URL.Path == "/v1/resolve" {
|
||||||
return &http.Response{
|
return &http.Response{
|
||||||
StatusCode: 500,
|
StatusCode: 500,
|
||||||
Header: make(http.Header),
|
Header: make(http.Header),
|
||||||
Body: io.NopCloser(strings.NewReader("page failure")),
|
Body: io.NopCloser(strings.NewReader("internal error")),
|
||||||
Request: req,
|
Request: req,
|
||||||
}, nil
|
}, nil
|
||||||
case req.URL.Host == "api.song.link":
|
}
|
||||||
body := `{"linksByPlatform":{"spotify":{"url":"https://open.spotify.com/track/testspotifyid"},"deezer":{"url":"https://www.deezer.com/track/908604612"},"amazonMusic":{"url":"https://music.amazon.com/albums/B086Q2QNLH?trackAsin=B086Q41M9C"},"tidal":{"url":"https://listen.tidal.com/track/134858527"},"qobuz":{"url":"https://open.qobuz.com/track/195125822"},"youtubeMusic":{"url":"https://music.youtube.com/watch?v=testvideoid1"}}}`
|
// SongLink fallback should be called
|
||||||
|
if req.URL.Host == "api.song.link" {
|
||||||
|
hitSongLink = true
|
||||||
|
body := `{"linksByPlatform":{"spotify":{"url":"https://open.spotify.com/track/testspotifyid"},"deezer":{"url":"https://www.deezer.com/track/908604612"},"tidal":{"url":"https://listen.tidal.com/track/134858527"}}}`
|
||||||
return &http.Response{
|
return &http.Response{
|
||||||
StatusCode: 200,
|
StatusCode: 200,
|
||||||
Header: make(http.Header),
|
Header: make(http.Header),
|
||||||
Body: io.NopCloser(strings.NewReader(body)),
|
Body: io.NopCloser(strings.NewReader(body)),
|
||||||
Request: req,
|
Request: req,
|
||||||
}, nil
|
}, nil
|
||||||
default:
|
|
||||||
t.Fatalf("unexpected request: %s", req.URL.String())
|
|
||||||
return nil, nil
|
|
||||||
}
|
}
|
||||||
|
t.Fatalf("unexpected request: %s %s", req.Method, req.URL.String())
|
||||||
|
return nil, nil
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
availability, err := client.CheckTrackAvailability("testspotifyid", "")
|
availability, err := client.CheckTrackAvailability("testspotifyid", "")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("expected SongLink fallback to succeed, got error: %v", err)
|
||||||
|
}
|
||||||
|
if !hitSongLink {
|
||||||
|
t.Fatal("expected fallback request to SongLink API, but it was never called")
|
||||||
|
}
|
||||||
|
if !availability.Deezer || availability.DeezerID != "908604612" {
|
||||||
|
t.Fatalf("Deezer availability via fallback = %+v, want DeezerID 908604612", availability)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCheckTrackAvailabilityFromSpotifyViaResolveAPIMixedSongURLShapes(t *testing.T) {
|
||||||
|
origRetryConfig := songLinkRetryConfig
|
||||||
|
defer func() { songLinkRetryConfig = origRetryConfig }()
|
||||||
|
|
||||||
|
client := &SongLinkClient{
|
||||||
|
client: &http.Client{
|
||||||
|
Transport: roundTripFunc(func(req *http.Request) (*http.Response, error) {
|
||||||
|
if req.URL.Host == "api.zarz.moe" && req.URL.Path == "/v1/resolve" && req.Method == "POST" {
|
||||||
|
body := `{"success":true,"isrc":"TCAHA2367688","songUrls":{"Spotify":"https://open.spotify.com/track/5glgyj6zH0irbNGfukHacv","Deezer":"https://www.deezer.com/track/2248583177","Tidal":"https://tidal.com/browse/track/290565315","AppleMusic":"https://geo.music.apple.com/us/album/example?i=1","YouTubeMusic":null,"YouTube":"https://www.youtube.com/watch?v=wD_e59XUNdQ","AmazonMusic":"https://music.amazon.com/tracks/B0C35TG38Y/?ref=dm_ff_amazonmusic_3p","Beatport":null,"BeatSource":null,"SoundCloud":null,"Qobuz":null,"Other":[]}}`
|
||||||
|
return &http.Response{
|
||||||
|
StatusCode: 200,
|
||||||
|
Header: make(http.Header),
|
||||||
|
Body: io.NopCloser(strings.NewReader(body)),
|
||||||
|
Request: req,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
t.Fatalf("unexpected request: %s %s", req.Method, req.URL.String())
|
||||||
|
return nil, nil
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
availability, err := client.CheckTrackAvailability("5glgyj6zH0irbNGfukHacv", "")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("CheckTrackAvailability() error = %v", err)
|
t.Fatalf("CheckTrackAvailability() error = %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if availability.SpotifyID != "testspotifyid" {
|
if availability.SpotifyID != "5glgyj6zH0irbNGfukHacv" {
|
||||||
t.Fatalf("SpotifyID = %q, want %q", availability.SpotifyID, "testspotifyid")
|
t.Fatalf("SpotifyID = %q, want %q", availability.SpotifyID, "5glgyj6zH0irbNGfukHacv")
|
||||||
}
|
}
|
||||||
if !availability.Deezer || availability.DeezerID != "908604612" {
|
if !availability.Deezer || availability.DeezerID != "2248583177" {
|
||||||
t.Fatalf("Deezer availability = %+v, want DeezerID 908604612", availability)
|
t.Fatalf("Deezer availability = %+v, want DeezerID 2248583177", availability)
|
||||||
}
|
}
|
||||||
if !availability.Amazon || !availability.Tidal || !availability.Qobuz || !availability.YouTube {
|
if !availability.Tidal || availability.TidalID != "290565315" {
|
||||||
t.Fatalf("availability flags = %+v, want Amazon/Tidal/Qobuz/YouTube true", availability)
|
t.Fatalf("Tidal availability = %+v, want TidalID 290565315", availability)
|
||||||
}
|
}
|
||||||
if availability.YouTubeID != "testvideoid1" {
|
if availability.Qobuz {
|
||||||
t.Fatalf("YouTubeID = %q, want %q", availability.YouTubeID, "testvideoid1")
|
t.Fatalf("Qobuz should remain false when resolve response contains null, got %+v", availability)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCheckAvailabilityFromDeezerUsesSongLink(t *testing.T) {
|
||||||
|
origRetryConfig := songLinkRetryConfig
|
||||||
|
songLinkRetryConfig = func() RetryConfig {
|
||||||
|
return RetryConfig{MaxRetries: 0, InitialDelay: 0, MaxDelay: 0, BackoffFactor: 1}
|
||||||
|
}
|
||||||
|
defer func() { songLinkRetryConfig = origRetryConfig }()
|
||||||
|
|
||||||
|
client := &SongLinkClient{
|
||||||
|
client: &http.Client{
|
||||||
|
Transport: roundTripFunc(func(req *http.Request) (*http.Response, error) {
|
||||||
|
// Non-Spotify should go to SongLink, not resolve API
|
||||||
|
if req.URL.Host == "api.zarz.moe" {
|
||||||
|
t.Fatalf("non-Spotify URL should not hit resolve API, got: %s", req.URL.String())
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
if req.URL.Host == "api.song.link" {
|
||||||
|
body := `{"linksByPlatform":{"spotify":{"url":"https://open.spotify.com/track/testid"},"deezer":{"url":"https://www.deezer.com/track/908604612"},"tidal":{"url":"https://listen.tidal.com/track/134858527"},"qobuz":{"url":"https://open.qobuz.com/track/195125822"},"youtubeMusic":{"url":"https://music.youtube.com/watch?v=testvid"}}}`
|
||||||
|
return &http.Response{
|
||||||
|
StatusCode: 200,
|
||||||
|
Header: make(http.Header),
|
||||||
|
Body: io.NopCloser(strings.NewReader(body)),
|
||||||
|
Request: req,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
t.Fatalf("unexpected request: %s %s", req.Method, req.URL.String())
|
||||||
|
return nil, nil
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
availability, err := client.checkAvailabilityFromDeezerSongLink("908604612")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("checkAvailabilityFromDeezerSongLink() error = %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !availability.Deezer || availability.DeezerID != "908604612" {
|
||||||
|
t.Fatalf("Deezer = %+v, want DeezerID 908604612", availability)
|
||||||
|
}
|
||||||
|
if availability.SpotifyID != "testid" {
|
||||||
|
t.Fatalf("SpotifyID = %q, want %q", availability.SpotifyID, "testid")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
+24
-19
@@ -875,8 +875,6 @@ func (t *TidalDownloader) SearchTracks(query string, limit int) ([]ExtTrackMetad
|
|||||||
return results, nil
|
return results, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// SearchAll searches Tidal for tracks, artists, and albums matching the query.
|
|
||||||
// Returns results in the same SearchAllResult format as Deezer's SearchAll.
|
|
||||||
func (t *TidalDownloader) SearchAll(query string, trackLimit, artistLimit int, filter string) (*SearchAllResult, error) {
|
func (t *TidalDownloader) SearchAll(query string, trackLimit, artistLimit int, filter string) (*SearchAllResult, error) {
|
||||||
GoLog("[Tidal] SearchAll: query=%q, trackLimit=%d, artistLimit=%d, filter=%q\n", query, trackLimit, artistLimit, filter)
|
GoLog("[Tidal] SearchAll: query=%q, trackLimit=%d, artistLimit=%d, filter=%q\n", query, trackLimit, artistLimit, filter)
|
||||||
|
|
||||||
@@ -1014,6 +1012,7 @@ func (t *TidalDownloader) GetAlbumMetadata(resourceID string) (*AlbumResponsePay
|
|||||||
}
|
}
|
||||||
|
|
||||||
tracks := make([]AlbumTrackMetadata, 0, len(itemsModule.PagedList.Items))
|
tracks := make([]AlbumTrackMetadata, 0, len(itemsModule.PagedList.Items))
|
||||||
|
totalDiscs := 0
|
||||||
for _, item := range itemsModule.PagedList.Items {
|
for _, item := range itemsModule.PagedList.Items {
|
||||||
track := item.Item
|
track := item.Item
|
||||||
track.Album.ID = headerModule.Album.ID
|
track.Album.ID = headerModule.Album.ID
|
||||||
@@ -1021,8 +1020,14 @@ func (t *TidalDownloader) GetAlbumMetadata(resourceID string) (*AlbumResponsePay
|
|||||||
track.Album.Cover = headerModule.Album.Cover
|
track.Album.Cover = headerModule.Album.Cover
|
||||||
track.Album.ReleaseDate = headerModule.Album.ReleaseDate
|
track.Album.ReleaseDate = headerModule.Album.ReleaseDate
|
||||||
track.Album.URL = headerModule.Album.URL
|
track.Album.URL = headerModule.Album.URL
|
||||||
|
if track.VolumeNumber > totalDiscs {
|
||||||
|
totalDiscs = track.VolumeNumber
|
||||||
|
}
|
||||||
tracks = append(tracks, tidalTrackToAlbumTrackMetadata(&track))
|
tracks = append(tracks, tidalTrackToAlbumTrackMetadata(&track))
|
||||||
}
|
}
|
||||||
|
for i := range tracks {
|
||||||
|
tracks[i].TotalDiscs = totalDiscs
|
||||||
|
}
|
||||||
|
|
||||||
return &AlbumResponsePayload{
|
return &AlbumResponsePayload{
|
||||||
AlbumInfo: tidalAlbumToAlbumInfo(&headerModule.Album),
|
AlbumInfo: tidalAlbumToAlbumInfo(&headerModule.Album),
|
||||||
@@ -1165,7 +1170,6 @@ type tidalAPIResult struct {
|
|||||||
duration time.Duration
|
duration time.Duration
|
||||||
}
|
}
|
||||||
|
|
||||||
// Mobile networks are more unstable, so we use longer timeouts
|
|
||||||
const (
|
const (
|
||||||
tidalAPITimeoutMobile = 25 * time.Second
|
tidalAPITimeoutMobile = 25 * time.Second
|
||||||
tidalMaxRetries = 2
|
tidalMaxRetries = 2
|
||||||
@@ -1211,7 +1215,6 @@ func fetchTidalURLWithRetry(api string, trackID int64, quality string, timeout t
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
// 429 rate limit - wait and retry
|
|
||||||
if resp.StatusCode == 429 {
|
if resp.StatusCode == 429 {
|
||||||
io.Copy(io.Discard, resp.Body)
|
io.Copy(io.Discard, resp.Body)
|
||||||
resp.Body.Close()
|
resp.Body.Close()
|
||||||
@@ -1233,7 +1236,6 @@ func fetchTidalURLWithRetry(api string, trackID int64, quality string, timeout t
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try V2 response format (with manifest)
|
|
||||||
var v2Response TidalAPIResponseV2
|
var v2Response TidalAPIResponseV2
|
||||||
if err := json.Unmarshal(body, &v2Response); err == nil && v2Response.Data.Manifest != "" {
|
if err := json.Unmarshal(body, &v2Response); err == nil && v2Response.Data.Manifest != "" {
|
||||||
if v2Response.Data.AssetPresentation == "PREVIEW" {
|
if v2Response.Data.AssetPresentation == "PREVIEW" {
|
||||||
@@ -1247,7 +1249,6 @@ func fetchTidalURLWithRetry(api string, trackID int64, quality string, timeout t
|
|||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try V1 response format
|
|
||||||
var v1Responses []struct {
|
var v1Responses []struct {
|
||||||
OriginalTrackURL string `json:"OriginalTrackUrl"`
|
OriginalTrackURL string `json:"OriginalTrackUrl"`
|
||||||
}
|
}
|
||||||
@@ -1602,10 +1603,6 @@ func (t *TidalDownloader) downloadFromManifest(ctx context.Context, manifestB64,
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// For DASH format, determine correct M4A path
|
|
||||||
// If outputPath already ends with .m4a, use it directly.
|
|
||||||
// If outputPath ends with .flac, convert .flac to .m4a.
|
|
||||||
// Otherwise (e.g., SAF /proc/self/fd/*), use outputPath as-is.
|
|
||||||
var m4aPath string
|
var m4aPath string
|
||||||
if strings.HasSuffix(outputPath, ".m4a") {
|
if strings.HasSuffix(outputPath, ".m4a") {
|
||||||
m4aPath = outputPath
|
m4aPath = outputPath
|
||||||
@@ -1879,8 +1876,6 @@ func titlesMatch(expectedTitle, foundTitle string) bool {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Emoji/symbol-only titles must be matched strictly to avoid false positives
|
|
||||||
// like mapping "🪐" to "Higher Power".
|
|
||||||
if (!hasAlphaNumericRunes(expectedTitle) || !hasAlphaNumericRunes(foundTitle)) &&
|
if (!hasAlphaNumericRunes(expectedTitle) || !hasAlphaNumericRunes(foundTitle)) &&
|
||||||
strings.TrimSpace(expectedTitle) != "" &&
|
strings.TrimSpace(expectedTitle) != "" &&
|
||||||
strings.TrimSpace(foundTitle) != "" {
|
strings.TrimSpace(foundTitle) != "" {
|
||||||
@@ -2111,7 +2106,6 @@ func resolveTidalTrackForRequest(req DownloadRequest, downloader *TidalDownloade
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Prefer Deezer-based SongLink lookup when DeezerID is available.
|
|
||||||
if req.DeezerID != "" {
|
if req.DeezerID != "" {
|
||||||
GoLog("[%s] Using Deezer ID for SongLink lookup: %s\n", logPrefix, req.DeezerID)
|
GoLog("[%s] Using Deezer ID for SongLink lookup: %s\n", logPrefix, req.DeezerID)
|
||||||
songlink := NewSongLinkClient()
|
songlink := NewSongLinkClient()
|
||||||
@@ -2150,11 +2144,9 @@ func resolveTidalTrackForRequest(req DownloadRequest, downloader *TidalDownloade
|
|||||||
return nil, fmt.Errorf("failed to find tidal track id from request/cache/songlink")
|
return nil, fmt.Errorf("failed to find tidal track id from request/cache/songlink")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Verify the resolved track matches the request.
|
|
||||||
actualTrack, fetchErr := tidalGetPublicTrackFunc(downloader, strconv.FormatInt(trackID, 10))
|
actualTrack, fetchErr := tidalGetPublicTrackFunc(downloader, strconv.FormatInt(trackID, 10))
|
||||||
if fetchErr != nil {
|
if fetchErr != nil {
|
||||||
GoLog("[%s] Warning: could not fetch Tidal track %d for verification: %v\n", logPrefix, trackID, fetchErr)
|
GoLog("[%s] Warning: could not fetch Tidal track %d for verification: %v\n", logPrefix, trackID, fetchErr)
|
||||||
// Continue without verification — better than failing entirely.
|
|
||||||
} else {
|
} else {
|
||||||
providerArtist := actualTrack.Artist.Name
|
providerArtist := actualTrack.Artist.Name
|
||||||
if providerArtist == "" && len(actualTrack.Artists) > 0 {
|
if providerArtist == "" && len(actualTrack.Artists) > 0 {
|
||||||
@@ -2168,7 +2160,6 @@ func resolveTidalTrackForRequest(req DownloadRequest, downloader *TidalDownloade
|
|||||||
SkipNameVerification: resolvedViaSongLink,
|
SkipNameVerification: resolvedViaSongLink,
|
||||||
}
|
}
|
||||||
if !trackMatchesRequest(req, resolved, logPrefix) {
|
if !trackMatchesRequest(req, resolved, logPrefix) {
|
||||||
// Invalidate the cached ID so future requests don't reuse it.
|
|
||||||
if req.ISRC != "" {
|
if req.ISRC != "" {
|
||||||
GetTrackIDCache().SetTidal(req.ISRC, 0)
|
GetTrackIDCache().SetTidal(req.ISRC, 0)
|
||||||
}
|
}
|
||||||
@@ -2178,13 +2169,26 @@ func resolveTidalTrackForRequest(req DownloadRequest, downloader *TidalDownloade
|
|||||||
GoLog("[%s] Track %d verified: '%s - %s' ✓\n", logPrefix, trackID, resolved.ArtistName, resolved.Title)
|
GoLog("[%s] Track %d verified: '%s - %s' ✓\n", logPrefix, trackID, resolved.ArtistName, resolved.Title)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Use track_number / disc_number from the actual Tidal API data when the
|
||||||
|
// request doesn't carry them (e.g. downloads from search results / popular).
|
||||||
|
resolvedTrackNumber := req.TrackNumber
|
||||||
|
resolvedDiscNumber := req.DiscNumber
|
||||||
|
if actualTrack != nil {
|
||||||
|
if resolvedTrackNumber == 0 && actualTrack.TrackNumber > 0 {
|
||||||
|
resolvedTrackNumber = actualTrack.TrackNumber
|
||||||
|
}
|
||||||
|
if resolvedDiscNumber == 0 && actualTrack.VolumeNumber > 0 {
|
||||||
|
resolvedDiscNumber = actualTrack.VolumeNumber
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
track := &TidalTrack{
|
track := &TidalTrack{
|
||||||
ID: trackID,
|
ID: trackID,
|
||||||
Title: strings.TrimSpace(req.TrackName),
|
Title: strings.TrimSpace(req.TrackName),
|
||||||
ISRC: strings.TrimSpace(req.ISRC),
|
ISRC: strings.TrimSpace(req.ISRC),
|
||||||
Duration: expectedDurationSec,
|
Duration: expectedDurationSec,
|
||||||
TrackNumber: req.TrackNumber,
|
TrackNumber: resolvedTrackNumber,
|
||||||
VolumeNumber: req.DiscNumber,
|
VolumeNumber: resolvedDiscNumber,
|
||||||
}
|
}
|
||||||
track.Artist.Name = strings.TrimSpace(req.ArtistName)
|
track.Artist.Name = strings.TrimSpace(req.ArtistName)
|
||||||
track.Album.Title = strings.TrimSpace(req.AlbumName)
|
track.Album.Title = strings.TrimSpace(req.AlbumName)
|
||||||
@@ -2363,10 +2367,12 @@ func downloadFromTidal(req DownloadRequest) (TidalDownloadResult, error) {
|
|||||||
TrackNumber: actualTrackNumber,
|
TrackNumber: actualTrackNumber,
|
||||||
TotalTracks: req.TotalTracks,
|
TotalTracks: req.TotalTracks,
|
||||||
DiscNumber: actualDiscNumber,
|
DiscNumber: actualDiscNumber,
|
||||||
|
TotalDiscs: req.TotalDiscs,
|
||||||
ISRC: track.ISRC,
|
ISRC: track.ISRC,
|
||||||
Genre: req.Genre,
|
Genre: req.Genre,
|
||||||
Label: req.Label,
|
Label: req.Label,
|
||||||
Copyright: req.Copyright,
|
Copyright: req.Copyright,
|
||||||
|
Composer: req.Composer,
|
||||||
}
|
}
|
||||||
|
|
||||||
var coverData []byte
|
var coverData []byte
|
||||||
@@ -2497,7 +2503,6 @@ func parseTidalURL(input string) (string, string, error) {
|
|||||||
|
|
||||||
parts := strings.Split(strings.Trim(parsed.Path, "/"), "/")
|
parts := strings.Split(strings.Trim(parsed.Path, "/"), "/")
|
||||||
|
|
||||||
// Handle /browse/track/123 format
|
|
||||||
if len(parts) > 0 && parts[0] == "browse" {
|
if len(parts) > 0 && parts[0] == "browse" {
|
||||||
parts = parts[1:]
|
parts = parts[1:]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -22,8 +22,6 @@ func writeNormalizedArtistRune(b *strings.Builder, r rune) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// normalizeLooseTitle collapses separators/punctuation so titles like
|
|
||||||
// "Doctor / Cops" and "Doctor _ Cops" can still match.
|
|
||||||
func normalizeLooseTitle(title string) string {
|
func normalizeLooseTitle(title string) string {
|
||||||
trimmed := strings.TrimSpace(strings.ToLower(title))
|
trimmed := strings.TrimSpace(strings.ToLower(title))
|
||||||
if trimmed == "" {
|
if trimmed == "" {
|
||||||
@@ -48,8 +46,6 @@ func normalizeLooseTitle(title string) string {
|
|||||||
return strings.Join(strings.Fields(b.String()), " ")
|
return strings.Join(strings.Fields(b.String()), " ")
|
||||||
}
|
}
|
||||||
|
|
||||||
// normalizeLooseArtistName folds diacritics and common separators so artist
|
|
||||||
// verification is resilient to variants like "Özkent" vs "Ozkent".
|
|
||||||
func normalizeLooseArtistName(name string) string {
|
func normalizeLooseArtistName(name string) string {
|
||||||
trimmed := strings.TrimSpace(strings.ToLower(name))
|
trimmed := strings.TrimSpace(strings.ToLower(name))
|
||||||
if trimmed == "" {
|
if trimmed == "" {
|
||||||
@@ -87,9 +83,6 @@ func hasAlphaNumericRunes(value string) bool {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
// normalizeSymbolOnlyTitle keeps symbol/emoji runes while dropping letters,
|
|
||||||
// digits, spaces and punctuation. This is useful for emoji-only titles such as
|
|
||||||
// "🪐", "🌎" etc, so we can compare them strictly and avoid false matches.
|
|
||||||
func normalizeSymbolOnlyTitle(title string) string {
|
func normalizeSymbolOnlyTitle(title string) string {
|
||||||
trimmed := strings.TrimSpace(strings.ToLower(title))
|
trimmed := strings.TrimSpace(strings.ToLower(title))
|
||||||
if trimmed == "" {
|
if trimmed == "" {
|
||||||
@@ -114,7 +107,6 @@ func normalizeSymbolOnlyTitle(title string) string {
|
|||||||
return b.String()
|
return b.String()
|
||||||
}
|
}
|
||||||
|
|
||||||
// resolvedTrackInfo holds the metadata fetched from a provider for verification.
|
|
||||||
type resolvedTrackInfo struct {
|
type resolvedTrackInfo struct {
|
||||||
Title string
|
Title string
|
||||||
ArtistName string
|
ArtistName string
|
||||||
@@ -123,8 +115,6 @@ type resolvedTrackInfo struct {
|
|||||||
SkipNameVerification bool
|
SkipNameVerification bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// trackMatchesRequest checks whether a resolved track from a provider matches
|
|
||||||
// the original download request. Returns true if the track is a plausible match.
|
|
||||||
func trackMatchesRequest(req DownloadRequest, resolved resolvedTrackInfo, logPrefix string) bool {
|
func trackMatchesRequest(req DownloadRequest, resolved resolvedTrackInfo, logPrefix string) bool {
|
||||||
exactISRCMatch := req.ISRC != "" &&
|
exactISRCMatch := req.ISRC != "" &&
|
||||||
resolved.ISRC != "" &&
|
resolved.ISRC != "" &&
|
||||||
|
|||||||
+33
@@ -27,6 +27,37 @@ require File.expand_path(File.join('packages', 'flutter_tools', 'bin', 'podhelpe
|
|||||||
|
|
||||||
flutter_ios_podfile_setup
|
flutter_ios_podfile_setup
|
||||||
|
|
||||||
|
def patch_device_info_plus_vision_selector
|
||||||
|
plugin_file = File.join(
|
||||||
|
__dir__,
|
||||||
|
'.symlinks',
|
||||||
|
'plugins',
|
||||||
|
'device_info_plus',
|
||||||
|
'ios',
|
||||||
|
'device_info_plus',
|
||||||
|
'Sources',
|
||||||
|
'device_info_plus',
|
||||||
|
'FPPDeviceInfoPlusPlugin.m'
|
||||||
|
)
|
||||||
|
return unless File.exist?(plugin_file)
|
||||||
|
|
||||||
|
source = File.read(plugin_file)
|
||||||
|
return if source.include?('FPPDeviceInfoPlusVisionCompat')
|
||||||
|
|
||||||
|
marker = "#import <sys/utsname.h>\n"
|
||||||
|
declaration = <<~OBJC
|
||||||
|
|
||||||
|
// Older Xcode SDKs do not declare this selector yet, but device_info_plus
|
||||||
|
// only calls it behind an availability check.
|
||||||
|
@interface NSProcessInfo (FPPDeviceInfoPlusVisionCompat)
|
||||||
|
- (BOOL)isiOSAppOnVision;
|
||||||
|
@end
|
||||||
|
OBJC
|
||||||
|
|
||||||
|
patched = source.sub(marker, "#{marker}#{declaration}\n")
|
||||||
|
File.write(plugin_file, patched) if patched != source
|
||||||
|
end
|
||||||
|
|
||||||
target 'Runner' do
|
target 'Runner' do
|
||||||
use_frameworks!
|
use_frameworks!
|
||||||
use_modular_headers!
|
use_modular_headers!
|
||||||
@@ -42,6 +73,8 @@ target 'RunnerTests' do
|
|||||||
end
|
end
|
||||||
|
|
||||||
post_install do |installer|
|
post_install do |installer|
|
||||||
|
patch_device_info_plus_vision_selector
|
||||||
|
|
||||||
installer.pods_project.targets.each do |target|
|
installer.pods_project.targets.each do |target|
|
||||||
flutter_additional_ios_build_settings(target)
|
flutter_additional_ios_build_settings(target)
|
||||||
target.build_configurations.each do |config|
|
target.build_configurations.each do |config|
|
||||||
|
|||||||
@@ -89,7 +89,7 @@ import Gobackend // Import Go framework
|
|||||||
}
|
}
|
||||||
self.lastDownloadProgressPayload = payload
|
self.lastDownloadProgressPayload = payload
|
||||||
DispatchQueue.main.async { [weak self] in
|
DispatchQueue.main.async { [weak self] in
|
||||||
self?.downloadProgressEventSink?(payload)
|
self?.downloadProgressEventSink?(self?.parseJsonPayload(payload))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
downloadProgressTimer = timer
|
downloadProgressTimer = timer
|
||||||
@@ -119,7 +119,7 @@ import Gobackend // Import Go framework
|
|||||||
}
|
}
|
||||||
self.lastLibraryScanProgressPayload = payload
|
self.lastLibraryScanProgressPayload = payload
|
||||||
DispatchQueue.main.async { [weak self] in
|
DispatchQueue.main.async { [weak self] in
|
||||||
self?.libraryScanProgressEventSink?(payload)
|
self?.libraryScanProgressEventSink?(self?.parseJsonPayload(payload))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
libraryScanProgressTimer = timer
|
libraryScanProgressTimer = timer
|
||||||
@@ -133,6 +133,17 @@ import Gobackend // Import Go framework
|
|||||||
libraryScanProgressEventSink = nil
|
libraryScanProgressEventSink = nil
|
||||||
lastLibraryScanProgressPayload = nil
|
lastLibraryScanProgressPayload = nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private func parseJsonPayload(_ payload: String) -> Any {
|
||||||
|
guard let data = payload.data(using: .utf8) else {
|
||||||
|
return payload
|
||||||
|
}
|
||||||
|
do {
|
||||||
|
return try JSONSerialization.jsonObject(with: data, options: [.fragmentsAllowed])
|
||||||
|
} catch {
|
||||||
|
return payload
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private func handleMethodCall(call: FlutterMethodCall, result: @escaping FlutterResult) {
|
private func handleMethodCall(call: FlutterMethodCall, result: @escaping FlutterResult) {
|
||||||
DispatchQueue.global(qos: .userInitiated).async {
|
DispatchQueue.global(qos: .userInitiated).async {
|
||||||
@@ -169,11 +180,11 @@ import Gobackend // Import Go framework
|
|||||||
|
|
||||||
case "getDownloadProgress":
|
case "getDownloadProgress":
|
||||||
let response = GobackendGetDownloadProgress()
|
let response = GobackendGetDownloadProgress()
|
||||||
return response
|
return parseJsonPayload(response as String? ?? "{}")
|
||||||
|
|
||||||
case "getAllDownloadProgress":
|
case "getAllDownloadProgress":
|
||||||
let response = GobackendGetAllDownloadProgress()
|
let response = GobackendGetAllDownloadProgress()
|
||||||
return response
|
return parseJsonPayload(response as String? ?? "{}")
|
||||||
|
|
||||||
case "initItemProgress":
|
case "initItemProgress":
|
||||||
let args = call.arguments as! [String: Any]
|
let args = call.arguments as! [String: Any]
|
||||||
@@ -296,6 +307,15 @@ import Gobackend // Import Go framework
|
|||||||
if let error = error { throw error }
|
if let error = error { throw error }
|
||||||
return response
|
return response
|
||||||
|
|
||||||
|
case "rewriteSplitArtistTags":
|
||||||
|
let args = call.arguments as! [String: Any]
|
||||||
|
let filePath = args["file_path"] as! String
|
||||||
|
let artist = args["artist"] as! String
|
||||||
|
let albumArtist = args["album_artist"] as! String
|
||||||
|
let response = GobackendRewriteSplitArtistTagsExport(filePath, artist, albumArtist, &error)
|
||||||
|
if let error = error { throw error }
|
||||||
|
return response
|
||||||
|
|
||||||
case "cleanupConnections":
|
case "cleanupConnections":
|
||||||
GobackendCleanupConnections()
|
GobackendCleanupConnections()
|
||||||
return nil
|
return nil
|
||||||
@@ -324,7 +344,8 @@ import Gobackend // Import Go framework
|
|||||||
let spotifyId = args["spotify_id"] as! String
|
let spotifyId = args["spotify_id"] as! String
|
||||||
let durationMs = args["duration_ms"] as? Int64 ?? 0
|
let durationMs = args["duration_ms"] as? Int64 ?? 0
|
||||||
let outputPath = args["output_path"] as! String
|
let outputPath = args["output_path"] as! String
|
||||||
GobackendFetchAndSaveLyrics(trackName, artistName, spotifyId, durationMs, outputPath, &error)
|
let audioFilePath = args["audio_file_path"] as? String ?? ""
|
||||||
|
GobackendFetchAndSaveLyrics(trackName, artistName, spotifyId, durationMs, outputPath, audioFilePath, &error)
|
||||||
if let error = error { throw error }
|
if let error = error { throw error }
|
||||||
return "{\"success\":true}"
|
return "{\"success\":true}"
|
||||||
|
|
||||||
@@ -923,7 +944,7 @@ import Gobackend // Import Go framework
|
|||||||
|
|
||||||
case "getLibraryScanProgress":
|
case "getLibraryScanProgress":
|
||||||
let response = GobackendGetLibraryScanProgressJSON()
|
let response = GobackendGetLibraryScanProgressJSON()
|
||||||
return response
|
return parseJsonPayload(response as String? ?? "{}")
|
||||||
|
|
||||||
case "cancelLibraryScan":
|
case "cancelLibraryScan":
|
||||||
GobackendCancelLibraryScanJSON()
|
GobackendCancelLibraryScanJSON()
|
||||||
|
|||||||
@@ -3,8 +3,8 @@ import 'package:flutter/foundation.dart';
|
|||||||
/// App version and info constants
|
/// App version and info constants
|
||||||
/// Update version here only - all other files will reference this
|
/// Update version here only - all other files will reference this
|
||||||
class AppInfo {
|
class AppInfo {
|
||||||
static const String version = '4.1.3';
|
static const String version = '4.2.1';
|
||||||
static const String buildNumber = '120';
|
static const String buildNumber = '122';
|
||||||
static const String fullVersion = '$version+$buildNumber';
|
static const String fullVersion = '$version+$buildNumber';
|
||||||
|
|
||||||
/// Shows "Internal" in debug builds, actual version in release.
|
/// Shows "Internal" in debug builds, actual version in release.
|
||||||
|
|||||||
@@ -256,6 +256,18 @@ abstract class AppLocalizations {
|
|||||||
/// **'Filename Format'**
|
/// **'Filename Format'**
|
||||||
String get downloadFilenameFormat;
|
String get downloadFilenameFormat;
|
||||||
|
|
||||||
|
/// Setting for output filename pattern for singles/EPs
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Single Filename Format'**
|
||||||
|
String get downloadSingleFilenameFormat;
|
||||||
|
|
||||||
|
/// Subtitle description for single filename format setting
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Filename pattern for singles and EPs. Uses the same tags as the album format.'**
|
||||||
|
String get downloadSingleFilenameFormatDescription;
|
||||||
|
|
||||||
/// Title of the folder organization picker bottom sheet
|
/// Title of the folder organization picker bottom sheet
|
||||||
///
|
///
|
||||||
/// In en, this message translates to:
|
/// In en, this message translates to:
|
||||||
@@ -400,6 +412,24 @@ abstract class AppLocalizations {
|
|||||||
/// **'Download highest resolution cover art'**
|
/// **'Download highest resolution cover art'**
|
||||||
String get optionsMaxQualityCoverSubtitle;
|
String get optionsMaxQualityCoverSubtitle;
|
||||||
|
|
||||||
|
/// Title for ReplayGain setting toggle
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'ReplayGain'**
|
||||||
|
String get optionsReplayGain;
|
||||||
|
|
||||||
|
/// Subtitle when ReplayGain is enabled
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Scan loudness and embed ReplayGain tags (EBU R128)'**
|
||||||
|
String get optionsReplayGainSubtitleOn;
|
||||||
|
|
||||||
|
/// Subtitle when ReplayGain is disabled
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Disabled: no loudness normalization tags'**
|
||||||
|
String get optionsReplayGainSubtitleOff;
|
||||||
|
|
||||||
/// Setting title for how artist metadata is written into files
|
/// Setting title for how artist metadata is written into files
|
||||||
///
|
///
|
||||||
/// In en, this message translates to:
|
/// In en, this message translates to:
|
||||||
@@ -2218,6 +2248,18 @@ abstract class AppLocalizations {
|
|||||||
/// **'Lyrics not available for this track'**
|
/// **'Lyrics not available for this track'**
|
||||||
String get trackLyricsNotAvailable;
|
String get trackLyricsNotAvailable;
|
||||||
|
|
||||||
|
/// Message when no embedded lyrics in audio file
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'No lyrics found in this file'**
|
||||||
|
String get trackLyricsNotInFile;
|
||||||
|
|
||||||
|
/// Action - fetch lyrics from online providers
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Fetch from Online'**
|
||||||
|
String get trackFetchOnlineLyrics;
|
||||||
|
|
||||||
/// Message when lyrics request times out
|
/// Message when lyrics request times out
|
||||||
///
|
///
|
||||||
/// In en, this message translates to:
|
/// In en, this message translates to:
|
||||||
@@ -4048,6 +4090,54 @@ abstract class AppLocalizations {
|
|||||||
/// **'Search metadata online and embed into file'**
|
/// **'Search metadata online and embed into file'**
|
||||||
String get trackReEnrichOnlineSubtitle;
|
String get trackReEnrichOnlineSubtitle;
|
||||||
|
|
||||||
|
/// Section title for field selection in re-enrich dialog
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Fields to update'**
|
||||||
|
String get trackReEnrichFieldsTitle;
|
||||||
|
|
||||||
|
/// Checkbox label for cover art field in re-enrich
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Cover Art'**
|
||||||
|
String get trackReEnrichFieldCover;
|
||||||
|
|
||||||
|
/// Checkbox label for lyrics field in re-enrich
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Lyrics'**
|
||||||
|
String get trackReEnrichFieldLyrics;
|
||||||
|
|
||||||
|
/// Checkbox label for basic tags in re-enrich (title/artist are never overwritten)
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Album, Album Artist'**
|
||||||
|
String get trackReEnrichFieldBasicTags;
|
||||||
|
|
||||||
|
/// Checkbox label for track info in re-enrich
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Track & Disc Number'**
|
||||||
|
String get trackReEnrichFieldTrackInfo;
|
||||||
|
|
||||||
|
/// Checkbox label for release info in re-enrich
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Date & ISRC'**
|
||||||
|
String get trackReEnrichFieldReleaseInfo;
|
||||||
|
|
||||||
|
/// Checkbox label for extra metadata in re-enrich
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Genre, Label, Copyright'**
|
||||||
|
String get trackReEnrichFieldExtra;
|
||||||
|
|
||||||
|
/// Select all fields checkbox in re-enrich
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Select All'**
|
||||||
|
String get trackReEnrichSelectAll;
|
||||||
|
|
||||||
/// Menu action - edit embedded metadata
|
/// Menu action - edit embedded metadata
|
||||||
///
|
///
|
||||||
/// In en, this message translates to:
|
/// In en, this message translates to:
|
||||||
@@ -4647,12 +4737,6 @@ abstract class AppLocalizations {
|
|||||||
/// **'You have unsaved changes that will be lost.'**
|
/// **'You have unsaved changes that will be lost.'**
|
||||||
String get lyricsProvidersDiscardContent;
|
String get lyricsProvidersDiscardContent;
|
||||||
|
|
||||||
/// Description for Spotify Lyrics API provider
|
|
||||||
///
|
|
||||||
/// In en, this message translates to:
|
|
||||||
/// **'Spotify-sourced synced lyrics via community API'**
|
|
||||||
String get lyricsProviderSpotifyApiDesc;
|
|
||||||
|
|
||||||
/// Description for LRCLIB provider
|
/// Description for LRCLIB provider
|
||||||
///
|
///
|
||||||
/// In en, this message translates to:
|
/// In en, this message translates to:
|
||||||
|
|||||||
@@ -76,6 +76,13 @@ class AppLocalizationsDe extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get downloadFilenameFormat => 'Dateinamenformat';
|
String get downloadFilenameFormat => 'Dateinamenformat';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormat => 'Single Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormatDescription =>
|
||||||
|
'Filename pattern for singles and EPs. Uses the same tags as the album format.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadFolderOrganization => 'Ordnerstruktur';
|
String get downloadFolderOrganization => 'Ordnerstruktur';
|
||||||
|
|
||||||
@@ -158,6 +165,17 @@ class AppLocalizationsDe extends AppLocalizations {
|
|||||||
String get optionsMaxQualityCoverSubtitle =>
|
String get optionsMaxQualityCoverSubtitle =>
|
||||||
'Cover in höchster Auflösung herunterladen';
|
'Cover in höchster Auflösung herunterladen';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGain => 'ReplayGain';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOn =>
|
||||||
|
'Scan loudness and embed ReplayGain tags (EBU R128)';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOff =>
|
||||||
|
'Disabled: no loudness normalization tags';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get optionsArtistTagMode => 'Artist Tag Mode';
|
String get optionsArtistTagMode => 'Artist Tag Mode';
|
||||||
|
|
||||||
@@ -1201,6 +1219,12 @@ class AppLocalizationsDe extends AppLocalizations {
|
|||||||
String get trackLyricsNotAvailable =>
|
String get trackLyricsNotAvailable =>
|
||||||
'Lyrics sind für diesen Titel nicht verfügbar';
|
'Lyrics sind für diesen Titel nicht verfügbar';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackLyricsNotInFile => 'No lyrics found in this file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackFetchOnlineLyrics => 'Fetch from Online';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsTimeout =>
|
String get trackLyricsTimeout =>
|
||||||
'Anfrage Timeout. Versuche es später erneut.';
|
'Anfrage Timeout. Versuche es später erneut.';
|
||||||
@@ -2290,6 +2314,30 @@ class AppLocalizationsDe extends AppLocalizations {
|
|||||||
String get trackReEnrichOnlineSubtitle =>
|
String get trackReEnrichOnlineSubtitle =>
|
||||||
'Metadaten online suchen und in Datei einbinden';
|
'Metadaten online suchen und in Datei einbinden';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldsTitle => 'Fields to update';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldCover => 'Cover Art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldLyrics => 'Lyrics';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldBasicTags => 'Album, Album Artist';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldTrackInfo => 'Track & Disc Number';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldReleaseInfo => 'Date & ISRC';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldExtra => 'Genre, Label, Copyright';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichSelectAll => 'Select All';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackEditMetadata => 'Metadaten bearbeiten';
|
String get trackEditMetadata => 'Metadaten bearbeiten';
|
||||||
|
|
||||||
@@ -2706,10 +2754,6 @@ class AppLocalizationsDe extends AppLocalizations {
|
|||||||
String get lyricsProvidersDiscardContent =>
|
String get lyricsProvidersDiscardContent =>
|
||||||
'You have unsaved changes that will be lost.';
|
'You have unsaved changes that will be lost.';
|
||||||
|
|
||||||
@override
|
|
||||||
String get lyricsProviderSpotifyApiDesc =>
|
|
||||||
'Spotify-sourced synced lyrics via community API';
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
||||||
|
|
||||||
|
|||||||
@@ -75,6 +75,13 @@ class AppLocalizationsEn extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get downloadFilenameFormat => 'Filename Format';
|
String get downloadFilenameFormat => 'Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormat => 'Single Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormatDescription =>
|
||||||
|
'Filename pattern for singles and EPs. Uses the same tags as the album format.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadFolderOrganization => 'Folder Organization';
|
String get downloadFolderOrganization => 'Folder Organization';
|
||||||
|
|
||||||
@@ -154,6 +161,17 @@ class AppLocalizationsEn extends AppLocalizations {
|
|||||||
String get optionsMaxQualityCoverSubtitle =>
|
String get optionsMaxQualityCoverSubtitle =>
|
||||||
'Download highest resolution cover art';
|
'Download highest resolution cover art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGain => 'ReplayGain';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOn =>
|
||||||
|
'Scan loudness and embed ReplayGain tags (EBU R128)';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOff =>
|
||||||
|
'Disabled: no loudness normalization tags';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get optionsArtistTagMode => 'Artist Tag Mode';
|
String get optionsArtistTagMode => 'Artist Tag Mode';
|
||||||
|
|
||||||
@@ -1182,6 +1200,12 @@ class AppLocalizationsEn extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackLyricsNotInFile => 'No lyrics found in this file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackFetchOnlineLyrics => 'Fetch from Online';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
||||||
|
|
||||||
@@ -2260,6 +2284,30 @@ class AppLocalizationsEn extends AppLocalizations {
|
|||||||
String get trackReEnrichOnlineSubtitle =>
|
String get trackReEnrichOnlineSubtitle =>
|
||||||
'Search metadata online and embed into file';
|
'Search metadata online and embed into file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldsTitle => 'Fields to update';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldCover => 'Cover Art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldLyrics => 'Lyrics';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldBasicTags => 'Album, Album Artist';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldTrackInfo => 'Track & Disc Number';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldReleaseInfo => 'Date & ISRC';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldExtra => 'Genre, Label, Copyright';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichSelectAll => 'Select All';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackEditMetadata => 'Edit Metadata';
|
String get trackEditMetadata => 'Edit Metadata';
|
||||||
|
|
||||||
@@ -2674,10 +2722,6 @@ class AppLocalizationsEn extends AppLocalizations {
|
|||||||
String get lyricsProvidersDiscardContent =>
|
String get lyricsProvidersDiscardContent =>
|
||||||
'You have unsaved changes that will be lost.';
|
'You have unsaved changes that will be lost.';
|
||||||
|
|
||||||
@override
|
|
||||||
String get lyricsProviderSpotifyApiDesc =>
|
|
||||||
'Spotify-sourced synced lyrics via community API';
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
||||||
|
|
||||||
|
|||||||
@@ -75,6 +75,13 @@ class AppLocalizationsEs extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get downloadFilenameFormat => 'Filename Format';
|
String get downloadFilenameFormat => 'Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormat => 'Single Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormatDescription =>
|
||||||
|
'Filename pattern for singles and EPs. Uses the same tags as the album format.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadFolderOrganization => 'Folder Organization';
|
String get downloadFolderOrganization => 'Folder Organization';
|
||||||
|
|
||||||
@@ -154,6 +161,17 @@ class AppLocalizationsEs extends AppLocalizations {
|
|||||||
String get optionsMaxQualityCoverSubtitle =>
|
String get optionsMaxQualityCoverSubtitle =>
|
||||||
'Download highest resolution cover art';
|
'Download highest resolution cover art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGain => 'ReplayGain';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOn =>
|
||||||
|
'Scan loudness and embed ReplayGain tags (EBU R128)';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOff =>
|
||||||
|
'Disabled: no loudness normalization tags';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get optionsArtistTagMode => 'Artist Tag Mode';
|
String get optionsArtistTagMode => 'Artist Tag Mode';
|
||||||
|
|
||||||
@@ -1182,6 +1200,12 @@ class AppLocalizationsEs extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackLyricsNotInFile => 'No lyrics found in this file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackFetchOnlineLyrics => 'Fetch from Online';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
||||||
|
|
||||||
@@ -2260,6 +2284,30 @@ class AppLocalizationsEs extends AppLocalizations {
|
|||||||
String get trackReEnrichOnlineSubtitle =>
|
String get trackReEnrichOnlineSubtitle =>
|
||||||
'Search metadata online and embed into file';
|
'Search metadata online and embed into file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldsTitle => 'Fields to update';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldCover => 'Cover Art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldLyrics => 'Lyrics';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldBasicTags => 'Album, Album Artist';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldTrackInfo => 'Track & Disc Number';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldReleaseInfo => 'Date & ISRC';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldExtra => 'Genre, Label, Copyright';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichSelectAll => 'Select All';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackEditMetadata => 'Edit Metadata';
|
String get trackEditMetadata => 'Edit Metadata';
|
||||||
|
|
||||||
@@ -2674,10 +2722,6 @@ class AppLocalizationsEs extends AppLocalizations {
|
|||||||
String get lyricsProvidersDiscardContent =>
|
String get lyricsProvidersDiscardContent =>
|
||||||
'You have unsaved changes that will be lost.';
|
'You have unsaved changes that will be lost.';
|
||||||
|
|
||||||
@override
|
|
||||||
String get lyricsProviderSpotifyApiDesc =>
|
|
||||||
'Spotify-sourced synced lyrics via community API';
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
||||||
|
|
||||||
|
|||||||
@@ -75,6 +75,13 @@ class AppLocalizationsFr extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get downloadFilenameFormat => 'Nom du fichier';
|
String get downloadFilenameFormat => 'Nom du fichier';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormat => 'Single Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormatDescription =>
|
||||||
|
'Filename pattern for singles and EPs. Uses the same tags as the album format.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadFolderOrganization => 'Organisation du dossier';
|
String get downloadFolderOrganization => 'Organisation du dossier';
|
||||||
|
|
||||||
@@ -156,6 +163,17 @@ class AppLocalizationsFr extends AppLocalizations {
|
|||||||
String get optionsMaxQualityCoverSubtitle =>
|
String get optionsMaxQualityCoverSubtitle =>
|
||||||
'Download highest resolution cover art';
|
'Download highest resolution cover art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGain => 'ReplayGain';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOn =>
|
||||||
|
'Scan loudness and embed ReplayGain tags (EBU R128)';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOff =>
|
||||||
|
'Disabled: no loudness normalization tags';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get optionsArtistTagMode => 'Artist Tag Mode';
|
String get optionsArtistTagMode => 'Artist Tag Mode';
|
||||||
|
|
||||||
@@ -1184,6 +1202,12 @@ class AppLocalizationsFr extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackLyricsNotInFile => 'No lyrics found in this file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackFetchOnlineLyrics => 'Fetch from Online';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
||||||
|
|
||||||
@@ -2262,6 +2286,30 @@ class AppLocalizationsFr extends AppLocalizations {
|
|||||||
String get trackReEnrichOnlineSubtitle =>
|
String get trackReEnrichOnlineSubtitle =>
|
||||||
'Search metadata online and embed into file';
|
'Search metadata online and embed into file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldsTitle => 'Fields to update';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldCover => 'Cover Art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldLyrics => 'Lyrics';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldBasicTags => 'Album, Album Artist';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldTrackInfo => 'Track & Disc Number';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldReleaseInfo => 'Date & ISRC';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldExtra => 'Genre, Label, Copyright';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichSelectAll => 'Select All';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackEditMetadata => 'Edit Metadata';
|
String get trackEditMetadata => 'Edit Metadata';
|
||||||
|
|
||||||
@@ -2675,10 +2723,6 @@ class AppLocalizationsFr extends AppLocalizations {
|
|||||||
String get lyricsProvidersDiscardContent =>
|
String get lyricsProvidersDiscardContent =>
|
||||||
'You have unsaved changes that will be lost.';
|
'You have unsaved changes that will be lost.';
|
||||||
|
|
||||||
@override
|
|
||||||
String get lyricsProviderSpotifyApiDesc =>
|
|
||||||
'Spotify-sourced synced lyrics via community API';
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
||||||
|
|
||||||
|
|||||||
@@ -75,6 +75,13 @@ class AppLocalizationsHi extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get downloadFilenameFormat => 'Filename Format';
|
String get downloadFilenameFormat => 'Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormat => 'Single Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormatDescription =>
|
||||||
|
'Filename pattern for singles and EPs. Uses the same tags as the album format.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadFolderOrganization => 'Folder Organization';
|
String get downloadFolderOrganization => 'Folder Organization';
|
||||||
|
|
||||||
@@ -154,6 +161,17 @@ class AppLocalizationsHi extends AppLocalizations {
|
|||||||
String get optionsMaxQualityCoverSubtitle =>
|
String get optionsMaxQualityCoverSubtitle =>
|
||||||
'Download highest resolution cover art';
|
'Download highest resolution cover art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGain => 'ReplayGain';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOn =>
|
||||||
|
'Scan loudness and embed ReplayGain tags (EBU R128)';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOff =>
|
||||||
|
'Disabled: no loudness normalization tags';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get optionsArtistTagMode => 'Artist Tag Mode';
|
String get optionsArtistTagMode => 'Artist Tag Mode';
|
||||||
|
|
||||||
@@ -1182,6 +1200,12 @@ class AppLocalizationsHi extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackLyricsNotInFile => 'No lyrics found in this file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackFetchOnlineLyrics => 'Fetch from Online';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
||||||
|
|
||||||
@@ -2260,6 +2284,30 @@ class AppLocalizationsHi extends AppLocalizations {
|
|||||||
String get trackReEnrichOnlineSubtitle =>
|
String get trackReEnrichOnlineSubtitle =>
|
||||||
'Search metadata online and embed into file';
|
'Search metadata online and embed into file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldsTitle => 'Fields to update';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldCover => 'Cover Art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldLyrics => 'Lyrics';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldBasicTags => 'Album, Album Artist';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldTrackInfo => 'Track & Disc Number';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldReleaseInfo => 'Date & ISRC';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldExtra => 'Genre, Label, Copyright';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichSelectAll => 'Select All';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackEditMetadata => 'Edit Metadata';
|
String get trackEditMetadata => 'Edit Metadata';
|
||||||
|
|
||||||
@@ -2673,10 +2721,6 @@ class AppLocalizationsHi extends AppLocalizations {
|
|||||||
String get lyricsProvidersDiscardContent =>
|
String get lyricsProvidersDiscardContent =>
|
||||||
'You have unsaved changes that will be lost.';
|
'You have unsaved changes that will be lost.';
|
||||||
|
|
||||||
@override
|
|
||||||
String get lyricsProviderSpotifyApiDesc =>
|
|
||||||
'Spotify-sourced synced lyrics via community API';
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
||||||
|
|
||||||
|
|||||||
@@ -76,6 +76,13 @@ class AppLocalizationsId extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get downloadFilenameFormat => 'Format Nama File';
|
String get downloadFilenameFormat => 'Format Nama File';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormat => 'Single Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormatDescription =>
|
||||||
|
'Filename pattern for singles and EPs. Uses the same tags as the album format.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadFolderOrganization => 'Organisasi Folder';
|
String get downloadFolderOrganization => 'Organisasi Folder';
|
||||||
|
|
||||||
@@ -158,6 +165,17 @@ class AppLocalizationsId extends AppLocalizations {
|
|||||||
String get optionsMaxQualityCoverSubtitle =>
|
String get optionsMaxQualityCoverSubtitle =>
|
||||||
'Unduh cover art resolusi tertinggi';
|
'Unduh cover art resolusi tertinggi';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGain => 'ReplayGain';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOn =>
|
||||||
|
'Scan loudness and embed ReplayGain tags (EBU R128)';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOff =>
|
||||||
|
'Disabled: no loudness normalization tags';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get optionsArtistTagMode => 'Artist Tag Mode';
|
String get optionsArtistTagMode => 'Artist Tag Mode';
|
||||||
|
|
||||||
@@ -1189,6 +1207,12 @@ class AppLocalizationsId extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get trackLyricsNotAvailable => 'Lirik tidak tersedia untuk lagu ini';
|
String get trackLyricsNotAvailable => 'Lirik tidak tersedia untuk lagu ini';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackLyricsNotInFile => 'No lyrics found in this file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackFetchOnlineLyrics => 'Fetch from Online';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsTimeout => 'Permintaan timeout. Coba lagi nanti.';
|
String get trackLyricsTimeout => 'Permintaan timeout. Coba lagi nanti.';
|
||||||
|
|
||||||
@@ -2270,6 +2294,30 @@ class AppLocalizationsId extends AppLocalizations {
|
|||||||
String get trackReEnrichOnlineSubtitle =>
|
String get trackReEnrichOnlineSubtitle =>
|
||||||
'Search metadata online and embed into file';
|
'Search metadata online and embed into file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldsTitle => 'Fields to update';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldCover => 'Cover Art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldLyrics => 'Lyrics';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldBasicTags => 'Album, Album Artist';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldTrackInfo => 'Track & Disc Number';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldReleaseInfo => 'Date & ISRC';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldExtra => 'Genre, Label, Copyright';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichSelectAll => 'Select All';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackEditMetadata => 'Edit Metadata';
|
String get trackEditMetadata => 'Edit Metadata';
|
||||||
|
|
||||||
@@ -2684,10 +2732,6 @@ class AppLocalizationsId extends AppLocalizations {
|
|||||||
String get lyricsProvidersDiscardContent =>
|
String get lyricsProvidersDiscardContent =>
|
||||||
'You have unsaved changes that will be lost.';
|
'You have unsaved changes that will be lost.';
|
||||||
|
|
||||||
@override
|
|
||||||
String get lyricsProviderSpotifyApiDesc =>
|
|
||||||
'Spotify-sourced synced lyrics via community API';
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
||||||
|
|
||||||
|
|||||||
@@ -75,6 +75,13 @@ class AppLocalizationsJa extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get downloadFilenameFormat => 'ファイル名の形式';
|
String get downloadFilenameFormat => 'ファイル名の形式';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormat => 'Single Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormatDescription =>
|
||||||
|
'Filename pattern for singles and EPs. Uses the same tags as the album format.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadFolderOrganization => 'フォルダ構成';
|
String get downloadFolderOrganization => 'フォルダ構成';
|
||||||
|
|
||||||
@@ -152,6 +159,17 @@ class AppLocalizationsJa extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get optionsMaxQualityCoverSubtitle => '最高解像度のカバーアートをダウンロード';
|
String get optionsMaxQualityCoverSubtitle => '最高解像度のカバーアートをダウンロード';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGain => 'ReplayGain';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOn =>
|
||||||
|
'Scan loudness and embed ReplayGain tags (EBU R128)';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOff =>
|
||||||
|
'Disabled: no loudness normalization tags';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get optionsArtistTagMode => 'Artist Tag Mode';
|
String get optionsArtistTagMode => 'Artist Tag Mode';
|
||||||
|
|
||||||
@@ -1176,6 +1194,12 @@ class AppLocalizationsJa extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get trackLyricsNotAvailable => 'このトラックの歌詞は利用できません';
|
String get trackLyricsNotAvailable => 'このトラックの歌詞は利用できません';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackLyricsNotInFile => 'No lyrics found in this file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackFetchOnlineLyrics => 'Fetch from Online';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsTimeout => 'リクエストがタイムアウトしました。後ほどお試しください。';
|
String get trackLyricsTimeout => 'リクエストがタイムアウトしました。後ほどお試しください。';
|
||||||
|
|
||||||
@@ -2247,6 +2271,30 @@ class AppLocalizationsJa extends AppLocalizations {
|
|||||||
String get trackReEnrichOnlineSubtitle =>
|
String get trackReEnrichOnlineSubtitle =>
|
||||||
'Search metadata online and embed into file';
|
'Search metadata online and embed into file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldsTitle => 'Fields to update';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldCover => 'Cover Art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldLyrics => 'Lyrics';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldBasicTags => 'Album, Album Artist';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldTrackInfo => 'Track & Disc Number';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldReleaseInfo => 'Date & ISRC';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldExtra => 'Genre, Label, Copyright';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichSelectAll => 'Select All';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackEditMetadata => 'メタデータを編集';
|
String get trackEditMetadata => 'メタデータを編集';
|
||||||
|
|
||||||
@@ -2660,10 +2708,6 @@ class AppLocalizationsJa extends AppLocalizations {
|
|||||||
String get lyricsProvidersDiscardContent =>
|
String get lyricsProvidersDiscardContent =>
|
||||||
'You have unsaved changes that will be lost.';
|
'You have unsaved changes that will be lost.';
|
||||||
|
|
||||||
@override
|
|
||||||
String get lyricsProviderSpotifyApiDesc =>
|
|
||||||
'Spotify-sourced synced lyrics via community API';
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
||||||
|
|
||||||
|
|||||||
@@ -74,6 +74,13 @@ class AppLocalizationsKo extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get downloadFilenameFormat => '파일 이름 형식';
|
String get downloadFilenameFormat => '파일 이름 형식';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormat => 'Single Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormatDescription =>
|
||||||
|
'Filename pattern for singles and EPs. Uses the same tags as the album format.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadFolderOrganization => '폴더 분류 형식';
|
String get downloadFolderOrganization => '폴더 분류 형식';
|
||||||
|
|
||||||
@@ -148,6 +155,17 @@ class AppLocalizationsKo extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get optionsMaxQualityCoverSubtitle => '최고 품질의 커버 이미지를 다운로드';
|
String get optionsMaxQualityCoverSubtitle => '최고 품질의 커버 이미지를 다운로드';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGain => 'ReplayGain';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOn =>
|
||||||
|
'Scan loudness and embed ReplayGain tags (EBU R128)';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOff =>
|
||||||
|
'Disabled: no loudness normalization tags';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get optionsArtistTagMode => 'Artist Tag Mode';
|
String get optionsArtistTagMode => 'Artist Tag Mode';
|
||||||
|
|
||||||
@@ -1162,6 +1180,12 @@ class AppLocalizationsKo extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackLyricsNotInFile => 'No lyrics found in this file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackFetchOnlineLyrics => 'Fetch from Online';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
||||||
|
|
||||||
@@ -2240,6 +2264,30 @@ class AppLocalizationsKo extends AppLocalizations {
|
|||||||
String get trackReEnrichOnlineSubtitle =>
|
String get trackReEnrichOnlineSubtitle =>
|
||||||
'Search metadata online and embed into file';
|
'Search metadata online and embed into file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldsTitle => 'Fields to update';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldCover => 'Cover Art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldLyrics => 'Lyrics';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldBasicTags => 'Album, Album Artist';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldTrackInfo => 'Track & Disc Number';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldReleaseInfo => 'Date & ISRC';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldExtra => 'Genre, Label, Copyright';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichSelectAll => 'Select All';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackEditMetadata => 'Edit Metadata';
|
String get trackEditMetadata => 'Edit Metadata';
|
||||||
|
|
||||||
@@ -2653,10 +2701,6 @@ class AppLocalizationsKo extends AppLocalizations {
|
|||||||
String get lyricsProvidersDiscardContent =>
|
String get lyricsProvidersDiscardContent =>
|
||||||
'You have unsaved changes that will be lost.';
|
'You have unsaved changes that will be lost.';
|
||||||
|
|
||||||
@override
|
|
||||||
String get lyricsProviderSpotifyApiDesc =>
|
|
||||||
'Spotify-sourced synced lyrics via community API';
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
||||||
|
|
||||||
|
|||||||
@@ -75,6 +75,13 @@ class AppLocalizationsNl extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get downloadFilenameFormat => 'Filename Format';
|
String get downloadFilenameFormat => 'Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormat => 'Single Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormatDescription =>
|
||||||
|
'Filename pattern for singles and EPs. Uses the same tags as the album format.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadFolderOrganization => 'Folder Organization';
|
String get downloadFolderOrganization => 'Folder Organization';
|
||||||
|
|
||||||
@@ -154,6 +161,17 @@ class AppLocalizationsNl extends AppLocalizations {
|
|||||||
String get optionsMaxQualityCoverSubtitle =>
|
String get optionsMaxQualityCoverSubtitle =>
|
||||||
'Download highest resolution cover art';
|
'Download highest resolution cover art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGain => 'ReplayGain';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOn =>
|
||||||
|
'Scan loudness and embed ReplayGain tags (EBU R128)';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOff =>
|
||||||
|
'Disabled: no loudness normalization tags';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get optionsArtistTagMode => 'Artist Tag Mode';
|
String get optionsArtistTagMode => 'Artist Tag Mode';
|
||||||
|
|
||||||
@@ -1182,6 +1200,12 @@ class AppLocalizationsNl extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackLyricsNotInFile => 'No lyrics found in this file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackFetchOnlineLyrics => 'Fetch from Online';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
||||||
|
|
||||||
@@ -2260,6 +2284,30 @@ class AppLocalizationsNl extends AppLocalizations {
|
|||||||
String get trackReEnrichOnlineSubtitle =>
|
String get trackReEnrichOnlineSubtitle =>
|
||||||
'Search metadata online and embed into file';
|
'Search metadata online and embed into file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldsTitle => 'Fields to update';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldCover => 'Cover Art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldLyrics => 'Lyrics';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldBasicTags => 'Album, Album Artist';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldTrackInfo => 'Track & Disc Number';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldReleaseInfo => 'Date & ISRC';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldExtra => 'Genre, Label, Copyright';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichSelectAll => 'Select All';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackEditMetadata => 'Edit Metadata';
|
String get trackEditMetadata => 'Edit Metadata';
|
||||||
|
|
||||||
@@ -2673,10 +2721,6 @@ class AppLocalizationsNl extends AppLocalizations {
|
|||||||
String get lyricsProvidersDiscardContent =>
|
String get lyricsProvidersDiscardContent =>
|
||||||
'You have unsaved changes that will be lost.';
|
'You have unsaved changes that will be lost.';
|
||||||
|
|
||||||
@override
|
|
||||||
String get lyricsProviderSpotifyApiDesc =>
|
|
||||||
'Spotify-sourced synced lyrics via community API';
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
||||||
|
|
||||||
|
|||||||
@@ -75,6 +75,13 @@ class AppLocalizationsPt extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get downloadFilenameFormat => 'Filename Format';
|
String get downloadFilenameFormat => 'Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormat => 'Single Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormatDescription =>
|
||||||
|
'Filename pattern for singles and EPs. Uses the same tags as the album format.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadFolderOrganization => 'Folder Organization';
|
String get downloadFolderOrganization => 'Folder Organization';
|
||||||
|
|
||||||
@@ -154,6 +161,17 @@ class AppLocalizationsPt extends AppLocalizations {
|
|||||||
String get optionsMaxQualityCoverSubtitle =>
|
String get optionsMaxQualityCoverSubtitle =>
|
||||||
'Download highest resolution cover art';
|
'Download highest resolution cover art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGain => 'ReplayGain';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOn =>
|
||||||
|
'Scan loudness and embed ReplayGain tags (EBU R128)';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOff =>
|
||||||
|
'Disabled: no loudness normalization tags';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get optionsArtistTagMode => 'Artist Tag Mode';
|
String get optionsArtistTagMode => 'Artist Tag Mode';
|
||||||
|
|
||||||
@@ -1182,6 +1200,12 @@ class AppLocalizationsPt extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackLyricsNotInFile => 'No lyrics found in this file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackFetchOnlineLyrics => 'Fetch from Online';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
||||||
|
|
||||||
@@ -2260,6 +2284,30 @@ class AppLocalizationsPt extends AppLocalizations {
|
|||||||
String get trackReEnrichOnlineSubtitle =>
|
String get trackReEnrichOnlineSubtitle =>
|
||||||
'Search metadata online and embed into file';
|
'Search metadata online and embed into file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldsTitle => 'Fields to update';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldCover => 'Cover Art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldLyrics => 'Lyrics';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldBasicTags => 'Album, Album Artist';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldTrackInfo => 'Track & Disc Number';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldReleaseInfo => 'Date & ISRC';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldExtra => 'Genre, Label, Copyright';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichSelectAll => 'Select All';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackEditMetadata => 'Edit Metadata';
|
String get trackEditMetadata => 'Edit Metadata';
|
||||||
|
|
||||||
@@ -2674,10 +2722,6 @@ class AppLocalizationsPt extends AppLocalizations {
|
|||||||
String get lyricsProvidersDiscardContent =>
|
String get lyricsProvidersDiscardContent =>
|
||||||
'You have unsaved changes that will be lost.';
|
'You have unsaved changes that will be lost.';
|
||||||
|
|
||||||
@override
|
|
||||||
String get lyricsProviderSpotifyApiDesc =>
|
|
||||||
'Spotify-sourced synced lyrics via community API';
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
||||||
|
|
||||||
|
|||||||
@@ -76,6 +76,13 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get downloadFilenameFormat => 'Формат имени файла';
|
String get downloadFilenameFormat => 'Формат имени файла';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormat => 'Single Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormatDescription =>
|
||||||
|
'Filename pattern for singles and EPs. Uses the same tags as the album format.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadFolderOrganization => 'Организация папок';
|
String get downloadFolderOrganization => 'Организация папок';
|
||||||
|
|
||||||
@@ -159,6 +166,17 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
String get optionsMaxQualityCoverSubtitle =>
|
String get optionsMaxQualityCoverSubtitle =>
|
||||||
'Скачивать обложку в макс. разрешении';
|
'Скачивать обложку в макс. разрешении';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGain => 'ReplayGain';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOn =>
|
||||||
|
'Scan loudness and embed ReplayGain tags (EBU R128)';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOff =>
|
||||||
|
'Disabled: no loudness normalization tags';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get optionsArtistTagMode => 'Artist Tag Mode';
|
String get optionsArtistTagMode => 'Artist Tag Mode';
|
||||||
|
|
||||||
@@ -1202,6 +1220,12 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
String get trackLyricsNotAvailable =>
|
String get trackLyricsNotAvailable =>
|
||||||
'Текст песни недоступен для этого трека';
|
'Текст песни недоступен для этого трека';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackLyricsNotInFile => 'No lyrics found in this file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackFetchOnlineLyrics => 'Fetch from Online';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsTimeout =>
|
String get trackLyricsTimeout =>
|
||||||
'Время ожидания запроса истекло. Повторите попытку позже.';
|
'Время ожидания запроса истекло. Повторите попытку позже.';
|
||||||
@@ -2312,6 +2336,30 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
String get trackReEnrichOnlineSubtitle =>
|
String get trackReEnrichOnlineSubtitle =>
|
||||||
'Поиск в сети метаданных и встраивание в файл';
|
'Поиск в сети метаданных и встраивание в файл';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldsTitle => 'Fields to update';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldCover => 'Cover Art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldLyrics => 'Lyrics';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldBasicTags => 'Album, Album Artist';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldTrackInfo => 'Track & Disc Number';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldReleaseInfo => 'Date & ISRC';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldExtra => 'Genre, Label, Copyright';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichSelectAll => 'Select All';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackEditMetadata => 'Редактировать метаданные';
|
String get trackEditMetadata => 'Редактировать метаданные';
|
||||||
|
|
||||||
@@ -2733,10 +2781,6 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
String get lyricsProvidersDiscardContent =>
|
String get lyricsProvidersDiscardContent =>
|
||||||
'You have unsaved changes that will be lost.';
|
'You have unsaved changes that will be lost.';
|
||||||
|
|
||||||
@override
|
|
||||||
String get lyricsProviderSpotifyApiDesc =>
|
|
||||||
'Spotify-sourced synced lyrics via community API';
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
||||||
|
|
||||||
|
|||||||
@@ -76,6 +76,13 @@ class AppLocalizationsTr extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get downloadFilenameFormat => 'Dosya adı formatı';
|
String get downloadFilenameFormat => 'Dosya adı formatı';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormat => 'Single Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormatDescription =>
|
||||||
|
'Filename pattern for singles and EPs. Uses the same tags as the album format.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadFolderOrganization => 'Dosya Organizasyonu';
|
String get downloadFolderOrganization => 'Dosya Organizasyonu';
|
||||||
|
|
||||||
@@ -157,6 +164,17 @@ class AppLocalizationsTr extends AppLocalizations {
|
|||||||
String get optionsMaxQualityCoverSubtitle =>
|
String get optionsMaxQualityCoverSubtitle =>
|
||||||
'En yüksek kalitedeki albüm kapaklarını indir';
|
'En yüksek kalitedeki albüm kapaklarını indir';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGain => 'ReplayGain';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOn =>
|
||||||
|
'Scan loudness and embed ReplayGain tags (EBU R128)';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOff =>
|
||||||
|
'Disabled: no loudness normalization tags';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get optionsArtistTagMode => 'Artist Tag Mode';
|
String get optionsArtistTagMode => 'Artist Tag Mode';
|
||||||
|
|
||||||
@@ -1188,6 +1206,12 @@ class AppLocalizationsTr extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackLyricsNotInFile => 'No lyrics found in this file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackFetchOnlineLyrics => 'Fetch from Online';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
||||||
|
|
||||||
@@ -2266,6 +2290,30 @@ class AppLocalizationsTr extends AppLocalizations {
|
|||||||
String get trackReEnrichOnlineSubtitle =>
|
String get trackReEnrichOnlineSubtitle =>
|
||||||
'Search metadata online and embed into file';
|
'Search metadata online and embed into file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldsTitle => 'Fields to update';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldCover => 'Cover Art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldLyrics => 'Lyrics';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldBasicTags => 'Album, Album Artist';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldTrackInfo => 'Track & Disc Number';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldReleaseInfo => 'Date & ISRC';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldExtra => 'Genre, Label, Copyright';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichSelectAll => 'Select All';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackEditMetadata => 'Edit Metadata';
|
String get trackEditMetadata => 'Edit Metadata';
|
||||||
|
|
||||||
@@ -2679,10 +2727,6 @@ class AppLocalizationsTr extends AppLocalizations {
|
|||||||
String get lyricsProvidersDiscardContent =>
|
String get lyricsProvidersDiscardContent =>
|
||||||
'You have unsaved changes that will be lost.';
|
'You have unsaved changes that will be lost.';
|
||||||
|
|
||||||
@override
|
|
||||||
String get lyricsProviderSpotifyApiDesc =>
|
|
||||||
'Spotify-sourced synced lyrics via community API';
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
||||||
|
|
||||||
|
|||||||
@@ -75,6 +75,13 @@ class AppLocalizationsZh extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get downloadFilenameFormat => 'Filename Format';
|
String get downloadFilenameFormat => 'Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormat => 'Single Filename Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get downloadSingleFilenameFormatDescription =>
|
||||||
|
'Filename pattern for singles and EPs. Uses the same tags as the album format.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadFolderOrganization => 'Folder Organization';
|
String get downloadFolderOrganization => 'Folder Organization';
|
||||||
|
|
||||||
@@ -154,6 +161,17 @@ class AppLocalizationsZh extends AppLocalizations {
|
|||||||
String get optionsMaxQualityCoverSubtitle =>
|
String get optionsMaxQualityCoverSubtitle =>
|
||||||
'Download highest resolution cover art';
|
'Download highest resolution cover art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGain => 'ReplayGain';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOn =>
|
||||||
|
'Scan loudness and embed ReplayGain tags (EBU R128)';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get optionsReplayGainSubtitleOff =>
|
||||||
|
'Disabled: no loudness normalization tags';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get optionsArtistTagMode => 'Artist Tag Mode';
|
String get optionsArtistTagMode => 'Artist Tag Mode';
|
||||||
|
|
||||||
@@ -1182,6 +1200,12 @@ class AppLocalizationsZh extends AppLocalizations {
|
|||||||
@override
|
@override
|
||||||
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackLyricsNotInFile => 'No lyrics found in this file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackFetchOnlineLyrics => 'Fetch from Online';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
||||||
|
|
||||||
@@ -2260,6 +2284,30 @@ class AppLocalizationsZh extends AppLocalizations {
|
|||||||
String get trackReEnrichOnlineSubtitle =>
|
String get trackReEnrichOnlineSubtitle =>
|
||||||
'Search metadata online and embed into file';
|
'Search metadata online and embed into file';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldsTitle => 'Fields to update';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldCover => 'Cover Art';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldLyrics => 'Lyrics';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldBasicTags => 'Album, Album Artist';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldTrackInfo => 'Track & Disc Number';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldReleaseInfo => 'Date & ISRC';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichFieldExtra => 'Genre, Label, Copyright';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get trackReEnrichSelectAll => 'Select All';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackEditMetadata => 'Edit Metadata';
|
String get trackEditMetadata => 'Edit Metadata';
|
||||||
|
|
||||||
@@ -2674,10 +2722,6 @@ class AppLocalizationsZh extends AppLocalizations {
|
|||||||
String get lyricsProvidersDiscardContent =>
|
String get lyricsProvidersDiscardContent =>
|
||||||
'You have unsaved changes that will be lost.';
|
'You have unsaved changes that will be lost.';
|
||||||
|
|
||||||
@override
|
|
||||||
String get lyricsProviderSpotifyApiDesc =>
|
|
||||||
'Spotify-sourced synced lyrics via community API';
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
String get lyricsProviderLrclibDesc => 'Open-source synced lyrics database';
|
||||||
|
|
||||||
|
|||||||
+60
-4
@@ -89,6 +89,14 @@
|
|||||||
"@downloadFilenameFormat": {
|
"@downloadFilenameFormat": {
|
||||||
"description": "Setting for output filename pattern"
|
"description": "Setting for output filename pattern"
|
||||||
},
|
},
|
||||||
|
"downloadSingleFilenameFormat": "Single Filename Format",
|
||||||
|
"@downloadSingleFilenameFormat": {
|
||||||
|
"description": "Setting for output filename pattern for singles/EPs"
|
||||||
|
},
|
||||||
|
"downloadSingleFilenameFormatDescription": "Filename pattern for singles and EPs. Uses the same tags as the album format.",
|
||||||
|
"@downloadSingleFilenameFormatDescription": {
|
||||||
|
"description": "Subtitle description for single filename format setting"
|
||||||
|
},
|
||||||
"downloadFolderOrganization": "Folder Organization",
|
"downloadFolderOrganization": "Folder Organization",
|
||||||
"@downloadFolderOrganization": {
|
"@downloadFolderOrganization": {
|
||||||
"description": "Setting for folder structure"
|
"description": "Setting for folder structure"
|
||||||
@@ -190,6 +198,18 @@
|
|||||||
"@optionsMaxQualityCoverSubtitle": {
|
"@optionsMaxQualityCoverSubtitle": {
|
||||||
"description": "Subtitle for max quality cover"
|
"description": "Subtitle for max quality cover"
|
||||||
},
|
},
|
||||||
|
"optionsReplayGain": "ReplayGain",
|
||||||
|
"@optionsReplayGain": {
|
||||||
|
"description": "Title for ReplayGain setting toggle"
|
||||||
|
},
|
||||||
|
"optionsReplayGainSubtitleOn": "Scan loudness and embed ReplayGain tags (EBU R128)",
|
||||||
|
"@optionsReplayGainSubtitleOn": {
|
||||||
|
"description": "Subtitle when ReplayGain is enabled"
|
||||||
|
},
|
||||||
|
"optionsReplayGainSubtitleOff": "Disabled: no loudness normalization tags",
|
||||||
|
"@optionsReplayGainSubtitleOff": {
|
||||||
|
"description": "Subtitle when ReplayGain is disabled"
|
||||||
|
},
|
||||||
"optionsArtistTagMode": "Artist Tag Mode",
|
"optionsArtistTagMode": "Artist Tag Mode",
|
||||||
"@optionsArtistTagMode": {
|
"@optionsArtistTagMode": {
|
||||||
"description": "Setting title for how artist metadata is written into files"
|
"description": "Setting title for how artist metadata is written into files"
|
||||||
@@ -1543,6 +1563,14 @@
|
|||||||
"@trackLyricsNotAvailable": {
|
"@trackLyricsNotAvailable": {
|
||||||
"description": "Message when lyrics not found"
|
"description": "Message when lyrics not found"
|
||||||
},
|
},
|
||||||
|
"trackLyricsNotInFile": "No lyrics found in this file",
|
||||||
|
"@trackLyricsNotInFile": {
|
||||||
|
"description": "Message when no embedded lyrics in audio file"
|
||||||
|
},
|
||||||
|
"trackFetchOnlineLyrics": "Fetch from Online",
|
||||||
|
"@trackFetchOnlineLyrics": {
|
||||||
|
"description": "Action - fetch lyrics from online providers"
|
||||||
|
},
|
||||||
"trackLyricsTimeout": "Request timed out. Try again later.",
|
"trackLyricsTimeout": "Request timed out. Try again later.",
|
||||||
"@trackLyricsTimeout": {
|
"@trackLyricsTimeout": {
|
||||||
"description": "Message when lyrics request times out"
|
"description": "Message when lyrics request times out"
|
||||||
@@ -2954,6 +2982,38 @@
|
|||||||
"@trackReEnrichOnlineSubtitle": {
|
"@trackReEnrichOnlineSubtitle": {
|
||||||
"description": "Subtitle for re-enrich metadata action for local items"
|
"description": "Subtitle for re-enrich metadata action for local items"
|
||||||
},
|
},
|
||||||
|
"trackReEnrichFieldsTitle": "Fields to update",
|
||||||
|
"@trackReEnrichFieldsTitle": {
|
||||||
|
"description": "Section title for field selection in re-enrich dialog"
|
||||||
|
},
|
||||||
|
"trackReEnrichFieldCover": "Cover Art",
|
||||||
|
"@trackReEnrichFieldCover": {
|
||||||
|
"description": "Checkbox label for cover art field in re-enrich"
|
||||||
|
},
|
||||||
|
"trackReEnrichFieldLyrics": "Lyrics",
|
||||||
|
"@trackReEnrichFieldLyrics": {
|
||||||
|
"description": "Checkbox label for lyrics field in re-enrich"
|
||||||
|
},
|
||||||
|
"trackReEnrichFieldBasicTags": "Album, Album Artist",
|
||||||
|
"@trackReEnrichFieldBasicTags": {
|
||||||
|
"description": "Checkbox label for basic tags in re-enrich (title/artist are never overwritten)"
|
||||||
|
},
|
||||||
|
"trackReEnrichFieldTrackInfo": "Track & Disc Number",
|
||||||
|
"@trackReEnrichFieldTrackInfo": {
|
||||||
|
"description": "Checkbox label for track info in re-enrich"
|
||||||
|
},
|
||||||
|
"trackReEnrichFieldReleaseInfo": "Date & ISRC",
|
||||||
|
"@trackReEnrichFieldReleaseInfo": {
|
||||||
|
"description": "Checkbox label for release info in re-enrich"
|
||||||
|
},
|
||||||
|
"trackReEnrichFieldExtra": "Genre, Label, Copyright",
|
||||||
|
"@trackReEnrichFieldExtra": {
|
||||||
|
"description": "Checkbox label for extra metadata in re-enrich"
|
||||||
|
},
|
||||||
|
"trackReEnrichSelectAll": "Select All",
|
||||||
|
"@trackReEnrichSelectAll": {
|
||||||
|
"description": "Select all fields checkbox in re-enrich"
|
||||||
|
},
|
||||||
"trackEditMetadata": "Edit Metadata",
|
"trackEditMetadata": "Edit Metadata",
|
||||||
"@trackEditMetadata": {
|
"@trackEditMetadata": {
|
||||||
"description": "Menu action - edit embedded metadata"
|
"description": "Menu action - edit embedded metadata"
|
||||||
@@ -3551,10 +3611,6 @@
|
|||||||
"@lyricsProvidersDiscardContent": {
|
"@lyricsProvidersDiscardContent": {
|
||||||
"description": "Body text of the discard-changes dialog on lyrics provider page"
|
"description": "Body text of the discard-changes dialog on lyrics provider page"
|
||||||
},
|
},
|
||||||
"lyricsProviderSpotifyApiDesc": "Spotify-sourced synced lyrics via community API",
|
|
||||||
"@lyricsProviderSpotifyApiDesc": {
|
|
||||||
"description": "Description for Spotify Lyrics API provider"
|
|
||||||
},
|
|
||||||
"lyricsProviderLrclibDesc": "Open-source synced lyrics database",
|
"lyricsProviderLrclibDesc": "Open-source synced lyrics database",
|
||||||
"@lyricsProviderLrclibDesc": {
|
"@lyricsProviderLrclibDesc": {
|
||||||
"description": "Description for LRCLIB provider"
|
"description": "Description for LRCLIB provider"
|
||||||
|
|||||||
@@ -82,7 +82,6 @@ class _RuntimeProfile {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Widget to eagerly initialize providers that need to load data on startup
|
|
||||||
class _EagerInitialization extends ConsumerStatefulWidget {
|
class _EagerInitialization extends ConsumerStatefulWidget {
|
||||||
const _EagerInitialization({required this.child});
|
const _EagerInitialization({required this.child});
|
||||||
final Widget child;
|
final Widget child;
|
||||||
@@ -170,10 +169,8 @@ class _EagerInitializationState extends ConsumerState<_EagerInitialization>
|
|||||||
const Duration(milliseconds: 1600),
|
const Duration(milliseconds: 1600),
|
||||||
() {
|
() {
|
||||||
ref.read(localLibraryProvider);
|
ref.read(localLibraryProvider);
|
||||||
// Trigger auto-scan after initial warmup on first app launch.
|
|
||||||
if (!_autoScanTriggeredOnLaunch) {
|
if (!_autoScanTriggeredOnLaunch) {
|
||||||
_autoScanTriggeredOnLaunch = true;
|
_autoScanTriggeredOnLaunch = true;
|
||||||
// Give the provider a moment to load existing data before scanning.
|
|
||||||
Future.delayed(const Duration(milliseconds: 500), () {
|
Future.delayed(const Duration(milliseconds: 500), () {
|
||||||
if (mounted) _maybeAutoScanLocalLibrary();
|
if (mounted) _maybeAutoScanLocalLibrary();
|
||||||
});
|
});
|
||||||
@@ -182,8 +179,6 @@ class _EagerInitializationState extends ConsumerState<_EagerInitialization>
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Checks whether an automatic incremental scan should be triggered based on
|
|
||||||
/// the user's auto-scan preference and the time since the last scan.
|
|
||||||
Future<void> _maybeAutoScanLocalLibrary() async {
|
Future<void> _maybeAutoScanLocalLibrary() async {
|
||||||
if (!mounted) return;
|
if (!mounted) return;
|
||||||
|
|
||||||
@@ -204,7 +199,6 @@ class _EagerInitializationState extends ConsumerState<_EagerInitialization>
|
|||||||
|
|
||||||
switch (settings.localLibraryAutoScan) {
|
switch (settings.localLibraryAutoScan) {
|
||||||
case 'on_open':
|
case 'on_open':
|
||||||
// Cooldown of 10 minutes to prevent rapid re-scans.
|
|
||||||
if (elapsed.inMinutes < 10) return;
|
if (elapsed.inMinutes < 10) return;
|
||||||
break;
|
break;
|
||||||
case 'daily':
|
case 'daily':
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ class AppSettings {
|
|||||||
final String
|
final String
|
||||||
artistTagMode; // 'joined' or 'split_vorbis' for Vorbis-based formats
|
artistTagMode; // 'joined' or 'split_vorbis' for Vorbis-based formats
|
||||||
final bool embedLyrics;
|
final bool embedLyrics;
|
||||||
|
final bool embedReplayGain; // Calculate and embed ReplayGain tags
|
||||||
final bool maxQualityCover;
|
final bool maxQualityCover;
|
||||||
final bool isFirstLaunch;
|
final bool isFirstLaunch;
|
||||||
final int concurrentDownloads;
|
final int concurrentDownloads;
|
||||||
@@ -30,15 +31,12 @@ class AppSettings {
|
|||||||
final String historyViewMode;
|
final String historyViewMode;
|
||||||
final String historyFilterMode;
|
final String historyFilterMode;
|
||||||
final bool askQualityBeforeDownload;
|
final bool askQualityBeforeDownload;
|
||||||
final String spotifyClientId;
|
|
||||||
final String spotifyClientSecret;
|
|
||||||
final bool useCustomSpotifyCredentials;
|
|
||||||
final String metadataSource;
|
|
||||||
final bool enableLogging;
|
final bool enableLogging;
|
||||||
final bool useExtensionProviders;
|
final bool useExtensionProviders;
|
||||||
final String? searchProvider;
|
final String? searchProvider;
|
||||||
final String? homeFeedProvider;
|
final String? homeFeedProvider;
|
||||||
final bool separateSingles;
|
final bool separateSingles;
|
||||||
|
final String singleFilenameFormat;
|
||||||
final String albumFolderStructure;
|
final String albumFolderStructure;
|
||||||
final bool showExtensionStore;
|
final bool showExtensionStore;
|
||||||
final String locale;
|
final String locale;
|
||||||
@@ -93,6 +91,7 @@ class AppSettings {
|
|||||||
this.embedMetadata = true,
|
this.embedMetadata = true,
|
||||||
this.artistTagMode = artistTagModeJoined,
|
this.artistTagMode = artistTagModeJoined,
|
||||||
this.embedLyrics = true,
|
this.embedLyrics = true,
|
||||||
|
this.embedReplayGain = false,
|
||||||
this.maxQualityCover = true,
|
this.maxQualityCover = true,
|
||||||
this.isFirstLaunch = true,
|
this.isFirstLaunch = true,
|
||||||
this.concurrentDownloads = 1,
|
this.concurrentDownloads = 1,
|
||||||
@@ -107,15 +106,12 @@ class AppSettings {
|
|||||||
this.historyViewMode = 'grid',
|
this.historyViewMode = 'grid',
|
||||||
this.historyFilterMode = 'all',
|
this.historyFilterMode = 'all',
|
||||||
this.askQualityBeforeDownload = true,
|
this.askQualityBeforeDownload = true,
|
||||||
this.spotifyClientId = '',
|
|
||||||
this.spotifyClientSecret = '',
|
|
||||||
this.useCustomSpotifyCredentials = false,
|
|
||||||
this.metadataSource = 'deezer',
|
|
||||||
this.enableLogging = false,
|
this.enableLogging = false,
|
||||||
this.useExtensionProviders = true,
|
this.useExtensionProviders = true,
|
||||||
this.searchProvider,
|
this.searchProvider,
|
||||||
this.homeFeedProvider,
|
this.homeFeedProvider,
|
||||||
this.separateSingles = false,
|
this.separateSingles = false,
|
||||||
|
this.singleFilenameFormat = '{title} - {artist}',
|
||||||
this.albumFolderStructure = 'artist_album',
|
this.albumFolderStructure = 'artist_album',
|
||||||
this.showExtensionStore = true,
|
this.showExtensionStore = true,
|
||||||
this.locale = 'system',
|
this.locale = 'system',
|
||||||
@@ -134,7 +130,6 @@ class AppSettings {
|
|||||||
this.hasCompletedTutorial = false,
|
this.hasCompletedTutorial = false,
|
||||||
this.lyricsProviders = const [
|
this.lyricsProviders = const [
|
||||||
'lrclib',
|
'lrclib',
|
||||||
'spotify_api',
|
|
||||||
'musixmatch',
|
'musixmatch',
|
||||||
'netease',
|
'netease',
|
||||||
'apple_music',
|
'apple_music',
|
||||||
@@ -158,6 +153,7 @@ class AppSettings {
|
|||||||
bool? embedMetadata,
|
bool? embedMetadata,
|
||||||
String? artistTagMode,
|
String? artistTagMode,
|
||||||
bool? embedLyrics,
|
bool? embedLyrics,
|
||||||
|
bool? embedReplayGain,
|
||||||
bool? maxQualityCover,
|
bool? maxQualityCover,
|
||||||
bool? isFirstLaunch,
|
bool? isFirstLaunch,
|
||||||
int? concurrentDownloads,
|
int? concurrentDownloads,
|
||||||
@@ -172,10 +168,6 @@ class AppSettings {
|
|||||||
String? historyViewMode,
|
String? historyViewMode,
|
||||||
String? historyFilterMode,
|
String? historyFilterMode,
|
||||||
bool? askQualityBeforeDownload,
|
bool? askQualityBeforeDownload,
|
||||||
String? spotifyClientId,
|
|
||||||
String? spotifyClientSecret,
|
|
||||||
bool? useCustomSpotifyCredentials,
|
|
||||||
String? metadataSource,
|
|
||||||
bool? enableLogging,
|
bool? enableLogging,
|
||||||
bool? useExtensionProviders,
|
bool? useExtensionProviders,
|
||||||
String? searchProvider,
|
String? searchProvider,
|
||||||
@@ -183,6 +175,7 @@ class AppSettings {
|
|||||||
String? homeFeedProvider,
|
String? homeFeedProvider,
|
||||||
bool clearHomeFeedProvider = false,
|
bool clearHomeFeedProvider = false,
|
||||||
bool? separateSingles,
|
bool? separateSingles,
|
||||||
|
String? singleFilenameFormat,
|
||||||
String? albumFolderStructure,
|
String? albumFolderStructure,
|
||||||
bool? showExtensionStore,
|
bool? showExtensionStore,
|
||||||
String? locale,
|
String? locale,
|
||||||
@@ -217,6 +210,7 @@ class AppSettings {
|
|||||||
embedMetadata: embedMetadata ?? this.embedMetadata,
|
embedMetadata: embedMetadata ?? this.embedMetadata,
|
||||||
artistTagMode: artistTagMode ?? this.artistTagMode,
|
artistTagMode: artistTagMode ?? this.artistTagMode,
|
||||||
embedLyrics: embedLyrics ?? this.embedLyrics,
|
embedLyrics: embedLyrics ?? this.embedLyrics,
|
||||||
|
embedReplayGain: embedReplayGain ?? this.embedReplayGain,
|
||||||
maxQualityCover: maxQualityCover ?? this.maxQualityCover,
|
maxQualityCover: maxQualityCover ?? this.maxQualityCover,
|
||||||
isFirstLaunch: isFirstLaunch ?? this.isFirstLaunch,
|
isFirstLaunch: isFirstLaunch ?? this.isFirstLaunch,
|
||||||
concurrentDownloads: concurrentDownloads ?? this.concurrentDownloads,
|
concurrentDownloads: concurrentDownloads ?? this.concurrentDownloads,
|
||||||
@@ -235,11 +229,6 @@ class AppSettings {
|
|||||||
historyFilterMode: historyFilterMode ?? this.historyFilterMode,
|
historyFilterMode: historyFilterMode ?? this.historyFilterMode,
|
||||||
askQualityBeforeDownload:
|
askQualityBeforeDownload:
|
||||||
askQualityBeforeDownload ?? this.askQualityBeforeDownload,
|
askQualityBeforeDownload ?? this.askQualityBeforeDownload,
|
||||||
spotifyClientId: spotifyClientId ?? this.spotifyClientId,
|
|
||||||
spotifyClientSecret: spotifyClientSecret ?? this.spotifyClientSecret,
|
|
||||||
useCustomSpotifyCredentials:
|
|
||||||
useCustomSpotifyCredentials ?? this.useCustomSpotifyCredentials,
|
|
||||||
metadataSource: metadataSource ?? this.metadataSource,
|
|
||||||
enableLogging: enableLogging ?? this.enableLogging,
|
enableLogging: enableLogging ?? this.enableLogging,
|
||||||
useExtensionProviders:
|
useExtensionProviders:
|
||||||
useExtensionProviders ?? this.useExtensionProviders,
|
useExtensionProviders ?? this.useExtensionProviders,
|
||||||
@@ -250,6 +239,7 @@ class AppSettings {
|
|||||||
? null
|
? null
|
||||||
: (homeFeedProvider ?? this.homeFeedProvider),
|
: (homeFeedProvider ?? this.homeFeedProvider),
|
||||||
separateSingles: separateSingles ?? this.separateSingles,
|
separateSingles: separateSingles ?? this.separateSingles,
|
||||||
|
singleFilenameFormat: singleFilenameFormat ?? this.singleFilenameFormat,
|
||||||
albumFolderStructure: albumFolderStructure ?? this.albumFolderStructure,
|
albumFolderStructure: albumFolderStructure ?? this.albumFolderStructure,
|
||||||
showExtensionStore: showExtensionStore ?? this.showExtensionStore,
|
showExtensionStore: showExtensionStore ?? this.showExtensionStore,
|
||||||
locale: locale ?? this.locale,
|
locale: locale ?? this.locale,
|
||||||
|
|||||||
@@ -15,8 +15,9 @@ AppSettings _$AppSettingsFromJson(Map<String, dynamic> json) => AppSettings(
|
|||||||
downloadTreeUri: json['downloadTreeUri'] as String? ?? '',
|
downloadTreeUri: json['downloadTreeUri'] as String? ?? '',
|
||||||
autoFallback: json['autoFallback'] as bool? ?? true,
|
autoFallback: json['autoFallback'] as bool? ?? true,
|
||||||
embedMetadata: json['embedMetadata'] as bool? ?? true,
|
embedMetadata: json['embedMetadata'] as bool? ?? true,
|
||||||
artistTagMode: json['artistTagMode'] as String? ?? 'joined',
|
artistTagMode: json['artistTagMode'] as String? ?? artistTagModeJoined,
|
||||||
embedLyrics: json['embedLyrics'] as bool? ?? true,
|
embedLyrics: json['embedLyrics'] as bool? ?? true,
|
||||||
|
embedReplayGain: json['embedReplayGain'] as bool? ?? false,
|
||||||
maxQualityCover: json['maxQualityCover'] as bool? ?? true,
|
maxQualityCover: json['maxQualityCover'] as bool? ?? true,
|
||||||
isFirstLaunch: json['isFirstLaunch'] as bool? ?? true,
|
isFirstLaunch: json['isFirstLaunch'] as bool? ?? true,
|
||||||
concurrentDownloads: (json['concurrentDownloads'] as num?)?.toInt() ?? 1,
|
concurrentDownloads: (json['concurrentDownloads'] as num?)?.toInt() ?? 1,
|
||||||
@@ -32,16 +33,13 @@ AppSettings _$AppSettingsFromJson(Map<String, dynamic> json) => AppSettings(
|
|||||||
historyViewMode: json['historyViewMode'] as String? ?? 'grid',
|
historyViewMode: json['historyViewMode'] as String? ?? 'grid',
|
||||||
historyFilterMode: json['historyFilterMode'] as String? ?? 'all',
|
historyFilterMode: json['historyFilterMode'] as String? ?? 'all',
|
||||||
askQualityBeforeDownload: json['askQualityBeforeDownload'] as bool? ?? true,
|
askQualityBeforeDownload: json['askQualityBeforeDownload'] as bool? ?? true,
|
||||||
spotifyClientId: json['spotifyClientId'] as String? ?? '',
|
|
||||||
spotifyClientSecret: json['spotifyClientSecret'] as String? ?? '',
|
|
||||||
useCustomSpotifyCredentials:
|
|
||||||
json['useCustomSpotifyCredentials'] as bool? ?? false,
|
|
||||||
metadataSource: json['metadataSource'] as String? ?? 'deezer',
|
|
||||||
enableLogging: json['enableLogging'] as bool? ?? false,
|
enableLogging: json['enableLogging'] as bool? ?? false,
|
||||||
useExtensionProviders: json['useExtensionProviders'] as bool? ?? true,
|
useExtensionProviders: json['useExtensionProviders'] as bool? ?? true,
|
||||||
searchProvider: json['searchProvider'] as String?,
|
searchProvider: json['searchProvider'] as String?,
|
||||||
homeFeedProvider: json['homeFeedProvider'] as String?,
|
homeFeedProvider: json['homeFeedProvider'] as String?,
|
||||||
separateSingles: json['separateSingles'] as bool? ?? false,
|
separateSingles: json['separateSingles'] as bool? ?? false,
|
||||||
|
singleFilenameFormat:
|
||||||
|
json['singleFilenameFormat'] as String? ?? '{title} - {artist}',
|
||||||
albumFolderStructure:
|
albumFolderStructure:
|
||||||
json['albumFolderStructure'] as String? ?? 'artist_album',
|
json['albumFolderStructure'] as String? ?? 'artist_album',
|
||||||
showExtensionStore: json['showExtensionStore'] as bool? ?? true,
|
showExtensionStore: json['showExtensionStore'] as bool? ?? true,
|
||||||
@@ -65,14 +63,7 @@ AppSettings _$AppSettingsFromJson(Map<String, dynamic> json) => AppSettings(
|
|||||||
(json['lyricsProviders'] as List<dynamic>?)
|
(json['lyricsProviders'] as List<dynamic>?)
|
||||||
?.map((e) => e as String)
|
?.map((e) => e as String)
|
||||||
.toList() ??
|
.toList() ??
|
||||||
const [
|
const ['lrclib', 'musixmatch', 'netease', 'apple_music', 'qqmusic'],
|
||||||
'lrclib',
|
|
||||||
'spotify_api',
|
|
||||||
'musixmatch',
|
|
||||||
'netease',
|
|
||||||
'apple_music',
|
|
||||||
'qqmusic',
|
|
||||||
],
|
|
||||||
lyricsIncludeTranslationNetease:
|
lyricsIncludeTranslationNetease:
|
||||||
json['lyricsIncludeTranslationNetease'] as bool? ?? false,
|
json['lyricsIncludeTranslationNetease'] as bool? ?? false,
|
||||||
lyricsIncludeRomanizationNetease:
|
lyricsIncludeRomanizationNetease:
|
||||||
@@ -96,6 +87,7 @@ Map<String, dynamic> _$AppSettingsToJson(
|
|||||||
'embedMetadata': instance.embedMetadata,
|
'embedMetadata': instance.embedMetadata,
|
||||||
'artistTagMode': instance.artistTagMode,
|
'artistTagMode': instance.artistTagMode,
|
||||||
'embedLyrics': instance.embedLyrics,
|
'embedLyrics': instance.embedLyrics,
|
||||||
|
'embedReplayGain': instance.embedReplayGain,
|
||||||
'maxQualityCover': instance.maxQualityCover,
|
'maxQualityCover': instance.maxQualityCover,
|
||||||
'isFirstLaunch': instance.isFirstLaunch,
|
'isFirstLaunch': instance.isFirstLaunch,
|
||||||
'concurrentDownloads': instance.concurrentDownloads,
|
'concurrentDownloads': instance.concurrentDownloads,
|
||||||
@@ -111,15 +103,12 @@ Map<String, dynamic> _$AppSettingsToJson(
|
|||||||
'historyViewMode': instance.historyViewMode,
|
'historyViewMode': instance.historyViewMode,
|
||||||
'historyFilterMode': instance.historyFilterMode,
|
'historyFilterMode': instance.historyFilterMode,
|
||||||
'askQualityBeforeDownload': instance.askQualityBeforeDownload,
|
'askQualityBeforeDownload': instance.askQualityBeforeDownload,
|
||||||
'spotifyClientId': instance.spotifyClientId,
|
|
||||||
'spotifyClientSecret': instance.spotifyClientSecret,
|
|
||||||
'useCustomSpotifyCredentials': instance.useCustomSpotifyCredentials,
|
|
||||||
'metadataSource': instance.metadataSource,
|
|
||||||
'enableLogging': instance.enableLogging,
|
'enableLogging': instance.enableLogging,
|
||||||
'useExtensionProviders': instance.useExtensionProviders,
|
'useExtensionProviders': instance.useExtensionProviders,
|
||||||
'searchProvider': instance.searchProvider,
|
'searchProvider': instance.searchProvider,
|
||||||
'homeFeedProvider': instance.homeFeedProvider,
|
'homeFeedProvider': instance.homeFeedProvider,
|
||||||
'separateSingles': instance.separateSingles,
|
'separateSingles': instance.separateSingles,
|
||||||
|
'singleFilenameFormat': instance.singleFilenameFormat,
|
||||||
'albumFolderStructure': instance.albumFolderStructure,
|
'albumFolderStructure': instance.albumFolderStructure,
|
||||||
'showExtensionStore': instance.showExtensionStore,
|
'showExtensionStore': instance.showExtensionStore,
|
||||||
'locale': instance.locale,
|
'locale': instance.locale,
|
||||||
|
|||||||
+10
-8
@@ -16,12 +16,14 @@ class Track {
|
|||||||
final int duration;
|
final int duration;
|
||||||
final int? trackNumber;
|
final int? trackNumber;
|
||||||
final int? discNumber;
|
final int? discNumber;
|
||||||
|
final int? totalDiscs;
|
||||||
final String? releaseDate;
|
final String? releaseDate;
|
||||||
final String? deezerId;
|
final String? deezerId;
|
||||||
final ServiceAvailability? availability;
|
final ServiceAvailability? availability;
|
||||||
final String? source;
|
final String? source;
|
||||||
final String? albumType;
|
final String? albumType;
|
||||||
final int? totalTracks;
|
final int? totalTracks;
|
||||||
|
final String? composer;
|
||||||
final String? itemType;
|
final String? itemType;
|
||||||
|
|
||||||
const Track({
|
const Track({
|
||||||
@@ -37,38 +39,38 @@ class Track {
|
|||||||
required this.duration,
|
required this.duration,
|
||||||
this.trackNumber,
|
this.trackNumber,
|
||||||
this.discNumber,
|
this.discNumber,
|
||||||
|
this.totalDiscs,
|
||||||
this.releaseDate,
|
this.releaseDate,
|
||||||
this.deezerId,
|
this.deezerId,
|
||||||
this.availability,
|
this.availability,
|
||||||
this.source,
|
this.source,
|
||||||
this.albumType,
|
this.albumType,
|
||||||
this.totalTracks,
|
this.totalTracks,
|
||||||
|
this.composer,
|
||||||
this.itemType,
|
this.itemType,
|
||||||
});
|
});
|
||||||
|
|
||||||
bool get isSingle {
|
bool get isSingle {
|
||||||
switch (albumType?.toLowerCase()) {
|
switch (albumType?.toLowerCase()) {
|
||||||
case 'single':
|
case 'single':
|
||||||
return true;
|
|
||||||
case 'ep':
|
case 'ep':
|
||||||
final count = totalTracks;
|
return true;
|
||||||
return count == null || count <= 1;
|
|
||||||
default:
|
default:
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool get isAlbumItem => itemType == 'album';
|
bool get isAlbumItem => itemType == 'album';
|
||||||
|
|
||||||
bool get isPlaylistItem => itemType == 'playlist';
|
bool get isPlaylistItem => itemType == 'playlist';
|
||||||
|
|
||||||
bool get isArtistItem => itemType == 'artist';
|
bool get isArtistItem => itemType == 'artist';
|
||||||
|
|
||||||
bool get isCollection => isAlbumItem || isPlaylistItem || isArtistItem;
|
bool get isCollection => isAlbumItem || isPlaylistItem || isArtistItem;
|
||||||
|
|
||||||
factory Track.fromJson(Map<String, dynamic> json) => _$TrackFromJson(json);
|
factory Track.fromJson(Map<String, dynamic> json) => _$TrackFromJson(json);
|
||||||
Map<String, dynamic> toJson() => _$TrackToJson(this);
|
Map<String, dynamic> toJson() => _$TrackToJson(this);
|
||||||
|
|
||||||
bool get isFromExtension => source != null && source!.isNotEmpty;
|
bool get isFromExtension => source != null && source!.isNotEmpty;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ Track _$TrackFromJson(Map<String, dynamic> json) => Track(
|
|||||||
duration: (json['duration'] as num).toInt(),
|
duration: (json['duration'] as num).toInt(),
|
||||||
trackNumber: (json['trackNumber'] as num?)?.toInt(),
|
trackNumber: (json['trackNumber'] as num?)?.toInt(),
|
||||||
discNumber: (json['discNumber'] as num?)?.toInt(),
|
discNumber: (json['discNumber'] as num?)?.toInt(),
|
||||||
|
totalDiscs: (json['totalDiscs'] as num?)?.toInt(),
|
||||||
releaseDate: json['releaseDate'] as String?,
|
releaseDate: json['releaseDate'] as String?,
|
||||||
deezerId: json['deezerId'] as String?,
|
deezerId: json['deezerId'] as String?,
|
||||||
availability: json['availability'] == null
|
availability: json['availability'] == null
|
||||||
@@ -29,6 +30,7 @@ Track _$TrackFromJson(Map<String, dynamic> json) => Track(
|
|||||||
source: json['source'] as String?,
|
source: json['source'] as String?,
|
||||||
albumType: json['albumType'] as String?,
|
albumType: json['albumType'] as String?,
|
||||||
totalTracks: (json['totalTracks'] as num?)?.toInt(),
|
totalTracks: (json['totalTracks'] as num?)?.toInt(),
|
||||||
|
composer: json['composer'] as String?,
|
||||||
itemType: json['itemType'] as String?,
|
itemType: json['itemType'] as String?,
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -45,12 +47,14 @@ Map<String, dynamic> _$TrackToJson(Track instance) => <String, dynamic>{
|
|||||||
'duration': instance.duration,
|
'duration': instance.duration,
|
||||||
'trackNumber': instance.trackNumber,
|
'trackNumber': instance.trackNumber,
|
||||||
'discNumber': instance.discNumber,
|
'discNumber': instance.discNumber,
|
||||||
|
'totalDiscs': instance.totalDiscs,
|
||||||
'releaseDate': instance.releaseDate,
|
'releaseDate': instance.releaseDate,
|
||||||
'deezerId': instance.deezerId,
|
'deezerId': instance.deezerId,
|
||||||
'availability': instance.availability,
|
'availability': instance.availability,
|
||||||
'source': instance.source,
|
'source': instance.source,
|
||||||
'albumType': instance.albumType,
|
'albumType': instance.albumType,
|
||||||
'totalTracks': instance.totalTracks,
|
'totalTracks': instance.totalTracks,
|
||||||
|
'composer': instance.composer,
|
||||||
'itemType': instance.itemType,
|
'itemType': instance.itemType,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -1,4 +1,5 @@
|
|||||||
import 'dart:convert';
|
import 'dart:convert';
|
||||||
|
import 'package:flutter/foundation.dart';
|
||||||
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
import 'package:spotiflac_android/services/platform_bridge.dart';
|
import 'package:spotiflac_android/services/platform_bridge.dart';
|
||||||
@@ -19,6 +20,7 @@ class ExploreItem {
|
|||||||
final String? providerId;
|
final String? providerId;
|
||||||
final String? albumId;
|
final String? albumId;
|
||||||
final String? albumName;
|
final String? albumName;
|
||||||
|
final String? releaseDate;
|
||||||
final int durationMs;
|
final int durationMs;
|
||||||
|
|
||||||
const ExploreItem({
|
const ExploreItem({
|
||||||
@@ -32,6 +34,7 @@ class ExploreItem {
|
|||||||
this.providerId,
|
this.providerId,
|
||||||
this.albumId,
|
this.albumId,
|
||||||
this.albumName,
|
this.albumName,
|
||||||
|
this.releaseDate,
|
||||||
this.durationMs = 0,
|
this.durationMs = 0,
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -47,6 +50,7 @@ class ExploreItem {
|
|||||||
providerId: json['provider_id'] as String?,
|
providerId: json['provider_id'] as String?,
|
||||||
albumId: json['album_id'] as String?,
|
albumId: json['album_id'] as String?,
|
||||||
albumName: json['album_name'] as String?,
|
albumName: json['album_name'] as String?,
|
||||||
|
releaseDate: json['release_date']?.toString(),
|
||||||
durationMs: json['duration_ms'] as int? ?? 0,
|
durationMs: json['duration_ms'] as int? ?? 0,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -62,6 +66,7 @@ class ExploreItem {
|
|||||||
'provider_id': providerId,
|
'provider_id': providerId,
|
||||||
'album_id': albumId,
|
'album_id': albumId,
|
||||||
'album_name': albumName,
|
'album_name': albumName,
|
||||||
|
'release_date': releaseDate,
|
||||||
'duration_ms': durationMs,
|
'duration_ms': durationMs,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -158,6 +163,52 @@ bool _isYTMusicQuickPicksItems(List<ExploreItem> items) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
List<Map<String, Object?>> _normalizeExploreSectionsPayload(
|
||||||
|
dynamic rawSections,
|
||||||
|
) {
|
||||||
|
if (rawSections is! List) return const [];
|
||||||
|
final sections = <Map<String, Object?>>[];
|
||||||
|
for (final rawSection in rawSections) {
|
||||||
|
if (rawSection is! Map) continue;
|
||||||
|
final section = Map<Object?, Object?>.from(rawSection);
|
||||||
|
final rawItems = section['items'];
|
||||||
|
final items = <Map<String, Object?>>[];
|
||||||
|
if (rawItems is List) {
|
||||||
|
for (final rawItem in rawItems) {
|
||||||
|
if (rawItem is! Map) continue;
|
||||||
|
items.add(Map<String, Object?>.from(rawItem));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
sections.add({
|
||||||
|
'uri': section['uri']?.toString() ?? '',
|
||||||
|
'title': section['title']?.toString() ?? '',
|
||||||
|
'items': items,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return sections;
|
||||||
|
}
|
||||||
|
|
||||||
|
List<Map<String, Object?>> _decodeExploreCacheSections(String rawCache) {
|
||||||
|
final decoded = jsonDecode(rawCache);
|
||||||
|
if (decoded is! Map) return const [];
|
||||||
|
return _normalizeExploreSectionsPayload(decoded['sections']);
|
||||||
|
}
|
||||||
|
|
||||||
|
String _encodeExploreCacheSections(List<Map<String, Object?>> sections) {
|
||||||
|
return jsonEncode({'sections': sections});
|
||||||
|
}
|
||||||
|
|
||||||
|
List<ExploreSection> _buildExploreSectionsFromNormalizedPayload(
|
||||||
|
List<Map<String, Object?>> normalizedSections,
|
||||||
|
) {
|
||||||
|
return normalizedSections
|
||||||
|
.map(
|
||||||
|
(section) =>
|
||||||
|
ExploreSection.fromJson(Map<String, dynamic>.from(section)),
|
||||||
|
)
|
||||||
|
.toList(growable: false);
|
||||||
|
}
|
||||||
|
|
||||||
class ExploreNotifier extends Notifier<ExploreState> {
|
class ExploreNotifier extends Notifier<ExploreState> {
|
||||||
static const _cacheKey = 'explore_home_feed_cache';
|
static const _cacheKey = 'explore_home_feed_cache';
|
||||||
static const _cacheTsKey = 'explore_home_feed_ts';
|
static const _cacheTsKey = 'explore_home_feed_ts';
|
||||||
@@ -175,11 +226,13 @@ class ExploreNotifier extends Notifier<ExploreState> {
|
|||||||
final cachedTs = prefs.getInt(_cacheTsKey);
|
final cachedTs = prefs.getInt(_cacheTsKey);
|
||||||
if (cached == null || cached.isEmpty) return;
|
if (cached == null || cached.isEmpty) return;
|
||||||
|
|
||||||
final data = jsonDecode(cached) as Map<String, dynamic>;
|
final normalizedSections = await compute(
|
||||||
final sectionsData = data['sections'] as List<dynamic>? ?? [];
|
_decodeExploreCacheSections,
|
||||||
final sections = sectionsData
|
cached,
|
||||||
.map((s) => ExploreSection.fromJson(s as Map<String, dynamic>))
|
);
|
||||||
.toList();
|
final sections = _buildExploreSectionsFromNormalizedPayload(
|
||||||
|
normalizedSections,
|
||||||
|
);
|
||||||
|
|
||||||
if (sections.isEmpty) return;
|
if (sections.isEmpty) return;
|
||||||
|
|
||||||
@@ -198,13 +251,18 @@ class ExploreNotifier extends Notifier<ExploreState> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _saveToCache(List<ExploreSection> sections) async {
|
Future<void> _saveToCache(
|
||||||
|
List<Map<String, Object?>> normalizedSections,
|
||||||
|
) async {
|
||||||
try {
|
try {
|
||||||
final prefs = await SharedPreferences.getInstance();
|
final prefs = await SharedPreferences.getInstance();
|
||||||
final data = {'sections': sections.map((s) => s.toJson()).toList()};
|
final encoded = await compute(
|
||||||
await prefs.setString(_cacheKey, jsonEncode(data));
|
_encodeExploreCacheSections,
|
||||||
|
normalizedSections,
|
||||||
|
);
|
||||||
|
await prefs.setString(_cacheKey, encoded);
|
||||||
await prefs.setInt(_cacheTsKey, DateTime.now().millisecondsSinceEpoch);
|
await prefs.setInt(_cacheTsKey, DateTime.now().millisecondsSinceEpoch);
|
||||||
_log.d('Saved ${sections.length} explore sections to cache');
|
_log.d('Saved ${normalizedSections.length} explore sections to cache');
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
_log.w('Failed to save explore cache: $e');
|
_log.w('Failed to save explore cache: $e');
|
||||||
}
|
}
|
||||||
@@ -286,10 +344,13 @@ class ExploreNotifier extends Notifier<ExploreState> {
|
|||||||
|
|
||||||
final greeting = result['greeting'] as String?;
|
final greeting = result['greeting'] as String?;
|
||||||
final sectionsData = result['sections'] as List<dynamic>? ?? [];
|
final sectionsData = result['sections'] as List<dynamic>? ?? [];
|
||||||
|
final normalizedSections = await compute(
|
||||||
final sections = sectionsData
|
_normalizeExploreSectionsPayload,
|
||||||
.map((s) => ExploreSection.fromJson(s as Map<String, dynamic>))
|
sectionsData,
|
||||||
.toList();
|
);
|
||||||
|
final sections = _buildExploreSectionsFromNormalizedPayload(
|
||||||
|
normalizedSections,
|
||||||
|
);
|
||||||
|
|
||||||
_log.i('Fetched ${sections.length} sections');
|
_log.i('Fetched ${sections.length} sections');
|
||||||
|
|
||||||
@@ -310,7 +371,7 @@ class ExploreNotifier extends Notifier<ExploreState> {
|
|||||||
lastFetched: DateTime.now(),
|
lastFetched: DateTime.now(),
|
||||||
);
|
);
|
||||||
|
|
||||||
_saveToCache(sections);
|
_saveToCache(normalizedSections);
|
||||||
} catch (e, stack) {
|
} catch (e, stack) {
|
||||||
_log.e('Error fetching home feed: $e', e, stack);
|
_log.e('Error fetching home feed: $e', e, stack);
|
||||||
state = state.copyWith(isLoading: false, error: e.toString());
|
state = state.copyWith(isLoading: false, error: e.toString());
|
||||||
|
|||||||
@@ -33,6 +33,7 @@ class Extension {
|
|||||||
final bool hasDownloadProvider;
|
final bool hasDownloadProvider;
|
||||||
final bool hasLyricsProvider;
|
final bool hasLyricsProvider;
|
||||||
final bool skipMetadataEnrichment;
|
final bool skipMetadataEnrichment;
|
||||||
|
final bool skipLyrics;
|
||||||
final SearchBehavior? searchBehavior;
|
final SearchBehavior? searchBehavior;
|
||||||
final URLHandler? urlHandler;
|
final URLHandler? urlHandler;
|
||||||
final TrackMatching? trackMatching;
|
final TrackMatching? trackMatching;
|
||||||
@@ -57,6 +58,7 @@ class Extension {
|
|||||||
this.hasDownloadProvider = false,
|
this.hasDownloadProvider = false,
|
||||||
this.hasLyricsProvider = false,
|
this.hasLyricsProvider = false,
|
||||||
this.skipMetadataEnrichment = false,
|
this.skipMetadataEnrichment = false,
|
||||||
|
this.skipLyrics = false,
|
||||||
this.searchBehavior,
|
this.searchBehavior,
|
||||||
this.urlHandler,
|
this.urlHandler,
|
||||||
this.trackMatching,
|
this.trackMatching,
|
||||||
@@ -94,6 +96,7 @@ class Extension {
|
|||||||
hasLyricsProvider: json['has_lyrics_provider'] as bool? ?? false,
|
hasLyricsProvider: json['has_lyrics_provider'] as bool? ?? false,
|
||||||
skipMetadataEnrichment:
|
skipMetadataEnrichment:
|
||||||
json['skip_metadata_enrichment'] as bool? ?? false,
|
json['skip_metadata_enrichment'] as bool? ?? false,
|
||||||
|
skipLyrics: json['skip_lyrics'] as bool? ?? false,
|
||||||
searchBehavior: json['search_behavior'] != null
|
searchBehavior: json['search_behavior'] != null
|
||||||
? SearchBehavior.fromJson(
|
? SearchBehavior.fromJson(
|
||||||
json['search_behavior'] as Map<String, dynamic>,
|
json['search_behavior'] as Map<String, dynamic>,
|
||||||
@@ -134,6 +137,7 @@ class Extension {
|
|||||||
bool? hasDownloadProvider,
|
bool? hasDownloadProvider,
|
||||||
bool? hasLyricsProvider,
|
bool? hasLyricsProvider,
|
||||||
bool? skipMetadataEnrichment,
|
bool? skipMetadataEnrichment,
|
||||||
|
bool? skipLyrics,
|
||||||
SearchBehavior? searchBehavior,
|
SearchBehavior? searchBehavior,
|
||||||
URLHandler? urlHandler,
|
URLHandler? urlHandler,
|
||||||
TrackMatching? trackMatching,
|
TrackMatching? trackMatching,
|
||||||
@@ -159,6 +163,7 @@ class Extension {
|
|||||||
hasLyricsProvider: hasLyricsProvider ?? this.hasLyricsProvider,
|
hasLyricsProvider: hasLyricsProvider ?? this.hasLyricsProvider,
|
||||||
skipMetadataEnrichment:
|
skipMetadataEnrichment:
|
||||||
skipMetadataEnrichment ?? this.skipMetadataEnrichment,
|
skipMetadataEnrichment ?? this.skipMetadataEnrichment,
|
||||||
|
skipLyrics: skipLyrics ?? this.skipLyrics,
|
||||||
searchBehavior: searchBehavior ?? this.searchBehavior,
|
searchBehavior: searchBehavior ?? this.searchBehavior,
|
||||||
urlHandler: urlHandler ?? this.urlHandler,
|
urlHandler: urlHandler ?? this.urlHandler,
|
||||||
trackMatching: trackMatching ?? this.trackMatching,
|
trackMatching: trackMatching ?? this.trackMatching,
|
||||||
@@ -662,9 +667,8 @@ class ExtensionNotifier extends Notifier<ExtensionState> {
|
|||||||
|
|
||||||
if (settings.searchProvider == extensionId) {
|
if (settings.searchProvider == extensionId) {
|
||||||
ref.read(settingsProvider.notifier).setSearchProvider(null);
|
ref.read(settingsProvider.notifier).setSearchProvider(null);
|
||||||
ref.read(settingsProvider.notifier).setMetadataSource('deezer');
|
|
||||||
_log.d(
|
_log.d(
|
||||||
'Cleared search provider and reset to Deezer because extension $extensionId was disabled',
|
'Cleared search provider because extension $extensionId was disabled',
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import 'dart:convert';
|
import 'dart:convert';
|
||||||
import 'dart:io';
|
import 'dart:io';
|
||||||
|
|
||||||
|
import 'package:flutter/foundation.dart';
|
||||||
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
||||||
import 'package:path/path.dart' as p;
|
import 'package:path/path.dart' as p;
|
||||||
import 'package:path_provider/path_provider.dart';
|
import 'package:path_provider/path_provider.dart';
|
||||||
@@ -127,6 +128,54 @@ class UserPlaylistCollection {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class PlaylistPickerSummary {
|
||||||
|
final String id;
|
||||||
|
final String name;
|
||||||
|
final String? coverImagePath;
|
||||||
|
final String? previewCover;
|
||||||
|
final DateTime createdAt;
|
||||||
|
final DateTime updatedAt;
|
||||||
|
final int trackCount;
|
||||||
|
final bool containsAllRequestedTracks;
|
||||||
|
|
||||||
|
const PlaylistPickerSummary({
|
||||||
|
required this.id,
|
||||||
|
required this.name,
|
||||||
|
this.coverImagePath,
|
||||||
|
this.previewCover,
|
||||||
|
required this.createdAt,
|
||||||
|
required this.updatedAt,
|
||||||
|
required this.trackCount,
|
||||||
|
required this.containsAllRequestedTracks,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
class PlaylistPickerSummaryRequest {
|
||||||
|
final List<String> trackKeys;
|
||||||
|
|
||||||
|
PlaylistPickerSummaryRequest._(this.trackKeys);
|
||||||
|
|
||||||
|
factory PlaylistPickerSummaryRequest.fromTracks(Iterable<Track> tracks) {
|
||||||
|
final keys =
|
||||||
|
tracks
|
||||||
|
.map(trackCollectionKey)
|
||||||
|
.where((key) => key.trim().isNotEmpty)
|
||||||
|
.toSet()
|
||||||
|
.toList(growable: false)
|
||||||
|
..sort();
|
||||||
|
return PlaylistPickerSummaryRequest._(keys);
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
bool operator ==(Object other) =>
|
||||||
|
identical(this, other) ||
|
||||||
|
other is PlaylistPickerSummaryRequest &&
|
||||||
|
listEquals(trackKeys, other.trackKeys);
|
||||||
|
|
||||||
|
@override
|
||||||
|
int get hashCode => Object.hashAll(trackKeys);
|
||||||
|
}
|
||||||
|
|
||||||
class LibraryCollectionsState {
|
class LibraryCollectionsState {
|
||||||
final List<CollectionTrackEntry> wishlist;
|
final List<CollectionTrackEntry> wishlist;
|
||||||
final List<CollectionTrackEntry> loved;
|
final List<CollectionTrackEntry> loved;
|
||||||
@@ -280,6 +329,10 @@ class LibraryCollectionsNotifier extends Notifier<LibraryCollectionsState> {
|
|||||||
final LibraryCollectionsDatabase _db = LibraryCollectionsDatabase.instance;
|
final LibraryCollectionsDatabase _db = LibraryCollectionsDatabase.instance;
|
||||||
Future<void>? _loadFuture;
|
Future<void>? _loadFuture;
|
||||||
|
|
||||||
|
void _invalidatePlaylistPickerSummaries() {
|
||||||
|
ref.invalidate(libraryPlaylistPickerSummariesProvider);
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
LibraryCollectionsState build() {
|
LibraryCollectionsState build() {
|
||||||
_loadFuture = _load();
|
_loadFuture = _load();
|
||||||
@@ -494,6 +547,7 @@ class LibraryCollectionsNotifier extends Notifier<LibraryCollectionsState> {
|
|||||||
updatedAt: now.toIso8601String(),
|
updatedAt: now.toIso8601String(),
|
||||||
);
|
);
|
||||||
state = state.copyWith(playlists: [playlist, ...state.playlists]);
|
state = state.copyWith(playlists: [playlist, ...state.playlists]);
|
||||||
|
_invalidatePlaylistPickerSummaries();
|
||||||
return id;
|
return id;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -513,6 +567,7 @@ class LibraryCollectionsNotifier extends Notifier<LibraryCollectionsState> {
|
|||||||
_replacePlaylistById(playlistId, (playlist) {
|
_replacePlaylistById(playlistId, (playlist) {
|
||||||
return playlist.copyWith(name: trimmed, updatedAt: now);
|
return playlist.copyWith(name: trimmed, updatedAt: now);
|
||||||
});
|
});
|
||||||
|
_invalidatePlaylistPickerSummaries();
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> deletePlaylist(String playlistId) async {
|
Future<void> deletePlaylist(String playlistId) async {
|
||||||
@@ -523,6 +578,7 @@ class LibraryCollectionsNotifier extends Notifier<LibraryCollectionsState> {
|
|||||||
await _db.deletePlaylist(playlistId);
|
await _db.deletePlaylist(playlistId);
|
||||||
final updatedPlaylists = [...state.playlists]..removeAt(playlistIndex);
|
final updatedPlaylists = [...state.playlists]..removeAt(playlistIndex);
|
||||||
state = state.copyWith(playlists: updatedPlaylists);
|
state = state.copyWith(playlists: updatedPlaylists);
|
||||||
|
_invalidatePlaylistPickerSummaries();
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<bool> addTrackToPlaylist(String playlistId, Track track) async {
|
Future<bool> addTrackToPlaylist(String playlistId, Track track) async {
|
||||||
@@ -550,6 +606,7 @@ class LibraryCollectionsNotifier extends Notifier<LibraryCollectionsState> {
|
|||||||
);
|
);
|
||||||
});
|
});
|
||||||
if (!changed) return false;
|
if (!changed) return false;
|
||||||
|
_invalidatePlaylistPickerSummaries();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -615,6 +672,7 @@ class LibraryCollectionsNotifier extends Notifier<LibraryCollectionsState> {
|
|||||||
alreadyInPlaylistCount: alreadyInPlaylistCount,
|
alreadyInPlaylistCount: alreadyInPlaylistCount,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
_invalidatePlaylistPickerSummaries();
|
||||||
return PlaylistAddBatchResult(
|
return PlaylistAddBatchResult(
|
||||||
addedCount: entriesToAdd.length,
|
addedCount: entriesToAdd.length,
|
||||||
alreadyInPlaylistCount: alreadyInPlaylistCount,
|
alreadyInPlaylistCount: alreadyInPlaylistCount,
|
||||||
@@ -642,6 +700,7 @@ class LibraryCollectionsNotifier extends Notifier<LibraryCollectionsState> {
|
|||||||
if (nextTracks.length == playlist.tracks.length) return playlist;
|
if (nextTracks.length == playlist.tracks.length) return playlist;
|
||||||
return playlist.copyWith(tracks: nextTracks, updatedAt: now);
|
return playlist.copyWith(tracks: nextTracks, updatedAt: now);
|
||||||
});
|
});
|
||||||
|
_invalidatePlaylistPickerSummaries();
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<Directory> _playlistCoversDir() async {
|
Future<Directory> _playlistCoversDir() async {
|
||||||
@@ -678,6 +737,7 @@ class LibraryCollectionsNotifier extends Notifier<LibraryCollectionsState> {
|
|||||||
if (playlist.coverImagePath == destPath) return playlist;
|
if (playlist.coverImagePath == destPath) return playlist;
|
||||||
return playlist.copyWith(coverImagePath: () => destPath, updatedAt: now);
|
return playlist.copyWith(coverImagePath: () => destPath, updatedAt: now);
|
||||||
});
|
});
|
||||||
|
_invalidatePlaylistPickerSummaries();
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> removePlaylistCover(String playlistId) async {
|
Future<void> removePlaylistCover(String playlistId) async {
|
||||||
@@ -703,6 +763,7 @@ class LibraryCollectionsNotifier extends Notifier<LibraryCollectionsState> {
|
|||||||
if (playlist.coverImagePath == null) return playlist;
|
if (playlist.coverImagePath == null) return playlist;
|
||||||
return playlist.copyWith(coverImagePath: () => null, updatedAt: now);
|
return playlist.copyWith(coverImagePath: () => null, updatedAt: now);
|
||||||
});
|
});
|
||||||
|
_invalidatePlaylistPickerSummaries();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -710,3 +771,27 @@ final libraryCollectionsProvider =
|
|||||||
NotifierProvider<LibraryCollectionsNotifier, LibraryCollectionsState>(
|
NotifierProvider<LibraryCollectionsNotifier, LibraryCollectionsState>(
|
||||||
LibraryCollectionsNotifier.new,
|
LibraryCollectionsNotifier.new,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
final libraryPlaylistPickerSummariesProvider =
|
||||||
|
FutureProvider.family<
|
||||||
|
List<PlaylistPickerSummary>,
|
||||||
|
PlaylistPickerSummaryRequest
|
||||||
|
>((ref, request) async {
|
||||||
|
final db = LibraryCollectionsDatabase.instance;
|
||||||
|
await db.migrateFromSharedPreferences();
|
||||||
|
final rows = await db.loadPlaylistPickerSummaries(request.trackKeys);
|
||||||
|
return rows
|
||||||
|
.map(
|
||||||
|
(row) => PlaylistPickerSummary(
|
||||||
|
id: row.id,
|
||||||
|
name: row.name,
|
||||||
|
coverImagePath: row.coverImagePath,
|
||||||
|
previewCover: row.previewCover,
|
||||||
|
createdAt: row.createdAt,
|
||||||
|
updatedAt: row.updatedAt,
|
||||||
|
trackCount: row.trackCount,
|
||||||
|
containsAllRequestedTracks: row.containsAllRequestedTracks,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.toList(growable: false);
|
||||||
|
});
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ import 'package:spotiflac_android/utils/logger.dart';
|
|||||||
|
|
||||||
const _settingsKey = 'app_settings';
|
const _settingsKey = 'app_settings';
|
||||||
const _migrationVersionKey = 'settings_migration_version';
|
const _migrationVersionKey = 'settings_migration_version';
|
||||||
const _currentMigrationVersion = 7;
|
const _currentMigrationVersion = 9;
|
||||||
const _spotifyClientSecretKey = 'spotify_client_secret';
|
const _spotifyClientSecretKey = 'spotify_client_secret';
|
||||||
final _log = AppLogger('SettingsProvider');
|
final _log = AppLogger('SettingsProvider');
|
||||||
|
|
||||||
@@ -44,7 +44,7 @@ class SettingsNotifier extends Notifier<AppSettings> {
|
|||||||
await _normalizeSongLinkRegionIfNeeded();
|
await _normalizeSongLinkRegionIfNeeded();
|
||||||
}
|
}
|
||||||
|
|
||||||
await _retireBuiltInSpotifyProvider();
|
await _cleanupRetiredSpotifySettings();
|
||||||
|
|
||||||
LogBuffer.loggingEnabled = state.enableLogging;
|
LogBuffer.loggingEnabled = state.enableLogging;
|
||||||
|
|
||||||
@@ -86,13 +86,6 @@ class SettingsNotifier extends Notifier<AppSettings> {
|
|||||||
Future<void> _runMigrations(SharedPreferences prefs) async {
|
Future<void> _runMigrations(SharedPreferences prefs) async {
|
||||||
final lastMigration = prefs.getInt(_migrationVersionKey) ?? 0;
|
final lastMigration = prefs.getInt(_migrationVersionKey) ?? 0;
|
||||||
|
|
||||||
if (lastMigration < 1) {
|
|
||||||
if (!state.useCustomSpotifyCredentials) {
|
|
||||||
state = state.copyWith(metadataSource: 'deezer');
|
|
||||||
await _saveSettings();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (lastMigration < _currentMigrationVersion) {
|
if (lastMigration < _currentMigrationVersion) {
|
||||||
if (state.downloadTreeUri.isNotEmpty && state.storageMode != 'saf') {
|
if (state.downloadTreeUri.isNotEmpty && state.storageMode != 'saf') {
|
||||||
state = state.copyWith(storageMode: 'saf');
|
state = state.copyWith(storageMode: 'saf');
|
||||||
@@ -101,26 +94,20 @@ class SettingsNotifier extends Notifier<AppSettings> {
|
|||||||
if (!state.isFirstLaunch && !state.hasCompletedTutorial) {
|
if (!state.isFirstLaunch && !state.hasCompletedTutorial) {
|
||||||
state = state.copyWith(hasCompletedTutorial: true);
|
state = state.copyWith(hasCompletedTutorial: true);
|
||||||
}
|
}
|
||||||
// Migration 4: include Spotify Lyrics API in provider order for existing users
|
if (state.lyricsProviders.contains('spotify_api')) {
|
||||||
if (!state.lyricsProviders.contains('spotify_api')) {
|
final updatedProviders = state.lyricsProviders
|
||||||
final updatedProviders = List<String>.from(state.lyricsProviders);
|
.where((provider) => provider != 'spotify_api')
|
||||||
final lrclibIndex = updatedProviders.indexOf('lrclib');
|
.toList();
|
||||||
if (lrclibIndex >= 0) {
|
|
||||||
updatedProviders.insert(lrclibIndex + 1, 'spotify_api');
|
|
||||||
} else {
|
|
||||||
updatedProviders.add('spotify_api');
|
|
||||||
}
|
|
||||||
state = state.copyWith(lyricsProviders: updatedProviders);
|
|
||||||
}
|
|
||||||
if (state.metadataSource != 'deezer' ||
|
|
||||||
state.spotifyClientId.isNotEmpty ||
|
|
||||||
state.spotifyClientSecret.isNotEmpty ||
|
|
||||||
state.useCustomSpotifyCredentials) {
|
|
||||||
state = state.copyWith(
|
state = state.copyWith(
|
||||||
metadataSource: 'deezer',
|
lyricsProviders: updatedProviders.isEmpty
|
||||||
spotifyClientId: '',
|
? const [
|
||||||
spotifyClientSecret: '',
|
'lrclib',
|
||||||
useCustomSpotifyCredentials: false,
|
'musixmatch',
|
||||||
|
'netease',
|
||||||
|
'apple_music',
|
||||||
|
'qqmusic',
|
||||||
|
]
|
||||||
|
: updatedProviders,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
state = state.copyWith(lastSeenVersion: AppInfo.version);
|
state = state.copyWith(lastSeenVersion: AppInfo.version);
|
||||||
@@ -134,8 +121,7 @@ class SettingsNotifier extends Notifier<AppSettings> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _saveSettings() async {
|
Future<void> _saveSettings() async {
|
||||||
final settingsToSave = state.copyWith(spotifyClientSecret: '');
|
_pendingSettingsJson = jsonEncode(state.toJson());
|
||||||
_pendingSettingsJson = jsonEncode(settingsToSave.toJson());
|
|
||||||
|
|
||||||
if (_isSavingSettings) {
|
if (_isSavingSettings) {
|
||||||
_saveQueued = true;
|
_saveQueued = true;
|
||||||
@@ -186,28 +172,13 @@ class SettingsNotifier extends Notifier<AppSettings> {
|
|||||||
await _saveSettings();
|
await _saveSettings();
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _retireBuiltInSpotifyProvider() async {
|
Future<void> _cleanupRetiredSpotifySettings() async {
|
||||||
final storedSecret = await _secureStorage.read(
|
final storedSecret = await _secureStorage.read(
|
||||||
key: _spotifyClientSecretKey,
|
key: _spotifyClientSecretKey,
|
||||||
);
|
);
|
||||||
if (storedSecret != null && storedSecret.isNotEmpty) {
|
if (storedSecret != null && storedSecret.isNotEmpty) {
|
||||||
await _secureStorage.delete(key: _spotifyClientSecretKey);
|
await _secureStorage.delete(key: _spotifyClientSecretKey);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (state.metadataSource == 'deezer' &&
|
|
||||||
state.spotifyClientId.isEmpty &&
|
|
||||||
state.spotifyClientSecret.isEmpty &&
|
|
||||||
!state.useCustomSpotifyCredentials) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
state = state.copyWith(
|
|
||||||
metadataSource: 'deezer',
|
|
||||||
spotifyClientId: '',
|
|
||||||
spotifyClientSecret: '',
|
|
||||||
useCustomSpotifyCredentials: false,
|
|
||||||
);
|
|
||||||
await _saveSettings();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void setDefaultService(String service) {
|
void setDefaultService(String service) {
|
||||||
@@ -225,6 +196,11 @@ class SettingsNotifier extends Notifier<AppSettings> {
|
|||||||
_saveSettings();
|
_saveSettings();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void setSingleFilenameFormat(String format) {
|
||||||
|
state = state.copyWith(singleFilenameFormat: format);
|
||||||
|
_saveSettings();
|
||||||
|
}
|
||||||
|
|
||||||
void setDownloadDirectory(String directory) {
|
void setDownloadDirectory(String directory) {
|
||||||
state = state.copyWith(downloadDirectory: directory);
|
state = state.copyWith(downloadDirectory: directory);
|
||||||
_saveSettings();
|
_saveSettings();
|
||||||
@@ -256,6 +232,11 @@ class SettingsNotifier extends Notifier<AppSettings> {
|
|||||||
_saveSettings();
|
_saveSettings();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void setEmbedReplayGain(bool enabled) {
|
||||||
|
state = state.copyWith(embedReplayGain: enabled);
|
||||||
|
_saveSettings();
|
||||||
|
}
|
||||||
|
|
||||||
void setEmbedMetadata(bool enabled) {
|
void setEmbedMetadata(bool enabled) {
|
||||||
state = state.copyWith(embedMetadata: enabled);
|
state = state.copyWith(embedMetadata: enabled);
|
||||||
_saveSettings();
|
_saveSettings();
|
||||||
@@ -380,11 +361,6 @@ class SettingsNotifier extends Notifier<AppSettings> {
|
|||||||
_saveSettings();
|
_saveSettings();
|
||||||
}
|
}
|
||||||
|
|
||||||
void setMetadataSource(String source) {
|
|
||||||
state = state.copyWith(metadataSource: source);
|
|
||||||
_saveSettings();
|
|
||||||
}
|
|
||||||
|
|
||||||
void setSearchProvider(String? provider) {
|
void setSearchProvider(String? provider) {
|
||||||
if (provider == null || provider.isEmpty) {
|
if (provider == null || provider.isEmpty) {
|
||||||
state = state.copyWith(clearSearchProvider: true);
|
state = state.copyWith(clearSearchProvider: true);
|
||||||
|
|||||||
@@ -146,7 +146,6 @@ class StoreState {
|
|||||||
this.registryUrl = '',
|
this.registryUrl = '',
|
||||||
});
|
});
|
||||||
|
|
||||||
/// Whether a registry URL has been configured by the user.
|
|
||||||
bool get hasRegistryUrl => registryUrl.isNotEmpty;
|
bool get hasRegistryUrl => registryUrl.isNotEmpty;
|
||||||
|
|
||||||
StoreState copyWith({
|
StoreState copyWith({
|
||||||
@@ -218,7 +217,6 @@ class StoreNotifier extends Notifier<StoreState> {
|
|||||||
Future<void> initialize(String cacheDir) async {
|
Future<void> initialize(String cacheDir) async {
|
||||||
if (state.isInitialized) return;
|
if (state.isInitialized) return;
|
||||||
|
|
||||||
// Load saved registry URL early to avoid UI flash (empty → setup screen)
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
final prefs = await SharedPreferences.getInstance();
|
||||||
final savedUrl = prefs.getString(_registryUrlPrefKey) ?? '';
|
final savedUrl = prefs.getString(_registryUrlPrefKey) ?? '';
|
||||||
|
|
||||||
@@ -246,8 +244,6 @@ class StoreNotifier extends Notifier<StoreState> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Sets the registry URL, saves it, and refreshes the store.
|
|
||||||
/// The Go backend handles URL normalisation (GitHub repo → raw URL, branch detection).
|
|
||||||
Future<void> setRegistryUrl(String url) async {
|
Future<void> setRegistryUrl(String url) async {
|
||||||
final trimmed = url.trim();
|
final trimmed = url.trim();
|
||||||
if (trimmed.isEmpty) {
|
if (trimmed.isEmpty) {
|
||||||
@@ -258,10 +254,8 @@ class StoreNotifier extends Notifier<StoreState> {
|
|||||||
state = state.copyWith(isLoading: true, clearError: true);
|
state = state.copyWith(isLoading: true, clearError: true);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// Go backend resolves GitHub URLs (detects default branch) and validates HTTPS.
|
|
||||||
await PlatformBridge.setStoreRegistryUrl(trimmed);
|
await PlatformBridge.setStoreRegistryUrl(trimmed);
|
||||||
|
|
||||||
// Read back the resolved URL (may differ from input after normalisation).
|
|
||||||
final resolvedUrl = await PlatformBridge.getStoreRegistryUrl();
|
final resolvedUrl = await PlatformBridge.getStoreRegistryUrl();
|
||||||
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
final prefs = await SharedPreferences.getInstance();
|
||||||
@@ -280,13 +274,11 @@ class StoreNotifier extends Notifier<StoreState> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Removes the saved registry URL and fully detaches the repo from backend.
|
|
||||||
Future<void> removeRegistryUrl() async {
|
Future<void> removeRegistryUrl() async {
|
||||||
try {
|
try {
|
||||||
final prefs = await SharedPreferences.getInstance();
|
final prefs = await SharedPreferences.getInstance();
|
||||||
await prefs.remove(_registryUrlPrefKey);
|
await prefs.remove(_registryUrlPrefKey);
|
||||||
|
|
||||||
// Reset the URL in Go backend memory AND clear its cache
|
|
||||||
await PlatformBridge.clearStoreRegistryUrl();
|
await PlatformBridge.clearStoreRegistryUrl();
|
||||||
|
|
||||||
state = state.copyWith(
|
state = state.copyWith(
|
||||||
|
|||||||
@@ -906,9 +906,11 @@ class TrackNotifier extends Notifier<TrackState> {
|
|||||||
duration: (durationMs / 1000).round(),
|
duration: (durationMs / 1000).round(),
|
||||||
trackNumber: data['track_number'] as int?,
|
trackNumber: data['track_number'] as int?,
|
||||||
discNumber: data['disc_number'] as int?,
|
discNumber: data['disc_number'] as int?,
|
||||||
|
totalDiscs: data['total_discs'] as int?,
|
||||||
releaseDate: data['release_date'] as String?,
|
releaseDate: data['release_date'] as String?,
|
||||||
albumType: data['album_type'] as String?,
|
albumType: data['album_type'] as String?,
|
||||||
totalTracks: data['total_tracks'] as int?,
|
totalTracks: data['total_tracks'] as int?,
|
||||||
|
composer: data['composer']?.toString(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -939,10 +941,12 @@ class TrackNotifier extends Notifier<TrackState> {
|
|||||||
duration: (durationMs / 1000).round(),
|
duration: (durationMs / 1000).round(),
|
||||||
trackNumber: data['track_number'] as int?,
|
trackNumber: data['track_number'] as int?,
|
||||||
discNumber: data['disc_number'] as int?,
|
discNumber: data['disc_number'] as int?,
|
||||||
|
totalDiscs: data['total_discs'] as int?,
|
||||||
releaseDate: data['release_date']?.toString(),
|
releaseDate: data['release_date']?.toString(),
|
||||||
totalTracks: data['total_tracks'] as int?,
|
totalTracks: data['total_tracks'] as int?,
|
||||||
source: effectiveSource,
|
source: effectiveSource,
|
||||||
albumType: data['album_type']?.toString(),
|
albumType: data['album_type']?.toString(),
|
||||||
|
composer: data['composer']?.toString(),
|
||||||
itemType: itemType,
|
itemType: itemType,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import 'package:spotiflac_android/providers/local_library_provider.dart';
|
|||||||
import 'package:spotiflac_android/providers/playback_provider.dart';
|
import 'package:spotiflac_android/providers/playback_provider.dart';
|
||||||
import 'package:spotiflac_android/services/platform_bridge.dart';
|
import 'package:spotiflac_android/services/platform_bridge.dart';
|
||||||
import 'package:spotiflac_android/utils/file_access.dart';
|
import 'package:spotiflac_android/utils/file_access.dart';
|
||||||
|
import 'package:spotiflac_android/utils/image_cache_utils.dart';
|
||||||
import 'package:spotiflac_android/utils/string_utils.dart';
|
import 'package:spotiflac_android/utils/string_utils.dart';
|
||||||
import 'package:spotiflac_android/widgets/track_collection_quick_actions.dart';
|
import 'package:spotiflac_android/widgets/track_collection_quick_actions.dart';
|
||||||
import 'package:spotiflac_android/widgets/download_service_picker.dart';
|
import 'package:spotiflac_android/widgets/download_service_picker.dart';
|
||||||
@@ -138,14 +139,11 @@ class _AlbumScreenState extends ConsumerState<AlbumScreen> {
|
|||||||
return (mediaSize.height * 0.55).clamp(360.0, 520.0);
|
return (mediaSize.height * 0.55).clamp(360.0, 520.0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Upgrade cover URL to a higher resolution for full-screen display.
|
|
||||||
String? _highResCoverUrl(String? url) {
|
String? _highResCoverUrl(String? url) {
|
||||||
if (url == null) return null;
|
if (url == null) return null;
|
||||||
// Spotify CDN: upgrade 300 → 640 only (no intermediate between 640 and 2000)
|
|
||||||
if (url.contains('ab67616d00001e02')) {
|
if (url.contains('ab67616d00001e02')) {
|
||||||
return url.replaceAll('ab67616d00001e02', 'ab67616d0000b273');
|
return url.replaceAll('ab67616d00001e02', 'ab67616d0000b273');
|
||||||
}
|
}
|
||||||
// Deezer CDN: upgrade to 1000x1000
|
|
||||||
final deezerRegex = RegExp(r'/(\d+)x(\d+)-(\d+)-(\d+)-(\d+)-(\d+)\.jpg$');
|
final deezerRegex = RegExp(r'/(\d+)x(\d+)-(\d+)-(\d+)-(\d+)-(\d+)\.jpg$');
|
||||||
if (url.contains('cdn-images.dzcdn.net') && deezerRegex.hasMatch(url)) {
|
if (url.contains('cdn-images.dzcdn.net') && deezerRegex.hasMatch(url)) {
|
||||||
return url.replaceAllMapped(
|
return url.replaceAllMapped(
|
||||||
@@ -301,9 +299,11 @@ class _AlbumScreenState extends ConsumerState<AlbumScreen> {
|
|||||||
duration: ((data['duration_ms'] as int? ?? 0) / 1000).round(),
|
duration: ((data['duration_ms'] as int? ?? 0) / 1000).round(),
|
||||||
trackNumber: data['track_number'] as int?,
|
trackNumber: data['track_number'] as int?,
|
||||||
discNumber: data['disc_number'] as int?,
|
discNumber: data['disc_number'] as int?,
|
||||||
|
totalDiscs: data['total_discs'] as int?,
|
||||||
releaseDate: data['release_date'] as String?,
|
releaseDate: data['release_date'] as String?,
|
||||||
albumType: data['album_type'] as String?,
|
albumType: data['album_type'] as String?,
|
||||||
totalTracks: data['total_tracks'] as int?,
|
totalTracks: data['total_tracks'] as int?,
|
||||||
|
composer: data['composer']?.toString(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -393,6 +393,7 @@ class _AlbumScreenState extends ConsumerState<AlbumScreen> {
|
|||||||
(constraints.maxHeight - kToolbarHeight) /
|
(constraints.maxHeight - kToolbarHeight) /
|
||||||
(expandedHeight - kToolbarHeight);
|
(expandedHeight - kToolbarHeight);
|
||||||
final showContent = collapseRatio > 0.3;
|
final showContent = collapseRatio > 0.3;
|
||||||
|
final cacheWidth = coverCacheWidthForViewport(context);
|
||||||
|
|
||||||
return FlexibleSpaceBar(
|
return FlexibleSpaceBar(
|
||||||
collapseMode: CollapseMode.pin,
|
collapseMode: CollapseMode.pin,
|
||||||
@@ -404,6 +405,7 @@ class _AlbumScreenState extends ConsumerState<AlbumScreen> {
|
|||||||
imageUrl:
|
imageUrl:
|
||||||
_highResCoverUrl(widget.coverUrl) ?? widget.coverUrl!,
|
_highResCoverUrl(widget.coverUrl) ?? widget.coverUrl!,
|
||||||
fit: BoxFit.cover,
|
fit: BoxFit.cover,
|
||||||
|
memCacheWidth: cacheWidth,
|
||||||
cacheManager: CoverCacheManager.instance,
|
cacheManager: CoverCacheManager.instance,
|
||||||
placeholder: (_, _) =>
|
placeholder: (_, _) =>
|
||||||
Container(color: colorScheme.surface),
|
Container(color: colorScheme.surface),
|
||||||
|
|||||||
@@ -228,7 +228,6 @@ class _ArtistScreenState extends ConsumerState<ArtistScreen> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void _onScroll() {
|
void _onScroll() {
|
||||||
// Show title when scrolled past the header (280px trigger)
|
|
||||||
final shouldShow = _scrollController.offset > 280;
|
final shouldShow = _scrollController.offset > 280;
|
||||||
if (shouldShow != _showTitleInAppBar) {
|
if (shouldShow != _showTitleInAppBar) {
|
||||||
setState(() => _showTitleInAppBar = shouldShow);
|
setState(() => _showTitleInAppBar = shouldShow);
|
||||||
@@ -411,9 +410,11 @@ class _ArtistScreenState extends ConsumerState<ArtistScreen> {
|
|||||||
duration: (durationMs / 1000).round(),
|
duration: (durationMs / 1000).round(),
|
||||||
trackNumber: data['track_number'] as int?,
|
trackNumber: data['track_number'] as int?,
|
||||||
discNumber: data['disc_number'] as int?,
|
discNumber: data['disc_number'] as int?,
|
||||||
|
totalDiscs: data['total_discs'] as int?,
|
||||||
releaseDate: data['release_date']?.toString(),
|
releaseDate: data['release_date']?.toString(),
|
||||||
albumType: data['album_type']?.toString() ?? album?.albumType,
|
albumType: data['album_type']?.toString() ?? album?.albumType,
|
||||||
totalTracks: data['total_tracks'] as int? ?? album?.totalTracks,
|
totalTracks: data['total_tracks'] as int? ?? album?.totalTracks,
|
||||||
|
composer: data['composer']?.toString(),
|
||||||
source: data['provider_id']?.toString() ?? widget.extensionId,
|
source: data['provider_id']?.toString() ?? widget.extensionId,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -1130,9 +1131,11 @@ class _ArtistScreenState extends ConsumerState<ArtistScreen> {
|
|||||||
trackNumber:
|
trackNumber:
|
||||||
data['track_position'] as int? ?? data['track_number'] as int?,
|
data['track_position'] as int? ?? data['track_number'] as int?,
|
||||||
discNumber: data['disk_number'] as int? ?? data['disc_number'] as int?,
|
discNumber: data['disk_number'] as int? ?? data['disc_number'] as int?,
|
||||||
|
totalDiscs: data['total_discs'] as int?,
|
||||||
releaseDate: album.releaseDate,
|
releaseDate: album.releaseDate,
|
||||||
albumType: album.albumType,
|
albumType: album.albumType,
|
||||||
totalTracks: album.totalTracks,
|
totalTracks: album.totalTracks,
|
||||||
|
composer: data['composer']?.toString(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -2013,7 +2016,6 @@ class _ArtistScreenState extends ConsumerState<ArtistScreen> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Option tile for discography download bottom sheet
|
|
||||||
class _DiscographyOptionTile extends StatelessWidget {
|
class _DiscographyOptionTile extends StatelessWidget {
|
||||||
final IconData icon;
|
final IconData icon;
|
||||||
final String title;
|
final String title;
|
||||||
@@ -2051,7 +2053,6 @@ class _DiscographyOptionTile extends StatelessWidget {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Progress dialog shown while fetching album tracks
|
|
||||||
class _FetchingProgressDialog extends StatefulWidget {
|
class _FetchingProgressDialog extends StatefulWidget {
|
||||||
final int totalAlbums;
|
final int totalAlbums;
|
||||||
final VoidCallback onCancel;
|
final VoidCallback onCancel;
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import 'package:spotiflac_android/services/platform_bridge.dart';
|
|||||||
import 'package:spotiflac_android/services/history_database.dart';
|
import 'package:spotiflac_android/services/history_database.dart';
|
||||||
import 'package:spotiflac_android/l10n/l10n.dart';
|
import 'package:spotiflac_android/l10n/l10n.dart';
|
||||||
import 'package:spotiflac_android/utils/file_access.dart';
|
import 'package:spotiflac_android/utils/file_access.dart';
|
||||||
|
import 'package:spotiflac_android/utils/image_cache_utils.dart';
|
||||||
import 'package:spotiflac_android/utils/lyrics_metadata_helper.dart';
|
import 'package:spotiflac_android/utils/lyrics_metadata_helper.dart';
|
||||||
import 'package:spotiflac_android/providers/download_queue_provider.dart';
|
import 'package:spotiflac_android/providers/download_queue_provider.dart';
|
||||||
import 'package:spotiflac_android/widgets/batch_progress_dialog.dart';
|
import 'package:spotiflac_android/widgets/batch_progress_dialog.dart';
|
||||||
@@ -95,7 +96,6 @@ class _DownloadedAlbumScreenState extends ConsumerState<DownloadedAlbumScreen> {
|
|||||||
return (mediaSize.height * 0.55).clamp(360.0, 520.0);
|
return (mediaSize.height * 0.55).clamp(360.0, 520.0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Upgrade cover URL to a reasonable resolution for full-screen display.
|
|
||||||
String? _highResCoverUrl(String? url) {
|
String? _highResCoverUrl(String? url) {
|
||||||
if (url == null) return null;
|
if (url == null) return null;
|
||||||
if (url.contains('ab67616d00001e02')) {
|
if (url.contains('ab67616d00001e02')) {
|
||||||
@@ -111,7 +111,6 @@ class _DownloadedAlbumScreenState extends ConsumerState<DownloadedAlbumScreen> {
|
|||||||
return url;
|
return url;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get tracks for this album from history provider (reactive)
|
|
||||||
List<DownloadHistoryItem> _getAlbumTracks(
|
List<DownloadHistoryItem> _getAlbumTracks(
|
||||||
List<DownloadHistoryItem> allItems,
|
List<DownloadHistoryItem> allItems,
|
||||||
) {
|
) {
|
||||||
@@ -464,6 +463,7 @@ class _DownloadedAlbumScreenState extends ConsumerState<DownloadedAlbumScreen> {
|
|||||||
(constraints.maxHeight - kToolbarHeight) /
|
(constraints.maxHeight - kToolbarHeight) /
|
||||||
(expandedHeight - kToolbarHeight);
|
(expandedHeight - kToolbarHeight);
|
||||||
final showContent = collapseRatio > 0.3;
|
final showContent = collapseRatio > 0.3;
|
||||||
|
final cacheWidth = coverCacheWidthForViewport(context);
|
||||||
|
|
||||||
return FlexibleSpaceBar(
|
return FlexibleSpaceBar(
|
||||||
collapseMode: CollapseMode.pin,
|
collapseMode: CollapseMode.pin,
|
||||||
@@ -474,6 +474,9 @@ class _DownloadedAlbumScreenState extends ConsumerState<DownloadedAlbumScreen> {
|
|||||||
Image.file(
|
Image.file(
|
||||||
File(embeddedCoverPath),
|
File(embeddedCoverPath),
|
||||||
fit: BoxFit.cover,
|
fit: BoxFit.cover,
|
||||||
|
cacheWidth: cacheWidth,
|
||||||
|
gaplessPlayback: true,
|
||||||
|
filterQuality: FilterQuality.low,
|
||||||
errorBuilder: (_, _, _) =>
|
errorBuilder: (_, _, _) =>
|
||||||
Container(color: colorScheme.surface),
|
Container(color: colorScheme.surface),
|
||||||
)
|
)
|
||||||
@@ -482,6 +485,7 @@ class _DownloadedAlbumScreenState extends ConsumerState<DownloadedAlbumScreen> {
|
|||||||
imageUrl:
|
imageUrl:
|
||||||
_highResCoverUrl(widget.coverUrl) ?? widget.coverUrl!,
|
_highResCoverUrl(widget.coverUrl) ?? widget.coverUrl!,
|
||||||
fit: BoxFit.cover,
|
fit: BoxFit.cover,
|
||||||
|
memCacheWidth: cacheWidth,
|
||||||
cacheManager: CoverCacheManager.instance,
|
cacheManager: CoverCacheManager.instance,
|
||||||
placeholder: (_, _) =>
|
placeholder: (_, _) =>
|
||||||
Container(color: colorScheme.surface),
|
Container(color: colorScheme.surface),
|
||||||
@@ -641,7 +645,6 @@ class _DownloadedAlbumScreenState extends ConsumerState<DownloadedAlbumScreen> {
|
|||||||
ColorScheme colorScheme,
|
ColorScheme colorScheme,
|
||||||
List<DownloadHistoryItem> tracks,
|
List<DownloadHistoryItem> tracks,
|
||||||
) {
|
) {
|
||||||
// Info is now displayed in the full-screen cover overlay
|
|
||||||
return const SliverToBoxAdapter(child: SizedBox.shrink());
|
return const SliverToBoxAdapter(child: SizedBox.shrink());
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -848,7 +851,6 @@ class _DownloadedAlbumScreenState extends ConsumerState<DownloadedAlbumScreen> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Share selected tracks via system share sheet
|
|
||||||
Future<void> _shareSelected(List<DownloadHistoryItem> allTracks) async {
|
Future<void> _shareSelected(List<DownloadHistoryItem> allTracks) async {
|
||||||
final tracksById = {for (final t in allTracks) t.id: t};
|
final tracksById = {for (final t in allTracks) t.id: t};
|
||||||
final safUris = <String>[];
|
final safUris = <String>[];
|
||||||
@@ -1091,7 +1093,6 @@ class _DownloadedAlbumScreenState extends ConsumerState<DownloadedAlbumScreen> {
|
|||||||
for (final id in _selectedIds) {
|
for (final id in _selectedIds) {
|
||||||
final item = tracksById[id];
|
final item = tracksById[id];
|
||||||
if (item == null) continue;
|
if (item == null) continue;
|
||||||
// For SAF items, use safFileName to detect format (filePath is content:// URI)
|
|
||||||
final nameToCheck =
|
final nameToCheck =
|
||||||
(item.safFileName != null && item.safFileName!.isNotEmpty)
|
(item.safFileName != null && item.safFileName!.isNotEmpty)
|
||||||
? item.safFileName!.toLowerCase()
|
? item.safFileName!.toLowerCase()
|
||||||
|
|||||||
+307
-165
@@ -258,6 +258,20 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
List<Track>? _searchBucketsSourceTracks;
|
List<Track>? _searchBucketsSourceTracks;
|
||||||
_SearchResultBuckets? _searchBucketsCache;
|
_SearchResultBuckets? _searchBucketsCache;
|
||||||
_SearchSortOption _searchSortOption = _SearchSortOption.defaultOrder;
|
_SearchSortOption _searchSortOption = _SearchSortOption.defaultOrder;
|
||||||
|
List<SearchArtist>? _sortedArtistsSource;
|
||||||
|
_SearchSortOption? _sortedArtistsMode;
|
||||||
|
List<SearchArtist>? _sortedArtistsCache;
|
||||||
|
List<SearchAlbum>? _sortedAlbumsSource;
|
||||||
|
_SearchSortOption? _sortedAlbumsMode;
|
||||||
|
List<SearchAlbum>? _sortedAlbumsCache;
|
||||||
|
List<SearchPlaylist>? _sortedPlaylistsSource;
|
||||||
|
_SearchSortOption? _sortedPlaylistsMode;
|
||||||
|
List<SearchPlaylist>? _sortedPlaylistsCache;
|
||||||
|
List<Track>? _sortedTracksSource;
|
||||||
|
List<int>? _sortedTrackIndexesSource;
|
||||||
|
_SearchSortOption? _sortedTracksMode;
|
||||||
|
List<Track>? _sortedTracksCache;
|
||||||
|
List<int>? _sortedTrackIndexesCache;
|
||||||
|
|
||||||
double _responsiveScale({
|
double _responsiveScale({
|
||||||
required BuildContext context,
|
required BuildContext context,
|
||||||
@@ -476,6 +490,23 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
return buckets;
|
return buckets;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void _invalidateSearchSortCaches() {
|
||||||
|
_sortedArtistsSource = null;
|
||||||
|
_sortedArtistsMode = null;
|
||||||
|
_sortedArtistsCache = null;
|
||||||
|
_sortedAlbumsSource = null;
|
||||||
|
_sortedAlbumsMode = null;
|
||||||
|
_sortedAlbumsCache = null;
|
||||||
|
_sortedPlaylistsSource = null;
|
||||||
|
_sortedPlaylistsMode = null;
|
||||||
|
_sortedPlaylistsCache = null;
|
||||||
|
_sortedTracksSource = null;
|
||||||
|
_sortedTrackIndexesSource = null;
|
||||||
|
_sortedTracksMode = null;
|
||||||
|
_sortedTracksCache = null;
|
||||||
|
_sortedTrackIndexesCache = null;
|
||||||
|
}
|
||||||
|
|
||||||
void _onSearchFocusChanged() {
|
void _onSearchFocusChanged() {
|
||||||
if (mounted) {
|
if (mounted) {
|
||||||
setState(() {});
|
setState(() {});
|
||||||
@@ -496,7 +527,6 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Check if live search is available (extension is set as search provider)
|
|
||||||
bool _isLiveSearchEnabled() {
|
bool _isLiveSearchEnabled() {
|
||||||
final settings = ref.read(settingsProvider);
|
final settings = ref.read(settingsProvider);
|
||||||
final extState = ref.read(extensionProvider);
|
final extState = ref.read(extensionProvider);
|
||||||
@@ -564,7 +594,6 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Built-in search providers that are not extensions
|
|
||||||
static const _builtInSearchProviders = {'tidal', 'qobuz'};
|
static const _builtInSearchProviders = {'tidal', 'qobuz'};
|
||||||
|
|
||||||
Future<void> _performSearch(String query, {String? filterOverride}) async {
|
Future<void> _performSearch(String query, {String? filterOverride}) async {
|
||||||
@@ -579,6 +608,7 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
if (_lastSearchQuery == searchKey) return;
|
if (_lastSearchQuery == searchKey) return;
|
||||||
_lastSearchQuery = searchKey;
|
_lastSearchQuery = searchKey;
|
||||||
_searchSortOption = _SearchSortOption.defaultOrder;
|
_searchSortOption = _SearchSortOption.defaultOrder;
|
||||||
|
_invalidateSearchSortCaches();
|
||||||
|
|
||||||
final isBuiltInProvider =
|
final isBuiltInProvider =
|
||||||
searchProvider != null &&
|
searchProvider != null &&
|
||||||
@@ -599,7 +629,6 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
.read(trackProvider.notifier)
|
.read(trackProvider.notifier)
|
||||||
.customSearch(searchProvider, query, options: options);
|
.customSearch(searchProvider, query, options: options);
|
||||||
} else if (isBuiltInProvider) {
|
} else if (isBuiltInProvider) {
|
||||||
// Use built-in Tidal or Qobuz search
|
|
||||||
await ref
|
await ref
|
||||||
.read(trackProvider.notifier)
|
.read(trackProvider.notifier)
|
||||||
.search(
|
.search(
|
||||||
@@ -757,7 +786,8 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
trackName: track.name,
|
trackName: track.name,
|
||||||
artistName: track.artistName,
|
artistName: track.artistName,
|
||||||
coverUrl: track.coverUrl,
|
coverUrl: track.coverUrl,
|
||||||
recommendedService: trackState.searchSource,
|
recommendedService:
|
||||||
|
trackState.searchExtensionId ?? trackState.searchSource,
|
||||||
onSelect: (quality, service) {
|
onSelect: (quality, service) {
|
||||||
ref
|
ref
|
||||||
.read(downloadQueueProvider.notifier)
|
.read(downloadQueueProvider.notifier)
|
||||||
@@ -1121,7 +1151,7 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
title: Text(
|
title: Text(
|
||||||
context.l10n.homeTitle,
|
context.l10n.homeTitle,
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
fontSize: 20 + (14 * expandRatio), // 20 -> 34
|
fontSize: 20 + (14 * expandRatio),
|
||||||
fontWeight: FontWeight.bold,
|
fontWeight: FontWeight.bold,
|
||||||
color: colorScheme.onSurface,
|
color: colorScheme.onSurface,
|
||||||
),
|
),
|
||||||
@@ -1407,10 +1437,6 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
itemCount: itemCount,
|
itemCount: itemCount,
|
||||||
itemBuilder: (context, index) {
|
itemBuilder: (context, index) {
|
||||||
final item = items[index];
|
final item = items[index];
|
||||||
final embeddedCoverPath = DownloadedEmbeddedCoverResolver.resolve(
|
|
||||||
item.filePath,
|
|
||||||
onChanged: _onEmbeddedCoverChanged,
|
|
||||||
);
|
|
||||||
return KeyedSubtree(
|
return KeyedSubtree(
|
||||||
key: ValueKey(item.id),
|
key: ValueKey(item.id),
|
||||||
child: Semantics(
|
child: Semantics(
|
||||||
@@ -1423,48 +1449,15 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
margin: const EdgeInsets.only(right: 12),
|
margin: const EdgeInsets.only(right: 12),
|
||||||
child: Column(
|
child: Column(
|
||||||
children: [
|
children: [
|
||||||
ClipRRect(
|
_DownloadedOrRemoteCover(
|
||||||
|
downloadedFilePath: item.filePath,
|
||||||
|
imageUrl: item.coverUrl,
|
||||||
|
width: coverSize,
|
||||||
|
height: coverSize,
|
||||||
borderRadius: BorderRadius.circular(12),
|
borderRadius: BorderRadius.circular(12),
|
||||||
child: embeddedCoverPath != null
|
fallbackIcon: Icons.music_note,
|
||||||
? Image.file(
|
fallbackIconSize: 32,
|
||||||
File(embeddedCoverPath),
|
colorScheme: colorScheme,
|
||||||
width: coverSize,
|
|
||||||
height: coverSize,
|
|
||||||
fit: BoxFit.cover,
|
|
||||||
cacheWidth: (coverSize * 2).round(),
|
|
||||||
cacheHeight: (coverSize * 2).round(),
|
|
||||||
errorBuilder: (_, _, _) => Container(
|
|
||||||
width: coverSize,
|
|
||||||
height: coverSize,
|
|
||||||
color:
|
|
||||||
colorScheme.surfaceContainerHighest,
|
|
||||||
child: Icon(
|
|
||||||
Icons.music_note,
|
|
||||||
color: colorScheme.onSurfaceVariant,
|
|
||||||
size: 32,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
: item.coverUrl != null
|
|
||||||
? CachedNetworkImage(
|
|
||||||
imageUrl: item.coverUrl!,
|
|
||||||
width: coverSize,
|
|
||||||
height: coverSize,
|
|
||||||
fit: BoxFit.cover,
|
|
||||||
memCacheWidth: (coverSize * 2).round(),
|
|
||||||
memCacheHeight: (coverSize * 2).round(),
|
|
||||||
cacheManager: CoverCacheManager.instance,
|
|
||||||
)
|
|
||||||
: Container(
|
|
||||||
width: coverSize,
|
|
||||||
height: coverSize,
|
|
||||||
color: colorScheme.surfaceContainerHighest,
|
|
||||||
child: Icon(
|
|
||||||
Icons.music_note,
|
|
||||||
color: colorScheme.onSurfaceVariant,
|
|
||||||
size: 32,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
const SizedBox(height: 6),
|
const SizedBox(height: 6),
|
||||||
Text(
|
Text(
|
||||||
@@ -1495,7 +1488,7 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
) {
|
) {
|
||||||
final hasGreeting = greeting != null && greeting.isNotEmpty;
|
final hasGreeting = greeting != null && greeting.isNotEmpty;
|
||||||
final sectionOffset = hasGreeting ? 1 : 0;
|
final sectionOffset = hasGreeting ? 1 : 0;
|
||||||
final totalCount = sections.length + sectionOffset + 1; // + bottom padding
|
final totalCount = sections.length + sectionOffset + 1;
|
||||||
|
|
||||||
return [
|
return [
|
||||||
SliverList(
|
SliverList(
|
||||||
@@ -1845,10 +1838,11 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
albumName: item.albumName ?? '',
|
albumName: item.albumName ?? '',
|
||||||
albumId: item.albumId,
|
albumId: item.albumId,
|
||||||
duration: item.durationMs ~/ 1000,
|
duration: item.durationMs ~/ 1000,
|
||||||
trackNumber: 1,
|
trackNumber: null,
|
||||||
discNumber: 1,
|
discNumber: null,
|
||||||
|
totalDiscs: null,
|
||||||
isrc: null,
|
isrc: null,
|
||||||
releaseDate: null,
|
releaseDate: item.releaseDate,
|
||||||
coverUrl: item.coverUrl,
|
coverUrl: item.coverUrl,
|
||||||
source: item.providerId ?? 'spotify-web',
|
source: item.providerId ?? 'spotify-web',
|
||||||
);
|
);
|
||||||
@@ -1997,12 +1991,6 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
IconData typeIcon;
|
IconData typeIcon;
|
||||||
String typeLabel;
|
String typeLabel;
|
||||||
final isDownloaded = item.providerId == 'download';
|
final isDownloaded = item.providerId == 'download';
|
||||||
final embeddedCoverPath = isDownloaded
|
|
||||||
? DownloadedEmbeddedCoverResolver.resolve(
|
|
||||||
downloadFilePathByRecentKey['${item.type.name}:${item.id}'],
|
|
||||||
onChanged: _onEmbeddedCoverChanged,
|
|
||||||
)
|
|
||||||
: null;
|
|
||||||
|
|
||||||
switch (item.type) {
|
switch (item.type) {
|
||||||
case RecentAccessType.artist:
|
case RecentAccessType.artist:
|
||||||
@@ -2028,55 +2016,18 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
padding: const EdgeInsets.symmetric(horizontal: 4, vertical: 8),
|
padding: const EdgeInsets.symmetric(horizontal: 4, vertical: 8),
|
||||||
child: Row(
|
child: Row(
|
||||||
children: [
|
children: [
|
||||||
ClipRRect(
|
_DownloadedOrRemoteCover(
|
||||||
|
downloadedFilePath: isDownloaded
|
||||||
|
? downloadFilePathByRecentKey['${item.type.name}:${item.id}']
|
||||||
|
: null,
|
||||||
|
imageUrl: item.imageUrl,
|
||||||
|
width: 56,
|
||||||
|
height: 56,
|
||||||
borderRadius: BorderRadius.circular(
|
borderRadius: BorderRadius.circular(
|
||||||
item.type == RecentAccessType.artist ? 28 : 4,
|
item.type == RecentAccessType.artist ? 28 : 4,
|
||||||
),
|
),
|
||||||
child: embeddedCoverPath != null
|
fallbackIcon: typeIcon,
|
||||||
? Image.file(
|
colorScheme: colorScheme,
|
||||||
File(embeddedCoverPath),
|
|
||||||
width: 56,
|
|
||||||
height: 56,
|
|
||||||
fit: BoxFit.cover,
|
|
||||||
cacheWidth: 112,
|
|
||||||
cacheHeight: 112,
|
|
||||||
errorBuilder: (context, error, stackTrace) => Container(
|
|
||||||
width: 56,
|
|
||||||
height: 56,
|
|
||||||
color: colorScheme.surfaceContainerHighest,
|
|
||||||
child: Icon(
|
|
||||||
typeIcon,
|
|
||||||
color: colorScheme.onSurfaceVariant,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
: item.imageUrl != null && item.imageUrl!.isNotEmpty
|
|
||||||
? CachedNetworkImage(
|
|
||||||
imageUrl: item.imageUrl!,
|
|
||||||
width: 56,
|
|
||||||
height: 56,
|
|
||||||
fit: BoxFit.cover,
|
|
||||||
memCacheWidth: 112,
|
|
||||||
cacheManager: CoverCacheManager.instance,
|
|
||||||
errorWidget: (context, url, error) => Container(
|
|
||||||
width: 56,
|
|
||||||
height: 56,
|
|
||||||
color: colorScheme.surfaceContainerHighest,
|
|
||||||
child: Icon(
|
|
||||||
typeIcon,
|
|
||||||
color: colorScheme.onSurfaceVariant,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
: Container(
|
|
||||||
width: 56,
|
|
||||||
height: 56,
|
|
||||||
color: colorScheme.surfaceContainerHighest,
|
|
||||||
child: Icon(
|
|
||||||
typeIcon,
|
|
||||||
color: colorScheme.onSurfaceVariant,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
const SizedBox(width: 12),
|
const SizedBox(width: 12),
|
||||||
Expanded(
|
Expanded(
|
||||||
@@ -2412,8 +2363,6 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// ── Search result sorting ──────────────────────────────────────────────
|
|
||||||
|
|
||||||
String _sortOptionLabel(_SearchSortOption option) {
|
String _sortOptionLabel(_SearchSortOption option) {
|
||||||
switch (option) {
|
switch (option) {
|
||||||
case _SearchSortOption.defaultOrder:
|
case _SearchSortOption.defaultOrder:
|
||||||
@@ -2574,6 +2523,114 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
return sorted;
|
return sorted;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
List<SearchArtist>? _sortSearchArtists(List<SearchArtist>? artists) {
|
||||||
|
if (artists == null ||
|
||||||
|
artists.isEmpty ||
|
||||||
|
_searchSortOption == _SearchSortOption.defaultOrder) {
|
||||||
|
return artists;
|
||||||
|
}
|
||||||
|
if (identical(artists, _sortedArtistsSource) &&
|
||||||
|
_sortedArtistsMode == _searchSortOption &&
|
||||||
|
_sortedArtistsCache != null) {
|
||||||
|
return _sortedArtistsCache;
|
||||||
|
}
|
||||||
|
final sorted = _applySortToList<SearchArtist>(
|
||||||
|
artists,
|
||||||
|
(a) => a.name,
|
||||||
|
(a) => a.name,
|
||||||
|
(a) => 0,
|
||||||
|
(a) => null,
|
||||||
|
);
|
||||||
|
_sortedArtistsSource = artists;
|
||||||
|
_sortedArtistsMode = _searchSortOption;
|
||||||
|
_sortedArtistsCache = sorted;
|
||||||
|
return sorted;
|
||||||
|
}
|
||||||
|
|
||||||
|
List<SearchAlbum>? _sortSearchAlbums(List<SearchAlbum>? albums) {
|
||||||
|
if (albums == null ||
|
||||||
|
albums.isEmpty ||
|
||||||
|
_searchSortOption == _SearchSortOption.defaultOrder) {
|
||||||
|
return albums;
|
||||||
|
}
|
||||||
|
if (identical(albums, _sortedAlbumsSource) &&
|
||||||
|
_sortedAlbumsMode == _searchSortOption &&
|
||||||
|
_sortedAlbumsCache != null) {
|
||||||
|
return _sortedAlbumsCache;
|
||||||
|
}
|
||||||
|
final sorted = _applySortToList<SearchAlbum>(
|
||||||
|
albums,
|
||||||
|
(a) => a.name,
|
||||||
|
(a) => a.artists,
|
||||||
|
(a) => 0,
|
||||||
|
(a) => a.releaseDate,
|
||||||
|
);
|
||||||
|
_sortedAlbumsSource = albums;
|
||||||
|
_sortedAlbumsMode = _searchSortOption;
|
||||||
|
_sortedAlbumsCache = sorted;
|
||||||
|
return sorted;
|
||||||
|
}
|
||||||
|
|
||||||
|
List<SearchPlaylist>? _sortSearchPlaylists(List<SearchPlaylist>? playlists) {
|
||||||
|
if (playlists == null ||
|
||||||
|
playlists.isEmpty ||
|
||||||
|
_searchSortOption == _SearchSortOption.defaultOrder) {
|
||||||
|
return playlists;
|
||||||
|
}
|
||||||
|
if (identical(playlists, _sortedPlaylistsSource) &&
|
||||||
|
_sortedPlaylistsMode == _searchSortOption &&
|
||||||
|
_sortedPlaylistsCache != null) {
|
||||||
|
return _sortedPlaylistsCache;
|
||||||
|
}
|
||||||
|
final sorted = _applySortToList<SearchPlaylist>(
|
||||||
|
playlists,
|
||||||
|
(p) => p.name,
|
||||||
|
(p) => p.owner,
|
||||||
|
(p) => 0,
|
||||||
|
(p) => null,
|
||||||
|
);
|
||||||
|
_sortedPlaylistsSource = playlists;
|
||||||
|
_sortedPlaylistsMode = _searchSortOption;
|
||||||
|
_sortedPlaylistsCache = sorted;
|
||||||
|
return sorted;
|
||||||
|
}
|
||||||
|
|
||||||
|
({List<Track> tracks, List<int> indexes}) _sortTrackResults(
|
||||||
|
List<Track> tracks,
|
||||||
|
List<int> indexes,
|
||||||
|
) {
|
||||||
|
if (tracks.isEmpty || _searchSortOption == _SearchSortOption.defaultOrder) {
|
||||||
|
return (tracks: tracks, indexes: indexes);
|
||||||
|
}
|
||||||
|
if (identical(tracks, _sortedTracksSource) &&
|
||||||
|
identical(indexes, _sortedTrackIndexesSource) &&
|
||||||
|
_sortedTracksMode == _searchSortOption &&
|
||||||
|
_sortedTracksCache != null &&
|
||||||
|
_sortedTrackIndexesCache != null) {
|
||||||
|
return (tracks: _sortedTracksCache!, indexes: _sortedTrackIndexesCache!);
|
||||||
|
}
|
||||||
|
final paired = List.generate(
|
||||||
|
tracks.length,
|
||||||
|
(i) => (tracks[i], indexes[i]),
|
||||||
|
growable: false,
|
||||||
|
);
|
||||||
|
final sortedPairs = _applySortToList<(Track, int)>(
|
||||||
|
paired,
|
||||||
|
(p) => p.$1.name,
|
||||||
|
(p) => p.$1.artistName,
|
||||||
|
(p) => p.$1.duration,
|
||||||
|
(p) => p.$1.releaseDate,
|
||||||
|
);
|
||||||
|
final sortedTracks = sortedPairs.map((p) => p.$1).toList(growable: false);
|
||||||
|
final sortedIndexes = sortedPairs.map((p) => p.$2).toList(growable: false);
|
||||||
|
_sortedTracksSource = tracks;
|
||||||
|
_sortedTrackIndexesSource = indexes;
|
||||||
|
_sortedTracksMode = _searchSortOption;
|
||||||
|
_sortedTracksCache = sortedTracks;
|
||||||
|
_sortedTrackIndexesCache = sortedIndexes;
|
||||||
|
return (tracks: sortedTracks, indexes: sortedIndexes);
|
||||||
|
}
|
||||||
|
|
||||||
List<Widget> _buildSearchResults({
|
List<Widget> _buildSearchResults({
|
||||||
required List<Track> tracks,
|
required List<Track> tracks,
|
||||||
required List<SearchArtist>? searchArtists,
|
required List<SearchArtist>? searchArtists,
|
||||||
@@ -2607,58 +2664,12 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
final playlistItems = buckets.playlistItems;
|
final playlistItems = buckets.playlistItems;
|
||||||
final artistItems = buckets.artistItems;
|
final artistItems = buckets.artistItems;
|
||||||
|
|
||||||
final sortedArtists = searchArtists != null && searchArtists.isNotEmpty
|
final sortedArtists = _sortSearchArtists(searchArtists);
|
||||||
? _applySortToList<SearchArtist>(
|
final sortedAlbums = _sortSearchAlbums(searchAlbums);
|
||||||
searchArtists,
|
final sortedPlaylists = _sortSearchPlaylists(searchPlaylists);
|
||||||
(a) => a.name,
|
final sortedTrackResults = _sortTrackResults(realTracks, realTrackIndexes);
|
||||||
(a) => a.name,
|
final sortedTracks = sortedTrackResults.tracks;
|
||||||
(a) => 0,
|
final sortedTrackIndexes = sortedTrackResults.indexes;
|
||||||
(a) => null,
|
|
||||||
)
|
|
||||||
: searchArtists;
|
|
||||||
|
|
||||||
final sortedAlbums = searchAlbums != null && searchAlbums.isNotEmpty
|
|
||||||
? _applySortToList<SearchAlbum>(
|
|
||||||
searchAlbums,
|
|
||||||
(a) => a.name,
|
|
||||||
(a) => a.artists,
|
|
||||||
(a) => 0,
|
|
||||||
(a) => a.releaseDate,
|
|
||||||
)
|
|
||||||
: searchAlbums;
|
|
||||||
|
|
||||||
final sortedPlaylists =
|
|
||||||
searchPlaylists != null && searchPlaylists.isNotEmpty
|
|
||||||
? _applySortToList<SearchPlaylist>(
|
|
||||||
searchPlaylists,
|
|
||||||
(p) => p.name,
|
|
||||||
(p) => p.owner,
|
|
||||||
(p) => 0,
|
|
||||||
(p) => null,
|
|
||||||
)
|
|
||||||
: searchPlaylists;
|
|
||||||
|
|
||||||
List<Track> sortedTracks;
|
|
||||||
List<int> sortedTrackIndexes;
|
|
||||||
if (realTracks.isNotEmpty &&
|
|
||||||
_searchSortOption != _SearchSortOption.defaultOrder) {
|
|
||||||
final paired = List.generate(
|
|
||||||
realTracks.length,
|
|
||||||
(i) => (realTracks[i], realTrackIndexes[i]),
|
|
||||||
);
|
|
||||||
final sortedPairs = _applySortToList<(Track, int)>(
|
|
||||||
paired,
|
|
||||||
(p) => p.$1.name,
|
|
||||||
(p) => p.$1.artistName,
|
|
||||||
(p) => p.$1.duration,
|
|
||||||
(p) => p.$1.releaseDate,
|
|
||||||
);
|
|
||||||
sortedTracks = sortedPairs.map((p) => p.$1).toList();
|
|
||||||
sortedTrackIndexes = sortedPairs.map((p) => p.$2).toList();
|
|
||||||
} else {
|
|
||||||
sortedTracks = realTracks;
|
|
||||||
sortedTrackIndexes = realTrackIndexes;
|
|
||||||
}
|
|
||||||
|
|
||||||
final slivers = <Widget>[
|
final slivers = <Widget>[
|
||||||
if (error != null)
|
if (error != null)
|
||||||
@@ -2940,7 +2951,7 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
albumId: album.id,
|
albumId: album.id,
|
||||||
albumName: album.name,
|
albumName: album.name,
|
||||||
coverUrl: album.imageUrl,
|
coverUrl: album.imageUrl,
|
||||||
tracks: const [], // Will be fetched by AlbumScreen
|
tracks: const [],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
@@ -2966,7 +2977,7 @@ class _HomeTabState extends ConsumerState<HomeTab>
|
|||||||
builder: (context) => PlaylistScreen(
|
builder: (context) => PlaylistScreen(
|
||||||
playlistName: playlist.name,
|
playlistName: playlist.name,
|
||||||
coverUrl: playlist.imageUrl,
|
coverUrl: playlist.imageUrl,
|
||||||
tracks: const [], // Will be fetched
|
tracks: const [],
|
||||||
playlistId: playlist.id,
|
playlistId: playlist.id,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -3693,9 +3704,7 @@ class _TrackItemWithStatus extends ConsumerWidget {
|
|||||||
Divider(
|
Divider(
|
||||||
height: 1,
|
height: 1,
|
||||||
thickness: 1,
|
thickness: 1,
|
||||||
indent:
|
indent: thumbWidth + 24,
|
||||||
thumbWidth +
|
|
||||||
24, // Adjust divider indent based on thumbnail width
|
|
||||||
endIndent: 12,
|
endIndent: 12,
|
||||||
color: colorScheme.outlineVariant.withValues(alpha: 0.3),
|
color: colorScheme.outlineVariant.withValues(alpha: 0.3),
|
||||||
),
|
),
|
||||||
@@ -4165,6 +4174,126 @@ class _SearchPlaylistItemWidget extends StatelessWidget {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class _DownloadedOrRemoteCover extends StatefulWidget {
|
||||||
|
final String? downloadedFilePath;
|
||||||
|
final String? imageUrl;
|
||||||
|
final double width;
|
||||||
|
final double height;
|
||||||
|
final BorderRadius borderRadius;
|
||||||
|
final IconData fallbackIcon;
|
||||||
|
final double fallbackIconSize;
|
||||||
|
final ColorScheme colorScheme;
|
||||||
|
|
||||||
|
const _DownloadedOrRemoteCover({
|
||||||
|
required this.downloadedFilePath,
|
||||||
|
required this.imageUrl,
|
||||||
|
required this.width,
|
||||||
|
required this.height,
|
||||||
|
required this.borderRadius,
|
||||||
|
required this.fallbackIcon,
|
||||||
|
required this.colorScheme,
|
||||||
|
this.fallbackIconSize = 24,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<_DownloadedOrRemoteCover> createState() =>
|
||||||
|
_DownloadedOrRemoteCoverState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _DownloadedOrRemoteCoverState extends State<_DownloadedOrRemoteCover> {
|
||||||
|
String? _embeddedCoverPath;
|
||||||
|
bool _refreshScheduled = false;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_embeddedCoverPath = _resolveEmbeddedCoverPath();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void didUpdateWidget(covariant _DownloadedOrRemoteCover oldWidget) {
|
||||||
|
super.didUpdateWidget(oldWidget);
|
||||||
|
if (oldWidget.downloadedFilePath != widget.downloadedFilePath ||
|
||||||
|
oldWidget.imageUrl != widget.imageUrl) {
|
||||||
|
final nextPath = _resolveEmbeddedCoverPath();
|
||||||
|
if (nextPath != _embeddedCoverPath) {
|
||||||
|
setState(() => _embeddedCoverPath = nextPath);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
String? _resolveEmbeddedCoverPath() {
|
||||||
|
final filePath = widget.downloadedFilePath;
|
||||||
|
if (filePath == null || filePath.isEmpty) return null;
|
||||||
|
return DownloadedEmbeddedCoverResolver.resolve(
|
||||||
|
filePath,
|
||||||
|
onChanged: _onEmbeddedCoverChanged,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
void _onEmbeddedCoverChanged() {
|
||||||
|
if (!mounted || _refreshScheduled) return;
|
||||||
|
_refreshScheduled = true;
|
||||||
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
||||||
|
_refreshScheduled = false;
|
||||||
|
if (!mounted) return;
|
||||||
|
final nextPath = _resolveEmbeddedCoverPath();
|
||||||
|
if (nextPath != _embeddedCoverPath) {
|
||||||
|
setState(() => _embeddedCoverPath = nextPath);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _fallback() {
|
||||||
|
return Container(
|
||||||
|
width: widget.width,
|
||||||
|
height: widget.height,
|
||||||
|
color: widget.colorScheme.surfaceContainerHighest,
|
||||||
|
child: Icon(
|
||||||
|
widget.fallbackIcon,
|
||||||
|
color: widget.colorScheme.onSurfaceVariant,
|
||||||
|
size: widget.fallbackIconSize,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
final cacheWidth = (widget.width * 2).round();
|
||||||
|
final cacheHeight = (widget.height * 2).round();
|
||||||
|
|
||||||
|
Widget child;
|
||||||
|
if (_embeddedCoverPath != null) {
|
||||||
|
child = Image.file(
|
||||||
|
File(_embeddedCoverPath!),
|
||||||
|
width: widget.width,
|
||||||
|
height: widget.height,
|
||||||
|
fit: BoxFit.cover,
|
||||||
|
cacheWidth: cacheWidth,
|
||||||
|
cacheHeight: cacheHeight,
|
||||||
|
gaplessPlayback: true,
|
||||||
|
filterQuality: FilterQuality.low,
|
||||||
|
errorBuilder: (_, _, _) => _fallback(),
|
||||||
|
);
|
||||||
|
} else if (widget.imageUrl != null && widget.imageUrl!.isNotEmpty) {
|
||||||
|
child = CachedNetworkImage(
|
||||||
|
imageUrl: widget.imageUrl!,
|
||||||
|
width: widget.width,
|
||||||
|
height: widget.height,
|
||||||
|
fit: BoxFit.cover,
|
||||||
|
memCacheWidth: cacheWidth,
|
||||||
|
memCacheHeight: cacheHeight,
|
||||||
|
cacheManager: CoverCacheManager.instance,
|
||||||
|
errorWidget: (_, _, _) => _fallback(),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
child = _fallback();
|
||||||
|
}
|
||||||
|
|
||||||
|
return ClipRRect(borderRadius: widget.borderRadius, child: child);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
class ExtensionAlbumScreen extends ConsumerStatefulWidget {
|
class ExtensionAlbumScreen extends ConsumerStatefulWidget {
|
||||||
final String extensionId;
|
final String extensionId;
|
||||||
final String albumId;
|
final String albumId;
|
||||||
@@ -4274,6 +4403,11 @@ class _ExtensionAlbumScreenState extends ConsumerState<ExtensionAlbumScreen> {
|
|||||||
isrc: data['isrc']?.toString(),
|
isrc: data['isrc']?.toString(),
|
||||||
duration: (durationMs / 1000).round(),
|
duration: (durationMs / 1000).round(),
|
||||||
trackNumber: data['track_number'] as int?,
|
trackNumber: data['track_number'] as int?,
|
||||||
|
discNumber: data['disc_number'] as int?,
|
||||||
|
totalDiscs: data['total_discs'] as int?,
|
||||||
|
releaseDate: data['release_date']?.toString(),
|
||||||
|
totalTracks: data['total_tracks'] as int?,
|
||||||
|
composer: data['composer']?.toString(),
|
||||||
source: widget.extensionId,
|
source: widget.extensionId,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -4431,6 +4565,11 @@ class _ExtensionPlaylistScreenState
|
|||||||
isrc: data['isrc']?.toString(),
|
isrc: data['isrc']?.toString(),
|
||||||
duration: (durationMs / 1000).round(),
|
duration: (durationMs / 1000).round(),
|
||||||
trackNumber: data['track_number'] as int?,
|
trackNumber: data['track_number'] as int?,
|
||||||
|
discNumber: data['disc_number'] as int?,
|
||||||
|
totalDiscs: data['total_discs'] as int?,
|
||||||
|
releaseDate: data['release_date']?.toString(),
|
||||||
|
totalTracks: data['total_tracks'] as int?,
|
||||||
|
composer: data['composer']?.toString(),
|
||||||
source: widget.extensionId,
|
source: widget.extensionId,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -4607,7 +4746,10 @@ class _ExtensionArtistScreenState extends ConsumerState<ExtensionArtistScreen> {
|
|||||||
duration: (durationMs / 1000).round(),
|
duration: (durationMs / 1000).round(),
|
||||||
trackNumber: data['track_number'] as int?,
|
trackNumber: data['track_number'] as int?,
|
||||||
discNumber: data['disc_number'] as int?,
|
discNumber: data['disc_number'] as int?,
|
||||||
|
totalDiscs: data['total_discs'] as int?,
|
||||||
releaseDate: data['release_date']?.toString(),
|
releaseDate: data['release_date']?.toString(),
|
||||||
|
totalTracks: data['total_tracks'] as int?,
|
||||||
|
composer: data['composer']?.toString(),
|
||||||
source: (data['provider_id'] ?? widget.extensionId).toString(),
|
source: (data['provider_id'] ?? widget.extensionId).toString(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,7 +16,6 @@ import 'package:spotiflac_android/providers/settings_provider.dart';
|
|||||||
import 'package:spotiflac_android/services/cover_cache_manager.dart';
|
import 'package:spotiflac_android/services/cover_cache_manager.dart';
|
||||||
import 'package:spotiflac_android/screens/track_metadata_screen.dart';
|
import 'package:spotiflac_android/screens/track_metadata_screen.dart';
|
||||||
import 'package:spotiflac_android/widgets/download_service_picker.dart';
|
import 'package:spotiflac_android/widgets/download_service_picker.dart';
|
||||||
import 'package:spotiflac_android/widgets/bottom_sheet_option_tile.dart';
|
|
||||||
import 'package:spotiflac_android/widgets/playlist_picker_sheet.dart';
|
import 'package:spotiflac_android/widgets/playlist_picker_sheet.dart';
|
||||||
import 'package:spotiflac_android/widgets/animation_utils.dart';
|
import 'package:spotiflac_android/widgets/animation_utils.dart';
|
||||||
|
|
||||||
@@ -598,7 +597,6 @@ class _LibraryTracksFolderScreenState
|
|||||||
final customCoverPath = playlist?.coverImagePath;
|
final customCoverPath = playlist?.coverImagePath;
|
||||||
final isLovedMode = widget.mode == LibraryTracksFolderMode.loved;
|
final isLovedMode = widget.mode == LibraryTracksFolderMode.loved;
|
||||||
final isPlaylistMode = widget.mode == LibraryTracksFolderMode.playlist;
|
final isPlaylistMode = widget.mode == LibraryTracksFolderMode.playlist;
|
||||||
// Loved always shows the heart icon (like Spotify's Liked Songs)
|
|
||||||
final coverUrl = isLovedMode ? null : _firstCoverUrl(entries, localState);
|
final coverUrl = isLovedMode ? null : _firstCoverUrl(entries, localState);
|
||||||
final hasCustomCover =
|
final hasCustomCover =
|
||||||
customCoverPath != null && customCoverPath.isNotEmpty;
|
customCoverPath != null && customCoverPath.isNotEmpty;
|
||||||
@@ -668,7 +666,6 @@ class _LibraryTracksFolderScreenState
|
|||||||
background: Stack(
|
background: Stack(
|
||||||
fit: StackFit.expand,
|
fit: StackFit.expand,
|
||||||
children: [
|
children: [
|
||||||
// Cover background: custom > first track URL > icon
|
|
||||||
if (hasCustomCover)
|
if (hasCustomCover)
|
||||||
Image.file(
|
Image.file(
|
||||||
File(customCoverPath),
|
File(customCoverPath),
|
||||||
@@ -1238,23 +1235,19 @@ class _CollectionTrackTile extends ConsumerWidget {
|
|||||||
trailing: isSelectionMode
|
trailing: isSelectionMode
|
||||||
? null
|
? null
|
||||||
: historyItem != null || localItem != null
|
: historyItem != null || localItem != null
|
||||||
? IconButton(
|
? IconButton(
|
||||||
tooltip: context.l10n.tooltipPlay,
|
tooltip: context.l10n.tooltipPlay,
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
ref
|
ref.read(playbackProvider.notifier).playTrackList([track]);
|
||||||
.read(playbackProvider.notifier)
|
},
|
||||||
.playTrackList([track]);
|
icon: Icon(Icons.play_arrow, color: colorScheme.primary),
|
||||||
},
|
style: IconButton.styleFrom(
|
||||||
icon: Icon(
|
backgroundColor: colorScheme.primaryContainer.withValues(
|
||||||
Icons.play_arrow,
|
alpha: 0.3,
|
||||||
color: colorScheme.primary,
|
),
|
||||||
),
|
),
|
||||||
style: IconButton.styleFrom(
|
)
|
||||||
backgroundColor: colorScheme.primaryContainer
|
: null,
|
||||||
.withValues(alpha: 0.3),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
: null,
|
|
||||||
onTap: isSelectionMode
|
onTap: isSelectionMode
|
||||||
? onTap
|
? onTap
|
||||||
: () {
|
: () {
|
||||||
@@ -1333,155 +1326,6 @@ class _CollectionTrackTile extends ConsumerWidget {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
void _showTrackOptionsSheet(BuildContext context, WidgetRef ref) {
|
|
||||||
final track = entry.track;
|
|
||||||
final effectiveCoverUrl = _resolveCoverUrl(track);
|
|
||||||
final colorScheme = Theme.of(context).colorScheme;
|
|
||||||
final historyState = ref.read(downloadHistoryProvider);
|
|
||||||
final isDownloaded =
|
|
||||||
historyState.isDownloaded(track.id) ||
|
|
||||||
(track.isrc != null &&
|
|
||||||
track.isrc!.isNotEmpty &&
|
|
||||||
historyState.getByIsrc(track.isrc!) != null) ||
|
|
||||||
historyState.findByTrackAndArtist(track.name, track.artistName) != null;
|
|
||||||
// Wishlist: only show "Add to Playlist" if track is already downloaded
|
|
||||||
final showAddToPlaylist =
|
|
||||||
mode != LibraryTracksFolderMode.wishlist || isDownloaded;
|
|
||||||
|
|
||||||
showModalBottomSheet<void>(
|
|
||||||
context: context,
|
|
||||||
useRootNavigator: true,
|
|
||||||
backgroundColor: colorScheme.surfaceContainerHigh,
|
|
||||||
shape: const RoundedRectangleBorder(
|
|
||||||
borderRadius: BorderRadius.vertical(top: Radius.circular(28)),
|
|
||||||
),
|
|
||||||
builder: (sheetContext) => SafeArea(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Column(
|
|
||||||
children: [
|
|
||||||
const SizedBox(height: 8),
|
|
||||||
Container(
|
|
||||||
width: 40,
|
|
||||||
height: 4,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: colorScheme.onSurfaceVariant.withValues(alpha: 0.4),
|
|
||||||
borderRadius: BorderRadius.circular(2),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.fromLTRB(16, 12, 16, 12),
|
|
||||||
child: Row(
|
|
||||||
children: [
|
|
||||||
ClipRRect(
|
|
||||||
borderRadius: BorderRadius.circular(8),
|
|
||||||
child:
|
|
||||||
effectiveCoverUrl != null &&
|
|
||||||
effectiveCoverUrl.isNotEmpty
|
|
||||||
? _buildTrackCover(context, effectiveCoverUrl, 56)
|
|
||||||
: Container(
|
|
||||||
width: 56,
|
|
||||||
height: 56,
|
|
||||||
color: colorScheme.surfaceContainerHighest,
|
|
||||||
child: Icon(
|
|
||||||
Icons.music_note,
|
|
||||||
color: colorScheme.onSurfaceVariant,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(width: 12),
|
|
||||||
Expanded(
|
|
||||||
child: Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
track.name,
|
|
||||||
style: Theme.of(context).textTheme.titleMedium
|
|
||||||
?.copyWith(fontWeight: FontWeight.w600),
|
|
||||||
maxLines: 1,
|
|
||||||
overflow: TextOverflow.ellipsis,
|
|
||||||
),
|
|
||||||
const SizedBox(height: 2),
|
|
||||||
Text(
|
|
||||||
track.artistName,
|
|
||||||
style: Theme.of(context).textTheme.bodyMedium
|
|
||||||
?.copyWith(
|
|
||||||
color: colorScheme.onSurfaceVariant,
|
|
||||||
),
|
|
||||||
maxLines: 1,
|
|
||||||
overflow: TextOverflow.ellipsis,
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
Divider(
|
|
||||||
height: 1,
|
|
||||||
color: colorScheme.outlineVariant.withValues(alpha: 0.5),
|
|
||||||
),
|
|
||||||
|
|
||||||
if (showAddToPlaylist)
|
|
||||||
BottomSheetOptionTile(
|
|
||||||
icon: Icons.playlist_add,
|
|
||||||
title: context.l10n.collectionAddToPlaylist,
|
|
||||||
onTap: () {
|
|
||||||
Navigator.pop(sheetContext);
|
|
||||||
showAddTrackToPlaylistSheet(context, ref, track);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
|
|
||||||
BottomSheetOptionTile(
|
|
||||||
icon: Icons.remove_circle_outline,
|
|
||||||
iconColor: colorScheme.error,
|
|
||||||
title: mode == LibraryTracksFolderMode.playlist
|
|
||||||
? context.l10n.collectionRemoveFromPlaylist
|
|
||||||
: context.l10n.collectionRemoveFromFolder,
|
|
||||||
onTap: () {
|
|
||||||
Navigator.pop(sheetContext);
|
|
||||||
_removeFromCurrentFolder(context, ref);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
|
|
||||||
const SizedBox(height: 16),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> _removeFromCurrentFolder(
|
|
||||||
BuildContext context,
|
|
||||||
WidgetRef ref,
|
|
||||||
) async {
|
|
||||||
final notifier = ref.read(libraryCollectionsProvider.notifier);
|
|
||||||
final key = entry.key;
|
|
||||||
|
|
||||||
switch (mode) {
|
|
||||||
case LibraryTracksFolderMode.wishlist:
|
|
||||||
await notifier.removeFromWishlist(key);
|
|
||||||
break;
|
|
||||||
case LibraryTracksFolderMode.loved:
|
|
||||||
await notifier.removeFromLoved(key);
|
|
||||||
break;
|
|
||||||
case LibraryTracksFolderMode.playlist:
|
|
||||||
if (playlistId != null) {
|
|
||||||
await notifier.removeTrackFromPlaylist(playlistId!, key);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!context.mounted) return;
|
|
||||||
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
SnackBar(content: Text(context.l10n.collectionRemoved(entry.track.name))),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void _downloadTrack(BuildContext context, WidgetRef ref) {
|
void _downloadTrack(BuildContext context, WidgetRef ref) {
|
||||||
final track = entry.track;
|
final track = entry.track;
|
||||||
final settings = ref.read(settingsProvider);
|
final settings = ref.read(settingsProvider);
|
||||||
@@ -1518,15 +1362,12 @@ class _CollectionTrackTile extends ConsumerWidget {
|
|||||||
final track = entry.track;
|
final track = entry.track;
|
||||||
final historyState = ref.read(downloadHistoryProvider);
|
final historyState = ref.read(downloadHistoryProvider);
|
||||||
|
|
||||||
// 1. Download history by Spotify ID
|
|
||||||
var historyItem = historyState.getBySpotifyId(track.id);
|
var historyItem = historyState.getBySpotifyId(track.id);
|
||||||
|
|
||||||
// 2. Download history by ISRC
|
|
||||||
if (historyItem == null && track.isrc != null && track.isrc!.isNotEmpty) {
|
if (historyItem == null && track.isrc != null && track.isrc!.isNotEmpty) {
|
||||||
historyItem = historyState.getByIsrc(track.isrc!);
|
historyItem = historyState.getByIsrc(track.isrc!);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3. Download history by track name + artist (handles ID/ISRC mismatch)
|
|
||||||
historyItem ??= historyState.findByTrackAndArtist(
|
historyItem ??= historyState.findByTrackAndArtist(
|
||||||
track.name,
|
track.name,
|
||||||
track.artistName,
|
track.artistName,
|
||||||
@@ -1539,14 +1380,12 @@ class _CollectionTrackTile extends ConsumerWidget {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 4. Local library by ISRC
|
|
||||||
final localState = ref.read(localLibraryProvider);
|
final localState = ref.read(localLibraryProvider);
|
||||||
LocalLibraryItem? localItem;
|
LocalLibraryItem? localItem;
|
||||||
if (track.isrc != null && track.isrc!.isNotEmpty) {
|
if (track.isrc != null && track.isrc!.isNotEmpty) {
|
||||||
localItem = localState.getByIsrc(track.isrc!);
|
localItem = localState.getByIsrc(track.isrc!);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 5. Local library by track name + artist
|
|
||||||
localItem ??= localState.findByTrackAndArtist(track.name, track.artistName);
|
localItem ??= localState.findByTrackAndArtist(track.name, track.artistName);
|
||||||
|
|
||||||
if (localItem != null) {
|
if (localItem != null) {
|
||||||
@@ -1556,7 +1395,6 @@ class _CollectionTrackTile extends ConsumerWidget {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 6. Not found anywhere — offer to download
|
|
||||||
_downloadTrack(context, ref);
|
_downloadTrack(context, ref);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,11 +9,13 @@ import 'package:spotiflac_android/providers/download_queue_provider.dart';
|
|||||||
import 'package:spotiflac_android/providers/extension_provider.dart';
|
import 'package:spotiflac_android/providers/extension_provider.dart';
|
||||||
import 'package:spotiflac_android/providers/settings_provider.dart';
|
import 'package:spotiflac_android/providers/settings_provider.dart';
|
||||||
import 'package:spotiflac_android/utils/file_access.dart';
|
import 'package:spotiflac_android/utils/file_access.dart';
|
||||||
|
import 'package:spotiflac_android/utils/image_cache_utils.dart';
|
||||||
import 'package:spotiflac_android/utils/lyrics_metadata_helper.dart';
|
import 'package:spotiflac_android/utils/lyrics_metadata_helper.dart';
|
||||||
import 'package:spotiflac_android/services/library_database.dart';
|
import 'package:spotiflac_android/services/library_database.dart';
|
||||||
import 'package:spotiflac_android/services/ffmpeg_service.dart';
|
import 'package:spotiflac_android/services/ffmpeg_service.dart';
|
||||||
import 'package:spotiflac_android/services/local_track_redownload_service.dart';
|
import 'package:spotiflac_android/services/local_track_redownload_service.dart';
|
||||||
import 'package:spotiflac_android/widgets/batch_progress_dialog.dart';
|
import 'package:spotiflac_android/widgets/batch_progress_dialog.dart';
|
||||||
|
import 'package:spotiflac_android/widgets/re_enrich_field_dialog.dart';
|
||||||
import 'package:spotiflac_android/services/platform_bridge.dart';
|
import 'package:spotiflac_android/services/platform_bridge.dart';
|
||||||
import 'package:spotiflac_android/providers/local_library_provider.dart';
|
import 'package:spotiflac_android/providers/local_library_provider.dart';
|
||||||
import 'package:spotiflac_android/providers/playback_provider.dart';
|
import 'package:spotiflac_android/providers/playback_provider.dart';
|
||||||
@@ -335,6 +337,7 @@ class _LocalAlbumScreenState extends ConsumerState<LocalAlbumScreen> {
|
|||||||
(constraints.maxHeight - kToolbarHeight) /
|
(constraints.maxHeight - kToolbarHeight) /
|
||||||
(expandedHeight - kToolbarHeight);
|
(expandedHeight - kToolbarHeight);
|
||||||
final showContent = collapseRatio > 0.3;
|
final showContent = collapseRatio > 0.3;
|
||||||
|
final cacheWidth = coverCacheWidthForViewport(context);
|
||||||
|
|
||||||
return FlexibleSpaceBar(
|
return FlexibleSpaceBar(
|
||||||
collapseMode: CollapseMode.pin,
|
collapseMode: CollapseMode.pin,
|
||||||
@@ -345,6 +348,9 @@ class _LocalAlbumScreenState extends ConsumerState<LocalAlbumScreen> {
|
|||||||
Image.file(
|
Image.file(
|
||||||
File(widget.coverPath!),
|
File(widget.coverPath!),
|
||||||
fit: BoxFit.cover,
|
fit: BoxFit.cover,
|
||||||
|
cacheWidth: cacheWidth,
|
||||||
|
gaplessPlayback: true,
|
||||||
|
filterQuality: FilterQuality.low,
|
||||||
errorBuilder: (_, _, _) =>
|
errorBuilder: (_, _, _) =>
|
||||||
Container(color: colorScheme.surface),
|
Container(color: colorScheme.surface),
|
||||||
)
|
)
|
||||||
@@ -525,7 +531,6 @@ class _LocalAlbumScreenState extends ConsumerState<LocalAlbumScreen> {
|
|||||||
ColorScheme colorScheme,
|
ColorScheme colorScheme,
|
||||||
List<LocalLibraryItem> tracks,
|
List<LocalLibraryItem> tracks,
|
||||||
) {
|
) {
|
||||||
// Info is now displayed in the full-screen cover overlay
|
|
||||||
return const SliverToBoxAdapter(child: SizedBox.shrink());
|
return const SliverToBoxAdapter(child: SizedBox.shrink());
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -824,12 +829,14 @@ class _LocalAlbumScreenState extends ConsumerState<LocalAlbumScreen> {
|
|||||||
mp3Path: ffmpegTarget,
|
mp3Path: ffmpegTarget,
|
||||||
coverPath: effectiveCoverPath,
|
coverPath: effectiveCoverPath,
|
||||||
metadata: metadata,
|
metadata: metadata,
|
||||||
|
preserveMetadata: true,
|
||||||
);
|
);
|
||||||
} else if (isM4A) {
|
} else if (isM4A) {
|
||||||
ffmpegResult = await FFmpegService.embedMetadataToM4a(
|
ffmpegResult = await FFmpegService.embedMetadataToM4a(
|
||||||
m4aPath: ffmpegTarget,
|
m4aPath: ffmpegTarget,
|
||||||
coverPath: effectiveCoverPath,
|
coverPath: effectiveCoverPath,
|
||||||
metadata: metadata,
|
metadata: metadata,
|
||||||
|
preserveMetadata: true,
|
||||||
);
|
);
|
||||||
} else if (isOpus) {
|
} else if (isOpus) {
|
||||||
ffmpegResult = await FFmpegService.embedMetadataToOpus(
|
ffmpegResult = await FFmpegService.embedMetadataToOpus(
|
||||||
@@ -837,6 +844,7 @@ class _LocalAlbumScreenState extends ConsumerState<LocalAlbumScreen> {
|
|||||||
coverPath: effectiveCoverPath,
|
coverPath: effectiveCoverPath,
|
||||||
metadata: metadata,
|
metadata: metadata,
|
||||||
artistTagMode: artistTagMode,
|
artistTagMode: artistTagMode,
|
||||||
|
preserveMetadata: true,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -867,7 +875,10 @@ class _LocalAlbumScreenState extends ConsumerState<LocalAlbumScreen> {
|
|||||||
return ffmpegResult != null;
|
return ffmpegResult != null;
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<bool> _reEnrichLocalTrack(LocalLibraryItem item) async {
|
Future<bool> _reEnrichLocalTrack(
|
||||||
|
LocalLibraryItem item, {
|
||||||
|
List<String>? updateFields,
|
||||||
|
}) async {
|
||||||
final durationMs = (item.duration ?? 0) * 1000;
|
final durationMs = (item.duration ?? 0) * 1000;
|
||||||
final artistTagMode = ref.read(settingsProvider).artistTagMode;
|
final artistTagMode = ref.read(settingsProvider).artistTagMode;
|
||||||
final request = <String, dynamic>{
|
final request = <String, dynamic>{
|
||||||
@@ -890,6 +901,8 @@ class _LocalAlbumScreenState extends ConsumerState<LocalAlbumScreen> {
|
|||||||
'copyright': '',
|
'copyright': '',
|
||||||
'duration_ms': durationMs,
|
'duration_ms': durationMs,
|
||||||
'search_online': true,
|
'search_online': true,
|
||||||
|
// ignore: use_null_aware_elements
|
||||||
|
if (updateFields != null) 'update_fields': updateFields,
|
||||||
};
|
};
|
||||||
|
|
||||||
final result = await PlatformBridge.reEnrichFile(request);
|
final result = await PlatformBridge.reEnrichFile(request);
|
||||||
@@ -1048,31 +1061,24 @@ class _LocalAlbumScreenState extends ConsumerState<LocalAlbumScreen> {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
final confirmed = await showDialog<bool>(
|
// The bar uses AnimatedPositioned (250ms), so wait for the slide-out.
|
||||||
context: context,
|
setState(() => _isSelectionMode = false);
|
||||||
builder: (ctx) => AlertDialog(
|
await Future<void>.delayed(const Duration(milliseconds: 300));
|
||||||
title: Text(context.l10n.trackReEnrich),
|
if (!mounted) return;
|
||||||
content: Text(
|
|
||||||
'${context.l10n.trackReEnrichOnlineSubtitle}\n\n'
|
final selection = await showReEnrichFieldDialog(
|
||||||
'${context.l10n.downloadedAlbumSelectedCount(selected.length)}',
|
context,
|
||||||
),
|
selectedCount: selected.length,
|
||||||
actions: [
|
|
||||||
TextButton(
|
|
||||||
onPressed: () => Navigator.pop(ctx, false),
|
|
||||||
child: Text(context.l10n.dialogCancel),
|
|
||||||
),
|
|
||||||
FilledButton(
|
|
||||||
onPressed: () => Navigator.pop(ctx, true),
|
|
||||||
child: Text(context.l10n.trackReEnrich),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
);
|
||||||
|
|
||||||
if (confirmed != true || !mounted) {
|
if (selection == null || !mounted) {
|
||||||
|
// Cancelled — restore selection mode (IDs are still intact).
|
||||||
|
if (mounted) setState(() => _isSelectionMode = true);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
final updateFields = selection.isAll ? null : selection.fields;
|
||||||
|
|
||||||
var successCount = 0;
|
var successCount = 0;
|
||||||
final total = selected.length;
|
final total = selected.length;
|
||||||
|
|
||||||
@@ -1098,7 +1104,7 @@ class _LocalAlbumScreenState extends ConsumerState<LocalAlbumScreen> {
|
|||||||
);
|
);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
final ok = await _reEnrichLocalTrack(item);
|
final ok = await _reEnrichLocalTrack(item, updateFields: updateFields);
|
||||||
if (ok) {
|
if (ok) {
|
||||||
successCount++;
|
successCount++;
|
||||||
}
|
}
|
||||||
@@ -1625,7 +1631,12 @@ class _LocalAlbumScreenState extends ConsumerState<LocalAlbumScreen> {
|
|||||||
try {
|
try {
|
||||||
await PlatformBridge.safDelete(item.filePath);
|
await PlatformBridge.safDelete(item.filePath);
|
||||||
} catch (_) {}
|
} catch (_) {}
|
||||||
await localDb.deleteByPath(item.filePath);
|
await localDb.replaceWithConvertedItem(
|
||||||
|
item: item,
|
||||||
|
newFilePath: safUri,
|
||||||
|
targetFormat: targetFormat,
|
||||||
|
bitrate: bitrate,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@@ -1637,8 +1648,12 @@ class _LocalAlbumScreenState extends ConsumerState<LocalAlbumScreen> {
|
|||||||
} catch (_) {}
|
} catch (_) {}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Regular file: just remove old entry, rescan will find the new one
|
await localDb.replaceWithConvertedItem(
|
||||||
await localDb.deleteByPath(item.filePath);
|
item: item,
|
||||||
|
newFilePath: newPath,
|
||||||
|
targetFormat: targetFormat,
|
||||||
|
bitrate: bitrate,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
successCount++;
|
successCount++;
|
||||||
|
|||||||
@@ -329,6 +329,8 @@ class _MainShellState extends ConsumerState<MainShell>
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!mounted) return;
|
||||||
|
|
||||||
final trackState = ref.read(trackProvider);
|
final trackState = ref.read(trackProvider);
|
||||||
|
|
||||||
final isKeyboardVisible = MediaQuery.of(context).viewInsets.bottom > 0;
|
final isKeyboardVisible = MediaQuery.of(context).viewInsets.bottom > 0;
|
||||||
@@ -346,7 +348,6 @@ class _MainShellState extends ConsumerState<MainShell>
|
|||||||
trackState.isShowingRecentAccess &&
|
trackState.isShowingRecentAccess &&
|
||||||
!trackState.isLoading &&
|
!trackState.isLoading &&
|
||||||
(trackState.hasSearchText || trackState.hasContent)) {
|
(trackState.hasSearchText || trackState.hasContent)) {
|
||||||
// Has recent access AND search content — clear everything at once
|
|
||||||
_log.i(
|
_log.i(
|
||||||
'Back: step 3a - dismiss recent access + clear search/content '
|
'Back: step 3a - dismiss recent access + clear search/content '
|
||||||
'(hasSearchText=${trackState.hasSearchText}, hasContent=${trackState.hasContent})',
|
'(hasSearchText=${trackState.hasSearchText}, hasContent=${trackState.hasContent})',
|
||||||
@@ -358,7 +359,6 @@ class _MainShellState extends ConsumerState<MainShell>
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (_currentIndex == 0 && trackState.isShowingRecentAccess) {
|
if (_currentIndex == 0 && trackState.isShowingRecentAccess) {
|
||||||
// Recent access overlay only (no search content) — just dismiss it
|
|
||||||
_log.i('Back: step 3b - dismiss recent access only');
|
_log.i('Back: step 3b - dismiss recent access only');
|
||||||
ref.read(trackProvider.notifier).setShowingRecentAccess(false);
|
ref.read(trackProvider.notifier).setShowingRecentAccess(false);
|
||||||
FocusManager.instance.primaryFocus?.unfocus();
|
FocusManager.instance.primaryFocus?.unfocus();
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ import 'package:spotiflac_android/models/track.dart';
|
|||||||
import 'package:spotiflac_android/providers/download_queue_provider.dart';
|
import 'package:spotiflac_android/providers/download_queue_provider.dart';
|
||||||
import 'package:spotiflac_android/providers/library_collections_provider.dart';
|
import 'package:spotiflac_android/providers/library_collections_provider.dart';
|
||||||
import 'package:spotiflac_android/utils/file_access.dart';
|
import 'package:spotiflac_android/utils/file_access.dart';
|
||||||
|
import 'package:spotiflac_android/utils/image_cache_utils.dart';
|
||||||
import 'package:spotiflac_android/utils/string_utils.dart';
|
import 'package:spotiflac_android/utils/string_utils.dart';
|
||||||
import 'package:spotiflac_android/providers/settings_provider.dart';
|
import 'package:spotiflac_android/providers/settings_provider.dart';
|
||||||
import 'package:spotiflac_android/providers/local_library_provider.dart';
|
import 'package:spotiflac_android/providers/local_library_provider.dart';
|
||||||
@@ -117,7 +118,6 @@ class _PlaylistScreenState extends ConsumerState<PlaylistScreen> {
|
|||||||
final playlistInfo = result['playlist_info'] as Map<String, dynamic>?;
|
final playlistInfo = result['playlist_info'] as Map<String, dynamic>?;
|
||||||
final owner = playlistInfo?['owner'] as Map<String, dynamic>?;
|
final owner = playlistInfo?['owner'] as Map<String, dynamic>?;
|
||||||
|
|
||||||
// Go backend returns 'track_list' not 'tracks'
|
|
||||||
final trackList = result['track_list'] as List<dynamic>? ?? [];
|
final trackList = result['track_list'] as List<dynamic>? ?? [];
|
||||||
final tracks = trackList
|
final tracks = trackList
|
||||||
.map((t) => _parseTrack(t as Map<String, dynamic>))
|
.map((t) => _parseTrack(t as Map<String, dynamic>))
|
||||||
@@ -164,7 +164,10 @@ class _PlaylistScreenState extends ConsumerState<PlaylistScreen> {
|
|||||||
duration: (durationMs / 1000).round(),
|
duration: (durationMs / 1000).round(),
|
||||||
trackNumber: data['track_number'] as int?,
|
trackNumber: data['track_number'] as int?,
|
||||||
discNumber: data['disc_number'] as int?,
|
discNumber: data['disc_number'] as int?,
|
||||||
|
totalDiscs: data['total_discs'] as int?,
|
||||||
releaseDate: data['release_date']?.toString(),
|
releaseDate: data['release_date']?.toString(),
|
||||||
|
totalTracks: data['total_tracks'] as int?,
|
||||||
|
composer: data['composer']?.toString(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -182,14 +185,11 @@ class _PlaylistScreenState extends ConsumerState<PlaylistScreen> {
|
|||||||
return (mediaSize.height * 0.55).clamp(360.0, 520.0);
|
return (mediaSize.height * 0.55).clamp(360.0, 520.0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Upgrade cover URL to a reasonable resolution for full-screen display.
|
|
||||||
String? _highResCoverUrl(String? url) {
|
String? _highResCoverUrl(String? url) {
|
||||||
if (url == null) return null;
|
if (url == null) return null;
|
||||||
// Spotify CDN: upgrade 300 → 640 only
|
|
||||||
if (url.contains('ab67616d00001e02')) {
|
if (url.contains('ab67616d00001e02')) {
|
||||||
return url.replaceAll('ab67616d00001e02', 'ab67616d0000b273');
|
return url.replaceAll('ab67616d00001e02', 'ab67616d0000b273');
|
||||||
}
|
}
|
||||||
// Deezer CDN: upgrade to 1000x1000
|
|
||||||
final deezerRegex = RegExp(r'/(\d+)x(\d+)-(\d+)-(\d+)-(\d+)-(\d+)\.jpg$');
|
final deezerRegex = RegExp(r'/(\d+)x(\d+)-(\d+)-(\d+)-(\d+)-(\d+)\.jpg$');
|
||||||
if (url.contains('cdn-images.dzcdn.net') && deezerRegex.hasMatch(url)) {
|
if (url.contains('cdn-images.dzcdn.net') && deezerRegex.hasMatch(url)) {
|
||||||
return url.replaceAllMapped(
|
return url.replaceAllMapped(
|
||||||
@@ -246,6 +246,7 @@ class _PlaylistScreenState extends ConsumerState<PlaylistScreen> {
|
|||||||
(constraints.maxHeight - kToolbarHeight) /
|
(constraints.maxHeight - kToolbarHeight) /
|
||||||
(expandedHeight - kToolbarHeight);
|
(expandedHeight - kToolbarHeight);
|
||||||
final showContent = collapseRatio > 0.3;
|
final showContent = collapseRatio > 0.3;
|
||||||
|
final cacheWidth = coverCacheWidthForViewport(context);
|
||||||
|
|
||||||
return FlexibleSpaceBar(
|
return FlexibleSpaceBar(
|
||||||
collapseMode: CollapseMode.pin,
|
collapseMode: CollapseMode.pin,
|
||||||
@@ -256,6 +257,7 @@ class _PlaylistScreenState extends ConsumerState<PlaylistScreen> {
|
|||||||
CachedNetworkImage(
|
CachedNetworkImage(
|
||||||
imageUrl: _highResCoverUrl(_coverUrl) ?? _coverUrl!,
|
imageUrl: _highResCoverUrl(_coverUrl) ?? _coverUrl!,
|
||||||
fit: BoxFit.cover,
|
fit: BoxFit.cover,
|
||||||
|
memCacheWidth: cacheWidth,
|
||||||
cacheManager: CoverCacheManager.instance,
|
cacheManager: CoverCacheManager.instance,
|
||||||
placeholder: (_, _) =>
|
placeholder: (_, _) =>
|
||||||
Container(color: colorScheme.surface),
|
Container(color: colorScheme.surface),
|
||||||
@@ -729,7 +731,6 @@ class _PlaylistScreenState extends ConsumerState<PlaylistScreen> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Separate Consumer widget for each track - only rebuilds when this specific track's status changes
|
|
||||||
class _PlaylistTrackItem extends ConsumerWidget {
|
class _PlaylistTrackItem extends ConsumerWidget {
|
||||||
final Track track;
|
final Track track;
|
||||||
final VoidCallback onDownload;
|
final VoidCallback onDownload;
|
||||||
|
|||||||
+154
-85
@@ -28,6 +28,7 @@ import 'package:spotiflac_android/services/history_database.dart';
|
|||||||
import 'package:spotiflac_android/services/downloaded_embedded_cover_resolver.dart';
|
import 'package:spotiflac_android/services/downloaded_embedded_cover_resolver.dart';
|
||||||
import 'package:spotiflac_android/screens/track_metadata_screen.dart';
|
import 'package:spotiflac_android/screens/track_metadata_screen.dart';
|
||||||
import 'package:spotiflac_android/screens/downloaded_album_screen.dart';
|
import 'package:spotiflac_android/screens/downloaded_album_screen.dart';
|
||||||
|
import 'package:spotiflac_android/widgets/re_enrich_field_dialog.dart';
|
||||||
import 'package:spotiflac_android/widgets/batch_progress_dialog.dart';
|
import 'package:spotiflac_android/widgets/batch_progress_dialog.dart';
|
||||||
import 'package:spotiflac_android/screens/library_tracks_folder_screen.dart';
|
import 'package:spotiflac_android/screens/library_tracks_folder_screen.dart';
|
||||||
import 'package:spotiflac_android/screens/local_album_screen.dart';
|
import 'package:spotiflac_android/screens/local_album_screen.dart';
|
||||||
@@ -155,7 +156,6 @@ class UnifiedLibraryItem {
|
|||||||
return 'builtin:$id';
|
return 'builtin:$id';
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Convert to a [Track] for adding to collections/playlists.
|
|
||||||
Track toTrack() {
|
Track toTrack() {
|
||||||
if (historyItem != null) {
|
if (historyItem != null) {
|
||||||
final h = historyItem!;
|
final h = historyItem!;
|
||||||
@@ -359,6 +359,24 @@ class _QueueGroupedAlbumFilterRequest {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class _QueueHistoryStatsMemoEntry {
|
||||||
|
final List<DownloadHistoryItem> historyItems;
|
||||||
|
final List<LocalLibraryItem> localItems;
|
||||||
|
final _HistoryStats stats;
|
||||||
|
|
||||||
|
const _QueueHistoryStatsMemoEntry({
|
||||||
|
required this.historyItems,
|
||||||
|
required this.localItems,
|
||||||
|
required this.stats,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
_QueueHistoryStatsMemoEntry? _queueHistoryStatsMemo;
|
||||||
|
|
||||||
|
String _queueHistoryAlbumKey(String albumName, String artistName) {
|
||||||
|
return '${albumName.toLowerCase()}|${artistName.toLowerCase()}';
|
||||||
|
}
|
||||||
|
|
||||||
String _queueFileExtLower(String filePath) {
|
String _queueFileExtLower(String filePath) {
|
||||||
final slashIndex = filePath.lastIndexOf('/');
|
final slashIndex = filePath.lastIndexOf('/');
|
||||||
final dotIndex = filePath.lastIndexOf('.');
|
final dotIndex = filePath.lastIndexOf('.');
|
||||||
@@ -558,21 +576,31 @@ _HistoryStats _buildQueueHistoryStats(
|
|||||||
List<DownloadHistoryItem> items, [
|
List<DownloadHistoryItem> items, [
|
||||||
List<LocalLibraryItem> localItems = const [],
|
List<LocalLibraryItem> localItems = const [],
|
||||||
]) {
|
]) {
|
||||||
|
final memo = _queueHistoryStatsMemo;
|
||||||
|
if (memo != null &&
|
||||||
|
identical(memo.historyItems, items) &&
|
||||||
|
identical(memo.localItems, localItems)) {
|
||||||
|
return memo.stats;
|
||||||
|
}
|
||||||
|
|
||||||
final albumCounts = <String, int>{};
|
final albumCounts = <String, int>{};
|
||||||
final albumMap = <String, List<DownloadHistoryItem>>{};
|
final albumMap = <String, List<DownloadHistoryItem>>{};
|
||||||
for (final item in items) {
|
for (final item in items) {
|
||||||
final key =
|
final key = _queueHistoryAlbumKey(
|
||||||
'${item.albumName.toLowerCase()}|${(item.albumArtist ?? item.artistName).toLowerCase()}';
|
item.albumName,
|
||||||
|
item.albumArtist ?? item.artistName,
|
||||||
|
);
|
||||||
albumCounts[key] = (albumCounts[key] ?? 0) + 1;
|
albumCounts[key] = (albumCounts[key] ?? 0) + 1;
|
||||||
albumMap.putIfAbsent(key, () => []).add(item);
|
albumMap.putIfAbsent(key, () => []).add(item);
|
||||||
}
|
}
|
||||||
|
|
||||||
var singleTracks = 0;
|
var singleTracks = 0;
|
||||||
for (final item in items) {
|
var albumCount = 0;
|
||||||
final key =
|
for (final count in albumCounts.values) {
|
||||||
'${item.albumName.toLowerCase()}|${(item.albumArtist ?? item.artistName).toLowerCase()}';
|
if (count > 1) {
|
||||||
if ((albumCounts[key] ?? 0) <= 1) {
|
albumCount++;
|
||||||
singleTracks++;
|
} else {
|
||||||
|
singleTracks += count;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -600,11 +628,6 @@ _HistoryStats _buildQueueHistoryStats(
|
|||||||
});
|
});
|
||||||
groupedAlbums.sort((a, b) => b.latestDownload.compareTo(a.latestDownload));
|
groupedAlbums.sort((a, b) => b.latestDownload.compareTo(a.latestDownload));
|
||||||
|
|
||||||
var albumCount = 0;
|
|
||||||
for (final count in albumCounts.values) {
|
|
||||||
if (count > 1) albumCount++;
|
|
||||||
}
|
|
||||||
|
|
||||||
final downloadedPathKeys = <String>{};
|
final downloadedPathKeys = <String>{};
|
||||||
for (final item in items) {
|
for (final item in items) {
|
||||||
downloadedPathKeys.addAll(buildPathMatchKeys(item.filePath));
|
downloadedPathKeys.addAll(buildPathMatchKeys(item.filePath));
|
||||||
@@ -620,8 +643,10 @@ _HistoryStats _buildQueueHistoryStats(
|
|||||||
final localAlbumCounts = <String, int>{};
|
final localAlbumCounts = <String, int>{};
|
||||||
final localAlbumMap = <String, List<LocalLibraryItem>>{};
|
final localAlbumMap = <String, List<LocalLibraryItem>>{};
|
||||||
for (final item in dedupedLocalItems) {
|
for (final item in dedupedLocalItems) {
|
||||||
final key =
|
final key = _queueHistoryAlbumKey(
|
||||||
'${item.albumName.toLowerCase()}|${(item.albumArtist ?? item.artistName).toLowerCase()}';
|
item.albumName,
|
||||||
|
item.albumArtist ?? item.artistName,
|
||||||
|
);
|
||||||
localAlbumCounts[key] = (localAlbumCounts[key] ?? 0) + 1;
|
localAlbumCounts[key] = (localAlbumCounts[key] ?? 0) + 1;
|
||||||
localAlbumMap.putIfAbsent(key, () => []).add(item);
|
localAlbumMap.putIfAbsent(key, () => []).add(item);
|
||||||
}
|
}
|
||||||
@@ -664,7 +689,7 @@ _HistoryStats _buildQueueHistoryStats(
|
|||||||
});
|
});
|
||||||
groupedLocalAlbums.sort((a, b) => b.latestScanned.compareTo(a.latestScanned));
|
groupedLocalAlbums.sort((a, b) => b.latestScanned.compareTo(a.latestScanned));
|
||||||
|
|
||||||
return _HistoryStats(
|
final stats = _HistoryStats(
|
||||||
albumCounts: albumCounts,
|
albumCounts: albumCounts,
|
||||||
localAlbumCounts: localAlbumCounts,
|
localAlbumCounts: localAlbumCounts,
|
||||||
groupedAlbums: groupedAlbums,
|
groupedAlbums: groupedAlbums,
|
||||||
@@ -674,6 +699,12 @@ _HistoryStats _buildQueueHistoryStats(
|
|||||||
localAlbumCount: localAlbumCount,
|
localAlbumCount: localAlbumCount,
|
||||||
localSingleTracks: localSingleTracks,
|
localSingleTracks: localSingleTracks,
|
||||||
);
|
);
|
||||||
|
_queueHistoryStatsMemo = _QueueHistoryStatsMemoEntry(
|
||||||
|
historyItems: items,
|
||||||
|
localItems: localItems,
|
||||||
|
stats: stats,
|
||||||
|
);
|
||||||
|
return stats;
|
||||||
}
|
}
|
||||||
|
|
||||||
List<_GroupedAlbum> _queueFilterGroupedAlbums(
|
List<_GroupedAlbum> _queueFilterGroupedAlbums(
|
||||||
@@ -1121,6 +1152,10 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
List<UnifiedLibraryItem> _cachedUnifiedLocal = const [];
|
List<UnifiedLibraryItem> _cachedUnifiedLocal = const [];
|
||||||
List<DownloadHistoryItem>? _cachedDownloadedPathKeysSource;
|
List<DownloadHistoryItem>? _cachedDownloadedPathKeysSource;
|
||||||
Set<String> _cachedDownloadedPathKeys = const <String>{};
|
Set<String> _cachedDownloadedPathKeys = const <String>{};
|
||||||
|
final Map<String, List<String>> _localPathMatchKeysCache = {};
|
||||||
|
List<LocalLibraryItem>? _cachedLocalSinglesSource;
|
||||||
|
Map<String, int>? _cachedLocalSinglesAlbumCountsSource;
|
||||||
|
List<LocalLibraryItem> _cachedLocalSingles = const [];
|
||||||
final Map<String, _FilterContentData> _filterContentDataCache = {};
|
final Map<String, _FilterContentData> _filterContentDataCache = {};
|
||||||
List<DownloadHistoryItem>? _filterCacheAllHistoryItems;
|
List<DownloadHistoryItem>? _filterCacheAllHistoryItems;
|
||||||
_HistoryStats? _filterCacheHistoryStats;
|
_HistoryStats? _filterCacheHistoryStats;
|
||||||
@@ -1132,11 +1167,11 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
String? _filterCacheFormat;
|
String? _filterCacheFormat;
|
||||||
String? _filterCacheMetadata;
|
String? _filterCacheMetadata;
|
||||||
String _filterCacheSortMode = 'latest';
|
String _filterCacheSortMode = 'latest';
|
||||||
String? _filterSource; // null = all, 'downloaded', 'local'
|
String? _filterSource;
|
||||||
String? _filterQuality; // null = all, 'hires', 'cd', 'lossy'
|
String? _filterQuality;
|
||||||
String? _filterFormat; // null = all, 'flac', 'mp3', 'm4a', 'opus', 'ogg'
|
String? _filterFormat;
|
||||||
String? _filterMetadata; // null = all, 'complete', 'missing-*'
|
String? _filterMetadata;
|
||||||
String _sortMode = 'latest'; // 'latest', 'oldest', 'a-z', 'z-a'
|
String _sortMode = 'latest';
|
||||||
|
|
||||||
double _effectiveTextScale() {
|
double _effectiveTextScale() {
|
||||||
final textScale = MediaQuery.textScalerOf(context).scale(1.0);
|
final textScale = MediaQuery.textScalerOf(context).scale(1.0);
|
||||||
@@ -1264,9 +1299,13 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
}
|
}
|
||||||
if (localChanged) {
|
if (localChanged) {
|
||||||
_localSearchIndexCache.clear();
|
_localSearchIndexCache.clear();
|
||||||
|
_localPathMatchKeysCache.clear();
|
||||||
_localFilterItemsCache = null;
|
_localFilterItemsCache = null;
|
||||||
_localFilterQueryCache = '';
|
_localFilterQueryCache = '';
|
||||||
_filteredLocalItemsCache = const [];
|
_filteredLocalItemsCache = const [];
|
||||||
|
_cachedLocalSinglesSource = null;
|
||||||
|
_cachedLocalSinglesAlbumCountsSource = null;
|
||||||
|
_cachedLocalSingles = const [];
|
||||||
_cachedUnifiedLocalSource = null;
|
_cachedUnifiedLocalSource = null;
|
||||||
_cachedUnifiedLocal = const [];
|
_cachedUnifiedLocal = const [];
|
||||||
}
|
}
|
||||||
@@ -1356,6 +1395,32 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
return _cachedDownloadedPathKeys;
|
return _cachedDownloadedPathKeys;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
List<String> _localPathMatchKeys(LocalLibraryItem item) {
|
||||||
|
final cached = _localPathMatchKeysCache[item.id];
|
||||||
|
if (cached != null) return cached;
|
||||||
|
final keys = buildPathMatchKeys(item.filePath).toList(growable: false);
|
||||||
|
_localPathMatchKeysCache[item.id] = keys;
|
||||||
|
return keys;
|
||||||
|
}
|
||||||
|
|
||||||
|
List<LocalLibraryItem> _localSingleItems(
|
||||||
|
List<LocalLibraryItem> items,
|
||||||
|
Map<String, int> localAlbumCounts,
|
||||||
|
) {
|
||||||
|
if (identical(items, _cachedLocalSinglesSource) &&
|
||||||
|
identical(localAlbumCounts, _cachedLocalSinglesAlbumCountsSource)) {
|
||||||
|
return _cachedLocalSingles;
|
||||||
|
}
|
||||||
|
|
||||||
|
final singles = items
|
||||||
|
.where((item) => (localAlbumCounts[item.albumKey] ?? 0) == 1)
|
||||||
|
.toList(growable: false);
|
||||||
|
_cachedLocalSinglesSource = items;
|
||||||
|
_cachedLocalSinglesAlbumCountsSource = localAlbumCounts;
|
||||||
|
_cachedLocalSingles = singles;
|
||||||
|
return singles;
|
||||||
|
}
|
||||||
|
|
||||||
List<LocalLibraryItem> _filterLocalItems(
|
List<LocalLibraryItem> _filterLocalItems(
|
||||||
List<LocalLibraryItem> items,
|
List<LocalLibraryItem> items,
|
||||||
String query,
|
String query,
|
||||||
@@ -2036,7 +2101,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
return quality.split('/').first;
|
return quality.split('/').first;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Supports "MP3 320k", "Opus 256kbps", etc.
|
|
||||||
final bitrateTextMatch = RegExp(
|
final bitrateTextMatch = RegExp(
|
||||||
r'(\d+)\s*k(?:bps)?',
|
r'(\d+)\s*k(?:bps)?',
|
||||||
caseSensitive: false,
|
caseSensitive: false,
|
||||||
@@ -2045,7 +2109,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
return '${bitrateTextMatch.group(1)}k';
|
return '${bitrateTextMatch.group(1)}k';
|
||||||
}
|
}
|
||||||
|
|
||||||
// Supports legacy quality IDs like "opus_256" / "mp3_320".
|
|
||||||
final bitrateIdMatch = RegExp(r'_(\d+)$').firstMatch(q);
|
final bitrateIdMatch = RegExp(r'_(\d+)$').firstMatch(q);
|
||||||
if (bitrateIdMatch != null) {
|
if (bitrateIdMatch != null) {
|
||||||
return '${bitrateIdMatch.group(1)}k';
|
return '${bitrateIdMatch.group(1)}k';
|
||||||
@@ -2100,7 +2163,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Reload local library if we deleted any local items
|
|
||||||
if (allItems.any(
|
if (allItems.any(
|
||||||
(i) =>
|
(i) =>
|
||||||
_selectedIds.contains(i.id) && i.source == LibraryItemSource.local,
|
_selectedIds.contains(i.id) && i.source == LibraryItemSource.local,
|
||||||
@@ -2120,7 +2182,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Strip EXISTS: prefix from file path (legacy history items)
|
|
||||||
String _cleanFilePath(String? filePath) {
|
String _cleanFilePath(String? filePath) {
|
||||||
return DownloadedEmbeddedCoverResolver.cleanFilePath(filePath);
|
return DownloadedEmbeddedCoverResolver.cleanFilePath(filePath);
|
||||||
}
|
}
|
||||||
@@ -2303,7 +2364,7 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
|
|
||||||
List<UnifiedLibraryItem> _applySorting(List<UnifiedLibraryItem> items) {
|
List<UnifiedLibraryItem> _applySorting(List<UnifiedLibraryItem> items) {
|
||||||
if (_sortMode == 'latest') {
|
if (_sortMode == 'latest') {
|
||||||
return items; // Already sorted newest first from _getUnifiedItems
|
return items;
|
||||||
}
|
}
|
||||||
final sorted = List<UnifiedLibraryItem>.of(items);
|
final sorted = List<UnifiedLibraryItem>.of(items);
|
||||||
switch (_sortMode) {
|
switch (_sortMode) {
|
||||||
@@ -2970,7 +3031,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Navigate with unfocus pattern — unfocuses search before and after navigation.
|
|
||||||
void _navigateWithUnfocus(Route<dynamic> route) {
|
void _navigateWithUnfocus(Route<dynamic> route) {
|
||||||
_searchFocusNode.unfocus();
|
_searchFocusNode.unfocus();
|
||||||
Navigator.of(context).push(route).then((_) => _searchFocusNode.unfocus());
|
Navigator.of(context).push(route).then((_) => _searchFocusNode.unfocus());
|
||||||
@@ -3200,14 +3260,12 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
}) async {
|
}) async {
|
||||||
final notifier = ref.read(libraryCollectionsProvider.notifier);
|
final notifier = ref.read(libraryCollectionsProvider.notifier);
|
||||||
|
|
||||||
// If in selection mode and the dragged item is selected, add ALL selected
|
|
||||||
if (_isSelectionMode &&
|
if (_isSelectionMode &&
|
||||||
_selectedIds.isNotEmpty &&
|
_selectedIds.isNotEmpty &&
|
||||||
_selectedIds.contains(item.id)) {
|
_selectedIds.contains(item.id)) {
|
||||||
final selectedItems = allItems
|
final selectedItems = allItems
|
||||||
.where((e) => _selectedIds.contains(e.id))
|
.where((e) => _selectedIds.contains(e.id))
|
||||||
.toList();
|
.toList();
|
||||||
// Fallback: if allItems is empty or no match, at least add the dragged item
|
|
||||||
if (selectedItems.isEmpty) {
|
if (selectedItems.isEmpty) {
|
||||||
selectedItems.add(item);
|
selectedItems.add(item);
|
||||||
}
|
}
|
||||||
@@ -3246,8 +3304,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Build a compact floating feedback widget shown while dragging a track.
|
|
||||||
/// Shows the count when multiple tracks are selected and being dragged.
|
|
||||||
Widget _buildDragFeedback(
|
Widget _buildDragFeedback(
|
||||||
BuildContext context,
|
BuildContext context,
|
||||||
UnifiedLibraryItem item,
|
UnifiedLibraryItem item,
|
||||||
@@ -3371,7 +3427,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
final selectionItems = getFilterData(
|
final selectionItems = getFilterData(
|
||||||
historyFilterMode,
|
historyFilterMode,
|
||||||
).filteredUnifiedItems;
|
).filteredUnifiedItems;
|
||||||
// Only sync overlays when selection mode is active
|
|
||||||
if (_isSelectionMode || _isPlaylistSelectionMode) {
|
if (_isSelectionMode || _isPlaylistSelectionMode) {
|
||||||
WidgetsBinding.instance.addPostFrameCallback((_) {
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
||||||
if (_isSelectionMode) {
|
if (_isSelectionMode) {
|
||||||
@@ -3627,12 +3682,10 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
if (filterMode == 'all') {
|
if (filterMode == 'all') {
|
||||||
localItemsForMerge = _filterLocalItems(localLibraryItems, query);
|
localItemsForMerge = _filterLocalItems(localLibraryItems, query);
|
||||||
} else {
|
} else {
|
||||||
final localSingles = localLibraryItems
|
final localSingles = _localSingleItems(
|
||||||
.where((item) {
|
localLibraryItems,
|
||||||
final count = localAlbumCounts[item.albumKey] ?? 0;
|
localAlbumCounts,
|
||||||
return count == 1;
|
);
|
||||||
})
|
|
||||||
.toList(growable: false);
|
|
||||||
localItemsForMerge = _filterLocalItems(localSingles, query);
|
localItemsForMerge = _filterLocalItems(localSingles, query);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -3641,7 +3694,10 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
|
|
||||||
final dedupedUnifiedLocal = <UnifiedLibraryItem>[];
|
final dedupedUnifiedLocal = <UnifiedLibraryItem>[];
|
||||||
for (final item in unifiedLocal) {
|
for (final item in unifiedLocal) {
|
||||||
final localPathKeys = buildPathMatchKeys(item.filePath);
|
final localSource = item.localItem;
|
||||||
|
final localPathKeys = localSource != null
|
||||||
|
? _localPathMatchKeys(localSource)
|
||||||
|
: buildPathMatchKeys(item.filePath);
|
||||||
final overlapsDownloaded = localPathKeys.any(downloadedPathKeys.contains);
|
final overlapsDownloaded = localPathKeys.any(downloadedPathKeys.contains);
|
||||||
if (!overlapsDownloaded) {
|
if (!overlapsDownloaded) {
|
||||||
dedupedUnifiedLocal.add(item);
|
dedupedUnifiedLocal.add(item);
|
||||||
@@ -3776,7 +3832,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Build a Spotify-style collection list item (Wishlist, Loved, Playlists)
|
|
||||||
Widget _buildCollectionListItem({
|
Widget _buildCollectionListItem({
|
||||||
required BuildContext context,
|
required BuildContext context,
|
||||||
required ColorScheme colorScheme,
|
required ColorScheme colorScheme,
|
||||||
@@ -3853,7 +3908,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Build a collection grid item for grid view mode
|
|
||||||
Widget _buildCollectionGridItem({
|
Widget _buildCollectionGridItem({
|
||||||
required BuildContext context,
|
required BuildContext context,
|
||||||
required ColorScheme colorScheme,
|
required ColorScheme colorScheme,
|
||||||
@@ -3936,7 +3990,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
return entries;
|
return entries;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Build a collection item for the unified "All" tab grid view.
|
|
||||||
Widget _buildAllTabGridCollectionItem({
|
Widget _buildAllTabGridCollectionItem({
|
||||||
required BuildContext context,
|
required BuildContext context,
|
||||||
required ColorScheme colorScheme,
|
required ColorScheme colorScheme,
|
||||||
@@ -4054,7 +4107,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Build a collection item for the unified "All" tab list view.
|
|
||||||
Widget _buildAllTabListCollectionItem({
|
Widget _buildAllTabListCollectionItem({
|
||||||
required BuildContext context,
|
required BuildContext context,
|
||||||
required ColorScheme colorScheme,
|
required ColorScheme colorScheme,
|
||||||
@@ -4207,8 +4259,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
// Collection folders as list items (Spotify-style) in "All" tab
|
|
||||||
// are now rendered inline with tracks below (unified sliver)
|
|
||||||
if ((filteredGroupedAlbums.isNotEmpty ||
|
if ((filteredGroupedAlbums.isNotEmpty ||
|
||||||
filteredGroupedLocalAlbums.isNotEmpty) &&
|
filteredGroupedLocalAlbums.isNotEmpty) &&
|
||||||
filterMode == 'albums')
|
filterMode == 'albums')
|
||||||
@@ -4695,7 +4745,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Album grid item for local library albums
|
|
||||||
Widget _buildLocalAlbumGridItem(
|
Widget _buildLocalAlbumGridItem(
|
||||||
BuildContext context,
|
BuildContext context,
|
||||||
_GroupedLocalAlbum album,
|
_GroupedLocalAlbum album,
|
||||||
@@ -4913,12 +4962,14 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
mp3Path: ffmpegTarget,
|
mp3Path: ffmpegTarget,
|
||||||
coverPath: effectiveCoverPath,
|
coverPath: effectiveCoverPath,
|
||||||
metadata: metadata,
|
metadata: metadata,
|
||||||
|
preserveMetadata: true,
|
||||||
);
|
);
|
||||||
} else if (isM4A) {
|
} else if (isM4A) {
|
||||||
ffmpegResult = await FFmpegService.embedMetadataToM4a(
|
ffmpegResult = await FFmpegService.embedMetadataToM4a(
|
||||||
m4aPath: ffmpegTarget,
|
m4aPath: ffmpegTarget,
|
||||||
coverPath: effectiveCoverPath,
|
coverPath: effectiveCoverPath,
|
||||||
metadata: metadata,
|
metadata: metadata,
|
||||||
|
preserveMetadata: true,
|
||||||
);
|
);
|
||||||
} else if (isOpus) {
|
} else if (isOpus) {
|
||||||
ffmpegResult = await FFmpegService.embedMetadataToOpus(
|
ffmpegResult = await FFmpegService.embedMetadataToOpus(
|
||||||
@@ -4926,6 +4977,7 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
coverPath: effectiveCoverPath,
|
coverPath: effectiveCoverPath,
|
||||||
metadata: metadata,
|
metadata: metadata,
|
||||||
artistTagMode: artistTagMode,
|
artistTagMode: artistTagMode,
|
||||||
|
preserveMetadata: true,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -4958,7 +5010,10 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
return ffmpegResult != null;
|
return ffmpegResult != null;
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<bool> _reEnrichQueueLocalTrack(LocalLibraryItem item) async {
|
Future<bool> _reEnrichQueueLocalTrack(
|
||||||
|
LocalLibraryItem item, {
|
||||||
|
List<String>? updateFields,
|
||||||
|
}) async {
|
||||||
final durationMs = (item.duration ?? 0) * 1000;
|
final durationMs = (item.duration ?? 0) * 1000;
|
||||||
final artistTagMode = ref.read(settingsProvider).artistTagMode;
|
final artistTagMode = ref.read(settingsProvider).artistTagMode;
|
||||||
final request = <String, dynamic>{
|
final request = <String, dynamic>{
|
||||||
@@ -4981,6 +5036,8 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
'copyright': '',
|
'copyright': '',
|
||||||
'duration_ms': durationMs,
|
'duration_ms': durationMs,
|
||||||
'search_online': true,
|
'search_online': true,
|
||||||
|
// ignore: use_null_aware_elements
|
||||||
|
if (updateFields != null) 'update_fields': updateFields,
|
||||||
};
|
};
|
||||||
|
|
||||||
final result = await PlatformBridge.reEnrichFile(request);
|
final result = await PlatformBridge.reEnrichFile(request);
|
||||||
@@ -5144,31 +5201,25 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
final confirmed = await showDialog<bool>(
|
// Hide the selection overlay: set the flag (prevents build() from
|
||||||
context: context,
|
// re-inserting via postFrameCallback) and remove the entry immediately.
|
||||||
builder: (ctx) => AlertDialog(
|
setState(() => _isSelectionMode = false);
|
||||||
title: Text(context.l10n.trackReEnrich),
|
_hideSelectionOverlay();
|
||||||
content: Text(
|
|
||||||
'${context.l10n.trackReEnrichOnlineSubtitle}\n\n'
|
final selection = await showReEnrichFieldDialog(
|
||||||
'${context.l10n.downloadedAlbumSelectedCount(selectedLocalItems.length)}',
|
context,
|
||||||
),
|
selectedCount: selectedLocalItems.length,
|
||||||
actions: [
|
|
||||||
TextButton(
|
|
||||||
onPressed: () => Navigator.pop(ctx, false),
|
|
||||||
child: Text(context.l10n.dialogCancel),
|
|
||||||
),
|
|
||||||
FilledButton(
|
|
||||||
onPressed: () => Navigator.pop(ctx, true),
|
|
||||||
child: Text(context.l10n.trackReEnrich),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
);
|
||||||
|
|
||||||
if (confirmed != true || !mounted) {
|
if (selection == null || !mounted) {
|
||||||
|
// Cancelled — restore selection mode; the next build cycle will
|
||||||
|
// re-create the overlay via _syncSelectionOverlay in postFrameCallback.
|
||||||
|
if (mounted) setState(() => _isSelectionMode = true);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
final updateFields = selection.isAll ? null : selection.fields;
|
||||||
|
|
||||||
var successCount = 0;
|
var successCount = 0;
|
||||||
final total = selectedLocalItems.length;
|
final total = selectedLocalItems.length;
|
||||||
|
|
||||||
@@ -5194,7 +5245,10 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
);
|
);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
final ok = await _reEnrichQueueLocalTrack(item);
|
final ok = await _reEnrichQueueLocalTrack(
|
||||||
|
item,
|
||||||
|
updateFields: updateFields,
|
||||||
|
);
|
||||||
if (ok) {
|
if (ok) {
|
||||||
successCount++;
|
successCount++;
|
||||||
}
|
}
|
||||||
@@ -5269,7 +5323,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Share SAF content URIs via native intent
|
|
||||||
if (safUris.isNotEmpty) {
|
if (safUris.isNotEmpty) {
|
||||||
try {
|
try {
|
||||||
if (safUris.length == 1) {
|
if (safUris.length == 1) {
|
||||||
@@ -5280,7 +5333,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
} catch (_) {}
|
} catch (_) {}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Share regular files via SharePlus
|
|
||||||
if (filesToShare.isNotEmpty) {
|
if (filesToShare.isNotEmpty) {
|
||||||
await SharePlus.instance.share(ShareParams(files: filesToShare));
|
await SharePlus.instance.share(ShareParams(files: filesToShare));
|
||||||
}
|
}
|
||||||
@@ -5801,13 +5853,27 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
final baseName = dotIdx > 0
|
final baseName = dotIdx > 0
|
||||||
? oldFileName.substring(0, dotIdx)
|
? oldFileName.substring(0, dotIdx)
|
||||||
: oldFileName;
|
: oldFileName;
|
||||||
final newExt = targetFormat.toLowerCase() == 'opus'
|
String newExt;
|
||||||
? '.opus'
|
String mimeType;
|
||||||
: '.mp3';
|
switch (targetFormat.toLowerCase()) {
|
||||||
|
case 'opus':
|
||||||
|
newExt = '.opus';
|
||||||
|
mimeType = 'audio/opus';
|
||||||
|
break;
|
||||||
|
case 'alac':
|
||||||
|
newExt = '.m4a';
|
||||||
|
mimeType = 'audio/mp4';
|
||||||
|
break;
|
||||||
|
case 'flac':
|
||||||
|
newExt = '.flac';
|
||||||
|
mimeType = 'audio/flac';
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
newExt = '.mp3';
|
||||||
|
mimeType = 'audio/mpeg';
|
||||||
|
break;
|
||||||
|
}
|
||||||
final newFileName = '$baseName$newExt';
|
final newFileName = '$baseName$newExt';
|
||||||
final mimeType = targetFormat.toLowerCase() == 'opus'
|
|
||||||
? 'audio/opus'
|
|
||||||
: 'audio/mpeg';
|
|
||||||
|
|
||||||
final safUri = await PlatformBridge.createSafFileFromPath(
|
final safUri = await PlatformBridge.createSafFileFromPath(
|
||||||
treeUri: treeUri,
|
treeUri: treeUri,
|
||||||
@@ -5832,7 +5898,12 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
try {
|
try {
|
||||||
await PlatformBridge.safDelete(item.filePath);
|
await PlatformBridge.safDelete(item.filePath);
|
||||||
} catch (_) {}
|
} catch (_) {}
|
||||||
await LibraryDatabase.instance.deleteByPath(item.filePath);
|
await LibraryDatabase.instance.replaceWithConvertedItem(
|
||||||
|
item: item.localItem!,
|
||||||
|
newFilePath: safUri,
|
||||||
|
targetFormat: targetFormat,
|
||||||
|
bitrate: bitrate,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@@ -5851,7 +5922,12 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
clearAudioSpecs: true,
|
clearAudioSpecs: true,
|
||||||
);
|
);
|
||||||
} else if (item.localItem != null) {
|
} else if (item.localItem != null) {
|
||||||
await LibraryDatabase.instance.deleteByPath(item.filePath);
|
await LibraryDatabase.instance.replaceWithConvertedItem(
|
||||||
|
item: item.localItem!,
|
||||||
|
newFilePath: newPath,
|
||||||
|
targetFormat: targetFormat,
|
||||||
|
bitrate: bitrate,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
successCount++;
|
successCount++;
|
||||||
@@ -6394,7 +6470,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Reusable filter button with badge showing active filter count.
|
|
||||||
Widget _buildFilterButton(
|
Widget _buildFilterButton(
|
||||||
BuildContext context,
|
BuildContext context,
|
||||||
List<UnifiedLibraryItem> unifiedItems,
|
List<UnifiedLibraryItem> unifiedItems,
|
||||||
@@ -6489,7 +6564,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Network URL cover (downloaded items)
|
|
||||||
if (item.coverUrl != null) {
|
if (item.coverUrl != null) {
|
||||||
return ClipRRect(
|
return ClipRRect(
|
||||||
borderRadius: BorderRadius.circular(8),
|
borderRadius: BorderRadius.circular(8),
|
||||||
@@ -6507,7 +6581,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Local file cover (from library scan)
|
|
||||||
if (item.localCoverPath != null && item.localCoverPath!.isNotEmpty) {
|
if (item.localCoverPath != null && item.localCoverPath!.isNotEmpty) {
|
||||||
return ClipRRect(
|
return ClipRRect(
|
||||||
borderRadius: BorderRadius.circular(8),
|
borderRadius: BorderRadius.circular(8),
|
||||||
@@ -6524,7 +6597,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Placeholder (no cover)
|
|
||||||
if (size != null) {
|
if (size != null) {
|
||||||
return buildPlaceholder();
|
return buildPlaceholder();
|
||||||
}
|
}
|
||||||
@@ -6534,7 +6606,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Build a unified library item (merged downloaded + local)
|
|
||||||
Widget _buildUnifiedLibraryItem(
|
Widget _buildUnifiedLibraryItem(
|
||||||
BuildContext context,
|
BuildContext context,
|
||||||
UnifiedLibraryItem item,
|
UnifiedLibraryItem item,
|
||||||
@@ -6742,7 +6813,6 @@ class _QueueTabState extends ConsumerState<QueueTab> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Build unified grid item for grid view mode
|
|
||||||
Widget _buildUnifiedGridItem(
|
Widget _buildUnifiedGridItem(
|
||||||
BuildContext context,
|
BuildContext context,
|
||||||
UnifiedLibraryItem item,
|
UnifiedLibraryItem item,
|
||||||
@@ -7032,7 +7102,6 @@ class _FilterChip extends StatelessWidget {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Reusable action button for selection mode bottom bar
|
|
||||||
class _SelectionActionButton extends StatelessWidget {
|
class _SelectionActionButton extends StatelessWidget {
|
||||||
final IconData icon;
|
final IconData icon;
|
||||||
final String label;
|
final String label;
|
||||||
|
|||||||
@@ -164,13 +164,7 @@ class _RecentDonorsCard extends StatelessWidget {
|
|||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final isDark = Theme.of(context).brightness == Brightness.dark;
|
final isDark = Theme.of(context).brightness == Brightness.dark;
|
||||||
const donorNames = <String>[
|
const donorNames = <String>['R4ND0MIZ3D', 'Isra', 'bigJr48'];
|
||||||
'McNuggets Jimmy',
|
|
||||||
'zcc09',
|
|
||||||
'micahRichie',
|
|
||||||
'a fan',
|
|
||||||
'CJBGR',
|
|
||||||
];
|
|
||||||
|
|
||||||
// Match SettingsGroup color logic
|
// Match SettingsGroup color logic
|
||||||
final cardColor = isDark
|
final cardColor = isDark
|
||||||
|
|||||||
@@ -592,6 +592,22 @@ class _DownloadSettingsPageState extends ConsumerState<DownloadSettingsPage> {
|
|||||||
settings.filenameFormat,
|
settings.filenameFormat,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
SettingsItem(
|
||||||
|
icon: Icons.music_note_outlined,
|
||||||
|
title: context.l10n.downloadSingleFilenameFormat,
|
||||||
|
subtitle: settings.singleFilenameFormat,
|
||||||
|
onTap: () => _showFormatEditor(
|
||||||
|
context,
|
||||||
|
ref,
|
||||||
|
settings.singleFilenameFormat,
|
||||||
|
onSave: ref
|
||||||
|
.read(settingsProvider.notifier)
|
||||||
|
.setSingleFilenameFormat,
|
||||||
|
title: context.l10n.downloadSingleFilenameFormat,
|
||||||
|
description:
|
||||||
|
context.l10n.downloadSingleFilenameFormatDescription,
|
||||||
|
),
|
||||||
|
),
|
||||||
SettingsItem(
|
SettingsItem(
|
||||||
icon: Icons.folder_outlined,
|
icon: Icons.folder_outlined,
|
||||||
title: context.l10n.downloadDirectory,
|
title: context.l10n.downloadDirectory,
|
||||||
@@ -952,7 +968,14 @@ class _DownloadSettingsPageState extends ConsumerState<DownloadSettingsPage> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
void _showFormatEditor(BuildContext context, WidgetRef ref, String current) {
|
void _showFormatEditor(
|
||||||
|
BuildContext context,
|
||||||
|
WidgetRef ref,
|
||||||
|
String current, {
|
||||||
|
void Function(String)? onSave,
|
||||||
|
String? title,
|
||||||
|
String? description,
|
||||||
|
}) {
|
||||||
final controller = TextEditingController(text: current);
|
final controller = TextEditingController(text: current);
|
||||||
final colorScheme = Theme.of(context).colorScheme;
|
final colorScheme = Theme.of(context).colorScheme;
|
||||||
|
|
||||||
@@ -1035,14 +1058,14 @@ class _DownloadSettingsPageState extends ConsumerState<DownloadSettingsPage> {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
Text(
|
Text(
|
||||||
context.l10n.filenameFormat,
|
title ?? context.l10n.filenameFormat,
|
||||||
style: Theme.of(context).textTheme.headlineSmall
|
style: Theme.of(context).textTheme.headlineSmall
|
||||||
?.copyWith(fontWeight: FontWeight.bold),
|
?.copyWith(fontWeight: FontWeight.bold),
|
||||||
textAlign: TextAlign.center,
|
textAlign: TextAlign.center,
|
||||||
),
|
),
|
||||||
const SizedBox(height: 8),
|
const SizedBox(height: 8),
|
||||||
Text(
|
Text(
|
||||||
context.l10n.downloadFilenameDescription,
|
description ?? context.l10n.downloadFilenameDescription,
|
||||||
style: Theme.of(context).textTheme.bodyMedium?.copyWith(
|
style: Theme.of(context).textTheme.bodyMedium?.copyWith(
|
||||||
color: colorScheme.onSurfaceVariant,
|
color: colorScheme.onSurfaceVariant,
|
||||||
),
|
),
|
||||||
@@ -1149,9 +1172,12 @@ class _DownloadSettingsPageState extends ConsumerState<DownloadSettingsPage> {
|
|||||||
flex: 2,
|
flex: 2,
|
||||||
child: FilledButton(
|
child: FilledButton(
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
ref
|
final save =
|
||||||
.read(settingsProvider.notifier)
|
onSave ??
|
||||||
.setFilenameFormat(controller.text);
|
ref
|
||||||
|
.read(settingsProvider.notifier)
|
||||||
|
.setFilenameFormat;
|
||||||
|
save(controller.text);
|
||||||
Navigator.pop(context);
|
Navigator.pop(context);
|
||||||
},
|
},
|
||||||
style: FilledButton.styleFrom(
|
style: FilledButton.styleFrom(
|
||||||
@@ -1563,7 +1589,6 @@ class _DownloadSettingsPageState extends ConsumerState<DownloadSettingsPage> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static const _providerDisplayNames = <String, String>{
|
static const _providerDisplayNames = <String, String>{
|
||||||
'spotify_api': 'Spotify Lyrics API',
|
|
||||||
'lrclib': 'LRCLIB',
|
'lrclib': 'LRCLIB',
|
||||||
'netease': 'Netease',
|
'netease': 'Netease',
|
||||||
'musixmatch': 'Musixmatch',
|
'musixmatch': 'Musixmatch',
|
||||||
|
|||||||
@@ -53,10 +53,8 @@ class _LibrarySettingsPageState extends ConsumerState<LibrarySettingsPage> {
|
|||||||
if (mounted) {
|
if (mounted) {
|
||||||
setState(() {
|
setState(() {
|
||||||
_androidSdkVersion = sdkVersion;
|
_androidSdkVersion = sdkVersion;
|
||||||
// SAF doesn't need storage permission on Android 10+
|
|
||||||
_hasStoragePermission = sdkVersion >= 29 ? true : false;
|
_hasStoragePermission = sdkVersion >= 29 ? true : false;
|
||||||
});
|
});
|
||||||
// For older Android, check legacy storage permission
|
|
||||||
if (sdkVersion < 29) {
|
if (sdkVersion < 29) {
|
||||||
final hasPermission = await Permission.storage.isGranted;
|
final hasPermission = await Permission.storage.isGranted;
|
||||||
if (mounted) {
|
if (mounted) {
|
||||||
@@ -65,7 +63,6 @@ class _LibrarySettingsPageState extends ConsumerState<LibrarySettingsPage> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (Platform.isIOS) {
|
} else if (Platform.isIOS) {
|
||||||
// iOS doesn't need explicit storage permission for app documents
|
|
||||||
setState(() => _hasStoragePermission = true);
|
setState(() => _hasStoragePermission = true);
|
||||||
} else {
|
} else {
|
||||||
setState(() => _hasStoragePermission = true);
|
setState(() => _hasStoragePermission = true);
|
||||||
@@ -74,7 +71,6 @@ class _LibrarySettingsPageState extends ConsumerState<LibrarySettingsPage> {
|
|||||||
|
|
||||||
Future<bool> _requestStoragePermission() async {
|
Future<bool> _requestStoragePermission() async {
|
||||||
if (!Platform.isAndroid) return true;
|
if (!Platform.isAndroid) return true;
|
||||||
// SAF on Android 10+ doesn't need MANAGE_EXTERNAL_STORAGE
|
|
||||||
if (_androidSdkVersion >= 29) return true;
|
if (_androidSdkVersion >= 29) return true;
|
||||||
|
|
||||||
final status = await Permission.storage.request();
|
final status = await Permission.storage.request();
|
||||||
@@ -125,12 +121,9 @@ class _LibrarySettingsPageState extends ConsumerState<LibrarySettingsPage> {
|
|||||||
final granted = await _requestStoragePermission();
|
final granted = await _requestStoragePermission();
|
||||||
if (!granted) return;
|
if (!granted) return;
|
||||||
}
|
}
|
||||||
// Fallback for older devices
|
|
||||||
final result = await FilePicker.platform.getDirectoryPath();
|
final result = await FilePicker.platform.getDirectoryPath();
|
||||||
if (result != null) {
|
if (result != null) {
|
||||||
if (Platform.isIOS) {
|
if (Platform.isIOS) {
|
||||||
// On iOS, create a security-scoped bookmark so we can access
|
|
||||||
// this folder across app restarts and from the Go backend.
|
|
||||||
final bookmark = await PlatformBridge.createIosBookmarkFromPath(
|
final bookmark = await PlatformBridge.createIosBookmarkFromPath(
|
||||||
result,
|
result,
|
||||||
);
|
);
|
||||||
@@ -139,8 +132,6 @@ class _LibrarySettingsPageState extends ConsumerState<LibrarySettingsPage> {
|
|||||||
.read(settingsProvider.notifier)
|
.read(settingsProvider.notifier)
|
||||||
.setLocalLibraryPathAndBookmark(result, bookmark);
|
.setLocalLibraryPathAndBookmark(result, bookmark);
|
||||||
} else {
|
} else {
|
||||||
// Bookmark creation failed; save path anyway (works for
|
|
||||||
// app-internal folders like Documents/).
|
|
||||||
ref.read(settingsProvider.notifier).setLocalLibraryPath(result);
|
ref.read(settingsProvider.notifier).setLocalLibraryPath(result);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@@ -162,13 +153,7 @@ class _LibrarySettingsPageState extends ConsumerState<LibrarySettingsPage> {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// On iOS with a bookmark, try resolving the bookmark first to validate
|
|
||||||
// access instead of checking the path directly (which may fail outside
|
|
||||||
// the app sandbox).
|
|
||||||
if (Platform.isIOS && iosBookmark.isNotEmpty) {
|
if (Platform.isIOS && iosBookmark.isNotEmpty) {
|
||||||
// Bookmark will be resolved inside startScan; skip Directory.exists
|
|
||||||
// check since security-scoped paths are not accessible without the
|
|
||||||
// bookmark being activated.
|
|
||||||
} else if (!libraryPath.startsWith('content://') &&
|
} else if (!libraryPath.startsWith('content://') &&
|
||||||
!await Directory(libraryPath).exists()) {
|
!await Directory(libraryPath).exists()) {
|
||||||
if (mounted) {
|
if (mounted) {
|
||||||
@@ -467,7 +452,6 @@ class _LibrarySettingsPageState extends ConsumerState<LibrarySettingsPage> {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
// Scan Actions Section
|
|
||||||
if (settings.localLibraryEnabled) ...[
|
if (settings.localLibraryEnabled) ...[
|
||||||
SliverToBoxAdapter(
|
SliverToBoxAdapter(
|
||||||
child: SettingsSectionHeader(title: context.l10n.libraryActions),
|
child: SettingsSectionHeader(title: context.l10n.libraryActions),
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user